From ba9ce1a0609a6d8f3c24619622b20d36b9d2c4c4 Mon Sep 17 00:00:00 2001 From: Harunobu Ishii Date: Wed, 27 Mar 2024 18:11:08 -0400 Subject: [PATCH] prettier applied --- .prettierignore | 3 +- .prettierrc | 6 +- README.md | 59 +- dist/jiff-client-websockets.js | 1990 +- dist/jiff-client.js | 111773 +++++++++------ index.html | 2 +- lib/client/api.js | 3 +- lib/client/api/bits.js | 2 +- lib/client/api/crypto_provider.js | 4 +- lib/client/api/custom.js | 8 +- lib/client/api/initialization.js | 4 +- lib/client/api/protocols.js | 2 +- lib/client/api/sharing.js | 8 +- lib/client/api/synchronization.js | 4 +- lib/client/arch/counters.js | 11 +- lib/client/arch/extensions.js | 2 +- lib/client/arch/hooks.js | 10 +- lib/client/handlers.js | 2 +- lib/client/handlers/crypto_provider.js | 2 +- lib/client/handlers/custom.js | 6 +- lib/client/handlers/initialization.js | 2 +- lib/client/handlers/sharing.js | 6 +- lib/client/preprocessing/api.js | 20 +- lib/client/preprocessing/daemon.js | 67 +- lib/client/preprocessing/handlers.js | 81 +- lib/client/preprocessing/map.js | 174 +- lib/client/protocols/arrays/open.js | 24 +- lib/client/protocols/arrays/share.js | 73 +- lib/client/protocols/arrays/util.js | 13 +- lib/client/protocols/bits/arithmetic.js | 375 +- lib/client/protocols/bits/comparison.js | 17 +- lib/client/protocols/bits/protocols.js | 5 +- lib/client/protocols/bits/sharing.js | 2 +- lib/client/protocols/booleans/boolean.js | 10 +- lib/client/protocols/generic.js | 13 +- lib/client/protocols/numbers/arithmetic.js | 41 +- lib/client/protocols/numbers/comparison.js | 39 +- lib/client/protocols/numbers/protocols.js | 2 +- lib/client/protocols/preprocessing/bits.js | 10 +- lib/client/protocols/preprocessing/numbers.js | 4 +- .../protocols/preprocessing/quotients.js | 4 +- .../protocols/preprocessing/sampling.js | 29 +- lib/client/protocols/shamir/open.js | 8 +- lib/client/protocols/shamir/reshare.js | 7 +- lib/client/protocols/shamir/share.js | 17 +- lib/client/share.js | 40 +- lib/client/shareHelpers.js | 16 +- lib/client/socket/events.js | 8 +- lib/client/socket/internal.js | 2 +- lib/client/socket/mailbox.js | 4 +- lib/client/util/constants.js | 2 +- lib/client/util/crypto.js | 2 +- lib/client/util/helpers.js | 14 +- lib/client/util/utils.js | 12 +- lib/common/helpers.js | 13 +- lib/common/linkedlist.js | 18 +- lib/ext/Hooks.md | 257 +- lib/ext/README.md | 41 +- lib/ext/jiff-client-bignumber.js | 44 +- lib/ext/jiff-client-debugging.js | 31 +- lib/ext/jiff-client-fixedpoint.js | 85 +- lib/ext/jiff-client-negativenumber.js | 95 +- lib/ext/jiff-client-performance.js | 4 +- lib/ext/jiff-client-restful.js | 17 +- lib/ext/jiff-client-websockets.js | 42 +- lib/ext/jiff-server-bignumber.js | 4 +- lib/ext/jiff-server-restful.js | 9 +- lib/ext/jiff-server-websockets.js | 13 +- lib/jiff-client.js | 12 +- lib/jiff-server.js | 2 +- lib/server/compute.js | 4 +- lib/server/constants.js | 2 +- lib/server/cryptoprovider.js | 8 +- lib/server/datastructures/intervals.js | 12 +- lib/server/extensions.js | 2 +- lib/server/handlers.js | 29 +- lib/server/hooks.js | 67 +- lib/server/mailbox.js | 6 +- lib/server/socket.js | 25 +- tests/dev/2D_array/mpc.js | 16 +- tests/dev/2D_array/party.js | 2 +- tests/dev/communication-test/client.html | 12 +- tests/dev/communication-test/client.js | 10 +- tests/dev/communication-test/mpc.js | 8 +- tests/dev/communication-test/party.js | 2 +- tests/live/console.js | 21 +- tests/live/index.html | 52 +- tests/live/server.js | 2 +- tests/suite/README.md | 78 +- tests/suite/computations-ifelse.js | 4 +- tests/suite/computations-share.js | 10 +- tests/suite/computations.js | 62 +- tests/suite/config/base/arithmetic.json | 2 +- tests/suite/config/base/comparison.json | 2 +- .../config/base/constant arithmetic.json | 22 +- .../config/base/constant comparison.json | 2 +- tests/suite/config/base/generation.js | 2 +- tests/suite/config/base/ifelse.json | 2 +- tests/suite/config/base/share.json | 4 +- .../bigNumber-negativeNumber/arithmetic.json | 4 +- .../bigNumber-negativeNumber/comparison.json | 4 +- .../bigNumber-negativeNumber/computations.js | 2 +- .../constant arithmetic.json | 4 +- .../constant comparison.json | 4 +- .../bigNumber-negativeNumber/generation.js | 1 - .../bigNumber-negativeNumber/ifelse.json | 4 +- .../bigNumber-negativeNumber/share.json | 4 +- tests/suite/config/bigNumber/arithmetic.json | 4 +- tests/suite/config/bigNumber/comparison.json | 4 +- tests/suite/config/bigNumber/computations.js | 2 +- .../config/bigNumber/constant arithmetic.json | 24 +- .../config/bigNumber/constant comparison.json | 4 +- tests/suite/config/bigNumber/generation.js | 2 +- tests/suite/config/bigNumber/helpers.js | 2 +- tests/suite/config/bigNumber/ifelse.json | 4 +- tests/suite/config/bigNumber/share.json | 4 +- .../suite/config/bits/arithmetic lengths.json | 4 +- tests/suite/config/bits/arithmetic.json | 4 +- .../suite/config/bits/comparison lengths.json | 2 +- tests/suite/config/bits/comparison.json | 2 +- .../config/bits/computations-rejection.js | 36 +- tests/suite/config/bits/computations.js | 84 +- .../config/bits/constant arithmetic.json | 24 +- .../config/bits/constant comparison.json | 2 +- tests/suite/config/bits/decomposition.json | 15 +- tests/suite/config/bits/generation.js | 11 +- .../suite/config/bits/rejection sampling.json | 205 +- .../fixedpoint-negativeNumber/arithmetic.json | 4 +- .../fixedpoint-negativeNumber/comparison.json | 4 +- .../fixedpoint-negativeNumber/computations.js | 2 +- .../constant arithmetic.json | 22 +- .../constant comparison.json | 4 +- .../fixedpoint-negativeNumber/generation.js | 6 +- .../fixedpoint-negativeNumber/ifelse.json | 4 +- .../fixedpoint-negativeNumber/share.json | 4 +- tests/suite/config/fixedpoint/arithmetic.json | 4 +- tests/suite/config/fixedpoint/comparison.json | 4 +- tests/suite/config/fixedpoint/computations.js | 2 +- .../fixedpoint/constant arithmetic.json | 16 +- .../fixedpoint/constant comparison.json | 4 +- tests/suite/config/fixedpoint/generation.js | 6 +- tests/suite/config/fixedpoint/ifelse.json | 4 +- tests/suite/config/fixedpoint/share.json | 4 +- .../config/jiff_websockets/arithmetic.json | 4 +- .../config/jiff_websockets/comparison.json | 4 +- .../jiff_websockets/constant arithmetic.json | 24 +- .../jiff_websockets/constant comparison.json | 4 +- .../suite/config/jiff_websockets/ifelse.json | 4 +- tests/suite/config/jiff_websockets/share.json | 4 +- .../config/negativeNumber/arithmetic.json | 6 +- .../config/negativeNumber/comparison.json | 4 +- .../config/negativeNumber/computations.js | 2 +- .../negativeNumber/constant arithmetic.json | 4 +- .../negativeNumber/constant comparison.json | 4 +- .../suite/config/negativeNumber/generation.js | 3 +- tests/suite/config/negativeNumber/ifelse.json | 4 +- tests/suite/config/negativeNumber/share.json | 4 +- tests/suite/config/restAPI/arithmetic.json | 6 +- tests/suite/config/restAPI/comparison.json | 4 +- .../config/restAPI/constant arithmetic.json | 4 +- .../config/restAPI/constant comparison.json | 4 +- tests/suite/config/restAPI/ifelse.json | 4 +- tests/suite/config/restAPI/share.json | 4 +- tests/suite/init.js | 4 +- tests/suite/server.js | 5 +- tests/utils/process.js | 2 +- 166 files changed, 68534 insertions(+), 48339 deletions(-) diff --git a/.prettierignore b/.prettierignore index 7d84bfd6d..588a2ec05 100644 --- a/.prettierignore +++ b/.prettierignore @@ -4,4 +4,5 @@ coverage demos .circleci .github -.travis.yml \ No newline at end of file +.travis.yml +CONTRIBUTING.md \ No newline at end of file diff --git a/.prettierrc b/.prettierrc index 0967ef424..a86dcc66b 100644 --- a/.prettierrc +++ b/.prettierrc @@ -1 +1,5 @@ -{} +{ + "printWidth": 200, + "singleQuote": true, + "trailingComma": "none" +} diff --git a/README.md b/README.md index 9b24301c8..69d3aa619 100644 --- a/README.md +++ b/README.md @@ -21,6 +21,7 @@ You can use browserify or similar tools to require JIFF via npm and bundle it wi ## Installation Add JIFF as a dependency to your project: + ```shell npm install jiff-mpc ``` @@ -29,6 +30,7 @@ npm install jiff-mpc After installing JIFF via npm, you can require the server module in your server code using: + ```javascript const { JIFFServer } = require('jiff-mpc'); @@ -40,6 +42,7 @@ http.listen(port, cb); ``` Similarly, you can require the client module in your node.js clients using: + ```javascript const { JIFFClient } = require('jiff-mpc'); @@ -55,15 +58,18 @@ let shares = jiffClient.share(); To use our bundle, include the provided pre-built JS file in a script tag. Make sure that you set up your web-server to serve this JS file. + ```html ``` Then inside a script tag (and after the page loads), initialize a JIFF object and set up a computation: + ```javascript const jiffClient = new JIFFClient("", "", ); ``` + The jiffClient object provides methods for sharing, opening, and performing operations on shares. Alternatively, you can use the same code for both Node.js and browser-based clients, using tools @@ -135,14 +141,17 @@ Each document is an independent tutorial. However, beginners are encouraged to v ## Running Demos and Examples Run a sample server from one of the demos under `demos` in the following way: + ```shell node demos//server.js ``` + The output from the example server will direct you to open `localhost:8080/demos//client.html` in a browser (you must open an instance in a separate window/tab for every distinct party participating in the protocol). You can then proceed with the protocol using the client interfaces. Note that you can run Node.js parties that can also participate in the protocol by executing (e.g., a separate terminal for each party): + ```shell node demos//party.js ``` @@ -150,15 +159,16 @@ node demos//party.js ## Documentation The latest documentation can be viewed on the [project page](https://multiparty.org/jiff/). The documentation can be generated using [JSDoc](http://usejsdoc.org/); you will find these docs in `docs/jsdocs/`: + ```shell ./node_modules/.bin/jsdoc -r -c docs/jsdoc.conf.json npm run-script gen-docs # shortcut ``` + ### Where to Look in the Docs The documentation for the client side library is separated into the distinct modules, namespaces, and classes: - ├─ modules │ └─ jiff-client Parent module: represents the exposed JIFFClient global variable ├─ classes @@ -174,14 +184,17 @@ The documentation for the client side library is separated into the distinct mod ## Running Tests All of the JIFF library test cases can be run in the following way: + ```shell npm test ``` Demos are accompanied by test cases. The following command can be used to run the demo servers and test cases: + ```shell npm run-script test-demo -- demos/ ``` + The command assumes that the server is located at demos//server.js and the test cases are located at demos//test.js See demos/run-test.sh for instructions on running test cases located in different directories or with different names. @@ -190,13 +203,14 @@ See the [testing suite framework documentation](tests/suite/README.md) for more ## Bundling If you made changes to the library and would like to bundle it again into a single browser-friendly file, you can run this command: + ```shell npm run-script build # will override dist/jiff-client.js ``` ## Development -The JIFF libraries allow developers to customize or extend their functionality by introducing new *hooks*. Multiple hooks can be combined to form a library *extension*. +The JIFF libraries allow developers to customize or extend their functionality by introducing new _hooks_. Multiple hooks can be combined to form a library _extension_. ### Hooks @@ -218,11 +232,13 @@ For examples on how to use an extension, see the following files: 2. `demos/sum-fixed/client.html`: using fixed point arithmetic extension in the browser. Run the bignumber test suite in the following way: + ```shell npm run-script test-bignumber ``` ## How to Contribute + Check out our contribution guidelines and resources @ [contributing](CONTRIBUTING.md). # For Cryptographers @@ -242,26 +258,27 @@ reduces to more traditional models in certain cases. For example, if the computa equivalent to 3-party computation with honest majority. ## Costs of Operations: [OUTDATED] -Below is a table of the current costs of operations in the *base* JIFF without extensions: - - -| Operation | Rounds | Total Messages | Preprocessing Rounds | Preprocessing Total Messages | Dependencies | -|-------------------|-------------------|-----------------------------------|----------------------|----------------------------------------------|--------------| -| Share | 1 | senders \* receivers | 0 | 0 | N/A | -| Open | 2 | sender + sender \* receivers | 1 | senders \* senders | N/A | -| +, -, c+, c-, c\* | 0 | 0 | 0 | 0 | N/A | -| \* | 2 | 2\*parties + parties\*(parties-1) | 2 | 2 \* (parties \* parties - 1) | triplet,open | -| <, <=, >, >= | 2\*(bits+3) | O( bits \* parties^2 ) | 3 | bits \* (2\*parties + parties^2) | \*, open | -| c<, c<=, c>, c>= | 2\*(bits+3) | O( bits \* parties^2 ) | 3 | bits \* (2\*parties + parties^2) | \*, open | -| =, c=, !=, c!= | 2\*(bits+4) | O( bits \* parties^2 ) | 3 | 2\*bits \* (2\*parties + parties^2) | c<, c>, \* | -| / | bits^2 + 5\*bits | O( bits^2 \* parties^2 ) | 3 | bits\*(2\*bits \* (2\*parties + parties^2)) | <, c<, \* | -| c/ | 2\*(bits+3) + 5 | O( bits \* parties^2 ) | 3 | 4 \* bits \* (2\*parties + parties^2) | open, \*, c< | -| bits+ | 8\*bits | O( parties^2 \* bits ) | 2 | 8 \* bits \* (parties \* parties - 1) | triplet,open | -| bits- | 8\*bits | O( parties^2 \* bits ) | 2 | 8 \* bits \* (parties \* parties - 1) | triplet,open | -| bits* | 12\*bits | O( parties^4 \* bits^2 ) | 2 | 12 \* bits^2 \* (parties \* parties - 1)^2 | triplet,open | -| bits/ | 25\*bits^2 | O( parties^2 \* bits^2 ) | 2 | 25 \* bits^2 \* (parties \* parties - 1) | triplet,open | + +Below is a table of the current costs of operations in the _base_ JIFF without extensions: + +| Operation | Rounds | Total Messages | Preprocessing Rounds | Preprocessing Total Messages | Dependencies | +| ----------------- | ---------------- | --------------------------------- | -------------------- | ------------------------------------------- | ------------ | +| Share | 1 | senders \* receivers | 0 | 0 | N/A | +| Open | 2 | sender + sender \* receivers | 1 | senders \* senders | N/A | +| +, -, c+, c-, c\* | 0 | 0 | 0 | 0 | N/A | +| \* | 2 | 2\*parties + parties\*(parties-1) | 2 | 2 \* (parties \* parties - 1) | triplet,open | +| <, <=, >, >= | 2\*(bits+3) | O( bits \* parties^2 ) | 3 | bits \* (2\*parties + parties^2) | \*, open | +| c<, c<=, c>, c>= | 2\*(bits+3) | O( bits \* parties^2 ) | 3 | bits \* (2\*parties + parties^2) | \*, open | +| =, c=, !=, c!= | 2\*(bits+4) | O( bits \* parties^2 ) | 3 | 2\*bits \* (2\*parties + parties^2) | c<, c>, \* | +| / | bits^2 + 5\*bits | O( bits^2 \* parties^2 ) | 3 | bits\*(2\*bits \* (2\*parties + parties^2)) | <, c<, \* | +| c/ | 2\*(bits+3) + 5 | O( bits \* parties^2 ) | 3 | 4 \* bits \* (2\*parties + parties^2) | open, \*, c< | +| bits+ | 8\*bits | O( parties^2 \* bits ) | 2 | 8 \* bits \* (parties \* parties - 1) | triplet,open | +| bits- | 8\*bits | O( parties^2 \* bits ) | 2 | 8 \* bits \* (parties \* parties - 1) | triplet,open | +| bits\* | 12\*bits | O( parties^4 \* bits^2 ) | 2 | 12 \* bits^2 \* (parties \* parties - 1)^2 | triplet,open | +| bits/ | 25\*bits^2 | O( parties^2 \* bits^2 ) | 2 | 25 \* bits^2 \* (parties \* parties - 1) | triplet,open | Some exact costs not shown in the table: + 1. Exact total number of messages for secret inequalities is: 3\*(parties + parties^2 + (bits+1) \* (2\*parties + parties\*(parties-1))) + 2\*parties + parties\*(parties-1) 2. Exact total number of messages for constant inequalities is: 2\*(parties + parties^2 + (bits+1) \* (2\*parties + parties\*(parties-1))) + 2\*parties + parties\*(parties-1) 3. Exact total number of messages for equality checks: 2\*(\*(parties + parties^2 + (bits+1) \* (2\*parties + parties\*(parties-1))) + 2\*parties + parties\*(parties-1)) + 2\*parties + parties\*(parties-1) @@ -269,6 +286,7 @@ Some exact costs not shown in the table: 5. Exact total number of messages for constant division is: 1 + 7\*parties + 4\*parties^2 + 8\*(parties + parties^2 + (bits+1) \* (2\*parties + parties\*(parties-1))) Dependencies: + 1. Multiplication has one message to synchronize beaver triplets and one open in sequence. 2. inequality tests has 3 less than half primes in parallel, each has an open and as many multiplication in sequence as bits. 3. constant inequality test has 2 less than half primes in parallel. @@ -277,7 +295,6 @@ Dependencies: 6. constant division has one open sequenced with 4 parallel constant inequality checks and two multiplications. 7. Secret XORs and ORs are equivalent to a single multiplication, constant XORs and ORs are free. - ## Information and Collaborators More information about this project, including collaborators and publications, can be found at [multiparty.org](https://multiparty.org/). diff --git a/dist/jiff-client-websockets.js b/dist/jiff-client-websockets.js index 3de653f11..7be81188f 100644 --- a/dist/jiff-client-websockets.js +++ b/dist/jiff-client-websockets.js @@ -1,429 +1,508 @@ -(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.jiff_websockets = f()}})(function(){var define,module,exports;return (function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i.connect() instead! - * @method - * @memberof handlers - */ - jiffClient.handlers.connected = function () { - console.log('Connected!', jiffClient.id); // TODO: remove debugging - jiffClient.initialization_counter++; - - if (jiffClient.secret_key == null && jiffClient.public_key == null) { - var key = jiffClient.hooks.generateKeyPair(jiffClient); - jiffClient.secret_key = key.secret_key; - jiffClient.public_key = key.public_key; - } - - // Initialization message - var msg = jiffClient.handlers.build_initialization_message(); - - // Emit initialization message to server - jiffClient.socket.emit('initialization', JSON.stringify(msg)); - }; - - /** - * Called after the server approves initialization of this instance. - * Sets the instance id, the count of parties in the computation, and the public keys - * of initialized parties. - * @method - * @memberof handlers - */ - jiffClient.handlers.initialized = function (msg) { - jiffClient.__initialized = true; - jiffClient.initialization_counter = 0; - - msg = JSON.parse(msg); - msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'initialization', msg], 2); - - jiffClient.id = msg.party_id; - jiffClient.party_count = msg.party_count; - - // Now: (1) this party is connect (2) server (and other parties) know this public key - // Resend all pending messages - jiffClient.socket.resend_mailbox(); - - // store the received public keys and resolve wait callbacks - jiffClient.handlers.store_public_keys(msg.public_keys); - }; - - /** - * Parse and store the given public keys - * @method - * @memberof handlers - * @param {object} keymap - maps party id to serialized public key. - */ - jiffClient.handlers.store_public_keys = function (keymap) { - var i; - for (i in keymap) { - if (keymap.hasOwnProperty(i) && jiffClient.keymap[i] == null) { - jiffClient.keymap[i] = jiffClient.hooks.parseKey(jiffClient, keymap[i]); - } - } - - // Resolve any pending messages that were received before the sender's public key was known - jiffClient.resolve_messages_waiting_for_keys(); - - // Resolve any pending waits that have satisfied conditions - jiffClient.execute_wait_callbacks(); - - // Check if all keys have been received - if (jiffClient.keymap['s1'] == null) { - return; - } - for (i = 1; i <= jiffClient.party_count; i++) { - if (jiffClient.keymap[i] == null) { - return; - } - } - - // all parties are connected; execute callback - if (jiffClient.__ready !== true && jiffClient.__initialized) { - jiffClient.__ready = true; - if (jiffClient.options.onConnect != null) { - jiffClient.options.onConnect(jiffClient); - } - } - }; -}; -},{}],5:[function(require,module,exports){ -// adds sharing related handlers -module.exports = function (jiffClient) { - /** - * Store the received share and resolves the corresponding - * deferred if needed. - * @method - * @memberof handlers - * @param {object} json_msg - the parsed json message as received. - */ - jiffClient.handlers.receive_share = function (json_msg) { - // Decrypt share - json_msg['share'] = jiffClient.hooks.decryptSign(jiffClient, json_msg['share'], jiffClient.secret_key, jiffClient.keymap[json_msg['party_id']]); - json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'share', json_msg], 2); - - var sender_id = json_msg['party_id']; - var op_id = json_msg['op_id']; - var share = json_msg['share']; - - // Call hook - share = jiffClient.hooks.execute_array_hooks('receiveShare', [jiffClient, sender_id, share], 2); - - // check if a deferred is set up (maybe the share was received early) - if (jiffClient.deferreds[op_id] == null) { - jiffClient.deferreds[op_id] = {}; - } - if (jiffClient.deferreds[op_id][sender_id] == null) { - // Share is received before deferred was setup, store it. - jiffClient.deferreds[op_id][sender_id] = new jiffClient.helpers.Deferred(); - } - - // Deferred is already setup, resolve it. - jiffClient.deferreds[op_id][sender_id].resolve(share); - }; - - /** - * Resolves the deferred corresponding to operation_id and sender_id. - * @method - * @memberof handlers - * @param {object} json_msg - the json message as received with the open event. - */ - jiffClient.handlers.receive_open = function (json_msg) { - // Decrypt share - if (json_msg['party_id'] !== jiffClient.id) { - json_msg['share'] = jiffClient.hooks.decryptSign(jiffClient, json_msg['share'], jiffClient.secret_key, jiffClient.keymap[json_msg['party_id']]); - json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'open', json_msg], 2); - } - - var sender_id = json_msg['party_id']; - var op_id = json_msg['op_id']; - var share = json_msg['share']; - var Zp = json_msg['Zp']; - - // call hook - share = jiffClient.hooks.execute_array_hooks('receiveOpen', [jiffClient, sender_id, share, Zp], 2); - - // Ensure deferred is setup - if (jiffClient.deferreds[op_id] == null) { - jiffClient.deferreds[op_id] = {}; - } - if (jiffClient.deferreds[op_id].shares == null) { - jiffClient.deferreds[op_id].shares = []; - } - - // Accumulate received shares - jiffClient.deferreds[op_id].shares.push({value: share, sender_id: sender_id, Zp: Zp}); - - // Resolve when ready - if (jiffClient.deferreds[op_id].shares.length === jiffClient.deferreds[op_id].threshold) { - jiffClient.deferreds[op_id].deferred.resolve(); - } - - // Clean up if done - if (jiffClient.deferreds[op_id] != null && jiffClient.deferreds[op_id].deferred === 'CLEAN' && jiffClient.deferreds[op_id].shares.length === jiffClient.deferreds[op_id].total) { - delete jiffClient.deferreds[op_id]; - } - } -}; -},{}],6:[function(require,module,exports){ -/** Doubly linked list with add and remove functions and pointers to head and tail**/ -module.exports = function () { - // attributes: list.head and list.tail - // functions: list.add(object) (returns pointer), list.remove(pointer) - // list.head/list.tail/any element contains: - // next: pointer to next, - // previous: pointer to previous, - // object: stored object. - var list = {head: null, tail: null}; - // TODO rename this to pushTail || push - list.add = function (obj) { - var node = { object: obj, next: null, previous: null }; - if (list.head == null) { - list.head = node; - list.tail = node; - } else { - list.tail.next = node; - node.previous = list.tail; - list.tail = node; - } - return node; - }; - - list.pushHead = function (obj) { - list.head = {object: obj, next : list.head, previous : null}; - if (list.head.next != null) { - list.head.next.previous = list.head; - } else { - list.tail = list.head; - } - }; - - list.popHead = function () { - var result = list.head; - if (list.head != null) { - list.head = list.head.next; - if (list.head == null) { - list.tail = null; - } else { - list.head.previous = null; - } - } - return result; - }; - - // merges two linked lists and return a pointer to the head of the merged list - // the head will be the head of list and the tail the tail of l2 - list.extend = function (l2) { - if (list.head == null) { - return l2; - } - if (l2.head == null) { - return list; - } - list.tail.next = l2.head; - l2.head.previous = list.tail; - list.tail = l2.tail; - - return list; - }; - - list.remove = function (ptr) { - var prev = ptr.previous; - var next = ptr.next; - - if (prev == null && list.head !== ptr) { - return; - } else if (next == null && list.tail !== ptr) { - return; - } - - if (prev == null) { // ptr is head (or both head and tail) - list.head = next; - if (list.head != null) { - list.head.previous = null; - } else { - list.tail = null; - } - } else if (next == null) { // ptr is tail (and not head) - list.tail = prev; - prev.next = null; - } else { // ptr is inside - prev.next = next; - next.previous = prev; - } - }; - list.slice = function (ptr) { // remove all elements from head to ptr (including ptr). - if (ptr == null) { - return; - } - - /* CONSERVATIVE: make sure ptr is part of the list then remove */ - var current = list.head; - while (current != null) { - if (current === ptr) { - list.head = ptr.next; - if (list.head == null) { - list.tail = null; +})(function () { + var define, module, exports; + return (function () { + function r(e, n, t) { + function o(i, f) { + if (!n[i]) { + if (!e[i]) { + var c = 'function' == typeof require && require; + if (!f && c) return c(i, !0); + if (u) return u(i, !0); + var a = new Error("Cannot find module '" + i + "'"); + throw ((a.code = 'MODULE_NOT_FOUND'), a); + } + var p = (n[i] = { exports: {} }); + e[i][0].call( + p.exports, + function (r) { + var n = e[i][1][r]; + return o(n || r); + }, + p, + p.exports, + r, + e, + n, + t + ); } - - return; + return n[i].exports; } - current = current.next; + for (var u = 'function' == typeof require && require, i = 0; i < t.length; i++) o(t[i]); + return o; } - - /* MORE AGGRESSIVE VERSION: will be incorrect if ptr is not in the list */ - /* + return r; + })()( + { + 1: [ + function (require, module, exports) { + var initializationHandlers = require('./handlers/initialization.js'); + var shareHandlers = require('./handlers/sharing.js'); + var customHandlers = require('./handlers/custom.js'); + var cryptoProviderHandlers = require('./handlers/crypto_provider.js'); + + /** + * Contains handlers for communication events + * @name handlers + * @alias handlers + * @namespace + */ + + // Add handlers implementations + module.exports = function (jiffClient) { + // fill in handlers + initializationHandlers(jiffClient); + shareHandlers(jiffClient); + customHandlers(jiffClient); + cryptoProviderHandlers(jiffClient); + }; + }, + { './handlers/crypto_provider.js': 2, './handlers/custom.js': 3, './handlers/initialization.js': 4, './handlers/sharing.js': 5 } + ], + 2: [ + function (require, module, exports) { + // setup handler for receiving messages from the crypto provider + module.exports = function (jiffClient) { + /** + * Parse crypto provider message and resolve associated promise. + * @method + * @memberof handlers + * @param {object} json_msg - the parsed json message as received by the crypto_provider event, contains 'values' and 'shares' attributes. + */ + jiffClient.handlers.receive_crypto_provider = function (json_msg) { + // Hook + json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'crypto_provider', json_msg], 2); + + var op_id = json_msg['op_id']; + if (jiffClient.deferreds[op_id] == null) { + return; // duplicate message: ignore + } + + // parse msg + var receivers_list = json_msg['receivers']; + var threshold = json_msg['threshold']; + var Zp = json_msg['Zp']; + + // construct secret share objects + var result = {}; + if (json_msg['values'] != null) { + result.values = json_msg['values']; + } + if (json_msg['shares'] != null) { + result.shares = []; + for (var i = 0; i < json_msg['shares'].length; i++) { + result.shares.push(new jiffClient.SecretShare(json_msg['shares'][i], receivers_list, threshold, Zp)); + } + } + + // resolve deferred + jiffClient.deferreds[op_id].resolve(result); + delete jiffClient.deferreds[op_id]; + }; + }; + }, + {} + ], + 3: [ + function (require, module, exports) { + module.exports = function (jiffClient) { + /** + * Called when this party receives a custom tag message from any party (including itself). + * If a custom listener was setup to listen to the tag, the message is passed to the listener. + * Otherwise, the message is stored until such a listener is provided. + * @method + * @memberof handlers + * @param {object} json_msg - the parsed json message as received by the custom event. + */ + jiffClient.handlers.receive_custom = function (json_msg) { + if (json_msg['party_id'] !== jiffClient.id) { + if (json_msg['encrypted'] === true) { + json_msg['message'] = jiffClient.hooks.decryptSign(jiffClient, json_msg['message'], jiffClient.secret_key, jiffClient.keymap[json_msg['party_id']]); + } + + json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'custom', json_msg], 2); + } + + var sender_id = json_msg['party_id']; + var tag = json_msg['tag']; + var message = json_msg['message']; + + if (jiffClient.listeners[tag] != null) { + jiffClient.listeners[tag](sender_id, message); + } else { + // Store message until listener is provided + var stored_messages = jiffClient.custom_messages_mailbox[tag]; + if (stored_messages == null) { + stored_messages = []; + jiffClient.custom_messages_mailbox[tag] = stored_messages; + } + + stored_messages.push({ sender_id: sender_id, message: message }); + } + }; + }; + }, + {} + ], + 4: [ + function (require, module, exports) { + // add handlers for initialization + module.exports = function (jiffClient) { + jiffClient.options.initialization = Object.assign({}, jiffClient.options.initialization); + + /** + * Called when an error occurs + * @method + * @memberof handlers + * @param {string} label - the name of message or operation causing the error + * @param {error|string} error - the error + */ + jiffClient.handlers.error = function (label, error) { + if (jiffClient.options.onError) { + jiffClient.options.onError(label, error); + } + + console.log(jiffClient.id, ':', 'Error from server:', label, '---', error); // TODO: remove debugging + if (label === 'initialization') { + jiffClient.socket.disconnect(); + + if (jiffClient.initialization_counter < jiffClient.options.maxInitializationRetries) { + console.log(jiffClient.id, ':', 'reconnecting..'); // TODO: remove debugging + setTimeout(jiffClient.connect, jiffClient.options.socketOptions.reconnectionDelay); + } + } + }; + + /** + * Builds the initialization message for this instance + * @method + * @memberof handlers + * @return {Object} + */ + jiffClient.handlers.build_initialization_message = function () { + var msg = { + computation_id: jiffClient.computation_id, + party_id: jiffClient.id, + party_count: jiffClient.party_count, + public_key: jiffClient.public_key != null ? jiffClient.hooks.dumpKey(jiffClient, jiffClient.public_key) : undefined + }; + msg = Object.assign(msg, jiffClient.options.initialization); + + // Initialization Hook + return jiffClient.hooks.execute_array_hooks('beforeOperation', [jiffClient, 'initialization', msg], 2); + }; + + /** + * Begins initialization of this instance by sending the initialization message to the server. + * Should only be called after connection is established. + * Do not call this manually unless you know what you are doing, use .connect() instead! + * @method + * @memberof handlers + */ + jiffClient.handlers.connected = function () { + console.log('Connected!', jiffClient.id); // TODO: remove debugging + jiffClient.initialization_counter++; + + if (jiffClient.secret_key == null && jiffClient.public_key == null) { + var key = jiffClient.hooks.generateKeyPair(jiffClient); + jiffClient.secret_key = key.secret_key; + jiffClient.public_key = key.public_key; + } + + // Initialization message + var msg = jiffClient.handlers.build_initialization_message(); + + // Emit initialization message to server + jiffClient.socket.emit('initialization', JSON.stringify(msg)); + }; + + /** + * Called after the server approves initialization of this instance. + * Sets the instance id, the count of parties in the computation, and the public keys + * of initialized parties. + * @method + * @memberof handlers + */ + jiffClient.handlers.initialized = function (msg) { + jiffClient.__initialized = true; + jiffClient.initialization_counter = 0; + + msg = JSON.parse(msg); + msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'initialization', msg], 2); + + jiffClient.id = msg.party_id; + jiffClient.party_count = msg.party_count; + + // Now: (1) this party is connect (2) server (and other parties) know this public key + // Resend all pending messages + jiffClient.socket.resend_mailbox(); + + // store the received public keys and resolve wait callbacks + jiffClient.handlers.store_public_keys(msg.public_keys); + }; + + /** + * Parse and store the given public keys + * @method + * @memberof handlers + * @param {object} keymap - maps party id to serialized public key. + */ + jiffClient.handlers.store_public_keys = function (keymap) { + var i; + for (i in keymap) { + if (keymap.hasOwnProperty(i) && jiffClient.keymap[i] == null) { + jiffClient.keymap[i] = jiffClient.hooks.parseKey(jiffClient, keymap[i]); + } + } + + // Resolve any pending messages that were received before the sender's public key was known + jiffClient.resolve_messages_waiting_for_keys(); + + // Resolve any pending waits that have satisfied conditions + jiffClient.execute_wait_callbacks(); + + // Check if all keys have been received + if (jiffClient.keymap['s1'] == null) { + return; + } + for (i = 1; i <= jiffClient.party_count; i++) { + if (jiffClient.keymap[i] == null) { + return; + } + } + + // all parties are connected; execute callback + if (jiffClient.__ready !== true && jiffClient.__initialized) { + jiffClient.__ready = true; + if (jiffClient.options.onConnect != null) { + jiffClient.options.onConnect(jiffClient); + } + } + }; + }; + }, + {} + ], + 5: [ + function (require, module, exports) { + // adds sharing related handlers + module.exports = function (jiffClient) { + /** + * Store the received share and resolves the corresponding + * deferred if needed. + * @method + * @memberof handlers + * @param {object} json_msg - the parsed json message as received. + */ + jiffClient.handlers.receive_share = function (json_msg) { + // Decrypt share + json_msg['share'] = jiffClient.hooks.decryptSign(jiffClient, json_msg['share'], jiffClient.secret_key, jiffClient.keymap[json_msg['party_id']]); + json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'share', json_msg], 2); + + var sender_id = json_msg['party_id']; + var op_id = json_msg['op_id']; + var share = json_msg['share']; + + // Call hook + share = jiffClient.hooks.execute_array_hooks('receiveShare', [jiffClient, sender_id, share], 2); + + // check if a deferred is set up (maybe the share was received early) + if (jiffClient.deferreds[op_id] == null) { + jiffClient.deferreds[op_id] = {}; + } + if (jiffClient.deferreds[op_id][sender_id] == null) { + // Share is received before deferred was setup, store it. + jiffClient.deferreds[op_id][sender_id] = new jiffClient.helpers.Deferred(); + } + + // Deferred is already setup, resolve it. + jiffClient.deferreds[op_id][sender_id].resolve(share); + }; + + /** + * Resolves the deferred corresponding to operation_id and sender_id. + * @method + * @memberof handlers + * @param {object} json_msg - the json message as received with the open event. + */ + jiffClient.handlers.receive_open = function (json_msg) { + // Decrypt share + if (json_msg['party_id'] !== jiffClient.id) { + json_msg['share'] = jiffClient.hooks.decryptSign(jiffClient, json_msg['share'], jiffClient.secret_key, jiffClient.keymap[json_msg['party_id']]); + json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'open', json_msg], 2); + } + + var sender_id = json_msg['party_id']; + var op_id = json_msg['op_id']; + var share = json_msg['share']; + var Zp = json_msg['Zp']; + + // call hook + share = jiffClient.hooks.execute_array_hooks('receiveOpen', [jiffClient, sender_id, share, Zp], 2); + + // Ensure deferred is setup + if (jiffClient.deferreds[op_id] == null) { + jiffClient.deferreds[op_id] = {}; + } + if (jiffClient.deferreds[op_id].shares == null) { + jiffClient.deferreds[op_id].shares = []; + } + + // Accumulate received shares + jiffClient.deferreds[op_id].shares.push({ value: share, sender_id: sender_id, Zp: Zp }); + + // Resolve when ready + if (jiffClient.deferreds[op_id].shares.length === jiffClient.deferreds[op_id].threshold) { + jiffClient.deferreds[op_id].deferred.resolve(); + } + + // Clean up if done + if (jiffClient.deferreds[op_id] != null && jiffClient.deferreds[op_id].deferred === 'CLEAN' && jiffClient.deferreds[op_id].shares.length === jiffClient.deferreds[op_id].total) { + delete jiffClient.deferreds[op_id]; + } + }; + }; + }, + {} + ], + 6: [ + function (require, module, exports) { + /** Doubly linked list with add and remove functions and pointers to head and tail**/ + module.exports = function () { + // attributes: list.head and list.tail + // functions: list.add(object) (returns pointer), list.remove(pointer) + // list.head/list.tail/any element contains: + // next: pointer to next, + // previous: pointer to previous, + // object: stored object. + var list = { head: null, tail: null }; + // TODO rename this to pushTail || push + list.add = function (obj) { + var node = { object: obj, next: null, previous: null }; + if (list.head == null) { + list.head = node; + list.tail = node; + } else { + list.tail.next = node; + node.previous = list.tail; + list.tail = node; + } + return node; + }; + + list.pushHead = function (obj) { + list.head = { object: obj, next: list.head, previous: null }; + if (list.head.next != null) { + list.head.next.previous = list.head; + } else { + list.tail = list.head; + } + }; + + list.popHead = function () { + var result = list.head; + if (list.head != null) { + list.head = list.head.next; + if (list.head == null) { + list.tail = null; + } else { + list.head.previous = null; + } + } + return result; + }; + + // merges two linked lists and return a pointer to the head of the merged list + // the head will be the head of list and the tail the tail of l2 + list.extend = function (l2) { + if (list.head == null) { + return l2; + } + if (l2.head == null) { + return list; + } + list.tail.next = l2.head; + l2.head.previous = list.tail; + list.tail = l2.tail; + + return list; + }; + + list.remove = function (ptr) { + var prev = ptr.previous; + var next = ptr.next; + + if (prev == null && list.head !== ptr) { + return; + } else if (next == null && list.tail !== ptr) { + return; + } + + if (prev == null) { + // ptr is head (or both head and tail) + list.head = next; + if (list.head != null) { + list.head.previous = null; + } else { + list.tail = null; + } + } else if (next == null) { + // ptr is tail (and not head) + list.tail = prev; + prev.next = null; + } else { + // ptr is inside + prev.next = next; + next.previous = prev; + } + }; + list.slice = function (ptr) { + // remove all elements from head to ptr (including ptr). + if (ptr == null) { + return; + } + + /* CONSERVATIVE: make sure ptr is part of the list then remove */ + var current = list.head; + while (current != null) { + if (current === ptr) { + list.head = ptr.next; + if (list.head == null) { + list.tail = null; + } + + return; + } + current = current.next; + } + + /* MORE AGGRESSIVE VERSION: will be incorrect if ptr is not in the list */ + /* list.head = ptr.next; if (list.head == null) { list.tail = null; } */ - }; - /* + }; + /* list._debug_length = function () { var l = 0; var current = list.head; @@ -434,562 +513,553 @@ module.exports = function () { return l; }; */ - return list; -}; - -},{}],7:[function(require,module,exports){ -(function (process,global){ -/** - * This defines a library extension for using websockets rather than socket.io for communication. This - * extension primarily edits/overwrites existing socket functions to use and be compatible with the - * ws library. - * @namespace jiffclient_websockets - * @version 1.0 - * - * REQUIREMENTS: - * You must apply this extension to your client and the server you're communicating with must apply jiffserver_websockets. - * When using this extension in browser, "/dist/jiff-client-websockets.js" must be loaded in client.html instead of this file. - */ - - - -(function (exports, node) { - /** - * The name of this extension: 'websocket' - * @type {string} - * @memberOf jiffclient_websockets - */ - - var ws; - var linkedList; - var handlers; - - linkedList = require('../common/linkedlist.js'); - handlers = require('../client/handlers.js'); - if (!process.browser) { - ws = require('ws'); - } else { - if (typeof WebSocket !== 'undefined') { - ws = WebSocket - } else if (typeof MozWebSocket !== 'undefined') { - ws = MozWebSocket - } else if (typeof global !== 'undefined') { - ws = global.WebSocket || global.MozWebSocket - } else if (typeof window !== 'undefined') { - ws = window.WebSocket || window.MozWebSocket - } else if (typeof self !== 'undefined') { - ws = self.WebSocket || self.MozWebSocket - } - } - - - // Take the jiff-client base instance and options for this extension, and use them - // to construct an instance for this extension. - function make_jiff(base_instance, options) { - var jiff = base_instance; - - // Parse options - if (options == null) { - options = {}; - } - - /* Functions that overwrite client/socket/events.js functionality */ - - /** - * initSocket's '.on' functions needed to be replaced since ws does - * not have as many protocols. Instead these functions are routed to - * when a message is received and a protocol is manually parsed. - */ - jiff.initSocket = function () { - var jiffClient = this; - - /* ws uses the 'open' protocol on connection. Should not conflict with the + return list; + }; + }, + {} + ], + 7: [ + function (require, module, exports) { + (function (process, global) { + /** + * This defines a library extension for using websockets rather than socket.io for communication. This + * extension primarily edits/overwrites existing socket functions to use and be compatible with the + * ws library. + * @namespace jiffclient_websockets + * @version 1.0 + * + * REQUIREMENTS: + * You must apply this extension to your client and the server you're communicating with must apply jiffserver_websockets. + * When using this extension in browser, "/dist/jiff-client-websockets.js" must be loaded in client.html instead of this file. + */ + + (function (exports, node) { + /** + * The name of this extension: 'websocket' + * @type {string} + * @memberOf jiffclient_websockets + */ + + var ws; + var linkedList; + var handlers; + + linkedList = require('../common/linkedlist.js'); + handlers = require('../client/handlers.js'); + if (!process.browser) { + ws = require('ws'); + } else { + if (typeof WebSocket !== 'undefined') { + ws = WebSocket; + } else if (typeof MozWebSocket !== 'undefined') { + ws = MozWebSocket; + } else if (typeof global !== 'undefined') { + ws = global.WebSocket || global.MozWebSocket; + } else if (typeof window !== 'undefined') { + ws = window.WebSocket || window.MozWebSocket; + } else if (typeof self !== 'undefined') { + ws = self.WebSocket || self.MozWebSocket; + } + } + + // Take the jiff-client base instance and options for this extension, and use them + // to construct an instance for this extension. + function make_jiff(base_instance, options) { + var jiff = base_instance; + + // Parse options + if (options == null) { + options = {}; + } + + /* Functions that overwrite client/socket/events.js functionality */ + + /** + * initSocket's '.on' functions needed to be replaced since ws does + * not have as many protocols. Instead these functions are routed to + * when a message is received and a protocol is manually parsed. + */ + jiff.initSocket = function () { + var jiffClient = this; + + /* ws uses the 'open' protocol on connection. Should not conflict with the JIFF open protocl as that will be sent as a message and ws will see it as a 'message' protocol. */ - this.socket.onopen = jiffClient.handlers.connected; - - // Public keys were updated on the server, and it sent us the updates - function publicKeysChanged(msg, callback) { - - msg = JSON.parse(msg); - msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'public_keys', msg], 2); - - jiffClient.handlers.store_public_keys(msg.public_keys); - } - - // Setup receiving matching shares - function share(msg, callback) { - - // parse message - var json_msg = JSON.parse(msg); - var sender_id = json_msg['party_id']; - - if (jiffClient.keymap[sender_id] != null) { - jiffClient.handlers.receive_share(json_msg); - } else { - if (jiffClient.messagesWaitingKeys[sender_id] == null) { - jiffClient.messagesWaitingKeys[sender_id] = []; + this.socket.onopen = jiffClient.handlers.connected; + + // Public keys were updated on the server, and it sent us the updates + function publicKeysChanged(msg, callback) { + msg = JSON.parse(msg); + msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'public_keys', msg], 2); + + jiffClient.handlers.store_public_keys(msg.public_keys); + } + + // Setup receiving matching shares + function share(msg, callback) { + // parse message + var json_msg = JSON.parse(msg); + var sender_id = json_msg['party_id']; + + if (jiffClient.keymap[sender_id] != null) { + jiffClient.handlers.receive_share(json_msg); + } else { + if (jiffClient.messagesWaitingKeys[sender_id] == null) { + jiffClient.messagesWaitingKeys[sender_id] = []; + } + jiffClient.messagesWaitingKeys[sender_id].push({ label: 'share', msg: json_msg }); + } + } + + function mpcOpen(msg, callback) { + // parse message + var json_msg = JSON.parse(msg); + var sender_id = json_msg['party_id']; + + if (jiffClient.keymap[sender_id] != null) { + jiffClient.handlers.receive_open(json_msg); + } else { + if (jiffClient.messagesWaitingKeys[sender_id] == null) { + jiffClient.messagesWaitingKeys[sender_id] = []; + } + jiffClient.messagesWaitingKeys[sender_id].push({ label: 'open', msg: json_msg }); + } + } + + // handle custom messages + function socketCustom(msg, callback) { + var json_msg = JSON.parse(msg); + var sender_id = json_msg['party_id']; + var encrypted = json_msg['encrypted']; + + if (jiffClient.keymap[sender_id] != null || encrypted !== true) { + jiffClient.handlers.receive_custom(json_msg); + } else { + // key must not exist yet for sender_id, and encrypted must be true + if (jiffClient.messagesWaitingKeys[sender_id] == null) { + jiffClient.messagesWaitingKeys[sender_id] = []; + } + jiffClient.messagesWaitingKeys[sender_id].push({ label: 'custom', msg: json_msg }); + } + } + + function cryptoProvider(msg, callback) { + jiffClient.handlers.receive_crypto_provider(JSON.parse(msg)); + } + + function onError(msg) { + try { + msg = JSON.parse(msg); + jiffClient.handlers.error(msg['label'], msg['error']); + } catch (error) { + jiffClient.handlers.error('socket.io', msg); + } + } + + function socketClose(reason) { + if (reason !== 'io client disconnect') { + // check that the reason is an error and not a user initiated disconnect + console.log('Disconnected!', jiffClient.id, reason); + } + + jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'disconnect', reason], -1); + } + + this.socket.onclose = function (reason) { + socketClose(reason.code); + }; + + /** + * In every message sent over ws, we will send along with it a socketProtocol string + * that will be parsed by the receiver to route the request to the correct function. The + * previous information sent by socket.io will be untouched, but now sent inside of msg.data. + */ + this.socket.onmessage = function (msg, callback) { + msg = JSON.parse(msg.data); + + switch (msg.socketProtocol) { + case 'initialization': + jiffClient.handlers.initialized(msg.data); + break; + case 'public_keys': + publicKeysChanged(msg.data, callback); + break; + case 'share': + share(msg.data, callback); + break; + case 'open': + mpcOpen(msg.data, callback); + break; + case 'custom': + socketCustom(msg.data, callback); + break; + case 'crypto_provider': + cryptoProvider(msg.data, callback); + break; + case 'close': + socketClose(msg.data); + break; + case 'disconnect': + socketClose(msg.data); + break; + case 'error': + onError(msg.data); + break; + default: + console.log('Uknown protocol, ' + msg.socketProtocol + ', received'); + } + }; + }; + + /* Overwrite the socketConnect function from jiff-client.js */ + + jiff.socketConnect = function (JIFFClientInstance) { + if (options.__internal_socket == null) { + /** + * Socket wrapper between this instance and the server, based on sockets.io + * @type {!GuardedSocket} + */ + JIFFClientInstance.socket = guardedSocket(JIFFClientInstance); + } else { + JIFFClientInstance.socket = internalSocket(JIFFClientInstance, options.__internal_socket); + } + + // set up socket event handlers + handlers(JIFFClientInstance); + + // Overwrite handlers.connected with our new ws connection handler + JIFFClientInstance.handlers.connected = function () { + JIFFClientInstance.initialization_counter++; + + if (JIFFClientInstance.secret_key == null && JIFFClientInstance.public_key == null) { + var key = JIFFClientInstance.hooks.generateKeyPair(JIFFClientInstance); + JIFFClientInstance.secret_key = key.secret_key; + JIFFClientInstance.public_key = key.public_key; + } + + // Initialization message + var msg = JIFFClientInstance.handlers.build_initialization_message(); + + // Double wrap the msg + msg = JSON.stringify(msg); + + // Emit initialization message to server + JIFFClientInstance.socket.send(JSON.stringify({ socketProtocol: 'initialization', data: msg })); + }; + + JIFFClientInstance.initSocket(); + }; + + /* Functions that overwrite client/socket/mailbox.js functionality */ + + function guardedSocket(jiffClient) { + // Create plain socket io object which we will wrap in this + var socket; + if (jiffClient.hostname.startsWith('http')) { + var modifiedHostName = 'ws' + jiffClient.hostname.substring(jiffClient.hostname.indexOf(':')); + socket = new ws(modifiedHostName); + } else { + socket = new ws(jiffClient.hostname); + } + + socket.old_disconnect = socket.close; + + socket.mailbox = linkedList(); // for outgoing messages + socket.empty_deferred = null; // gets resolved whenever the mailbox is empty + socket.jiffClient = jiffClient; + + // add functionality to socket + socket.safe_emit = safe_emit.bind(socket); + socket.resend_mailbox = resend_mailbox.bind(socket); + socket.disconnect = disconnect.bind(socket); + socket.safe_disconnect = safe_disconnect.bind(socket); + socket.is_empty = is_empty.bind(socket); + + return socket; + } + + function safe_emit(label, msg) { + // add message to mailbox + var mailbox_pointer = this.mailbox.add({ label: label, msg: msg }); + if (this.readyState === 1) { + var self = this; + // emit the message, if an acknowledgment is received, remove it from mailbox + + this.send(JSON.stringify({ socketProtocol: label, data: msg }), null, function (status) { + self.mailbox.remove(mailbox_pointer); + + if (self.is_empty() && self.empty_deferred != null) { + self.empty_deferred.resolve(); + } + + if (label === 'free') { + self.jiffClient.hooks.execute_array_hooks('afterOperation', [self.jiffClient, 'free', msg], 2); + } + }); + } + } + + function resend_mailbox() { + // Create a new mailbox, since the current mailbox will be resent and + // will contain new backups. + var old_mailbox = this.mailbox; + this.mailbox = linkedList(); + + // loop over all stored messages and emit them + var current_node = old_mailbox.head; + while (current_node != null) { + var label = current_node.object.label; + var msg = current_node.object.msg; + this.safe_emit(label, msg); + current_node = current_node.next; + } + } + + function disconnect() { + this.jiffClient.hooks.execute_array_hooks('beforeOperation', [this.jiffClient, 'disconnect', {}], -1); + + this.old_disconnect.apply(this, arguments); + } + + function safe_disconnect(free, callback) { + if (this.is_empty()) { + if (free) { + this.jiffClient.free(); + free = false; + } else { + // T: Should remain "disconnect" since we override the .disconnect, no need to change to close + this.disconnect(); + if (callback != null) { + callback(); + } + return; + } + } + + this.empty_deferred = new this.jiffClient.helpers.Deferred(); + this.empty_deferred.promise.then(this.safe_disconnect.bind(this, free, callback)); + } + + function is_empty() { + return this.mailbox.head == null && this.jiffClient.counters.pending_opens === 0; + } + + /* PREPROCESSING IS THE SAME */ + jiff.preprocessing_function_map[exports.name] = {}; + + return jiff; + } + // Expose the API for this extension. + exports.make_jiff = make_jiff; + })(typeof exports === 'undefined' ? (this.jiff_websockets = {}) : exports, typeof exports !== 'undefined'); + }).call(this, require('_process'), typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : typeof window !== 'undefined' ? window : {}); + }, + { '../client/handlers.js': 1, '../common/linkedlist.js': 6, _process: 8, ws: 9 } + ], + 8: [ + function (require, module, exports) { + // shim for using process in browser + var process = (module.exports = {}); + + // cached from whatever global is present so that test runners that stub it + // don't break things. But we need to wrap it in a try catch in case it is + // wrapped in strict mode code which doesn't define any globals. It's inside a + // function because try/catches deoptimize in certain engines. + + var cachedSetTimeout; + var cachedClearTimeout; + + function defaultSetTimout() { + throw new Error('setTimeout has not been defined'); } - jiffClient.messagesWaitingKeys[sender_id].push({ label: 'share', msg: json_msg }); - } - } - - function mpcOpen(msg, callback) { - // parse message - var json_msg = JSON.parse(msg); - var sender_id = json_msg['party_id']; - - if (jiffClient.keymap[sender_id] != null) { - jiffClient.handlers.receive_open(json_msg); - } else { - if (jiffClient.messagesWaitingKeys[sender_id] == null) { - jiffClient.messagesWaitingKeys[sender_id] = []; + function defaultClearTimeout() { + throw new Error('clearTimeout has not been defined'); } - jiffClient.messagesWaitingKeys[sender_id].push({ label: 'open', msg: json_msg }); - } - } - - // handle custom messages - function socketCustom(msg, callback) { - var json_msg = JSON.parse(msg); - var sender_id = json_msg['party_id']; - var encrypted = json_msg['encrypted']; - - if (jiffClient.keymap[sender_id] != null || encrypted !== true) { - jiffClient.handlers.receive_custom(json_msg); - } else { - // key must not exist yet for sender_id, and encrypted must be true - if (jiffClient.messagesWaitingKeys[sender_id] == null) { - jiffClient.messagesWaitingKeys[sender_id] = []; + (function () { + try { + if (typeof setTimeout === 'function') { + cachedSetTimeout = setTimeout; + } else { + cachedSetTimeout = defaultSetTimout; + } + } catch (e) { + cachedSetTimeout = defaultSetTimout; + } + try { + if (typeof clearTimeout === 'function') { + cachedClearTimeout = clearTimeout; + } else { + cachedClearTimeout = defaultClearTimeout; + } + } catch (e) { + cachedClearTimeout = defaultClearTimeout; + } + })(); + function runTimeout(fun) { + if (cachedSetTimeout === setTimeout) { + //normal enviroments in sane situations + return setTimeout(fun, 0); + } + // if setTimeout wasn't available but was latter defined + if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) { + cachedSetTimeout = setTimeout; + return setTimeout(fun, 0); + } + try { + // when when somebody has screwed with setTimeout but no I.E. maddness + return cachedSetTimeout(fun, 0); + } catch (e) { + try { + // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally + return cachedSetTimeout.call(null, fun, 0); + } catch (e) { + // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error + return cachedSetTimeout.call(this, fun, 0); + } + } } - jiffClient.messagesWaitingKeys[sender_id].push({ label: 'custom', msg: json_msg }); - } - } - - function cryptoProvider(msg, callback) { - jiffClient.handlers.receive_crypto_provider(JSON.parse(msg)); - } - - function onError(msg) { - try { - msg = JSON.parse(msg); - jiffClient.handlers.error(msg['label'], msg['error']); - } catch (error) { - jiffClient.handlers.error('socket.io', msg); - } - } - - function socketClose(reason) { - if (reason !== 'io client disconnect') { - // check that the reason is an error and not a user initiated disconnect - console.log('Disconnected!', jiffClient.id, reason); - } - - jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'disconnect', reason], -1); - } - - this.socket.onclose = function (reason) { - socketClose(reason.code); - } - - /** - * In every message sent over ws, we will send along with it a socketProtocol string - * that will be parsed by the receiver to route the request to the correct function. The - * previous information sent by socket.io will be untouched, but now sent inside of msg.data. - */ - this.socket.onmessage = function (msg, callback) { - msg = JSON.parse(msg.data); - - switch (msg.socketProtocol) { - case 'initialization': - jiffClient.handlers.initialized(msg.data); - break; - case 'public_keys': - publicKeysChanged(msg.data, callback); - break; - case 'share': - share(msg.data, callback); - break; - case 'open': - mpcOpen(msg.data, callback); - break; - case 'custom': - socketCustom(msg.data, callback); - break; - case 'crypto_provider': - cryptoProvider(msg.data, callback); - break; - case 'close': - socketClose(msg.data); - break; - case 'disconnect': - socketClose(msg.data); - break; - case 'error': - onError(msg.data); - break; - default: - console.log('Uknown protocol, ' + msg.socketProtocol + ', received'); - } - } - - }; - - /* Overwrite the socketConnect function from jiff-client.js */ - - jiff.socketConnect = function (JIFFClientInstance) { - - if (options.__internal_socket == null) { - /** - * Socket wrapper between this instance and the server, based on sockets.io - * @type {!GuardedSocket} - */ - JIFFClientInstance.socket = guardedSocket(JIFFClientInstance); - } else { - JIFFClientInstance.socket = internalSocket(JIFFClientInstance, options.__internal_socket); - } - - // set up socket event handlers - handlers(JIFFClientInstance); - - // Overwrite handlers.connected with our new ws connection handler - JIFFClientInstance.handlers.connected = function () { - JIFFClientInstance.initialization_counter++; - - if (JIFFClientInstance.secret_key == null && JIFFClientInstance.public_key == null) { - var key = JIFFClientInstance.hooks.generateKeyPair(JIFFClientInstance); - JIFFClientInstance.secret_key = key.secret_key; - JIFFClientInstance.public_key = key.public_key; - } - - // Initialization message - var msg = JIFFClientInstance.handlers.build_initialization_message(); - - // Double wrap the msg - msg = JSON.stringify(msg); - - // Emit initialization message to server - JIFFClientInstance.socket.send(JSON.stringify({ socketProtocol: 'initialization', data: msg })); - }; - - - JIFFClientInstance.initSocket(); - } - - /* Functions that overwrite client/socket/mailbox.js functionality */ - - function guardedSocket(jiffClient) { - // Create plain socket io object which we will wrap in this - var socket; - if (jiffClient.hostname.startsWith("http")) { - var modifiedHostName = "ws" + jiffClient.hostname.substring(jiffClient.hostname.indexOf(":")) - socket = new ws(modifiedHostName) - } else { - socket = new ws(jiffClient.hostname); - } - - - socket.old_disconnect = socket.close; - - socket.mailbox = linkedList(); // for outgoing messages - socket.empty_deferred = null; // gets resolved whenever the mailbox is empty - socket.jiffClient = jiffClient; - - // add functionality to socket - socket.safe_emit = safe_emit.bind(socket); - socket.resend_mailbox = resend_mailbox.bind(socket); - socket.disconnect = disconnect.bind(socket); - socket.safe_disconnect = safe_disconnect.bind(socket); - socket.is_empty = is_empty.bind(socket); - - return socket; - } - - function safe_emit(label, msg) { - // add message to mailbox - var mailbox_pointer = this.mailbox.add({ label: label, msg: msg }); - if (this.readyState === 1) { - var self = this; - // emit the message, if an acknowledgment is received, remove it from mailbox - - this.send(JSON.stringify({ socketProtocol: label, data: msg }), null, function (status) { - - self.mailbox.remove(mailbox_pointer); - - if (self.is_empty() && self.empty_deferred != null) { - self.empty_deferred.resolve(); + function runClearTimeout(marker) { + if (cachedClearTimeout === clearTimeout) { + //normal enviroments in sane situations + return clearTimeout(marker); + } + // if clearTimeout wasn't available but was latter defined + if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) { + cachedClearTimeout = clearTimeout; + return clearTimeout(marker); + } + try { + // when when somebody has screwed with setTimeout but no I.E. maddness + return cachedClearTimeout(marker); + } catch (e) { + try { + // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally + return cachedClearTimeout.call(null, marker); + } catch (e) { + // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error. + // Some versions of I.E. have different rules for clearTimeout vs setTimeout + return cachedClearTimeout.call(this, marker); + } + } } - - if (label === 'free') { - self.jiffClient.hooks.execute_array_hooks('afterOperation', [self.jiffClient, 'free', msg], 2); + var queue = []; + var draining = false; + var currentQueue; + var queueIndex = -1; + + function cleanUpNextTick() { + if (!draining || !currentQueue) { + return; + } + draining = false; + if (currentQueue.length) { + queue = currentQueue.concat(queue); + } else { + queueIndex = -1; + } + if (queue.length) { + drainQueue(); + } } - }); - } - - } - - function resend_mailbox() { - // Create a new mailbox, since the current mailbox will be resent and - // will contain new backups. - var old_mailbox = this.mailbox; - this.mailbox = linkedList(); - - // loop over all stored messages and emit them - var current_node = old_mailbox.head; - while (current_node != null) { - var label = current_node.object.label; - var msg = current_node.object.msg; - this.safe_emit(label, msg); - current_node = current_node.next; - } - - } - - function disconnect() { - this.jiffClient.hooks.execute_array_hooks('beforeOperation', [this.jiffClient, 'disconnect', {}], -1); - - - this.old_disconnect.apply(this, arguments); - } - - function safe_disconnect(free, callback) { - - if (this.is_empty()) { - - if (free) { - this.jiffClient.free(); - free = false; - } else { - // T: Should remain "disconnect" since we override the .disconnect, no need to change to close - this.disconnect(); - if (callback != null) { - callback(); + function drainQueue() { + if (draining) { + return; + } + var timeout = runTimeout(cleanUpNextTick); + draining = true; + + var len = queue.length; + while (len) { + currentQueue = queue; + queue = []; + while (++queueIndex < len) { + if (currentQueue) { + currentQueue[queueIndex].run(); + } + } + queueIndex = -1; + len = queue.length; + } + currentQueue = null; + draining = false; + runClearTimeout(timeout); } - return; - } - } - - this.empty_deferred = new this.jiffClient.helpers.Deferred(); - this.empty_deferred.promise.then(this.safe_disconnect.bind(this, free, callback)); - } - - function is_empty() { - return this.mailbox.head == null && this.jiffClient.counters.pending_opens === 0; - - } - - /* PREPROCESSING IS THE SAME */ - jiff.preprocessing_function_map[exports.name] = {}; - - - return jiff; - } - // Expose the API for this extension. - exports.make_jiff = make_jiff; - -}((typeof exports === 'undefined' ? this.jiff_websockets = {} : exports), typeof exports !== 'undefined')); - -}).call(this,require('_process'),typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) - -},{"../client/handlers.js":1,"../common/linkedlist.js":6,"_process":8,"ws":9}],8:[function(require,module,exports){ -// shim for using process in browser -var process = module.exports = {}; - -// cached from whatever global is present so that test runners that stub it -// don't break things. But we need to wrap it in a try catch in case it is -// wrapped in strict mode code which doesn't define any globals. It's inside a -// function because try/catches deoptimize in certain engines. - -var cachedSetTimeout; -var cachedClearTimeout; - -function defaultSetTimout() { - throw new Error('setTimeout has not been defined'); -} -function defaultClearTimeout () { - throw new Error('clearTimeout has not been defined'); -} -(function () { - try { - if (typeof setTimeout === 'function') { - cachedSetTimeout = setTimeout; - } else { - cachedSetTimeout = defaultSetTimout; - } - } catch (e) { - cachedSetTimeout = defaultSetTimout; - } - try { - if (typeof clearTimeout === 'function') { - cachedClearTimeout = clearTimeout; - } else { - cachedClearTimeout = defaultClearTimeout; - } - } catch (e) { - cachedClearTimeout = defaultClearTimeout; - } -} ()) -function runTimeout(fun) { - if (cachedSetTimeout === setTimeout) { - //normal enviroments in sane situations - return setTimeout(fun, 0); - } - // if setTimeout wasn't available but was latter defined - if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) { - cachedSetTimeout = setTimeout; - return setTimeout(fun, 0); - } - try { - // when when somebody has screwed with setTimeout but no I.E. maddness - return cachedSetTimeout(fun, 0); - } catch(e){ - try { - // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally - return cachedSetTimeout.call(null, fun, 0); - } catch(e){ - // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error - return cachedSetTimeout.call(this, fun, 0); - } - } - - -} -function runClearTimeout(marker) { - if (cachedClearTimeout === clearTimeout) { - //normal enviroments in sane situations - return clearTimeout(marker); - } - // if clearTimeout wasn't available but was latter defined - if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) { - cachedClearTimeout = clearTimeout; - return clearTimeout(marker); - } - try { - // when when somebody has screwed with setTimeout but no I.E. maddness - return cachedClearTimeout(marker); - } catch (e){ - try { - // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally - return cachedClearTimeout.call(null, marker); - } catch (e){ - // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error. - // Some versions of I.E. have different rules for clearTimeout vs setTimeout - return cachedClearTimeout.call(this, marker); - } - } - - - -} -var queue = []; -var draining = false; -var currentQueue; -var queueIndex = -1; - -function cleanUpNextTick() { - if (!draining || !currentQueue) { - return; - } - draining = false; - if (currentQueue.length) { - queue = currentQueue.concat(queue); - } else { - queueIndex = -1; - } - if (queue.length) { - drainQueue(); - } -} - -function drainQueue() { - if (draining) { - return; - } - var timeout = runTimeout(cleanUpNextTick); - draining = true; - - var len = queue.length; - while(len) { - currentQueue = queue; - queue = []; - while (++queueIndex < len) { - if (currentQueue) { - currentQueue[queueIndex].run(); + process.nextTick = function (fun) { + var args = new Array(arguments.length - 1); + if (arguments.length > 1) { + for (var i = 1; i < arguments.length; i++) { + args[i - 1] = arguments[i]; + } } - } - queueIndex = -1; - len = queue.length; - } - currentQueue = null; - draining = false; - runClearTimeout(timeout); -} - -process.nextTick = function (fun) { - var args = new Array(arguments.length - 1); - if (arguments.length > 1) { - for (var i = 1; i < arguments.length; i++) { - args[i - 1] = arguments[i]; - } - } - queue.push(new Item(fun, args)); - if (queue.length === 1 && !draining) { - runTimeout(drainQueue); - } -}; - -// v8 likes predictible objects -function Item(fun, array) { - this.fun = fun; - this.array = array; -} -Item.prototype.run = function () { - this.fun.apply(null, this.array); -}; -process.title = 'browser'; -process.browser = true; -process.env = {}; -process.argv = []; -process.version = ''; // empty string to avoid regexp issues -process.versions = {}; - -function noop() {} - -process.on = noop; -process.addListener = noop; -process.once = noop; -process.off = noop; -process.removeListener = noop; -process.removeAllListeners = noop; -process.emit = noop; -process.prependListener = noop; -process.prependOnceListener = noop; - -process.listeners = function (name) { return [] } - -process.binding = function (name) { - throw new Error('process.binding is not supported'); -}; - -process.cwd = function () { return '/' }; -process.chdir = function (dir) { - throw new Error('process.chdir is not supported'); -}; -process.umask = function() { return 0; }; - -},{}],9:[function(require,module,exports){ -'use strict'; - -module.exports = function () { - throw new Error( - 'ws does not work in the browser. Browser clients must use the native ' + - 'WebSocket object' - ); -}; - -},{}]},{},[7])(7) + queue.push(new Item(fun, args)); + if (queue.length === 1 && !draining) { + runTimeout(drainQueue); + } + }; + + // v8 likes predictible objects + function Item(fun, array) { + this.fun = fun; + this.array = array; + } + Item.prototype.run = function () { + this.fun.apply(null, this.array); + }; + process.title = 'browser'; + process.browser = true; + process.env = {}; + process.argv = []; + process.version = ''; // empty string to avoid regexp issues + process.versions = {}; + + function noop() {} + + process.on = noop; + process.addListener = noop; + process.once = noop; + process.off = noop; + process.removeListener = noop; + process.removeAllListeners = noop; + process.emit = noop; + process.prependListener = noop; + process.prependOnceListener = noop; + + process.listeners = function (name) { + return []; + }; + + process.binding = function (name) { + throw new Error('process.binding is not supported'); + }; + + process.cwd = function () { + return '/'; + }; + process.chdir = function (dir) { + throw new Error('process.chdir is not supported'); + }; + process.umask = function () { + return 0; + }; + }, + {} + ], + 9: [ + function (require, module, exports) { + 'use strict'; + + module.exports = function () { + throw new Error('ws does not work in the browser. Browser clients must use the native ' + 'WebSocket object'); + }; + }, + {} + ] + }, + {}, + [7] + )(7); }); //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm5vZGVfbW9kdWxlcy9icm93c2VyLXBhY2svX3ByZWx1ZGUuanMiLCJsaWIvY2xpZW50L2hhbmRsZXJzLmpzIiwibGliL2NsaWVudC9oYW5kbGVycy9jcnlwdG9fcHJvdmlkZXIuanMiLCJsaWIvY2xpZW50L2hhbmRsZXJzL2N1c3RvbS5qcyIsImxpYi9jbGllbnQvaGFuZGxlcnMvaW5pdGlhbGl6YXRpb24uanMiLCJsaWIvY2xpZW50L2hhbmRsZXJzL3NoYXJpbmcuanMiLCJsaWIvY29tbW9uL2xpbmtlZGxpc3QuanMiLCJsaWIvZXh0L2ppZmYtY2xpZW50LXdlYnNvY2tldHMuanMiLCJub2RlX21vZHVsZXMvcHJvY2Vzcy9icm93c2VyLmpzIiwibm9kZV9tb2R1bGVzL3dzL2Jyb3dzZXIuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7QUNBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQ25CQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FDdENBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUNuQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUNySUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUM1RUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7OztBQzlIQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7OztBQy9WQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQ3hMQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJmaWxlIjoiZ2VuZXJhdGVkLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXNDb250ZW50IjpbIihmdW5jdGlvbigpe2Z1bmN0aW9uIHIoZSxuLHQpe2Z1bmN0aW9uIG8oaSxmKXtpZighbltpXSl7aWYoIWVbaV0pe3ZhciBjPVwiZnVuY3Rpb25cIj09dHlwZW9mIHJlcXVpcmUmJnJlcXVpcmU7aWYoIWYmJmMpcmV0dXJuIGMoaSwhMCk7aWYodSlyZXR1cm4gdShpLCEwKTt2YXIgYT1uZXcgRXJyb3IoXCJDYW5ub3QgZmluZCBtb2R1bGUgJ1wiK2krXCInXCIpO3Rocm93IGEuY29kZT1cIk1PRFVMRV9OT1RfRk9VTkRcIixhfXZhciBwPW5baV09e2V4cG9ydHM6e319O2VbaV1bMF0uY2FsbChwLmV4cG9ydHMsZnVuY3Rpb24ocil7dmFyIG49ZVtpXVsxXVtyXTtyZXR1cm4gbyhufHxyKX0scCxwLmV4cG9ydHMscixlLG4sdCl9cmV0dXJuIG5baV0uZXhwb3J0c31mb3IodmFyIHU9XCJmdW5jdGlvblwiPT10eXBlb2YgcmVxdWlyZSYmcmVxdWlyZSxpPTA7aTx0Lmxlbmd0aDtpKyspbyh0W2ldKTtyZXR1cm4gb31yZXR1cm4gcn0pKCkiLCJ2YXIgaW5pdGlhbGl6YXRpb25IYW5kbGVycyA9IHJlcXVpcmUoJy4vaGFuZGxlcnMvaW5pdGlhbGl6YXRpb24uanMnKTtcbnZhciBzaGFyZUhhbmRsZXJzID0gcmVxdWlyZSgnLi9oYW5kbGVycy9zaGFyaW5nLmpzJyk7XG52YXIgY3VzdG9tSGFuZGxlcnMgPSByZXF1aXJlKCcuL2hhbmRsZXJzL2N1c3RvbS5qcycpO1xudmFyIGNyeXB0b1Byb3ZpZGVySGFuZGxlcnMgPSByZXF1aXJlKCcuL2hhbmRsZXJzL2NyeXB0b19wcm92aWRlci5qcycpO1xuXG4vKipcbiAqIENvbnRhaW5zIGhhbmRsZXJzIGZvciBjb21tdW5pY2F0aW9uIGV2ZW50c1xuICogQG5hbWUgaGFuZGxlcnNcbiAqIEBhbGlhcyBoYW5kbGVyc1xuICogQG5hbWVzcGFjZVxuICovXG5cbi8vIEFkZCBoYW5kbGVycyBpbXBsZW1lbnRhdGlvbnNcbm1vZHVsZS5leHBvcnRzID0gZnVuY3Rpb24gKGppZmZDbGllbnQpIHtcbiAgLy8gZmlsbCBpbiBoYW5kbGVyc1xuICBpbml0aWFsaXphdGlvbkhhbmRsZXJzKGppZmZDbGllbnQpO1xuICBzaGFyZUhhbmRsZXJzKGppZmZDbGllbnQpO1xuICBjdXN0b21IYW5kbGVycyhqaWZmQ2xpZW50KTtcbiAgY3J5cHRvUHJvdmlkZXJIYW5kbGVycyhqaWZmQ2xpZW50KTtcbn07IiwiLy8gc2V0dXAgaGFuZGxlciBmb3IgcmVjZWl2aW5nIG1lc3NhZ2VzIGZyb20gdGhlIGNyeXB0byBwcm92aWRlclxubW9kdWxlLmV4cG9ydHMgPSBmdW5jdGlvbiAoamlmZkNsaWVudCkge1xuICAvKipcbiAgICogUGFyc2UgY3J5cHRvIHByb3ZpZGVyIG1lc3NhZ2UgYW5kIHJlc29sdmUgYXNzb2NpYXRlZCBwcm9taXNlLlxuICAgKiBAbWV0aG9kXG4gICAqIEBtZW1iZXJvZiBoYW5kbGVyc1xuICAgKiBAcGFyYW0ge29iamVjdH0ganNvbl9tc2cgLSB0aGUgcGFyc2VkIGpzb24gbWVzc2FnZSBhcyByZWNlaXZlZCBieSB0aGUgY3J5cHRvX3Byb3ZpZGVyIGV2ZW50LCBjb250YWlucyAndmFsdWVzJyBhbmQgJ3NoYXJlcycgYXR0cmlidXRlcy5cbiAgICovXG4gIGppZmZDbGllbnQuaGFuZGxlcnMucmVjZWl2ZV9jcnlwdG9fcHJvdmlkZXIgPSBmdW5jdGlvbiAoanNvbl9tc2cpIHtcbiAgICAvLyBIb29rXG4gICAganNvbl9tc2cgPSBqaWZmQ2xpZW50Lmhvb2tzLmV4ZWN1dGVfYXJyYXlfaG9va3MoJ2FmdGVyT3BlcmF0aW9uJywgW2ppZmZDbGllbnQsICdjcnlwdG9fcHJvdmlkZXInLCBqc29uX21zZ10sIDIpO1xuXG4gICAgdmFyIG9wX2lkID0ganNvbl9tc2dbJ29wX2lkJ107XG4gICAgaWYgKGppZmZDbGllbnQuZGVmZXJyZWRzW29wX2lkXSA9PSBudWxsKSB7XG4gICAgICByZXR1cm47IC8vIGR1cGxpY2F0ZSBtZXNzYWdlOiBpZ25vcmVcbiAgICB9XG5cbiAgICAvLyBwYXJzZSBtc2dcbiAgICB2YXIgcmVjZWl2ZXJzX2xpc3QgPSBqc29uX21zZ1sncmVjZWl2ZXJzJ107XG4gICAgdmFyIHRocmVzaG9sZCA9IGpzb25fbXNnWyd0aHJlc2hvbGQnXTtcbiAgICB2YXIgWnAgPSBqc29uX21zZ1snWnAnXTtcblxuICAgIC8vIGNvbnN0cnVjdCBzZWNyZXQgc2hhcmUgb2JqZWN0c1xuICAgIHZhciByZXN1bHQgPSB7fTtcbiAgICBpZiAoanNvbl9tc2dbJ3ZhbHVlcyddICE9IG51bGwpIHtcbiAgICAgIHJlc3VsdC52YWx1ZXMgPSBqc29uX21zZ1sndmFsdWVzJ107XG4gICAgfVxuICAgIGlmIChqc29uX21zZ1snc2hhcmVzJ10gIT0gbnVsbCkge1xuICAgICAgcmVzdWx0LnNoYXJlcyA9IFtdO1xuICAgICAgZm9yICh2YXIgaSA9IDA7IGkgPCBqc29uX21zZ1snc2hhcmVzJ10ubGVuZ3RoOyBpKyspIHtcbiAgICAgICAgcmVzdWx0LnNoYXJlcy5wdXNoKG5ldyBqaWZmQ2xpZW50LlNlY3JldFNoYXJlKGpzb25fbXNnWydzaGFyZXMnXVtpXSwgcmVjZWl2ZXJzX2xpc3QsIHRocmVzaG9sZCwgWnApKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyByZXNvbHZlIGRlZmVycmVkXG4gICAgamlmZkNsaWVudC5kZWZlcnJlZHNbb3BfaWRdLnJlc29sdmUocmVzdWx0KTtcbiAgICBkZWxldGUgamlmZkNsaWVudC5kZWZlcnJlZHNbb3BfaWRdO1xuICB9O1xufTsiLCJtb2R1bGUuZXhwb3J0cyA9IGZ1bmN0aW9uIChqaWZmQ2xpZW50KSB7XG4gIC8qKlxuICAgKiBDYWxsZWQgd2hlbiB0aGlzIHBhcnR5IHJlY2VpdmVzIGEgY3VzdG9tIHRhZyBtZXNzYWdlIGZyb20gYW55IHBhcnR5IChpbmNsdWRpbmcgaXRzZWxmKS5cbiAgICogSWYgYSBjdXN0b20gbGlzdGVuZXIgd2FzIHNldHVwIHRvIGxpc3RlbiB0byB0aGUgdGFnLCB0aGUgbWVzc2FnZSBpcyBwYXNzZWQgdG8gdGhlIGxpc3RlbmVyLlxuICAgKiBPdGhlcndpc2UsIHRoZSBtZXNzYWdlIGlzIHN0b3JlZCB1bnRpbCBzdWNoIGEgbGlzdGVuZXIgaXMgcHJvdmlkZWQuXG4gICAqIEBtZXRob2RcbiAgICogQG1lbWJlcm9mIGhhbmRsZXJzXG4gICAqIEBwYXJhbSB7b2JqZWN0fSBqc29uX21zZyAtIHRoZSBwYXJzZWQganNvbiBtZXNzYWdlIGFzIHJlY2VpdmVkIGJ5IHRoZSBjdXN0b20gZXZlbnQuXG4gICAqL1xuICBqaWZmQ2xpZW50LmhhbmRsZXJzLnJlY2VpdmVfY3VzdG9tID0gZnVuY3Rpb24gKGpzb25fbXNnKSB7XG4gICAgaWYgKGpzb25fbXNnWydwYXJ0eV9pZCddICE9PSBqaWZmQ2xpZW50LmlkKSB7XG4gICAgICBpZiAoanNvbl9tc2dbJ2VuY3J5cHRlZCddID09PSB0cnVlKSB7XG4gICAgICAgIGpzb25fbXNnWydtZXNzYWdlJ10gPSBqaWZmQ2xpZW50Lmhvb2tzLmRlY3J5cHRTaWduKGppZmZDbGllbnQsIGpzb25fbXNnWydtZXNzYWdlJ10sIGppZmZDbGllbnQuc2VjcmV0X2tleSwgamlmZkNsaWVudC5rZXltYXBbanNvbl9tc2dbJ3BhcnR5X2lkJ11dKTtcbiAgICAgIH1cblxuICAgICAganNvbl9tc2cgPSBqaWZmQ2xpZW50Lmhvb2tzLmV4ZWN1dGVfYXJyYXlfaG9va3MoJ2FmdGVyT3BlcmF0aW9uJywgW2ppZmZDbGllbnQsICdjdXN0b20nLCBqc29uX21zZ10sIDIpO1xuICAgIH1cblxuICAgIHZhciBzZW5kZXJfaWQgPSBqc29uX21zZ1sncGFydHlfaWQnXTtcbiAgICB2YXIgdGFnID0ganNvbl9tc2dbJ3RhZyddO1xuICAgIHZhciBtZXNzYWdlID0ganNvbl9tc2dbJ21lc3NhZ2UnXTtcblxuICAgIGlmIChqaWZmQ2xpZW50Lmxpc3RlbmVyc1t0YWddICE9IG51bGwpIHtcbiAgICAgIGppZmZDbGllbnQubGlzdGVuZXJzW3RhZ10oc2VuZGVyX2lkLCBtZXNzYWdlKTtcbiAgICB9IGVsc2Uge1xuICAgICAgLy8gU3RvcmUgbWVzc2FnZSB1bnRpbCBsaXN0ZW5lciBpcyBwcm92aWRlZFxuICAgICAgdmFyIHN0b3JlZF9tZXNzYWdlcyA9IGppZmZDbGllbnQuY3VzdG9tX21lc3NhZ2VzX21haWxib3hbdGFnXTtcbiAgICAgIGlmIChzdG9yZWRfbWVzc2FnZXMgPT0gbnVsbCkge1xuICAgICAgICBzdG9yZWRfbWVzc2FnZXMgPSBbXTtcbiAgICAgICAgamlmZkNsaWVudC5jdXN0b21fbWVzc2FnZXNfbWFpbGJveFt0YWddID0gc3RvcmVkX21lc3NhZ2VzO1xuICAgICAgfVxuXG4gICAgICBzdG9yZWRfbWVzc2FnZXMucHVzaCh7c2VuZGVyX2lkOiBzZW5kZXJfaWQsIG1lc3NhZ2U6IG1lc3NhZ2V9KTtcbiAgICB9XG4gIH1cbn07IiwiLy8gYWRkIGhhbmRsZXJzIGZvciBpbml0aWFsaXphdGlvblxubW9kdWxlLmV4cG9ydHMgPSBmdW5jdGlvbiAoamlmZkNsaWVudCkge1xuICBqaWZmQ2xpZW50Lm9wdGlvbnMuaW5pdGlhbGl6YXRpb24gPSBPYmplY3QuYXNzaWduKHt9LCBqaWZmQ2xpZW50Lm9wdGlvbnMuaW5pdGlhbGl6YXRpb24pO1xuXG4gIC8qKlxuICAgKiBDYWxsZWQgd2hlbiBhbiBlcnJvciBvY2N1cnNcbiAgICogQG1ldGhvZFxuICAgKiBAbWVtYmVyb2YgaGFuZGxlcnNcbiAgICogQHBhcmFtIHtzdHJpbmd9IGxhYmVsIC0gdGhlIG5hbWUgb2YgbWVzc2FnZSBvciBvcGVyYXRpb24gY2F1c2luZyB0aGUgZXJyb3JcbiAgICogQHBhcmFtIHtlcnJvcnxzdHJpbmd9IGVycm9yIC0gdGhlIGVycm9yXG4gICAqL1xuICBqaWZmQ2xpZW50LmhhbmRsZXJzLmVycm9yID0gZnVuY3Rpb24gKGxhYmVsLCBlcnJvcikge1xuICAgIGlmIChqaWZmQ2xpZW50Lm9wdGlvbnMub25FcnJvcikge1xuICAgICAgamlmZkNsaWVudC5vcHRpb25zLm9uRXJyb3IobGFiZWwsIGVycm9yKTtcbiAgICB9XG5cbiAgICBjb25zb2xlLmxvZyhqaWZmQ2xpZW50LmlkLCAnOicsICdFcnJvciBmcm9tIHNlcnZlcjonLCBsYWJlbCwgJy0tLScsIGVycm9yKTsgLy8gVE9ETzogcmVtb3ZlIGRlYnVnZ2luZ1xuICAgIGlmIChsYWJlbCA9PT0gJ2luaXRpYWxpemF0aW9uJykge1xuICAgICAgamlmZkNsaWVudC5zb2NrZXQuZGlzY29ubmVjdCgpO1xuXG4gICAgICBpZiAoamlmZkNsaWVudC5pbml0aWFsaXphdGlvbl9jb3VudGVyIDwgamlmZkNsaWVudC5vcHRpb25zLm1heEluaXRpYWxpemF0aW9uUmV0cmllcykge1xuICAgICAgICBjb25zb2xlLmxvZyhqaWZmQ2xpZW50LmlkLCAnOicsICdyZWNvbm5lY3RpbmcuLicpOyAvLyBUT0RPOiByZW1vdmUgZGVidWdnaW5nXG4gICAgICAgIHNldFRpbWVvdXQoamlmZkNsaWVudC5jb25uZWN0LCBqaWZmQ2xpZW50Lm9wdGlvbnMuc29ja2V0T3B0aW9ucy5yZWNvbm5lY3Rpb25EZWxheSk7XG4gICAgICB9XG4gICAgfVxuICB9O1xuXG4gIC8qKlxuICAgKiBCdWlsZHMgdGhlIGluaXRpYWxpemF0aW9uIG1lc3NhZ2UgZm9yIHRoaXMgaW5zdGFuY2VcbiAgICogQG1ldGhvZFxuICAgKiBAbWVtYmVyb2YgaGFuZGxlcnNcbiAgICogQHJldHVybiB7T2JqZWN0fVxuICAgKi9cbiAgamlmZkNsaWVudC5oYW5kbGVycy5idWlsZF9pbml0aWFsaXphdGlvbl9tZXNzYWdlID0gZnVuY3Rpb24gKCkge1xuICAgIHZhciBtc2cgPSB7XG4gICAgICBjb21wdXRhdGlvbl9pZDogamlmZkNsaWVudC5jb21wdXRhdGlvbl9pZCxcbiAgICAgIHBhcnR5X2lkOiBqaWZmQ2xpZW50LmlkLFxuICAgICAgcGFydHlfY291bnQ6IGppZmZDbGllbnQucGFydHlfY291bnQsXG4gICAgICBwdWJsaWNfa2V5OiBqaWZmQ2xpZW50LnB1YmxpY19rZXkgIT0gbnVsbCA/IGppZmZDbGllbnQuaG9va3MuZHVtcEtleShqaWZmQ2xpZW50LCBqaWZmQ2xpZW50LnB1YmxpY19rZXkpIDogdW5kZWZpbmVkXG4gICAgfTtcbiAgICBtc2cgPSBPYmplY3QuYXNzaWduKG1zZywgamlmZkNsaWVudC5vcHRpb25zLmluaXRpYWxpemF0aW9uKTtcblxuICAgIC8vIEluaXRpYWxpemF0aW9uIEhvb2tcbiAgICByZXR1cm4gamlmZkNsaWVudC5ob29rcy5leGVjdXRlX2FycmF5X2hvb2tzKCdiZWZvcmVPcGVyYXRpb24nLCBbamlmZkNsaWVudCwgJ2luaXRpYWxpemF0aW9uJywgbXNnXSwgMik7XG4gIH07XG5cbiAgLyoqXG4gICAqIEJlZ2lucyBpbml0aWFsaXphdGlvbiBvZiB0aGlzIGluc3RhbmNlIGJ5IHNlbmRpbmcgdGhlIGluaXRpYWxpemF0aW9uIG1lc3NhZ2UgdG8gdGhlIHNlcnZlci5cbiAgICogU2hvdWxkIG9ubHkgYmUgY2FsbGVkIGFmdGVyIGNvbm5lY3Rpb24gaXMgZXN0YWJsaXNoZWQuXG4gICAqIERvIG5vdCBjYWxsIHRoaXMgbWFudWFsbHkgdW5sZXNzIHlvdSBrbm93IHdoYXQgeW91IGFyZSBkb2luZywgdXNlIDxqaWZmX2luc3RhbmNlPi5jb25uZWN0KCkgaW5zdGVhZCFcbiAgICogQG1ldGhvZFxuICAgKiBAbWVtYmVyb2YgaGFuZGxlcnNcbiAgICovXG4gIGppZmZDbGllbnQuaGFuZGxlcnMuY29ubmVjdGVkID0gZnVuY3Rpb24gKCkge1xuICAgIGNvbnNvbGUubG9nKCdDb25uZWN0ZWQhJywgamlmZkNsaWVudC5pZCk7IC8vIFRPRE86IHJlbW92ZSBkZWJ1Z2dpbmdcbiAgICBqaWZmQ2xpZW50LmluaXRpYWxpemF0aW9uX2NvdW50ZXIrKztcblxuICAgIGlmIChqaWZmQ2xpZW50LnNlY3JldF9rZXkgPT0gbnVsbCAmJiBqaWZmQ2xpZW50LnB1YmxpY19rZXkgPT0gbnVsbCkge1xuICAgICAgdmFyIGtleSA9IGppZmZDbGllbnQuaG9va3MuZ2VuZXJhdGVLZXlQYWlyKGppZmZDbGllbnQpO1xuICAgICAgamlmZkNsaWVudC5zZWNyZXRfa2V5ID0ga2V5LnNlY3JldF9rZXk7XG4gICAgICBqaWZmQ2xpZW50LnB1YmxpY19rZXkgPSBrZXkucHVibGljX2tleTtcbiAgICB9XG5cbiAgICAvLyBJbml0aWFsaXphdGlvbiBtZXNzYWdlXG4gICAgdmFyIG1zZyA9IGppZmZDbGllbnQuaGFuZGxlcnMuYnVpbGRfaW5pdGlhbGl6YXRpb25fbWVzc2FnZSgpO1xuXG4gICAgLy8gRW1pdCBpbml0aWFsaXphdGlvbiBtZXNzYWdlIHRvIHNlcnZlclxuICAgIGppZmZDbGllbnQuc29ja2V0LmVtaXQoJ2luaXRpYWxpemF0aW9uJywgSlNPTi5zdHJpbmdpZnkobXNnKSk7XG4gIH07XG5cbiAgLyoqXG4gICAqIENhbGxlZCBhZnRlciB0aGUgc2VydmVyIGFwcHJvdmVzIGluaXRpYWxpemF0aW9uIG9mIHRoaXMgaW5zdGFuY2UuXG4gICAqIFNldHMgdGhlIGluc3RhbmNlIGlkLCB0aGUgY291bnQgb2YgcGFydGllcyBpbiB0aGUgY29tcHV0YXRpb24sIGFuZCB0aGUgcHVibGljIGtleXNcbiAgICogb2YgaW5pdGlhbGl6ZWQgcGFydGllcy5cbiAgICogQG1ldGhvZFxuICAgKiBAbWVtYmVyb2YgaGFuZGxlcnNcbiAgICovXG4gIGppZmZDbGllbnQuaGFuZGxlcnMuaW5pdGlhbGl6ZWQgPSBmdW5jdGlvbiAobXNnKSB7XG4gICAgamlmZkNsaWVudC5fX2luaXRpYWxpemVkID0gdHJ1ZTtcbiAgICBqaWZmQ2xpZW50LmluaXRpYWxpemF0aW9uX2NvdW50ZXIgPSAwO1xuXG4gICAgbXNnID0gSlNPTi5wYXJzZShtc2cpO1xuICAgIG1zZyA9IGppZmZDbGllbnQuaG9va3MuZXhlY3V0ZV9hcnJheV9ob29rcygnYWZ0ZXJPcGVyYXRpb24nLCBbamlmZkNsaWVudCwgJ2luaXRpYWxpemF0aW9uJywgbXNnXSwgMik7XG5cbiAgICBqaWZmQ2xpZW50LmlkID0gbXNnLnBhcnR5X2lkO1xuICAgIGppZmZDbGllbnQucGFydHlfY291bnQgPSBtc2cucGFydHlfY291bnQ7XG5cbiAgICAvLyBOb3c6ICgxKSB0aGlzIHBhcnR5IGlzIGNvbm5lY3QgKDIpIHNlcnZlciAoYW5kIG90aGVyIHBhcnRpZXMpIGtub3cgdGhpcyBwdWJsaWMga2V5XG4gICAgLy8gUmVzZW5kIGFsbCBwZW5kaW5nIG1lc3NhZ2VzXG4gICAgamlmZkNsaWVudC5zb2NrZXQucmVzZW5kX21haWxib3goKTtcblxuICAgIC8vIHN0b3JlIHRoZSByZWNlaXZlZCBwdWJsaWMga2V5cyBhbmQgcmVzb2x2ZSB3YWl0IGNhbGxiYWNrc1xuICAgIGppZmZDbGllbnQuaGFuZGxlcnMuc3RvcmVfcHVibGljX2tleXMobXNnLnB1YmxpY19rZXlzKTtcbiAgfTtcblxuICAvKipcbiAgICogUGFyc2UgYW5kIHN0b3JlIHRoZSBnaXZlbiBwdWJsaWMga2V5c1xuICAgKiBAbWV0aG9kXG4gICAqIEBtZW1iZXJvZiBoYW5kbGVyc1xuICAgKiBAcGFyYW0ge29iamVjdH0ga2V5bWFwIC0gbWFwcyBwYXJ0eSBpZCB0byBzZXJpYWxpemVkIHB1YmxpYyBrZXkuXG4gICAqL1xuICBqaWZmQ2xpZW50LmhhbmRsZXJzLnN0b3JlX3B1YmxpY19rZXlzID0gZnVuY3Rpb24gKGtleW1hcCkge1xuICAgIHZhciBpO1xuICAgIGZvciAoaSBpbiBrZXltYXApIHtcbiAgICAgIGlmIChrZXltYXAuaGFzT3duUHJvcGVydHkoaSkgJiYgamlmZkNsaWVudC5rZXltYXBbaV0gPT0gbnVsbCkge1xuICAgICAgICBqaWZmQ2xpZW50LmtleW1hcFtpXSA9IGppZmZDbGllbnQuaG9va3MucGFyc2VLZXkoamlmZkNsaWVudCwga2V5bWFwW2ldKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBSZXNvbHZlIGFueSBwZW5kaW5nIG1lc3NhZ2VzIHRoYXQgd2VyZSByZWNlaXZlZCBiZWZvcmUgdGhlIHNlbmRlcidzIHB1YmxpYyBrZXkgd2FzIGtub3duXG4gICAgamlmZkNsaWVudC5yZXNvbHZlX21lc3NhZ2VzX3dhaXRpbmdfZm9yX2tleXMoKTtcblxuICAgIC8vIFJlc29sdmUgYW55IHBlbmRpbmcgd2FpdHMgdGhhdCBoYXZlIHNhdGlzZmllZCBjb25kaXRpb25zXG4gICAgamlmZkNsaWVudC5leGVjdXRlX3dhaXRfY2FsbGJhY2tzKCk7XG5cbiAgICAvLyBDaGVjayBpZiBhbGwga2V5cyBoYXZlIGJlZW4gcmVjZWl2ZWRcbiAgICBpZiAoamlmZkNsaWVudC5rZXltYXBbJ3MxJ10gPT0gbnVsbCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBmb3IgKGkgPSAxOyBpIDw9IGppZmZDbGllbnQucGFydHlfY291bnQ7IGkrKykge1xuICAgICAgaWYgKGppZmZDbGllbnQua2V5bWFwW2ldID09IG51bGwpIHtcbiAgICAgICAgcmV0dXJuO1xuICAgICAgfVxuICAgIH1cblxuICAgIC8vIGFsbCBwYXJ0aWVzIGFyZSBjb25uZWN0ZWQ7IGV4ZWN1dGUgY2FsbGJhY2tcbiAgICBpZiAoamlmZkNsaWVudC5fX3JlYWR5ICE9PSB0cnVlICYmIGppZmZDbGllbnQuX19pbml0aWFsaXplZCkge1xuICAgICAgamlmZkNsaWVudC5fX3JlYWR5ID0gdHJ1ZTtcbiAgICAgIGlmIChqaWZmQ2xpZW50Lm9wdGlvbnMub25Db25uZWN0ICE9IG51bGwpIHtcbiAgICAgICAgamlmZkNsaWVudC5vcHRpb25zLm9uQ29ubmVjdChqaWZmQ2xpZW50KTtcbiAgICAgIH1cbiAgICB9XG4gIH07XG59OyIsIi8vIGFkZHMgc2hhcmluZyByZWxhdGVkIGhhbmRsZXJzXG5tb2R1bGUuZXhwb3J0cyA9IGZ1bmN0aW9uIChqaWZmQ2xpZW50KSB7XG4gIC8qKlxuICAgKiBTdG9yZSB0aGUgcmVjZWl2ZWQgc2hhcmUgYW5kIHJlc29sdmVzIHRoZSBjb3JyZXNwb25kaW5nXG4gICAqIGRlZmVycmVkIGlmIG5lZWRlZC5cbiAgICogQG1ldGhvZFxuICAgKiBAbWVtYmVyb2YgaGFuZGxlcnNcbiAgICogQHBhcmFtIHtvYmplY3R9IGpzb25fbXNnIC0gdGhlIHBhcnNlZCBqc29uIG1lc3NhZ2UgYXMgcmVjZWl2ZWQuXG4gICAqL1xuICBqaWZmQ2xpZW50LmhhbmRsZXJzLnJlY2VpdmVfc2hhcmUgPSBmdW5jdGlvbiAoanNvbl9tc2cpIHtcbiAgICAvLyBEZWNyeXB0IHNoYXJlXG4gICAganNvbl9tc2dbJ3NoYXJlJ10gPSBqaWZmQ2xpZW50Lmhvb2tzLmRlY3J5cHRTaWduKGppZmZDbGllbnQsIGpzb25fbXNnWydzaGFyZSddLCBqaWZmQ2xpZW50LnNlY3JldF9rZXksIGppZmZDbGllbnQua2V5bWFwW2pzb25fbXNnWydwYXJ0eV9pZCddXSk7XG4gICAganNvbl9tc2cgPSBqaWZmQ2xpZW50Lmhvb2tzLmV4ZWN1dGVfYXJyYXlfaG9va3MoJ2FmdGVyT3BlcmF0aW9uJywgW2ppZmZDbGllbnQsICdzaGFyZScsIGpzb25fbXNnXSwgMik7XG5cbiAgICB2YXIgc2VuZGVyX2lkID0ganNvbl9tc2dbJ3BhcnR5X2lkJ107XG4gICAgdmFyIG9wX2lkID0ganNvbl9tc2dbJ29wX2lkJ107XG4gICAgdmFyIHNoYXJlID0ganNvbl9tc2dbJ3NoYXJlJ107XG5cbiAgICAvLyBDYWxsIGhvb2tcbiAgICBzaGFyZSA9IGppZmZDbGllbnQuaG9va3MuZXhlY3V0ZV9hcnJheV9ob29rcygncmVjZWl2ZVNoYXJlJywgW2ppZmZDbGllbnQsIHNlbmRlcl9pZCwgc2hhcmVdLCAyKTtcblxuICAgIC8vIGNoZWNrIGlmIGEgZGVmZXJyZWQgaXMgc2V0IHVwIChtYXliZSB0aGUgc2hhcmUgd2FzIHJlY2VpdmVkIGVhcmx5KVxuICAgIGlmIChqaWZmQ2xpZW50LmRlZmVycmVkc1tvcF9pZF0gPT0gbnVsbCkge1xuICAgICAgamlmZkNsaWVudC5kZWZlcnJlZHNbb3BfaWRdID0ge307XG4gICAgfVxuICAgIGlmIChqaWZmQ2xpZW50LmRlZmVycmVkc1tvcF9pZF1bc2VuZGVyX2lkXSA9PSBudWxsKSB7XG4gICAgICAvLyBTaGFyZSBpcyByZWNlaXZlZCBiZWZvcmUgZGVmZXJyZWQgd2FzIHNldHVwLCBzdG9yZSBpdC5cbiAgICAgIGppZmZDbGllbnQuZGVmZXJyZWRzW29wX2lkXVtzZW5kZXJfaWRdID0gbmV3IGppZmZDbGllbnQuaGVscGVycy5EZWZlcnJlZCgpO1xuICAgIH1cblxuICAgIC8vIERlZmVycmVkIGlzIGFscmVhZHkgc2V0dXAsIHJlc29sdmUgaXQuXG4gICAgamlmZkNsaWVudC5kZWZlcnJlZHNbb3BfaWRdW3NlbmRlcl9pZF0ucmVzb2x2ZShzaGFyZSk7XG4gIH07XG5cbiAgLyoqXG4gICAqIFJlc29sdmVzIHRoZSBkZWZlcnJlZCBjb3JyZXNwb25kaW5nIHRvIG9wZXJhdGlvbl9pZCBhbmQgc2VuZGVyX2lkLlxuICAgKiBAbWV0aG9kXG4gICAqIEBtZW1iZXJvZiBoYW5kbGVyc1xuICAgKiBAcGFyYW0ge29iamVjdH0ganNvbl9tc2cgLSB0aGUganNvbiBtZXNzYWdlIGFzIHJlY2VpdmVkIHdpdGggdGhlIG9wZW4gZXZlbnQuXG4gICAqL1xuICBqaWZmQ2xpZW50LmhhbmRsZXJzLnJlY2VpdmVfb3BlbiA9IGZ1bmN0aW9uIChqc29uX21zZykge1xuICAgIC8vIERlY3J5cHQgc2hhcmVcbiAgICBpZiAoanNvbl9tc2dbJ3BhcnR5X2lkJ10gIT09IGppZmZDbGllbnQuaWQpIHtcbiAgICAgIGpzb25fbXNnWydzaGFyZSddID0gamlmZkNsaWVudC5ob29rcy5kZWNyeXB0U2lnbihqaWZmQ2xpZW50LCBqc29uX21zZ1snc2hhcmUnXSwgamlmZkNsaWVudC5zZWNyZXRfa2V5LCBqaWZmQ2xpZW50LmtleW1hcFtqc29uX21zZ1sncGFydHlfaWQnXV0pO1xuICAgICAganNvbl9tc2cgPSBqaWZmQ2xpZW50Lmhvb2tzLmV4ZWN1dGVfYXJyYXlfaG9va3MoJ2FmdGVyT3BlcmF0aW9uJywgW2ppZmZDbGllbnQsICdvcGVuJywganNvbl9tc2ddLCAyKTtcbiAgICB9XG5cbiAgICB2YXIgc2VuZGVyX2lkID0ganNvbl9tc2dbJ3BhcnR5X2lkJ107XG4gICAgdmFyIG9wX2lkID0ganNvbl9tc2dbJ29wX2lkJ107XG4gICAgdmFyIHNoYXJlID0ganNvbl9tc2dbJ3NoYXJlJ107XG4gICAgdmFyIFpwID0ganNvbl9tc2dbJ1pwJ107XG5cbiAgICAvLyBjYWxsIGhvb2tcbiAgICBzaGFyZSA9IGppZmZDbGllbnQuaG9va3MuZXhlY3V0ZV9hcnJheV9ob29rcygncmVjZWl2ZU9wZW4nLCBbamlmZkNsaWVudCwgc2VuZGVyX2lkLCBzaGFyZSwgWnBdLCAyKTtcblxuICAgIC8vIEVuc3VyZSBkZWZlcnJlZCBpcyBzZXR1cFxuICAgIGlmIChqaWZmQ2xpZW50LmRlZmVycmVkc1tvcF9pZF0gPT0gbnVsbCkge1xuICAgICAgamlmZkNsaWVudC5kZWZlcnJlZHNbb3BfaWRdID0ge307XG4gICAgfVxuICAgIGlmIChqaWZmQ2xpZW50LmRlZmVycmVkc1tvcF9pZF0uc2hhcmVzID09IG51bGwpIHtcbiAgICAgIGppZmZDbGllbnQuZGVmZXJyZWRzW29wX2lkXS5zaGFyZXMgPSBbXTtcbiAgICB9XG5cbiAgICAvLyBBY2N1bXVsYXRlIHJlY2VpdmVkIHNoYXJlc1xuICAgIGppZmZDbGllbnQuZGVmZXJyZWRzW29wX2lkXS5zaGFyZXMucHVzaCh7dmFsdWU6IHNoYXJlLCBzZW5kZXJfaWQ6IHNlbmRlcl9pZCwgWnA6IFpwfSk7XG5cbiAgICAvLyBSZXNvbHZlIHdoZW4gcmVhZHlcbiAgICBpZiAoamlmZkNsaWVudC5kZWZlcnJlZHNbb3BfaWRdLnNoYXJlcy5sZW5ndGggPT09IGppZmZDbGllbnQuZGVmZXJyZWRzW29wX2lkXS50aHJlc2hvbGQpIHtcbiAgICAgIGppZmZDbGllbnQuZGVmZXJyZWRzW29wX2lkXS5kZWZlcnJlZC5yZXNvbHZlKCk7XG4gICAgfVxuXG4gICAgLy8gQ2xlYW4gdXAgaWYgZG9uZVxuICAgIGlmIChqaWZmQ2xpZW50LmRlZmVycmVkc1tvcF9pZF0gIT0gbnVsbCAmJiBqaWZmQ2xpZW50LmRlZmVycmVkc1tvcF9pZF0uZGVmZXJyZWQgPT09ICdDTEVBTicgJiYgamlmZkNsaWVudC5kZWZlcnJlZHNbb3BfaWRdLnNoYXJlcy5sZW5ndGggPT09IGppZmZDbGllbnQuZGVmZXJyZWRzW29wX2lkXS50b3RhbCkge1xuICAgICAgZGVsZXRlIGppZmZDbGllbnQuZGVmZXJyZWRzW29wX2lkXTtcbiAgICB9XG4gIH1cbn07IiwiLyoqIERvdWJseSBsaW5rZWQgbGlzdCB3aXRoIGFkZCBhbmQgcmVtb3ZlIGZ1bmN0aW9ucyBhbmQgcG9pbnRlcnMgdG8gaGVhZCBhbmQgdGFpbCoqL1xubW9kdWxlLmV4cG9ydHMgPSBmdW5jdGlvbiAoKSB7XG4gIC8vIGF0dHJpYnV0ZXM6IGxpc3QuaGVhZCBhbmQgbGlzdC50YWlsXG4gIC8vIGZ1bmN0aW9uczogbGlzdC5hZGQob2JqZWN0KSAocmV0dXJucyBwb2ludGVyKSwgbGlzdC5yZW1vdmUocG9pbnRlcilcbiAgLy8gbGlzdC5oZWFkL2xpc3QudGFpbC9hbnkgZWxlbWVudCBjb250YWluczpcbiAgLy8gICAgbmV4dDogcG9pbnRlciB0byBuZXh0LFxuICAvLyAgICBwcmV2aW91czogcG9pbnRlciB0byBwcmV2aW91cyxcbiAgLy8gICAgb2JqZWN0OiBzdG9yZWQgb2JqZWN0LlxuICB2YXIgbGlzdCA9IHtoZWFkOiBudWxsLCB0YWlsOiBudWxsfTtcbiAgLy8gVE9ETyByZW5hbWUgdGhpcyB0byBwdXNoVGFpbCB8fCBwdXNoXG4gIGxpc3QuYWRkID0gZnVuY3Rpb24gKG9iaikge1xuICAgIHZhciBub2RlID0geyBvYmplY3Q6IG9iaiwgbmV4dDogbnVsbCwgcHJldmlvdXM6IG51bGwgfTtcbiAgICBpZiAobGlzdC5oZWFkID09IG51bGwpIHtcbiAgICAgIGxpc3QuaGVhZCA9IG5vZGU7XG4gICAgICBsaXN0LnRhaWwgPSBub2RlO1xuICAgIH0gZWxzZSB7XG4gICAgICBsaXN0LnRhaWwubmV4dCA9IG5vZGU7XG4gICAgICBub2RlLnByZXZpb3VzID0gbGlzdC50YWlsO1xuICAgICAgbGlzdC50YWlsID0gbm9kZTtcbiAgICB9XG4gICAgcmV0dXJuIG5vZGU7XG4gIH07XG5cbiAgbGlzdC5wdXNoSGVhZCA9IGZ1bmN0aW9uIChvYmopIHtcbiAgICBsaXN0LmhlYWQgPSB7b2JqZWN0OiBvYmosIG5leHQgOiBsaXN0LmhlYWQsIHByZXZpb3VzIDogbnVsbH07XG4gICAgaWYgKGxpc3QuaGVhZC5uZXh0ICE9IG51bGwpIHtcbiAgICAgIGxpc3QuaGVhZC5uZXh0LnByZXZpb3VzID0gbGlzdC5oZWFkO1xuICAgIH0gZWxzZSB7XG4gICAgICBsaXN0LnRhaWwgPSBsaXN0LmhlYWQ7XG4gICAgfVxuICB9O1xuXG4gIGxpc3QucG9wSGVhZCA9IGZ1bmN0aW9uICgpIHtcbiAgICB2YXIgcmVzdWx0ID0gbGlzdC5oZWFkO1xuICAgIGlmIChsaXN0LmhlYWQgIT0gbnVsbCkge1xuICAgICAgbGlzdC5oZWFkID0gbGlzdC5oZWFkLm5leHQ7XG4gICAgICBpZiAobGlzdC5oZWFkID09IG51bGwpIHtcbiAgICAgICAgbGlzdC50YWlsID0gbnVsbDtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGxpc3QuaGVhZC5wcmV2aW91cyAgPSBudWxsO1xuICAgICAgfVxuICAgIH1cbiAgICByZXR1cm4gcmVzdWx0O1xuICB9O1xuXG4gIC8vIG1lcmdlcyB0d28gbGlua2VkIGxpc3RzIGFuZCByZXR1cm4gYSBwb2ludGVyIHRvIHRoZSBoZWFkIG9mIHRoZSBtZXJnZWQgbGlzdFxuICAvLyB0aGUgaGVhZCB3aWxsIGJlIHRoZSBoZWFkIG9mIGxpc3QgYW5kIHRoZSB0YWlsIHRoZSB0YWlsIG9mIGwyXG4gIGxpc3QuZXh0ZW5kID0gZnVuY3Rpb24gKGwyKSB7XG4gICAgaWYgKGxpc3QuaGVhZCA9PSBudWxsKSB7XG4gICAgICByZXR1cm4gbDI7XG4gICAgfVxuICAgIGlmIChsMi5oZWFkID09IG51bGwpIHtcbiAgICAgIHJldHVybiBsaXN0O1xuICAgIH1cbiAgICBsaXN0LnRhaWwubmV4dCA9IGwyLmhlYWQ7XG4gICAgbDIuaGVhZC5wcmV2aW91cyA9IGxpc3QudGFpbDtcbiAgICBsaXN0LnRhaWwgPSBsMi50YWlsO1xuXG4gICAgcmV0dXJuIGxpc3Q7XG4gIH07XG5cbiAgbGlzdC5yZW1vdmUgPSBmdW5jdGlvbiAocHRyKSB7XG4gICAgdmFyIHByZXYgPSBwdHIucHJldmlvdXM7XG4gICAgdmFyIG5leHQgPSBwdHIubmV4dDtcblxuICAgIGlmIChwcmV2ID09IG51bGwgJiYgbGlzdC5oZWFkICE9PSBwdHIpIHtcbiAgICAgIHJldHVybjtcbiAgICB9IGVsc2UgaWYgKG5leHQgPT0gbnVsbCAmJiBsaXN0LnRhaWwgIT09IHB0cikge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGlmIChwcmV2ID09IG51bGwpIHsgLy8gcHRyIGlzIGhlYWQgKG9yIGJvdGggaGVhZCBhbmQgdGFpbClcbiAgICAgIGxpc3QuaGVhZCA9IG5leHQ7XG4gICAgICBpZiAobGlzdC5oZWFkICE9IG51bGwpIHtcbiAgICAgICAgbGlzdC5oZWFkLnByZXZpb3VzID0gbnVsbDtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGxpc3QudGFpbCA9IG51bGw7XG4gICAgICB9XG4gICAgfSBlbHNlIGlmIChuZXh0ID09IG51bGwpIHsgLy8gcHRyIGlzIHRhaWwgKGFuZCBub3QgaGVhZClcbiAgICAgIGxpc3QudGFpbCA9IHByZXY7XG4gICAgICBwcmV2Lm5leHQgPSBudWxsO1xuICAgIH0gZWxzZSB7IC8vIHB0ciBpcyBpbnNpZGVcbiAgICAgIHByZXYubmV4dCA9IG5leHQ7XG4gICAgICBuZXh0LnByZXZpb3VzID0gcHJldjtcbiAgICB9XG4gIH07XG4gIGxpc3Quc2xpY2UgPSBmdW5jdGlvbiAocHRyKSB7IC8vIHJlbW92ZSBhbGwgZWxlbWVudHMgZnJvbSBoZWFkIHRvIHB0ciAoaW5jbHVkaW5nIHB0cikuXG4gICAgaWYgKHB0ciA9PSBudWxsKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgLyogQ09OU0VSVkFUSVZFOiBtYWtlIHN1cmUgcHRyIGlzIHBhcnQgb2YgdGhlIGxpc3QgdGhlbiByZW1vdmUgKi9cbiAgICB2YXIgY3VycmVudCA9IGxpc3QuaGVhZDtcbiAgICB3aGlsZSAoY3VycmVudCAhPSBudWxsKSB7XG4gICAgICBpZiAoY3VycmVudCA9PT0gcHRyKSB7XG4gICAgICAgIGxpc3QuaGVhZCA9IHB0ci5uZXh0O1xuICAgICAgICBpZiAobGlzdC5oZWFkID09IG51bGwpIHtcbiAgICAgICAgICBsaXN0LnRhaWwgPSBudWxsO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuO1xuICAgICAgfVxuICAgICAgY3VycmVudCA9IGN1cnJlbnQubmV4dDtcbiAgICB9XG5cbiAgICAvKiBNT1JFIEFHR1JFU1NJVkUgVkVSU0lPTjogd2lsbCBiZSBpbmNvcnJlY3QgaWYgcHRyIGlzIG5vdCBpbiB0aGUgbGlzdCAqL1xuICAgIC8qXG4gICAgbGlzdC5oZWFkID0gcHRyLm5leHQ7XG4gICAgaWYgKGxpc3QuaGVhZCA9PSBudWxsKSB7XG4gICAgICBsaXN0LnRhaWwgPSBudWxsO1xuICAgIH1cbiAgICAqL1xuICB9O1xuICAvKlxuICBsaXN0Ll9kZWJ1Z19sZW5ndGggPSBmdW5jdGlvbiAoKSB7XG4gICAgdmFyIGwgPSAwO1xuICAgIHZhciBjdXJyZW50ID0gbGlzdC5oZWFkO1xuICAgIHdoaWxlIChjdXJyZW50ICE9IG51bGwpIHtcbiAgICAgIGN1cnJlbnQgPSBjdXJyZW50Lm5leHQ7XG4gICAgICBsKys7XG4gICAgfVxuICAgIHJldHVybiBsO1xuICB9O1xuICAqL1xuICByZXR1cm4gbGlzdDtcbn07XG4iLCIvKipcbiAqIFRoaXMgZGVmaW5lcyBhIGxpYnJhcnkgZXh0ZW5zaW9uIGZvciB1c2luZyB3ZWJzb2NrZXRzIHJhdGhlciB0aGFuIHNvY2tldC5pbyBmb3IgY29tbXVuaWNhdGlvbi4gVGhpc1xuICogZXh0ZW5zaW9uIHByaW1hcmlseSBlZGl0cy9vdmVyd3JpdGVzIGV4aXN0aW5nIHNvY2tldCBmdW5jdGlvbnMgdG8gdXNlIGFuZCBiZSBjb21wYXRpYmxlIHdpdGggdGhlXG4gKiB3cyBsaWJyYXJ5LlxuICogQG5hbWVzcGFjZSBqaWZmY2xpZW50X3dlYnNvY2tldHNcbiAqIEB2ZXJzaW9uIDEuMFxuICpcbiAqIFJFUVVJUkVNRU5UUzpcbiAqIFlvdSBtdXN0IGFwcGx5IHRoaXMgZXh0ZW5zaW9uIHRvIHlvdXIgY2xpZW50IGFuZCB0aGUgc2VydmVyIHlvdSdyZSBjb21tdW5pY2F0aW5nIHdpdGggbXVzdCBhcHBseSBqaWZmc2VydmVyX3dlYnNvY2tldHMuXG4gKiBXaGVuIHVzaW5nIHRoaXMgZXh0ZW5zaW9uIGluIGJyb3dzZXIsIFwiL2Rpc3QvamlmZi1jbGllbnQtd2Vic29ja2V0cy5qc1wiIG11c3QgYmUgbG9hZGVkIGluIGNsaWVudC5odG1sIGluc3RlYWQgb2YgdGhpcyBmaWxlLlxuICovXG5cblxuXG4oZnVuY3Rpb24gKGV4cG9ydHMsIG5vZGUpIHtcbiAgLyoqXG4gICAqIFRoZSBuYW1lIG9mIHRoaXMgZXh0ZW5zaW9uOiAnd2Vic29ja2V0J1xuICAgKiBAdHlwZSB7c3RyaW5nfVxuICAgKiBAbWVtYmVyT2YgamlmZmNsaWVudF93ZWJzb2NrZXRzXG4gICAqL1xuXG4gIHZhciB3cztcbiAgdmFyIGxpbmtlZExpc3Q7XG4gIHZhciBoYW5kbGVycztcblxuICBsaW5rZWRMaXN0ID0gcmVxdWlyZSgnLi4vY29tbW9uL2xpbmtlZGxpc3QuanMnKTtcbiAgaGFuZGxlcnMgPSByZXF1aXJlKCcuLi9jbGllbnQvaGFuZGxlcnMuanMnKTtcbiAgaWYgKCFwcm9jZXNzLmJyb3dzZXIpIHtcbiAgICB3cyA9IHJlcXVpcmUoJ3dzJyk7XG4gIH0gZWxzZSB7XG4gICAgaWYgKHR5cGVvZiBXZWJTb2NrZXQgIT09ICd1bmRlZmluZWQnKSB7XG4gICAgICB3cyA9IFdlYlNvY2tldFxuICAgIH0gZWxzZSBpZiAodHlwZW9mIE1veldlYlNvY2tldCAhPT0gJ3VuZGVmaW5lZCcpIHtcbiAgICAgIHdzID0gTW96V2ViU29ja2V0XG4gICAgfSBlbHNlIGlmICh0eXBlb2YgZ2xvYmFsICE9PSAndW5kZWZpbmVkJykge1xuICAgICAgd3MgPSBnbG9iYWwuV2ViU29ja2V0IHx8IGdsb2JhbC5Nb3pXZWJTb2NrZXRcbiAgICB9IGVsc2UgaWYgKHR5cGVvZiB3aW5kb3cgIT09ICd1bmRlZmluZWQnKSB7XG4gICAgICB3cyA9IHdpbmRvdy5XZWJTb2NrZXQgfHwgd2luZG93Lk1veldlYlNvY2tldFxuICAgIH0gZWxzZSBpZiAodHlwZW9mIHNlbGYgIT09ICd1bmRlZmluZWQnKSB7XG4gICAgICB3cyA9IHNlbGYuV2ViU29ja2V0IHx8IHNlbGYuTW96V2ViU29ja2V0XG4gICAgfVxuICB9XG5cblxuICAvLyBUYWtlIHRoZSBqaWZmLWNsaWVudCBiYXNlIGluc3RhbmNlIGFuZCBvcHRpb25zIGZvciB0aGlzIGV4dGVuc2lvbiwgYW5kIHVzZSB0aGVtXG4gIC8vIHRvIGNvbnN0cnVjdCBhbiBpbnN0YW5jZSBmb3IgdGhpcyBleHRlbnNpb24uXG4gIGZ1bmN0aW9uIG1ha2VfamlmZihiYXNlX2luc3RhbmNlLCBvcHRpb25zKSB7XG4gICAgdmFyIGppZmYgPSBiYXNlX2luc3RhbmNlO1xuXG4gICAgLy8gUGFyc2Ugb3B0aW9uc1xuICAgIGlmIChvcHRpb25zID09IG51bGwpIHtcbiAgICAgIG9wdGlvbnMgPSB7fTtcbiAgICB9XG5cbiAgICAvKiBGdW5jdGlvbnMgdGhhdCBvdmVyd3JpdGUgY2xpZW50L3NvY2tldC9ldmVudHMuanMgZnVuY3Rpb25hbGl0eSAqL1xuXG4gICAgLyoqXG4gICAgICogaW5pdFNvY2tldCdzICcub24nIGZ1bmN0aW9ucyBuZWVkZWQgdG8gYmUgcmVwbGFjZWQgc2luY2Ugd3MgZG9lc1xuICAgICAqIG5vdCBoYXZlIGFzIG1hbnkgcHJvdG9jb2xzLiBJbnN0ZWFkIHRoZXNlIGZ1bmN0aW9ucyBhcmUgcm91dGVkIHRvXG4gICAgICogd2hlbiBhIG1lc3NhZ2UgaXMgcmVjZWl2ZWQgYW5kIGEgcHJvdG9jb2wgaXMgbWFudWFsbHkgcGFyc2VkLlxuICAgICAqL1xuICAgIGppZmYuaW5pdFNvY2tldCA9IGZ1bmN0aW9uICgpIHtcbiAgICAgIHZhciBqaWZmQ2xpZW50ID0gdGhpcztcblxuICAgICAgLyogd3MgdXNlcyB0aGUgJ29wZW4nIHByb3RvY29sIG9uIGNvbm5lY3Rpb24uIFNob3VsZCBub3QgY29uZmxpY3Qgd2l0aCB0aGVcbiAgICAgICAgICAgSklGRiBvcGVuIHByb3RvY2wgYXMgdGhhdCB3aWxsIGJlIHNlbnQgYXMgYSBtZXNzYWdlIGFuZCB3c1xuICAgICAgICAgICB3aWxsIHNlZSBpdCBhcyBhICdtZXNzYWdlJyBwcm90b2NvbC4gKi9cbiAgICAgIHRoaXMuc29ja2V0Lm9ub3BlbiA9IGppZmZDbGllbnQuaGFuZGxlcnMuY29ubmVjdGVkO1xuXG4gICAgICAvLyBQdWJsaWMga2V5cyB3ZXJlIHVwZGF0ZWQgb24gdGhlIHNlcnZlciwgYW5kIGl0IHNlbnQgdXMgdGhlIHVwZGF0ZXNcbiAgICAgIGZ1bmN0aW9uIHB1YmxpY0tleXNDaGFuZ2VkKG1zZywgY2FsbGJhY2spIHtcblxuICAgICAgICBtc2cgPSBKU09OLnBhcnNlKG1zZyk7XG4gICAgICAgIG1zZyA9IGppZmZDbGllbnQuaG9va3MuZXhlY3V0ZV9hcnJheV9ob29rcygnYWZ0ZXJPcGVyYXRpb24nLCBbamlmZkNsaWVudCwgJ3B1YmxpY19rZXlzJywgbXNnXSwgMik7XG5cbiAgICAgICAgamlmZkNsaWVudC5oYW5kbGVycy5zdG9yZV9wdWJsaWNfa2V5cyhtc2cucHVibGljX2tleXMpO1xuICAgICAgfVxuXG4gICAgICAvLyBTZXR1cCByZWNlaXZpbmcgbWF0Y2hpbmcgc2hhcmVzXG4gICAgICBmdW5jdGlvbiBzaGFyZShtc2csIGNhbGxiYWNrKSB7XG5cbiAgICAgICAgLy8gcGFyc2UgbWVzc2FnZVxuICAgICAgICB2YXIganNvbl9tc2cgPSBKU09OLnBhcnNlKG1zZyk7XG4gICAgICAgIHZhciBzZW5kZXJfaWQgPSBqc29uX21zZ1sncGFydHlfaWQnXTtcblxuICAgICAgICBpZiAoamlmZkNsaWVudC5rZXltYXBbc2VuZGVyX2lkXSAhPSBudWxsKSB7XG4gICAgICAgICAgamlmZkNsaWVudC5oYW5kbGVycy5yZWNlaXZlX3NoYXJlKGpzb25fbXNnKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBpZiAoamlmZkNsaWVudC5tZXNzYWdlc1dhaXRpbmdLZXlzW3NlbmRlcl9pZF0gPT0gbnVsbCkge1xuICAgICAgICAgICAgamlmZkNsaWVudC5tZXNzYWdlc1dhaXRpbmdLZXlzW3NlbmRlcl9pZF0gPSBbXTtcbiAgICAgICAgICB9XG4gICAgICAgICAgamlmZkNsaWVudC5tZXNzYWdlc1dhaXRpbmdLZXlzW3NlbmRlcl9pZF0ucHVzaCh7IGxhYmVsOiAnc2hhcmUnLCBtc2c6IGpzb25fbXNnIH0pO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIGZ1bmN0aW9uIG1wY09wZW4obXNnLCBjYWxsYmFjaykge1xuICAgICAgICAvLyBwYXJzZSBtZXNzYWdlXG4gICAgICAgIHZhciBqc29uX21zZyA9IEpTT04ucGFyc2UobXNnKTtcbiAgICAgICAgdmFyIHNlbmRlcl9pZCA9IGpzb25fbXNnWydwYXJ0eV9pZCddO1xuXG4gICAgICAgIGlmIChqaWZmQ2xpZW50LmtleW1hcFtzZW5kZXJfaWRdICE9IG51bGwpIHtcbiAgICAgICAgICBqaWZmQ2xpZW50LmhhbmRsZXJzLnJlY2VpdmVfb3Blbihqc29uX21zZyk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgaWYgKGppZmZDbGllbnQubWVzc2FnZXNXYWl0aW5nS2V5c1tzZW5kZXJfaWRdID09IG51bGwpIHtcbiAgICAgICAgICAgIGppZmZDbGllbnQubWVzc2FnZXNXYWl0aW5nS2V5c1tzZW5kZXJfaWRdID0gW107XG4gICAgICAgICAgfVxuICAgICAgICAgIGppZmZDbGllbnQubWVzc2FnZXNXYWl0aW5nS2V5c1tzZW5kZXJfaWRdLnB1c2goeyBsYWJlbDogJ29wZW4nLCBtc2c6IGpzb25fbXNnIH0pO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIGhhbmRsZSBjdXN0b20gbWVzc2FnZXNcbiAgICAgIGZ1bmN0aW9uIHNvY2tldEN1c3RvbShtc2csIGNhbGxiYWNrKSB7XG4gICAgICAgIHZhciBqc29uX21zZyA9IEpTT04ucGFyc2UobXNnKTtcbiAgICAgICAgdmFyIHNlbmRlcl9pZCA9IGpzb25fbXNnWydwYXJ0eV9pZCddO1xuICAgICAgICB2YXIgZW5jcnlwdGVkID0ganNvbl9tc2dbJ2VuY3J5cHRlZCddO1xuXG4gICAgICAgIGlmIChqaWZmQ2xpZW50LmtleW1hcFtzZW5kZXJfaWRdICE9IG51bGwgfHwgZW5jcnlwdGVkICE9PSB0cnVlKSB7XG4gICAgICAgICAgamlmZkNsaWVudC5oYW5kbGVycy5yZWNlaXZlX2N1c3RvbShqc29uX21zZyk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgLy8ga2V5IG11c3Qgbm90IGV4aXN0IHlldCBmb3Igc2VuZGVyX2lkLCBhbmQgZW5jcnlwdGVkIG11c3QgYmUgdHJ1ZVxuICAgICAgICAgIGlmIChqaWZmQ2xpZW50Lm1lc3NhZ2VzV2FpdGluZ0tleXNbc2VuZGVyX2lkXSA9PSBudWxsKSB7XG4gICAgICAgICAgICBqaWZmQ2xpZW50Lm1lc3NhZ2VzV2FpdGluZ0tleXNbc2VuZGVyX2lkXSA9IFtdO1xuICAgICAgICAgIH1cbiAgICAgICAgICBqaWZmQ2xpZW50Lm1lc3NhZ2VzV2FpdGluZ0tleXNbc2VuZGVyX2lkXS5wdXNoKHsgbGFiZWw6ICdjdXN0b20nLCBtc2c6IGpzb25fbXNnIH0pO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIGZ1bmN0aW9uIGNyeXB0b1Byb3ZpZGVyKG1zZywgY2FsbGJhY2spIHtcbiAgICAgICAgamlmZkNsaWVudC5oYW5kbGVycy5yZWNlaXZlX2NyeXB0b19wcm92aWRlcihKU09OLnBhcnNlKG1zZykpO1xuICAgICAgfVxuXG4gICAgICBmdW5jdGlvbiBvbkVycm9yKG1zZykge1xuICAgICAgICB0cnkge1xuICAgICAgICAgIG1zZyA9IEpTT04ucGFyc2UobXNnKTtcbiAgICAgICAgICBqaWZmQ2xpZW50LmhhbmRsZXJzLmVycm9yKG1zZ1snbGFiZWwnXSwgbXNnWydlcnJvciddKTtcbiAgICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgICBqaWZmQ2xpZW50LmhhbmRsZXJzLmVycm9yKCdzb2NrZXQuaW8nLCBtc2cpO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIGZ1bmN0aW9uIHNvY2tldENsb3NlKHJlYXNvbikge1xuICAgICAgICBpZiAocmVhc29uICE9PSAnaW8gY2xpZW50IGRpc2Nvbm5lY3QnKSB7XG4gICAgICAgICAgLy8gY2hlY2sgdGhhdCB0aGUgcmVhc29uIGlzIGFuIGVycm9yIGFuZCBub3QgYSB1c2VyIGluaXRpYXRlZCBkaXNjb25uZWN0XG4gICAgICAgICAgY29uc29sZS5sb2coJ0Rpc2Nvbm5lY3RlZCEnLCBqaWZmQ2xpZW50LmlkLCByZWFzb24pO1xuICAgICAgICB9XG5cbiAgICAgICAgamlmZkNsaWVudC5ob29rcy5leGVjdXRlX2FycmF5X2hvb2tzKCdhZnRlck9wZXJhdGlvbicsIFtqaWZmQ2xpZW50LCAnZGlzY29ubmVjdCcsIHJlYXNvbl0sIC0xKTtcbiAgICAgIH1cblxuICAgICAgdGhpcy5zb2NrZXQub25jbG9zZSA9IGZ1bmN0aW9uIChyZWFzb24pIHtcbiAgICAgICAgc29ja2V0Q2xvc2UocmVhc29uLmNvZGUpO1xuICAgICAgfVxuXG4gICAgICAvKipcbiAgICAgICAqIEluIGV2ZXJ5IG1lc3NhZ2Ugc2VudCBvdmVyIHdzLCB3ZSB3aWxsIHNlbmQgYWxvbmcgd2l0aCBpdCBhIHNvY2tldFByb3RvY29sIHN0cmluZ1xuICAgICAgICogdGhhdCB3aWxsIGJlIHBhcnNlZCBieSB0aGUgcmVjZWl2ZXIgdG8gcm91dGUgdGhlIHJlcXVlc3QgdG8gdGhlIGNvcnJlY3QgZnVuY3Rpb24uIFRoZVxuICAgICAgICogcHJldmlvdXMgaW5mb3JtYXRpb24gc2VudCBieSBzb2NrZXQuaW8gd2lsbCBiZSB1bnRvdWNoZWQsIGJ1dCBub3cgc2VudCBpbnNpZGUgb2YgbXNnLmRhdGEuXG4gICAgICAgKi9cbiAgICAgIHRoaXMuc29ja2V0Lm9ubWVzc2FnZSA9IGZ1bmN0aW9uIChtc2csIGNhbGxiYWNrKSB7XG4gICAgICAgIG1zZyA9IEpTT04ucGFyc2UobXNnLmRhdGEpO1xuXG4gICAgICAgIHN3aXRjaCAobXNnLnNvY2tldFByb3RvY29sKSB7XG4gICAgICAgICAgY2FzZSAnaW5pdGlhbGl6YXRpb24nOlxuICAgICAgICAgICAgamlmZkNsaWVudC5oYW5kbGVycy5pbml0aWFsaXplZChtc2cuZGF0YSk7XG4gICAgICAgICAgICBicmVhaztcbiAgICAgICAgICBjYXNlICdwdWJsaWNfa2V5cyc6XG4gICAgICAgICAgICBwdWJsaWNLZXlzQ2hhbmdlZChtc2cuZGF0YSwgY2FsbGJhY2spO1xuICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgY2FzZSAnc2hhcmUnOlxuICAgICAgICAgICAgc2hhcmUobXNnLmRhdGEsIGNhbGxiYWNrKTtcbiAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgIGNhc2UgJ29wZW4nOlxuICAgICAgICAgICAgbXBjT3Blbihtc2cuZGF0YSwgY2FsbGJhY2spO1xuICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgY2FzZSAnY3VzdG9tJzpcbiAgICAgICAgICAgIHNvY2tldEN1c3RvbShtc2cuZGF0YSwgY2FsbGJhY2spO1xuICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgY2FzZSAnY3J5cHRvX3Byb3ZpZGVyJzpcbiAgICAgICAgICAgIGNyeXB0b1Byb3ZpZGVyKG1zZy5kYXRhLCBjYWxsYmFjayk7XG4gICAgICAgICAgICBicmVhaztcbiAgICAgICAgICBjYXNlICdjbG9zZSc6XG4gICAgICAgICAgICBzb2NrZXRDbG9zZShtc2cuZGF0YSk7XG4gICAgICAgICAgICBicmVhaztcbiAgICAgICAgICBjYXNlICdkaXNjb25uZWN0JzpcbiAgICAgICAgICAgIHNvY2tldENsb3NlKG1zZy5kYXRhKTtcbiAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgIGNhc2UgJ2Vycm9yJzpcbiAgICAgICAgICAgIG9uRXJyb3IobXNnLmRhdGEpO1xuICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgZGVmYXVsdDpcbiAgICAgICAgICAgIGNvbnNvbGUubG9nKCdVa25vd24gcHJvdG9jb2wsICcgKyBtc2cuc29ja2V0UHJvdG9jb2wgKyAnLCByZWNlaXZlZCcpO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICB9O1xuXG4gICAgLyogT3ZlcndyaXRlIHRoZSBzb2NrZXRDb25uZWN0IGZ1bmN0aW9uIGZyb20gamlmZi1jbGllbnQuanMgKi9cblxuICAgIGppZmYuc29ja2V0Q29ubmVjdCA9IGZ1bmN0aW9uIChKSUZGQ2xpZW50SW5zdGFuY2UpIHtcblxuICAgICAgaWYgKG9wdGlvbnMuX19pbnRlcm5hbF9zb2NrZXQgPT0gbnVsbCkge1xuICAgICAgICAvKipcbiAgICAgICAgICogU29ja2V0IHdyYXBwZXIgYmV0d2VlbiB0aGlzIGluc3RhbmNlIGFuZCB0aGUgc2VydmVyLCBiYXNlZCBvbiBzb2NrZXRzLmlvXG4gICAgICAgICAqIEB0eXBlIHshR3VhcmRlZFNvY2tldH1cbiAgICAgICAgICovXG4gICAgICAgIEpJRkZDbGllbnRJbnN0YW5jZS5zb2NrZXQgPSBndWFyZGVkU29ja2V0KEpJRkZDbGllbnRJbnN0YW5jZSk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBKSUZGQ2xpZW50SW5zdGFuY2Uuc29ja2V0ID0gaW50ZXJuYWxTb2NrZXQoSklGRkNsaWVudEluc3RhbmNlLCBvcHRpb25zLl9faW50ZXJuYWxfc29ja2V0KTtcbiAgICAgIH1cblxuICAgICAgLy8gc2V0IHVwIHNvY2tldCBldmVudCBoYW5kbGVyc1xuICAgICAgaGFuZGxlcnMoSklGRkNsaWVudEluc3RhbmNlKTtcblxuICAgICAgLy8gT3ZlcndyaXRlIGhhbmRsZXJzLmNvbm5lY3RlZCB3aXRoIG91ciBuZXcgd3MgY29ubmVjdGlvbiBoYW5kbGVyXG4gICAgICBKSUZGQ2xpZW50SW5zdGFuY2UuaGFuZGxlcnMuY29ubmVjdGVkID0gZnVuY3Rpb24gKCkge1xuICAgICAgICBKSUZGQ2xpZW50SW5zdGFuY2UuaW5pdGlhbGl6YXRpb25fY291bnRlcisrO1xuXG4gICAgICAgIGlmIChKSUZGQ2xpZW50SW5zdGFuY2Uuc2VjcmV0X2tleSA9PSBudWxsICYmIEpJRkZDbGllbnRJbnN0YW5jZS5wdWJsaWNfa2V5ID09IG51bGwpIHtcbiAgICAgICAgICB2YXIga2V5ID0gSklGRkNsaWVudEluc3RhbmNlLmhvb2tzLmdlbmVyYXRlS2V5UGFpcihKSUZGQ2xpZW50SW5zdGFuY2UpO1xuICAgICAgICAgIEpJRkZDbGllbnRJbnN0YW5jZS5zZWNyZXRfa2V5ID0ga2V5LnNlY3JldF9rZXk7XG4gICAgICAgICAgSklGRkNsaWVudEluc3RhbmNlLnB1YmxpY19rZXkgPSBrZXkucHVibGljX2tleTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIEluaXRpYWxpemF0aW9uIG1lc3NhZ2VcbiAgICAgICAgdmFyIG1zZyA9IEpJRkZDbGllbnRJbnN0YW5jZS5oYW5kbGVycy5idWlsZF9pbml0aWFsaXphdGlvbl9tZXNzYWdlKCk7XG5cbiAgICAgICAgLy8gRG91YmxlIHdyYXAgdGhlIG1zZ1xuICAgICAgICBtc2cgPSBKU09OLnN0cmluZ2lmeShtc2cpO1xuXG4gICAgICAgIC8vIEVtaXQgaW5pdGlhbGl6YXRpb24gbWVzc2FnZSB0byBzZXJ2ZXJcbiAgICAgICAgSklGRkNsaWVudEluc3RhbmNlLnNvY2tldC5zZW5kKEpTT04uc3RyaW5naWZ5KHsgc29ja2V0UHJvdG9jb2w6ICdpbml0aWFsaXphdGlvbicsIGRhdGE6IG1zZyB9KSk7XG4gICAgICB9O1xuXG5cbiAgICAgIEpJRkZDbGllbnRJbnN0YW5jZS5pbml0U29ja2V0KCk7XG4gICAgfVxuXG4gICAgLyogRnVuY3Rpb25zIHRoYXQgb3ZlcndyaXRlIGNsaWVudC9zb2NrZXQvbWFpbGJveC5qcyBmdW5jdGlvbmFsaXR5ICovXG5cbiAgICBmdW5jdGlvbiBndWFyZGVkU29ja2V0KGppZmZDbGllbnQpIHtcbiAgICAgIC8vIENyZWF0ZSBwbGFpbiBzb2NrZXQgaW8gb2JqZWN0IHdoaWNoIHdlIHdpbGwgd3JhcCBpbiB0aGlzXG4gICAgICB2YXIgc29ja2V0O1xuICAgICAgaWYgKGppZmZDbGllbnQuaG9zdG5hbWUuc3RhcnRzV2l0aChcImh0dHBcIikpIHtcbiAgICAgICAgdmFyIG1vZGlmaWVkSG9zdE5hbWUgPSBcIndzXCIgKyBqaWZmQ2xpZW50Lmhvc3RuYW1lLnN1YnN0cmluZyhqaWZmQ2xpZW50Lmhvc3RuYW1lLmluZGV4T2YoXCI6XCIpKVxuICAgICAgICBzb2NrZXQgPSBuZXcgd3MobW9kaWZpZWRIb3N0TmFtZSlcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHNvY2tldCA9IG5ldyB3cyhqaWZmQ2xpZW50Lmhvc3RuYW1lKTtcbiAgICAgIH1cblxuXG4gICAgICBzb2NrZXQub2xkX2Rpc2Nvbm5lY3QgPSBzb2NrZXQuY2xvc2U7XG5cbiAgICAgIHNvY2tldC5tYWlsYm94ID0gbGlua2VkTGlzdCgpOyAvLyBmb3Igb3V0Z29pbmcgbWVzc2FnZXNcbiAgICAgIHNvY2tldC5lbXB0eV9kZWZlcnJlZCA9IG51bGw7IC8vIGdldHMgcmVzb2x2ZWQgd2hlbmV2ZXIgdGhlIG1haWxib3ggaXMgZW1wdHlcbiAgICAgIHNvY2tldC5qaWZmQ2xpZW50ID0gamlmZkNsaWVudDtcblxuICAgICAgLy8gYWRkIGZ1bmN0aW9uYWxpdHkgdG8gc29ja2V0XG4gICAgICBzb2NrZXQuc2FmZV9lbWl0ID0gc2FmZV9lbWl0LmJpbmQoc29ja2V0KTtcbiAgICAgIHNvY2tldC5yZXNlbmRfbWFpbGJveCA9IHJlc2VuZF9tYWlsYm94LmJpbmQoc29ja2V0KTtcbiAgICAgIHNvY2tldC5kaXNjb25uZWN0ID0gZGlzY29ubmVjdC5iaW5kKHNvY2tldCk7XG4gICAgICBzb2NrZXQuc2FmZV9kaXNjb25uZWN0ID0gc2FmZV9kaXNjb25uZWN0LmJpbmQoc29ja2V0KTtcbiAgICAgIHNvY2tldC5pc19lbXB0eSA9IGlzX2VtcHR5LmJpbmQoc29ja2V0KTtcblxuICAgICAgcmV0dXJuIHNvY2tldDtcbiAgICB9XG5cbiAgICBmdW5jdGlvbiBzYWZlX2VtaXQobGFiZWwsIG1zZykge1xuICAgICAgLy8gYWRkIG1lc3NhZ2UgdG8gbWFpbGJveFxuICAgICAgdmFyIG1haWxib3hfcG9pbnRlciA9IHRoaXMubWFpbGJveC5hZGQoeyBsYWJlbDogbGFiZWwsIG1zZzogbXNnIH0pO1xuICAgICAgaWYgKHRoaXMucmVhZHlTdGF0ZSA9PT0gMSkge1xuICAgICAgICB2YXIgc2VsZiA9IHRoaXM7XG4gICAgICAgIC8vIGVtaXQgdGhlIG1lc3NhZ2UsIGlmIGFuIGFja25vd2xlZGdtZW50IGlzIHJlY2VpdmVkLCByZW1vdmUgaXQgZnJvbSBtYWlsYm94XG5cbiAgICAgICAgdGhpcy5zZW5kKEpTT04uc3RyaW5naWZ5KHsgc29ja2V0UHJvdG9jb2w6IGxhYmVsLCBkYXRhOiBtc2cgfSksIG51bGwsIGZ1bmN0aW9uIChzdGF0dXMpIHtcblxuICAgICAgICAgIHNlbGYubWFpbGJveC5yZW1vdmUobWFpbGJveF9wb2ludGVyKTtcblxuICAgICAgICAgIGlmIChzZWxmLmlzX2VtcHR5KCkgJiYgc2VsZi5lbXB0eV9kZWZlcnJlZCAhPSBudWxsKSB7XG4gICAgICAgICAgICBzZWxmLmVtcHR5X2RlZmVycmVkLnJlc29sdmUoKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBpZiAobGFiZWwgPT09ICdmcmVlJykge1xuICAgICAgICAgICAgc2VsZi5qaWZmQ2xpZW50Lmhvb2tzLmV4ZWN1dGVfYXJyYXlfaG9va3MoJ2FmdGVyT3BlcmF0aW9uJywgW3NlbGYuamlmZkNsaWVudCwgJ2ZyZWUnLCBtc2ddLCAyKTtcbiAgICAgICAgICB9XG4gICAgICAgIH0pO1xuICAgICAgfVxuXG4gICAgfVxuXG4gICAgZnVuY3Rpb24gcmVzZW5kX21haWxib3goKSB7XG4gICAgICAvLyBDcmVhdGUgYSBuZXcgbWFpbGJveCwgc2luY2UgdGhlIGN1cnJlbnQgbWFpbGJveCB3aWxsIGJlIHJlc2VudCBhbmRcbiAgICAgIC8vIHdpbGwgY29udGFpbiBuZXcgYmFja3Vwcy5cbiAgICAgIHZhciBvbGRfbWFpbGJveCA9IHRoaXMubWFpbGJveDtcbiAgICAgIHRoaXMubWFpbGJveCA9IGxpbmtlZExpc3QoKTtcblxuICAgICAgLy8gbG9vcCBvdmVyIGFsbCBzdG9yZWQgbWVzc2FnZXMgYW5kIGVtaXQgdGhlbVxuICAgICAgdmFyIGN1cnJlbnRfbm9kZSA9IG9sZF9tYWlsYm94LmhlYWQ7XG4gICAgICB3aGlsZSAoY3VycmVudF9ub2RlICE9IG51bGwpIHtcbiAgICAgICAgdmFyIGxhYmVsID0gY3VycmVudF9ub2RlLm9iamVjdC5sYWJlbDtcbiAgICAgICAgdmFyIG1zZyA9IGN1cnJlbnRfbm9kZS5vYmplY3QubXNnO1xuICAgICAgICB0aGlzLnNhZmVfZW1pdChsYWJlbCwgbXNnKTtcbiAgICAgICAgY3VycmVudF9ub2RlID0gY3VycmVudF9ub2RlLm5leHQ7XG4gICAgICB9XG5cbiAgICB9XG5cbiAgICBmdW5jdGlvbiBkaXNjb25uZWN0KCkge1xuXG4gICAgICB0aGlzLmppZmZDbGllbnQuaG9va3MuZXhlY3V0ZV9hcnJheV9ob29rcygnYmVmb3JlT3BlcmF0aW9uJywgW3RoaXMuamlmZkNsaWVudCwgJ2Rpc2Nvbm5lY3QnLCB7fV0sIC0xKTtcblxuXG4gICAgICB0aGlzLm9sZF9kaXNjb25uZWN0LmFwcGx5KHRoaXMsIGFyZ3VtZW50cyk7XG4gICAgfVxuXG4gICAgZnVuY3Rpb24gc2FmZV9kaXNjb25uZWN0KGZyZWUsIGNhbGxiYWNrKSB7XG5cbiAgICAgIGlmICh0aGlzLmlzX2VtcHR5KCkpIHtcblxuICAgICAgICBpZiAoZnJlZSkge1xuICAgICAgICAgIHRoaXMuamlmZkNsaWVudC5mcmVlKCk7XG4gICAgICAgICAgZnJlZSA9IGZhbHNlO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIC8vIFQ6IFNob3VsZCByZW1haW4gXCJkaXNjb25uZWN0XCIgc2luY2Ugd2Ugb3ZlcnJpZGUgdGhlIC5kaXNjb25uZWN0LCBubyBuZWVkIHRvIGNoYW5nZSB0byBjbG9zZVxuICAgICAgICAgIHRoaXMuZGlzY29ubmVjdCgpO1xuICAgICAgICAgIGlmIChjYWxsYmFjayAhPSBudWxsKSB7XG4gICAgICAgICAgICBjYWxsYmFjaygpO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgdGhpcy5lbXB0eV9kZWZlcnJlZCA9IG5ldyB0aGlzLmppZmZDbGllbnQuaGVscGVycy5EZWZlcnJlZCgpO1xuICAgICAgdGhpcy5lbXB0eV9kZWZlcnJlZC5wcm9taXNlLnRoZW4odGhpcy5zYWZlX2Rpc2Nvbm5lY3QuYmluZCh0aGlzLCBmcmVlLCBjYWxsYmFjaykpO1xuXG4gICAgfVxuXG4gICAgZnVuY3Rpb24gaXNfZW1wdHkoKSB7XG4gICAgICByZXR1cm4gdGhpcy5tYWlsYm94LmhlYWQgPT0gbnVsbCAmJiB0aGlzLmppZmZDbGllbnQuY291bnRlcnMucGVuZGluZ19vcGVucyA9PT0gMDtcblxuICAgIH1cblxuICAgIC8qIFBSRVBST0NFU1NJTkcgSVMgVEhFIFNBTUUgKi9cbiAgICBqaWZmLnByZXByb2Nlc3NpbmdfZnVuY3Rpb25fbWFwW2V4cG9ydHMubmFtZV0gPSB7fTtcblxuXG4gICAgcmV0dXJuIGppZmY7XG4gIH1cbiAgLy8gRXhwb3NlIHRoZSBBUEkgZm9yIHRoaXMgZXh0ZW5zaW9uLlxuICBleHBvcnRzLm1ha2VfamlmZiA9IG1ha2VfamlmZjtcblxufSgodHlwZW9mIGV4cG9ydHMgPT09ICd1bmRlZmluZWQnID8gdGhpcy5qaWZmX3dlYnNvY2tldHMgPSB7fSA6IGV4cG9ydHMpLCB0eXBlb2YgZXhwb3J0cyAhPT0gJ3VuZGVmaW5lZCcpKTtcbiIsIi8vIHNoaW0gZm9yIHVzaW5nIHByb2Nlc3MgaW4gYnJvd3NlclxudmFyIHByb2Nlc3MgPSBtb2R1bGUuZXhwb3J0cyA9IHt9O1xuXG4vLyBjYWNoZWQgZnJvbSB3aGF0ZXZlciBnbG9iYWwgaXMgcHJlc2VudCBzbyB0aGF0IHRlc3QgcnVubmVycyB0aGF0IHN0dWIgaXRcbi8vIGRvbid0IGJyZWFrIHRoaW5ncy4gIEJ1dCB3ZSBuZWVkIHRvIHdyYXAgaXQgaW4gYSB0cnkgY2F0Y2ggaW4gY2FzZSBpdCBpc1xuLy8gd3JhcHBlZCBpbiBzdHJpY3QgbW9kZSBjb2RlIHdoaWNoIGRvZXNuJ3QgZGVmaW5lIGFueSBnbG9iYWxzLiAgSXQncyBpbnNpZGUgYVxuLy8gZnVuY3Rpb24gYmVjYXVzZSB0cnkvY2F0Y2hlcyBkZW9wdGltaXplIGluIGNlcnRhaW4gZW5naW5lcy5cblxudmFyIGNhY2hlZFNldFRpbWVvdXQ7XG52YXIgY2FjaGVkQ2xlYXJUaW1lb3V0O1xuXG5mdW5jdGlvbiBkZWZhdWx0U2V0VGltb3V0KCkge1xuICAgIHRocm93IG5ldyBFcnJvcignc2V0VGltZW91dCBoYXMgbm90IGJlZW4gZGVmaW5lZCcpO1xufVxuZnVuY3Rpb24gZGVmYXVsdENsZWFyVGltZW91dCAoKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKCdjbGVhclRpbWVvdXQgaGFzIG5vdCBiZWVuIGRlZmluZWQnKTtcbn1cbihmdW5jdGlvbiAoKSB7XG4gICAgdHJ5IHtcbiAgICAgICAgaWYgKHR5cGVvZiBzZXRUaW1lb3V0ID09PSAnZnVuY3Rpb24nKSB7XG4gICAgICAgICAgICBjYWNoZWRTZXRUaW1lb3V0ID0gc2V0VGltZW91dDtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIGNhY2hlZFNldFRpbWVvdXQgPSBkZWZhdWx0U2V0VGltb3V0O1xuICAgICAgICB9XG4gICAgfSBjYXRjaCAoZSkge1xuICAgICAgICBjYWNoZWRTZXRUaW1lb3V0ID0gZGVmYXVsdFNldFRpbW91dDtcbiAgICB9XG4gICAgdHJ5IHtcbiAgICAgICAgaWYgKHR5cGVvZiBjbGVhclRpbWVvdXQgPT09ICdmdW5jdGlvbicpIHtcbiAgICAgICAgICAgIGNhY2hlZENsZWFyVGltZW91dCA9IGNsZWFyVGltZW91dDtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIGNhY2hlZENsZWFyVGltZW91dCA9IGRlZmF1bHRDbGVhclRpbWVvdXQ7XG4gICAgICAgIH1cbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICAgIGNhY2hlZENsZWFyVGltZW91dCA9IGRlZmF1bHRDbGVhclRpbWVvdXQ7XG4gICAgfVxufSAoKSlcbmZ1bmN0aW9uIHJ1blRpbWVvdXQoZnVuKSB7XG4gICAgaWYgKGNhY2hlZFNldFRpbWVvdXQgPT09IHNldFRpbWVvdXQpIHtcbiAgICAgICAgLy9ub3JtYWwgZW52aXJvbWVudHMgaW4gc2FuZSBzaXR1YXRpb25zXG4gICAgICAgIHJldHVybiBzZXRUaW1lb3V0KGZ1biwgMCk7XG4gICAgfVxuICAgIC8vIGlmIHNldFRpbWVvdXQgd2Fzbid0IGF2YWlsYWJsZSBidXQgd2FzIGxhdHRlciBkZWZpbmVkXG4gICAgaWYgKChjYWNoZWRTZXRUaW1lb3V0ID09PSBkZWZhdWx0U2V0VGltb3V0IHx8ICFjYWNoZWRTZXRUaW1lb3V0KSAmJiBzZXRUaW1lb3V0KSB7XG4gICAgICAgIGNhY2hlZFNldFRpbWVvdXQgPSBzZXRUaW1lb3V0O1xuICAgICAgICByZXR1cm4gc2V0VGltZW91dChmdW4sIDApO1xuICAgIH1cbiAgICB0cnkge1xuICAgICAgICAvLyB3aGVuIHdoZW4gc29tZWJvZHkgaGFzIHNjcmV3ZWQgd2l0aCBzZXRUaW1lb3V0IGJ1dCBubyBJLkUuIG1hZGRuZXNzXG4gICAgICAgIHJldHVybiBjYWNoZWRTZXRUaW1lb3V0KGZ1biwgMCk7XG4gICAgfSBjYXRjaChlKXtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIC8vIFdoZW4gd2UgYXJlIGluIEkuRS4gYnV0IHRoZSBzY3JpcHQgaGFzIGJlZW4gZXZhbGVkIHNvIEkuRS4gZG9lc24ndCB0cnVzdCB0aGUgZ2xvYmFsIG9iamVjdCB3aGVuIGNhbGxlZCBub3JtYWxseVxuICAgICAgICAgICAgcmV0dXJuIGNhY2hlZFNldFRpbWVvdXQuY2FsbChudWxsLCBmdW4sIDApO1xuICAgICAgICB9IGNhdGNoKGUpe1xuICAgICAgICAgICAgLy8gc2FtZSBhcyBhYm92ZSBidXQgd2hlbiBpdCdzIGEgdmVyc2lvbiBvZiBJLkUuIHRoYXQgbXVzdCBoYXZlIHRoZSBnbG9iYWwgb2JqZWN0IGZvciAndGhpcycsIGhvcGZ1bGx5IG91ciBjb250ZXh0IGNvcnJlY3Qgb3RoZXJ3aXNlIGl0IHdpbGwgdGhyb3cgYSBnbG9iYWwgZXJyb3JcbiAgICAgICAgICAgIHJldHVybiBjYWNoZWRTZXRUaW1lb3V0LmNhbGwodGhpcywgZnVuLCAwKTtcbiAgICAgICAgfVxuICAgIH1cblxuXG59XG5mdW5jdGlvbiBydW5DbGVhclRpbWVvdXQobWFya2VyKSB7XG4gICAgaWYgKGNhY2hlZENsZWFyVGltZW91dCA9PT0gY2xlYXJUaW1lb3V0KSB7XG4gICAgICAgIC8vbm9ybWFsIGVudmlyb21lbnRzIGluIHNhbmUgc2l0dWF0aW9uc1xuICAgICAgICByZXR1cm4gY2xlYXJUaW1lb3V0KG1hcmtlcik7XG4gICAgfVxuICAgIC8vIGlmIGNsZWFyVGltZW91dCB3YXNuJ3QgYXZhaWxhYmxlIGJ1dCB3YXMgbGF0dGVyIGRlZmluZWRcbiAgICBpZiAoKGNhY2hlZENsZWFyVGltZW91dCA9PT0gZGVmYXVsdENsZWFyVGltZW91dCB8fCAhY2FjaGVkQ2xlYXJUaW1lb3V0KSAmJiBjbGVhclRpbWVvdXQpIHtcbiAgICAgICAgY2FjaGVkQ2xlYXJUaW1lb3V0ID0gY2xlYXJUaW1lb3V0O1xuICAgICAgICByZXR1cm4gY2xlYXJUaW1lb3V0KG1hcmtlcik7XG4gICAgfVxuICAgIHRyeSB7XG4gICAgICAgIC8vIHdoZW4gd2hlbiBzb21lYm9keSBoYXMgc2NyZXdlZCB3aXRoIHNldFRpbWVvdXQgYnV0IG5vIEkuRS4gbWFkZG5lc3NcbiAgICAgICAgcmV0dXJuIGNhY2hlZENsZWFyVGltZW91dChtYXJrZXIpO1xuICAgIH0gY2F0Y2ggKGUpe1xuICAgICAgICB0cnkge1xuICAgICAgICAgICAgLy8gV2hlbiB3ZSBhcmUgaW4gSS5FLiBidXQgdGhlIHNjcmlwdCBoYXMgYmVlbiBldmFsZWQgc28gSS5FLiBkb2Vzbid0ICB0cnVzdCB0aGUgZ2xvYmFsIG9iamVjdCB3aGVuIGNhbGxlZCBub3JtYWxseVxuICAgICAgICAgICAgcmV0dXJuIGNhY2hlZENsZWFyVGltZW91dC5jYWxsKG51bGwsIG1hcmtlcik7XG4gICAgICAgIH0gY2F0Y2ggKGUpe1xuICAgICAgICAgICAgLy8gc2FtZSBhcyBhYm92ZSBidXQgd2hlbiBpdCdzIGEgdmVyc2lvbiBvZiBJLkUuIHRoYXQgbXVzdCBoYXZlIHRoZSBnbG9iYWwgb2JqZWN0IGZvciAndGhpcycsIGhvcGZ1bGx5IG91ciBjb250ZXh0IGNvcnJlY3Qgb3RoZXJ3aXNlIGl0IHdpbGwgdGhyb3cgYSBnbG9iYWwgZXJyb3IuXG4gICAgICAgICAgICAvLyBTb21lIHZlcnNpb25zIG9mIEkuRS4gaGF2ZSBkaWZmZXJlbnQgcnVsZXMgZm9yIGNsZWFyVGltZW91dCB2cyBzZXRUaW1lb3V0XG4gICAgICAgICAgICByZXR1cm4gY2FjaGVkQ2xlYXJUaW1lb3V0LmNhbGwodGhpcywgbWFya2VyKTtcbiAgICAgICAgfVxuICAgIH1cblxuXG5cbn1cbnZhciBxdWV1ZSA9IFtdO1xudmFyIGRyYWluaW5nID0gZmFsc2U7XG52YXIgY3VycmVudFF1ZXVlO1xudmFyIHF1ZXVlSW5kZXggPSAtMTtcblxuZnVuY3Rpb24gY2xlYW5VcE5leHRUaWNrKCkge1xuICAgIGlmICghZHJhaW5pbmcgfHwgIWN1cnJlbnRRdWV1ZSkge1xuICAgICAgICByZXR1cm47XG4gICAgfVxuICAgIGRyYWluaW5nID0gZmFsc2U7XG4gICAgaWYgKGN1cnJlbnRRdWV1ZS5sZW5ndGgpIHtcbiAgICAgICAgcXVldWUgPSBjdXJyZW50UXVldWUuY29uY2F0KHF1ZXVlKTtcbiAgICB9IGVsc2Uge1xuICAgICAgICBxdWV1ZUluZGV4ID0gLTE7XG4gICAgfVxuICAgIGlmIChxdWV1ZS5sZW5ndGgpIHtcbiAgICAgICAgZHJhaW5RdWV1ZSgpO1xuICAgIH1cbn1cblxuZnVuY3Rpb24gZHJhaW5RdWV1ZSgpIHtcbiAgICBpZiAoZHJhaW5pbmcpIHtcbiAgICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB2YXIgdGltZW91dCA9IHJ1blRpbWVvdXQoY2xlYW5VcE5leHRUaWNrKTtcbiAgICBkcmFpbmluZyA9IHRydWU7XG5cbiAgICB2YXIgbGVuID0gcXVldWUubGVuZ3RoO1xuICAgIHdoaWxlKGxlbikge1xuICAgICAgICBjdXJyZW50UXVldWUgPSBxdWV1ZTtcbiAgICAgICAgcXVldWUgPSBbXTtcbiAgICAgICAgd2hpbGUgKCsrcXVldWVJbmRleCA8IGxlbikge1xuICAgICAgICAgICAgaWYgKGN1cnJlbnRRdWV1ZSkge1xuICAgICAgICAgICAgICAgIGN1cnJlbnRRdWV1ZVtxdWV1ZUluZGV4XS5ydW4oKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICBxdWV1ZUluZGV4ID0gLTE7XG4gICAgICAgIGxlbiA9IHF1ZXVlLmxlbmd0aDtcbiAgICB9XG4gICAgY3VycmVudFF1ZXVlID0gbnVsbDtcbiAgICBkcmFpbmluZyA9IGZhbHNlO1xuICAgIHJ1bkNsZWFyVGltZW91dCh0aW1lb3V0KTtcbn1cblxucHJvY2Vzcy5uZXh0VGljayA9IGZ1bmN0aW9uIChmdW4pIHtcbiAgICB2YXIgYXJncyA9IG5ldyBBcnJheShhcmd1bWVudHMubGVuZ3RoIC0gMSk7XG4gICAgaWYgKGFyZ3VtZW50cy5sZW5ndGggPiAxKSB7XG4gICAgICAgIGZvciAodmFyIGkgPSAxOyBpIDwgYXJndW1lbnRzLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgICAgICBhcmdzW2kgLSAxXSA9IGFyZ3VtZW50c1tpXTtcbiAgICAgICAgfVxuICAgIH1cbiAgICBxdWV1ZS5wdXNoKG5ldyBJdGVtKGZ1biwgYXJncykpO1xuICAgIGlmIChxdWV1ZS5sZW5ndGggPT09IDEgJiYgIWRyYWluaW5nKSB7XG4gICAgICAgIHJ1blRpbWVvdXQoZHJhaW5RdWV1ZSk7XG4gICAgfVxufTtcblxuLy8gdjggbGlrZXMgcHJlZGljdGlibGUgb2JqZWN0c1xuZnVuY3Rpb24gSXRlbShmdW4sIGFycmF5KSB7XG4gICAgdGhpcy5mdW4gPSBmdW47XG4gICAgdGhpcy5hcnJheSA9IGFycmF5O1xufVxuSXRlbS5wcm90b3R5cGUucnVuID0gZnVuY3Rpb24gKCkge1xuICAgIHRoaXMuZnVuLmFwcGx5KG51bGwsIHRoaXMuYXJyYXkpO1xufTtcbnByb2Nlc3MudGl0bGUgPSAnYnJvd3Nlcic7XG5wcm9jZXNzLmJyb3dzZXIgPSB0cnVlO1xucHJvY2Vzcy5lbnYgPSB7fTtcbnByb2Nlc3MuYXJndiA9IFtdO1xucHJvY2Vzcy52ZXJzaW9uID0gJyc7IC8vIGVtcHR5IHN0cmluZyB0byBhdm9pZCByZWdleHAgaXNzdWVzXG5wcm9jZXNzLnZlcnNpb25zID0ge307XG5cbmZ1bmN0aW9uIG5vb3AoKSB7fVxuXG5wcm9jZXNzLm9uID0gbm9vcDtcbnByb2Nlc3MuYWRkTGlzdGVuZXIgPSBub29wO1xucHJvY2Vzcy5vbmNlID0gbm9vcDtcbnByb2Nlc3Mub2ZmID0gbm9vcDtcbnByb2Nlc3MucmVtb3ZlTGlzdGVuZXIgPSBub29wO1xucHJvY2Vzcy5yZW1vdmVBbGxMaXN0ZW5lcnMgPSBub29wO1xucHJvY2Vzcy5lbWl0ID0gbm9vcDtcbnByb2Nlc3MucHJlcGVuZExpc3RlbmVyID0gbm9vcDtcbnByb2Nlc3MucHJlcGVuZE9uY2VMaXN0ZW5lciA9IG5vb3A7XG5cbnByb2Nlc3MubGlzdGVuZXJzID0gZnVuY3Rpb24gKG5hbWUpIHsgcmV0dXJuIFtdIH1cblxucHJvY2Vzcy5iaW5kaW5nID0gZnVuY3Rpb24gKG5hbWUpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ3Byb2Nlc3MuYmluZGluZyBpcyBub3Qgc3VwcG9ydGVkJyk7XG59O1xuXG5wcm9jZXNzLmN3ZCA9IGZ1bmN0aW9uICgpIHsgcmV0dXJuICcvJyB9O1xucHJvY2Vzcy5jaGRpciA9IGZ1bmN0aW9uIChkaXIpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ3Byb2Nlc3MuY2hkaXIgaXMgbm90IHN1cHBvcnRlZCcpO1xufTtcbnByb2Nlc3MudW1hc2sgPSBmdW5jdGlvbigpIHsgcmV0dXJuIDA7IH07XG4iLCIndXNlIHN0cmljdCc7XG5cbm1vZHVsZS5leHBvcnRzID0gZnVuY3Rpb24gKCkge1xuICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgJ3dzIGRvZXMgbm90IHdvcmsgaW4gdGhlIGJyb3dzZXIuIEJyb3dzZXIgY2xpZW50cyBtdXN0IHVzZSB0aGUgbmF0aXZlICcgK1xuICAgICAgJ1dlYlNvY2tldCBvYmplY3QnXG4gICk7XG59O1xuIl19 diff --git a/dist/jiff-client.js b/dist/jiff-client.js index a507cb9a2..4c2ad6e78 100644 --- a/dist/jiff-client.js +++ b/dist/jiff-client.js @@ -1,48110 +1,68015 @@ -(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.JIFFClient = f()}})(function(){var define,module,exports;return (function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;iImportant: bit protocols (including bit_decomposition) are unaware of any extension specific customizations, and will operate as - * on the given shares as if they are natural numbers in Zp. Make sure to take into consideration any magnification/shift transformations - * needed to translate correctly between plain representations and extension representations of bits! - * @alias bits - * @namespace - */ - -module.exports = function (jiffClient) { - /** - * Compute sum of bitwise secret shared number and a constant - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - the bit wise secret shares - * @param {number} constant - the constant - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length here will be max(|bits|, |constant|) + 1 - * in case of potential overflow / carry - */ - jiffClient.protocols.bits.cadd = arithmetic.cadd.bind(null, jiffClient); - /** - * Compute [secret bits] - [constant bits] - * @method - * @memberof bits - * @param {number} constant - the constant - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - the bit wise secret shares - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where - * the bit at index 0 is the least significant bit. The bit at index 1 is the most significant bit, - * and the bit at index |bits| is 1 if the result overflows, or 0 otherwise - */ - jiffClient.protocols.bits.csubl = arithmetic.csubl.bind(null, jiffClient); - /** - * Compute [constant bits] - [secret bits] - * @method - * @memberof bits - * @param {number} constant - the constant - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - the bit wise secret shares - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where - * the bit at index 0 is the least significant bit. The bit at index 1 is the most significant bit, - * and the bit at index |bits| is 1 if the result overflows, or 0 otherwise - */ - jiffClient.protocols.bits.csubr = arithmetic.csubr.bind(null, jiffClient); - /** - * - * Compute [secret bits1] + [secret bits2] - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - the first bitwise shared number: array of secrets with index 0 being least significant bit - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number (length may be different) - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where - * the bit at index 0 is the least significant bit - */ - jiffClient.protocols.bits.sadd = arithmetic.sadd.bind(null, jiffClient); - /** - * Compute [secret bits1] - [secret bits2] - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - first bitwise secret shared number: lower indices represent less significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - second bitwise secret shared number (length may be different) - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where - * the bit at index 0 is the least significant bit. The bit at index 1 is the most significant bit, - * and the bit at index |bits| is 1 if the result overflows, or 0 otherwise - */ - jiffClient.protocols.bits.ssub = arithmetic.ssub.bind(null, jiffClient); - /** - * Compute [secret bits] * constant - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - bitwise shared secret to multiply: lower indices represent less significant bits - * @param {number} constant - constant to multiply with - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result, the length of the result will be bits.length + ceil(log2(constant)), except - * if constant is zero, the result will then be [ zero share ] - */ - jiffClient.protocols.bits.cmult = arithmetic.cmult.bind(null, jiffClient); - /** - * Compute [secret bits1] * [secret bits2] - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - bitwise shared secret to multiply: lower indices represent less significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - bitwise shared secret to multiply - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result, the length of the result will be bits1.length + bits2.length - */ - jiffClient.protocols.bits.smult = arithmetic.smult.bind(null, jiffClient); - /** - * Computes integer division of [secret bits 1] / [secret bits 2] - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {{quotient: module:jiff-client~JIFFClient#SecretShare[], remainder: module:jiff-client~JIFFClient#SecretShare[]}} the quotient and remainder bits arrays, note that - * the quotient array has the same length as bits1, - * and the remainder array has the same length as bits2 or bits1, whichever is smaller. - * Note: if bits2 represent 0, the returned result is the maximum - * number that fits in the number of bits (all 1), and the remainder - * is equal to bits1 - */ - jiffClient.protocols.bits.sdiv = arithmetic.sdiv.bind(null, jiffClient); - /** - * Computes integer division of [secret bits] / constant - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - numerator: an array of secret shares of bits, starting from least to most significant bits - * @param {number} constant - the denominator number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {{quotient: module:jiff-client~JIFFClient#SecretShare[], remainder: module:jiff-client~JIFFClient#SecretShare[]}} the quotient and remainder bits arrays, note that - * the quotient array has the same length as bits, - * and the remainder array has the same length as - * constant or bits, whichever is smaller - * @throws if constant is 0. - */ - jiffClient.protocols.bits.cdivl = arithmetic.cdivl.bind(null, jiffClient); - /** - * Computes integer division of constant / [secret bits] - * @method - * @memberof bits - * @param {number} constant - the numerator number - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - denominator: an array of secret shares of bits, starting from least to most significant bits - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {{quotient: module:jiff-client~JIFFClient#SecretShare[], remainder: module:jiff-client~JIFFClient#SecretShare[]}} the quotient and remainder bits arrays, note that - * the quotient array has the same length as the number of bits in constant, - * and the remainder array has the same length as bits or constant, whichever is smaller. - * Note: if bits represent 0, the returned result is the maximum - * number that fits in its bits (all 1), and the remainder - * is equal to constant - */ - jiffClient.protocols.bits.cdivr = arithmetic.cdivr.bind(null, jiffClient); - - /** - * Checks whether the given bitwise secret shared number and numeric constant are equal - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of secret shares of bits, starting from least to most significant bits - * @param {number} constant - the constant number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {SecretShare|boolean} a secret share of 1 if parameters are equal, 0 otherwise. If result is known - * (e.g. constant has a greater non-zero bit than bits' most significant bit), the result is - * returned immediately as a boolean - */ - jiffClient.protocols.bits.ceq = comparison.ceq.bind(null, jiffClient); - /** - * Checks whether the given bitwise secret shared number and numeric constant are not equal - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of secret shares of bits, starting from least to most significant bits - * @param {number} constant - the constant number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {SecretShare|boolean} a secret share of 1 if parameters are not equal, 0 otherwise. If result is known - * (e.g. constant has a greater non-zero bit than bits' most significant bit), the result is - * returned immediately as a boolean - */ - jiffClient.protocols.bits.cneq = comparison.cneq.bind(null, jiffClient); - /** - * Checks whether given secret shared bits are greater than the given constant - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits - * @param {number} constant - the constant number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {SecretShare|boolean} a secret share of 1 if bits are greater than constant, 0 otherwise, if result is known - * (e.g. constant has greater non-zero bit than bits' most significant bit), the result is - * returned immediately as a boolean - */ - jiffClient.protocols.bits.cgt = comparison.cgt.bind(null, jiffClient); - /** - * Checks whether given secret shared bits are greater or equal to the given constant - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits - * @param {number} constant - the constant number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {SecretShare|boolean} a secret share of 1 if bits are greater or equal to constant, 0 otherwise, if result is known - * (e.g. constant has greater non-zero bit than bits' most significant bit or constant is zero), the result is - * returned immediately as a boolean - */ - jiffClient.protocols.bits.cgteq = comparison.cgteq.bind(null, jiffClient); - /** - * Checks whether given secret shared bits are less than the given constant - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits - * @param {number} constant - the constant number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {SecretShare|boolean} a secret share of 1 if bits are less than the constant, 0 otherwise, if result is known - * (e.g. constant has greater non-zero bit than bits' most significant bit), the result is - * returned immediately as a boolean - */ - jiffClient.protocols.bits.clt = comparison.clt.bind(null, jiffClient); - /** - * Checks whether given secret shared bits are less or equal to the given constant - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits - * @param {number} constant - the constant number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {SecretShare|boolean} a secret share of 1 if bits are less or equal to constant, 0 otherwise, if result is known - * (e.g. constant has greater non-zero bit than bits' most significant bit), the result is - * returned immediately as a boolean - */ - jiffClient.protocols.bits.clteq = comparison.clteq.bind(null, jiffClient); - /** - * Checks whether the two given bitwise secret shared numbers are equal - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if bits are equal, 0 otherwise - */ - jiffClient.protocols.bits.seq = comparison.seq.bind(null, jiffClient); - /** - * Checks whether the two given bitwise secret shared numbers are not equal - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if bits are not equal, 0 otherwise - */ - jiffClient.protocols.bits.sneq = comparison.sneq.bind(null, jiffClient); - /** - * Checks whether the first given bitwise secret shared number is greater than the second bitwise secret shared number - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is greater than the second, 0 otherwise - */ - jiffClient.protocols.bits.sgt = comparison.sgt.bind(null, jiffClient); - /** - * Checks whether the first given bitwise secret shared number is greater than or equal to the second bitwise secret shared number - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is greater or equal to the second, 0 otherwise - */ - jiffClient.protocols.bits.sgteq = comparison.sgteq.bind(null, jiffClient); - /** - * Checks whether the first given bitwise secret shared number is less than the second bitwise secret shared number - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is less than the second, 0 otherwise - */ - jiffClient.protocols.bits.slt = comparison.slt.bind(null, jiffClient); - /** - * Checks whether the first given bitwise secret shared number is less or equal to the second bitwise secret shared number - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is less than or equal to the second, 0 otherwise - */ - jiffClient.protocols.bits.slteq = comparison.slteq.bind(null, jiffClient); - - /** - * Retrieves preprocessed rejection_sampling bits or performs the rejection sampling on the fly if crypto_provider is enabled - * @method - * @memberof bits - * @param {number} lower_bound - the lower bound, included (can be a bigNumber if using bigNumber extension) - * @param {number} upper_bound - the upper bound, excluded (can be a bigNumber if using bigNumber extension) - * @param {number} [threshold=parties.length] - the threshold of the resulting shares after sampling - * @param {Array} [parties=all_parties] - array of party ids that want to receive (or compute if needed) the sampling shares, by default, this includes all parties - * @param {number} [Zp=jiff-instance.Zp] - the mod (if null then the default Zp for the instance is used) - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly - * @returns {SecretShare[]} an array of secret shares, each representing a bit from the sampled value (from least to most significant) - */ - jiffClient.protocols.bits.rejection_sampling = otherProtocols.rejection_sampling.bind(null, jiffClient); - - /** - * Creates a secret share of the number represented by the given array of secret shared bits. - * Requires no communication, only local operations - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits. - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of the number represented by bits. - */ - jiffClient.protocols.bits.bit_composition = otherProtocols.bit_composition; - - /** - * Share a number as an array of secret bits - * This takes the same parameters as jiff-instance.share, but returns an array of secret bit shares per sending party. - * Each bit array starts with the least significant bit at index 0, and most significant bit at index length-1 - * @method - * @memberof bits - * @param {number} secret - the number to share (this party's input) - * @param {number} [bit_length=jiff_instance.Zp] - the number of generated bits, if the secret has less bits, it will be - * padded with zeros - * @param {number} [threshold=receivers_list.length] - threshold of each shared bit - * @param {Array} [receivers_list=all_parties] - receivers of every bits - * @param {Array} [senders_list=all_parties] - senders of evey bit - * @param {number} [Zp=jiff_instance.Zp] - the field of sharing for every bit - * @param {string|number} [share_id=auto_gen()] - synchronization id - * @returns {object} a map (of size equal to the number of parties) - * where the key is the party id (from 1 to n) - * and the value is an array of secret shared bits - */ - jiffClient.protocols.bits.share = sharing.share_bits.bind(null, jiffClient); - /** - * Opens the given array of secret shared bits. - * This works regardless of whether the represented value fit inside the corresponding field or not - * @method - * @memberof bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits - * @param {number[]} parties - parties to open (same as jiff_instance.open) - * @param {string|number} [op_id=auto_gen()] - same as jiff_instance.open - * @returns {promise} a promise to the number represented by bits - */ - jiffClient.protocols.bits.open = sharing.open_bits.bind(null, jiffClient); - /** - * Receives an opening of an array of secret bits without owning shares of the underlying value. - * Similar to jiff.receive_open() but for bits. - * This works regardless of whether the represented value fit inside the corresponding field or not - * @method - * @memberOf jiff-instance.protocols.bits - * @param {Array} senders - an array with party ids (1 to n) specifying the parties sending the shares - * @param {Array} [receivers=all_parties] - an array with party ids (1 to n) specifying the parties receiving the result - * @param {number} [count=ceil(log2(Zp))] - the number of bits being opened - * @param {number} [threshold=parties.length] - the min number of parties needed to reconstruct the secret, defaults to all the senders - * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used) - * @param {string|number|object} [op_id=auto_gen()] - unique and consistent synchronization id between all parties - * @returns {promise} a (JQuery) promise to the open value of the secret - */ - jiffClient.protocols.bits.receive_open = sharing.receive_open_bits.bind(null, jiffClient); -}; -},{"../protocols/bits/arithmetic.js":25,"../protocols/bits/comparison.js":26,"../protocols/bits/protocols.js":27,"../protocols/bits/sharing.js":28}],3:[function(require,module,exports){ -module.exports = function (jiffClient) { - /** - * Requests secret(s) from the server (crypto provider) of type matching the given label. - * @method from_crypto_provider - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string} label - the type of secret(s) being requested from crypto_provider (e.g. triplet, quotient, numbers, etc) - * @param {Array} [receivers_list=all_parties] - array of party ids that want to receive the secret(s), by default, this includes all parties. - * @param {number} [threshold=receivers_list.length] - the min number of parties needed to reconstruct the secret(s). - * @param {number} [Zp=jiff_instance.Zp] - the mod, defaults to the Zp of the instance. - * @param {string} [op_id=auto_Gen()] - an id which is used to identify the secret requested, so that every party - * gets a share from the same secret for every matching instruction. An automatic id - * is generated by increasing a local counter per label, default ids suffice when all - * parties execute all instructions in the same order. - * @param {object} [params={}] - any additional parameters specific to the label, these are defined by the label handler at the server side. - * some of these parameters may be optional, while others may be required. - * @returns {promise} a promise to the secret(s) provided by the server/crypto provider, the promise returns an object with the given format: - * { values: , shares: } - */ - jiffClient.from_crypto_provider = function (label, receivers_list, threshold, Zp, op_id, params) { - // defaults - if (Zp == null) { - Zp = jiffClient.Zp; - } - if (receivers_list == null) { - receivers_list = []; - for (var i = 1; i <= jiffClient.party_count; i++) { - receivers_list.push(i); - } +(function (f) { + if (typeof exports === 'object' && typeof module !== 'undefined') { + module.exports = f(); + } else if (typeof define === 'function' && define.amd) { + define([], f); + } else { + var g; + if (typeof window !== 'undefined') { + g = window; + } else if (typeof global !== 'undefined') { + g = global; + } else if (typeof self !== 'undefined') { + g = self; } else { - jiffClient.helpers.sort_ids(receivers_list); - } - if (threshold == null) { - threshold = receivers_list.length; - } - if (op_id == null) { - op_id = jiffClient.counters.gen_op_id('crypto_provider:' + label, receivers_list); - } - if (params == null) { - params = {}; + g = this; + } + g.JIFFClient = f(); + } +})(function () { + var define, module, exports; + return (function () { + function r(e, n, t) { + function o(i, f) { + if (!n[i]) { + if (!e[i]) { + var c = 'function' == typeof require && require; + if (!f && c) return c(i, !0); + if (u) return u(i, !0); + var a = new Error("Cannot find module '" + i + "'"); + throw ((a.code = 'MODULE_NOT_FOUND'), a); + } + var p = (n[i] = { exports: {} }); + e[i][0].call( + p.exports, + function (r) { + var n = e[i][1][r]; + return o(n || r); + }, + p, + p.exports, + r, + e, + n, + t + ); + } + return n[i].exports; + } + for (var u = 'function' == typeof require && require, i = 0; i < t.length; i++) o(t[i]); + return o; } + return r; + })()( + { + 1: [ + function (require, module, exports) { + var initialization = require('./api/initialization.js'); + var sharing = require('./api/sharing.js'); + var custom = require('./api/custom.js'); + var crypto_provider = require('./api/crypto_provider.js'); + var synchronization = require('./api/synchronization.js'); + + var protocols = require('./api/protocols.js'); + var bitsProtocols = require('./api/bits.js'); + + module.exports = function (jiffClient) { + initialization(jiffClient); + sharing(jiffClient); + custom(jiffClient); + crypto_provider(jiffClient); + synchronization(jiffClient); + + protocols(jiffClient); + bitsProtocols(jiffClient); + }; + }, + { './api/bits.js': 2, './api/crypto_provider.js': 3, './api/custom.js': 4, './api/initialization.js': 5, './api/protocols.js': 6, './api/sharing.js': 7, './api/synchronization.js': 8 } + ], + 2: [ + function (require, module, exports) { + var arithmetic = require('../protocols/bits/arithmetic.js'); + var comparison = require('../protocols/bits/comparison.js'); + var otherProtocols = require('../protocols/bits/protocols.js'); + var sharing = require('../protocols/bits/sharing.js'); + + /** + * Contains bits protocols (including rejection sampling and bits operations) + * + * Important: bit protocols (including bit_decomposition) are unaware of any extension specific customizations, and will operate as + * on the given shares as if they are natural numbers in Zp. Make sure to take into consideration any magnification/shift transformations + * needed to translate correctly between plain representations and extension representations of bits! + * @alias bits + * @namespace + */ + + module.exports = function (jiffClient) { + /** + * Compute sum of bitwise secret shared number and a constant + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - the bit wise secret shares + * @param {number} constant - the constant + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length here will be max(|bits|, |constant|) + 1 + * in case of potential overflow / carry + */ + jiffClient.protocols.bits.cadd = arithmetic.cadd.bind(null, jiffClient); + /** + * Compute [secret bits] - [constant bits] + * @method + * @memberof bits + * @param {number} constant - the constant + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - the bit wise secret shares + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where + * the bit at index 0 is the least significant bit. The bit at index 1 is the most significant bit, + * and the bit at index |bits| is 1 if the result overflows, or 0 otherwise + */ + jiffClient.protocols.bits.csubl = arithmetic.csubl.bind(null, jiffClient); + /** + * Compute [constant bits] - [secret bits] + * @method + * @memberof bits + * @param {number} constant - the constant + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - the bit wise secret shares + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where + * the bit at index 0 is the least significant bit. The bit at index 1 is the most significant bit, + * and the bit at index |bits| is 1 if the result overflows, or 0 otherwise + */ + jiffClient.protocols.bits.csubr = arithmetic.csubr.bind(null, jiffClient); + /** + * + * Compute [secret bits1] + [secret bits2] + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - the first bitwise shared number: array of secrets with index 0 being least significant bit + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number (length may be different) + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where + * the bit at index 0 is the least significant bit + */ + jiffClient.protocols.bits.sadd = arithmetic.sadd.bind(null, jiffClient); + /** + * Compute [secret bits1] - [secret bits2] + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - first bitwise secret shared number: lower indices represent less significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - second bitwise secret shared number (length may be different) + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where + * the bit at index 0 is the least significant bit. The bit at index 1 is the most significant bit, + * and the bit at index |bits| is 1 if the result overflows, or 0 otherwise + */ + jiffClient.protocols.bits.ssub = arithmetic.ssub.bind(null, jiffClient); + /** + * Compute [secret bits] * constant + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - bitwise shared secret to multiply: lower indices represent less significant bits + * @param {number} constant - constant to multiply with + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result, the length of the result will be bits.length + ceil(log2(constant)), except + * if constant is zero, the result will then be [ zero share ] + */ + jiffClient.protocols.bits.cmult = arithmetic.cmult.bind(null, jiffClient); + /** + * Compute [secret bits1] * [secret bits2] + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - bitwise shared secret to multiply: lower indices represent less significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - bitwise shared secret to multiply + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result, the length of the result will be bits1.length + bits2.length + */ + jiffClient.protocols.bits.smult = arithmetic.smult.bind(null, jiffClient); + /** + * Computes integer division of [secret bits 1] / [secret bits 2] + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {{quotient: module:jiff-client~JIFFClient#SecretShare[], remainder: module:jiff-client~JIFFClient#SecretShare[]}} the quotient and remainder bits arrays, note that + * the quotient array has the same length as bits1, + * and the remainder array has the same length as bits2 or bits1, whichever is smaller. + * Note: if bits2 represent 0, the returned result is the maximum + * number that fits in the number of bits (all 1), and the remainder + * is equal to bits1 + */ + jiffClient.protocols.bits.sdiv = arithmetic.sdiv.bind(null, jiffClient); + /** + * Computes integer division of [secret bits] / constant + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - numerator: an array of secret shares of bits, starting from least to most significant bits + * @param {number} constant - the denominator number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {{quotient: module:jiff-client~JIFFClient#SecretShare[], remainder: module:jiff-client~JIFFClient#SecretShare[]}} the quotient and remainder bits arrays, note that + * the quotient array has the same length as bits, + * and the remainder array has the same length as + * constant or bits, whichever is smaller + * @throws if constant is 0. + */ + jiffClient.protocols.bits.cdivl = arithmetic.cdivl.bind(null, jiffClient); + /** + * Computes integer division of constant / [secret bits] + * @method + * @memberof bits + * @param {number} constant - the numerator number + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - denominator: an array of secret shares of bits, starting from least to most significant bits + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {{quotient: module:jiff-client~JIFFClient#SecretShare[], remainder: module:jiff-client~JIFFClient#SecretShare[]}} the quotient and remainder bits arrays, note that + * the quotient array has the same length as the number of bits in constant, + * and the remainder array has the same length as bits or constant, whichever is smaller. + * Note: if bits represent 0, the returned result is the maximum + * number that fits in its bits (all 1), and the remainder + * is equal to constant + */ + jiffClient.protocols.bits.cdivr = arithmetic.cdivr.bind(null, jiffClient); + + /** + * Checks whether the given bitwise secret shared number and numeric constant are equal + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of secret shares of bits, starting from least to most significant bits + * @param {number} constant - the constant number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {SecretShare|boolean} a secret share of 1 if parameters are equal, 0 otherwise. If result is known + * (e.g. constant has a greater non-zero bit than bits' most significant bit), the result is + * returned immediately as a boolean + */ + jiffClient.protocols.bits.ceq = comparison.ceq.bind(null, jiffClient); + /** + * Checks whether the given bitwise secret shared number and numeric constant are not equal + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of secret shares of bits, starting from least to most significant bits + * @param {number} constant - the constant number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {SecretShare|boolean} a secret share of 1 if parameters are not equal, 0 otherwise. If result is known + * (e.g. constant has a greater non-zero bit than bits' most significant bit), the result is + * returned immediately as a boolean + */ + jiffClient.protocols.bits.cneq = comparison.cneq.bind(null, jiffClient); + /** + * Checks whether given secret shared bits are greater than the given constant + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits + * @param {number} constant - the constant number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {SecretShare|boolean} a secret share of 1 if bits are greater than constant, 0 otherwise, if result is known + * (e.g. constant has greater non-zero bit than bits' most significant bit), the result is + * returned immediately as a boolean + */ + jiffClient.protocols.bits.cgt = comparison.cgt.bind(null, jiffClient); + /** + * Checks whether given secret shared bits are greater or equal to the given constant + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits + * @param {number} constant - the constant number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {SecretShare|boolean} a secret share of 1 if bits are greater or equal to constant, 0 otherwise, if result is known + * (e.g. constant has greater non-zero bit than bits' most significant bit or constant is zero), the result is + * returned immediately as a boolean + */ + jiffClient.protocols.bits.cgteq = comparison.cgteq.bind(null, jiffClient); + /** + * Checks whether given secret shared bits are less than the given constant + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits + * @param {number} constant - the constant number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {SecretShare|boolean} a secret share of 1 if bits are less than the constant, 0 otherwise, if result is known + * (e.g. constant has greater non-zero bit than bits' most significant bit), the result is + * returned immediately as a boolean + */ + jiffClient.protocols.bits.clt = comparison.clt.bind(null, jiffClient); + /** + * Checks whether given secret shared bits are less or equal to the given constant + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits + * @param {number} constant - the constant number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {SecretShare|boolean} a secret share of 1 if bits are less or equal to constant, 0 otherwise, if result is known + * (e.g. constant has greater non-zero bit than bits' most significant bit), the result is + * returned immediately as a boolean + */ + jiffClient.protocols.bits.clteq = comparison.clteq.bind(null, jiffClient); + /** + * Checks whether the two given bitwise secret shared numbers are equal + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if bits are equal, 0 otherwise + */ + jiffClient.protocols.bits.seq = comparison.seq.bind(null, jiffClient); + /** + * Checks whether the two given bitwise secret shared numbers are not equal + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if bits are not equal, 0 otherwise + */ + jiffClient.protocols.bits.sneq = comparison.sneq.bind(null, jiffClient); + /** + * Checks whether the first given bitwise secret shared number is greater than the second bitwise secret shared number + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is greater than the second, 0 otherwise + */ + jiffClient.protocols.bits.sgt = comparison.sgt.bind(null, jiffClient); + /** + * Checks whether the first given bitwise secret shared number is greater than or equal to the second bitwise secret shared number + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is greater or equal to the second, 0 otherwise + */ + jiffClient.protocols.bits.sgteq = comparison.sgteq.bind(null, jiffClient); + /** + * Checks whether the first given bitwise secret shared number is less than the second bitwise secret shared number + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is less than the second, 0 otherwise + */ + jiffClient.protocols.bits.slt = comparison.slt.bind(null, jiffClient); + /** + * Checks whether the first given bitwise secret shared number is less or equal to the second bitwise secret shared number + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is less than or equal to the second, 0 otherwise + */ + jiffClient.protocols.bits.slteq = comparison.slteq.bind(null, jiffClient); + + /** + * Retrieves preprocessed rejection_sampling bits or performs the rejection sampling on the fly if crypto_provider is enabled + * @method + * @memberof bits + * @param {number} lower_bound - the lower bound, included (can be a bigNumber if using bigNumber extension) + * @param {number} upper_bound - the upper bound, excluded (can be a bigNumber if using bigNumber extension) + * @param {number} [threshold=parties.length] - the threshold of the resulting shares after sampling + * @param {Array} [parties=all_parties] - array of party ids that want to receive (or compute if needed) the sampling shares, by default, this includes all parties + * @param {number} [Zp=jiff-instance.Zp] - the mod (if null then the default Zp for the instance is used) + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly + * @returns {SecretShare[]} an array of secret shares, each representing a bit from the sampled value (from least to most significant) + */ + jiffClient.protocols.bits.rejection_sampling = otherProtocols.rejection_sampling.bind(null, jiffClient); + + /** + * Creates a secret share of the number represented by the given array of secret shared bits. + * Requires no communication, only local operations + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits. + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of the number represented by bits. + */ + jiffClient.protocols.bits.bit_composition = otherProtocols.bit_composition; + + /** + * Share a number as an array of secret bits + * This takes the same parameters as jiff-instance.share, but returns an array of secret bit shares per sending party. + * Each bit array starts with the least significant bit at index 0, and most significant bit at index length-1 + * @method + * @memberof bits + * @param {number} secret - the number to share (this party's input) + * @param {number} [bit_length=jiff_instance.Zp] - the number of generated bits, if the secret has less bits, it will be + * padded with zeros + * @param {number} [threshold=receivers_list.length] - threshold of each shared bit + * @param {Array} [receivers_list=all_parties] - receivers of every bits + * @param {Array} [senders_list=all_parties] - senders of evey bit + * @param {number} [Zp=jiff_instance.Zp] - the field of sharing for every bit + * @param {string|number} [share_id=auto_gen()] - synchronization id + * @returns {object} a map (of size equal to the number of parties) + * where the key is the party id (from 1 to n) + * and the value is an array of secret shared bits + */ + jiffClient.protocols.bits.share = sharing.share_bits.bind(null, jiffClient); + /** + * Opens the given array of secret shared bits. + * This works regardless of whether the represented value fit inside the corresponding field or not + * @method + * @memberof bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits + * @param {number[]} parties - parties to open (same as jiff_instance.open) + * @param {string|number} [op_id=auto_gen()] - same as jiff_instance.open + * @returns {promise} a promise to the number represented by bits + */ + jiffClient.protocols.bits.open = sharing.open_bits.bind(null, jiffClient); + /** + * Receives an opening of an array of secret bits without owning shares of the underlying value. + * Similar to jiff.receive_open() but for bits. + * This works regardless of whether the represented value fit inside the corresponding field or not + * @method + * @memberOf jiff-instance.protocols.bits + * @param {Array} senders - an array with party ids (1 to n) specifying the parties sending the shares + * @param {Array} [receivers=all_parties] - an array with party ids (1 to n) specifying the parties receiving the result + * @param {number} [count=ceil(log2(Zp))] - the number of bits being opened + * @param {number} [threshold=parties.length] - the min number of parties needed to reconstruct the secret, defaults to all the senders + * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used) + * @param {string|number|object} [op_id=auto_gen()] - unique and consistent synchronization id between all parties + * @returns {promise} a (JQuery) promise to the open value of the secret + */ + jiffClient.protocols.bits.receive_open = sharing.receive_open_bits.bind(null, jiffClient); + }; + }, + { '../protocols/bits/arithmetic.js': 25, '../protocols/bits/comparison.js': 26, '../protocols/bits/protocols.js': 27, '../protocols/bits/sharing.js': 28 } + ], + 3: [ + function (require, module, exports) { + module.exports = function (jiffClient) { + /** + * Requests secret(s) from the server (crypto provider) of type matching the given label. + * @method from_crypto_provider + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string} label - the type of secret(s) being requested from crypto_provider (e.g. triplet, quotient, numbers, etc) + * @param {Array} [receivers_list=all_parties] - array of party ids that want to receive the secret(s), by default, this includes all parties. + * @param {number} [threshold=receivers_list.length] - the min number of parties needed to reconstruct the secret(s). + * @param {number} [Zp=jiff_instance.Zp] - the mod, defaults to the Zp of the instance. + * @param {string} [op_id=auto_Gen()] - an id which is used to identify the secret requested, so that every party + * gets a share from the same secret for every matching instruction. An automatic id + * is generated by increasing a local counter per label, default ids suffice when all + * parties execute all instructions in the same order. + * @param {object} [params={}] - any additional parameters specific to the label, these are defined by the label handler at the server side. + * some of these parameters may be optional, while others may be required. + * @returns {promise} a promise to the secret(s) provided by the server/crypto provider, the promise returns an object with the given format: + * { values: , shares: } + */ + jiffClient.from_crypto_provider = function (label, receivers_list, threshold, Zp, op_id, params) { + // defaults + if (Zp == null) { + Zp = jiffClient.Zp; + } + if (receivers_list == null) { + receivers_list = []; + for (var i = 1; i <= jiffClient.party_count; i++) { + receivers_list.push(i); + } + } else { + jiffClient.helpers.sort_ids(receivers_list); + } + if (threshold == null) { + threshold = receivers_list.length; + } + if (op_id == null) { + op_id = jiffClient.counters.gen_op_id('crypto_provider:' + label, receivers_list); + } + if (params == null) { + params = {}; + } + + // Send a request to the server + var msg = { label: label, op_id: op_id, receivers: receivers_list, threshold: threshold, Zp: Zp, params: params }; + msg = jiffClient.hooks.execute_array_hooks('beforeOperation', [jiffClient, 'crypto_provider', msg], 2); + msg = JSON.stringify(msg); + + // Setup deferred to handle receiving the result later. + jiffClient.deferreds[op_id] = new jiffClient.helpers.Deferred(); + var result = jiffClient.deferreds[op_id].promise; + + // send a request to the server. + jiffClient.socket.safe_emit('crypto_provider', msg); + return result; + }; + }; + }, + {} + ], + 4: [ + function (require, module, exports) { + module.exports = function (jiffClient) { + /** + * Sends a custom message to a subset of parties + * @memberof module:jiff-client~JIFFClient + * @method emit + * @instance + * @param {string} tag - the tag to attach to the message + * @param {Array} [receivers=all_parties] - contains the party ids to receive the message, defaults to all parties + * @param {string} message - the message to send + * @param {boolean} [encrypt=true] - if true, messages will be encrypted + */ + jiffClient.emit = function (tag, receivers, message, encrypt) { + if (typeof message !== 'string') { + throw new Error('Emit: message must be a string'); + } + + if (receivers == null) { + receivers = []; + for (var i = 1; i <= jiffClient.party_count; i++) { + receivers.push(i); + } + } + + // send to all other parties + for (var p = 0; p < receivers.length; p++) { + if (receivers[p] === jiffClient.id) { + continue; + } + + var message_to_send = { tag: tag, party_id: receivers[p], message: message, encrypted: encrypt }; + message_to_send = jiffClient.hooks.execute_array_hooks('beforeOperation', [jiffClient, 'custom', message_to_send], 2); + + if (message_to_send['encrypted'] !== false) { + message_to_send['message'] = jiffClient.hooks.encryptSign(jiffClient, message_to_send['message'], jiffClient.keymap[message_to_send['party_id']], jiffClient.secret_key); + message_to_send['encrypted'] = true; + } + + jiffClient.socket.safe_emit('custom', JSON.stringify(message_to_send)); + } + + // receive our own message if specified + if (receivers.indexOf(jiffClient.id) > -1) { + jiffClient.handlers.receive_custom({ tag: tag, party_id: jiffClient.id, message: message, encrypted: false }); + } + }; + + /** + * Registers the given function as a listener for messages with the given tag. + * Removes any previously set listener for this tag. + * @memberof module:jiff-client~JIFFClient + * @method listen + * @instance + * @param {string} tag - the tag to listen for. + * @param {function(party_id, string)} handler - the function that handles the received message: takes the sender id and the message as parameters. + */ + jiffClient.listen = function (tag, handler) { + jiffClient.listeners[tag] = handler; + + var stored_messages = jiffClient.custom_messages_mailbox[tag]; + if (stored_messages == null) { + return; + } + + for (var i = 0; i < stored_messages.length; i++) { + var sender_id = stored_messages[i].sender_id; + var message = stored_messages[i].message; + handler(sender_id, message); + } + + delete jiffClient.custom_messages_mailbox[tag]; + }; + + /** + * Removes the custom message listener attached to the given tag + * @memberof module:jiff-client~JIFFClient + * @method remove_listener + * @instance + * @param {string} tag - the tag of the listener to remove + */ + jiffClient.remove_listener = function (tag) { + delete jiffClient.listeners[tag]; + }; + }; + }, + {} + ], + 5: [ + function (require, module, exports) { + module.exports = function (jiffClient) { + /** + * Wait until the public keys of these parties are known. + * The public keys may be known before the parties connect (if provided in the options), + * or they could be sent by the server after the parties connect. + * Computation specified in the callback may assume that these parties are connected, + * if they are not, the server will handle storing and relaying the needed messages + * to them when they connect. + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {Array} parties - an array of party ids to wait for, must explicitly include 's1' if callback must wait for the server. + * @param {function(jiff-instance)} callback - the function to execute when these parties are known. + * @param {boolean} [wait_for_initialization=true] - specifies whether to wait for initialization to be complete + * before executing the callback (even if parties are available). + * Set this to false if you do not need the party count and this + * party's id, or if you already have them, and you are certain + * they will be accepted by the server on initialization. + */ + jiffClient.wait_for = function (parties, callback, wait_for_initialization) { + if (wait_for_initialization == null) { + wait_for_initialization = true; + } + + jiffClient.wait_callbacks.push({ parties: parties, callback: callback, initialization: wait_for_initialization }); + jiffClient.execute_wait_callbacks(); // See if the callback can be executed immediately + }; + + /** + * Disconnects from the computation. + * Allows the client program to exit. + * @method disconnect + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {boolean} [safe=false] - if true, jiff will disconnect safely (i.e. after ensuring all + * outgoing pending messages were delivered). + * @param {boolean} [free=false] - if set to true, it means this party's disconnection is final, and all resources + * associated with this party must be freed. + * If all parties in a computation are freed, then all resources associated with the + * computation are freed, and any subsequent reconnection to the computation is as + * if a the connection is for a fresh new computation. + * @param {function()} [callback] - executed after the instance safely disconnects, if safe is set to false, this + * parameter is ignored. + */ + jiffClient.disconnect = function (safe, free, callback) { + if (safe) { + jiffClient.socket.safe_disconnect(free, callback); + } else { + if (free) { + jiffClient.free(); + } + jiffClient.socket.disconnect(); + } + }; + + /** + * Emits event to free up all the resources allocated for this party on the server. + * It is best not to call this function directly, as it can break things if resources still need to be used. + * Instead, use jiff.disconnect(safe, free, callback) to free after safely disconnecting. + * @see {@link module:jiff-client~JIFFClient#disconnect} + * @method free + * @memberof module:jiff-client~JIFFClient + * @instance + */ + jiffClient.free = function () { + var msg = jiffClient.hooks.execute_array_hooks('beforeOperation', [jiffClient, 'free', {}], 2); + jiffClient.socket.safe_emit('free', JSON.stringify(msg)); + }; + }; + }, + {} + ], + 6: [ + function (require, module, exports) { + var numbers = require('../protocols/preprocessing/numbers.js'); + var bits = require('../protocols/preprocessing/bits.js'); + var triplets = require('../protocols/preprocessing/triplets.js'); + var quotients = require('../protocols/preprocessing/quotients.js'); + var sampling = require('../protocols/preprocessing/sampling.js'); + + /** + * Contains miscellaneous protocols (mostly used in preprocessing) + * @name protocols + * @alias protocols + * @namespace + */ + module.exports = function (jiffClient) { + /** + * Creates shares of an unknown random number. Every party comes up with its own random number and shares it. + * Then every party combines all the received shares to construct one share of the random unknown number. + * @method + * @memberof protocols + * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed. + * @param {Array} receivers_list - array of party ids to receive the result. + * @param {Array} compute_list - array of party ids to perform the protocol. + * @param {number} Zp - the mod. + * @param {object} params - an object containing extra parameters passed by the user. + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated. + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * @return {Object} contains 'share' (this party's share of the result) and 'promise'. + */ + jiffClient.protocols.generate_random_number = numbers.generate_random_number.bind(null, jiffClient); + + /** + * Creates shares of 0, such that no party knows the other parties' shares. + * Every party secret shares 0, then every party sums all the shares they received, resulting + * in a new share of 0 for every party. + * @method + * @memberof protocols + * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed. + * @param {Array} receivers_list - array of party ids to receive the result. + * @param {Array} compute_list - array of party ids to perform the protocol. + * @param {number} Zp - the mod. + * @param {object} params - an object containing extra parameters passed by the user. + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated. + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * @return {Object} contains 'share' (this party's share of the result) and 'promise'. + */ + jiffClient.protocols.generate_zero = numbers.generate_zero.bind(null, jiffClient); + + /** + * Generates a random bit under MPC by xoring all bits sent by participating parties using smult / beaver triples + * @method + * @memberof protocols + * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed. + * @param {Array} receivers_list - array of party ids to receive the result. + * @param {Array} compute_list - array of party ids to perform the protocol. + * @param {number} Zp - the mod. + * @param {object} params - an object containing extra parameters passed by the user. + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated. + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * @return {Object} contains 'share' (this party's share of the generated bit) and 'promise'. + */ + jiffClient.protocols.generate_random_bit_smult = bits.generate_random_bit_smult.bind(null, jiffClient); + + /** + * Generates a random bit under MPC by xoring all bits sent by participating parties using smult_bgw + * @method + * @memberof protocols + * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed. + * @param {Array} receivers_list - array of party ids to receive the result. + * @param {Array} compute_list - array of party ids to perform the protocol. + * @param {number} Zp - the mod. + * @param {object} params - an object containing extra parameters passed by the user. + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated. + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * @return {Object} contains 'share' (this party's share of the generated bit) and 'promise'. + */ + jiffClient.protocols.generate_random_bit_bgw = bits.generate_random_bit_bgw.bind(null, jiffClient); + + /** + * Generates a sequence of random bits under MPC. + * @method + * @memberof protocols + * @param {number} [threshold=receivers_list.length] - the threshold of the bit when stored by receivers after generation. * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed. + * @param {Array} receivers_list - array of party ids to receive the result. + * @param {Array} compute_list - array of party ids to perform the protocol. + * @param {number} Zp - the mod. + * @param {object} params - an object containing extra parameters passed by the user. + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated. + * - count: how many random bits to generate. + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * @param {object} protocols - the protocols to use for preprocessing. + * @return {Object} contains 'share' (array of secret shares bits) and 'promise'. + */ + jiffClient.protocols.generate_random_bits = bits.generate_random_bits.bind(null, jiffClient); + + /** + * Generation of beaver triplet via MPC, uses the server for communication channels, but not for generation + * @method + * @memberof protocols + * @param {number} threshold - the threshold of the triplets when stored by receivers after generation + * @param {Array} receivers_list - array of party ids that want to receive the triplet shares + * @param {Array} compute_list - array of party ids that will perform this protocol + * @param {number} Zp - the mod + * @param {object} params - an object containing extra parameters passed by the user + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated + * - an optional number compute_threshold parameter, which specifies threshold used + * during the protocol execution. By default, this is the length of the (compute_list+1)/2 + * @param {object} protocols - the sub protocols to use for preprocessing + * @return {object} all pre-processing protocols must return an object with these keys: + * { + * 'share': the share(s)/value(s) to store attached to op_id for later use by the computation (i.e. the result of preprocessing), + * 'promise': a promise for when this protocol is fully completed (could be null if the protocol was already completed) + * } + * In this case, 'share' is an array of this party's shares of the resulting triplet, a,b,c such that a*b=c + */ + jiffClient.protocols.generate_beaver_bgw = triplets.bind(null, jiffClient); + + /** + * Creates shares of r and x, such that r is a uniform random number between 0 and Zp, and x is floor(r/constant) + * where constant is provided by the extra params + * @method + * @memberof protocols + * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed + * @param {Array} receivers_list - array of party ids to receive the result + * @param {Array} compute_list - array of party ids to perform the protocol + * @param {number} Zp - the mod + * @param {object} params - an object containing extra parameters passed by the user + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * - constant: the constant to divide the random number by. + * - output_op_id: the set op id of the output quotient and noise + * @return {Object} contains 'share' (this party's share of the result) and 'promise' + */ + jiffClient.protocols.generate_random_and_quotient = quotients.bind(null, jiffClient); + + /** + * Wrapper for when doing rejection sampling during pre processing + * + * Do not use this function directly, especially during online computation time, use jiffClient.protocols.bits.rejection_sampling instead + * + * @method + * @memberof protocols + * @param {number} [threshold=receivers_list.length] - the threshold of the resulting shares after sampling + * @param {Array} [receivers_list=all_parties] - array of party ids that want to receive the sampling shares, by default, this includes all parties + * @param {Array} [compute_list=all_parties] - array of party ids that will perform this protocol, by default, this includes all parties + * @param {number} [Zp=jiff-instance.Zp] - the mod (if null then the default Zp for the instance is used) + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly + * @param {object} [params={}] - an object containing extra parameters passed by the user + * Expects: + * - an optional number compute_threshold parameter, which specifies threshold used + * during the protocol execution. By default, this is (|compute_list|+1)/2 + * - optional 'lower_bound' and 'upper_bound' numeric parameters, default to 0 and Zp respectively + * - op_id, the base op_id to tag operations inside this protocol with, defaults to auto generated + * - output_op_id, the tag id for the output result + * - retry_count, how many times rejection sampling have been retried! + * @param {object} [protocols=defaults] - the protocols to use for preprocessing, any protocol(s) not provided will be replaced with defaults + * @returns {Object} an object containing keys: 'share', and 'promise'. The promise is resolved when the rejection sampling is completed. + * The object is consumed by .preprocessing: + * - 'share' attribute contains the resulting array of secret shared bits representing the sampled value, and is stored in the preprocessing table internally + * - The promise is consumed and a new promise is returned by .preprocessing that is resolved after this returned promise (and all other promise generated by that .preprocessing call) are resolved + */ + jiffClient.protocols.rejection_sampling = sampling.bind(null, jiffClient); + }; + }, + { + '../protocols/preprocessing/bits.js': 34, + '../protocols/preprocessing/numbers.js': 35, + '../protocols/preprocessing/quotients.js': 36, + '../protocols/preprocessing/sampling.js': 37, + '../protocols/preprocessing/triplets.js': 38 + } + ], + 7: [ + function (require, module, exports) { + var shareProtocol = require('../protocols/shamir/share.js'); + var openProtocol = require('../protocols/shamir/open.js'); + var reshareProtocol = require('../protocols/shamir/reshare.js'); + var arraysSharing = require('../protocols/arrays/api.js'); + + module.exports = function (jiffClient) { + /** + * Share a secret input + * + * Can be overriden by extensions to customize behavior + * + * @method share + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {number} secret - the number to share (this party's input) + * @param {number} [threshold=receivers_list.length] - the minimum number of parties needed to reconstruct the secret, defaults to all the receivers + * @param {Array} [receivers_list=all_parties] - array of party ids to share with, by default, this includes all parties + * @param {Array} [senders_list=all_parties] - array of party ids to receive from, by default, this includes all parties + * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used) + * @param {string|number} [share_id=auto_gen()] - the tag used to tag the messages sent by this share operation, this tag is used + * so that parties distinguish messages belonging to this share operation from other + * share operations between the same parties (when the order of execution is not + * deterministic). An automatic id is generated by increasing a local counter, default + * ids suffice when all parties execute all sharing operations with the same senders + * and receivers in the same order + * @returns {object} a map (of size equal to the number of sending parties) + * where the key is the party id (between 1 and n), or 's1' if 's1' is specified in the senders_list, + * and the value is the share object that wraps the value received from that party (the internal value maybe + * deferred). + * + * @example + * // share an input value with all parties, and receive all other parties' inputs + * var shares = jiffClient.share(input); + * // my party id is '1', so the first share is mine (technically my share of my input value) + * var my_share = shares[1]; + * // my share of party 2's input + * var p2_share = shares[2]; + */ + jiffClient.share = function (secret, threshold, receivers_list, senders_list, Zp, share_id) { + // type check to confirm the secret to be shared is a number + // for fixed-point extension it should allow non-ints + if (secret != null && (typeof secret !== 'number' || Math.floor(secret) !== secret || secret < 0)) { + throw new Error("secret '" + secret + "' must be a non-negative whole number"); + } + if (secret != null && secret >= (Zp == null ? jiffClient.Zp : Zp)) { + throw new Error("secret '" + secret + "' must fit inside Zp"); + } + return jiffClient.internal_share(secret, threshold, receivers_list, senders_list, Zp, share_id); + }; + + /** + * Same as share, but used by internal JIFF primitives/protocols, do not override this! + * @see {@link module:jiff-client~JIFFClient#share} + * @method internal_share + * @instance + * @memberof module:jiff-client~JIFFClient + */ + jiffClient.internal_share = shareProtocol.jiff_share.bind(null, jiffClient); + + /** + * re-share an existing share (value) under a new threshold or to a new set of parties or both. + * Do not use this to refresh a share (use {@link module:jiff-client~JIFFClient#SecretShare#refresh} instead) + * @method reshare + * @instance + * @memberof module:jiff-client~JIFFClient + * @param {module:jiff-client~JIFFClient#SecretShare} [share=null] - the share you would like to reshare (null if you are a receiver but not a sender) + * @param {number} [threshold=receivers_list.length] - the new threshold, defaults to the length of receivers_list param + * @param {Array} [receivers_list=all_parties] - array of party ids to receive from, by default, this includes all parties + * @param {Array} [senders_list=all_parties] - array of party ids that posses the share and will reshare it with the receivers, by default, this includes all parties + * @param {number} [Zp=jiff.Zp] - the Zp of the existing share + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this multiplication (and internally, the corresponding beaver triplet). + * This id must be unique, and must be passed by all parties to the same instruction. + * this ensures that every party gets a share from the same triplet for every matching instruction. An automatic id + * is generated by increasing a local counter, default ids suffice when all parties execute the + * instructions in the same order + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result under the new threshold, or null if this party is not a receiver + */ + jiffClient.reshare = reshareProtocol.bind(null, jiffClient); + + /** + * Open a secret share to reconstruct secret. + * @method open + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {module:jiff-client~JIFFClient#SecretShare} share - this party's share of the secret to reconstruct. + * @param {Array} [parties=all_parties] - an array with party ids (1 to n) of receiving parties. + * @param {string|number} [op_id=auto_gen()] - the operation id to be used to tag outgoing messages. + * @returns {?promise} a (JQuery) promise to the open value of the secret, null if the party is not specified in the parties array as a receiver. + * @example + * var shares = jiff_instance.share(input); + * //multiply the inputs of party 1 and 2 together + * var result = shares[1].mult(shares[2]); + * // reveal the result of the multiplication to all parties + * return jiff_instance.open(result); + */ + jiffClient.open = openProtocol.jiff_open.bind(null, jiffClient); + + /** + * Same as open, but used by internal JIFF primitives/protocols, do not override this! + * @see {@link module:jiff-client~JIFFClient#open} + * @method internal_open + * @instance + * @memberof module:jiff-client~JIFFClient + */ + jiffClient.internal_open = jiffClient.open; + + /** + * Receive shares from the specified parties and reconstruct their secret. + * Use this function in a party that will receive some answer/value but does not have a share of it. + * @method receive_open + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {Array} senders - an array with party ids (1 to n) specifying the parties sending the shares, this must be provided! + * @param {Array} [receivers=all_parties] - an array with party ids (1 to n) specifying the parties receiving the result + * @param {number} [threshold=senders.length] - the min number of parties needed to reconstruct the secret, defaults to all the senders + * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used) + * @param {string|number} [op_id=auto_Gen()] - same as jiff-instance.open + * @returns {!promise} a (JQuery) promise to the open value of the secret. + */ + jiffClient.receive_open = function (senders, receivers, threshold, Zp, op_id) { + if (senders == null) { + throw new Error('Must provide "senders" parameter in receive_open'); + } + + jiffClient.helpers.sort_ids(senders); + if (receivers == null) { + receivers = []; + for (var i = 1; i <= jiffClient.party_count; i++) { + receivers.push(i); + } + } else { + jiffClient.helpers.sort_ids(receivers); + } + if (Zp == null) { + Zp = jiffClient.Zp; + } + if (threshold == null) { + threshold = senders.length; + } + + var imitationSecretShare = new jiffClient.SecretShare({}, senders, threshold, Zp); + return jiffClient.open(imitationSecretShare, receivers, op_id); + }; + + /** + * Receive arrays of shares from the specified parties and reconstruct their secrets. + * Use this function in a party that will receive some answer/value but does not have a share of it. + * @method receive_open + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {Array} senders - an array with party ids (1 to n) specifying the parties sending the shares, this must be provided! + * @param {number} [threshold=senders.length] - the min number of parties needed to reconstruct the secret, defaults to all the senders + * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used) + * @param {string|number} [op_id=auto_Gen()] - same as jiff-instance.open + * @returns {!promise} a (JQuery) promise to the open value of the secret. + */ + jiffClient.receive_open_array = arraysSharing.jiff_receive_open_ND_array.bind(null, jiffClient); + + /** + * Share an array of values. Each sender may have an array of different length. This is handled by the lengths parameter. + * This function will reveal the lengths of the shared array. + * + * If parties would like to keep the lengths of their arrays secret, they should agree on some public "max" length apriori (either under MPC + * or as part of the logistics of the computation), all their arrays should be padded to that length by using appropriate default/identity + * values + * @method share_array + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {Array} array - the array to be shared. + * @param {null|number|object} [lengths] - the lengths of the arrays to be shared, has the following options:
+ * 1. null: lengths are unknown, each sender will publicly reveal the lengths of its own array.
+ * 2. number: all arrays are of this length
+ * 3. object: { : length }: must specify the length of the array for each sender.
+ * @param {number} [threshold=receivers_list.length] - the min number of parties needed to reconstruct the secret, defaults to all the receivers. + * @param {Array} [receivers_list=all_parties] - array of party ids to share with, by default, this includes all parties. + * @param {Array} [senders_list=all_parties] - array of party ids to receive from, by default, this includes all parties. + * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used). + * @param {string|number} [share_id=auto_gen()] - the base tag used to tag the messages sent by this share operation, every element of the array + * will get a unique id based on the concatenation of base_share_id and the index of the element. + * This tag is used so that parties distinguish messages belonging to this share operation from + * other share operations between the same parties (when the order of execution is not + * deterministic). An automatic id is generated by increasing a local counter, default + * ids suffice when all parties execute all sharing operations with the same senders + * and receivers in the same order. + * @returns {?promise} if the calling party is a receiver then a promise to the shared arrays is returned, the promise will provide an object + * formatted as follows: { : [ <1st_share>, <2nd_share>, ..., <(lengths[party_id])th_share> ] } + * where the party_ids are those of the senders. + * if the calling party is not a receiver, then null is returned. + */ + jiffClient.share_array = arraysSharing.jiff_share_array.bind(null, jiffClient); + + /** + * Share an array of values. Each sender may have an array of different length. This is handled by the lengths parameter. + * This function will reveal the lengths of the shared array. + * + * If parties would like to keep the lengths of their arrays secret, they should agree on some "max" length apriori (either under MPC + * or as part of the logistics of the computation), all their arrays should be padded to that length by using appropriate default/identity + * values. + * @method share_2D_array + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {Array} array - the array to be shared. + * @param {null|number|object} lengths - the lengths of the arrays to be shared. For this to work successfully, the + * same exact value must be used in the calling code for each party. Any missing + * lengths for a row will be automatically publicly revealed by this function. + * Must have the following format: + * 1. null: lengths are unknown, each sender will publicly reveal the lengths of its own array. + * 2. { rows: , cols: , 0: , 1: , ...}: all parties have arrays + * with the given number of rows and cols. In case of jagged 2D arrays, different rows + * can have a different number of cols specified by using : . + * rows is mandatory, cols and any other number matching a specific row are optional. + * 3. { : }: must specify the lengths for each party by using + * an object with the same format as 2. Must include every party. + * @param {number} [threshold=receivers_list.length] - the min number of parties needed to reconstruct the secret, defaults to all the receivers. + * @param {Array} [receivers_list=all_parties] - array of party ids to share with, by default, this includes all parties. + * @param {Array} [senders_list=all_parties] - array of party ids to receive from, by default, this includes all parties. + * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used). + * @param {string|number} [share_id=auto_gen()] - the base tag used to tag the messages sent by this share operation, every element of the array + * will get a unique id based on the concatenation of base_share_id and the index of the element. + * This tag is used so that parties distinguish messages belonging to this share operation from + * other share operations between the same parties (when the order of execution is not + * deterministic). An automatic id is generated by increasing a local counter, default + * ids suffice when all parties execute all sharing operations with the same senders + * and receivers in the same order. + * @returns {promise} if the calling party is a receiver then a promise to the shared arrays is returned, the promise will provide an object + * formatted as follows: { : [ [ <1st_row_shares> ], [<2nd_row_share> ], ..., [ <(lengths[party_id])th_row_shares> ] ] } + * where the party_ids are those of the senders. + * if the calling party is not a receiver, then null is returned. + */ + jiffClient.share_2D_array = arraysSharing.jiff_share_2D_array.bind(null, jiffClient); + + /** + * Share an n-dimensional array of secrets + * The arrays can be of different lengths and dimensions. + * @method share_ND_array + * @memberof module:jiff-client~JIFFClient + * @instance + * @returns {object|promise} + */ + jiffClient.share_ND_array = arraysSharing.jiff_share_ND_array.bind(null, jiffClient); + + /** + * Helper function of share_ND_array + * This method gets called if share_ND_array does not have enough information + * to output an array of shares immediately. Instead it + * returns a promise to an object holding the unknown-size + * arrays from each sender. + * @method share_ND_array_deferred + * @memberof module:jiff-client~JIFFClient + * @instance + * @returns {promise} + */ + //jiffClient.share_ND_array_deferred = arraysSharing.jiff_share_ND_array_deferred.bind(null, jiffClient); + + /** + * Helper function of share_ND_array + * This method gets called if share_ND_array when share_ND_array was given + * sufficients information (in the form of array skeletons) to + * infer the size and shape of each of the senders' arrays. + * This allows it to immediately return an object containing + * the secret-shared arrays from each sender. + * @method share_ND_array_static + * @memberof module:jiff-client~JIFFClient + * @instance + * @returns {object} + */ + //jiffClient.share_ND_array_static = arraysSharing.jiff_share_ND_array_static.bind(null, jiffClient); + + /* + * Wipe a secret array of all secrets but preserve the shape + */ + jiffClient.skeleton_of = arraysSharing.jiff_skeleton_of.bind(null, jiffClient); + + /** + * Opens an array of secret shares. + * @method open_array + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} shares - an array containing this party's shares of the secrets to reconstruct. + * @param {Array} [parties=all_parties] - an array with party ids of receiving parties. + * This must be one of 3 cases: + * 1. null: open all shares to all parties. + * 2. array of numbers: open all shares to all the parties specified in the array. + * 3. array of array of numbers: open share with index i to the parties specified + * in the nested array at parties[i]. if parties[i] was null, + * then shares[i] will be opened to all parties. + * @param {string|number|object} [op_ids=auto_gen()] - an optional mapping that specifies the ID/Tag associated with each + * open message sent. Since open_array involves sending many messages per party, + * this parameter only specifies the BASE OPERATION ID. Each message sent will + * have this base id attached to it concatenated to a counter. + * If this is an object, then it should map an id of a receiving parties + * to the base op_id that should be used to tag the messages sent to that party. + * Parties left unmapped by this object will get an automatically generated id. + * If this is a number/string, then it will be used as the base id tagging all messages + * sent by this open to all parties. + * You can safely ignore this unless you have multiple opens each containing other opens. + * In that case, the order by which these opens are executed is not fully deterministic + * and depends on the order of arriving messages. In this case, use this parameter + * with every nested_open, to ensure ids are unique and define a total ordering on + * the execution of the opens (check implementation of slt for an example). + * @returns {promise} a (JQuery) promise to ALL the open values of the secret, the promise will yield + * a 2D array of values, each corresponding to the given share in the shares parameter + * at the same index. In the case where different values are opened to different parties, the order + * of the values will be preserved, but not the indices, there will be no blanks in the resulting arrays, + * the first share that is opened to this party will appear at index [0], even if it was not initially + * at [0]. + * @throws error if some shares does not belong to the passed jiff instance. + */ + jiffClient.open_array = arraysSharing.jiff_open_array.bind(null, jiffClient); + + /** + * Opens an n-dimensional array of secret shares. + * @method open_ND_array + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {module:jiff-client~JIFFClient#SecretShare|module:jiff-client~JIFFClient#SecretShare[]|module:jiff-client~JIFFClient#SecretShare[][]} shares - an n-dimensional array containing this party's shares of the secrets to reconstruct. + * @param {Array[]} [parties=all_parties] - an array with party ids (1 to n) of receiving parties. + * @param {string|number|object} [op_ids=auto_gen()] - an optional ID/Tag associated with these open operations + * @returns {promise} a (JQuery) promise to ALL the open values of the secret, the promise will yield + * an n-dimensional array of values, each corresponding to the given share in the shares parameter + * at the same index. In the case where different values are opened to different parties, the order + * of the values will be preserved, but not the indices, there will be no blanks in the resulting arrays, + * the first share that is opened to this party will appear at indices [0][0], even if it was not initially + * at [0][0]. + * @throws error if some shares does not belong to the passed jiff instance. + */ + jiffClient.open_ND_array = arraysSharing.jiff_open_ND_array.bind(null, jiffClient); + }; + }, + { '../protocols/arrays/api.js': 21, '../protocols/shamir/open.js': 39, '../protocols/shamir/reshare.js': 40, '../protocols/shamir/share.js': 41 } + ], + 8: [ + function (require, module, exports) { + module.exports = function (jiffClient) { + var maxBarrierId = 10000000; + var currentBarrierId = 0; + var openBarriers = 0; + + /** + * Starts a new barrier, all promises and secret shares created between this call and the corresponding start_barrier + * call will be part of this barrier. start_barrier may be called before previous barriers are resolved, in which + * case promises / secret shares created will be part of the new barrier as well as any previous barriers. + * @memberof module:jiff-client~JIFFClient + * @method start_barrier + * @instance + * @returns {number} a barrier id that identifies this barrier. + */ + jiffClient.start_barrier = function () { + openBarriers++; + currentBarrierId = currentBarrierId + (1 % maxBarrierId); + jiffClient.barriers[currentBarrierId] = []; + return currentBarrierId; + }; + + /** + * Adds given promise to all active barriers. + * @memberof module:jiff-client~JIFFClient + * @method add_to_barriers + * @instance + * @param {promise} promise - the promise to add. + */ + jiffClient.add_to_barriers = function (promise) { + if (openBarriers > 0) { + for (var id in jiffClient.barriers) { + if (jiffClient.barriers.hasOwnProperty(id)) { + jiffClient.barriers[id].push(promise); + } + } + } + }; + + /** + * Executes the callback only after all promises / secret shares in the barrier were resolved. + * @memberof module:jiff-client~JIFFClient + * @method end_barrier + * @instance + * @param {number} [barrier_id=jiff.barriers.length - 1] - identifies the barrier, should be returned by start_barrier. + * by default, barrier_id will refer to the last barrier. + * @returns {promise} a promise that resolves after the secret shares are resolved. + */ + jiffClient.end_barrier = function (barrier_id) { + if (openBarriers === 0) { + return; + } + + openBarriers--; + if (barrier_id == null) { + barrier_id = currentBarrierId; + } + + var promise = Promise.all(jiffClient.barriers[barrier_id]); + delete jiffClient.barriers[barrier_id]; + return promise; + }; + }; + }, + {} + ], + 9: [ + function (require, module, exports) { + // Manages op_id counters and generation + module.exports = function (jiffClient) { + /** + * Resets all the counters for op_ids + * @method counters.reset + * @memberof module:jiff-client~JIFFClient + * @instance + */ + jiffClient.counters.reset = function () { + jiffClient.counters.op_count = {}; + jiffClient.counters.op_count_preprocessing = {}; + + if (jiffClient.counters.pending_opens == null) { + jiffClient.counters.pending_opens = 0; + } + + // stores a seed for generating unique op_ids. + jiffClient.op_id_seed = ''; + }; + + // initialize the counters for the first time + jiffClient.counters.reset(); + + /** + * Shorthand for generating unique operation ids. + * All primitives called after this seed will use their usual default ids prefixed by the seed. + * Helpful when we have nested callbacks/functions (e.g. share_arrays) that may be executed in arbitrary order, + * using this function as a the first call inside such callbacks with an appropriate deterministic unique base_op_id + * ensures that regardless of the order of execution, operations in the same callback are matched correctly across + * all parties. + * Check out demos/graph-pip/mpc.js for an example on using this. + * @method seed_ids + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string|number} base_op_id - the base seed to use as a prefix for all future op_ids. + */ + jiffClient.seed_ids = function (base_op_id) { + if (base_op_id == null || base_op_id === '') { + jiffClient.op_id_seed = ''; + } else { + jiffClient.op_id_seed = base_op_id.toString() + ':'; + } + }; + + /** + * Generate a unique operation id for a new operation object. + * The returned op_id will be unique with respect to other operations, and identifies the same + * operation across all parties, as long as all parties are executing instructions in the same order. + * @method gen_op_id + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string} op - the type/name of operation performed. + * @param {Array} holders - an array containing the ids of all the parties carrying out the operation. + * @return {string} the op_id for the operation. + */ + jiffClient.counters.gen_op_id = function (op, holders) { + var label = jiffClient.op_id_seed + op + ':' + holders.join(','); + if (jiffClient.counters.op_count[label] == null) { + jiffClient.counters.op_count[label] = 0; + } + return label + ':' + jiffClient.counters.op_count[label]++; + }; + + /** + * Generate a unique operation id for a new operation object given two distinct executing parties lists. + * For example, when sharing, this is given two potentially different lists of senders and receivers. + * The returned op_id will be unique with respect to other operations, and identifies the same + * operation across all parties, as long as all parties are executing instructions in the same order. + * @method gen_op_id2 + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string} op - the type/name of operation performed. + * @param {Array} receivers - an array containing the ids of all the parties carrying out the receiving portion of the operation. + * @param {Array} senders - an array containing the ids of all the parties carrying out the sending portion of the operation. + * @return {string} the op_id for the operation. + */ + jiffClient.counters.gen_op_id2 = function (op, receivers, senders) { + var label = jiffClient.op_id_seed + op + ':' + senders.join(',') + ':' + receivers.join(','); + if (jiffClient.counters.op_count[label] == null) { + jiffClient.counters.op_count[label] = 0; + } + return label + ':' + jiffClient.counters.op_count[label]++; + }; + + /** + * Generate a unique operation id for a new operation object (for preprocessing) + * The returned op_id will be unique with respect to other operations, and identifies the same + * operation across all parties, as long as all parties are executing instructions in the same order. + * @method gen_op_id_preprocessing + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string} op - the type/name of operation performed. + * @param {Array} holders - an array containing the ids of all the parties carrying out the operation. + * @return {string} the op_id for the operation. + */ + jiffClient.counters.gen_op_id_preprocessing = function (op, holders) { + var label = jiffClient.op_id_seed + op + ':' + holders.join(','); + if (jiffClient.counters.op_count_preprocessing[label] == null) { + jiffClient.counters.op_count_preprocessing[label] = 0; + } + return label + ':' + jiffClient.counters.op_count_preprocessing[label]++; + }; + + /** + * Generate a unique operation id for a new operation object given two distinct executing parties lists (for preprocessing). + * For example, when sharing, this is given two potentially different lists of senders and receivers. + * The returned op_id will be unique with respect to other operations, and identifies the same + * operation across all parties, as long as all parties are executing instructions in the same order. + * @method gen_op_id2_preprocessing + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string} op - the type/name of operation performed. + * @param {Array} receivers - an array containing the ids of all the parties carrying out the receiving portion of the operation. + * @param {Array} senders - an array containing the ids of all the parties carrying out the sending portion of the operation. + * @return {string} the op_id for the operation. + */ + jiffClient.counters.gen_op_id2_preprocessing = function (op, receivers, senders) { + var label = jiffClient.op_id_seed + op + ':' + senders.join(',') + ':' + receivers.join(','); + if (jiffClient.counters.op_count_preprocessing[label] == null) { + jiffClient.counters.op_count_preprocessing[label] = 0; + } + return label + ':' + jiffClient.counters.op_count_preprocessing[label]++; + }; + }; + }, + {} + ], + 10: [ + function (require, module, exports) { + // Server extensions management system + module.exports = function (JIFFClient) { + /** + * Checks if the given extension is applied. + * @method has_extension + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string} name - the extension name (found in the extension filename as jiff-client-[name].js). + * @return {boolean} true if the extension was applied, false otherwise. + */ + JIFFClient.prototype.has_extension = function (name) { + return this.extensions.indexOf(name) > -1; + }; + + /** + * Checks if a given extension can be safely applied to the instance + * @method can_apply_extension + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string} name - the extension name (found in the filename at jiff-client[name].js) + * @return {boolean} true if the extension can be safely applied, otherwise returns an error message. + */ + JIFFClient.prototype.can_apply_extension = function (name) { + return true; + }; + + /** + * Applies the given extension. + * If the extension is safe (as per can_apply_extension), it will be applied successfully. + * If the extension is not safe to be applied, an exception will be thrown with an appropriate error message. + * @see {@link module:jiff-client~JIFFClient#can_apply_extension} + * @method apply_extension + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {object} ext - the namespace of the extension acquired when the extension is imported, should contain a make_jiff function. + * @param {object} [options={}] - optional options to be passed to the extension. + */ + JIFFClient.prototype.apply_extension = function (ext, options) { + if (options == null) { + options = {}; + } + + var name = ext.name; + var status = this.can_apply_extension(name); + + if (status === true) { + ext.make_jiff(this, options); + + this.extensions.push(name); + this.extension_applied(name, options); + } else { + throw status; + } + }; + + /** + * Called when an extension is applied successfully. Override to change behavior of your extension based on future extensions. + * @method extension_applied + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string} name - the name of the applied extension. + * @param {object} [options={}] - the options passed by the user to the newly applied extension. + */ + JIFFClient.prototype.extension_applied = function (name, options) {}; + }; + }, + {} + ], + 11: [ + function (require, module, exports) { + /** + * The hooks for this instance. + * Checkout the hooks documentation + * @see {@link module:jiff-client~JIFFClient#hooks} + * @name hooks + * @alias hooks + * @namespace + */ + + var crypto = require('../util/crypto.js'); + var shamir_share = require('../protocols/shamir/share.js'); + var shamir_open = require('../protocols/shamir/open.js'); + + function Hooks(jiffClient) { + this.jiffClient = jiffClient; + + // avoid sharing aliases to the same array + for (hook in Hooks.prototype) { + if (Hooks.prototype.hasOwnProperty(hook) && typeof Hooks.prototype[hook].length === 'number' && Hooks.prototype[hook].slice) { + this[hook] = Hooks.prototype[hook].slice(); + } + } - // Send a request to the server - var msg = {label: label, op_id: op_id, receivers: receivers_list, threshold: threshold, Zp: Zp, params: params}; - msg = jiffClient.hooks.execute_array_hooks('beforeOperation', [jiffClient, 'crypto_provider', msg], 2); - msg = JSON.stringify(msg); + // fill in hooks from options + var optionHooks = jiffClient.options.hooks || {}; + for (var hook in optionHooks) { + if (hook === 'afterOperation') { + this[hook] = optionHooks[hook].concat(this[hook]); + } else if (optionHooks.hasOwnProperty(hook)) { + this[hook] = optionHooks[hook]; + } + } + } - // Setup deferred to handle receiving the result later. - jiffClient.deferreds[op_id] = new jiffClient.helpers.Deferred(); - var result = jiffClient.deferreds[op_id].promise; + /** + * Hook for computing shares of a secret + * @method computeShares + * @memberof hooks + * @param jiffClient {module:jiff-client~JIFFClient} - the jiff client instance + * @param secret {number} - the secret to share + * @param parties_list {number[]} - array of party ids to share with + * @param threshold {number} - threshold of sharing + * @param Zp {number} - the field prime + */ + Hooks.prototype.computeShares = shamir_share.jiff_compute_shares; + Hooks.prototype.reconstructShare = shamir_open.jiff_lagrange; + + // Crypto hooks + Hooks.prototype.encryptSign = function (jiffClient, message) { + if (jiffClient.sodium_ !== false) { + return crypto.encrypt_and_sign.apply(null, arguments); + } else { + return message; + } + }; - // send a request to the server. - jiffClient.socket.safe_emit('crypto_provider', msg); - return result; - }; -}; -},{}],4:[function(require,module,exports){ -module.exports = function (jiffClient) { - /** - * Sends a custom message to a subset of parties - * @memberof module:jiff-client~JIFFClient - * @method emit - * @instance - * @param {string} tag - the tag to attach to the message - * @param {Array} [receivers=all_parties] - contains the party ids to receive the message, defaults to all parties - * @param {string} message - the message to send - * @param {boolean} [encrypt=true] - if true, messages will be encrypted - */ - jiffClient.emit = function (tag, receivers, message, encrypt) { - if (typeof(message) !== 'string') { - throw new Error('Emit: message must be a string'); - } + Hooks.prototype.decryptSign = function (jiffClient, cipher) { + if (jiffClient.sodium_ !== false) { + return crypto.decrypt_and_sign.apply(null, arguments); + } else { + return cipher; + } + }; + + Hooks.prototype.generateKeyPair = function (jiffClient) { + if (jiffClient.sodium_ !== false) { + var key = jiffClient.sodium_.crypto_box_keypair(); // this party's public and secret key + return { public_key: key.publicKey, secret_key: key.privateKey }; + } else { + return { public_key: '', secret_key: '' }; + } + }; - if (receivers == null) { - receivers = []; - for (var i = 1; i <= jiffClient.party_count; i++) { - receivers.push(i); - } - } + Hooks.prototype.parseKey = function (jiffClient, keyString) { + if (jiffClient.sodium_ !== false) { + return new Uint8Array(JSON.parse(keyString)); + } else { + return ''; + } + }; - // send to all other parties - for (var p = 0; p < receivers.length; p++) { - if (receivers[p] === jiffClient.id) { - continue; - } + Hooks.prototype.dumpKey = function (jiffClient, key) { + if (jiffClient.sodium_ !== false) { + return '[' + key.toString() + ']'; + } else { + return ''; + } + }; + + // Array Hooks + Hooks.prototype.beforeShare = []; + Hooks.prototype.afterComputeShare = []; + Hooks.prototype.receiveShare = []; + + Hooks.prototype.beforeOpen = []; + Hooks.prototype.receiveOpen = []; + Hooks.prototype.afterReconstructShare = []; + + Hooks.prototype.createSecretShare = []; + + Hooks.prototype.beforeOperation = []; + Hooks.prototype.afterOperation = [ + // parse content of share/open messages to be integers (instead of strings due to encryption/decryption) + function (jiff, label, msg) { + if (label === 'share' || label === 'open') { + msg['share'] = parseInt(msg['share'], 10); + } + return msg; + } + ]; + + /** + * Execute all hooks attached to the given name in order. + * Hooks are executed sequentially such that the first hook's return value is passed into the second and so on. + * @method execute_array_hooks + * @memberof hooks + * @param {string} hook_name - the name of the hook + * @param {Array} params - parameters to pass to the hooks + * @param {number} acc_index - the index in params in which the result of the hooks must be saved, if no hooks + * exist for the name, then params[acc_index] is returned. + * @return {object} returns the result of the last hook. + */ + Hooks.prototype.execute_array_hooks = function (hook_name, params, acc_index) { + var arr = this.jiffClient.hooks[hook_name]; + arr = arr == null ? [] : arr; + + for (var i = 0; i < arr.length; i++) { + params[acc_index] = arr[i].apply(this.jiffClient, params); + } + return params[acc_index]; + }; - var message_to_send = {tag: tag, party_id: receivers[p], message: message, encrypted: encrypt}; - message_to_send = jiffClient.hooks.execute_array_hooks('beforeOperation', [jiffClient, 'custom', message_to_send], 2); + module.exports = Hooks; + }, + { '../protocols/shamir/open.js': 39, '../protocols/shamir/share.js': 41, '../util/crypto.js': 48 } + ], + 12: [ + function (require, module, exports) { + var initializationHandlers = require('./handlers/initialization.js'); + var shareHandlers = require('./handlers/sharing.js'); + var customHandlers = require('./handlers/custom.js'); + var cryptoProviderHandlers = require('./handlers/crypto_provider.js'); + + /** + * Contains handlers for communication events + * @name handlers + * @alias handlers + * @namespace + */ + + // Add handlers implementations + module.exports = function (jiffClient) { + // fill in handlers + initializationHandlers(jiffClient); + shareHandlers(jiffClient); + customHandlers(jiffClient); + cryptoProviderHandlers(jiffClient); + }; + }, + { './handlers/crypto_provider.js': 13, './handlers/custom.js': 14, './handlers/initialization.js': 15, './handlers/sharing.js': 16 } + ], + 13: [ + function (require, module, exports) { + // setup handler for receiving messages from the crypto provider + module.exports = function (jiffClient) { + /** + * Parse crypto provider message and resolve associated promise. + * @method + * @memberof handlers + * @param {object} json_msg - the parsed json message as received by the crypto_provider event, contains 'values' and 'shares' attributes. + */ + jiffClient.handlers.receive_crypto_provider = function (json_msg) { + // Hook + json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'crypto_provider', json_msg], 2); + + var op_id = json_msg['op_id']; + if (jiffClient.deferreds[op_id] == null) { + return; // duplicate message: ignore + } + + // parse msg + var receivers_list = json_msg['receivers']; + var threshold = json_msg['threshold']; + var Zp = json_msg['Zp']; + + // construct secret share objects + var result = {}; + if (json_msg['values'] != null) { + result.values = json_msg['values']; + } + if (json_msg['shares'] != null) { + result.shares = []; + for (var i = 0; i < json_msg['shares'].length; i++) { + result.shares.push(new jiffClient.SecretShare(json_msg['shares'][i], receivers_list, threshold, Zp)); + } + } + + // resolve deferred + jiffClient.deferreds[op_id].resolve(result); + delete jiffClient.deferreds[op_id]; + }; + }; + }, + {} + ], + 14: [ + function (require, module, exports) { + module.exports = function (jiffClient) { + /** + * Called when this party receives a custom tag message from any party (including itself). + * If a custom listener was setup to listen to the tag, the message is passed to the listener. + * Otherwise, the message is stored until such a listener is provided. + * @method + * @memberof handlers + * @param {object} json_msg - the parsed json message as received by the custom event. + */ + jiffClient.handlers.receive_custom = function (json_msg) { + if (json_msg['party_id'] !== jiffClient.id) { + if (json_msg['encrypted'] === true) { + json_msg['message'] = jiffClient.hooks.decryptSign(jiffClient, json_msg['message'], jiffClient.secret_key, jiffClient.keymap[json_msg['party_id']]); + } + + json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'custom', json_msg], 2); + } + + var sender_id = json_msg['party_id']; + var tag = json_msg['tag']; + var message = json_msg['message']; + + if (jiffClient.listeners[tag] != null) { + jiffClient.listeners[tag](sender_id, message); + } else { + // Store message until listener is provided + var stored_messages = jiffClient.custom_messages_mailbox[tag]; + if (stored_messages == null) { + stored_messages = []; + jiffClient.custom_messages_mailbox[tag] = stored_messages; + } + + stored_messages.push({ sender_id: sender_id, message: message }); + } + }; + }; + }, + {} + ], + 15: [ + function (require, module, exports) { + // add handlers for initialization + module.exports = function (jiffClient) { + jiffClient.options.initialization = Object.assign({}, jiffClient.options.initialization); + + /** + * Called when an error occurs + * @method + * @memberof handlers + * @param {string} label - the name of message or operation causing the error + * @param {error|string} error - the error + */ + jiffClient.handlers.error = function (label, error) { + if (jiffClient.options.onError) { + jiffClient.options.onError(label, error); + } + + console.log(jiffClient.id, ':', 'Error from server:', label, '---', error); // TODO: remove debugging + if (label === 'initialization') { + jiffClient.socket.disconnect(); + + if (jiffClient.initialization_counter < jiffClient.options.maxInitializationRetries) { + console.log(jiffClient.id, ':', 'reconnecting..'); // TODO: remove debugging + setTimeout(jiffClient.connect, jiffClient.options.socketOptions.reconnectionDelay); + } + } + }; + + /** + * Builds the initialization message for this instance + * @method + * @memberof handlers + * @return {Object} + */ + jiffClient.handlers.build_initialization_message = function () { + var msg = { + computation_id: jiffClient.computation_id, + party_id: jiffClient.id, + party_count: jiffClient.party_count, + public_key: jiffClient.public_key != null ? jiffClient.hooks.dumpKey(jiffClient, jiffClient.public_key) : undefined + }; + msg = Object.assign(msg, jiffClient.options.initialization); + + // Initialization Hook + return jiffClient.hooks.execute_array_hooks('beforeOperation', [jiffClient, 'initialization', msg], 2); + }; + + /** + * Begins initialization of this instance by sending the initialization message to the server. + * Should only be called after connection is established. + * Do not call this manually unless you know what you are doing, use .connect() instead! + * @method + * @memberof handlers + */ + jiffClient.handlers.connected = function () { + console.log('Connected!', jiffClient.id); // TODO: remove debugging + jiffClient.initialization_counter++; + + if (jiffClient.secret_key == null && jiffClient.public_key == null) { + var key = jiffClient.hooks.generateKeyPair(jiffClient); + jiffClient.secret_key = key.secret_key; + jiffClient.public_key = key.public_key; + } + + // Initialization message + var msg = jiffClient.handlers.build_initialization_message(); + + // Emit initialization message to server + jiffClient.socket.emit('initialization', JSON.stringify(msg)); + }; + + /** + * Called after the server approves initialization of this instance. + * Sets the instance id, the count of parties in the computation, and the public keys + * of initialized parties. + * @method + * @memberof handlers + */ + jiffClient.handlers.initialized = function (msg) { + jiffClient.__initialized = true; + jiffClient.initialization_counter = 0; + + msg = JSON.parse(msg); + msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'initialization', msg], 2); + + jiffClient.id = msg.party_id; + jiffClient.party_count = msg.party_count; + + // Now: (1) this party is connect (2) server (and other parties) know this public key + // Resend all pending messages + jiffClient.socket.resend_mailbox(); + + // store the received public keys and resolve wait callbacks + jiffClient.handlers.store_public_keys(msg.public_keys); + }; + + /** + * Parse and store the given public keys + * @method + * @memberof handlers + * @param {object} keymap - maps party id to serialized public key. + */ + jiffClient.handlers.store_public_keys = function (keymap) { + var i; + for (i in keymap) { + if (keymap.hasOwnProperty(i) && jiffClient.keymap[i] == null) { + jiffClient.keymap[i] = jiffClient.hooks.parseKey(jiffClient, keymap[i]); + } + } + + // Resolve any pending messages that were received before the sender's public key was known + jiffClient.resolve_messages_waiting_for_keys(); + + // Resolve any pending waits that have satisfied conditions + jiffClient.execute_wait_callbacks(); + + // Check if all keys have been received + if (jiffClient.keymap['s1'] == null) { + return; + } + for (i = 1; i <= jiffClient.party_count; i++) { + if (jiffClient.keymap[i] == null) { + return; + } + } + + // all parties are connected; execute callback + if (jiffClient.__ready !== true && jiffClient.__initialized) { + jiffClient.__ready = true; + if (jiffClient.options.onConnect != null) { + jiffClient.options.onConnect(jiffClient); + } + } + }; + }; + }, + {} + ], + 16: [ + function (require, module, exports) { + // adds sharing related handlers + module.exports = function (jiffClient) { + /** + * Store the received share and resolves the corresponding + * deferred if needed. + * @method + * @memberof handlers + * @param {object} json_msg - the parsed json message as received. + */ + jiffClient.handlers.receive_share = function (json_msg) { + // Decrypt share + json_msg['share'] = jiffClient.hooks.decryptSign(jiffClient, json_msg['share'], jiffClient.secret_key, jiffClient.keymap[json_msg['party_id']]); + json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'share', json_msg], 2); + + var sender_id = json_msg['party_id']; + var op_id = json_msg['op_id']; + var share = json_msg['share']; + + // Call hook + share = jiffClient.hooks.execute_array_hooks('receiveShare', [jiffClient, sender_id, share], 2); + + // check if a deferred is set up (maybe the share was received early) + if (jiffClient.deferreds[op_id] == null) { + jiffClient.deferreds[op_id] = {}; + } + if (jiffClient.deferreds[op_id][sender_id] == null) { + // Share is received before deferred was setup, store it. + jiffClient.deferreds[op_id][sender_id] = new jiffClient.helpers.Deferred(); + } + + // Deferred is already setup, resolve it. + jiffClient.deferreds[op_id][sender_id].resolve(share); + }; + + /** + * Resolves the deferred corresponding to operation_id and sender_id. + * @method + * @memberof handlers + * @param {object} json_msg - the json message as received with the open event. + */ + jiffClient.handlers.receive_open = function (json_msg) { + // Decrypt share + if (json_msg['party_id'] !== jiffClient.id) { + json_msg['share'] = jiffClient.hooks.decryptSign(jiffClient, json_msg['share'], jiffClient.secret_key, jiffClient.keymap[json_msg['party_id']]); + json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'open', json_msg], 2); + } + + var sender_id = json_msg['party_id']; + var op_id = json_msg['op_id']; + var share = json_msg['share']; + var Zp = json_msg['Zp']; + + // call hook + share = jiffClient.hooks.execute_array_hooks('receiveOpen', [jiffClient, sender_id, share, Zp], 2); + + // Ensure deferred is setup + if (jiffClient.deferreds[op_id] == null) { + jiffClient.deferreds[op_id] = {}; + } + if (jiffClient.deferreds[op_id].shares == null) { + jiffClient.deferreds[op_id].shares = []; + } + + // Accumulate received shares + jiffClient.deferreds[op_id].shares.push({ value: share, sender_id: sender_id, Zp: Zp }); + + // Resolve when ready + if (jiffClient.deferreds[op_id].shares.length === jiffClient.deferreds[op_id].threshold) { + jiffClient.deferreds[op_id].deferred.resolve(); + } + + // Clean up if done + if (jiffClient.deferreds[op_id] != null && jiffClient.deferreds[op_id].deferred === 'CLEAN' && jiffClient.deferreds[op_id].shares.length === jiffClient.deferreds[op_id].total) { + delete jiffClient.deferreds[op_id]; + } + }; + }; + }, + {} + ], + 17: [ + function (require, module, exports) { + var linkedList = require('../../common/linkedlist.js'); + + module.exports = function (jiffClient) { + var isRunning = false; + var userCallbacks = []; + var preprocessingTasks = [linkedList()]; + + /** + * Checks if the given operation uses preprocessed values + * @method has_preprocessing + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string} op - name of the operation to check + * @return {boolean} true if the op uses preprocessing, false otherwise + */ + jiffClient.has_preprocessing = function (op) { + for (var i = 0; i < jiffClient.extensions.length; i++) { + if (jiffClient.preprocessing_function_map[jiffClient.extensions[i]][op] != null) { + return true; + } + } + + return false; + }; + + /** + * Get a preprocessed share/value by associated op_id. If value does not exist + * Fallback to some user specified way for creating it + * @method get_preprocessing + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string} op_id - the op_id associated with the preprocessed value/share + * @return {object} the preprocessed share(s) + */ + jiffClient.get_preprocessing = function (op_id) { + var values = jiffClient.preprocessing_table[op_id]; + if (values != null) { + return values; + } + if (jiffClient.crypto_provider === true) { + return null; + } + throw new Error('No preprocessed value(s) that correspond to the op_id "' + op_id + '"'); + }; + + /** + * Store a pair of op_id and associated pre-processed value/share + * The value/share can be accessed later during the computation through jiffClient.get_preprocessing(op_id) + * @method store_preprocessing + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string} op_id - the op_id associated with the preprocessed value/share + * @param {SecretShare} share - the share/value to store + */ + jiffClient.store_preprocessing = function (op_id, share) { + if (share != null) { + jiffClient.preprocessing_table[op_id] = share; + } + }; + + /** + * Generate values used for JIFF operations in advance of the computation. + * + * Calling this function does not begin preprocessing, it just creates a preprocessing task. After you created + * your desired tasks, you can ask JIFF to execute them via {@link module:jiff-client~JIFFClient#executePreprocessing}. + * + * @method preprocessing + * @memberof module:jiff-client~JIFFClient + * @instance + * @param {string} dependent_op - name of the operation that will later use the pre_processed values + * @param {Number} [count=1] - number of times the protocol should be performed, number of values that will be generated + * @param {Object} [protocols=defaults] - a mapping from base preprocessing elements ('beaver', 'bits', 'sampling') to functions that can pre-process them + * the function must implement the same interface as the JIFF provided protocols (e.g. jiffClient.protocols.generate_beaver_bgw), + * missing mappings indicate that JIFF must use the default protocols + * @param {Number} [threshold=receivers_list.length] - the threshold of the preprocessed shares + * @param {Array} [receivers_list=all_parties] - the parties that will receive the preprocsssed shares + * @param {Array} [compute_list=all_parties] - the parties that will compute the preprocsssed shares + * @param {Number} [Zp=jiffClient.Zp] - the Zp of the preprocessed shares + * @param {Array} [id_list=auto_gen()] - array of ids to be used sequentially to identify the pre_processed values + * @param {Object} [params={}] - any additional protocol-specific parameters + * @return {promise} a promise that is resolved when preprocessing is completed, null if this is called by a party that is neither a compute nor receiver party + * @see {@link module:jiff-client~JIFFClient#executePreprocessing} + */ + jiffClient.preprocessing = function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, id_list, params) { + // defaults! + if (receivers_list == null) { + receivers_list = []; + for (var p = 1; p <= jiffClient.party_count; p++) { + receivers_list.push(p); + } + } else { + jiffClient.helpers.sort_ids(receivers_list); + } + if (compute_list == null) { + compute_list = []; + for (var c = 1; c <= jiffClient.party_count; c++) { + compute_list.push(c); + } + } else { + jiffClient.helpers.sort_ids(compute_list); + } + + // not a receiver nor a sender + if (receivers_list.indexOf(jiffClient.id) === -1 && compute_list.indexOf(jiffClient.id) === -1) { + return null; + } + + // more defaults + if (Zp == null) { + Zp = jiffClient.Zp; + } + if (threshold == null) { + threshold = receivers_list.length; + } + if (protocols == null) { + protocols = jiffClient.default_preprocessing_protocols; + } + + // actual preprocessing + if (count == null || count <= 0) { + count = 1; + } + if (params == null) { + params = {}; + } + if (params['namespace'] == null) { + params['namespace'] = jiffClient.extensions[jiffClient.extensions.length - 1]; + } + + // Create preprocessing tasks + var task = { + dependent_op: dependent_op, + count: count, + threshold: threshold, + receivers_list: receivers_list, + compute_list: compute_list, + Zp: Zp, + id_list: id_list, + id: null, + params: params, + protocols: protocols, + deferred: new jiffClient.helpers.Deferred() + }; + + preprocessingTasks[preprocessingTasks.length - 1].add(task); + + return task.deferred.promise; + }; + + /** + * Ask JIFF to start executing preprocessing for tasks previously added by {@link module:jiff-client~JIFFClient#preprocessing}. + * + * Calls the provided callback when the preprocessing tasks are all done. + * + * @method executePreprocessing + * @memberof module:jiff-client~JIFFClient + * @instance + * @param callback {!Function} - the callback to execute when preprocessing is finished. + * {@link module:jiff-client~JIFFClient#preprocessing} + */ + jiffClient.executePreprocessing = function (callback) { + userCallbacks.push(callback); + preprocessingTasks.push(linkedList()); + + if (!isRunning) { + __executePreprocessing(); + } + }; + + // called only when preprocessing can run RIGHT NOW + var __executePreprocessing = function () { + isRunning = true; + + jiffClient.currentPreprocessingTasks = preprocessingTasks.shift(); + var currentCallback = userCallbacks.shift(); + + jiffClient.preprocessingCallback = function () { + if (currentCallback != null) { + currentCallback.apply(null, currentCallback); + } + + if (userCallbacks.length > 0) { + __executePreprocessing(); + } else { + isRunning = false; + } + }; + + jiffClient.preprocessingDaemon(); + }; + }; + }, + { '../../common/linkedlist.js': 52 } + ], + 18: [ + function (require, module, exports) { + module.exports = function (jiffClient) { + var linkedList = require('../../common/linkedlist.js'); + var currentBatchLoad = 0; + var suspendedTasks = 0; + + var getFirstTask = function (task) { + if (task.count > 1) { + var remainingTasks = Object.assign({}, task); + + var deferred1 = new jiffClient.helpers.Deferred(); + var deferred2 = new jiffClient.helpers.Deferred(); + Promise.all([deferred1.promise, deferred2.promise]).then(task.deferred.resolve); + task.deferred = deferred1; + remainingTasks.deferred = deferred2; + + remainingTasks.count--; + task.count = 1; + if (task.id_list != null) { + task.id = remainingTasks.id_list.shift(); + task.id_list = null; + } + jiffClient.currentPreprocessingTasks.pushHead(remainingTasks); + } + if (task.id_list != null) { + task.id = task.id_list[0]; + task.id_list = null; + } + return task; + }; + + var checkIfDone = function () { + if (currentBatchLoad === 0 && suspendedTasks === 0) { + var callback = jiffClient.preprocessingCallback; + jiffClient.preprocessingCallback = null; + callback(jiffClient); + } + }; + + var buildID = function (task) { + // Two kinds of operations: one that relies on different sets of senders and receivers, and one that has a set of holders + if (task.dependent_op === 'open' || task.dependent_op === 'bits.open') { + // TODO: make this part of the description in table + var open_parties = task.params['open_parties'] != null ? task.params['open_parties'] : task.receivers_list; + task.id = jiffClient.counters.gen_op_id2_preprocessing(task.dependent_op, open_parties, task.receivers_list); + } else { + task.id = jiffClient.counters.gen_op_id_preprocessing(task.dependent_op, task.receivers_list); + } + }; + + var taskIsExecutable = function (task) { + // if the protocol name is in the map, it can be directly executed + var namespace = find_closest_namespace(task.dependent_op, task.params['namespace']); + return namespace == null; + }; + + var find_closest_namespace = function (op, starting_namespace) { + var namespace_index = jiffClient.extensions.indexOf(starting_namespace); + while (namespace_index >= 0) { + var namespace = jiffClient.extensions[namespace_index]; + if (jiffClient.preprocessing_function_map[namespace] != null && jiffClient.preprocessing_function_map[namespace][op] != null) { + return namespace; + } + namespace_index--; + } + + return null; + }; + + // execute a task and handle it upon completion + var executeTask = function (task) { + currentBatchLoad++; + + var _params = Object.assign({}, task.params); + _params.output_op_id = task.id; + + var protocol = task.protocols[task.dependent_op] || jiffClient.default_preprocessing_protocols[task.dependent_op]; + var result = protocol(task.threshold, task.receivers_list, task.compute_list, task.Zp, _params, task.protocols); + + if (result.promise == null || result.promise.then == null) { + taskFinished(task, result); + } else { + result.promise.then(taskFinished.bind(null, task, result)); + } + }; + var taskFinished = function (task, result) { + currentBatchLoad--; + + if (task.receivers_list.indexOf(jiffClient.id) > -1) { + jiffClient.store_preprocessing(task.id, result.share); + } + task.deferred.resolve(); + jiffClient.preprocessingDaemon(); + }; + + // expand task by one level and replace the node in the task list + var expandTask = function (task) { + // copy of params + var _params = Object.assign({}, task.params); + + // Recursively follow jiffClient.preprocessing_function_map + // to figure out the sub-components/nested primitives of the given operation + // and pre-process those with the right op_ids. + + // ID should never be null + var namespace = find_closest_namespace(task.dependent_op, _params['namespace']); + var preprocessing_dependencies = jiffClient.preprocessing_function_map[namespace][task.dependent_op]; + + if (typeof preprocessing_dependencies === 'function') { + preprocessing_dependencies = preprocessing_dependencies( + task.dependent_op, + task.count, + task.protocols, + task.threshold, + task.receivers_list, + task.compute_list, + task.Zp, + task.id, + _params, + task, + jiffClient + ); + } + + var newTasks = linkedList(); + var deferredChain = []; + // build linked list of new dependencies, afterwords merge them with current tasks list + for (var k = 0; k < preprocessing_dependencies.length; k++) { + var dependency = preprocessing_dependencies[k]; + var next_op = dependency['op']; + + // copy both the originally given extra_params and the extra params of the dependency and merge them + // together, dependency params overwrite duplicate keys. + // If params are ever needed in non-leaf operations, this must be changed to accommodate + var extra_params = Object.assign({}, _params, dependency['params']); + extra_params['namespace'] = dependency['namespace'] != null ? dependency['namespace'] : 'base'; + if (dependency.handler != null) { + extra_params = dependency.handler(task.threshold, task.receivers_list, task.compute_list, task.Zp, task.id, extra_params, task, jiffClient); + } + if (extra_params.ignore === true) { + continue; + } + + // compose ids similar to how the actual operation is implemented + var next_id_list = []; + var next_count = dependency['count']; + + if (next_count == null) { + next_count = 1; + next_id_list[0] = dependency['absolute_op_id'] || task.id + dependency['op_id']; + } else { + next_count = next_count(task.threshold, task.receivers_list, task.compute_list, task.Zp, task.id, extra_params); + for (var j = 0; j < next_count; j++) { + next_id_list.push(dependency['absolute_op_id'] || task.id + dependency['op_id'] + j); + } + } + + var nextTask = { + dependent_op: next_op, + count: next_count, + threshold: dependency['threshold'] || task.threshold, + receivers_list: dependency['receivers_list'] || task.receivers_list, + compute_list: dependency['compute_list'] || task.compute_list, + Zp: dependency['Zp'] || task.Zp, + id_list: next_id_list, + id: null, + params: extra_params, + protocols: task.protocols, + deferred: new jiffClient.helpers.Deferred() + }; + + deferredChain.push(nextTask.deferred.promise); + if (dependency.requires != null) { + nextTask.wait = true; + var required_ops = []; + for (var r = 0; r < dependency.requires.length; r++) { + if (dependency.requires[r] >= k) { + throw new Error( + 'Preprocessing dependency "' + + next_op + + '" in preprocessingMap for "' + + task.dependent_op + + '" at namespace "' + + namespace + + '" cannot require subsequent dependency ' + + dependency.requires[r] + ); + } + required_ops.push(deferredChain[dependency.requires[r]]); + } + + Promise.all(required_ops).then( + function (nextTask) { + delete nextTask['wait']; + jiffClient.preprocessingDaemon(); + }.bind(null, nextTask) + ); + + // add waiting task to the tail of the queue + // jiffClient.currentPreprocessingTasks.add(nextTask); + newTasks.add(nextTask); + } else { + // non-waiting tasks are added to the head of the queue + newTasks.add(nextTask); + } + } + + Promise.all(deferredChain).then(task.deferred.resolve); + jiffClient.currentPreprocessingTasks = newTasks.extend(jiffClient.currentPreprocessingTasks); + }; + + /** + * Preprocessing Daemon that executes all currently scheduled preprocessing tasks (entries in jiffClient.currentPreprocessingTasks array) in order. + * @method preprocessingDaemon + * @memberof module:jiff-client~JIFFClient + * @instance + */ + jiffClient.preprocessingDaemon = function () { + while (currentBatchLoad < jiffClient.preprocessingBatchSize) { + var task = jiffClient.currentPreprocessingTasks.popHead(); + + if (task == null) { + checkIfDone(); + return; + } + + if (task.object.wait) { + jiffClient.currentPreprocessingTasks.pushHead(task.object); + break; + } + + task = getFirstTask(task.object); + if (task.id == null) { + buildID(task); + } + + // check if task is executable or no + if (taskIsExecutable(task)) { + executeTask(task); // co-recursively calls preprocessingDaemon() + } else { + //expand single task + expandTask(task); + } + } + }; + }; + }, + { '../../common/linkedlist.js': 52 } + ], + 19: [ + function (require, module, exports) { + // internal functions for use in preprocessing function map + module.exports = { + bits_count: function (threshold, receivers_list, compute_list, Zp, op_id, params) { + var bitLength = params.bitLength; + if (bitLength == null) { + bitLength = Zp.toString(2).length; + } + return bitLength; + }, + constant_bits_count: function () { + return module.exports.bits_count.apply(null, arguments) - 1; + }, + dynamic_bits_cmult: function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params) { + // constant bit length + var constantBits = Zp.toString(2).length; + if (params.constantBits != null) { + constantBits = params.constantBits; + } + // secret bit length + var bitLength = params.bitLength; + if (bitLength == null) { + bitLength = Zp.toString(2).length; + } + // for every bit from constant, pre-process for one bits.sadd of the right size + var ops = []; + for (var i = 0; i < constantBits; i++) { + var accLength = i === 0 ? 1 : bitLength + i; + ops.push({ op: 'bits.sadd', op_id: ':bits.sadd:' + i, params: { bitLengthLeft: accLength, bitLengthRight: bitLength + i } }); + } + return ops; + }, + dynamic_bits_smult: function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params) { + var bitLength = params.bitLength; + if (bitLength == null) { + bitLength = Zp.toString(2).length; + } + + var left = params.bitLengthLeft; + var right = params.bitLengthRight; + left = left != null ? left : bitLength; + right = right != null ? right : bitLength; + var max = Math.max(left, right); + var min = Math.max(left, right); + + var ops = []; + for (var i = 0; i < min; i++) { + for (var j = 0; j < max + i; j++) { + ops.push({ op: 'if_else', op_id: ':if_else:' + i + ':' + j }); + } + var accLength = i === 0 ? min : max + i; + ops.push({ op: 'bits.sadd', op_id: ':bits.sadd:' + i, params: { bitLengthLeft: accLength, bitLengthRight: max + i } }); + } + return ops; + }, + choice_bits_count: function (choice, offset) { + if (offset == null) { + offset = 0; + } + return function (threshold, receivers_list, compute_list, Zp, op_id, params) { + var bitLength = params.bitLength; + if (bitLength == null) { + bitLength = Zp.toString(2).length; + } + + var left = params.bitLengthLeft; + var right = params.bitLengthRight; + left = left != null ? left : bitLength; + right = right != null ? right : bitLength; + + return choice(left, right) + offset; + }; + }, + decomposition_ifelse_count: function (threshold, receivers_list, compute_list, Zp, op_id, params) { + return Zp.toString(2).length; + }, + dynamic_bits_sdiv: function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params) { + var bitLength = params.bitLength; + if (bitLength == null) { + bitLength = Zp.toString(2).length; + } + + var left = params.bitLengthLeft; + var right = params.bitLengthRight; + left = left != null ? left : bitLength; + right = right != null ? right : bitLength; + var min = Math.min(left, right); + + var ops = []; + for (var i = 0; i < left; i++) { + var accLength = Math.min(i + 1, min + 1); + ops.push({ op: 'bits.ssub', op_id: ':bits.ssub:' + i, params: { bitLengthLeft: accLength, bitLengthRight: right } }); + for (var j = 0; j < accLength; j++) { + ops.push({ op: 'if_else', op_id: ':if_else:' + i + ':' + j }); + } + } + return ops; + }, + dynamic_bits_cdiv: function (dir) { + return function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params) { + var constantBits = Zp.toString(2).length; + if (params.constantBits != null) { + constantBits = params.constantBits; + } + var bitLength = params.bitLength; + if (bitLength == null) { + bitLength = Zp.toString(2).length; + } + var min = Math.min(bitLength, constantBits); + + var ops = []; + var loopCounter = dir === 'left' ? bitLength : constantBits; + for (var i = 0; i < loopCounter; i++) { + var accLength = Math.min(i + 1, min + 1); + if (dir === 'left') { + ops.push({ + op: 'bits.csubl', + op_id: ':bits.csubl:' + i, + params: { bitLength: accLength, constantBits: constantBits } + }); + } else { + ops.push({ + op: 'bits.ssub', + op_id: ':bits.ssub:' + i, + params: { bitLengthLeft: accLength, bitLengthRight: bitLength } + }); + } + + for (var j = 0; j < accLength; j++) { + ops.push({ op: 'if_else', op_id: ':if_else:' + i + ':' + j }); + } + } + return ops; + }; + }, + // rejection sampling + dynamic_rejection_sampling: function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params, task, jiff) { + var previousPreprocessing = jiff.preprocessing_table[task.id]; + params.reject_count = params.reject_count == null ? -1 : params.reject_count; + + if (previousPreprocessing == null || previousPreprocessing === 'RETRY' || (previousPreprocessing[0] != null && previousPreprocessing[0].value === 'RETRY')) { + if (!params.defaultBounds && (params.lower_bound == null || params.upper_bound == null)) { + jiff.store_preprocessing(task.id, { ondemand: true }); + return []; + } + + var compute_threshold = params.compute_threshold; + if (compute_threshold == null) { + // honest majority BGW + compute_threshold = Math.floor((compute_list.length + 1) / 2); + } + + var extra_params = { compute_threshold: compute_threshold }; + var reject_count = ++params.reject_count; + + Zp = Zp != null ? Zp : jiff.Zp; + params.lower_bound = params.lower_bound || 0; + params.upper_bound = params.upper_bound != null ? params.upper_bound : Zp; + + var range; + if (params.upper_bound.isBigNumber === true) { + range = params.upper_bound.minus(params.lower_bound); + } else { + range = params.upper_bound - params.lower_bound; + } + + // handle special cases + if (range.toString() === '0') { + throw new Error('rejection sampling preprocessing called with range 0, no numbers to sample!'); + } + if (range.toString() === '1') { + return [{ op: 'sampling', op_id: '', params: extra_params }]; + } + + var bitLength = jiff.helpers.ceil(jiff.helpers.bLog(range, 2)); + bitLength = parseInt(bitLength.toString(), 10); + params.bitLength = bitLength; + + // fill in dependencies according to the lower and upper bounds + var dependent_ops = []; + var requires = []; + if (jiff.helpers.bLog(range, 2).toString().indexOf('.') > -1) { + dependent_ops = [ + { op: 'bits.clt', op_id: ':bits.clt:' + reject_count, params: extra_params, threshold: compute_threshold, receivers_list: compute_list }, + { op: 'open', op_id: ':open:' + reject_count, params: extra_params, threshold: compute_threshold, receivers_list: compute_list } + ]; + requires = [0, 1]; + } + + if (params.lower_bound.toString() !== '0' && bitLength > 1) { + dependent_ops.push({ op: 'bits.cadd', op_id: ':bits.cadd:' + reject_count, params: extra_params, threshold: compute_threshold, receivers_list: compute_list }); + requires.push(dependent_ops.length - 1); + } + + dependent_ops.push({ op: 'sampling', op_id: '', requires: requires, params: extra_params }, { op: 'rejection_sampling', op_id: '', requires: [dependent_ops.length] }); + + return dependent_ops; + } + + if (previousPreprocessing[0] != null && previousPreprocessing[0].value === 'PLACEHOLDER') { + delete jiff.preprocessing_table[task.id]; + } + + return []; + }, + // random quotients for cdiv + dynamic_random_and_quotient: function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params, task, jiff) { + var constantNotProvided = params.constant == null; + receivers_list = constantNotProvided ? compute_list : receivers_list; + Zp = Zp ? Zp : jiff.Zp; + + var newParams = { compute_threshold: params.compute_threshold }; + if (params.compute_threshold == null) { + // honest majority BGW + newParams.compute_threshold = Math.floor((compute_list.length + 1) / 2); + } + threshold = newParams.compute_threshold; + + var dependent_ops = [ + { op: 'bits.cgteq', op_id: ':bits_cgteq', receivers_list: receivers_list, threshold: threshold, params: newParams }, + { op: 'if_else', op_id: ':ifelse1', receivers_list: receivers_list, threshold: threshold, params: newParams }, + { op: 'if_else', op_id: ':ifelse2', receivers_list: receivers_list, threshold: threshold, params: newParams }, + { op: 'rejection_sampling', op_id: ':rejection1', receivers_list: receivers_list, threshold: threshold, params: { lower_bound: 0, upper_bound: Zp, compute_threshold: threshold } } + ]; + + if (constantNotProvided) { + jiff.store_preprocessing(task_id, { ondemand: true }); + return dependent_ops; + } + + // we want to sample uniformly in [0, largest multiple of constant <= Zp) and [0, constant) + var largestQuotient = jiff.share_helpers['floor'](jiff.share_helpers['/'](Zp, params.constant)); + + dependent_ops.push( + { op: 'rejection_sampling', op_id: ':rejection2', threshold: threshold, params: { lower_bound: 0, upper_bound: largestQuotient, compute_threshold: threshold } }, + { op: 'rejection_sampling', op_id: ':rejection3', threshold: threshold, params: { lower_bound: 0, upper_bound: params.constant, compute_threshold: threshold } }, + { op: 'generate_random_and_quotient', op_id: '', requires: [0, 1, 2, 3, 4, 5], params: newParams } + ); + + return dependent_ops; + }, + // fast exponentiation + dynamic_fast_exponentiation: function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params, task, jiff) { + Zp = Zp ? Zp : jiff.Zp; + var constantNotProvided = params.constant == null; + var constant = params.constant; + var constantBits = params.constantBits == null ? Zp.toString(2).length : params.constantBits; + + // special case + if (!constantNotProvided && constant.toString() === '0') { + return []; + } + + var ops = []; + if (constantNotProvided) { + for (var i = 0; i < constantBits - 1; i++) { + ops.push({ op: 'smult', op_id: ':smult0:' + i }); + ops.push({ op: 'smult', op_id: ':smult1:' + i }); + } + ops.push({ op: 'smult', op_id: ':smult0:' + i }); + return ops; + } + + // handle big number + if (jiff.helpers.BigNumber) { + constant = jiff.helpers.BigNumber(constant); + } + + // preprocess for exactly the needed amount of multiplications + for (i = 0; jiff.share_helpers['<'](1, constant); i++) { + ops.push({ op: 'smult', op_id: ':smult0:' + i }); + if (!jiff.share_helpers['even'](constant)) { + ops.push({ op: 'smult', op_id: ':smult1:' + i }); + } + constant = jiff.share_helpers['floor'](jiff.share_helpers['/'](constant, 2)); + } + ops.push({ op: 'smult', op_id: ':smult0:' + i }); + return ops; + }, + // for various equality tests, preprocess of cpow(Zp - 1) (Fermat's little theorem) + handler_cpow_Zp_minus_1: function (threshold, receivers_list, compute_list, Zp, op_id, params, task, jiff) { + Zp = Zp ? Zp : jiff.Zp; + params.constant = jiff.share_helpers['-'](Zp, 1); + return params; + } + }; + }, + {} + ], + 20: [ + function (require, module, exports) { + module.exports = function (jiffClient) { + var handlers = require('./handlers.js'); + + jiffClient.default_preprocessing_protocols = { + generate_beaver: jiffClient.protocols.generate_beaver_bgw, + generate_random_number: jiffClient.protocols.generate_random_number, + sampling: jiffClient.protocols.rejection_sampling, + generate_random_bits: jiffClient.protocols.generate_random_bits, + generate_random_bit: jiffClient.protocols.generate_random_bit_bgw, + generate_zero: jiffClient.protocols.generate_zero, + generate_random_and_quotient: jiffClient.protocols.generate_random_and_quotient + }; + + jiffClient.preprocessing_function_map = { + base: { + // arithmetic sharing protocols + smult: [ + { op: 'open', op_id: ':open1' }, + { op: 'generate_beaver', op_id: ':triplet' }, + { op: 'open', op_id: ':open2' } + ], + sxor_bit: [{ op: 'smult', op_id: ':smult1' }], + slt: [ + { op: 'lt_halfprime', op_id: ':halfprime:1' }, + { op: 'lt_halfprime', op_id: ':halfprime:2' }, + { op: 'lt_halfprime', op_id: ':halfprime:3' }, + { op: 'smult', op_id: ':smult1' }, + { op: 'smult', op_id: ':smult2' } + ], + cgt: [ + { op: 'lt_halfprime', op_id: ':halfprime:1' }, + { op: 'lt_halfprime', op_id: ':halfprime:2' }, + { op: 'smult', op_id: ':smult1' } + ], + clt: [ + { op: 'lt_halfprime', op_id: ':halfprime:1' }, + { op: 'lt_halfprime', op_id: ':halfprime:2' }, + { op: 'smult', op_id: ':smult1' } + ], + lt_halfprime: [ + { op: 'rejection_sampling', op_id: ':sampling', params: { defaultBounds: true } }, + { op: 'smult', op_id: ':smult1' }, + { op: 'bits.cgt', op_id: ':bits.cgt' }, + { op: 'sxor_bit', op_id: ':sxor_bit' }, + { op: 'open', op_id: ':open' } + ], + cneq: [{ op: 'cpow', op_id: ':cpow', handler: handlers.handler_cpow_Zp_minus_1 }], + cpow: handlers.dynamic_fast_exponentiation, + cdiv: [ + { op: 'cgt', op_id: ':wrap_cgt' }, + { op: 'cgteq', op_id: ':cor1' }, + { op: 'cgteq', op_id: ':cor2' }, + { op: 'smult', op_id: ':smult' }, + { op: 'clt', op_id: ':zero_check' }, + { op: 'smult', op_id: ':zero_it' }, + { op: 'open', op_id: ':open' }, + { op: 'quotient', op_id: ':quotient' } + ], + sdiv: [ + { op: 'bit_decomposition', op_id: ':decomposition1' }, + { op: 'bit_decomposition', op_id: ':decomposition2' }, + { op: 'bits.sdiv', op_id: ':bits.sdiv' } + ], + if_else: [{ op: 'smult', op_id: ':smult' }], + // bits protocols + bit_decomposition: [ + { op: 'rejection_sampling', op_id: ':sampling', params: { defaultBounds: true } }, + { op: 'bits.csubr', op_id: ':bits.csubr:1' }, + { op: 'bits.csubr', op_id: ':bits.csubr:2' }, + { op: 'if_else', op_id: ':if_else:', count: handlers.decomposition_ifelse_count }, + { op: 'open', op_id: ':open' } + ], + // comparisons + 'bits.cgteq': [{ op: 'smult', op_id: ':smult:', count: handlers.constant_bits_count }], + 'bits.cneq': [{ op: 'sor_bit', op_id: ':sor_bit:', count: handlers.constant_bits_count }], + 'bits.sneq': [ + { op: 'sxor_bit', op_id: ':sxor_bit:initial' }, + { op: 'sxor_bit', op_id: ':sxor_bit:', count: handlers.choice_bits_count(Math.min, -1) }, + { op: 'sor_bit', op_id: ':sor_bit:', count: handlers.choice_bits_count(Math.max, -1) } + ], + 'bits.sgteq': [ + { op: 'smult', op_id: ':smult:initial' }, + { op: 'smult', op_id: ':smult1:', count: handlers.choice_bits_count(Math.max, -1) }, + { op: 'sxor_bit', op_id: ':sxor_bit1:', count: handlers.choice_bits_count(Math.min, -1) }, + { op: 'smult', op_id: ':smult2:', count: handlers.choice_bits_count(Math.min, -1) } + ], + 'bits.sgt': [ + { op: 'bits.sgteq', op_id: ':bits.sgteq' }, + { op: 'bits.sneq', op_id: ':bits.sneq' }, + { op: 'smult', op_id: ':smult' } + ], + // constant arithmetic + 'bits.cadd': [ + { op: 'smult', op_id: ':smult:', count: handlers.constant_bits_count }, + { op: 'sxor_bit', op_id: ':sxor_bit:', count: handlers.constant_bits_count } + ], + 'bits.cmult': handlers.dynamic_bits_cmult, + 'bits.cdivl': handlers.dynamic_bits_cdiv('left'), + 'bits.cdivr': handlers.dynamic_bits_cdiv('right'), + // secret arithmetic + 'bits.sadd': [ + { op: 'sxor_bit', op_id: ':sxor_bit:initial' }, + { op: 'smult', op_id: ':smult:initial' }, + { op: 'smult', op_id: ':smult1:', count: handlers.choice_bits_count(Math.max, -1) }, + { op: 'sxor_bit', op_id: ':sxor_bit1:', count: handlers.choice_bits_count(Math.max, -1) }, + { op: 'smult', op_id: ':smult2:', count: handlers.choice_bits_count(Math.min, -1) }, + { op: 'sxor_bit', op_id: ':sxor_bit2:', count: handlers.choice_bits_count(Math.min, -1) } + ], + 'bits.smult': handlers.dynamic_bits_smult, + 'bits.sdiv': handlers.dynamic_bits_sdiv, + 'bits.open': [{ op: 'open', op_id: ':', count: handlers.bits_count }], + // refresh/open + refresh: [{ op: 'generate_zero', op_id: '' }], + open: [{ op: 'refresh', op_id: ':refresh' }], + // generating a random number and its quotient / constant + quotient: handlers.dynamic_random_and_quotient, + // rejection sampling + rejection_sampling: handlers.dynamic_rejection_sampling + } + }; + + // arithmetic protocols + jiffClient.preprocessing_function_map['base']['sor_bit'] = jiffClient.preprocessing_function_map['base']['sxor_bit']; + jiffClient.preprocessing_function_map['base']['smod'] = jiffClient.preprocessing_function_map['base']['sdiv']; + jiffClient.preprocessing_function_map['base']['slteq'] = jiffClient.preprocessing_function_map['base']['slt']; + jiffClient.preprocessing_function_map['base']['sgteq'] = jiffClient.preprocessing_function_map['base']['slt']; + jiffClient.preprocessing_function_map['base']['sgt'] = jiffClient.preprocessing_function_map['base']['slt']; + jiffClient.preprocessing_function_map['base']['clteq'] = jiffClient.preprocessing_function_map['base']['cgt']; + jiffClient.preprocessing_function_map['base']['cgteq'] = jiffClient.preprocessing_function_map['base']['clt']; + jiffClient.preprocessing_function_map['base']['seq'] = jiffClient.preprocessing_function_map['base']['cneq']; + jiffClient.preprocessing_function_map['base']['sneq'] = jiffClient.preprocessing_function_map['base']['cneq']; + jiffClient.preprocessing_function_map['base']['ceq'] = jiffClient.preprocessing_function_map['base']['cneq']; + + // bits protocols + jiffClient.preprocessing_function_map['base']['bits.clt'] = jiffClient.preprocessing_function_map['base']['bits.cgteq']; + jiffClient.preprocessing_function_map['base']['bits.clteq'] = jiffClient.preprocessing_function_map['base']['bits.cgteq']; + jiffClient.preprocessing_function_map['base']['bits.cgt'] = jiffClient.preprocessing_function_map['base']['bits.cgteq']; + jiffClient.preprocessing_function_map['base']['bits.ceq'] = jiffClient.preprocessing_function_map['base']['bits.cneq']; + jiffClient.preprocessing_function_map['base']['bits.slt'] = jiffClient.preprocessing_function_map['base']['bits.sgteq']; + jiffClient.preprocessing_function_map['base']['bits.slteq'] = jiffClient.preprocessing_function_map['base']['bits.sgt']; + jiffClient.preprocessing_function_map['base']['bits.seq'] = jiffClient.preprocessing_function_map['base']['bits.sneq']; + jiffClient.preprocessing_function_map['base']['bits.csubl'] = jiffClient.preprocessing_function_map['base']['bits.cadd']; + jiffClient.preprocessing_function_map['base']['bits.csubr'] = jiffClient.preprocessing_function_map['base']['bits.cadd']; + jiffClient.preprocessing_function_map['base']['bits.ssub'] = jiffClient.preprocessing_function_map['base']['bits.sadd']; + }; + }, + { './handlers.js': 19 } + ], + 21: [ + function (require, module, exports) { + const share = require('./share.js'); + const open = require('./open.js'); + const util = require('./util.js'); + + module.exports = { + jiff_share_array: share.share_array, + jiff_share_2D_array: share.share_2D_array, + jiff_share_ND_array: share.share_ND_array, + jiff_share_ND_array_static: share.share_ND_array_static, + jiff_share_ND_array_deferred: share.share_ND_array_deferred, + + jiff_skeleton_of: util.skeleton_of, + + jiff_open_array: open.open_array, + jiff_open_ND_array: open.open_ND_array, + jiff_receive_open_ND_array: open.receive_open_ND_array + }; + }, + { './open.js': 22, './share.js': 23, './util.js': 24 } + ], + 22: [ + function (require, module, exports) { + const open_array = function (jiff, shares, parties, op_ids) { + var i; + + // A base operation id is provided to use for all opens. + if (typeof op_ids === 'string' || typeof op_ids === 'number') { + var tmp = { s1: op_ids }; + for (i = 1; i <= jiff.party_count; i++) { + tmp[i] = op_ids; + } + op_ids = tmp; + } - if (message_to_send['encrypted'] !== false) { - message_to_send['message'] = jiffClient.hooks.encryptSign(jiffClient, message_to_send['message'], jiffClient.keymap[message_to_send['party_id']], jiffClient.secret_key); - message_to_send['encrypted'] = true; - } + return open_ND_array(jiff, shares, parties, null, op_ids); + }; + + const open_ND_array = function (jiff, shares, receivers_list, senders_list, op_ids) { + if (senders_list == null) { + senders_list = []; + for (var i = 1; i <= jiff.party_count; i++) { + senders_list.push(i); + } + if (receivers_list == null) { + receivers_list = Array.from(senders_list); + } + } + var is_sending = senders_list.indexOf(jiff.id) > -1; + + // Compute operation id + /* + *if (op_ids == null) { + * op_ids = jiff.counters.gen_op_id('open_ND_array', receivers_list.concat(senders_list)); + *} + */ + + var final_deferred = new jiff.helpers.Deferred(); + var final_promise = final_deferred.promise; + var resolve_open = function (shares) { + final_deferred.resolve( + (function __open_ND_array(shares, parties, op_ids) { + if (typeof shares.length === 'undefined') { + return jiff.internal_open(shares, parties); + } else if (shares.length === 0) { + return Promise.resolve([]); + } else { + var promised_array = []; + for (var i = 0; i < shares.length; i++) { + promised_array.push(__open_ND_array(shares[i], parties, op_ids + ':' + i)); + } + return Promise.all(promised_array); + } + })(shares, receivers_list, op_ids) + ); + }; + + if (is_sending) { + // Must emit the skeleton for any parties that are not holders but are receiving the open + var skeleton = (function __unwipe(nd_array, replace) { + if (!(typeof nd_array.length === 'undefined') || nd_array.length === 0) { + var unwiped_array = []; + for (var k = 0; k < nd_array.length; k++) { + unwiped_array.push(__unwipe(nd_array[k], replace)); + } + return unwiped_array; + } + return replace; + })(shares, null); + jiff.emit(op_ids + 'skeleton', receivers_list, JSON.stringify(skeleton)); + + resolve_open(shares); + } else { + // Populate skeleton with imitation shares + var revive_shares = function (skeleton) { + const share = new jiff.SecretShare({}, senders_list, senders_list.length, jiff.Zp); + return (function __unwipe(nd_array, replace) { + if (nd_array != null && !(typeof nd_array.length === 'undefined' && nd_array.length > 0)) { + var unwiped_array = []; + for (var k = 0; k < nd_array.length; k++) { + unwiped_array.push(__unwipe(nd_array[k], replace)); + } + return unwiped_array; + } + return replace; + })(skeleton, share); + }; + + // If this party is not a sender, then the variable `shares` may be a skeleton + if (shares != null) { + // Use existing shares as skeleton to revive + shares = revive_shares(shares); + resolve_open(shares); + } else { + // Receive skeleton from senders + jiff.listen(op_ids + 'skeleton', function (sender, skeleton) { + jiff.remove_listener(op_ids + 'skeleton'); // This doesn't seem to work + + if (typeof skeleton === 'string') { + skeleton = JSON.parse(skeleton); + } + + shares = revive_shares(skeleton); + resolve_open(shares); + }); + } + } - jiffClient.socket.safe_emit('custom', JSON.stringify(message_to_send)); - } + return final_promise; + }; - // receive our own message if specified - if (receivers.indexOf(jiffClient.id) > -1) { - jiffClient.handlers.receive_custom({tag: tag, party_id: jiffClient.id, message: message, encrypted: false}); - } - }; + const receive_open_ND_array = function (jiff, receivers_list, senders_list, threshold, Zp, op_ids) { + return open_ND_array(jiff, null, receivers_list, senders_list, op_ids); + }; - /** - * Registers the given function as a listener for messages with the given tag. - * Removes any previously set listener for this tag. - * @memberof module:jiff-client~JIFFClient - * @method listen - * @instance - * @param {string} tag - the tag to listen for. - * @param {function(party_id, string)} handler - the function that handles the received message: takes the sender id and the message as parameters. - */ - jiffClient.listen = function (tag, handler) { - jiffClient.listeners[tag] = handler; - - var stored_messages = jiffClient.custom_messages_mailbox[tag]; - if (stored_messages == null) { - return; - } + module.exports = { + open_array: open_array, + open_ND_array: open_ND_array, + receive_open_ND_array: receive_open_ND_array + }; + }, + {} + ], + 23: [ + function (require, module, exports) { + const util = require('./util.js'); + + const share_array = function (jiff, array, lengths, threshold, receivers_list, senders_list, Zp, share_id) { + var skeletons = null; + if (lengths != null) { + // Check format of lengths + if (lengths != null && typeof lengths !== 'number' && typeof lengths !== 'object') { + throw new Error('share_array: unrecognized lengths'); + } + + // Get senders list for later checking + if (senders_list == null) { + senders_list = []; + for (i = 1; i <= jiff.party_count; i++) { + senders_list.push(i); + } + } + + // Generate skeletons from lengths + skeletons = {}; + if (typeof lengths === 'number') { + // All arrays are of the same length + var skeleton = []; + for (var i = 0; i < lengths; i++) { + skeleton.push(null); + } + for (i = 0; i < senders_list.length; i++) { + skeletons[senders_list[i]] = skeleton; + } + } else { + // Lengths of the different arrays are all provided + for (i = 0; i < senders_list.length; i++) { + if (lengths[senders_list[i]] == null) { + throw new Error('share_array: missing length'); + } else { + skeletons[senders_list[i]] = []; + for (var j = 0; j < lengths[senders_list[i]]; j++) { + skeletons[senders_list[i]].push(null); + } + } + } + } + } - for (var i = 0; i < stored_messages.length; i++) { - var sender_id = stored_messages[i].sender_id; - var message = stored_messages[i].message; - handler(sender_id, message); - } + return share_ND_array(jiff, array, skeletons, threshold, receivers_list, senders_list, Zp, share_id); + }; - delete jiffClient.custom_messages_mailbox[tag]; - }; + const share_2D_array = function (jiff, array, lengths, threshold, receivers_list, senders_list, Zp, share_id) { + var i; - /** - * Removes the custom message listener attached to the given tag - * @memberof module:jiff-client~JIFFClient - * @method remove_listener - * @instance - * @param {string} tag - the tag of the listener to remove - */ - jiffClient.remove_listener = function (tag) { - delete jiffClient.listeners[tag]; - }; -}; -},{}],5:[function(require,module,exports){ -module.exports = function (jiffClient) { - /** - * Wait until the public keys of these parties are known. - * The public keys may be known before the parties connect (if provided in the options), - * or they could be sent by the server after the parties connect. - * Computation specified in the callback may assume that these parties are connected, - * if they are not, the server will handle storing and relaying the needed messages - * to them when they connect. - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {Array} parties - an array of party ids to wait for, must explicitly include 's1' if callback must wait for the server. - * @param {function(jiff-instance)} callback - the function to execute when these parties are known. - * @param {boolean} [wait_for_initialization=true] - specifies whether to wait for initialization to be complete - * before executing the callback (even if parties are available). - * Set this to false if you do not need the party count and this - * party's id, or if you already have them, and you are certain - * they will be accepted by the server on initialization. - */ - jiffClient.wait_for = function (parties, callback, wait_for_initialization) { - if (wait_for_initialization == null) { - wait_for_initialization = true; - } + // Check format of lengths + if (lengths != null && typeof lengths !== 'object') { + throw new Error('share_array: unrecognized lengths'); + } - jiffClient.wait_callbacks.push({parties: parties, callback: callback, initialization: wait_for_initialization}); - jiffClient.execute_wait_callbacks(); // See if the callback can be executed immediately - }; + // Default values + if (receivers_list == null) { + receivers_list = []; + for (i = 1; i <= jiff.party_count; i++) { + receivers_list.push(i); + } + } + if (senders_list == null) { + senders_list = []; + for (i = 1; i <= jiff.party_count; i++) { + senders_list.push(i); + } + } - /** - * Disconnects from the computation. - * Allows the client program to exit. - * @method disconnect - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {boolean} [safe=false] - if true, jiff will disconnect safely (i.e. after ensuring all - * outgoing pending messages were delivered). - * @param {boolean} [free=false] - if set to true, it means this party's disconnection is final, and all resources - * associated with this party must be freed. - * If all parties in a computation are freed, then all resources associated with the - * computation are freed, and any subsequent reconnection to the computation is as - * if a the connection is for a fresh new computation. - * @param {function()} [callback] - executed after the instance safely disconnects, if safe is set to false, this - * parameter is ignored. - */ - jiffClient.disconnect = function (safe, free, callback) { - if (safe) { - jiffClient.socket.safe_disconnect(free, callback); - } else { - if (free) { - jiffClient.free(); - } - jiffClient.socket.disconnect(); - } - }; + var isReceiving = receivers_list.indexOf(jiff.id) > -1; + if (senders_list.indexOf(jiff.id) === -1 && !isReceiving) { + // This party is neither a sender nor a receiver, do nothing! + return null; + } - /** - * Emits event to free up all the resources allocated for this party on the server. - * It is best not to call this function directly, as it can break things if resources still need to be used. - * Instead, use jiff.disconnect(safe, free, callback) to free after safely disconnecting. - * @see {@link module:jiff-client~JIFFClient#disconnect} - * @method free - * @memberof module:jiff-client~JIFFClient - * @instance - */ - jiffClient.free = function () { - var msg = jiffClient.hooks.execute_array_hooks('beforeOperation', [jiffClient, 'free', {}], 2); - jiffClient.socket.safe_emit('free', JSON.stringify(msg)); - }; -}; -},{}],6:[function(require,module,exports){ -var numbers = require('../protocols/preprocessing/numbers.js'); -var bits = require('../protocols/preprocessing/bits.js'); -var triplets = require('../protocols/preprocessing/triplets.js'); -var quotients = require('../protocols/preprocessing/quotients.js'); -var sampling = require('../protocols/preprocessing/sampling.js'); - -/** - * Contains miscellaneous protocols (mostly used in preprocessing) - * @name protocols - * @alias protocols - * @namespace - */ -module.exports = function (jiffClient) { - /** - * Creates shares of an unknown random number. Every party comes up with its own random number and shares it. - * Then every party combines all the received shares to construct one share of the random unknown number. - * @method - * @memberof protocols - * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed. - * @param {Array} receivers_list - array of party ids to receive the result. - * @param {Array} compute_list - array of party ids to perform the protocol. - * @param {number} Zp - the mod. - * @param {object} params - an object containing extra parameters passed by the user. - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated. - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * @return {Object} contains 'share' (this party's share of the result) and 'promise'. - */ - jiffClient.protocols.generate_random_number = numbers.generate_random_number.bind(null, jiffClient); - - /** - * Creates shares of 0, such that no party knows the other parties' shares. - * Every party secret shares 0, then every party sums all the shares they received, resulting - * in a new share of 0 for every party. - * @method - * @memberof protocols - * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed. - * @param {Array} receivers_list - array of party ids to receive the result. - * @param {Array} compute_list - array of party ids to perform the protocol. - * @param {number} Zp - the mod. - * @param {object} params - an object containing extra parameters passed by the user. - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated. - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * @return {Object} contains 'share' (this party's share of the result) and 'promise'. - */ - jiffClient.protocols.generate_zero = numbers.generate_zero.bind(null, jiffClient); - - /** - * Generates a random bit under MPC by xoring all bits sent by participating parties using smult / beaver triples - * @method - * @memberof protocols - * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed. - * @param {Array} receivers_list - array of party ids to receive the result. - * @param {Array} compute_list - array of party ids to perform the protocol. - * @param {number} Zp - the mod. - * @param {object} params - an object containing extra parameters passed by the user. - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated. - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * @return {Object} contains 'share' (this party's share of the generated bit) and 'promise'. - */ - jiffClient.protocols.generate_random_bit_smult = bits.generate_random_bit_smult.bind(null, jiffClient); - - /** - * Generates a random bit under MPC by xoring all bits sent by participating parties using smult_bgw - * @method - * @memberof protocols - * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed. - * @param {Array} receivers_list - array of party ids to receive the result. - * @param {Array} compute_list - array of party ids to perform the protocol. - * @param {number} Zp - the mod. - * @param {object} params - an object containing extra parameters passed by the user. - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated. - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * @return {Object} contains 'share' (this party's share of the generated bit) and 'promise'. - */ - jiffClient.protocols.generate_random_bit_bgw = bits.generate_random_bit_bgw.bind(null, jiffClient); - - /** - * Generates a sequence of random bits under MPC. - * @method - * @memberof protocols - * @param {number} [threshold=receivers_list.length] - the threshold of the bit when stored by receivers after generation. * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed. - * @param {Array} receivers_list - array of party ids to receive the result. - * @param {Array} compute_list - array of party ids to perform the protocol. - * @param {number} Zp - the mod. - * @param {object} params - an object containing extra parameters passed by the user. - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated. - * - count: how many random bits to generate. - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * @param {object} protocols - the protocols to use for preprocessing. - * @return {Object} contains 'share' (array of secret shares bits) and 'promise'. - */ - jiffClient.protocols.generate_random_bits = bits.generate_random_bits.bind(null, jiffClient); - - /** - * Generation of beaver triplet via MPC, uses the server for communication channels, but not for generation - * @method - * @memberof protocols - * @param {number} threshold - the threshold of the triplets when stored by receivers after generation - * @param {Array} receivers_list - array of party ids that want to receive the triplet shares - * @param {Array} compute_list - array of party ids that will perform this protocol - * @param {number} Zp - the mod - * @param {object} params - an object containing extra parameters passed by the user - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated - * - an optional number compute_threshold parameter, which specifies threshold used - * during the protocol execution. By default, this is the length of the (compute_list+1)/2 - * @param {object} protocols - the sub protocols to use for preprocessing - * @return {object} all pre-processing protocols must return an object with these keys: - * { - * 'share': the share(s)/value(s) to store attached to op_id for later use by the computation (i.e. the result of preprocessing), - * 'promise': a promise for when this protocol is fully completed (could be null if the protocol was already completed) - * } - * In this case, 'share' is an array of this party's shares of the resulting triplet, a,b,c such that a*b=c - */ - jiffClient.protocols.generate_beaver_bgw = triplets.bind(null, jiffClient); - - /** - * Creates shares of r and x, such that r is a uniform random number between 0 and Zp, and x is floor(r/constant) - * where constant is provided by the extra params - * @method - * @memberof protocols - * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed - * @param {Array} receivers_list - array of party ids to receive the result - * @param {Array} compute_list - array of party ids to perform the protocol - * @param {number} Zp - the mod - * @param {object} params - an object containing extra parameters passed by the user - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * - constant: the constant to divide the random number by. - * - output_op_id: the set op id of the output quotient and noise - * @return {Object} contains 'share' (this party's share of the result) and 'promise' - */ - jiffClient.protocols.generate_random_and_quotient = quotients.bind(null, jiffClient); - - /** - * Wrapper for when doing rejection sampling during pre processing - * - * Do not use this function directly, especially during online computation time, use jiffClient.protocols.bits.rejection_sampling instead - * - * @method - * @memberof protocols - * @param {number} [threshold=receivers_list.length] - the threshold of the resulting shares after sampling - * @param {Array} [receivers_list=all_parties] - array of party ids that want to receive the sampling shares, by default, this includes all parties - * @param {Array} [compute_list=all_parties] - array of party ids that will perform this protocol, by default, this includes all parties - * @param {number} [Zp=jiff-instance.Zp] - the mod (if null then the default Zp for the instance is used) - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly - * @param {object} [params={}] - an object containing extra parameters passed by the user - * Expects: - * - an optional number compute_threshold parameter, which specifies threshold used - * during the protocol execution. By default, this is (|compute_list|+1)/2 - * - optional 'lower_bound' and 'upper_bound' numeric parameters, default to 0 and Zp respectively - * - op_id, the base op_id to tag operations inside this protocol with, defaults to auto generated - * - output_op_id, the tag id for the output result - * - retry_count, how many times rejection sampling have been retried! - * @param {object} [protocols=defaults] - the protocols to use for preprocessing, any protocol(s) not provided will be replaced with defaults - * @returns {Object} an object containing keys: 'share', and 'promise'. The promise is resolved when the rejection sampling is completed. - * The object is consumed by .preprocessing: - * - 'share' attribute contains the resulting array of secret shared bits representing the sampled value, and is stored in the preprocessing table internally - * - The promise is consumed and a new promise is returned by .preprocessing that is resolved after this returned promise (and all other promise generated by that .preprocessing call) are resolved - */ - jiffClient.protocols.rejection_sampling = sampling.bind(null, jiffClient); -}; -},{"../protocols/preprocessing/bits.js":34,"../protocols/preprocessing/numbers.js":35,"../protocols/preprocessing/quotients.js":36,"../protocols/preprocessing/sampling.js":37,"../protocols/preprocessing/triplets.js":38}],7:[function(require,module,exports){ -var shareProtocol = require('../protocols/shamir/share.js'); -var openProtocol = require('../protocols/shamir/open.js'); -var reshareProtocol = require('../protocols/shamir/reshare.js'); -var arraysSharing = require('../protocols/arrays/api.js'); - -module.exports = function (jiffClient) { - /** - * Share a secret input - * - * Can be overriden by extensions to customize behavior - * - * @method share - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {number} secret - the number to share (this party's input) - * @param {number} [threshold=receivers_list.length] - the minimum number of parties needed to reconstruct the secret, defaults to all the receivers - * @param {Array} [receivers_list=all_parties] - array of party ids to share with, by default, this includes all parties - * @param {Array} [senders_list=all_parties] - array of party ids to receive from, by default, this includes all parties - * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used) - * @param {string|number} [share_id=auto_gen()] - the tag used to tag the messages sent by this share operation, this tag is used - * so that parties distinguish messages belonging to this share operation from other - * share operations between the same parties (when the order of execution is not - * deterministic). An automatic id is generated by increasing a local counter, default - * ids suffice when all parties execute all sharing operations with the same senders - * and receivers in the same order - * @returns {object} a map (of size equal to the number of sending parties) - * where the key is the party id (between 1 and n), or 's1' if 's1' is specified in the senders_list, - * and the value is the share object that wraps the value received from that party (the internal value maybe - * deferred). - * - * @example - * // share an input value with all parties, and receive all other parties' inputs - * var shares = jiffClient.share(input); - * // my party id is '1', so the first share is mine (technically my share of my input value) - * var my_share = shares[1]; - * // my share of party 2's input - * var p2_share = shares[2]; - */ - jiffClient.share = function (secret, threshold, receivers_list, senders_list, Zp, share_id) { - // type check to confirm the secret to be shared is a number - // for fixed-point extension it should allow non-ints - if (secret != null && (typeof(secret) !== 'number' || Math.floor(secret) !== secret || secret < 0)) { - throw new Error('secret \'' + secret + '\' must be a non-negative whole number'); - } - if (secret != null && (secret >= (Zp == null ? jiffClient.Zp : Zp))) { - throw new Error('secret \'' + secret + '\' must fit inside Zp'); - } - return jiffClient.internal_share(secret, threshold, receivers_list, senders_list, Zp, share_id); - }; + // compute operation id + receivers_list.sort(); // sort to get the same order + senders_list.sort(); + if (share_id == null) { + share_id = jiff.counters.gen_share_id(receivers_list, senders_list) + ':array:'; + } - /** - * Same as share, but used by internal JIFF primitives/protocols, do not override this! - * @see {@link module:jiff-client~JIFFClient#share} - * @method internal_share - * @instance - * @memberof module:jiff-client~JIFFClient - */ - jiffClient.internal_share = shareProtocol.jiff_share.bind(null, jiffClient); - - /** - * re-share an existing share (value) under a new threshold or to a new set of parties or both. - * Do not use this to refresh a share (use {@link module:jiff-client~JIFFClient#SecretShare#refresh} instead) - * @method reshare - * @instance - * @memberof module:jiff-client~JIFFClient - * @param {module:jiff-client~JIFFClient#SecretShare} [share=null] - the share you would like to reshare (null if you are a receiver but not a sender) - * @param {number} [threshold=receivers_list.length] - the new threshold, defaults to the length of receivers_list param - * @param {Array} [receivers_list=all_parties] - array of party ids to receive from, by default, this includes all parties - * @param {Array} [senders_list=all_parties] - array of party ids that posses the share and will reshare it with the receivers, by default, this includes all parties - * @param {number} [Zp=jiff.Zp] - the Zp of the existing share - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this multiplication (and internally, the corresponding beaver triplet). - * This id must be unique, and must be passed by all parties to the same instruction. - * this ensures that every party gets a share from the same triplet for every matching instruction. An automatic id - * is generated by increasing a local counter, default ids suffice when all parties execute the - * instructions in the same order - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result under the new threshold, or null if this party is not a receiver - */ - jiffClient.reshare = reshareProtocol.bind(null, jiffClient); - - /** - * Open a secret share to reconstruct secret. - * @method open - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {module:jiff-client~JIFFClient#SecretShare} share - this party's share of the secret to reconstruct. - * @param {Array} [parties=all_parties] - an array with party ids (1 to n) of receiving parties. - * @param {string|number} [op_id=auto_gen()] - the operation id to be used to tag outgoing messages. - * @returns {?promise} a (JQuery) promise to the open value of the secret, null if the party is not specified in the parties array as a receiver. - * @example - * var shares = jiff_instance.share(input); - * //multiply the inputs of party 1 and 2 together - * var result = shares[1].mult(shares[2]); - * // reveal the result of the multiplication to all parties - * return jiff_instance.open(result); - */ - jiffClient.open = openProtocol.jiff_open.bind(null, jiffClient); - - /** - * Same as open, but used by internal JIFF primitives/protocols, do not override this! - * @see {@link module:jiff-client~JIFFClient#open} - * @method internal_open - * @instance - * @memberof module:jiff-client~JIFFClient - */ - jiffClient.internal_open = jiffClient.open; - - /** - * Receive shares from the specified parties and reconstruct their secret. - * Use this function in a party that will receive some answer/value but does not have a share of it. - * @method receive_open - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {Array} senders - an array with party ids (1 to n) specifying the parties sending the shares, this must be provided! - * @param {Array} [receivers=all_parties] - an array with party ids (1 to n) specifying the parties receiving the result - * @param {number} [threshold=senders.length] - the min number of parties needed to reconstruct the secret, defaults to all the senders - * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used) - * @param {string|number} [op_id=auto_Gen()] - same as jiff-instance.open - * @returns {!promise} a (JQuery) promise to the open value of the secret. - */ - jiffClient.receive_open = function (senders, receivers, threshold, Zp, op_id) { - if (senders == null) { - throw new Error('Must provide "senders" parameter in receive_open'); - } + // wrap around result of share_array + var lengths_deferred = new jiff.helpers.Deferred(); + var lengths_promise = lengths_deferred.promise; + + // figure out lengths by having each party emit their length publicly + if (lengths == null) { + lengths = {}; + var total = 0; + if (senders_list.indexOf(jiff.id) > -1) { + lengths[jiff.id] = array.length; + + // send the length of this party's array to all receivers + jiff.emit(share_id + 'length', receivers_list, array.length.toString(10)); + } + + jiff.listen(share_id + 'length', function (sender, message) { + lengths[sender] = { rows: parseInt(message, 10) }; + total++; + if (total === senders_list.length) { + jiff.remove_listener(share_id + 'length'); + lengths_deferred.resolve(lengths); + } + }); + } else if (typeof lengths.rows === 'number') { + // All arrays are of the same length + var l = lengths; + lengths = {}; + for (i = 0; i < senders_list.length; i++) { + lengths[senders_list[i]] = l; + } + + lengths_deferred.resolve(lengths); + } else { + // Lengths of the different arrays are all provided + for (i = 0; i < senders_list.length; i++) { + if (lengths[senders_list[i]] == null || lengths[senders_list[i]].rows == null) { + throw new Error('share_2D_array: missing rows length'); + } + } + + lengths_deferred.resolve(lengths); + } - jiffClient.helpers.sort_ids(senders); - if (receivers == null) { - receivers = []; - for (var i = 1; i <= jiffClient.party_count; i++) { - receivers.push(i); - } - } else { - jiffClient.helpers.sort_ids(receivers); - } - if (Zp == null) { - Zp = jiffClient.Zp; - } - if (threshold == null) { - threshold = senders.length; - } + // Final results + var share_array_deferred = new jiff.helpers.Deferred(); + var share_array_promise = share_array_deferred.promise; + + // lengths are now set, start sharing + lengths_promise.then(function (lengths) { + // compute the number of sharing rounds + var max = 0; + for (i = 0; i < senders_list.length; i++) { + var l = lengths[senders_list[i]].rows; + max = l > max ? l : max; + } + + // share every round + var promises = []; + for (var r = 0; r < max; r++) { + var round_senders = []; + for (i = 0; i < senders_list.length; i++) { + if (lengths[senders_list[i]].rows > r) { + round_senders.push(senders_list[i]); + } + } + + var row_lengths = {}; + var empty = false; + for (var p = 0; p < round_senders.length; p++) { + var pid = round_senders[p]; + row_lengths[pid] = lengths[pid].cols; + if (lengths[pid][r] != null) { + row_lengths[pid] = lengths[pid][r]; + } + if (row_lengths[pid] == null) { + empty = true; + } + } + + var row = r < array.length ? array[r] : []; + row_lengths = empty ? null : row_lengths; + var round_results = share_array(jiff, row, row_lengths, threshold, receivers_list, round_senders, Zp, share_id + 'row' + r + ':'); + promises.push(round_results); + } + + // Wait for every promises corresponding to every row + return Promise.all(promises).then(function (intermediate_results) { + // Store results here + var results = {}; + if (isReceiving) { + for (i = 0; i < senders_list.length; i++) { + results[senders_list[i]] = []; + } + } + + for (i = 0; i < intermediate_results.length; i++) { + var round = intermediate_results[i]; + for (var sender_id in round) { + if (round.hasOwnProperty(sender_id)) { + results[sender_id].push(round[sender_id]); + } + } + } + + share_array_deferred.resolve(results); + }); + }); + + return isReceiving ? share_array_promise : null; + }; + + const share_from_skeleton_unbound = function (jiff, that, sender, skeleton) { + var shares = typeof skeleton === 'string' ? JSON.parse(skeleton) : skeleton; + + var promise = share_array_single_sender(jiff, shares, that.threshold, that.receivers_list, sender, that.Zp, that.share_id + ':p_id_' + sender); + + promise.then( + function (sender, array) { + that.deferreds[sender].resolve(array); + }.bind(null, sender) + ); + }; + + const share_array_single_sender = function (jiff, secrets, threshold, receivers_list, sender, Zp, share_id) { + if (secrets != null && secrets.length === 0) { + return Promise.resolve([]); + } else if (secrets != null && Array.isArray(secrets)) { + var promised_array = []; + for (var j = 0; j < secrets.length; j++) { + promised_array.push(share_array_single_sender(jiff, secrets[j], threshold, receivers_list, sender, Zp, share_id + ':' + j)); + } + + var isReceiving = receivers_list.indexOf(jiff.id) > -1; + + if (isReceiving) { + var deferred_array = new jiff.helpers.Deferred(); + Promise.all(promised_array).then(function (array) { + deferred_array.resolve(array); + }); + } + + return isReceiving ? deferred_array.promise : Promise.resolve({}); + } else { + // Create and distribute the share - Note: Senders are reorganized in the final array. + // The return value of jiff.share is an array, [sender: share], and we only need to return share by itself. + return Promise.resolve(jiff.share(secrets, threshold, receivers_list, [sender], Zp, 'share:' + share_id)[sender]); + } + }; - var imitationSecretShare = new jiffClient.SecretShare({}, senders, threshold, Zp); - return jiffClient.open(imitationSecretShare, receivers, op_id); - }; + const share_ND_array_deferred = function (jiff, secrets, skeletons, threshold, receivers_list, senders_list, Zp, share_id) { + var parameters = [receivers_list, senders_list, threshold, Zp, share_id]; + [receivers_list, senders_list, threshold, Zp, share_id] = util.sanitize_array_params.bind(null, jiff).apply(null, parameters); - /** - * Receive arrays of shares from the specified parties and reconstruct their secrets. - * Use this function in a party that will receive some answer/value but does not have a share of it. - * @method receive_open - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {Array} senders - an array with party ids (1 to n) specifying the parties sending the shares, this must be provided! - * @param {number} [threshold=senders.length] - the min number of parties needed to reconstruct the secret, defaults to all the senders - * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used) - * @param {string|number} [op_id=auto_Gen()] - same as jiff-instance.open - * @returns {!promise} a (JQuery) promise to the open value of the secret. - */ - jiffClient.receive_open_array = arraysSharing.jiff_receive_open_ND_array.bind(null, jiffClient); - - /** - * Share an array of values. Each sender may have an array of different length. This is handled by the lengths parameter. - * This function will reveal the lengths of the shared array. - * - * If parties would like to keep the lengths of their arrays secret, they should agree on some public "max" length apriori (either under MPC - * or as part of the logistics of the computation), all their arrays should be padded to that length by using appropriate default/identity - * values - * @method share_array - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {Array} array - the array to be shared. - * @param {null|number|object} [lengths] - the lengths of the arrays to be shared, has the following options:
- * 1. null: lengths are unknown, each sender will publicly reveal the lengths of its own array.
- * 2. number: all arrays are of this length
- * 3. object: { : length }: must specify the length of the array for each sender.
- * @param {number} [threshold=receivers_list.length] - the min number of parties needed to reconstruct the secret, defaults to all the receivers. - * @param {Array} [receivers_list=all_parties] - array of party ids to share with, by default, this includes all parties. - * @param {Array} [senders_list=all_parties] - array of party ids to receive from, by default, this includes all parties. - * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used). - * @param {string|number} [share_id=auto_gen()] - the base tag used to tag the messages sent by this share operation, every element of the array - * will get a unique id based on the concatenation of base_share_id and the index of the element. - * This tag is used so that parties distinguish messages belonging to this share operation from - * other share operations between the same parties (when the order of execution is not - * deterministic). An automatic id is generated by increasing a local counter, default - * ids suffice when all parties execute all sharing operations with the same senders - * and receivers in the same order. - * @returns {?promise} if the calling party is a receiver then a promise to the shared arrays is returned, the promise will provide an object - * formatted as follows: { : [ <1st_share>, <2nd_share>, ..., <(lengths[party_id])th_share> ] } - * where the party_ids are those of the senders. - * if the calling party is not a receiver, then null is returned. - */ - jiffClient.share_array = arraysSharing.jiff_share_array.bind(null, jiffClient); - - /** - * Share an array of values. Each sender may have an array of different length. This is handled by the lengths parameter. - * This function will reveal the lengths of the shared array. - * - * If parties would like to keep the lengths of their arrays secret, they should agree on some "max" length apriori (either under MPC - * or as part of the logistics of the computation), all their arrays should be padded to that length by using appropriate default/identity - * values. - * @method share_2D_array - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {Array} array - the array to be shared. - * @param {null|number|object} lengths - the lengths of the arrays to be shared. For this to work successfully, the - * same exact value must be used in the calling code for each party. Any missing - * lengths for a row will be automatically publicly revealed by this function. - * Must have the following format: - * 1. null: lengths are unknown, each sender will publicly reveal the lengths of its own array. - * 2. { rows: , cols: , 0: , 1: , ...}: all parties have arrays - * with the given number of rows and cols. In case of jagged 2D arrays, different rows - * can have a different number of cols specified by using : . - * rows is mandatory, cols and any other number matching a specific row are optional. - * 3. { : }: must specify the lengths for each party by using - * an object with the same format as 2. Must include every party. - * @param {number} [threshold=receivers_list.length] - the min number of parties needed to reconstruct the secret, defaults to all the receivers. - * @param {Array} [receivers_list=all_parties] - array of party ids to share with, by default, this includes all parties. - * @param {Array} [senders_list=all_parties] - array of party ids to receive from, by default, this includes all parties. - * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used). - * @param {string|number} [share_id=auto_gen()] - the base tag used to tag the messages sent by this share operation, every element of the array - * will get a unique id based on the concatenation of base_share_id and the index of the element. - * This tag is used so that parties distinguish messages belonging to this share operation from - * other share operations between the same parties (when the order of execution is not - * deterministic). An automatic id is generated by increasing a local counter, default - * ids suffice when all parties execute all sharing operations with the same senders - * and receivers in the same order. - * @returns {promise} if the calling party is a receiver then a promise to the shared arrays is returned, the promise will provide an object - * formatted as follows: { : [ [ <1st_row_shares> ], [<2nd_row_share> ], ..., [ <(lengths[party_id])th_row_shares> ] ] } - * where the party_ids are those of the senders. - * if the calling party is not a receiver, then null is returned. - */ - jiffClient.share_2D_array = arraysSharing.jiff_share_2D_array.bind(null, jiffClient); - - /** - * Share an n-dimensional array of secrets - * The arrays can be of different lengths and dimensions. - * @method share_ND_array - * @memberof module:jiff-client~JIFFClient - * @instance - * @returns {object|promise} - */ - jiffClient.share_ND_array = arraysSharing.jiff_share_ND_array.bind(null, jiffClient); - - /** - * Helper function of share_ND_array - * This method gets called if share_ND_array does not have enough information - * to output an array of shares immediately. Instead it - * returns a promise to an object holding the unknown-size - * arrays from each sender. - * @method share_ND_array_deferred - * @memberof module:jiff-client~JIFFClient - * @instance - * @returns {promise} - */ - //jiffClient.share_ND_array_deferred = arraysSharing.jiff_share_ND_array_deferred.bind(null, jiffClient); - - /** - * Helper function of share_ND_array - * This method gets called if share_ND_array when share_ND_array was given - * sufficients information (in the form of array skeletons) to - * infer the size and shape of each of the senders' arrays. - * This allows it to immediately return an object containing - * the secret-shared arrays from each sender. - * @method share_ND_array_static - * @memberof module:jiff-client~JIFFClient - * @instance - * @returns {object} - */ - //jiffClient.share_ND_array_static = arraysSharing.jiff_share_ND_array_static.bind(null, jiffClient); - - /* - * Wipe a secret array of all secrets but preserve the shape - */ - jiffClient.skeleton_of = arraysSharing.jiff_skeleton_of.bind(null, jiffClient); - - /** - * Opens an array of secret shares. - * @method open_array - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} shares - an array containing this party's shares of the secrets to reconstruct. - * @param {Array} [parties=all_parties] - an array with party ids of receiving parties. - * This must be one of 3 cases: - * 1. null: open all shares to all parties. - * 2. array of numbers: open all shares to all the parties specified in the array. - * 3. array of array of numbers: open share with index i to the parties specified - * in the nested array at parties[i]. if parties[i] was null, - * then shares[i] will be opened to all parties. - * @param {string|number|object} [op_ids=auto_gen()] - an optional mapping that specifies the ID/Tag associated with each - * open message sent. Since open_array involves sending many messages per party, - * this parameter only specifies the BASE OPERATION ID. Each message sent will - * have this base id attached to it concatenated to a counter. - * If this is an object, then it should map an id of a receiving parties - * to the base op_id that should be used to tag the messages sent to that party. - * Parties left unmapped by this object will get an automatically generated id. - * If this is a number/string, then it will be used as the base id tagging all messages - * sent by this open to all parties. - * You can safely ignore this unless you have multiple opens each containing other opens. - * In that case, the order by which these opens are executed is not fully deterministic - * and depends on the order of arriving messages. In this case, use this parameter - * with every nested_open, to ensure ids are unique and define a total ordering on - * the execution of the opens (check implementation of slt for an example). - * @returns {promise} a (JQuery) promise to ALL the open values of the secret, the promise will yield - * a 2D array of values, each corresponding to the given share in the shares parameter - * at the same index. In the case where different values are opened to different parties, the order - * of the values will be preserved, but not the indices, there will be no blanks in the resulting arrays, - * the first share that is opened to this party will appear at index [0], even if it was not initially - * at [0]. - * @throws error if some shares does not belong to the passed jiff instance. - */ - jiffClient.open_array = arraysSharing.jiff_open_array.bind(null, jiffClient); - - /** - * Opens an n-dimensional array of secret shares. - * @method open_ND_array - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {module:jiff-client~JIFFClient#SecretShare|module:jiff-client~JIFFClient#SecretShare[]|module:jiff-client~JIFFClient#SecretShare[][]} shares - an n-dimensional array containing this party's shares of the secrets to reconstruct. - * @param {Array[]} [parties=all_parties] - an array with party ids (1 to n) of receiving parties. - * @param {string|number|object} [op_ids=auto_gen()] - an optional ID/Tag associated with these open operations - * @returns {promise} a (JQuery) promise to ALL the open values of the secret, the promise will yield - * an n-dimensional array of values, each corresponding to the given share in the shares parameter - * at the same index. In the case where different values are opened to different parties, the order - * of the values will be preserved, but not the indices, there will be no blanks in the resulting arrays, - * the first share that is opened to this party will appear at indices [0][0], even if it was not initially - * at [0][0]. - * @throws error if some shares does not belong to the passed jiff instance. - */ - jiffClient.open_ND_array = arraysSharing.jiff_open_ND_array.bind(null, jiffClient); -}; - -},{"../protocols/arrays/api.js":21,"../protocols/shamir/open.js":39,"../protocols/shamir/reshare.js":40,"../protocols/shamir/share.js":41}],8:[function(require,module,exports){ -module.exports = function (jiffClient) { - var maxBarrierId = 10000000; - var currentBarrierId = 0; - var openBarriers = 0; - - /** - * Starts a new barrier, all promises and secret shares created between this call and the corresponding start_barrier - * call will be part of this barrier. start_barrier may be called before previous barriers are resolved, in which - * case promises / secret shares created will be part of the new barrier as well as any previous barriers. - * @memberof module:jiff-client~JIFFClient - * @method start_barrier - * @instance - * @returns {number} a barrier id that identifies this barrier. - */ - jiffClient.start_barrier = function () { - openBarriers++; - currentBarrierId = (currentBarrierId + 1 % maxBarrierId); - jiffClient.barriers[currentBarrierId] = []; - return currentBarrierId; - }; + var i, p_id, skeleton; - /** - * Adds given promise to all active barriers. - * @memberof module:jiff-client~JIFFClient - * @method add_to_barriers - * @instance - * @param {promise} promise - the promise to add. - */ - jiffClient.add_to_barriers = function (promise) { - if (openBarriers > 0) { - for (var id in jiffClient.barriers) { - if (jiffClient.barriers.hasOwnProperty(id)) { - jiffClient.barriers[id].push(promise); - } - } - } - }; + var isReceiving = receivers_list.indexOf(jiff.id) > -1; + var isSending = senders_list.indexOf(jiff.id) > -1; + if (!isSending && !isReceiving) { + return null; // This party is neither a sender nor a receiver, do nothing! + } - /** - * Executes the callback only after all promises / secret shares in the barrier were resolved. - * @memberof module:jiff-client~JIFFClient - * @method end_barrier - * @instance - * @param {number} [barrier_id=jiff.barriers.length - 1] - identifies the barrier, should be returned by start_barrier. - * by default, barrier_id will refer to the last barrier. - * @returns {promise} a promise that resolves after the secret shares are resolved. - */ - jiffClient.end_barrier = function (barrier_id) { - if (openBarriers === 0) { - return; - } + // Setup deferreds: required because we don't yet know how many shares to account for + var final_deferreds = []; + for (i = 0; i < senders_list.length; i++) { + p_id = senders_list[i]; + final_deferreds[p_id] = new jiff.helpers.Deferred(); + } - openBarriers--; - if (barrier_id == null) { - barrier_id = currentBarrierId; - } + const share_from_skeleton = share_from_skeleton_unbound.bind(null, jiff).bind(null, { + deferreds: final_deferreds, + threshold: threshold, + receivers_list: receivers_list, + Zp: Zp, + share_id: share_id + }); + + if (skeletons == null) { + if (isSending) { + // Send the shape and lengths of this party's array to all receivers + skeleton = jiff.skeleton_of(secrets); // All secrets are removed while maintaing the array's orginial structure. + var skeleton_str = JSON.stringify(skeleton); // serialize for emit + jiff.emit(share_id + 'skeleton', receivers_list, skeleton_str); + share_from_skeleton(jiff.id, secrets); // Share the real values matching the emitted skeleton + } + if (isReceiving) { + jiff.listen(share_id + 'skeleton', share_from_skeleton); // Receive shares when dimensions are known + } + } else { + senders_list = Array.from(senders_list); // remove jiff helpers' internal properties + util.match_skeletons(jiff, skeletons, senders_list); // Saftey check array dimention presets + for (i in senders_list) { + // Share each party's array + p_id = senders_list[i]; + var myself = p_id === jiff.id; + skeleton = skeletons[p_id]; + share_from_skeleton(p_id, myself ? secrets : skeleton); + } + } - var promise = Promise.all(jiffClient.barriers[barrier_id]); - delete jiffClient.barriers[barrier_id]; - return promise; - }; -}; -},{}],9:[function(require,module,exports){ -// Manages op_id counters and generation -module.exports = function (jiffClient) { - - /** - * Resets all the counters for op_ids - * @method counters.reset - * @memberof module:jiff-client~JIFFClient - * @instance - */ - jiffClient.counters.reset = function () { - jiffClient.counters.op_count = {}; - jiffClient.counters.op_count_preprocessing = {}; - - if (jiffClient.counters.pending_opens == null) { - jiffClient.counters.pending_opens = 0; - } + // Combine all promises and re-index final array map + var final_deferred = new jiff.helpers.Deferred(); + var final_promise = isReceiving ? final_deferred.promise : Promise.resolve({}); + Promise.all( + (function () { + var all_promises = []; + for (var i = 0; i < senders_list.length; i++) { + var p_id = senders_list[i]; + all_promises.push(final_deferreds[p_id].promise); + } + return all_promises; + })() + ).then(function (array) { + var shares = {}; + for (var i = 0; i < senders_list.length; i++) { + var p_id = senders_list[i]; + shares[p_id] = array[i]; + } + final_deferred.resolve(shares); + jiff.remove_listener(share_id + 'skeleton'); + }); + + return final_promise; // Return promise to map of secret-shared arrays + }; + + const share_ND_array_static = function (jiff, secrets, skeletons, threshold, receivers_list, senders_list, Zp, share_id) { + var parameters = [receivers_list, senders_list, threshold, Zp, share_id]; + [receivers_list, senders_list, threshold, Zp, share_id] = util.sanitize_array_params.bind(null, jiff).apply(null, parameters); + + var i; + + var shares = {}; + + var isReceiving = receivers_list.indexOf(jiff.id) > -1; + var isSending = senders_list.indexOf(jiff.id) > -1; + if (isSending || isReceiving) { + // Static version of share_from_skeleton + const share_from_skeleton = function (sender, skeleton) { + return (function share_recursive(__secrets, share_id) { + if (__secrets != null && __secrets.length === 0) { + return []; + } else if (__secrets != null && Array.isArray(__secrets)) { + var array = []; + for (var j = 0; j < __secrets.length; j++) { + array.push(share_recursive(__secrets[j], share_id + ':' + j)); + } + return isReceiving ? array : {}; + } else { + return jiff.share(__secrets, threshold, receivers_list, [sender], Zp, 'share:' + share_id)[sender]; + } + })(skeleton, share_id + ':p_id_' + sender); + }; + + senders_list = Array.from(senders_list); // remove jiff helpers' internal properties + util.match_skeletons(jiff, skeletons, senders_list); // Saftey check array dimention presets + for (i in senders_list) { + // Share each party's array + var p_id = senders_list[i]; + var skeleton = skeletons[p_id]; + shares[p_id] = share_from_skeleton(p_id, p_id === jiff.id ? secrets : skeleton); + } + } - // stores a seed for generating unique op_ids. - jiffClient.op_id_seed = ''; - }; + return isReceiving ? shares : {}; // Return promise to map of secret-shared arrays + }; + + const share_ND_array = function (jiff, secrets, skeletons, threshold, receivers_list, senders_list, Zp, share_id) { + var share_ND_array = skeletons != null ? share_ND_array_static : share_ND_array_deferred; + return share_ND_array(jiff, secrets, skeletons, threshold, receivers_list, senders_list, Zp, share_id); + }; + + module.exports = { + share_array: share_array, + share_2D_array: share_2D_array, + share_ND_array: share_ND_array + }; + }, + { './util.js': 24 } + ], + 24: [ + function (require, module, exports) { + const sanitize_array_params = function (jiff, receivers_list, senders_list, threshold, Zp, share_id) { + [receivers_list, senders_list] = [receivers_list, senders_list].map(function (party_list) { + if (party_list == null) { + party_list = []; + for (var i = 1; i <= jiff.party_count; i++) { + party_list.push(i); + } + } + jiff.helpers.sort_ids(party_list); // sort to get the same order + return party_list; + }); + + if (threshold == null) { + threshold = receivers_list.length; + } else if (threshold < 0) { + threshold = 2; + } else if (threshold > receivers_list.length) { + threshold = receivers_list.length; + } - // initialize the counters for the first time - jiffClient.counters.reset(); - - /** - * Shorthand for generating unique operation ids. - * All primitives called after this seed will use their usual default ids prefixed by the seed. - * Helpful when we have nested callbacks/functions (e.g. share_arrays) that may be executed in arbitrary order, - * using this function as a the first call inside such callbacks with an appropriate deterministic unique base_op_id - * ensures that regardless of the order of execution, operations in the same callback are matched correctly across - * all parties. - * Check out demos/graph-pip/mpc.js for an example on using this. - * @method seed_ids - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string|number} base_op_id - the base seed to use as a prefix for all future op_ids. - */ - jiffClient.seed_ids = function (base_op_id) { - if (base_op_id == null || base_op_id === '') { - jiffClient.op_id_seed = ''; - } else { - jiffClient.op_id_seed = base_op_id.toString() + ':'; - } - }; + if (Zp == null) { + Zp = jiff.Zp; + } - /** - * Generate a unique operation id for a new operation object. - * The returned op_id will be unique with respect to other operations, and identifies the same - * operation across all parties, as long as all parties are executing instructions in the same order. - * @method gen_op_id - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string} op - the type/name of operation performed. - * @param {Array} holders - an array containing the ids of all the parties carrying out the operation. - * @return {string} the op_id for the operation. - */ - jiffClient.counters.gen_op_id = function (op, holders) { - var label = jiffClient.op_id_seed + op + ':' + holders.join(','); - if (jiffClient.counters.op_count[label] == null) { - jiffClient.counters.op_count[label] = 0; - } - return label + ':' + (jiffClient.counters.op_count[label]++); - }; + if (share_id == null) { + // Make a unique id for this array + share_id = jiff.counters.gen_op_id2('share_array', receivers_list, senders_list); + } - /** - * Generate a unique operation id for a new operation object given two distinct executing parties lists. - * For example, when sharing, this is given two potentially different lists of senders and receivers. - * The returned op_id will be unique with respect to other operations, and identifies the same - * operation across all parties, as long as all parties are executing instructions in the same order. - * @method gen_op_id2 - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string} op - the type/name of operation performed. - * @param {Array} receivers - an array containing the ids of all the parties carrying out the receiving portion of the operation. - * @param {Array} senders - an array containing the ids of all the parties carrying out the sending portion of the operation. - * @return {string} the op_id for the operation. - */ - jiffClient.counters.gen_op_id2 = function (op, receivers, senders) { - var label = jiffClient.op_id_seed + op + ':' + senders.join(',') + ':' + receivers.join(','); - if (jiffClient.counters.op_count[label] == null) { - jiffClient.counters.op_count[label] = 0; - } - return label + ':' + (jiffClient.counters.op_count[label]++); - }; + return [receivers_list, senders_list, threshold, Zp, share_id]; + }; - /** - * Generate a unique operation id for a new operation object (for preprocessing) - * The returned op_id will be unique with respect to other operations, and identifies the same - * operation across all parties, as long as all parties are executing instructions in the same order. - * @method gen_op_id_preprocessing - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string} op - the type/name of operation performed. - * @param {Array} holders - an array containing the ids of all the parties carrying out the operation. - * @return {string} the op_id for the operation. - */ - jiffClient.counters.gen_op_id_preprocessing = function (op, holders) { - var label = jiffClient.op_id_seed + op + ':' + holders.join(','); - if (jiffClient.counters.op_count_preprocessing[label] == null) { - jiffClient.counters.op_count_preprocessing[label] = 0; - } - return label + ':' + (jiffClient.counters.op_count_preprocessing[label]++); - }; + const skeleton_of = function (jiff, nd_array, replace) { + if (typeof replace === 'undefined') { + replace = null; + } + if (!(typeof nd_array.length === 'undefined') || nd_array.length === 0) { + var wiped_array = []; + for (var k = 0; k < nd_array.length; k++) { + wiped_array.push(jiff.skeleton_of(nd_array[k], replace)); + } + return wiped_array; + } + return replace; + }; + + const match_skeletons = function (jiff, skeletons, senders_list) { + var keys = Object.keys(skeletons); + var expected_keys = senders_list.map(String); + if (keys.length === senders_list.length) { + for (var i = 0; i < senders_list.length; i++) { + if (!(keys[i] === expected_keys[i])) { + console.log('senders: ', senders_list); + console.log('skeleton keys', keys); + throw new Error("Keys of parameter 'skeletons' should be the same as the senders"); + } + } + } else { + throw new Error("Invalid parameter 'skeletons'"); + } + }; + + module.exports = { + sanitize_array_params: sanitize_array_params, + skeleton_of: skeleton_of, + match_skeletons: match_skeletons + }; + }, + {} + ], + 25: [ + function (require, module, exports) { + module.exports = { + /** + * Compute sum of bitwise secret shared number and a constant + * @function cadd + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - the bit wise secret shares + * @param {number} constant - the constant + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length here will be max(|bits|, |constant|) + 1 + * in case of potential overflow / carry + */ + cadd: function (jiff, bits, constant, op_id) { + if (!bits[0].isConstant(constant)) { + throw new Error('parameter should be a number (bits.cadd)'); + } + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.cadd', bits[0].holders); + } + + if (constant.toString() === '0') { + return bits; + } + + // copy to avoid aliasing problems during execution + bits = bits.slice(); + + // decompose constant into bits + var constant_bits = jiff.helpers.number_to_bits(constant, bits.length); // pads with zeros to bits.length + + // initialize results + var result = jiff.utils.many_secret_shares(Math.max(constant_bits.length, bits.length), bits[0].holders, bits[0].threshold, bits[0].Zp); + var deferreds = result.deferreds; + result = result.shares; + + var sum = bits[0].icxor_bit(constant_bits[0]); + var carry = bits[0].icmult(constant_bits[0]); + + // put initial bit at head of result array + result.unshift(sum); + deferreds.unshift(null); + + // compute sum one bit at a time, propagating carry + jiff.utils.bit_combinator(deferreds[deferreds.length - 1], 1, deferreds.length - 1, carry, function (i, carry) { + var sum; + if (i < bits.length) { + var and = bits[i].icmult(constant_bits[i]); + var xor = bits[i].icxor_bit(constant_bits[i]); + var xorAndCarry = xor.ismult(carry, op_id + ':smult:' + (i - 1)); + + sum = xor.isxor_bit(carry, op_id + ':sxor_bit:' + (i - 1)); + carry = and.isadd(xorAndCarry); // cheap or, xor and and cannot both be true! + } else { + // bits.length <= i < constant_bits.length + // and is zero, xor is constant_bits[i] + sum = carry.icxor_bit(constant_bits[i]); + carry = carry.icmult(constant_bits[i]); + } + + sum.wThen(deferreds[i].resolve); + return carry; + }); + + return result; + }, + /** + * Compute [secret bits] - [constant bits] + * @function csubl + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {number} constant - the constant + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - the bit wise secret shares + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where + * the bit at index 0 is the least significant bit. The bit at index 1 is the most significant bit, + * and the bit at index |bits| is 1 if the result overflows, or 0 otherwise + */ + csubl: function (jiff, bits, constant, op_id) { + if (!bits[0].isConstant(constant)) { + throw new Error('parameter should be a number (bits.csubl)'); + } + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.csubl', bits[0].holders); + } + + if (constant.toString() === '0') { + return bits; + } + + // copy to avoid aliasing problems during execution + bits = bits.slice(); + + // decompose constant into bits + var constant_bits = jiff.helpers.number_to_bits(constant, bits.length); // pads with zeros to bits.length + + // initialize results + var result = jiff.utils.many_secret_shares(Math.max(constant_bits.length, bits.length), bits[0].holders, bits[0].threshold, bits[0].Zp); + var deferreds = result.deferreds; + result = result.shares; + + var diff = bits[0].icxor_bit(constant_bits[0]); + var borrow = bits[0].inot().icmult(constant_bits[0]); + + // put initial bit at head of result array + result.unshift(diff); + deferreds.unshift(null); + + // compute diff one bit at a time, propagating borrow + jiff.utils.bit_combinator(deferreds[deferreds.length - 1], 1, deferreds.length - 1, borrow, function (i, borrow) { + var diff; + if (i < bits.length) { + var xor = bits[i].icxor_bit(constant_bits[i]); + var andNot = bits[i].inot().icmult(constant_bits[i]); + + // save and update borrow + diff = xor.isxor_bit(borrow, op_id + ':sxor_bit:' + (i - 1)); + borrow = xor.inot().ismult(borrow, op_id + ':smult:' + (i - 1)); + borrow = borrow.isadd(andNot); + } else { + // bits.length <= i < constant_bits.length + // xor and andNot are equal to the constant bit value since secret bit is always zero here + diff = borrow.icxor_bit(constant_bits[i]); + borrow = borrow.issub(borrow.icmult(constant_bits[i])); + borrow = borrow.icadd(constant_bits[i]); + } + + diff.wThen(deferreds[i].resolve); + return borrow; + }); + + return result; + }, + /** + * Compute [constant bits] - [secret bits] + * @function csubr + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {number} constant - the constant + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - the bit wise secret shares + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where + * the bit at index 0 is the least significant bit. The bit at index 1 is the most significant bit, + * and the bit at index |bits| is 1 if the result overflows, or 0 otherwise + */ + csubr: function (jiff, constant, bits, op_id) { + if (!bits[0].isConstant(constant)) { + throw new Error('parameter should be a number (bits.csubr)'); + } + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.csubr', bits[0].holders); + } + + // copy to avoid aliasing problems during execution + bits = bits.slice(); + + // decompose constant into bits + var constant_bits = jiff.helpers.number_to_bits(constant, bits.length); // pads with zeros to bits.length + + // initialize results + var result = jiff.utils.many_secret_shares(Math.max(constant_bits.length, bits.length), bits[0].holders, bits[0].threshold, bits[0].Zp); + var deferreds = result.deferreds; + result = result.shares; + + var diff = bits[0].icxor_bit(constant_bits[0]); + var borrow = bits[0].issub(bits[0].icmult(constant_bits[0])); + + // put initial bit at head of result array + result.unshift(diff); + deferreds.unshift(null); + + // compute diff one bit at a time, propagating borrow + jiff.utils.bit_combinator(deferreds[deferreds.length - 1], 1, deferreds.length - 1, borrow, function (i, borrow) { + var diff; + if (i < bits.length) { + var xor = bits[i].icxor_bit(constant_bits[i]); + var andNot = bits[i].issub(bits[i].icmult(constant_bits[i])); + + // save and update borrow + diff = xor.isxor_bit(borrow, op_id + ':sxor_bit:' + (i - 1)); + borrow = xor.inot().ismult(borrow, op_id + ':smult:' + (i - 1)); + borrow = borrow.isadd(andNot); + } else { + // andNot is zero and xor is equal to the constant bit since secret bit is always zero here. + diff = borrow.icxor_bit(constant_bits[i]); + borrow = borrow.icmult(constant_bits[i] === 1 ? 0 : 1); + } + + diff.wThen(deferreds[i].resolve); + return borrow; + }); + + return result; + }, + /** + * + * Compute [secret bits1] + [secret bits2] + * @function sadd + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - the first bitwise shared number: array of secrets with index 0 being least significant bit + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number (length may be different) + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where + * the bit at index 0 is the least significant bit + */ + sadd: function (jiff, bits1, bits2, op_id) { + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.sadd', bits1[0].holders); + } + + // copy to avoid aliasing problems during execution + bits1 = bits1.slice(); + bits2 = bits2.slice(); + + var tmp = bits1.length > bits2.length ? bits1 : bits2; + bits2 = bits1.length > bits2.length ? bits2 : bits1; // shortest array + bits1 = tmp; // longest array + + // initialize results + var result = jiff.utils.many_secret_shares(bits1.length, bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); + var deferreds = result.deferreds; + result = result.shares; + + var sum = bits1[0].isxor_bit(bits2[0], op_id + ':sxor_bit:initial'); + var carry = bits1[0].ismult(bits2[0], op_id + ':smult:initial'); + + // put initial bit at head of result array + result.unshift(sum); + deferreds.unshift(null); + + // compute sum one bit at a time, propagating carry + jiff.utils.bit_combinator(deferreds[deferreds.length - 1], 1, deferreds.length - 1, carry, function (i, carry) { + var sum; + if (i < bits2.length) { + var and = bits1[i].ismult(bits2[i], op_id + ':smult1:' + (i - 1)); + var xor = bits1[i].isxor_bit(bits2[i], op_id + ':sxor_bit1:' + (i - 1)); + var xorAndCarry = xor.ismult(carry, op_id + ':smult2:' + (i - 1)); + + sum = xor.isxor_bit(carry, op_id + ':sxor_bit2:' + (i - 1)); + carry = and.isadd(xorAndCarry); // cheap or, xor and and cannot both be true! + } else { + // and is always zero, xor is equal to bits1[i] + sum = bits1[i].isxor_bit(carry, op_id + ':sxor_bit1:' + (i - 1)); + carry = bits1[i].ismult(carry, op_id + ':smult1:' + (i - 1)); + } + + sum.wThen(deferreds[i].resolve); + return carry; + }); + + return result; + }, + /** + * Compute [secret bits1] - [secret bits2] + * @function ssub + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - first bitwise secret shared number: lower indices represent less significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - second bitwise secret shared number (length may be different) + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where + * the bit at index 0 is the least significant bit. The bit at index 1 is the most significant bit, + * and the bit at index |bits| is 1 if the result overflows, or 0 otherwise + */ + ssub: function (jiff, bits1, bits2, op_id) { + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.ssub', bits1[0].holders); + } + + // copy to avoid aliasing problems during execution + bits1 = bits1.slice(); + bits2 = bits2.slice(); + + // initialize results + var result = jiff.utils.many_secret_shares(Math.max(bits1.length, bits2.length), bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); + var deferreds = result.deferreds; + result = result.shares; + + var diff = bits1[0].isxor_bit(bits2[0], op_id + ':sxor_bit:initial'); + var borrow = bits1[0].inot().ismult(bits2[0], op_id + ':smult:initial'); + + // put initial bit at head of result array + result.unshift(diff); + deferreds.unshift(null); + + // compute diff one bit at a time, propagating borrow + jiff.utils.bit_combinator(deferreds[deferreds.length - 1], 1, deferreds.length - 1, borrow, function (i, borrow) { + var diff; + if (i < bits1.length && i < bits2.length) { + var xor = bits1[i].isxor_bit(bits2[i], op_id + ':sxor_bit1:' + (i - 1)); + var andNot = bits1[i].inot().ismult(bits2[i], op_id + ':smult1:' + (i - 1)); + + // save and update borrow + diff = xor.isxor_bit(borrow, op_id + ':sxor_bit2:' + (i - 1)); + borrow = xor.inot().ismult(borrow, op_id + ':smult2:' + (i - 1)); + borrow = borrow.isadd(andNot); + } else if (i < bits1.length) { + // xor is equal to the value of bits1[i], andNot is equal to 0, since bits[2] is all zeros here + diff = bits1[i].isxor_bit(borrow, op_id + ':sxor_bit1:' + (i - 1)); + borrow = bits1[i].inot().ismult(borrow, op_id + ':smult1:' + (i - 1)); + } else { + // i < bits2.length + // xor and andNot are equal to the value of bits2[i] + diff = bits2[i].isxor_bit(borrow, op_id + ':sxor_bit1:' + (i - 1)); + borrow = bits2[i].inot().ismult(borrow, op_id + ':smult1:' + (i - 1)); + borrow = borrow.isadd(bits2[i]); + } + + diff.wThen(deferreds[i].resolve); + return borrow; + }); + + return result; + }, + /** + * Compute [secret bits] * constant + * @function cmult + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - bitwise shared secret to multiply: lower indices represent less significant bits + * @param {number} constant - constant to multiply with + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result, the length of the result will be bits.length + ceil(log2(constant)), except + * if constant is zero, the result will then be [ zero share ] + */ + cmult: function (jiff, bits, constant, op_id) { + if (!bits[0].isConstant(constant)) { + throw new Error('parameter should be a number (bits.cmult)'); + } + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.cmult', bits[0].holders); + } + + // copy to avoid aliasing problems during execution + bits = bits.slice(); + + // decompose constant into bits + var constant_bits = jiff.helpers.number_to_bits(constant); // do not pad + + // Initialize the result + var result = jiff.utils.many_secret_shares(bits.length + constant_bits.length, bits[0].holders, bits[0].threshold, bits[0].Zp); + var deferreds = result.deferreds; + result = result.shares; + + // Resolve result when ready + var final_deferred = new jiff.helpers.Deferred(); + final_deferred.promise.then(jiff.utils.resolve_many_secrets.bind(null, deferreds)); + + // get useless share of zero (just for padding) + var zero = new jiff.SecretShare(0, bits[0].holders, bits[0].threshold, bits[0].Zp); + var initial = [zero]; + + // special case + if (constant.toString() === '0') { + return initial; + } + + // main function + jiff.utils.bit_combinator( + final_deferred, + 0, + constant_bits.length, + initial, + function (i, intermediate) { + // Shift bits to create the intermediate values, + // and sum if the corresponding bit in a is 1 + if (constant_bits[i].toString() === '1') { + intermediate = jiff.protocols.bits.sadd(intermediate, bits, op_id + ':bits.sadd:' + i); + } + + bits.unshift(zero); + return intermediate; + }, + function (intermediate) { + // promise-ify an array of intermediate results + var promises = []; + for (var i = 0; i < intermediate.length; i++) { + promises.push(intermediate[i].value); + } + return Promise.all(promises); + }, + function (result) { + // identity + return result; + } + ); + + return result; + }, + /** + * Compute [secret bits1] * [secret bits2] + * @function smult + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - bitwise shared secret to multiply: lower indices represent less significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - bitwise shared secret to multiply + * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result, the length of the result will be bits1.length + bits2.length + */ + smult: function (jiff, bits1, bits2, op_id) { + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.smult', bits1[0].holders); + } + + // copy to avoid aliasing problems during execution + bits1 = bits1.slice(); + bits2 = bits2.slice(); + + // bits1 will be the longest array, bits2 will be the shortest + var tmp = bits1.length > bits2.length ? bits1 : bits2; + bits2 = bits1.length > bits2.length ? bits2 : bits1; + bits1 = tmp; + + // Initialize the result + var offset = bits2.length === 1 ? -1 : 0; + var result = jiff.utils.many_secret_shares(bits1.length + bits2.length + offset, bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); + var deferreds = result.deferreds; + result = result.shares; + + // Resolve result when ready + var final_deferred = new jiff.helpers.Deferred(); + final_deferred.promise.then(jiff.utils.resolve_many_secrets.bind(null, deferreds)); + + // Loop over *shortest* array one bit at a time + jiff.utils.bit_combinator( + final_deferred, + 0, + bits2.length, + bits2, + function (i, intermediate) { + var this_bit = bits2[i]; + var bit_mult = []; // add bits1 or 0 to the result according to this bit + for (var j = 0; j < bits1.length; j++) { + bit_mult[j] = this_bit.iif_else(bits1[j], 0, op_id + ':if_else:' + i + ':' + j); + } + bits1.unshift(0); // increase magnitude + + if (i === 0) { + return bit_mult; + } + + return jiff.protocols.bits.sadd(intermediate, bit_mult, op_id + ':bits.sadd:' + i); + }, + function (intermediate) { + // promise-ify an array of intermediate results + var promises = []; + for (var i = 0; i < intermediate.length; i++) { + promises.push(intermediate[i].value); + } + return Promise.all(promises); + }, + function (result) { + // identity + return result; + } + ); + + return result; + }, + /** + * Computes integer division of [secret bits 1] / [secret bits 2] + * @function sdiv + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {{quotient: module:jiff-client~JIFFClient#SecretShare[], remainder: module:jiff-client~JIFFClient#SecretShare[]}} the quotient and remainder bits arrays, note that + * the quotient array has the same length as bits1, + * and the remainder array has the same length as bits2 or bits1, whichever is smaller. + * Note: if bits2 represent 0, the returned result is the maximum + * number that fits in the number of bits (all 1), and the remainder + * is equal to bits1 + */ + sdiv: function (jiff, bits1, bits2, op_id) { + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.sdiv', bits1[0].holders); + } + + // copy to avoid aliasing problems during execution + bits1 = bits1.slice(); + bits2 = bits2.slice(); + + // Initialize the result + var quotient = jiff.utils.many_secret_shares(bits1.length, bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); + var quotientDeferreds = quotient.deferreds; + quotient = quotient.shares; + + var remainder = jiff.utils.many_secret_shares(Math.min(bits1.length, bits2.length), bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); + var remainderDeferreds = remainder.deferreds; + remainder = remainder.shares; + + // Resolve result when ready + var final_deferred = new jiff.helpers.Deferred(); + final_deferred.promise.then(function (result) { + jiff.utils.resolve_many_secrets(remainderDeferreds, result); + }); + + var initial = []; // initial remainder + jiff.utils.bit_combinator( + final_deferred, + bits1.length - 1, + -1, + initial, + function (i, _remainder) { + var iterationCounter = bits1.length - i - 1; + + // add bit i to the head of remainder (least significant bit) + _remainder.unshift(bits1[i]); + + // Get the next bit of the quotient + // and conditionally subtract b from the + // intermediate remainder to continue + var sub = jiff.protocols.bits.ssub(_remainder, bits2, op_id + ':bits.ssub:' + iterationCounter); + var noUnderflow = sub.pop().inot(); // get the overflow bit, sub is now the result of subtraction + + // Get next bit of quotient + noUnderflow.wThen(quotientDeferreds[i].resolve); + + // Update remainder + for (var j = 0; j < _remainder.length; j++) { + // note, if noUnderflow, then |# bits in sub| <= |# bits in remainder| + _remainder[j] = noUnderflow.iif_else(sub[j], _remainder[j], op_id + ':if_else:' + iterationCounter + ':' + j); + } + + // Remainder cannot be greater than divisor at this point + while (_remainder.length > remainder.length) { + _remainder.pop(); + } + + return _remainder; + }, + function (intermediate) { + // promise-ify an array of intermediate results + var promises = []; + for (var i = 0; i < intermediate.length; i++) { + promises.push(intermediate[i].value); + } + return Promise.all(promises); + }, + function (result) { + // identity + return result; + } + ); + + return { quotient: quotient, remainder: remainder }; + }, + /** + * Computes integer division of [secret bits] / constant + * @function cdivl + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - numerator: an array of secret shares of bits, starting from least to most significant bits + * @param {number} constant - the denominator number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {{quotient: module:jiff-client~JIFFClient#SecretShare[], remainder: module:jiff-client~JIFFClient#SecretShare[]}} the quotient and remainder bits arrays, note that + * the quotient array has the same length as bits, + * and the remainder array has the same length as + * constant or bits, whichever is smaller + * @throws if constant is 0. + */ + cdivl: function (jiff, bits, constant, op_id) { + if (!bits[0].isConstant(constant)) { + throw new Error('parameter should be a number (bits.cdivl)'); + } + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.cdivl', bits[0].holders); + } + + if (constant.toString() === '0') { + throw new Error('constant cannot be 0 in bits.cdiv'); + } + + // copy to avoid aliasing problems during execution + bits = bits.slice(); + + // special case, divide by 1 + if (constant.toString() === '1') { + return { + quotient: bits, + remainder: [new jiff.SecretShare(0, bits[0].holders, bits[0].threshold, bits[0].Zp)] + }; + } + + // Initialize the result + var quotient = jiff.utils.many_secret_shares(bits.length, bits[0].holders, bits[0].threshold, bits[0].Zp); + var quotientDeferreds = quotient.deferreds; + quotient = quotient.shares; + + var constantLessBits = jiff.helpers.ceil(jiff.helpers.bLog(constant, 2)); + constantLessBits = parseInt(constantLessBits.toString(), 10); + var remainder = jiff.utils.many_secret_shares(Math.min(constantLessBits, bits.length), bits[0].holders, bits[0].threshold, bits[0].Zp); + var remainderDeferreds = remainder.deferreds; + remainder = remainder.shares; + + // Resolve result when ready + var final_deferred = new jiff.helpers.Deferred(); + final_deferred.promise.then(jiff.utils.resolve_many_secrets.bind(null, remainderDeferreds)); + + var initial = []; // initial remainder + jiff.utils.bit_combinator( + final_deferred, + bits.length - 1, + -1, + initial, + function (i, _remainder) { + var iterationCounter = bits.length - i - 1; + + // add bit i to the head of remainder (least significant bit) + _remainder.unshift(bits[i]); + + // Get the next bit of the quotient + // and conditionally subtract b from the + // intermediate remainder to continue + var sub = jiff.protocols.bits.csubl(_remainder, constant, op_id + ':bits.csubl:' + iterationCounter); + var noUnderflow = sub.pop().inot(); // get the overflow bit, sub is now the result of subtraction + + // Get next bit of quotient + noUnderflow.wThen(quotientDeferreds[i].resolve); + + // Update remainder + for (var j = 0; j < _remainder.length; j++) { + // note, if noUnderflow, then |# bits in sub| <= |# bits in remainder| + _remainder[j] = noUnderflow.iif_else(sub[j], _remainder[j], op_id + ':if_else:' + iterationCounter + ':' + j); + } + + // Remainder cannot be greater than constant at this point + while (_remainder.length > remainder.length) { + _remainder.pop(); + } + + return _remainder; + }, + function (intermediate) { + // promise-ify an array of intermediate results + var promises = []; + for (var i = 0; i < intermediate.length; i++) { + promises.push(intermediate[i].value); + } + return Promise.all(promises); + }, + function (result) { + // identity + return result; + } + ); + + return { quotient: quotient, remainder: remainder }; + }, + /** + * Computes integer division of constant / [secret bits] + * @function cdivr + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {number} constant - the numerator number + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - denominator: an array of secret shares of bits, starting from least to most significant bits + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {{quotient: module:jiff-client~JIFFClient#SecretShare[], remainder: module:jiff-client~JIFFClient#SecretShare[]}} the quotient and remainder bits arrays, note that + * the quotient array has the same length as the number of bits in constant, + * and the remainder array has the same length as bits or constant, whichever is smaller. + * Note: if bits represent 0, the returned result is the maximum + * number that fits in its bits (all 1), and the remainder + * is equal to constant + */ + cdivr: function (jiff, constant, bits, op_id) { + if (!bits[0].isConstant(constant)) { + throw new Error('parameter should be a number (bits.cdivr)'); + } + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.cdivr', bits[0].holders); + } + + // copy to avoid aliasing problems during execution + bits = bits.slice(); + + // do not pad + var constant_bits = jiff.helpers.number_to_bits(constant); + + // Initialize the result + var quotient = jiff.utils.many_secret_shares(constant_bits.length, bits[0].holders, bits[0].threshold, bits[0].Zp); + var quotientDeferreds = quotient.deferreds; + quotient = quotient.shares; + + var remainder = jiff.utils.many_secret_shares(Math.min(constant_bits.length, bits.length), bits[0].holders, bits[0].threshold, bits[0].Zp); + var remainderDeferreds = remainder.deferreds; + remainder = remainder.shares; + + // Resolve result when ready + var final_deferred = new jiff.helpers.Deferred(); + final_deferred.promise.then(jiff.utils.resolve_many_secrets.bind(null, remainderDeferreds)); + + var initial = []; // initial remainder + jiff.utils.bit_combinator( + final_deferred, + constant_bits.length - 1, + -1, + initial, + function (i, _remainder) { + var iterationCounter = constant_bits.length - i - 1; + + // add bit i to the head of remainder (least significant bit) + // turn into a secret without communication, just for typing + var cbit_share = new jiff.SecretShare(constant_bits[i], bits[0].holders, bits[0].threshold, bits[0].Zp); + _remainder.unshift(cbit_share); + + // Get the next bit of the quotient + // and conditionally subtract b from the + // intermediate remainder to continue + var sub = jiff.protocols.bits.ssub(_remainder, bits, op_id + ':bits.ssub:' + iterationCounter); + var noUnderflow = sub.pop().inot(); // get the overflow bit, sub is now the result of subtraction + + // Get next bit of quotient + noUnderflow.wThen(quotientDeferreds[i].resolve); + + // Update remainder + for (var j = 0; j < _remainder.length; j++) { + // note, if noUnderflow, then |# bits in sub| <= |# bits in remainder| + _remainder[j] = noUnderflow.iif_else(sub[j], _remainder[j], op_id + ':if_else:' + iterationCounter + ':' + j); + } + + // cannot be bigger than divisor at this point + while (_remainder.length > remainder.length) { + _remainder.pop(); + } + + return _remainder; + }, + function (intermediate) { + // promise-ify an array of intermediate results + var promises = []; + for (var i = 0; i < intermediate.length; i++) { + promises.push(intermediate[i].value); + } + return Promise.all(promises); + }, + function (result) { + // identity + return result; + } + ); + + return { quotient: quotient, remainder: remainder }; + } + }; + }, + {} + ], + 26: [ + function (require, module, exports) { + module.exports = { + /** + * Checks whether the given bitwise secret shared number and numeric constant are equal + * @function ceq + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of secret shares of bits, starting from least to most significant bits + * @param {number} constant - the constant number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {SecretShare|boolean} a secret share of 1 if parameters are equal, 0 otherwise. If result is known + * (e.g. constant has a greater non-zero bit than bits' most significant bit), the result is + * returned immediately as a boolean + */ + ceq: function (jiff, bits, constant, op_id) { + if (!bits[0].isConstant(constant)) { + throw new Error('parameter should be a number (bits.ceq)'); + } + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.ceq', bits[0].holders); + } + var result = jiff.protocols.bits.cneq(bits, constant, op_id); + if (result === true || result === false) { + return !result; + } + return result.inot(); + }, + /** + * Checks whether the given bitwise secret shared number and numeric constant are not equal + * @function cneq + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of secret shares of bits, starting from least to most significant bits + * @param {number} constant - the constant number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {SecretShare|boolean} a secret share of 1 if parameters are not equal, 0 otherwise. If result is known + * (e.g. constant has a greater non-zero bit than bits' most significant bit), the result is + * returned immediately as a boolean + */ + cneq: function (jiff, bits, constant, op_id) { + if (!bits[0].isConstant(constant)) { + throw new Error('parameter should be a number (bits.cneq)'); + } + + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.cneq', bits[0].holders); + } + + // copy to avoid aliasing problems during execution + bits = bits.slice(); + + var constant_bits = jiff.helpers.number_to_bits(constant, bits.length); + if (constant_bits.length > bits.length) { + // Optimization: if constant has more bits, one of them must be 1, constant must be greater than bits. + return true; + } + + var deferred = new jiff.helpers.Deferred(); + var result = new jiff.SecretShare(deferred.promise, bits[0].holders, bits[0].threshold, bits[0].Zp); + + // big or of bitwise XORs + var initial = bits[0].icxor_bit(constant_bits[0]); + jiff.utils.bit_combinator(deferred, 1, bits.length, initial, function (i, prev) { + var xor = bits[i].icxor_bit(constant_bits[i]); + xor = prev.isor_bit(xor, op_id + ':sor_bit:' + (i - 1)); + return xor; + }); + + return result; + }, + /** + * Checks whether given secret shared bits are greater than the given constant + * @function cgt + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits + * @param {number} constant - the constant number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {SecretShare|boolean} a secret share of 1 if bits are greater than constant, 0 otherwise, if result is known + * (e.g. constant has greater non-zero bit than bits' most significant bit), the result is + * returned immediately as a boolean + */ + cgt: function (jiff, bits, constant, op_id) { + if (!bits[0].isConstant(constant)) { + throw new Error('parameter should be a number (bits.cgt)'); + } + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.cgt', bits[0].holders); + } + return jiff.protocols.bits.cgteq(bits, constant + 1, op_id); + }, + /** + * Checks whether given secret shared bits are greater or equal to the given constant + * @function cgteq + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits + * @param {number} constant - the constant number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {SecretShare|boolean} a secret share of 1 if bits are greater or equal to constant, 0 otherwise, if result is known + * (e.g. constant has greater non-zero bit than bits' most significant bit or constant is zero), the result is + * returned immediately as a boolean + */ + cgteq: function (jiff, bits, constant, op_id) { + if (!bits[0].isConstant(constant)) { + throw new Error('parameter should be a number (bits.cgteq)'); + } + + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.cgteq', bits[0].holders); + } + + // copy to avoid aliasing problems during execution + bits = bits.slice(); + + // Optimization: the bits are a share of non-negative number, if constant <= 0, return true + if (constant.toString().startsWith('-') || constant.toString() === '0') { + return true; + } + + // decompose result into bits + var constant_bits = jiff.helpers.number_to_bits(constant, bits.length); + if (constant_bits.length > bits.length) { + // Optimization: if constant has more bits, one of them must be 1, constant must be greater than bits. + return false; + } + + // initialize result + var deferred = new jiff.helpers.Deferred(); + var result = new jiff.SecretShare(deferred.promise, bits[0].holders, bits[0].threshold, bits[0].Zp); + + // Subtract bits2 from bits1, only keeping track of borrow + var borrow = bits[0].inot().icmult(constant_bits[0]); + + // compute one bit at a time, propagating borrow + jiff.utils.bit_combinator(deferred, 1, bits.length, borrow, function (i, borrow) { + var xor = bits[i].icxor_bit(constant_bits[i]); + var andNot = bits[i].inot().icmult(constant_bits[i]); + + // save and update borrow + borrow = xor.inot().ismult(borrow, op_id + ':smult:' + (i - 1)); + return borrow.isadd(andNot); + }); + + return result.inot(); + }, + /** + * Checks whether given secret shared bits are less than the given constant + * @function clt + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits + * @param {number} constant - the constant number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {SecretShare|boolean} a secret share of 1 if bits are less than the constant, 0 otherwise, if result is known + * (e.g. constant has greater non-zero bit than bits' most significant bit), the result is + * returned immediately as a boolean + */ + clt: function (jiff, bits, constant, op_id) { + if (!bits[0].isConstant(constant)) { + throw new Error('parameter should be a number (bits.clt)'); + } + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.clt', bits[0].holders); + } + var result = jiff.protocols.bits.cgteq(bits, constant, op_id); + if (result === true || result === false) { + return !result; + } + return result.inot(); + }, + /** + * Checks whether given secret shared bits are less or equal to the given constant + * @function clteq + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits + * @param {number} constant - the constant number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {SecretShare|boolean} a secret share of 1 if bits are less or equal to constant, 0 otherwise, if result is known + * (e.g. constant has greater non-zero bit than bits' most significant bit), the result is + * returned immediately as a boolean + */ + clteq: function (jiff, bits, constant, op_id) { + if (!bits[0].isConstant(constant)) { + throw new Error('parameter should be a number (bits.clteq)'); + } + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.clteq', bits[0].holders); + } + var result = jiff.protocols.bits.cgt(bits, constant, op_id); + if (result === true || result === false) { + return !result; + } + return result.inot(); + }, + /** + * Checks whether the two given bitwise secret shared numbers are equal + * @function seq + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if bits are equal, 0 otherwise + */ + seq: function (jiff, bits1, bits2, op_id) { + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.seq', bits1[0].holders); + } + return jiff.protocols.bits.sneq(bits1, bits2, op_id).inot(); + }, + /** + * Checks whether the two given bitwise secret shared numbers are not equal + * @function sneq + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if bits are not equal, 0 otherwise + */ + sneq: function (jiff, bits1, bits2, op_id) { + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.sneq', bits1[0].holders); + } + + var tmp = bits1.length > bits2.length ? bits1 : bits2; + bits2 = bits1.length > bits2.length ? bits2 : bits1; // shortest array + bits1 = tmp; // longest array + + // copy to avoid aliasing problems during execution + bits1 = bits1.slice(); + bits2 = bits2.slice(); + + // initialize result + var deferred = new jiff.helpers.Deferred(); + var result = new jiff.SecretShare(deferred.promise, bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); + + // big or of bitwise XORs + var initial = bits1[0].isxor_bit(bits2[0], op_id + ':sxor_bit:initial'); + jiff.utils.bit_combinator(deferred, 1, bits1.length, initial, function (i, prev) { + var next; + if (i < bits2.length) { + var xor = bits1[i].isxor_bit(bits2[i], op_id + ':sxor_bit:' + (i - 1)); + next = prev.isor_bit(xor, op_id + ':sor_bit:' + (i - 1)); + } else { + // xor is equal to bits1[i] since bits2[i] is zero + next = prev.isor_bit(bits1[i], op_id + ':sor_bit:' + (i - 1)); + } + return next; + }); + + return result; + }, + /** + * Checks whether the first given bitwise secret shared number is greater than the second bitwise secret shared number + * @function sgt + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is greater than the second, 0 otherwise + */ + sgt: function (jiff, bits1, bits2, op_id) { + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.sgt', bits1[0].holders); + } + + var gteq = jiff.protocols.bits.sgteq(bits1, bits2, op_id + ':bits.sgteq'); + var neq = jiff.protocols.bits.sneq(bits1, bits2, op_id + ':bits.sneq'); + return gteq.ismult(neq, op_id + ':smult'); + }, + /** + * Checks whether the first given bitwise secret shared number is greater than or equal to the second bitwise secret shared number + * @function sgteq + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is greater or equal to the second, 0 otherwise + */ + sgteq: function (jiff, bits1, bits2, op_id) { + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.sgteq', bits1[0].holders); + } + + // copy to avoid aliasing problems during execution + bits1 = bits1.slice(); + bits2 = bits2.slice(); + + // initialize result + var deferred = new jiff.helpers.Deferred(); + var result = new jiff.SecretShare(deferred.promise, bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); + + // Subtract bits2 from bits1, only keeping track of borrow + var borrow = bits1[0].inot().ismult(bits2[0], op_id + ':smult:initial'); + var n = Math.max(bits1.length, bits2.length); + jiff.utils.bit_combinator(deferred, 1, n, borrow, function (i, borrow) { + if (i < bits1.length && i < bits2.length) { + var xor = bits1[i].isxor_bit(bits2[i], op_id + ':sxor_bit1:' + (i - 1)); + var andNot = bits1[i].inot().ismult(bits2[i], op_id + ':smult1:' + (i - 1)); + + // save and update borrow + borrow = xor.inot().ismult(borrow, op_id + ':smult2:' + (i - 1)); + borrow = borrow.isadd(andNot); + } else if (i < bits1.length) { + // xor is equal to the value of bits1[i], andNot is equal to 0, since bits[2] is all zeros here + borrow = bits1[i].inot().ismult(borrow, op_id + ':smult1:' + (i - 1)); + } else { + // i < bits2.length + // xor and andNot are equal to the value of bits2[i] + borrow = bits2[i].inot().ismult(borrow, op_id + ':smult1:' + (i - 1)); + borrow = borrow.isadd(bits2[i]); + } + + return borrow; + }); + + return result.inot(); + }, + /** + * Checks whether the first given bitwise secret shared number is less than the second bitwise secret shared number + * @function slt + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is less than the second, 0 otherwise + */ + slt: function (jiff, bits1, bits2, op_id) { + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.slt', bits1[0].holders); + } + var result = jiff.protocols.bits.sgteq(bits1, bits2, op_id); + return result.inot(); + }, + /** + * Checks whether the first given bitwise secret shared number is less or equal to the second bitwise secret shared number + * @function slteq + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number + * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. + * default value should suffice when the code of all parties executes all instructions + * in the same exact order, otherwise, a unique base name is needed here + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is less than or equal to the second, 0 otherwise + */ + slteq: function (jiff, bits1, bits2, op_id) { + if (op_id == null) { + op_id = jiff.counters.gen_op_id('bits.slteq', bits1[0].holders); + } + var result = jiff.protocols.bits.sgt(bits1, bits2, op_id); + return result.inot(); + } + }; + }, + {} + ], + 27: [ + function (require, module, exports) { + module.exports = { + /** + * Retrieves preprocessed rejection_sampling bits or performs the rejection sampling on the fly if crypto_provider is enabled + * @function rejection_sampling + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff instance + * @param {number} [lower_bound=0] - the lower bound, included (can be a bigNumber if using bigNumber extension) + * @param {number} [upper_bound=jiff-instance.Zp] - the upper bound, excluded (can be a bigNumber if using bigNumber extension) + * @param {number} [threshold=parties.length] - the threshold of the resulting shares after sampling + * @param {Array} [parties=all_parties] - array of party ids that want to receive (or compute if needed) the sampling shares, by default, this includes all parties + * @param {number} [Zp=jiff-instance.Zp] - the mod (if null then the default Zp for the instance is used) + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly + * @returns {SecretShare[]} an array of secret shares, each representing a bit from the sampled value (from least to most significant) + */ + rejection_sampling: function (jiff, lower_bound, upper_bound, threshold, parties, Zp, op_id) { + // defaults + if (parties == null) { + parties = []; + for (var i = 1; i <= jiff.party_count; i++) { + parties.push(i); + } + } else { + jiff.helpers.sort_ids(parties); + } + + if (op_id == null) { + op_id = jiff.counters.gen_op_id('rejection_sampling', parties); + } + + // try to get preprocessed samples + var result = jiff.get_preprocessing(op_id); + if (result != null && result.ondemand !== true) { + return result; + } + + // Not ready, either preprocess it on demand, or use crypto provider! + lower_bound = lower_bound ? lower_bound : 0; + upper_bound = upper_bound ? upper_bound : jiff.Zp; + if (threshold == null) { + threshold = parties.length; + } + if (Zp == null) { + Zp = jiff.Zp; + } + + var finalLength = jiff.helpers.ceil(jiff.helpers.bLog(upper_bound, 2)); + finalLength = parseInt(finalLength.toString(), 10); + finalLength = Math.max(finalLength, 1); // special case: when upper_bound is 1! + + var many_shares = jiff.utils.many_secret_shares(finalLength, parties, threshold, Zp); + var final_deferreds = many_shares.deferreds; + + // Crypto provider + if (result == null) { + var promise = jiff.from_crypto_provider('numbers', parties, threshold, Zp, op_id, { + max: upper_bound, + min: lower_bound, + bitLength: finalLength, + count: 1 + }); + promise.then(function (result) { + jiff.utils.resolve_many_secrets(final_deferreds, result['shares']); + }); + } else { + // preprocess on demand + delete jiff.preprocessing_table[op_id]; + jiff.preprocessing('rejection_sampling', 1, null, threshold, parties, parties, Zp, [op_id], { + lower_bound: lower_bound, + upper_bound: upper_bound + }); + jiff.executePreprocessing(function () { + jiff.utils.resolve_many_secrets(final_deferreds, jiff.get_preprocessing(op_id)); + }); + } + + return many_shares.shares; + }, + + /** + * Creates a secret share of the number represented by the given array of secret shared bits. + * Requires no communication, only local operations + * @function bit_composition + * @ignore + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits. + * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of the number represented by bits. + */ + bit_composition: function (bits) { + var result = bits[0]; + var pow = 1; + for (var i = 1; i < bits.length; i++) { + pow = pow * 2; + result = result.isadd(bits[i].icmult(pow)); + } + return result; + } + }; + }, + {} + ], + 28: [ + function (require, module, exports) { + module.exports = { + /** + * Share a number as an array of secret bits + * This takes the same parameters as jiff-instance.share, but returns an array of secret bit shares per sending party. + * Each bit array starts with the least significant bit at index 0, and most significant bit at index length-1 + * @function share_bits + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {number} secret - the number to share (this party's input) + * @param {number} [bit_length=jiff_instance.Zp] - the number of generated bits, if the secret has less bits, it will be + * padded with zeros + * @param {number} [threshold=receivers_list.length] - threshold of each shared bit + * @param {Array} [receivers_list=all_parties] - receivers of every bits + * @param {Array} [senders_list=all_parties] - senders of evey bit + * @param {number} [Zp=jiff_instance.Zp] - the field of sharing for every bit + * @param {string|number} [share_id=auto_gen()] - synchronization id + * @returns {object} a map (of size equal to the number of parties) + * where the key is the party id (from 1 to n) + * and the value is an array of secret shared bits + */ + share_bits: function (jiff, secret, bit_length, threshold, receivers_list, senders_list, Zp, share_id) { + var i; + if (Zp == null) { + Zp = jiff.Zp; + } + if (receivers_list == null) { + receivers_list = []; + for (i = 1; i <= jiff.party_count; i++) { + receivers_list.push(i); + } + } else { + jiff.helpers.sort_ids(receivers_list); + } + if (senders_list == null) { + senders_list = []; + for (i = 1; i <= jiff.party_count; i++) { + senders_list.push(i); + } + } else { + jiff.helpers.sort_ids(senders_list); + } + + if (share_id == null) { + share_id = jiff.counters.gen_op_id2('share_bits', receivers_list, senders_list); + } + + if (bit_length == null) { + bit_length = Zp.toString(2).length; + } + + // to allow for secret=null when party is not a sender + var local_bits = []; + if (secret != null) { + local_bits = jiff.helpers.number_to_bits(secret, bit_length); + } + + var shared_bits = {}; + for (i = 0; i < senders_list.length; i++) { + shared_bits[senders_list[i]] = []; + } + + for (i = 0; i < bit_length; i++) { + var round = jiff.internal_share(local_bits[i], threshold, receivers_list, senders_list, Zp, share_id + ':' + i); + for (var si = 0; si < senders_list.length; si++) { + var pid = senders_list[si]; + shared_bits[pid].push(round[pid]); + } + } + + return shared_bits; + }, + /** + * Opens the given array of secret shared bits. + * This works regardless of whether the represented value fit inside the corresponding field or not + * @function open_bits + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits + * @param {number[]} parties - parties to open (same as jiff_instance.open) + * @param {string|number} [op_id=auto_gen()] - same as jiff_instance.open + * @returns {promise} a promise to the number represented by bits + */ + open_bits: function (jiff, bits, parties, op_id) { + // Default values + if (parties == null) { + parties = []; + for (var p = 1; p <= jiff.party_count; p++) { + parties.push(p); + } + } else { + jiff.helpers.sort_ids(parties); + } + + // Compute operation ids (one for each party that will receive a result + if (op_id == null) { + op_id = jiff.counters.gen_op_id2('bits.open', parties, bits[0].holders); + } + + var opened_bits = []; + for (var i = 0; i < bits.length; i++) { + opened_bits[i] = jiff.internal_open(bits[i], parties, op_id + ':' + i); + } + + return Promise.all(opened_bits).then(function (bits) { + return jiff.helpers.bits_to_number(bits, bits.length); + }); + }, + /** + * Receives an opening of an array of secret bits without owning shares of the underlying value. + * Similar to jiff.receive_open() but for bits. + * This works regardless of whether the represented value fit inside the corresponding field or not + * @function receive_open_bits + * @memberOf jiff-instance.protocols.bits + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {Array} senders - an array with party ids (1 to n) specifying the parties sending the shares + * @param {Array} [receivers=all_parties] - an array with party ids (1 to n) specifying the parties receiving the result + * @param {number} [count=ceil(log2(Zp))] - the number of bits being opened + * @param {number} [threshold=parties.length] - the min number of parties needed to reconstruct the secret, defaults to all the senders + * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used) + * @param {string|number|object} [op_id=auto_gen()] - unique and consistent synchronization id between all parties + * @returns {promise} a (JQuery) promise to the open value of the secret + */ + receive_open_bits: function (jiff, senders, receivers, count, threshold, Zp, op_id) { + if (senders == null) { + throw new Error('Must provide "senders" parameter in receive_open'); + } + jiff.helpers.sort_ids(senders); + // Default values + if (receivers == null) { + receivers = []; + for (i = 1; i <= jiff.party_count; i++) { + receivers.push(i); + } + } else { + jiff.helpers.sort_ids(receivers); + } + + if (op_id == null) { + op_id = jiff.counters.gen_op_id2('bits.open', receivers, senders); + } + + if (count == null) { + if (Zp == null) { + Zp = jiff.Zp; + } + count = Zp.toString(2).length; + } + + var opened_bits = []; + for (var i = 0; i < count; i++) { + opened_bits[i] = jiff.receive_open(senders, receivers, threshold, Zp, op_id + ':' + i); + } + + return Promise.all(opened_bits).then(function (bits) { + return jiff.helpers.bits_to_number(bits, bits.length); + }); + } + }; + }, + {} + ], + 29: [ + function (require, module, exports) { + // Generic version of operations + module.exports = function (SecretShare) { + /** + * bitwise-XOR with a constant (BOTH BITS). + * @method cxor_bit + * @param {number} cst - the constant bit to XOR with (0 or 1). + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.cxor_bit = function (cst) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (^)'); + } + if (!this.jiff.share_helpers['binary'](cst)) { + throw new Error('parameter should be binary (^)'); + } + + return this.icadd(cst).issub(this.icmult(cst).icmult(2)); + }; + + /** + * bitwise-OR with a constant (BOTH BITS). + * @method cor_bit + * @param {number} cst - the constant bit to OR with (0 or 1). + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.cor_bit = function (cst) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (|)'); + } + if (!this.jiff.share_helpers['binary'](cst)) { + throw new Error('parameter should be binary (|)'); + } + + return this.icadd(cst).issub(this.icmult(cst)); + }; + + /** + * bitwise-XOR of two secret shares OF BITS. + * @method sxor_bit + * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to XOR with. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this < o, and 0 otherwise. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.sxor_bit = function (o, op_id) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (^)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (^)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (^)'); + } + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('sxor_bit', this.holders); + } + + return this.isadd(o).issub(this.ismult(o, op_id + ':smult1').icmult(2)); + }; + + /** + * OR of two secret shares OF BITS. + * @method sor_bit + * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to OR with. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this < o, and 0 otherwise. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.sor_bit = function (o, op_id) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (|)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (|)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (|)'); + } + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('sor_bit', this.holders); + } + + return this.isadd(o).issub(this.ismult(o, op_id + ':smult1')); + }; + + /** + * Negation of a bit. + * This has to be a share of a BIT in order for this to work properly. + * @method not + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result (negated bit). + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.not = function () { + return this.icmult(-1).icadd(1); + }; + + /** + * Simulate an oblivious If-else statement with a single return value. + * Should be called on a secret share of a bit: 0 representing false, and 1 representing true + * If this is a share of 1, a new sharing of the element represented by the first parameter is returned, + * otherwise, a new sharing of the second is returned. + * @method if_else + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + * @param {module:jiff-client~JIFFClient#SecretShare | number} trueVal - the value/share to return if this is a sharing of 1. + * @param {module:jiff-client~JIFFClient#SecretShare | number} falseVal - the value/share to return if this is a sharing of 0. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} a new sharing of the result of the if. + * + * @example + * // a and b are secret shares + * // cmp will be a secret share of either 1 or 0, depending on whether a or b is greater + * var cmp = a.gt(b); + * + * // max is set to the greater value, without revealing the value or the result of the inequality + * var max = cmp.if_else(a, b); + */ + SecretShare.prototype.if_else = function (trueVal, falseVal, op_id) { + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('if_else', this.holders); + } + + var const1 = this.isConstant(trueVal); + var const2 = this.isConstant(falseVal); + if (const1 && const2) { + return this.icmult(trueVal).isadd(this.inot().icmult(falseVal)); + } else if (const1) { + return this.inot() + .ismult(falseVal.icsub(trueVal), op_id + ':smult') + .icadd(trueVal); + } else if (const2) { + return this.ismult(trueVal.icsub(falseVal), op_id + ':smult').icadd(falseVal); + } else { + return this.ismult(trueVal.issub(falseVal), op_id + ':smult').isadd(falseVal); + } + }; + }; + }, + {} + ], + 30: [ + function (require, module, exports) { + // Generic version of operations + module.exports = function (SecretShare) { + /** + * Checks if the given parameter is a constant, used to determine whether constant or secret + * operations should be executed when the generic version of an operation is called + * @method isConstant + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + * @param {number/object} o - the parameter to determine. + * @return {boolean} true if o is a valid constant, false otherwise. + */ + SecretShare.prototype.isConstant = function (o) { + return typeof o === 'number'; + }; + + /** + * Generic Addition. + * Uses either the constant or secret version of this operator depending on type of paramter. + * @method add + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * + * @example + * var shares = jiffClient.share(input); + * // this will add two secret shared values together + * var sum1 = shares[1].add(shares[2]); + * // this will add 3 to the secret input from party 1 + * var sum2 = shares[1].add(3); + * // both sum1 and sum2 are SecretShares + */ + SecretShare.prototype.add = function (o) { + if (this.isConstant(o)) { + return this.cadd(o); + } + return this.sadd(o); + }; + + /** + * Generic Subtraction. + * Uses either the constant or secret version of this operator depending on type of paramter. + * @method sub + * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.sub = function (o) { + if (this.isConstant(o)) { + return this.csub(o); + } + return this.ssub(o); + }; + + /** + * Generic Multiplication. + * Uses either the constant or secret version of this operator depending on type of paramter. + * @method mult + * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this multiplication (and internally, the corresponding beaver triplet). + * This id must be unique, and must be passed by all parties to the same instruction. + * this ensures that every party gets a share from the same triplet for every matching instruction. An automatic id + * is generated by increasing a local counter, default ids suffice when all parties execute the + * instructions in the same order. Only used if secret multiplication is used. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.mult = function (o, op_id) { + if (this.isConstant(o)) { + return this.cmult(o); + } + return this.smult(o, op_id); + }; + + /** + * Generic XOR for bits (both this and o have to be bits to work correctly). + * Uses either the constant or secret version of this operator depending on type of paramter. + * @method xor_bit + * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * Only used if secret xor is used.. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.xor_bit = function (o, op_id) { + if (this.isConstant(o)) { + return this.cxor_bit(o); + } + return this.sxor_bit(o, op_id); + }; + + /** + * Generic OR for bits (both this and o have to be bits to work correctly). + * Uses either the constant or secret version of this operator depending on type of paramter. + * @method or_bit + * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * Only used if secret or is used.. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.or_bit = function (o, op_id) { + if (this.isConstant(o)) { + return this.cor_bit(o); + } + return this.sor_bit(o, op_id); + }; + + /** + * Generic Greater or equal. + * Uses either the constant or secret version of this operator depending on type of paramter. + * @method gteq + * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.gteq = function (o, op_id) { + if (this.isConstant(o)) { + return this.cgteq(o, op_id); + } + return this.sgteq(o); + }; + + /** + * Generic Greater than. + * Uses either the constant or secret version of this operator depending on type of paramter. + * @method gt + * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.gt = function (o, op_id) { + if (this.isConstant(o)) { + return this.cgt(o, op_id); + } + return this.sgt(o, op_id); + }; + + /** + * Generic Less or equal. + * Uses either the constant or secret version of this operator depending on type of paramter. + * @method lteq + * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.lteq = function (o, op_id) { + if (this.isConstant(o)) { + return this.clteq(o, op_id); + } + return this.slteq(o, op_id); + }; + + /** + * Generic Less than. + * Uses either the constant or secret version of this operator depending on type of paramter. + * @method lt + * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.lt = function (o, op_id) { + if (this.isConstant(o)) { + return this.clt(o, op_id); + } + return this.slt(o, op_id); + }; + + /** + * Generic Equals. + * Uses either the constant or secret version of this operator depending on type of paramter. + * @method eq + * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.eq = function (o, op_id) { + if (this.isConstant(o)) { + return this.ceq(o, op_id); + } + return this.seq(o, op_id); + }; + + /** + * Generic Not Equals. + * Uses either the constant or secret version of this operator depending on type of paramter. + * @method neq + * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.neq = function (o, op_id) { + if (this.isConstant(o)) { + return this.cneq(o, op_id); + } + return this.sneq(o, op_id); + }; + + /** + * Generic Integer Divison. + * Uses either the constant or secret version of this operator depending on type of paramter. + * @method div + * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). + * @param {number} l - the maximum bit length of the two shares. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.div = function (o, l, op_id) { + if (this.isConstant(o)) { + return this.cdiv(o, l, op_id); + } + return this.sdiv(o, l, op_id); + }; + }; + }, + {} + ], + 31: [ + function (require, module, exports) { + // Arithmetic operations on shares + module.exports = function (SecretShare) { + /** + * Addition with a constant. + * @method cadd + * @param {number} cst - the constant to add. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.cadd = function (cst) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (+)'); + } + + var self = this; + var ready = function () { + return self.jiff.helpers.mod(self.jiff.share_helpers['+'](self.value, cst), self.Zp); + }; + + return new this.jiff.SecretShare(this.wThen(ready), this.holders, this.threshold, this.Zp); + }; + + /** + * Subtraction with a constant. + * @method csub + * @param {number} cst - the constant to subtract from this share. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.csub = function (cst) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (-)'); + } + + var self = this; + var ready = function () { + return self.jiff.helpers.mod(self.jiff.share_helpers['-'](self.value, cst), self.Zp); + }; + + return new this.jiff.SecretShare(this.wThen(ready), this.holders, this.threshold, this.Zp); + }; + + /** + * Multiplication by a constant. + * @method cmult + * @param {number} cst - the constant to multiply to this share. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.cmult = function (cst) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (*)'); + } + + var self = this; + var ready = function () { + return self.jiff.helpers.mod(self.jiff.share_helpers['*'](self.value, cst), self.Zp); + }; + + return new this.jiff.SecretShare(this.wThen(ready), this.holders, this.threshold, this.Zp); + }; + + /** + * Division by a constant factor of the number represented by the share. + * @method cdivfac + * @param {number} cst - the constant by which to divide the share. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.cdivfac = function (cst) { + if (!this.isConstant(cst)) { + throw new Error('Parameter should be a number (cdivfac)'); + } + + var inv = this.jiff.helpers.extended_gcd(cst, this.Zp)[0]; + + var self = this; + var ready = function () { + return self.jiff.helpers.mod(self.jiff.share_helpers['*'](self.value, inv), self.Zp); + }; + + return new this.jiff.SecretShare(this.wThen(ready), this.holders, this.threshold, this.Zp); + }; + + /** + * Addition of two secret shares. + * @method sadd + * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to add to this share. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + * + * @example + * // share a value with all parties, and sum the values of all shares + * var shares = jiff_instance.share(x); + * var sum = shares[1]; + * for (var i = 2; i <= jiff_instance.party_count; i++) { + * sum = sum.sadd(shares[i]); + * } + * + */ + SecretShare.prototype.sadd = function (o) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (+)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (+)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (+)'); + } + + // add the two shares when ready locally + var self = this; + var ready = function () { + return self.jiff.helpers.mod(self.jiff.share_helpers['+'](self.value, o.value), self.Zp); + }; + + // promise to execute ready_add when both are ready + return new this.jiff.SecretShare(this.when_both_ready(o, ready), this.holders, Math.max(this.threshold, o.threshold), this.Zp); + }; + + /** + * Subtraction of two secret shares. + * @method ssub + * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to subtract from this share. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.ssub = function (o) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (-)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (-)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (-)'); + } + + // subtract the two shares when ready locally + var self = this; + var ready = function () { + return self.jiff.helpers.mod(self.jiff.share_helpers['-'](self.value, o.value), self.Zp); + }; + + // promise to execute ready_add when both are ready + return new this.jiff.SecretShare(this.when_both_ready(o, ready), this.holders, Math.max(this.threshold, o.threshold), this.Zp); + }; + + /** + * Multiplication of two secret shares through Beaver Triplets. + * @method smult + * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to multiply with. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this multiplication (and internally, the corresponding beaver triplet). + * This id must be unique, and must be passed by all parties to the same instruction. + * this ensures that every party gets a share from the same triplet for every matching instruction. An automatic id + * is generated by increasing a local counter, default ids suffice when all parties execute the + * instructions in the same order. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.smult = function (o, op_id) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (*)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (*)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (*)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('smult', this.holders); + } + + // final result + var final_deferred = new this.jiff.helpers.Deferred(); + var final_promise = final_deferred.promise; + var result = new this.jiff.SecretShare(final_promise, this.holders, Math.max(this.threshold, o.threshold), this.Zp); + + // called when triplet is ready + var self = this; + var ready_triplet = function (triplet) { + var a = triplet[0]; + var b = triplet[1]; + var c = triplet[2]; + + // d = s - a. e = o - b. + var d = self.isadd(a.icmult(-1)); + var e = o.isadd(b.icmult(-1)); + + // Open d and e. + // The only communication cost. + var e_promise = self.jiff.internal_open(e, e.holders, op_id + ':open1'); + var d_promise = self.jiff.internal_open(d, d.holders, op_id + ':open2'); + Promise.all([e_promise, d_promise]).then(function (arr) { + var e_open = arr[0]; + var d_open = arr[1]; + + // result_share = d_open * e_open + d_open * b_share + e_open * a_share + c. + var t1 = self.jiff.helpers.mod(self.jiff.share_helpers['*'](d_open, e_open), self.Zp); + var t2 = b.icmult(d_open); + var t3 = a.icmult(e_open); + + // All this happens locally. + var final_result = t2.icadd(t1); + final_result = final_result.isadd(t3); + final_result = final_result.isadd(c); + + final_result.wThen(final_deferred.resolve); + }); + }; + + // Get shares of triplets. + var triplet = this.jiff.get_preprocessing(op_id + ':triplet'); + if (triplet == null) { + var promise = this.jiff.from_crypto_provider('triplet', this.holders, Math.max(this.threshold, o.threshold), this.Zp, op_id + ':triplet'); + promise.then(function (msg) { + ready_triplet(msg['shares']); + }); + } else { + ready_triplet(triplet); + } + + return result; + }; + + /** + * Multiplication of two secret shares through BGW protocol. + * @method smult_bgw + * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to multiply with. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this multiplication (and internally, the corresponding beaver triplet). + * This id must be unique, and must be passed by all parties to the same instruction. + * this ensures that every party gets a share from the same triplet for every matching instruction. An automatic id + * is generated by increasing a local counter, default ids suffice when all parties execute the + * instructions in the same order. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.smult_bgw = function (o, op_id) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (bgw*)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (bgw*)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (bgw*)'); + } + if (this.threshold - 1 + (o.threshold - 1) > this.holders.length - 1) { + throw new Error('threshold too high for BGW (*)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('smult_bgw', this.holders); + } + + // ensure thresholds are fine + var new_threshold = this.threshold - 1 + (o.threshold - 1) + 1; + if (new_threshold > this.holders) { + var errorMsg = 'Threshold too large for smult_bgw: ' + new_threshold; + errorMsg += '. Shares: ' + this.toString() + ', ' + o.toString(); + throw new Error(errorMsg); + } + + // multiply via the BGW protocol + var self = this; + var ready = function () { + return self.jiff.helpers.mod(self.jiff.share_helpers['*'](self.value, o.value), self.Zp); + }; + + // reshare to reduce threshold and return when ready + var result = new this.jiff.SecretShare(this.when_both_ready(o, ready), this.holders, new_threshold, this.Zp); + return this.jiff.reshare(result, Math.max(this.threshold, o.threshold), result.holders, result.holders, result.Zp, op_id + ':threshold'); + }; + + /** + * Integer divison with two shares (this / o) + * @method sdiv + * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to divide by. + * @param {number} [l=ceil(log_2(this.Zp))] - the maximum bit length of either operands. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.sdiv = function (o, l, op_id) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (!=)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (!=)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (!=)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('sdiv', this.holders); + } + + // figure out maximum output bit length + var lZp = this.jiff.share_helpers['ceil'](this.jiff.helpers.bLog(this.Zp, 2)); + l = l != null && l < lZp ? l : lZp; + + // Convert to bits + var dividend_bits = this.bit_decomposition(op_id + ':decomposition1').slice(0, l); + var divisor_bits = o.bit_decomposition(op_id + ':decomposition2').slice(0, l); + + // Compute by long division + var quotient_bits = this.jiff.protocols.bits.sdiv(dividend_bits, divisor_bits, op_id + ':bits.sdiv').quotient; + + // Convert to number and return + return this.jiff.protocols.bits.bit_composition(quotient_bits); + }; + + /** + * Integer divison with a share and a constant (this / cst). + * @method cdiv + * @param {module:jiff-client~JIFFClient#SecretShare} cst - the constant to divide by. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.cdiv = function (cst, op_id) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (/)'); + } + + if (this.jiff.share_helpers['<='](cst, 0)) { + throw new Error('divisor must be > 0 (cst/): ' + cst); + } + + if (this.jiff.share_helpers['<='](this.Zp, cst)) { + throw new Error('divisor must be < share.Zp (' + this.Zp + ') in (cst/): ' + cst); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('cdiv', this.holders); + } + + // Allocate share for result to which the answer will be resolved once available + var final_deferred = new this.jiff.helpers.Deferred(); + var final_promise = final_deferred.promise; + var result = new this.jiff.SecretShare(final_promise, this.holders, this.threshold, this.Zp); + + // Execute protocol when random in noise in [0, Zp) and quotient floor(noise/constant) is ready! + var self = this; + var ready_quotient = function (noise, nOVERc) { + // Use noise + var noisyX = self.isadd(noise); + self.jiff.internal_open(noisyX, noisyX.holders, op_id + ':open').then(function (noisyX) { + var wrapped = self.icgt(noisyX, op_id + ':wrap_cgt'); // 1 => x + noise wrapped around Zp, 0 otherwise + + // if we did not wrap + var noWrapDiv = self.jiff.share_helpers['floor/'](noisyX, cst); + var unCorrectedQuotient = nOVERc.icmult(-1).icadd(noWrapDiv).icsub(1); + var verify = self.issub(unCorrectedQuotient.icmult(cst)); + var isNotCorrect = verify.icgteq(cst, op_id + ':cor1'); + var noWrapAnswer = unCorrectedQuotient.isadd(isNotCorrect); // if incorrect => isNotCorrect = 1 => quotient = unCorrectedQuotient - 1 + + // if we wrapped + var wrapDiv = self.jiff.share_helpers['floor/'](self.jiff.share_helpers['+'](noisyX, self.Zp), cst); + unCorrectedQuotient = nOVERc.icmult(-1).icadd(wrapDiv).icsub(1); + verify = self.issub(unCorrectedQuotient.icmult(cst)); + isNotCorrect = verify.icgteq(cst, op_id + ':cor2'); + var wrapAnswer = unCorrectedQuotient.isadd(isNotCorrect); // if incorrect => isNotCorrect = 1 => quotient = unCorrectedQuotient - 1 + + var answer = noWrapAnswer.isadd(wrapped.ismult(wrapAnswer.issub(noWrapAnswer), op_id + ':smult')); + answer.wThen(final_deferred.resolve); + }); + }; + + // Preprocessing cases + var quotient = this.jiff.get_preprocessing(op_id + ':quotient'); + if (quotient == null) { + // case 1: no preprocessing with crypto provider! + var promise = this.jiff.from_crypto_provider('quotient', this.holders, this.threshold, this.Zp, op_id + ':quotient', { constant: cst }); + promise.then(function (msg) { + ready_quotient(msg['shares'][0], msg['shares'][1]); + }); + } else if (quotient.ondemand === true) { + // case 2: constant was not available at preprocessing time, must do it now! + this.jiff.preprocessing('quotient', 1, null, this.threshold, this.holders, this.holders, this.Zp, [op_id + ':quotient'], { constant: cst, namespace: 'base' }); + this.jiff.executePreprocessing(function () { + var quotient = self.jiff.get_preprocessing(op_id + ':quotient'); + ready_quotient(quotient.r, quotient.q); + }); + } else { + // case 3: preprocessing is completed! + ready_quotient(quotient.r, quotient.q); + } + + // special case, if result is zero, sometimes we will get to -1 due to how correction happens above (.csub(1) and then compare) + var zeroIt = this.iclt(cst, op_id + ':zero_check').inot(); + return result.ismult(zeroIt, op_id + ':zero_it'); + }; + + /** + * Remainder with two shares (this % o) + * @method smod + * @param {module:jiff-client~JIFFClient#SecretShare} o - the modulus to apply + * @param {number} [l=ceil(log_2(this.Zp))] - the maximum bit length of either operands. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.smod = function (o, l, op_id) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (!=)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (!=)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (!=)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('smod', this.holders); + } + + // figure out maximum output bit length + var lZp = this.jiff.share_helpers['ceil'](this.jiff.helpers.bLog(this.Zp, 2)); + l = l != null && l < lZp ? l : lZp; + + // Convert to bits + var dividend_bits = this.bit_decomposition(op_id + ':decomposition1').slice(0, l); + var divisor_bits = o.bit_decomposition(op_id + ':decomposition2').slice(0, l); + + // Compute by long division + var remainder_bits = this.jiff.protocols.bits.sdiv(dividend_bits, divisor_bits, op_id + ':bits.sdiv').remainder; + + // Convert to number and return + return this.jiff.protocols.bits.bit_composition(remainder_bits); + }; + + /** + * Fast (modular) exponentiation with constant exponent via repeated squaring. + * @method cpow + * @param {number} cst - the constant to multiply to this share. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.cpow = function (cst, op_id) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (*)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('cpow', this.holders); + } + + // handle big numbers + var one = 1; + if (this.jiff.has_extension('bignumber')) { + one = this.jiff.helpers.BigNumber(1); + cst = this.jiff.helpers.BigNumber(cst); + } + + // ensure exponent is non-negative + if (this.jiff.share_helpers['<'](cst, 0)) { + throw new Error('cpow supports non-negative exponents only, given ' + cst.toString()); + } + + // begin protocol + var evens = this; + var odds = new this.jiff.SecretShare(one, this.holders, this.threshold, this.Zp); + + // special case + if (cst.toString() === '0') { + return odds; + } + + for (var i = 0; this.jiff.share_helpers['<'](1, cst); i++) { + if (this.jiff.share_helpers['even'](cst)) { + evens = evens.ismult(evens, op_id + ':smult0:' + i); + cst = this.jiff.share_helpers['/'](cst, 2); + } else { + odds = evens.ismult(odds, op_id + ':smult0:' + i); + evens = evens.ismult(evens, op_id + ':smult1:' + i); + cst = this.jiff.share_helpers['/'](this.jiff.share_helpers['-'](cst, 1), 2); + } + } + + return evens.ismult(odds, op_id + ':smult0:' + i); + }; + }; + }, + {} + ], + 32: [ + function (require, module, exports) { + // Comparison operations on shares + module.exports = function (SecretShare) { + /** + * Greater than or equal with another share. + * @method sgteq + * @param {module:jiff-client~JIFFClient#SecretShare} o - the other share. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this >= o, and 0 otherwise. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.sgteq = function (o, op_id) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (>=)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (>=)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (>=)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('sgteq', this.holders); + } + + return this.islt(o, op_id).inot(); + }; + + /** + * Greater than with another share. + * @method sgt + * @param {module:jiff-client~JIFFClient#SecretShare} o - the other share. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this > o, and 0 otherwise. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.sgt = function (o, op_id) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (>)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (>)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (>)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('sgt', this.holders); + } + + return o.islt(this, op_id); + }; + + /** + * Less than or equal with another share. + * @method slteq + * @param {module:jiff-client~JIFFClient#SecretShare} o - the other share. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this <= o, and 0 otherwise. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.slteq = function (o, op_id) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (<=)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (<=)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (<=)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('slteq', this.holders); + } + + return o.islt(this, op_id).inot(); + }; + + /** + * Less than with another share. + * @method slt + * @param {module:jiff-client~JIFFClient#SecretShare} o - the other share. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this < o, and 0 otherwise. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.slt = function (o, op_id) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (<)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (<)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (<)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('slt', this.holders); + } + + var final_deferred = new this.jiff.helpers.Deferred(); + var final_promise = final_deferred.promise; + var result = new this.jiff.SecretShare(final_promise, this.holders, Math.max(this.threshold, o.threshold), this.Zp); + + var w = this.ilt_halfprime(op_id + ':halfprime:1'); + + var self = this; + w.wThen(function () { + var x = o.ilt_halfprime(op_id + ':halfprime:2'); + x.wThen(function () { + var y = self.issub(o).ilt_halfprime(op_id + ':halfprime:3'); + y.wThen(function () { + var xy = x.ismult(y, op_id + ':smult1'); + var answer = x + .icmult(-1) + .icadd(1) + .issub(y) + .isadd(xy) + .isadd(w.ismult(x.isadd(y).issub(xy.icmult(2)), op_id + ':smult2')); + answer.wThen(final_deferred.resolve); + }); + }); + }); + + return result; + }; + + /** + * Greater than or equal with a constant. + * @method cgteqn + * @param {number} cst - the constant to compare with. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this >= cst, and 0 otherwise. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.cgteq = function (cst, op_id) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (>=)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('cgteq', this.holders); + } + + return this.iclt(cst, op_id).inot(); + }; + + /** + * Greater than with a constant. + * @method cgt + * @param {number} cst - the constant to compare with. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly.default ids suffice when all parties execute the + * instructions in the same order. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this > cst, and 0 otherwise. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.cgt = function (cst, op_id) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (>)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('cgt', this.holders); + } + + var final_deferred = new this.jiff.helpers.Deferred(); + var final_promise = final_deferred.promise; + var result = new this.jiff.SecretShare(final_promise, this.holders, this.threshold, this.Zp); + + var w = this.jiff.share_helpers['<'](cst, this.jiff.share_helpers['/'](this.Zp, 2)) ? 1 : 0; + var x = this.ilt_halfprime(op_id + ':halfprime:1'); + + var self = this; + x.wThen(function () { + var y = self + .icmult(-1) + .icadd(cst) + .ilt_halfprime(op_id + ':halfprime:2'); + y.wThen(function () { + var xy = y.ismult(x, op_id + ':smult1'); + var answer = x + .icmult(-1) + .icadd(1) + .issub(y) + .isadd(xy) + .isadd(x.isadd(y).issub(xy.icmult(2)).icmult(w)); + answer.wThen(final_deferred.resolve); + }); + }); + + return result; + }; + + /** + * Less than or equal with a constant. + * @method clteq + * @param {number} cst - the constant to compare with. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this <= cst, and 0 otherwise. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.clteq = function (cst, op_id) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (<=)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('clteq', this.holders); + } + + return this.icgt(cst, op_id).inot(); + }; + + /** + * Less than with a constant. + * @method clt + * @param {number} cst - the constant to compare with. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this < cst, and 0 otherwise. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.clt = function (cst, op_id) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (<)'); + } + + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('clt', this.holders); + } + + var final_deferred = new this.jiff.helpers.Deferred(); + var final_promise = final_deferred.promise; + var result = new this.jiff.SecretShare(final_promise, this.holders, this.threshold, this.Zp); + + var w = this.ilt_halfprime(op_id + ':halfprime:1'); + + var self = this; + w.wThen(function () { + var x = self.jiff.share_helpers['<'](cst, self.jiff.share_helpers['/'](self.Zp, 2)) ? 1 : 0; + var y = self.icsub(cst).ilt_halfprime(op_id + ':halfprime:2'); + y.wThen(function () { + var xy = y.icmult(x); + var answer = y + .icmult(-1) + .icadd(1 - x) + .isadd(xy) + .isadd(w.ismult(y.icadd(x).issub(xy.icmult(2)), op_id + ':smult1')); + answer.wThen(final_deferred.resolve); + }); + }); + + return result; + }; + + /** + * Equality test with two shares. + * @method seq + * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to compare with. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this = o, and 0 otherwise. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.seq = function (o, op_id) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (==)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (==)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (==)'); + } + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('seq', this.holders); + } + + return this.isneq(o, op_id).inot(); + }; + + /** + * Unequality test with two shares. + * @method sneq + * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to compare with. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 0 if this = o, and 1 otherwise. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.sneq = function (o, op_id) { + if (!(o.jiff === this.jiff)) { + throw new Error('shares do not belong to the same instance (!=)'); + } + if (!this.jiff.helpers.Zp_equals(this, o)) { + throw new Error('shares must belong to the same field (!=)'); + } + if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { + throw new Error('shares must be held by the same parties (!=)'); + } + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('sneq', this.holders); + } + + return this.issub(o).icpow(this.jiff.share_helpers['-'](this.Zp, 1), op_id + ':cpow'); + }; + + /** + * Equality test with a constant. + * @method ceq + * @param {number} cst - the constant to compare with. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 0 if this = o, and 1 otherwise. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.ceq = function (cst, op_id) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (==)'); + } + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('ceq', this.holders); + } + + return this.icneq(cst, op_id).inot(); + }; + + /** + * Unequality test with a constant. + * @method cneq + * @param {number} cst - the constant to compare with. + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 0 if this = o, and 1 otherwise. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.cneq = function (cst, op_id) { + if (!this.isConstant(cst)) { + throw new Error('parameter should be a number (!=)'); + } + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('cneq', this.holders); + } + + return this.icsub(cst).icpow(this.jiff.share_helpers['-'](this.Zp, 1), op_id + ':cpow'); + }; + + /** + * Checks whether the share is less than half the field size. + * @method lt_halfprime + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. + */ + SecretShare.prototype.lt_halfprime = function (op_id) { + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('lt_halfprime', this.holders); + } + + // if share is even, then this is less than half the prime, otherwise, share is greater than half the prime + var share = this.icmult(2); + + // to check if share is even, we will use pre-shared bits as some form of a bit mask + var bitLength = this.jiff.share_helpers['ceil'](this.jiff.helpers.bLog(share.Zp, 2)); + + // Create result share + var final_deferred = new this.jiff.helpers.Deferred(); + var final_promise = final_deferred.promise; + var result = new this.jiff.SecretShare(final_promise, this.holders, this.threshold, this.Zp); + + // Execute protocol when randomly sampled bit-wise random number is ready + var self = this; + var ready_sampling = function (bits) { + // if 2*this is even, then this is less than half prime, otherwise this is greater or equal to half prime + if (bits.length !== bitLength) { + throw new Error('Preprocessed bits sequence has incorrect length, expected: ' + bitLength + ' actual: ' + bits.length); + } + + // bit composition: r = (rl ... r1 r0)_10 + var r = self.jiff.protocols.bits.bit_composition(bits); + // open share + noise, and utilize opened value with shared bit representation of noise to check the least significant digit of share. + share.jiff.internal_open(r.isadd(share), share.holders, op_id + ':open').then(function (result) { + var wrapped = self.jiff.protocols.bits.cgt(bits, result, op_id + ':bits.cgt'); + var isOdd = self.jiff.helpers.mod(result, 2); + isOdd = bits[0].icxor_bit(isOdd); + isOdd = isOdd.isxor_bit(wrapped, op_id + ':sxor_bit'); + + var answer = isOdd.inot(); + answer.wThen(final_deferred.resolve); + }); + }; + + // generate the bits of a random number less than our prime + var bits = this.jiff.get_preprocessing(op_id + ':sampling'); + if (bits == null) { + var promise = this.jiff.from_crypto_provider('numbers', this.holders, this.threshold, this.Zp, op_id + ':sampling', { + bitLength: bitLength, + count: 1, + max: this.Zp + }); + promise.then(function (msg) { + ready_sampling(msg['shares']); + }); + } else { + ready_sampling(bits); + } + + return result; + }; + }; + }, + {} + ], + 33: [ + function (require, module, exports) { + // general arithmetic protocols + module.exports = function (SecretShare) { + /** + * Reshares/refreshes the sharing of this number, used before opening to keep the share secret. + * @method refresh + * @param {string} [op_id=auto_gen()] - the operation id with which to tag the messages sent by this refresh, by default + * an automatic operation id is generated by increasing a local counter, default operation ids + * suffice when all parties execute the instructions in the same order. + * @returns {module:jiff-client~JIFFClient#SecretShare} a new share of the same number. + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.refresh = function (op_id) { + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('refresh', this.holders); + } + + // final result + var final_deferred = new this.jiff.helpers.Deferred(); + var final_promise = final_deferred.promise; + var result = new this.jiff.SecretShare(final_promise, this.holders, this.threshold, this.Zp); + + // refresh + var self = this; + var ready_number = function (zero) { + self.isadd(zero).wThen(final_deferred.resolve); + }; + + // get shares of zero + var zero = this.jiff.get_preprocessing(op_id); + if (zero == null) { + var promise = this.jiff.from_crypto_provider('numbers', this.holders, this.threshold, this.Zp, op_id, { + number: 0, + count: 1 + }); + promise.then(function (msg) { + ready_number(msg['shares'][0]); + }); + } else { + ready_number(zero); + } + + return result; + }; + + /** + * Bit Decomposition: Transform existing share to an array of bit shares. + * @method bit_decomposition + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly. + * @returns {module:jiff-client~JIFFClient#SecretShare[]} an array of secret shares of bits of length [ceil(log_2(this.Zp))], where + * index 0 represents the least significant bit. + */ + SecretShare.prototype.bit_decomposition = function (op_id) { + if (op_id == null) { + op_id = this.jiff.counters.gen_op_id('bit_decomposition', this.holders); + } + + // bit length of this secret + var bitLength = this.jiff.share_helpers['ceil'](this.jiff.helpers.bLog(this.Zp, 2)); + + // Create deferred shares to resolve to later when the computation completes + var many_shares = this.jiff.utils.many_secret_shares(bitLength, this.holders, this.threshold, this.Zp); + var deferreds = many_shares.deferreds; + var result = many_shares.shares; + + // Execute protocol when randomly sampled bit-wise random number is ready + var self = this; + var ready_sampling = function (bits) { + var r = self.jiff.protocols.bits.bit_composition(bits); + // add and reveal random number to this + self.jiff.internal_open(r.isadd(self), self.holders, op_id + ':open').then(function (result) { + // compute bits assuming r+this < Zp + var noWrap = self.jiff.protocols.bits.csubr(result, bits, op_id + ':bits.csubr:1'); + var didWrap = noWrap.pop(); + + // compute bits assuming r+this >= Zp + var withWrap = self.jiff.protocols.bits.csubr(self.jiff.share_helpers['+'](result, self.Zp), bits, op_id + ':bits.csubr:2'); + withWrap.pop(); // withWrap cannot underflow! + + // choose noWrap if first subtraction does not overflow (sign bit is zero), otherwise choose withWrap. + for (var i = 0; i < bitLength; i++) { + withWrap[i] = didWrap.iif_else(withWrap[i], noWrap[i], op_id + ':if_else:' + i); + } + self.jiff.utils.resolve_many_secrets(deferreds, withWrap); + }); + }; + + // generate the bits of a random number less than our prime + var bits = this.jiff.get_preprocessing(op_id + ':sampling'); + if (bits == null) { + var promise = this.jiff.from_crypto_provider('numbers', this.holders, this.threshold, this.Zp, op_id + ':sampling', { + bitLength: bitLength, + count: 1, + max: this.Zp + }); + promise.then(function (msg) { + ready_sampling(msg['shares']); + }); + } else { + ready_sampling(bits); + } + + return result; + }; + }; + }, + {} + ], + 34: [ + function (require, module, exports) { + module.exports = { + /** + * Generates a random bit under MPC by xoring all bits sent by participating parties using smult / beaver triples + * @function generate_random_bit_bgw + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed + * @param {Array} receivers_list - array of party ids to receive the result + * @param {Array} compute_list - array of party ids to perform the protocol + * @param {number} Zp - the mod + * @param {object} params - an object containing extra parameters passed by the user + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * @return {Object} contains 'share' (this party's share of the generated bit) and 'promise' + */ + generate_random_bit_smult: function (jiff, threshold, receivers_list, compute_list, Zp, params) { + if (params.compute_threshold == null) { + params.compute_threshold = threshold; + } + + if (params.op_id == null && params.output_op_id == null) { + params.op_id = jiff.counters.gen_op_id2('generate_random_bit_smult', receivers_list, compute_list); + } else if (params.op_id == null) { + params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); + } + var op_id = params.op_id; + + // Generate random bit + var random_bit, promise; + if (compute_list.indexOf(jiff.id) > -1) { + var bit = jiff.helpers.random(2); + var bit_shares = jiff.internal_share(bit, params.compute_threshold, compute_list, compute_list, Zp, op_id + ':share'); + + random_bit = bit_shares[compute_list[0]]; + for (var i = 1; i < compute_list.length; i++) { + var party_id = compute_list[i]; + var obit = bit_shares[party_id]; + random_bit = random_bit.isxor_bit(obit, op_id + ':sxor_bit:' + i); + } + + promise = random_bit.value; + } + + // Reshare + random_bit = jiff.reshare(random_bit, threshold, receivers_list, compute_list, Zp, op_id + ':reshare'); + if (receivers_list.indexOf(jiff.id) > -1) { + promise = random_bit.value; + } + return { share: random_bit, promise: promise }; + }, + /** + * Generates a random bit under MPC by xoring all bits sent by participating parties using smult_bgw + * @function generate_random_bit_bgw + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed + * @param {Array} receivers_list - array of party ids to receive the result + * @param {Array} compute_list - array of party ids to perform the protocol + * @param {number} Zp - the mod + * @param {object} params - an object containing extra parameters passed by the user + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * @return {Object} contains 'share' (this party's share of the generated bit) and 'promise' + */ + generate_random_bit_bgw: function (jiff, threshold, receivers_list, compute_list, Zp, params) { + if (params.compute_threshold == null) { + params.compute_threshold = Math.floor((compute_list.length + 1) / 2); // honest majority BGW + } + + if (params.op_id == null && params.output_op_id == null) { + params.op_id = jiff.counters.gen_op_id2('generate_random_bit_bgw', receivers_list, compute_list); + } else if (params.op_id == null) { + params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); + } + + var op_id = params.op_id; + + // Generate random bit + var random_bit, promise; + if (compute_list.indexOf(jiff.id) > -1) { + var bit = jiff.helpers.random(2); + var bit_shares = jiff.internal_share(bit, params.compute_threshold, compute_list, compute_list, Zp, op_id + ':share'); + + random_bit = bit_shares[compute_list[0]]; + for (var i = 1; i < compute_list.length; i++) { + var party_id = compute_list[i]; + var obit = bit_shares[party_id]; + random_bit = random_bit.isadd(obit).issub(random_bit.ismult_bgw(obit, op_id + ':smult' + i).icmult(2)); + } + + promise = random_bit.value; + } + + // Reshare + random_bit = jiff.reshare(random_bit, threshold, receivers_list, compute_list, Zp, op_id + ':reshare'); + if (receivers_list.indexOf(jiff.id) > -1) { + promise = random_bit.value; + } + return { share: random_bit, promise: promise }; + }, + /** + * Generates a sequence of random bits under MPC + * @function generate_random_bits + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {number} [threshold=receivers_list.length] - the threshold of the bit when stored by receivers after generation + * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed + * @param {Array} receivers_list - array of party ids to receive the result + * @param {Array} compute_list - array of party ids to perform the protocol + * @param {number} Zp - the mod + * @param {object} params - an object containing extra parameters passed by the user + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated + * - count: how many random bits to generate + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * @param {object} protocols - the protocols to use for preprocessing + * @return {Object} contains 'share' (array of secret shares bits) and 'promise' + */ + generate_random_bits: function (jiff, threshold, receivers_list, compute_list, Zp, params, protocols) { + if (params.count == null) { + params.count = 1; + } + + if (params.op_id == null && params.output_op_id == null) { + params.op_id = jiff.counters.gen_op_id2('generate_random_bits', receivers_list, compute_list); + } else if (params.op_id == null) { + params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); + } + + var op_id = params.op_id; + var _params = params; + + var promises = []; + var bits = []; + for (var i = 0; i < params.count; i++) { + params = Object.assign({}, _params); + params.op_id = op_id + ':' + i; + + var bit = protocols.generate_random_bit(threshold, receivers_list, compute_list, Zp, params, protocols); + + promises.push(bit.value); + if (bit.share != null) { + bits.push(bit.share); + } + } + + if (bits.length === 0) { + bits = null; + } + return { share: bits, promise: Promise.all(promises) }; + } + }; + }, + {} + ], + 35: [ + function (require, module, exports) { + /** + * Can be used to generate shares of a random number, or shares of zero. + * For a random number, every party generates a local random number and secret share it, + * then every party sums its share, resulting in a single share of an unknown random number for every party. + * The same approach is followed for zero, but instead, all the parties know that the total number is zero, but they + * do not know the value of any resulting share (except their own) + * @function jiff_share_all_number + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {number} n - the number to share + * @param {number} [threshold=receivers_list.length] - the min number of parties needed to reconstruct the secret, defaults to all the receivers + * @param {Array} [receivers_list=all_parties] - array of party ids to receive the result, by default, this includes all parties + * @param {Array} [compute_list=all_parties] - array of party ids to perform the protocol, by default, this includes all parties + * @param {number} [Zp=jiff.Zp] - the mod + * @param {object} [params={}] - an object containing extra parameters passed by the user + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * @return {Object} contains 'share' (this party's share of the result) and 'promise' + */ + var jiff_share_all_number = function (jiff, n, threshold, receivers_list, compute_list, Zp, params) { + var isSender = compute_list.indexOf(jiff.id) > -1; + var isReceiver = receivers_list.indexOf(jiff.id) > -1; + + if (!isSender && !isReceiver) { + return {}; + } - /** - * Generate a unique operation id for a new operation object given two distinct executing parties lists (for preprocessing). - * For example, when sharing, this is given two potentially different lists of senders and receivers. - * The returned op_id will be unique with respect to other operations, and identifies the same - * operation across all parties, as long as all parties are executing instructions in the same order. - * @method gen_op_id2_preprocessing - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string} op - the type/name of operation performed. - * @param {Array} receivers - an array containing the ids of all the parties carrying out the receiving portion of the operation. - * @param {Array} senders - an array containing the ids of all the parties carrying out the sending portion of the operation. - * @return {string} the op_id for the operation. - */ - jiffClient.counters.gen_op_id2_preprocessing = function (op, receivers, senders) { - var label = jiffClient.op_id_seed + op + ':' + senders.join(',') + ':' + receivers.join(','); - if (jiffClient.counters.op_count_preprocessing[label] == null) { - jiffClient.counters.op_count_preprocessing[label] = 0; - } - return label + ':' + (jiffClient.counters.op_count_preprocessing[label]++); - }; -}; -},{}],10:[function(require,module,exports){ -// Server extensions management system -module.exports = function (JIFFClient) { - /** - * Checks if the given extension is applied. - * @method has_extension - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string} name - the extension name (found in the extension filename as jiff-client-[name].js). - * @return {boolean} true if the extension was applied, false otherwise. - */ - JIFFClient.prototype.has_extension = function (name) { - return this.extensions.indexOf(name) > -1; - }; + if (params.compute_threshold == null) { + params.compute_threshold = Math.min(threshold, compute_list.length); + } - /** - * Checks if a given extension can be safely applied to the instance - * @method can_apply_extension - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string} name - the extension name (found in the filename at jiff-client[name].js) - * @return {boolean} true if the extension can be safely applied, otherwise returns an error message. - */ - JIFFClient.prototype.can_apply_extension = function (name) { - return true; - }; + var result, promise; + if (isSender) { + var shares = jiff.internal_share(n, params.compute_threshold, compute_list, compute_list, Zp, params.op_id + ':share'); + result = shares[compute_list[0]]; + for (var i = 1; i < compute_list.length; i++) { + result = result.isadd(shares[compute_list[i]]); + } + promise = result.value; + } - /** - * Applies the given extension. - * If the extension is safe (as per can_apply_extension), it will be applied successfully. - * If the extension is not safe to be applied, an exception will be thrown with an appropriate error message. - * @see {@link module:jiff-client~JIFFClient#can_apply_extension} - * @method apply_extension - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {object} ext - the namespace of the extension acquired when the extension is imported, should contain a make_jiff function. - * @param {object} [options={}] - optional options to be passed to the extension. - */ - JIFFClient.prototype.apply_extension = function (ext, options) { - if (options == null) { - options = {}; - } + result = jiff.reshare(result, threshold, receivers_list, compute_list, Zp, params.op_id + ':reshare'); + if (receivers_list.indexOf(jiff.id) > -1) { + promise = result.value; + } - var name = ext.name; - var status = this.can_apply_extension(name); + return { share: result, promise: promise }; + }; + + module.exports = { + /** + * Creates shares of an unknown random number. Every party comes up with its own random number and shares it. + * Then every party combines all the received shares to construct one share of the random unknown number + * @function generate_random_number + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed + * @param {Array} receivers_list - array of party ids to receive the result + * @param {Array} compute_list - array of party ids to perform the protocol + * @param {number} Zp - the mod + * @param {object} params - an object containing extra parameters passed by the user + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * @return {Object} contains 'share' (this party's share of the result) and 'promise' + */ + generate_random_number: function (jiff, threshold, receivers_list, compute_list, Zp, params) { + if (params.op_id == null && params.output_op_id == null) { + params.op_id = jiff.counters.gen_op_id2('generate_random_number', receivers_list, compute_list); + } else if (params.op_id == null) { + params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); + } + return jiff_share_all_number(jiff, jiff.helpers.random(Zp), threshold, receivers_list, compute_list, Zp, params); + }, + /** + * Creates shares of 0, such that no party knows the other parties' shares. + * Every party secret shares 0, then every party sums all the shares they received, resulting + * in a new share of 0 for every party + * @function generate_zero + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff instance + * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed + * @param {Array} receivers_list - array of party ids to receive the result + * @param {Array} compute_list - array of party ids to perform the protocol + * @param {number} Zp - the mod + * @param {object} params - an object containing extra parameters passed by the user + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * @return {Object} contains 'share' (this party's share of the result) and 'promise' + */ + generate_zero: function (jiff, threshold, receivers_list, compute_list, Zp, params) { + if (params.op_id == null && params.output_op_id == null) { + params.op_id = jiff.counters.gen_op_id2('generate_zero', receivers_list, compute_list); + } else if (params.op_id == null) { + params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); + } + return jiff_share_all_number(jiff, 0, threshold, receivers_list, compute_list, Zp, params); + } + }; + }, + {} + ], + 36: [ + function (require, module, exports) { + /** + * Creates shares of r and x, such that r is a uniform random number between 0 and Zp, and x is floor(r/constant) + * where constant is provided by the extra params + * @function generate_random_and_quotient + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed + * @param {Array} receivers_list - array of party ids to receive the result + * @param {Array} compute_list - array of party ids to perform the protocol + * @param {number} Zp - the mod + * @param {object} params - an object containing extra parameters passed by the user + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated + * - compute_threshold: the threshold to use during computation: defaults to compute_list.length + * - constant: the constant to divide the random number by. + * - output_op_id: the set op id of the output quotient and noise + * @return {Object} contains 'share' (this party's share of the result) and 'promise' + */ + module.exports = function (jiff, threshold, receivers_list, compute_list, Zp, params, protocols) { + // consistent and unique op_id for compute and receiver parties + if (params.op_id == null && params.output_op_id == null) { + params.op_id = jiff.counters.gen_op_id2('generate_random_and_quotient', receivers_list, compute_list); + } else if (params.op_id == null) { + params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); + } - if (status === true) { - ext.make_jiff(this, options); + if (params.compute_threshold == null) { + params.compute_threshold = Math.floor((compute_list.length + 1) / 2); // honest majority BGW + } - this.extensions.push(name); - this.extension_applied(name, options); - } else { - throw status; - } - }; + var constant = params.constant; + var op_id = params.op_id; + Zp = Zp ? jiff.Zp : Zp; + + // stores the result + var r, q; + + // for compute parties + var promise; + if (compute_list.indexOf(jiff.id) > -1) { + var largestQuotient = jiff.share_helpers['floor'](jiff.share_helpers['/'](Zp, constant)); + var largestMultiple = jiff.share_helpers['*'](largestQuotient, constant); + + // Uniform random number between [0, Zp) + var r_bits = jiff.protocols.bits.rejection_sampling(0, Zp, params.compute_threshold, compute_list, Zp, params.output_op_id + ':rejection1'); + var cmp = jiff.protocols.bits.cgteq(r_bits, largestMultiple, params.output_op_id + ':bits_cgteq'); + var r1 = jiff.protocols.bits.bit_composition(r_bits); + + // Uniform random number between [0, Math.floor(Zp / constant)) + var quotient = jiff.protocols.bits.rejection_sampling(0, largestQuotient, params.compute_threshold, compute_list, Zp, params.output_op_id + ':rejection2'); + quotient = jiff.protocols.bits.bit_composition(quotient); + + // Uniform random number between [0, constant) + var remainder = jiff.protocols.bits.rejection_sampling(0, constant, params.compute_threshold, compute_list, Zp, params.output_op_id + ':rejection3'); + remainder = jiff.protocols.bits.bit_composition(remainder); + var r2 = quotient.icmult(constant).isadd(remainder); + + // choose either (r1, largestQuotient) or (r2, quotient) based on cmp result + r = cmp.iif_else(r1, r2, params.output_op_id + ':ifelse1'); + q = cmp.iif_else(largestQuotient, quotient, params.output_op_id + ':ifelse2'); + promise = Promise.all([r.value, q.value]); + } - /** - * Called when an extension is applied successfully. Override to change behavior of your extension based on future extensions. - * @method extension_applied - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string} name - the name of the applied extension. - * @param {object} [options={}] - the options passed by the user to the newly applied extension. - */ - JIFFClient.prototype.extension_applied = function (name, options) {}; -}; -},{}],11:[function(require,module,exports){ -/** - * The hooks for this instance. - * Checkout the hooks documentation - * @see {@link module:jiff-client~JIFFClient#hooks} - * @name hooks - * @alias hooks - * @namespace - */ + // reshare the result with the designated receivers + r = jiff.reshare(r, threshold, receivers_list, compute_list, Zp, op_id + ':reshare1'); + q = jiff.reshare(q, threshold, receivers_list, compute_list, Zp, op_id + ':reshare2'); -var crypto = require('../util/crypto.js'); -var shamir_share = require('../protocols/shamir/share.js'); -var shamir_open = require('../protocols/shamir/open.js'); + // return result + if (receivers_list.indexOf(jiff.id) > -1) { + promise = Promise.all([r.value, q.value]); + } + return { share: { r: r, q: q }, promise: promise }; + }; + }, + {} + ], + 37: [ + function (require, module, exports) { + var sample = function (jiff, range, compute_list, Zp, params, protocols, reject_count) { + // Transform sampling range into bit size + var bitLength = jiff.helpers.ceil(jiff.helpers.bLog(range, 2)); + bitLength = parseInt(bitLength.toString(), 10); + + var paramsCopy = Object.assign({}, params); + paramsCopy['count'] = bitLength; + paramsCopy['op_id'] = params.op_id + ':sampling:' + reject_count; + return protocols.generate_random_bits(params.compute_threshold, compute_list, compute_list, Zp, paramsCopy, protocols).share; + }; + + var one_round_sampling = function (jiff, lower_bound, upper_bound, compute_list, Zp, params, protocols, finalLength, reject_count) { + // Figure out sampling range + var range; + if (upper_bound.isBigNumber === true) { + range = upper_bound.minus(lower_bound); + } else { + range = upper_bound - lower_bound; + } -function Hooks(jiffClient) { - this.jiffClient = jiffClient; + // Special cases + if (range.toString() === '0') { + throw new Error('rejection sampling called with range 0, no numbers to sample!'); + } + if (range.toString() === '1') { + var zero = protocols.generate_zero(params.compute_threshold, compute_list, compute_list, Zp, params, protocols).share; + // special case: cadd can be performed locally on bit arrays of length 1! + var resultOne = jiff.protocols.bits.cadd([zero], lower_bound); + while (resultOne.length > finalLength) { + resultOne.pop(); + } + return { share: resultOne, promise: true }; + } - // avoid sharing aliases to the same array - for (hook in Hooks.prototype) { - if (Hooks.prototype.hasOwnProperty(hook) && typeof(Hooks.prototype[hook].length) === 'number' && Hooks.prototype[hook].slice) { - this[hook] = Hooks.prototype[hook].slice(); - } - } + // Rejection protocol + var bits = sample(jiff, range, compute_list, Zp, params, protocols, reject_count); + var cmp = jiff.protocols.bits.clt(bits, range, params.output_op_id + ':bits.clt:' + reject_count); + var bits_add = jiff.protocols.bits.cadd(bits, lower_bound, params.output_op_id + ':bits.cadd:' + reject_count); - // fill in hooks from options - var optionHooks = jiffClient.options.hooks || {}; - for (var hook in optionHooks) { - if (hook === 'afterOperation') { - this[hook] = optionHooks[hook].concat(this[hook]); - } else if (optionHooks.hasOwnProperty(hook)) { - this[hook] = optionHooks[hook]; - } - } -} - -/** - * Hook for computing shares of a secret - * @method computeShares - * @memberof hooks - * @param jiffClient {module:jiff-client~JIFFClient} - the jiff client instance - * @param secret {number} - the secret to share - * @param parties_list {number[]} - array of party ids to share with - * @param threshold {number} - threshold of sharing - * @param Zp {number} - the field prime - */ -Hooks.prototype.computeShares = shamir_share.jiff_compute_shares; -Hooks.prototype.reconstructShare = shamir_open.jiff_lagrange; + if (cmp === true) { + return { share: bits_add, promise: true }; + } else if (cmp === false) { + // need to resample + return { share: bits_add, promise: false }; + } -// Crypto hooks -Hooks.prototype.encryptSign = function (jiffClient, message) { - if (jiffClient.sodium_ !== false) { - return crypto.encrypt_and_sign.apply(null, arguments); - } else { - return message; - } -}; + var promise = jiff.internal_open(cmp, compute_list, params.output_op_id + ':open:' + reject_count); + return { + share: bits_add, + promise: promise.then(function (cmp) { + return cmp.toString() === '1'; + }) + }; + }; + + var computeParty = function (jiff, lower_bound, upper_bound, threshold, receivers_list, compute_list, Zp, params, protocols, op_id, finalLength, reject_count) { + var result = one_round_sampling(jiff, lower_bound, upper_bound, compute_list, Zp, params, protocols, finalLength, reject_count); + + // Case 1: we know whether sampling succeeded or not + if (result.promise === true) { + return reshareResult(jiff, upper_bound, threshold, receivers_list, compute_list, Zp, op_id, finalLength, result.share); + } + if (result.promise === false) { + return { share: 'RETRY', promise: jiff.utils.all_promises(result.share) }; + } -Hooks.prototype.decryptSign = function (jiffClient, cipher) { - if (jiffClient.sodium_ !== false) { - return crypto.decrypt_and_sign.apply(null, arguments); - } else { - return cipher; - } -}; + // Case 2: we only have a promise to whether the sampling succeeded or not + var many_shares = jiff.utils.many_secret_shares(finalLength, compute_list, threshold, Zp); + var final_deferreds = many_shares.deferreds; + + result.promise.then(function (promiseVal) { + // RETRY and PLACEHOLDER shares are cleaned up later in the preprocessing pipeline + if (promiseVal === false) { + for (var i = 0; i < final_deferreds.length; i++) { + final_deferreds[i].resolve('RETRY'); + } + return; + } + + // Need to make sure party only executes the reshare operation if sampling succeeds + var reshared = reshareResult(jiff, upper_bound, threshold, receivers_list, compute_list, Zp, op_id, finalLength, result.share); + + if (receivers_list.indexOf(jiff.id) > -1) { + jiff.utils.resolve_many_secrets(final_deferreds, reshared.share); + } else { + for (i = 0; i < final_deferreds.length; i++) { + final_deferreds[i].resolve('PLACEHOLDER'); + } + } + }); + + return { share: many_shares.shares, promise: jiff.utils.all_promises(many_shares.shares) }; + }; + + var reshareResult = function (jiff, upper_bound, threshold, receivers_list, compute_list, Zp, op_id, finalLength, shares) { + // fix threshold and parties + var promises = []; + for (var i = 0; i < finalLength; i++) { + if (compute_list.indexOf(jiff.id) > -1) { + promises[i] = shares[i].value; + } + shares[i] = jiff.reshare(shares[i], threshold, receivers_list, compute_list, Zp, op_id + ':reshare:' + i); + if (receivers_list.indexOf(jiff.id) > -1) { + promises[i] = shares[i].value; + } + } -Hooks.prototype.generateKeyPair = function (jiffClient) { - if (jiffClient.sodium_ !== false) { - var key = jiffClient.sodium_.crypto_box_keypair(); // this party's public and secret key - return { public_key: key.publicKey, secret_key: key.privateKey } - } else { - return { public_key: '', secret_key: ''}; - } -}; + // return output + if (receivers_list.indexOf(jiff.id) === -1) { + shares = null; + } -Hooks.prototype.parseKey = function (jiffClient, keyString) { - if (jiffClient.sodium_ !== false) { - return new Uint8Array(JSON.parse(keyString)); - } else { - return ''; - } -}; + // handle rejection case + return { share: shares, promise: Promise.all(promises) }; + }; + + /** + * Wrapper for when doing rejection sampling during pre processing + * + * Do not use this function directly, especially during online computation time, use jiffClient.protocols.bits.rejection_sampling instead + * + * @function sampling + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {number} [threshold=receivers_list.length] - the threshold of the resulting shares after sampling + * @param {Array} [receivers_list=all_parties] - array of party ids that want to receive the sampling shares, by default, this includes all parties + * @param {Array} [compute_list=all_parties] - array of party ids that will perform this protocol, by default, this includes all parties + * @param {number} [Zp=jiff-instance.Zp] - the mod (if null then the default Zp for the instance is used) + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. + * This id must be unique, and must be passed by all parties to the same instruction, to + * ensure that corresponding instructions across different parties are matched correctly + * @param {object} [params={}] - an object containing extra parameters passed by the user + * Expects: + * - compute_threshold, an optional number compute_threshold parameter, which specifies threshold used + * during the protocol execution. By default, this is (|compute_list|+1)/2 + * - optional 'lower_bound' and 'upper_bound', numeric parameters, default to 0 and Zp respectively + * - op_id, the base op_id to tag operations inside this protocol with, defaults to auto generated + * - output_op_id, the tag id for the output result + * - retry_count, how many times rejection sampling have been retried! + * @param {object} [protocols=defaults] - the protocols to use for preprocessing, any protocol(s) not provided will be replaced with defaults + * @returns {Object} an object containing keys: 'share', and 'promise'. The promise is resolved when the rejection sampling is completed. + * The object is consumed by .preprocessing: + * - 'share' attribute contains the resulting array of secret shared bits representing the sampled value, and is stored in the preprocessing table internally + * - The promise is consumed and a new promise is returned by .preprocessing that is resolved after this returned promise (and all other promise generated by that .preprocessing call) are resolved + */ + module.exports = function (jiff, threshold, receivers_list, compute_list, Zp, params, protocols) { + // Internal version: set parameters (e.g. receivers_list) do not need defaults + // defaults (for internal preprocessing) + var lower_bound = params.lower_bound != null ? params.lower_bound : 0; + var upper_bound = params.upper_bound != null ? params.upper_bound : Zp; + if (params.compute_threshold == null) { + // honest majority BGW + params.compute_threshold = Math.floor((compute_list.length + 1) / 2); + } -Hooks.prototype.dumpKey = function (jiffClient, key) { - if (jiffClient.sodium_ !== false) { - return '[' + key.toString() + ']'; - } else { - return ''; - } -}; + // Figure out final bit size (after adding back lower) + var finalLength = jiff.helpers.ceil(jiff.helpers.bLog(upper_bound, 2)); + finalLength = parseInt(finalLength.toString(), 10); + finalLength = Math.max(finalLength, 1); // special case: when upper_bound is 1! -// Array Hooks -Hooks.prototype.beforeShare = []; -Hooks.prototype.afterComputeShare = []; -Hooks.prototype.receiveShare = []; + if (params.op_id == null && params.output_op_id == null) { + params.op_id = jiff.counters.gen_op_id2('rejection_sampling', receivers_list, compute_list); + } else if (params.op_id == null) { + params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); + } + var op_id = params.op_id; -Hooks.prototype.beforeOpen = []; -Hooks.prototype.receiveOpen = []; -Hooks.prototype.afterReconstructShare = []; + // Rejection count + var reject_count = params.reject_count || 0; -Hooks.prototype.createSecretShare = []; + if (compute_list.indexOf(jiff.id) === -1) { + return reshareResult(jiff, upper_bound, threshold, receivers_list, compute_list, Zp, op_id, finalLength, []); + } -Hooks.prototype.beforeOperation = []; -Hooks.prototype.afterOperation = [ - // parse content of share/open messages to be integers (instead of strings due to encryption/decryption) - function (jiff, label, msg) { - if (label === 'share' || label === 'open') { - msg['share'] = parseInt(msg['share'], 10); - } - return msg; - } -]; - -/** - * Execute all hooks attached to the given name in order. - * Hooks are executed sequentially such that the first hook's return value is passed into the second and so on. - * @method execute_array_hooks - * @memberof hooks - * @param {string} hook_name - the name of the hook - * @param {Array} params - parameters to pass to the hooks - * @param {number} acc_index - the index in params in which the result of the hooks must be saved, if no hooks - * exist for the name, then params[acc_index] is returned. - * @return {object} returns the result of the last hook. - */ -Hooks.prototype.execute_array_hooks = function (hook_name, params, acc_index) { - var arr = this.jiffClient.hooks[hook_name]; - arr = (arr == null ? [] : arr); + return computeParty(jiff, lower_bound, upper_bound, threshold, receivers_list, compute_list, Zp, params, protocols, op_id, finalLength, reject_count); + }; + }, + {} + ], + 38: [ + function (require, module, exports) { + /** + * Generation of beaver triplet via MPC, uses the server for communication channels, but not for generation + * @function generate_beaver_bgw + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {number} threshold - the threshold of the triplets when stored by receivers after generation + * @param {Array} receivers_list - array of party ids that want to receive the triplet shares + * @param {Array} compute_list - array of party ids that will perform this protocol + * @param {number} Zp - the mod + * @param {object} params - an object containing extra parameters passed by the user + * Expects: + * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated + * - an optional number compute_threshold parameter, which specifies threshold used + * during the protocol execution. By default, this is the length of the (compute_list+1)/2 + * @param {object} protocols - the sub protocols to use for preprocessing + * @return {object} all pre-processing protocols must return an object with these keys: + * { + * 'share': the share(s)/value(s) to store attached to op_id for later use by the computation (i.e. the result of preprocessing), + * 'promise': a promise for when this protocol is fully completed (could be null if the protocol was already completed) + * } + * In this case, 'share' is an array of this party's shares of the resulting triplet, a,b,c such that a*b=c + */ + module.exports = function (jiff, threshold, receivers_list, compute_list, Zp, params, protocols) { + if (params.compute_threshold == null) { + params.compute_threshold = Math.floor((compute_list.length + 1) / 2); // honest majority BGW + } + if (params.op_id == null && params.output_op_id == null) { + params.op_id = jiff.counters.gen_op_id2('generate_beaver_bgw', receivers_list, compute_list); + } else if (params.op_id == null) { + params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); + } - for (var i = 0; i < arr.length; i++) { - params[acc_index] = arr[i].apply(this.jiffClient, params); - } - return params[acc_index]; -}; - -module.exports = Hooks; -},{"../protocols/shamir/open.js":39,"../protocols/shamir/share.js":41,"../util/crypto.js":48}],12:[function(require,module,exports){ -var initializationHandlers = require('./handlers/initialization.js'); -var shareHandlers = require('./handlers/sharing.js'); -var customHandlers = require('./handlers/custom.js'); -var cryptoProviderHandlers = require('./handlers/crypto_provider.js'); - -/** - * Contains handlers for communication events - * @name handlers - * @alias handlers - * @namespace - */ + var op_id = params.op_id; + var _params = params; -// Add handlers implementations -module.exports = function (jiffClient) { - // fill in handlers - initializationHandlers(jiffClient); - shareHandlers(jiffClient); - customHandlers(jiffClient); - cryptoProviderHandlers(jiffClient); -}; -},{"./handlers/crypto_provider.js":13,"./handlers/custom.js":14,"./handlers/initialization.js":15,"./handlers/sharing.js":16}],13:[function(require,module,exports){ -// setup handler for receiving messages from the crypto provider -module.exports = function (jiffClient) { - /** - * Parse crypto provider message and resolve associated promise. - * @method - * @memberof handlers - * @param {object} json_msg - the parsed json message as received by the crypto_provider event, contains 'values' and 'shares' attributes. - */ - jiffClient.handlers.receive_crypto_provider = function (json_msg) { - // Hook - json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'crypto_provider', json_msg], 2); - - var op_id = json_msg['op_id']; - if (jiffClient.deferreds[op_id] == null) { - return; // duplicate message: ignore - } + var a, b, c, promises; + if (compute_list.indexOf(jiff.id) > -1) { + params = Object.assign({}, _params); + params.op_id = op_id + ':share_a'; + a = protocols.generate_random_number(params.compute_threshold, compute_list, compute_list, Zp, params, protocols).share; - // parse msg - var receivers_list = json_msg['receivers']; - var threshold = json_msg['threshold']; - var Zp = json_msg['Zp']; + params = Object.assign({}, _params); + params.op_id = op_id + ':share_b'; + b = protocols.generate_random_number(params.compute_threshold, compute_list, compute_list, Zp, params, protocols).share; - // construct secret share objects - var result = {}; - if (json_msg['values'] != null) { - result.values = json_msg['values']; - } - if (json_msg['shares'] != null) { - result.shares = []; - for (var i = 0; i < json_msg['shares'].length; i++) { - result.shares.push(new jiffClient.SecretShare(json_msg['shares'][i], receivers_list, threshold, Zp)); - } - } + c = a.ismult_bgw(b, op_id + ':smult_bgw'); + promises = [a.value, b.value, c.value]; + } - // resolve deferred - jiffClient.deferreds[op_id].resolve(result); - delete jiffClient.deferreds[op_id]; - }; -}; -},{}],14:[function(require,module,exports){ -module.exports = function (jiffClient) { - /** - * Called when this party receives a custom tag message from any party (including itself). - * If a custom listener was setup to listen to the tag, the message is passed to the listener. - * Otherwise, the message is stored until such a listener is provided. - * @method - * @memberof handlers - * @param {object} json_msg - the parsed json message as received by the custom event. - */ - jiffClient.handlers.receive_custom = function (json_msg) { - if (json_msg['party_id'] !== jiffClient.id) { - if (json_msg['encrypted'] === true) { - json_msg['message'] = jiffClient.hooks.decryptSign(jiffClient, json_msg['message'], jiffClient.secret_key, jiffClient.keymap[json_msg['party_id']]); - } + a = jiff.reshare(a, threshold, receivers_list, compute_list, Zp, op_id + ':reshare_a'); + b = jiff.reshare(b, threshold, receivers_list, compute_list, Zp, op_id + ':reshare_b'); + c = jiff.reshare(c, threshold, receivers_list, compute_list, Zp, op_id + ':reshare_c'); + if (receivers_list.indexOf(jiff.id) > -1) { + promises = [a.value, b.value, c.value]; + } - json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'custom', json_msg], 2); - } + return { share: [a, b, c], promise: Promise.all(promises) }; + }; + }, + {} + ], + 39: [ + function (require, module, exports) { + /** + * Share the given share to all the parties in the jiff instance. + * @function jiff_broadcast + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance. + * @param {module:jiff-client~JIFFClient#SecretShare} share - the share. + * @param {Array} parties - the parties to broadcast the share to. + * @param {number|string} op_id - a unique operation id, used to tag outgoing messages. + * + */ + var jiff_broadcast = function (jiff, share, parties, op_id) { + for (var index = 0; index < parties.length; index++) { + var i = parties[index]; // Party id + if (i === jiff.id) { + jiff.handlers.receive_open({ party_id: i, share: share.value, op_id: op_id, Zp: share.Zp }); + continue; + } + + // encrypt, sign and send + var msg = { party_id: i, share: share.value, op_id: op_id, Zp: share.Zp }; + msg = jiff.hooks.execute_array_hooks('beforeOperation', [jiff, 'open', msg], 2); + + msg['share'] = jiff.hooks.encryptSign(jiff, msg['share'].toString(), jiff.keymap[msg['party_id']], jiff.secret_key); + jiff.socket.safe_emit('open', JSON.stringify(msg)); + } + }; + + module.exports = { + /** + * Open up the given share to the participating parties. + * @function jiff_open + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance. + * @param {module:jiff-client~JIFFClient#SecretShare} share - the share of the secret to open that belongs to this party + * @param {Array} [parties=all_parties] - an array with party ids of receiving parties + * @param {string|number} [op_id=auto_gen()] - the operation id to be used to tag outgoing messages + * @returns {?promise} a (JQuery) promise to the open value of the secret, or null if the calling party is not a receiving party + * + */ + jiff_open: function (jiff, share, parties, op_id) { + var i; + + if (!(share.jiff === jiff)) { + throw 'share does not belong to given instance'; + } + + // Default values + if (parties == null || parties === []) { + parties = []; + for (i = 1; i <= jiff.party_count; i++) { + parties.push(i); + } + } else { + jiff.helpers.sort_ids(parties); + } + + // If not a receiver nor holder, do nothing + if (share.holders.indexOf(jiff.id) === -1 && parties.indexOf(jiff.id) === -1) { + return null; + } + + // Compute operation ids (one for each party that will receive a result + if (op_id == null) { + op_id = jiff.counters.gen_op_id2('open', parties, share.holders); + } + + // Party is a holder + if (share.holders.indexOf(jiff.id) > -1) { + // Call hook + share = jiff.hooks.execute_array_hooks('beforeOpen', [jiff, share, parties], 1); + + // refresh/reshare, so that the original share remains secret, instead + // a new share is sent/open without changing the actual value. + share = share.refresh(op_id + ':refresh'); + + // The given share has been computed, broadcast it to all parties + jiff.counters.pending_opens++; + share.wThen(function () { + jiff.counters.pending_opens--; + jiff_broadcast(jiff, share, parties, op_id); + }, share.error); + } + + // Party is a receiver + if (parties.indexOf(jiff.id) > -1) { + var final_deferred = new jiff.helpers.Deferred(); // will be resolved when the final value is reconstructed + var final_promise = final_deferred.promise; + + if (jiff.deferreds[op_id] == null) { + jiff.deferreds[op_id] = {}; + } + + jiff.deferreds[op_id].deferred = final_deferred; + jiff.deferreds[op_id].threshold = share.threshold; + jiff.deferreds[op_id].total = share.holders.length; + if (jiff.deferreds[op_id].shares != null && jiff.deferreds[op_id].shares.length >= share.threshold) { + final_deferred.resolve(); + } + + return final_promise.then(function () { + var shares = jiff.deferreds[op_id].shares; + + if (shares.length === jiff.deferreds[op_id].total) { + delete jiff.deferreds[op_id]; + } else { + jiff.deferreds[op_id].deferred = 'CLEAN'; + } + + var recons_secret = jiff.hooks.reconstructShare(jiff, shares); + recons_secret = jiff.hooks.execute_array_hooks('afterReconstructShare', [jiff, recons_secret], 1); + return recons_secret; + }); + } + + return null; + }, + /** + * Uses Lagrange polynomials to interpolate the polynomial + * described by the given shares (points) + * @function jiff_lagrange + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {Array} shares - an array of objects representing shares to reconstruct, every object has 3 attributes: value, sender_id, Zp + * @returns {number} the value of the polynomial at x=0 (the secret value) + * + */ + jiff_lagrange: function (jiff, shares) { + var lagrange_coeff = []; // will contain shares.length many elements. + + // Compute the Langrange coefficients at 0. + for (var i = 0; i < shares.length; i++) { + var pi = jiff.helpers.get_party_number(shares[i].sender_id); + lagrange_coeff[pi] = 1; + + for (var j = 0; j < shares.length; j++) { + var pj = jiff.helpers.get_party_number(shares[j].sender_id); + if (pj !== pi) { + var inv = jiff.helpers.extended_gcd(pi - pj, shares[i].Zp)[0]; + lagrange_coeff[pi] = jiff.helpers.mod(lagrange_coeff[pi] * (0 - pj), shares[i].Zp) * inv; + lagrange_coeff[pi] = jiff.helpers.mod(lagrange_coeff[pi], shares[i].Zp); + } + } + } + + // Reconstruct the secret via Lagrange interpolation + var recons_secret = 0; + for (var p = 0; p < shares.length; p++) { + var party = jiff.helpers.get_party_number(shares[p].sender_id); + var tmp = jiff.helpers.mod(shares[p].value * lagrange_coeff[party], shares[p].Zp); + recons_secret = jiff.helpers.mod(recons_secret + tmp, shares[p].Zp); + } + + return recons_secret; + } + }; + }, + {} + ], + 40: [ + function (require, module, exports) { + /** + * re-share an existing share (value) under a new threshold or to a new set of parties or both. + * Do not use this to refresh a share (use {@link jiff-client~JIFFClient#SecretShare#refresh} instead) + * @function reshare + * @ignore + * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance + * @param {module:jiff-client~JIFFClient#SecretShare} [share=null] - the share you would like to reshare (null if you are a receiver but not a sender) + * @param {number} [threshold=receivers_list.length] - the new threshold, defaults to the length of receivers_list param + * @param {Array} [receivers_list=all_parties] - array of party ids to receive from, by default, this includes all parties + * @param {Array} [senders_list=all_parties] - array of party ids that posses the share and will reshare it with the receivers, by default, this includes all parties + * @param {number} [Zp=jiff.Zp] - the Zp of the existing share + * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this multiplication (and internally, the corresponding beaver triplet). + * This id must be unique, and must be passed by all parties to the same instruction. + * this ensures that every party gets a share from the same triplet for every matching instruction. An automatic id + * is generated by increasing a local counter, default ids suffice when all parties execute the + * instructions in the same order + * @return {SecretShare} this party's share of the result under the new threshold, or null if this party is not a receiver + */ + module.exports = function (jiff, share, threshold, receivers_list, senders_list, Zp, op_id) { + var i; + + // default values + if (receivers_list == null) { + receivers_list = []; + for (i = 1; i <= jiff.party_count; i++) { + receivers_list.push(i); + } + } else { + jiff.helpers.sort_ids(receivers_list); + } + if (senders_list == null) { + senders_list = []; + for (i = 1; i <= jiff.party_count; i++) { + senders_list.push(i); + } + } else { + jiff.helpers.sort_ids(senders_list); + } - var sender_id = json_msg['party_id']; - var tag = json_msg['tag']; - var message = json_msg['message']; + if (threshold == null) { + threshold = receivers_list.length; + } + if (Zp == null) { + Zp = jiff.Zp; + } - if (jiffClient.listeners[tag] != null) { - jiffClient.listeners[tag](sender_id, message); - } else { - // Store message until listener is provided - var stored_messages = jiffClient.custom_messages_mailbox[tag]; - if (stored_messages == null) { - stored_messages = []; - jiffClient.custom_messages_mailbox[tag] = stored_messages; - } + if (op_id == null) { + op_id = jiff.counters.gen_op_id('reshare', senders_list); + } - stored_messages.push({sender_id: sender_id, message: message}); - } - } -}; -},{}],15:[function(require,module,exports){ -// add handlers for initialization -module.exports = function (jiffClient) { - jiffClient.options.initialization = Object.assign({}, jiffClient.options.initialization); - - /** - * Called when an error occurs - * @method - * @memberof handlers - * @param {string} label - the name of message or operation causing the error - * @param {error|string} error - the error - */ - jiffClient.handlers.error = function (label, error) { - if (jiffClient.options.onError) { - jiffClient.options.onError(label, error); - } + // Check if this party is a sender or receiver + var isSender = senders_list.indexOf(jiff.id) > -1; + var isReceiver = receivers_list.indexOf(jiff.id) > -1; + if (!isSender && !isReceiver) { + return null; + } - console.log(jiffClient.id, ':', 'Error from server:', label, '---', error); // TODO: remove debugging - if (label === 'initialization') { - jiffClient.socket.disconnect(); + // optimization, if nothing changes, keep share + if (share != null && JSON.stringify(receivers_list) === JSON.stringify(senders_list) && threshold === share.threshold) { + return share; + } - if (jiffClient.initialization_counter < jiffClient.options.maxInitializationRetries) { - console.log(jiffClient.id, ':', 'reconnecting..'); // TODO: remove debugging - setTimeout(jiffClient.connect, jiffClient.options.socketOptions.reconnectionDelay); - } - } - }; + // Setup the result + var final_deferred; + var result = null; + if (isReceiver) { + final_deferred = new jiff.helpers.Deferred(); + result = new jiff.SecretShare(final_deferred.promise, receivers_list, threshold, Zp); + } - /** - * Builds the initialization message for this instance - * @method - * @memberof handlers - * @return {Object} - */ - jiffClient.handlers.build_initialization_message = function () { - var msg = { - computation_id: jiffClient.computation_id, - party_id: jiffClient.id, - party_count: jiffClient.party_count, - public_key: jiffClient.public_key != null ? jiffClient.hooks.dumpKey(jiffClient, jiffClient.public_key) : undefined - }; - msg = Object.assign(msg, jiffClient.options.initialization); - - // Initialization Hook - return jiffClient.hooks.execute_array_hooks('beforeOperation', [jiffClient, 'initialization', msg], 2); - }; + // This function is called when the share is ready: the value of the share has been received. + var ready_share = function () { + var intermediate_shares = jiff.internal_share(isSender ? share.value : null, threshold, receivers_list, senders_list, Zp, op_id); + + if (isReceiver) { + var promises = []; + for (var i = 0; i < senders_list.length; i++) { + var party_id = senders_list[i]; + promises.push(intermediate_shares[party_id].value); + } + + // Reconstruct share under new threshold + Promise.all(promises).then(function () { + var reconstruct_parts = []; + for (var i = 0; i < senders_list.length; i++) { + var party_id = senders_list[i]; + //shamir reconstruct takes an array of objects + //has attributes: {value: x, sender_id: y, Zp: jiff_instance.Zp} + reconstruct_parts[i] = { value: intermediate_shares[party_id].value, sender_id: party_id, Zp: Zp }; + } + var value = jiff.hooks.reconstructShare(jiff, reconstruct_parts); + final_deferred.resolve(value); + }); + } + }; + + if (isSender && !share.ready) { + share.value.then(ready_share); + } else { + // either a receiver or share is ready + ready_share(); + } - /** - * Begins initialization of this instance by sending the initialization message to the server. - * Should only be called after connection is established. - * Do not call this manually unless you know what you are doing, use .connect() instead! - * @method - * @memberof handlers - */ - jiffClient.handlers.connected = function () { - console.log('Connected!', jiffClient.id); // TODO: remove debugging - jiffClient.initialization_counter++; - - if (jiffClient.secret_key == null && jiffClient.public_key == null) { - var key = jiffClient.hooks.generateKeyPair(jiffClient); - jiffClient.secret_key = key.secret_key; - jiffClient.public_key = key.public_key; - } + return result; + }; + }, + {} + ], + 41: [ + function (require, module, exports) { + module.exports = { + /** + * Default way of computing shares (can be overridden using hooks). + * Compute the shares of the secret (as many shares as parties) using Shamir secret sharing + * @ignore + * @function jiff_compute_shares + * @param {module:jiff-client~JIFFClient} jiff - the jiff instance + * @param {number} secret - the secret to share. + * @param {Array} parties_list - array of party ids to share with. + * @param {number} threshold - the min number of parties needed to reconstruct the secret, defaults to all the receivers. + * @param {number} Zp - the mod. + * @returns {object} a map between party number and its share, this means that (party number, share) is a + * point from the polynomial. + * + */ + jiff_compute_shares: function (jiff, secret, parties_list, threshold, Zp) { + var shares = {}; // Keeps the shares + var i; + + // Each player's random polynomial f must have + // degree threshold - 1, so that threshold many points are needed + // to interpolate/reconstruct. + var t = threshold - 1; + var polynomial = Array(t + 1); // stores the coefficients + + // Each players's random polynomial f must be constructed + // such that f(0) = secret + polynomial[0] = secret; + + // Compute the random polynomial f's coefficients + for (i = 1; i <= t; i++) { + polynomial[i] = jiff.helpers.random(Zp); + } + + // Compute each players share such that share[i] = f(i) + for (i = 0; i < parties_list.length; i++) { + var p_id = parties_list[i]; + shares[p_id] = polynomial[0]; + var power = jiff.helpers.get_party_number(p_id); + + for (var j = 1; j < polynomial.length; j++) { + var tmp = jiff.helpers.mod(polynomial[j] * power, Zp); + shares[p_id] = jiff.helpers.mod(shares[p_id] + tmp, Zp); + power = jiff.helpers.mod(power * jiff.helpers.get_party_number(p_id), Zp); + } + } + + return shares; + }, + /** + * Share given secret to the participating parties. + * @ignore + * @function jiff_share + * @param {module:jiff-client~JIFFClient} jiff - the jiff instance + * @param {number} secret - the secret to share. + * @param {number} [threshold=receivers_list.length] - the min number of parties needed to reconstruct the secret, defaults to all the receivers. + * @param {Array} [receivers_list=all_parties] - array of party ids to share with, by default, this includes all parties. + * @param {Array} [senders_list=all_parties] - array of party ids to receive from, by default, this includes all parties. + * @param {number} [Zp=jiff.Zp] - the mod (if null then the default Zp for the instance is used). + * @param {string|number} [share_id=auto_gen()] - the tag used to tag the messages sent by this share operation, this tag is used + * so that parties distinguish messages belonging to this share operation from other + * share operations between the same parties (when the order of execution is not + * deterministic). An automatic id is generated by increasing a local counter, default + * ids suffice when all parties execute all sharing operations with the same senders + * and receivers in the same order. + * @returns {object} a map where the key is the sender party id + * and the value is the share object that wraps + * what was sent from that party (the internal value maybe deferred). + * if the party that calls this function is not a receiver then the map + * will be empty. + */ + jiff_share: function (jiff, secret, threshold, receivers_list, senders_list, Zp, share_id) { + var i, p_id; + + // defaults + if (Zp == null) { + Zp = jiff.Zp; + } + if (receivers_list == null) { + receivers_list = []; + for (i = 1; i <= jiff.party_count; i++) { + receivers_list.push(i); + } + } else { + jiff.helpers.sort_ids(receivers_list); + } + if (senders_list == null) { + senders_list = []; + for (i = 1; i <= jiff.party_count; i++) { + senders_list.push(i); + } + } else { + jiff.helpers.sort_ids(senders_list); + } + if (threshold == null) { + threshold = receivers_list.length; + } + if (threshold < 0) { + threshold = 2; + } + if (threshold > receivers_list.length) { + threshold = receivers_list.length; + } + + // if party is uninvolved in the share, do nothing + if (receivers_list.indexOf(jiff.id) === -1 && senders_list.indexOf(jiff.id) === -1) { + return {}; + } + + // compute operation id + if (share_id == null) { + share_id = jiff.counters.gen_op_id2('share', receivers_list, senders_list); + } + + // stage sending of shares + if (senders_list.indexOf(jiff.id) > -1) { + // Call hook + secret = jiff.hooks.execute_array_hooks('beforeShare', [jiff, secret, threshold, receivers_list, senders_list, Zp], 1); + + // compute shares + var shares = jiff.hooks.computeShares(jiff, secret, receivers_list, threshold, Zp); + + // Call hook + shares = jiff.hooks.execute_array_hooks('afterComputeShare', [jiff, shares, threshold, receivers_list, senders_list, Zp], 1); + + // send shares + for (i = 0; i < receivers_list.length; i++) { + p_id = receivers_list[i]; + if (p_id === jiff.id) { + continue; + } + + // send encrypted and signed shares_id[p_id] to party p_id + var msg = { party_id: p_id, share: shares[p_id], op_id: share_id }; + msg = jiff.hooks.execute_array_hooks('beforeOperation', [jiff, 'share', msg], 2); + + msg['share'] = jiff.hooks.encryptSign(jiff, msg['share'].toString(10), jiff.keymap[msg['party_id']], jiff.secret_key); + jiff.socket.safe_emit('share', JSON.stringify(msg)); + } + } + + // stage receiving of shares + var result = {}; + if (receivers_list.indexOf(jiff.id) > -1) { + // setup a map of deferred for every received share + if (jiff.deferreds[share_id] == null) { + jiff.deferreds[share_id] = {}; + } + + var _remaining = senders_list.length; + for (i = 0; i < senders_list.length; i++) { + p_id = senders_list[i]; + if (p_id === jiff.id) { + // Keep party's own share + var my_share = jiff.hooks.execute_array_hooks('receiveShare', [jiff, p_id, shares[p_id]], 2); + result[p_id] = new jiff.SecretShare(my_share, receivers_list, threshold, Zp); + _remaining--; + continue; + } + + // check if a deferred is set up (maybe the message was previously received) + if (jiff.deferreds[share_id][p_id] == null) { + // not ready, setup a deferred + jiff.deferreds[share_id][p_id] = new jiff.helpers.Deferred(); + } + + var promise = jiff.deferreds[share_id][p_id].promise; + // destroy deferred when done + (function (promise, p_id) { + // p_id is modified in a for loop, must do this to avoid scoping issues. + promise.then(function () { + delete jiff.deferreds[share_id][p_id]; + _remaining--; + if (_remaining === 0) { + delete jiff.deferreds[share_id]; + } + }); + })(promise, p_id); + + // receive share_i[id] from party p_id + result[p_id] = new jiff.SecretShare(promise, receivers_list, threshold, Zp); + } + } + + return result; + } + }; + }, + {} + ], + 42: [ + function (require, module, exports) { + var genericProtocols = require('./protocols/generic.js'); + var arithmetic = require('./protocols/numbers/arithmetic.js'); + var comparison = require('./protocols/numbers/comparison.js'); + var protocols = require('./protocols/numbers/protocols.js'); + var booleans = require('./protocols/booleans/boolean.js'); + + // a metaclass that creates SecretShare classes when given a jiff instance + // alternatively, we can think of this as a factory for a secret share prototypes/constructors given a jiff instance + module.exports = function (jiff) { + // Look at jiff-client#SecretShare + function SecretShare(value, holders, threshold, Zp) { + // sort holders + jiff.helpers.sort_ids(holders); + + /** + * Indicates if the secret share's value is ready or still a promise + * @member {boolean} ready + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + this.ready = value.then == null; + + /** + * The value of the share (or a promise to it) + * @member {number|promise} value + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + this.value = value; + + /** + * Array of party ids who hold shares of the corresponding secret + * @member {Array} holders + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + this.holders = holders; + /** + * The sharing threshold + * @member {number} threshold + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + this.threshold = threshold; + + /** + * The field prime under which the corresponding secret is shared + * @member {number} Zp + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + this.Zp = Zp; + + // when the promise is resolved, acquire the value of the share and set ready to true + if (!this.ready) { + this.value = this.value.then(this.promise_handler.bind(this), this.error.bind(this)); + this.jiff.add_to_barriers(this.value); + } + + // return the share + return jiff.hooks.execute_array_hooks('createSecretShare', [jiff, this], 1); + } - // Initialization message - var msg = jiffClient.handlers.build_initialization_message(); + /** + * The jiff client instance this share belongs to + * @member {module:jiff-client~JIFFClient} jiff + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.jiff = jiff; + + // Basic operations in prototype of SecretShare + /** + * Gets the value of this share + * @method valueOf + * @returns {number} the value (undefined if not ready yet) + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.valueOf = function () { + if (this.ready) { + return this.value; + } else { + return undefined; + } + }; + + /** + * Gets a string representation of this share + * @method toString + * @returns {string} the id and value of the share as a string + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + */ + SecretShare.prototype.toString = function () { + var val = this.ready ? this.value : ''; + return 'share: ' + val + '. Holders: ' + JSON.stringify(this.holders) + '. Threshold: ' + this.threshold + '. Zp: ' + this.Zp.toString() + '.'; + }; + + /** + * Logs an error. Passes the error on to the associated jiff client instance's {@link handlers.error} + * @method error + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + * @param {string|error} error - the error to log + */ + SecretShare.prototype.error = function (error) { + this.jiff.handlers.error('SecretShare', error); + }; + + /** + * Logs the value represented by this share to the console + * WARNING: THIS LEAKS INFORMATION AND MUST BE USED ONLY TO DEBUG ON FAKE DATA + * @method logLEAK + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + * @param {string} tag - accompanying tag to display in the console + * @param {Array} [parties=[holders[0]] - the parties which will display the log + * @param {string|number|object} [op_id=auto_gen()] - same as {@link module:jiff-client:JIFFClient#open} + * @return {?promise} a promise to the value represented by this share after logging it, null if party is not in parties + */ + SecretShare.prototype.logLEAK = function (tag, parties, op_id) { + if (parties == null) { + parties = [this.holders[0]]; + } + var promise = this.open(parties, tag, op_id); + if (promise != null) { + promise = promise.then(function (result) { + console.log(tag, result.toString()); + return result; + }, this.error); + } + return promise; + }; + + /** + * Handler for when this share's promise (if any) is resolved + * @method promise_handler + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + * @param {number} value - the value of the share after it was resolved + */ + SecretShare.prototype.promise_handler = function (value) { + this.value = value; + this.ready = true; + return this.value; + }; + + /** + * Executes callback when both this share and o are ready and returns the result (or a promise to the result) + * @method when_both_ready + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + * @param {SecretShare} o - the other share object. + * @param {function()} cb - the callback to execute. + * @returns {value|promise} either the return value of cb() or a promise to it + */ + SecretShare.prototype.when_both_ready = function (o, cb) { + if (this.ready && o.ready) { + return cb(); + } + + if (this.ready) { + return o.value.then(cb, this.error); + } else if (o.ready) { + return this.value.then(cb, this.error); + } else { + return Promise.all([this.value, o.value]).then(cb, this.error); + } + }; + + /** + * Shortcut for opening/revealing the value of this share. Alias for open in jiff-instance + * @see jiff-instance#open + * @method open + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + * @param {Array} [parties=all_parties] - an array with party ids (1 to n) of receiving parties + * @param {string|number|object} [op_id=auto_gen()] - same as {@link module:jiff-client:JIFFClient#open} + * @returns {?promise} a (JQuery) promise to the open value of the secret, null if the party is not specified in + * the parties array as a receiver + */ + SecretShare.prototype.open = function (parties, op_id) { + return this.jiff.open(this, parties, op_id); + }; + + /** + * Wrapper around share.value.then. + * In case share is ready (its promise is resolved and cleared) + * The callback is executed immediately. + * Does not support chaining + * @method wThen + * @memberof module:jiff-client~JIFFClient#SecretShare + * @instance + * @param {function} onFulfilled - callback for success, called with this.value as parameter + * @param {function} [onRejected=this.error] - callback for errors + * @return {promise|value} either the result of executing onFulfilled or a promise to it + */ + SecretShare.prototype.wThen = function (onFulfilled, onRejected) { + if (this.ready) { + return onFulfilled(this.value); + } else { + if (onRejected == null) { + onRejected = this.error; + } + return this.value.then(onFulfilled, onRejected); + } + }; + + // Complex protocols in prototype of SecretShare + genericProtocols(SecretShare); + arithmetic(SecretShare); + comparison(SecretShare); + protocols(SecretShare); + booleans(SecretShare); + + // internal variant of primitives, to use internally by other primitives + var internals = [ + 'cadd', + 'csub', + 'cmult', + 'sadd', + 'ssub', + 'smult', + 'smult_bgw', + 'cxor_bit', + 'sxor_bit', + 'cor_bit', + 'sor_bit', + 'slt', + 'slteq', + 'sgt', + 'sgteq', + 'seq', + 'sneq', + 'clt', + 'clteq', + 'cgt', + 'cgteq', + 'ceq', + 'cneq', + 'sdiv', + 'cdiv', + 'not', + 'cpow', + 'lt_halfprime', + 'if_else' + ]; + for (var i = 0; i < internals.length; i++) { + var key = internals[i]; + SecretShare.prototype['i' + key] = SecretShare.prototype[key]; + } - // Emit initialization message to server - jiffClient.socket.emit('initialization', JSON.stringify(msg)); - }; + return SecretShare; + }; + }, + { + './protocols/booleans/boolean.js': 29, + './protocols/generic.js': 30, + './protocols/numbers/arithmetic.js': 31, + './protocols/numbers/comparison.js': 32, + './protocols/numbers/protocols.js': 33 + } + ], + 43: [ + function (require, module, exports) { + module.exports = { + '+': function (v1, v2) { + return v1 + v2; + }, + '-': function (v1, v2) { + return v1 - v2; + }, + '*': function (v1, v2) { + return v1 * v2; + }, + '/': function (v1, v2) { + return v1 / v2; + }, + '<': function (v1, v2) { + return v1 < v2; + }, + '<=': function (v1, v2) { + return v1 <= v2; + }, + floor: function (v) { + return Math.floor(v); + }, + ceil: function (v) { + return Math.ceil(v); + }, + 'floor/': function (v1, v2) { + return Math.floor(v1 / v2); + }, + pow: function (v1, v2) { + return Math.pow(v1, v2); + }, + binary: function (v) { + return v === 1 || v === 0; + }, + abs: function (v) { + return Math.abs(v); + }, + '==': function (v1, v2) { + return v1 === v2; + }, + even: function (v1) { + return v1 % 2 === 0; + } + }; + }, + {} + ], + 44: [ + function (require, module, exports) { + module.exports = function (JIFFClient) { + /** + * Initialize socket listeners and events + * @memberof module:jiff-client.JIFFClient + * @method + */ + JIFFClient.prototype.initSocket = function () { + var jiffClient = this; + + // set on('connect') handler once! + this.socket.on('connect', jiffClient.handlers.connected); + + // Store the id when server sends it back + this.socket.on('initialization', jiffClient.handlers.initialized); + + // Public keys were updated on the server, and it sent us the updates + jiffClient.socket.on('public_keys', function (msg, callback) { + callback(true); + + msg = JSON.parse(msg); + msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'public_keys', msg], 2); + + jiffClient.handlers.store_public_keys(msg.public_keys); + }); + + // Setup receiving matching shares + this.socket.on('share', function (msg, callback) { + callback(true); // send ack to server + + // parse message + var json_msg = JSON.parse(msg); + var sender_id = json_msg['party_id']; + + if (jiffClient.keymap[sender_id] != null) { + jiffClient.handlers.receive_share(json_msg); + } else { + if (jiffClient.messagesWaitingKeys[sender_id] == null) { + jiffClient.messagesWaitingKeys[sender_id] = []; + } + jiffClient.messagesWaitingKeys[sender_id].push({ label: 'share', msg: json_msg }); + } + }); + + this.socket.on('open', function (msg, callback) { + callback(true); // send ack to server + + // parse message + var json_msg = JSON.parse(msg); + var sender_id = json_msg['party_id']; + + if (jiffClient.keymap[sender_id] != null) { + jiffClient.handlers.receive_open(json_msg); + } else { + if (jiffClient.messagesWaitingKeys[sender_id] == null) { + jiffClient.messagesWaitingKeys[sender_id] = []; + } + jiffClient.messagesWaitingKeys[sender_id].push({ label: 'open', msg: json_msg }); + } + }); + + // handle custom messages + this.socket.on('custom', function (msg, callback) { + callback(true); // send ack to server + + var json_msg = JSON.parse(msg); + var sender_id = json_msg['party_id']; + var encrypted = json_msg['encrypted']; + + if (jiffClient.keymap[sender_id] != null || encrypted !== true) { + jiffClient.handlers.receive_custom(json_msg); + } else { + // key must not exist yet for sender_id, and encrypted must be true + if (jiffClient.messagesWaitingKeys[sender_id] == null) { + jiffClient.messagesWaitingKeys[sender_id] = []; + } + jiffClient.messagesWaitingKeys[sender_id].push({ label: 'custom', msg: json_msg }); + } + }); + + this.socket.on('crypto_provider', function (msg, callback) { + callback(true); // send ack to server + jiffClient.handlers.receive_crypto_provider(JSON.parse(msg)); + }); + + this.socket.on('error', function (msg) { + try { + msg = JSON.parse(msg); + jiffClient.handlers.error(msg['label'], msg['error']); + } catch (error) { + jiffClient.handlers.error('socket.io', msg); + } + }); + + this.socket.on('disconnect', function (reason) { + if (reason !== 'io client disconnect') { + // check that the reason is an error and not a user initiated disconnect + console.log('Disconnected!', jiffClient.id, reason); + } + + jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'disconnect', reason], -1); + }); + }; + + /** + * Executes all callbacks for which the wait condition has been satisfied. + * Remove all executed callbacks so that they would not be executed in the future. + * @memberof module:jiff-client.JIFFClient + * @method + */ + JIFFClient.prototype.execute_wait_callbacks = function () { + var copy_callbacks = this.wait_callbacks; + this.wait_callbacks = []; + for (var i = 0; i < copy_callbacks.length; i++) { + var wait = copy_callbacks[i]; + var parties = wait.parties; + var callback = wait.callback; + var initialization = wait.initialization; + + // Check if the parties to wait for are now known + var parties_satisfied = this.__initialized || !initialization; + for (var j = 0; j < parties.length; j++) { + var party_id = parties[j]; + if (this.keymap == null || this.keymap[party_id] == null) { + parties_satisfied = false; + break; + } + } + + if (parties_satisfied) { + callback(this); + } else { + this.wait_callbacks.push(wait); + } + } + }; + + /** + * Resolves all messages that were pending because their senders primary key was previously unknown. + * These messages are decrypted and verified and handled appropriatly before being removed from the wait queue. + * @memberof module:jiff-client.JIFFClient + * @method + */ + JIFFClient.prototype.resolve_messages_waiting_for_keys = function () { + for (var party_id in this.keymap) { + if (!this.keymap.hasOwnProperty(party_id)) { + continue; + } + + var messageQueue = this.messagesWaitingKeys[party_id]; + if (messageQueue == null) { + continue; + } + for (var i = 0; i < messageQueue.length; i++) { + var msg = messageQueue[i]; + if (msg.label === 'share') { + this.handlers.receive_share(msg.msg); + } else if (msg.label === 'open') { + this.handlers.receive_open(msg.msg); + } else if (msg.label === 'custom') { + this.handlers.receive_custom(msg.msg); + } else { + throw new Error('Error resolving pending message: unknown label ' + msg.label); + } + } + + this.messagesWaitingKeys[party_id] = null; + } + }; + }; + }, + {} + ], + 45: [ + function (require, module, exports) { + module.exports = function (jiffClient, __internal_socket) { + __internal_socket.safe_emit = function (label, msg) { + if (label === 'free') { + jiffClient.hooks.execute_array_hooks('afterOperation', [this.jiffClient, 'free', msg], 2); + return; + } + + __internal_socket.emit(label, msg); + }; + + __internal_socket.resend_mailbox = function () {}; + + __internal_socket.disconnect = function () { + jiffClient.hooks.execute_array_hooks('beforeOperation', [jiffClient, 'disconnect', {}], -1); + }; + + __internal_socket.safe_disconnect = function (free, callback) { + if (free) { + jiffClient.free(); + } + jiffClient.socket.disconnect(); + if (callback != null) { + callback(); + } + }; + + return __internal_socket; + }; + }, + {} + ], + 46: [ + function (require, module, exports) { + var io = require('socket.io-client'); + + var linked_list = require('../../common/linkedlist.js'); + var constants = require('../util/constants.js'); + + var defaultSocketOptions = { + reconnectionDelay: constants.reconnectionDelay, + reconnectionDelayMax: constants.reconnectionDelayMax, + randomizationFactor: constants.randomizationFactor, + autoConnect: false + }; + + /** + * A guarded socket with an attached mailbox. + * + * The socket uses the mailbox to store all outgoing messages, and removes them from the mailbox only when + * the server acknowledges their receipt. The socket resends mailbox upon re-connection. Extends {@link https://socket.io/docs/client-api/#Socket}. + * @see {@link module:jiff-client~JIFFClient#socket} + * @name GuardedSocket + * @alias GuardedSocket + * @constructor + */ + + function guardedSocket(jiffClient) { + jiffClient.options.socketOptions = Object.assign({}, defaultSocketOptions, jiffClient.options.socketOptions); + + // Create plain socket io object which we will wrap in this + var socket = io(jiffClient.hostname, jiffClient.options.socketOptions); + socket.old_disconnect = socket.disconnect; + socket.mailbox = linked_list(); // for outgoing messages + socket.empty_deferred = null; // gets resolved whenever the mailbox is empty + socket.jiffClient = jiffClient; + + // add functionality to socket + socket.safe_emit = safe_emit.bind(socket); + socket.resend_mailbox = resend_mailbox.bind(socket); + socket.disconnect = disconnect.bind(socket); + socket.safe_disconnect = safe_disconnect.bind(socket); + socket.is_empty = is_empty.bind(socket); + + return socket; + } - /** - * Called after the server approves initialization of this instance. - * Sets the instance id, the count of parties in the computation, and the public keys - * of initialized parties. - * @method - * @memberof handlers - */ - jiffClient.handlers.initialized = function (msg) { - jiffClient.__initialized = true; - jiffClient.initialization_counter = 0; - - msg = JSON.parse(msg); - msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'initialization', msg], 2); - - jiffClient.id = msg.party_id; - jiffClient.party_count = msg.party_count; - - // Now: (1) this party is connect (2) server (and other parties) know this public key - // Resend all pending messages - jiffClient.socket.resend_mailbox(); - - // store the received public keys and resolve wait callbacks - jiffClient.handlers.store_public_keys(msg.public_keys); - }; + /** + * Safe emit: stores message in the mailbox until acknowledgment is received, results in socket.emit(label, msg) call(s) + * @method safe_emit + * @memberof GuardedSocket + * @instance + * @param {string} label - the label given to the message + * @param {string} msg - the message to send + */ + var safe_emit = function (label, msg) { + // add message to mailbox + var mailbox_pointer = this.mailbox.add({ label: label, msg: msg }); + if (this.connected) { + var self = this; + // emit the message, if an acknowledgment is received, remove it from mailbox + this.emit(label, msg, function (status) { + if (status) { + self.mailbox.remove(mailbox_pointer); + if (this.is_empty() && self.empty_deferred != null) { + self.empty_deferred.resolve(); + } + + if (label === 'free') { + this.jiffClient.hooks.execute_array_hooks('afterOperation', [this.jiffClient, 'free', msg], 2); + } + } + }); + } + }; + + /** + * Re-sends all pending messages + * @method resend_mailbox + * @memberof GuardedSocket + * @instance + */ + var resend_mailbox = function () { + // Create a new mailbox, since the current mailbox will be resent and + // will contain new backups. + var old_mailbox = this.mailbox; + this.mailbox = linked_list(); + + // loop over all stored messages and emit them + var current_node = old_mailbox.head; + while (current_node != null) { + var label = current_node.object.label; + var msg = current_node.object.msg; + this.safe_emit(label, msg); + current_node = current_node.next; + } + }; + + /** + * Wraps socket.io regular disconnect with a call to a hook before disconnection + * @method disconnect + * @memberof GuardedSocket + * @instance + */ + var disconnect = function () { + this.jiffClient.hooks.execute_array_hooks('beforeOperation', [this.jiffClient, 'disconnect', {}], -1); + this.old_disconnect.apply(this, arguments); + }; + + /** + * Safe disconnect: disconnect only after all messages (including free) were acknowledged and + * all pending opens were resolved + * @method safe_disconnect + * @memberof GuardedSocket + * @instance + * @param {boolean} [free=false] - if true, a free message will be issued prior to disconnecting + * @param {function()} [callback] - given callback will be executed after safe disconnection is complete + */ + var safe_disconnect = function (free, callback) { + if (this.is_empty()) { + if (free) { + this.jiffClient.free(); + free = false; + } else { + this.disconnect(); + if (callback != null) { + callback(); + } + return; + } + } - /** - * Parse and store the given public keys - * @method - * @memberof handlers - * @param {object} keymap - maps party id to serialized public key. - */ - jiffClient.handlers.store_public_keys = function (keymap) { - var i; - for (i in keymap) { - if (keymap.hasOwnProperty(i) && jiffClient.keymap[i] == null) { - jiffClient.keymap[i] = jiffClient.hooks.parseKey(jiffClient, keymap[i]); - } - } + this.empty_deferred = new this.jiffClient.helpers.Deferred(); + this.empty_deferred.promise.then(this.safe_disconnect.bind(this, free, callback)); + }; + + /** + * Checks if the socket mailbox is empty (all communication was done and acknowledged), + * used in safe_disconnect + * @method is_empty + * @memberof GuardedSocket + * @instance + */ + var is_empty = function () { + return this.mailbox.head == null && this.jiffClient.counters.pending_opens === 0; + }; + + module.exports = guardedSocket; + }, + { '../../common/linkedlist.js': 52, '../util/constants.js': 47, 'socket.io-client': 278 } + ], + 47: [ + function (require, module, exports) { + module.exports = { + /** + * The default mod to be used in a jiff instance if a custom mod was not provided. + */ + gZp: 16777729, + + /** + * Socket connection timeouts + */ + reconnectionDelay: 25000, + reconnectionDelayMax: 27500, + randomizationFactor: 0.1, + + /** + * Maximum numbers of retries on failed initialization. + */ + maxInitializationRetries: 2 + }; + }, + {} + ], + 48: [ + function (require, module, exports) { + /** + * Encrypts and signs the given message. + * @ignore + * @memberof jiff.utils + * @param {number|string} message - the message to encrypt. + * @param {Uint8Array} encryption_public_key - ascii-armored public key to encrypt with. + * @param {Uint8Array} signing_private_key - the private key of the encrypting party to sign with. + * @returns {object} the signed cipher, includes two properties: 'cipher' and 'nonce'. + */ + exports.encrypt_and_sign = function (jiff, message, encryption_public_key, signing_private_key) { + var nonce = jiff.sodium_.randombytes_buf(jiff.sodium_.crypto_box_NONCEBYTES); + var cipher = jiff.sodium_.crypto_box_easy(message, nonce, encryption_public_key, signing_private_key); + + var result = { nonce: '[' + nonce.toString() + ']', cipher: '[' + cipher.toString() + ']' }; + return result; + }; + + /** + * Decrypts and checks the signature of the given cipher text. + * @ignore + * @memberof jiff.utils + * @param {object} cipher_text - the cipher text to decrypt, includes two properties: 'cipher' and 'nonce'. + * @param {Uint8Array} decryption_secret_key - the secret key to decrypt with. + * @param {Uint8Array} signing_public_key - ascii-armored public key to verify against signature. + * @returns {number|string} the decrypted message if the signature was correct, the decrypted message type should + * the type of operation, such that the returned value has the appropriate type and does + * not need any type modifications. + * @throws error if signature or nonce was forged/incorrect. + */ + exports.decrypt_and_sign = function (jiff, cipher_text, decryption_secret_key, signing_public_key) { + var nonce = new Uint8Array(JSON.parse(cipher_text.nonce)); + cipher_text = new Uint8Array(JSON.parse(cipher_text.cipher)); + + try { + return jiff.sodium_.crypto_box_open_easy(cipher_text, nonce, signing_public_key, decryption_secret_key, 'text'); + } catch (_) { + throw new Error('Bad signature or Bad nonce: Cipher: ' + cipher_text + '. DecSKey: ' + decryption_secret_key + '. SignPKey: ' + signing_public_key); + } + }; + }, + {} + ], + 49: [ + function (require, module, exports) { + var helpers = require('../../common/helpers.js'); + + /** + * Contains helper functions: these may be overriden by extensions to customize behavior + * @see {@link module:jiff-client~JIFFClient#helpers} + * @name helpers + * @alias helpers + * @namespace + */ + + module.exports = function (jiffClient) { + /** + * Polyfill for jQuery Deferred + * From https://developer.mozilla.org/en-US/docs/Mozilla/JavaScript_code_modules/Promise.jsm/Deferred + * @memberof helpers + * @constructor Deferred + */ + jiffClient.helpers.Deferred = function () { + /** + * A method to resolve the associate Promise with the value passed. + * @method resolve + * @memberof helpers.Deferred + * @instance + * @param {*} value - the value to resolve the promise with + */ + this.resolve = null; + + /** + * A method to reject the associated Promise with the value passed. + * If the promise is already settled it does nothing. + * @method reject + * @memberof helpers.Deferred + * @instance + * @param {*} reason - The reason for the rejection of the Promise. + * Generally its an Error object. If however a Promise is passed, then the Promise + * itself will be the reason for rejection no matter the state of the Promise. + */ + this.reject = null; + + /** + * A newly created Promise object. + * Initially in pending state. + * @memberof helpers.Deferred + * @member {Promise} promise + * @instance + */ + this.promise = new Promise( + function (resolve, reject) { + this.resolve = resolve; + this.reject = reject; + }.bind(this) + ); + + Object.freeze(this); + }; + + /** + * Correct Mod instead of javascript's remainder (%). + * @memberof helpers + * @method + * @param {number} x - the number. + * @param {number} y - the mod. + * @return {number} x mod y. + */ + jiffClient.helpers.mod = helpers.mod; + + /** + * Ceil of a number. + * @memberof helpers + * @method + * @param {number} x - the number to ceil. + * @return {number} ceil of x. + */ + jiffClient.helpers.ceil = Math.ceil; + + /** + * Floor of a number + * @memberof helpers + * @method + * @param {number} x - the number to floor. + * @return {number} floor of x. + */ + jiffClient.helpers.floor = Math.floor; + + /** + * Fast Exponentiation Mod + * @memberof helpers + * @method + * @param {number} a - the base number + * @param {number} b - the power + * @param {number} n - the mod + * @return {number} (base^pow) mod m + */ + jiffClient.helpers.pow_mod = function (a, b, n) { + a = jiffClient.helpers.mod(a, n); + var result = 1; + var x = a; + while (b > 0) { + var leastSignificantBit = jiffClient.helpers.mod(b, 2); + b = Math.floor(b / 2); + if (leastSignificantBit === 1) { + result = result * x; + result = jiffClient.helpers.mod(result, n); + } + x = x * x; + x = jiffClient.helpers.mod(x, n); + } + return result; + }; + + /** + * Extended Euclidean for finding inverses. + * @method + * @memberof helpers + * @param {number} a - the number to find inverse for. + * @param {number} b - the mod. + * @return {number[]} [inverse of a mod b, coefficient for a, coefficient for b]. + */ + jiffClient.helpers.extended_gcd = function (a, b) { + if (b === 0) { + return [1, 0, a]; + } + + var temp = jiffClient.helpers.extended_gcd(b, jiffClient.helpers.mod(a, b)); + var x = temp[0]; + var y = temp[1]; + var d = temp[2]; + return [y, x - y * Math.floor(a / b), d]; + }; + + /** + * Compute Log to a given base. + * @method + * @memberof helpers + * @param {number} value - the number to find log for. + * @param {number} [base=2] - the base (2 by default). + * @return {number} log(value) with the given base. + */ + jiffClient.helpers.bLog = function (value, base) { + if (base == null) { + base = 2; + } + return Math.log(value) / Math.log(base); + }; + + /** + * Check that two sorted arrays are equal. + * @method + * @memberof helpers + * @param {Array} arr1 - the first array. + * @param {Array} arr2 - the second array. + * @return {boolean} true if arr1 is equal to arr2, false otherwise. + */ + jiffClient.helpers.array_equals = function (arr1, arr2) { + if (arr1.length !== arr2.length) { + return false; + } + + for (var i = 0; i < arr1.length; i++) { + if (arr1[i] !== arr2[i]) { + return false; + } + } + + return true; + }; + + /** + * Check that two Zps are equal. Used to determine if shares can be computed on or not. + * @method + * @memberof helpers + * @param {SecretShare} s1 - the first share. + * @param {SecretShare} s2 - the second share. + * @return {boolean} true both shares have the same Zp, false otherwise. + */ + jiffClient.helpers.Zp_equals = function (s1, s2) { + return s1.Zp === s2.Zp; + }; + + /** + * Generate a random integer between 0 and max-1 [inclusive]. + * Modify this to change the source of randomness and how it is generated. + * @method + * @memberof helpers + * @param {number} max - the maximum number. + * @return {number} the random number. + */ + jiffClient.helpers.random = helpers.random; + + /** + * Get the party number from the given party_id, the number is used to compute/open shares. + * If party id was a number (regular party), that number is returned, + * If party id refers to the ith server, then party_count + i is returned (i > 0). + * @method + * @memberof helpers + * @param {number|string} party_id - the party id from which to compute the number. + * @return {number} the party number (> 0). + */ + jiffClient.helpers.get_party_number = helpers.get_party_number; + + /** + * Transforms the given number to an array of bits (numbers). + * Lower indices in the returned array corresponding to less significant bits. + * @memberof helpers + * @method + * @param {number} number - the number to transform to binary + * @param {length} [length=ceil(log2(number))] - if provided, then the given array will be padded with zeros to the length. + * @return {number[]} the array of bits. + */ + jiffClient.helpers.number_to_bits = helpers.number_to_bits; + + /** + * Transforms the given array of bits to a number. + * @memberof helpers + * @method + * @param {number[]} bits - the array of bits to compose as a number, starting from least to most significant bits. + * @param {number} [length = bits.length] - if provided, only the first 'length' bits will be used + * @return {number} the array of bits. + */ + jiffClient.helpers.bits_to_number = function (bits, length) { + if (length == null || length > bits.length) { + length = bits.length; + } + return parseInt(bits.slice(0, length).reverse().join(''), 2); + }; + + /** + * Checks if the given number is prime using AKS primality test + * @method + * @memberof helpers + * @param {number} p - the number to check + * @return {boolean} true if p is prime, false otherwise + */ + jiffClient.helpers.is_prime = function (p) { + // AKS Primality Test + + if (p === 2) { + return true; + } else if (p === 3) { + return true; + } else if (p % 2 === 0) { + return false; + } else if (p % 3 === 0) { + return false; + } + + var i = 5; + var n = 2; + while (i * i <= p) { + if (p % i === 0) { + return false; + } + i += n; + n = 6 - n; + } + + return true; + }; + + /** + * sorts an array of ids (in place) according to a consistent ordering + * @method + * @memberof helpers + * @param {array} ids - array of ids containing numbers or "s1" + */ + jiffClient.helpers.sort_ids = function (ids) { + if (ids.__jiff_sorted) { + return; + } + + ids.sort(function (e1, e2) { + if (e1 === e2) { + throw new Error('ids array has duplicated: ' + ids.toString()); + } + if (e1 === 's1') { + return 1; + } + if (e2 === 's1') { + return -1; + } + return e1 - e2; + }); + ids.__jiff_sorted = true; + }; + }; + }, + { '../../common/helpers.js': 51 } + ], + 50: [ + function (require, module, exports) { + /** + * Contains utility functions (higher order combinators) + * @see {@link module:jiff-client~JIFFClient#utils} + * @name utils + * @alias utils + * @namespace + */ + + module.exports = function (jiffClient) { + /** + * Create an array of secret shares and associated deferred. + * @method + * @memberof utils + * @param {number} count - number of secret shares + * @param {Array} holders - the parties that hold all the corresponding shares (must be sorted) + * @param {number} threshold - the min number of parties needed to reconstruct the secret + * @param {number} Zp - the mod under which this share was created + * @return {object} the secret share object containing the give value + * + */ + jiffClient.utils.many_secret_shares = function (count, holders, threshold, Zp) { + var deferreds = []; + var shares = []; + for (var i = 0; i < count; i++) { + var deferred = new jiffClient.helpers.Deferred(); + shares.push(new jiffClient.SecretShare(deferred.promise, holders, threshold, Zp)); + deferreds.push(deferred); + } + + return { shares: shares, deferreds: deferreds }; + }; + + /** + * Resolve the array of deferreds with the values of the given shares when ready, matched by index + * @method + * @memberof utils + * @param {Deferred[]} deferreds - the deferred to resolve + * @param {SecretShare[]} shares - the shares to resolve with + */ + jiffClient.utils.resolve_many_secrets = function (deferreds, shares) { + for (var i = 0; i < deferreds.length; i++) { + shares[i].wThen(deferreds[i].resolve); + } + }; + + /** + * Combines all the promises of the given share into a single promise that is resolved when + * all shares are resolved + * @method + * @methodof utils + * @param {SecretShare[]} shares - the shares whose promises should be joined + */ + jiffClient.utils.all_promises = function (shares) { + var promises = []; + for (var i = 0; i < shares.length; i++) { + promises.push(shares[i].value); + } + return Promise.all(promises); + }; + + /** + * A high level combinator for iteration of bit arrays + * It executes a round of (func) starting from index start to the length (supports backwards if start > length) excluding length + * Every round is blocked until the previous one finishes and the promise produced by it + * is resolved + * The final value is used to resolve deferred + * @method + * @memberof utils + * @param {!Deferred} deferred - the deferred to resolve with the final output + * @param {!number} start - the index to start from + * @param {!number} length - the index to stop at (excluding it) + * @param {?object} initial - an initial aggregator value + * @param {!function(number, object)} func - the aggregator function to apply to the elements in order, takes the index and the aggregator value so far + * @param {?function(object)} [promisify] - if initial is not null, this is called prior to starting combinator, to turn initial + * into a promise of the actually used initial value(in case it has to be resolved), + * defaults to promisifying a SecretShare with .wThen + * @param {?function(object)} [valufy] - applied to the final result to turn it into a value, which is then used to resolve deferred, + * defaults to getting value of a SecretShare + */ + jiffClient.utils.bit_combinator = function (deferred, start, length, initial, func, promisify, valufy) { + if (promisify == null) { + promisify = function (share) { + return { then: share.wThen.bind(share) }; + }; + } + + if (valufy == null) { + valufy = function (share) { + return share.value; + }; + } + + var next = start <= length ? 1 : -1; + var __bit_combinator = function (start, val) { + if (start === length) { + // done + deferred.resolve(valufy(val)); + return; + } + + // execute func once + val = func(start, val); + + // when done, do next iteration + promisify(val).then(function () { + __bit_combinator(start + next, val); + }); + }; + + // start combinator + if (initial == null) { + __bit_combinator(start, initial); + } else { + promisify(initial).then(function () { + __bit_combinator(start, initial); + }); + } + }; + }; + }, + {} + ], + 51: [ + function (require, module, exports) { + var crypto_; + if (typeof window === 'undefined') { + crypto_ = require('crypto'); + crypto_.__randomBytesWrapper = crypto_.randomBytes; + } else { + crypto_ = window.crypto || window.msCrypto; + crypto_.__randomBytesWrapper = function (bytesNeeded) { + var randomBytes = new Uint8Array(bytesNeeded); + crypto_.getRandomValues(randomBytes); + return randomBytes; + }; + } - // Resolve any pending messages that were received before the sender's public key was known - jiffClient.resolve_messages_waiting_for_keys(); + // Secure randomness via rejection sampling. + exports.random = function (max) { + // Use rejection sampling to get random value within bounds + // Generate random Uint8 values of 1 byte larger than the max parameter + // Reject if random is larger than quotient * max (remainder would cause biased distribution), then try again - // Resolve any pending waits that have satisfied conditions - jiffClient.execute_wait_callbacks(); + // Values up to 2^53 should be supported, but log2(2^49) === log2(2^49+1), so we lack the precision to easily + // determine how many bytes are required + if (max > 562949953421312) { + throw new RangeError('Max value should be smaller than or equal to 2^49'); + } - // Check if all keys have been received - if (jiffClient.keymap['s1'] == null) { - return; - } - for (i = 1; i <= jiffClient.party_count; i++) { - if (jiffClient.keymap[i] == null) { - return; - } - } + var bitsNeeded = Math.ceil(Math.log(max) / Math.log(2)); + var bytesNeeded = Math.ceil(bitsNeeded / 8); + var maxValue = Math.pow(256, bytesNeeded); - // all parties are connected; execute callback - if (jiffClient.__ready !== true && jiffClient.__initialized) { - jiffClient.__ready = true; - if (jiffClient.options.onConnect != null) { - jiffClient.options.onConnect(jiffClient); - } - } - }; -}; -},{}],16:[function(require,module,exports){ -// adds sharing related handlers -module.exports = function (jiffClient) { - /** - * Store the received share and resolves the corresponding - * deferred if needed. - * @method - * @memberof handlers - * @param {object} json_msg - the parsed json message as received. - */ - jiffClient.handlers.receive_share = function (json_msg) { - // Decrypt share - json_msg['share'] = jiffClient.hooks.decryptSign(jiffClient, json_msg['share'], jiffClient.secret_key, jiffClient.keymap[json_msg['party_id']]); - json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'share', json_msg], 2); - - var sender_id = json_msg['party_id']; - var op_id = json_msg['op_id']; - var share = json_msg['share']; - - // Call hook - share = jiffClient.hooks.execute_array_hooks('receiveShare', [jiffClient, sender_id, share], 2); - - // check if a deferred is set up (maybe the share was received early) - if (jiffClient.deferreds[op_id] == null) { - jiffClient.deferreds[op_id] = {}; - } - if (jiffClient.deferreds[op_id][sender_id] == null) { - // Share is received before deferred was setup, store it. - jiffClient.deferreds[op_id][sender_id] = new jiffClient.helpers.Deferred(); - } - - // Deferred is already setup, resolve it. - jiffClient.deferreds[op_id][sender_id].resolve(share); - }; + // Keep trying until we find a random value within bounds + while (true) { + // eslint-disable-line + var randomBytes = crypto_.__randomBytesWrapper(bytesNeeded); + var randomValue = 0; - /** - * Resolves the deferred corresponding to operation_id and sender_id. - * @method - * @memberof handlers - * @param {object} json_msg - the json message as received with the open event. - */ - jiffClient.handlers.receive_open = function (json_msg) { - // Decrypt share - if (json_msg['party_id'] !== jiffClient.id) { - json_msg['share'] = jiffClient.hooks.decryptSign(jiffClient, json_msg['share'], jiffClient.secret_key, jiffClient.keymap[json_msg['party_id']]); - json_msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'open', json_msg], 2); - } + for (var i = 0; i < bytesNeeded; i++) { + randomValue = randomValue * 256 + (randomBytes.readUInt8 ? randomBytes.readUInt8(i) : randomBytes[i]); + } - var sender_id = json_msg['party_id']; - var op_id = json_msg['op_id']; - var share = json_msg['share']; - var Zp = json_msg['Zp']; + // randomValue should be smaller than largest multiple of max within maxBytes + if (randomValue < maxValue - (maxValue % max)) { + return randomValue % max; + } + } + }; - // call hook - share = jiffClient.hooks.execute_array_hooks('receiveOpen', [jiffClient, sender_id, share, Zp], 2); + // actual mode + exports.mod = function (x, y) { + if (x < 0) { + return (x % y) + y; + } + return x % y; + }; - // Ensure deferred is setup - if (jiffClient.deferreds[op_id] == null) { - jiffClient.deferreds[op_id] = {}; + // get the party number from the given party_id, the number is used to compute/open shares + exports.get_party_number = function (party_id) { + if (typeof party_id === 'number') { + return party_id; + } + if (party_id.startsWith('s')) { + return -1 * parseInt(party_id.substring(1), 10); + } + return parseInt(party_id, 10); + }; + + // transform number to bit array + exports.number_to_bits = function (number, length) { + number = number.toString(2); + var bits = []; + for (var i = 0; i < number.length; i++) { + bits[i] = parseInt(number.charAt(number.length - 1 - i)); + } + while (length != null && bits.length < length) { + bits.push(0); + } + return bits; + }; + }, + { crypto: 136 } + ], + 52: [ + function (require, module, exports) { + /** Doubly linked list with add and remove functions and pointers to head and tail**/ + module.exports = function () { + // attributes: list.head and list.tail + // functions: list.add(object) (returns pointer), list.remove(pointer) + // list.head/list.tail/any element contains: + // next: pointer to next, + // previous: pointer to previous, + // object: stored object. + var list = { head: null, tail: null }; + // TODO rename this to pushTail || push + list.add = function (obj) { + var node = { object: obj, next: null, previous: null }; + if (list.head == null) { + list.head = node; + list.tail = node; + } else { + list.tail.next = node; + node.previous = list.tail; + list.tail = node; + } + return node; + }; + + list.pushHead = function (obj) { + list.head = { object: obj, next: list.head, previous: null }; + if (list.head.next != null) { + list.head.next.previous = list.head; + } else { + list.tail = list.head; + } + }; + + list.popHead = function () { + var result = list.head; + if (list.head != null) { + list.head = list.head.next; + if (list.head == null) { + list.tail = null; + } else { + list.head.previous = null; + } + } + return result; + }; + + // merges two linked lists and return a pointer to the head of the merged list + // the head will be the head of list and the tail the tail of l2 + list.extend = function (l2) { + if (list.head == null) { + return l2; + } + if (l2.head == null) { + return list; + } + list.tail.next = l2.head; + l2.head.previous = list.tail; + list.tail = l2.tail; + + return list; + }; + + list.remove = function (ptr) { + var prev = ptr.previous; + var next = ptr.next; + + if (prev == null && list.head !== ptr) { + return; + } else if (next == null && list.tail !== ptr) { + return; + } + + if (prev == null) { + // ptr is head (or both head and tail) + list.head = next; + if (list.head != null) { + list.head.previous = null; + } else { + list.tail = null; + } + } else if (next == null) { + // ptr is tail (and not head) + list.tail = prev; + prev.next = null; + } else { + // ptr is inside + prev.next = next; + next.previous = prev; + } + }; + list.slice = function (ptr) { + // remove all elements from head to ptr (including ptr). + if (ptr == null) { + return; + } + + /* CONSERVATIVE: make sure ptr is part of the list then remove */ + var current = list.head; + while (current != null) { + if (current === ptr) { + list.head = ptr.next; + if (list.head == null) { + list.tail = null; + } + + return; + } + current = current.next; + } + + /* MORE AGGRESSIVE VERSION: will be incorrect if ptr is not in the list */ + /* + list.head = ptr.next; + if (list.head == null) { + list.tail = null; } - if (jiffClient.deferreds[op_id].shares == null) { - jiffClient.deferreds[op_id].shares = []; + */ + }; + /* + list._debug_length = function () { + var l = 0; + var current = list.head; + while (current != null) { + current = current.next; + l++; } + return l; + }; + */ + return list; + }; + }, + {} + ], + 53: [ + function (require, module, exports) { + /** + * JIFF Client. + * + * Exposes the constructor for the {@link module:jiff-client~JIFFClient} class. + * + * In the browser, this adds `JIFFClient` as a global identifier. + * + * In the browser, this can be accessed via: + *

+           *   <script src="jiff-client.js"></script>
+           *   <script type="text/javascript">
+           *     var jiffClientInstance = new JIFFClient(hostname, computationId, options);
+           *   </script>
+           * 
+ * + * In node.js, this can be accessed via: + *

+           *   const JIFFClient = require('jiffClient');
+           *   const jiffClientInstance = new JIFFClient(hostname, computationId, options);
+           *
+           * 
+ * + * @module jiff-client + * @alias jiff-client + */ + + // browserify bundles this into our code bundle + var sodium = require('libsodium-wrappers'); + + // utils and helpers + var constants = require('./client/util/constants.js'); + var helpers = require('./client/util/helpers.js'); + var utils = require('./client/util/utils.js'); + var linkedList = require('./common/linkedlist.js'); + + // hooks + var Hooks = require('./client/arch/hooks.js'); + + // extensions management + var extensions = require('./client/arch/extensions.js'); + + // op ids and other counters + var counters = require('./client/arch/counters.js'); + + // socket and events + var guardedSocket = require('./client/socket/mailbox.js'); + var internalSocket = require('./client/socket/internal.js'); + var socketEvents = require('./client/socket/events.js'); + + // handlers for communication + var handlers = require('./client/handlers.js'); + + // secret shares + var SecretShareMetaClass = require('./client/share.js'); + var share_helpers = require('./client/shareHelpers.js'); + + // jiff client instance API + var api = require('./client/api.js'); + + // preprocessing + var preprocessingMap = require('./client/preprocessing/map.js'); + var preprocessingAPI = require('./client/preprocessing/api.js'); + var preprocessingDaemon = require('./client/preprocessing/daemon.js'); + + /** + * Creates a new jiff client instance. + * @class + * @name JIFFClient + * @param {!string} hostname - server hostname/ip and port. + * @param {!string} computation_id - the id of the computation of this instance. + * @param {?object} [options={}] - javascript object with additional options. + * all parameters are optional, However, private and public key must either be both provided or neither of them provided. +
+ {
+   "party_id": number,
+   "party_count": number,
+   "secret_key": Uint8Array to be used with libsodium-wrappers [(check Library Specs)]{@link https://download.libsodium.org/doc/public-key_cryptography/authenticated_encryption.html},
+   "public_key": Uint8Array to be used with libsodium-wrappers [(check Library Specs)]{@link https://download.libsodium.org/doc/public-key_cryptography/authenticated_encryption.html},
+   "public_keys": { 1: "Uint8Array PublicKey", 2: "Uint8Array PublicKey", ... },
+   "Zp": default mod to use (prime number),
+   "autoConnect": true/false,
+   "hooks": { 'check out hooks documentation' },
+   "listeners" : A map from custom tags to listeners (of type function(sender_id, message_string)) that handle custom messages with that tag.
+   "onConnect": function(jiff_instance),
+   "onError": function(label, error): called when errors occured in client code or during handling requests from this client at the server side
+                                      label is a string indicating where the error occured, and error is a string or an exception object.
+   "safemod": boolean (whether or not to check if provided Zp is prime, may be slow for big primes, defaults to false),
+   "crypto_provider": a boolean that flags whether to get beaver triplets and other preprocessing entities from the server (defaults to false),
+   "socketOptions": an object, passed directly to socket.io constructor,
+   "sodium": boolean, if false messages between clients will not be encrypted (useful for debugging),
+   "maxInitializationRetries": how many consecutive times to retry to initialize with the server if initialization fails, defaults to 2,
+   "preprocessingBatchSize": how many base level preprocessing tasks to execute in parallel.
+ }
+ 
+ * + * @example + * var JIFFClient = require('jiffClient'); // only for node.js + * // for the browser + * // build a jiff instance which will connect to a server running on the local machine + * var instance = new JIFFClient('http://localhost:8080', 'computation-1', {party_count: 2}); + */ + function JIFFClient(hostname, computation_id, options) { + var jiffClient = this; + options = Object.assign({}, options); + + /** + * The server hostname, ends with a slash, includes port and protocol (http/https). + * @type {!string} + */ + this.hostname = hostname.trim(); + if (!this.hostname.endsWith('/')) { + this.hostname = this.hostname + '/'; + } - // Accumulate received shares - jiffClient.deferreds[op_id].shares.push({value: share, sender_id: sender_id, Zp: Zp}); + /** + * Stores the computation id. + * @type {!string} + */ + this.computation_id = computation_id; + + /** + * Private. Do not use directly externally; use isReady() instead. + * @type {!boolean} + * @see {@link module:jiff-client~JIFFClient#isReady} + */ + this.__ready = false; + + /** + * Private. Do not use directly externally; use isInitialized() instead. + * @type {!boolean} + * @see {@link module:jiff-client~JIFFClient#isInitialized} + */ + this.__initialized = false; + + /** + * Returns whether this instance is capable of starting the computation. + * In other words, the public keys for all parties and servers are known, + * and this party successfully initialized with the server. + * @returns {!boolean} + */ + this.isReady = function () { + return this.__ready; + }; + + /** + * Returns whether this instance initialized successfully with the server. + * Note that this can be true even when isReady() returns false, in case where some other parties have not + * initialized yet! + * @returns {!boolean} + */ + this.isInitialized = function () { + return this.__initialized; + }; + + /** + * Helper functions [DO NOT MODIFY UNLESS YOU KNOW WHAT YOU ARE DOING]. + * @type {!helpers} + */ + this.helpers = {}; + helpers(this); + + /** + * Shallow copy of the options passed to the constructor. + * @type {!Object} + */ + + this.options = options; + + // Parse and verify options + options.maxInitializationRetries = options.maxInitializationRetries || constants.maxInitializationRetries; + if (typeof options.Zp === 'number' && options.safemod !== false) { + // big numbers are checked by extension + if (!this.helpers.is_prime(options.Zp)) { + throw new Error('Zp = ' + options.Zp + ' is not prime. Please use a prime number for the modulus or set safemod to false.'); + } + } - // Resolve when ready - if (jiffClient.deferreds[op_id].shares.length === jiffClient.deferreds[op_id].threshold) { - jiffClient.deferreds[op_id].deferred.resolve(); - } + /** + * The default Zp for this instance. + * @type {!number} + */ + this.Zp = options.Zp || constants.gZp; + + /** + * The id of this party. + * @type {number} + */ + this.id = options.party_id; + + /** + * Total party count in the computation, parties will take ids between 1 to party_count (inclusive). + * @type {number} + */ + this.party_count = options.party_count; + + /** + * sodium wrappers either imported via require (if in nodejs) or from the bundle (in the browser). + * This will be false if options.sodium is false. + * @see {@link https://www.npmjs.com/package/libsodium-wrappers} + * @type {?sodium} + */ + this.sodium_ = options.sodium !== false ? sodium : false; + + /** + * A map from party id to public key. Where key is the party id (number), and + * value is the public key, which by default follows libsodium's specs (Uint8Array). + * @see {@link https://download.libsodium.org/doc/public-key_cryptography/authenticated_encryption.html} + * @type {!object} + */ + this.keymap = Object.assign({}, options.public_keys); + + /** + * The secret key of this party, by default this follows libsodium's specs. + * @see {@link https://download.libsodium.org/doc/public-key_cryptography/authenticated_encryption.html} + * @type {?Uint8Array} + */ + this.secret_key = options.secret_key; + + /** + * The public key of this party, by default this follows libsodium's specs. + * @see {@link https://download.libsodium.org/doc/public-key_cryptography/authenticated_encryption.html} + * @type {?Uint8Array} + */ + this.public_key = options.public_key; + + /** + * Flags whether to use the server as a fallback for objects that were not pre-processed properly + * @type {!boolean} + */ + this.crypto_provider = options.crypto_provider === true; + + /** + * Stores messages that are received with a signature prior to acquiring the public keys of the sender. + * { 'party_id': [ { 'label': 'share/open/custom', } ] } + * @type {object} + */ + this.messagesWaitingKeys = {}; + + /** + * A map from tags to listeners (functions that take a sender_id and a string message). + * + * Stores listeners that are attached to this JIFF instance, listeners listen to custom messages sent by other parties. + * @type {!object} + */ + this.listeners = Object.assign({}, options.listeners); + + /** + * Stores custom messages that are received before their listeners are set. Messages are stored in order. + * { 'tag' => [ { "sender_id": , "message": }, ... ] } + * + * Once a listener has been set, the corresponding messages are sent to it in order. + * @type {!object} + */ + this.custom_messages_mailbox = {}; + + /** + * Stores all promises created within some barrier. + * @type {!object} + */ + this.barriers = {}; + + /** + * Stores the parties and callbacks for every .wait_for() registered by the user. + * @type {!Array} + */ + this.wait_callbacks = []; + + /** + * Counts how many times JIFF attempted to initialize with the server + * without success consecutively. + * @type {!number} + * + */ + this.initialization_counter = 0; + + /** + * Utility functions + * @type {!utils} + */ + this.utils = {}; + utils(this); + + /** + * An array containing the names (jiff-client-[name].js) of the extensions that are + * applied to this instance. + * @type {string[]} + */ + this.extensions = ['base']; + + /** + * Internal helpers for operations inside/on a share. Modify existing helpers or add more in your extensions + * to avoid having to re-write and duplicate the code for primitives. + * @type {!object} + */ + this.share_helpers = share_helpers; + + /** + * The constructor function used by JIFF to create a new share. This can be overloaded by extensions to create custom shares. + * Modifying this will modify how shares are generated in the BASE JIFF implementation. + * A share is a value/promise wrapped with a share object. + * A share also has methods for performing operations. + * @constructor + * @param {number|promise} value - the value of the share, or a promise to it. + * @param {Array} holders - the parties that hold all the corresponding shares (must be sorted). + * @param {number} threshold - the min number of parties needed to reconstruct the secret. + * @param {number} Zp - the mod under which this share was created. + * + * @example + * // A share whose value is 10: the secret is still unknown, 10 is only one share + * var share = new jiffClient.SecretShare(10, [1, 2, 3], 3, jiffClient.Zp); + * + * @example + * // A share whose value depends on some promise + * var share = new jiffClient.SecretShare(promise, [1, 2, 3, 4], 4, jiffClient.Zp); + */ + this.SecretShare = SecretShareMetaClass(this); + + /** + * A collection of useful protocols to be used during computation or preprocessing: extensions are encouraged to add useful + * common protocols here, under a sub namespace corresponding to the extension name. + * @type {!protocols} + */ + this.protocols = {}; + + /** + * A collection of useful protocols for manipulating bitwise shared numbers, and transforming them from and to regular numeric shares. + * @member {!bits} bits + * @memberof protocols + */ + this.protocols.bits = {}; + + /** + * Stores pre-computed values (beaver triples, random bits, etc) used to aid/speed up the main processes. + * @type {!object} + */ + this.preprocessing_table = {}; + + /** + * Sets batch size for base level preprocessing tasks + * @type {!Number} + */ + this.preprocessingBatchSize = options.preprocessingBatchSize || 10; + + /** + * maps all primitive operations to the other operations they are dependent on, until leaves are primitives for which preprocessing protocols are defined, + * this map is traversed during preprocessing to guide preprocessing of high level operations. Extensions should modify this map to reflect + * any required changes to preprocessing of modified primitives + * @type {!object} + */ + this.preprocessing_function_map = {}; + + /** + * Store the default preprocessing protocols for each type of preprocessing value + * @type {!object} + */ + this.default_preprocessing_protocols = {}; + + /** + * Stores currently executing preprocessing tasks. + * @type {!linkedlist} + */ + this.currentPreprocessingTasks = linkedList(); + + /** + * Callback to execute when preprocessing is done! + * @type {?function} + */ + this.preprocessingCallback = null; + + /** + * Used for logging/debugging + * @type {!Array} + */ + this.logs = []; + + /** + * A map from open operation id to the corresponding shares received for that open operation + * @type {!object} + */ + this.shares = {}; + + /** + * A map from some message operation id to a deferred resolved when that message is received. + * @type {!object} + */ + this.deferreds = {}; + + /** + * Store sharing and shares counter which keeps track of the count of + * sharing operations (share and open) and the total number of shares + * respectively (used to get a unique id for each share operation and + * share object). + * @type {!object} + */ + this.counters = {}; + + /** + * A prefix attached to all op_ids, can be changed using {@link module:jiff-client~JIFFClient#seed_ids} + * to guarantee uniqueness of auto generate ids in a user-side callback + * or event handler. + * @type {string} + * @see {@link module:jiff-client~JIFFClient#seed_ids} + */ + this.op_id_seed = ''; + + /** + * The hooks for this instance. + * Checkout the hooks documentation + * @type {!hooks} + */ + this.hooks = new Hooks(this); + + /** + * Contains handlers for communication events + * @type {!handlers} + */ + this.handlers = {}; + + // Add user facing API + api(this); + + // Preprocessing + preprocessingMap(this); + preprocessingAPI(this); + preprocessingDaemon(this); + + // set up counters for op_ids + counters(this); + + this.socketConnect = function (JIFFClientInstance) { + if (options.__internal_socket == null) { + /** + * Socket wrapper between this instance and the server, based on sockets.io + * @type {!GuardedSocket} + */ + JIFFClientInstance.socket = guardedSocket(JIFFClientInstance); + } else { + JIFFClientInstance.socket = internalSocket(JIFFClientInstance, options.__internal_socket); + } + + // set up socket event handlers + handlers(JIFFClientInstance); + + JIFFClientInstance.initSocket(); + + JIFFClientInstance.socket.connect(); + }; + + /** + * Connect to the server and starts listening. + */ + this.connect = function () { + // Ask socket to connect, which will automatically trigger a call to 'initialize()' when connection is established! + + // Wait to let JIFF know that we are connected until sodium is ready + // Trigger the onConnect call + var JIFFClientInstance = this; + if (jiffClient.sodium_ === false) { + JIFFClientInstance.socketConnect(JIFFClientInstance); + } else { + jiffClient.sodium_.ready.then(function () { + JIFFClientInstance.socketConnect(JIFFClientInstance); + }); + } + }; + + // Connect when all is done + if (options.autoConnect !== false) { + this.connect(); + } + } - // Clean up if done - if (jiffClient.deferreds[op_id] != null && jiffClient.deferreds[op_id].deferred === 'CLEAN' && jiffClient.deferreds[op_id].shares.length === jiffClient.deferreds[op_id].total) { - delete jiffClient.deferreds[op_id]; - } - } -}; -},{}],17:[function(require,module,exports){ -var linkedList = require('../../common/linkedlist.js'); - -module.exports = function (jiffClient) { - var isRunning = false; - var userCallbacks = []; - var preprocessingTasks = [linkedList()]; - - /** - * Checks if the given operation uses preprocessed values - * @method has_preprocessing - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string} op - name of the operation to check - * @return {boolean} true if the op uses preprocessing, false otherwise - */ - jiffClient.has_preprocessing = function (op) { - for (var i = 0; i < jiffClient.extensions.length; i++) { - if (jiffClient.preprocessing_function_map[jiffClient.extensions[i]][op] != null) { - return true; - } - } + // Add socket event handlers to prototype + socketEvents(JIFFClient); + + // Add extension management to prototype + extensions(JIFFClient); + + // export JIFFClient class + module.exports = JIFFClient; + }, + { + './client/api.js': 1, + './client/arch/counters.js': 9, + './client/arch/extensions.js': 10, + './client/arch/hooks.js': 11, + './client/handlers.js': 12, + './client/preprocessing/api.js': 17, + './client/preprocessing/daemon.js': 18, + './client/preprocessing/map.js': 20, + './client/share.js': 42, + './client/shareHelpers.js': 43, + './client/socket/events.js': 44, + './client/socket/internal.js': 45, + './client/socket/mailbox.js': 46, + './client/util/constants.js': 47, + './client/util/helpers.js': 49, + './client/util/utils.js': 50, + './common/linkedlist.js': 52, + 'libsodium-wrappers': 221 + } + ], + 54: [ + function (require, module, exports) { + module.exports = after; + + function after(count, callback, err_cb) { + var bail = false; + err_cb = err_cb || noop; + proxy.count = count; + + return count === 0 ? callback() : proxy; + + function proxy(err, result) { + if (proxy.count <= 0) { + throw new Error('after called too many times'); + } + --proxy.count; + + // after first error, rest are passed to err_cb + if (err) { + bail = true; + callback(err); + // future error callbacks will go to error handler + callback = err_cb; + } else if (proxy.count === 0 && !bail) { + callback(null, result); + } + } + } - return false; - }; + function noop() {} + }, + {} + ], + 55: [ + function (require, module, exports) { + /** + * An abstraction for slicing an arraybuffer even when + * ArrayBuffer.prototype.slice is not supported + * + * @api public + */ + + module.exports = function (arraybuffer, start, end) { + var bytes = arraybuffer.byteLength; + start = start || 0; + end = end || bytes; + + if (arraybuffer.slice) { + return arraybuffer.slice(start, end); + } - /** - * Get a preprocessed share/value by associated op_id. If value does not exist - * Fallback to some user specified way for creating it - * @method get_preprocessing - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string} op_id - the op_id associated with the preprocessed value/share - * @return {object} the preprocessed share(s) - */ - jiffClient.get_preprocessing = function (op_id) { - var values = jiffClient.preprocessing_table[op_id]; - if (values != null) { - return values; - } - if (jiffClient.crypto_provider === true) { - return null; - } - throw new Error('No preprocessed value(s) that correspond to the op_id "' + op_id + '"'); - }; + if (start < 0) { + start += bytes; + } + if (end < 0) { + end += bytes; + } + if (end > bytes) { + end = bytes; + } - /** - * Store a pair of op_id and associated pre-processed value/share - * The value/share can be accessed later during the computation through jiffClient.get_preprocessing(op_id) - * @method store_preprocessing - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string} op_id - the op_id associated with the preprocessed value/share - * @param {SecretShare} share - the share/value to store - */ - jiffClient.store_preprocessing = function (op_id, share) { - if (share != null) { - jiffClient.preprocessing_table[op_id] = share; - } - }; + if (start >= bytes || start >= end || bytes === 0) { + return new ArrayBuffer(0); + } - /** - * Generate values used for JIFF operations in advance of the computation. - * - * Calling this function does not begin preprocessing, it just creates a preprocessing task. After you created - * your desired tasks, you can ask JIFF to execute them via {@link module:jiff-client~JIFFClient#executePreprocessing}. - * - * @method preprocessing - * @memberof module:jiff-client~JIFFClient - * @instance - * @param {string} dependent_op - name of the operation that will later use the pre_processed values - * @param {Number} [count=1] - number of times the protocol should be performed, number of values that will be generated - * @param {Object} [protocols=defaults] - a mapping from base preprocessing elements ('beaver', 'bits', 'sampling') to functions that can pre-process them - * the function must implement the same interface as the JIFF provided protocols (e.g. jiffClient.protocols.generate_beaver_bgw), - * missing mappings indicate that JIFF must use the default protocols - * @param {Number} [threshold=receivers_list.length] - the threshold of the preprocessed shares - * @param {Array} [receivers_list=all_parties] - the parties that will receive the preprocsssed shares - * @param {Array} [compute_list=all_parties] - the parties that will compute the preprocsssed shares - * @param {Number} [Zp=jiffClient.Zp] - the Zp of the preprocessed shares - * @param {Array} [id_list=auto_gen()] - array of ids to be used sequentially to identify the pre_processed values - * @param {Object} [params={}] - any additional protocol-specific parameters - * @return {promise} a promise that is resolved when preprocessing is completed, null if this is called by a party that is neither a compute nor receiver party - * @see {@link module:jiff-client~JIFFClient#executePreprocessing} - */ - jiffClient.preprocessing = function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, id_list, params) { - // defaults! - if (receivers_list == null) { - receivers_list = []; - for (var p = 1; p <= jiffClient.party_count; p++) { - receivers_list.push(p); - } - } else { - jiffClient.helpers.sort_ids(receivers_list); - } - if (compute_list == null) { - compute_list = []; - for (var c = 1; c <= jiffClient.party_count; c++) { - compute_list.push(c); - } - } else { - jiffClient.helpers.sort_ids(compute_list); - } + var abv = new Uint8Array(arraybuffer); + var result = new Uint8Array(end - start); + for (var i = start, ii = 0; i < end; i++, ii++) { + result[ii] = abv[i]; + } + return result.buffer; + }; + }, + {} + ], + 56: [ + function (require, module, exports) { + var asn1 = exports; + + asn1.bignum = require('bn.js'); + + asn1.define = require('./asn1/api').define; + asn1.base = require('./asn1/base'); + asn1.constants = require('./asn1/constants'); + asn1.decoders = require('./asn1/decoders'); + asn1.encoders = require('./asn1/encoders'); + }, + { './asn1/api': 57, './asn1/base': 59, './asn1/constants': 63, './asn1/decoders': 65, './asn1/encoders': 68, 'bn.js': 70 } + ], + 57: [ + function (require, module, exports) { + var asn1 = require('../asn1'); + var inherits = require('inherits'); - // not a receiver nor a sender - if (receivers_list.indexOf(jiffClient.id) === -1 && compute_list.indexOf(jiffClient.id) === -1) { - return null; - } + var api = exports; - // more defaults - if (Zp == null) { - Zp = jiffClient.Zp; - } - if (threshold == null) { - threshold = receivers_list.length; - } - if (protocols == null) { - protocols = jiffClient.default_preprocessing_protocols; - } + api.define = function define(name, body) { + return new Entity(name, body); + }; - // actual preprocessing - if (count == null || count <= 0) { - count = 1; - } - if (params == null) { - params = {}; - } - if (params['namespace'] == null) { - params['namespace'] = jiffClient.extensions[jiffClient.extensions.length - 1]; - } + function Entity(name, body) { + this.name = name; + this.body = body; - // Create preprocessing tasks - var task = { - dependent_op : dependent_op, - count : count, - threshold : threshold, - receivers_list : receivers_list, - compute_list : compute_list, - Zp : Zp, - id_list : id_list, - id : null, - params : params, - protocols : protocols, - deferred: new jiffClient.helpers.Deferred() - }; - - preprocessingTasks[preprocessingTasks.length - 1].add(task); - - return task.deferred.promise; - }; + this.decoders = {}; + this.encoders = {}; + } - /** - * Ask JIFF to start executing preprocessing for tasks previously added by {@link module:jiff-client~JIFFClient#preprocessing}. - * - * Calls the provided callback when the preprocessing tasks are all done. - * - * @method executePreprocessing - * @memberof module:jiff-client~JIFFClient - * @instance - * @param callback {!Function} - the callback to execute when preprocessing is finished. - * {@link module:jiff-client~JIFFClient#preprocessing} - */ - jiffClient.executePreprocessing = function (callback) { - userCallbacks.push(callback); - preprocessingTasks.push(linkedList()); - - if (!isRunning) { - __executePreprocessing(); - } - }; + Entity.prototype._createNamed = function createNamed(base) { + var named; + try { + named = require('vm').runInThisContext('(function ' + this.name + '(entity) {\n' + ' this._initNamed(entity);\n' + '})'); + } catch (e) { + named = function (entity) { + this._initNamed(entity); + }; + } + inherits(named, base); + named.prototype._initNamed = function initnamed(entity) { + base.call(this, entity); + }; + + return new named(this); + }; + + Entity.prototype._getDecoder = function _getDecoder(enc) { + enc = enc || 'der'; + // Lazily create decoder + if (!this.decoders.hasOwnProperty(enc)) this.decoders[enc] = this._createNamed(asn1.decoders[enc]); + return this.decoders[enc]; + }; + + Entity.prototype.decode = function decode(data, enc, options) { + return this._getDecoder(enc).decode(data, options); + }; + + Entity.prototype._getEncoder = function _getEncoder(enc) { + enc = enc || 'der'; + // Lazily create encoder + if (!this.encoders.hasOwnProperty(enc)) this.encoders[enc] = this._createNamed(asn1.encoders[enc]); + return this.encoders[enc]; + }; + + Entity.prototype.encode = function encode(data, enc, /* internal */ reporter) { + return this._getEncoder(enc).encode(data, reporter); + }; + }, + { '../asn1': 56, inherits: 218, vm: 296 } + ], + 58: [ + function (require, module, exports) { + var inherits = require('inherits'); + var Reporter = require('../base').Reporter; + var Buffer = require('buffer').Buffer; + + function DecoderBuffer(base, options) { + Reporter.call(this, options); + if (!Buffer.isBuffer(base)) { + this.error('Input not Buffer'); + return; + } - // called only when preprocessing can run RIGHT NOW - var __executePreprocessing = function () { - isRunning = true; + this.base = base; + this.offset = 0; + this.length = base.length; + } + inherits(DecoderBuffer, Reporter); + exports.DecoderBuffer = DecoderBuffer; + + DecoderBuffer.prototype.save = function save() { + return { offset: this.offset, reporter: Reporter.prototype.save.call(this) }; + }; + + DecoderBuffer.prototype.restore = function restore(save) { + // Return skipped data + var res = new DecoderBuffer(this.base); + res.offset = save.offset; + res.length = this.offset; + + this.offset = save.offset; + Reporter.prototype.restore.call(this, save.reporter); + + return res; + }; + + DecoderBuffer.prototype.isEmpty = function isEmpty() { + return this.offset === this.length; + }; + + DecoderBuffer.prototype.readUInt8 = function readUInt8(fail) { + if (this.offset + 1 <= this.length) return this.base.readUInt8(this.offset++, true); + else return this.error(fail || 'DecoderBuffer overrun'); + }; + + DecoderBuffer.prototype.skip = function skip(bytes, fail) { + if (!(this.offset + bytes <= this.length)) return this.error(fail || 'DecoderBuffer overrun'); + + var res = new DecoderBuffer(this.base); + + // Share reporter state + res._reporterState = this._reporterState; + + res.offset = this.offset; + res.length = this.offset + bytes; + this.offset += bytes; + return res; + }; + + DecoderBuffer.prototype.raw = function raw(save) { + return this.base.slice(save ? save.offset : this.offset, this.length); + }; + + function EncoderBuffer(value, reporter) { + if (Array.isArray(value)) { + this.length = 0; + this.value = value.map(function (item) { + if (!(item instanceof EncoderBuffer)) item = new EncoderBuffer(item, reporter); + this.length += item.length; + return item; + }, this); + } else if (typeof value === 'number') { + if (!(0 <= value && value <= 0xff)) return reporter.error('non-byte EncoderBuffer value'); + this.value = value; + this.length = 1; + } else if (typeof value === 'string') { + this.value = value; + this.length = Buffer.byteLength(value); + } else if (Buffer.isBuffer(value)) { + this.value = value; + this.length = value.length; + } else { + return reporter.error('Unsupported type: ' + typeof value); + } + } + exports.EncoderBuffer = EncoderBuffer; + + EncoderBuffer.prototype.join = function join(out, offset) { + if (!out) out = new Buffer(this.length); + if (!offset) offset = 0; + + if (this.length === 0) return out; + + if (Array.isArray(this.value)) { + this.value.forEach(function (item) { + item.join(out, offset); + offset += item.length; + }); + } else { + if (typeof this.value === 'number') out[offset] = this.value; + else if (typeof this.value === 'string') out.write(this.value, offset); + else if (Buffer.isBuffer(this.value)) this.value.copy(out, offset); + offset += this.length; + } - jiffClient.currentPreprocessingTasks = preprocessingTasks.shift(); - var currentCallback = userCallbacks.shift(); + return out; + }; + }, + { '../base': 59, buffer: 124, inherits: 218 } + ], + 59: [ + function (require, module, exports) { + var base = exports; + + base.Reporter = require('./reporter').Reporter; + base.DecoderBuffer = require('./buffer').DecoderBuffer; + base.EncoderBuffer = require('./buffer').EncoderBuffer; + base.Node = require('./node'); + }, + { './buffer': 58, './node': 60, './reporter': 61 } + ], + 60: [ + function (require, module, exports) { + var Reporter = require('../base').Reporter; + var EncoderBuffer = require('../base').EncoderBuffer; + var DecoderBuffer = require('../base').DecoderBuffer; + var assert = require('minimalistic-assert'); + + // Supported tags + var tags = [ + 'seq', + 'seqof', + 'set', + 'setof', + 'objid', + 'bool', + 'gentime', + 'utctime', + 'null_', + 'enum', + 'int', + 'objDesc', + 'bitstr', + 'bmpstr', + 'charstr', + 'genstr', + 'graphstr', + 'ia5str', + 'iso646str', + 'numstr', + 'octstr', + 'printstr', + 't61str', + 'unistr', + 'utf8str', + 'videostr' + ]; + + // Public methods list + var methods = ['key', 'obj', 'use', 'optional', 'explicit', 'implicit', 'def', 'choice', 'any', 'contains'].concat(tags); + + // Overrided methods list + var overrided = [ + '_peekTag', + '_decodeTag', + '_use', + '_decodeStr', + '_decodeObjid', + '_decodeTime', + '_decodeNull', + '_decodeInt', + '_decodeBool', + '_decodeList', + + '_encodeComposite', + '_encodeStr', + '_encodeObjid', + '_encodeTime', + '_encodeNull', + '_encodeInt', + '_encodeBool' + ]; + + function Node(enc, parent) { + var state = {}; + this._baseState = state; + + state.enc = enc; + + state.parent = parent || null; + state.children = null; + + // State + state.tag = null; + state.args = null; + state.reverseArgs = null; + state.choice = null; + state.optional = false; + state.any = false; + state.obj = false; + state.use = null; + state.useDecoder = null; + state.key = null; + state['default'] = null; + state.explicit = null; + state.implicit = null; + state.contains = null; + + // Should create new instance on each method + if (!state.parent) { + state.children = []; + this._wrap(); + } + } + module.exports = Node; + + var stateProps = ['enc', 'parent', 'children', 'tag', 'args', 'reverseArgs', 'choice', 'optional', 'any', 'obj', 'use', 'alteredUse', 'key', 'default', 'explicit', 'implicit', 'contains']; + + Node.prototype.clone = function clone() { + var state = this._baseState; + var cstate = {}; + stateProps.forEach(function (prop) { + cstate[prop] = state[prop]; + }); + var res = new this.constructor(cstate.parent); + res._baseState = cstate; + return res; + }; + + Node.prototype._wrap = function wrap() { + var state = this._baseState; + methods.forEach(function (method) { + this[method] = function _wrappedMethod() { + var clone = new this.constructor(this); + state.children.push(clone); + return clone[method].apply(clone, arguments); + }; + }, this); + }; + + Node.prototype._init = function init(body) { + var state = this._baseState; + + assert(state.parent === null); + body.call(this); + + // Filter children + state.children = state.children.filter(function (child) { + return child._baseState.parent === this; + }, this); + assert.equal(state.children.length, 1, 'Root node can have only one child'); + }; + + Node.prototype._useArgs = function useArgs(args) { + var state = this._baseState; + + // Filter children and args + var children = args.filter(function (arg) { + return arg instanceof this.constructor; + }, this); + args = args.filter(function (arg) { + return !(arg instanceof this.constructor); + }, this); + + if (children.length !== 0) { + assert(state.children === null); + state.children = children; + + // Replace parent to maintain backward link + children.forEach(function (child) { + child._baseState.parent = this; + }, this); + } + if (args.length !== 0) { + assert(state.args === null); + state.args = args; + state.reverseArgs = args.map(function (arg) { + if (typeof arg !== 'object' || arg.constructor !== Object) return arg; + + var res = {}; + Object.keys(arg).forEach(function (key) { + if (key == (key | 0)) key |= 0; + var value = arg[key]; + res[value] = key; + }); + return res; + }); + } + }; - jiffClient.preprocessingCallback = function () { - if (currentCallback != null) { - currentCallback.apply(null, currentCallback); - } + // + // Overrided methods + // - if (userCallbacks.length > 0) { - __executePreprocessing(); - } else { - isRunning = false; - } - }; + overrided.forEach(function (method) { + Node.prototype[method] = function _overrided() { + var state = this._baseState; + throw new Error(method + ' not implemented for encoding: ' + state.enc); + }; + }); - jiffClient.preprocessingDaemon(); - }; -}; - -},{"../../common/linkedlist.js":52}],18:[function(require,module,exports){ -module.exports = function (jiffClient) { - var linkedList = require('../../common/linkedlist.js'); - var currentBatchLoad = 0; - var suspendedTasks = 0; - - var getFirstTask = function (task) { - if (task.count > 1) { - var remainingTasks = Object.assign({}, task); - - var deferred1 = new jiffClient.helpers.Deferred(); - var deferred2 = new jiffClient.helpers.Deferred(); - Promise.all([deferred1.promise, deferred2.promise]).then(task.deferred.resolve); - task.deferred = deferred1; - remainingTasks.deferred = deferred2; - - remainingTasks.count--; - task.count = 1; - if (task.id_list != null) { - task.id = remainingTasks.id_list.shift(); - task.id_list = null; - } - jiffClient.currentPreprocessingTasks.pushHead(remainingTasks); - } - if (task.id_list != null) { - task.id = task.id_list[0]; - task.id_list = null; - } - return task; - }; + // + // Public methods + // - var checkIfDone = function () { - if (currentBatchLoad === 0 && suspendedTasks === 0) { - var callback = jiffClient.preprocessingCallback; - jiffClient.preprocessingCallback = null; - callback(jiffClient); - } - }; + tags.forEach(function (tag) { + Node.prototype[tag] = function _tagMethod() { + var state = this._baseState; + var args = Array.prototype.slice.call(arguments); - var buildID = function (task) { - // Two kinds of operations: one that relies on different sets of senders and receivers, and one that has a set of holders - if (task.dependent_op === 'open' || task.dependent_op === 'bits.open') { // TODO: make this part of the description in table - var open_parties = task.params['open_parties'] != null ? task.params['open_parties'] : task.receivers_list; - task.id = jiffClient.counters.gen_op_id2_preprocessing(task.dependent_op, open_parties, task.receivers_list); - } else { - task.id = jiffClient.counters.gen_op_id_preprocessing(task.dependent_op, task.receivers_list); - } - }; + assert(state.tag === null); + state.tag = tag; - var taskIsExecutable = function (task) { - // if the protocol name is in the map, it can be directly executed - var namespace = find_closest_namespace(task.dependent_op, task.params['namespace']); - return (namespace == null); - }; + this._useArgs(args); - var find_closest_namespace = function (op, starting_namespace) { - var namespace_index = jiffClient.extensions.indexOf(starting_namespace); - while (namespace_index >= 0) { - var namespace = jiffClient.extensions[namespace_index]; - if (jiffClient.preprocessing_function_map[namespace] != null && jiffClient.preprocessing_function_map[namespace][op] != null) { - return namespace; - } - namespace_index--; - } + return this; + }; + }); - return null; - }; + Node.prototype.use = function use(item) { + assert(item); + var state = this._baseState; - // execute a task and handle it upon completion - var executeTask = function (task) { - currentBatchLoad++; + assert(state.use === null); + state.use = item; - var _params = Object.assign({}, task.params); - _params.output_op_id = task.id; + return this; + }; - var protocol = task.protocols[task.dependent_op] || jiffClient.default_preprocessing_protocols[task.dependent_op]; - var result = protocol(task.threshold, task.receivers_list, task.compute_list, task.Zp, _params, task.protocols); + Node.prototype.optional = function optional() { + var state = this._baseState; - if (result.promise == null || result.promise.then == null) { - taskFinished(task, result); - } else { - result.promise.then(taskFinished.bind(null, task, result)); - } - }; - var taskFinished = function (task, result) { - currentBatchLoad--; + state.optional = true; - if (task.receivers_list.indexOf(jiffClient.id) > -1) { - jiffClient.store_preprocessing(task.id, result.share); - } - task.deferred.resolve(); - jiffClient.preprocessingDaemon(); - }; + return this; + }; - // expand task by one level and replace the node in the task list - var expandTask = function (task) { - // copy of params - var _params = Object.assign({}, task.params); + Node.prototype.def = function def(val) { + var state = this._baseState; - // Recursively follow jiffClient.preprocessing_function_map - // to figure out the sub-components/nested primitives of the given operation - // and pre-process those with the right op_ids. + assert(state['default'] === null); + state['default'] = val; + state.optional = true; - // ID should never be null - var namespace = find_closest_namespace(task.dependent_op, _params['namespace']); - var preprocessing_dependencies = jiffClient.preprocessing_function_map[namespace][task.dependent_op]; + return this; + }; - if (typeof(preprocessing_dependencies) === 'function') { - preprocessing_dependencies = preprocessing_dependencies(task.dependent_op, task.count, task.protocols, task.threshold, task.receivers_list, task.compute_list, task.Zp, task.id, _params, task, jiffClient); - } + Node.prototype.explicit = function explicit(num) { + var state = this._baseState; - var newTasks = linkedList(); - var deferredChain = []; - // build linked list of new dependencies, afterwords merge them with current tasks list - for (var k = 0; k < preprocessing_dependencies.length; k++) { - var dependency = preprocessing_dependencies[k]; - var next_op = dependency['op']; - - // copy both the originally given extra_params and the extra params of the dependency and merge them - // together, dependency params overwrite duplicate keys. - // If params are ever needed in non-leaf operations, this must be changed to accommodate - var extra_params = Object.assign({}, _params, dependency['params']); - extra_params['namespace'] = dependency['namespace'] != null ? dependency['namespace'] : 'base'; - if (dependency.handler != null) { - extra_params = dependency.handler(task.threshold, task.receivers_list, task.compute_list, task.Zp, task.id, extra_params, task, jiffClient); - } - if (extra_params.ignore === true) { - continue; - } + assert(state.explicit === null && state.implicit === null); + state.explicit = num; - // compose ids similar to how the actual operation is implemented - var next_id_list = []; - var next_count = dependency['count']; - - if (next_count == null) { - next_count = 1; - next_id_list[0] = dependency['absolute_op_id'] || (task.id + dependency['op_id']); - } else { - next_count = next_count(task.threshold, task.receivers_list, task.compute_list, task.Zp, task.id, extra_params); - for (var j = 0; j < next_count; j++) { - next_id_list.push(dependency['absolute_op_id'] || (task.id + dependency['op_id'] + j)); - } - } + return this; + }; - var nextTask = { - dependent_op : next_op, - count : next_count, - threshold : dependency['threshold'] || task.threshold, - receivers_list : dependency['receivers_list'] || task.receivers_list, - compute_list : dependency['compute_list'] || task.compute_list, - Zp : dependency['Zp'] || task.Zp, - id_list : next_id_list, - id : null, - params : extra_params, - protocols : task.protocols, - deferred: new jiffClient.helpers.Deferred() - }; - - deferredChain.push(nextTask.deferred.promise); - if (dependency.requires != null) { - nextTask.wait = true; - var required_ops = []; - for (var r = 0; r < dependency.requires.length; r++) { - if (dependency.requires[r] >= k) { - throw new Error('Preprocessing dependency "' + next_op + '" in preprocessingMap for "' + task.dependent_op - + '" at namespace "' + namespace + '" cannot require subsequent dependency ' + dependency.requires[r]); - } - required_ops.push(deferredChain[dependency.requires[r]]); - } + Node.prototype.implicit = function implicit(num) { + var state = this._baseState; - Promise.all(required_ops).then(function (nextTask) { - delete nextTask['wait']; - jiffClient.preprocessingDaemon(); - }.bind(null, nextTask)); - - // add waiting task to the tail of the queue - // jiffClient.currentPreprocessingTasks.add(nextTask); - newTasks.add(nextTask); - } else { - // non-waiting tasks are added to the head of the queue - newTasks.add(nextTask); - } - } + assert(state.explicit === null && state.implicit === null); + state.implicit = num; - Promise.all(deferredChain).then(task.deferred.resolve); - jiffClient.currentPreprocessingTasks = newTasks.extend(jiffClient.currentPreprocessingTasks); - }; + return this; + }; - /** - * Preprocessing Daemon that executes all currently scheduled preprocessing tasks (entries in jiffClient.currentPreprocessingTasks array) in order. - * @method preprocessingDaemon - * @memberof module:jiff-client~JIFFClient - * @instance - */ - jiffClient.preprocessingDaemon = function () { - while (currentBatchLoad < jiffClient.preprocessingBatchSize) { - var task = jiffClient.currentPreprocessingTasks.popHead(); - - if (task == null) { - checkIfDone(); - return; - } + Node.prototype.obj = function obj() { + var state = this._baseState; + var args = Array.prototype.slice.call(arguments); - if (task.object.wait) { - jiffClient.currentPreprocessingTasks.pushHead(task.object); - break; - } + state.obj = true; - task = getFirstTask(task.object); - if (task.id == null) { - buildID(task); - } + if (args.length !== 0) this._useArgs(args); - // check if task is executable or no - if (taskIsExecutable(task)) { - executeTask(task); // co-recursively calls preprocessingDaemon() - } else { - //expand single task - expandTask(task); - } - } - }; -}; - -},{"../../common/linkedlist.js":52}],19:[function(require,module,exports){ -// internal functions for use in preprocessing function map -module.exports = { - bits_count: function (threshold, receivers_list, compute_list, Zp, op_id, params) { - var bitLength = params.bitLength; - if (bitLength == null) { - bitLength = Zp.toString(2).length; - } - return bitLength; - }, - constant_bits_count: function () { - return module.exports.bits_count.apply(null, arguments) - 1; - }, - dynamic_bits_cmult: function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params) { - // constant bit length - var constantBits = Zp.toString(2).length; - if (params.constantBits != null) { - constantBits = params.constantBits; - } - // secret bit length - var bitLength = params.bitLength; - if (bitLength == null) { - bitLength = Zp.toString(2).length; - } - // for every bit from constant, pre-process for one bits.sadd of the right size - var ops = []; - for (var i = 0; i < constantBits; i++) { - var accLength = i === 0 ? 1 : (bitLength + i); - ops.push({ op: 'bits.sadd', op_id: ':bits.sadd:' + i, params: {bitLengthLeft: accLength, bitLengthRight: bitLength + i}}); - } - return ops; - }, - dynamic_bits_smult: function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params) { - var bitLength = params.bitLength; - if (bitLength == null) { - bitLength = Zp.toString(2).length; - } + return this; + }; - var left = params.bitLengthLeft; - var right = params.bitLengthRight; - left = left != null ? left : bitLength; - right = right != null ? right : bitLength; - var max = Math.max(left, right); - var min = Math.max(left, right); - - var ops = []; - for (var i = 0; i < min; i++) { - for (var j = 0; j < max + i; j++) { - ops.push({ op: 'if_else', op_id: ':if_else:' + i + ':' + j }); - } - var accLength = i === 0 ? min : (max + i); - ops.push({ op: 'bits.sadd', op_id: ':bits.sadd:'+i, params: {bitLengthLeft: accLength, bitLengthRight: max + i}}); - } - return ops; - }, - choice_bits_count: function (choice, offset) { - if (offset == null) { - offset = 0; - } - return function (threshold, receivers_list, compute_list, Zp, op_id, params) { - var bitLength = params.bitLength; - if (bitLength == null) { - bitLength = Zp.toString(2).length; - } + Node.prototype.key = function key(newKey) { + var state = this._baseState; - var left = params.bitLengthLeft; - var right = params.bitLengthRight; - left = left != null ? left : bitLength; - right = right != null ? right : bitLength; - - return choice(left, right) + offset; - }; - }, - decomposition_ifelse_count: function (threshold, receivers_list, compute_list, Zp, op_id, params) { - return Zp.toString(2).length; - }, - dynamic_bits_sdiv: function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params) { - var bitLength = params.bitLength; - if (bitLength == null) { - bitLength = Zp.toString(2).length; - } + assert(state.key === null); + state.key = newKey; - var left = params.bitLengthLeft; - var right = params.bitLengthRight; - left = left != null ? left : bitLength; - right = right != null ? right : bitLength; - var min = Math.min(left, right); - - var ops = []; - for (var i = 0; i < left; i++) { - var accLength = Math.min(i+1, min+1); - ops.push({ op: 'bits.ssub', op_id: ':bits.ssub:'+i, params: {bitLengthLeft: accLength, bitLengthRight: right}}); - for (var j = 0; j < accLength; j++) { - ops.push({ op: 'if_else', op_id: ':if_else:' + i + ':' + j }); - } - } - return ops; - }, - dynamic_bits_cdiv: function (dir) { - return function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params) { - var constantBits = Zp.toString(2).length; - if (params.constantBits != null) { - constantBits = params.constantBits; - } - var bitLength = params.bitLength; - if (bitLength == null) { - bitLength = Zp.toString(2).length; - } - var min = Math.min(bitLength, constantBits); - - var ops = []; - var loopCounter = (dir === 'left') ? bitLength : constantBits; - for (var i = 0; i < loopCounter; i++) { - var accLength = Math.min(i + 1, min + 1); - if (dir === 'left') { - ops.push({ - op: 'bits.csubl', - op_id: ':bits.csubl:' + i, - params: {bitLength: accLength, constantBits: constantBits} - }); - } else { - ops.push({ - op: 'bits.ssub', - op_id: ':bits.ssub:' + i, - params: {bitLengthLeft: accLength, bitLengthRight: bitLength} - }); - } + return this; + }; - for (var j = 0; j < accLength; j++) { - ops.push({op: 'if_else', op_id: ':if_else:' + i + ':' + j}); - } - } - return ops; - } - }, - // rejection sampling - dynamic_rejection_sampling: function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params, task, jiff) { - var previousPreprocessing = jiff.preprocessing_table[task.id]; - params.reject_count = params.reject_count == null ? -1 : params.reject_count; - - if (previousPreprocessing == null || previousPreprocessing === 'RETRY' || (previousPreprocessing[0] != null && previousPreprocessing[0].value === 'RETRY')) { - if (!params.defaultBounds && (params.lower_bound == null || params.upper_bound == null)) { - jiff.store_preprocessing(task.id, {ondemand: true}); - return []; - } + Node.prototype.any = function any() { + var state = this._baseState; - var compute_threshold = params.compute_threshold; - if (compute_threshold == null) { // honest majority BGW - compute_threshold = Math.floor((compute_list.length + 1) / 2); - } + state.any = true; - var extra_params = {compute_threshold: compute_threshold}; - var reject_count = ++params.reject_count; + return this; + }; - Zp = Zp != null ? Zp : jiff.Zp; - params.lower_bound = params.lower_bound || 0; - params.upper_bound = params.upper_bound != null ? params.upper_bound : Zp; + Node.prototype.choice = function choice(obj) { + var state = this._baseState; - var range; - if (params.upper_bound.isBigNumber === true) { - range = params.upper_bound.minus(params.lower_bound); - } else { - range = params.upper_bound - params.lower_bound; - } + assert(state.choice === null); + state.choice = obj; + this._useArgs( + Object.keys(obj).map(function (key) { + return obj[key]; + }) + ); - // handle special cases - if (range.toString() === '0') { - throw new Error('rejection sampling preprocessing called with range 0, no numbers to sample!'); - } - if (range.toString() === '1') { - return [{op: 'sampling', op_id: '', params: extra_params}]; - } + return this; + }; - var bitLength = jiff.helpers.ceil(jiff.helpers.bLog(range, 2)); - bitLength = parseInt(bitLength.toString(), 10); - params.bitLength = bitLength; - - // fill in dependencies according to the lower and upper bounds - var dependent_ops = []; - var requires = []; - if (jiff.helpers.bLog(range, 2).toString().indexOf('.') > -1) { - dependent_ops = [ - {op: 'bits.clt', op_id: ':bits.clt:' + reject_count, params: extra_params, threshold: compute_threshold, receivers_list: compute_list}, - {op: 'open', op_id: ':open:' + reject_count, params: extra_params, threshold: compute_threshold, receivers_list: compute_list} - ]; - requires = [0, 1]; - } + Node.prototype.contains = function contains(item) { + var state = this._baseState; - if (params.lower_bound.toString() !== '0' && bitLength > 1) { - dependent_ops.push({op: 'bits.cadd', op_id: ':bits.cadd:' + reject_count, params: extra_params, threshold: compute_threshold, receivers_list: compute_list}); - requires.push(dependent_ops.length - 1); - } + assert(state.use === null); + state.contains = item; - dependent_ops.push( - {op: 'sampling', op_id: '', requires: requires, params: extra_params}, - {op: 'rejection_sampling', op_id: '', requires: [dependent_ops.length]} - ); + return this; + }; - return dependent_ops; - } + // + // Decoding + // - if (previousPreprocessing[0] != null && previousPreprocessing[0].value === 'PLACEHOLDER') { - delete jiff.preprocessing_table[task.id]; - } + Node.prototype._decode = function decode(input, options) { + var state = this._baseState; - return []; - }, - // random quotients for cdiv - dynamic_random_and_quotient: function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params, task, jiff) { - var constantNotProvided = (params.constant == null); - receivers_list = constantNotProvided ? compute_list : receivers_list; - Zp = Zp ? Zp : jiff.Zp; - - var newParams = {compute_threshold: params.compute_threshold}; - if (params.compute_threshold == null) { // honest majority BGW - newParams.compute_threshold = Math.floor((compute_list.length + 1) / 2); - } - threshold = newParams.compute_threshold; - - var dependent_ops = [ - {op: 'bits.cgteq', op_id: ':bits_cgteq', receivers_list: receivers_list, threshold: threshold, params: newParams}, - {op: 'if_else', op_id: ':ifelse1', receivers_list: receivers_list, threshold: threshold, params: newParams}, - {op: 'if_else', op_id: ':ifelse2', receivers_list: receivers_list, threshold: threshold, params: newParams}, - {op: 'rejection_sampling', op_id: ':rejection1', receivers_list: receivers_list, threshold: threshold, params: {lower_bound: 0, upper_bound: Zp, compute_threshold: threshold}} - ]; - - if (constantNotProvided) { - jiff.store_preprocessing(task_id, {ondemand: true}); - return dependent_ops; - } + // Decode root node + if (state.parent === null) return input.wrapResult(state.children[0]._decode(input, options)); - // we want to sample uniformly in [0, largest multiple of constant <= Zp) and [0, constant) - var largestQuotient = jiff.share_helpers['floor'](jiff.share_helpers['/'](Zp, params.constant)); - - dependent_ops.push( - {op: 'rejection_sampling', op_id: ':rejection2', threshold: threshold, params: {lower_bound: 0, upper_bound: largestQuotient, compute_threshold: threshold}}, - {op: 'rejection_sampling', op_id: ':rejection3', threshold: threshold, params: {lower_bound: 0, upper_bound: params.constant, compute_threshold: threshold}}, - {op: 'generate_random_and_quotient', op_id: '', requires: [0, 1, 2, 3, 4, 5], params: newParams} - ); - - return dependent_ops; - }, - // fast exponentiation - dynamic_fast_exponentiation: function (dependent_op, count, protocols, threshold, receivers_list, compute_list, Zp, task_id, params, task, jiff) { - Zp = Zp ? Zp : jiff.Zp; - var constantNotProvided = params.constant == null; - var constant = params.constant; - var constantBits = params.constantBits == null ? Zp.toString(2).length : params.constantBits; - - // special case - if (!constantNotProvided && constant.toString() === '0') { - return []; - } + var result = state['default']; + var present = true; - var ops = []; - if (constantNotProvided) { - for (var i = 0; i < constantBits-1; i++) { - ops.push({op: 'smult', op_id: ':smult0:' + i}); - ops.push({op: 'smult', op_id: ':smult1:' + i}); - } - ops.push({op: 'smult', op_id: ':smult0:' + i}); - return ops; - } + var prevKey = null; + if (state.key !== null) prevKey = input.enterKey(state.key); - // handle big number - if (jiff.helpers.BigNumber) { - constant = jiff.helpers.BigNumber(constant); - } + // Check if tag is there + if (state.optional) { + var tag = null; + if (state.explicit !== null) tag = state.explicit; + else if (state.implicit !== null) tag = state.implicit; + else if (state.tag !== null) tag = state.tag; - // preprocess for exactly the needed amount of multiplications - for (i = 0; jiff.share_helpers['<'](1, constant); i++) { - ops.push({op: 'smult', op_id: ':smult0:' + i}); - if (!jiff.share_helpers['even'](constant)) { - ops.push({op: 'smult', op_id: ':smult1:' + i}); - } - constant = jiff.share_helpers['floor'](jiff.share_helpers['/'](constant, 2)); - } - ops.push({op: 'smult', op_id: ':smult0:' + i}); - return ops; - }, - // for various equality tests, preprocess of cpow(Zp - 1) (Fermat's little theorem) - handler_cpow_Zp_minus_1: function (threshold, receivers_list, compute_list, Zp, op_id, params, task, jiff) { - Zp = Zp ? Zp : jiff.Zp; - params.constant = jiff.share_helpers['-'](Zp, 1); - return params; - } -}; -},{}],20:[function(require,module,exports){ -module.exports = function (jiffClient) { - var handlers = require('./handlers.js'); - - jiffClient.default_preprocessing_protocols = { - generate_beaver: jiffClient.protocols.generate_beaver_bgw, - generate_random_number: jiffClient.protocols.generate_random_number, - sampling: jiffClient.protocols.rejection_sampling, - generate_random_bits: jiffClient.protocols.generate_random_bits, - generate_random_bit: jiffClient.protocols.generate_random_bit_bgw, - generate_zero: jiffClient.protocols.generate_zero, - generate_random_and_quotient: jiffClient.protocols.generate_random_and_quotient - }; + if (tag === null && !state.any) { + // Trial and Error + var save = input.save(); + try { + if (state.choice === null) this._decodeGeneric(state.tag, input, options); + else this._decodeChoice(input, options); + present = true; + } catch (e) { + present = false; + } + input.restore(save); + } else { + present = this._peekTag(input, tag, state.any); - jiffClient.preprocessing_function_map = { - base: { - // arithmetic sharing protocols - 'smult': [ - {op: 'open', op_id: ':open1'}, - {op: 'generate_beaver', op_id: ':triplet'}, - {op: 'open', op_id: ':open2'} - ], - 'sxor_bit': [ - {op: 'smult', op_id: ':smult1'} - ], - 'slt': [ - {op: 'lt_halfprime', op_id: ':halfprime:1'}, - {op: 'lt_halfprime', op_id: ':halfprime:2'}, - {op: 'lt_halfprime', op_id: ':halfprime:3'}, - {op: 'smult', op_id: ':smult1'}, - {op: 'smult', op_id: ':smult2'} - ], - 'cgt': [ - {op: 'lt_halfprime', op_id: ':halfprime:1'}, - {op: 'lt_halfprime', op_id: ':halfprime:2'}, - {op: 'smult', op_id: ':smult1'} - ], - 'clt': [ - {op: 'lt_halfprime', op_id: ':halfprime:1'}, - {op: 'lt_halfprime', op_id: ':halfprime:2'}, - {op: 'smult', op_id: ':smult1'} - ], - 'lt_halfprime': [ - {op: 'rejection_sampling', op_id: ':sampling', params: {defaultBounds: true}}, - {op: 'smult', op_id: ':smult1'}, - {op: 'bits.cgt', op_id: ':bits.cgt'}, - {op: 'sxor_bit', op_id: ':sxor_bit'}, - {op: 'open', op_id: ':open'} - ], - 'cneq': [ - {op: 'cpow', op_id: ':cpow', handler: handlers.handler_cpow_Zp_minus_1} - ], - 'cpow': handlers.dynamic_fast_exponentiation, - 'cdiv': [ - {op: 'cgt', op_id: ':wrap_cgt'}, - {op: 'cgteq', op_id: ':cor1'}, - {op: 'cgteq', op_id: ':cor2'}, - {op: 'smult', op_id: ':smult'}, - {op: 'clt', op_id: ':zero_check'}, - {op: 'smult', op_id: ':zero_it'}, - {op: 'open', op_id: ':open'}, - {op: 'quotient', op_id: ':quotient'} - ], - 'sdiv': [ - {op: 'bit_decomposition', op_id: ':decomposition1'}, - {op: 'bit_decomposition', op_id: ':decomposition2'}, - {op: 'bits.sdiv', op_id: ':bits.sdiv'} - ], - 'if_else': [ - {op: 'smult', op_id: ':smult'} - ], - // bits protocols - 'bit_decomposition': [ - {op: 'rejection_sampling', op_id: ':sampling', params: {defaultBounds: true}}, - {op: 'bits.csubr', op_id: ':bits.csubr:1'}, - {op: 'bits.csubr', op_id: ':bits.csubr:2'}, - {op: 'if_else', op_id: ':if_else:', count: handlers.decomposition_ifelse_count}, - {op: 'open', op_id: ':open'} - ], - // comparisons - 'bits.cgteq': [ - {op: 'smult', op_id: ':smult:', count: handlers.constant_bits_count} - ], - 'bits.cneq': [ - {op: 'sor_bit', op_id: ':sor_bit:', count: handlers.constant_bits_count} - ], - 'bits.sneq': [ - {op: 'sxor_bit', op_id: ':sxor_bit:initial'}, - {op: 'sxor_bit', op_id: ':sxor_bit:', count: handlers.choice_bits_count(Math.min, -1)}, - {op: 'sor_bit', op_id: ':sor_bit:', count: handlers.choice_bits_count(Math.max, -1)} - ], - 'bits.sgteq': [ - {op: 'smult', op_id: ':smult:initial'}, - {op: 'smult', op_id: ':smult1:', count: handlers.choice_bits_count(Math.max, -1)}, - {op: 'sxor_bit', op_id: ':sxor_bit1:', count: handlers.choice_bits_count(Math.min, -1)}, - {op: 'smult', op_id: ':smult2:', count: handlers.choice_bits_count(Math.min, -1)} - ], - 'bits.sgt': [ - {op: 'bits.sgteq', op_id: ':bits.sgteq'}, - {op: 'bits.sneq', op_id: ':bits.sneq'}, - {op: 'smult', op_id: ':smult'} - ], - // constant arithmetic - 'bits.cadd': [ - {op: 'smult', op_id: ':smult:', count: handlers.constant_bits_count}, - {op: 'sxor_bit', op_id: ':sxor_bit:', count: handlers.constant_bits_count} - ], - 'bits.cmult': handlers.dynamic_bits_cmult, - 'bits.cdivl': handlers.dynamic_bits_cdiv('left'), - 'bits.cdivr': handlers.dynamic_bits_cdiv('right'), - // secret arithmetic - 'bits.sadd': [ - {op: 'sxor_bit', op_id: ':sxor_bit:initial'}, - {op: 'smult', op_id: ':smult:initial'}, - {op: 'smult', op_id: ':smult1:', count: handlers.choice_bits_count(Math.max, -1)}, - {op: 'sxor_bit', op_id: ':sxor_bit1:', count: handlers.choice_bits_count(Math.max, -1)}, - {op: 'smult', op_id: ':smult2:', count: handlers.choice_bits_count(Math.min, -1)}, - {op: 'sxor_bit', op_id: ':sxor_bit2:', count: handlers.choice_bits_count(Math.min, -1)} - ], - 'bits.smult': handlers.dynamic_bits_smult, - 'bits.sdiv': handlers.dynamic_bits_sdiv, - 'bits.open': [ - {op: 'open', op_id: ':', count: handlers.bits_count} - ], - // refresh/open - 'refresh': [ - {op: 'generate_zero', op_id: ''} - ], - 'open': [ - {op: 'refresh', op_id: ':refresh'} - ], - // generating a random number and its quotient / constant - 'quotient': handlers.dynamic_random_and_quotient, - // rejection sampling - 'rejection_sampling': handlers.dynamic_rejection_sampling - } - }; + if (input.isError(present)) return present; + } + } - // arithmetic protocols - jiffClient.preprocessing_function_map['base']['sor_bit'] = jiffClient.preprocessing_function_map['base']['sxor_bit']; - jiffClient.preprocessing_function_map['base']['smod'] = jiffClient.preprocessing_function_map['base']['sdiv']; - jiffClient.preprocessing_function_map['base']['slteq'] = jiffClient.preprocessing_function_map['base']['slt']; - jiffClient.preprocessing_function_map['base']['sgteq'] = jiffClient.preprocessing_function_map['base']['slt']; - jiffClient.preprocessing_function_map['base']['sgt'] = jiffClient.preprocessing_function_map['base']['slt']; - jiffClient.preprocessing_function_map['base']['clteq'] = jiffClient.preprocessing_function_map['base']['cgt']; - jiffClient.preprocessing_function_map['base']['cgteq'] = jiffClient.preprocessing_function_map['base']['clt']; - jiffClient.preprocessing_function_map['base']['seq'] = jiffClient.preprocessing_function_map['base']['cneq']; - jiffClient.preprocessing_function_map['base']['sneq'] = jiffClient.preprocessing_function_map['base']['cneq']; - jiffClient.preprocessing_function_map['base']['ceq'] = jiffClient.preprocessing_function_map['base']['cneq']; - - // bits protocols - jiffClient.preprocessing_function_map['base']['bits.clt'] = jiffClient.preprocessing_function_map['base']['bits.cgteq']; - jiffClient.preprocessing_function_map['base']['bits.clteq'] = jiffClient.preprocessing_function_map['base']['bits.cgteq']; - jiffClient.preprocessing_function_map['base']['bits.cgt'] = jiffClient.preprocessing_function_map['base']['bits.cgteq']; - jiffClient.preprocessing_function_map['base']['bits.ceq'] = jiffClient.preprocessing_function_map['base']['bits.cneq']; - jiffClient.preprocessing_function_map['base']['bits.slt'] = jiffClient.preprocessing_function_map['base']['bits.sgteq']; - jiffClient.preprocessing_function_map['base']['bits.slteq'] = jiffClient.preprocessing_function_map['base']['bits.sgt']; - jiffClient.preprocessing_function_map['base']['bits.seq'] = jiffClient.preprocessing_function_map['base']['bits.sneq']; - jiffClient.preprocessing_function_map['base']['bits.csubl'] = jiffClient.preprocessing_function_map['base']['bits.cadd']; - jiffClient.preprocessing_function_map['base']['bits.csubr'] = jiffClient.preprocessing_function_map['base']['bits.cadd']; - jiffClient.preprocessing_function_map['base']['bits.ssub'] = jiffClient.preprocessing_function_map['base']['bits.sadd']; -}; - -},{"./handlers.js":19}],21:[function(require,module,exports){ -const share = require('./share.js'); -const open = require('./open.js'); -const util = require('./util.js'); - -module.exports = { - jiff_share_array: share.share_array, - jiff_share_2D_array: share.share_2D_array, - jiff_share_ND_array: share.share_ND_array, - jiff_share_ND_array_static: share.share_ND_array_static, - jiff_share_ND_array_deferred: share.share_ND_array_deferred, - - jiff_skeleton_of: util.skeleton_of, - - jiff_open_array: open.open_array, - jiff_open_ND_array: open.open_ND_array, - jiff_receive_open_ND_array: open.receive_open_ND_array -}; - -},{"./open.js":22,"./share.js":23,"./util.js":24}],22:[function(require,module,exports){ -const open_array = function (jiff, shares, parties, op_ids) { - var i; - - // A base operation id is provided to use for all opens. - if (typeof(op_ids) === 'string' || typeof(op_ids) === 'number') { - var tmp = { s1: op_ids }; - for (i = 1; i <= jiff.party_count; i++) { - tmp[i] = op_ids; - } - op_ids = tmp; - } + // Push object on stack + var prevObj; + if (state.obj && present) prevObj = input.enterObject(); + + if (present) { + // Unwrap explicit values + if (state.explicit !== null) { + var explicit = this._decodeTag(input, state.explicit); + if (input.isError(explicit)) return explicit; + input = explicit; + } + + var start = input.offset; + + // Unwrap implicit and normal values + if (state.use === null && state.choice === null) { + if (state.any) var save = input.save(); + var body = this._decodeTag(input, state.implicit !== null ? state.implicit : state.tag, state.any); + if (input.isError(body)) return body; + + if (state.any) result = input.raw(save); + else input = body; + } + + if (options && options.track && state.tag !== null) options.track(input.path(), start, input.length, 'tagged'); + + if (options && options.track && state.tag !== null) options.track(input.path(), input.offset, input.length, 'content'); + + // Select proper method for tag + if (state.any) result = result; + else if (state.choice === null) result = this._decodeGeneric(state.tag, input, options); + else result = this._decodeChoice(input, options); + + if (input.isError(result)) return result; + + // Decode children + if (!state.any && state.choice === null && state.children !== null) { + state.children.forEach(function decodeChildren(child) { + // NOTE: We are ignoring errors here, to let parser continue with other + // parts of encoded data + child._decode(input, options); + }); + } + + // Decode contained/encoded by schema, only in bit or octet strings + if (state.contains && (state.tag === 'octstr' || state.tag === 'bitstr')) { + var data = new DecoderBuffer(result); + result = this._getUse(state.contains, input._reporterState.obj)._decode(data, options); + } + } - return open_ND_array(jiff, shares, parties, null, op_ids); -}; + // Pop object + if (state.obj && present) result = input.leaveObject(prevObj); + + // Set key + if (state.key !== null && (result !== null || present === true)) input.leaveKey(prevKey, state.key, result); + else if (prevKey !== null) input.exitKey(prevKey); + + return result; + }; + + Node.prototype._decodeGeneric = function decodeGeneric(tag, input, options) { + var state = this._baseState; + + if (tag === 'seq' || tag === 'set') return null; + if (tag === 'seqof' || tag === 'setof') return this._decodeList(input, tag, state.args[0], options); + else if (/str$/.test(tag)) return this._decodeStr(input, tag, options); + else if (tag === 'objid' && state.args) return this._decodeObjid(input, state.args[0], state.args[1], options); + else if (tag === 'objid') return this._decodeObjid(input, null, null, options); + else if (tag === 'gentime' || tag === 'utctime') return this._decodeTime(input, tag, options); + else if (tag === 'null_') return this._decodeNull(input, options); + else if (tag === 'bool') return this._decodeBool(input, options); + else if (tag === 'objDesc') return this._decodeStr(input, tag, options); + else if (tag === 'int' || tag === 'enum') return this._decodeInt(input, state.args && state.args[0], options); + + if (state.use !== null) { + return this._getUse(state.use, input._reporterState.obj)._decode(input, options); + } else { + return input.error('unknown tag: ' + tag); + } + }; + + Node.prototype._getUse = function _getUse(entity, obj) { + var state = this._baseState; + // Create altered use decoder if implicit is set + state.useDecoder = this._use(entity, obj); + assert(state.useDecoder._baseState.parent === null); + state.useDecoder = state.useDecoder._baseState.children[0]; + if (state.implicit !== state.useDecoder._baseState.implicit) { + state.useDecoder = state.useDecoder.clone(); + state.useDecoder._baseState.implicit = state.implicit; + } + return state.useDecoder; + }; -const open_ND_array = function (jiff, shares, receivers_list, senders_list, op_ids) { - if (senders_list == null) { - senders_list = []; - for (var i = 1; i <= jiff.party_count; i++) { - senders_list.push(i); - } - if (receivers_list == null) { - receivers_list = Array.from(senders_list); - } - } - var is_sending = senders_list.indexOf(jiff.id) > -1; - - // Compute operation id - /* - *if (op_ids == null) { - * op_ids = jiff.counters.gen_op_id('open_ND_array', receivers_list.concat(senders_list)); - *} - */ - - var final_deferred = new jiff.helpers.Deferred(); - var final_promise = final_deferred.promise; - var resolve_open = function (shares) { - final_deferred.resolve(( - function __open_ND_array(shares, parties, op_ids) { - if (typeof(shares.length) === 'undefined') { - return jiff.internal_open(shares, parties); - } else if (shares.length === 0) { - return Promise.resolve([]); - } else { - var promised_array = []; - for (var i = 0; i < shares.length; i++) { - promised_array.push(__open_ND_array(shares[i], parties, op_ids + ':' + i)); - } - return Promise.all(promised_array); - } - } - )(shares, receivers_list, op_ids)); - }; + Node.prototype._decodeChoice = function decodeChoice(input, options) { + var state = this._baseState; + var result = null; + var match = false; - if (is_sending) { - // Must emit the skeleton for any parties that are not holders but are receiving the open - var skeleton = (function __unwipe(nd_array, replace) { - if (!(typeof (nd_array.length) === 'undefined') || nd_array.length === 0) { - var unwiped_array = []; - for (var k = 0; k < nd_array.length; k++) { - unwiped_array.push(__unwipe(nd_array[k], replace)); - } - return unwiped_array; - } - return replace; - })(shares, null); - jiff.emit(op_ids + 'skeleton', receivers_list, JSON.stringify(skeleton)); + Object.keys(state.choice).some(function (key) { + var save = input.save(); + var node = state.choice[key]; + try { + var value = node._decode(input, options); + if (input.isError(value)) return false; - resolve_open(shares); - } else { - // Populate skeleton with imitation shares - var revive_shares = function (skeleton) { - const share = new jiff.SecretShare({}, senders_list, senders_list.length, jiff.Zp); - return (function __unwipe(nd_array, replace) { - if (nd_array != null && !(typeof (nd_array.length) === 'undefined' && nd_array.length > 0)) { - var unwiped_array = []; - for (var k = 0; k < nd_array.length; k++) { - unwiped_array.push(__unwipe(nd_array[k], replace)); - } - return unwiped_array; - } - return replace; - })(skeleton, share); - } + result = { type: key, value: value }; + match = true; + } catch (e) { + input.restore(save); + return false; + } + return true; + }, this); - // If this party is not a sender, then the variable `shares` may be a skeleton - if (shares != null) { - // Use existing shares as skeleton to revive - shares = revive_shares(shares); - resolve_open(shares); - } else { - // Receive skeleton from senders - jiff.listen(op_ids + 'skeleton', function (sender, skeleton) { - jiff.remove_listener(op_ids + 'skeleton'); // This doesn't seem to work + if (!match) return input.error('Choice not matched'); - if (typeof(skeleton) === 'string') { - skeleton = JSON.parse(skeleton); - } + return result; + }; - shares = revive_shares(skeleton); - resolve_open(shares); - }); - } - } + // + // Encoding + // - return final_promise; -}; + Node.prototype._createEncoderBuffer = function createEncoderBuffer(data) { + return new EncoderBuffer(data, this.reporter); + }; -const receive_open_ND_array = function (jiff, receivers_list, senders_list, threshold, Zp, op_ids) { - return open_ND_array(jiff, null, receivers_list, senders_list, op_ids); -}; + Node.prototype._encode = function encode(data, reporter, parent) { + var state = this._baseState; + if (state['default'] !== null && state['default'] === data) return; -module.exports = { - open_array: open_array, - open_ND_array: open_ND_array, - receive_open_ND_array: receive_open_ND_array -}; + var result = this._encodeValue(data, reporter, parent); + if (result === undefined) return; -},{}],23:[function(require,module,exports){ -const util = require('./util.js'); + if (this._skipDefault(result, reporter, parent)) return; -const share_array = function (jiff, array, lengths, threshold, receivers_list, senders_list, Zp, share_id) { - var skeletons = null; - if (lengths != null) { - // Check format of lengths - if (lengths != null && typeof(lengths) !== 'number' && typeof(lengths) !== 'object') { - throw new Error('share_array: unrecognized lengths'); - } + return result; + }; - // Get senders list for later checking - if (senders_list == null) { - senders_list = []; - for (i = 1; i <= jiff.party_count; i++) { - senders_list.push(i); - } - } + Node.prototype._encodeValue = function encode(data, reporter, parent) { + var state = this._baseState; - // Generate skeletons from lengths - skeletons = {}; - if (typeof(lengths) === 'number') { - // All arrays are of the same length - var skeleton = []; - for (var i = 0; i < lengths; i++) { - skeleton.push(null); - } - for (i = 0; i < senders_list.length; i++) { - skeletons[senders_list[i]] = skeleton; - } - } else { - // Lengths of the different arrays are all provided - for (i = 0; i < senders_list.length; i++) { - if (lengths[senders_list[i]] == null) { - throw new Error('share_array: missing length'); - } else { - skeletons[senders_list[i]] = []; - for (var j = 0; j < lengths[senders_list[i]]; j++) { - skeletons[senders_list[i]].push(null); - } - } - } - } - } + // Decode root node + if (state.parent === null) return state.children[0]._encode(data, reporter || new Reporter()); - return share_ND_array(jiff, array, skeletons, threshold, receivers_list, senders_list, Zp, share_id); -}; + var result = null; -const share_2D_array = function (jiff, array, lengths, threshold, receivers_list, senders_list, Zp, share_id) { - var i; + // Set reporter to share it with a child class + this.reporter = reporter; - // Check format of lengths - if (lengths != null && typeof(lengths) !== 'object') { - throw new Error('share_array: unrecognized lengths'); - } + // Check if data is there + if (state.optional && data === undefined) { + if (state['default'] !== null) data = state['default']; + else return; + } - // Default values - if (receivers_list == null) { - receivers_list = []; - for (i = 1; i <= jiff.party_count; i++) { - receivers_list.push(i); - } - } - if (senders_list == null) { - senders_list = []; - for (i = 1; i <= jiff.party_count; i++) { - senders_list.push(i); - } - } + // Encode children first + var content = null; + var primitive = false; + if (state.any) { + // Anything that was given is translated to buffer + result = this._createEncoderBuffer(data); + } else if (state.choice) { + result = this._encodeChoice(data, reporter); + } else if (state.contains) { + content = this._getUse(state.contains, parent)._encode(data, reporter); + primitive = true; + } else if (state.children) { + content = state.children + .map(function (child) { + if (child._baseState.tag === 'null_') return child._encode(null, reporter, data); + + if (child._baseState.key === null) return reporter.error('Child should have a key'); + var prevKey = reporter.enterKey(child._baseState.key); + + if (typeof data !== 'object') return reporter.error('Child expected, but input is not object'); + + var res = child._encode(data[child._baseState.key], reporter, data); + reporter.leaveKey(prevKey); + + return res; + }, this) + .filter(function (child) { + return child; + }); + content = this._createEncoderBuffer(content); + } else { + if (state.tag === 'seqof' || state.tag === 'setof') { + // TODO(indutny): this should be thrown on DSL level + if (!(state.args && state.args.length === 1)) return reporter.error('Too many args for : ' + state.tag); + + if (!Array.isArray(data)) return reporter.error('seqof/setof, but data is not Array'); + + var child = this.clone(); + child._baseState.implicit = null; + content = this._createEncoderBuffer( + data.map(function (item) { + var state = this._baseState; + + return this._getUse(state.args[0], data)._encode(item, reporter); + }, child) + ); + } else if (state.use !== null) { + result = this._getUse(state.use, parent)._encode(data, reporter); + } else { + content = this._encodePrimitive(state.tag, data); + primitive = true; + } + } - var isReceiving = receivers_list.indexOf(jiff.id) > -1; - if (senders_list.indexOf(jiff.id) === -1 && !isReceiving) { - // This party is neither a sender nor a receiver, do nothing! - return null; - } + // Encode data itself + var result; + if (!state.any && state.choice === null) { + var tag = state.implicit !== null ? state.implicit : state.tag; + var cls = state.implicit === null ? 'universal' : 'context'; + + if (tag === null) { + if (state.use === null) reporter.error('Tag could be omitted only for .use()'); + } else { + if (state.use === null) result = this._encodeComposite(tag, primitive, cls, content); + } + } - // compute operation id - receivers_list.sort(); // sort to get the same order - senders_list.sort(); - if (share_id == null) { - share_id = jiff.counters.gen_share_id(receivers_list, senders_list) + ':array:'; - } + // Wrap in explicit + if (state.explicit !== null) result = this._encodeComposite(state.explicit, false, 'context', result); - // wrap around result of share_array - var lengths_deferred = new jiff.helpers.Deferred(); - var lengths_promise = lengths_deferred.promise; + return result; + }; - // figure out lengths by having each party emit their length publicly - if (lengths == null) { - lengths = {}; - var total = 0; - if (senders_list.indexOf(jiff.id) > -1) { - lengths[jiff.id] = array.length; + Node.prototype._encodeChoice = function encodeChoice(data, reporter) { + var state = this._baseState; - // send the length of this party's array to all receivers - jiff.emit(share_id + 'length', receivers_list, array.length.toString(10)); - } + var node = state.choice[data.type]; + if (!node) { + assert(false, data.type + ' not found in ' + JSON.stringify(Object.keys(state.choice))); + } + return node._encode(data.value, reporter); + }; + + Node.prototype._encodePrimitive = function encodePrimitive(tag, data) { + var state = this._baseState; + + if (/str$/.test(tag)) return this._encodeStr(data, tag); + else if (tag === 'objid' && state.args) return this._encodeObjid(data, state.reverseArgs[0], state.args[1]); + else if (tag === 'objid') return this._encodeObjid(data, null, null); + else if (tag === 'gentime' || tag === 'utctime') return this._encodeTime(data, tag); + else if (tag === 'null_') return this._encodeNull(); + else if (tag === 'int' || tag === 'enum') return this._encodeInt(data, state.args && state.reverseArgs[0]); + else if (tag === 'bool') return this._encodeBool(data); + else if (tag === 'objDesc') return this._encodeStr(data, tag); + else throw new Error('Unsupported tag: ' + tag); + }; + + Node.prototype._isNumstr = function isNumstr(str) { + return /^[0-9 ]*$/.test(str); + }; + + Node.prototype._isPrintstr = function isPrintstr(str) { + return /^[A-Za-z0-9 '\(\)\+,\-\.\/:=\?]*$/.test(str); + }; + }, + { '../base': 59, 'minimalistic-assert': 226 } + ], + 61: [ + function (require, module, exports) { + var inherits = require('inherits'); + + function Reporter(options) { + this._reporterState = { + obj: null, + path: [], + options: options || {}, + errors: [] + }; + } + exports.Reporter = Reporter; + + Reporter.prototype.isError = function isError(obj) { + return obj instanceof ReporterError; + }; + + Reporter.prototype.save = function save() { + var state = this._reporterState; + + return { obj: state.obj, pathLen: state.path.length }; + }; + + Reporter.prototype.restore = function restore(data) { + var state = this._reporterState; + + state.obj = data.obj; + state.path = state.path.slice(0, data.pathLen); + }; + + Reporter.prototype.enterKey = function enterKey(key) { + return this._reporterState.path.push(key); + }; + + Reporter.prototype.exitKey = function exitKey(index) { + var state = this._reporterState; + + state.path = state.path.slice(0, index - 1); + }; + + Reporter.prototype.leaveKey = function leaveKey(index, key, value) { + var state = this._reporterState; + + this.exitKey(index); + if (state.obj !== null) state.obj[key] = value; + }; + + Reporter.prototype.path = function path() { + return this._reporterState.path.join('/'); + }; + + Reporter.prototype.enterObject = function enterObject() { + var state = this._reporterState; + + var prev = state.obj; + state.obj = {}; + return prev; + }; + + Reporter.prototype.leaveObject = function leaveObject(prev) { + var state = this._reporterState; + + var now = state.obj; + state.obj = prev; + return now; + }; + + Reporter.prototype.error = function error(msg) { + var err; + var state = this._reporterState; + + var inherited = msg instanceof ReporterError; + if (inherited) { + err = msg; + } else { + err = new ReporterError( + state.path + .map(function (elem) { + return '[' + JSON.stringify(elem) + ']'; + }) + .join(''), + msg.message || msg, + msg.stack + ); + } - jiff.listen(share_id + 'length', function (sender, message) { - lengths[sender] = { rows: parseInt(message, 10) }; - total++; - if (total === senders_list.length) { - jiff.remove_listener(share_id + 'length'); - lengths_deferred.resolve(lengths); - } - }); - } else if (typeof(lengths.rows) === 'number') { - // All arrays are of the same length - var l = lengths; - lengths = {}; - for (i = 0; i < senders_list.length; i++) { - lengths[senders_list[i]] = l; - } + if (!state.options.partial) throw err; - lengths_deferred.resolve(lengths); - } else { - // Lengths of the different arrays are all provided - for (i = 0; i < senders_list.length; i++) { - if (lengths[senders_list[i]] == null || lengths[senders_list[i]].rows == null) { - throw new Error('share_2D_array: missing rows length'); - } - } + if (!inherited) state.errors.push(err); - lengths_deferred.resolve(lengths); - } + return err; + }; - // Final results - var share_array_deferred = new jiff.helpers.Deferred(); - var share_array_promise = share_array_deferred.promise; - - // lengths are now set, start sharing - lengths_promise.then(function (lengths) { - // compute the number of sharing rounds - var max = 0; - for (i = 0; i < senders_list.length; i++) { - var l = lengths[senders_list[i]].rows; - max = l > max ? l : max; - } + Reporter.prototype.wrapResult = function wrapResult(result) { + var state = this._reporterState; + if (!state.options.partial) return result; - // share every round - var promises = []; - for (var r = 0; r < max; r++) { - var round_senders = []; - for (i = 0; i < senders_list.length; i++) { - if (lengths[senders_list[i]].rows > r) { - round_senders.push(senders_list[i]); - } - } + return { + result: this.isError(result) ? null : result, + errors: state.errors + }; + }; - var row_lengths = {}; - var empty = false; - for (var p = 0; p < round_senders.length; p++) { - var pid = round_senders[p]; - row_lengths[pid] = lengths[pid].cols; - if (lengths[pid][r] != null) { - row_lengths[pid] = lengths[pid][r]; - } - if (row_lengths[pid] == null) { - empty = true; - } - } + function ReporterError(path, msg) { + this.path = path; + this.rethrow(msg); + } + inherits(ReporterError, Error); + + ReporterError.prototype.rethrow = function rethrow(msg) { + this.message = msg + ' at: ' + (this.path || '(shallow)'); + if (Error.captureStackTrace) Error.captureStackTrace(this, ReporterError); + + if (!this.stack) { + try { + // IE only adds stack when thrown + throw new Error(this.message); + } catch (e) { + this.stack = e.stack; + } + } + return this; + }; + }, + { inherits: 218 } + ], + 62: [ + function (require, module, exports) { + var constants = require('../constants'); + + exports.tagClass = { + 0: 'universal', + 1: 'application', + 2: 'context', + 3: 'private' + }; + exports.tagClassByName = constants._reverse(exports.tagClass); + + exports.tag = { + 0x00: 'end', + 0x01: 'bool', + 0x02: 'int', + 0x03: 'bitstr', + 0x04: 'octstr', + 0x05: 'null_', + 0x06: 'objid', + 0x07: 'objDesc', + 0x08: 'external', + 0x09: 'real', + 0x0a: 'enum', + 0x0b: 'embed', + 0x0c: 'utf8str', + 0x0d: 'relativeOid', + 0x10: 'seq', + 0x11: 'set', + 0x12: 'numstr', + 0x13: 'printstr', + 0x14: 't61str', + 0x15: 'videostr', + 0x16: 'ia5str', + 0x17: 'utctime', + 0x18: 'gentime', + 0x19: 'graphstr', + 0x1a: 'iso646str', + 0x1b: 'genstr', + 0x1c: 'unistr', + 0x1d: 'charstr', + 0x1e: 'bmpstr' + }; + exports.tagByName = constants._reverse(exports.tag); + }, + { '../constants': 63 } + ], + 63: [ + function (require, module, exports) { + var constants = exports; - var row = r < array.length ? array[r] : []; - row_lengths = empty ? null : row_lengths; - var round_results = share_array(jiff, row, row_lengths, threshold, receivers_list, round_senders, Zp, share_id + 'row' + r + ':'); - promises.push(round_results); - } + // Helper + constants._reverse = function reverse(map) { + var res = {}; - // Wait for every promises corresponding to every row - return Promise.all(promises).then(function (intermediate_results) { - // Store results here - var results = {}; - if (isReceiving) { - for (i = 0; i < senders_list.length; i++) { - results[senders_list[i]] = []; - } - } + Object.keys(map).forEach(function (key) { + // Convert key to integer if it is stringified + if ((key | 0) == key) key = key | 0; - for (i = 0; i < intermediate_results.length; i++) { - var round = intermediate_results[i]; - for (var sender_id in round) { - if (round.hasOwnProperty(sender_id)) { - results[sender_id].push(round[sender_id]); - } - } - } + var value = map[key]; + res[value] = key; + }); - share_array_deferred.resolve(results); - }); - }); + return res; + }; - return isReceiving ? share_array_promise : null; -}; + constants.der = require('./der'); + }, + { './der': 62 } + ], + 64: [ + function (require, module, exports) { + var inherits = require('inherits'); -const share_from_skeleton_unbound = function (jiff, that, sender, skeleton) { - var shares = typeof(skeleton) === 'string' ? JSON.parse(skeleton) : skeleton; + var asn1 = require('../../asn1'); + var base = asn1.base; + var bignum = asn1.bignum; - var promise = share_array_single_sender(jiff, shares, that.threshold, that.receivers_list, sender, that.Zp, that.share_id + ':p_id_' + sender); + // Import DER constants + var der = asn1.constants.der; - promise.then(function (sender, array) { - that.deferreds[sender].resolve(array); - }.bind(null, sender)); -}; + function DERDecoder(entity) { + this.enc = 'der'; + this.name = entity.name; + this.entity = entity; -const share_array_single_sender = function (jiff, secrets, threshold, receivers_list, sender, Zp, share_id) { - if (secrets != null && secrets.length === 0) { - return Promise.resolve([]); - } else if (secrets != null && Array.isArray(secrets)) { - var promised_array = []; - for (var j = 0; j < secrets.length; j++) { - promised_array.push(share_array_single_sender(jiff, secrets[j], threshold, receivers_list, sender, Zp, share_id + ':' + j)); - } + // Construct base tree + this.tree = new DERNode(); + this.tree._init(entity.body); + } + module.exports = DERDecoder; - var isReceiving = receivers_list.indexOf(jiff.id) > -1; + DERDecoder.prototype.decode = function decode(data, options) { + if (!(data instanceof base.DecoderBuffer)) data = new base.DecoderBuffer(data, options); - if (isReceiving) { - var deferred_array = new jiff.helpers.Deferred(); - Promise.all(promised_array).then(function (array) { - deferred_array.resolve(array); - }); - } + return this.tree._decode(data, options); + }; - return isReceiving ? deferred_array.promise : Promise.resolve({}); - } else { - // Create and distribute the share - Note: Senders are reorganized in the final array. - // The return value of jiff.share is an array, [sender: share], and we only need to return share by itself. - return Promise.resolve(jiff.share(secrets, threshold, receivers_list, [sender], Zp, 'share:' + share_id)[sender]); - } -}; + // Tree methods -const share_ND_array_deferred = function (jiff, secrets, skeletons, threshold, receivers_list, senders_list, Zp, share_id) { - var parameters = [receivers_list, senders_list, threshold, Zp, share_id]; - [receivers_list, senders_list, threshold, Zp, share_id] = util.sanitize_array_params.bind(null, jiff).apply(null, parameters); + function DERNode(parent) { + base.Node.call(this, 'der', parent); + } + inherits(DERNode, base.Node); - var i, p_id, skeleton; + DERNode.prototype._peekTag = function peekTag(buffer, tag, any) { + if (buffer.isEmpty()) return false; - var isReceiving = receivers_list.indexOf(jiff.id) > -1; - var isSending = senders_list.indexOf(jiff.id) > -1; - if (!isSending && !isReceiving) { - return null; // This party is neither a sender nor a receiver, do nothing! - } + var state = buffer.save(); + var decodedTag = derDecodeTag(buffer, 'Failed to peek tag: "' + tag + '"'); + if (buffer.isError(decodedTag)) return decodedTag; - // Setup deferreds: required because we don't yet know how many shares to account for - var final_deferreds = []; - for (i = 0; i < senders_list.length; i++) { - p_id = senders_list[i]; - final_deferreds[p_id] = new jiff.helpers.Deferred(); - } + buffer.restore(state); - const share_from_skeleton = share_from_skeleton_unbound.bind(null, jiff).bind(null, { - deferreds: final_deferreds, - threshold: threshold, - receivers_list: receivers_list, - Zp: Zp, share_id: share_id - }); - - if (skeletons == null) { - if (isSending) { // Send the shape and lengths of this party's array to all receivers - skeleton = jiff.skeleton_of(secrets); // All secrets are removed while maintaing the array's orginial structure. - var skeleton_str = JSON.stringify(skeleton); // serialize for emit - jiff.emit(share_id + 'skeleton', receivers_list, skeleton_str); - share_from_skeleton(jiff.id, secrets); // Share the real values matching the emitted skeleton - } - if (isReceiving) { - jiff.listen(share_id + 'skeleton', share_from_skeleton); // Receive shares when dimensions are known - } - } else { - senders_list = Array.from(senders_list); // remove jiff helpers' internal properties - util.match_skeletons(jiff, skeletons, senders_list); // Saftey check array dimention presets - for (i in senders_list) { // Share each party's array - p_id = senders_list[i]; - var myself = p_id === jiff.id; - skeleton = skeletons[p_id]; - share_from_skeleton(p_id, myself ? secrets : skeleton); - } - } + return decodedTag.tag === tag || decodedTag.tagStr === tag || decodedTag.tagStr + 'of' === tag || any; + }; + DERNode.prototype._decodeTag = function decodeTag(buffer, tag, any) { + var decodedTag = derDecodeTag(buffer, 'Failed to decode tag of "' + tag + '"'); + if (buffer.isError(decodedTag)) return decodedTag; - // Combine all promises and re-index final array map - var final_deferred = new jiff.helpers.Deferred(); - var final_promise = isReceiving ? final_deferred.promise : Promise.resolve({}); - Promise.all((function () { - var all_promises = []; - for (var i = 0; i < senders_list.length; i++) { - var p_id = senders_list[i] - all_promises.push(final_deferreds[p_id].promise); - } - return all_promises; - })()).then(function (array) { - var shares = {}; - for (var i = 0; i < senders_list.length; i++) { - var p_id = senders_list[i]; - shares[p_id] = array[i]; - } - final_deferred.resolve(shares); - jiff.remove_listener(share_id + 'skeleton'); - }); - - return final_promise; // Return promise to map of secret-shared arrays -}; - -const share_ND_array_static = function (jiff, secrets, skeletons, threshold, receivers_list, senders_list, Zp, share_id) { - var parameters = [receivers_list, senders_list, threshold, Zp, share_id]; - [receivers_list, senders_list, threshold, Zp, share_id] = util.sanitize_array_params.bind(null, jiff).apply(null, parameters); - - var i; - - var shares = {}; - - var isReceiving = receivers_list.indexOf(jiff.id) > -1; - var isSending = senders_list.indexOf(jiff.id) > -1; - if (isSending || isReceiving) { - // Static version of share_from_skeleton - const share_from_skeleton = function (sender, skeleton) { - return (function share_recursive(__secrets, share_id) { - if (__secrets != null && __secrets.length === 0) { - return []; - } else if (__secrets != null && Array.isArray(__secrets)) { - var array = []; - for (var j = 0; j < __secrets.length; j++) { - array.push(share_recursive(__secrets[j], share_id + ':' + j)); - } - return isReceiving ? array : {}; - } else { - return jiff.share(__secrets, threshold, receivers_list, [sender], Zp, 'share:' + share_id)[sender]; - } - })(skeleton, share_id + ':p_id_' + sender); - } + var len = derDecodeLen(buffer, decodedTag.primitive, 'Failed to get length of "' + tag + '"'); - senders_list = Array.from(senders_list); // remove jiff helpers' internal properties - util.match_skeletons(jiff, skeletons, senders_list); // Saftey check array dimention presets - for (i in senders_list) { // Share each party's array - var p_id = senders_list[i]; - var skeleton = skeletons[p_id]; - shares[p_id] = share_from_skeleton(p_id, p_id === jiff.id ? secrets : skeleton); - } - } + // Failure + if (buffer.isError(len)) return len; - return isReceiving ? shares : {}; // Return promise to map of secret-shared arrays -}; - -const share_ND_array = function (jiff, secrets, skeletons, threshold, receivers_list, senders_list, Zp, share_id) { - var share_ND_array = skeletons != null ? share_ND_array_static : share_ND_array_deferred; - return share_ND_array(jiff, secrets, skeletons, threshold, receivers_list, senders_list, Zp, share_id); -}; - -module.exports = { - share_array: share_array, - share_2D_array: share_2D_array, - share_ND_array: share_ND_array -}; - -},{"./util.js":24}],24:[function(require,module,exports){ -const sanitize_array_params = function (jiff, receivers_list, senders_list, threshold, Zp, share_id) { - [receivers_list, senders_list] = [receivers_list, senders_list].map(function (party_list) { - if (party_list == null) { - party_list = []; - for (var i = 1; i <= jiff.party_count; i++) { - party_list.push(i); - } - } - jiff.helpers.sort_ids(party_list); // sort to get the same order - return party_list; - }); - - if (threshold == null) { - threshold = receivers_list.length; - } else if (threshold < 0) { - threshold = 2; - } else if (threshold > receivers_list.length) { - threshold = receivers_list.length; - } + if (!any && decodedTag.tag !== tag && decodedTag.tagStr !== tag && decodedTag.tagStr + 'of' !== tag) { + return buffer.error('Failed to match tag: "' + tag + '"'); + } - if (Zp == null) { - Zp = jiff.Zp; - } + if (decodedTag.primitive || len !== null) return buffer.skip(len, 'Failed to match body of: "' + tag + '"'); - if (share_id == null) { // Make a unique id for this array - share_id = jiff.counters.gen_op_id2('share_array', receivers_list, senders_list); - } + // Indefinite length... find END tag + var state = buffer.save(); + var res = this._skipUntilEnd(buffer, 'Failed to skip indefinite length body: "' + this.tag + '"'); + if (buffer.isError(res)) return res; - return [receivers_list, senders_list, threshold, Zp, share_id]; -}; + len = buffer.offset - state.offset; + buffer.restore(state); + return buffer.skip(len, 'Failed to match body of: "' + tag + '"'); + }; -const skeleton_of = function (jiff, nd_array, replace) { - if (typeof(replace) === 'undefined') { - replace = null; - } - if (!(typeof(nd_array.length) === 'undefined') || nd_array.length === 0) { - var wiped_array = []; - for (var k = 0; k < nd_array.length; k++) { - wiped_array.push(jiff.skeleton_of(nd_array[k], replace)); - } - return wiped_array; - } - return replace; -}; - -const match_skeletons = function (jiff, skeletons, senders_list) { - var keys = Object.keys(skeletons); - var expected_keys = senders_list.map(String); - if (keys.length === senders_list.length) { - for (var i = 0; i < senders_list.length; i++) { - if (!(keys[i] === expected_keys[i])) { - console.log('senders: ', senders_list); - console.log('skeleton keys', keys); - throw new Error('Keys of parameter \'skeletons\' should be the same as the senders'); - } - } - } else { - throw new Error('Invalid parameter \'skeletons\''); - } -}; - -module.exports = { - sanitize_array_params: sanitize_array_params, - skeleton_of: skeleton_of, - match_skeletons: match_skeletons -}; - -},{}],25:[function(require,module,exports){ -module.exports = { - /** - * Compute sum of bitwise secret shared number and a constant - * @function cadd - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - the bit wise secret shares - * @param {number} constant - the constant - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length here will be max(|bits|, |constant|) + 1 - * in case of potential overflow / carry - */ - cadd: function (jiff, bits, constant, op_id) { - if (!(bits[0].isConstant(constant))) { - throw new Error('parameter should be a number (bits.cadd)'); - } - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.cadd', bits[0].holders); - } + DERNode.prototype._skipUntilEnd = function skipUntilEnd(buffer, fail) { + while (true) { + var tag = derDecodeTag(buffer, fail); + if (buffer.isError(tag)) return tag; + var len = derDecodeLen(buffer, tag.primitive, fail); + if (buffer.isError(len)) return len; - if (constant.toString() === '0') { - return bits; - } + var res; + if (tag.primitive || len !== null) res = buffer.skip(len); + else res = this._skipUntilEnd(buffer, fail); - // copy to avoid aliasing problems during execution - bits = bits.slice(); - - // decompose constant into bits - var constant_bits = jiff.helpers.number_to_bits(constant, bits.length); // pads with zeros to bits.length - - // initialize results - var result = jiff.utils.many_secret_shares(Math.max(constant_bits.length, bits.length), bits[0].holders, bits[0].threshold, bits[0].Zp); - var deferreds = result.deferreds; - result = result.shares; - - var sum = bits[0].icxor_bit(constant_bits[0]); - var carry = bits[0].icmult(constant_bits[0]); - - // put initial bit at head of result array - result.unshift(sum); - deferreds.unshift(null); - - // compute sum one bit at a time, propagating carry - jiff.utils.bit_combinator(deferreds[deferreds.length - 1], 1, deferreds.length - 1, carry, function (i, carry) { - var sum; - if (i < bits.length) { - var and = bits[i].icmult(constant_bits[i]); - var xor = bits[i].icxor_bit(constant_bits[i]); - var xorAndCarry = xor.ismult(carry, op_id + ':smult:' + (i - 1)); - - sum = xor.isxor_bit(carry, op_id + ':sxor_bit:' + (i - 1)); - carry = and.isadd(xorAndCarry); // cheap or, xor and and cannot both be true! - } else { - // bits.length <= i < constant_bits.length - // and is zero, xor is constant_bits[i] - sum = carry.icxor_bit(constant_bits[i]); - carry = carry.icmult(constant_bits[i]); - } + // Failure + if (buffer.isError(res)) return res; - sum.wThen(deferreds[i].resolve); - return carry; - }); - - return result; - }, - /** - * Compute [secret bits] - [constant bits] - * @function csubl - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {number} constant - the constant - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - the bit wise secret shares - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where - * the bit at index 0 is the least significant bit. The bit at index 1 is the most significant bit, - * and the bit at index |bits| is 1 if the result overflows, or 0 otherwise - */ - csubl: function (jiff, bits, constant, op_id) { - if (!(bits[0].isConstant(constant))) { - throw new Error('parameter should be a number (bits.csubl)'); - } - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.csubl', bits[0].holders); - } + if (tag.tagStr === 'end') break; + } + }; - if (constant.toString() === '0') { - return bits; - } + DERNode.prototype._decodeList = function decodeList(buffer, tag, decoder, options) { + var result = []; + while (!buffer.isEmpty()) { + var possibleEnd = this._peekTag(buffer, 'end'); + if (buffer.isError(possibleEnd)) return possibleEnd; - // copy to avoid aliasing problems during execution - bits = bits.slice(); - - // decompose constant into bits - var constant_bits = jiff.helpers.number_to_bits(constant, bits.length); // pads with zeros to bits.length - - // initialize results - var result = jiff.utils.many_secret_shares(Math.max(constant_bits.length, bits.length), bits[0].holders, bits[0].threshold, bits[0].Zp); - var deferreds = result.deferreds; - result = result.shares; - - var diff = bits[0].icxor_bit(constant_bits[0]); - var borrow = bits[0].inot().icmult(constant_bits[0]); - - // put initial bit at head of result array - result.unshift(diff); - deferreds.unshift(null); - - // compute diff one bit at a time, propagating borrow - jiff.utils.bit_combinator(deferreds[deferreds.length - 1], 1, deferreds.length - 1, borrow, function (i, borrow) { - var diff; - if (i < bits.length) { - var xor = bits[i].icxor_bit(constant_bits[i]); - var andNot = bits[i].inot().icmult(constant_bits[i]); - - // save and update borrow - diff = xor.isxor_bit(borrow, op_id + ':sxor_bit:' + (i - 1)); - borrow = xor.inot().ismult(borrow, op_id + ':smult:' + (i - 1)); - borrow = borrow.isadd(andNot); - } else { - // bits.length <= i < constant_bits.length - // xor and andNot are equal to the constant bit value since secret bit is always zero here - diff = borrow.icxor_bit(constant_bits[i]); - borrow = borrow.issub(borrow.icmult(constant_bits[i])); - borrow = borrow.icadd(constant_bits[i]); - } + var res = decoder.decode(buffer, 'der', options); + if (buffer.isError(res) && possibleEnd) break; + result.push(res); + } + return result; + }; + + DERNode.prototype._decodeStr = function decodeStr(buffer, tag) { + if (tag === 'bitstr') { + var unused = buffer.readUInt8(); + if (buffer.isError(unused)) return unused; + return { unused: unused, data: buffer.raw() }; + } else if (tag === 'bmpstr') { + var raw = buffer.raw(); + if (raw.length % 2 === 1) return buffer.error('Decoding of string type: bmpstr length mismatch'); + + var str = ''; + for (var i = 0; i < raw.length / 2; i++) { + str += String.fromCharCode(raw.readUInt16BE(i * 2)); + } + return str; + } else if (tag === 'numstr') { + var numstr = buffer.raw().toString('ascii'); + if (!this._isNumstr(numstr)) { + return buffer.error('Decoding of string type: ' + 'numstr unsupported characters'); + } + return numstr; + } else if (tag === 'octstr') { + return buffer.raw(); + } else if (tag === 'objDesc') { + return buffer.raw(); + } else if (tag === 'printstr') { + var printstr = buffer.raw().toString('ascii'); + if (!this._isPrintstr(printstr)) { + return buffer.error('Decoding of string type: ' + 'printstr unsupported characters'); + } + return printstr; + } else if (/str$/.test(tag)) { + return buffer.raw().toString(); + } else { + return buffer.error('Decoding of string type: ' + tag + ' unsupported'); + } + }; + + DERNode.prototype._decodeObjid = function decodeObjid(buffer, values, relative) { + var result; + var identifiers = []; + var ident = 0; + while (!buffer.isEmpty()) { + var subident = buffer.readUInt8(); + ident <<= 7; + ident |= subident & 0x7f; + if ((subident & 0x80) === 0) { + identifiers.push(ident); + ident = 0; + } + } + if (subident & 0x80) identifiers.push(ident); - diff.wThen(deferreds[i].resolve); - return borrow; - }); - - return result; - }, - /** - * Compute [constant bits] - [secret bits] - * @function csubr - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {number} constant - the constant - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - the bit wise secret shares - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where - * the bit at index 0 is the least significant bit. The bit at index 1 is the most significant bit, - * and the bit at index |bits| is 1 if the result overflows, or 0 otherwise - */ - csubr: function (jiff, constant, bits, op_id) { - if (!(bits[0].isConstant(constant))) { - throw new Error('parameter should be a number (bits.csubr)'); - } - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.csubr', bits[0].holders); - } + var first = (identifiers[0] / 40) | 0; + var second = identifiers[0] % 40; - // copy to avoid aliasing problems during execution - bits = bits.slice(); - - // decompose constant into bits - var constant_bits = jiff.helpers.number_to_bits(constant, bits.length); // pads with zeros to bits.length - - // initialize results - var result = jiff.utils.many_secret_shares(Math.max(constant_bits.length, bits.length), bits[0].holders, bits[0].threshold, bits[0].Zp); - var deferreds = result.deferreds; - result = result.shares; - - var diff = bits[0].icxor_bit(constant_bits[0]); - var borrow = bits[0].issub(bits[0].icmult(constant_bits[0])); - - // put initial bit at head of result array - result.unshift(diff); - deferreds.unshift(null); - - // compute diff one bit at a time, propagating borrow - jiff.utils.bit_combinator(deferreds[deferreds.length - 1], 1, deferreds.length - 1, borrow, function (i, borrow) { - var diff; - if (i < bits.length) { - var xor = bits[i].icxor_bit(constant_bits[i]); - var andNot = bits[i].issub(bits[i].icmult(constant_bits[i])); - - // save and update borrow - diff = xor.isxor_bit(borrow, op_id + ':sxor_bit:' + (i - 1)); - borrow = xor.inot().ismult(borrow, op_id + ':smult:' + (i - 1)); - borrow = borrow.isadd(andNot); - } else { - // andNot is zero and xor is equal to the constant bit since secret bit is always zero here. - diff = borrow.icxor_bit(constant_bits[i]); - borrow = borrow.icmult(constant_bits[i] === 1 ? 0 : 1); - } + if (relative) result = identifiers; + else result = [first, second].concat(identifiers.slice(1)); - diff.wThen(deferreds[i].resolve); - return borrow; - }); - - return result; - }, - /** - * - * Compute [secret bits1] + [secret bits2] - * @function sadd - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - the first bitwise shared number: array of secrets with index 0 being least significant bit - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number (length may be different) - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where - * the bit at index 0 is the least significant bit - */ - sadd: function (jiff, bits1, bits2, op_id) { - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.sadd', bits1[0].holders); - } + if (values) { + var tmp = values[result.join(' ')]; + if (tmp === undefined) tmp = values[result.join('.')]; + if (tmp !== undefined) result = tmp; + } - // copy to avoid aliasing problems during execution - bits1 = bits1.slice(); - bits2 = bits2.slice(); - - var tmp = bits1.length > bits2.length ? bits1 : bits2; - bits2 = bits1.length > bits2.length ? bits2 : bits1; // shortest array - bits1 = tmp; // longest array - - // initialize results - var result = jiff.utils.many_secret_shares(bits1.length, bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); - var deferreds = result.deferreds; - result = result.shares; - - var sum = bits1[0].isxor_bit(bits2[0], op_id + ':sxor_bit:initial'); - var carry = bits1[0].ismult(bits2[0], op_id + ':smult:initial'); - - // put initial bit at head of result array - result.unshift(sum); - deferreds.unshift(null); - - // compute sum one bit at a time, propagating carry - jiff.utils.bit_combinator(deferreds[deferreds.length - 1], 1, deferreds.length - 1, carry, function (i, carry) { - var sum; - if (i < bits2.length) { - var and = bits1[i].ismult(bits2[i], op_id + ':smult1:' + (i - 1)); - var xor = bits1[i].isxor_bit(bits2[i], op_id + ':sxor_bit1:' + (i - 1)); - var xorAndCarry = xor.ismult(carry, op_id + ':smult2:' + (i - 1)); - - sum = xor.isxor_bit(carry, op_id + ':sxor_bit2:' + (i - 1)); - carry = and.isadd(xorAndCarry); // cheap or, xor and and cannot both be true! - } else { - // and is always zero, xor is equal to bits1[i] - sum = bits1[i].isxor_bit(carry, op_id + ':sxor_bit1:' + (i - 1)); - carry = bits1[i].ismult(carry, op_id + ':smult1:' + (i - 1)); - } + return result; + }; + + DERNode.prototype._decodeTime = function decodeTime(buffer, tag) { + var str = buffer.raw().toString(); + if (tag === 'gentime') { + var year = str.slice(0, 4) | 0; + var mon = str.slice(4, 6) | 0; + var day = str.slice(6, 8) | 0; + var hour = str.slice(8, 10) | 0; + var min = str.slice(10, 12) | 0; + var sec = str.slice(12, 14) | 0; + } else if (tag === 'utctime') { + var year = str.slice(0, 2) | 0; + var mon = str.slice(2, 4) | 0; + var day = str.slice(4, 6) | 0; + var hour = str.slice(6, 8) | 0; + var min = str.slice(8, 10) | 0; + var sec = str.slice(10, 12) | 0; + if (year < 70) year = 2000 + year; + else year = 1900 + year; + } else { + return buffer.error('Decoding ' + tag + ' time is not supported yet'); + } - sum.wThen(deferreds[i].resolve); - return carry; - }); - - return result; - }, - /** - * Compute [secret bits1] - [secret bits2] - * @function ssub - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - first bitwise secret shared number: lower indices represent less significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - second bitwise secret shared number (length may be different) - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result. Note that the length of the returned result is |bits|+1, where - * the bit at index 0 is the least significant bit. The bit at index 1 is the most significant bit, - * and the bit at index |bits| is 1 if the result overflows, or 0 otherwise - */ - ssub: function (jiff, bits1, bits2, op_id) { - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.ssub', bits1[0].holders); - } + return Date.UTC(year, mon - 1, day, hour, min, sec, 0); + }; - // copy to avoid aliasing problems during execution - bits1 = bits1.slice(); - bits2 = bits2.slice(); - - // initialize results - var result = jiff.utils.many_secret_shares(Math.max(bits1.length, bits2.length), bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); - var deferreds = result.deferreds; - result = result.shares; - - var diff = bits1[0].isxor_bit(bits2[0], op_id + ':sxor_bit:initial'); - var borrow = bits1[0].inot().ismult(bits2[0], op_id + ':smult:initial'); - - // put initial bit at head of result array - result.unshift(diff); - deferreds.unshift(null); - - // compute diff one bit at a time, propagating borrow - jiff.utils.bit_combinator(deferreds[deferreds.length - 1], 1, deferreds.length - 1, borrow, function (i, borrow) { - var diff; - if (i < bits1.length && i < bits2.length) { - var xor = bits1[i].isxor_bit(bits2[i], op_id + ':sxor_bit1:' + (i - 1)); - var andNot = bits1[i].inot().ismult(bits2[i], op_id + ':smult1:' + (i - 1)); - - // save and update borrow - diff = xor.isxor_bit(borrow, op_id + ':sxor_bit2:' + (i - 1)); - borrow = xor.inot().ismult(borrow, op_id + ':smult2:' + (i - 1)); - borrow = borrow.isadd(andNot); - } else if (i < bits1.length) { - // xor is equal to the value of bits1[i], andNot is equal to 0, since bits[2] is all zeros here - diff = bits1[i].isxor_bit(borrow, op_id + ':sxor_bit1:' + (i - 1)); - borrow = bits1[i].inot().ismult(borrow, op_id + ':smult1:' + (i - 1)); - } else { // i < bits2.length - // xor and andNot are equal to the value of bits2[i] - diff = bits2[i].isxor_bit(borrow, op_id + ':sxor_bit1:' + (i - 1)); - borrow = bits2[i].inot().ismult(borrow, op_id + ':smult1:' + (i - 1)); - borrow = borrow.isadd(bits2[i]); - } + DERNode.prototype._decodeNull = function decodeNull(buffer) { + return null; + }; - diff.wThen(deferreds[i].resolve); - return borrow; - }); - - return result; - }, - /** - * Compute [secret bits] * constant - * @function cmult - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - bitwise shared secret to multiply: lower indices represent less significant bits - * @param {number} constant - constant to multiply with - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result, the length of the result will be bits.length + ceil(log2(constant)), except - * if constant is zero, the result will then be [ zero share ] - */ - cmult: function (jiff, bits, constant, op_id) { - if (!(bits[0].isConstant(constant))) { - throw new Error('parameter should be a number (bits.cmult)'); - } - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.cmult', bits[0].holders); - } + DERNode.prototype._decodeBool = function decodeBool(buffer) { + var res = buffer.readUInt8(); + if (buffer.isError(res)) return res; + else return res !== 0; + }; - // copy to avoid aliasing problems during execution - bits = bits.slice(); + DERNode.prototype._decodeInt = function decodeInt(buffer, values) { + // Bigint, return as it is (assume big endian) + var raw = buffer.raw(); + var res = new bignum(raw); - // decompose constant into bits - var constant_bits = jiff.helpers.number_to_bits(constant); // do not pad + if (values) res = values[res.toString(10)] || res; - // Initialize the result - var result = jiff.utils.many_secret_shares(bits.length + constant_bits.length, bits[0].holders, bits[0].threshold, bits[0].Zp); - var deferreds = result.deferreds; - result = result.shares; + return res; + }; - // Resolve result when ready - var final_deferred = new jiff.helpers.Deferred(); - final_deferred.promise.then(jiff.utils.resolve_many_secrets.bind(null, deferreds)); + DERNode.prototype._use = function use(entity, obj) { + if (typeof entity === 'function') entity = entity(obj); + return entity._getDecoder('der').tree; + }; - // get useless share of zero (just for padding) - var zero = new jiff.SecretShare(0, bits[0].holders, bits[0].threshold, bits[0].Zp); - var initial = [zero]; + // Utility methods - // special case - if (constant.toString() === '0') { - return initial; - } + function derDecodeTag(buf, fail) { + var tag = buf.readUInt8(fail); + if (buf.isError(tag)) return tag; - // main function - jiff.utils.bit_combinator(final_deferred, 0, constant_bits.length, initial, function (i, intermediate) { - // Shift bits to create the intermediate values, - // and sum if the corresponding bit in a is 1 - if (constant_bits[i].toString() === '1') { - intermediate = jiff.protocols.bits.sadd(intermediate, bits, op_id + ':bits.sadd:' + i); - } + var cls = der.tagClass[tag >> 6]; + var primitive = (tag & 0x20) === 0; - bits.unshift(zero); - return intermediate; - }, function (intermediate) { - // promise-ify an array of intermediate results - var promises = []; - for (var i = 0; i < intermediate.length; i++) { - promises.push(intermediate[i].value); - } - return Promise.all(promises); - }, function (result) { - // identity - return result; - }); - - return result; - }, - /** - * Compute [secret bits1] * [secret bits2] - * @function smult - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - bitwise shared secret to multiply: lower indices represent less significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - bitwise shared secret to multiply - * @param {string} [op_id=] - the base operation id to use when generating unique ids for communication. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare[]} bitwise sharing of the result, the length of the result will be bits1.length + bits2.length - */ - smult: function (jiff, bits1, bits2, op_id) { - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.smult', bits1[0].holders); - } + // Multi-octet tag - load + if ((tag & 0x1f) === 0x1f) { + var oct = tag; + tag = 0; + while ((oct & 0x80) === 0x80) { + oct = buf.readUInt8(fail); + if (buf.isError(oct)) return oct; - // copy to avoid aliasing problems during execution - bits1 = bits1.slice(); - bits2 = bits2.slice(); - - // bits1 will be the longest array, bits2 will be the shortest - var tmp = bits1.length > bits2.length ? bits1 : bits2; - bits2 = bits1.length > bits2.length ? bits2 : bits1; - bits1 = tmp; - - // Initialize the result - var offset = bits2.length === 1 ? -1 : 0; - var result = jiff.utils.many_secret_shares(bits1.length + bits2.length + offset, bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); - var deferreds = result.deferreds; - result = result.shares; - - // Resolve result when ready - var final_deferred = new jiff.helpers.Deferred(); - final_deferred.promise.then(jiff.utils.resolve_many_secrets.bind(null, deferreds)); - - // Loop over *shortest* array one bit at a time - jiff.utils.bit_combinator(final_deferred, 0, bits2.length, bits2, function (i, intermediate) { - var this_bit = bits2[i]; - var bit_mult = []; // add bits1 or 0 to the result according to this bit - for (var j = 0; j < bits1.length; j++) { - bit_mult[j] = this_bit.iif_else(bits1[j], 0, op_id + ':if_else:' + i + ':' + j); - } - bits1.unshift(0); // increase magnitude + tag <<= 7; + tag |= oct & 0x7f; + } + } else { + tag &= 0x1f; + } + var tagStr = der.tag[tag]; + + return { + cls: cls, + primitive: primitive, + tag: tag, + tagStr: tagStr + }; + } - if (i === 0) { - return bit_mult; - } + function derDecodeLen(buf, primitive, fail) { + var len = buf.readUInt8(fail); + if (buf.isError(len)) return len; - return jiff.protocols.bits.sadd(intermediate, bit_mult, op_id + ':bits.sadd:' + i); - }, function (intermediate) { - // promise-ify an array of intermediate results - var promises = []; - for (var i = 0; i < intermediate.length; i++) { - promises.push(intermediate[i].value); - } - return Promise.all(promises); - }, function (result) { - // identity - return result; - }); - - return result; - }, - /** - * Computes integer division of [secret bits 1] / [secret bits 2] - * @function sdiv - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {{quotient: module:jiff-client~JIFFClient#SecretShare[], remainder: module:jiff-client~JIFFClient#SecretShare[]}} the quotient and remainder bits arrays, note that - * the quotient array has the same length as bits1, - * and the remainder array has the same length as bits2 or bits1, whichever is smaller. - * Note: if bits2 represent 0, the returned result is the maximum - * number that fits in the number of bits (all 1), and the remainder - * is equal to bits1 - */ - sdiv: function (jiff, bits1, bits2, op_id) { - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.sdiv', bits1[0].holders); - } + // Indefinite form + if (!primitive && len === 0x80) return null; - // copy to avoid aliasing problems during execution - bits1 = bits1.slice(); - bits2 = bits2.slice(); - - // Initialize the result - var quotient = jiff.utils.many_secret_shares(bits1.length, bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); - var quotientDeferreds = quotient.deferreds; - quotient = quotient.shares; - - var remainder = jiff.utils.many_secret_shares(Math.min(bits1.length, bits2.length), bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); - var remainderDeferreds = remainder.deferreds; - remainder = remainder.shares; - - // Resolve result when ready - var final_deferred = new jiff.helpers.Deferred(); - final_deferred.promise.then(function (result) { - jiff.utils.resolve_many_secrets(remainderDeferreds, result); - }); - - var initial = []; // initial remainder - jiff.utils.bit_combinator(final_deferred, bits1.length - 1, -1, initial, function (i, _remainder) { - var iterationCounter = (bits1.length - i - 1); - - // add bit i to the head of remainder (least significant bit) - _remainder.unshift(bits1[i]); - - // Get the next bit of the quotient - // and conditionally subtract b from the - // intermediate remainder to continue - var sub = jiff.protocols.bits.ssub(_remainder, bits2, op_id + ':bits.ssub:' + iterationCounter); - var noUnderflow = sub.pop().inot(); // get the overflow bit, sub is now the result of subtraction - - // Get next bit of quotient - noUnderflow.wThen(quotientDeferreds[i].resolve); - - // Update remainder - for (var j = 0; j < _remainder.length; j++) { - // note, if noUnderflow, then |# bits in sub| <= |# bits in remainder| - _remainder[j] = noUnderflow.iif_else(sub[j], _remainder[j], op_id + ':if_else:' + iterationCounter + ':' + j); - } + // Definite form + if ((len & 0x80) === 0) { + // Short form + return len; + } - // Remainder cannot be greater than divisor at this point - while (_remainder.length > remainder.length) { - _remainder.pop(); - } + // Long form + var num = len & 0x7f; + if (num > 4) return buf.error('length octect is too long'); - return _remainder; - }, function (intermediate) { - // promise-ify an array of intermediate results - var promises = []; - for (var i = 0; i < intermediate.length; i++) { - promises.push(intermediate[i].value); - } - return Promise.all(promises); - }, function (result) { - // identity - return result; - }); - - return {quotient: quotient, remainder: remainder} - }, - /** - * Computes integer division of [secret bits] / constant - * @function cdivl - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - numerator: an array of secret shares of bits, starting from least to most significant bits - * @param {number} constant - the denominator number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {{quotient: module:jiff-client~JIFFClient#SecretShare[], remainder: module:jiff-client~JIFFClient#SecretShare[]}} the quotient and remainder bits arrays, note that - * the quotient array has the same length as bits, - * and the remainder array has the same length as - * constant or bits, whichever is smaller - * @throws if constant is 0. - */ - cdivl: function (jiff, bits, constant, op_id) { - if (!(bits[0].isConstant(constant))) { - throw new Error('parameter should be a number (bits.cdivl)'); - } - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.cdivl', bits[0].holders); - } + len = 0; + for (var i = 0; i < num; i++) { + len <<= 8; + var j = buf.readUInt8(fail); + if (buf.isError(j)) return j; + len |= j; + } - if (constant.toString() === '0') { - throw new Error('constant cannot be 0 in bits.cdiv'); - } + return len; + } + }, + { '../../asn1': 56, inherits: 218 } + ], + 65: [ + function (require, module, exports) { + var decoders = exports; + + decoders.der = require('./der'); + decoders.pem = require('./pem'); + }, + { './der': 64, './pem': 66 } + ], + 66: [ + function (require, module, exports) { + var inherits = require('inherits'); + var Buffer = require('buffer').Buffer; - // copy to avoid aliasing problems during execution - bits = bits.slice(); + var DERDecoder = require('./der'); - // special case, divide by 1 - if (constant.toString() === '1') { - return { - quotient: bits, - remainder: [new jiff.SecretShare(0, bits[0].holders, bits[0].threshold, bits[0].Zp)] - } - } + function PEMDecoder(entity) { + DERDecoder.call(this, entity); + this.enc = 'pem'; + } + inherits(PEMDecoder, DERDecoder); + module.exports = PEMDecoder; + + PEMDecoder.prototype.decode = function decode(data, options) { + var lines = data.toString().split(/[\r\n]+/g); + + var label = options.label.toUpperCase(); + + var re = /^-----(BEGIN|END) ([^-]+)-----$/; + var start = -1; + var end = -1; + for (var i = 0; i < lines.length; i++) { + var match = lines[i].match(re); + if (match === null) continue; + + if (match[2] !== label) continue; + + if (start === -1) { + if (match[1] !== 'BEGIN') break; + start = i; + } else { + if (match[1] !== 'END') break; + end = i; + break; + } + } + if (start === -1 || end === -1) throw new Error('PEM section not found for: ' + label); - // Initialize the result - var quotient = jiff.utils.many_secret_shares(bits.length, bits[0].holders, bits[0].threshold, bits[0].Zp); - var quotientDeferreds = quotient.deferreds; - quotient = quotient.shares; - - var constantLessBits = jiff.helpers.ceil(jiff.helpers.bLog(constant, 2)); - constantLessBits = parseInt(constantLessBits.toString(), 10); - var remainder = jiff.utils.many_secret_shares(Math.min(constantLessBits, bits.length), bits[0].holders, bits[0].threshold, bits[0].Zp); - var remainderDeferreds = remainder.deferreds; - remainder = remainder.shares; - - // Resolve result when ready - var final_deferred = new jiff.helpers.Deferred(); - final_deferred.promise.then(jiff.utils.resolve_many_secrets.bind(null, remainderDeferreds)); - - var initial = []; // initial remainder - jiff.utils.bit_combinator(final_deferred, bits.length - 1, -1, initial, function (i, _remainder) { - var iterationCounter = (bits.length - i - 1); - - // add bit i to the head of remainder (least significant bit) - _remainder.unshift(bits[i]); - - // Get the next bit of the quotient - // and conditionally subtract b from the - // intermediate remainder to continue - var sub = jiff.protocols.bits.csubl(_remainder, constant, op_id + ':bits.csubl:' + iterationCounter); - var noUnderflow = sub.pop().inot(); // get the overflow bit, sub is now the result of subtraction - - // Get next bit of quotient - noUnderflow.wThen(quotientDeferreds[i].resolve); - - // Update remainder - for (var j = 0; j < _remainder.length; j++) { - // note, if noUnderflow, then |# bits in sub| <= |# bits in remainder| - _remainder[j] = noUnderflow.iif_else(sub[j], _remainder[j], op_id + ':if_else:' + iterationCounter + ':' + j); - } + var base64 = lines.slice(start + 1, end).join(''); + // Remove excessive symbols + base64.replace(/[^a-z0-9\+\/=]+/gi, ''); - // Remainder cannot be greater than constant at this point - while (_remainder.length > remainder.length) { - _remainder.pop(); - } + var input = new Buffer(base64, 'base64'); + return DERDecoder.prototype.decode.call(this, input, options); + }; + }, + { './der': 64, buffer: 124, inherits: 218 } + ], + 67: [ + function (require, module, exports) { + var inherits = require('inherits'); + var Buffer = require('buffer').Buffer; - return _remainder; - }, function (intermediate) { - // promise-ify an array of intermediate results - var promises = []; - for (var i = 0; i < intermediate.length; i++) { - promises.push(intermediate[i].value); - } - return Promise.all(promises); - }, function (result) { - // identity - return result; - }); - - return {quotient: quotient, remainder: remainder}; - }, - /** - * Computes integer division of constant / [secret bits] - * @function cdivr - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {number} constant - the numerator number - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - denominator: an array of secret shares of bits, starting from least to most significant bits - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {{quotient: module:jiff-client~JIFFClient#SecretShare[], remainder: module:jiff-client~JIFFClient#SecretShare[]}} the quotient and remainder bits arrays, note that - * the quotient array has the same length as the number of bits in constant, - * and the remainder array has the same length as bits or constant, whichever is smaller. - * Note: if bits represent 0, the returned result is the maximum - * number that fits in its bits (all 1), and the remainder - * is equal to constant - */ - cdivr: function (jiff, constant, bits, op_id) { - if (!(bits[0].isConstant(constant))) { - throw new Error('parameter should be a number (bits.cdivr)'); - } - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.cdivr', bits[0].holders); - } + var asn1 = require('../../asn1'); + var base = asn1.base; - // copy to avoid aliasing problems during execution - bits = bits.slice(); + // Import DER constants + var der = asn1.constants.der; - // do not pad - var constant_bits = jiff.helpers.number_to_bits(constant); + function DEREncoder(entity) { + this.enc = 'der'; + this.name = entity.name; + this.entity = entity; - // Initialize the result - var quotient = jiff.utils.many_secret_shares(constant_bits.length, bits[0].holders, bits[0].threshold, bits[0].Zp); - var quotientDeferreds = quotient.deferreds; - quotient = quotient.shares; + // Construct base tree + this.tree = new DERNode(); + this.tree._init(entity.body); + } + module.exports = DEREncoder; - var remainder = jiff.utils.many_secret_shares(Math.min(constant_bits.length, bits.length), bits[0].holders, bits[0].threshold, bits[0].Zp); - var remainderDeferreds = remainder.deferreds; - remainder = remainder.shares; + DEREncoder.prototype.encode = function encode(data, reporter) { + return this.tree._encode(data, reporter).join(); + }; - // Resolve result when ready - var final_deferred = new jiff.helpers.Deferred(); - final_deferred.promise.then(jiff.utils.resolve_many_secrets.bind(null, remainderDeferreds)); + // Tree methods - var initial = []; // initial remainder - jiff.utils.bit_combinator(final_deferred, constant_bits.length - 1, -1, initial, function (i, _remainder) { - var iterationCounter = (constant_bits.length - i - 1); + function DERNode(parent) { + base.Node.call(this, 'der', parent); + } + inherits(DERNode, base.Node); - // add bit i to the head of remainder (least significant bit) - // turn into a secret without communication, just for typing - var cbit_share = new jiff.SecretShare(constant_bits[i], bits[0].holders, bits[0].threshold, bits[0].Zp); - _remainder.unshift(cbit_share); + DERNode.prototype._encodeComposite = function encodeComposite(tag, primitive, cls, content) { + var encodedTag = encodeTag(tag, primitive, cls, this.reporter); - // Get the next bit of the quotient - // and conditionally subtract b from the - // intermediate remainder to continue - var sub = jiff.protocols.bits.ssub(_remainder, bits, op_id + ':bits.ssub:' + iterationCounter); - var noUnderflow = sub.pop().inot(); // get the overflow bit, sub is now the result of subtraction + // Short form + if (content.length < 0x80) { + var header = new Buffer(2); + header[0] = encodedTag; + header[1] = content.length; + return this._createEncoderBuffer([header, content]); + } - // Get next bit of quotient - noUnderflow.wThen(quotientDeferreds[i].resolve); + // Long form + // Count octets required to store length + var lenOctets = 1; + for (var i = content.length; i >= 0x100; i >>= 8) lenOctets++; + + var header = new Buffer(1 + 1 + lenOctets); + header[0] = encodedTag; + header[1] = 0x80 | lenOctets; + + for (var i = 1 + lenOctets, j = content.length; j > 0; i--, j >>= 8) header[i] = j & 0xff; + + return this._createEncoderBuffer([header, content]); + }; + + DERNode.prototype._encodeStr = function encodeStr(str, tag) { + if (tag === 'bitstr') { + return this._createEncoderBuffer([str.unused | 0, str.data]); + } else if (tag === 'bmpstr') { + var buf = new Buffer(str.length * 2); + for (var i = 0; i < str.length; i++) { + buf.writeUInt16BE(str.charCodeAt(i), i * 2); + } + return this._createEncoderBuffer(buf); + } else if (tag === 'numstr') { + if (!this._isNumstr(str)) { + return this.reporter.error('Encoding of string type: numstr supports ' + 'only digits and space'); + } + return this._createEncoderBuffer(str); + } else if (tag === 'printstr') { + if (!this._isPrintstr(str)) { + return this.reporter.error( + 'Encoding of string type: printstr supports ' + + 'only latin upper and lower case letters, ' + + 'digits, space, apostrophe, left and rigth ' + + 'parenthesis, plus sign, comma, hyphen, ' + + 'dot, slash, colon, equal sign, ' + + 'question mark' + ); + } + return this._createEncoderBuffer(str); + } else if (/str$/.test(tag)) { + return this._createEncoderBuffer(str); + } else if (tag === 'objDesc') { + return this._createEncoderBuffer(str); + } else { + return this.reporter.error('Encoding of string type: ' + tag + ' unsupported'); + } + }; + + DERNode.prototype._encodeObjid = function encodeObjid(id, values, relative) { + if (typeof id === 'string') { + if (!values) return this.reporter.error('string objid given, but no values map found'); + if (!values.hasOwnProperty(id)) return this.reporter.error('objid not found in values map'); + id = values[id].split(/[\s\.]+/g); + for (var i = 0; i < id.length; i++) id[i] |= 0; + } else if (Array.isArray(id)) { + id = id.slice(); + for (var i = 0; i < id.length; i++) id[i] |= 0; + } - // Update remainder - for (var j = 0; j < _remainder.length; j++) { - // note, if noUnderflow, then |# bits in sub| <= |# bits in remainder| - _remainder[j] = noUnderflow.iif_else(sub[j], _remainder[j], op_id + ':if_else:' + iterationCounter + ':' + j); - } + if (!Array.isArray(id)) { + return this.reporter.error('objid() should be either array or string, ' + 'got: ' + JSON.stringify(id)); + } - // cannot be bigger than divisor at this point - while (_remainder.length > remainder.length) { - _remainder.pop(); - } + if (!relative) { + if (id[1] >= 40) return this.reporter.error('Second objid identifier OOB'); + id.splice(0, 2, id[0] * 40 + id[1]); + } - return _remainder; - }, function (intermediate) { - // promise-ify an array of intermediate results - var promises = []; - for (var i = 0; i < intermediate.length; i++) { - promises.push(intermediate[i].value); - } - return Promise.all(promises); - }, function (result) { - // identity - return result; - }); + // Count number of octets + var size = 0; + for (var i = 0; i < id.length; i++) { + var ident = id[i]; + for (size++; ident >= 0x80; ident >>= 7) size++; + } - return {quotient: quotient, remainder: remainder}; - } -}; -},{}],26:[function(require,module,exports){ -module.exports = { - /** - * Checks whether the given bitwise secret shared number and numeric constant are equal - * @function ceq - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of secret shares of bits, starting from least to most significant bits - * @param {number} constant - the constant number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {SecretShare|boolean} a secret share of 1 if parameters are equal, 0 otherwise. If result is known - * (e.g. constant has a greater non-zero bit than bits' most significant bit), the result is - * returned immediately as a boolean - */ - ceq: function (jiff, bits, constant, op_id) { - if (!(bits[0].isConstant(constant))) { - throw new Error('parameter should be a number (bits.ceq)'); - } - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.ceq', bits[0].holders); - } - var result = jiff.protocols.bits.cneq(bits, constant, op_id); - if (result === true || result === false) { - return !result; - } - return result.inot(); - }, - /** - * Checks whether the given bitwise secret shared number and numeric constant are not equal - * @function cneq - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of secret shares of bits, starting from least to most significant bits - * @param {number} constant - the constant number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {SecretShare|boolean} a secret share of 1 if parameters are not equal, 0 otherwise. If result is known - * (e.g. constant has a greater non-zero bit than bits' most significant bit), the result is - * returned immediately as a boolean - */ - cneq: function (jiff, bits, constant, op_id) { - if (!(bits[0].isConstant(constant))) { - throw new Error('parameter should be a number (bits.cneq)'); - } + var objid = new Buffer(size); + var offset = objid.length - 1; + for (var i = id.length - 1; i >= 0; i--) { + var ident = id[i]; + objid[offset--] = ident & 0x7f; + while ((ident >>= 7) > 0) objid[offset--] = 0x80 | (ident & 0x7f); + } - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.cneq', bits[0].holders); - } + return this._createEncoderBuffer(objid); + }; - // copy to avoid aliasing problems during execution - bits = bits.slice(); + function two(num) { + if (num < 10) return '0' + num; + else return num; + } - var constant_bits = jiff.helpers.number_to_bits(constant, bits.length); - if (constant_bits.length > bits.length) { - // Optimization: if constant has more bits, one of them must be 1, constant must be greater than bits. - return true; - } + DERNode.prototype._encodeTime = function encodeTime(time, tag) { + var str; + var date = new Date(time); - var deferred = new jiff.helpers.Deferred(); - var result = new jiff.SecretShare(deferred.promise, bits[0].holders, bits[0].threshold, bits[0].Zp); - - // big or of bitwise XORs - var initial = bits[0].icxor_bit(constant_bits[0]); - jiff.utils.bit_combinator(deferred, 1, bits.length, initial, function (i, prev) { - var xor = bits[i].icxor_bit(constant_bits[i]); - xor = prev.isor_bit(xor, op_id + ':sor_bit:' + (i - 1)); - return xor; - }); - - return result; - }, - /** - * Checks whether given secret shared bits are greater than the given constant - * @function cgt - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits - * @param {number} constant - the constant number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {SecretShare|boolean} a secret share of 1 if bits are greater than constant, 0 otherwise, if result is known - * (e.g. constant has greater non-zero bit than bits' most significant bit), the result is - * returned immediately as a boolean - */ - cgt: function (jiff, bits, constant, op_id) { - if (!(bits[0].isConstant(constant))) { - throw new Error('parameter should be a number (bits.cgt)'); - } - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.cgt', bits[0].holders); - } - return jiff.protocols.bits.cgteq(bits, constant + 1, op_id); - }, - /** - * Checks whether given secret shared bits are greater or equal to the given constant - * @function cgteq - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits - * @param {number} constant - the constant number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {SecretShare|boolean} a secret share of 1 if bits are greater or equal to constant, 0 otherwise, if result is known - * (e.g. constant has greater non-zero bit than bits' most significant bit or constant is zero), the result is - * returned immediately as a boolean - */ - cgteq: function (jiff, bits, constant, op_id) { - if (!(bits[0].isConstant(constant))) { - throw new Error('parameter should be a number (bits.cgteq)'); - } + if (tag === 'gentime') { + str = [two(date.getFullYear()), two(date.getUTCMonth() + 1), two(date.getUTCDate()), two(date.getUTCHours()), two(date.getUTCMinutes()), two(date.getUTCSeconds()), 'Z'].join(''); + } else if (tag === 'utctime') { + str = [two(date.getFullYear() % 100), two(date.getUTCMonth() + 1), two(date.getUTCDate()), two(date.getUTCHours()), two(date.getUTCMinutes()), two(date.getUTCSeconds()), 'Z'].join(''); + } else { + this.reporter.error('Encoding ' + tag + ' time is not supported yet'); + } - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.cgteq', bits[0].holders); - } + return this._encodeStr(str, 'octstr'); + }; - // copy to avoid aliasing problems during execution - bits = bits.slice(); + DERNode.prototype._encodeNull = function encodeNull() { + return this._createEncoderBuffer(''); + }; - // Optimization: the bits are a share of non-negative number, if constant <= 0, return true - if (constant.toString().startsWith('-') || constant.toString() === '0') { - return true; - } + DERNode.prototype._encodeInt = function encodeInt(num, values) { + if (typeof num === 'string') { + if (!values) return this.reporter.error('String int or enum given, but no values map'); + if (!values.hasOwnProperty(num)) { + return this.reporter.error("Values map doesn't contain: " + JSON.stringify(num)); + } + num = values[num]; + } - // decompose result into bits - var constant_bits = jiff.helpers.number_to_bits(constant, bits.length); - if (constant_bits.length > bits.length) { - // Optimization: if constant has more bits, one of them must be 1, constant must be greater than bits. - return false; - } + // Bignum, assume big endian + if (typeof num !== 'number' && !Buffer.isBuffer(num)) { + var numArray = num.toArray(); + if (!num.sign && numArray[0] & 0x80) { + numArray.unshift(0); + } + num = new Buffer(numArray); + } - // initialize result - var deferred = new jiff.helpers.Deferred(); - var result = new jiff.SecretShare(deferred.promise, bits[0].holders, bits[0].threshold, bits[0].Zp); - - // Subtract bits2 from bits1, only keeping track of borrow - var borrow = bits[0].inot().icmult(constant_bits[0]); - - // compute one bit at a time, propagating borrow - jiff.utils.bit_combinator(deferred, 1, bits.length, borrow, function (i, borrow) { - var xor = bits[i].icxor_bit(constant_bits[i]); - var andNot = bits[i].inot().icmult(constant_bits[i]); - - // save and update borrow - borrow = xor.inot().ismult(borrow, op_id + ':smult:' + (i - 1)); - return borrow.isadd(andNot); - }); - - return result.inot(); - }, - /** - * Checks whether given secret shared bits are less than the given constant - * @function clt - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits - * @param {number} constant - the constant number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {SecretShare|boolean} a secret share of 1 if bits are less than the constant, 0 otherwise, if result is known - * (e.g. constant has greater non-zero bit than bits' most significant bit), the result is - * returned immediately as a boolean - */ - clt: function (jiff, bits, constant, op_id) { - if (!(bits[0].isConstant(constant))) { - throw new Error('parameter should be a number (bits.clt)'); - } - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.clt', bits[0].holders); - } - var result = jiff.protocols.bits.cgteq(bits, constant, op_id); - if (result === true || result === false) { - return !result; - } - return result.inot(); - }, - /** - * Checks whether given secret shared bits are less or equal to the given constant - * @function clteq - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits - * @param {number} constant - the constant number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {SecretShare|boolean} a secret share of 1 if bits are less or equal to constant, 0 otherwise, if result is known - * (e.g. constant has greater non-zero bit than bits' most significant bit), the result is - * returned immediately as a boolean - */ - clteq: function (jiff, bits, constant, op_id) { - if (!(bits[0].isConstant(constant))) { - throw new Error('parameter should be a number (bits.clteq)'); - } - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.clteq', bits[0].holders); - } - var result = jiff.protocols.bits.cgt(bits, constant, op_id); - if (result === true || result === false) { - return !result; - } - return result.inot(); - }, - /** - * Checks whether the two given bitwise secret shared numbers are equal - * @function seq - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if bits are equal, 0 otherwise - */ - seq: function (jiff, bits1, bits2, op_id) { - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.seq', bits1[0].holders); - } - return jiff.protocols.bits.sneq(bits1, bits2, op_id).inot(); - }, - /** - * Checks whether the two given bitwise secret shared numbers are not equal - * @function sneq - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if bits are not equal, 0 otherwise - */ - sneq: function (jiff, bits1, bits2, op_id) { - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.sneq', bits1[0].holders); - } + if (Buffer.isBuffer(num)) { + var size = num.length; + if (num.length === 0) size++; - var tmp = bits1.length > bits2.length ? bits1 : bits2; - bits2 = bits1.length > bits2.length ? bits2 : bits1; // shortest array - bits1 = tmp; // longest array - - // copy to avoid aliasing problems during execution - bits1 = bits1.slice(); - bits2 = bits2.slice(); - - // initialize result - var deferred = new jiff.helpers.Deferred(); - var result = new jiff.SecretShare(deferred.promise, bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); - - // big or of bitwise XORs - var initial = bits1[0].isxor_bit(bits2[0], op_id + ':sxor_bit:initial'); - jiff.utils.bit_combinator(deferred, 1, bits1.length, initial, function (i, prev) { - var next; - if (i < bits2.length) { - var xor = bits1[i].isxor_bit(bits2[i], op_id + ':sxor_bit:' + (i - 1)); - next = prev.isor_bit(xor, op_id + ':sor_bit:' + (i - 1)); - } else { - // xor is equal to bits1[i] since bits2[i] is zero - next = prev.isor_bit(bits1[i], op_id + ':sor_bit:' + (i - 1)); - } - return next; - }); - - return result; - }, - /** - * Checks whether the first given bitwise secret shared number is greater than the second bitwise secret shared number - * @function sgt - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is greater than the second, 0 otherwise - */ - sgt: function (jiff, bits1, bits2, op_id) { - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.sgt', bits1[0].holders); - } + var out = new Buffer(size); + num.copy(out); + if (num.length === 0) out[0] = 0; + return this._createEncoderBuffer(out); + } - var gteq = jiff.protocols.bits.sgteq(bits1, bits2, op_id + ':bits.sgteq'); - var neq = jiff.protocols.bits.sneq(bits1, bits2, op_id + ':bits.sneq'); - return gteq.ismult(neq, op_id + ':smult'); - }, - /** - * Checks whether the first given bitwise secret shared number is greater than or equal to the second bitwise secret shared number - * @function sgteq - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is greater or equal to the second, 0 otherwise - */ - sgteq: function (jiff, bits1, bits2, op_id) { - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.sgteq', bits1[0].holders); - } + if (num < 0x80) return this._createEncoderBuffer(num); - // copy to avoid aliasing problems during execution - bits1 = bits1.slice(); - bits2 = bits2.slice(); - - // initialize result - var deferred = new jiff.helpers.Deferred(); - var result = new jiff.SecretShare(deferred.promise, bits1[0].holders, Math.max(bits1[0].threshold, bits2[0].threshold), bits1[0].Zp); - - // Subtract bits2 from bits1, only keeping track of borrow - var borrow = bits1[0].inot().ismult(bits2[0], op_id + ':smult:initial'); - var n = Math.max(bits1.length, bits2.length); - jiff.utils.bit_combinator(deferred, 1, n, borrow, function (i, borrow) { - if (i < bits1.length && i < bits2.length) { - var xor = bits1[i].isxor_bit(bits2[i], op_id + ':sxor_bit1:' + (i - 1)); - var andNot = bits1[i].inot().ismult(bits2[i], op_id + ':smult1:' + (i - 1)); - - // save and update borrow - borrow = xor.inot().ismult(borrow, op_id + ':smult2:' + (i - 1)); - borrow = borrow.isadd(andNot); - } else if (i < bits1.length) { - // xor is equal to the value of bits1[i], andNot is equal to 0, since bits[2] is all zeros here - borrow = bits1[i].inot().ismult(borrow, op_id + ':smult1:' + (i - 1)); - } else { // i < bits2.length - // xor and andNot are equal to the value of bits2[i] - borrow = bits2[i].inot().ismult(borrow, op_id + ':smult1:' + (i - 1)); - borrow = borrow.isadd(bits2[i]); - } + if (num < 0x100) return this._createEncoderBuffer([0, num]); - return borrow; - }); - - return result.inot(); - }, - /** - * Checks whether the first given bitwise secret shared number is less than the second bitwise secret shared number - * @function slt - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is less than the second, 0 otherwise - */ - slt: function (jiff, bits1, bits2, op_id) { - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.slt', bits1[0].holders); - } - var result = jiff.protocols.bits.sgteq(bits1, bits2, op_id); - return result.inot(); - }, - /** - * Checks whether the first given bitwise secret shared number is less or equal to the second bitwise secret shared number - * @function slteq - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits1 - an array of secret shares of bits, starting from least to most significant bits - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits2 - the second bitwise shared number - * @param {string} [op_id=] - the base operation id to use when generating unique ids for multiplications. - * default value should suffice when the code of all parties executes all instructions - * in the same exact order, otherwise, a unique base name is needed here - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of 1 if the first number is less than or equal to the second, 0 otherwise - */ - slteq: function (jiff, bits1, bits2, op_id) { - if (op_id == null) { - op_id = jiff.counters.gen_op_id('bits.slteq', bits1[0].holders); - } - var result = jiff.protocols.bits.sgt(bits1, bits2, op_id); - return result.inot(); - } -}; -},{}],27:[function(require,module,exports){ -module.exports = { - /** - * Retrieves preprocessed rejection_sampling bits or performs the rejection sampling on the fly if crypto_provider is enabled - * @function rejection_sampling - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff instance - * @param {number} [lower_bound=0] - the lower bound, included (can be a bigNumber if using bigNumber extension) - * @param {number} [upper_bound=jiff-instance.Zp] - the upper bound, excluded (can be a bigNumber if using bigNumber extension) - * @param {number} [threshold=parties.length] - the threshold of the resulting shares after sampling - * @param {Array} [parties=all_parties] - array of party ids that want to receive (or compute if needed) the sampling shares, by default, this includes all parties - * @param {number} [Zp=jiff-instance.Zp] - the mod (if null then the default Zp for the instance is used) - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly - * @returns {SecretShare[]} an array of secret shares, each representing a bit from the sampled value (from least to most significant) - */ - rejection_sampling: function (jiff, lower_bound, upper_bound, threshold, parties, Zp, op_id) { - // defaults - if (parties == null) { - parties = []; - for (var i = 1; i <= jiff.party_count; i++) { - parties.push(i); - } - } else { - jiff.helpers.sort_ids(parties); - } + var size = 1; + for (var i = num; i >= 0x100; i >>= 8) size++; - if (op_id == null) { - op_id = jiff.counters.gen_op_id('rejection_sampling', parties); - } + var out = new Array(size); + for (var i = out.length - 1; i >= 0; i--) { + out[i] = num & 0xff; + num >>= 8; + } + if (out[0] & 0x80) { + out.unshift(0); + } - // try to get preprocessed samples - var result = jiff.get_preprocessing(op_id); - if (result != null && result.ondemand !== true) { - return result; - } + return this._createEncoderBuffer(new Buffer(out)); + }; - // Not ready, either preprocess it on demand, or use crypto provider! - lower_bound = lower_bound ? lower_bound : 0; - upper_bound = upper_bound ? upper_bound : jiff.Zp; - if (threshold == null) { - threshold = parties.length; - } - if (Zp == null) { - Zp = jiff.Zp; - } + DERNode.prototype._encodeBool = function encodeBool(value) { + return this._createEncoderBuffer(value ? 0xff : 0); + }; - var finalLength = jiff.helpers.ceil(jiff.helpers.bLog(upper_bound, 2)); - finalLength = parseInt(finalLength.toString(), 10); - finalLength = Math.max(finalLength, 1); // special case: when upper_bound is 1! - - var many_shares = jiff.utils.many_secret_shares(finalLength, parties, threshold, Zp); - var final_deferreds = many_shares.deferreds; - - // Crypto provider - if (result == null) { - var promise = jiff.from_crypto_provider('numbers', parties, threshold, Zp, op_id, { - max: upper_bound, - min: lower_bound, - bitLength: finalLength, - count: 1 - }); - promise.then(function (result) { - jiff.utils.resolve_many_secrets(final_deferreds, result['shares']); - }); - } else { // preprocess on demand - delete jiff.preprocessing_table[op_id]; - jiff.preprocessing('rejection_sampling', 1, null, threshold, parties, parties, Zp, [op_id], { - lower_bound: lower_bound, - upper_bound: upper_bound - }); - jiff.executePreprocessing(function () { - jiff.utils.resolve_many_secrets(final_deferreds, jiff.get_preprocessing(op_id)); - }); - } + DERNode.prototype._use = function use(entity, obj) { + if (typeof entity === 'function') entity = entity(obj); + return entity._getEncoder('der').tree; + }; - return many_shares.shares; - }, - - /** - * Creates a secret share of the number represented by the given array of secret shared bits. - * Requires no communication, only local operations - * @function bit_composition - * @ignore - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits. - * @returns {module:jiff-client~JIFFClient#SecretShare} a secret share of the number represented by bits. - */ - bit_composition: function (bits) { - var result = bits[0]; - var pow = 1; - for (var i = 1; i < bits.length; i++) { - pow = pow * 2; - result = result.isadd(bits[i].icmult(pow)); - } - return result; - } -}; -},{}],28:[function(require,module,exports){ -module.exports = { - /** - * Share a number as an array of secret bits - * This takes the same parameters as jiff-instance.share, but returns an array of secret bit shares per sending party. - * Each bit array starts with the least significant bit at index 0, and most significant bit at index length-1 - * @function share_bits - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {number} secret - the number to share (this party's input) - * @param {number} [bit_length=jiff_instance.Zp] - the number of generated bits, if the secret has less bits, it will be - * padded with zeros - * @param {number} [threshold=receivers_list.length] - threshold of each shared bit - * @param {Array} [receivers_list=all_parties] - receivers of every bits - * @param {Array} [senders_list=all_parties] - senders of evey bit - * @param {number} [Zp=jiff_instance.Zp] - the field of sharing for every bit - * @param {string|number} [share_id=auto_gen()] - synchronization id - * @returns {object} a map (of size equal to the number of parties) - * where the key is the party id (from 1 to n) - * and the value is an array of secret shared bits - */ - share_bits: function (jiff, secret, bit_length, threshold, receivers_list, senders_list, Zp, share_id) { - var i; - if (Zp == null) { - Zp = jiff.Zp; - } - if (receivers_list == null) { - receivers_list = []; - for (i = 1; i <= jiff.party_count; i++) { - receivers_list.push(i); - } - } else { - jiff.helpers.sort_ids(receivers_list); - } - if (senders_list == null) { - senders_list = []; - for (i = 1; i <= jiff.party_count; i++) { - senders_list.push(i); - } - } else { - jiff.helpers.sort_ids(senders_list); - } + DERNode.prototype._skipDefault = function skipDefault(dataBuffer, reporter, parent) { + var state = this._baseState; + var i; + if (state['default'] === null) return false; - if (share_id == null) { - share_id = jiff.counters.gen_op_id2('share_bits', receivers_list, senders_list); - } + var data = dataBuffer.join(); + if (state.defaultBuffer === undefined) state.defaultBuffer = this._encodeValue(state['default'], reporter, parent).join(); - if (bit_length == null) { - bit_length = Zp.toString(2).length; - } + if (data.length !== state.defaultBuffer.length) return false; - // to allow for secret=null when party is not a sender - var local_bits = []; - if (secret != null) { - local_bits = jiff.helpers.number_to_bits(secret, bit_length); - } + for (i = 0; i < data.length; i++) if (data[i] !== state.defaultBuffer[i]) return false; - var shared_bits = {}; - for (i = 0; i < senders_list.length; i++) { - shared_bits[senders_list[i]] = []; - } + return true; + }; - for (i = 0; i < bit_length; i++) { - var round = jiff.internal_share(local_bits[i], threshold, receivers_list, senders_list, Zp, share_id + ':' + i); - for (var si = 0; si < senders_list.length; si++) { - var pid = senders_list[si]; - shared_bits[pid].push(round[pid]); - } - } + // Utility methods - return shared_bits; - }, - /** - * Opens the given array of secret shared bits. - * This works regardless of whether the represented value fit inside the corresponding field or not - * @function open_bits - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare[]} bits - an array of the secret shares of bits, starting from least to most significant bits - * @param {number[]} parties - parties to open (same as jiff_instance.open) - * @param {string|number} [op_id=auto_gen()] - same as jiff_instance.open - * @returns {promise} a promise to the number represented by bits - */ - open_bits: function (jiff, bits, parties, op_id) { - // Default values - if (parties == null) { - parties = []; - for (var p = 1; p <= jiff.party_count; p++) { - parties.push(p); - } - } else { - jiff.helpers.sort_ids(parties); - } + function encodeTag(tag, primitive, cls, reporter) { + var res; - // Compute operation ids (one for each party that will receive a result - if (op_id == null) { - op_id = jiff.counters.gen_op_id2('bits.open', parties, bits[0].holders); - } + if (tag === 'seqof') tag = 'seq'; + else if (tag === 'setof') tag = 'set'; - var opened_bits = []; - for (var i = 0; i < bits.length; i++) { - opened_bits[i] = jiff.internal_open(bits[i], parties, op_id + ':' + i); - } + if (der.tagByName.hasOwnProperty(tag)) res = der.tagByName[tag]; + else if (typeof tag === 'number' && (tag | 0) === tag) res = tag; + else return reporter.error('Unknown tag: ' + tag); - return Promise.all(opened_bits).then(function (bits) { - return jiff.helpers.bits_to_number(bits, bits.length); - }); - }, - /** - * Receives an opening of an array of secret bits without owning shares of the underlying value. - * Similar to jiff.receive_open() but for bits. - * This works regardless of whether the represented value fit inside the corresponding field or not - * @function receive_open_bits - * @memberOf jiff-instance.protocols.bits - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {Array} senders - an array with party ids (1 to n) specifying the parties sending the shares - * @param {Array} [receivers=all_parties] - an array with party ids (1 to n) specifying the parties receiving the result - * @param {number} [count=ceil(log2(Zp))] - the number of bits being opened - * @param {number} [threshold=parties.length] - the min number of parties needed to reconstruct the secret, defaults to all the senders - * @param {number} [Zp=jiff_instance.Zp] - the mod (if null then the default Zp for the instance is used) - * @param {string|number|object} [op_id=auto_gen()] - unique and consistent synchronization id between all parties - * @returns {promise} a (JQuery) promise to the open value of the secret - */ - receive_open_bits: function (jiff, senders, receivers, count, threshold, Zp, op_id) { - if (senders == null) { - throw new Error('Must provide "senders" parameter in receive_open'); - } - jiff.helpers.sort_ids(senders); - // Default values - if (receivers == null) { - receivers = []; - for (i = 1; i <= jiff.party_count; i++) { - receivers.push(i); - } - } else { - jiff.helpers.sort_ids(receivers); - } + if (res >= 0x1f) return reporter.error('Multi-octet tag encoding unsupported'); - if (op_id == null) { - op_id = jiff.counters.gen_op_id2('bits.open', receivers, senders); - } + if (!primitive) res |= 0x20; - if (count == null) { - if (Zp == null) { - Zp = jiff.Zp; - } - count = Zp.toString(2).length; - } + res |= der.tagClassByName[cls || 'universal'] << 6; - var opened_bits = []; - for (var i = 0; i < count; i++) { - opened_bits[i] = jiff.receive_open(senders, receivers, threshold, Zp, op_id + ':' + i); - } + return res; + } + }, + { '../../asn1': 56, buffer: 124, inherits: 218 } + ], + 68: [ + function (require, module, exports) { + var encoders = exports; + + encoders.der = require('./der'); + encoders.pem = require('./pem'); + }, + { './der': 67, './pem': 69 } + ], + 69: [ + function (require, module, exports) { + var inherits = require('inherits'); - return Promise.all(opened_bits).then(function (bits) { - return jiff.helpers.bits_to_number(bits, bits.length); - }); - } -}; -},{}],29:[function(require,module,exports){ -// Generic version of operations -module.exports = function (SecretShare) { - /** - * bitwise-XOR with a constant (BOTH BITS). - * @method cxor_bit - * @param {number} cst - the constant bit to XOR with (0 or 1). - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.cxor_bit = function (cst) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (^)'); - } - if (!this.jiff.share_helpers['binary'](cst)) { - throw new Error('parameter should be binary (^)'); - } + var DEREncoder = require('./der'); - return this.icadd(cst).issub(this.icmult(cst).icmult(2)); - }; + function PEMEncoder(entity) { + DEREncoder.call(this, entity); + this.enc = 'pem'; + } + inherits(PEMEncoder, DEREncoder); + module.exports = PEMEncoder; + + PEMEncoder.prototype.encode = function encode(data, options) { + var buf = DEREncoder.prototype.encode.call(this, data); + + var p = buf.toString('base64'); + var out = ['-----BEGIN ' + options.label + '-----']; + for (var i = 0; i < p.length; i += 64) out.push(p.slice(i, i + 64)); + out.push('-----END ' + options.label + '-----'); + return out.join('\n'); + }; + }, + { './der': 67, inherits: 218 } + ], + 70: [ + function (require, module, exports) { + (function (module, exports) { + 'use strict'; + + // Utils + function assert(val, msg) { + if (!val) throw new Error(msg || 'Assertion failed'); + } - /** - * bitwise-OR with a constant (BOTH BITS). - * @method cor_bit - * @param {number} cst - the constant bit to OR with (0 or 1). - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.cor_bit = function (cst) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (|)'); - } - if (!this.jiff.share_helpers['binary'](cst)) { - throw new Error('parameter should be binary (|)'); - } + // Could use `inherits` module, but don't want to move from single file + // architecture yet. + function inherits(ctor, superCtor) { + ctor.super_ = superCtor; + var TempCtor = function () {}; + TempCtor.prototype = superCtor.prototype; + ctor.prototype = new TempCtor(); + ctor.prototype.constructor = ctor; + } - return this.icadd(cst).issub(this.icmult(cst)); - }; + // BN - /** - * bitwise-XOR of two secret shares OF BITS. - * @method sxor_bit - * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to XOR with. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this < o, and 0 otherwise. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.sxor_bit = function (o, op_id) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (^)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (^)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (^)'); - } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('sxor_bit', this.holders); - } + function BN(number, base, endian) { + if (BN.isBN(number)) { + return number; + } - return this.isadd(o).issub(this.ismult(o, op_id + ':smult1').icmult(2)); - }; + this.negative = 0; + this.words = null; + this.length = 0; - /** - * OR of two secret shares OF BITS. - * @method sor_bit - * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to OR with. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this < o, and 0 otherwise. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.sor_bit = function (o, op_id) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (|)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (|)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (|)'); - } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('sor_bit', this.holders); - } + // Reduction context + this.red = null; - return this.isadd(o).issub(this.ismult(o, op_id + ':smult1')); - }; + if (number !== null) { + if (base === 'le' || base === 'be') { + endian = base; + base = 10; + } - /** - * Negation of a bit. - * This has to be a share of a BIT in order for this to work properly. - * @method not - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result (negated bit). - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.not = function () { - return this.icmult(-1).icadd(1); - }; + this._init(number || 0, base || 10, endian || 'be'); + } + } + if (typeof module === 'object') { + module.exports = BN; + } else { + exports.BN = BN; + } - /** - * Simulate an oblivious If-else statement with a single return value. - * Should be called on a secret share of a bit: 0 representing false, and 1 representing true - * If this is a share of 1, a new sharing of the element represented by the first parameter is returned, - * otherwise, a new sharing of the second is returned. - * @method if_else - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - * @param {module:jiff-client~JIFFClient#SecretShare | number} trueVal - the value/share to return if this is a sharing of 1. - * @param {module:jiff-client~JIFFClient#SecretShare | number} falseVal - the value/share to return if this is a sharing of 0. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} a new sharing of the result of the if. - * - * @example - * // a and b are secret shares - * // cmp will be a secret share of either 1 or 0, depending on whether a or b is greater - * var cmp = a.gt(b); - * - * // max is set to the greater value, without revealing the value or the result of the inequality - * var max = cmp.if_else(a, b); - */ - SecretShare.prototype.if_else = function (trueVal, falseVal, op_id) { - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('if_else', this.holders); - } + BN.BN = BN; + BN.wordSize = 26; + + var Buffer; + try { + Buffer = require('buffer').Buffer; + } catch (e) {} + + BN.isBN = function isBN(num) { + if (num instanceof BN) { + return true; + } + + return num !== null && typeof num === 'object' && num.constructor.wordSize === BN.wordSize && Array.isArray(num.words); + }; + + BN.max = function max(left, right) { + if (left.cmp(right) > 0) return left; + return right; + }; + + BN.min = function min(left, right) { + if (left.cmp(right) < 0) return left; + return right; + }; + + BN.prototype._init = function init(number, base, endian) { + if (typeof number === 'number') { + return this._initNumber(number, base, endian); + } + + if (typeof number === 'object') { + return this._initArray(number, base, endian); + } + + if (base === 'hex') { + base = 16; + } + assert(base === (base | 0) && base >= 2 && base <= 36); + + number = number.toString().replace(/\s+/g, ''); + var start = 0; + if (number[0] === '-') { + start++; + } + + if (base === 16) { + this._parseHex(number, start); + } else { + this._parseBase(number, base, start); + } + + if (number[0] === '-') { + this.negative = 1; + } + + this.strip(); + + if (endian !== 'le') return; + + this._initArray(this.toArray(), base, endian); + }; + + BN.prototype._initNumber = function _initNumber(number, base, endian) { + if (number < 0) { + this.negative = 1; + number = -number; + } + if (number < 0x4000000) { + this.words = [number & 0x3ffffff]; + this.length = 1; + } else if (number < 0x10000000000000) { + this.words = [number & 0x3ffffff, (number / 0x4000000) & 0x3ffffff]; + this.length = 2; + } else { + assert(number < 0x20000000000000); // 2 ^ 53 (unsafe) + this.words = [number & 0x3ffffff, (number / 0x4000000) & 0x3ffffff, 1]; + this.length = 3; + } + + if (endian !== 'le') return; + + // Reverse the bytes + this._initArray(this.toArray(), base, endian); + }; + + BN.prototype._initArray = function _initArray(number, base, endian) { + // Perhaps a Uint8Array + assert(typeof number.length === 'number'); + if (number.length <= 0) { + this.words = [0]; + this.length = 1; + return this; + } + + this.length = Math.ceil(number.length / 3); + this.words = new Array(this.length); + for (var i = 0; i < this.length; i++) { + this.words[i] = 0; + } + + var j, w; + var off = 0; + if (endian === 'be') { + for (i = number.length - 1, j = 0; i >= 0; i -= 3) { + w = number[i] | (number[i - 1] << 8) | (number[i - 2] << 16); + this.words[j] |= (w << off) & 0x3ffffff; + this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff; + off += 24; + if (off >= 26) { + off -= 26; + j++; + } + } + } else if (endian === 'le') { + for (i = 0, j = 0; i < number.length; i += 3) { + w = number[i] | (number[i + 1] << 8) | (number[i + 2] << 16); + this.words[j] |= (w << off) & 0x3ffffff; + this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff; + off += 24; + if (off >= 26) { + off -= 26; + j++; + } + } + } + return this.strip(); + }; + + function parseHex(str, start, end) { + var r = 0; + var len = Math.min(str.length, end); + for (var i = start; i < len; i++) { + var c = str.charCodeAt(i) - 48; + + r <<= 4; + + // 'a' - 'f' + if (c >= 49 && c <= 54) { + r |= c - 49 + 0xa; + + // 'A' - 'F' + } else if (c >= 17 && c <= 22) { + r |= c - 17 + 0xa; + + // '0' - '9' + } else { + r |= c & 0xf; + } + } + return r; + } - var const1 = this.isConstant(trueVal); - var const2 = this.isConstant(falseVal); - if (const1 && const2) { - return this.icmult(trueVal).isadd(this.inot().icmult(falseVal)); - } else if (const1) { - return this.inot().ismult(falseVal.icsub(trueVal), op_id + ':smult').icadd(trueVal); - } else if (const2) { - return this.ismult(trueVal.icsub(falseVal), op_id + ':smult').icadd(falseVal); - } else { - return this.ismult(trueVal.issub(falseVal), op_id + ':smult').isadd(falseVal); - } - }; -}; -},{}],30:[function(require,module,exports){ -// Generic version of operations -module.exports = function (SecretShare) { - /** - * Checks if the given parameter is a constant, used to determine whether constant or secret - * operations should be executed when the generic version of an operation is called - * @method isConstant - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - * @param {number/object} o - the parameter to determine. - * @return {boolean} true if o is a valid constant, false otherwise. - */ - SecretShare.prototype.isConstant = function (o) { - return typeof(o) === 'number'; - }; + BN.prototype._parseHex = function _parseHex(number, start) { + // Create possibly bigger array to ensure that it fits the number + this.length = Math.ceil((number.length - start) / 6); + this.words = new Array(this.length); + for (var i = 0; i < this.length; i++) { + this.words[i] = 0; + } + + var j, w; + // Scan 24-bit chunks and add them to the number + var off = 0; + for (i = number.length - 6, j = 0; i >= start; i -= 6) { + w = parseHex(number, i, i + 6); + this.words[j] |= (w << off) & 0x3ffffff; + // NOTE: `0x3fffff` is intentional here, 26bits max shift + 24bit hex limb + this.words[j + 1] |= (w >>> (26 - off)) & 0x3fffff; + off += 24; + if (off >= 26) { + off -= 26; + j++; + } + } + if (i + 6 !== start) { + w = parseHex(number, start, i + 6); + this.words[j] |= (w << off) & 0x3ffffff; + this.words[j + 1] |= (w >>> (26 - off)) & 0x3fffff; + } + this.strip(); + }; + + function parseBase(str, start, end, mul) { + var r = 0; + var len = Math.min(str.length, end); + for (var i = start; i < len; i++) { + var c = str.charCodeAt(i) - 48; + + r *= mul; + + // 'a' + if (c >= 49) { + r += c - 49 + 0xa; + + // 'A' + } else if (c >= 17) { + r += c - 17 + 0xa; + + // '0' - '9' + } else { + r += c; + } + } + return r; + } - /** - * Generic Addition. - * Uses either the constant or secret version of this operator depending on type of paramter. - * @method add - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * - * @example - * var shares = jiffClient.share(input); - * // this will add two secret shared values together - * var sum1 = shares[1].add(shares[2]); - * // this will add 3 to the secret input from party 1 - * var sum2 = shares[1].add(3); - * // both sum1 and sum2 are SecretShares - */ - SecretShare.prototype.add = function (o) { - if (this.isConstant(o)) { - return this.cadd(o); - } - return this.sadd(o); - }; + BN.prototype._parseBase = function _parseBase(number, base, start) { + // Initialize as zero + this.words = [0]; + this.length = 1; + + // Find length of limb in base + for (var limbLen = 0, limbPow = 1; limbPow <= 0x3ffffff; limbPow *= base) { + limbLen++; + } + limbLen--; + limbPow = (limbPow / base) | 0; + + var total = number.length - start; + var mod = total % limbLen; + var end = Math.min(total, total - mod) + start; + + var word = 0; + for (var i = start; i < end; i += limbLen) { + word = parseBase(number, i, i + limbLen, base); + + this.imuln(limbPow); + if (this.words[0] + word < 0x4000000) { + this.words[0] += word; + } else { + this._iaddn(word); + } + } + + if (mod !== 0) { + var pow = 1; + word = parseBase(number, i, number.length, base); + + for (i = 0; i < mod; i++) { + pow *= base; + } + + this.imuln(pow); + if (this.words[0] + word < 0x4000000) { + this.words[0] += word; + } else { + this._iaddn(word); + } + } + }; + + BN.prototype.copy = function copy(dest) { + dest.words = new Array(this.length); + for (var i = 0; i < this.length; i++) { + dest.words[i] = this.words[i]; + } + dest.length = this.length; + dest.negative = this.negative; + dest.red = this.red; + }; + + BN.prototype.clone = function clone() { + var r = new BN(null); + this.copy(r); + return r; + }; + + BN.prototype._expand = function _expand(size) { + while (this.length < size) { + this.words[this.length++] = 0; + } + return this; + }; + + // Remove leading `0` from `this` + BN.prototype.strip = function strip() { + while (this.length > 1 && this.words[this.length - 1] === 0) { + this.length--; + } + return this._normSign(); + }; + + BN.prototype._normSign = function _normSign() { + // -0 = 0 + if (this.length === 1 && this.words[0] === 0) { + this.negative = 0; + } + return this; + }; + + BN.prototype.inspect = function inspect() { + return (this.red ? ''; + }; + + /* + var zeros = []; + var groupSizes = []; + var groupBases = []; - /** - * Generic Subtraction. - * Uses either the constant or secret version of this operator depending on type of paramter. - * @method sub - * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.sub = function (o) { - if (this.isConstant(o)) { - return this.csub(o); + var s = ''; + var i = -1; + while (++i < BN.wordSize) { + zeros[i] = s; + s += '0'; + } + groupSizes[0] = 0; + groupSizes[1] = 0; + groupBases[0] = 0; + groupBases[1] = 0; + var base = 2 - 1; + while (++base < 36 + 1) { + var groupSize = 0; + var groupBase = 1; + while (groupBase < (1 << BN.wordSize) / base) { + groupBase *= base; + groupSize += 1; } - return this.ssub(o); - }; + groupSizes[base] = groupSize; + groupBases[base] = groupBase; + } + */ - /** - * Generic Multiplication. - * Uses either the constant or secret version of this operator depending on type of paramter. - * @method mult - * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this multiplication (and internally, the corresponding beaver triplet). - * This id must be unique, and must be passed by all parties to the same instruction. - * this ensures that every party gets a share from the same triplet for every matching instruction. An automatic id - * is generated by increasing a local counter, default ids suffice when all parties execute the - * instructions in the same order. Only used if secret multiplication is used. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.mult = function (o, op_id) { - if (this.isConstant(o)) { - return this.cmult(o); - } - return this.smult(o, op_id); - }; + var zeros = [ + '', + '0', + '00', + '000', + '0000', + '00000', + '000000', + '0000000', + '00000000', + '000000000', + '0000000000', + '00000000000', + '000000000000', + '0000000000000', + '00000000000000', + '000000000000000', + '0000000000000000', + '00000000000000000', + '000000000000000000', + '0000000000000000000', + '00000000000000000000', + '000000000000000000000', + '0000000000000000000000', + '00000000000000000000000', + '000000000000000000000000', + '0000000000000000000000000' + ]; + + var groupSizes = [0, 0, 25, 16, 12, 11, 10, 9, 8, 8, 7, 7, 7, 7, 6, 6, 6, 6, 6, 6, 6, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5]; + + var groupBases = [ + 0, 0, 33554432, 43046721, 16777216, 48828125, 60466176, 40353607, 16777216, 43046721, 10000000, 19487171, 35831808, 62748517, 7529536, 11390625, 16777216, 24137569, 34012224, 47045881, + 64000000, 4084101, 5153632, 6436343, 7962624, 9765625, 11881376, 14348907, 17210368, 20511149, 24300000, 28629151, 33554432, 39135393, 45435424, 52521875, 60466176 + ]; + + BN.prototype.toString = function toString(base, padding) { + base = base || 10; + padding = padding | 0 || 1; + + var out; + if (base === 16 || base === 'hex') { + out = ''; + var off = 0; + var carry = 0; + for (var i = 0; i < this.length; i++) { + var w = this.words[i]; + var word = (((w << off) | carry) & 0xffffff).toString(16); + carry = (w >>> (24 - off)) & 0xffffff; + if (carry !== 0 || i !== this.length - 1) { + out = zeros[6 - word.length] + word + out; + } else { + out = word + out; + } + off += 2; + if (off >= 26) { + off -= 26; + i--; + } + } + if (carry !== 0) { + out = carry.toString(16) + out; + } + while (out.length % padding !== 0) { + out = '0' + out; + } + if (this.negative !== 0) { + out = '-' + out; + } + return out; + } + + if (base === (base | 0) && base >= 2 && base <= 36) { + // var groupSize = Math.floor(BN.wordSize * Math.LN2 / Math.log(base)); + var groupSize = groupSizes[base]; + // var groupBase = Math.pow(base, groupSize); + var groupBase = groupBases[base]; + out = ''; + var c = this.clone(); + c.negative = 0; + while (!c.isZero()) { + var r = c.modn(groupBase).toString(base); + c = c.idivn(groupBase); + + if (!c.isZero()) { + out = zeros[groupSize - r.length] + r + out; + } else { + out = r + out; + } + } + if (this.isZero()) { + out = '0' + out; + } + while (out.length % padding !== 0) { + out = '0' + out; + } + if (this.negative !== 0) { + out = '-' + out; + } + return out; + } + + assert(false, 'Base should be between 2 and 36'); + }; + + BN.prototype.toNumber = function toNumber() { + var ret = this.words[0]; + if (this.length === 2) { + ret += this.words[1] * 0x4000000; + } else if (this.length === 3 && this.words[2] === 0x01) { + // NOTE: at this stage it is known that the top bit is set + ret += 0x10000000000000 + this.words[1] * 0x4000000; + } else if (this.length > 2) { + assert(false, 'Number can only safely store up to 53 bits'); + } + return this.negative !== 0 ? -ret : ret; + }; + + BN.prototype.toJSON = function toJSON() { + return this.toString(16); + }; + + BN.prototype.toBuffer = function toBuffer(endian, length) { + assert(typeof Buffer !== 'undefined'); + return this.toArrayLike(Buffer, endian, length); + }; + + BN.prototype.toArray = function toArray(endian, length) { + return this.toArrayLike(Array, endian, length); + }; + + BN.prototype.toArrayLike = function toArrayLike(ArrayType, endian, length) { + var byteLength = this.byteLength(); + var reqLength = length || Math.max(1, byteLength); + assert(byteLength <= reqLength, 'byte array longer than desired length'); + assert(reqLength > 0, 'Requested array length <= 0'); + + this.strip(); + var littleEndian = endian === 'le'; + var res = new ArrayType(reqLength); + + var b, i; + var q = this.clone(); + if (!littleEndian) { + // Assume big-endian + for (i = 0; i < reqLength - byteLength; i++) { + res[i] = 0; + } + + for (i = 0; !q.isZero(); i++) { + b = q.andln(0xff); + q.iushrn(8); + + res[reqLength - i - 1] = b; + } + } else { + for (i = 0; !q.isZero(); i++) { + b = q.andln(0xff); + q.iushrn(8); + + res[i] = b; + } + + for (; i < reqLength; i++) { + res[i] = 0; + } + } + + return res; + }; + + if (Math.clz32) { + BN.prototype._countBits = function _countBits(w) { + return 32 - Math.clz32(w); + }; + } else { + BN.prototype._countBits = function _countBits(w) { + var t = w; + var r = 0; + if (t >= 0x1000) { + r += 13; + t >>>= 13; + } + if (t >= 0x40) { + r += 7; + t >>>= 7; + } + if (t >= 0x8) { + r += 4; + t >>>= 4; + } + if (t >= 0x02) { + r += 2; + t >>>= 2; + } + return r + t; + }; + } + BN.prototype._zeroBits = function _zeroBits(w) { + // Short-cut + if (w === 0) return 26; + + var t = w; + var r = 0; + if ((t & 0x1fff) === 0) { + r += 13; + t >>>= 13; + } + if ((t & 0x7f) === 0) { + r += 7; + t >>>= 7; + } + if ((t & 0xf) === 0) { + r += 4; + t >>>= 4; + } + if ((t & 0x3) === 0) { + r += 2; + t >>>= 2; + } + if ((t & 0x1) === 0) { + r++; + } + return r; + }; + + // Return number of used bits in a BN + BN.prototype.bitLength = function bitLength() { + var w = this.words[this.length - 1]; + var hi = this._countBits(w); + return (this.length - 1) * 26 + hi; + }; + + function toBitArray(num) { + var w = new Array(num.bitLength()); + + for (var bit = 0; bit < w.length; bit++) { + var off = (bit / 26) | 0; + var wbit = bit % 26; + + w[bit] = (num.words[off] & (1 << wbit)) >>> wbit; + } + + return w; + } - /** - * Generic XOR for bits (both this and o have to be bits to work correctly). - * Uses either the constant or secret version of this operator depending on type of paramter. - * @method xor_bit - * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * Only used if secret xor is used.. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.xor_bit = function (o, op_id) { - if (this.isConstant(o)) { - return this.cxor_bit(o); - } - return this.sxor_bit(o, op_id); - }; + // Number of trailing zero bits + BN.prototype.zeroBits = function zeroBits() { + if (this.isZero()) return 0; + + var r = 0; + for (var i = 0; i < this.length; i++) { + var b = this._zeroBits(this.words[i]); + r += b; + if (b !== 26) break; + } + return r; + }; + + BN.prototype.byteLength = function byteLength() { + return Math.ceil(this.bitLength() / 8); + }; + + BN.prototype.toTwos = function toTwos(width) { + if (this.negative !== 0) { + return this.abs().inotn(width).iaddn(1); + } + return this.clone(); + }; + + BN.prototype.fromTwos = function fromTwos(width) { + if (this.testn(width - 1)) { + return this.notn(width).iaddn(1).ineg(); + } + return this.clone(); + }; + + BN.prototype.isNeg = function isNeg() { + return this.negative !== 0; + }; + + // Return negative clone of `this` + BN.prototype.neg = function neg() { + return this.clone().ineg(); + }; + + BN.prototype.ineg = function ineg() { + if (!this.isZero()) { + this.negative ^= 1; + } + + return this; + }; + + // Or `num` with `this` in-place + BN.prototype.iuor = function iuor(num) { + while (this.length < num.length) { + this.words[this.length++] = 0; + } + + for (var i = 0; i < num.length; i++) { + this.words[i] = this.words[i] | num.words[i]; + } + + return this.strip(); + }; + + BN.prototype.ior = function ior(num) { + assert((this.negative | num.negative) === 0); + return this.iuor(num); + }; + + // Or `num` with `this` + BN.prototype.or = function or(num) { + if (this.length > num.length) return this.clone().ior(num); + return num.clone().ior(this); + }; + + BN.prototype.uor = function uor(num) { + if (this.length > num.length) return this.clone().iuor(num); + return num.clone().iuor(this); + }; + + // And `num` with `this` in-place + BN.prototype.iuand = function iuand(num) { + // b = min-length(num, this) + var b; + if (this.length > num.length) { + b = num; + } else { + b = this; + } + + for (var i = 0; i < b.length; i++) { + this.words[i] = this.words[i] & num.words[i]; + } + + this.length = b.length; + + return this.strip(); + }; + + BN.prototype.iand = function iand(num) { + assert((this.negative | num.negative) === 0); + return this.iuand(num); + }; + + // And `num` with `this` + BN.prototype.and = function and(num) { + if (this.length > num.length) return this.clone().iand(num); + return num.clone().iand(this); + }; + + BN.prototype.uand = function uand(num) { + if (this.length > num.length) return this.clone().iuand(num); + return num.clone().iuand(this); + }; + + // Xor `num` with `this` in-place + BN.prototype.iuxor = function iuxor(num) { + // a.length > b.length + var a; + var b; + if (this.length > num.length) { + a = this; + b = num; + } else { + a = num; + b = this; + } + + for (var i = 0; i < b.length; i++) { + this.words[i] = a.words[i] ^ b.words[i]; + } + + if (this !== a) { + for (; i < a.length; i++) { + this.words[i] = a.words[i]; + } + } + + this.length = a.length; + + return this.strip(); + }; + + BN.prototype.ixor = function ixor(num) { + assert((this.negative | num.negative) === 0); + return this.iuxor(num); + }; + + // Xor `num` with `this` + BN.prototype.xor = function xor(num) { + if (this.length > num.length) return this.clone().ixor(num); + return num.clone().ixor(this); + }; + + BN.prototype.uxor = function uxor(num) { + if (this.length > num.length) return this.clone().iuxor(num); + return num.clone().iuxor(this); + }; + + // Not ``this`` with ``width`` bitwidth + BN.prototype.inotn = function inotn(width) { + assert(typeof width === 'number' && width >= 0); + + var bytesNeeded = Math.ceil(width / 26) | 0; + var bitsLeft = width % 26; + + // Extend the buffer with leading zeroes + this._expand(bytesNeeded); + + if (bitsLeft > 0) { + bytesNeeded--; + } + + // Handle complete words + for (var i = 0; i < bytesNeeded; i++) { + this.words[i] = ~this.words[i] & 0x3ffffff; + } + + // Handle the residue + if (bitsLeft > 0) { + this.words[i] = ~this.words[i] & (0x3ffffff >> (26 - bitsLeft)); + } + + // And remove leading zeroes + return this.strip(); + }; + + BN.prototype.notn = function notn(width) { + return this.clone().inotn(width); + }; + + // Set `bit` of `this` + BN.prototype.setn = function setn(bit, val) { + assert(typeof bit === 'number' && bit >= 0); + + var off = (bit / 26) | 0; + var wbit = bit % 26; + + this._expand(off + 1); + + if (val) { + this.words[off] = this.words[off] | (1 << wbit); + } else { + this.words[off] = this.words[off] & ~(1 << wbit); + } + + return this.strip(); + }; + + // Add `num` to `this` in-place + BN.prototype.iadd = function iadd(num) { + var r; + + // negative + positive + if (this.negative !== 0 && num.negative === 0) { + this.negative = 0; + r = this.isub(num); + this.negative ^= 1; + return this._normSign(); + + // positive + negative + } else if (this.negative === 0 && num.negative !== 0) { + num.negative = 0; + r = this.isub(num); + num.negative = 1; + return r._normSign(); + } + + // a.length > b.length + var a, b; + if (this.length > num.length) { + a = this; + b = num; + } else { + a = num; + b = this; + } + + var carry = 0; + for (var i = 0; i < b.length; i++) { + r = (a.words[i] | 0) + (b.words[i] | 0) + carry; + this.words[i] = r & 0x3ffffff; + carry = r >>> 26; + } + for (; carry !== 0 && i < a.length; i++) { + r = (a.words[i] | 0) + carry; + this.words[i] = r & 0x3ffffff; + carry = r >>> 26; + } + + this.length = a.length; + if (carry !== 0) { + this.words[this.length] = carry; + this.length++; + // Copy the rest of the words + } else if (a !== this) { + for (; i < a.length; i++) { + this.words[i] = a.words[i]; + } + } + + return this; + }; + + // Add `num` to `this` + BN.prototype.add = function add(num) { + var res; + if (num.negative !== 0 && this.negative === 0) { + num.negative = 0; + res = this.sub(num); + num.negative ^= 1; + return res; + } else if (num.negative === 0 && this.negative !== 0) { + this.negative = 0; + res = num.sub(this); + this.negative = 1; + return res; + } + + if (this.length > num.length) return this.clone().iadd(num); + + return num.clone().iadd(this); + }; + + // Subtract `num` from `this` in-place + BN.prototype.isub = function isub(num) { + // this - (-num) = this + num + if (num.negative !== 0) { + num.negative = 0; + var r = this.iadd(num); + num.negative = 1; + return r._normSign(); + + // -this - num = -(this + num) + } else if (this.negative !== 0) { + this.negative = 0; + this.iadd(num); + this.negative = 1; + return this._normSign(); + } + + // At this point both numbers are positive + var cmp = this.cmp(num); + + // Optimization - zeroify + if (cmp === 0) { + this.negative = 0; + this.length = 1; + this.words[0] = 0; + return this; + } + + // a > b + var a, b; + if (cmp > 0) { + a = this; + b = num; + } else { + a = num; + b = this; + } + + var carry = 0; + for (var i = 0; i < b.length; i++) { + r = (a.words[i] | 0) - (b.words[i] | 0) + carry; + carry = r >> 26; + this.words[i] = r & 0x3ffffff; + } + for (; carry !== 0 && i < a.length; i++) { + r = (a.words[i] | 0) + carry; + carry = r >> 26; + this.words[i] = r & 0x3ffffff; + } + + // Copy rest of the words + if (carry === 0 && i < a.length && a !== this) { + for (; i < a.length; i++) { + this.words[i] = a.words[i]; + } + } + + this.length = Math.max(this.length, i); + + if (a !== this) { + this.negative = 1; + } + + return this.strip(); + }; + + // Subtract `num` from `this` + BN.prototype.sub = function sub(num) { + return this.clone().isub(num); + }; + + function smallMulTo(self, num, out) { + out.negative = num.negative ^ self.negative; + var len = (self.length + num.length) | 0; + out.length = len; + len = (len - 1) | 0; + + // Peel one iteration (compiler can't do it, because of code complexity) + var a = self.words[0] | 0; + var b = num.words[0] | 0; + var r = a * b; + + var lo = r & 0x3ffffff; + var carry = (r / 0x4000000) | 0; + out.words[0] = lo; + + for (var k = 1; k < len; k++) { + // Sum all words with the same `i + j = k` and accumulate `ncarry`, + // note that ncarry could be >= 0x3ffffff + var ncarry = carry >>> 26; + var rword = carry & 0x3ffffff; + var maxJ = Math.min(k, num.length - 1); + for (var j = Math.max(0, k - self.length + 1); j <= maxJ; j++) { + var i = (k - j) | 0; + a = self.words[i] | 0; + b = num.words[j] | 0; + r = a * b + rword; + ncarry += (r / 0x4000000) | 0; + rword = r & 0x3ffffff; + } + out.words[k] = rword | 0; + carry = ncarry | 0; + } + if (carry !== 0) { + out.words[k] = carry | 0; + } else { + out.length--; + } + + return out.strip(); + } - /** - * Generic OR for bits (both this and o have to be bits to work correctly). - * Uses either the constant or secret version of this operator depending on type of paramter. - * @method or_bit - * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * Only used if secret or is used.. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.or_bit = function (o, op_id) { - if (this.isConstant(o)) { - return this.cor_bit(o); - } - return this.sor_bit(o, op_id); - }; + // TODO(indutny): it may be reasonable to omit it for users who don't need + // to work with 256-bit numbers, otherwise it gives 20% improvement for 256-bit + // multiplication (like elliptic secp256k1). + var comb10MulTo = function comb10MulTo(self, num, out) { + var a = self.words; + var b = num.words; + var o = out.words; + var c = 0; + var lo; + var mid; + var hi; + var a0 = a[0] | 0; + var al0 = a0 & 0x1fff; + var ah0 = a0 >>> 13; + var a1 = a[1] | 0; + var al1 = a1 & 0x1fff; + var ah1 = a1 >>> 13; + var a2 = a[2] | 0; + var al2 = a2 & 0x1fff; + var ah2 = a2 >>> 13; + var a3 = a[3] | 0; + var al3 = a3 & 0x1fff; + var ah3 = a3 >>> 13; + var a4 = a[4] | 0; + var al4 = a4 & 0x1fff; + var ah4 = a4 >>> 13; + var a5 = a[5] | 0; + var al5 = a5 & 0x1fff; + var ah5 = a5 >>> 13; + var a6 = a[6] | 0; + var al6 = a6 & 0x1fff; + var ah6 = a6 >>> 13; + var a7 = a[7] | 0; + var al7 = a7 & 0x1fff; + var ah7 = a7 >>> 13; + var a8 = a[8] | 0; + var al8 = a8 & 0x1fff; + var ah8 = a8 >>> 13; + var a9 = a[9] | 0; + var al9 = a9 & 0x1fff; + var ah9 = a9 >>> 13; + var b0 = b[0] | 0; + var bl0 = b0 & 0x1fff; + var bh0 = b0 >>> 13; + var b1 = b[1] | 0; + var bl1 = b1 & 0x1fff; + var bh1 = b1 >>> 13; + var b2 = b[2] | 0; + var bl2 = b2 & 0x1fff; + var bh2 = b2 >>> 13; + var b3 = b[3] | 0; + var bl3 = b3 & 0x1fff; + var bh3 = b3 >>> 13; + var b4 = b[4] | 0; + var bl4 = b4 & 0x1fff; + var bh4 = b4 >>> 13; + var b5 = b[5] | 0; + var bl5 = b5 & 0x1fff; + var bh5 = b5 >>> 13; + var b6 = b[6] | 0; + var bl6 = b6 & 0x1fff; + var bh6 = b6 >>> 13; + var b7 = b[7] | 0; + var bl7 = b7 & 0x1fff; + var bh7 = b7 >>> 13; + var b8 = b[8] | 0; + var bl8 = b8 & 0x1fff; + var bh8 = b8 >>> 13; + var b9 = b[9] | 0; + var bl9 = b9 & 0x1fff; + var bh9 = b9 >>> 13; + + out.negative = self.negative ^ num.negative; + out.length = 19; + /* k = 0 */ + lo = Math.imul(al0, bl0); + mid = Math.imul(al0, bh0); + mid = (mid + Math.imul(ah0, bl0)) | 0; + hi = Math.imul(ah0, bh0); + var w0 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w0 >>> 26)) | 0; + w0 &= 0x3ffffff; + /* k = 1 */ + lo = Math.imul(al1, bl0); + mid = Math.imul(al1, bh0); + mid = (mid + Math.imul(ah1, bl0)) | 0; + hi = Math.imul(ah1, bh0); + lo = (lo + Math.imul(al0, bl1)) | 0; + mid = (mid + Math.imul(al0, bh1)) | 0; + mid = (mid + Math.imul(ah0, bl1)) | 0; + hi = (hi + Math.imul(ah0, bh1)) | 0; + var w1 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w1 >>> 26)) | 0; + w1 &= 0x3ffffff; + /* k = 2 */ + lo = Math.imul(al2, bl0); + mid = Math.imul(al2, bh0); + mid = (mid + Math.imul(ah2, bl0)) | 0; + hi = Math.imul(ah2, bh0); + lo = (lo + Math.imul(al1, bl1)) | 0; + mid = (mid + Math.imul(al1, bh1)) | 0; + mid = (mid + Math.imul(ah1, bl1)) | 0; + hi = (hi + Math.imul(ah1, bh1)) | 0; + lo = (lo + Math.imul(al0, bl2)) | 0; + mid = (mid + Math.imul(al0, bh2)) | 0; + mid = (mid + Math.imul(ah0, bl2)) | 0; + hi = (hi + Math.imul(ah0, bh2)) | 0; + var w2 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w2 >>> 26)) | 0; + w2 &= 0x3ffffff; + /* k = 3 */ + lo = Math.imul(al3, bl0); + mid = Math.imul(al3, bh0); + mid = (mid + Math.imul(ah3, bl0)) | 0; + hi = Math.imul(ah3, bh0); + lo = (lo + Math.imul(al2, bl1)) | 0; + mid = (mid + Math.imul(al2, bh1)) | 0; + mid = (mid + Math.imul(ah2, bl1)) | 0; + hi = (hi + Math.imul(ah2, bh1)) | 0; + lo = (lo + Math.imul(al1, bl2)) | 0; + mid = (mid + Math.imul(al1, bh2)) | 0; + mid = (mid + Math.imul(ah1, bl2)) | 0; + hi = (hi + Math.imul(ah1, bh2)) | 0; + lo = (lo + Math.imul(al0, bl3)) | 0; + mid = (mid + Math.imul(al0, bh3)) | 0; + mid = (mid + Math.imul(ah0, bl3)) | 0; + hi = (hi + Math.imul(ah0, bh3)) | 0; + var w3 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w3 >>> 26)) | 0; + w3 &= 0x3ffffff; + /* k = 4 */ + lo = Math.imul(al4, bl0); + mid = Math.imul(al4, bh0); + mid = (mid + Math.imul(ah4, bl0)) | 0; + hi = Math.imul(ah4, bh0); + lo = (lo + Math.imul(al3, bl1)) | 0; + mid = (mid + Math.imul(al3, bh1)) | 0; + mid = (mid + Math.imul(ah3, bl1)) | 0; + hi = (hi + Math.imul(ah3, bh1)) | 0; + lo = (lo + Math.imul(al2, bl2)) | 0; + mid = (mid + Math.imul(al2, bh2)) | 0; + mid = (mid + Math.imul(ah2, bl2)) | 0; + hi = (hi + Math.imul(ah2, bh2)) | 0; + lo = (lo + Math.imul(al1, bl3)) | 0; + mid = (mid + Math.imul(al1, bh3)) | 0; + mid = (mid + Math.imul(ah1, bl3)) | 0; + hi = (hi + Math.imul(ah1, bh3)) | 0; + lo = (lo + Math.imul(al0, bl4)) | 0; + mid = (mid + Math.imul(al0, bh4)) | 0; + mid = (mid + Math.imul(ah0, bl4)) | 0; + hi = (hi + Math.imul(ah0, bh4)) | 0; + var w4 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w4 >>> 26)) | 0; + w4 &= 0x3ffffff; + /* k = 5 */ + lo = Math.imul(al5, bl0); + mid = Math.imul(al5, bh0); + mid = (mid + Math.imul(ah5, bl0)) | 0; + hi = Math.imul(ah5, bh0); + lo = (lo + Math.imul(al4, bl1)) | 0; + mid = (mid + Math.imul(al4, bh1)) | 0; + mid = (mid + Math.imul(ah4, bl1)) | 0; + hi = (hi + Math.imul(ah4, bh1)) | 0; + lo = (lo + Math.imul(al3, bl2)) | 0; + mid = (mid + Math.imul(al3, bh2)) | 0; + mid = (mid + Math.imul(ah3, bl2)) | 0; + hi = (hi + Math.imul(ah3, bh2)) | 0; + lo = (lo + Math.imul(al2, bl3)) | 0; + mid = (mid + Math.imul(al2, bh3)) | 0; + mid = (mid + Math.imul(ah2, bl3)) | 0; + hi = (hi + Math.imul(ah2, bh3)) | 0; + lo = (lo + Math.imul(al1, bl4)) | 0; + mid = (mid + Math.imul(al1, bh4)) | 0; + mid = (mid + Math.imul(ah1, bl4)) | 0; + hi = (hi + Math.imul(ah1, bh4)) | 0; + lo = (lo + Math.imul(al0, bl5)) | 0; + mid = (mid + Math.imul(al0, bh5)) | 0; + mid = (mid + Math.imul(ah0, bl5)) | 0; + hi = (hi + Math.imul(ah0, bh5)) | 0; + var w5 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w5 >>> 26)) | 0; + w5 &= 0x3ffffff; + /* k = 6 */ + lo = Math.imul(al6, bl0); + mid = Math.imul(al6, bh0); + mid = (mid + Math.imul(ah6, bl0)) | 0; + hi = Math.imul(ah6, bh0); + lo = (lo + Math.imul(al5, bl1)) | 0; + mid = (mid + Math.imul(al5, bh1)) | 0; + mid = (mid + Math.imul(ah5, bl1)) | 0; + hi = (hi + Math.imul(ah5, bh1)) | 0; + lo = (lo + Math.imul(al4, bl2)) | 0; + mid = (mid + Math.imul(al4, bh2)) | 0; + mid = (mid + Math.imul(ah4, bl2)) | 0; + hi = (hi + Math.imul(ah4, bh2)) | 0; + lo = (lo + Math.imul(al3, bl3)) | 0; + mid = (mid + Math.imul(al3, bh3)) | 0; + mid = (mid + Math.imul(ah3, bl3)) | 0; + hi = (hi + Math.imul(ah3, bh3)) | 0; + lo = (lo + Math.imul(al2, bl4)) | 0; + mid = (mid + Math.imul(al2, bh4)) | 0; + mid = (mid + Math.imul(ah2, bl4)) | 0; + hi = (hi + Math.imul(ah2, bh4)) | 0; + lo = (lo + Math.imul(al1, bl5)) | 0; + mid = (mid + Math.imul(al1, bh5)) | 0; + mid = (mid + Math.imul(ah1, bl5)) | 0; + hi = (hi + Math.imul(ah1, bh5)) | 0; + lo = (lo + Math.imul(al0, bl6)) | 0; + mid = (mid + Math.imul(al0, bh6)) | 0; + mid = (mid + Math.imul(ah0, bl6)) | 0; + hi = (hi + Math.imul(ah0, bh6)) | 0; + var w6 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w6 >>> 26)) | 0; + w6 &= 0x3ffffff; + /* k = 7 */ + lo = Math.imul(al7, bl0); + mid = Math.imul(al7, bh0); + mid = (mid + Math.imul(ah7, bl0)) | 0; + hi = Math.imul(ah7, bh0); + lo = (lo + Math.imul(al6, bl1)) | 0; + mid = (mid + Math.imul(al6, bh1)) | 0; + mid = (mid + Math.imul(ah6, bl1)) | 0; + hi = (hi + Math.imul(ah6, bh1)) | 0; + lo = (lo + Math.imul(al5, bl2)) | 0; + mid = (mid + Math.imul(al5, bh2)) | 0; + mid = (mid + Math.imul(ah5, bl2)) | 0; + hi = (hi + Math.imul(ah5, bh2)) | 0; + lo = (lo + Math.imul(al4, bl3)) | 0; + mid = (mid + Math.imul(al4, bh3)) | 0; + mid = (mid + Math.imul(ah4, bl3)) | 0; + hi = (hi + Math.imul(ah4, bh3)) | 0; + lo = (lo + Math.imul(al3, bl4)) | 0; + mid = (mid + Math.imul(al3, bh4)) | 0; + mid = (mid + Math.imul(ah3, bl4)) | 0; + hi = (hi + Math.imul(ah3, bh4)) | 0; + lo = (lo + Math.imul(al2, bl5)) | 0; + mid = (mid + Math.imul(al2, bh5)) | 0; + mid = (mid + Math.imul(ah2, bl5)) | 0; + hi = (hi + Math.imul(ah2, bh5)) | 0; + lo = (lo + Math.imul(al1, bl6)) | 0; + mid = (mid + Math.imul(al1, bh6)) | 0; + mid = (mid + Math.imul(ah1, bl6)) | 0; + hi = (hi + Math.imul(ah1, bh6)) | 0; + lo = (lo + Math.imul(al0, bl7)) | 0; + mid = (mid + Math.imul(al0, bh7)) | 0; + mid = (mid + Math.imul(ah0, bl7)) | 0; + hi = (hi + Math.imul(ah0, bh7)) | 0; + var w7 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w7 >>> 26)) | 0; + w7 &= 0x3ffffff; + /* k = 8 */ + lo = Math.imul(al8, bl0); + mid = Math.imul(al8, bh0); + mid = (mid + Math.imul(ah8, bl0)) | 0; + hi = Math.imul(ah8, bh0); + lo = (lo + Math.imul(al7, bl1)) | 0; + mid = (mid + Math.imul(al7, bh1)) | 0; + mid = (mid + Math.imul(ah7, bl1)) | 0; + hi = (hi + Math.imul(ah7, bh1)) | 0; + lo = (lo + Math.imul(al6, bl2)) | 0; + mid = (mid + Math.imul(al6, bh2)) | 0; + mid = (mid + Math.imul(ah6, bl2)) | 0; + hi = (hi + Math.imul(ah6, bh2)) | 0; + lo = (lo + Math.imul(al5, bl3)) | 0; + mid = (mid + Math.imul(al5, bh3)) | 0; + mid = (mid + Math.imul(ah5, bl3)) | 0; + hi = (hi + Math.imul(ah5, bh3)) | 0; + lo = (lo + Math.imul(al4, bl4)) | 0; + mid = (mid + Math.imul(al4, bh4)) | 0; + mid = (mid + Math.imul(ah4, bl4)) | 0; + hi = (hi + Math.imul(ah4, bh4)) | 0; + lo = (lo + Math.imul(al3, bl5)) | 0; + mid = (mid + Math.imul(al3, bh5)) | 0; + mid = (mid + Math.imul(ah3, bl5)) | 0; + hi = (hi + Math.imul(ah3, bh5)) | 0; + lo = (lo + Math.imul(al2, bl6)) | 0; + mid = (mid + Math.imul(al2, bh6)) | 0; + mid = (mid + Math.imul(ah2, bl6)) | 0; + hi = (hi + Math.imul(ah2, bh6)) | 0; + lo = (lo + Math.imul(al1, bl7)) | 0; + mid = (mid + Math.imul(al1, bh7)) | 0; + mid = (mid + Math.imul(ah1, bl7)) | 0; + hi = (hi + Math.imul(ah1, bh7)) | 0; + lo = (lo + Math.imul(al0, bl8)) | 0; + mid = (mid + Math.imul(al0, bh8)) | 0; + mid = (mid + Math.imul(ah0, bl8)) | 0; + hi = (hi + Math.imul(ah0, bh8)) | 0; + var w8 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w8 >>> 26)) | 0; + w8 &= 0x3ffffff; + /* k = 9 */ + lo = Math.imul(al9, bl0); + mid = Math.imul(al9, bh0); + mid = (mid + Math.imul(ah9, bl0)) | 0; + hi = Math.imul(ah9, bh0); + lo = (lo + Math.imul(al8, bl1)) | 0; + mid = (mid + Math.imul(al8, bh1)) | 0; + mid = (mid + Math.imul(ah8, bl1)) | 0; + hi = (hi + Math.imul(ah8, bh1)) | 0; + lo = (lo + Math.imul(al7, bl2)) | 0; + mid = (mid + Math.imul(al7, bh2)) | 0; + mid = (mid + Math.imul(ah7, bl2)) | 0; + hi = (hi + Math.imul(ah7, bh2)) | 0; + lo = (lo + Math.imul(al6, bl3)) | 0; + mid = (mid + Math.imul(al6, bh3)) | 0; + mid = (mid + Math.imul(ah6, bl3)) | 0; + hi = (hi + Math.imul(ah6, bh3)) | 0; + lo = (lo + Math.imul(al5, bl4)) | 0; + mid = (mid + Math.imul(al5, bh4)) | 0; + mid = (mid + Math.imul(ah5, bl4)) | 0; + hi = (hi + Math.imul(ah5, bh4)) | 0; + lo = (lo + Math.imul(al4, bl5)) | 0; + mid = (mid + Math.imul(al4, bh5)) | 0; + mid = (mid + Math.imul(ah4, bl5)) | 0; + hi = (hi + Math.imul(ah4, bh5)) | 0; + lo = (lo + Math.imul(al3, bl6)) | 0; + mid = (mid + Math.imul(al3, bh6)) | 0; + mid = (mid + Math.imul(ah3, bl6)) | 0; + hi = (hi + Math.imul(ah3, bh6)) | 0; + lo = (lo + Math.imul(al2, bl7)) | 0; + mid = (mid + Math.imul(al2, bh7)) | 0; + mid = (mid + Math.imul(ah2, bl7)) | 0; + hi = (hi + Math.imul(ah2, bh7)) | 0; + lo = (lo + Math.imul(al1, bl8)) | 0; + mid = (mid + Math.imul(al1, bh8)) | 0; + mid = (mid + Math.imul(ah1, bl8)) | 0; + hi = (hi + Math.imul(ah1, bh8)) | 0; + lo = (lo + Math.imul(al0, bl9)) | 0; + mid = (mid + Math.imul(al0, bh9)) | 0; + mid = (mid + Math.imul(ah0, bl9)) | 0; + hi = (hi + Math.imul(ah0, bh9)) | 0; + var w9 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w9 >>> 26)) | 0; + w9 &= 0x3ffffff; + /* k = 10 */ + lo = Math.imul(al9, bl1); + mid = Math.imul(al9, bh1); + mid = (mid + Math.imul(ah9, bl1)) | 0; + hi = Math.imul(ah9, bh1); + lo = (lo + Math.imul(al8, bl2)) | 0; + mid = (mid + Math.imul(al8, bh2)) | 0; + mid = (mid + Math.imul(ah8, bl2)) | 0; + hi = (hi + Math.imul(ah8, bh2)) | 0; + lo = (lo + Math.imul(al7, bl3)) | 0; + mid = (mid + Math.imul(al7, bh3)) | 0; + mid = (mid + Math.imul(ah7, bl3)) | 0; + hi = (hi + Math.imul(ah7, bh3)) | 0; + lo = (lo + Math.imul(al6, bl4)) | 0; + mid = (mid + Math.imul(al6, bh4)) | 0; + mid = (mid + Math.imul(ah6, bl4)) | 0; + hi = (hi + Math.imul(ah6, bh4)) | 0; + lo = (lo + Math.imul(al5, bl5)) | 0; + mid = (mid + Math.imul(al5, bh5)) | 0; + mid = (mid + Math.imul(ah5, bl5)) | 0; + hi = (hi + Math.imul(ah5, bh5)) | 0; + lo = (lo + Math.imul(al4, bl6)) | 0; + mid = (mid + Math.imul(al4, bh6)) | 0; + mid = (mid + Math.imul(ah4, bl6)) | 0; + hi = (hi + Math.imul(ah4, bh6)) | 0; + lo = (lo + Math.imul(al3, bl7)) | 0; + mid = (mid + Math.imul(al3, bh7)) | 0; + mid = (mid + Math.imul(ah3, bl7)) | 0; + hi = (hi + Math.imul(ah3, bh7)) | 0; + lo = (lo + Math.imul(al2, bl8)) | 0; + mid = (mid + Math.imul(al2, bh8)) | 0; + mid = (mid + Math.imul(ah2, bl8)) | 0; + hi = (hi + Math.imul(ah2, bh8)) | 0; + lo = (lo + Math.imul(al1, bl9)) | 0; + mid = (mid + Math.imul(al1, bh9)) | 0; + mid = (mid + Math.imul(ah1, bl9)) | 0; + hi = (hi + Math.imul(ah1, bh9)) | 0; + var w10 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w10 >>> 26)) | 0; + w10 &= 0x3ffffff; + /* k = 11 */ + lo = Math.imul(al9, bl2); + mid = Math.imul(al9, bh2); + mid = (mid + Math.imul(ah9, bl2)) | 0; + hi = Math.imul(ah9, bh2); + lo = (lo + Math.imul(al8, bl3)) | 0; + mid = (mid + Math.imul(al8, bh3)) | 0; + mid = (mid + Math.imul(ah8, bl3)) | 0; + hi = (hi + Math.imul(ah8, bh3)) | 0; + lo = (lo + Math.imul(al7, bl4)) | 0; + mid = (mid + Math.imul(al7, bh4)) | 0; + mid = (mid + Math.imul(ah7, bl4)) | 0; + hi = (hi + Math.imul(ah7, bh4)) | 0; + lo = (lo + Math.imul(al6, bl5)) | 0; + mid = (mid + Math.imul(al6, bh5)) | 0; + mid = (mid + Math.imul(ah6, bl5)) | 0; + hi = (hi + Math.imul(ah6, bh5)) | 0; + lo = (lo + Math.imul(al5, bl6)) | 0; + mid = (mid + Math.imul(al5, bh6)) | 0; + mid = (mid + Math.imul(ah5, bl6)) | 0; + hi = (hi + Math.imul(ah5, bh6)) | 0; + lo = (lo + Math.imul(al4, bl7)) | 0; + mid = (mid + Math.imul(al4, bh7)) | 0; + mid = (mid + Math.imul(ah4, bl7)) | 0; + hi = (hi + Math.imul(ah4, bh7)) | 0; + lo = (lo + Math.imul(al3, bl8)) | 0; + mid = (mid + Math.imul(al3, bh8)) | 0; + mid = (mid + Math.imul(ah3, bl8)) | 0; + hi = (hi + Math.imul(ah3, bh8)) | 0; + lo = (lo + Math.imul(al2, bl9)) | 0; + mid = (mid + Math.imul(al2, bh9)) | 0; + mid = (mid + Math.imul(ah2, bl9)) | 0; + hi = (hi + Math.imul(ah2, bh9)) | 0; + var w11 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w11 >>> 26)) | 0; + w11 &= 0x3ffffff; + /* k = 12 */ + lo = Math.imul(al9, bl3); + mid = Math.imul(al9, bh3); + mid = (mid + Math.imul(ah9, bl3)) | 0; + hi = Math.imul(ah9, bh3); + lo = (lo + Math.imul(al8, bl4)) | 0; + mid = (mid + Math.imul(al8, bh4)) | 0; + mid = (mid + Math.imul(ah8, bl4)) | 0; + hi = (hi + Math.imul(ah8, bh4)) | 0; + lo = (lo + Math.imul(al7, bl5)) | 0; + mid = (mid + Math.imul(al7, bh5)) | 0; + mid = (mid + Math.imul(ah7, bl5)) | 0; + hi = (hi + Math.imul(ah7, bh5)) | 0; + lo = (lo + Math.imul(al6, bl6)) | 0; + mid = (mid + Math.imul(al6, bh6)) | 0; + mid = (mid + Math.imul(ah6, bl6)) | 0; + hi = (hi + Math.imul(ah6, bh6)) | 0; + lo = (lo + Math.imul(al5, bl7)) | 0; + mid = (mid + Math.imul(al5, bh7)) | 0; + mid = (mid + Math.imul(ah5, bl7)) | 0; + hi = (hi + Math.imul(ah5, bh7)) | 0; + lo = (lo + Math.imul(al4, bl8)) | 0; + mid = (mid + Math.imul(al4, bh8)) | 0; + mid = (mid + Math.imul(ah4, bl8)) | 0; + hi = (hi + Math.imul(ah4, bh8)) | 0; + lo = (lo + Math.imul(al3, bl9)) | 0; + mid = (mid + Math.imul(al3, bh9)) | 0; + mid = (mid + Math.imul(ah3, bl9)) | 0; + hi = (hi + Math.imul(ah3, bh9)) | 0; + var w12 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w12 >>> 26)) | 0; + w12 &= 0x3ffffff; + /* k = 13 */ + lo = Math.imul(al9, bl4); + mid = Math.imul(al9, bh4); + mid = (mid + Math.imul(ah9, bl4)) | 0; + hi = Math.imul(ah9, bh4); + lo = (lo + Math.imul(al8, bl5)) | 0; + mid = (mid + Math.imul(al8, bh5)) | 0; + mid = (mid + Math.imul(ah8, bl5)) | 0; + hi = (hi + Math.imul(ah8, bh5)) | 0; + lo = (lo + Math.imul(al7, bl6)) | 0; + mid = (mid + Math.imul(al7, bh6)) | 0; + mid = (mid + Math.imul(ah7, bl6)) | 0; + hi = (hi + Math.imul(ah7, bh6)) | 0; + lo = (lo + Math.imul(al6, bl7)) | 0; + mid = (mid + Math.imul(al6, bh7)) | 0; + mid = (mid + Math.imul(ah6, bl7)) | 0; + hi = (hi + Math.imul(ah6, bh7)) | 0; + lo = (lo + Math.imul(al5, bl8)) | 0; + mid = (mid + Math.imul(al5, bh8)) | 0; + mid = (mid + Math.imul(ah5, bl8)) | 0; + hi = (hi + Math.imul(ah5, bh8)) | 0; + lo = (lo + Math.imul(al4, bl9)) | 0; + mid = (mid + Math.imul(al4, bh9)) | 0; + mid = (mid + Math.imul(ah4, bl9)) | 0; + hi = (hi + Math.imul(ah4, bh9)) | 0; + var w13 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w13 >>> 26)) | 0; + w13 &= 0x3ffffff; + /* k = 14 */ + lo = Math.imul(al9, bl5); + mid = Math.imul(al9, bh5); + mid = (mid + Math.imul(ah9, bl5)) | 0; + hi = Math.imul(ah9, bh5); + lo = (lo + Math.imul(al8, bl6)) | 0; + mid = (mid + Math.imul(al8, bh6)) | 0; + mid = (mid + Math.imul(ah8, bl6)) | 0; + hi = (hi + Math.imul(ah8, bh6)) | 0; + lo = (lo + Math.imul(al7, bl7)) | 0; + mid = (mid + Math.imul(al7, bh7)) | 0; + mid = (mid + Math.imul(ah7, bl7)) | 0; + hi = (hi + Math.imul(ah7, bh7)) | 0; + lo = (lo + Math.imul(al6, bl8)) | 0; + mid = (mid + Math.imul(al6, bh8)) | 0; + mid = (mid + Math.imul(ah6, bl8)) | 0; + hi = (hi + Math.imul(ah6, bh8)) | 0; + lo = (lo + Math.imul(al5, bl9)) | 0; + mid = (mid + Math.imul(al5, bh9)) | 0; + mid = (mid + Math.imul(ah5, bl9)) | 0; + hi = (hi + Math.imul(ah5, bh9)) | 0; + var w14 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w14 >>> 26)) | 0; + w14 &= 0x3ffffff; + /* k = 15 */ + lo = Math.imul(al9, bl6); + mid = Math.imul(al9, bh6); + mid = (mid + Math.imul(ah9, bl6)) | 0; + hi = Math.imul(ah9, bh6); + lo = (lo + Math.imul(al8, bl7)) | 0; + mid = (mid + Math.imul(al8, bh7)) | 0; + mid = (mid + Math.imul(ah8, bl7)) | 0; + hi = (hi + Math.imul(ah8, bh7)) | 0; + lo = (lo + Math.imul(al7, bl8)) | 0; + mid = (mid + Math.imul(al7, bh8)) | 0; + mid = (mid + Math.imul(ah7, bl8)) | 0; + hi = (hi + Math.imul(ah7, bh8)) | 0; + lo = (lo + Math.imul(al6, bl9)) | 0; + mid = (mid + Math.imul(al6, bh9)) | 0; + mid = (mid + Math.imul(ah6, bl9)) | 0; + hi = (hi + Math.imul(ah6, bh9)) | 0; + var w15 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w15 >>> 26)) | 0; + w15 &= 0x3ffffff; + /* k = 16 */ + lo = Math.imul(al9, bl7); + mid = Math.imul(al9, bh7); + mid = (mid + Math.imul(ah9, bl7)) | 0; + hi = Math.imul(ah9, bh7); + lo = (lo + Math.imul(al8, bl8)) | 0; + mid = (mid + Math.imul(al8, bh8)) | 0; + mid = (mid + Math.imul(ah8, bl8)) | 0; + hi = (hi + Math.imul(ah8, bh8)) | 0; + lo = (lo + Math.imul(al7, bl9)) | 0; + mid = (mid + Math.imul(al7, bh9)) | 0; + mid = (mid + Math.imul(ah7, bl9)) | 0; + hi = (hi + Math.imul(ah7, bh9)) | 0; + var w16 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w16 >>> 26)) | 0; + w16 &= 0x3ffffff; + /* k = 17 */ + lo = Math.imul(al9, bl8); + mid = Math.imul(al9, bh8); + mid = (mid + Math.imul(ah9, bl8)) | 0; + hi = Math.imul(ah9, bh8); + lo = (lo + Math.imul(al8, bl9)) | 0; + mid = (mid + Math.imul(al8, bh9)) | 0; + mid = (mid + Math.imul(ah8, bl9)) | 0; + hi = (hi + Math.imul(ah8, bh9)) | 0; + var w17 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w17 >>> 26)) | 0; + w17 &= 0x3ffffff; + /* k = 18 */ + lo = Math.imul(al9, bl9); + mid = Math.imul(al9, bh9); + mid = (mid + Math.imul(ah9, bl9)) | 0; + hi = Math.imul(ah9, bh9); + var w18 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w18 >>> 26)) | 0; + w18 &= 0x3ffffff; + o[0] = w0; + o[1] = w1; + o[2] = w2; + o[3] = w3; + o[4] = w4; + o[5] = w5; + o[6] = w6; + o[7] = w7; + o[8] = w8; + o[9] = w9; + o[10] = w10; + o[11] = w11; + o[12] = w12; + o[13] = w13; + o[14] = w14; + o[15] = w15; + o[16] = w16; + o[17] = w17; + o[18] = w18; + if (c !== 0) { + o[19] = c; + out.length++; + } + return out; + }; + + // Polyfill comb + if (!Math.imul) { + comb10MulTo = smallMulTo; + } - /** - * Generic Greater or equal. - * Uses either the constant or secret version of this operator depending on type of paramter. - * @method gteq - * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.gteq = function (o, op_id) { - if (this.isConstant(o)) { - return this.cgteq(o, op_id); - } - return this.sgteq(o); - }; + function bigMulTo(self, num, out) { + out.negative = num.negative ^ self.negative; + out.length = self.length + num.length; + + var carry = 0; + var hncarry = 0; + for (var k = 0; k < out.length - 1; k++) { + // Sum all words with the same `i + j = k` and accumulate `ncarry`, + // note that ncarry could be >= 0x3ffffff + var ncarry = hncarry; + hncarry = 0; + var rword = carry & 0x3ffffff; + var maxJ = Math.min(k, num.length - 1); + for (var j = Math.max(0, k - self.length + 1); j <= maxJ; j++) { + var i = k - j; + var a = self.words[i] | 0; + var b = num.words[j] | 0; + var r = a * b; + + var lo = r & 0x3ffffff; + ncarry = (ncarry + ((r / 0x4000000) | 0)) | 0; + lo = (lo + rword) | 0; + rword = lo & 0x3ffffff; + ncarry = (ncarry + (lo >>> 26)) | 0; + + hncarry += ncarry >>> 26; + ncarry &= 0x3ffffff; + } + out.words[k] = rword; + carry = ncarry; + ncarry = hncarry; + } + if (carry !== 0) { + out.words[k] = carry; + } else { + out.length--; + } + + return out.strip(); + } + function jumboMulTo(self, num, out) { + var fftm = new FFTM(); + return fftm.mulp(self, num, out); + } - /** - * Generic Greater than. - * Uses either the constant or secret version of this operator depending on type of paramter. - * @method gt - * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.gt = function (o, op_id) { - if (this.isConstant(o)) { - return this.cgt(o, op_id); - } - return this.sgt(o, op_id); - }; + BN.prototype.mulTo = function mulTo(num, out) { + var res; + var len = this.length + num.length; + if (this.length === 10 && num.length === 10) { + res = comb10MulTo(this, num, out); + } else if (len < 63) { + res = smallMulTo(this, num, out); + } else if (len < 1024) { + res = bigMulTo(this, num, out); + } else { + res = jumboMulTo(this, num, out); + } + + return res; + }; + + // Cooley-Tukey algorithm for FFT + // slightly revisited to rely on looping instead of recursion + + function FFTM(x, y) { + this.x = x; + this.y = y; + } + FFTM.prototype.makeRBT = function makeRBT(N) { + var t = new Array(N); + var l = BN.prototype._countBits(N) - 1; + for (var i = 0; i < N; i++) { + t[i] = this.revBin(i, l, N); + } - /** - * Generic Less or equal. - * Uses either the constant or secret version of this operator depending on type of paramter. - * @method lteq - * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.lteq = function (o, op_id) { - if (this.isConstant(o)) { - return this.clteq(o, op_id); - } - return this.slteq(o, op_id); - }; + return t; + }; + // Returns binary-reversed representation of `x` + FFTM.prototype.revBin = function revBin(x, l, N) { + if (x === 0 || x === N - 1) return x; - /** - * Generic Less than. - * Uses either the constant or secret version of this operator depending on type of paramter. - * @method lt - * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.lt = function (o, op_id) { - if (this.isConstant(o)) { - return this.clt(o, op_id); - } - return this.slt(o, op_id); - }; + var rb = 0; + for (var i = 0; i < l; i++) { + rb |= (x & 1) << (l - i - 1); + x >>= 1; + } + return rb; + }; + + // Performs "tweedling" phase, therefore 'emulating' + // behaviour of the recursive algorithm + FFTM.prototype.permute = function permute(rbt, rws, iws, rtws, itws, N) { + for (var i = 0; i < N; i++) { + rtws[i] = rws[rbt[i]]; + itws[i] = iws[rbt[i]]; + } + }; + + FFTM.prototype.transform = function transform(rws, iws, rtws, itws, N, rbt) { + this.permute(rbt, rws, iws, rtws, itws, N); + + for (var s = 1; s < N; s <<= 1) { + var l = s << 1; + + var rtwdf = Math.cos((2 * Math.PI) / l); + var itwdf = Math.sin((2 * Math.PI) / l); + + for (var p = 0; p < N; p += l) { + var rtwdf_ = rtwdf; + var itwdf_ = itwdf; + + for (var j = 0; j < s; j++) { + var re = rtws[p + j]; + var ie = itws[p + j]; + + var ro = rtws[p + j + s]; + var io = itws[p + j + s]; + + var rx = rtwdf_ * ro - itwdf_ * io; - /** - * Generic Equals. - * Uses either the constant or secret version of this operator depending on type of paramter. - * @method eq - * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.eq = function (o, op_id) { - if (this.isConstant(o)) { - return this.ceq(o, op_id); - } - return this.seq(o, op_id); - }; + io = rtwdf_ * io + itwdf_ * ro; + ro = rx; + rtws[p + j] = re + ro; + itws[p + j] = ie + io; - /** - * Generic Not Equals. - * Uses either the constant or secret version of this operator depending on type of paramter. - * @method neq - * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.neq = function (o, op_id) { - if (this.isConstant(o)) { - return this.cneq(o, op_id); - } - return this.sneq(o, op_id); - }; + rtws[p + j + s] = re - ro; + itws[p + j + s] = ie - io; + /* jshint maxdepth : false */ + if (j !== l) { + rx = rtwdf * rtwdf_ - itwdf * itwdf_; - /** - * Generic Integer Divison. - * Uses either the constant or secret version of this operator depending on type of paramter. - * @method div - * @param {number|module:jiff-client~JIFFClient#SecretShare} o - the other operand (can be either number or share). - * @param {number} l - the maximum bit length of the two shares. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.div = function (o, l, op_id) { - if (this.isConstant(o)) { - return this.cdiv(o, l, op_id); - } - return this.sdiv(o, l, op_id); - }; -}; -},{}],31:[function(require,module,exports){ -// Arithmetic operations on shares -module.exports = function (SecretShare) { - /** - * Addition with a constant. - * @method cadd - * @param {number} cst - the constant to add. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.cadd = function (cst) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (+)'); - } + itwdf_ = rtwdf * itwdf_ + itwdf * rtwdf_; + rtwdf_ = rx; + } + } + } + } + }; - var self = this; - var ready = function () { - return self.jiff.helpers.mod(self.jiff.share_helpers['+'](self.value, cst), self.Zp); - }; + FFTM.prototype.guessLen13b = function guessLen13b(n, m) { + var N = Math.max(m, n) | 1; + var odd = N & 1; + var i = 0; + for (N = (N / 2) | 0; N; N = N >>> 1) { + i++; + } - return new this.jiff.SecretShare(this.wThen(ready), this.holders, this.threshold, this.Zp); - }; + return 1 << (i + 1 + odd); + }; - /** - * Subtraction with a constant. - * @method csub - * @param {number} cst - the constant to subtract from this share. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.csub = function (cst) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (-)'); - } + FFTM.prototype.conjugate = function conjugate(rws, iws, N) { + if (N <= 1) return; - var self = this; - var ready = function () { - return self.jiff.helpers.mod(self.jiff.share_helpers['-'](self.value, cst), self.Zp); - }; + for (var i = 0; i < N / 2; i++) { + var t = rws[i]; + + rws[i] = rws[N - i - 1]; + rws[N - i - 1] = t; + + t = iws[i]; + + iws[i] = -iws[N - i - 1]; + iws[N - i - 1] = -t; + } + }; + + FFTM.prototype.normalize13b = function normalize13b(ws, N) { + var carry = 0; + for (var i = 0; i < N / 2; i++) { + var w = Math.round(ws[2 * i + 1] / N) * 0x2000 + Math.round(ws[2 * i] / N) + carry; + + ws[i] = w & 0x3ffffff; + + if (w < 0x4000000) { + carry = 0; + } else { + carry = (w / 0x4000000) | 0; + } + } + + return ws; + }; + + FFTM.prototype.convert13b = function convert13b(ws, len, rws, N) { + var carry = 0; + for (var i = 0; i < len; i++) { + carry = carry + (ws[i] | 0); + + rws[2 * i] = carry & 0x1fff; + carry = carry >>> 13; + rws[2 * i + 1] = carry & 0x1fff; + carry = carry >>> 13; + } + + // Pad with zeroes + for (i = 2 * len; i < N; ++i) { + rws[i] = 0; + } + + assert(carry === 0); + assert((carry & ~0x1fff) === 0); + }; + + FFTM.prototype.stub = function stub(N) { + var ph = new Array(N); + for (var i = 0; i < N; i++) { + ph[i] = 0; + } + + return ph; + }; + + FFTM.prototype.mulp = function mulp(x, y, out) { + var N = 2 * this.guessLen13b(x.length, y.length); + + var rbt = this.makeRBT(N); + + var _ = this.stub(N); + + var rws = new Array(N); + var rwst = new Array(N); + var iwst = new Array(N); + + var nrws = new Array(N); + var nrwst = new Array(N); + var niwst = new Array(N); + + var rmws = out.words; + rmws.length = N; + + this.convert13b(x.words, x.length, rws, N); + this.convert13b(y.words, y.length, nrws, N); + + this.transform(rws, _, rwst, iwst, N, rbt); + this.transform(nrws, _, nrwst, niwst, N, rbt); + + for (var i = 0; i < N; i++) { + var rx = rwst[i] * nrwst[i] - iwst[i] * niwst[i]; + iwst[i] = rwst[i] * niwst[i] + iwst[i] * nrwst[i]; + rwst[i] = rx; + } + + this.conjugate(rwst, iwst, N); + this.transform(rwst, iwst, rmws, _, N, rbt); + this.conjugate(rmws, _, N); + this.normalize13b(rmws, N); + + out.negative = x.negative ^ y.negative; + out.length = x.length + y.length; + return out.strip(); + }; + + // Multiply `this` by `num` + BN.prototype.mul = function mul(num) { + var out = new BN(null); + out.words = new Array(this.length + num.length); + return this.mulTo(num, out); + }; + + // Multiply employing FFT + BN.prototype.mulf = function mulf(num) { + var out = new BN(null); + out.words = new Array(this.length + num.length); + return jumboMulTo(this, num, out); + }; + + // In-place Multiplication + BN.prototype.imul = function imul(num) { + return this.clone().mulTo(num, this); + }; + + BN.prototype.imuln = function imuln(num) { + assert(typeof num === 'number'); + assert(num < 0x4000000); + + // Carry + var carry = 0; + for (var i = 0; i < this.length; i++) { + var w = (this.words[i] | 0) * num; + var lo = (w & 0x3ffffff) + (carry & 0x3ffffff); + carry >>= 26; + carry += (w / 0x4000000) | 0; + // NOTE: lo is 27bit maximum + carry += lo >>> 26; + this.words[i] = lo & 0x3ffffff; + } + + if (carry !== 0) { + this.words[i] = carry; + this.length++; + } + + return this; + }; + + BN.prototype.muln = function muln(num) { + return this.clone().imuln(num); + }; + + // `this` * `this` + BN.prototype.sqr = function sqr() { + return this.mul(this); + }; + + // `this` * `this` in-place + BN.prototype.isqr = function isqr() { + return this.imul(this.clone()); + }; + + // Math.pow(`this`, `num`) + BN.prototype.pow = function pow(num) { + var w = toBitArray(num); + if (w.length === 0) return new BN(1); + + // Skip leading zeroes + var res = this; + for (var i = 0; i < w.length; i++, res = res.sqr()) { + if (w[i] !== 0) break; + } + + if (++i < w.length) { + for (var q = res.sqr(); i < w.length; i++, q = q.sqr()) { + if (w[i] === 0) continue; + + res = res.mul(q); + } + } + + return res; + }; + + // Shift-left in-place + BN.prototype.iushln = function iushln(bits) { + assert(typeof bits === 'number' && bits >= 0); + var r = bits % 26; + var s = (bits - r) / 26; + var carryMask = (0x3ffffff >>> (26 - r)) << (26 - r); + var i; + + if (r !== 0) { + var carry = 0; + + for (i = 0; i < this.length; i++) { + var newCarry = this.words[i] & carryMask; + var c = ((this.words[i] | 0) - newCarry) << r; + this.words[i] = c | carry; + carry = newCarry >>> (26 - r); + } + + if (carry) { + this.words[i] = carry; + this.length++; + } + } + + if (s !== 0) { + for (i = this.length - 1; i >= 0; i--) { + this.words[i + s] = this.words[i]; + } + + for (i = 0; i < s; i++) { + this.words[i] = 0; + } + + this.length += s; + } + + return this.strip(); + }; + + BN.prototype.ishln = function ishln(bits) { + // TODO(indutny): implement me + assert(this.negative === 0); + return this.iushln(bits); + }; + + // Shift-right in-place + // NOTE: `hint` is a lowest bit before trailing zeroes + // NOTE: if `extended` is present - it will be filled with destroyed bits + BN.prototype.iushrn = function iushrn(bits, hint, extended) { + assert(typeof bits === 'number' && bits >= 0); + var h; + if (hint) { + h = (hint - (hint % 26)) / 26; + } else { + h = 0; + } + + var r = bits % 26; + var s = Math.min((bits - r) / 26, this.length); + var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r); + var maskedWords = extended; + + h -= s; + h = Math.max(0, h); + + // Extended mode, copy masked part + if (maskedWords) { + for (var i = 0; i < s; i++) { + maskedWords.words[i] = this.words[i]; + } + maskedWords.length = s; + } + + if (s === 0) { + // No-op, we should not move anything at all + } else if (this.length > s) { + this.length -= s; + for (i = 0; i < this.length; i++) { + this.words[i] = this.words[i + s]; + } + } else { + this.words[0] = 0; + this.length = 1; + } + + var carry = 0; + for (i = this.length - 1; i >= 0 && (carry !== 0 || i >= h); i--) { + var word = this.words[i] | 0; + this.words[i] = (carry << (26 - r)) | (word >>> r); + carry = word & mask; + } + + // Push carried bits as a mask + if (maskedWords && carry !== 0) { + maskedWords.words[maskedWords.length++] = carry; + } + + if (this.length === 0) { + this.words[0] = 0; + this.length = 1; + } + + return this.strip(); + }; + + BN.prototype.ishrn = function ishrn(bits, hint, extended) { + // TODO(indutny): implement me + assert(this.negative === 0); + return this.iushrn(bits, hint, extended); + }; + + // Shift-left + BN.prototype.shln = function shln(bits) { + return this.clone().ishln(bits); + }; + + BN.prototype.ushln = function ushln(bits) { + return this.clone().iushln(bits); + }; + + // Shift-right + BN.prototype.shrn = function shrn(bits) { + return this.clone().ishrn(bits); + }; + + BN.prototype.ushrn = function ushrn(bits) { + return this.clone().iushrn(bits); + }; + + // Test if n bit is set + BN.prototype.testn = function testn(bit) { + assert(typeof bit === 'number' && bit >= 0); + var r = bit % 26; + var s = (bit - r) / 26; + var q = 1 << r; + + // Fast case: bit is much higher than all existing words + if (this.length <= s) return false; + + // Check bit and return + var w = this.words[s]; + + return !!(w & q); + }; + + // Return only lowers bits of number (in-place) + BN.prototype.imaskn = function imaskn(bits) { + assert(typeof bits === 'number' && bits >= 0); + var r = bits % 26; + var s = (bits - r) / 26; + + assert(this.negative === 0, 'imaskn works only with positive numbers'); + + if (this.length <= s) { + return this; + } + + if (r !== 0) { + s++; + } + this.length = Math.min(s, this.length); + + if (r !== 0) { + var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r); + this.words[this.length - 1] &= mask; + } + + return this.strip(); + }; + + // Return only lowers bits of number + BN.prototype.maskn = function maskn(bits) { + return this.clone().imaskn(bits); + }; + + // Add plain number `num` to `this` + BN.prototype.iaddn = function iaddn(num) { + assert(typeof num === 'number'); + assert(num < 0x4000000); + if (num < 0) return this.isubn(-num); + + // Possible sign change + if (this.negative !== 0) { + if (this.length === 1 && (this.words[0] | 0) < num) { + this.words[0] = num - (this.words[0] | 0); + this.negative = 0; + return this; + } + + this.negative = 0; + this.isubn(num); + this.negative = 1; + return this; + } + + // Add without checks + return this._iaddn(num); + }; + + BN.prototype._iaddn = function _iaddn(num) { + this.words[0] += num; + + // Carry + for (var i = 0; i < this.length && this.words[i] >= 0x4000000; i++) { + this.words[i] -= 0x4000000; + if (i === this.length - 1) { + this.words[i + 1] = 1; + } else { + this.words[i + 1]++; + } + } + this.length = Math.max(this.length, i + 1); + + return this; + }; + + // Subtract plain number `num` from `this` + BN.prototype.isubn = function isubn(num) { + assert(typeof num === 'number'); + assert(num < 0x4000000); + if (num < 0) return this.iaddn(-num); + + if (this.negative !== 0) { + this.negative = 0; + this.iaddn(num); + this.negative = 1; + return this; + } + + this.words[0] -= num; + + if (this.length === 1 && this.words[0] < 0) { + this.words[0] = -this.words[0]; + this.negative = 1; + } else { + // Carry + for (var i = 0; i < this.length && this.words[i] < 0; i++) { + this.words[i] += 0x4000000; + this.words[i + 1] -= 1; + } + } + + return this.strip(); + }; + + BN.prototype.addn = function addn(num) { + return this.clone().iaddn(num); + }; + + BN.prototype.subn = function subn(num) { + return this.clone().isubn(num); + }; + + BN.prototype.iabs = function iabs() { + this.negative = 0; + + return this; + }; + + BN.prototype.abs = function abs() { + return this.clone().iabs(); + }; + + BN.prototype._ishlnsubmul = function _ishlnsubmul(num, mul, shift) { + var len = num.length + shift; + var i; + + this._expand(len); + + var w; + var carry = 0; + for (i = 0; i < num.length; i++) { + w = (this.words[i + shift] | 0) + carry; + var right = (num.words[i] | 0) * mul; + w -= right & 0x3ffffff; + carry = (w >> 26) - ((right / 0x4000000) | 0); + this.words[i + shift] = w & 0x3ffffff; + } + for (; i < this.length - shift; i++) { + w = (this.words[i + shift] | 0) + carry; + carry = w >> 26; + this.words[i + shift] = w & 0x3ffffff; + } + + if (carry === 0) return this.strip(); + + // Subtraction overflow + assert(carry === -1); + carry = 0; + for (i = 0; i < this.length; i++) { + w = -(this.words[i] | 0) + carry; + carry = w >> 26; + this.words[i] = w & 0x3ffffff; + } + this.negative = 1; + + return this.strip(); + }; + + BN.prototype._wordDiv = function _wordDiv(num, mode) { + var shift = this.length - num.length; + + var a = this.clone(); + var b = num; + + // Normalize + var bhi = b.words[b.length - 1] | 0; + var bhiBits = this._countBits(bhi); + shift = 26 - bhiBits; + if (shift !== 0) { + b = b.ushln(shift); + a.iushln(shift); + bhi = b.words[b.length - 1] | 0; + } + + // Initialize quotient + var m = a.length - b.length; + var q; + + if (mode !== 'mod') { + q = new BN(null); + q.length = m + 1; + q.words = new Array(q.length); + for (var i = 0; i < q.length; i++) { + q.words[i] = 0; + } + } + + var diff = a.clone()._ishlnsubmul(b, 1, m); + if (diff.negative === 0) { + a = diff; + if (q) { + q.words[m] = 1; + } + } + + for (var j = m - 1; j >= 0; j--) { + var qj = (a.words[b.length + j] | 0) * 0x4000000 + (a.words[b.length + j - 1] | 0); + + // NOTE: (qj / bhi) is (0x3ffffff * 0x4000000 + 0x3ffffff) / 0x2000000 max + // (0x7ffffff) + qj = Math.min((qj / bhi) | 0, 0x3ffffff); + + a._ishlnsubmul(b, qj, j); + while (a.negative !== 0) { + qj--; + a.negative = 0; + a._ishlnsubmul(b, 1, j); + if (!a.isZero()) { + a.negative ^= 1; + } + } + if (q) { + q.words[j] = qj; + } + } + if (q) { + q.strip(); + } + a.strip(); + + // Denormalize + if (mode !== 'div' && shift !== 0) { + a.iushrn(shift); + } + + return { + div: q || null, + mod: a + }; + }; + + // NOTE: 1) `mode` can be set to `mod` to request mod only, + // to `div` to request div only, or be absent to + // request both div & mod + // 2) `positive` is true if unsigned mod is requested + BN.prototype.divmod = function divmod(num, mode, positive) { + assert(!num.isZero()); + + if (this.isZero()) { + return { + div: new BN(0), + mod: new BN(0) + }; + } + + var div, mod, res; + if (this.negative !== 0 && num.negative === 0) { + res = this.neg().divmod(num, mode); + + if (mode !== 'mod') { + div = res.div.neg(); + } + + if (mode !== 'div') { + mod = res.mod.neg(); + if (positive && mod.negative !== 0) { + mod.iadd(num); + } + } + + return { + div: div, + mod: mod + }; + } + + if (this.negative === 0 && num.negative !== 0) { + res = this.divmod(num.neg(), mode); + + if (mode !== 'mod') { + div = res.div.neg(); + } + + return { + div: div, + mod: res.mod + }; + } + + if ((this.negative & num.negative) !== 0) { + res = this.neg().divmod(num.neg(), mode); + + if (mode !== 'div') { + mod = res.mod.neg(); + if (positive && mod.negative !== 0) { + mod.isub(num); + } + } + + return { + div: res.div, + mod: mod + }; + } + + // Both numbers are positive at this point + + // Strip both numbers to approximate shift value + if (num.length > this.length || this.cmp(num) < 0) { + return { + div: new BN(0), + mod: this + }; + } + + // Very short reduction + if (num.length === 1) { + if (mode === 'div') { + return { + div: this.divn(num.words[0]), + mod: null + }; + } + + if (mode === 'mod') { + return { + div: null, + mod: new BN(this.modn(num.words[0])) + }; + } + + return { + div: this.divn(num.words[0]), + mod: new BN(this.modn(num.words[0])) + }; + } + + return this._wordDiv(num, mode); + }; + + // Find `this` / `num` + BN.prototype.div = function div(num) { + return this.divmod(num, 'div', false).div; + }; + + // Find `this` % `num` + BN.prototype.mod = function mod(num) { + return this.divmod(num, 'mod', false).mod; + }; + + BN.prototype.umod = function umod(num) { + return this.divmod(num, 'mod', true).mod; + }; + + // Find Round(`this` / `num`) + BN.prototype.divRound = function divRound(num) { + var dm = this.divmod(num); + + // Fast case - exact division + if (dm.mod.isZero()) return dm.div; + + var mod = dm.div.negative !== 0 ? dm.mod.isub(num) : dm.mod; + + var half = num.ushrn(1); + var r2 = num.andln(1); + var cmp = mod.cmp(half); + + // Round down + if (cmp < 0 || (r2 === 1 && cmp === 0)) return dm.div; + + // Round up + return dm.div.negative !== 0 ? dm.div.isubn(1) : dm.div.iaddn(1); + }; + + BN.prototype.modn = function modn(num) { + assert(num <= 0x3ffffff); + var p = (1 << 26) % num; + + var acc = 0; + for (var i = this.length - 1; i >= 0; i--) { + acc = (p * acc + (this.words[i] | 0)) % num; + } + + return acc; + }; + + // In-place division by number + BN.prototype.idivn = function idivn(num) { + assert(num <= 0x3ffffff); + + var carry = 0; + for (var i = this.length - 1; i >= 0; i--) { + var w = (this.words[i] | 0) + carry * 0x4000000; + this.words[i] = (w / num) | 0; + carry = w % num; + } + + return this.strip(); + }; + + BN.prototype.divn = function divn(num) { + return this.clone().idivn(num); + }; + + BN.prototype.egcd = function egcd(p) { + assert(p.negative === 0); + assert(!p.isZero()); + + var x = this; + var y = p.clone(); + + if (x.negative !== 0) { + x = x.umod(p); + } else { + x = x.clone(); + } + + // A * x + B * y = x + var A = new BN(1); + var B = new BN(0); + + // C * x + D * y = y + var C = new BN(0); + var D = new BN(1); + + var g = 0; + + while (x.isEven() && y.isEven()) { + x.iushrn(1); + y.iushrn(1); + ++g; + } + + var yp = y.clone(); + var xp = x.clone(); + + while (!x.isZero()) { + for (var i = 0, im = 1; (x.words[0] & im) === 0 && i < 26; ++i, im <<= 1); + if (i > 0) { + x.iushrn(i); + while (i-- > 0) { + if (A.isOdd() || B.isOdd()) { + A.iadd(yp); + B.isub(xp); + } + + A.iushrn(1); + B.iushrn(1); + } + } + + for (var j = 0, jm = 1; (y.words[0] & jm) === 0 && j < 26; ++j, jm <<= 1); + if (j > 0) { + y.iushrn(j); + while (j-- > 0) { + if (C.isOdd() || D.isOdd()) { + C.iadd(yp); + D.isub(xp); + } + + C.iushrn(1); + D.iushrn(1); + } + } + + if (x.cmp(y) >= 0) { + x.isub(y); + A.isub(C); + B.isub(D); + } else { + y.isub(x); + C.isub(A); + D.isub(B); + } + } + + return { + a: C, + b: D, + gcd: y.iushln(g) + }; + }; + + // This is reduced incarnation of the binary EEA + // above, designated to invert members of the + // _prime_ fields F(p) at a maximal speed + BN.prototype._invmp = function _invmp(p) { + assert(p.negative === 0); + assert(!p.isZero()); + + var a = this; + var b = p.clone(); + + if (a.negative !== 0) { + a = a.umod(p); + } else { + a = a.clone(); + } + + var x1 = new BN(1); + var x2 = new BN(0); + + var delta = b.clone(); + + while (a.cmpn(1) > 0 && b.cmpn(1) > 0) { + for (var i = 0, im = 1; (a.words[0] & im) === 0 && i < 26; ++i, im <<= 1); + if (i > 0) { + a.iushrn(i); + while (i-- > 0) { + if (x1.isOdd()) { + x1.iadd(delta); + } + + x1.iushrn(1); + } + } + + for (var j = 0, jm = 1; (b.words[0] & jm) === 0 && j < 26; ++j, jm <<= 1); + if (j > 0) { + b.iushrn(j); + while (j-- > 0) { + if (x2.isOdd()) { + x2.iadd(delta); + } + + x2.iushrn(1); + } + } + + if (a.cmp(b) >= 0) { + a.isub(b); + x1.isub(x2); + } else { + b.isub(a); + x2.isub(x1); + } + } + + var res; + if (a.cmpn(1) === 0) { + res = x1; + } else { + res = x2; + } + + if (res.cmpn(0) < 0) { + res.iadd(p); + } + + return res; + }; + + BN.prototype.gcd = function gcd(num) { + if (this.isZero()) return num.abs(); + if (num.isZero()) return this.abs(); + + var a = this.clone(); + var b = num.clone(); + a.negative = 0; + b.negative = 0; + + // Remove common factor of two + for (var shift = 0; a.isEven() && b.isEven(); shift++) { + a.iushrn(1); + b.iushrn(1); + } + + do { + while (a.isEven()) { + a.iushrn(1); + } + while (b.isEven()) { + b.iushrn(1); + } + + var r = a.cmp(b); + if (r < 0) { + // Swap `a` and `b` to make `a` always bigger than `b` + var t = a; + a = b; + b = t; + } else if (r === 0 || b.cmpn(1) === 0) { + break; + } + + a.isub(b); + } while (true); + + return b.iushln(shift); + }; + + // Invert number in the field F(num) + BN.prototype.invm = function invm(num) { + return this.egcd(num).a.umod(num); + }; + + BN.prototype.isEven = function isEven() { + return (this.words[0] & 1) === 0; + }; + + BN.prototype.isOdd = function isOdd() { + return (this.words[0] & 1) === 1; + }; + + // And first word and num + BN.prototype.andln = function andln(num) { + return this.words[0] & num; + }; + + // Increment at the bit position in-line + BN.prototype.bincn = function bincn(bit) { + assert(typeof bit === 'number'); + var r = bit % 26; + var s = (bit - r) / 26; + var q = 1 << r; + + // Fast case: bit is much higher than all existing words + if (this.length <= s) { + this._expand(s + 1); + this.words[s] |= q; + return this; + } + + // Add bit and propagate, if needed + var carry = q; + for (var i = s; carry !== 0 && i < this.length; i++) { + var w = this.words[i] | 0; + w += carry; + carry = w >>> 26; + w &= 0x3ffffff; + this.words[i] = w; + } + if (carry !== 0) { + this.words[i] = carry; + this.length++; + } + return this; + }; + + BN.prototype.isZero = function isZero() { + return this.length === 1 && this.words[0] === 0; + }; + + BN.prototype.cmpn = function cmpn(num) { + var negative = num < 0; + + if (this.negative !== 0 && !negative) return -1; + if (this.negative === 0 && negative) return 1; + + this.strip(); + + var res; + if (this.length > 1) { + res = 1; + } else { + if (negative) { + num = -num; + } + + assert(num <= 0x3ffffff, 'Number is too big'); + + var w = this.words[0] | 0; + res = w === num ? 0 : w < num ? -1 : 1; + } + if (this.negative !== 0) return -res | 0; + return res; + }; + + // Compare two numbers and return: + // 1 - if `this` > `num` + // 0 - if `this` == `num` + // -1 - if `this` < `num` + BN.prototype.cmp = function cmp(num) { + if (this.negative !== 0 && num.negative === 0) return -1; + if (this.negative === 0 && num.negative !== 0) return 1; + + var res = this.ucmp(num); + if (this.negative !== 0) return -res | 0; + return res; + }; + + // Unsigned comparison + BN.prototype.ucmp = function ucmp(num) { + // At this point both numbers have the same sign + if (this.length > num.length) return 1; + if (this.length < num.length) return -1; + + var res = 0; + for (var i = this.length - 1; i >= 0; i--) { + var a = this.words[i] | 0; + var b = num.words[i] | 0; + + if (a === b) continue; + if (a < b) { + res = -1; + } else if (a > b) { + res = 1; + } + break; + } + return res; + }; + + BN.prototype.gtn = function gtn(num) { + return this.cmpn(num) === 1; + }; + + BN.prototype.gt = function gt(num) { + return this.cmp(num) === 1; + }; + + BN.prototype.gten = function gten(num) { + return this.cmpn(num) >= 0; + }; + + BN.prototype.gte = function gte(num) { + return this.cmp(num) >= 0; + }; + + BN.prototype.ltn = function ltn(num) { + return this.cmpn(num) === -1; + }; + + BN.prototype.lt = function lt(num) { + return this.cmp(num) === -1; + }; + + BN.prototype.lten = function lten(num) { + return this.cmpn(num) <= 0; + }; + + BN.prototype.lte = function lte(num) { + return this.cmp(num) <= 0; + }; + + BN.prototype.eqn = function eqn(num) { + return this.cmpn(num) === 0; + }; + + BN.prototype.eq = function eq(num) { + return this.cmp(num) === 0; + }; + + // + // A reduce context, could be using montgomery or something better, depending + // on the `m` itself. + // + BN.red = function red(num) { + return new Red(num); + }; + + BN.prototype.toRed = function toRed(ctx) { + assert(!this.red, 'Already a number in reduction context'); + assert(this.negative === 0, 'red works only with positives'); + return ctx.convertTo(this)._forceRed(ctx); + }; + + BN.prototype.fromRed = function fromRed() { + assert(this.red, 'fromRed works only with numbers in reduction context'); + return this.red.convertFrom(this); + }; + + BN.prototype._forceRed = function _forceRed(ctx) { + this.red = ctx; + return this; + }; + + BN.prototype.forceRed = function forceRed(ctx) { + assert(!this.red, 'Already a number in reduction context'); + return this._forceRed(ctx); + }; + + BN.prototype.redAdd = function redAdd(num) { + assert(this.red, 'redAdd works only with red numbers'); + return this.red.add(this, num); + }; + + BN.prototype.redIAdd = function redIAdd(num) { + assert(this.red, 'redIAdd works only with red numbers'); + return this.red.iadd(this, num); + }; + + BN.prototype.redSub = function redSub(num) { + assert(this.red, 'redSub works only with red numbers'); + return this.red.sub(this, num); + }; + + BN.prototype.redISub = function redISub(num) { + assert(this.red, 'redISub works only with red numbers'); + return this.red.isub(this, num); + }; + + BN.prototype.redShl = function redShl(num) { + assert(this.red, 'redShl works only with red numbers'); + return this.red.shl(this, num); + }; + + BN.prototype.redMul = function redMul(num) { + assert(this.red, 'redMul works only with red numbers'); + this.red._verify2(this, num); + return this.red.mul(this, num); + }; + + BN.prototype.redIMul = function redIMul(num) { + assert(this.red, 'redMul works only with red numbers'); + this.red._verify2(this, num); + return this.red.imul(this, num); + }; + + BN.prototype.redSqr = function redSqr() { + assert(this.red, 'redSqr works only with red numbers'); + this.red._verify1(this); + return this.red.sqr(this); + }; + + BN.prototype.redISqr = function redISqr() { + assert(this.red, 'redISqr works only with red numbers'); + this.red._verify1(this); + return this.red.isqr(this); + }; + + // Square root over p + BN.prototype.redSqrt = function redSqrt() { + assert(this.red, 'redSqrt works only with red numbers'); + this.red._verify1(this); + return this.red.sqrt(this); + }; + + BN.prototype.redInvm = function redInvm() { + assert(this.red, 'redInvm works only with red numbers'); + this.red._verify1(this); + return this.red.invm(this); + }; + + // Return negative clone of `this` % `red modulo` + BN.prototype.redNeg = function redNeg() { + assert(this.red, 'redNeg works only with red numbers'); + this.red._verify1(this); + return this.red.neg(this); + }; + + BN.prototype.redPow = function redPow(num) { + assert(this.red && !num.red, 'redPow(normalNum)'); + this.red._verify1(this); + return this.red.pow(this, num); + }; + + // Prime numbers with efficient reduction + var primes = { + k256: null, + p224: null, + p192: null, + p25519: null + }; + + // Pseudo-Mersenne prime + function MPrime(name, p) { + // P = 2 ^ N - K + this.name = name; + this.p = new BN(p, 16); + this.n = this.p.bitLength(); + this.k = new BN(1).iushln(this.n).isub(this.p); + + this.tmp = this._tmp(); + } - return new this.jiff.SecretShare(this.wThen(ready), this.holders, this.threshold, this.Zp); - }; + MPrime.prototype._tmp = function _tmp() { + var tmp = new BN(null); + tmp.words = new Array(Math.ceil(this.n / 13)); + return tmp; + }; + + MPrime.prototype.ireduce = function ireduce(num) { + // Assumes that `num` is less than `P^2` + // num = HI * (2 ^ N - K) + HI * K + LO = HI * K + LO (mod P) + var r = num; + var rlen; + + do { + this.split(r, this.tmp); + r = this.imulK(r); + r = r.iadd(this.tmp); + rlen = r.bitLength(); + } while (rlen > this.n); + + var cmp = rlen < this.n ? -1 : r.ucmp(this.p); + if (cmp === 0) { + r.words[0] = 0; + r.length = 1; + } else if (cmp > 0) { + r.isub(this.p); + } else { + if (r.strip !== undefined) { + // r is BN v4 instance + r.strip(); + } else { + // r is BN v5 instance + r._strip(); + } + } + + return r; + }; + + MPrime.prototype.split = function split(input, out) { + input.iushrn(this.n, 0, out); + }; + + MPrime.prototype.imulK = function imulK(num) { + return num.imul(this.k); + }; + + function K256() { + MPrime.call(this, 'k256', 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff fffffffe fffffc2f'); + } + inherits(K256, MPrime); + + K256.prototype.split = function split(input, output) { + // 256 = 9 * 26 + 22 + var mask = 0x3fffff; + + var outLen = Math.min(input.length, 9); + for (var i = 0; i < outLen; i++) { + output.words[i] = input.words[i]; + } + output.length = outLen; + + if (input.length <= 9) { + input.words[0] = 0; + input.length = 1; + return; + } + + // Shift by 9 limbs + var prev = input.words[9]; + output.words[output.length++] = prev & mask; + + for (i = 10; i < input.length; i++) { + var next = input.words[i] | 0; + input.words[i - 10] = ((next & mask) << 4) | (prev >>> 22); + prev = next; + } + prev >>>= 22; + input.words[i - 10] = prev; + if (prev === 0 && input.length > 10) { + input.length -= 10; + } else { + input.length -= 9; + } + }; + + K256.prototype.imulK = function imulK(num) { + // K = 0x1000003d1 = [ 0x40, 0x3d1 ] + num.words[num.length] = 0; + num.words[num.length + 1] = 0; + num.length += 2; + + // bounded at: 0x40 * 0x3ffffff + 0x3d0 = 0x100000390 + var lo = 0; + for (var i = 0; i < num.length; i++) { + var w = num.words[i] | 0; + lo += w * 0x3d1; + num.words[i] = lo & 0x3ffffff; + lo = w * 0x40 + ((lo / 0x4000000) | 0); + } + + // Fast length reduction + if (num.words[num.length - 1] === 0) { + num.length--; + if (num.words[num.length - 1] === 0) { + num.length--; + } + } + return num; + }; + + function P224() { + MPrime.call(this, 'p224', 'ffffffff ffffffff ffffffff ffffffff 00000000 00000000 00000001'); + } + inherits(P224, MPrime); - /** - * Multiplication by a constant. - * @method cmult - * @param {number} cst - the constant to multiply to this share. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.cmult = function (cst) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (*)'); - } + function P192() { + MPrime.call(this, 'p192', 'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff'); + } + inherits(P192, MPrime); - var self = this; - var ready = function () { - return self.jiff.helpers.mod(self.jiff.share_helpers['*'](self.value, cst), self.Zp); - }; + function P25519() { + // 2 ^ 255 - 19 + MPrime.call(this, '25519', '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed'); + } + inherits(P25519, MPrime); + + P25519.prototype.imulK = function imulK(num) { + // K = 0x13 + var carry = 0; + for (var i = 0; i < num.length; i++) { + var hi = (num.words[i] | 0) * 0x13 + carry; + var lo = hi & 0x3ffffff; + hi >>>= 26; + + num.words[i] = lo; + carry = hi; + } + if (carry !== 0) { + num.words[num.length++] = carry; + } + return num; + }; + + // Exported mostly for testing purposes, use plain name instead + BN._prime = function prime(name) { + // Cached version of prime + if (primes[name]) return primes[name]; + + var prime; + if (name === 'k256') { + prime = new K256(); + } else if (name === 'p224') { + prime = new P224(); + } else if (name === 'p192') { + prime = new P192(); + } else if (name === 'p25519') { + prime = new P25519(); + } else { + throw new Error('Unknown prime ' + name); + } + primes[name] = prime; + + return prime; + }; + + // + // Base reduction engine + // + function Red(m) { + if (typeof m === 'string') { + var prime = BN._prime(m); + this.m = prime.p; + this.prime = prime; + } else { + assert(m.gtn(1), 'modulus must be greater than 1'); + this.m = m; + this.prime = null; + } + } - return new this.jiff.SecretShare(this.wThen(ready), this.holders, this.threshold, this.Zp); - }; + Red.prototype._verify1 = function _verify1(a) { + assert(a.negative === 0, 'red works only with positives'); + assert(a.red, 'red works only with red numbers'); + }; + + Red.prototype._verify2 = function _verify2(a, b) { + assert((a.negative | b.negative) === 0, 'red works only with positives'); + assert(a.red && a.red === b.red, 'red works only with red numbers'); + }; + + Red.prototype.imod = function imod(a) { + if (this.prime) return this.prime.ireduce(a)._forceRed(this); + return a.umod(this.m)._forceRed(this); + }; + + Red.prototype.neg = function neg(a) { + if (a.isZero()) { + return a.clone(); + } + + return this.m.sub(a)._forceRed(this); + }; + + Red.prototype.add = function add(a, b) { + this._verify2(a, b); + + var res = a.add(b); + if (res.cmp(this.m) >= 0) { + res.isub(this.m); + } + return res._forceRed(this); + }; + + Red.prototype.iadd = function iadd(a, b) { + this._verify2(a, b); + + var res = a.iadd(b); + if (res.cmp(this.m) >= 0) { + res.isub(this.m); + } + return res; + }; + + Red.prototype.sub = function sub(a, b) { + this._verify2(a, b); + + var res = a.sub(b); + if (res.cmpn(0) < 0) { + res.iadd(this.m); + } + return res._forceRed(this); + }; + + Red.prototype.isub = function isub(a, b) { + this._verify2(a, b); + + var res = a.isub(b); + if (res.cmpn(0) < 0) { + res.iadd(this.m); + } + return res; + }; + + Red.prototype.shl = function shl(a, num) { + this._verify1(a); + return this.imod(a.ushln(num)); + }; + + Red.prototype.imul = function imul(a, b) { + this._verify2(a, b); + return this.imod(a.imul(b)); + }; + + Red.prototype.mul = function mul(a, b) { + this._verify2(a, b); + return this.imod(a.mul(b)); + }; + + Red.prototype.isqr = function isqr(a) { + return this.imul(a, a.clone()); + }; + + Red.prototype.sqr = function sqr(a) { + return this.mul(a, a); + }; + + Red.prototype.sqrt = function sqrt(a) { + if (a.isZero()) return a.clone(); + + var mod3 = this.m.andln(3); + assert(mod3 % 2 === 1); + + // Fast case + if (mod3 === 3) { + var pow = this.m.add(new BN(1)).iushrn(2); + return this.pow(a, pow); + } + + // Tonelli-Shanks algorithm (Totally unoptimized and slow) + // + // Find Q and S, that Q * 2 ^ S = (P - 1) + var q = this.m.subn(1); + var s = 0; + while (!q.isZero() && q.andln(1) === 0) { + s++; + q.iushrn(1); + } + assert(!q.isZero()); + + var one = new BN(1).toRed(this); + var nOne = one.redNeg(); + + // Find quadratic non-residue + // NOTE: Max is such because of generalized Riemann hypothesis. + var lpow = this.m.subn(1).iushrn(1); + var z = this.m.bitLength(); + z = new BN(2 * z * z).toRed(this); + + while (this.pow(z, lpow).cmp(nOne) !== 0) { + z.redIAdd(nOne); + } + + var c = this.pow(z, q); + var r = this.pow(a, q.addn(1).iushrn(1)); + var t = this.pow(a, q); + var m = s; + while (t.cmp(one) !== 0) { + var tmp = t; + for (var i = 0; tmp.cmp(one) !== 0; i++) { + tmp = tmp.redSqr(); + } + assert(i < m); + var b = this.pow(c, new BN(1).iushln(m - i - 1)); + + r = r.redMul(b); + c = b.redSqr(); + t = t.redMul(c); + m = i; + } + + return r; + }; + + Red.prototype.invm = function invm(a) { + var inv = a._invmp(this.m); + if (inv.negative !== 0) { + inv.negative = 0; + return this.imod(inv).redNeg(); + } else { + return this.imod(inv); + } + }; + + Red.prototype.pow = function pow(a, num) { + if (num.isZero()) return new BN(1).toRed(this); + if (num.cmpn(1) === 0) return a.clone(); + + var windowSize = 4; + var wnd = new Array(1 << windowSize); + wnd[0] = new BN(1).toRed(this); + wnd[1] = a; + for (var i = 2; i < wnd.length; i++) { + wnd[i] = this.mul(wnd[i - 1], a); + } + + var res = wnd[0]; + var current = 0; + var currentLen = 0; + var start = num.bitLength() % 26; + if (start === 0) { + start = 26; + } + + for (i = num.length - 1; i >= 0; i--) { + var word = num.words[i]; + for (var j = start - 1; j >= 0; j--) { + var bit = (word >> j) & 1; + if (res !== wnd[0]) { + res = this.sqr(res); + } + + if (bit === 0 && current === 0) { + currentLen = 0; + continue; + } + + current <<= 1; + current |= bit; + currentLen++; + if (currentLen !== windowSize && (i !== 0 || j !== 0)) continue; + + res = this.mul(res, wnd[current]); + currentLen = 0; + current = 0; + } + start = 26; + } + + return res; + }; + + Red.prototype.convertTo = function convertTo(num) { + var r = num.umod(this.m); + + return r === num ? r.clone() : r; + }; + + Red.prototype.convertFrom = function convertFrom(num) { + var res = num.clone(); + res.red = null; + return res; + }; + + // + // Montgomery method engine + // + + BN.mont = function mont(num) { + return new Mont(num); + }; + + function Mont(m) { + Red.call(this, m); + + this.shift = this.m.bitLength(); + if (this.shift % 26 !== 0) { + this.shift += 26 - (this.shift % 26); + } + + this.r = new BN(1).iushln(this.shift); + this.r2 = this.imod(this.r.sqr()); + this.rinv = this.r._invmp(this.m); + + this.minv = this.rinv.mul(this.r).isubn(1).div(this.m); + this.minv = this.minv.umod(this.r); + this.minv = this.r.sub(this.minv); + } + inherits(Mont, Red); + + Mont.prototype.convertTo = function convertTo(num) { + return this.imod(num.ushln(this.shift)); + }; + + Mont.prototype.convertFrom = function convertFrom(num) { + var r = this.imod(num.mul(this.rinv)); + r.red = null; + return r; + }; + + Mont.prototype.imul = function imul(a, b) { + if (a.isZero() || b.isZero()) { + a.words[0] = 0; + a.length = 1; + return a; + } + + var t = a.imul(b); + var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m); + var u = t.isub(c).iushrn(this.shift); + var res = u; + + if (u.cmp(this.m) >= 0) { + res = u.isub(this.m); + } else if (u.cmpn(0) < 0) { + res = u.iadd(this.m); + } + + return res._forceRed(this); + }; + + Mont.prototype.mul = function mul(a, b) { + if (a.isZero() || b.isZero()) return new BN(0)._forceRed(this); + + var t = a.mul(b); + var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m); + var u = t.isub(c).iushrn(this.shift); + var res = u; + if (u.cmp(this.m) >= 0) { + res = u.isub(this.m); + } else if (u.cmpn(0) < 0) { + res = u.iadd(this.m); + } + + return res._forceRed(this); + }; + + Mont.prototype.invm = function invm(a) { + // (AR)^-1 * R^2 = (A^-1 * R^-1) * R^2 = A^-1 * R + var res = this.imod(a._invmp(this.m).mul(this.r2)); + return res._forceRed(this); + }; + })(typeof module === 'undefined' || module, this); + }, + { buffer: 77 } + ], + 71: [ + function (require, module, exports) { + /** + * Expose `Backoff`. + */ + + module.exports = Backoff; + + /** + * Initialize backoff timer with `opts`. + * + * - `min` initial timeout in milliseconds [100] + * - `max` max timeout [10000] + * - `jitter` [0] + * - `factor` [2] + * + * @param {Object} opts + * @api public + */ + + function Backoff(opts) { + opts = opts || {}; + this.ms = opts.min || 100; + this.max = opts.max || 10000; + this.factor = opts.factor || 2; + this.jitter = opts.jitter > 0 && opts.jitter <= 1 ? opts.jitter : 0; + this.attempts = 0; + } - /** - * Division by a constant factor of the number represented by the share. - * @method cdivfac - * @param {number} cst - the constant by which to divide the share. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.cdivfac = function (cst) { - if (!(this.isConstant(cst))) { - throw new Error('Parameter should be a number (cdivfac)'); - } + /** + * Return the backoff duration. + * + * @return {Number} + * @api public + */ + + Backoff.prototype.duration = function () { + var ms = this.ms * Math.pow(this.factor, this.attempts++); + if (this.jitter) { + var rand = Math.random(); + var deviation = Math.floor(rand * this.jitter * ms); + ms = (Math.floor(rand * 10) & 1) == 0 ? ms - deviation : ms + deviation; + } + return Math.min(ms, this.max) | 0; + }; + + /** + * Reset the number of attempts. + * + * @api public + */ + + Backoff.prototype.reset = function () { + this.attempts = 0; + }; + + /** + * Set the minimum duration + * + * @api public + */ + + Backoff.prototype.setMin = function (min) { + this.ms = min; + }; + + /** + * Set the maximum duration + * + * @api public + */ + + Backoff.prototype.setMax = function (max) { + this.max = max; + }; + + /** + * Set the jitter + * + * @api public + */ + + Backoff.prototype.setJitter = function (jitter) { + this.jitter = jitter; + }; + }, + {} + ], + 72: [ + function (require, module, exports) { + /* + * base64-arraybuffer + * https://github.com/niklasvh/base64-arraybuffer + * + * Copyright (c) 2012 Niklas von Hertzen + * Licensed under the MIT license. + */ + (function () { + 'use strict'; + + var chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'; + + // Use a lookup table to find the index. + var lookup = new Uint8Array(256); + for (var i = 0; i < chars.length; i++) { + lookup[chars.charCodeAt(i)] = i; + } - var inv = this.jiff.helpers.extended_gcd(cst, this.Zp)[0]; + exports.encode = function (arraybuffer) { + var bytes = new Uint8Array(arraybuffer), + i, + len = bytes.length, + base64 = ''; + + for (i = 0; i < len; i += 3) { + base64 += chars[bytes[i] >> 2]; + base64 += chars[((bytes[i] & 3) << 4) | (bytes[i + 1] >> 4)]; + base64 += chars[((bytes[i + 1] & 15) << 2) | (bytes[i + 2] >> 6)]; + base64 += chars[bytes[i + 2] & 63]; + } + + if (len % 3 === 2) { + base64 = base64.substring(0, base64.length - 1) + '='; + } else if (len % 3 === 1) { + base64 = base64.substring(0, base64.length - 2) + '=='; + } + + return base64; + }; + + exports.decode = function (base64) { + var bufferLength = base64.length * 0.75, + len = base64.length, + i, + p = 0, + encoded1, + encoded2, + encoded3, + encoded4; + + if (base64[base64.length - 1] === '=') { + bufferLength--; + if (base64[base64.length - 2] === '=') { + bufferLength--; + } + } + + var arraybuffer = new ArrayBuffer(bufferLength), + bytes = new Uint8Array(arraybuffer); + + for (i = 0; i < len; i += 4) { + encoded1 = lookup[base64.charCodeAt(i)]; + encoded2 = lookup[base64.charCodeAt(i + 1)]; + encoded3 = lookup[base64.charCodeAt(i + 2)]; + encoded4 = lookup[base64.charCodeAt(i + 3)]; + + bytes[p++] = (encoded1 << 2) | (encoded2 >> 4); + bytes[p++] = ((encoded2 & 15) << 4) | (encoded3 >> 2); + bytes[p++] = ((encoded3 & 3) << 6) | (encoded4 & 63); + } + + return arraybuffer; + }; + })(); + }, + {} + ], + 73: [ + function (require, module, exports) { + 'use strict'; + + exports.byteLength = byteLength; + exports.toByteArray = toByteArray; + exports.fromByteArray = fromByteArray; + + var lookup = []; + var revLookup = []; + var Arr = typeof Uint8Array !== 'undefined' ? Uint8Array : Array; + + var code = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'; + for (var i = 0, len = code.length; i < len; ++i) { + lookup[i] = code[i]; + revLookup[code.charCodeAt(i)] = i; + } - var self = this; - var ready = function () { - return self.jiff.helpers.mod(self.jiff.share_helpers['*'](self.value, inv), self.Zp); - }; + // Support decoding URL-safe base64 strings, as Node.js does. + // See: https://en.wikipedia.org/wiki/Base64#URL_applications + revLookup['-'.charCodeAt(0)] = 62; + revLookup['_'.charCodeAt(0)] = 63; - return new this.jiff.SecretShare(this.wThen(ready), this.holders, this.threshold, this.Zp); - }; + function getLens(b64) { + var len = b64.length; - /** - * Addition of two secret shares. - * @method sadd - * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to add to this share. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - * - * @example - * // share a value with all parties, and sum the values of all shares - * var shares = jiff_instance.share(x); - * var sum = shares[1]; - * for (var i = 2; i <= jiff_instance.party_count; i++) { - * sum = sum.sadd(shares[i]); - * } - * - */ - SecretShare.prototype.sadd = function (o) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (+)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (+)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (+)'); - } + if (len % 4 > 0) { + throw new Error('Invalid string. Length must be a multiple of 4'); + } - // add the two shares when ready locally - var self = this; - var ready = function () { - return self.jiff.helpers.mod(self.jiff.share_helpers['+'](self.value, o.value), self.Zp); - }; + // Trim off extra bytes after placeholder bytes are found + // See: https://github.com/beatgammit/base64-js/issues/42 + var validLen = b64.indexOf('='); + if (validLen === -1) validLen = len; - // promise to execute ready_add when both are ready - return new this.jiff.SecretShare(this.when_both_ready(o, ready), this.holders, Math.max(this.threshold, o.threshold), this.Zp); - }; + var placeHoldersLen = validLen === len ? 0 : 4 - (validLen % 4); - /** - * Subtraction of two secret shares. - * @method ssub - * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to subtract from this share. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.ssub = function (o) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (-)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (-)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (-)'); - } + return [validLen, placeHoldersLen]; + } - // subtract the two shares when ready locally - var self = this; - var ready = function () { - return self.jiff.helpers.mod(self.jiff.share_helpers['-'](self.value, o.value), self.Zp); - }; + // base64 is 4/3 + up to two characters of the original data + function byteLength(b64) { + var lens = getLens(b64); + var validLen = lens[0]; + var placeHoldersLen = lens[1]; + return ((validLen + placeHoldersLen) * 3) / 4 - placeHoldersLen; + } - // promise to execute ready_add when both are ready - return new this.jiff.SecretShare(this.when_both_ready(o, ready), this.holders, Math.max(this.threshold, o.threshold), this.Zp); - }; + function _byteLength(b64, validLen, placeHoldersLen) { + return ((validLen + placeHoldersLen) * 3) / 4 - placeHoldersLen; + } - /** - * Multiplication of two secret shares through Beaver Triplets. - * @method smult - * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to multiply with. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this multiplication (and internally, the corresponding beaver triplet). - * This id must be unique, and must be passed by all parties to the same instruction. - * this ensures that every party gets a share from the same triplet for every matching instruction. An automatic id - * is generated by increasing a local counter, default ids suffice when all parties execute the - * instructions in the same order. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.smult = function (o, op_id) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (*)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (*)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (*)'); - } + function toByteArray(b64) { + var tmp; + var lens = getLens(b64); + var validLen = lens[0]; + var placeHoldersLen = lens[1]; - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('smult', this.holders); - } + var arr = new Arr(_byteLength(b64, validLen, placeHoldersLen)); - // final result - var final_deferred = new this.jiff.helpers.Deferred(); - var final_promise = final_deferred.promise; - var result = new this.jiff.SecretShare(final_promise, this.holders, Math.max(this.threshold, o.threshold), this.Zp); - - // called when triplet is ready - var self = this; - var ready_triplet = function (triplet) { - var a = triplet[0]; - var b = triplet[1]; - var c = triplet[2]; - - // d = s - a. e = o - b. - var d = self.isadd(a.icmult(-1)); - var e = o.isadd(b.icmult(-1)); - - // Open d and e. - // The only communication cost. - var e_promise = self.jiff.internal_open(e, e.holders, op_id + ':open1'); - var d_promise = self.jiff.internal_open(d, d.holders, op_id + ':open2'); - Promise.all([e_promise, d_promise]).then(function (arr) { - var e_open = arr[0]; - var d_open = arr[1]; - - // result_share = d_open * e_open + d_open * b_share + e_open * a_share + c. - var t1 = self.jiff.helpers.mod(self.jiff.share_helpers['*'](d_open, e_open), self.Zp); - var t2 = b.icmult(d_open); - var t3 = a.icmult(e_open); - - // All this happens locally. - var final_result = t2.icadd(t1); - final_result = final_result.isadd(t3); - final_result = final_result.isadd(c); - - final_result.wThen(final_deferred.resolve); - }); - }; - - // Get shares of triplets. - var triplet = this.jiff.get_preprocessing(op_id + ':triplet'); - if (triplet == null) { - var promise = this.jiff.from_crypto_provider('triplet', this.holders, Math.max(this.threshold, o.threshold), this.Zp, op_id + ':triplet'); - promise.then(function (msg) { - ready_triplet(msg['shares']); - }); - } else { - ready_triplet(triplet); - } + var curByte = 0; - return result; - }; + // if there are placeholders, only get up to the last complete 4 chars + var len = placeHoldersLen > 0 ? validLen - 4 : validLen; - /** - * Multiplication of two secret shares through BGW protocol. - * @method smult_bgw - * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to multiply with. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this multiplication (and internally, the corresponding beaver triplet). - * This id must be unique, and must be passed by all parties to the same instruction. - * this ensures that every party gets a share from the same triplet for every matching instruction. An automatic id - * is generated by increasing a local counter, default ids suffice when all parties execute the - * instructions in the same order. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.smult_bgw = function (o, op_id) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (bgw*)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (bgw*)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (bgw*)'); - } - if ((this.threshold - 1) + (o.threshold - 1) > this.holders.length - 1) { - throw new Error('threshold too high for BGW (*)'); - } + var i; + for (i = 0; i < len; i += 4) { + tmp = (revLookup[b64.charCodeAt(i)] << 18) | (revLookup[b64.charCodeAt(i + 1)] << 12) | (revLookup[b64.charCodeAt(i + 2)] << 6) | revLookup[b64.charCodeAt(i + 3)]; + arr[curByte++] = (tmp >> 16) & 0xff; + arr[curByte++] = (tmp >> 8) & 0xff; + arr[curByte++] = tmp & 0xff; + } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('smult_bgw', this.holders); - } + if (placeHoldersLen === 2) { + tmp = (revLookup[b64.charCodeAt(i)] << 2) | (revLookup[b64.charCodeAt(i + 1)] >> 4); + arr[curByte++] = tmp & 0xff; + } - // ensure thresholds are fine - var new_threshold = (this.threshold - 1) + (o.threshold - 1) + 1; - if (new_threshold > this.holders) { - var errorMsg = 'Threshold too large for smult_bgw: ' + new_threshold; - errorMsg += '. Shares: ' + this.toString() + ', ' + o.toString(); - throw new Error(errorMsg); - } + if (placeHoldersLen === 1) { + tmp = (revLookup[b64.charCodeAt(i)] << 10) | (revLookup[b64.charCodeAt(i + 1)] << 4) | (revLookup[b64.charCodeAt(i + 2)] >> 2); + arr[curByte++] = (tmp >> 8) & 0xff; + arr[curByte++] = tmp & 0xff; + } - // multiply via the BGW protocol - var self = this; - var ready = function () { - return self.jiff.helpers.mod(self.jiff.share_helpers['*'](self.value, o.value), self.Zp); - }; + return arr; + } - // reshare to reduce threshold and return when ready - var result = new this.jiff.SecretShare(this.when_both_ready(o, ready), this.holders, new_threshold, this.Zp); - return this.jiff.reshare(result, Math.max(this.threshold, o.threshold), result.holders, result.holders, result.Zp, op_id + ':threshold'); - }; + function tripletToBase64(num) { + return lookup[(num >> 18) & 0x3f] + lookup[(num >> 12) & 0x3f] + lookup[(num >> 6) & 0x3f] + lookup[num & 0x3f]; + } - /** - * Integer divison with two shares (this / o) - * @method sdiv - * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to divide by. - * @param {number} [l=ceil(log_2(this.Zp))] - the maximum bit length of either operands. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.sdiv = function (o, l, op_id) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (!=)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (!=)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (!=)'); - } + function encodeChunk(uint8, start, end) { + var tmp; + var output = []; + for (var i = start; i < end; i += 3) { + tmp = ((uint8[i] << 16) & 0xff0000) + ((uint8[i + 1] << 8) & 0xff00) + (uint8[i + 2] & 0xff); + output.push(tripletToBase64(tmp)); + } + return output.join(''); + } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('sdiv', this.holders); - } + function fromByteArray(uint8) { + var tmp; + var len = uint8.length; + var extraBytes = len % 3; // if we have 1 byte left, pad 2 bytes + var parts = []; + var maxChunkLength = 16383; // must be multiple of 3 - // figure out maximum output bit length - var lZp = this.jiff.share_helpers['ceil'](this.jiff.helpers.bLog(this.Zp, 2)); - l = (l != null && l < lZp) ? l : lZp; + // go through the array every three bytes, we'll deal with trailing stuff later + for (var i = 0, len2 = len - extraBytes; i < len2; i += maxChunkLength) { + parts.push(encodeChunk(uint8, i, i + maxChunkLength > len2 ? len2 : i + maxChunkLength)); + } - // Convert to bits - var dividend_bits = this.bit_decomposition(op_id + ':decomposition1').slice(0, l); - var divisor_bits = o.bit_decomposition(op_id + ':decomposition2').slice(0, l); + // pad the end with zeros, but make sure to not forget the extra bytes + if (extraBytes === 1) { + tmp = uint8[len - 1]; + parts.push(lookup[tmp >> 2] + lookup[(tmp << 4) & 0x3f] + '=='); + } else if (extraBytes === 2) { + tmp = (uint8[len - 2] << 8) + uint8[len - 1]; + parts.push(lookup[tmp >> 10] + lookup[(tmp >> 4) & 0x3f] + lookup[(tmp << 2) & 0x3f] + '='); + } - // Compute by long division - var quotient_bits = this.jiff.protocols.bits.sdiv(dividend_bits, divisor_bits, op_id + ':bits.sdiv').quotient; + return parts.join(''); + } + }, + {} + ], + 74: [ + function (require, module, exports) { + /** + * Create a blob builder even when vendor prefixes exist + */ + + var BlobBuilder = + typeof BlobBuilder !== 'undefined' + ? BlobBuilder + : typeof WebKitBlobBuilder !== 'undefined' + ? WebKitBlobBuilder + : typeof MSBlobBuilder !== 'undefined' + ? MSBlobBuilder + : typeof MozBlobBuilder !== 'undefined' + ? MozBlobBuilder + : false; + + /** + * Check if Blob constructor is supported + */ + + var blobSupported = (function () { + try { + var a = new Blob(['hi']); + return a.size === 2; + } catch (e) { + return false; + } + })(); + + /** + * Check if Blob constructor supports ArrayBufferViews + * Fails in Safari 6, so we need to map to ArrayBuffers there. + */ + + var blobSupportsArrayBufferView = + blobSupported && + (function () { + try { + var b = new Blob([new Uint8Array([1, 2])]); + return b.size === 2; + } catch (e) { + return false; + } + })(); + + /** + * Check if BlobBuilder is supported + */ + + var blobBuilderSupported = BlobBuilder && BlobBuilder.prototype.append && BlobBuilder.prototype.getBlob; + + /** + * Helper function that maps ArrayBufferViews to ArrayBuffers + * Used by BlobBuilder constructor and old browsers that didn't + * support it in the Blob constructor. + */ + + function mapArrayBufferViews(ary) { + return ary.map(function (chunk) { + if (chunk.buffer instanceof ArrayBuffer) { + var buf = chunk.buffer; + + // if this is a subarray, make a copy so we only + // include the subarray region from the underlying buffer + if (chunk.byteLength !== buf.byteLength) { + var copy = new Uint8Array(chunk.byteLength); + copy.set(new Uint8Array(buf, chunk.byteOffset, chunk.byteLength)); + buf = copy.buffer; + } + + return buf; + } + + return chunk; + }); + } - // Convert to number and return - return this.jiff.protocols.bits.bit_composition(quotient_bits); - }; + function BlobBuilderConstructor(ary, options) { + options = options || {}; - /** - * Integer divison with a share and a constant (this / cst). - * @method cdiv - * @param {module:jiff-client~JIFFClient#SecretShare} cst - the constant to divide by. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.cdiv = function (cst, op_id) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (/)'); - } + var bb = new BlobBuilder(); + mapArrayBufferViews(ary).forEach(function (part) { + bb.append(part); + }); - if (this.jiff.share_helpers['<='](cst, 0)) { - throw new Error('divisor must be > 0 (cst/): ' + cst); - } + return options.type ? bb.getBlob(options.type) : bb.getBlob(); + } - if (this.jiff.share_helpers['<='](this.Zp, cst)) { - throw new Error('divisor must be < share.Zp (' + this.Zp + ') in (cst/): ' + cst); - } + function BlobConstructor(ary, options) { + return new Blob(mapArrayBufferViews(ary), options || {}); + } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('cdiv', this.holders); - } + if (typeof Blob !== 'undefined') { + BlobBuilderConstructor.prototype = Blob.prototype; + BlobConstructor.prototype = Blob.prototype; + } - // Allocate share for result to which the answer will be resolved once available - var final_deferred = new this.jiff.helpers.Deferred(); - var final_promise = final_deferred.promise; - var result = new this.jiff.SecretShare(final_promise, this.holders, this.threshold, this.Zp); - - // Execute protocol when random in noise in [0, Zp) and quotient floor(noise/constant) is ready! - var self = this; - var ready_quotient = function (noise, nOVERc) { - // Use noise - var noisyX = self.isadd(noise); - self.jiff.internal_open(noisyX, noisyX.holders, op_id + ':open').then(function (noisyX) { - var wrapped = self.icgt(noisyX, op_id + ':wrap_cgt'); // 1 => x + noise wrapped around Zp, 0 otherwise - - // if we did not wrap - var noWrapDiv = self.jiff.share_helpers['floor/'](noisyX, cst); - var unCorrectedQuotient = nOVERc.icmult(-1).icadd(noWrapDiv).icsub(1); - var verify = self.issub(unCorrectedQuotient.icmult(cst)); - var isNotCorrect = verify.icgteq(cst, op_id + ':cor1'); - var noWrapAnswer = unCorrectedQuotient.isadd(isNotCorrect); // if incorrect => isNotCorrect = 1 => quotient = unCorrectedQuotient - 1 - - // if we wrapped - var wrapDiv = self.jiff.share_helpers['floor/'](self.jiff.share_helpers['+'](noisyX, self.Zp), cst); - unCorrectedQuotient = nOVERc.icmult(-1).icadd(wrapDiv).icsub(1); - verify = self.issub(unCorrectedQuotient.icmult(cst)); - isNotCorrect = verify.icgteq(cst, op_id + ':cor2'); - var wrapAnswer = unCorrectedQuotient.isadd(isNotCorrect); // if incorrect => isNotCorrect = 1 => quotient = unCorrectedQuotient - 1 - - var answer = noWrapAnswer.isadd(wrapped.ismult(wrapAnswer.issub(noWrapAnswer), op_id + ':smult')); - answer.wThen(final_deferred.resolve); - }); - }; - - // Preprocessing cases - var quotient = this.jiff.get_preprocessing(op_id + ':quotient'); - if (quotient == null) { // case 1: no preprocessing with crypto provider! - var promise = this.jiff.from_crypto_provider('quotient', this.holders, this.threshold, this.Zp, op_id + ':quotient', {constant: cst}); - promise.then(function (msg) { - ready_quotient(msg['shares'][0], msg['shares'][1]); - }); - } else if (quotient.ondemand === true) { // case 2: constant was not available at preprocessing time, must do it now! - this.jiff.preprocessing('quotient', 1, null, this.threshold, this.holders, this.holders, this.Zp, [op_id + ':quotient'], {constant: cst, namespace: 'base'}); - this.jiff.executePreprocessing(function () { - var quotient = self.jiff.get_preprocessing(op_id + ':quotient'); - ready_quotient(quotient.r, quotient.q); - }); - } else { // case 3: preprocessing is completed! - ready_quotient(quotient.r, quotient.q); - } + module.exports = (function () { + if (blobSupported) { + return blobSupportsArrayBufferView ? Blob : BlobConstructor; + } else if (blobBuilderSupported) { + return BlobBuilderConstructor; + } else { + return undefined; + } + })(); + }, + {} + ], + 75: [ + function (require, module, exports) { + (function (module, exports) { + 'use strict'; + + // Utils + function assert(val, msg) { + if (!val) throw new Error(msg || 'Assertion failed'); + } - // special case, if result is zero, sometimes we will get to -1 due to how correction happens above (.csub(1) and then compare) - var zeroIt = this.iclt(cst, op_id + ':zero_check').inot(); - return result.ismult(zeroIt, op_id + ':zero_it'); - }; + // Could use `inherits` module, but don't want to move from single file + // architecture yet. + function inherits(ctor, superCtor) { + ctor.super_ = superCtor; + var TempCtor = function () {}; + TempCtor.prototype = superCtor.prototype; + ctor.prototype = new TempCtor(); + ctor.prototype.constructor = ctor; + } - /** - * Remainder with two shares (this % o) - * @method smod - * @param {module:jiff-client~JIFFClient#SecretShare} o - the modulus to apply - * @param {number} [l=ceil(log_2(this.Zp))] - the maximum bit length of either operands. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.smod = function (o, l, op_id) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (!=)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (!=)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (!=)'); - } + // BN - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('smod', this.holders); - } + function BN(number, base, endian) { + if (BN.isBN(number)) { + return number; + } - // figure out maximum output bit length - var lZp = this.jiff.share_helpers['ceil'](this.jiff.helpers.bLog(this.Zp, 2)); - l = (l != null && l < lZp) ? l : lZp; + this.negative = 0; + this.words = null; + this.length = 0; - // Convert to bits - var dividend_bits = this.bit_decomposition(op_id + ':decomposition1').slice(0, l); - var divisor_bits = o.bit_decomposition(op_id + ':decomposition2').slice(0, l); + // Reduction context + this.red = null; - // Compute by long division - var remainder_bits = this.jiff.protocols.bits.sdiv(dividend_bits, divisor_bits, op_id + ':bits.sdiv').remainder; + if (number !== null) { + if (base === 'le' || base === 'be') { + endian = base; + base = 10; + } - // Convert to number and return - return this.jiff.protocols.bits.bit_composition(remainder_bits); - }; + this._init(number || 0, base || 10, endian || 'be'); + } + } + if (typeof module === 'object') { + module.exports = BN; + } else { + exports.BN = BN; + } - /** - * Fast (modular) exponentiation with constant exponent via repeated squaring. - * @method cpow - * @param {number} cst - the constant to multiply to this share. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.cpow = function (cst, op_id) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (*)'); - } + BN.BN = BN; + BN.wordSize = 26; + + var Buffer; + try { + Buffer = require('buffer').Buffer; + } catch (e) {} + + BN.isBN = function isBN(num) { + if (num instanceof BN) { + return true; + } + + return num !== null && typeof num === 'object' && num.constructor.wordSize === BN.wordSize && Array.isArray(num.words); + }; + + BN.max = function max(left, right) { + if (left.cmp(right) > 0) return left; + return right; + }; + + BN.min = function min(left, right) { + if (left.cmp(right) < 0) return left; + return right; + }; + + BN.prototype._init = function init(number, base, endian) { + if (typeof number === 'number') { + return this._initNumber(number, base, endian); + } + + if (typeof number === 'object') { + return this._initArray(number, base, endian); + } + + if (base === 'hex') { + base = 16; + } + assert(base === (base | 0) && base >= 2 && base <= 36); + + number = number.toString().replace(/\s+/g, ''); + var start = 0; + if (number[0] === '-') { + start++; + } + + if (base === 16) { + this._parseHex(number, start); + } else { + this._parseBase(number, base, start); + } + + if (number[0] === '-') { + this.negative = 1; + } + + this._strip(); + + if (endian !== 'le') return; + + this._initArray(this.toArray(), base, endian); + }; + + BN.prototype._initNumber = function _initNumber(number, base, endian) { + if (number < 0) { + this.negative = 1; + number = -number; + } + if (number < 0x4000000) { + this.words = [number & 0x3ffffff]; + this.length = 1; + } else if (number < 0x10000000000000) { + this.words = [number & 0x3ffffff, (number / 0x4000000) & 0x3ffffff]; + this.length = 2; + } else { + assert(number < 0x20000000000000); // 2 ^ 53 (unsafe) + this.words = [number & 0x3ffffff, (number / 0x4000000) & 0x3ffffff, 1]; + this.length = 3; + } + + if (endian !== 'le') return; + + // Reverse the bytes + this._initArray(this.toArray(), base, endian); + }; + + BN.prototype._initArray = function _initArray(number, base, endian) { + // Perhaps a Uint8Array + assert(typeof number.length === 'number'); + if (number.length <= 0) { + this.words = [0]; + this.length = 1; + return this; + } + + this.length = Math.ceil(number.length / 3); + this.words = new Array(this.length); + for (var i = 0; i < this.length; i++) { + this.words[i] = 0; + } + + var j, w; + var off = 0; + if (endian === 'be') { + for (i = number.length - 1, j = 0; i >= 0; i -= 3) { + w = number[i] | (number[i - 1] << 8) | (number[i - 2] << 16); + this.words[j] |= (w << off) & 0x3ffffff; + this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff; + off += 24; + if (off >= 26) { + off -= 26; + j++; + } + } + } else if (endian === 'le') { + for (i = 0, j = 0; i < number.length; i += 3) { + w = number[i] | (number[i + 1] << 8) | (number[i + 2] << 16); + this.words[j] |= (w << off) & 0x3ffffff; + this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff; + off += 24; + if (off >= 26) { + off -= 26; + j++; + } + } + } + return this._strip(); + }; + + function parseHex(str, start, end) { + var r = 0; + var len = Math.min(str.length, end); + var z = 0; + for (var i = start; i < len; i++) { + var c = str.charCodeAt(i) - 48; + + r <<= 4; + + var b; + + // 'a' - 'f' + if (c >= 49 && c <= 54) { + b = c - 49 + 0xa; + + // 'A' - 'F' + } else if (c >= 17 && c <= 22) { + b = c - 17 + 0xa; + + // '0' - '9' + } else { + b = c; + } + + r |= b; + z |= b; + } + + assert(!(z & 0xf0), 'Invalid character in ' + str); + return r; + } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('cpow', this.holders); - } + BN.prototype._parseHex = function _parseHex(number, start) { + // Create possibly bigger array to ensure that it fits the number + this.length = Math.ceil((number.length - start) / 6); + this.words = new Array(this.length); + for (var i = 0; i < this.length; i++) { + this.words[i] = 0; + } + + var j, w; + // Scan 24-bit chunks and add them to the number + var off = 0; + for (i = number.length - 6, j = 0; i >= start; i -= 6) { + w = parseHex(number, i, i + 6); + this.words[j] |= (w << off) & 0x3ffffff; + // NOTE: `0x3fffff` is intentional here, 26bits max shift + 24bit hex limb + this.words[j + 1] |= (w >>> (26 - off)) & 0x3fffff; + off += 24; + if (off >= 26) { + off -= 26; + j++; + } + } + if (i + 6 !== start) { + w = parseHex(number, start, i + 6); + this.words[j] |= (w << off) & 0x3ffffff; + this.words[j + 1] |= (w >>> (26 - off)) & 0x3fffff; + } + this._strip(); + }; + + function parseBase(str, start, end, mul) { + var r = 0; + var b = 0; + var len = Math.min(str.length, end); + for (var i = start; i < len; i++) { + var c = str.charCodeAt(i) - 48; + + r *= mul; + + // 'a' + if (c >= 49) { + b = c - 49 + 0xa; + + // 'A' + } else if (c >= 17) { + b = c - 17 + 0xa; + + // '0' - '9' + } else { + b = c; + } + assert(c >= 0 && b < mul, 'Invalid character'); + r += b; + } + return r; + } - // handle big numbers - var one = 1; - if (this.jiff.has_extension('bignumber')) { - one = this.jiff.helpers.BigNumber(1); - cst = this.jiff.helpers.BigNumber(cst); - } + BN.prototype._parseBase = function _parseBase(number, base, start) { + // Initialize as zero + this.words = [0]; + this.length = 1; + + // Find length of limb in base + for (var limbLen = 0, limbPow = 1; limbPow <= 0x3ffffff; limbPow *= base) { + limbLen++; + } + limbLen--; + limbPow = (limbPow / base) | 0; + + var total = number.length - start; + var mod = total % limbLen; + var end = Math.min(total, total - mod) + start; + + var word = 0; + for (var i = start; i < end; i += limbLen) { + word = parseBase(number, i, i + limbLen, base); + + this.imuln(limbPow); + if (this.words[0] + word < 0x4000000) { + this.words[0] += word; + } else { + this._iaddn(word); + } + } + + if (mod !== 0) { + var pow = 1; + word = parseBase(number, i, number.length, base); + + for (i = 0; i < mod; i++) { + pow *= base; + } + + this.imuln(pow); + if (this.words[0] + word < 0x4000000) { + this.words[0] += word; + } else { + this._iaddn(word); + } + } + }; + + BN.prototype.copy = function copy(dest) { + dest.words = new Array(this.length); + for (var i = 0; i < this.length; i++) { + dest.words[i] = this.words[i]; + } + dest.length = this.length; + dest.negative = this.negative; + dest.red = this.red; + }; + + function move(dest, src) { + dest.words = src.words; + dest.length = src.length; + dest.negative = src.negative; + dest.red = src.red; + } - // ensure exponent is non-negative - if (this.jiff.share_helpers['<'](cst, 0)) { - throw new Error('cpow supports non-negative exponents only, given ' + cst.toString()); - } + BN.prototype._move = function _move(dest) { + move(dest, this); + }; + + BN.prototype.clone = function clone() { + var r = new BN(null); + this.copy(r); + return r; + }; + + BN.prototype._expand = function _expand(size) { + while (this.length < size) { + this.words[this.length++] = 0; + } + return this; + }; + + // Remove leading `0` from `this` + BN.prototype._strip = function strip() { + while (this.length > 1 && this.words[this.length - 1] === 0) { + this.length--; + } + return this._normSign(); + }; + + BN.prototype._normSign = function _normSign() { + // -0 = 0 + if (this.length === 1 && this.words[0] === 0) { + this.negative = 0; + } + return this; + }; + + // Check Symbol.for because not everywhere where Symbol defined + // See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Symbol#Browser_compatibility + if (typeof Symbol !== 'undefined' && typeof Symbol.for === 'function') { + BN.prototype[Symbol.for('nodejs.util.inspect.custom')] = inspect; + } else { + BN.prototype.inspect = inspect; + } - // begin protocol - var evens = this; - var odds = new this.jiff.SecretShare(one, this.holders, this.threshold, this.Zp); + function inspect() { + return (this.red ? ''; + } - // special case - if (cst.toString() === '0') { - return odds; - } + /* - for (var i = 0; this.jiff.share_helpers['<'](1, cst); i++) { - if (this.jiff.share_helpers['even'](cst)) { - evens = evens.ismult(evens, op_id + ':smult0:'+i); - cst = this.jiff.share_helpers['/'](cst, 2); - } else { - odds = evens.ismult(odds, op_id + ':smult0:'+i); - evens = evens.ismult(evens, op_id + ':smult1:'+i); - cst = this.jiff.share_helpers['/'](this.jiff.share_helpers['-'](cst, 1), 2); - } - } + var zeros = []; + var groupSizes = []; + var groupBases = []; - return evens.ismult(odds, op_id + ':smult0:' + i); - }; -}; -},{}],32:[function(require,module,exports){ -// Comparison operations on shares -module.exports = function (SecretShare) { - /** - * Greater than or equal with another share. - * @method sgteq - * @param {module:jiff-client~JIFFClient#SecretShare} o - the other share. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this >= o, and 0 otherwise. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.sgteq = function (o, op_id) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (>=)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (>=)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (>=)'); + var s = ''; + var i = -1; + while (++i < BN.wordSize) { + zeros[i] = s; + s += '0'; + } + groupSizes[0] = 0; + groupSizes[1] = 0; + groupBases[0] = 0; + groupBases[1] = 0; + var base = 2 - 1; + while (++base < 36 + 1) { + var groupSize = 0; + var groupBase = 1; + while (groupBase < (1 << BN.wordSize) / base) { + groupBase *= base; + groupSize += 1; } + groupSizes[base] = groupSize; + groupBases[base] = groupBase; + } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('sgteq', this.holders); - } + */ - return this.islt(o, op_id).inot(); - }; + var zeros = [ + '', + '0', + '00', + '000', + '0000', + '00000', + '000000', + '0000000', + '00000000', + '000000000', + '0000000000', + '00000000000', + '000000000000', + '0000000000000', + '00000000000000', + '000000000000000', + '0000000000000000', + '00000000000000000', + '000000000000000000', + '0000000000000000000', + '00000000000000000000', + '000000000000000000000', + '0000000000000000000000', + '00000000000000000000000', + '000000000000000000000000', + '0000000000000000000000000' + ]; + + var groupSizes = [0, 0, 25, 16, 12, 11, 10, 9, 8, 8, 7, 7, 7, 7, 6, 6, 6, 6, 6, 6, 6, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5]; + + var groupBases = [ + 0, 0, 33554432, 43046721, 16777216, 48828125, 60466176, 40353607, 16777216, 43046721, 10000000, 19487171, 35831808, 62748517, 7529536, 11390625, 16777216, 24137569, 34012224, 47045881, + 64000000, 4084101, 5153632, 6436343, 7962624, 9765625, 11881376, 14348907, 17210368, 20511149, 24300000, 28629151, 33554432, 39135393, 45435424, 52521875, 60466176 + ]; + + BN.prototype.toString = function toString(base, padding) { + base = base || 10; + padding = padding | 0 || 1; + + var out; + if (base === 16 || base === 'hex') { + out = ''; + var off = 0; + var carry = 0; + for (var i = 0; i < this.length; i++) { + var w = this.words[i]; + var word = (((w << off) | carry) & 0xffffff).toString(16); + carry = (w >>> (24 - off)) & 0xffffff; + if (carry !== 0 || i !== this.length - 1) { + out = zeros[6 - word.length] + word + out; + } else { + out = word + out; + } + off += 2; + if (off >= 26) { + off -= 26; + i--; + } + } + if (carry !== 0) { + out = carry.toString(16) + out; + } + while (out.length % padding !== 0) { + out = '0' + out; + } + if (this.negative !== 0) { + out = '-' + out; + } + return out; + } + + if (base === (base | 0) && base >= 2 && base <= 36) { + // var groupSize = Math.floor(BN.wordSize * Math.LN2 / Math.log(base)); + var groupSize = groupSizes[base]; + // var groupBase = Math.pow(base, groupSize); + var groupBase = groupBases[base]; + out = ''; + var c = this.clone(); + c.negative = 0; + while (!c.isZero()) { + var r = c.modrn(groupBase).toString(base); + c = c.idivn(groupBase); + + if (!c.isZero()) { + out = zeros[groupSize - r.length] + r + out; + } else { + out = r + out; + } + } + if (this.isZero()) { + out = '0' + out; + } + while (out.length % padding !== 0) { + out = '0' + out; + } + if (this.negative !== 0) { + out = '-' + out; + } + return out; + } + + assert(false, 'Base should be between 2 and 36'); + }; + + BN.prototype.toNumber = function toNumber() { + var ret = this.words[0]; + if (this.length === 2) { + ret += this.words[1] * 0x4000000; + } else if (this.length === 3 && this.words[2] === 0x01) { + // NOTE: at this stage it is known that the top bit is set + ret += 0x10000000000000 + this.words[1] * 0x4000000; + } else if (this.length > 2) { + assert(false, 'Number can only safely store up to 53 bits'); + } + return this.negative !== 0 ? -ret : ret; + }; + + BN.prototype.toJSON = function toJSON() { + return this.toString(16, 2); + }; + + if (Buffer) { + BN.prototype.toBuffer = function toBuffer(endian, length) { + return this.toArrayLike(Buffer, endian, length); + }; + } - /** - * Greater than with another share. - * @method sgt - * @param {module:jiff-client~JIFFClient#SecretShare} o - the other share. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this > o, and 0 otherwise. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.sgt = function (o, op_id) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (>)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (>)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (>)'); - } + BN.prototype.toArray = function toArray(endian, length) { + return this.toArrayLike(Array, endian, length); + }; + + var allocate = function allocate(ArrayType, size) { + if (ArrayType.allocUnsafe) { + return ArrayType.allocUnsafe(size); + } + return new ArrayType(size); + }; + + BN.prototype.toArrayLike = function toArrayLike(ArrayType, endian, length) { + this._strip(); + + var byteLength = this.byteLength(); + var reqLength = length || Math.max(1, byteLength); + assert(byteLength <= reqLength, 'byte array longer than desired length'); + assert(reqLength > 0, 'Requested array length <= 0'); + + var res = allocate(ArrayType, reqLength); + var postfix = endian === 'le' ? 'LE' : 'BE'; + this['_toArrayLike' + postfix](res, byteLength); + return res; + }; + + BN.prototype._toArrayLikeLE = function _toArrayLikeLE(res, byteLength) { + var position = 0; + var carry = 0; + + for (var i = 0, shift = 0; i < this.length; i++) { + var word = (this.words[i] << shift) | carry; + + res[position++] = word & 0xff; + if (position < res.length) { + res[position++] = (word >> 8) & 0xff; + } + if (position < res.length) { + res[position++] = (word >> 16) & 0xff; + } + + if (shift === 6) { + if (position < res.length) { + res[position++] = (word >> 24) & 0xff; + } + carry = 0; + shift = 0; + } else { + carry = word >>> 24; + shift += 2; + } + } + + if (position < res.length) { + res[position++] = carry; + + while (position < res.length) { + res[position++] = 0; + } + } + }; + + BN.prototype._toArrayLikeBE = function _toArrayLikeBE(res, byteLength) { + var position = res.length - 1; + var carry = 0; + + for (var i = 0, shift = 0; i < this.length; i++) { + var word = (this.words[i] << shift) | carry; + + res[position--] = word & 0xff; + if (position >= 0) { + res[position--] = (word >> 8) & 0xff; + } + if (position >= 0) { + res[position--] = (word >> 16) & 0xff; + } + + if (shift === 6) { + if (position >= 0) { + res[position--] = (word >> 24) & 0xff; + } + carry = 0; + shift = 0; + } else { + carry = word >>> 24; + shift += 2; + } + } + + if (position >= 0) { + res[position--] = carry; + + while (position >= 0) { + res[position--] = 0; + } + } + }; + + if (Math.clz32) { + BN.prototype._countBits = function _countBits(w) { + return 32 - Math.clz32(w); + }; + } else { + BN.prototype._countBits = function _countBits(w) { + var t = w; + var r = 0; + if (t >= 0x1000) { + r += 13; + t >>>= 13; + } + if (t >= 0x40) { + r += 7; + t >>>= 7; + } + if (t >= 0x8) { + r += 4; + t >>>= 4; + } + if (t >= 0x02) { + r += 2; + t >>>= 2; + } + return r + t; + }; + } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('sgt', this.holders); - } + BN.prototype._zeroBits = function _zeroBits(w) { + // Short-cut + if (w === 0) return 26; + + var t = w; + var r = 0; + if ((t & 0x1fff) === 0) { + r += 13; + t >>>= 13; + } + if ((t & 0x7f) === 0) { + r += 7; + t >>>= 7; + } + if ((t & 0xf) === 0) { + r += 4; + t >>>= 4; + } + if ((t & 0x3) === 0) { + r += 2; + t >>>= 2; + } + if ((t & 0x1) === 0) { + r++; + } + return r; + }; + + // Return number of used bits in a BN + BN.prototype.bitLength = function bitLength() { + var w = this.words[this.length - 1]; + var hi = this._countBits(w); + return (this.length - 1) * 26 + hi; + }; + + function toBitArray(num) { + var w = new Array(num.bitLength()); + + for (var bit = 0; bit < w.length; bit++) { + var off = (bit / 26) | 0; + var wbit = bit % 26; + + w[bit] = (num.words[off] >>> wbit) & 0x01; + } + + return w; + } - return o.islt(this, op_id); - }; + // Number of trailing zero bits + BN.prototype.zeroBits = function zeroBits() { + if (this.isZero()) return 0; + + var r = 0; + for (var i = 0; i < this.length; i++) { + var b = this._zeroBits(this.words[i]); + r += b; + if (b !== 26) break; + } + return r; + }; + + BN.prototype.byteLength = function byteLength() { + return Math.ceil(this.bitLength() / 8); + }; + + BN.prototype.toTwos = function toTwos(width) { + if (this.negative !== 0) { + return this.abs().inotn(width).iaddn(1); + } + return this.clone(); + }; + + BN.prototype.fromTwos = function fromTwos(width) { + if (this.testn(width - 1)) { + return this.notn(width).iaddn(1).ineg(); + } + return this.clone(); + }; + + BN.prototype.isNeg = function isNeg() { + return this.negative !== 0; + }; + + // Return negative clone of `this` + BN.prototype.neg = function neg() { + return this.clone().ineg(); + }; + + BN.prototype.ineg = function ineg() { + if (!this.isZero()) { + this.negative ^= 1; + } + + return this; + }; + + // Or `num` with `this` in-place + BN.prototype.iuor = function iuor(num) { + while (this.length < num.length) { + this.words[this.length++] = 0; + } + + for (var i = 0; i < num.length; i++) { + this.words[i] = this.words[i] | num.words[i]; + } + + return this._strip(); + }; + + BN.prototype.ior = function ior(num) { + assert((this.negative | num.negative) === 0); + return this.iuor(num); + }; + + // Or `num` with `this` + BN.prototype.or = function or(num) { + if (this.length > num.length) return this.clone().ior(num); + return num.clone().ior(this); + }; + + BN.prototype.uor = function uor(num) { + if (this.length > num.length) return this.clone().iuor(num); + return num.clone().iuor(this); + }; + + // And `num` with `this` in-place + BN.prototype.iuand = function iuand(num) { + // b = min-length(num, this) + var b; + if (this.length > num.length) { + b = num; + } else { + b = this; + } + + for (var i = 0; i < b.length; i++) { + this.words[i] = this.words[i] & num.words[i]; + } + + this.length = b.length; + + return this._strip(); + }; + + BN.prototype.iand = function iand(num) { + assert((this.negative | num.negative) === 0); + return this.iuand(num); + }; + + // And `num` with `this` + BN.prototype.and = function and(num) { + if (this.length > num.length) return this.clone().iand(num); + return num.clone().iand(this); + }; + + BN.prototype.uand = function uand(num) { + if (this.length > num.length) return this.clone().iuand(num); + return num.clone().iuand(this); + }; + + // Xor `num` with `this` in-place + BN.prototype.iuxor = function iuxor(num) { + // a.length > b.length + var a; + var b; + if (this.length > num.length) { + a = this; + b = num; + } else { + a = num; + b = this; + } + + for (var i = 0; i < b.length; i++) { + this.words[i] = a.words[i] ^ b.words[i]; + } + + if (this !== a) { + for (; i < a.length; i++) { + this.words[i] = a.words[i]; + } + } + + this.length = a.length; + + return this._strip(); + }; + + BN.prototype.ixor = function ixor(num) { + assert((this.negative | num.negative) === 0); + return this.iuxor(num); + }; + + // Xor `num` with `this` + BN.prototype.xor = function xor(num) { + if (this.length > num.length) return this.clone().ixor(num); + return num.clone().ixor(this); + }; + + BN.prototype.uxor = function uxor(num) { + if (this.length > num.length) return this.clone().iuxor(num); + return num.clone().iuxor(this); + }; + + // Not ``this`` with ``width`` bitwidth + BN.prototype.inotn = function inotn(width) { + assert(typeof width === 'number' && width >= 0); + + var bytesNeeded = Math.ceil(width / 26) | 0; + var bitsLeft = width % 26; + + // Extend the buffer with leading zeroes + this._expand(bytesNeeded); + + if (bitsLeft > 0) { + bytesNeeded--; + } + + // Handle complete words + for (var i = 0; i < bytesNeeded; i++) { + this.words[i] = ~this.words[i] & 0x3ffffff; + } + + // Handle the residue + if (bitsLeft > 0) { + this.words[i] = ~this.words[i] & (0x3ffffff >> (26 - bitsLeft)); + } + + // And remove leading zeroes + return this._strip(); + }; + + BN.prototype.notn = function notn(width) { + return this.clone().inotn(width); + }; + + // Set `bit` of `this` + BN.prototype.setn = function setn(bit, val) { + assert(typeof bit === 'number' && bit >= 0); + + var off = (bit / 26) | 0; + var wbit = bit % 26; + + this._expand(off + 1); + + if (val) { + this.words[off] = this.words[off] | (1 << wbit); + } else { + this.words[off] = this.words[off] & ~(1 << wbit); + } + + return this._strip(); + }; + + // Add `num` to `this` in-place + BN.prototype.iadd = function iadd(num) { + var r; + + // negative + positive + if (this.negative !== 0 && num.negative === 0) { + this.negative = 0; + r = this.isub(num); + this.negative ^= 1; + return this._normSign(); + + // positive + negative + } else if (this.negative === 0 && num.negative !== 0) { + num.negative = 0; + r = this.isub(num); + num.negative = 1; + return r._normSign(); + } + + // a.length > b.length + var a, b; + if (this.length > num.length) { + a = this; + b = num; + } else { + a = num; + b = this; + } + + var carry = 0; + for (var i = 0; i < b.length; i++) { + r = (a.words[i] | 0) + (b.words[i] | 0) + carry; + this.words[i] = r & 0x3ffffff; + carry = r >>> 26; + } + for (; carry !== 0 && i < a.length; i++) { + r = (a.words[i] | 0) + carry; + this.words[i] = r & 0x3ffffff; + carry = r >>> 26; + } + + this.length = a.length; + if (carry !== 0) { + this.words[this.length] = carry; + this.length++; + // Copy the rest of the words + } else if (a !== this) { + for (; i < a.length; i++) { + this.words[i] = a.words[i]; + } + } + + return this; + }; + + // Add `num` to `this` + BN.prototype.add = function add(num) { + var res; + if (num.negative !== 0 && this.negative === 0) { + num.negative = 0; + res = this.sub(num); + num.negative ^= 1; + return res; + } else if (num.negative === 0 && this.negative !== 0) { + this.negative = 0; + res = num.sub(this); + this.negative = 1; + return res; + } + + if (this.length > num.length) return this.clone().iadd(num); + + return num.clone().iadd(this); + }; + + // Subtract `num` from `this` in-place + BN.prototype.isub = function isub(num) { + // this - (-num) = this + num + if (num.negative !== 0) { + num.negative = 0; + var r = this.iadd(num); + num.negative = 1; + return r._normSign(); + + // -this - num = -(this + num) + } else if (this.negative !== 0) { + this.negative = 0; + this.iadd(num); + this.negative = 1; + return this._normSign(); + } + + // At this point both numbers are positive + var cmp = this.cmp(num); + + // Optimization - zeroify + if (cmp === 0) { + this.negative = 0; + this.length = 1; + this.words[0] = 0; + return this; + } + + // a > b + var a, b; + if (cmp > 0) { + a = this; + b = num; + } else { + a = num; + b = this; + } + + var carry = 0; + for (var i = 0; i < b.length; i++) { + r = (a.words[i] | 0) - (b.words[i] | 0) + carry; + carry = r >> 26; + this.words[i] = r & 0x3ffffff; + } + for (; carry !== 0 && i < a.length; i++) { + r = (a.words[i] | 0) + carry; + carry = r >> 26; + this.words[i] = r & 0x3ffffff; + } + + // Copy rest of the words + if (carry === 0 && i < a.length && a !== this) { + for (; i < a.length; i++) { + this.words[i] = a.words[i]; + } + } + + this.length = Math.max(this.length, i); + + if (a !== this) { + this.negative = 1; + } + + return this._strip(); + }; + + // Subtract `num` from `this` + BN.prototype.sub = function sub(num) { + return this.clone().isub(num); + }; + + function smallMulTo(self, num, out) { + out.negative = num.negative ^ self.negative; + var len = (self.length + num.length) | 0; + out.length = len; + len = (len - 1) | 0; + + // Peel one iteration (compiler can't do it, because of code complexity) + var a = self.words[0] | 0; + var b = num.words[0] | 0; + var r = a * b; + + var lo = r & 0x3ffffff; + var carry = (r / 0x4000000) | 0; + out.words[0] = lo; + + for (var k = 1; k < len; k++) { + // Sum all words with the same `i + j = k` and accumulate `ncarry`, + // note that ncarry could be >= 0x3ffffff + var ncarry = carry >>> 26; + var rword = carry & 0x3ffffff; + var maxJ = Math.min(k, num.length - 1); + for (var j = Math.max(0, k - self.length + 1); j <= maxJ; j++) { + var i = (k - j) | 0; + a = self.words[i] | 0; + b = num.words[j] | 0; + r = a * b + rword; + ncarry += (r / 0x4000000) | 0; + rword = r & 0x3ffffff; + } + out.words[k] = rword | 0; + carry = ncarry | 0; + } + if (carry !== 0) { + out.words[k] = carry | 0; + } else { + out.length--; + } + + return out._strip(); + } - /** - * Less than or equal with another share. - * @method slteq - * @param {module:jiff-client~JIFFClient#SecretShare} o - the other share. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this <= o, and 0 otherwise. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.slteq = function (o, op_id) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (<=)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (<=)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (<=)'); - } + // TODO(indutny): it may be reasonable to omit it for users who don't need + // to work with 256-bit numbers, otherwise it gives 20% improvement for 256-bit + // multiplication (like elliptic secp256k1). + var comb10MulTo = function comb10MulTo(self, num, out) { + var a = self.words; + var b = num.words; + var o = out.words; + var c = 0; + var lo; + var mid; + var hi; + var a0 = a[0] | 0; + var al0 = a0 & 0x1fff; + var ah0 = a0 >>> 13; + var a1 = a[1] | 0; + var al1 = a1 & 0x1fff; + var ah1 = a1 >>> 13; + var a2 = a[2] | 0; + var al2 = a2 & 0x1fff; + var ah2 = a2 >>> 13; + var a3 = a[3] | 0; + var al3 = a3 & 0x1fff; + var ah3 = a3 >>> 13; + var a4 = a[4] | 0; + var al4 = a4 & 0x1fff; + var ah4 = a4 >>> 13; + var a5 = a[5] | 0; + var al5 = a5 & 0x1fff; + var ah5 = a5 >>> 13; + var a6 = a[6] | 0; + var al6 = a6 & 0x1fff; + var ah6 = a6 >>> 13; + var a7 = a[7] | 0; + var al7 = a7 & 0x1fff; + var ah7 = a7 >>> 13; + var a8 = a[8] | 0; + var al8 = a8 & 0x1fff; + var ah8 = a8 >>> 13; + var a9 = a[9] | 0; + var al9 = a9 & 0x1fff; + var ah9 = a9 >>> 13; + var b0 = b[0] | 0; + var bl0 = b0 & 0x1fff; + var bh0 = b0 >>> 13; + var b1 = b[1] | 0; + var bl1 = b1 & 0x1fff; + var bh1 = b1 >>> 13; + var b2 = b[2] | 0; + var bl2 = b2 & 0x1fff; + var bh2 = b2 >>> 13; + var b3 = b[3] | 0; + var bl3 = b3 & 0x1fff; + var bh3 = b3 >>> 13; + var b4 = b[4] | 0; + var bl4 = b4 & 0x1fff; + var bh4 = b4 >>> 13; + var b5 = b[5] | 0; + var bl5 = b5 & 0x1fff; + var bh5 = b5 >>> 13; + var b6 = b[6] | 0; + var bl6 = b6 & 0x1fff; + var bh6 = b6 >>> 13; + var b7 = b[7] | 0; + var bl7 = b7 & 0x1fff; + var bh7 = b7 >>> 13; + var b8 = b[8] | 0; + var bl8 = b8 & 0x1fff; + var bh8 = b8 >>> 13; + var b9 = b[9] | 0; + var bl9 = b9 & 0x1fff; + var bh9 = b9 >>> 13; + + out.negative = self.negative ^ num.negative; + out.length = 19; + /* k = 0 */ + lo = Math.imul(al0, bl0); + mid = Math.imul(al0, bh0); + mid = (mid + Math.imul(ah0, bl0)) | 0; + hi = Math.imul(ah0, bh0); + var w0 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w0 >>> 26)) | 0; + w0 &= 0x3ffffff; + /* k = 1 */ + lo = Math.imul(al1, bl0); + mid = Math.imul(al1, bh0); + mid = (mid + Math.imul(ah1, bl0)) | 0; + hi = Math.imul(ah1, bh0); + lo = (lo + Math.imul(al0, bl1)) | 0; + mid = (mid + Math.imul(al0, bh1)) | 0; + mid = (mid + Math.imul(ah0, bl1)) | 0; + hi = (hi + Math.imul(ah0, bh1)) | 0; + var w1 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w1 >>> 26)) | 0; + w1 &= 0x3ffffff; + /* k = 2 */ + lo = Math.imul(al2, bl0); + mid = Math.imul(al2, bh0); + mid = (mid + Math.imul(ah2, bl0)) | 0; + hi = Math.imul(ah2, bh0); + lo = (lo + Math.imul(al1, bl1)) | 0; + mid = (mid + Math.imul(al1, bh1)) | 0; + mid = (mid + Math.imul(ah1, bl1)) | 0; + hi = (hi + Math.imul(ah1, bh1)) | 0; + lo = (lo + Math.imul(al0, bl2)) | 0; + mid = (mid + Math.imul(al0, bh2)) | 0; + mid = (mid + Math.imul(ah0, bl2)) | 0; + hi = (hi + Math.imul(ah0, bh2)) | 0; + var w2 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w2 >>> 26)) | 0; + w2 &= 0x3ffffff; + /* k = 3 */ + lo = Math.imul(al3, bl0); + mid = Math.imul(al3, bh0); + mid = (mid + Math.imul(ah3, bl0)) | 0; + hi = Math.imul(ah3, bh0); + lo = (lo + Math.imul(al2, bl1)) | 0; + mid = (mid + Math.imul(al2, bh1)) | 0; + mid = (mid + Math.imul(ah2, bl1)) | 0; + hi = (hi + Math.imul(ah2, bh1)) | 0; + lo = (lo + Math.imul(al1, bl2)) | 0; + mid = (mid + Math.imul(al1, bh2)) | 0; + mid = (mid + Math.imul(ah1, bl2)) | 0; + hi = (hi + Math.imul(ah1, bh2)) | 0; + lo = (lo + Math.imul(al0, bl3)) | 0; + mid = (mid + Math.imul(al0, bh3)) | 0; + mid = (mid + Math.imul(ah0, bl3)) | 0; + hi = (hi + Math.imul(ah0, bh3)) | 0; + var w3 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w3 >>> 26)) | 0; + w3 &= 0x3ffffff; + /* k = 4 */ + lo = Math.imul(al4, bl0); + mid = Math.imul(al4, bh0); + mid = (mid + Math.imul(ah4, bl0)) | 0; + hi = Math.imul(ah4, bh0); + lo = (lo + Math.imul(al3, bl1)) | 0; + mid = (mid + Math.imul(al3, bh1)) | 0; + mid = (mid + Math.imul(ah3, bl1)) | 0; + hi = (hi + Math.imul(ah3, bh1)) | 0; + lo = (lo + Math.imul(al2, bl2)) | 0; + mid = (mid + Math.imul(al2, bh2)) | 0; + mid = (mid + Math.imul(ah2, bl2)) | 0; + hi = (hi + Math.imul(ah2, bh2)) | 0; + lo = (lo + Math.imul(al1, bl3)) | 0; + mid = (mid + Math.imul(al1, bh3)) | 0; + mid = (mid + Math.imul(ah1, bl3)) | 0; + hi = (hi + Math.imul(ah1, bh3)) | 0; + lo = (lo + Math.imul(al0, bl4)) | 0; + mid = (mid + Math.imul(al0, bh4)) | 0; + mid = (mid + Math.imul(ah0, bl4)) | 0; + hi = (hi + Math.imul(ah0, bh4)) | 0; + var w4 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w4 >>> 26)) | 0; + w4 &= 0x3ffffff; + /* k = 5 */ + lo = Math.imul(al5, bl0); + mid = Math.imul(al5, bh0); + mid = (mid + Math.imul(ah5, bl0)) | 0; + hi = Math.imul(ah5, bh0); + lo = (lo + Math.imul(al4, bl1)) | 0; + mid = (mid + Math.imul(al4, bh1)) | 0; + mid = (mid + Math.imul(ah4, bl1)) | 0; + hi = (hi + Math.imul(ah4, bh1)) | 0; + lo = (lo + Math.imul(al3, bl2)) | 0; + mid = (mid + Math.imul(al3, bh2)) | 0; + mid = (mid + Math.imul(ah3, bl2)) | 0; + hi = (hi + Math.imul(ah3, bh2)) | 0; + lo = (lo + Math.imul(al2, bl3)) | 0; + mid = (mid + Math.imul(al2, bh3)) | 0; + mid = (mid + Math.imul(ah2, bl3)) | 0; + hi = (hi + Math.imul(ah2, bh3)) | 0; + lo = (lo + Math.imul(al1, bl4)) | 0; + mid = (mid + Math.imul(al1, bh4)) | 0; + mid = (mid + Math.imul(ah1, bl4)) | 0; + hi = (hi + Math.imul(ah1, bh4)) | 0; + lo = (lo + Math.imul(al0, bl5)) | 0; + mid = (mid + Math.imul(al0, bh5)) | 0; + mid = (mid + Math.imul(ah0, bl5)) | 0; + hi = (hi + Math.imul(ah0, bh5)) | 0; + var w5 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w5 >>> 26)) | 0; + w5 &= 0x3ffffff; + /* k = 6 */ + lo = Math.imul(al6, bl0); + mid = Math.imul(al6, bh0); + mid = (mid + Math.imul(ah6, bl0)) | 0; + hi = Math.imul(ah6, bh0); + lo = (lo + Math.imul(al5, bl1)) | 0; + mid = (mid + Math.imul(al5, bh1)) | 0; + mid = (mid + Math.imul(ah5, bl1)) | 0; + hi = (hi + Math.imul(ah5, bh1)) | 0; + lo = (lo + Math.imul(al4, bl2)) | 0; + mid = (mid + Math.imul(al4, bh2)) | 0; + mid = (mid + Math.imul(ah4, bl2)) | 0; + hi = (hi + Math.imul(ah4, bh2)) | 0; + lo = (lo + Math.imul(al3, bl3)) | 0; + mid = (mid + Math.imul(al3, bh3)) | 0; + mid = (mid + Math.imul(ah3, bl3)) | 0; + hi = (hi + Math.imul(ah3, bh3)) | 0; + lo = (lo + Math.imul(al2, bl4)) | 0; + mid = (mid + Math.imul(al2, bh4)) | 0; + mid = (mid + Math.imul(ah2, bl4)) | 0; + hi = (hi + Math.imul(ah2, bh4)) | 0; + lo = (lo + Math.imul(al1, bl5)) | 0; + mid = (mid + Math.imul(al1, bh5)) | 0; + mid = (mid + Math.imul(ah1, bl5)) | 0; + hi = (hi + Math.imul(ah1, bh5)) | 0; + lo = (lo + Math.imul(al0, bl6)) | 0; + mid = (mid + Math.imul(al0, bh6)) | 0; + mid = (mid + Math.imul(ah0, bl6)) | 0; + hi = (hi + Math.imul(ah0, bh6)) | 0; + var w6 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w6 >>> 26)) | 0; + w6 &= 0x3ffffff; + /* k = 7 */ + lo = Math.imul(al7, bl0); + mid = Math.imul(al7, bh0); + mid = (mid + Math.imul(ah7, bl0)) | 0; + hi = Math.imul(ah7, bh0); + lo = (lo + Math.imul(al6, bl1)) | 0; + mid = (mid + Math.imul(al6, bh1)) | 0; + mid = (mid + Math.imul(ah6, bl1)) | 0; + hi = (hi + Math.imul(ah6, bh1)) | 0; + lo = (lo + Math.imul(al5, bl2)) | 0; + mid = (mid + Math.imul(al5, bh2)) | 0; + mid = (mid + Math.imul(ah5, bl2)) | 0; + hi = (hi + Math.imul(ah5, bh2)) | 0; + lo = (lo + Math.imul(al4, bl3)) | 0; + mid = (mid + Math.imul(al4, bh3)) | 0; + mid = (mid + Math.imul(ah4, bl3)) | 0; + hi = (hi + Math.imul(ah4, bh3)) | 0; + lo = (lo + Math.imul(al3, bl4)) | 0; + mid = (mid + Math.imul(al3, bh4)) | 0; + mid = (mid + Math.imul(ah3, bl4)) | 0; + hi = (hi + Math.imul(ah3, bh4)) | 0; + lo = (lo + Math.imul(al2, bl5)) | 0; + mid = (mid + Math.imul(al2, bh5)) | 0; + mid = (mid + Math.imul(ah2, bl5)) | 0; + hi = (hi + Math.imul(ah2, bh5)) | 0; + lo = (lo + Math.imul(al1, bl6)) | 0; + mid = (mid + Math.imul(al1, bh6)) | 0; + mid = (mid + Math.imul(ah1, bl6)) | 0; + hi = (hi + Math.imul(ah1, bh6)) | 0; + lo = (lo + Math.imul(al0, bl7)) | 0; + mid = (mid + Math.imul(al0, bh7)) | 0; + mid = (mid + Math.imul(ah0, bl7)) | 0; + hi = (hi + Math.imul(ah0, bh7)) | 0; + var w7 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w7 >>> 26)) | 0; + w7 &= 0x3ffffff; + /* k = 8 */ + lo = Math.imul(al8, bl0); + mid = Math.imul(al8, bh0); + mid = (mid + Math.imul(ah8, bl0)) | 0; + hi = Math.imul(ah8, bh0); + lo = (lo + Math.imul(al7, bl1)) | 0; + mid = (mid + Math.imul(al7, bh1)) | 0; + mid = (mid + Math.imul(ah7, bl1)) | 0; + hi = (hi + Math.imul(ah7, bh1)) | 0; + lo = (lo + Math.imul(al6, bl2)) | 0; + mid = (mid + Math.imul(al6, bh2)) | 0; + mid = (mid + Math.imul(ah6, bl2)) | 0; + hi = (hi + Math.imul(ah6, bh2)) | 0; + lo = (lo + Math.imul(al5, bl3)) | 0; + mid = (mid + Math.imul(al5, bh3)) | 0; + mid = (mid + Math.imul(ah5, bl3)) | 0; + hi = (hi + Math.imul(ah5, bh3)) | 0; + lo = (lo + Math.imul(al4, bl4)) | 0; + mid = (mid + Math.imul(al4, bh4)) | 0; + mid = (mid + Math.imul(ah4, bl4)) | 0; + hi = (hi + Math.imul(ah4, bh4)) | 0; + lo = (lo + Math.imul(al3, bl5)) | 0; + mid = (mid + Math.imul(al3, bh5)) | 0; + mid = (mid + Math.imul(ah3, bl5)) | 0; + hi = (hi + Math.imul(ah3, bh5)) | 0; + lo = (lo + Math.imul(al2, bl6)) | 0; + mid = (mid + Math.imul(al2, bh6)) | 0; + mid = (mid + Math.imul(ah2, bl6)) | 0; + hi = (hi + Math.imul(ah2, bh6)) | 0; + lo = (lo + Math.imul(al1, bl7)) | 0; + mid = (mid + Math.imul(al1, bh7)) | 0; + mid = (mid + Math.imul(ah1, bl7)) | 0; + hi = (hi + Math.imul(ah1, bh7)) | 0; + lo = (lo + Math.imul(al0, bl8)) | 0; + mid = (mid + Math.imul(al0, bh8)) | 0; + mid = (mid + Math.imul(ah0, bl8)) | 0; + hi = (hi + Math.imul(ah0, bh8)) | 0; + var w8 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w8 >>> 26)) | 0; + w8 &= 0x3ffffff; + /* k = 9 */ + lo = Math.imul(al9, bl0); + mid = Math.imul(al9, bh0); + mid = (mid + Math.imul(ah9, bl0)) | 0; + hi = Math.imul(ah9, bh0); + lo = (lo + Math.imul(al8, bl1)) | 0; + mid = (mid + Math.imul(al8, bh1)) | 0; + mid = (mid + Math.imul(ah8, bl1)) | 0; + hi = (hi + Math.imul(ah8, bh1)) | 0; + lo = (lo + Math.imul(al7, bl2)) | 0; + mid = (mid + Math.imul(al7, bh2)) | 0; + mid = (mid + Math.imul(ah7, bl2)) | 0; + hi = (hi + Math.imul(ah7, bh2)) | 0; + lo = (lo + Math.imul(al6, bl3)) | 0; + mid = (mid + Math.imul(al6, bh3)) | 0; + mid = (mid + Math.imul(ah6, bl3)) | 0; + hi = (hi + Math.imul(ah6, bh3)) | 0; + lo = (lo + Math.imul(al5, bl4)) | 0; + mid = (mid + Math.imul(al5, bh4)) | 0; + mid = (mid + Math.imul(ah5, bl4)) | 0; + hi = (hi + Math.imul(ah5, bh4)) | 0; + lo = (lo + Math.imul(al4, bl5)) | 0; + mid = (mid + Math.imul(al4, bh5)) | 0; + mid = (mid + Math.imul(ah4, bl5)) | 0; + hi = (hi + Math.imul(ah4, bh5)) | 0; + lo = (lo + Math.imul(al3, bl6)) | 0; + mid = (mid + Math.imul(al3, bh6)) | 0; + mid = (mid + Math.imul(ah3, bl6)) | 0; + hi = (hi + Math.imul(ah3, bh6)) | 0; + lo = (lo + Math.imul(al2, bl7)) | 0; + mid = (mid + Math.imul(al2, bh7)) | 0; + mid = (mid + Math.imul(ah2, bl7)) | 0; + hi = (hi + Math.imul(ah2, bh7)) | 0; + lo = (lo + Math.imul(al1, bl8)) | 0; + mid = (mid + Math.imul(al1, bh8)) | 0; + mid = (mid + Math.imul(ah1, bl8)) | 0; + hi = (hi + Math.imul(ah1, bh8)) | 0; + lo = (lo + Math.imul(al0, bl9)) | 0; + mid = (mid + Math.imul(al0, bh9)) | 0; + mid = (mid + Math.imul(ah0, bl9)) | 0; + hi = (hi + Math.imul(ah0, bh9)) | 0; + var w9 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w9 >>> 26)) | 0; + w9 &= 0x3ffffff; + /* k = 10 */ + lo = Math.imul(al9, bl1); + mid = Math.imul(al9, bh1); + mid = (mid + Math.imul(ah9, bl1)) | 0; + hi = Math.imul(ah9, bh1); + lo = (lo + Math.imul(al8, bl2)) | 0; + mid = (mid + Math.imul(al8, bh2)) | 0; + mid = (mid + Math.imul(ah8, bl2)) | 0; + hi = (hi + Math.imul(ah8, bh2)) | 0; + lo = (lo + Math.imul(al7, bl3)) | 0; + mid = (mid + Math.imul(al7, bh3)) | 0; + mid = (mid + Math.imul(ah7, bl3)) | 0; + hi = (hi + Math.imul(ah7, bh3)) | 0; + lo = (lo + Math.imul(al6, bl4)) | 0; + mid = (mid + Math.imul(al6, bh4)) | 0; + mid = (mid + Math.imul(ah6, bl4)) | 0; + hi = (hi + Math.imul(ah6, bh4)) | 0; + lo = (lo + Math.imul(al5, bl5)) | 0; + mid = (mid + Math.imul(al5, bh5)) | 0; + mid = (mid + Math.imul(ah5, bl5)) | 0; + hi = (hi + Math.imul(ah5, bh5)) | 0; + lo = (lo + Math.imul(al4, bl6)) | 0; + mid = (mid + Math.imul(al4, bh6)) | 0; + mid = (mid + Math.imul(ah4, bl6)) | 0; + hi = (hi + Math.imul(ah4, bh6)) | 0; + lo = (lo + Math.imul(al3, bl7)) | 0; + mid = (mid + Math.imul(al3, bh7)) | 0; + mid = (mid + Math.imul(ah3, bl7)) | 0; + hi = (hi + Math.imul(ah3, bh7)) | 0; + lo = (lo + Math.imul(al2, bl8)) | 0; + mid = (mid + Math.imul(al2, bh8)) | 0; + mid = (mid + Math.imul(ah2, bl8)) | 0; + hi = (hi + Math.imul(ah2, bh8)) | 0; + lo = (lo + Math.imul(al1, bl9)) | 0; + mid = (mid + Math.imul(al1, bh9)) | 0; + mid = (mid + Math.imul(ah1, bl9)) | 0; + hi = (hi + Math.imul(ah1, bh9)) | 0; + var w10 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w10 >>> 26)) | 0; + w10 &= 0x3ffffff; + /* k = 11 */ + lo = Math.imul(al9, bl2); + mid = Math.imul(al9, bh2); + mid = (mid + Math.imul(ah9, bl2)) | 0; + hi = Math.imul(ah9, bh2); + lo = (lo + Math.imul(al8, bl3)) | 0; + mid = (mid + Math.imul(al8, bh3)) | 0; + mid = (mid + Math.imul(ah8, bl3)) | 0; + hi = (hi + Math.imul(ah8, bh3)) | 0; + lo = (lo + Math.imul(al7, bl4)) | 0; + mid = (mid + Math.imul(al7, bh4)) | 0; + mid = (mid + Math.imul(ah7, bl4)) | 0; + hi = (hi + Math.imul(ah7, bh4)) | 0; + lo = (lo + Math.imul(al6, bl5)) | 0; + mid = (mid + Math.imul(al6, bh5)) | 0; + mid = (mid + Math.imul(ah6, bl5)) | 0; + hi = (hi + Math.imul(ah6, bh5)) | 0; + lo = (lo + Math.imul(al5, bl6)) | 0; + mid = (mid + Math.imul(al5, bh6)) | 0; + mid = (mid + Math.imul(ah5, bl6)) | 0; + hi = (hi + Math.imul(ah5, bh6)) | 0; + lo = (lo + Math.imul(al4, bl7)) | 0; + mid = (mid + Math.imul(al4, bh7)) | 0; + mid = (mid + Math.imul(ah4, bl7)) | 0; + hi = (hi + Math.imul(ah4, bh7)) | 0; + lo = (lo + Math.imul(al3, bl8)) | 0; + mid = (mid + Math.imul(al3, bh8)) | 0; + mid = (mid + Math.imul(ah3, bl8)) | 0; + hi = (hi + Math.imul(ah3, bh8)) | 0; + lo = (lo + Math.imul(al2, bl9)) | 0; + mid = (mid + Math.imul(al2, bh9)) | 0; + mid = (mid + Math.imul(ah2, bl9)) | 0; + hi = (hi + Math.imul(ah2, bh9)) | 0; + var w11 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w11 >>> 26)) | 0; + w11 &= 0x3ffffff; + /* k = 12 */ + lo = Math.imul(al9, bl3); + mid = Math.imul(al9, bh3); + mid = (mid + Math.imul(ah9, bl3)) | 0; + hi = Math.imul(ah9, bh3); + lo = (lo + Math.imul(al8, bl4)) | 0; + mid = (mid + Math.imul(al8, bh4)) | 0; + mid = (mid + Math.imul(ah8, bl4)) | 0; + hi = (hi + Math.imul(ah8, bh4)) | 0; + lo = (lo + Math.imul(al7, bl5)) | 0; + mid = (mid + Math.imul(al7, bh5)) | 0; + mid = (mid + Math.imul(ah7, bl5)) | 0; + hi = (hi + Math.imul(ah7, bh5)) | 0; + lo = (lo + Math.imul(al6, bl6)) | 0; + mid = (mid + Math.imul(al6, bh6)) | 0; + mid = (mid + Math.imul(ah6, bl6)) | 0; + hi = (hi + Math.imul(ah6, bh6)) | 0; + lo = (lo + Math.imul(al5, bl7)) | 0; + mid = (mid + Math.imul(al5, bh7)) | 0; + mid = (mid + Math.imul(ah5, bl7)) | 0; + hi = (hi + Math.imul(ah5, bh7)) | 0; + lo = (lo + Math.imul(al4, bl8)) | 0; + mid = (mid + Math.imul(al4, bh8)) | 0; + mid = (mid + Math.imul(ah4, bl8)) | 0; + hi = (hi + Math.imul(ah4, bh8)) | 0; + lo = (lo + Math.imul(al3, bl9)) | 0; + mid = (mid + Math.imul(al3, bh9)) | 0; + mid = (mid + Math.imul(ah3, bl9)) | 0; + hi = (hi + Math.imul(ah3, bh9)) | 0; + var w12 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w12 >>> 26)) | 0; + w12 &= 0x3ffffff; + /* k = 13 */ + lo = Math.imul(al9, bl4); + mid = Math.imul(al9, bh4); + mid = (mid + Math.imul(ah9, bl4)) | 0; + hi = Math.imul(ah9, bh4); + lo = (lo + Math.imul(al8, bl5)) | 0; + mid = (mid + Math.imul(al8, bh5)) | 0; + mid = (mid + Math.imul(ah8, bl5)) | 0; + hi = (hi + Math.imul(ah8, bh5)) | 0; + lo = (lo + Math.imul(al7, bl6)) | 0; + mid = (mid + Math.imul(al7, bh6)) | 0; + mid = (mid + Math.imul(ah7, bl6)) | 0; + hi = (hi + Math.imul(ah7, bh6)) | 0; + lo = (lo + Math.imul(al6, bl7)) | 0; + mid = (mid + Math.imul(al6, bh7)) | 0; + mid = (mid + Math.imul(ah6, bl7)) | 0; + hi = (hi + Math.imul(ah6, bh7)) | 0; + lo = (lo + Math.imul(al5, bl8)) | 0; + mid = (mid + Math.imul(al5, bh8)) | 0; + mid = (mid + Math.imul(ah5, bl8)) | 0; + hi = (hi + Math.imul(ah5, bh8)) | 0; + lo = (lo + Math.imul(al4, bl9)) | 0; + mid = (mid + Math.imul(al4, bh9)) | 0; + mid = (mid + Math.imul(ah4, bl9)) | 0; + hi = (hi + Math.imul(ah4, bh9)) | 0; + var w13 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w13 >>> 26)) | 0; + w13 &= 0x3ffffff; + /* k = 14 */ + lo = Math.imul(al9, bl5); + mid = Math.imul(al9, bh5); + mid = (mid + Math.imul(ah9, bl5)) | 0; + hi = Math.imul(ah9, bh5); + lo = (lo + Math.imul(al8, bl6)) | 0; + mid = (mid + Math.imul(al8, bh6)) | 0; + mid = (mid + Math.imul(ah8, bl6)) | 0; + hi = (hi + Math.imul(ah8, bh6)) | 0; + lo = (lo + Math.imul(al7, bl7)) | 0; + mid = (mid + Math.imul(al7, bh7)) | 0; + mid = (mid + Math.imul(ah7, bl7)) | 0; + hi = (hi + Math.imul(ah7, bh7)) | 0; + lo = (lo + Math.imul(al6, bl8)) | 0; + mid = (mid + Math.imul(al6, bh8)) | 0; + mid = (mid + Math.imul(ah6, bl8)) | 0; + hi = (hi + Math.imul(ah6, bh8)) | 0; + lo = (lo + Math.imul(al5, bl9)) | 0; + mid = (mid + Math.imul(al5, bh9)) | 0; + mid = (mid + Math.imul(ah5, bl9)) | 0; + hi = (hi + Math.imul(ah5, bh9)) | 0; + var w14 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w14 >>> 26)) | 0; + w14 &= 0x3ffffff; + /* k = 15 */ + lo = Math.imul(al9, bl6); + mid = Math.imul(al9, bh6); + mid = (mid + Math.imul(ah9, bl6)) | 0; + hi = Math.imul(ah9, bh6); + lo = (lo + Math.imul(al8, bl7)) | 0; + mid = (mid + Math.imul(al8, bh7)) | 0; + mid = (mid + Math.imul(ah8, bl7)) | 0; + hi = (hi + Math.imul(ah8, bh7)) | 0; + lo = (lo + Math.imul(al7, bl8)) | 0; + mid = (mid + Math.imul(al7, bh8)) | 0; + mid = (mid + Math.imul(ah7, bl8)) | 0; + hi = (hi + Math.imul(ah7, bh8)) | 0; + lo = (lo + Math.imul(al6, bl9)) | 0; + mid = (mid + Math.imul(al6, bh9)) | 0; + mid = (mid + Math.imul(ah6, bl9)) | 0; + hi = (hi + Math.imul(ah6, bh9)) | 0; + var w15 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w15 >>> 26)) | 0; + w15 &= 0x3ffffff; + /* k = 16 */ + lo = Math.imul(al9, bl7); + mid = Math.imul(al9, bh7); + mid = (mid + Math.imul(ah9, bl7)) | 0; + hi = Math.imul(ah9, bh7); + lo = (lo + Math.imul(al8, bl8)) | 0; + mid = (mid + Math.imul(al8, bh8)) | 0; + mid = (mid + Math.imul(ah8, bl8)) | 0; + hi = (hi + Math.imul(ah8, bh8)) | 0; + lo = (lo + Math.imul(al7, bl9)) | 0; + mid = (mid + Math.imul(al7, bh9)) | 0; + mid = (mid + Math.imul(ah7, bl9)) | 0; + hi = (hi + Math.imul(ah7, bh9)) | 0; + var w16 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w16 >>> 26)) | 0; + w16 &= 0x3ffffff; + /* k = 17 */ + lo = Math.imul(al9, bl8); + mid = Math.imul(al9, bh8); + mid = (mid + Math.imul(ah9, bl8)) | 0; + hi = Math.imul(ah9, bh8); + lo = (lo + Math.imul(al8, bl9)) | 0; + mid = (mid + Math.imul(al8, bh9)) | 0; + mid = (mid + Math.imul(ah8, bl9)) | 0; + hi = (hi + Math.imul(ah8, bh9)) | 0; + var w17 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w17 >>> 26)) | 0; + w17 &= 0x3ffffff; + /* k = 18 */ + lo = Math.imul(al9, bl9); + mid = Math.imul(al9, bh9); + mid = (mid + Math.imul(ah9, bl9)) | 0; + hi = Math.imul(ah9, bh9); + var w18 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; + c = (((hi + (mid >>> 13)) | 0) + (w18 >>> 26)) | 0; + w18 &= 0x3ffffff; + o[0] = w0; + o[1] = w1; + o[2] = w2; + o[3] = w3; + o[4] = w4; + o[5] = w5; + o[6] = w6; + o[7] = w7; + o[8] = w8; + o[9] = w9; + o[10] = w10; + o[11] = w11; + o[12] = w12; + o[13] = w13; + o[14] = w14; + o[15] = w15; + o[16] = w16; + o[17] = w17; + o[18] = w18; + if (c !== 0) { + o[19] = c; + out.length++; + } + return out; + }; + + // Polyfill comb + if (!Math.imul) { + comb10MulTo = smallMulTo; + } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('slteq', this.holders); - } + function bigMulTo(self, num, out) { + out.negative = num.negative ^ self.negative; + out.length = self.length + num.length; + + var carry = 0; + var hncarry = 0; + for (var k = 0; k < out.length - 1; k++) { + // Sum all words with the same `i + j = k` and accumulate `ncarry`, + // note that ncarry could be >= 0x3ffffff + var ncarry = hncarry; + hncarry = 0; + var rword = carry & 0x3ffffff; + var maxJ = Math.min(k, num.length - 1); + for (var j = Math.max(0, k - self.length + 1); j <= maxJ; j++) { + var i = k - j; + var a = self.words[i] | 0; + var b = num.words[j] | 0; + var r = a * b; + + var lo = r & 0x3ffffff; + ncarry = (ncarry + ((r / 0x4000000) | 0)) | 0; + lo = (lo + rword) | 0; + rword = lo & 0x3ffffff; + ncarry = (ncarry + (lo >>> 26)) | 0; + + hncarry += ncarry >>> 26; + ncarry &= 0x3ffffff; + } + out.words[k] = rword; + carry = ncarry; + ncarry = hncarry; + } + if (carry !== 0) { + out.words[k] = carry; + } else { + out.length--; + } + + return out._strip(); + } - return o.islt(this, op_id).inot(); - }; + function jumboMulTo(self, num, out) { + // Temporary disable, see https://github.com/indutny/bn.js/issues/211 + // var fftm = new FFTM(); + // return fftm.mulp(self, num, out); + return bigMulTo(self, num, out); + } - /** - * Less than with another share. - * @method slt - * @param {module:jiff-client~JIFFClient#SecretShare} o - the other share. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this < o, and 0 otherwise. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.slt = function (o, op_id) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (<)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (<)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (<)'); - } + BN.prototype.mulTo = function mulTo(num, out) { + var res; + var len = this.length + num.length; + if (this.length === 10 && num.length === 10) { + res = comb10MulTo(this, num, out); + } else if (len < 63) { + res = smallMulTo(this, num, out); + } else if (len < 1024) { + res = bigMulTo(this, num, out); + } else { + res = jumboMulTo(this, num, out); + } + + return res; + }; + + // Cooley-Tukey algorithm for FFT + // slightly revisited to rely on looping instead of recursion + + function FFTM(x, y) { + this.x = x; + this.y = y; + } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('slt', this.holders); - } + FFTM.prototype.makeRBT = function makeRBT(N) { + var t = new Array(N); + var l = BN.prototype._countBits(N) - 1; + for (var i = 0; i < N; i++) { + t[i] = this.revBin(i, l, N); + } - var final_deferred = new this.jiff.helpers.Deferred(); - var final_promise = final_deferred.promise; - var result = new this.jiff.SecretShare(final_promise, this.holders, Math.max(this.threshold, o.threshold), this.Zp); - - var w = this.ilt_halfprime(op_id + ':halfprime:1'); - - var self = this; - w.wThen(function () { - var x = o.ilt_halfprime(op_id + ':halfprime:2'); - x.wThen(function () { - var y = self.issub(o).ilt_halfprime(op_id + ':halfprime:3'); - y.wThen(function () { - var xy = x.ismult(y, op_id + ':smult1'); - var answer = x.icmult(-1).icadd(1).issub(y).isadd(xy).isadd(w.ismult(x.isadd(y).issub(xy.icmult(2)), op_id + ':smult2')); - answer.wThen(final_deferred.resolve); - }); - }); - }); - - return result; - }; + return t; + }; - /** - * Greater than or equal with a constant. - * @method cgteqn - * @param {number} cst - the constant to compare with. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this >= cst, and 0 otherwise. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.cgteq = function (cst, op_id) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (>=)'); - } + // Returns binary-reversed representation of `x` + FFTM.prototype.revBin = function revBin(x, l, N) { + if (x === 0 || x === N - 1) return x; - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('cgteq', this.holders); - } + var rb = 0; + for (var i = 0; i < l; i++) { + rb |= (x & 1) << (l - i - 1); + x >>= 1; + } - return this.iclt(cst, op_id).inot(); - }; + return rb; + }; + + // Performs "tweedling" phase, therefore 'emulating' + // behaviour of the recursive algorithm + FFTM.prototype.permute = function permute(rbt, rws, iws, rtws, itws, N) { + for (var i = 0; i < N; i++) { + rtws[i] = rws[rbt[i]]; + itws[i] = iws[rbt[i]]; + } + }; + + FFTM.prototype.transform = function transform(rws, iws, rtws, itws, N, rbt) { + this.permute(rbt, rws, iws, rtws, itws, N); + + for (var s = 1; s < N; s <<= 1) { + var l = s << 1; + + var rtwdf = Math.cos((2 * Math.PI) / l); + var itwdf = Math.sin((2 * Math.PI) / l); + + for (var p = 0; p < N; p += l) { + var rtwdf_ = rtwdf; + var itwdf_ = itwdf; + + for (var j = 0; j < s; j++) { + var re = rtws[p + j]; + var ie = itws[p + j]; + + var ro = rtws[p + j + s]; + var io = itws[p + j + s]; + + var rx = rtwdf_ * ro - itwdf_ * io; - /** - * Greater than with a constant. - * @method cgt - * @param {number} cst - the constant to compare with. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly.default ids suffice when all parties execute the - * instructions in the same order. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this > cst, and 0 otherwise. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.cgt = function (cst, op_id) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (>)'); - } + io = rtwdf_ * io + itwdf_ * ro; + ro = rx; - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('cgt', this.holders); - } + rtws[p + j] = re + ro; + itws[p + j] = ie + io; - var final_deferred = new this.jiff.helpers.Deferred(); - var final_promise = final_deferred.promise; - var result = new this.jiff.SecretShare(final_promise, this.holders, this.threshold, this.Zp); + rtws[p + j + s] = re - ro; + itws[p + j + s] = ie - io; - var w = this.jiff.share_helpers['<'](cst, this.jiff.share_helpers['/'](this.Zp, 2)) ? 1 : 0; - var x = this.ilt_halfprime(op_id + ':halfprime:1'); + /* jshint maxdepth : false */ + if (j !== l) { + rx = rtwdf * rtwdf_ - itwdf * itwdf_; - var self = this; - x.wThen(function () { - var y = self.icmult(-1).icadd(cst).ilt_halfprime(op_id + ':halfprime:2'); - y.wThen(function () { - var xy = y.ismult(x, op_id + ':smult1'); - var answer = x.icmult(-1).icadd(1).issub(y).isadd(xy).isadd(x.isadd(y).issub(xy.icmult(2)).icmult(w)); - answer.wThen(final_deferred.resolve); - }); - }); + itwdf_ = rtwdf * itwdf_ + itwdf * rtwdf_; + rtwdf_ = rx; + } + } + } + } + }; - return result; - }; + FFTM.prototype.guessLen13b = function guessLen13b(n, m) { + var N = Math.max(m, n) | 1; + var odd = N & 1; + var i = 0; + for (N = (N / 2) | 0; N; N = N >>> 1) { + i++; + } - /** - * Less than or equal with a constant. - * @method clteq - * @param {number} cst - the constant to compare with. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this <= cst, and 0 otherwise. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.clteq = function (cst, op_id) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (<=)'); - } + return 1 << (i + 1 + odd); + }; - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('clteq', this.holders); - } + FFTM.prototype.conjugate = function conjugate(rws, iws, N) { + if (N <= 1) return; - return this.icgt(cst, op_id).inot(); - }; + for (var i = 0; i < N / 2; i++) { + var t = rws[i]; + + rws[i] = rws[N - i - 1]; + rws[N - i - 1] = t; + + t = iws[i]; + + iws[i] = -iws[N - i - 1]; + iws[N - i - 1] = -t; + } + }; + + FFTM.prototype.normalize13b = function normalize13b(ws, N) { + var carry = 0; + for (var i = 0; i < N / 2; i++) { + var w = Math.round(ws[2 * i + 1] / N) * 0x2000 + Math.round(ws[2 * i] / N) + carry; + + ws[i] = w & 0x3ffffff; + + if (w < 0x4000000) { + carry = 0; + } else { + carry = (w / 0x4000000) | 0; + } + } + + return ws; + }; + + FFTM.prototype.convert13b = function convert13b(ws, len, rws, N) { + var carry = 0; + for (var i = 0; i < len; i++) { + carry = carry + (ws[i] | 0); + + rws[2 * i] = carry & 0x1fff; + carry = carry >>> 13; + rws[2 * i + 1] = carry & 0x1fff; + carry = carry >>> 13; + } + + // Pad with zeroes + for (i = 2 * len; i < N; ++i) { + rws[i] = 0; + } + + assert(carry === 0); + assert((carry & ~0x1fff) === 0); + }; + + FFTM.prototype.stub = function stub(N) { + var ph = new Array(N); + for (var i = 0; i < N; i++) { + ph[i] = 0; + } + + return ph; + }; + + FFTM.prototype.mulp = function mulp(x, y, out) { + var N = 2 * this.guessLen13b(x.length, y.length); + + var rbt = this.makeRBT(N); + + var _ = this.stub(N); + + var rws = new Array(N); + var rwst = new Array(N); + var iwst = new Array(N); + + var nrws = new Array(N); + var nrwst = new Array(N); + var niwst = new Array(N); + + var rmws = out.words; + rmws.length = N; + + this.convert13b(x.words, x.length, rws, N); + this.convert13b(y.words, y.length, nrws, N); + + this.transform(rws, _, rwst, iwst, N, rbt); + this.transform(nrws, _, nrwst, niwst, N, rbt); + + for (var i = 0; i < N; i++) { + var rx = rwst[i] * nrwst[i] - iwst[i] * niwst[i]; + iwst[i] = rwst[i] * niwst[i] + iwst[i] * nrwst[i]; + rwst[i] = rx; + } + + this.conjugate(rwst, iwst, N); + this.transform(rwst, iwst, rmws, _, N, rbt); + this.conjugate(rmws, _, N); + this.normalize13b(rmws, N); + + out.negative = x.negative ^ y.negative; + out.length = x.length + y.length; + return out._strip(); + }; + + // Multiply `this` by `num` + BN.prototype.mul = function mul(num) { + var out = new BN(null); + out.words = new Array(this.length + num.length); + return this.mulTo(num, out); + }; + + // Multiply employing FFT + BN.prototype.mulf = function mulf(num) { + var out = new BN(null); + out.words = new Array(this.length + num.length); + return jumboMulTo(this, num, out); + }; + + // In-place Multiplication + BN.prototype.imul = function imul(num) { + return this.clone().mulTo(num, this); + }; + + BN.prototype.imuln = function imuln(num) { + var isNegNum = num < 0; + if (isNegNum) num = -num; + + assert(typeof num === 'number'); + assert(num < 0x4000000); + + // Carry + var carry = 0; + for (var i = 0; i < this.length; i++) { + var w = (this.words[i] | 0) * num; + var lo = (w & 0x3ffffff) + (carry & 0x3ffffff); + carry >>= 26; + carry += (w / 0x4000000) | 0; + // NOTE: lo is 27bit maximum + carry += lo >>> 26; + this.words[i] = lo & 0x3ffffff; + } + + if (carry !== 0) { + this.words[i] = carry; + this.length++; + } + + return isNegNum ? this.ineg() : this; + }; + + BN.prototype.muln = function muln(num) { + return this.clone().imuln(num); + }; + + // `this` * `this` + BN.prototype.sqr = function sqr() { + return this.mul(this); + }; + + // `this` * `this` in-place + BN.prototype.isqr = function isqr() { + return this.imul(this.clone()); + }; + + // Math.pow(`this`, `num`) + BN.prototype.pow = function pow(num) { + var w = toBitArray(num); + if (w.length === 0) return new BN(1); + + // Skip leading zeroes + var res = this; + for (var i = 0; i < w.length; i++, res = res.sqr()) { + if (w[i] !== 0) break; + } + + if (++i < w.length) { + for (var q = res.sqr(); i < w.length; i++, q = q.sqr()) { + if (w[i] === 0) continue; + + res = res.mul(q); + } + } + + return res; + }; + + // Shift-left in-place + BN.prototype.iushln = function iushln(bits) { + assert(typeof bits === 'number' && bits >= 0); + var r = bits % 26; + var s = (bits - r) / 26; + var carryMask = (0x3ffffff >>> (26 - r)) << (26 - r); + var i; + + if (r !== 0) { + var carry = 0; + + for (i = 0; i < this.length; i++) { + var newCarry = this.words[i] & carryMask; + var c = ((this.words[i] | 0) - newCarry) << r; + this.words[i] = c | carry; + carry = newCarry >>> (26 - r); + } + + if (carry) { + this.words[i] = carry; + this.length++; + } + } + + if (s !== 0) { + for (i = this.length - 1; i >= 0; i--) { + this.words[i + s] = this.words[i]; + } + + for (i = 0; i < s; i++) { + this.words[i] = 0; + } + + this.length += s; + } + + return this._strip(); + }; + + BN.prototype.ishln = function ishln(bits) { + // TODO(indutny): implement me + assert(this.negative === 0); + return this.iushln(bits); + }; + + // Shift-right in-place + // NOTE: `hint` is a lowest bit before trailing zeroes + // NOTE: if `extended` is present - it will be filled with destroyed bits + BN.prototype.iushrn = function iushrn(bits, hint, extended) { + assert(typeof bits === 'number' && bits >= 0); + var h; + if (hint) { + h = (hint - (hint % 26)) / 26; + } else { + h = 0; + } + + var r = bits % 26; + var s = Math.min((bits - r) / 26, this.length); + var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r); + var maskedWords = extended; + + h -= s; + h = Math.max(0, h); + + // Extended mode, copy masked part + if (maskedWords) { + for (var i = 0; i < s; i++) { + maskedWords.words[i] = this.words[i]; + } + maskedWords.length = s; + } + + if (s === 0) { + // No-op, we should not move anything at all + } else if (this.length > s) { + this.length -= s; + for (i = 0; i < this.length; i++) { + this.words[i] = this.words[i + s]; + } + } else { + this.words[0] = 0; + this.length = 1; + } + + var carry = 0; + for (i = this.length - 1; i >= 0 && (carry !== 0 || i >= h); i--) { + var word = this.words[i] | 0; + this.words[i] = (carry << (26 - r)) | (word >>> r); + carry = word & mask; + } + + // Push carried bits as a mask + if (maskedWords && carry !== 0) { + maskedWords.words[maskedWords.length++] = carry; + } + + if (this.length === 0) { + this.words[0] = 0; + this.length = 1; + } + + return this._strip(); + }; + + BN.prototype.ishrn = function ishrn(bits, hint, extended) { + // TODO(indutny): implement me + assert(this.negative === 0); + return this.iushrn(bits, hint, extended); + }; + + // Shift-left + BN.prototype.shln = function shln(bits) { + return this.clone().ishln(bits); + }; + + BN.prototype.ushln = function ushln(bits) { + return this.clone().iushln(bits); + }; + + // Shift-right + BN.prototype.shrn = function shrn(bits) { + return this.clone().ishrn(bits); + }; + + BN.prototype.ushrn = function ushrn(bits) { + return this.clone().iushrn(bits); + }; + + // Test if n bit is set + BN.prototype.testn = function testn(bit) { + assert(typeof bit === 'number' && bit >= 0); + var r = bit % 26; + var s = (bit - r) / 26; + var q = 1 << r; + + // Fast case: bit is much higher than all existing words + if (this.length <= s) return false; + + // Check bit and return + var w = this.words[s]; + + return !!(w & q); + }; + + // Return only lowers bits of number (in-place) + BN.prototype.imaskn = function imaskn(bits) { + assert(typeof bits === 'number' && bits >= 0); + var r = bits % 26; + var s = (bits - r) / 26; + + assert(this.negative === 0, 'imaskn works only with positive numbers'); + + if (this.length <= s) { + return this; + } + + if (r !== 0) { + s++; + } + this.length = Math.min(s, this.length); + + if (r !== 0) { + var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r); + this.words[this.length - 1] &= mask; + } + + return this._strip(); + }; + + // Return only lowers bits of number + BN.prototype.maskn = function maskn(bits) { + return this.clone().imaskn(bits); + }; + + // Add plain number `num` to `this` + BN.prototype.iaddn = function iaddn(num) { + assert(typeof num === 'number'); + assert(num < 0x4000000); + if (num < 0) return this.isubn(-num); + + // Possible sign change + if (this.negative !== 0) { + if (this.length === 1 && (this.words[0] | 0) <= num) { + this.words[0] = num - (this.words[0] | 0); + this.negative = 0; + return this; + } + + this.negative = 0; + this.isubn(num); + this.negative = 1; + return this; + } + + // Add without checks + return this._iaddn(num); + }; + + BN.prototype._iaddn = function _iaddn(num) { + this.words[0] += num; + + // Carry + for (var i = 0; i < this.length && this.words[i] >= 0x4000000; i++) { + this.words[i] -= 0x4000000; + if (i === this.length - 1) { + this.words[i + 1] = 1; + } else { + this.words[i + 1]++; + } + } + this.length = Math.max(this.length, i + 1); + + return this; + }; + + // Subtract plain number `num` from `this` + BN.prototype.isubn = function isubn(num) { + assert(typeof num === 'number'); + assert(num < 0x4000000); + if (num < 0) return this.iaddn(-num); + + if (this.negative !== 0) { + this.negative = 0; + this.iaddn(num); + this.negative = 1; + return this; + } + + this.words[0] -= num; + + if (this.length === 1 && this.words[0] < 0) { + this.words[0] = -this.words[0]; + this.negative = 1; + } else { + // Carry + for (var i = 0; i < this.length && this.words[i] < 0; i++) { + this.words[i] += 0x4000000; + this.words[i + 1] -= 1; + } + } + + return this._strip(); + }; + + BN.prototype.addn = function addn(num) { + return this.clone().iaddn(num); + }; + + BN.prototype.subn = function subn(num) { + return this.clone().isubn(num); + }; + + BN.prototype.iabs = function iabs() { + this.negative = 0; + + return this; + }; + + BN.prototype.abs = function abs() { + return this.clone().iabs(); + }; + + BN.prototype._ishlnsubmul = function _ishlnsubmul(num, mul, shift) { + var len = num.length + shift; + var i; + + this._expand(len); + + var w; + var carry = 0; + for (i = 0; i < num.length; i++) { + w = (this.words[i + shift] | 0) + carry; + var right = (num.words[i] | 0) * mul; + w -= right & 0x3ffffff; + carry = (w >> 26) - ((right / 0x4000000) | 0); + this.words[i + shift] = w & 0x3ffffff; + } + for (; i < this.length - shift; i++) { + w = (this.words[i + shift] | 0) + carry; + carry = w >> 26; + this.words[i + shift] = w & 0x3ffffff; + } + + if (carry === 0) return this._strip(); + + // Subtraction overflow + assert(carry === -1); + carry = 0; + for (i = 0; i < this.length; i++) { + w = -(this.words[i] | 0) + carry; + carry = w >> 26; + this.words[i] = w & 0x3ffffff; + } + this.negative = 1; + + return this._strip(); + }; + + BN.prototype._wordDiv = function _wordDiv(num, mode) { + var shift = this.length - num.length; + + var a = this.clone(); + var b = num; + + // Normalize + var bhi = b.words[b.length - 1] | 0; + var bhiBits = this._countBits(bhi); + shift = 26 - bhiBits; + if (shift !== 0) { + b = b.ushln(shift); + a.iushln(shift); + bhi = b.words[b.length - 1] | 0; + } + + // Initialize quotient + var m = a.length - b.length; + var q; + + if (mode !== 'mod') { + q = new BN(null); + q.length = m + 1; + q.words = new Array(q.length); + for (var i = 0; i < q.length; i++) { + q.words[i] = 0; + } + } + + var diff = a.clone()._ishlnsubmul(b, 1, m); + if (diff.negative === 0) { + a = diff; + if (q) { + q.words[m] = 1; + } + } + + for (var j = m - 1; j >= 0; j--) { + var qj = (a.words[b.length + j] | 0) * 0x4000000 + (a.words[b.length + j - 1] | 0); + + // NOTE: (qj / bhi) is (0x3ffffff * 0x4000000 + 0x3ffffff) / 0x2000000 max + // (0x7ffffff) + qj = Math.min((qj / bhi) | 0, 0x3ffffff); + + a._ishlnsubmul(b, qj, j); + while (a.negative !== 0) { + qj--; + a.negative = 0; + a._ishlnsubmul(b, 1, j); + if (!a.isZero()) { + a.negative ^= 1; + } + } + if (q) { + q.words[j] = qj; + } + } + if (q) { + q._strip(); + } + a._strip(); + + // Denormalize + if (mode !== 'div' && shift !== 0) { + a.iushrn(shift); + } + + return { + div: q || null, + mod: a + }; + }; + + // NOTE: 1) `mode` can be set to `mod` to request mod only, + // to `div` to request div only, or be absent to + // request both div & mod + // 2) `positive` is true if unsigned mod is requested + BN.prototype.divmod = function divmod(num, mode, positive) { + assert(!num.isZero()); + + if (this.isZero()) { + return { + div: new BN(0), + mod: new BN(0) + }; + } + + var div, mod, res; + if (this.negative !== 0 && num.negative === 0) { + res = this.neg().divmod(num, mode); + + if (mode !== 'mod') { + div = res.div.neg(); + } + + if (mode !== 'div') { + mod = res.mod.neg(); + if (positive && mod.negative !== 0) { + mod.iadd(num); + } + } + + return { + div: div, + mod: mod + }; + } + + if (this.negative === 0 && num.negative !== 0) { + res = this.divmod(num.neg(), mode); + + if (mode !== 'mod') { + div = res.div.neg(); + } + + return { + div: div, + mod: res.mod + }; + } + + if ((this.negative & num.negative) !== 0) { + res = this.neg().divmod(num.neg(), mode); + + if (mode !== 'div') { + mod = res.mod.neg(); + if (positive && mod.negative !== 0) { + mod.isub(num); + } + } + + return { + div: res.div, + mod: mod + }; + } + + // Both numbers are positive at this point + + // Strip both numbers to approximate shift value + if (num.length > this.length || this.cmp(num) < 0) { + return { + div: new BN(0), + mod: this + }; + } + + // Very short reduction + if (num.length === 1) { + if (mode === 'div') { + return { + div: this.divn(num.words[0]), + mod: null + }; + } + + if (mode === 'mod') { + return { + div: null, + mod: new BN(this.modrn(num.words[0])) + }; + } + + return { + div: this.divn(num.words[0]), + mod: new BN(this.modrn(num.words[0])) + }; + } + + return this._wordDiv(num, mode); + }; + + // Find `this` / `num` + BN.prototype.div = function div(num) { + return this.divmod(num, 'div', false).div; + }; + + // Find `this` % `num` + BN.prototype.mod = function mod(num) { + return this.divmod(num, 'mod', false).mod; + }; + + BN.prototype.umod = function umod(num) { + return this.divmod(num, 'mod', true).mod; + }; + + // Find Round(`this` / `num`) + BN.prototype.divRound = function divRound(num) { + var dm = this.divmod(num); + + // Fast case - exact division + if (dm.mod.isZero()) return dm.div; + + var mod = dm.div.negative !== 0 ? dm.mod.isub(num) : dm.mod; + + var half = num.ushrn(1); + var r2 = num.andln(1); + var cmp = mod.cmp(half); + + // Round down + if (cmp < 0 || (r2 === 1 && cmp === 0)) return dm.div; + + // Round up + return dm.div.negative !== 0 ? dm.div.isubn(1) : dm.div.iaddn(1); + }; + + BN.prototype.modrn = function modrn(num) { + var isNegNum = num < 0; + if (isNegNum) num = -num; + + assert(num <= 0x3ffffff); + var p = (1 << 26) % num; + + var acc = 0; + for (var i = this.length - 1; i >= 0; i--) { + acc = (p * acc + (this.words[i] | 0)) % num; + } + + return isNegNum ? -acc : acc; + }; + + // WARNING: DEPRECATED + BN.prototype.modn = function modn(num) { + return this.modrn(num); + }; + + // In-place division by number + BN.prototype.idivn = function idivn(num) { + var isNegNum = num < 0; + if (isNegNum) num = -num; + + assert(num <= 0x3ffffff); + + var carry = 0; + for (var i = this.length - 1; i >= 0; i--) { + var w = (this.words[i] | 0) + carry * 0x4000000; + this.words[i] = (w / num) | 0; + carry = w % num; + } + + this._strip(); + return isNegNum ? this.ineg() : this; + }; + + BN.prototype.divn = function divn(num) { + return this.clone().idivn(num); + }; + + BN.prototype.egcd = function egcd(p) { + assert(p.negative === 0); + assert(!p.isZero()); + + var x = this; + var y = p.clone(); + + if (x.negative !== 0) { + x = x.umod(p); + } else { + x = x.clone(); + } + + // A * x + B * y = x + var A = new BN(1); + var B = new BN(0); + + // C * x + D * y = y + var C = new BN(0); + var D = new BN(1); + + var g = 0; + + while (x.isEven() && y.isEven()) { + x.iushrn(1); + y.iushrn(1); + ++g; + } + + var yp = y.clone(); + var xp = x.clone(); + + while (!x.isZero()) { + for (var i = 0, im = 1; (x.words[0] & im) === 0 && i < 26; ++i, im <<= 1); + if (i > 0) { + x.iushrn(i); + while (i-- > 0) { + if (A.isOdd() || B.isOdd()) { + A.iadd(yp); + B.isub(xp); + } + + A.iushrn(1); + B.iushrn(1); + } + } + + for (var j = 0, jm = 1; (y.words[0] & jm) === 0 && j < 26; ++j, jm <<= 1); + if (j > 0) { + y.iushrn(j); + while (j-- > 0) { + if (C.isOdd() || D.isOdd()) { + C.iadd(yp); + D.isub(xp); + } + + C.iushrn(1); + D.iushrn(1); + } + } + + if (x.cmp(y) >= 0) { + x.isub(y); + A.isub(C); + B.isub(D); + } else { + y.isub(x); + C.isub(A); + D.isub(B); + } + } + + return { + a: C, + b: D, + gcd: y.iushln(g) + }; + }; + + // This is reduced incarnation of the binary EEA + // above, designated to invert members of the + // _prime_ fields F(p) at a maximal speed + BN.prototype._invmp = function _invmp(p) { + assert(p.negative === 0); + assert(!p.isZero()); + + var a = this; + var b = p.clone(); + + if (a.negative !== 0) { + a = a.umod(p); + } else { + a = a.clone(); + } + + var x1 = new BN(1); + var x2 = new BN(0); + + var delta = b.clone(); + + while (a.cmpn(1) > 0 && b.cmpn(1) > 0) { + for (var i = 0, im = 1; (a.words[0] & im) === 0 && i < 26; ++i, im <<= 1); + if (i > 0) { + a.iushrn(i); + while (i-- > 0) { + if (x1.isOdd()) { + x1.iadd(delta); + } + + x1.iushrn(1); + } + } + + for (var j = 0, jm = 1; (b.words[0] & jm) === 0 && j < 26; ++j, jm <<= 1); + if (j > 0) { + b.iushrn(j); + while (j-- > 0) { + if (x2.isOdd()) { + x2.iadd(delta); + } + + x2.iushrn(1); + } + } + + if (a.cmp(b) >= 0) { + a.isub(b); + x1.isub(x2); + } else { + b.isub(a); + x2.isub(x1); + } + } + + var res; + if (a.cmpn(1) === 0) { + res = x1; + } else { + res = x2; + } + + if (res.cmpn(0) < 0) { + res.iadd(p); + } + + return res; + }; + + BN.prototype.gcd = function gcd(num) { + if (this.isZero()) return num.abs(); + if (num.isZero()) return this.abs(); + + var a = this.clone(); + var b = num.clone(); + a.negative = 0; + b.negative = 0; + + // Remove common factor of two + for (var shift = 0; a.isEven() && b.isEven(); shift++) { + a.iushrn(1); + b.iushrn(1); + } + + do { + while (a.isEven()) { + a.iushrn(1); + } + while (b.isEven()) { + b.iushrn(1); + } + + var r = a.cmp(b); + if (r < 0) { + // Swap `a` and `b` to make `a` always bigger than `b` + var t = a; + a = b; + b = t; + } else if (r === 0 || b.cmpn(1) === 0) { + break; + } + + a.isub(b); + } while (true); + + return b.iushln(shift); + }; + + // Invert number in the field F(num) + BN.prototype.invm = function invm(num) { + return this.egcd(num).a.umod(num); + }; + + BN.prototype.isEven = function isEven() { + return (this.words[0] & 1) === 0; + }; + + BN.prototype.isOdd = function isOdd() { + return (this.words[0] & 1) === 1; + }; + + // And first word and num + BN.prototype.andln = function andln(num) { + return this.words[0] & num; + }; + + // Increment at the bit position in-line + BN.prototype.bincn = function bincn(bit) { + assert(typeof bit === 'number'); + var r = bit % 26; + var s = (bit - r) / 26; + var q = 1 << r; + + // Fast case: bit is much higher than all existing words + if (this.length <= s) { + this._expand(s + 1); + this.words[s] |= q; + return this; + } + + // Add bit and propagate, if needed + var carry = q; + for (var i = s; carry !== 0 && i < this.length; i++) { + var w = this.words[i] | 0; + w += carry; + carry = w >>> 26; + w &= 0x3ffffff; + this.words[i] = w; + } + if (carry !== 0) { + this.words[i] = carry; + this.length++; + } + return this; + }; + + BN.prototype.isZero = function isZero() { + return this.length === 1 && this.words[0] === 0; + }; + + BN.prototype.cmpn = function cmpn(num) { + var negative = num < 0; + + if (this.negative !== 0 && !negative) return -1; + if (this.negative === 0 && negative) return 1; + + this._strip(); + + var res; + if (this.length > 1) { + res = 1; + } else { + if (negative) { + num = -num; + } + + assert(num <= 0x3ffffff, 'Number is too big'); + + var w = this.words[0] | 0; + res = w === num ? 0 : w < num ? -1 : 1; + } + if (this.negative !== 0) return -res | 0; + return res; + }; + + // Compare two numbers and return: + // 1 - if `this` > `num` + // 0 - if `this` == `num` + // -1 - if `this` < `num` + BN.prototype.cmp = function cmp(num) { + if (this.negative !== 0 && num.negative === 0) return -1; + if (this.negative === 0 && num.negative !== 0) return 1; + + var res = this.ucmp(num); + if (this.negative !== 0) return -res | 0; + return res; + }; + + // Unsigned comparison + BN.prototype.ucmp = function ucmp(num) { + // At this point both numbers have the same sign + if (this.length > num.length) return 1; + if (this.length < num.length) return -1; + + var res = 0; + for (var i = this.length - 1; i >= 0; i--) { + var a = this.words[i] | 0; + var b = num.words[i] | 0; + + if (a === b) continue; + if (a < b) { + res = -1; + } else if (a > b) { + res = 1; + } + break; + } + return res; + }; + + BN.prototype.gtn = function gtn(num) { + return this.cmpn(num) === 1; + }; + + BN.prototype.gt = function gt(num) { + return this.cmp(num) === 1; + }; + + BN.prototype.gten = function gten(num) { + return this.cmpn(num) >= 0; + }; + + BN.prototype.gte = function gte(num) { + return this.cmp(num) >= 0; + }; + + BN.prototype.ltn = function ltn(num) { + return this.cmpn(num) === -1; + }; + + BN.prototype.lt = function lt(num) { + return this.cmp(num) === -1; + }; + + BN.prototype.lten = function lten(num) { + return this.cmpn(num) <= 0; + }; + + BN.prototype.lte = function lte(num) { + return this.cmp(num) <= 0; + }; + + BN.prototype.eqn = function eqn(num) { + return this.cmpn(num) === 0; + }; + + BN.prototype.eq = function eq(num) { + return this.cmp(num) === 0; + }; + + // + // A reduce context, could be using montgomery or something better, depending + // on the `m` itself. + // + BN.red = function red(num) { + return new Red(num); + }; + + BN.prototype.toRed = function toRed(ctx) { + assert(!this.red, 'Already a number in reduction context'); + assert(this.negative === 0, 'red works only with positives'); + return ctx.convertTo(this)._forceRed(ctx); + }; + + BN.prototype.fromRed = function fromRed() { + assert(this.red, 'fromRed works only with numbers in reduction context'); + return this.red.convertFrom(this); + }; + + BN.prototype._forceRed = function _forceRed(ctx) { + this.red = ctx; + return this; + }; + + BN.prototype.forceRed = function forceRed(ctx) { + assert(!this.red, 'Already a number in reduction context'); + return this._forceRed(ctx); + }; + + BN.prototype.redAdd = function redAdd(num) { + assert(this.red, 'redAdd works only with red numbers'); + return this.red.add(this, num); + }; + + BN.prototype.redIAdd = function redIAdd(num) { + assert(this.red, 'redIAdd works only with red numbers'); + return this.red.iadd(this, num); + }; + + BN.prototype.redSub = function redSub(num) { + assert(this.red, 'redSub works only with red numbers'); + return this.red.sub(this, num); + }; + + BN.prototype.redISub = function redISub(num) { + assert(this.red, 'redISub works only with red numbers'); + return this.red.isub(this, num); + }; + + BN.prototype.redShl = function redShl(num) { + assert(this.red, 'redShl works only with red numbers'); + return this.red.shl(this, num); + }; + + BN.prototype.redMul = function redMul(num) { + assert(this.red, 'redMul works only with red numbers'); + this.red._verify2(this, num); + return this.red.mul(this, num); + }; + + BN.prototype.redIMul = function redIMul(num) { + assert(this.red, 'redMul works only with red numbers'); + this.red._verify2(this, num); + return this.red.imul(this, num); + }; + + BN.prototype.redSqr = function redSqr() { + assert(this.red, 'redSqr works only with red numbers'); + this.red._verify1(this); + return this.red.sqr(this); + }; + + BN.prototype.redISqr = function redISqr() { + assert(this.red, 'redISqr works only with red numbers'); + this.red._verify1(this); + return this.red.isqr(this); + }; + + // Square root over p + BN.prototype.redSqrt = function redSqrt() { + assert(this.red, 'redSqrt works only with red numbers'); + this.red._verify1(this); + return this.red.sqrt(this); + }; + + BN.prototype.redInvm = function redInvm() { + assert(this.red, 'redInvm works only with red numbers'); + this.red._verify1(this); + return this.red.invm(this); + }; + + // Return negative clone of `this` % `red modulo` + BN.prototype.redNeg = function redNeg() { + assert(this.red, 'redNeg works only with red numbers'); + this.red._verify1(this); + return this.red.neg(this); + }; + + BN.prototype.redPow = function redPow(num) { + assert(this.red && !num.red, 'redPow(normalNum)'); + this.red._verify1(this); + return this.red.pow(this, num); + }; + + // Prime numbers with efficient reduction + var primes = { + k256: null, + p224: null, + p192: null, + p25519: null + }; + + // Pseudo-Mersenne prime + function MPrime(name, p) { + // P = 2 ^ N - K + this.name = name; + this.p = new BN(p, 16); + this.n = this.p.bitLength(); + this.k = new BN(1).iushln(this.n).isub(this.p); + + this.tmp = this._tmp(); + } - /** - * Less than with a constant. - * @method clt - * @param {number} cst - the constant to compare with. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this < cst, and 0 otherwise. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.clt = function (cst, op_id) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (<)'); - } + MPrime.prototype._tmp = function _tmp() { + var tmp = new BN(null); + tmp.words = new Array(Math.ceil(this.n / 13)); + return tmp; + }; + + MPrime.prototype.ireduce = function ireduce(num) { + // Assumes that `num` is less than `P^2` + // num = HI * (2 ^ N - K) + HI * K + LO = HI * K + LO (mod P) + var r = num; + var rlen; + + do { + this.split(r, this.tmp); + r = this.imulK(r); + r = r.iadd(this.tmp); + rlen = r.bitLength(); + } while (rlen > this.n); + + var cmp = rlen < this.n ? -1 : r.ucmp(this.p); + if (cmp === 0) { + r.words[0] = 0; + r.length = 1; + } else if (cmp > 0) { + r.isub(this.p); + } else { + if (r.strip !== undefined) { + // r is a BN v4 instance + r.strip(); + } else { + // r is a BN v5 instance + r._strip(); + } + } + + return r; + }; + + MPrime.prototype.split = function split(input, out) { + input.iushrn(this.n, 0, out); + }; + + MPrime.prototype.imulK = function imulK(num) { + return num.imul(this.k); + }; + + function K256() { + MPrime.call(this, 'k256', 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff fffffffe fffffc2f'); + } + inherits(K256, MPrime); + + K256.prototype.split = function split(input, output) { + // 256 = 9 * 26 + 22 + var mask = 0x3fffff; + + var outLen = Math.min(input.length, 9); + for (var i = 0; i < outLen; i++) { + output.words[i] = input.words[i]; + } + output.length = outLen; + + if (input.length <= 9) { + input.words[0] = 0; + input.length = 1; + return; + } + + // Shift by 9 limbs + var prev = input.words[9]; + output.words[output.length++] = prev & mask; + + for (i = 10; i < input.length; i++) { + var next = input.words[i] | 0; + input.words[i - 10] = ((next & mask) << 4) | (prev >>> 22); + prev = next; + } + prev >>>= 22; + input.words[i - 10] = prev; + if (prev === 0 && input.length > 10) { + input.length -= 10; + } else { + input.length -= 9; + } + }; + + K256.prototype.imulK = function imulK(num) { + // K = 0x1000003d1 = [ 0x40, 0x3d1 ] + num.words[num.length] = 0; + num.words[num.length + 1] = 0; + num.length += 2; + + // bounded at: 0x40 * 0x3ffffff + 0x3d0 = 0x100000390 + var lo = 0; + for (var i = 0; i < num.length; i++) { + var w = num.words[i] | 0; + lo += w * 0x3d1; + num.words[i] = lo & 0x3ffffff; + lo = w * 0x40 + ((lo / 0x4000000) | 0); + } + + // Fast length reduction + if (num.words[num.length - 1] === 0) { + num.length--; + if (num.words[num.length - 1] === 0) { + num.length--; + } + } + return num; + }; + + function P224() { + MPrime.call(this, 'p224', 'ffffffff ffffffff ffffffff ffffffff 00000000 00000000 00000001'); + } + inherits(P224, MPrime); - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('clt', this.holders); - } + function P192() { + MPrime.call(this, 'p192', 'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff'); + } + inherits(P192, MPrime); - var final_deferred = new this.jiff.helpers.Deferred(); - var final_promise = final_deferred.promise; - var result = new this.jiff.SecretShare(final_promise, this.holders, this.threshold, this.Zp); + function P25519() { + // 2 ^ 255 - 19 + MPrime.call(this, '25519', '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed'); + } + inherits(P25519, MPrime); + + P25519.prototype.imulK = function imulK(num) { + // K = 0x13 + var carry = 0; + for (var i = 0; i < num.length; i++) { + var hi = (num.words[i] | 0) * 0x13 + carry; + var lo = hi & 0x3ffffff; + hi >>>= 26; + + num.words[i] = lo; + carry = hi; + } + if (carry !== 0) { + num.words[num.length++] = carry; + } + return num; + }; + + // Exported mostly for testing purposes, use plain name instead + BN._prime = function prime(name) { + // Cached version of prime + if (primes[name]) return primes[name]; + + var prime; + if (name === 'k256') { + prime = new K256(); + } else if (name === 'p224') { + prime = new P224(); + } else if (name === 'p192') { + prime = new P192(); + } else if (name === 'p25519') { + prime = new P25519(); + } else { + throw new Error('Unknown prime ' + name); + } + primes[name] = prime; + + return prime; + }; + + // + // Base reduction engine + // + function Red(m) { + if (typeof m === 'string') { + var prime = BN._prime(m); + this.m = prime.p; + this.prime = prime; + } else { + assert(m.gtn(1), 'modulus must be greater than 1'); + this.m = m; + this.prime = null; + } + } - var w = this.ilt_halfprime(op_id + ':halfprime:1'); + Red.prototype._verify1 = function _verify1(a) { + assert(a.negative === 0, 'red works only with positives'); + assert(a.red, 'red works only with red numbers'); + }; + + Red.prototype._verify2 = function _verify2(a, b) { + assert((a.negative | b.negative) === 0, 'red works only with positives'); + assert(a.red && a.red === b.red, 'red works only with red numbers'); + }; + + Red.prototype.imod = function imod(a) { + if (this.prime) return this.prime.ireduce(a)._forceRed(this); + + move(a, a.umod(this.m)._forceRed(this)); + return a; + }; + + Red.prototype.neg = function neg(a) { + if (a.isZero()) { + return a.clone(); + } + + return this.m.sub(a)._forceRed(this); + }; + + Red.prototype.add = function add(a, b) { + this._verify2(a, b); + + var res = a.add(b); + if (res.cmp(this.m) >= 0) { + res.isub(this.m); + } + return res._forceRed(this); + }; + + Red.prototype.iadd = function iadd(a, b) { + this._verify2(a, b); + + var res = a.iadd(b); + if (res.cmp(this.m) >= 0) { + res.isub(this.m); + } + return res; + }; + + Red.prototype.sub = function sub(a, b) { + this._verify2(a, b); + + var res = a.sub(b); + if (res.cmpn(0) < 0) { + res.iadd(this.m); + } + return res._forceRed(this); + }; + + Red.prototype.isub = function isub(a, b) { + this._verify2(a, b); + + var res = a.isub(b); + if (res.cmpn(0) < 0) { + res.iadd(this.m); + } + return res; + }; + + Red.prototype.shl = function shl(a, num) { + this._verify1(a); + return this.imod(a.ushln(num)); + }; + + Red.prototype.imul = function imul(a, b) { + this._verify2(a, b); + return this.imod(a.imul(b)); + }; + + Red.prototype.mul = function mul(a, b) { + this._verify2(a, b); + return this.imod(a.mul(b)); + }; + + Red.prototype.isqr = function isqr(a) { + return this.imul(a, a.clone()); + }; + + Red.prototype.sqr = function sqr(a) { + return this.mul(a, a); + }; + + Red.prototype.sqrt = function sqrt(a) { + if (a.isZero()) return a.clone(); + + var mod3 = this.m.andln(3); + assert(mod3 % 2 === 1); + + // Fast case + if (mod3 === 3) { + var pow = this.m.add(new BN(1)).iushrn(2); + return this.pow(a, pow); + } + + // Tonelli-Shanks algorithm (Totally unoptimized and slow) + // + // Find Q and S, that Q * 2 ^ S = (P - 1) + var q = this.m.subn(1); + var s = 0; + while (!q.isZero() && q.andln(1) === 0) { + s++; + q.iushrn(1); + } + assert(!q.isZero()); + + var one = new BN(1).toRed(this); + var nOne = one.redNeg(); + + // Find quadratic non-residue + // NOTE: Max is such because of generalized Riemann hypothesis. + var lpow = this.m.subn(1).iushrn(1); + var z = this.m.bitLength(); + z = new BN(2 * z * z).toRed(this); + + while (this.pow(z, lpow).cmp(nOne) !== 0) { + z.redIAdd(nOne); + } + + var c = this.pow(z, q); + var r = this.pow(a, q.addn(1).iushrn(1)); + var t = this.pow(a, q); + var m = s; + while (t.cmp(one) !== 0) { + var tmp = t; + for (var i = 0; tmp.cmp(one) !== 0; i++) { + tmp = tmp.redSqr(); + } + assert(i < m); + var b = this.pow(c, new BN(1).iushln(m - i - 1)); + + r = r.redMul(b); + c = b.redSqr(); + t = t.redMul(c); + m = i; + } + + return r; + }; + + Red.prototype.invm = function invm(a) { + var inv = a._invmp(this.m); + if (inv.negative !== 0) { + inv.negative = 0; + return this.imod(inv).redNeg(); + } else { + return this.imod(inv); + } + }; + + Red.prototype.pow = function pow(a, num) { + if (num.isZero()) return new BN(1).toRed(this); + if (num.cmpn(1) === 0) return a.clone(); + + var windowSize = 4; + var wnd = new Array(1 << windowSize); + wnd[0] = new BN(1).toRed(this); + wnd[1] = a; + for (var i = 2; i < wnd.length; i++) { + wnd[i] = this.mul(wnd[i - 1], a); + } + + var res = wnd[0]; + var current = 0; + var currentLen = 0; + var start = num.bitLength() % 26; + if (start === 0) { + start = 26; + } + + for (i = num.length - 1; i >= 0; i--) { + var word = num.words[i]; + for (var j = start - 1; j >= 0; j--) { + var bit = (word >> j) & 1; + if (res !== wnd[0]) { + res = this.sqr(res); + } + + if (bit === 0 && current === 0) { + currentLen = 0; + continue; + } + + current <<= 1; + current |= bit; + currentLen++; + if (currentLen !== windowSize && (i !== 0 || j !== 0)) continue; + + res = this.mul(res, wnd[current]); + currentLen = 0; + current = 0; + } + start = 26; + } + + return res; + }; + + Red.prototype.convertTo = function convertTo(num) { + var r = num.umod(this.m); + + return r === num ? r.clone() : r; + }; + + Red.prototype.convertFrom = function convertFrom(num) { + var res = num.clone(); + res.red = null; + return res; + }; + + // + // Montgomery method engine + // + + BN.mont = function mont(num) { + return new Mont(num); + }; + + function Mont(m) { + Red.call(this, m); + + this.shift = this.m.bitLength(); + if (this.shift % 26 !== 0) { + this.shift += 26 - (this.shift % 26); + } + + this.r = new BN(1).iushln(this.shift); + this.r2 = this.imod(this.r.sqr()); + this.rinv = this.r._invmp(this.m); + + this.minv = this.rinv.mul(this.r).isubn(1).div(this.m); + this.minv = this.minv.umod(this.r); + this.minv = this.r.sub(this.minv); + } + inherits(Mont, Red); + + Mont.prototype.convertTo = function convertTo(num) { + return this.imod(num.ushln(this.shift)); + }; + + Mont.prototype.convertFrom = function convertFrom(num) { + var r = this.imod(num.mul(this.rinv)); + r.red = null; + return r; + }; + + Mont.prototype.imul = function imul(a, b) { + if (a.isZero() || b.isZero()) { + a.words[0] = 0; + a.length = 1; + return a; + } + + var t = a.imul(b); + var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m); + var u = t.isub(c).iushrn(this.shift); + var res = u; + + if (u.cmp(this.m) >= 0) { + res = u.isub(this.m); + } else if (u.cmpn(0) < 0) { + res = u.iadd(this.m); + } + + return res._forceRed(this); + }; + + Mont.prototype.mul = function mul(a, b) { + if (a.isZero() || b.isZero()) return new BN(0)._forceRed(this); + + var t = a.mul(b); + var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m); + var u = t.isub(c).iushrn(this.shift); + var res = u; + if (u.cmp(this.m) >= 0) { + res = u.isub(this.m); + } else if (u.cmpn(0) < 0) { + res = u.iadd(this.m); + } + + return res._forceRed(this); + }; + + Mont.prototype.invm = function invm(a) { + // (AR)^-1 * R^2 = (A^-1 * R^-1) * R^2 = A^-1 * R + var res = this.imod(a._invmp(this.m).mul(this.r2)); + return res._forceRed(this); + }; + })(typeof module === 'undefined' || module, this); + }, + { buffer: 77 } + ], + 76: [ + function (require, module, exports) { + var r; - var self = this; - w.wThen(function () { - var x = self.jiff.share_helpers['<'](cst, self.jiff.share_helpers['/'](self.Zp, 2)) ? 1 : 0; - var y = self.icsub(cst).ilt_halfprime(op_id + ':halfprime:2'); - y.wThen(function () { - var xy = y.icmult(x); - var answer = y.icmult(-1).icadd(1 - x).isadd(xy).isadd(w.ismult(y.icadd(x).issub(xy.icmult(2)), op_id + ':smult1')); - answer.wThen(final_deferred.resolve); - }); - }); + module.exports = function rand(len) { + if (!r) r = new Rand(null); - return result; - }; + return r.generate(len); + }; - /** - * Equality test with two shares. - * @method seq - * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to compare with. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 1 if this = o, and 0 otherwise. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.seq = function (o, op_id) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (==)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (==)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (==)'); - } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('seq', this.holders); - } + function Rand(rand) { + this.rand = rand; + } + module.exports.Rand = Rand; + + Rand.prototype.generate = function generate(len) { + return this._rand(len); + }; + + // Emulate crypto API using randy + Rand.prototype._rand = function _rand(n) { + if (this.rand.getBytes) return this.rand.getBytes(n); + + var res = new Uint8Array(n); + for (var i = 0; i < res.length; i++) res[i] = this.rand.getByte(); + return res; + }; + + if (typeof self === 'object') { + if (self.crypto && self.crypto.getRandomValues) { + // Modern browsers + Rand.prototype._rand = function _rand(n) { + var arr = new Uint8Array(n); + self.crypto.getRandomValues(arr); + return arr; + }; + } else if (self.msCrypto && self.msCrypto.getRandomValues) { + // IE + Rand.prototype._rand = function _rand(n) { + var arr = new Uint8Array(n); + self.msCrypto.getRandomValues(arr); + return arr; + }; + + // Safari's WebWorkers do not have `crypto` + } else if (typeof window === 'object') { + // Old junk + Rand.prototype._rand = function () { + throw new Error('Not implemented yet'); + }; + } + } else { + // Node.js or Web worker with no crypto support + try { + var crypto = require('crypto'); + if (typeof crypto.randomBytes !== 'function') throw new Error('Not supported'); + + Rand.prototype._rand = function _rand(n) { + return crypto.randomBytes(n); + }; + } catch (e) {} + } + }, + { crypto: 77 } + ], + 77: [function (require, module, exports) {}, {}], + 78: [ + function (require, module, exports) { + // based on the aes implimentation in triple sec + // https://github.com/keybase/triplesec + // which is in turn based on the one from crypto-js + // https://code.google.com/p/crypto-js/ - return this.isneq(o, op_id).inot(); - }; + var Buffer = require('safe-buffer').Buffer; - /** - * Unequality test with two shares. - * @method sneq - * @param {module:jiff-client~JIFFClient#SecretShare} o - the share to compare with. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 0 if this = o, and 1 otherwise. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.sneq = function (o, op_id) { - if (!(o.jiff === this.jiff)) { - throw new Error('shares do not belong to the same instance (!=)'); - } - if (!this.jiff.helpers.Zp_equals(this, o)) { - throw new Error('shares must belong to the same field (!=)'); - } - if (!this.jiff.helpers.array_equals(this.holders, o.holders)) { - throw new Error('shares must be held by the same parties (!=)'); - } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('sneq', this.holders); - } + function asUInt32Array(buf) { + if (!Buffer.isBuffer(buf)) buf = Buffer.from(buf); - return this.issub(o).icpow(this.jiff.share_helpers['-'](this.Zp, 1), op_id + ':cpow'); - }; + var len = (buf.length / 4) | 0; + var out = new Array(len); - /** - * Equality test with a constant. - * @method ceq - * @param {number} cst - the constant to compare with. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 0 if this = o, and 1 otherwise. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.ceq = function (cst, op_id) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (==)'); - } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('ceq', this.holders); - } + for (var i = 0; i < len; i++) { + out[i] = buf.readUInt32BE(i * 4); + } - return this.icneq(cst, op_id).inot(); - }; + return out; + } - /** - * Unequality test with a constant. - * @method cneq - * @param {number} cst - the constant to compare with. - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result, the final result is 0 if this = o, and 1 otherwise. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.cneq = function (cst, op_id) { - if (!(this.isConstant(cst))) { - throw new Error('parameter should be a number (!=)'); - } - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('cneq', this.holders); - } + function scrubVec(v) { + for (var i = 0; i < v.length; v++) { + v[i] = 0; + } + } - return this.icsub(cst).icpow(this.jiff.share_helpers['-'](this.Zp, 1), op_id + ':cpow'); - }; + function cryptBlock(M, keySchedule, SUB_MIX, SBOX, nRounds) { + var SUB_MIX0 = SUB_MIX[0]; + var SUB_MIX1 = SUB_MIX[1]; + var SUB_MIX2 = SUB_MIX[2]; + var SUB_MIX3 = SUB_MIX[3]; + + var s0 = M[0] ^ keySchedule[0]; + var s1 = M[1] ^ keySchedule[1]; + var s2 = M[2] ^ keySchedule[2]; + var s3 = M[3] ^ keySchedule[3]; + var t0, t1, t2, t3; + var ksRow = 4; + + for (var round = 1; round < nRounds; round++) { + t0 = SUB_MIX0[s0 >>> 24] ^ SUB_MIX1[(s1 >>> 16) & 0xff] ^ SUB_MIX2[(s2 >>> 8) & 0xff] ^ SUB_MIX3[s3 & 0xff] ^ keySchedule[ksRow++]; + t1 = SUB_MIX0[s1 >>> 24] ^ SUB_MIX1[(s2 >>> 16) & 0xff] ^ SUB_MIX2[(s3 >>> 8) & 0xff] ^ SUB_MIX3[s0 & 0xff] ^ keySchedule[ksRow++]; + t2 = SUB_MIX0[s2 >>> 24] ^ SUB_MIX1[(s3 >>> 16) & 0xff] ^ SUB_MIX2[(s0 >>> 8) & 0xff] ^ SUB_MIX3[s1 & 0xff] ^ keySchedule[ksRow++]; + t3 = SUB_MIX0[s3 >>> 24] ^ SUB_MIX1[(s0 >>> 16) & 0xff] ^ SUB_MIX2[(s1 >>> 8) & 0xff] ^ SUB_MIX3[s2 & 0xff] ^ keySchedule[ksRow++]; + s0 = t0; + s1 = t1; + s2 = t2; + s3 = t3; + } - /** - * Checks whether the share is less than half the field size. - * @method lt_halfprime - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @return {module:jiff-client~JIFFClient#SecretShare} this party's share of the result. - */ - SecretShare.prototype.lt_halfprime = function (op_id) { - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('lt_halfprime', this.holders); - } + t0 = ((SBOX[s0 >>> 24] << 24) | (SBOX[(s1 >>> 16) & 0xff] << 16) | (SBOX[(s2 >>> 8) & 0xff] << 8) | SBOX[s3 & 0xff]) ^ keySchedule[ksRow++]; + t1 = ((SBOX[s1 >>> 24] << 24) | (SBOX[(s2 >>> 16) & 0xff] << 16) | (SBOX[(s3 >>> 8) & 0xff] << 8) | SBOX[s0 & 0xff]) ^ keySchedule[ksRow++]; + t2 = ((SBOX[s2 >>> 24] << 24) | (SBOX[(s3 >>> 16) & 0xff] << 16) | (SBOX[(s0 >>> 8) & 0xff] << 8) | SBOX[s1 & 0xff]) ^ keySchedule[ksRow++]; + t3 = ((SBOX[s3 >>> 24] << 24) | (SBOX[(s0 >>> 16) & 0xff] << 16) | (SBOX[(s1 >>> 8) & 0xff] << 8) | SBOX[s2 & 0xff]) ^ keySchedule[ksRow++]; + t0 = t0 >>> 0; + t1 = t1 >>> 0; + t2 = t2 >>> 0; + t3 = t3 >>> 0; - // if share is even, then this is less than half the prime, otherwise, share is greater than half the prime - var share = this.icmult(2); + return [t0, t1, t2, t3]; + } - // to check if share is even, we will use pre-shared bits as some form of a bit mask - var bitLength = this.jiff.share_helpers['ceil'](this.jiff.helpers.bLog(share.Zp, 2)); + // AES constants + var RCON = [0x00, 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36]; + var G = (function () { + // Compute double table + var d = new Array(256); + for (var j = 0; j < 256; j++) { + if (j < 128) { + d[j] = j << 1; + } else { + d[j] = (j << 1) ^ 0x11b; + } + } - // Create result share - var final_deferred = new this.jiff.helpers.Deferred(); - var final_promise = final_deferred.promise; - var result = new this.jiff.SecretShare(final_promise, this.holders, this.threshold, this.Zp); + var SBOX = []; + var INV_SBOX = []; + var SUB_MIX = [[], [], [], []]; + var INV_SUB_MIX = [[], [], [], []]; + + // Walk GF(2^8) + var x = 0; + var xi = 0; + for (var i = 0; i < 256; ++i) { + // Compute sbox + var sx = xi ^ (xi << 1) ^ (xi << 2) ^ (xi << 3) ^ (xi << 4); + sx = (sx >>> 8) ^ (sx & 0xff) ^ 0x63; + SBOX[x] = sx; + INV_SBOX[sx] = x; + + // Compute multiplication + var x2 = d[x]; + var x4 = d[x2]; + var x8 = d[x4]; + + // Compute sub bytes, mix columns tables + var t = (d[sx] * 0x101) ^ (sx * 0x1010100); + SUB_MIX[0][x] = (t << 24) | (t >>> 8); + SUB_MIX[1][x] = (t << 16) | (t >>> 16); + SUB_MIX[2][x] = (t << 8) | (t >>> 24); + SUB_MIX[3][x] = t; + + // Compute inv sub bytes, inv mix columns tables + t = (x8 * 0x1010101) ^ (x4 * 0x10001) ^ (x2 * 0x101) ^ (x * 0x1010100); + INV_SUB_MIX[0][sx] = (t << 24) | (t >>> 8); + INV_SUB_MIX[1][sx] = (t << 16) | (t >>> 16); + INV_SUB_MIX[2][sx] = (t << 8) | (t >>> 24); + INV_SUB_MIX[3][sx] = t; + + if (x === 0) { + x = xi = 1; + } else { + x = x2 ^ d[d[d[x8 ^ x2]]]; + xi ^= d[d[xi]]; + } + } - // Execute protocol when randomly sampled bit-wise random number is ready - var self = this; - var ready_sampling = function (bits) { - // if 2*this is even, then this is less than half prime, otherwise this is greater or equal to half prime - if (bits.length !== bitLength) { - throw new Error('Preprocessed bits sequence has incorrect length, expected: ' + bitLength + ' actual: ' + bits.length); - } + return { + SBOX: SBOX, + INV_SBOX: INV_SBOX, + SUB_MIX: SUB_MIX, + INV_SUB_MIX: INV_SUB_MIX + }; + })(); + + function AES(key) { + this._key = asUInt32Array(key); + this._reset(); + } - // bit composition: r = (rl ... r1 r0)_10 - var r = self.jiff.protocols.bits.bit_composition(bits); - // open share + noise, and utilize opened value with shared bit representation of noise to check the least significant digit of share. - share.jiff.internal_open(r.isadd(share), share.holders, op_id + ':open').then(function (result) { - var wrapped = self.jiff.protocols.bits.cgt(bits, result, op_id + ':bits.cgt'); - var isOdd = self.jiff.helpers.mod(result, 2); - isOdd = bits[0].icxor_bit(isOdd); - isOdd = isOdd.isxor_bit(wrapped, op_id + ':sxor_bit'); - - var answer = isOdd.inot(); - answer.wThen(final_deferred.resolve); - }); - }; - - // generate the bits of a random number less than our prime - var bits = this.jiff.get_preprocessing(op_id + ':sampling'); - if (bits == null) { - var promise = this.jiff.from_crypto_provider('numbers', this.holders, this.threshold, this.Zp, op_id + ':sampling', { - bitLength: bitLength, - count: 1, - max: this.Zp - }); - promise.then(function (msg) { - ready_sampling(msg['shares']); - }); - } else { - ready_sampling(bits); - } + AES.blockSize = 4 * 4; + AES.keySize = 256 / 8; + AES.prototype.blockSize = AES.blockSize; + AES.prototype.keySize = AES.keySize; + AES.prototype._reset = function () { + var keyWords = this._key; + var keySize = keyWords.length; + var nRounds = keySize + 6; + var ksRows = (nRounds + 1) * 4; + + var keySchedule = []; + for (var k = 0; k < keySize; k++) { + keySchedule[k] = keyWords[k]; + } - return result; - }; -}; -},{}],33:[function(require,module,exports){ -// general arithmetic protocols -module.exports = function (SecretShare) { - /** - * Reshares/refreshes the sharing of this number, used before opening to keep the share secret. - * @method refresh - * @param {string} [op_id=auto_gen()] - the operation id with which to tag the messages sent by this refresh, by default - * an automatic operation id is generated by increasing a local counter, default operation ids - * suffice when all parties execute the instructions in the same order. - * @returns {module:jiff-client~JIFFClient#SecretShare} a new share of the same number. - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.refresh = function (op_id) { - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('refresh', this.holders); - } + for (k = keySize; k < ksRows; k++) { + var t = keySchedule[k - 1]; - // final result - var final_deferred = new this.jiff.helpers.Deferred(); - var final_promise = final_deferred.promise; - var result = new this.jiff.SecretShare(final_promise, this.holders, this.threshold, this.Zp); - - // refresh - var self = this; - var ready_number = function (zero) { - self.isadd(zero).wThen(final_deferred.resolve); - }; - - // get shares of zero - var zero = this.jiff.get_preprocessing(op_id); - if (zero == null) { - var promise = this.jiff.from_crypto_provider('numbers', this.holders, this.threshold, this.Zp, op_id, { - number: 0, - count: 1 - }); - promise.then(function (msg) { - ready_number(msg['shares'][0]); - }); - } else { - ready_number(zero); - } + if (k % keySize === 0) { + t = (t << 8) | (t >>> 24); + t = (G.SBOX[t >>> 24] << 24) | (G.SBOX[(t >>> 16) & 0xff] << 16) | (G.SBOX[(t >>> 8) & 0xff] << 8) | G.SBOX[t & 0xff]; - return result; - }; + t ^= RCON[(k / keySize) | 0] << 24; + } else if (keySize > 6 && k % keySize === 4) { + t = (G.SBOX[t >>> 24] << 24) | (G.SBOX[(t >>> 16) & 0xff] << 16) | (G.SBOX[(t >>> 8) & 0xff] << 8) | G.SBOX[t & 0xff]; + } - /** - * Bit Decomposition: Transform existing share to an array of bit shares. - * @method bit_decomposition - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly. - * @returns {module:jiff-client~JIFFClient#SecretShare[]} an array of secret shares of bits of length [ceil(log_2(this.Zp))], where - * index 0 represents the least significant bit. - */ - SecretShare.prototype.bit_decomposition = function (op_id) { - if (op_id == null) { - op_id = this.jiff.counters.gen_op_id('bit_decomposition', this.holders); - } + keySchedule[k] = keySchedule[k - keySize] ^ t; + } - // bit length of this secret - var bitLength = this.jiff.share_helpers['ceil'](this.jiff.helpers.bLog(this.Zp, 2)); - - // Create deferred shares to resolve to later when the computation completes - var many_shares = this.jiff.utils.many_secret_shares(bitLength, this.holders, this.threshold, this.Zp); - var deferreds = many_shares.deferreds; - var result = many_shares.shares; - - // Execute protocol when randomly sampled bit-wise random number is ready - var self = this; - var ready_sampling = function (bits) { - var r = self.jiff.protocols.bits.bit_composition(bits); - // add and reveal random number to this - self.jiff.internal_open(r.isadd(self), self.holders, op_id + ':open').then(function (result) { - // compute bits assuming r+this < Zp - var noWrap = self.jiff.protocols.bits.csubr(result, bits, op_id + ':bits.csubr:1'); - var didWrap = noWrap.pop(); - - // compute bits assuming r+this >= Zp - var withWrap = self.jiff.protocols.bits.csubr(self.jiff.share_helpers['+'](result, self.Zp), bits, op_id + ':bits.csubr:2'); - withWrap.pop(); // withWrap cannot underflow! - - // choose noWrap if first subtraction does not overflow (sign bit is zero), otherwise choose withWrap. - for (var i = 0; i < bitLength; i++) { - withWrap[i] = didWrap.iif_else(withWrap[i], noWrap[i], op_id + ':if_else:' + i); - } - self.jiff.utils.resolve_many_secrets(deferreds, withWrap); - }); - }; - - // generate the bits of a random number less than our prime - var bits = this.jiff.get_preprocessing(op_id + ':sampling'); - if (bits == null) { - var promise = this.jiff.from_crypto_provider('numbers', this.holders, this.threshold, this.Zp, op_id + ':sampling', { - bitLength: bitLength, - count: 1, - max: this.Zp - }); - promise.then(function (msg) { - ready_sampling(msg['shares']); - }); - } else { - ready_sampling(bits); - } + var invKeySchedule = []; + for (var ik = 0; ik < ksRows; ik++) { + var ksR = ksRows - ik; + var tt = keySchedule[ksR - (ik % 4 ? 0 : 4)]; + + if (ik < 4 || ksR <= 4) { + invKeySchedule[ik] = tt; + } else { + invKeySchedule[ik] = + G.INV_SUB_MIX[0][G.SBOX[tt >>> 24]] ^ G.INV_SUB_MIX[1][G.SBOX[(tt >>> 16) & 0xff]] ^ G.INV_SUB_MIX[2][G.SBOX[(tt >>> 8) & 0xff]] ^ G.INV_SUB_MIX[3][G.SBOX[tt & 0xff]]; + } + } - return result; - }; -}; -},{}],34:[function(require,module,exports){ -module.exports = { - /** - * Generates a random bit under MPC by xoring all bits sent by participating parties using smult / beaver triples - * @function generate_random_bit_bgw - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed - * @param {Array} receivers_list - array of party ids to receive the result - * @param {Array} compute_list - array of party ids to perform the protocol - * @param {number} Zp - the mod - * @param {object} params - an object containing extra parameters passed by the user - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * @return {Object} contains 'share' (this party's share of the generated bit) and 'promise' - */ - generate_random_bit_smult: function (jiff, threshold, receivers_list, compute_list, Zp, params) { - if (params.compute_threshold == null) { - params.compute_threshold = threshold; - } + this._nRounds = nRounds; + this._keySchedule = keySchedule; + this._invKeySchedule = invKeySchedule; + }; + + AES.prototype.encryptBlockRaw = function (M) { + M = asUInt32Array(M); + return cryptBlock(M, this._keySchedule, G.SUB_MIX, G.SBOX, this._nRounds); + }; + + AES.prototype.encryptBlock = function (M) { + var out = this.encryptBlockRaw(M); + var buf = Buffer.allocUnsafe(16); + buf.writeUInt32BE(out[0], 0); + buf.writeUInt32BE(out[1], 4); + buf.writeUInt32BE(out[2], 8); + buf.writeUInt32BE(out[3], 12); + return buf; + }; + + AES.prototype.decryptBlock = function (M) { + M = asUInt32Array(M); + + // swap + var m1 = M[1]; + M[1] = M[3]; + M[3] = m1; + + var out = cryptBlock(M, this._invKeySchedule, G.INV_SUB_MIX, G.INV_SBOX, this._nRounds); + var buf = Buffer.allocUnsafe(16); + buf.writeUInt32BE(out[0], 0); + buf.writeUInt32BE(out[3], 4); + buf.writeUInt32BE(out[2], 8); + buf.writeUInt32BE(out[1], 12); + return buf; + }; + + AES.prototype.scrub = function () { + scrubVec(this._keySchedule); + scrubVec(this._invKeySchedule); + scrubVec(this._key); + }; + + module.exports.AES = AES; + }, + { 'safe-buffer': 269 } + ], + 79: [ + function (require, module, exports) { + var aes = require('./aes'); + var Buffer = require('safe-buffer').Buffer; + var Transform = require('cipher-base'); + var inherits = require('inherits'); + var GHASH = require('./ghash'); + var xor = require('buffer-xor'); + var incr32 = require('./incr32'); + + function xorTest(a, b) { + var out = 0; + if (a.length !== b.length) out++; + + var len = Math.min(a.length, b.length); + for (var i = 0; i < len; ++i) { + out += a[i] ^ b[i]; + } - if (params.op_id == null && params.output_op_id == null) { - params.op_id = jiff.counters.gen_op_id2('generate_random_bit_smult', receivers_list, compute_list); - } else if (params.op_id == null) { - params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); - } - var op_id = params.op_id; - - // Generate random bit - var random_bit, promise; - if (compute_list.indexOf(jiff.id) > -1) { - var bit = jiff.helpers.random(2); - var bit_shares = jiff.internal_share(bit, params.compute_threshold, compute_list, compute_list, Zp, op_id + ':share'); - - random_bit = bit_shares[compute_list[0]]; - for (var i = 1; i < compute_list.length; i++) { - var party_id = compute_list[i]; - var obit = bit_shares[party_id]; - random_bit = random_bit.isxor_bit(obit, op_id + ':sxor_bit:' + i); - } + return out; + } - promise = random_bit.value; - } + function calcIv(self, iv, ck) { + if (iv.length === 12) { + self._finID = Buffer.concat([iv, Buffer.from([0, 0, 0, 1])]); + return Buffer.concat([iv, Buffer.from([0, 0, 0, 2])]); + } + var ghash = new GHASH(ck); + var len = iv.length; + var toPad = len % 16; + ghash.update(iv); + if (toPad) { + toPad = 16 - toPad; + ghash.update(Buffer.alloc(toPad, 0)); + } + ghash.update(Buffer.alloc(8, 0)); + var ivBits = len * 8; + var tail = Buffer.alloc(8); + tail.writeUIntBE(ivBits, 0, 8); + ghash.update(tail); + self._finID = ghash.state; + var out = Buffer.from(self._finID); + incr32(out); + return out; + } + function StreamCipher(mode, key, iv, decrypt) { + Transform.call(this); + + var h = Buffer.alloc(4, 0); + + this._cipher = new aes.AES(key); + var ck = this._cipher.encryptBlock(h); + this._ghash = new GHASH(ck); + iv = calcIv(this, iv, ck); + + this._prev = Buffer.from(iv); + this._cache = Buffer.allocUnsafe(0); + this._secCache = Buffer.allocUnsafe(0); + this._decrypt = decrypt; + this._alen = 0; + this._len = 0; + this._mode = mode; + + this._authTag = null; + this._called = false; + } - // Reshare - random_bit = jiff.reshare(random_bit, threshold, receivers_list, compute_list, Zp, op_id + ':reshare'); - if (receivers_list.indexOf(jiff.id) > -1) { - promise = random_bit.value; - } - return {share: random_bit, promise: promise}; - }, - /** - * Generates a random bit under MPC by xoring all bits sent by participating parties using smult_bgw - * @function generate_random_bit_bgw - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed - * @param {Array} receivers_list - array of party ids to receive the result - * @param {Array} compute_list - array of party ids to perform the protocol - * @param {number} Zp - the mod - * @param {object} params - an object containing extra parameters passed by the user - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * @return {Object} contains 'share' (this party's share of the generated bit) and 'promise' - */ - generate_random_bit_bgw: function (jiff, threshold, receivers_list, compute_list, Zp, params) { - if (params.compute_threshold == null) { - params.compute_threshold = Math.floor((compute_list.length + 1) / 2); // honest majority BGW - } + inherits(StreamCipher, Transform); - if (params.op_id == null && params.output_op_id == null) { - params.op_id = jiff.counters.gen_op_id2('generate_random_bit_bgw', receivers_list, compute_list); - } else if (params.op_id == null) { - params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); - } + StreamCipher.prototype._update = function (chunk) { + if (!this._called && this._alen) { + var rump = 16 - (this._alen % 16); + if (rump < 16) { + rump = Buffer.alloc(rump, 0); + this._ghash.update(rump); + } + } - var op_id = params.op_id; + this._called = true; + var out = this._mode.encrypt(this, chunk); + if (this._decrypt) { + this._ghash.update(chunk); + } else { + this._ghash.update(out); + } + this._len += chunk.length; + return out; + }; - // Generate random bit - var random_bit, promise; - if (compute_list.indexOf(jiff.id) > -1) { - var bit = jiff.helpers.random(2); - var bit_shares = jiff.internal_share(bit, params.compute_threshold, compute_list, compute_list, Zp, op_id + ':share'); + StreamCipher.prototype._final = function () { + if (this._decrypt && !this._authTag) throw new Error('Unsupported state or unable to authenticate data'); - random_bit = bit_shares[compute_list[0]]; - for (var i = 1; i < compute_list.length; i++) { - var party_id = compute_list[i]; - var obit = bit_shares[party_id]; - random_bit = random_bit.isadd(obit).issub(random_bit.ismult_bgw(obit, op_id + ':smult' + i).icmult(2)); - } + var tag = xor(this._ghash.final(this._alen * 8, this._len * 8), this._cipher.encryptBlock(this._finID)); + if (this._decrypt && xorTest(tag, this._authTag)) throw new Error('Unsupported state or unable to authenticate data'); - promise = random_bit.value; - } + this._authTag = tag; + this._cipher.scrub(); + }; - // Reshare - random_bit = jiff.reshare(random_bit, threshold, receivers_list, compute_list, Zp, op_id + ':reshare'); - if (receivers_list.indexOf(jiff.id) > -1) { - promise = random_bit.value; - } - return {share: random_bit, promise: promise}; - }, - /** - * Generates a sequence of random bits under MPC - * @function generate_random_bits - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {number} [threshold=receivers_list.length] - the threshold of the bit when stored by receivers after generation - * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed - * @param {Array} receivers_list - array of party ids to receive the result - * @param {Array} compute_list - array of party ids to perform the protocol - * @param {number} Zp - the mod - * @param {object} params - an object containing extra parameters passed by the user - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated - * - count: how many random bits to generate - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * @param {object} protocols - the protocols to use for preprocessing - * @return {Object} contains 'share' (array of secret shares bits) and 'promise' - */ - generate_random_bits: function (jiff, threshold, receivers_list, compute_list, Zp, params, protocols) { - if (params.count == null) { - params.count = 1; - } + StreamCipher.prototype.getAuthTag = function getAuthTag() { + if (this._decrypt || !Buffer.isBuffer(this._authTag)) throw new Error('Attempting to get auth tag in unsupported state'); - if (params.op_id == null && params.output_op_id == null) { - params.op_id = jiff.counters.gen_op_id2('generate_random_bits', receivers_list, compute_list); - } else if (params.op_id == null) { - params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); - } + return this._authTag; + }; - var op_id = params.op_id; - var _params = params; + StreamCipher.prototype.setAuthTag = function setAuthTag(tag) { + if (!this._decrypt) throw new Error('Attempting to set auth tag in unsupported state'); - var promises = []; - var bits = []; - for (var i = 0; i < params.count; i++) { - params = Object.assign({}, _params); - params.op_id = op_id + ':' + i; + this._authTag = tag; + }; - var bit = protocols.generate_random_bit(threshold, receivers_list, compute_list, Zp, params, protocols); + StreamCipher.prototype.setAAD = function setAAD(buf) { + if (this._called) throw new Error('Attempting to set AAD in unsupported state'); - promises.push(bit.value); - if (bit.share != null) { - bits.push(bit.share); - } - } + this._ghash.update(buf); + this._alen += buf.length; + }; - if (bits.length === 0) { - bits = null; - } - return {share: bits, promise: Promise.all(promises)}; - } -}; -},{}],35:[function(require,module,exports){ -/** - * Can be used to generate shares of a random number, or shares of zero. - * For a random number, every party generates a local random number and secret share it, - * then every party sums its share, resulting in a single share of an unknown random number for every party. - * The same approach is followed for zero, but instead, all the parties know that the total number is zero, but they - * do not know the value of any resulting share (except their own) - * @function jiff_share_all_number - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {number} n - the number to share - * @param {number} [threshold=receivers_list.length] - the min number of parties needed to reconstruct the secret, defaults to all the receivers - * @param {Array} [receivers_list=all_parties] - array of party ids to receive the result, by default, this includes all parties - * @param {Array} [compute_list=all_parties] - array of party ids to perform the protocol, by default, this includes all parties - * @param {number} [Zp=jiff.Zp] - the mod - * @param {object} [params={}] - an object containing extra parameters passed by the user - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * @return {Object} contains 'share' (this party's share of the result) and 'promise' - */ -var jiff_share_all_number = function (jiff, n, threshold, receivers_list, compute_list, Zp, params) { - var isSender = compute_list.indexOf(jiff.id) > -1; - var isReceiver = receivers_list.indexOf(jiff.id) > -1; - - if (!isSender && !isReceiver) { - return {}; - } - - if (params.compute_threshold == null) { - params.compute_threshold = Math.min(threshold, compute_list.length); - } - - var result, promise; - if (isSender) { - var shares = jiff.internal_share(n, params.compute_threshold, compute_list, compute_list, Zp, params.op_id + ':share'); - result = shares[compute_list[0]]; - for (var i = 1; i < compute_list.length; i++) { - result = result.isadd(shares[compute_list[i]]); - } - promise = result.value; - } - - result = jiff.reshare(result, threshold, receivers_list, compute_list, Zp, params.op_id + ':reshare'); - if (receivers_list.indexOf(jiff.id) > -1) { - promise = result.value; - } - - return {share: result, promise: promise}; -}; - -module.exports = { - /** - * Creates shares of an unknown random number. Every party comes up with its own random number and shares it. - * Then every party combines all the received shares to construct one share of the random unknown number - * @function generate_random_number - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed - * @param {Array} receivers_list - array of party ids to receive the result - * @param {Array} compute_list - array of party ids to perform the protocol - * @param {number} Zp - the mod - * @param {object} params - an object containing extra parameters passed by the user - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * @return {Object} contains 'share' (this party's share of the result) and 'promise' - */ - generate_random_number: function (jiff, threshold, receivers_list, compute_list, Zp, params) { - if (params.op_id == null && params.output_op_id == null) { - params.op_id = jiff.counters.gen_op_id2('generate_random_number', receivers_list, compute_list); - } else if (params.op_id == null) { - params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); - } - return jiff_share_all_number(jiff, jiff.helpers.random(Zp), threshold, receivers_list, compute_list, Zp, params); - }, - /** - * Creates shares of 0, such that no party knows the other parties' shares. - * Every party secret shares 0, then every party sums all the shares they received, resulting - * in a new share of 0 for every party - * @function generate_zero - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff instance - * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed - * @param {Array} receivers_list - array of party ids to receive the result - * @param {Array} compute_list - array of party ids to perform the protocol - * @param {number} Zp - the mod - * @param {object} params - an object containing extra parameters passed by the user - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * @return {Object} contains 'share' (this party's share of the result) and 'promise' - */ - generate_zero: function (jiff, threshold, receivers_list, compute_list, Zp, params) { - if (params.op_id == null && params.output_op_id == null) { - params.op_id = jiff.counters.gen_op_id2('generate_zero', receivers_list, compute_list); - } else if (params.op_id == null) { - params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); - } - return jiff_share_all_number(jiff, 0, threshold, receivers_list, compute_list, Zp, params); - } -}; -},{}],36:[function(require,module,exports){ -/** - * Creates shares of r and x, such that r is a uniform random number between 0 and Zp, and x is floor(r/constant) - * where constant is provided by the extra params - * @function generate_random_and_quotient - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {number} threshold - the min number of parties needed to reconstruct the secret after it is computed - * @param {Array} receivers_list - array of party ids to receive the result - * @param {Array} compute_list - array of party ids to perform the protocol - * @param {number} Zp - the mod - * @param {object} params - an object containing extra parameters passed by the user - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated - * - compute_threshold: the threshold to use during computation: defaults to compute_list.length - * - constant: the constant to divide the random number by. - * - output_op_id: the set op id of the output quotient and noise - * @return {Object} contains 'share' (this party's share of the result) and 'promise' - */ -module.exports = function (jiff, threshold, receivers_list, compute_list, Zp, params, protocols) { - // consistent and unique op_id for compute and receiver parties - if (params.op_id == null && params.output_op_id == null) { - params.op_id = jiff.counters.gen_op_id2('generate_random_and_quotient', receivers_list, compute_list); - } else if (params.op_id == null) { - params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); - } - - if (params.compute_threshold == null) { - params.compute_threshold = Math.floor((compute_list.length + 1) / 2); // honest majority BGW - } - - var constant = params.constant; - var op_id = params.op_id; - Zp = Zp ? jiff.Zp : Zp; - - // stores the result - var r, q; - - // for compute parties - var promise; - if (compute_list.indexOf(jiff.id) > -1) { - var largestQuotient = jiff.share_helpers['floor'](jiff.share_helpers['/'](Zp, constant)); - var largestMultiple = jiff.share_helpers['*'](largestQuotient, constant); - - // Uniform random number between [0, Zp) - var r_bits = jiff.protocols.bits.rejection_sampling(0, Zp, params.compute_threshold, compute_list, Zp, params.output_op_id + ':rejection1'); - var cmp = jiff.protocols.bits.cgteq(r_bits, largestMultiple, params.output_op_id + ':bits_cgteq'); - var r1 = jiff.protocols.bits.bit_composition(r_bits); - - // Uniform random number between [0, Math.floor(Zp / constant)) - var quotient = jiff.protocols.bits.rejection_sampling(0, largestQuotient, params.compute_threshold, compute_list, Zp, params.output_op_id + ':rejection2'); - quotient = jiff.protocols.bits.bit_composition(quotient); - - // Uniform random number between [0, constant) - var remainder = jiff.protocols.bits.rejection_sampling(0, constant, params.compute_threshold, compute_list, Zp, params.output_op_id + ':rejection3'); - remainder = jiff.protocols.bits.bit_composition(remainder); - var r2 = quotient.icmult(constant).isadd(remainder); - - // choose either (r1, largestQuotient) or (r2, quotient) based on cmp result - r = cmp.iif_else(r1, r2, params.output_op_id + ':ifelse1'); - q = cmp.iif_else(largestQuotient, quotient, params.output_op_id + ':ifelse2'); - promise = Promise.all([r.value, q.value]); - } - - // reshare the result with the designated receivers - r = jiff.reshare(r, threshold, receivers_list, compute_list, Zp, op_id + ':reshare1'); - q = jiff.reshare(q, threshold, receivers_list, compute_list, Zp, op_id + ':reshare2'); - - // return result - if (receivers_list.indexOf(jiff.id) > -1) { - promise = Promise.all([r.value, q.value]); - } - return {share: {r: r, q: q}, promise: promise}; -}; -},{}],37:[function(require,module,exports){ -var sample = function (jiff, range, compute_list, Zp, params, protocols, reject_count) { - // Transform sampling range into bit size - var bitLength = jiff.helpers.ceil(jiff.helpers.bLog(range, 2)); - bitLength = parseInt(bitLength.toString(), 10); - - var paramsCopy = Object.assign({}, params); - paramsCopy['count'] = bitLength; - paramsCopy['op_id'] = params.op_id + ':sampling:' + reject_count; - return protocols.generate_random_bits(params.compute_threshold, compute_list, compute_list, Zp, paramsCopy, protocols).share; -}; - -var one_round_sampling = function (jiff, lower_bound, upper_bound, compute_list, Zp, params, protocols, finalLength, reject_count) { - // Figure out sampling range - var range; - if (upper_bound.isBigNumber === true) { - range = upper_bound.minus(lower_bound); - } else { - range = upper_bound - lower_bound; - } - - // Special cases - if (range.toString() === '0') { - throw new Error('rejection sampling called with range 0, no numbers to sample!'); - } - if (range.toString() === '1') { - var zero = protocols.generate_zero(params.compute_threshold, compute_list, compute_list, Zp, params, protocols).share; - // special case: cadd can be performed locally on bit arrays of length 1! - var resultOne = jiff.protocols.bits.cadd([zero], lower_bound); - while (resultOne.length > finalLength) { - resultOne.pop(); - } - return {share: resultOne, promise: true}; - } - - // Rejection protocol - var bits = sample(jiff, range, compute_list, Zp, params, protocols, reject_count); - var cmp = jiff.protocols.bits.clt(bits, range, params.output_op_id + ':bits.clt:' + reject_count); - var bits_add = jiff.protocols.bits.cadd(bits, lower_bound, params.output_op_id + ':bits.cadd:' + reject_count); - - if (cmp === true) { - return {share: bits_add, promise: true}; - } else if (cmp === false) { // need to resample - return {share: bits_add, promise: false}; - } - - var promise = jiff.internal_open(cmp, compute_list, params.output_op_id + ':open:' + reject_count); - return {share: bits_add, promise: promise.then( - function (cmp) { - return cmp.toString() === '1'; - } - )}; -}; - -var computeParty = function (jiff, lower_bound, upper_bound, threshold, receivers_list, compute_list, Zp, params, protocols, op_id, finalLength, reject_count) { - var result = one_round_sampling(jiff, lower_bound, upper_bound, compute_list, Zp, params, protocols, finalLength, reject_count); - - // Case 1: we know whether sampling succeeded or not - if (result.promise === true) { - return reshareResult(jiff, upper_bound, threshold, receivers_list, compute_list, Zp, op_id, finalLength, result.share); - } - if (result.promise === false) { - return {share: 'RETRY', promise: jiff.utils.all_promises(result.share)}; - } - - // Case 2: we only have a promise to whether the sampling succeeded or not - var many_shares = jiff.utils.many_secret_shares(finalLength, compute_list, threshold, Zp); - var final_deferreds = many_shares.deferreds; - - result.promise.then(function (promiseVal) { - // RETRY and PLACEHOLDER shares are cleaned up later in the preprocessing pipeline - if (promiseVal === false) { - for (var i = 0; i < final_deferreds.length; i++) { - final_deferreds[i].resolve('RETRY'); - } - return; - } - - // Need to make sure party only executes the reshare operation if sampling succeeds - var reshared = reshareResult(jiff, upper_bound, threshold, receivers_list, compute_list, Zp, op_id, finalLength, result.share); - - if (receivers_list.indexOf(jiff.id) > -1) { - jiff.utils.resolve_many_secrets(final_deferreds, reshared.share); - } else { - for (i = 0; i < final_deferreds.length; i++) { - final_deferreds[i].resolve('PLACEHOLDER'); - } - } - }); - - return {share: many_shares.shares, promise: jiff.utils.all_promises(many_shares.shares)}; -}; - -var reshareResult = function (jiff, upper_bound, threshold, receivers_list, compute_list, Zp, op_id, finalLength, shares) { - // fix threshold and parties - var promises = []; - for (var i = 0; i < finalLength; i++) { - if (compute_list.indexOf(jiff.id) > -1) { - promises[i] = shares[i].value; - } - shares[i] = jiff.reshare(shares[i], threshold, receivers_list, compute_list, Zp, op_id + ':reshare:' + i); - if (receivers_list.indexOf(jiff.id) > -1) { - promises[i] = shares[i].value; - } - } - - // return output - if (receivers_list.indexOf(jiff.id) === -1) { - shares = null; - } - - // handle rejection case - return {share: shares, promise: Promise.all(promises)}; -}; - -/** - * Wrapper for when doing rejection sampling during pre processing - * - * Do not use this function directly, especially during online computation time, use jiffClient.protocols.bits.rejection_sampling instead - * - * @function sampling - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {number} [threshold=receivers_list.length] - the threshold of the resulting shares after sampling - * @param {Array} [receivers_list=all_parties] - array of party ids that want to receive the sampling shares, by default, this includes all parties - * @param {Array} [compute_list=all_parties] - array of party ids that will perform this protocol, by default, this includes all parties - * @param {number} [Zp=jiff-instance.Zp] - the mod (if null then the default Zp for the instance is used) - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this operation. - * This id must be unique, and must be passed by all parties to the same instruction, to - * ensure that corresponding instructions across different parties are matched correctly - * @param {object} [params={}] - an object containing extra parameters passed by the user - * Expects: - * - compute_threshold, an optional number compute_threshold parameter, which specifies threshold used - * during the protocol execution. By default, this is (|compute_list|+1)/2 - * - optional 'lower_bound' and 'upper_bound', numeric parameters, default to 0 and Zp respectively - * - op_id, the base op_id to tag operations inside this protocol with, defaults to auto generated - * - output_op_id, the tag id for the output result - * - retry_count, how many times rejection sampling have been retried! - * @param {object} [protocols=defaults] - the protocols to use for preprocessing, any protocol(s) not provided will be replaced with defaults - * @returns {Object} an object containing keys: 'share', and 'promise'. The promise is resolved when the rejection sampling is completed. - * The object is consumed by .preprocessing: - * - 'share' attribute contains the resulting array of secret shared bits representing the sampled value, and is stored in the preprocessing table internally - * - The promise is consumed and a new promise is returned by .preprocessing that is resolved after this returned promise (and all other promise generated by that .preprocessing call) are resolved - */ -module.exports = function (jiff, threshold, receivers_list, compute_list, Zp, params, protocols) { - // Internal version: set parameters (e.g. receivers_list) do not need defaults - // defaults (for internal preprocessing) - var lower_bound = params.lower_bound != null ? params.lower_bound : 0; - var upper_bound = params.upper_bound != null ? params.upper_bound : Zp; - if (params.compute_threshold == null) { // honest majority BGW - params.compute_threshold = Math.floor((compute_list.length + 1) / 2); - } - - // Figure out final bit size (after adding back lower) - var finalLength = jiff.helpers.ceil(jiff.helpers.bLog(upper_bound, 2)); - finalLength = parseInt(finalLength.toString(), 10); - finalLength = Math.max(finalLength, 1); // special case: when upper_bound is 1! - - if (params.op_id == null && params.output_op_id == null) { - params.op_id = jiff.counters.gen_op_id2('rejection_sampling', receivers_list, compute_list); - } else if (params.op_id == null) { - params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); - } - var op_id = params.op_id; - - // Rejection count - var reject_count = params.reject_count || 0; - - if (compute_list.indexOf(jiff.id) === -1) { - return reshareResult(jiff, upper_bound, threshold, receivers_list, compute_list, Zp, op_id, finalLength, []) - } - - return computeParty(jiff, lower_bound, upper_bound, threshold, receivers_list, compute_list, Zp, params, protocols, op_id, finalLength, reject_count); -}; - -},{}],38:[function(require,module,exports){ -/** - * Generation of beaver triplet via MPC, uses the server for communication channels, but not for generation - * @function generate_beaver_bgw - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {number} threshold - the threshold of the triplets when stored by receivers after generation - * @param {Array} receivers_list - array of party ids that want to receive the triplet shares - * @param {Array} compute_list - array of party ids that will perform this protocol - * @param {number} Zp - the mod - * @param {object} params - an object containing extra parameters passed by the user - * Expects: - * - op_id: the base id to use for operation during the execution of this protocol, defaults to auto generated - * - an optional number compute_threshold parameter, which specifies threshold used - * during the protocol execution. By default, this is the length of the (compute_list+1)/2 - * @param {object} protocols - the sub protocols to use for preprocessing - * @return {object} all pre-processing protocols must return an object with these keys: - * { - * 'share': the share(s)/value(s) to store attached to op_id for later use by the computation (i.e. the result of preprocessing), - * 'promise': a promise for when this protocol is fully completed (could be null if the protocol was already completed) - * } - * In this case, 'share' is an array of this party's shares of the resulting triplet, a,b,c such that a*b=c - */ -module.exports = function (jiff, threshold, receivers_list, compute_list, Zp, params, protocols) { - if (params.compute_threshold == null) { - params.compute_threshold = Math.floor((compute_list.length + 1) / 2); // honest majority BGW - } - if (params.op_id == null && params.output_op_id == null) { - params.op_id = jiff.counters.gen_op_id2('generate_beaver_bgw', receivers_list, compute_list); - } else if (params.op_id == null) { - params.op_id = 'preprocessing:' + params.output_op_id + ':' + compute_list.join(','); - } - - var op_id = params.op_id; - var _params = params; - - var a, b, c, promises; - if (compute_list.indexOf(jiff.id) > -1) { - params = Object.assign({}, _params); - params.op_id = op_id + ':share_a'; - a = protocols.generate_random_number(params.compute_threshold, compute_list, compute_list, Zp, params, protocols).share; - - params = Object.assign({}, _params); - params.op_id = op_id + ':share_b'; - b = protocols.generate_random_number(params.compute_threshold, compute_list, compute_list, Zp, params, protocols).share; - - c = a.ismult_bgw(b, op_id + ':smult_bgw'); - promises = [a.value, b.value, c.value]; - } - - a = jiff.reshare(a, threshold, receivers_list, compute_list, Zp, op_id + ':reshare_a'); - b = jiff.reshare(b, threshold, receivers_list, compute_list, Zp, op_id + ':reshare_b'); - c = jiff.reshare(c, threshold, receivers_list, compute_list, Zp, op_id + ':reshare_c'); - if (receivers_list.indexOf(jiff.id) > -1) { - promises = [a.value, b.value, c.value]; - } - - return { share: [a, b, c], promise: Promise.all(promises) }; -}; - -},{}],39:[function(require,module,exports){ -/** - * Share the given share to all the parties in the jiff instance. - * @function jiff_broadcast - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance. - * @param {module:jiff-client~JIFFClient#SecretShare} share - the share. - * @param {Array} parties - the parties to broadcast the share to. - * @param {number|string} op_id - a unique operation id, used to tag outgoing messages. - * - */ -var jiff_broadcast = function (jiff, share, parties, op_id) { - for (var index = 0; index < parties.length; index++) { - var i = parties[index]; // Party id - if (i === jiff.id) { - jiff.handlers.receive_open({ party_id: i, share: share.value, op_id: op_id, Zp: share.Zp }); - continue; - } - - // encrypt, sign and send - var msg = {party_id: i, share: share.value, op_id: op_id, Zp: share.Zp}; - msg = jiff.hooks.execute_array_hooks('beforeOperation', [jiff, 'open', msg], 2); - - msg['share'] = jiff.hooks.encryptSign(jiff, msg['share'].toString(), jiff.keymap[msg['party_id']], jiff.secret_key); - jiff.socket.safe_emit('open', JSON.stringify(msg)); - } -}; - -module.exports = { - /** - * Open up the given share to the participating parties. - * @function jiff_open - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance. - * @param {module:jiff-client~JIFFClient#SecretShare} share - the share of the secret to open that belongs to this party - * @param {Array} [parties=all_parties] - an array with party ids of receiving parties - * @param {string|number} [op_id=auto_gen()] - the operation id to be used to tag outgoing messages - * @returns {?promise} a (JQuery) promise to the open value of the secret, or null if the calling party is not a receiving party - * - */ - jiff_open: function (jiff, share, parties, op_id) { - var i; - - if (!(share.jiff === jiff)) { - throw 'share does not belong to given instance'; - } - - // Default values - if (parties == null || parties === []) { - parties = []; - for (i = 1; i <= jiff.party_count; i++) { - parties.push(i); - } - } else { - jiff.helpers.sort_ids(parties); - } - - // If not a receiver nor holder, do nothing - if (share.holders.indexOf(jiff.id) === -1 && parties.indexOf(jiff.id) === -1) { - return null; - } - - // Compute operation ids (one for each party that will receive a result - if (op_id == null) { - op_id = jiff.counters.gen_op_id2('open', parties, share.holders); - } - - // Party is a holder - if (share.holders.indexOf(jiff.id) > -1) { - // Call hook - share = jiff.hooks.execute_array_hooks('beforeOpen', [jiff, share, parties], 1); - - // refresh/reshare, so that the original share remains secret, instead - // a new share is sent/open without changing the actual value. - share = share.refresh(op_id + ':refresh'); - - // The given share has been computed, broadcast it to all parties - jiff.counters.pending_opens++; - share.wThen(function () { - jiff.counters.pending_opens--; - jiff_broadcast(jiff, share, parties, op_id); - }, share.error); - } - - // Party is a receiver - if (parties.indexOf(jiff.id) > -1) { - var final_deferred = new jiff.helpers.Deferred(); // will be resolved when the final value is reconstructed - var final_promise = final_deferred.promise; - - if (jiff.deferreds[op_id] == null) { - jiff.deferreds[op_id] = {}; - } - - jiff.deferreds[op_id].deferred = final_deferred; - jiff.deferreds[op_id].threshold = share.threshold; - jiff.deferreds[op_id].total = share.holders.length; - if (jiff.deferreds[op_id].shares != null && jiff.deferreds[op_id].shares.length >= share.threshold) { - final_deferred.resolve(); - } - - return final_promise.then(function () { - var shares = jiff.deferreds[op_id].shares; - - if (shares.length === jiff.deferreds[op_id].total) { - delete jiff.deferreds[op_id]; - } else { - jiff.deferreds[op_id].deferred = 'CLEAN'; - } - - var recons_secret = jiff.hooks.reconstructShare(jiff, shares); - recons_secret = jiff.hooks.execute_array_hooks('afterReconstructShare', [jiff, recons_secret], 1); - return recons_secret; - }); - } - - return null; - }, - /** - * Uses Lagrange polynomials to interpolate the polynomial - * described by the given shares (points) - * @function jiff_lagrange - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {Array} shares - an array of objects representing shares to reconstruct, every object has 3 attributes: value, sender_id, Zp - * @returns {number} the value of the polynomial at x=0 (the secret value) - * - */ - jiff_lagrange: function (jiff, shares) { - var lagrange_coeff = []; // will contain shares.length many elements. - - // Compute the Langrange coefficients at 0. - for (var i = 0; i < shares.length; i++) { - var pi = jiff.helpers.get_party_number(shares[i].sender_id); - lagrange_coeff[pi] = 1; - - for (var j = 0; j < shares.length; j++) { - var pj = jiff.helpers.get_party_number(shares[j].sender_id); - if (pj !== pi) { - var inv = jiff.helpers.extended_gcd(pi - pj, shares[i].Zp)[0]; - lagrange_coeff[pi] = jiff.helpers.mod(lagrange_coeff[pi] * (0 - pj), shares[i].Zp) * inv; - lagrange_coeff[pi] = jiff.helpers.mod(lagrange_coeff[pi], shares[i].Zp); - } - } - } - - // Reconstruct the secret via Lagrange interpolation - var recons_secret = 0; - for (var p = 0; p < shares.length; p++) { - var party = jiff.helpers.get_party_number(shares[p].sender_id); - var tmp = jiff.helpers.mod((shares[p].value * lagrange_coeff[party]), shares[p].Zp); - recons_secret = jiff.helpers.mod((recons_secret + tmp), shares[p].Zp); - } - - return recons_secret; - } -}; -},{}],40:[function(require,module,exports){ -/** - * re-share an existing share (value) under a new threshold or to a new set of parties or both. - * Do not use this to refresh a share (use {@link jiff-client~JIFFClient#SecretShare#refresh} instead) - * @function reshare - * @ignore - * @param {module:jiff-client~JIFFClient} jiff - the jiff client instance - * @param {module:jiff-client~JIFFClient#SecretShare} [share=null] - the share you would like to reshare (null if you are a receiver but not a sender) - * @param {number} [threshold=receivers_list.length] - the new threshold, defaults to the length of receivers_list param - * @param {Array} [receivers_list=all_parties] - array of party ids to receive from, by default, this includes all parties - * @param {Array} [senders_list=all_parties] - array of party ids that posses the share and will reshare it with the receivers, by default, this includes all parties - * @param {number} [Zp=jiff.Zp] - the Zp of the existing share - * @param {string} [op_id=auto_gen()] - the operation id which is used to identify this multiplication (and internally, the corresponding beaver triplet). - * This id must be unique, and must be passed by all parties to the same instruction. - * this ensures that every party gets a share from the same triplet for every matching instruction. An automatic id - * is generated by increasing a local counter, default ids suffice when all parties execute the - * instructions in the same order - * @return {SecretShare} this party's share of the result under the new threshold, or null if this party is not a receiver - */ -module.exports = function (jiff, share, threshold, receivers_list, senders_list, Zp, op_id) { - var i; - - // default values - if (receivers_list == null) { - receivers_list = []; - for (i = 1; i <= jiff.party_count; i++) { - receivers_list.push(i); - } - } else { - jiff.helpers.sort_ids(receivers_list); - } - if (senders_list == null) { - senders_list = []; - for (i = 1; i <= jiff.party_count; i++) { - senders_list.push(i); - } - } else { - jiff.helpers.sort_ids(senders_list); - } - - if (threshold == null) { - threshold = receivers_list.length; - } - if (Zp == null) { - Zp = jiff.Zp; - } - - if (op_id == null) { - op_id = jiff.counters.gen_op_id('reshare', senders_list); - } - - // Check if this party is a sender or receiver - var isSender = senders_list.indexOf(jiff.id) > -1; - var isReceiver = receivers_list.indexOf(jiff.id) > -1; - if (!isSender && !isReceiver) { - return null; - } - - // optimization, if nothing changes, keep share - if (share != null && JSON.stringify(receivers_list) === JSON.stringify(senders_list) && threshold === share.threshold) { - return share; - } + module.exports = StreamCipher; + }, + { './aes': 78, './ghash': 83, './incr32': 84, 'buffer-xor': 123, 'cipher-base': 125, inherits: 218, 'safe-buffer': 269 } + ], + 80: [ + function (require, module, exports) { + var ciphers = require('./encrypter'); + var deciphers = require('./decrypter'); + var modes = require('./modes/list.json'); + + function getCiphers() { + return Object.keys(modes); + } - // Setup the result - var final_deferred; - var result = null; - if (isReceiver) { - final_deferred = new jiff.helpers.Deferred(); - result = new jiff.SecretShare(final_deferred.promise, receivers_list, threshold, Zp); - } + exports.createCipher = exports.Cipher = ciphers.createCipher; + exports.createCipheriv = exports.Cipheriv = ciphers.createCipheriv; + exports.createDecipher = exports.Decipher = deciphers.createDecipher; + exports.createDecipheriv = exports.Decipheriv = deciphers.createDecipheriv; + exports.listCiphers = exports.getCiphers = getCiphers; + }, + { './decrypter': 81, './encrypter': 82, './modes/list.json': 92 } + ], + 81: [ + function (require, module, exports) { + var AuthCipher = require('./authCipher'); + var Buffer = require('safe-buffer').Buffer; + var MODES = require('./modes'); + var StreamCipher = require('./streamCipher'); + var Transform = require('cipher-base'); + var aes = require('./aes'); + var ebtk = require('evp_bytestokey'); + var inherits = require('inherits'); + + function Decipher(mode, key, iv) { + Transform.call(this); + + this._cache = new Splitter(); + this._last = void 0; + this._cipher = new aes.AES(key); + this._prev = Buffer.from(iv); + this._mode = mode; + this._autopadding = true; + } - // This function is called when the share is ready: the value of the share has been received. - var ready_share = function () { - var intermediate_shares = jiff.internal_share(isSender ? share.value : null, threshold, receivers_list, senders_list, Zp, op_id); + inherits(Decipher, Transform); - if (isReceiver) { - var promises = []; - for (var i = 0; i < senders_list.length; i++) { - var party_id = senders_list[i]; - promises.push(intermediate_shares[party_id].value); - } + Decipher.prototype._update = function (data) { + this._cache.add(data); + var chunk; + var thing; + var out = []; + while ((chunk = this._cache.get(this._autopadding))) { + thing = this._mode.decrypt(this, chunk); + out.push(thing); + } + return Buffer.concat(out); + }; + + Decipher.prototype._final = function () { + var chunk = this._cache.flush(); + if (this._autopadding) { + return unpad(this._mode.decrypt(this, chunk)); + } else if (chunk) { + throw new Error('data not multiple of block length'); + } + }; - // Reconstruct share under new threshold - Promise.all(promises).then(function () { - var reconstruct_parts = []; - for (var i = 0; i < senders_list.length; i++) { - var party_id = senders_list[i]; - //shamir reconstruct takes an array of objects - //has attributes: {value: x, sender_id: y, Zp: jiff_instance.Zp} - reconstruct_parts[i] = {value: intermediate_shares[party_id].value, sender_id: party_id, Zp: Zp}; - } - var value = jiff.hooks.reconstructShare(jiff, reconstruct_parts); - final_deferred.resolve(value); - }); - } - }; + Decipher.prototype.setAutoPadding = function (setTo) { + this._autopadding = !!setTo; + return this; + }; - if (isSender && !share.ready) { - share.value.then(ready_share); - } else { // either a receiver or share is ready - ready_share(); - } + function Splitter() { + this.cache = Buffer.allocUnsafe(0); + } - return result; -}; -},{}],41:[function(require,module,exports){ -module.exports = { - /** - * Default way of computing shares (can be overridden using hooks). - * Compute the shares of the secret (as many shares as parties) using Shamir secret sharing - * @ignore - * @function jiff_compute_shares - * @param {module:jiff-client~JIFFClient} jiff - the jiff instance - * @param {number} secret - the secret to share. - * @param {Array} parties_list - array of party ids to share with. - * @param {number} threshold - the min number of parties needed to reconstruct the secret, defaults to all the receivers. - * @param {number} Zp - the mod. - * @returns {object} a map between party number and its share, this means that (party number, share) is a - * point from the polynomial. - * - */ - jiff_compute_shares: function (jiff, secret, parties_list, threshold, Zp) { - var shares = {}; // Keeps the shares - var i; - - // Each player's random polynomial f must have - // degree threshold - 1, so that threshold many points are needed - // to interpolate/reconstruct. - var t = threshold - 1; - var polynomial = Array(t + 1); // stores the coefficients - - // Each players's random polynomial f must be constructed - // such that f(0) = secret - polynomial[0] = secret; - - // Compute the random polynomial f's coefficients - for (i = 1; i <= t; i++) { - polynomial[i] = jiff.helpers.random(Zp); - } + Splitter.prototype.add = function (data) { + this.cache = Buffer.concat([this.cache, data]); + }; + + Splitter.prototype.get = function (autoPadding) { + var out; + if (autoPadding) { + if (this.cache.length > 16) { + out = this.cache.slice(0, 16); + this.cache = this.cache.slice(16); + return out; + } + } else { + if (this.cache.length >= 16) { + out = this.cache.slice(0, 16); + this.cache = this.cache.slice(16); + return out; + } + } - // Compute each players share such that share[i] = f(i) - for (i = 0; i < parties_list.length; i++) { - var p_id = parties_list[i]; - shares[p_id] = polynomial[0]; - var power = jiff.helpers.get_party_number(p_id); + return null; + }; - for (var j = 1; j < polynomial.length; j++) { - var tmp = jiff.helpers.mod((polynomial[j] * power), Zp); - shares[p_id] = jiff.helpers.mod((shares[p_id] + tmp), Zp); - power = jiff.helpers.mod(power * jiff.helpers.get_party_number(p_id), Zp); - } - } + Splitter.prototype.flush = function () { + if (this.cache.length) return this.cache; + }; - return shares; - }, - /** - * Share given secret to the participating parties. - * @ignore - * @function jiff_share - * @param {module:jiff-client~JIFFClient} jiff - the jiff instance - * @param {number} secret - the secret to share. - * @param {number} [threshold=receivers_list.length] - the min number of parties needed to reconstruct the secret, defaults to all the receivers. - * @param {Array} [receivers_list=all_parties] - array of party ids to share with, by default, this includes all parties. - * @param {Array} [senders_list=all_parties] - array of party ids to receive from, by default, this includes all parties. - * @param {number} [Zp=jiff.Zp] - the mod (if null then the default Zp for the instance is used). - * @param {string|number} [share_id=auto_gen()] - the tag used to tag the messages sent by this share operation, this tag is used - * so that parties distinguish messages belonging to this share operation from other - * share operations between the same parties (when the order of execution is not - * deterministic). An automatic id is generated by increasing a local counter, default - * ids suffice when all parties execute all sharing operations with the same senders - * and receivers in the same order. - * @returns {object} a map where the key is the sender party id - * and the value is the share object that wraps - * what was sent from that party (the internal value maybe deferred). - * if the party that calls this function is not a receiver then the map - * will be empty. - */ - jiff_share: function (jiff, secret, threshold, receivers_list, senders_list, Zp, share_id) { - var i, p_id; - - // defaults - if (Zp == null) { - Zp = jiff.Zp; - } - if (receivers_list == null) { - receivers_list = []; - for (i = 1; i <= jiff.party_count; i++) { - receivers_list.push(i); - } - } else { - jiff.helpers.sort_ids(receivers_list); - } - if (senders_list == null) { - senders_list = []; - for (i = 1; i <= jiff.party_count; i++) { - senders_list.push(i); - } - } else { - jiff.helpers.sort_ids(senders_list); - } - if (threshold == null) { - threshold = receivers_list.length; - } - if (threshold < 0) { - threshold = 2; - } - if (threshold > receivers_list.length) { - threshold = receivers_list.length; - } + function unpad(last) { + var padded = last[15]; + if (padded < 1 || padded > 16) { + throw new Error('unable to decrypt data'); + } + var i = -1; + while (++i < padded) { + if (last[i + (16 - padded)] !== padded) { + throw new Error('unable to decrypt data'); + } + } + if (padded === 16) return; - // if party is uninvolved in the share, do nothing - if (receivers_list.indexOf(jiff.id) === -1 && senders_list.indexOf(jiff.id) === -1) { - return {}; - } + return last.slice(0, 16 - padded); + } - // compute operation id - if (share_id == null) { - share_id = jiff.counters.gen_op_id2('share', receivers_list, senders_list); - } + function createDecipheriv(suite, password, iv) { + var config = MODES[suite.toLowerCase()]; + if (!config) throw new TypeError('invalid suite type'); - // stage sending of shares - if (senders_list.indexOf(jiff.id) > -1) { - // Call hook - secret = jiff.hooks.execute_array_hooks('beforeShare', [jiff, secret, threshold, receivers_list, senders_list, Zp], 1); + if (typeof iv === 'string') iv = Buffer.from(iv); + if (config.mode !== 'GCM' && iv.length !== config.iv) throw new TypeError('invalid iv length ' + iv.length); - // compute shares - var shares = jiff.hooks.computeShares(jiff, secret, receivers_list, threshold, Zp); + if (typeof password === 'string') password = Buffer.from(password); + if (password.length !== config.key / 8) throw new TypeError('invalid key length ' + password.length); - // Call hook - shares = jiff.hooks.execute_array_hooks('afterComputeShare', [jiff, shares, threshold, receivers_list, senders_list, Zp], 1); + if (config.type === 'stream') { + return new StreamCipher(config.module, password, iv, true); + } else if (config.type === 'auth') { + return new AuthCipher(config.module, password, iv, true); + } - // send shares - for (i = 0; i < receivers_list.length; i++) { - p_id = receivers_list[i]; - if (p_id === jiff.id) { - continue; - } + return new Decipher(config.module, password, iv); + } - // send encrypted and signed shares_id[p_id] to party p_id - var msg = {party_id: p_id, share: shares[p_id], op_id: share_id}; - msg = jiff.hooks.execute_array_hooks('beforeOperation', [jiff, 'share', msg], 2); + function createDecipher(suite, password) { + var config = MODES[suite.toLowerCase()]; + if (!config) throw new TypeError('invalid suite type'); - msg['share'] = jiff.hooks.encryptSign(jiff, msg['share'].toString(10), jiff.keymap[msg['party_id']], jiff.secret_key); - jiff.socket.safe_emit('share', JSON.stringify(msg)); - } - } + var keys = ebtk(password, false, config.key, config.iv); + return createDecipheriv(suite, keys.key, keys.iv); + } - // stage receiving of shares - var result = {}; - if (receivers_list.indexOf(jiff.id) > -1) { - // setup a map of deferred for every received share - if (jiff.deferreds[share_id] == null) { - jiff.deferreds[share_id] = {}; - } + exports.createDecipher = createDecipher; + exports.createDecipheriv = createDecipheriv; + }, + { './aes': 78, './authCipher': 79, './modes': 91, './streamCipher': 94, 'cipher-base': 125, evp_bytestokey: 182, inherits: 218, 'safe-buffer': 269 } + ], + 82: [ + function (require, module, exports) { + var MODES = require('./modes'); + var AuthCipher = require('./authCipher'); + var Buffer = require('safe-buffer').Buffer; + var StreamCipher = require('./streamCipher'); + var Transform = require('cipher-base'); + var aes = require('./aes'); + var ebtk = require('evp_bytestokey'); + var inherits = require('inherits'); + + function Cipher(mode, key, iv) { + Transform.call(this); + + this._cache = new Splitter(); + this._cipher = new aes.AES(key); + this._prev = Buffer.from(iv); + this._mode = mode; + this._autopadding = true; + } - var _remaining = senders_list.length; - for (i = 0; i < senders_list.length; i++) { - p_id = senders_list[i]; - if (p_id === jiff.id) { // Keep party's own share - var my_share = jiff.hooks.execute_array_hooks('receiveShare', [jiff, p_id, shares[p_id]], 2); - result[p_id] = new jiff.SecretShare(my_share, receivers_list, threshold, Zp); - _remaining--; - continue; - } + inherits(Cipher, Transform); - // check if a deferred is set up (maybe the message was previously received) - if (jiff.deferreds[share_id][p_id] == null) { // not ready, setup a deferred - jiff.deferreds[share_id][p_id] = new jiff.helpers.Deferred(); - } + Cipher.prototype._update = function (data) { + this._cache.add(data); + var chunk; + var thing; + var out = []; - var promise = jiff.deferreds[share_id][p_id].promise; - // destroy deferred when done - (function (promise, p_id) { // p_id is modified in a for loop, must do this to avoid scoping issues. - promise.then(function () { - delete jiff.deferreds[share_id][p_id]; - _remaining--; - if (_remaining === 0) { - delete jiff.deferreds[share_id]; + while ((chunk = this._cache.get())) { + thing = this._mode.encrypt(this, chunk); + out.push(thing); } - }); - })(promise, p_id); - - // receive share_i[id] from party p_id - result[p_id] = new jiff.SecretShare(promise, receivers_list, threshold, Zp); - } - } - return result; - } -}; -},{}],42:[function(require,module,exports){ -var genericProtocols = require('./protocols/generic.js'); -var arithmetic = require('./protocols/numbers/arithmetic.js'); -var comparison = require('./protocols/numbers/comparison.js'); -var protocols = require('./protocols/numbers/protocols.js'); -var booleans = require('./protocols/booleans/boolean.js'); - -// a metaclass that creates SecretShare classes when given a jiff instance -// alternatively, we can think of this as a factory for a secret share prototypes/constructors given a jiff instance -module.exports = function (jiff) { - // Look at jiff-client#SecretShare - function SecretShare(value, holders, threshold, Zp) { - // sort holders - jiff.helpers.sort_ids(holders); - - /** - * Indicates if the secret share's value is ready or still a promise - * @member {boolean} ready - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - this.ready = (value.then == null); - - /** - * The value of the share (or a promise to it) - * @member {number|promise} value - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - this.value = value; - - /** - * Array of party ids who hold shares of the corresponding secret - * @member {Array} holders - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - this.holders = holders; - /** - * The sharing threshold - * @member {number} threshold - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - this.threshold = threshold; - - /** - * The field prime under which the corresponding secret is shared - * @member {number} Zp - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - this.Zp = Zp; - - // when the promise is resolved, acquire the value of the share and set ready to true - if (!this.ready) { - this.value = this.value.then(this.promise_handler.bind(this), this.error.bind(this)); - this.jiff.add_to_barriers(this.value); - } - - // return the share - return jiff.hooks.execute_array_hooks('createSecretShare', [jiff, this], 1); - } - - /** - * The jiff client instance this share belongs to - * @member {module:jiff-client~JIFFClient} jiff - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.jiff = jiff; - - // Basic operations in prototype of SecretShare - /** - * Gets the value of this share - * @method valueOf - * @returns {number} the value (undefined if not ready yet) - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.valueOf = function () { - if (this.ready) { - return this.value; - } else { - return undefined; - } - }; + return Buffer.concat(out); + }; - /** - * Gets a string representation of this share - * @method toString - * @returns {string} the id and value of the share as a string - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - */ - SecretShare.prototype.toString = function () { - var val = this.ready ? this.value : ''; - return 'share: ' + val + '. Holders: ' + JSON.stringify(this.holders) + '. Threshold: ' + this.threshold + '. Zp: ' + this.Zp.toString() + '.'; - }; - - /** - * Logs an error. Passes the error on to the associated jiff client instance's {@link handlers.error} - * @method error - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - * @param {string|error} error - the error to log - */ - SecretShare.prototype.error = function (error) { - this.jiff.handlers.error('SecretShare', error); - }; - - /** - * Logs the value represented by this share to the console - * WARNING: THIS LEAKS INFORMATION AND MUST BE USED ONLY TO DEBUG ON FAKE DATA - * @method logLEAK - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - * @param {string} tag - accompanying tag to display in the console - * @param {Array} [parties=[holders[0]] - the parties which will display the log - * @param {string|number|object} [op_id=auto_gen()] - same as {@link module:jiff-client:JIFFClient#open} - * @return {?promise} a promise to the value represented by this share after logging it, null if party is not in parties - */ - SecretShare.prototype.logLEAK = function (tag, parties, op_id) { - if (parties == null) { - parties = [this.holders[0]]; - } - var promise = this.open(parties, tag, op_id); - if (promise != null) { - promise = promise.then(function (result) { - console.log(tag, result.toString()); - return result; - }, this.error); - } - return promise; - }; + var PADDING = Buffer.alloc(16, 0x10); - /** - * Handler for when this share's promise (if any) is resolved - * @method promise_handler - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - * @param {number} value - the value of the share after it was resolved - */ - SecretShare.prototype.promise_handler = function (value) { - this.value = value; - this.ready = true; - return this.value; - }; + Cipher.prototype._final = function () { + var chunk = this._cache.flush(); + if (this._autopadding) { + chunk = this._mode.encrypt(this, chunk); + this._cipher.scrub(); + return chunk; + } - /** - * Executes callback when both this share and o are ready and returns the result (or a promise to the result) - * @method when_both_ready - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - * @param {SecretShare} o - the other share object. - * @param {function()} cb - the callback to execute. - * @returns {value|promise} either the return value of cb() or a promise to it - */ - SecretShare.prototype.when_both_ready = function (o, cb) { - if (this.ready && o.ready) { - return cb(); - } + if (!chunk.equals(PADDING)) { + this._cipher.scrub(); + throw new Error('data not multiple of block length'); + } + }; - if (this.ready) { - return o.value.then(cb, this.error); - } else if (o.ready) { - return this.value.then(cb, this.error); - } else { - return Promise.all([this.value, o.value]).then(cb, this.error); - } - }; + Cipher.prototype.setAutoPadding = function (setTo) { + this._autopadding = !!setTo; + return this; + }; - /** - * Shortcut for opening/revealing the value of this share. Alias for open in jiff-instance - * @see jiff-instance#open - * @method open - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - * @param {Array} [parties=all_parties] - an array with party ids (1 to n) of receiving parties - * @param {string|number|object} [op_id=auto_gen()] - same as {@link module:jiff-client:JIFFClient#open} - * @returns {?promise} a (JQuery) promise to the open value of the secret, null if the party is not specified in - * the parties array as a receiver - */ - SecretShare.prototype.open = function (parties, op_id) { - return this.jiff.open(this, parties, op_id); - }; + function Splitter() { + this.cache = Buffer.allocUnsafe(0); + } - /** - * Wrapper around share.value.then. - * In case share is ready (its promise is resolved and cleared) - * The callback is executed immediately. - * Does not support chaining - * @method wThen - * @memberof module:jiff-client~JIFFClient#SecretShare - * @instance - * @param {function} onFulfilled - callback for success, called with this.value as parameter - * @param {function} [onRejected=this.error] - callback for errors - * @return {promise|value} either the result of executing onFulfilled or a promise to it - */ - SecretShare.prototype.wThen = function (onFulfilled, onRejected) { - if (this.ready) { - return onFulfilled(this.value); - } else { - if (onRejected == null) { - onRejected = this.error; - } - return this.value.then(onFulfilled, onRejected); - } - }; + Splitter.prototype.add = function (data) { + this.cache = Buffer.concat([this.cache, data]); + }; - // Complex protocols in prototype of SecretShare - genericProtocols(SecretShare); - arithmetic(SecretShare); - comparison(SecretShare); - protocols(SecretShare); - booleans(SecretShare); - - // internal variant of primitives, to use internally by other primitives - var internals = ['cadd', 'csub', 'cmult', 'sadd', 'ssub', 'smult', 'smult_bgw', - 'cxor_bit', 'sxor_bit', 'cor_bit', 'sor_bit', - 'slt', 'slteq', 'sgt', 'sgteq', 'seq', 'sneq', - 'clt', 'clteq', 'cgt', 'cgteq', 'ceq', 'cneq', - 'sdiv', 'cdiv', 'not', 'cpow', 'lt_halfprime', 'if_else']; - for (var i = 0; i < internals.length; i++) { - var key = internals[i]; - SecretShare.prototype['i' + key] = SecretShare.prototype[key]; - } + Splitter.prototype.get = function () { + if (this.cache.length > 15) { + var out = this.cache.slice(0, 16); + this.cache = this.cache.slice(16); + return out; + } + return null; + }; - return SecretShare; -}; -},{"./protocols/booleans/boolean.js":29,"./protocols/generic.js":30,"./protocols/numbers/arithmetic.js":31,"./protocols/numbers/comparison.js":32,"./protocols/numbers/protocols.js":33}],43:[function(require,module,exports){ -module.exports = { - '+': function (v1, v2) { - return v1 + v2; - }, - '-': function (v1, v2) { - return v1 - v2; - }, - '*': function (v1, v2) { - return v1 * v2; - }, - '/': function (v1, v2) { - return v1 / v2; - }, - '<': function (v1, v2) { - return v1 < v2; - }, - '<=': function (v1, v2) { - return v1 <= v2; - }, - 'floor': function (v) { - return Math.floor(v); - }, - 'ceil': function (v) { - return Math.ceil(v); - }, - 'floor/': function (v1, v2) { - return Math.floor(v1 / v2); - }, - 'pow': function (v1, v2) { - return Math.pow(v1, v2); - }, - 'binary': function (v) { - return v === 1 || v === 0; - }, - 'abs': function (v) { - return Math.abs(v); - }, - '==': function (v1, v2) { - return v1 === v2; - }, - 'even': function (v1) { - return (v1 % 2) === 0; - } -}; -},{}],44:[function(require,module,exports){ -module.exports = function (JIFFClient) { - /** - * Initialize socket listeners and events - * @memberof module:jiff-client.JIFFClient - * @method - */ - JIFFClient.prototype.initSocket = function () { - var jiffClient = this; - - // set on('connect') handler once! - this.socket.on('connect', jiffClient.handlers.connected); - - // Store the id when server sends it back - this.socket.on('initialization', jiffClient.handlers.initialized); - - // Public keys were updated on the server, and it sent us the updates - jiffClient.socket.on('public_keys', function (msg, callback) { - callback(true); - - msg = JSON.parse(msg); - msg = jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'public_keys', msg], 2); - - jiffClient.handlers.store_public_keys(msg.public_keys); - }); - - // Setup receiving matching shares - this.socket.on('share', function (msg, callback) { - callback(true); // send ack to server - - // parse message - var json_msg = JSON.parse(msg); - var sender_id = json_msg['party_id']; - - if (jiffClient.keymap[sender_id] != null) { - jiffClient.handlers.receive_share(json_msg); - } else { - if (jiffClient.messagesWaitingKeys[sender_id] == null) { - jiffClient.messagesWaitingKeys[sender_id] = []; - } - jiffClient.messagesWaitingKeys[sender_id].push({label: 'share', msg: json_msg}); - } - }); + Splitter.prototype.flush = function () { + var len = 16 - this.cache.length; + var padBuff = Buffer.allocUnsafe(len); - this.socket.on('open', function (msg, callback) { - callback(true); // send ack to server + var i = -1; + while (++i < len) { + padBuff.writeUInt8(len, i); + } - // parse message - var json_msg = JSON.parse(msg); - var sender_id = json_msg['party_id']; + return Buffer.concat([this.cache, padBuff]); + }; - if (jiffClient.keymap[sender_id] != null) { - jiffClient.handlers.receive_open(json_msg); - } else { - if (jiffClient.messagesWaitingKeys[sender_id] == null) { - jiffClient.messagesWaitingKeys[sender_id] = []; - } - jiffClient.messagesWaitingKeys[sender_id].push({label: 'open', msg: json_msg}); - } - }); - - // handle custom messages - this.socket.on('custom', function (msg, callback) { - callback(true); // send ack to server - - var json_msg = JSON.parse(msg); - var sender_id = json_msg['party_id']; - var encrypted = json_msg['encrypted']; - - if (jiffClient.keymap[sender_id] != null || encrypted !== true) { - jiffClient.handlers.receive_custom(json_msg); - } else { - // key must not exist yet for sender_id, and encrypted must be true - if (jiffClient.messagesWaitingKeys[sender_id] == null) { - jiffClient.messagesWaitingKeys[sender_id] = []; - } - jiffClient.messagesWaitingKeys[sender_id].push({label: 'custom', msg: json_msg}); - } - }); - - this.socket.on('crypto_provider', function (msg, callback) { - callback(true); // send ack to server - jiffClient.handlers.receive_crypto_provider(JSON.parse(msg)); - }); - - this.socket.on('error', function (msg) { - try { - msg = JSON.parse(msg); - jiffClient.handlers.error(msg['label'], msg['error']); - } catch (error) { - jiffClient.handlers.error('socket.io', msg); - } - }); + function createCipheriv(suite, password, iv) { + var config = MODES[suite.toLowerCase()]; + if (!config) throw new TypeError('invalid suite type'); - this.socket.on('disconnect', function (reason) { - if (reason !== 'io client disconnect') { - // check that the reason is an error and not a user initiated disconnect - console.log('Disconnected!', jiffClient.id, reason); - } + if (typeof password === 'string') password = Buffer.from(password); + if (password.length !== config.key / 8) throw new TypeError('invalid key length ' + password.length); - jiffClient.hooks.execute_array_hooks('afterOperation', [jiffClient, 'disconnect', reason], -1); - }); - }; + if (typeof iv === 'string') iv = Buffer.from(iv); + if (config.mode !== 'GCM' && iv.length !== config.iv) throw new TypeError('invalid iv length ' + iv.length); - /** - * Executes all callbacks for which the wait condition has been satisfied. - * Remove all executed callbacks so that they would not be executed in the future. - * @memberof module:jiff-client.JIFFClient - * @method - */ - JIFFClient.prototype.execute_wait_callbacks = function () { - var copy_callbacks = this.wait_callbacks; - this.wait_callbacks = []; - for (var i = 0; i < copy_callbacks.length; i++) { - var wait = copy_callbacks[i]; - var parties = wait.parties; - var callback = wait.callback; - var initialization = wait.initialization; - - // Check if the parties to wait for are now known - var parties_satisfied = this.__initialized || !initialization; - for (var j = 0; j < parties.length; j++) { - var party_id = parties[j]; - if (this.keymap == null || this.keymap[party_id] == null) { - parties_satisfied = false; - break; - } - } + if (config.type === 'stream') { + return new StreamCipher(config.module, password, iv); + } else if (config.type === 'auth') { + return new AuthCipher(config.module, password, iv); + } - if (parties_satisfied) { - callback(this); - } else { - this.wait_callbacks.push(wait); - } - } - }; + return new Cipher(config.module, password, iv); + } - /** - * Resolves all messages that were pending because their senders primary key was previously unknown. - * These messages are decrypted and verified and handled appropriatly before being removed from the wait queue. - * @memberof module:jiff-client.JIFFClient - * @method - */ - JIFFClient.prototype.resolve_messages_waiting_for_keys = function () { - for (var party_id in this.keymap) { - if (!this.keymap.hasOwnProperty(party_id)) { - continue; - } + function createCipher(suite, password) { + var config = MODES[suite.toLowerCase()]; + if (!config) throw new TypeError('invalid suite type'); - var messageQueue = this.messagesWaitingKeys[party_id]; - if (messageQueue == null) { - continue; - } - for (var i = 0; i < messageQueue.length; i++) { - var msg = messageQueue[i]; - if (msg.label === 'share') { - this.handlers.receive_share(msg.msg); - } else if (msg.label === 'open') { - this.handlers.receive_open(msg.msg); - } else if (msg.label === 'custom') { - this.handlers.receive_custom(msg.msg); - } else { - throw new Error('Error resolving pending message: unknown label ' + msg.label); - } - } + var keys = ebtk(password, false, config.key, config.iv); + return createCipheriv(suite, keys.key, keys.iv); + } - this.messagesWaitingKeys[party_id] = null; - } - }; -}; -},{}],45:[function(require,module,exports){ -module.exports = function (jiffClient, __internal_socket) { - __internal_socket.safe_emit = function (label, msg) { - if (label === 'free') { - jiffClient.hooks.execute_array_hooks('afterOperation', [this.jiffClient, 'free', msg], 2); - return; - } + exports.createCipheriv = createCipheriv; + exports.createCipher = createCipher; + }, + { './aes': 78, './authCipher': 79, './modes': 91, './streamCipher': 94, 'cipher-base': 125, evp_bytestokey: 182, inherits: 218, 'safe-buffer': 269 } + ], + 83: [ + function (require, module, exports) { + var Buffer = require('safe-buffer').Buffer; + var ZEROES = Buffer.alloc(16, 0); - __internal_socket.emit(label, msg); - }; + function toArray(buf) { + return [buf.readUInt32BE(0), buf.readUInt32BE(4), buf.readUInt32BE(8), buf.readUInt32BE(12)]; + } - __internal_socket.resend_mailbox = function () {}; + function fromArray(out) { + var buf = Buffer.allocUnsafe(16); + buf.writeUInt32BE(out[0] >>> 0, 0); + buf.writeUInt32BE(out[1] >>> 0, 4); + buf.writeUInt32BE(out[2] >>> 0, 8); + buf.writeUInt32BE(out[3] >>> 0, 12); + return buf; + } - __internal_socket.disconnect = function () { - jiffClient.hooks.execute_array_hooks('beforeOperation', [jiffClient, 'disconnect', {}], -1); - }; + function GHASH(key) { + this.h = key; + this.state = Buffer.alloc(16, 0); + this.cache = Buffer.allocUnsafe(0); + } - __internal_socket.safe_disconnect = function (free, callback) { - if (free) { - jiffClient.free(); - } - jiffClient.socket.disconnect(); - if (callback != null) { - callback(); - } - }; + // from http://bitwiseshiftleft.github.io/sjcl/doc/symbols/src/core_gcm.js.html + // by Juho Vähä-Herttua + GHASH.prototype.ghash = function (block) { + var i = -1; + while (++i < block.length) { + this.state[i] ^= block[i]; + } + this._multiply(); + }; + + GHASH.prototype._multiply = function () { + var Vi = toArray(this.h); + var Zi = [0, 0, 0, 0]; + var j, xi, lsbVi; + var i = -1; + while (++i < 128) { + xi = (this.state[~~(i / 8)] & (1 << (7 - (i % 8)))) !== 0; + if (xi) { + // Z_i+1 = Z_i ^ V_i + Zi[0] ^= Vi[0]; + Zi[1] ^= Vi[1]; + Zi[2] ^= Vi[2]; + Zi[3] ^= Vi[3]; + } + + // Store the value of LSB(V_i) + lsbVi = (Vi[3] & 1) !== 0; + + // V_i+1 = V_i >> 1 + for (j = 3; j > 0; j--) { + Vi[j] = (Vi[j] >>> 1) | ((Vi[j - 1] & 1) << 31); + } + Vi[0] = Vi[0] >>> 1; + + // If LSB(V_i) is 1, V_i+1 = (V_i >> 1) ^ R + if (lsbVi) { + Vi[0] = Vi[0] ^ (0xe1 << 24); + } + } + this.state = fromArray(Zi); + }; + + GHASH.prototype.update = function (buf) { + this.cache = Buffer.concat([this.cache, buf]); + var chunk; + while (this.cache.length >= 16) { + chunk = this.cache.slice(0, 16); + this.cache = this.cache.slice(16); + this.ghash(chunk); + } + }; - return __internal_socket; -}; -},{}],46:[function(require,module,exports){ -var io = require('socket.io-client'); + GHASH.prototype.final = function (abl, bl) { + if (this.cache.length) { + this.ghash(Buffer.concat([this.cache, ZEROES], 16)); + } -var linked_list = require('../../common/linkedlist.js'); -var constants = require('../util/constants.js'); + this.ghash(fromArray([0, abl, 0, bl])); + return this.state; + }; -var defaultSocketOptions = { - reconnectionDelay: constants.reconnectionDelay, - reconnectionDelayMax: constants.reconnectionDelayMax, - randomizationFactor: constants.randomizationFactor, - autoConnect: false -}; + module.exports = GHASH; + }, + { 'safe-buffer': 269 } + ], + 84: [ + function (require, module, exports) { + function incr32(iv) { + var len = iv.length; + var item; + while (len--) { + item = iv.readUInt8(len); + if (item === 255) { + iv.writeUInt8(0, len); + } else { + item++; + iv.writeUInt8(item, len); + break; + } + } + } + module.exports = incr32; + }, + {} + ], + 85: [ + function (require, module, exports) { + var xor = require('buffer-xor'); -/** - * A guarded socket with an attached mailbox. - * - * The socket uses the mailbox to store all outgoing messages, and removes them from the mailbox only when - * the server acknowledges their receipt. The socket resends mailbox upon re-connection. Extends {@link https://socket.io/docs/client-api/#Socket}. - * @see {@link module:jiff-client~JIFFClient#socket} - * @name GuardedSocket - * @alias GuardedSocket - * @constructor - */ + exports.encrypt = function (self, block) { + var data = xor(block, self._prev); -function guardedSocket(jiffClient) { - jiffClient.options.socketOptions = Object.assign({}, defaultSocketOptions, jiffClient.options.socketOptions); - - // Create plain socket io object which we will wrap in this - var socket = io(jiffClient.hostname, jiffClient.options.socketOptions); - socket.old_disconnect = socket.disconnect; - socket.mailbox = linked_list(); // for outgoing messages - socket.empty_deferred = null; // gets resolved whenever the mailbox is empty - socket.jiffClient = jiffClient; - - // add functionality to socket - socket.safe_emit = safe_emit.bind(socket); - socket.resend_mailbox = resend_mailbox.bind(socket); - socket.disconnect = disconnect.bind(socket); - socket.safe_disconnect = safe_disconnect.bind(socket); - socket.is_empty = is_empty.bind(socket); - - return socket; -} - -/** - * Safe emit: stores message in the mailbox until acknowledgment is received, results in socket.emit(label, msg) call(s) - * @method safe_emit - * @memberof GuardedSocket - * @instance - * @param {string} label - the label given to the message - * @param {string} msg - the message to send - */ -var safe_emit = function (label, msg) { - // add message to mailbox - var mailbox_pointer = this.mailbox.add({ label: label, msg: msg }); - if (this.connected) { - var self = this; - // emit the message, if an acknowledgment is received, remove it from mailbox - this.emit(label, msg, function (status) { - if (status) { - self.mailbox.remove(mailbox_pointer); - if (this.is_empty() && self.empty_deferred != null) { - self.empty_deferred.resolve(); - } + self._prev = self._cipher.encryptBlock(data); + return self._prev; + }; - if (label === 'free') { - this.jiffClient.hooks.execute_array_hooks('afterOperation', [this.jiffClient, 'free', msg], 2); - } - } - }); - } -}; + exports.decrypt = function (self, block) { + var pad = self._prev; -/** - * Re-sends all pending messages - * @method resend_mailbox - * @memberof GuardedSocket - * @instance - */ -var resend_mailbox = function () { - // Create a new mailbox, since the current mailbox will be resent and - // will contain new backups. - var old_mailbox = this.mailbox; - this.mailbox = linked_list(); - - // loop over all stored messages and emit them - var current_node = old_mailbox.head; - while (current_node != null) { - var label = current_node.object.label; - var msg = current_node.object.msg; - this.safe_emit(label, msg); - current_node = current_node.next; - } -}; - -/** - * Wraps socket.io regular disconnect with a call to a hook before disconnection - * @method disconnect - * @memberof GuardedSocket - * @instance - */ -var disconnect = function () { - this.jiffClient.hooks.execute_array_hooks('beforeOperation', [this.jiffClient, 'disconnect', {}], -1); - this.old_disconnect.apply(this, arguments); -}; - -/** - * Safe disconnect: disconnect only after all messages (including free) were acknowledged and - * all pending opens were resolved - * @method safe_disconnect - * @memberof GuardedSocket - * @instance - * @param {boolean} [free=false] - if true, a free message will be issued prior to disconnecting - * @param {function()} [callback] - given callback will be executed after safe disconnection is complete - */ -var safe_disconnect = function (free, callback) { - if (this.is_empty()) { - if (free) { - this.jiffClient.free(); - free = false; - } else { - this.disconnect(); - if (callback != null) { - callback(); - } - return; - } - } - - this.empty_deferred = new this.jiffClient.helpers.Deferred(); - this.empty_deferred.promise.then(this.safe_disconnect.bind(this, free, callback)); -}; - -/** - * Checks if the socket mailbox is empty (all communication was done and acknowledged), - * used in safe_disconnect - * @method is_empty - * @memberof GuardedSocket - * @instance - */ -var is_empty = function () { - return this.mailbox.head == null && this.jiffClient.counters.pending_opens === 0; -}; - -module.exports = guardedSocket; -},{"../../common/linkedlist.js":52,"../util/constants.js":47,"socket.io-client":278}],47:[function(require,module,exports){ -module.exports = { - /** - * The default mod to be used in a jiff instance if a custom mod was not provided. - */ - gZp: 16777729, - - /** - * Socket connection timeouts - */ - reconnectionDelay: 25000, - reconnectionDelayMax: 27500, - randomizationFactor: 0.1, - - /** - * Maximum numbers of retries on failed initialization. - */ - maxInitializationRetries: 2 -}; -},{}],48:[function(require,module,exports){ -/** - * Encrypts and signs the given message. - * @ignore - * @memberof jiff.utils - * @param {number|string} message - the message to encrypt. - * @param {Uint8Array} encryption_public_key - ascii-armored public key to encrypt with. - * @param {Uint8Array} signing_private_key - the private key of the encrypting party to sign with. - * @returns {object} the signed cipher, includes two properties: 'cipher' and 'nonce'. - */ -exports.encrypt_and_sign = function (jiff, message, encryption_public_key, signing_private_key) { - var nonce = jiff.sodium_.randombytes_buf(jiff.sodium_.crypto_box_NONCEBYTES); - var cipher = jiff.sodium_.crypto_box_easy(message, nonce, encryption_public_key, signing_private_key); - - var result = { nonce: '[' + nonce.toString() + ']', cipher: '[' + cipher.toString() + ']' }; - return result; -}; - -/** - * Decrypts and checks the signature of the given cipher text. - * @ignore - * @memberof jiff.utils - * @param {object} cipher_text - the cipher text to decrypt, includes two properties: 'cipher' and 'nonce'. - * @param {Uint8Array} decryption_secret_key - the secret key to decrypt with. - * @param {Uint8Array} signing_public_key - ascii-armored public key to verify against signature. - * @returns {number|string} the decrypted message if the signature was correct, the decrypted message type should - * the type of operation, such that the returned value has the appropriate type and does - * not need any type modifications. - * @throws error if signature or nonce was forged/incorrect. - */ -exports.decrypt_and_sign = function (jiff, cipher_text, decryption_secret_key, signing_public_key) { - var nonce = new Uint8Array(JSON.parse(cipher_text.nonce)); - cipher_text = new Uint8Array(JSON.parse(cipher_text.cipher)); - - try { - return jiff.sodium_.crypto_box_open_easy(cipher_text, nonce, signing_public_key, decryption_secret_key, 'text'); - } catch (_) { - throw new Error('Bad signature or Bad nonce: Cipher: ' + cipher_text + '. DecSKey: ' + decryption_secret_key + '. SignPKey: ' + signing_public_key); - } -}; -},{}],49:[function(require,module,exports){ -var helpers = require('../../common/helpers.js'); - -/** - * Contains helper functions: these may be overriden by extensions to customize behavior - * @see {@link module:jiff-client~JIFFClient#helpers} - * @name helpers - * @alias helpers - * @namespace - */ - -module.exports = function (jiffClient) { - /** - * Polyfill for jQuery Deferred - * From https://developer.mozilla.org/en-US/docs/Mozilla/JavaScript_code_modules/Promise.jsm/Deferred - * @memberof helpers - * @constructor Deferred - */ - jiffClient.helpers.Deferred = function () { - /** - * A method to resolve the associate Promise with the value passed. - * @method resolve - * @memberof helpers.Deferred - * @instance - * @param {*} value - the value to resolve the promise with - */ - this.resolve = null; - - /** - * A method to reject the associated Promise with the value passed. - * If the promise is already settled it does nothing. - * @method reject - * @memberof helpers.Deferred - * @instance - * @param {*} reason - The reason for the rejection of the Promise. - * Generally its an Error object. If however a Promise is passed, then the Promise - * itself will be the reason for rejection no matter the state of the Promise. - */ - this.reject = null; - - /** - * A newly created Promise object. - * Initially in pending state. - * @memberof helpers.Deferred - * @member {Promise} promise - * @instance - */ - this.promise = new Promise(function (resolve, reject) { - this.resolve = resolve; - this.reject = reject; - }.bind(this)); - - Object.freeze(this); - }; - - /** - * Correct Mod instead of javascript's remainder (%). - * @memberof helpers - * @method - * @param {number} x - the number. - * @param {number} y - the mod. - * @return {number} x mod y. - */ - jiffClient.helpers.mod = helpers.mod; - - /** - * Ceil of a number. - * @memberof helpers - * @method - * @param {number} x - the number to ceil. - * @return {number} ceil of x. - */ - jiffClient.helpers.ceil = Math.ceil; - - /** - * Floor of a number - * @memberof helpers - * @method - * @param {number} x - the number to floor. - * @return {number} floor of x. - */ - jiffClient.helpers.floor = Math.floor; - - /** - * Fast Exponentiation Mod - * @memberof helpers - * @method - * @param {number} a - the base number - * @param {number} b - the power - * @param {number} n - the mod - * @return {number} (base^pow) mod m - */ - jiffClient.helpers.pow_mod = function (a, b, n) { - a = jiffClient.helpers.mod(a, n); - var result = 1; - var x = a; - while (b > 0) { - var leastSignificantBit = jiffClient.helpers.mod(b, 2); - b = Math.floor(b / 2); - if (leastSignificantBit === 1) { - result = result * x; - result = jiffClient.helpers.mod(result, n); - } - x = x * x; - x = jiffClient.helpers.mod(x, n); - } - return result; - }; - - /** - * Extended Euclidean for finding inverses. - * @method - * @memberof helpers - * @param {number} a - the number to find inverse for. - * @param {number} b - the mod. - * @return {number[]} [inverse of a mod b, coefficient for a, coefficient for b]. - */ - jiffClient.helpers.extended_gcd = function (a, b) { - if (b === 0) { - return [1, 0, a]; - } - - var temp = jiffClient.helpers.extended_gcd(b, jiffClient.helpers.mod(a, b)); - var x = temp[0]; - var y = temp[1]; - var d = temp[2]; - return [y, x - y * Math.floor(a / b), d]; - }; - - /** - * Compute Log to a given base. - * @method - * @memberof helpers - * @param {number} value - the number to find log for. - * @param {number} [base=2] - the base (2 by default). - * @return {number} log(value) with the given base. - */ - jiffClient.helpers.bLog = function (value, base) { - if (base == null) { - base = 2; - } - return Math.log(value) / Math.log(base); - }; - - /** - * Check that two sorted arrays are equal. - * @method - * @memberof helpers - * @param {Array} arr1 - the first array. - * @param {Array} arr2 - the second array. - * @return {boolean} true if arr1 is equal to arr2, false otherwise. - */ - jiffClient.helpers.array_equals = function (arr1, arr2) { - if (arr1.length !== arr2.length) { - return false; - } - - for (var i = 0; i < arr1.length; i++) { - if (arr1[i] !== arr2[i]) { - return false; - } - } - - return true; - }; - - /** - * Check that two Zps are equal. Used to determine if shares can be computed on or not. - * @method - * @memberof helpers - * @param {SecretShare} s1 - the first share. - * @param {SecretShare} s2 - the second share. - * @return {boolean} true both shares have the same Zp, false otherwise. - */ - jiffClient.helpers.Zp_equals = function (s1, s2) { - return s1.Zp === s2.Zp; - }; - - /** - * Generate a random integer between 0 and max-1 [inclusive]. - * Modify this to change the source of randomness and how it is generated. - * @method - * @memberof helpers - * @param {number} max - the maximum number. - * @return {number} the random number. - */ - jiffClient.helpers.random = helpers.random; - - /** - * Get the party number from the given party_id, the number is used to compute/open shares. - * If party id was a number (regular party), that number is returned, - * If party id refers to the ith server, then party_count + i is returned (i > 0). - * @method - * @memberof helpers - * @param {number|string} party_id - the party id from which to compute the number. - * @return {number} the party number (> 0). - */ - jiffClient.helpers.get_party_number = helpers.get_party_number; - - /** - * Transforms the given number to an array of bits (numbers). - * Lower indices in the returned array corresponding to less significant bits. - * @memberof helpers - * @method - * @param {number} number - the number to transform to binary - * @param {length} [length=ceil(log2(number))] - if provided, then the given array will be padded with zeros to the length. - * @return {number[]} the array of bits. - */ - jiffClient.helpers.number_to_bits = helpers.number_to_bits; - - /** - * Transforms the given array of bits to a number. - * @memberof helpers - * @method - * @param {number[]} bits - the array of bits to compose as a number, starting from least to most significant bits. - * @param {number} [length = bits.length] - if provided, only the first 'length' bits will be used - * @return {number} the array of bits. - */ - jiffClient.helpers.bits_to_number = function (bits, length) { - if (length == null || length > bits.length) { - length = bits.length; - } - return parseInt(bits.slice(0, length).reverse().join(''), 2); - }; - - /** - * Checks if the given number is prime using AKS primality test - * @method - * @memberof helpers - * @param {number} p - the number to check - * @return {boolean} true if p is prime, false otherwise - */ - jiffClient.helpers.is_prime = function (p) { - // AKS Primality Test - - if (p === 2) { - return true; - } else if (p === 3) { - return true; - } else if (p % 2 === 0) { - return false; - } else if (p % 3 === 0) { - return false; - } - - var i = 5; - var n = 2; - while (i * i <= p) { - if (p % i === 0) { - return false; - } - i += n; - n = 6 - n; - } - - return true; - }; - - /** - * sorts an array of ids (in place) according to a consistent ordering - * @method - * @memberof helpers - * @param {array} ids - array of ids containing numbers or "s1" - */ - jiffClient.helpers.sort_ids = function (ids) { - if (ids.__jiff_sorted) { - return; - } - - ids.sort(function (e1, e2) { - if (e1 === e2) { - throw new Error('ids array has duplicated: ' + ids.toString()); - } - if (e1 === 's1') { - return 1; - } - if (e2 === 's1') { - return -1; - } - return e1 - e2; - }); - ids.__jiff_sorted = true; - } -}; -},{"../../common/helpers.js":51}],50:[function(require,module,exports){ -/** - * Contains utility functions (higher order combinators) - * @see {@link module:jiff-client~JIFFClient#utils} - * @name utils - * @alias utils - * @namespace - */ - -module.exports = function (jiffClient) { - /** - * Create an array of secret shares and associated deferred. - * @method - * @memberof utils - * @param {number} count - number of secret shares - * @param {Array} holders - the parties that hold all the corresponding shares (must be sorted) - * @param {number} threshold - the min number of parties needed to reconstruct the secret - * @param {number} Zp - the mod under which this share was created - * @return {object} the secret share object containing the give value - * - */ - jiffClient.utils.many_secret_shares = function (count, holders, threshold, Zp) { - var deferreds = []; - var shares = []; - for (var i = 0; i < count; i++) { - var deferred = new jiffClient.helpers.Deferred(); - shares.push(new jiffClient.SecretShare(deferred.promise, holders, threshold, Zp)); - deferreds.push(deferred); - } - - return {shares: shares, deferreds: deferreds}; - }; - - /** - * Resolve the array of deferreds with the values of the given shares when ready, matched by index - * @method - * @memberof utils - * @param {Deferred[]} deferreds - the deferred to resolve - * @param {SecretShare[]} shares - the shares to resolve with - */ - jiffClient.utils.resolve_many_secrets = function (deferreds, shares) { - for (var i = 0; i < deferreds.length; i++) { - shares[i].wThen(deferreds[i].resolve); - } - }; - - /** - * Combines all the promises of the given share into a single promise that is resolved when - * all shares are resolved - * @method - * @methodof utils - * @param {SecretShare[]} shares - the shares whose promises should be joined - */ - jiffClient.utils.all_promises = function (shares) { - var promises = []; - for (var i = 0; i < shares.length; i++) { - promises.push(shares[i].value); - } - return Promise.all(promises); - }; - - /** - * A high level combinator for iteration of bit arrays - * It executes a round of (func) starting from index start to the length (supports backwards if start > length) excluding length - * Every round is blocked until the previous one finishes and the promise produced by it - * is resolved - * The final value is used to resolve deferred - * @method - * @memberof utils - * @param {!Deferred} deferred - the deferred to resolve with the final output - * @param {!number} start - the index to start from - * @param {!number} length - the index to stop at (excluding it) - * @param {?object} initial - an initial aggregator value - * @param {!function(number, object)} func - the aggregator function to apply to the elements in order, takes the index and the aggregator value so far - * @param {?function(object)} [promisify] - if initial is not null, this is called prior to starting combinator, to turn initial - * into a promise of the actually used initial value(in case it has to be resolved), - * defaults to promisifying a SecretShare with .wThen - * @param {?function(object)} [valufy] - applied to the final result to turn it into a value, which is then used to resolve deferred, - * defaults to getting value of a SecretShare - */ - jiffClient.utils.bit_combinator = function (deferred, start, length, initial, func, promisify, valufy) { - if (promisify == null) { - promisify = function (share) { - return {then: share.wThen.bind(share)}; - } - } - - if (valufy == null) { - valufy = function (share) { - return share.value; - } - } - - var next = start <= length ? 1 : -1; - var __bit_combinator = function (start, val) { - if (start === length) { - // done - deferred.resolve(valufy(val)); - return; - } - - // execute func once - val = func(start, val); - - // when done, do next iteration - promisify(val).then(function () { - __bit_combinator(start + next, val); - }); - }; - - // start combinator - if (initial == null) { - __bit_combinator(start, initial); - } else { - promisify(initial).then(function () { - __bit_combinator(start, initial); - }); - } - } -}; -},{}],51:[function(require,module,exports){ -var crypto_; -if (typeof(window) === 'undefined') { - crypto_ = require('crypto'); - crypto_.__randomBytesWrapper = crypto_.randomBytes; -} else { - crypto_ = window.crypto || window.msCrypto; - crypto_.__randomBytesWrapper = function (bytesNeeded) { - var randomBytes = new Uint8Array(bytesNeeded); - crypto_.getRandomValues(randomBytes); - return randomBytes; - }; -} - -// Secure randomness via rejection sampling. -exports.random = function (max) { - // Use rejection sampling to get random value within bounds - // Generate random Uint8 values of 1 byte larger than the max parameter - // Reject if random is larger than quotient * max (remainder would cause biased distribution), then try again - - // Values up to 2^53 should be supported, but log2(2^49) === log2(2^49+1), so we lack the precision to easily - // determine how many bytes are required - if (max > 562949953421312) { - throw new RangeError('Max value should be smaller than or equal to 2^49'); - } - - var bitsNeeded = Math.ceil(Math.log(max)/Math.log(2)); - var bytesNeeded = Math.ceil(bitsNeeded / 8); - var maxValue = Math.pow(256, bytesNeeded); - - // Keep trying until we find a random value within bounds - while (true) { // eslint-disable-line - var randomBytes = crypto_.__randomBytesWrapper(bytesNeeded); - var randomValue = 0; - - for (var i = 0; i < bytesNeeded; i++) { - randomValue = randomValue * 256 + (randomBytes.readUInt8 ? randomBytes.readUInt8(i) : randomBytes[i]); - } - - // randomValue should be smaller than largest multiple of max within maxBytes - if (randomValue < maxValue - maxValue % max) { - return randomValue % max; - } - } -}; - -// actual mode -exports.mod = function (x, y) { - if (x < 0) { - return (x % y) + y; - } - return x % y; -}; - -// get the party number from the given party_id, the number is used to compute/open shares -exports.get_party_number = function (party_id) { - if (typeof(party_id) === 'number') { - return party_id; - } - if (party_id.startsWith('s')) { - return -1 * parseInt(party_id.substring(1), 10); - } - return parseInt(party_id, 10); -}; - -// transform number to bit array -exports.number_to_bits = function (number, length) { - number = number.toString(2); - var bits = []; - for (var i = 0; i < number.length; i++) { - bits[i] = parseInt(number.charAt(number.length - 1 - i)); - } - while (length != null && bits.length < length) { - bits.push(0); - } - return bits; -}; -},{"crypto":136}],52:[function(require,module,exports){ -/** Doubly linked list with add and remove functions and pointers to head and tail**/ -module.exports = function () { - // attributes: list.head and list.tail - // functions: list.add(object) (returns pointer), list.remove(pointer) - // list.head/list.tail/any element contains: - // next: pointer to next, - // previous: pointer to previous, - // object: stored object. - var list = {head: null, tail: null}; - // TODO rename this to pushTail || push - list.add = function (obj) { - var node = { object: obj, next: null, previous: null }; - if (list.head == null) { - list.head = node; - list.tail = node; - } else { - list.tail.next = node; - node.previous = list.tail; - list.tail = node; - } - return node; - }; - - list.pushHead = function (obj) { - list.head = {object: obj, next : list.head, previous : null}; - if (list.head.next != null) { - list.head.next.previous = list.head; - } else { - list.tail = list.head; - } - }; - - list.popHead = function () { - var result = list.head; - if (list.head != null) { - list.head = list.head.next; - if (list.head == null) { - list.tail = null; - } else { - list.head.previous = null; - } - } - return result; - }; - - // merges two linked lists and return a pointer to the head of the merged list - // the head will be the head of list and the tail the tail of l2 - list.extend = function (l2) { - if (list.head == null) { - return l2; - } - if (l2.head == null) { - return list; - } - list.tail.next = l2.head; - l2.head.previous = list.tail; - list.tail = l2.tail; - - return list; - }; - - list.remove = function (ptr) { - var prev = ptr.previous; - var next = ptr.next; - - if (prev == null && list.head !== ptr) { - return; - } else if (next == null && list.tail !== ptr) { - return; - } - - if (prev == null) { // ptr is head (or both head and tail) - list.head = next; - if (list.head != null) { - list.head.previous = null; - } else { - list.tail = null; - } - } else if (next == null) { // ptr is tail (and not head) - list.tail = prev; - prev.next = null; - } else { // ptr is inside - prev.next = next; - next.previous = prev; - } - }; - list.slice = function (ptr) { // remove all elements from head to ptr (including ptr). - if (ptr == null) { - return; - } - - /* CONSERVATIVE: make sure ptr is part of the list then remove */ - var current = list.head; - while (current != null) { - if (current === ptr) { - list.head = ptr.next; - if (list.head == null) { - list.tail = null; - } - - return; - } - current = current.next; - } - - /* MORE AGGRESSIVE VERSION: will be incorrect if ptr is not in the list */ - /* - list.head = ptr.next; - if (list.head == null) { - list.tail = null; - } - */ - }; - /* - list._debug_length = function () { - var l = 0; - var current = list.head; - while (current != null) { - current = current.next; - l++; - } - return l; - }; - */ - return list; -}; - -},{}],53:[function(require,module,exports){ -/** - * JIFF Client. - * - * Exposes the constructor for the {@link module:jiff-client~JIFFClient} class. - * - * In the browser, this adds `JIFFClient` as a global identifier. - * - * In the browser, this can be accessed via: - *

- *   <script src="jiff-client.js"></script>
- *   <script type="text/javascript">
- *     var jiffClientInstance = new JIFFClient(hostname, computationId, options);
- *   </script>
- * 
- * - * In node.js, this can be accessed via: - *

- *   const JIFFClient = require('jiffClient');
- *   const jiffClientInstance = new JIFFClient(hostname, computationId, options);
- *
- * 
- * - * @module jiff-client - * @alias jiff-client - */ - -// browserify bundles this into our code bundle -var sodium = require('libsodium-wrappers'); - -// utils and helpers -var constants = require('./client/util/constants.js'); -var helpers = require('./client/util/helpers.js'); -var utils = require('./client/util/utils.js'); -var linkedList = require('./common/linkedlist.js'); - -// hooks -var Hooks = require('./client/arch/hooks.js'); - -// extensions management -var extensions = require('./client/arch/extensions.js'); - -// op ids and other counters -var counters = require('./client/arch/counters.js'); - -// socket and events -var guardedSocket = require('./client/socket/mailbox.js'); -var internalSocket = require('./client/socket/internal.js'); -var socketEvents = require('./client/socket/events.js'); - -// handlers for communication -var handlers = require('./client/handlers.js'); - -// secret shares -var SecretShareMetaClass = require('./client/share.js'); -var share_helpers = require('./client/shareHelpers.js'); - -// jiff client instance API -var api = require('./client/api.js'); - -// preprocessing -var preprocessingMap = require('./client/preprocessing/map.js'); -var preprocessingAPI = require('./client/preprocessing/api.js'); -var preprocessingDaemon = require('./client/preprocessing/daemon.js'); - -/** - * Creates a new jiff client instance. - * @class - * @name JIFFClient - * @param {!string} hostname - server hostname/ip and port. - * @param {!string} computation_id - the id of the computation of this instance. - * @param {?object} [options={}] - javascript object with additional options. - * all parameters are optional, However, private and public key must either be both provided or neither of them provided. -
- {
-   "party_id": number,
-   "party_count": number,
-   "secret_key": Uint8Array to be used with libsodium-wrappers [(check Library Specs)]{@link https://download.libsodium.org/doc/public-key_cryptography/authenticated_encryption.html},
-   "public_key": Uint8Array to be used with libsodium-wrappers [(check Library Specs)]{@link https://download.libsodium.org/doc/public-key_cryptography/authenticated_encryption.html},
-   "public_keys": { 1: "Uint8Array PublicKey", 2: "Uint8Array PublicKey", ... },
-   "Zp": default mod to use (prime number),
-   "autoConnect": true/false,
-   "hooks": { 'check out hooks documentation' },
-   "listeners" : A map from custom tags to listeners (of type function(sender_id, message_string)) that handle custom messages with that tag.
-   "onConnect": function(jiff_instance),
-   "onError": function(label, error): called when errors occured in client code or during handling requests from this client at the server side
-                                      label is a string indicating where the error occured, and error is a string or an exception object.
-   "safemod": boolean (whether or not to check if provided Zp is prime, may be slow for big primes, defaults to false),
-   "crypto_provider": a boolean that flags whether to get beaver triplets and other preprocessing entities from the server (defaults to false),
-   "socketOptions": an object, passed directly to socket.io constructor,
-   "sodium": boolean, if false messages between clients will not be encrypted (useful for debugging),
-   "maxInitializationRetries": how many consecutive times to retry to initialize with the server if initialization fails, defaults to 2,
-   "preprocessingBatchSize": how many base level preprocessing tasks to execute in parallel.
- }
- 
- * - * @example - * var JIFFClient = require('jiffClient'); // only for node.js - * // for the browser - * // build a jiff instance which will connect to a server running on the local machine - * var instance = new JIFFClient('http://localhost:8080', 'computation-1', {party_count: 2}); - */ -function JIFFClient(hostname, computation_id, options) { - var jiffClient = this; - options = Object.assign({}, options); - - /** - * The server hostname, ends with a slash, includes port and protocol (http/https). - * @type {!string} - */ - this.hostname = hostname.trim(); - if (!this.hostname.endsWith('/')) { - this.hostname = this.hostname + '/'; - } - - /** - * Stores the computation id. - * @type {!string} - */ - this.computation_id = computation_id; - - /** - * Private. Do not use directly externally; use isReady() instead. - * @type {!boolean} - * @see {@link module:jiff-client~JIFFClient#isReady} - */ - this.__ready = false; - - /** - * Private. Do not use directly externally; use isInitialized() instead. - * @type {!boolean} - * @see {@link module:jiff-client~JIFFClient#isInitialized} - */ - this.__initialized = false; - - /** - * Returns whether this instance is capable of starting the computation. - * In other words, the public keys for all parties and servers are known, - * and this party successfully initialized with the server. - * @returns {!boolean} - */ - this.isReady = function () { - return this.__ready; - }; - - /** - * Returns whether this instance initialized successfully with the server. - * Note that this can be true even when isReady() returns false, in case where some other parties have not - * initialized yet! - * @returns {!boolean} - */ - this.isInitialized = function () { - return this.__initialized; - }; - - /** - * Helper functions [DO NOT MODIFY UNLESS YOU KNOW WHAT YOU ARE DOING]. - * @type {!helpers} - */ - this.helpers = {}; - helpers(this); - - /** - * Shallow copy of the options passed to the constructor. - * @type {!Object} - */ - - this.options = options; - - - // Parse and verify options - options.maxInitializationRetries = options.maxInitializationRetries || constants.maxInitializationRetries; - if (typeof(options.Zp) === 'number' && options.safemod !== false) { // big numbers are checked by extension - if (!this.helpers.is_prime(options.Zp)) { - throw new Error('Zp = ' + options.Zp + ' is not prime. Please use a prime number for the modulus or set safemod to false.'); - } - } - - /** - * The default Zp for this instance. - * @type {!number} - */ - this.Zp = options.Zp || constants.gZp; - - /** - * The id of this party. - * @type {number} - */ - this.id = options.party_id; - - /** - * Total party count in the computation, parties will take ids between 1 to party_count (inclusive). - * @type {number} - */ - this.party_count = options.party_count; - - /** - * sodium wrappers either imported via require (if in nodejs) or from the bundle (in the browser). - * This will be false if options.sodium is false. - * @see {@link https://www.npmjs.com/package/libsodium-wrappers} - * @type {?sodium} - */ - this.sodium_ = options.sodium !== false ? sodium : false; - - /** - * A map from party id to public key. Where key is the party id (number), and - * value is the public key, which by default follows libsodium's specs (Uint8Array). - * @see {@link https://download.libsodium.org/doc/public-key_cryptography/authenticated_encryption.html} - * @type {!object} - */ - this.keymap = Object.assign({}, options.public_keys); - - /** - * The secret key of this party, by default this follows libsodium's specs. - * @see {@link https://download.libsodium.org/doc/public-key_cryptography/authenticated_encryption.html} - * @type {?Uint8Array} - */ - this.secret_key = options.secret_key; - - /** - * The public key of this party, by default this follows libsodium's specs. - * @see {@link https://download.libsodium.org/doc/public-key_cryptography/authenticated_encryption.html} - * @type {?Uint8Array} - */ - this.public_key = options.public_key; - - /** - * Flags whether to use the server as a fallback for objects that were not pre-processed properly - * @type {!boolean} - */ - this.crypto_provider = (options.crypto_provider === true); - - /** - * Stores messages that are received with a signature prior to acquiring the public keys of the sender. - * { 'party_id': [ { 'label': 'share/open/custom', } ] } - * @type {object} - */ - this.messagesWaitingKeys = {}; - - /** - * A map from tags to listeners (functions that take a sender_id and a string message). - * - * Stores listeners that are attached to this JIFF instance, listeners listen to custom messages sent by other parties. - * @type {!object} - */ - this.listeners = Object.assign({}, options.listeners); - - /** - * Stores custom messages that are received before their listeners are set. Messages are stored in order. - * { 'tag' => [ { "sender_id": , "message": }, ... ] } - * - * Once a listener has been set, the corresponding messages are sent to it in order. - * @type {!object} - */ - this.custom_messages_mailbox = {}; - - /** - * Stores all promises created within some barrier. - * @type {!object} - */ - this.barriers = {}; - - /** - * Stores the parties and callbacks for every .wait_for() registered by the user. - * @type {!Array} - */ - this.wait_callbacks = []; - - /** - * Counts how many times JIFF attempted to initialize with the server - * without success consecutively. - * @type {!number} - * - */ - this.initialization_counter = 0; - - /** - * Utility functions - * @type {!utils} - */ - this.utils = {}; - utils(this); - - /** - * An array containing the names (jiff-client-[name].js) of the extensions that are - * applied to this instance. - * @type {string[]} - */ - this.extensions = ['base']; - - /** - * Internal helpers for operations inside/on a share. Modify existing helpers or add more in your extensions - * to avoid having to re-write and duplicate the code for primitives. - * @type {!object} - */ - this.share_helpers = share_helpers; - - /** - * The constructor function used by JIFF to create a new share. This can be overloaded by extensions to create custom shares. - * Modifying this will modify how shares are generated in the BASE JIFF implementation. - * A share is a value/promise wrapped with a share object. - * A share also has methods for performing operations. - * @constructor - * @param {number|promise} value - the value of the share, or a promise to it. - * @param {Array} holders - the parties that hold all the corresponding shares (must be sorted). - * @param {number} threshold - the min number of parties needed to reconstruct the secret. - * @param {number} Zp - the mod under which this share was created. - * - * @example - * // A share whose value is 10: the secret is still unknown, 10 is only one share - * var share = new jiffClient.SecretShare(10, [1, 2, 3], 3, jiffClient.Zp); - * - * @example - * // A share whose value depends on some promise - * var share = new jiffClient.SecretShare(promise, [1, 2, 3, 4], 4, jiffClient.Zp); - */ - this.SecretShare = SecretShareMetaClass(this); - - /** - * A collection of useful protocols to be used during computation or preprocessing: extensions are encouraged to add useful - * common protocols here, under a sub namespace corresponding to the extension name. - * @type {!protocols} - */ - this.protocols = {}; - - /** - * A collection of useful protocols for manipulating bitwise shared numbers, and transforming them from and to regular numeric shares. - * @member {!bits} bits - * @memberof protocols - */ - this.protocols.bits = {}; - - /** - * Stores pre-computed values (beaver triples, random bits, etc) used to aid/speed up the main processes. - * @type {!object} - */ - this.preprocessing_table = {}; - - /** - * Sets batch size for base level preprocessing tasks - * @type {!Number} - */ - this.preprocessingBatchSize = options.preprocessingBatchSize || 10; - - /** - * maps all primitive operations to the other operations they are dependent on, until leaves are primitives for which preprocessing protocols are defined, - * this map is traversed during preprocessing to guide preprocessing of high level operations. Extensions should modify this map to reflect - * any required changes to preprocessing of modified primitives - * @type {!object} - */ - this.preprocessing_function_map = {}; - - /** - * Store the default preprocessing protocols for each type of preprocessing value - * @type {!object} - */ - this.default_preprocessing_protocols = {}; - - /** - * Stores currently executing preprocessing tasks. - * @type {!linkedlist} - */ - this.currentPreprocessingTasks = linkedList(); - - /** - * Callback to execute when preprocessing is done! - * @type {?function} - */ - this.preprocessingCallback = null; - - /** - * Used for logging/debugging - * @type {!Array} - */ - this.logs = []; - - /** - * A map from open operation id to the corresponding shares received for that open operation - * @type {!object} - */ - this.shares = {}; - - /** - * A map from some message operation id to a deferred resolved when that message is received. - * @type {!object} - */ - this.deferreds = {}; - - /** - * Store sharing and shares counter which keeps track of the count of - * sharing operations (share and open) and the total number of shares - * respectively (used to get a unique id for each share operation and - * share object). - * @type {!object} - */ - this.counters = {}; - - /** - * A prefix attached to all op_ids, can be changed using {@link module:jiff-client~JIFFClient#seed_ids} - * to guarantee uniqueness of auto generate ids in a user-side callback - * or event handler. - * @type {string} - * @see {@link module:jiff-client~JIFFClient#seed_ids} - */ - this.op_id_seed = ''; - - /** - * The hooks for this instance. - * Checkout the hooks documentation - * @type {!hooks} - */ - this.hooks = new Hooks(this); - - /** - * Contains handlers for communication events - * @type {!handlers} - */ - this.handlers = {}; - - // Add user facing API - api(this); - - // Preprocessing - preprocessingMap(this); - preprocessingAPI(this); - preprocessingDaemon(this); - - // set up counters for op_ids - counters(this); - - - this.socketConnect = function (JIFFClientInstance) { - - if (options.__internal_socket == null) { - /** - * Socket wrapper between this instance and the server, based on sockets.io - * @type {!GuardedSocket} - */ - JIFFClientInstance.socket = guardedSocket(JIFFClientInstance); - } else { - JIFFClientInstance.socket = internalSocket(JIFFClientInstance, options.__internal_socket); - } - - // set up socket event handlers - handlers(JIFFClientInstance); - - JIFFClientInstance.initSocket(); - - - JIFFClientInstance.socket.connect(); - } - - /** - * Connect to the server and starts listening. - */ - this.connect = function () { - // Ask socket to connect, which will automatically trigger a call to 'initialize()' when connection is established! - - // Wait to let JIFF know that we are connected until sodium is ready - // Trigger the onConnect call - var JIFFClientInstance = this; - if (jiffClient.sodium_ === false) { - JIFFClientInstance.socketConnect(JIFFClientInstance); - } else { - jiffClient.sodium_.ready.then(function () { - JIFFClientInstance.socketConnect(JIFFClientInstance); - }); - } - }; - - // Connect when all is done - if (options.autoConnect !== false) { - this.connect(); - } -} - - -// Add socket event handlers to prototype -socketEvents(JIFFClient); - -// Add extension management to prototype -extensions(JIFFClient); - -// export JIFFClient class -module.exports = JIFFClient; - -},{"./client/api.js":1,"./client/arch/counters.js":9,"./client/arch/extensions.js":10,"./client/arch/hooks.js":11,"./client/handlers.js":12,"./client/preprocessing/api.js":17,"./client/preprocessing/daemon.js":18,"./client/preprocessing/map.js":20,"./client/share.js":42,"./client/shareHelpers.js":43,"./client/socket/events.js":44,"./client/socket/internal.js":45,"./client/socket/mailbox.js":46,"./client/util/constants.js":47,"./client/util/helpers.js":49,"./client/util/utils.js":50,"./common/linkedlist.js":52,"libsodium-wrappers":221}],54:[function(require,module,exports){ -module.exports = after - -function after(count, callback, err_cb) { - var bail = false - err_cb = err_cb || noop - proxy.count = count - - return (count === 0) ? callback() : proxy - - function proxy(err, result) { - if (proxy.count <= 0) { - throw new Error('after called too many times') - } - --proxy.count - - // after first error, rest are passed to err_cb - if (err) { - bail = true - callback(err) - // future error callbacks will go to error handler - callback = err_cb - } else if (proxy.count === 0 && !bail) { - callback(null, result) - } - } -} - -function noop() {} - -},{}],55:[function(require,module,exports){ -/** - * An abstraction for slicing an arraybuffer even when - * ArrayBuffer.prototype.slice is not supported - * - * @api public - */ - -module.exports = function(arraybuffer, start, end) { - var bytes = arraybuffer.byteLength; - start = start || 0; - end = end || bytes; - - if (arraybuffer.slice) { return arraybuffer.slice(start, end); } - - if (start < 0) { start += bytes; } - if (end < 0) { end += bytes; } - if (end > bytes) { end = bytes; } - - if (start >= bytes || start >= end || bytes === 0) { - return new ArrayBuffer(0); - } - - var abv = new Uint8Array(arraybuffer); - var result = new Uint8Array(end - start); - for (var i = start, ii = 0; i < end; i++, ii++) { - result[ii] = abv[i]; - } - return result.buffer; -}; - -},{}],56:[function(require,module,exports){ -var asn1 = exports; - -asn1.bignum = require('bn.js'); - -asn1.define = require('./asn1/api').define; -asn1.base = require('./asn1/base'); -asn1.constants = require('./asn1/constants'); -asn1.decoders = require('./asn1/decoders'); -asn1.encoders = require('./asn1/encoders'); - -},{"./asn1/api":57,"./asn1/base":59,"./asn1/constants":63,"./asn1/decoders":65,"./asn1/encoders":68,"bn.js":70}],57:[function(require,module,exports){ -var asn1 = require('../asn1'); -var inherits = require('inherits'); - -var api = exports; - -api.define = function define(name, body) { - return new Entity(name, body); -}; - -function Entity(name, body) { - this.name = name; - this.body = body; - - this.decoders = {}; - this.encoders = {}; -}; - -Entity.prototype._createNamed = function createNamed(base) { - var named; - try { - named = require('vm').runInThisContext( - '(function ' + this.name + '(entity) {\n' + - ' this._initNamed(entity);\n' + - '})' - ); - } catch (e) { - named = function (entity) { - this._initNamed(entity); - }; - } - inherits(named, base); - named.prototype._initNamed = function initnamed(entity) { - base.call(this, entity); - }; - - return new named(this); -}; - -Entity.prototype._getDecoder = function _getDecoder(enc) { - enc = enc || 'der'; - // Lazily create decoder - if (!this.decoders.hasOwnProperty(enc)) - this.decoders[enc] = this._createNamed(asn1.decoders[enc]); - return this.decoders[enc]; -}; - -Entity.prototype.decode = function decode(data, enc, options) { - return this._getDecoder(enc).decode(data, options); -}; - -Entity.prototype._getEncoder = function _getEncoder(enc) { - enc = enc || 'der'; - // Lazily create encoder - if (!this.encoders.hasOwnProperty(enc)) - this.encoders[enc] = this._createNamed(asn1.encoders[enc]); - return this.encoders[enc]; -}; - -Entity.prototype.encode = function encode(data, enc, /* internal */ reporter) { - return this._getEncoder(enc).encode(data, reporter); -}; - -},{"../asn1":56,"inherits":218,"vm":296}],58:[function(require,module,exports){ -var inherits = require('inherits'); -var Reporter = require('../base').Reporter; -var Buffer = require('buffer').Buffer; - -function DecoderBuffer(base, options) { - Reporter.call(this, options); - if (!Buffer.isBuffer(base)) { - this.error('Input not Buffer'); - return; - } - - this.base = base; - this.offset = 0; - this.length = base.length; -} -inherits(DecoderBuffer, Reporter); -exports.DecoderBuffer = DecoderBuffer; - -DecoderBuffer.prototype.save = function save() { - return { offset: this.offset, reporter: Reporter.prototype.save.call(this) }; -}; - -DecoderBuffer.prototype.restore = function restore(save) { - // Return skipped data - var res = new DecoderBuffer(this.base); - res.offset = save.offset; - res.length = this.offset; - - this.offset = save.offset; - Reporter.prototype.restore.call(this, save.reporter); - - return res; -}; - -DecoderBuffer.prototype.isEmpty = function isEmpty() { - return this.offset === this.length; -}; - -DecoderBuffer.prototype.readUInt8 = function readUInt8(fail) { - if (this.offset + 1 <= this.length) - return this.base.readUInt8(this.offset++, true); - else - return this.error(fail || 'DecoderBuffer overrun'); -} - -DecoderBuffer.prototype.skip = function skip(bytes, fail) { - if (!(this.offset + bytes <= this.length)) - return this.error(fail || 'DecoderBuffer overrun'); - - var res = new DecoderBuffer(this.base); - - // Share reporter state - res._reporterState = this._reporterState; - - res.offset = this.offset; - res.length = this.offset + bytes; - this.offset += bytes; - return res; -} - -DecoderBuffer.prototype.raw = function raw(save) { - return this.base.slice(save ? save.offset : this.offset, this.length); -} - -function EncoderBuffer(value, reporter) { - if (Array.isArray(value)) { - this.length = 0; - this.value = value.map(function(item) { - if (!(item instanceof EncoderBuffer)) - item = new EncoderBuffer(item, reporter); - this.length += item.length; - return item; - }, this); - } else if (typeof value === 'number') { - if (!(0 <= value && value <= 0xff)) - return reporter.error('non-byte EncoderBuffer value'); - this.value = value; - this.length = 1; - } else if (typeof value === 'string') { - this.value = value; - this.length = Buffer.byteLength(value); - } else if (Buffer.isBuffer(value)) { - this.value = value; - this.length = value.length; - } else { - return reporter.error('Unsupported type: ' + typeof value); - } -} -exports.EncoderBuffer = EncoderBuffer; - -EncoderBuffer.prototype.join = function join(out, offset) { - if (!out) - out = new Buffer(this.length); - if (!offset) - offset = 0; - - if (this.length === 0) - return out; - - if (Array.isArray(this.value)) { - this.value.forEach(function(item) { - item.join(out, offset); - offset += item.length; - }); - } else { - if (typeof this.value === 'number') - out[offset] = this.value; - else if (typeof this.value === 'string') - out.write(this.value, offset); - else if (Buffer.isBuffer(this.value)) - this.value.copy(out, offset); - offset += this.length; - } - - return out; -}; - -},{"../base":59,"buffer":124,"inherits":218}],59:[function(require,module,exports){ -var base = exports; - -base.Reporter = require('./reporter').Reporter; -base.DecoderBuffer = require('./buffer').DecoderBuffer; -base.EncoderBuffer = require('./buffer').EncoderBuffer; -base.Node = require('./node'); - -},{"./buffer":58,"./node":60,"./reporter":61}],60:[function(require,module,exports){ -var Reporter = require('../base').Reporter; -var EncoderBuffer = require('../base').EncoderBuffer; -var DecoderBuffer = require('../base').DecoderBuffer; -var assert = require('minimalistic-assert'); - -// Supported tags -var tags = [ - 'seq', 'seqof', 'set', 'setof', 'objid', 'bool', - 'gentime', 'utctime', 'null_', 'enum', 'int', 'objDesc', - 'bitstr', 'bmpstr', 'charstr', 'genstr', 'graphstr', 'ia5str', 'iso646str', - 'numstr', 'octstr', 'printstr', 't61str', 'unistr', 'utf8str', 'videostr' -]; - -// Public methods list -var methods = [ - 'key', 'obj', 'use', 'optional', 'explicit', 'implicit', 'def', 'choice', - 'any', 'contains' -].concat(tags); - -// Overrided methods list -var overrided = [ - '_peekTag', '_decodeTag', '_use', - '_decodeStr', '_decodeObjid', '_decodeTime', - '_decodeNull', '_decodeInt', '_decodeBool', '_decodeList', - - '_encodeComposite', '_encodeStr', '_encodeObjid', '_encodeTime', - '_encodeNull', '_encodeInt', '_encodeBool' -]; - -function Node(enc, parent) { - var state = {}; - this._baseState = state; - - state.enc = enc; - - state.parent = parent || null; - state.children = null; - - // State - state.tag = null; - state.args = null; - state.reverseArgs = null; - state.choice = null; - state.optional = false; - state.any = false; - state.obj = false; - state.use = null; - state.useDecoder = null; - state.key = null; - state['default'] = null; - state.explicit = null; - state.implicit = null; - state.contains = null; - - // Should create new instance on each method - if (!state.parent) { - state.children = []; - this._wrap(); - } -} -module.exports = Node; - -var stateProps = [ - 'enc', 'parent', 'children', 'tag', 'args', 'reverseArgs', 'choice', - 'optional', 'any', 'obj', 'use', 'alteredUse', 'key', 'default', 'explicit', - 'implicit', 'contains' -]; - -Node.prototype.clone = function clone() { - var state = this._baseState; - var cstate = {}; - stateProps.forEach(function(prop) { - cstate[prop] = state[prop]; - }); - var res = new this.constructor(cstate.parent); - res._baseState = cstate; - return res; -}; - -Node.prototype._wrap = function wrap() { - var state = this._baseState; - methods.forEach(function(method) { - this[method] = function _wrappedMethod() { - var clone = new this.constructor(this); - state.children.push(clone); - return clone[method].apply(clone, arguments); - }; - }, this); -}; - -Node.prototype._init = function init(body) { - var state = this._baseState; - - assert(state.parent === null); - body.call(this); - - // Filter children - state.children = state.children.filter(function(child) { - return child._baseState.parent === this; - }, this); - assert.equal(state.children.length, 1, 'Root node can have only one child'); -}; - -Node.prototype._useArgs = function useArgs(args) { - var state = this._baseState; - - // Filter children and args - var children = args.filter(function(arg) { - return arg instanceof this.constructor; - }, this); - args = args.filter(function(arg) { - return !(arg instanceof this.constructor); - }, this); - - if (children.length !== 0) { - assert(state.children === null); - state.children = children; - - // Replace parent to maintain backward link - children.forEach(function(child) { - child._baseState.parent = this; - }, this); - } - if (args.length !== 0) { - assert(state.args === null); - state.args = args; - state.reverseArgs = args.map(function(arg) { - if (typeof arg !== 'object' || arg.constructor !== Object) - return arg; - - var res = {}; - Object.keys(arg).forEach(function(key) { - if (key == (key | 0)) - key |= 0; - var value = arg[key]; - res[value] = key; - }); - return res; - }); - } -}; - -// -// Overrided methods -// - -overrided.forEach(function(method) { - Node.prototype[method] = function _overrided() { - var state = this._baseState; - throw new Error(method + ' not implemented for encoding: ' + state.enc); - }; -}); - -// -// Public methods -// - -tags.forEach(function(tag) { - Node.prototype[tag] = function _tagMethod() { - var state = this._baseState; - var args = Array.prototype.slice.call(arguments); - - assert(state.tag === null); - state.tag = tag; - - this._useArgs(args); - - return this; - }; -}); - -Node.prototype.use = function use(item) { - assert(item); - var state = this._baseState; - - assert(state.use === null); - state.use = item; - - return this; -}; - -Node.prototype.optional = function optional() { - var state = this._baseState; - - state.optional = true; - - return this; -}; - -Node.prototype.def = function def(val) { - var state = this._baseState; - - assert(state['default'] === null); - state['default'] = val; - state.optional = true; - - return this; -}; - -Node.prototype.explicit = function explicit(num) { - var state = this._baseState; - - assert(state.explicit === null && state.implicit === null); - state.explicit = num; - - return this; -}; - -Node.prototype.implicit = function implicit(num) { - var state = this._baseState; - - assert(state.explicit === null && state.implicit === null); - state.implicit = num; - - return this; -}; - -Node.prototype.obj = function obj() { - var state = this._baseState; - var args = Array.prototype.slice.call(arguments); - - state.obj = true; - - if (args.length !== 0) - this._useArgs(args); - - return this; -}; - -Node.prototype.key = function key(newKey) { - var state = this._baseState; - - assert(state.key === null); - state.key = newKey; - - return this; -}; - -Node.prototype.any = function any() { - var state = this._baseState; - - state.any = true; - - return this; -}; - -Node.prototype.choice = function choice(obj) { - var state = this._baseState; - - assert(state.choice === null); - state.choice = obj; - this._useArgs(Object.keys(obj).map(function(key) { - return obj[key]; - })); - - return this; -}; - -Node.prototype.contains = function contains(item) { - var state = this._baseState; - - assert(state.use === null); - state.contains = item; - - return this; -}; - -// -// Decoding -// - -Node.prototype._decode = function decode(input, options) { - var state = this._baseState; - - // Decode root node - if (state.parent === null) - return input.wrapResult(state.children[0]._decode(input, options)); - - var result = state['default']; - var present = true; - - var prevKey = null; - if (state.key !== null) - prevKey = input.enterKey(state.key); - - // Check if tag is there - if (state.optional) { - var tag = null; - if (state.explicit !== null) - tag = state.explicit; - else if (state.implicit !== null) - tag = state.implicit; - else if (state.tag !== null) - tag = state.tag; - - if (tag === null && !state.any) { - // Trial and Error - var save = input.save(); - try { - if (state.choice === null) - this._decodeGeneric(state.tag, input, options); - else - this._decodeChoice(input, options); - present = true; - } catch (e) { - present = false; - } - input.restore(save); - } else { - present = this._peekTag(input, tag, state.any); - - if (input.isError(present)) - return present; - } - } - - // Push object on stack - var prevObj; - if (state.obj && present) - prevObj = input.enterObject(); - - if (present) { - // Unwrap explicit values - if (state.explicit !== null) { - var explicit = this._decodeTag(input, state.explicit); - if (input.isError(explicit)) - return explicit; - input = explicit; - } - - var start = input.offset; - - // Unwrap implicit and normal values - if (state.use === null && state.choice === null) { - if (state.any) - var save = input.save(); - var body = this._decodeTag( - input, - state.implicit !== null ? state.implicit : state.tag, - state.any - ); - if (input.isError(body)) - return body; - - if (state.any) - result = input.raw(save); - else - input = body; - } - - if (options && options.track && state.tag !== null) - options.track(input.path(), start, input.length, 'tagged'); - - if (options && options.track && state.tag !== null) - options.track(input.path(), input.offset, input.length, 'content'); - - // Select proper method for tag - if (state.any) - result = result; - else if (state.choice === null) - result = this._decodeGeneric(state.tag, input, options); - else - result = this._decodeChoice(input, options); - - if (input.isError(result)) - return result; - - // Decode children - if (!state.any && state.choice === null && state.children !== null) { - state.children.forEach(function decodeChildren(child) { - // NOTE: We are ignoring errors here, to let parser continue with other - // parts of encoded data - child._decode(input, options); - }); - } - - // Decode contained/encoded by schema, only in bit or octet strings - if (state.contains && (state.tag === 'octstr' || state.tag === 'bitstr')) { - var data = new DecoderBuffer(result); - result = this._getUse(state.contains, input._reporterState.obj) - ._decode(data, options); - } - } - - // Pop object - if (state.obj && present) - result = input.leaveObject(prevObj); - - // Set key - if (state.key !== null && (result !== null || present === true)) - input.leaveKey(prevKey, state.key, result); - else if (prevKey !== null) - input.exitKey(prevKey); - - return result; -}; - -Node.prototype._decodeGeneric = function decodeGeneric(tag, input, options) { - var state = this._baseState; - - if (tag === 'seq' || tag === 'set') - return null; - if (tag === 'seqof' || tag === 'setof') - return this._decodeList(input, tag, state.args[0], options); - else if (/str$/.test(tag)) - return this._decodeStr(input, tag, options); - else if (tag === 'objid' && state.args) - return this._decodeObjid(input, state.args[0], state.args[1], options); - else if (tag === 'objid') - return this._decodeObjid(input, null, null, options); - else if (tag === 'gentime' || tag === 'utctime') - return this._decodeTime(input, tag, options); - else if (tag === 'null_') - return this._decodeNull(input, options); - else if (tag === 'bool') - return this._decodeBool(input, options); - else if (tag === 'objDesc') - return this._decodeStr(input, tag, options); - else if (tag === 'int' || tag === 'enum') - return this._decodeInt(input, state.args && state.args[0], options); - - if (state.use !== null) { - return this._getUse(state.use, input._reporterState.obj) - ._decode(input, options); - } else { - return input.error('unknown tag: ' + tag); - } -}; - -Node.prototype._getUse = function _getUse(entity, obj) { - - var state = this._baseState; - // Create altered use decoder if implicit is set - state.useDecoder = this._use(entity, obj); - assert(state.useDecoder._baseState.parent === null); - state.useDecoder = state.useDecoder._baseState.children[0]; - if (state.implicit !== state.useDecoder._baseState.implicit) { - state.useDecoder = state.useDecoder.clone(); - state.useDecoder._baseState.implicit = state.implicit; - } - return state.useDecoder; -}; - -Node.prototype._decodeChoice = function decodeChoice(input, options) { - var state = this._baseState; - var result = null; - var match = false; - - Object.keys(state.choice).some(function(key) { - var save = input.save(); - var node = state.choice[key]; - try { - var value = node._decode(input, options); - if (input.isError(value)) - return false; - - result = { type: key, value: value }; - match = true; - } catch (e) { - input.restore(save); - return false; - } - return true; - }, this); - - if (!match) - return input.error('Choice not matched'); - - return result; -}; - -// -// Encoding -// - -Node.prototype._createEncoderBuffer = function createEncoderBuffer(data) { - return new EncoderBuffer(data, this.reporter); -}; - -Node.prototype._encode = function encode(data, reporter, parent) { - var state = this._baseState; - if (state['default'] !== null && state['default'] === data) - return; - - var result = this._encodeValue(data, reporter, parent); - if (result === undefined) - return; - - if (this._skipDefault(result, reporter, parent)) - return; - - return result; -}; - -Node.prototype._encodeValue = function encode(data, reporter, parent) { - var state = this._baseState; - - // Decode root node - if (state.parent === null) - return state.children[0]._encode(data, reporter || new Reporter()); - - var result = null; - - // Set reporter to share it with a child class - this.reporter = reporter; - - // Check if data is there - if (state.optional && data === undefined) { - if (state['default'] !== null) - data = state['default'] - else - return; - } - - // Encode children first - var content = null; - var primitive = false; - if (state.any) { - // Anything that was given is translated to buffer - result = this._createEncoderBuffer(data); - } else if (state.choice) { - result = this._encodeChoice(data, reporter); - } else if (state.contains) { - content = this._getUse(state.contains, parent)._encode(data, reporter); - primitive = true; - } else if (state.children) { - content = state.children.map(function(child) { - if (child._baseState.tag === 'null_') - return child._encode(null, reporter, data); - - if (child._baseState.key === null) - return reporter.error('Child should have a key'); - var prevKey = reporter.enterKey(child._baseState.key); - - if (typeof data !== 'object') - return reporter.error('Child expected, but input is not object'); - - var res = child._encode(data[child._baseState.key], reporter, data); - reporter.leaveKey(prevKey); - - return res; - }, this).filter(function(child) { - return child; - }); - content = this._createEncoderBuffer(content); - } else { - if (state.tag === 'seqof' || state.tag === 'setof') { - // TODO(indutny): this should be thrown on DSL level - if (!(state.args && state.args.length === 1)) - return reporter.error('Too many args for : ' + state.tag); - - if (!Array.isArray(data)) - return reporter.error('seqof/setof, but data is not Array'); - - var child = this.clone(); - child._baseState.implicit = null; - content = this._createEncoderBuffer(data.map(function(item) { - var state = this._baseState; - - return this._getUse(state.args[0], data)._encode(item, reporter); - }, child)); - } else if (state.use !== null) { - result = this._getUse(state.use, parent)._encode(data, reporter); - } else { - content = this._encodePrimitive(state.tag, data); - primitive = true; - } - } - - // Encode data itself - var result; - if (!state.any && state.choice === null) { - var tag = state.implicit !== null ? state.implicit : state.tag; - var cls = state.implicit === null ? 'universal' : 'context'; - - if (tag === null) { - if (state.use === null) - reporter.error('Tag could be omitted only for .use()'); - } else { - if (state.use === null) - result = this._encodeComposite(tag, primitive, cls, content); - } - } - - // Wrap in explicit - if (state.explicit !== null) - result = this._encodeComposite(state.explicit, false, 'context', result); - - return result; -}; - -Node.prototype._encodeChoice = function encodeChoice(data, reporter) { - var state = this._baseState; - - var node = state.choice[data.type]; - if (!node) { - assert( - false, - data.type + ' not found in ' + - JSON.stringify(Object.keys(state.choice))); - } - return node._encode(data.value, reporter); -}; - -Node.prototype._encodePrimitive = function encodePrimitive(tag, data) { - var state = this._baseState; - - if (/str$/.test(tag)) - return this._encodeStr(data, tag); - else if (tag === 'objid' && state.args) - return this._encodeObjid(data, state.reverseArgs[0], state.args[1]); - else if (tag === 'objid') - return this._encodeObjid(data, null, null); - else if (tag === 'gentime' || tag === 'utctime') - return this._encodeTime(data, tag); - else if (tag === 'null_') - return this._encodeNull(); - else if (tag === 'int' || tag === 'enum') - return this._encodeInt(data, state.args && state.reverseArgs[0]); - else if (tag === 'bool') - return this._encodeBool(data); - else if (tag === 'objDesc') - return this._encodeStr(data, tag); - else - throw new Error('Unsupported tag: ' + tag); -}; - -Node.prototype._isNumstr = function isNumstr(str) { - return /^[0-9 ]*$/.test(str); -}; - -Node.prototype._isPrintstr = function isPrintstr(str) { - return /^[A-Za-z0-9 '\(\)\+,\-\.\/:=\?]*$/.test(str); -}; - -},{"../base":59,"minimalistic-assert":226}],61:[function(require,module,exports){ -var inherits = require('inherits'); - -function Reporter(options) { - this._reporterState = { - obj: null, - path: [], - options: options || {}, - errors: [] - }; -} -exports.Reporter = Reporter; - -Reporter.prototype.isError = function isError(obj) { - return obj instanceof ReporterError; -}; - -Reporter.prototype.save = function save() { - var state = this._reporterState; - - return { obj: state.obj, pathLen: state.path.length }; -}; - -Reporter.prototype.restore = function restore(data) { - var state = this._reporterState; - - state.obj = data.obj; - state.path = state.path.slice(0, data.pathLen); -}; - -Reporter.prototype.enterKey = function enterKey(key) { - return this._reporterState.path.push(key); -}; - -Reporter.prototype.exitKey = function exitKey(index) { - var state = this._reporterState; - - state.path = state.path.slice(0, index - 1); -}; - -Reporter.prototype.leaveKey = function leaveKey(index, key, value) { - var state = this._reporterState; - - this.exitKey(index); - if (state.obj !== null) - state.obj[key] = value; -}; - -Reporter.prototype.path = function path() { - return this._reporterState.path.join('/'); -}; - -Reporter.prototype.enterObject = function enterObject() { - var state = this._reporterState; - - var prev = state.obj; - state.obj = {}; - return prev; -}; - -Reporter.prototype.leaveObject = function leaveObject(prev) { - var state = this._reporterState; - - var now = state.obj; - state.obj = prev; - return now; -}; - -Reporter.prototype.error = function error(msg) { - var err; - var state = this._reporterState; - - var inherited = msg instanceof ReporterError; - if (inherited) { - err = msg; - } else { - err = new ReporterError(state.path.map(function(elem) { - return '[' + JSON.stringify(elem) + ']'; - }).join(''), msg.message || msg, msg.stack); - } - - if (!state.options.partial) - throw err; - - if (!inherited) - state.errors.push(err); - - return err; -}; - -Reporter.prototype.wrapResult = function wrapResult(result) { - var state = this._reporterState; - if (!state.options.partial) - return result; - - return { - result: this.isError(result) ? null : result, - errors: state.errors - }; -}; - -function ReporterError(path, msg) { - this.path = path; - this.rethrow(msg); -}; -inherits(ReporterError, Error); - -ReporterError.prototype.rethrow = function rethrow(msg) { - this.message = msg + ' at: ' + (this.path || '(shallow)'); - if (Error.captureStackTrace) - Error.captureStackTrace(this, ReporterError); - - if (!this.stack) { - try { - // IE only adds stack when thrown - throw new Error(this.message); - } catch (e) { - this.stack = e.stack; - } - } - return this; -}; - -},{"inherits":218}],62:[function(require,module,exports){ -var constants = require('../constants'); - -exports.tagClass = { - 0: 'universal', - 1: 'application', - 2: 'context', - 3: 'private' -}; -exports.tagClassByName = constants._reverse(exports.tagClass); - -exports.tag = { - 0x00: 'end', - 0x01: 'bool', - 0x02: 'int', - 0x03: 'bitstr', - 0x04: 'octstr', - 0x05: 'null_', - 0x06: 'objid', - 0x07: 'objDesc', - 0x08: 'external', - 0x09: 'real', - 0x0a: 'enum', - 0x0b: 'embed', - 0x0c: 'utf8str', - 0x0d: 'relativeOid', - 0x10: 'seq', - 0x11: 'set', - 0x12: 'numstr', - 0x13: 'printstr', - 0x14: 't61str', - 0x15: 'videostr', - 0x16: 'ia5str', - 0x17: 'utctime', - 0x18: 'gentime', - 0x19: 'graphstr', - 0x1a: 'iso646str', - 0x1b: 'genstr', - 0x1c: 'unistr', - 0x1d: 'charstr', - 0x1e: 'bmpstr' -}; -exports.tagByName = constants._reverse(exports.tag); - -},{"../constants":63}],63:[function(require,module,exports){ -var constants = exports; - -// Helper -constants._reverse = function reverse(map) { - var res = {}; - - Object.keys(map).forEach(function(key) { - // Convert key to integer if it is stringified - if ((key | 0) == key) - key = key | 0; - - var value = map[key]; - res[value] = key; - }); - - return res; -}; - -constants.der = require('./der'); - -},{"./der":62}],64:[function(require,module,exports){ -var inherits = require('inherits'); - -var asn1 = require('../../asn1'); -var base = asn1.base; -var bignum = asn1.bignum; - -// Import DER constants -var der = asn1.constants.der; - -function DERDecoder(entity) { - this.enc = 'der'; - this.name = entity.name; - this.entity = entity; - - // Construct base tree - this.tree = new DERNode(); - this.tree._init(entity.body); -}; -module.exports = DERDecoder; - -DERDecoder.prototype.decode = function decode(data, options) { - if (!(data instanceof base.DecoderBuffer)) - data = new base.DecoderBuffer(data, options); - - return this.tree._decode(data, options); -}; - -// Tree methods - -function DERNode(parent) { - base.Node.call(this, 'der', parent); -} -inherits(DERNode, base.Node); - -DERNode.prototype._peekTag = function peekTag(buffer, tag, any) { - if (buffer.isEmpty()) - return false; - - var state = buffer.save(); - var decodedTag = derDecodeTag(buffer, 'Failed to peek tag: "' + tag + '"'); - if (buffer.isError(decodedTag)) - return decodedTag; - - buffer.restore(state); - - return decodedTag.tag === tag || decodedTag.tagStr === tag || - (decodedTag.tagStr + 'of') === tag || any; -}; - -DERNode.prototype._decodeTag = function decodeTag(buffer, tag, any) { - var decodedTag = derDecodeTag(buffer, - 'Failed to decode tag of "' + tag + '"'); - if (buffer.isError(decodedTag)) - return decodedTag; - - var len = derDecodeLen(buffer, - decodedTag.primitive, - 'Failed to get length of "' + tag + '"'); - - // Failure - if (buffer.isError(len)) - return len; - - if (!any && - decodedTag.tag !== tag && - decodedTag.tagStr !== tag && - decodedTag.tagStr + 'of' !== tag) { - return buffer.error('Failed to match tag: "' + tag + '"'); - } - - if (decodedTag.primitive || len !== null) - return buffer.skip(len, 'Failed to match body of: "' + tag + '"'); - - // Indefinite length... find END tag - var state = buffer.save(); - var res = this._skipUntilEnd( - buffer, - 'Failed to skip indefinite length body: "' + this.tag + '"'); - if (buffer.isError(res)) - return res; - - len = buffer.offset - state.offset; - buffer.restore(state); - return buffer.skip(len, 'Failed to match body of: "' + tag + '"'); -}; - -DERNode.prototype._skipUntilEnd = function skipUntilEnd(buffer, fail) { - while (true) { - var tag = derDecodeTag(buffer, fail); - if (buffer.isError(tag)) - return tag; - var len = derDecodeLen(buffer, tag.primitive, fail); - if (buffer.isError(len)) - return len; - - var res; - if (tag.primitive || len !== null) - res = buffer.skip(len) - else - res = this._skipUntilEnd(buffer, fail); - - // Failure - if (buffer.isError(res)) - return res; - - if (tag.tagStr === 'end') - break; - } -}; - -DERNode.prototype._decodeList = function decodeList(buffer, tag, decoder, - options) { - var result = []; - while (!buffer.isEmpty()) { - var possibleEnd = this._peekTag(buffer, 'end'); - if (buffer.isError(possibleEnd)) - return possibleEnd; - - var res = decoder.decode(buffer, 'der', options); - if (buffer.isError(res) && possibleEnd) - break; - result.push(res); - } - return result; -}; - -DERNode.prototype._decodeStr = function decodeStr(buffer, tag) { - if (tag === 'bitstr') { - var unused = buffer.readUInt8(); - if (buffer.isError(unused)) - return unused; - return { unused: unused, data: buffer.raw() }; - } else if (tag === 'bmpstr') { - var raw = buffer.raw(); - if (raw.length % 2 === 1) - return buffer.error('Decoding of string type: bmpstr length mismatch'); - - var str = ''; - for (var i = 0; i < raw.length / 2; i++) { - str += String.fromCharCode(raw.readUInt16BE(i * 2)); - } - return str; - } else if (tag === 'numstr') { - var numstr = buffer.raw().toString('ascii'); - if (!this._isNumstr(numstr)) { - return buffer.error('Decoding of string type: ' + - 'numstr unsupported characters'); - } - return numstr; - } else if (tag === 'octstr') { - return buffer.raw(); - } else if (tag === 'objDesc') { - return buffer.raw(); - } else if (tag === 'printstr') { - var printstr = buffer.raw().toString('ascii'); - if (!this._isPrintstr(printstr)) { - return buffer.error('Decoding of string type: ' + - 'printstr unsupported characters'); - } - return printstr; - } else if (/str$/.test(tag)) { - return buffer.raw().toString(); - } else { - return buffer.error('Decoding of string type: ' + tag + ' unsupported'); - } -}; - -DERNode.prototype._decodeObjid = function decodeObjid(buffer, values, relative) { - var result; - var identifiers = []; - var ident = 0; - while (!buffer.isEmpty()) { - var subident = buffer.readUInt8(); - ident <<= 7; - ident |= subident & 0x7f; - if ((subident & 0x80) === 0) { - identifiers.push(ident); - ident = 0; - } - } - if (subident & 0x80) - identifiers.push(ident); - - var first = (identifiers[0] / 40) | 0; - var second = identifiers[0] % 40; - - if (relative) - result = identifiers; - else - result = [first, second].concat(identifiers.slice(1)); - - if (values) { - var tmp = values[result.join(' ')]; - if (tmp === undefined) - tmp = values[result.join('.')]; - if (tmp !== undefined) - result = tmp; - } - - return result; -}; - -DERNode.prototype._decodeTime = function decodeTime(buffer, tag) { - var str = buffer.raw().toString(); - if (tag === 'gentime') { - var year = str.slice(0, 4) | 0; - var mon = str.slice(4, 6) | 0; - var day = str.slice(6, 8) | 0; - var hour = str.slice(8, 10) | 0; - var min = str.slice(10, 12) | 0; - var sec = str.slice(12, 14) | 0; - } else if (tag === 'utctime') { - var year = str.slice(0, 2) | 0; - var mon = str.slice(2, 4) | 0; - var day = str.slice(4, 6) | 0; - var hour = str.slice(6, 8) | 0; - var min = str.slice(8, 10) | 0; - var sec = str.slice(10, 12) | 0; - if (year < 70) - year = 2000 + year; - else - year = 1900 + year; - } else { - return buffer.error('Decoding ' + tag + ' time is not supported yet'); - } - - return Date.UTC(year, mon - 1, day, hour, min, sec, 0); -}; - -DERNode.prototype._decodeNull = function decodeNull(buffer) { - return null; -}; - -DERNode.prototype._decodeBool = function decodeBool(buffer) { - var res = buffer.readUInt8(); - if (buffer.isError(res)) - return res; - else - return res !== 0; -}; - -DERNode.prototype._decodeInt = function decodeInt(buffer, values) { - // Bigint, return as it is (assume big endian) - var raw = buffer.raw(); - var res = new bignum(raw); - - if (values) - res = values[res.toString(10)] || res; - - return res; -}; - -DERNode.prototype._use = function use(entity, obj) { - if (typeof entity === 'function') - entity = entity(obj); - return entity._getDecoder('der').tree; -}; - -// Utility methods - -function derDecodeTag(buf, fail) { - var tag = buf.readUInt8(fail); - if (buf.isError(tag)) - return tag; - - var cls = der.tagClass[tag >> 6]; - var primitive = (tag & 0x20) === 0; - - // Multi-octet tag - load - if ((tag & 0x1f) === 0x1f) { - var oct = tag; - tag = 0; - while ((oct & 0x80) === 0x80) { - oct = buf.readUInt8(fail); - if (buf.isError(oct)) - return oct; - - tag <<= 7; - tag |= oct & 0x7f; - } - } else { - tag &= 0x1f; - } - var tagStr = der.tag[tag]; - - return { - cls: cls, - primitive: primitive, - tag: tag, - tagStr: tagStr - }; -} - -function derDecodeLen(buf, primitive, fail) { - var len = buf.readUInt8(fail); - if (buf.isError(len)) - return len; - - // Indefinite form - if (!primitive && len === 0x80) - return null; - - // Definite form - if ((len & 0x80) === 0) { - // Short form - return len; - } - - // Long form - var num = len & 0x7f; - if (num > 4) - return buf.error('length octect is too long'); - - len = 0; - for (var i = 0; i < num; i++) { - len <<= 8; - var j = buf.readUInt8(fail); - if (buf.isError(j)) - return j; - len |= j; - } - - return len; -} - -},{"../../asn1":56,"inherits":218}],65:[function(require,module,exports){ -var decoders = exports; - -decoders.der = require('./der'); -decoders.pem = require('./pem'); - -},{"./der":64,"./pem":66}],66:[function(require,module,exports){ -var inherits = require('inherits'); -var Buffer = require('buffer').Buffer; - -var DERDecoder = require('./der'); - -function PEMDecoder(entity) { - DERDecoder.call(this, entity); - this.enc = 'pem'; -}; -inherits(PEMDecoder, DERDecoder); -module.exports = PEMDecoder; - -PEMDecoder.prototype.decode = function decode(data, options) { - var lines = data.toString().split(/[\r\n]+/g); - - var label = options.label.toUpperCase(); - - var re = /^-----(BEGIN|END) ([^-]+)-----$/; - var start = -1; - var end = -1; - for (var i = 0; i < lines.length; i++) { - var match = lines[i].match(re); - if (match === null) - continue; - - if (match[2] !== label) - continue; - - if (start === -1) { - if (match[1] !== 'BEGIN') - break; - start = i; - } else { - if (match[1] !== 'END') - break; - end = i; - break; - } - } - if (start === -1 || end === -1) - throw new Error('PEM section not found for: ' + label); - - var base64 = lines.slice(start + 1, end).join(''); - // Remove excessive symbols - base64.replace(/[^a-z0-9\+\/=]+/gi, ''); - - var input = new Buffer(base64, 'base64'); - return DERDecoder.prototype.decode.call(this, input, options); -}; - -},{"./der":64,"buffer":124,"inherits":218}],67:[function(require,module,exports){ -var inherits = require('inherits'); -var Buffer = require('buffer').Buffer; - -var asn1 = require('../../asn1'); -var base = asn1.base; - -// Import DER constants -var der = asn1.constants.der; - -function DEREncoder(entity) { - this.enc = 'der'; - this.name = entity.name; - this.entity = entity; - - // Construct base tree - this.tree = new DERNode(); - this.tree._init(entity.body); -}; -module.exports = DEREncoder; - -DEREncoder.prototype.encode = function encode(data, reporter) { - return this.tree._encode(data, reporter).join(); -}; - -// Tree methods - -function DERNode(parent) { - base.Node.call(this, 'der', parent); -} -inherits(DERNode, base.Node); - -DERNode.prototype._encodeComposite = function encodeComposite(tag, - primitive, - cls, - content) { - var encodedTag = encodeTag(tag, primitive, cls, this.reporter); - - // Short form - if (content.length < 0x80) { - var header = new Buffer(2); - header[0] = encodedTag; - header[1] = content.length; - return this._createEncoderBuffer([ header, content ]); - } - - // Long form - // Count octets required to store length - var lenOctets = 1; - for (var i = content.length; i >= 0x100; i >>= 8) - lenOctets++; - - var header = new Buffer(1 + 1 + lenOctets); - header[0] = encodedTag; - header[1] = 0x80 | lenOctets; - - for (var i = 1 + lenOctets, j = content.length; j > 0; i--, j >>= 8) - header[i] = j & 0xff; - - return this._createEncoderBuffer([ header, content ]); -}; - -DERNode.prototype._encodeStr = function encodeStr(str, tag) { - if (tag === 'bitstr') { - return this._createEncoderBuffer([ str.unused | 0, str.data ]); - } else if (tag === 'bmpstr') { - var buf = new Buffer(str.length * 2); - for (var i = 0; i < str.length; i++) { - buf.writeUInt16BE(str.charCodeAt(i), i * 2); - } - return this._createEncoderBuffer(buf); - } else if (tag === 'numstr') { - if (!this._isNumstr(str)) { - return this.reporter.error('Encoding of string type: numstr supports ' + - 'only digits and space'); - } - return this._createEncoderBuffer(str); - } else if (tag === 'printstr') { - if (!this._isPrintstr(str)) { - return this.reporter.error('Encoding of string type: printstr supports ' + - 'only latin upper and lower case letters, ' + - 'digits, space, apostrophe, left and rigth ' + - 'parenthesis, plus sign, comma, hyphen, ' + - 'dot, slash, colon, equal sign, ' + - 'question mark'); - } - return this._createEncoderBuffer(str); - } else if (/str$/.test(tag)) { - return this._createEncoderBuffer(str); - } else if (tag === 'objDesc') { - return this._createEncoderBuffer(str); - } else { - return this.reporter.error('Encoding of string type: ' + tag + - ' unsupported'); - } -}; - -DERNode.prototype._encodeObjid = function encodeObjid(id, values, relative) { - if (typeof id === 'string') { - if (!values) - return this.reporter.error('string objid given, but no values map found'); - if (!values.hasOwnProperty(id)) - return this.reporter.error('objid not found in values map'); - id = values[id].split(/[\s\.]+/g); - for (var i = 0; i < id.length; i++) - id[i] |= 0; - } else if (Array.isArray(id)) { - id = id.slice(); - for (var i = 0; i < id.length; i++) - id[i] |= 0; - } - - if (!Array.isArray(id)) { - return this.reporter.error('objid() should be either array or string, ' + - 'got: ' + JSON.stringify(id)); - } - - if (!relative) { - if (id[1] >= 40) - return this.reporter.error('Second objid identifier OOB'); - id.splice(0, 2, id[0] * 40 + id[1]); - } - - // Count number of octets - var size = 0; - for (var i = 0; i < id.length; i++) { - var ident = id[i]; - for (size++; ident >= 0x80; ident >>= 7) - size++; - } - - var objid = new Buffer(size); - var offset = objid.length - 1; - for (var i = id.length - 1; i >= 0; i--) { - var ident = id[i]; - objid[offset--] = ident & 0x7f; - while ((ident >>= 7) > 0) - objid[offset--] = 0x80 | (ident & 0x7f); - } - - return this._createEncoderBuffer(objid); -}; - -function two(num) { - if (num < 10) - return '0' + num; - else - return num; -} - -DERNode.prototype._encodeTime = function encodeTime(time, tag) { - var str; - var date = new Date(time); - - if (tag === 'gentime') { - str = [ - two(date.getFullYear()), - two(date.getUTCMonth() + 1), - two(date.getUTCDate()), - two(date.getUTCHours()), - two(date.getUTCMinutes()), - two(date.getUTCSeconds()), - 'Z' - ].join(''); - } else if (tag === 'utctime') { - str = [ - two(date.getFullYear() % 100), - two(date.getUTCMonth() + 1), - two(date.getUTCDate()), - two(date.getUTCHours()), - two(date.getUTCMinutes()), - two(date.getUTCSeconds()), - 'Z' - ].join(''); - } else { - this.reporter.error('Encoding ' + tag + ' time is not supported yet'); - } - - return this._encodeStr(str, 'octstr'); -}; - -DERNode.prototype._encodeNull = function encodeNull() { - return this._createEncoderBuffer(''); -}; - -DERNode.prototype._encodeInt = function encodeInt(num, values) { - if (typeof num === 'string') { - if (!values) - return this.reporter.error('String int or enum given, but no values map'); - if (!values.hasOwnProperty(num)) { - return this.reporter.error('Values map doesn\'t contain: ' + - JSON.stringify(num)); - } - num = values[num]; - } - - // Bignum, assume big endian - if (typeof num !== 'number' && !Buffer.isBuffer(num)) { - var numArray = num.toArray(); - if (!num.sign && numArray[0] & 0x80) { - numArray.unshift(0); - } - num = new Buffer(numArray); - } - - if (Buffer.isBuffer(num)) { - var size = num.length; - if (num.length === 0) - size++; - - var out = new Buffer(size); - num.copy(out); - if (num.length === 0) - out[0] = 0 - return this._createEncoderBuffer(out); - } - - if (num < 0x80) - return this._createEncoderBuffer(num); - - if (num < 0x100) - return this._createEncoderBuffer([0, num]); - - var size = 1; - for (var i = num; i >= 0x100; i >>= 8) - size++; - - var out = new Array(size); - for (var i = out.length - 1; i >= 0; i--) { - out[i] = num & 0xff; - num >>= 8; - } - if(out[0] & 0x80) { - out.unshift(0); - } - - return this._createEncoderBuffer(new Buffer(out)); -}; - -DERNode.prototype._encodeBool = function encodeBool(value) { - return this._createEncoderBuffer(value ? 0xff : 0); -}; - -DERNode.prototype._use = function use(entity, obj) { - if (typeof entity === 'function') - entity = entity(obj); - return entity._getEncoder('der').tree; -}; - -DERNode.prototype._skipDefault = function skipDefault(dataBuffer, reporter, parent) { - var state = this._baseState; - var i; - if (state['default'] === null) - return false; - - var data = dataBuffer.join(); - if (state.defaultBuffer === undefined) - state.defaultBuffer = this._encodeValue(state['default'], reporter, parent).join(); - - if (data.length !== state.defaultBuffer.length) - return false; - - for (i=0; i < data.length; i++) - if (data[i] !== state.defaultBuffer[i]) - return false; - - return true; -}; - -// Utility methods - -function encodeTag(tag, primitive, cls, reporter) { - var res; - - if (tag === 'seqof') - tag = 'seq'; - else if (tag === 'setof') - tag = 'set'; - - if (der.tagByName.hasOwnProperty(tag)) - res = der.tagByName[tag]; - else if (typeof tag === 'number' && (tag | 0) === tag) - res = tag; - else - return reporter.error('Unknown tag: ' + tag); - - if (res >= 0x1f) - return reporter.error('Multi-octet tag encoding unsupported'); - - if (!primitive) - res |= 0x20; - - res |= (der.tagClassByName[cls || 'universal'] << 6); - - return res; -} - -},{"../../asn1":56,"buffer":124,"inherits":218}],68:[function(require,module,exports){ -var encoders = exports; - -encoders.der = require('./der'); -encoders.pem = require('./pem'); - -},{"./der":67,"./pem":69}],69:[function(require,module,exports){ -var inherits = require('inherits'); - -var DEREncoder = require('./der'); - -function PEMEncoder(entity) { - DEREncoder.call(this, entity); - this.enc = 'pem'; -}; -inherits(PEMEncoder, DEREncoder); -module.exports = PEMEncoder; - -PEMEncoder.prototype.encode = function encode(data, options) { - var buf = DEREncoder.prototype.encode.call(this, data); - - var p = buf.toString('base64'); - var out = [ '-----BEGIN ' + options.label + '-----' ]; - for (var i = 0; i < p.length; i += 64) - out.push(p.slice(i, i + 64)); - out.push('-----END ' + options.label + '-----'); - return out.join('\n'); -}; - -},{"./der":67,"inherits":218}],70:[function(require,module,exports){ -(function (module, exports) { - 'use strict'; - - // Utils - function assert (val, msg) { - if (!val) throw new Error(msg || 'Assertion failed'); - } - - // Could use `inherits` module, but don't want to move from single file - // architecture yet. - function inherits (ctor, superCtor) { - ctor.super_ = superCtor; - var TempCtor = function () {}; - TempCtor.prototype = superCtor.prototype; - ctor.prototype = new TempCtor(); - ctor.prototype.constructor = ctor; - } - - // BN - - function BN (number, base, endian) { - if (BN.isBN(number)) { - return number; - } - - this.negative = 0; - this.words = null; - this.length = 0; - - // Reduction context - this.red = null; - - if (number !== null) { - if (base === 'le' || base === 'be') { - endian = base; - base = 10; - } - - this._init(number || 0, base || 10, endian || 'be'); - } - } - if (typeof module === 'object') { - module.exports = BN; - } else { - exports.BN = BN; - } - - BN.BN = BN; - BN.wordSize = 26; - - var Buffer; - try { - Buffer = require('buffer').Buffer; - } catch (e) { - } - - BN.isBN = function isBN (num) { - if (num instanceof BN) { - return true; - } - - return num !== null && typeof num === 'object' && - num.constructor.wordSize === BN.wordSize && Array.isArray(num.words); - }; - - BN.max = function max (left, right) { - if (left.cmp(right) > 0) return left; - return right; - }; - - BN.min = function min (left, right) { - if (left.cmp(right) < 0) return left; - return right; - }; - - BN.prototype._init = function init (number, base, endian) { - if (typeof number === 'number') { - return this._initNumber(number, base, endian); - } - - if (typeof number === 'object') { - return this._initArray(number, base, endian); - } - - if (base === 'hex') { - base = 16; - } - assert(base === (base | 0) && base >= 2 && base <= 36); - - number = number.toString().replace(/\s+/g, ''); - var start = 0; - if (number[0] === '-') { - start++; - } - - if (base === 16) { - this._parseHex(number, start); - } else { - this._parseBase(number, base, start); - } - - if (number[0] === '-') { - this.negative = 1; - } - - this.strip(); - - if (endian !== 'le') return; - - this._initArray(this.toArray(), base, endian); - }; - - BN.prototype._initNumber = function _initNumber (number, base, endian) { - if (number < 0) { - this.negative = 1; - number = -number; - } - if (number < 0x4000000) { - this.words = [ number & 0x3ffffff ]; - this.length = 1; - } else if (number < 0x10000000000000) { - this.words = [ - number & 0x3ffffff, - (number / 0x4000000) & 0x3ffffff - ]; - this.length = 2; - } else { - assert(number < 0x20000000000000); // 2 ^ 53 (unsafe) - this.words = [ - number & 0x3ffffff, - (number / 0x4000000) & 0x3ffffff, - 1 - ]; - this.length = 3; - } - - if (endian !== 'le') return; - - // Reverse the bytes - this._initArray(this.toArray(), base, endian); - }; - - BN.prototype._initArray = function _initArray (number, base, endian) { - // Perhaps a Uint8Array - assert(typeof number.length === 'number'); - if (number.length <= 0) { - this.words = [ 0 ]; - this.length = 1; - return this; - } - - this.length = Math.ceil(number.length / 3); - this.words = new Array(this.length); - for (var i = 0; i < this.length; i++) { - this.words[i] = 0; - } - - var j, w; - var off = 0; - if (endian === 'be') { - for (i = number.length - 1, j = 0; i >= 0; i -= 3) { - w = number[i] | (number[i - 1] << 8) | (number[i - 2] << 16); - this.words[j] |= (w << off) & 0x3ffffff; - this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff; - off += 24; - if (off >= 26) { - off -= 26; - j++; - } - } - } else if (endian === 'le') { - for (i = 0, j = 0; i < number.length; i += 3) { - w = number[i] | (number[i + 1] << 8) | (number[i + 2] << 16); - this.words[j] |= (w << off) & 0x3ffffff; - this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff; - off += 24; - if (off >= 26) { - off -= 26; - j++; - } - } - } - return this.strip(); - }; - - function parseHex (str, start, end) { - var r = 0; - var len = Math.min(str.length, end); - for (var i = start; i < len; i++) { - var c = str.charCodeAt(i) - 48; - - r <<= 4; - - // 'a' - 'f' - if (c >= 49 && c <= 54) { - r |= c - 49 + 0xa; - - // 'A' - 'F' - } else if (c >= 17 && c <= 22) { - r |= c - 17 + 0xa; - - // '0' - '9' - } else { - r |= c & 0xf; - } - } - return r; - } - - BN.prototype._parseHex = function _parseHex (number, start) { - // Create possibly bigger array to ensure that it fits the number - this.length = Math.ceil((number.length - start) / 6); - this.words = new Array(this.length); - for (var i = 0; i < this.length; i++) { - this.words[i] = 0; - } - - var j, w; - // Scan 24-bit chunks and add them to the number - var off = 0; - for (i = number.length - 6, j = 0; i >= start; i -= 6) { - w = parseHex(number, i, i + 6); - this.words[j] |= (w << off) & 0x3ffffff; - // NOTE: `0x3fffff` is intentional here, 26bits max shift + 24bit hex limb - this.words[j + 1] |= w >>> (26 - off) & 0x3fffff; - off += 24; - if (off >= 26) { - off -= 26; - j++; - } - } - if (i + 6 !== start) { - w = parseHex(number, start, i + 6); - this.words[j] |= (w << off) & 0x3ffffff; - this.words[j + 1] |= w >>> (26 - off) & 0x3fffff; - } - this.strip(); - }; - - function parseBase (str, start, end, mul) { - var r = 0; - var len = Math.min(str.length, end); - for (var i = start; i < len; i++) { - var c = str.charCodeAt(i) - 48; - - r *= mul; - - // 'a' - if (c >= 49) { - r += c - 49 + 0xa; - - // 'A' - } else if (c >= 17) { - r += c - 17 + 0xa; - - // '0' - '9' - } else { - r += c; - } - } - return r; - } - - BN.prototype._parseBase = function _parseBase (number, base, start) { - // Initialize as zero - this.words = [ 0 ]; - this.length = 1; - - // Find length of limb in base - for (var limbLen = 0, limbPow = 1; limbPow <= 0x3ffffff; limbPow *= base) { - limbLen++; - } - limbLen--; - limbPow = (limbPow / base) | 0; - - var total = number.length - start; - var mod = total % limbLen; - var end = Math.min(total, total - mod) + start; - - var word = 0; - for (var i = start; i < end; i += limbLen) { - word = parseBase(number, i, i + limbLen, base); - - this.imuln(limbPow); - if (this.words[0] + word < 0x4000000) { - this.words[0] += word; - } else { - this._iaddn(word); - } - } - - if (mod !== 0) { - var pow = 1; - word = parseBase(number, i, number.length, base); - - for (i = 0; i < mod; i++) { - pow *= base; - } - - this.imuln(pow); - if (this.words[0] + word < 0x4000000) { - this.words[0] += word; - } else { - this._iaddn(word); - } - } - }; - - BN.prototype.copy = function copy (dest) { - dest.words = new Array(this.length); - for (var i = 0; i < this.length; i++) { - dest.words[i] = this.words[i]; - } - dest.length = this.length; - dest.negative = this.negative; - dest.red = this.red; - }; - - BN.prototype.clone = function clone () { - var r = new BN(null); - this.copy(r); - return r; - }; - - BN.prototype._expand = function _expand (size) { - while (this.length < size) { - this.words[this.length++] = 0; - } - return this; - }; - - // Remove leading `0` from `this` - BN.prototype.strip = function strip () { - while (this.length > 1 && this.words[this.length - 1] === 0) { - this.length--; - } - return this._normSign(); - }; - - BN.prototype._normSign = function _normSign () { - // -0 = 0 - if (this.length === 1 && this.words[0] === 0) { - this.negative = 0; - } - return this; - }; - - BN.prototype.inspect = function inspect () { - return (this.red ? ''; - }; - - /* - - var zeros = []; - var groupSizes = []; - var groupBases = []; - - var s = ''; - var i = -1; - while (++i < BN.wordSize) { - zeros[i] = s; - s += '0'; - } - groupSizes[0] = 0; - groupSizes[1] = 0; - groupBases[0] = 0; - groupBases[1] = 0; - var base = 2 - 1; - while (++base < 36 + 1) { - var groupSize = 0; - var groupBase = 1; - while (groupBase < (1 << BN.wordSize) / base) { - groupBase *= base; - groupSize += 1; - } - groupSizes[base] = groupSize; - groupBases[base] = groupBase; - } - - */ - - var zeros = [ - '', - '0', - '00', - '000', - '0000', - '00000', - '000000', - '0000000', - '00000000', - '000000000', - '0000000000', - '00000000000', - '000000000000', - '0000000000000', - '00000000000000', - '000000000000000', - '0000000000000000', - '00000000000000000', - '000000000000000000', - '0000000000000000000', - '00000000000000000000', - '000000000000000000000', - '0000000000000000000000', - '00000000000000000000000', - '000000000000000000000000', - '0000000000000000000000000' - ]; - - var groupSizes = [ - 0, 0, - 25, 16, 12, 11, 10, 9, 8, - 8, 7, 7, 7, 7, 6, 6, - 6, 6, 6, 6, 6, 5, 5, - 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5 - ]; - - var groupBases = [ - 0, 0, - 33554432, 43046721, 16777216, 48828125, 60466176, 40353607, 16777216, - 43046721, 10000000, 19487171, 35831808, 62748517, 7529536, 11390625, - 16777216, 24137569, 34012224, 47045881, 64000000, 4084101, 5153632, - 6436343, 7962624, 9765625, 11881376, 14348907, 17210368, 20511149, - 24300000, 28629151, 33554432, 39135393, 45435424, 52521875, 60466176 - ]; - - BN.prototype.toString = function toString (base, padding) { - base = base || 10; - padding = padding | 0 || 1; - - var out; - if (base === 16 || base === 'hex') { - out = ''; - var off = 0; - var carry = 0; - for (var i = 0; i < this.length; i++) { - var w = this.words[i]; - var word = (((w << off) | carry) & 0xffffff).toString(16); - carry = (w >>> (24 - off)) & 0xffffff; - if (carry !== 0 || i !== this.length - 1) { - out = zeros[6 - word.length] + word + out; - } else { - out = word + out; - } - off += 2; - if (off >= 26) { - off -= 26; - i--; - } - } - if (carry !== 0) { - out = carry.toString(16) + out; - } - while (out.length % padding !== 0) { - out = '0' + out; - } - if (this.negative !== 0) { - out = '-' + out; - } - return out; - } - - if (base === (base | 0) && base >= 2 && base <= 36) { - // var groupSize = Math.floor(BN.wordSize * Math.LN2 / Math.log(base)); - var groupSize = groupSizes[base]; - // var groupBase = Math.pow(base, groupSize); - var groupBase = groupBases[base]; - out = ''; - var c = this.clone(); - c.negative = 0; - while (!c.isZero()) { - var r = c.modn(groupBase).toString(base); - c = c.idivn(groupBase); - - if (!c.isZero()) { - out = zeros[groupSize - r.length] + r + out; - } else { - out = r + out; - } - } - if (this.isZero()) { - out = '0' + out; - } - while (out.length % padding !== 0) { - out = '0' + out; - } - if (this.negative !== 0) { - out = '-' + out; - } - return out; - } - - assert(false, 'Base should be between 2 and 36'); - }; - - BN.prototype.toNumber = function toNumber () { - var ret = this.words[0]; - if (this.length === 2) { - ret += this.words[1] * 0x4000000; - } else if (this.length === 3 && this.words[2] === 0x01) { - // NOTE: at this stage it is known that the top bit is set - ret += 0x10000000000000 + (this.words[1] * 0x4000000); - } else if (this.length > 2) { - assert(false, 'Number can only safely store up to 53 bits'); - } - return (this.negative !== 0) ? -ret : ret; - }; - - BN.prototype.toJSON = function toJSON () { - return this.toString(16); - }; - - BN.prototype.toBuffer = function toBuffer (endian, length) { - assert(typeof Buffer !== 'undefined'); - return this.toArrayLike(Buffer, endian, length); - }; - - BN.prototype.toArray = function toArray (endian, length) { - return this.toArrayLike(Array, endian, length); - }; - - BN.prototype.toArrayLike = function toArrayLike (ArrayType, endian, length) { - var byteLength = this.byteLength(); - var reqLength = length || Math.max(1, byteLength); - assert(byteLength <= reqLength, 'byte array longer than desired length'); - assert(reqLength > 0, 'Requested array length <= 0'); - - this.strip(); - var littleEndian = endian === 'le'; - var res = new ArrayType(reqLength); - - var b, i; - var q = this.clone(); - if (!littleEndian) { - // Assume big-endian - for (i = 0; i < reqLength - byteLength; i++) { - res[i] = 0; - } - - for (i = 0; !q.isZero(); i++) { - b = q.andln(0xff); - q.iushrn(8); - - res[reqLength - i - 1] = b; - } - } else { - for (i = 0; !q.isZero(); i++) { - b = q.andln(0xff); - q.iushrn(8); - - res[i] = b; - } - - for (; i < reqLength; i++) { - res[i] = 0; - } - } - - return res; - }; - - if (Math.clz32) { - BN.prototype._countBits = function _countBits (w) { - return 32 - Math.clz32(w); - }; - } else { - BN.prototype._countBits = function _countBits (w) { - var t = w; - var r = 0; - if (t >= 0x1000) { - r += 13; - t >>>= 13; - } - if (t >= 0x40) { - r += 7; - t >>>= 7; - } - if (t >= 0x8) { - r += 4; - t >>>= 4; - } - if (t >= 0x02) { - r += 2; - t >>>= 2; - } - return r + t; - }; - } - - BN.prototype._zeroBits = function _zeroBits (w) { - // Short-cut - if (w === 0) return 26; - - var t = w; - var r = 0; - if ((t & 0x1fff) === 0) { - r += 13; - t >>>= 13; - } - if ((t & 0x7f) === 0) { - r += 7; - t >>>= 7; - } - if ((t & 0xf) === 0) { - r += 4; - t >>>= 4; - } - if ((t & 0x3) === 0) { - r += 2; - t >>>= 2; - } - if ((t & 0x1) === 0) { - r++; - } - return r; - }; - - // Return number of used bits in a BN - BN.prototype.bitLength = function bitLength () { - var w = this.words[this.length - 1]; - var hi = this._countBits(w); - return (this.length - 1) * 26 + hi; - }; - - function toBitArray (num) { - var w = new Array(num.bitLength()); - - for (var bit = 0; bit < w.length; bit++) { - var off = (bit / 26) | 0; - var wbit = bit % 26; - - w[bit] = (num.words[off] & (1 << wbit)) >>> wbit; - } - - return w; - } - - // Number of trailing zero bits - BN.prototype.zeroBits = function zeroBits () { - if (this.isZero()) return 0; - - var r = 0; - for (var i = 0; i < this.length; i++) { - var b = this._zeroBits(this.words[i]); - r += b; - if (b !== 26) break; - } - return r; - }; - - BN.prototype.byteLength = function byteLength () { - return Math.ceil(this.bitLength() / 8); - }; - - BN.prototype.toTwos = function toTwos (width) { - if (this.negative !== 0) { - return this.abs().inotn(width).iaddn(1); - } - return this.clone(); - }; - - BN.prototype.fromTwos = function fromTwos (width) { - if (this.testn(width - 1)) { - return this.notn(width).iaddn(1).ineg(); - } - return this.clone(); - }; - - BN.prototype.isNeg = function isNeg () { - return this.negative !== 0; - }; - - // Return negative clone of `this` - BN.prototype.neg = function neg () { - return this.clone().ineg(); - }; - - BN.prototype.ineg = function ineg () { - if (!this.isZero()) { - this.negative ^= 1; - } - - return this; - }; - - // Or `num` with `this` in-place - BN.prototype.iuor = function iuor (num) { - while (this.length < num.length) { - this.words[this.length++] = 0; - } - - for (var i = 0; i < num.length; i++) { - this.words[i] = this.words[i] | num.words[i]; - } - - return this.strip(); - }; - - BN.prototype.ior = function ior (num) { - assert((this.negative | num.negative) === 0); - return this.iuor(num); - }; - - // Or `num` with `this` - BN.prototype.or = function or (num) { - if (this.length > num.length) return this.clone().ior(num); - return num.clone().ior(this); - }; - - BN.prototype.uor = function uor (num) { - if (this.length > num.length) return this.clone().iuor(num); - return num.clone().iuor(this); - }; - - // And `num` with `this` in-place - BN.prototype.iuand = function iuand (num) { - // b = min-length(num, this) - var b; - if (this.length > num.length) { - b = num; - } else { - b = this; - } - - for (var i = 0; i < b.length; i++) { - this.words[i] = this.words[i] & num.words[i]; - } - - this.length = b.length; - - return this.strip(); - }; - - BN.prototype.iand = function iand (num) { - assert((this.negative | num.negative) === 0); - return this.iuand(num); - }; - - // And `num` with `this` - BN.prototype.and = function and (num) { - if (this.length > num.length) return this.clone().iand(num); - return num.clone().iand(this); - }; - - BN.prototype.uand = function uand (num) { - if (this.length > num.length) return this.clone().iuand(num); - return num.clone().iuand(this); - }; - - // Xor `num` with `this` in-place - BN.prototype.iuxor = function iuxor (num) { - // a.length > b.length - var a; - var b; - if (this.length > num.length) { - a = this; - b = num; - } else { - a = num; - b = this; - } - - for (var i = 0; i < b.length; i++) { - this.words[i] = a.words[i] ^ b.words[i]; - } - - if (this !== a) { - for (; i < a.length; i++) { - this.words[i] = a.words[i]; - } - } - - this.length = a.length; - - return this.strip(); - }; - - BN.prototype.ixor = function ixor (num) { - assert((this.negative | num.negative) === 0); - return this.iuxor(num); - }; - - // Xor `num` with `this` - BN.prototype.xor = function xor (num) { - if (this.length > num.length) return this.clone().ixor(num); - return num.clone().ixor(this); - }; - - BN.prototype.uxor = function uxor (num) { - if (this.length > num.length) return this.clone().iuxor(num); - return num.clone().iuxor(this); - }; - - // Not ``this`` with ``width`` bitwidth - BN.prototype.inotn = function inotn (width) { - assert(typeof width === 'number' && width >= 0); - - var bytesNeeded = Math.ceil(width / 26) | 0; - var bitsLeft = width % 26; - - // Extend the buffer with leading zeroes - this._expand(bytesNeeded); - - if (bitsLeft > 0) { - bytesNeeded--; - } - - // Handle complete words - for (var i = 0; i < bytesNeeded; i++) { - this.words[i] = ~this.words[i] & 0x3ffffff; - } - - // Handle the residue - if (bitsLeft > 0) { - this.words[i] = ~this.words[i] & (0x3ffffff >> (26 - bitsLeft)); - } - - // And remove leading zeroes - return this.strip(); - }; - - BN.prototype.notn = function notn (width) { - return this.clone().inotn(width); - }; - - // Set `bit` of `this` - BN.prototype.setn = function setn (bit, val) { - assert(typeof bit === 'number' && bit >= 0); - - var off = (bit / 26) | 0; - var wbit = bit % 26; - - this._expand(off + 1); - - if (val) { - this.words[off] = this.words[off] | (1 << wbit); - } else { - this.words[off] = this.words[off] & ~(1 << wbit); - } - - return this.strip(); - }; - - // Add `num` to `this` in-place - BN.prototype.iadd = function iadd (num) { - var r; - - // negative + positive - if (this.negative !== 0 && num.negative === 0) { - this.negative = 0; - r = this.isub(num); - this.negative ^= 1; - return this._normSign(); - - // positive + negative - } else if (this.negative === 0 && num.negative !== 0) { - num.negative = 0; - r = this.isub(num); - num.negative = 1; - return r._normSign(); - } - - // a.length > b.length - var a, b; - if (this.length > num.length) { - a = this; - b = num; - } else { - a = num; - b = this; - } - - var carry = 0; - for (var i = 0; i < b.length; i++) { - r = (a.words[i] | 0) + (b.words[i] | 0) + carry; - this.words[i] = r & 0x3ffffff; - carry = r >>> 26; - } - for (; carry !== 0 && i < a.length; i++) { - r = (a.words[i] | 0) + carry; - this.words[i] = r & 0x3ffffff; - carry = r >>> 26; - } - - this.length = a.length; - if (carry !== 0) { - this.words[this.length] = carry; - this.length++; - // Copy the rest of the words - } else if (a !== this) { - for (; i < a.length; i++) { - this.words[i] = a.words[i]; - } - } - - return this; - }; - - // Add `num` to `this` - BN.prototype.add = function add (num) { - var res; - if (num.negative !== 0 && this.negative === 0) { - num.negative = 0; - res = this.sub(num); - num.negative ^= 1; - return res; - } else if (num.negative === 0 && this.negative !== 0) { - this.negative = 0; - res = num.sub(this); - this.negative = 1; - return res; - } - - if (this.length > num.length) return this.clone().iadd(num); - - return num.clone().iadd(this); - }; - - // Subtract `num` from `this` in-place - BN.prototype.isub = function isub (num) { - // this - (-num) = this + num - if (num.negative !== 0) { - num.negative = 0; - var r = this.iadd(num); - num.negative = 1; - return r._normSign(); - - // -this - num = -(this + num) - } else if (this.negative !== 0) { - this.negative = 0; - this.iadd(num); - this.negative = 1; - return this._normSign(); - } - - // At this point both numbers are positive - var cmp = this.cmp(num); - - // Optimization - zeroify - if (cmp === 0) { - this.negative = 0; - this.length = 1; - this.words[0] = 0; - return this; - } - - // a > b - var a, b; - if (cmp > 0) { - a = this; - b = num; - } else { - a = num; - b = this; - } - - var carry = 0; - for (var i = 0; i < b.length; i++) { - r = (a.words[i] | 0) - (b.words[i] | 0) + carry; - carry = r >> 26; - this.words[i] = r & 0x3ffffff; - } - for (; carry !== 0 && i < a.length; i++) { - r = (a.words[i] | 0) + carry; - carry = r >> 26; - this.words[i] = r & 0x3ffffff; - } - - // Copy rest of the words - if (carry === 0 && i < a.length && a !== this) { - for (; i < a.length; i++) { - this.words[i] = a.words[i]; - } - } - - this.length = Math.max(this.length, i); - - if (a !== this) { - this.negative = 1; - } - - return this.strip(); - }; - - // Subtract `num` from `this` - BN.prototype.sub = function sub (num) { - return this.clone().isub(num); - }; - - function smallMulTo (self, num, out) { - out.negative = num.negative ^ self.negative; - var len = (self.length + num.length) | 0; - out.length = len; - len = (len - 1) | 0; - - // Peel one iteration (compiler can't do it, because of code complexity) - var a = self.words[0] | 0; - var b = num.words[0] | 0; - var r = a * b; - - var lo = r & 0x3ffffff; - var carry = (r / 0x4000000) | 0; - out.words[0] = lo; - - for (var k = 1; k < len; k++) { - // Sum all words with the same `i + j = k` and accumulate `ncarry`, - // note that ncarry could be >= 0x3ffffff - var ncarry = carry >>> 26; - var rword = carry & 0x3ffffff; - var maxJ = Math.min(k, num.length - 1); - for (var j = Math.max(0, k - self.length + 1); j <= maxJ; j++) { - var i = (k - j) | 0; - a = self.words[i] | 0; - b = num.words[j] | 0; - r = a * b + rword; - ncarry += (r / 0x4000000) | 0; - rword = r & 0x3ffffff; - } - out.words[k] = rword | 0; - carry = ncarry | 0; - } - if (carry !== 0) { - out.words[k] = carry | 0; - } else { - out.length--; - } - - return out.strip(); - } - - // TODO(indutny): it may be reasonable to omit it for users who don't need - // to work with 256-bit numbers, otherwise it gives 20% improvement for 256-bit - // multiplication (like elliptic secp256k1). - var comb10MulTo = function comb10MulTo (self, num, out) { - var a = self.words; - var b = num.words; - var o = out.words; - var c = 0; - var lo; - var mid; - var hi; - var a0 = a[0] | 0; - var al0 = a0 & 0x1fff; - var ah0 = a0 >>> 13; - var a1 = a[1] | 0; - var al1 = a1 & 0x1fff; - var ah1 = a1 >>> 13; - var a2 = a[2] | 0; - var al2 = a2 & 0x1fff; - var ah2 = a2 >>> 13; - var a3 = a[3] | 0; - var al3 = a3 & 0x1fff; - var ah3 = a3 >>> 13; - var a4 = a[4] | 0; - var al4 = a4 & 0x1fff; - var ah4 = a4 >>> 13; - var a5 = a[5] | 0; - var al5 = a5 & 0x1fff; - var ah5 = a5 >>> 13; - var a6 = a[6] | 0; - var al6 = a6 & 0x1fff; - var ah6 = a6 >>> 13; - var a7 = a[7] | 0; - var al7 = a7 & 0x1fff; - var ah7 = a7 >>> 13; - var a8 = a[8] | 0; - var al8 = a8 & 0x1fff; - var ah8 = a8 >>> 13; - var a9 = a[9] | 0; - var al9 = a9 & 0x1fff; - var ah9 = a9 >>> 13; - var b0 = b[0] | 0; - var bl0 = b0 & 0x1fff; - var bh0 = b0 >>> 13; - var b1 = b[1] | 0; - var bl1 = b1 & 0x1fff; - var bh1 = b1 >>> 13; - var b2 = b[2] | 0; - var bl2 = b2 & 0x1fff; - var bh2 = b2 >>> 13; - var b3 = b[3] | 0; - var bl3 = b3 & 0x1fff; - var bh3 = b3 >>> 13; - var b4 = b[4] | 0; - var bl4 = b4 & 0x1fff; - var bh4 = b4 >>> 13; - var b5 = b[5] | 0; - var bl5 = b5 & 0x1fff; - var bh5 = b5 >>> 13; - var b6 = b[6] | 0; - var bl6 = b6 & 0x1fff; - var bh6 = b6 >>> 13; - var b7 = b[7] | 0; - var bl7 = b7 & 0x1fff; - var bh7 = b7 >>> 13; - var b8 = b[8] | 0; - var bl8 = b8 & 0x1fff; - var bh8 = b8 >>> 13; - var b9 = b[9] | 0; - var bl9 = b9 & 0x1fff; - var bh9 = b9 >>> 13; - - out.negative = self.negative ^ num.negative; - out.length = 19; - /* k = 0 */ - lo = Math.imul(al0, bl0); - mid = Math.imul(al0, bh0); - mid = (mid + Math.imul(ah0, bl0)) | 0; - hi = Math.imul(ah0, bh0); - var w0 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w0 >>> 26)) | 0; - w0 &= 0x3ffffff; - /* k = 1 */ - lo = Math.imul(al1, bl0); - mid = Math.imul(al1, bh0); - mid = (mid + Math.imul(ah1, bl0)) | 0; - hi = Math.imul(ah1, bh0); - lo = (lo + Math.imul(al0, bl1)) | 0; - mid = (mid + Math.imul(al0, bh1)) | 0; - mid = (mid + Math.imul(ah0, bl1)) | 0; - hi = (hi + Math.imul(ah0, bh1)) | 0; - var w1 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w1 >>> 26)) | 0; - w1 &= 0x3ffffff; - /* k = 2 */ - lo = Math.imul(al2, bl0); - mid = Math.imul(al2, bh0); - mid = (mid + Math.imul(ah2, bl0)) | 0; - hi = Math.imul(ah2, bh0); - lo = (lo + Math.imul(al1, bl1)) | 0; - mid = (mid + Math.imul(al1, bh1)) | 0; - mid = (mid + Math.imul(ah1, bl1)) | 0; - hi = (hi + Math.imul(ah1, bh1)) | 0; - lo = (lo + Math.imul(al0, bl2)) | 0; - mid = (mid + Math.imul(al0, bh2)) | 0; - mid = (mid + Math.imul(ah0, bl2)) | 0; - hi = (hi + Math.imul(ah0, bh2)) | 0; - var w2 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w2 >>> 26)) | 0; - w2 &= 0x3ffffff; - /* k = 3 */ - lo = Math.imul(al3, bl0); - mid = Math.imul(al3, bh0); - mid = (mid + Math.imul(ah3, bl0)) | 0; - hi = Math.imul(ah3, bh0); - lo = (lo + Math.imul(al2, bl1)) | 0; - mid = (mid + Math.imul(al2, bh1)) | 0; - mid = (mid + Math.imul(ah2, bl1)) | 0; - hi = (hi + Math.imul(ah2, bh1)) | 0; - lo = (lo + Math.imul(al1, bl2)) | 0; - mid = (mid + Math.imul(al1, bh2)) | 0; - mid = (mid + Math.imul(ah1, bl2)) | 0; - hi = (hi + Math.imul(ah1, bh2)) | 0; - lo = (lo + Math.imul(al0, bl3)) | 0; - mid = (mid + Math.imul(al0, bh3)) | 0; - mid = (mid + Math.imul(ah0, bl3)) | 0; - hi = (hi + Math.imul(ah0, bh3)) | 0; - var w3 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w3 >>> 26)) | 0; - w3 &= 0x3ffffff; - /* k = 4 */ - lo = Math.imul(al4, bl0); - mid = Math.imul(al4, bh0); - mid = (mid + Math.imul(ah4, bl0)) | 0; - hi = Math.imul(ah4, bh0); - lo = (lo + Math.imul(al3, bl1)) | 0; - mid = (mid + Math.imul(al3, bh1)) | 0; - mid = (mid + Math.imul(ah3, bl1)) | 0; - hi = (hi + Math.imul(ah3, bh1)) | 0; - lo = (lo + Math.imul(al2, bl2)) | 0; - mid = (mid + Math.imul(al2, bh2)) | 0; - mid = (mid + Math.imul(ah2, bl2)) | 0; - hi = (hi + Math.imul(ah2, bh2)) | 0; - lo = (lo + Math.imul(al1, bl3)) | 0; - mid = (mid + Math.imul(al1, bh3)) | 0; - mid = (mid + Math.imul(ah1, bl3)) | 0; - hi = (hi + Math.imul(ah1, bh3)) | 0; - lo = (lo + Math.imul(al0, bl4)) | 0; - mid = (mid + Math.imul(al0, bh4)) | 0; - mid = (mid + Math.imul(ah0, bl4)) | 0; - hi = (hi + Math.imul(ah0, bh4)) | 0; - var w4 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w4 >>> 26)) | 0; - w4 &= 0x3ffffff; - /* k = 5 */ - lo = Math.imul(al5, bl0); - mid = Math.imul(al5, bh0); - mid = (mid + Math.imul(ah5, bl0)) | 0; - hi = Math.imul(ah5, bh0); - lo = (lo + Math.imul(al4, bl1)) | 0; - mid = (mid + Math.imul(al4, bh1)) | 0; - mid = (mid + Math.imul(ah4, bl1)) | 0; - hi = (hi + Math.imul(ah4, bh1)) | 0; - lo = (lo + Math.imul(al3, bl2)) | 0; - mid = (mid + Math.imul(al3, bh2)) | 0; - mid = (mid + Math.imul(ah3, bl2)) | 0; - hi = (hi + Math.imul(ah3, bh2)) | 0; - lo = (lo + Math.imul(al2, bl3)) | 0; - mid = (mid + Math.imul(al2, bh3)) | 0; - mid = (mid + Math.imul(ah2, bl3)) | 0; - hi = (hi + Math.imul(ah2, bh3)) | 0; - lo = (lo + Math.imul(al1, bl4)) | 0; - mid = (mid + Math.imul(al1, bh4)) | 0; - mid = (mid + Math.imul(ah1, bl4)) | 0; - hi = (hi + Math.imul(ah1, bh4)) | 0; - lo = (lo + Math.imul(al0, bl5)) | 0; - mid = (mid + Math.imul(al0, bh5)) | 0; - mid = (mid + Math.imul(ah0, bl5)) | 0; - hi = (hi + Math.imul(ah0, bh5)) | 0; - var w5 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w5 >>> 26)) | 0; - w5 &= 0x3ffffff; - /* k = 6 */ - lo = Math.imul(al6, bl0); - mid = Math.imul(al6, bh0); - mid = (mid + Math.imul(ah6, bl0)) | 0; - hi = Math.imul(ah6, bh0); - lo = (lo + Math.imul(al5, bl1)) | 0; - mid = (mid + Math.imul(al5, bh1)) | 0; - mid = (mid + Math.imul(ah5, bl1)) | 0; - hi = (hi + Math.imul(ah5, bh1)) | 0; - lo = (lo + Math.imul(al4, bl2)) | 0; - mid = (mid + Math.imul(al4, bh2)) | 0; - mid = (mid + Math.imul(ah4, bl2)) | 0; - hi = (hi + Math.imul(ah4, bh2)) | 0; - lo = (lo + Math.imul(al3, bl3)) | 0; - mid = (mid + Math.imul(al3, bh3)) | 0; - mid = (mid + Math.imul(ah3, bl3)) | 0; - hi = (hi + Math.imul(ah3, bh3)) | 0; - lo = (lo + Math.imul(al2, bl4)) | 0; - mid = (mid + Math.imul(al2, bh4)) | 0; - mid = (mid + Math.imul(ah2, bl4)) | 0; - hi = (hi + Math.imul(ah2, bh4)) | 0; - lo = (lo + Math.imul(al1, bl5)) | 0; - mid = (mid + Math.imul(al1, bh5)) | 0; - mid = (mid + Math.imul(ah1, bl5)) | 0; - hi = (hi + Math.imul(ah1, bh5)) | 0; - lo = (lo + Math.imul(al0, bl6)) | 0; - mid = (mid + Math.imul(al0, bh6)) | 0; - mid = (mid + Math.imul(ah0, bl6)) | 0; - hi = (hi + Math.imul(ah0, bh6)) | 0; - var w6 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w6 >>> 26)) | 0; - w6 &= 0x3ffffff; - /* k = 7 */ - lo = Math.imul(al7, bl0); - mid = Math.imul(al7, bh0); - mid = (mid + Math.imul(ah7, bl0)) | 0; - hi = Math.imul(ah7, bh0); - lo = (lo + Math.imul(al6, bl1)) | 0; - mid = (mid + Math.imul(al6, bh1)) | 0; - mid = (mid + Math.imul(ah6, bl1)) | 0; - hi = (hi + Math.imul(ah6, bh1)) | 0; - lo = (lo + Math.imul(al5, bl2)) | 0; - mid = (mid + Math.imul(al5, bh2)) | 0; - mid = (mid + Math.imul(ah5, bl2)) | 0; - hi = (hi + Math.imul(ah5, bh2)) | 0; - lo = (lo + Math.imul(al4, bl3)) | 0; - mid = (mid + Math.imul(al4, bh3)) | 0; - mid = (mid + Math.imul(ah4, bl3)) | 0; - hi = (hi + Math.imul(ah4, bh3)) | 0; - lo = (lo + Math.imul(al3, bl4)) | 0; - mid = (mid + Math.imul(al3, bh4)) | 0; - mid = (mid + Math.imul(ah3, bl4)) | 0; - hi = (hi + Math.imul(ah3, bh4)) | 0; - lo = (lo + Math.imul(al2, bl5)) | 0; - mid = (mid + Math.imul(al2, bh5)) | 0; - mid = (mid + Math.imul(ah2, bl5)) | 0; - hi = (hi + Math.imul(ah2, bh5)) | 0; - lo = (lo + Math.imul(al1, bl6)) | 0; - mid = (mid + Math.imul(al1, bh6)) | 0; - mid = (mid + Math.imul(ah1, bl6)) | 0; - hi = (hi + Math.imul(ah1, bh6)) | 0; - lo = (lo + Math.imul(al0, bl7)) | 0; - mid = (mid + Math.imul(al0, bh7)) | 0; - mid = (mid + Math.imul(ah0, bl7)) | 0; - hi = (hi + Math.imul(ah0, bh7)) | 0; - var w7 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w7 >>> 26)) | 0; - w7 &= 0x3ffffff; - /* k = 8 */ - lo = Math.imul(al8, bl0); - mid = Math.imul(al8, bh0); - mid = (mid + Math.imul(ah8, bl0)) | 0; - hi = Math.imul(ah8, bh0); - lo = (lo + Math.imul(al7, bl1)) | 0; - mid = (mid + Math.imul(al7, bh1)) | 0; - mid = (mid + Math.imul(ah7, bl1)) | 0; - hi = (hi + Math.imul(ah7, bh1)) | 0; - lo = (lo + Math.imul(al6, bl2)) | 0; - mid = (mid + Math.imul(al6, bh2)) | 0; - mid = (mid + Math.imul(ah6, bl2)) | 0; - hi = (hi + Math.imul(ah6, bh2)) | 0; - lo = (lo + Math.imul(al5, bl3)) | 0; - mid = (mid + Math.imul(al5, bh3)) | 0; - mid = (mid + Math.imul(ah5, bl3)) | 0; - hi = (hi + Math.imul(ah5, bh3)) | 0; - lo = (lo + Math.imul(al4, bl4)) | 0; - mid = (mid + Math.imul(al4, bh4)) | 0; - mid = (mid + Math.imul(ah4, bl4)) | 0; - hi = (hi + Math.imul(ah4, bh4)) | 0; - lo = (lo + Math.imul(al3, bl5)) | 0; - mid = (mid + Math.imul(al3, bh5)) | 0; - mid = (mid + Math.imul(ah3, bl5)) | 0; - hi = (hi + Math.imul(ah3, bh5)) | 0; - lo = (lo + Math.imul(al2, bl6)) | 0; - mid = (mid + Math.imul(al2, bh6)) | 0; - mid = (mid + Math.imul(ah2, bl6)) | 0; - hi = (hi + Math.imul(ah2, bh6)) | 0; - lo = (lo + Math.imul(al1, bl7)) | 0; - mid = (mid + Math.imul(al1, bh7)) | 0; - mid = (mid + Math.imul(ah1, bl7)) | 0; - hi = (hi + Math.imul(ah1, bh7)) | 0; - lo = (lo + Math.imul(al0, bl8)) | 0; - mid = (mid + Math.imul(al0, bh8)) | 0; - mid = (mid + Math.imul(ah0, bl8)) | 0; - hi = (hi + Math.imul(ah0, bh8)) | 0; - var w8 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w8 >>> 26)) | 0; - w8 &= 0x3ffffff; - /* k = 9 */ - lo = Math.imul(al9, bl0); - mid = Math.imul(al9, bh0); - mid = (mid + Math.imul(ah9, bl0)) | 0; - hi = Math.imul(ah9, bh0); - lo = (lo + Math.imul(al8, bl1)) | 0; - mid = (mid + Math.imul(al8, bh1)) | 0; - mid = (mid + Math.imul(ah8, bl1)) | 0; - hi = (hi + Math.imul(ah8, bh1)) | 0; - lo = (lo + Math.imul(al7, bl2)) | 0; - mid = (mid + Math.imul(al7, bh2)) | 0; - mid = (mid + Math.imul(ah7, bl2)) | 0; - hi = (hi + Math.imul(ah7, bh2)) | 0; - lo = (lo + Math.imul(al6, bl3)) | 0; - mid = (mid + Math.imul(al6, bh3)) | 0; - mid = (mid + Math.imul(ah6, bl3)) | 0; - hi = (hi + Math.imul(ah6, bh3)) | 0; - lo = (lo + Math.imul(al5, bl4)) | 0; - mid = (mid + Math.imul(al5, bh4)) | 0; - mid = (mid + Math.imul(ah5, bl4)) | 0; - hi = (hi + Math.imul(ah5, bh4)) | 0; - lo = (lo + Math.imul(al4, bl5)) | 0; - mid = (mid + Math.imul(al4, bh5)) | 0; - mid = (mid + Math.imul(ah4, bl5)) | 0; - hi = (hi + Math.imul(ah4, bh5)) | 0; - lo = (lo + Math.imul(al3, bl6)) | 0; - mid = (mid + Math.imul(al3, bh6)) | 0; - mid = (mid + Math.imul(ah3, bl6)) | 0; - hi = (hi + Math.imul(ah3, bh6)) | 0; - lo = (lo + Math.imul(al2, bl7)) | 0; - mid = (mid + Math.imul(al2, bh7)) | 0; - mid = (mid + Math.imul(ah2, bl7)) | 0; - hi = (hi + Math.imul(ah2, bh7)) | 0; - lo = (lo + Math.imul(al1, bl8)) | 0; - mid = (mid + Math.imul(al1, bh8)) | 0; - mid = (mid + Math.imul(ah1, bl8)) | 0; - hi = (hi + Math.imul(ah1, bh8)) | 0; - lo = (lo + Math.imul(al0, bl9)) | 0; - mid = (mid + Math.imul(al0, bh9)) | 0; - mid = (mid + Math.imul(ah0, bl9)) | 0; - hi = (hi + Math.imul(ah0, bh9)) | 0; - var w9 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w9 >>> 26)) | 0; - w9 &= 0x3ffffff; - /* k = 10 */ - lo = Math.imul(al9, bl1); - mid = Math.imul(al9, bh1); - mid = (mid + Math.imul(ah9, bl1)) | 0; - hi = Math.imul(ah9, bh1); - lo = (lo + Math.imul(al8, bl2)) | 0; - mid = (mid + Math.imul(al8, bh2)) | 0; - mid = (mid + Math.imul(ah8, bl2)) | 0; - hi = (hi + Math.imul(ah8, bh2)) | 0; - lo = (lo + Math.imul(al7, bl3)) | 0; - mid = (mid + Math.imul(al7, bh3)) | 0; - mid = (mid + Math.imul(ah7, bl3)) | 0; - hi = (hi + Math.imul(ah7, bh3)) | 0; - lo = (lo + Math.imul(al6, bl4)) | 0; - mid = (mid + Math.imul(al6, bh4)) | 0; - mid = (mid + Math.imul(ah6, bl4)) | 0; - hi = (hi + Math.imul(ah6, bh4)) | 0; - lo = (lo + Math.imul(al5, bl5)) | 0; - mid = (mid + Math.imul(al5, bh5)) | 0; - mid = (mid + Math.imul(ah5, bl5)) | 0; - hi = (hi + Math.imul(ah5, bh5)) | 0; - lo = (lo + Math.imul(al4, bl6)) | 0; - mid = (mid + Math.imul(al4, bh6)) | 0; - mid = (mid + Math.imul(ah4, bl6)) | 0; - hi = (hi + Math.imul(ah4, bh6)) | 0; - lo = (lo + Math.imul(al3, bl7)) | 0; - mid = (mid + Math.imul(al3, bh7)) | 0; - mid = (mid + Math.imul(ah3, bl7)) | 0; - hi = (hi + Math.imul(ah3, bh7)) | 0; - lo = (lo + Math.imul(al2, bl8)) | 0; - mid = (mid + Math.imul(al2, bh8)) | 0; - mid = (mid + Math.imul(ah2, bl8)) | 0; - hi = (hi + Math.imul(ah2, bh8)) | 0; - lo = (lo + Math.imul(al1, bl9)) | 0; - mid = (mid + Math.imul(al1, bh9)) | 0; - mid = (mid + Math.imul(ah1, bl9)) | 0; - hi = (hi + Math.imul(ah1, bh9)) | 0; - var w10 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w10 >>> 26)) | 0; - w10 &= 0x3ffffff; - /* k = 11 */ - lo = Math.imul(al9, bl2); - mid = Math.imul(al9, bh2); - mid = (mid + Math.imul(ah9, bl2)) | 0; - hi = Math.imul(ah9, bh2); - lo = (lo + Math.imul(al8, bl3)) | 0; - mid = (mid + Math.imul(al8, bh3)) | 0; - mid = (mid + Math.imul(ah8, bl3)) | 0; - hi = (hi + Math.imul(ah8, bh3)) | 0; - lo = (lo + Math.imul(al7, bl4)) | 0; - mid = (mid + Math.imul(al7, bh4)) | 0; - mid = (mid + Math.imul(ah7, bl4)) | 0; - hi = (hi + Math.imul(ah7, bh4)) | 0; - lo = (lo + Math.imul(al6, bl5)) | 0; - mid = (mid + Math.imul(al6, bh5)) | 0; - mid = (mid + Math.imul(ah6, bl5)) | 0; - hi = (hi + Math.imul(ah6, bh5)) | 0; - lo = (lo + Math.imul(al5, bl6)) | 0; - mid = (mid + Math.imul(al5, bh6)) | 0; - mid = (mid + Math.imul(ah5, bl6)) | 0; - hi = (hi + Math.imul(ah5, bh6)) | 0; - lo = (lo + Math.imul(al4, bl7)) | 0; - mid = (mid + Math.imul(al4, bh7)) | 0; - mid = (mid + Math.imul(ah4, bl7)) | 0; - hi = (hi + Math.imul(ah4, bh7)) | 0; - lo = (lo + Math.imul(al3, bl8)) | 0; - mid = (mid + Math.imul(al3, bh8)) | 0; - mid = (mid + Math.imul(ah3, bl8)) | 0; - hi = (hi + Math.imul(ah3, bh8)) | 0; - lo = (lo + Math.imul(al2, bl9)) | 0; - mid = (mid + Math.imul(al2, bh9)) | 0; - mid = (mid + Math.imul(ah2, bl9)) | 0; - hi = (hi + Math.imul(ah2, bh9)) | 0; - var w11 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w11 >>> 26)) | 0; - w11 &= 0x3ffffff; - /* k = 12 */ - lo = Math.imul(al9, bl3); - mid = Math.imul(al9, bh3); - mid = (mid + Math.imul(ah9, bl3)) | 0; - hi = Math.imul(ah9, bh3); - lo = (lo + Math.imul(al8, bl4)) | 0; - mid = (mid + Math.imul(al8, bh4)) | 0; - mid = (mid + Math.imul(ah8, bl4)) | 0; - hi = (hi + Math.imul(ah8, bh4)) | 0; - lo = (lo + Math.imul(al7, bl5)) | 0; - mid = (mid + Math.imul(al7, bh5)) | 0; - mid = (mid + Math.imul(ah7, bl5)) | 0; - hi = (hi + Math.imul(ah7, bh5)) | 0; - lo = (lo + Math.imul(al6, bl6)) | 0; - mid = (mid + Math.imul(al6, bh6)) | 0; - mid = (mid + Math.imul(ah6, bl6)) | 0; - hi = (hi + Math.imul(ah6, bh6)) | 0; - lo = (lo + Math.imul(al5, bl7)) | 0; - mid = (mid + Math.imul(al5, bh7)) | 0; - mid = (mid + Math.imul(ah5, bl7)) | 0; - hi = (hi + Math.imul(ah5, bh7)) | 0; - lo = (lo + Math.imul(al4, bl8)) | 0; - mid = (mid + Math.imul(al4, bh8)) | 0; - mid = (mid + Math.imul(ah4, bl8)) | 0; - hi = (hi + Math.imul(ah4, bh8)) | 0; - lo = (lo + Math.imul(al3, bl9)) | 0; - mid = (mid + Math.imul(al3, bh9)) | 0; - mid = (mid + Math.imul(ah3, bl9)) | 0; - hi = (hi + Math.imul(ah3, bh9)) | 0; - var w12 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w12 >>> 26)) | 0; - w12 &= 0x3ffffff; - /* k = 13 */ - lo = Math.imul(al9, bl4); - mid = Math.imul(al9, bh4); - mid = (mid + Math.imul(ah9, bl4)) | 0; - hi = Math.imul(ah9, bh4); - lo = (lo + Math.imul(al8, bl5)) | 0; - mid = (mid + Math.imul(al8, bh5)) | 0; - mid = (mid + Math.imul(ah8, bl5)) | 0; - hi = (hi + Math.imul(ah8, bh5)) | 0; - lo = (lo + Math.imul(al7, bl6)) | 0; - mid = (mid + Math.imul(al7, bh6)) | 0; - mid = (mid + Math.imul(ah7, bl6)) | 0; - hi = (hi + Math.imul(ah7, bh6)) | 0; - lo = (lo + Math.imul(al6, bl7)) | 0; - mid = (mid + Math.imul(al6, bh7)) | 0; - mid = (mid + Math.imul(ah6, bl7)) | 0; - hi = (hi + Math.imul(ah6, bh7)) | 0; - lo = (lo + Math.imul(al5, bl8)) | 0; - mid = (mid + Math.imul(al5, bh8)) | 0; - mid = (mid + Math.imul(ah5, bl8)) | 0; - hi = (hi + Math.imul(ah5, bh8)) | 0; - lo = (lo + Math.imul(al4, bl9)) | 0; - mid = (mid + Math.imul(al4, bh9)) | 0; - mid = (mid + Math.imul(ah4, bl9)) | 0; - hi = (hi + Math.imul(ah4, bh9)) | 0; - var w13 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w13 >>> 26)) | 0; - w13 &= 0x3ffffff; - /* k = 14 */ - lo = Math.imul(al9, bl5); - mid = Math.imul(al9, bh5); - mid = (mid + Math.imul(ah9, bl5)) | 0; - hi = Math.imul(ah9, bh5); - lo = (lo + Math.imul(al8, bl6)) | 0; - mid = (mid + Math.imul(al8, bh6)) | 0; - mid = (mid + Math.imul(ah8, bl6)) | 0; - hi = (hi + Math.imul(ah8, bh6)) | 0; - lo = (lo + Math.imul(al7, bl7)) | 0; - mid = (mid + Math.imul(al7, bh7)) | 0; - mid = (mid + Math.imul(ah7, bl7)) | 0; - hi = (hi + Math.imul(ah7, bh7)) | 0; - lo = (lo + Math.imul(al6, bl8)) | 0; - mid = (mid + Math.imul(al6, bh8)) | 0; - mid = (mid + Math.imul(ah6, bl8)) | 0; - hi = (hi + Math.imul(ah6, bh8)) | 0; - lo = (lo + Math.imul(al5, bl9)) | 0; - mid = (mid + Math.imul(al5, bh9)) | 0; - mid = (mid + Math.imul(ah5, bl9)) | 0; - hi = (hi + Math.imul(ah5, bh9)) | 0; - var w14 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w14 >>> 26)) | 0; - w14 &= 0x3ffffff; - /* k = 15 */ - lo = Math.imul(al9, bl6); - mid = Math.imul(al9, bh6); - mid = (mid + Math.imul(ah9, bl6)) | 0; - hi = Math.imul(ah9, bh6); - lo = (lo + Math.imul(al8, bl7)) | 0; - mid = (mid + Math.imul(al8, bh7)) | 0; - mid = (mid + Math.imul(ah8, bl7)) | 0; - hi = (hi + Math.imul(ah8, bh7)) | 0; - lo = (lo + Math.imul(al7, bl8)) | 0; - mid = (mid + Math.imul(al7, bh8)) | 0; - mid = (mid + Math.imul(ah7, bl8)) | 0; - hi = (hi + Math.imul(ah7, bh8)) | 0; - lo = (lo + Math.imul(al6, bl9)) | 0; - mid = (mid + Math.imul(al6, bh9)) | 0; - mid = (mid + Math.imul(ah6, bl9)) | 0; - hi = (hi + Math.imul(ah6, bh9)) | 0; - var w15 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w15 >>> 26)) | 0; - w15 &= 0x3ffffff; - /* k = 16 */ - lo = Math.imul(al9, bl7); - mid = Math.imul(al9, bh7); - mid = (mid + Math.imul(ah9, bl7)) | 0; - hi = Math.imul(ah9, bh7); - lo = (lo + Math.imul(al8, bl8)) | 0; - mid = (mid + Math.imul(al8, bh8)) | 0; - mid = (mid + Math.imul(ah8, bl8)) | 0; - hi = (hi + Math.imul(ah8, bh8)) | 0; - lo = (lo + Math.imul(al7, bl9)) | 0; - mid = (mid + Math.imul(al7, bh9)) | 0; - mid = (mid + Math.imul(ah7, bl9)) | 0; - hi = (hi + Math.imul(ah7, bh9)) | 0; - var w16 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w16 >>> 26)) | 0; - w16 &= 0x3ffffff; - /* k = 17 */ - lo = Math.imul(al9, bl8); - mid = Math.imul(al9, bh8); - mid = (mid + Math.imul(ah9, bl8)) | 0; - hi = Math.imul(ah9, bh8); - lo = (lo + Math.imul(al8, bl9)) | 0; - mid = (mid + Math.imul(al8, bh9)) | 0; - mid = (mid + Math.imul(ah8, bl9)) | 0; - hi = (hi + Math.imul(ah8, bh9)) | 0; - var w17 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w17 >>> 26)) | 0; - w17 &= 0x3ffffff; - /* k = 18 */ - lo = Math.imul(al9, bl9); - mid = Math.imul(al9, bh9); - mid = (mid + Math.imul(ah9, bl9)) | 0; - hi = Math.imul(ah9, bh9); - var w18 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w18 >>> 26)) | 0; - w18 &= 0x3ffffff; - o[0] = w0; - o[1] = w1; - o[2] = w2; - o[3] = w3; - o[4] = w4; - o[5] = w5; - o[6] = w6; - o[7] = w7; - o[8] = w8; - o[9] = w9; - o[10] = w10; - o[11] = w11; - o[12] = w12; - o[13] = w13; - o[14] = w14; - o[15] = w15; - o[16] = w16; - o[17] = w17; - o[18] = w18; - if (c !== 0) { - o[19] = c; - out.length++; - } - return out; - }; - - // Polyfill comb - if (!Math.imul) { - comb10MulTo = smallMulTo; - } - - function bigMulTo (self, num, out) { - out.negative = num.negative ^ self.negative; - out.length = self.length + num.length; - - var carry = 0; - var hncarry = 0; - for (var k = 0; k < out.length - 1; k++) { - // Sum all words with the same `i + j = k` and accumulate `ncarry`, - // note that ncarry could be >= 0x3ffffff - var ncarry = hncarry; - hncarry = 0; - var rword = carry & 0x3ffffff; - var maxJ = Math.min(k, num.length - 1); - for (var j = Math.max(0, k - self.length + 1); j <= maxJ; j++) { - var i = k - j; - var a = self.words[i] | 0; - var b = num.words[j] | 0; - var r = a * b; - - var lo = r & 0x3ffffff; - ncarry = (ncarry + ((r / 0x4000000) | 0)) | 0; - lo = (lo + rword) | 0; - rword = lo & 0x3ffffff; - ncarry = (ncarry + (lo >>> 26)) | 0; - - hncarry += ncarry >>> 26; - ncarry &= 0x3ffffff; - } - out.words[k] = rword; - carry = ncarry; - ncarry = hncarry; - } - if (carry !== 0) { - out.words[k] = carry; - } else { - out.length--; - } - - return out.strip(); - } - - function jumboMulTo (self, num, out) { - var fftm = new FFTM(); - return fftm.mulp(self, num, out); - } - - BN.prototype.mulTo = function mulTo (num, out) { - var res; - var len = this.length + num.length; - if (this.length === 10 && num.length === 10) { - res = comb10MulTo(this, num, out); - } else if (len < 63) { - res = smallMulTo(this, num, out); - } else if (len < 1024) { - res = bigMulTo(this, num, out); - } else { - res = jumboMulTo(this, num, out); - } - - return res; - }; - - // Cooley-Tukey algorithm for FFT - // slightly revisited to rely on looping instead of recursion - - function FFTM (x, y) { - this.x = x; - this.y = y; - } - - FFTM.prototype.makeRBT = function makeRBT (N) { - var t = new Array(N); - var l = BN.prototype._countBits(N) - 1; - for (var i = 0; i < N; i++) { - t[i] = this.revBin(i, l, N); - } - - return t; - }; - - // Returns binary-reversed representation of `x` - FFTM.prototype.revBin = function revBin (x, l, N) { - if (x === 0 || x === N - 1) return x; - - var rb = 0; - for (var i = 0; i < l; i++) { - rb |= (x & 1) << (l - i - 1); - x >>= 1; - } - - return rb; - }; - - // Performs "tweedling" phase, therefore 'emulating' - // behaviour of the recursive algorithm - FFTM.prototype.permute = function permute (rbt, rws, iws, rtws, itws, N) { - for (var i = 0; i < N; i++) { - rtws[i] = rws[rbt[i]]; - itws[i] = iws[rbt[i]]; - } - }; - - FFTM.prototype.transform = function transform (rws, iws, rtws, itws, N, rbt) { - this.permute(rbt, rws, iws, rtws, itws, N); - - for (var s = 1; s < N; s <<= 1) { - var l = s << 1; - - var rtwdf = Math.cos(2 * Math.PI / l); - var itwdf = Math.sin(2 * Math.PI / l); - - for (var p = 0; p < N; p += l) { - var rtwdf_ = rtwdf; - var itwdf_ = itwdf; - - for (var j = 0; j < s; j++) { - var re = rtws[p + j]; - var ie = itws[p + j]; - - var ro = rtws[p + j + s]; - var io = itws[p + j + s]; - - var rx = rtwdf_ * ro - itwdf_ * io; - - io = rtwdf_ * io + itwdf_ * ro; - ro = rx; - - rtws[p + j] = re + ro; - itws[p + j] = ie + io; - - rtws[p + j + s] = re - ro; - itws[p + j + s] = ie - io; - - /* jshint maxdepth : false */ - if (j !== l) { - rx = rtwdf * rtwdf_ - itwdf * itwdf_; - - itwdf_ = rtwdf * itwdf_ + itwdf * rtwdf_; - rtwdf_ = rx; - } - } - } - } - }; - - FFTM.prototype.guessLen13b = function guessLen13b (n, m) { - var N = Math.max(m, n) | 1; - var odd = N & 1; - var i = 0; - for (N = N / 2 | 0; N; N = N >>> 1) { - i++; - } - - return 1 << i + 1 + odd; - }; - - FFTM.prototype.conjugate = function conjugate (rws, iws, N) { - if (N <= 1) return; - - for (var i = 0; i < N / 2; i++) { - var t = rws[i]; - - rws[i] = rws[N - i - 1]; - rws[N - i - 1] = t; - - t = iws[i]; - - iws[i] = -iws[N - i - 1]; - iws[N - i - 1] = -t; - } - }; - - FFTM.prototype.normalize13b = function normalize13b (ws, N) { - var carry = 0; - for (var i = 0; i < N / 2; i++) { - var w = Math.round(ws[2 * i + 1] / N) * 0x2000 + - Math.round(ws[2 * i] / N) + - carry; - - ws[i] = w & 0x3ffffff; - - if (w < 0x4000000) { - carry = 0; - } else { - carry = w / 0x4000000 | 0; - } - } - - return ws; - }; - - FFTM.prototype.convert13b = function convert13b (ws, len, rws, N) { - var carry = 0; - for (var i = 0; i < len; i++) { - carry = carry + (ws[i] | 0); - - rws[2 * i] = carry & 0x1fff; carry = carry >>> 13; - rws[2 * i + 1] = carry & 0x1fff; carry = carry >>> 13; - } - - // Pad with zeroes - for (i = 2 * len; i < N; ++i) { - rws[i] = 0; - } - - assert(carry === 0); - assert((carry & ~0x1fff) === 0); - }; - - FFTM.prototype.stub = function stub (N) { - var ph = new Array(N); - for (var i = 0; i < N; i++) { - ph[i] = 0; - } - - return ph; - }; - - FFTM.prototype.mulp = function mulp (x, y, out) { - var N = 2 * this.guessLen13b(x.length, y.length); - - var rbt = this.makeRBT(N); - - var _ = this.stub(N); - - var rws = new Array(N); - var rwst = new Array(N); - var iwst = new Array(N); - - var nrws = new Array(N); - var nrwst = new Array(N); - var niwst = new Array(N); - - var rmws = out.words; - rmws.length = N; - - this.convert13b(x.words, x.length, rws, N); - this.convert13b(y.words, y.length, nrws, N); - - this.transform(rws, _, rwst, iwst, N, rbt); - this.transform(nrws, _, nrwst, niwst, N, rbt); - - for (var i = 0; i < N; i++) { - var rx = rwst[i] * nrwst[i] - iwst[i] * niwst[i]; - iwst[i] = rwst[i] * niwst[i] + iwst[i] * nrwst[i]; - rwst[i] = rx; - } - - this.conjugate(rwst, iwst, N); - this.transform(rwst, iwst, rmws, _, N, rbt); - this.conjugate(rmws, _, N); - this.normalize13b(rmws, N); - - out.negative = x.negative ^ y.negative; - out.length = x.length + y.length; - return out.strip(); - }; - - // Multiply `this` by `num` - BN.prototype.mul = function mul (num) { - var out = new BN(null); - out.words = new Array(this.length + num.length); - return this.mulTo(num, out); - }; - - // Multiply employing FFT - BN.prototype.mulf = function mulf (num) { - var out = new BN(null); - out.words = new Array(this.length + num.length); - return jumboMulTo(this, num, out); - }; - - // In-place Multiplication - BN.prototype.imul = function imul (num) { - return this.clone().mulTo(num, this); - }; - - BN.prototype.imuln = function imuln (num) { - assert(typeof num === 'number'); - assert(num < 0x4000000); - - // Carry - var carry = 0; - for (var i = 0; i < this.length; i++) { - var w = (this.words[i] | 0) * num; - var lo = (w & 0x3ffffff) + (carry & 0x3ffffff); - carry >>= 26; - carry += (w / 0x4000000) | 0; - // NOTE: lo is 27bit maximum - carry += lo >>> 26; - this.words[i] = lo & 0x3ffffff; - } - - if (carry !== 0) { - this.words[i] = carry; - this.length++; - } - - return this; - }; - - BN.prototype.muln = function muln (num) { - return this.clone().imuln(num); - }; - - // `this` * `this` - BN.prototype.sqr = function sqr () { - return this.mul(this); - }; - - // `this` * `this` in-place - BN.prototype.isqr = function isqr () { - return this.imul(this.clone()); - }; - - // Math.pow(`this`, `num`) - BN.prototype.pow = function pow (num) { - var w = toBitArray(num); - if (w.length === 0) return new BN(1); - - // Skip leading zeroes - var res = this; - for (var i = 0; i < w.length; i++, res = res.sqr()) { - if (w[i] !== 0) break; - } - - if (++i < w.length) { - for (var q = res.sqr(); i < w.length; i++, q = q.sqr()) { - if (w[i] === 0) continue; - - res = res.mul(q); - } - } - - return res; - }; - - // Shift-left in-place - BN.prototype.iushln = function iushln (bits) { - assert(typeof bits === 'number' && bits >= 0); - var r = bits % 26; - var s = (bits - r) / 26; - var carryMask = (0x3ffffff >>> (26 - r)) << (26 - r); - var i; - - if (r !== 0) { - var carry = 0; - - for (i = 0; i < this.length; i++) { - var newCarry = this.words[i] & carryMask; - var c = ((this.words[i] | 0) - newCarry) << r; - this.words[i] = c | carry; - carry = newCarry >>> (26 - r); - } - - if (carry) { - this.words[i] = carry; - this.length++; - } - } - - if (s !== 0) { - for (i = this.length - 1; i >= 0; i--) { - this.words[i + s] = this.words[i]; - } - - for (i = 0; i < s; i++) { - this.words[i] = 0; - } - - this.length += s; - } - - return this.strip(); - }; - - BN.prototype.ishln = function ishln (bits) { - // TODO(indutny): implement me - assert(this.negative === 0); - return this.iushln(bits); - }; - - // Shift-right in-place - // NOTE: `hint` is a lowest bit before trailing zeroes - // NOTE: if `extended` is present - it will be filled with destroyed bits - BN.prototype.iushrn = function iushrn (bits, hint, extended) { - assert(typeof bits === 'number' && bits >= 0); - var h; - if (hint) { - h = (hint - (hint % 26)) / 26; - } else { - h = 0; - } - - var r = bits % 26; - var s = Math.min((bits - r) / 26, this.length); - var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r); - var maskedWords = extended; - - h -= s; - h = Math.max(0, h); - - // Extended mode, copy masked part - if (maskedWords) { - for (var i = 0; i < s; i++) { - maskedWords.words[i] = this.words[i]; - } - maskedWords.length = s; - } - - if (s === 0) { - // No-op, we should not move anything at all - } else if (this.length > s) { - this.length -= s; - for (i = 0; i < this.length; i++) { - this.words[i] = this.words[i + s]; - } - } else { - this.words[0] = 0; - this.length = 1; - } - - var carry = 0; - for (i = this.length - 1; i >= 0 && (carry !== 0 || i >= h); i--) { - var word = this.words[i] | 0; - this.words[i] = (carry << (26 - r)) | (word >>> r); - carry = word & mask; - } - - // Push carried bits as a mask - if (maskedWords && carry !== 0) { - maskedWords.words[maskedWords.length++] = carry; - } - - if (this.length === 0) { - this.words[0] = 0; - this.length = 1; - } - - return this.strip(); - }; - - BN.prototype.ishrn = function ishrn (bits, hint, extended) { - // TODO(indutny): implement me - assert(this.negative === 0); - return this.iushrn(bits, hint, extended); - }; - - // Shift-left - BN.prototype.shln = function shln (bits) { - return this.clone().ishln(bits); - }; - - BN.prototype.ushln = function ushln (bits) { - return this.clone().iushln(bits); - }; - - // Shift-right - BN.prototype.shrn = function shrn (bits) { - return this.clone().ishrn(bits); - }; - - BN.prototype.ushrn = function ushrn (bits) { - return this.clone().iushrn(bits); - }; - - // Test if n bit is set - BN.prototype.testn = function testn (bit) { - assert(typeof bit === 'number' && bit >= 0); - var r = bit % 26; - var s = (bit - r) / 26; - var q = 1 << r; - - // Fast case: bit is much higher than all existing words - if (this.length <= s) return false; - - // Check bit and return - var w = this.words[s]; - - return !!(w & q); - }; - - // Return only lowers bits of number (in-place) - BN.prototype.imaskn = function imaskn (bits) { - assert(typeof bits === 'number' && bits >= 0); - var r = bits % 26; - var s = (bits - r) / 26; - - assert(this.negative === 0, 'imaskn works only with positive numbers'); - - if (this.length <= s) { - return this; - } - - if (r !== 0) { - s++; - } - this.length = Math.min(s, this.length); - - if (r !== 0) { - var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r); - this.words[this.length - 1] &= mask; - } - - return this.strip(); - }; - - // Return only lowers bits of number - BN.prototype.maskn = function maskn (bits) { - return this.clone().imaskn(bits); - }; - - // Add plain number `num` to `this` - BN.prototype.iaddn = function iaddn (num) { - assert(typeof num === 'number'); - assert(num < 0x4000000); - if (num < 0) return this.isubn(-num); - - // Possible sign change - if (this.negative !== 0) { - if (this.length === 1 && (this.words[0] | 0) < num) { - this.words[0] = num - (this.words[0] | 0); - this.negative = 0; - return this; - } - - this.negative = 0; - this.isubn(num); - this.negative = 1; - return this; - } - - // Add without checks - return this._iaddn(num); - }; - - BN.prototype._iaddn = function _iaddn (num) { - this.words[0] += num; - - // Carry - for (var i = 0; i < this.length && this.words[i] >= 0x4000000; i++) { - this.words[i] -= 0x4000000; - if (i === this.length - 1) { - this.words[i + 1] = 1; - } else { - this.words[i + 1]++; - } - } - this.length = Math.max(this.length, i + 1); - - return this; - }; - - // Subtract plain number `num` from `this` - BN.prototype.isubn = function isubn (num) { - assert(typeof num === 'number'); - assert(num < 0x4000000); - if (num < 0) return this.iaddn(-num); - - if (this.negative !== 0) { - this.negative = 0; - this.iaddn(num); - this.negative = 1; - return this; - } - - this.words[0] -= num; - - if (this.length === 1 && this.words[0] < 0) { - this.words[0] = -this.words[0]; - this.negative = 1; - } else { - // Carry - for (var i = 0; i < this.length && this.words[i] < 0; i++) { - this.words[i] += 0x4000000; - this.words[i + 1] -= 1; - } - } - - return this.strip(); - }; - - BN.prototype.addn = function addn (num) { - return this.clone().iaddn(num); - }; - - BN.prototype.subn = function subn (num) { - return this.clone().isubn(num); - }; - - BN.prototype.iabs = function iabs () { - this.negative = 0; - - return this; - }; - - BN.prototype.abs = function abs () { - return this.clone().iabs(); - }; - - BN.prototype._ishlnsubmul = function _ishlnsubmul (num, mul, shift) { - var len = num.length + shift; - var i; - - this._expand(len); - - var w; - var carry = 0; - for (i = 0; i < num.length; i++) { - w = (this.words[i + shift] | 0) + carry; - var right = (num.words[i] | 0) * mul; - w -= right & 0x3ffffff; - carry = (w >> 26) - ((right / 0x4000000) | 0); - this.words[i + shift] = w & 0x3ffffff; - } - for (; i < this.length - shift; i++) { - w = (this.words[i + shift] | 0) + carry; - carry = w >> 26; - this.words[i + shift] = w & 0x3ffffff; - } - - if (carry === 0) return this.strip(); - - // Subtraction overflow - assert(carry === -1); - carry = 0; - for (i = 0; i < this.length; i++) { - w = -(this.words[i] | 0) + carry; - carry = w >> 26; - this.words[i] = w & 0x3ffffff; - } - this.negative = 1; - - return this.strip(); - }; - - BN.prototype._wordDiv = function _wordDiv (num, mode) { - var shift = this.length - num.length; - - var a = this.clone(); - var b = num; - - // Normalize - var bhi = b.words[b.length - 1] | 0; - var bhiBits = this._countBits(bhi); - shift = 26 - bhiBits; - if (shift !== 0) { - b = b.ushln(shift); - a.iushln(shift); - bhi = b.words[b.length - 1] | 0; - } - - // Initialize quotient - var m = a.length - b.length; - var q; - - if (mode !== 'mod') { - q = new BN(null); - q.length = m + 1; - q.words = new Array(q.length); - for (var i = 0; i < q.length; i++) { - q.words[i] = 0; - } - } - - var diff = a.clone()._ishlnsubmul(b, 1, m); - if (diff.negative === 0) { - a = diff; - if (q) { - q.words[m] = 1; - } - } - - for (var j = m - 1; j >= 0; j--) { - var qj = (a.words[b.length + j] | 0) * 0x4000000 + - (a.words[b.length + j - 1] | 0); - - // NOTE: (qj / bhi) is (0x3ffffff * 0x4000000 + 0x3ffffff) / 0x2000000 max - // (0x7ffffff) - qj = Math.min((qj / bhi) | 0, 0x3ffffff); - - a._ishlnsubmul(b, qj, j); - while (a.negative !== 0) { - qj--; - a.negative = 0; - a._ishlnsubmul(b, 1, j); - if (!a.isZero()) { - a.negative ^= 1; - } - } - if (q) { - q.words[j] = qj; - } - } - if (q) { - q.strip(); - } - a.strip(); - - // Denormalize - if (mode !== 'div' && shift !== 0) { - a.iushrn(shift); - } - - return { - div: q || null, - mod: a - }; - }; - - // NOTE: 1) `mode` can be set to `mod` to request mod only, - // to `div` to request div only, or be absent to - // request both div & mod - // 2) `positive` is true if unsigned mod is requested - BN.prototype.divmod = function divmod (num, mode, positive) { - assert(!num.isZero()); - - if (this.isZero()) { - return { - div: new BN(0), - mod: new BN(0) - }; - } - - var div, mod, res; - if (this.negative !== 0 && num.negative === 0) { - res = this.neg().divmod(num, mode); - - if (mode !== 'mod') { - div = res.div.neg(); - } - - if (mode !== 'div') { - mod = res.mod.neg(); - if (positive && mod.negative !== 0) { - mod.iadd(num); - } - } - - return { - div: div, - mod: mod - }; - } - - if (this.negative === 0 && num.negative !== 0) { - res = this.divmod(num.neg(), mode); - - if (mode !== 'mod') { - div = res.div.neg(); - } - - return { - div: div, - mod: res.mod - }; - } - - if ((this.negative & num.negative) !== 0) { - res = this.neg().divmod(num.neg(), mode); - - if (mode !== 'div') { - mod = res.mod.neg(); - if (positive && mod.negative !== 0) { - mod.isub(num); - } - } - - return { - div: res.div, - mod: mod - }; - } - - // Both numbers are positive at this point - - // Strip both numbers to approximate shift value - if (num.length > this.length || this.cmp(num) < 0) { - return { - div: new BN(0), - mod: this - }; - } - - // Very short reduction - if (num.length === 1) { - if (mode === 'div') { - return { - div: this.divn(num.words[0]), - mod: null - }; - } - - if (mode === 'mod') { - return { - div: null, - mod: new BN(this.modn(num.words[0])) - }; - } - - return { - div: this.divn(num.words[0]), - mod: new BN(this.modn(num.words[0])) - }; - } - - return this._wordDiv(num, mode); - }; - - // Find `this` / `num` - BN.prototype.div = function div (num) { - return this.divmod(num, 'div', false).div; - }; - - // Find `this` % `num` - BN.prototype.mod = function mod (num) { - return this.divmod(num, 'mod', false).mod; - }; - - BN.prototype.umod = function umod (num) { - return this.divmod(num, 'mod', true).mod; - }; - - // Find Round(`this` / `num`) - BN.prototype.divRound = function divRound (num) { - var dm = this.divmod(num); - - // Fast case - exact division - if (dm.mod.isZero()) return dm.div; - - var mod = dm.div.negative !== 0 ? dm.mod.isub(num) : dm.mod; - - var half = num.ushrn(1); - var r2 = num.andln(1); - var cmp = mod.cmp(half); - - // Round down - if (cmp < 0 || r2 === 1 && cmp === 0) return dm.div; - - // Round up - return dm.div.negative !== 0 ? dm.div.isubn(1) : dm.div.iaddn(1); - }; - - BN.prototype.modn = function modn (num) { - assert(num <= 0x3ffffff); - var p = (1 << 26) % num; - - var acc = 0; - for (var i = this.length - 1; i >= 0; i--) { - acc = (p * acc + (this.words[i] | 0)) % num; - } - - return acc; - }; - - // In-place division by number - BN.prototype.idivn = function idivn (num) { - assert(num <= 0x3ffffff); - - var carry = 0; - for (var i = this.length - 1; i >= 0; i--) { - var w = (this.words[i] | 0) + carry * 0x4000000; - this.words[i] = (w / num) | 0; - carry = w % num; - } - - return this.strip(); - }; - - BN.prototype.divn = function divn (num) { - return this.clone().idivn(num); - }; - - BN.prototype.egcd = function egcd (p) { - assert(p.negative === 0); - assert(!p.isZero()); - - var x = this; - var y = p.clone(); - - if (x.negative !== 0) { - x = x.umod(p); - } else { - x = x.clone(); - } - - // A * x + B * y = x - var A = new BN(1); - var B = new BN(0); - - // C * x + D * y = y - var C = new BN(0); - var D = new BN(1); - - var g = 0; - - while (x.isEven() && y.isEven()) { - x.iushrn(1); - y.iushrn(1); - ++g; - } - - var yp = y.clone(); - var xp = x.clone(); - - while (!x.isZero()) { - for (var i = 0, im = 1; (x.words[0] & im) === 0 && i < 26; ++i, im <<= 1); - if (i > 0) { - x.iushrn(i); - while (i-- > 0) { - if (A.isOdd() || B.isOdd()) { - A.iadd(yp); - B.isub(xp); - } - - A.iushrn(1); - B.iushrn(1); - } - } - - for (var j = 0, jm = 1; (y.words[0] & jm) === 0 && j < 26; ++j, jm <<= 1); - if (j > 0) { - y.iushrn(j); - while (j-- > 0) { - if (C.isOdd() || D.isOdd()) { - C.iadd(yp); - D.isub(xp); - } - - C.iushrn(1); - D.iushrn(1); - } - } - - if (x.cmp(y) >= 0) { - x.isub(y); - A.isub(C); - B.isub(D); - } else { - y.isub(x); - C.isub(A); - D.isub(B); - } - } - - return { - a: C, - b: D, - gcd: y.iushln(g) - }; - }; - - // This is reduced incarnation of the binary EEA - // above, designated to invert members of the - // _prime_ fields F(p) at a maximal speed - BN.prototype._invmp = function _invmp (p) { - assert(p.negative === 0); - assert(!p.isZero()); - - var a = this; - var b = p.clone(); - - if (a.negative !== 0) { - a = a.umod(p); - } else { - a = a.clone(); - } - - var x1 = new BN(1); - var x2 = new BN(0); - - var delta = b.clone(); - - while (a.cmpn(1) > 0 && b.cmpn(1) > 0) { - for (var i = 0, im = 1; (a.words[0] & im) === 0 && i < 26; ++i, im <<= 1); - if (i > 0) { - a.iushrn(i); - while (i-- > 0) { - if (x1.isOdd()) { - x1.iadd(delta); - } - - x1.iushrn(1); - } - } - - for (var j = 0, jm = 1; (b.words[0] & jm) === 0 && j < 26; ++j, jm <<= 1); - if (j > 0) { - b.iushrn(j); - while (j-- > 0) { - if (x2.isOdd()) { - x2.iadd(delta); - } - - x2.iushrn(1); - } - } - - if (a.cmp(b) >= 0) { - a.isub(b); - x1.isub(x2); - } else { - b.isub(a); - x2.isub(x1); - } - } - - var res; - if (a.cmpn(1) === 0) { - res = x1; - } else { - res = x2; - } - - if (res.cmpn(0) < 0) { - res.iadd(p); - } - - return res; - }; - - BN.prototype.gcd = function gcd (num) { - if (this.isZero()) return num.abs(); - if (num.isZero()) return this.abs(); - - var a = this.clone(); - var b = num.clone(); - a.negative = 0; - b.negative = 0; - - // Remove common factor of two - for (var shift = 0; a.isEven() && b.isEven(); shift++) { - a.iushrn(1); - b.iushrn(1); - } - - do { - while (a.isEven()) { - a.iushrn(1); - } - while (b.isEven()) { - b.iushrn(1); - } - - var r = a.cmp(b); - if (r < 0) { - // Swap `a` and `b` to make `a` always bigger than `b` - var t = a; - a = b; - b = t; - } else if (r === 0 || b.cmpn(1) === 0) { - break; - } - - a.isub(b); - } while (true); - - return b.iushln(shift); - }; - - // Invert number in the field F(num) - BN.prototype.invm = function invm (num) { - return this.egcd(num).a.umod(num); - }; - - BN.prototype.isEven = function isEven () { - return (this.words[0] & 1) === 0; - }; - - BN.prototype.isOdd = function isOdd () { - return (this.words[0] & 1) === 1; - }; - - // And first word and num - BN.prototype.andln = function andln (num) { - return this.words[0] & num; - }; - - // Increment at the bit position in-line - BN.prototype.bincn = function bincn (bit) { - assert(typeof bit === 'number'); - var r = bit % 26; - var s = (bit - r) / 26; - var q = 1 << r; - - // Fast case: bit is much higher than all existing words - if (this.length <= s) { - this._expand(s + 1); - this.words[s] |= q; - return this; - } - - // Add bit and propagate, if needed - var carry = q; - for (var i = s; carry !== 0 && i < this.length; i++) { - var w = this.words[i] | 0; - w += carry; - carry = w >>> 26; - w &= 0x3ffffff; - this.words[i] = w; - } - if (carry !== 0) { - this.words[i] = carry; - this.length++; - } - return this; - }; - - BN.prototype.isZero = function isZero () { - return this.length === 1 && this.words[0] === 0; - }; - - BN.prototype.cmpn = function cmpn (num) { - var negative = num < 0; - - if (this.negative !== 0 && !negative) return -1; - if (this.negative === 0 && negative) return 1; - - this.strip(); - - var res; - if (this.length > 1) { - res = 1; - } else { - if (negative) { - num = -num; - } - - assert(num <= 0x3ffffff, 'Number is too big'); - - var w = this.words[0] | 0; - res = w === num ? 0 : w < num ? -1 : 1; - } - if (this.negative !== 0) return -res | 0; - return res; - }; - - // Compare two numbers and return: - // 1 - if `this` > `num` - // 0 - if `this` == `num` - // -1 - if `this` < `num` - BN.prototype.cmp = function cmp (num) { - if (this.negative !== 0 && num.negative === 0) return -1; - if (this.negative === 0 && num.negative !== 0) return 1; - - var res = this.ucmp(num); - if (this.negative !== 0) return -res | 0; - return res; - }; - - // Unsigned comparison - BN.prototype.ucmp = function ucmp (num) { - // At this point both numbers have the same sign - if (this.length > num.length) return 1; - if (this.length < num.length) return -1; - - var res = 0; - for (var i = this.length - 1; i >= 0; i--) { - var a = this.words[i] | 0; - var b = num.words[i] | 0; - - if (a === b) continue; - if (a < b) { - res = -1; - } else if (a > b) { - res = 1; - } - break; - } - return res; - }; - - BN.prototype.gtn = function gtn (num) { - return this.cmpn(num) === 1; - }; - - BN.prototype.gt = function gt (num) { - return this.cmp(num) === 1; - }; - - BN.prototype.gten = function gten (num) { - return this.cmpn(num) >= 0; - }; - - BN.prototype.gte = function gte (num) { - return this.cmp(num) >= 0; - }; - - BN.prototype.ltn = function ltn (num) { - return this.cmpn(num) === -1; - }; - - BN.prototype.lt = function lt (num) { - return this.cmp(num) === -1; - }; - - BN.prototype.lten = function lten (num) { - return this.cmpn(num) <= 0; - }; - - BN.prototype.lte = function lte (num) { - return this.cmp(num) <= 0; - }; - - BN.prototype.eqn = function eqn (num) { - return this.cmpn(num) === 0; - }; - - BN.prototype.eq = function eq (num) { - return this.cmp(num) === 0; - }; - - // - // A reduce context, could be using montgomery or something better, depending - // on the `m` itself. - // - BN.red = function red (num) { - return new Red(num); - }; - - BN.prototype.toRed = function toRed (ctx) { - assert(!this.red, 'Already a number in reduction context'); - assert(this.negative === 0, 'red works only with positives'); - return ctx.convertTo(this)._forceRed(ctx); - }; - - BN.prototype.fromRed = function fromRed () { - assert(this.red, 'fromRed works only with numbers in reduction context'); - return this.red.convertFrom(this); - }; - - BN.prototype._forceRed = function _forceRed (ctx) { - this.red = ctx; - return this; - }; - - BN.prototype.forceRed = function forceRed (ctx) { - assert(!this.red, 'Already a number in reduction context'); - return this._forceRed(ctx); - }; - - BN.prototype.redAdd = function redAdd (num) { - assert(this.red, 'redAdd works only with red numbers'); - return this.red.add(this, num); - }; - - BN.prototype.redIAdd = function redIAdd (num) { - assert(this.red, 'redIAdd works only with red numbers'); - return this.red.iadd(this, num); - }; - - BN.prototype.redSub = function redSub (num) { - assert(this.red, 'redSub works only with red numbers'); - return this.red.sub(this, num); - }; - - BN.prototype.redISub = function redISub (num) { - assert(this.red, 'redISub works only with red numbers'); - return this.red.isub(this, num); - }; - - BN.prototype.redShl = function redShl (num) { - assert(this.red, 'redShl works only with red numbers'); - return this.red.shl(this, num); - }; - - BN.prototype.redMul = function redMul (num) { - assert(this.red, 'redMul works only with red numbers'); - this.red._verify2(this, num); - return this.red.mul(this, num); - }; - - BN.prototype.redIMul = function redIMul (num) { - assert(this.red, 'redMul works only with red numbers'); - this.red._verify2(this, num); - return this.red.imul(this, num); - }; - - BN.prototype.redSqr = function redSqr () { - assert(this.red, 'redSqr works only with red numbers'); - this.red._verify1(this); - return this.red.sqr(this); - }; - - BN.prototype.redISqr = function redISqr () { - assert(this.red, 'redISqr works only with red numbers'); - this.red._verify1(this); - return this.red.isqr(this); - }; - - // Square root over p - BN.prototype.redSqrt = function redSqrt () { - assert(this.red, 'redSqrt works only with red numbers'); - this.red._verify1(this); - return this.red.sqrt(this); - }; - - BN.prototype.redInvm = function redInvm () { - assert(this.red, 'redInvm works only with red numbers'); - this.red._verify1(this); - return this.red.invm(this); - }; - - // Return negative clone of `this` % `red modulo` - BN.prototype.redNeg = function redNeg () { - assert(this.red, 'redNeg works only with red numbers'); - this.red._verify1(this); - return this.red.neg(this); - }; - - BN.prototype.redPow = function redPow (num) { - assert(this.red && !num.red, 'redPow(normalNum)'); - this.red._verify1(this); - return this.red.pow(this, num); - }; - - // Prime numbers with efficient reduction - var primes = { - k256: null, - p224: null, - p192: null, - p25519: null - }; - - // Pseudo-Mersenne prime - function MPrime (name, p) { - // P = 2 ^ N - K - this.name = name; - this.p = new BN(p, 16); - this.n = this.p.bitLength(); - this.k = new BN(1).iushln(this.n).isub(this.p); - - this.tmp = this._tmp(); - } - - MPrime.prototype._tmp = function _tmp () { - var tmp = new BN(null); - tmp.words = new Array(Math.ceil(this.n / 13)); - return tmp; - }; - - MPrime.prototype.ireduce = function ireduce (num) { - // Assumes that `num` is less than `P^2` - // num = HI * (2 ^ N - K) + HI * K + LO = HI * K + LO (mod P) - var r = num; - var rlen; - - do { - this.split(r, this.tmp); - r = this.imulK(r); - r = r.iadd(this.tmp); - rlen = r.bitLength(); - } while (rlen > this.n); - - var cmp = rlen < this.n ? -1 : r.ucmp(this.p); - if (cmp === 0) { - r.words[0] = 0; - r.length = 1; - } else if (cmp > 0) { - r.isub(this.p); - } else { - if (r.strip !== undefined) { - // r is BN v4 instance - r.strip(); - } else { - // r is BN v5 instance - r._strip(); - } - } - - return r; - }; - - MPrime.prototype.split = function split (input, out) { - input.iushrn(this.n, 0, out); - }; - - MPrime.prototype.imulK = function imulK (num) { - return num.imul(this.k); - }; - - function K256 () { - MPrime.call( - this, - 'k256', - 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff fffffffe fffffc2f'); - } - inherits(K256, MPrime); - - K256.prototype.split = function split (input, output) { - // 256 = 9 * 26 + 22 - var mask = 0x3fffff; - - var outLen = Math.min(input.length, 9); - for (var i = 0; i < outLen; i++) { - output.words[i] = input.words[i]; - } - output.length = outLen; - - if (input.length <= 9) { - input.words[0] = 0; - input.length = 1; - return; - } - - // Shift by 9 limbs - var prev = input.words[9]; - output.words[output.length++] = prev & mask; - - for (i = 10; i < input.length; i++) { - var next = input.words[i] | 0; - input.words[i - 10] = ((next & mask) << 4) | (prev >>> 22); - prev = next; - } - prev >>>= 22; - input.words[i - 10] = prev; - if (prev === 0 && input.length > 10) { - input.length -= 10; - } else { - input.length -= 9; - } - }; - - K256.prototype.imulK = function imulK (num) { - // K = 0x1000003d1 = [ 0x40, 0x3d1 ] - num.words[num.length] = 0; - num.words[num.length + 1] = 0; - num.length += 2; - - // bounded at: 0x40 * 0x3ffffff + 0x3d0 = 0x100000390 - var lo = 0; - for (var i = 0; i < num.length; i++) { - var w = num.words[i] | 0; - lo += w * 0x3d1; - num.words[i] = lo & 0x3ffffff; - lo = w * 0x40 + ((lo / 0x4000000) | 0); - } - - // Fast length reduction - if (num.words[num.length - 1] === 0) { - num.length--; - if (num.words[num.length - 1] === 0) { - num.length--; - } - } - return num; - }; - - function P224 () { - MPrime.call( - this, - 'p224', - 'ffffffff ffffffff ffffffff ffffffff 00000000 00000000 00000001'); - } - inherits(P224, MPrime); - - function P192 () { - MPrime.call( - this, - 'p192', - 'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff'); - } - inherits(P192, MPrime); - - function P25519 () { - // 2 ^ 255 - 19 - MPrime.call( - this, - '25519', - '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed'); - } - inherits(P25519, MPrime); - - P25519.prototype.imulK = function imulK (num) { - // K = 0x13 - var carry = 0; - for (var i = 0; i < num.length; i++) { - var hi = (num.words[i] | 0) * 0x13 + carry; - var lo = hi & 0x3ffffff; - hi >>>= 26; - - num.words[i] = lo; - carry = hi; - } - if (carry !== 0) { - num.words[num.length++] = carry; - } - return num; - }; - - // Exported mostly for testing purposes, use plain name instead - BN._prime = function prime (name) { - // Cached version of prime - if (primes[name]) return primes[name]; - - var prime; - if (name === 'k256') { - prime = new K256(); - } else if (name === 'p224') { - prime = new P224(); - } else if (name === 'p192') { - prime = new P192(); - } else if (name === 'p25519') { - prime = new P25519(); - } else { - throw new Error('Unknown prime ' + name); - } - primes[name] = prime; - - return prime; - }; - - // - // Base reduction engine - // - function Red (m) { - if (typeof m === 'string') { - var prime = BN._prime(m); - this.m = prime.p; - this.prime = prime; - } else { - assert(m.gtn(1), 'modulus must be greater than 1'); - this.m = m; - this.prime = null; - } - } - - Red.prototype._verify1 = function _verify1 (a) { - assert(a.negative === 0, 'red works only with positives'); - assert(a.red, 'red works only with red numbers'); - }; - - Red.prototype._verify2 = function _verify2 (a, b) { - assert((a.negative | b.negative) === 0, 'red works only with positives'); - assert(a.red && a.red === b.red, - 'red works only with red numbers'); - }; - - Red.prototype.imod = function imod (a) { - if (this.prime) return this.prime.ireduce(a)._forceRed(this); - return a.umod(this.m)._forceRed(this); - }; - - Red.prototype.neg = function neg (a) { - if (a.isZero()) { - return a.clone(); - } - - return this.m.sub(a)._forceRed(this); - }; - - Red.prototype.add = function add (a, b) { - this._verify2(a, b); - - var res = a.add(b); - if (res.cmp(this.m) >= 0) { - res.isub(this.m); - } - return res._forceRed(this); - }; - - Red.prototype.iadd = function iadd (a, b) { - this._verify2(a, b); - - var res = a.iadd(b); - if (res.cmp(this.m) >= 0) { - res.isub(this.m); - } - return res; - }; - - Red.prototype.sub = function sub (a, b) { - this._verify2(a, b); - - var res = a.sub(b); - if (res.cmpn(0) < 0) { - res.iadd(this.m); - } - return res._forceRed(this); - }; - - Red.prototype.isub = function isub (a, b) { - this._verify2(a, b); - - var res = a.isub(b); - if (res.cmpn(0) < 0) { - res.iadd(this.m); - } - return res; - }; - - Red.prototype.shl = function shl (a, num) { - this._verify1(a); - return this.imod(a.ushln(num)); - }; - - Red.prototype.imul = function imul (a, b) { - this._verify2(a, b); - return this.imod(a.imul(b)); - }; - - Red.prototype.mul = function mul (a, b) { - this._verify2(a, b); - return this.imod(a.mul(b)); - }; - - Red.prototype.isqr = function isqr (a) { - return this.imul(a, a.clone()); - }; - - Red.prototype.sqr = function sqr (a) { - return this.mul(a, a); - }; - - Red.prototype.sqrt = function sqrt (a) { - if (a.isZero()) return a.clone(); - - var mod3 = this.m.andln(3); - assert(mod3 % 2 === 1); - - // Fast case - if (mod3 === 3) { - var pow = this.m.add(new BN(1)).iushrn(2); - return this.pow(a, pow); - } - - // Tonelli-Shanks algorithm (Totally unoptimized and slow) - // - // Find Q and S, that Q * 2 ^ S = (P - 1) - var q = this.m.subn(1); - var s = 0; - while (!q.isZero() && q.andln(1) === 0) { - s++; - q.iushrn(1); - } - assert(!q.isZero()); - - var one = new BN(1).toRed(this); - var nOne = one.redNeg(); - - // Find quadratic non-residue - // NOTE: Max is such because of generalized Riemann hypothesis. - var lpow = this.m.subn(1).iushrn(1); - var z = this.m.bitLength(); - z = new BN(2 * z * z).toRed(this); - - while (this.pow(z, lpow).cmp(nOne) !== 0) { - z.redIAdd(nOne); - } - - var c = this.pow(z, q); - var r = this.pow(a, q.addn(1).iushrn(1)); - var t = this.pow(a, q); - var m = s; - while (t.cmp(one) !== 0) { - var tmp = t; - for (var i = 0; tmp.cmp(one) !== 0; i++) { - tmp = tmp.redSqr(); - } - assert(i < m); - var b = this.pow(c, new BN(1).iushln(m - i - 1)); - - r = r.redMul(b); - c = b.redSqr(); - t = t.redMul(c); - m = i; - } - - return r; - }; - - Red.prototype.invm = function invm (a) { - var inv = a._invmp(this.m); - if (inv.negative !== 0) { - inv.negative = 0; - return this.imod(inv).redNeg(); - } else { - return this.imod(inv); - } - }; - - Red.prototype.pow = function pow (a, num) { - if (num.isZero()) return new BN(1).toRed(this); - if (num.cmpn(1) === 0) return a.clone(); - - var windowSize = 4; - var wnd = new Array(1 << windowSize); - wnd[0] = new BN(1).toRed(this); - wnd[1] = a; - for (var i = 2; i < wnd.length; i++) { - wnd[i] = this.mul(wnd[i - 1], a); - } - - var res = wnd[0]; - var current = 0; - var currentLen = 0; - var start = num.bitLength() % 26; - if (start === 0) { - start = 26; - } - - for (i = num.length - 1; i >= 0; i--) { - var word = num.words[i]; - for (var j = start - 1; j >= 0; j--) { - var bit = (word >> j) & 1; - if (res !== wnd[0]) { - res = this.sqr(res); - } - - if (bit === 0 && current === 0) { - currentLen = 0; - continue; - } - - current <<= 1; - current |= bit; - currentLen++; - if (currentLen !== windowSize && (i !== 0 || j !== 0)) continue; - - res = this.mul(res, wnd[current]); - currentLen = 0; - current = 0; - } - start = 26; - } - - return res; - }; - - Red.prototype.convertTo = function convertTo (num) { - var r = num.umod(this.m); - - return r === num ? r.clone() : r; - }; - - Red.prototype.convertFrom = function convertFrom (num) { - var res = num.clone(); - res.red = null; - return res; - }; - - // - // Montgomery method engine - // - - BN.mont = function mont (num) { - return new Mont(num); - }; - - function Mont (m) { - Red.call(this, m); - - this.shift = this.m.bitLength(); - if (this.shift % 26 !== 0) { - this.shift += 26 - (this.shift % 26); - } - - this.r = new BN(1).iushln(this.shift); - this.r2 = this.imod(this.r.sqr()); - this.rinv = this.r._invmp(this.m); - - this.minv = this.rinv.mul(this.r).isubn(1).div(this.m); - this.minv = this.minv.umod(this.r); - this.minv = this.r.sub(this.minv); - } - inherits(Mont, Red); - - Mont.prototype.convertTo = function convertTo (num) { - return this.imod(num.ushln(this.shift)); - }; - - Mont.prototype.convertFrom = function convertFrom (num) { - var r = this.imod(num.mul(this.rinv)); - r.red = null; - return r; - }; - - Mont.prototype.imul = function imul (a, b) { - if (a.isZero() || b.isZero()) { - a.words[0] = 0; - a.length = 1; - return a; - } - - var t = a.imul(b); - var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m); - var u = t.isub(c).iushrn(this.shift); - var res = u; - - if (u.cmp(this.m) >= 0) { - res = u.isub(this.m); - } else if (u.cmpn(0) < 0) { - res = u.iadd(this.m); - } - - return res._forceRed(this); - }; - - Mont.prototype.mul = function mul (a, b) { - if (a.isZero() || b.isZero()) return new BN(0)._forceRed(this); - - var t = a.mul(b); - var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m); - var u = t.isub(c).iushrn(this.shift); - var res = u; - if (u.cmp(this.m) >= 0) { - res = u.isub(this.m); - } else if (u.cmpn(0) < 0) { - res = u.iadd(this.m); - } - - return res._forceRed(this); - }; - - Mont.prototype.invm = function invm (a) { - // (AR)^-1 * R^2 = (A^-1 * R^-1) * R^2 = A^-1 * R - var res = this.imod(a._invmp(this.m).mul(this.r2)); - return res._forceRed(this); - }; -})(typeof module === 'undefined' || module, this); - -},{"buffer":77}],71:[function(require,module,exports){ - -/** - * Expose `Backoff`. - */ - -module.exports = Backoff; - -/** - * Initialize backoff timer with `opts`. - * - * - `min` initial timeout in milliseconds [100] - * - `max` max timeout [10000] - * - `jitter` [0] - * - `factor` [2] - * - * @param {Object} opts - * @api public - */ - -function Backoff(opts) { - opts = opts || {}; - this.ms = opts.min || 100; - this.max = opts.max || 10000; - this.factor = opts.factor || 2; - this.jitter = opts.jitter > 0 && opts.jitter <= 1 ? opts.jitter : 0; - this.attempts = 0; -} - -/** - * Return the backoff duration. - * - * @return {Number} - * @api public - */ - -Backoff.prototype.duration = function(){ - var ms = this.ms * Math.pow(this.factor, this.attempts++); - if (this.jitter) { - var rand = Math.random(); - var deviation = Math.floor(rand * this.jitter * ms); - ms = (Math.floor(rand * 10) & 1) == 0 ? ms - deviation : ms + deviation; - } - return Math.min(ms, this.max) | 0; -}; - -/** - * Reset the number of attempts. - * - * @api public - */ - -Backoff.prototype.reset = function(){ - this.attempts = 0; -}; - -/** - * Set the minimum duration - * - * @api public - */ - -Backoff.prototype.setMin = function(min){ - this.ms = min; -}; - -/** - * Set the maximum duration - * - * @api public - */ - -Backoff.prototype.setMax = function(max){ - this.max = max; -}; - -/** - * Set the jitter - * - * @api public - */ - -Backoff.prototype.setJitter = function(jitter){ - this.jitter = jitter; -}; - - -},{}],72:[function(require,module,exports){ -/* - * base64-arraybuffer - * https://github.com/niklasvh/base64-arraybuffer - * - * Copyright (c) 2012 Niklas von Hertzen - * Licensed under the MIT license. - */ -(function(){ - "use strict"; - - var chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; - - // Use a lookup table to find the index. - var lookup = new Uint8Array(256); - for (var i = 0; i < chars.length; i++) { - lookup[chars.charCodeAt(i)] = i; - } - - exports.encode = function(arraybuffer) { - var bytes = new Uint8Array(arraybuffer), - i, len = bytes.length, base64 = ""; - - for (i = 0; i < len; i+=3) { - base64 += chars[bytes[i] >> 2]; - base64 += chars[((bytes[i] & 3) << 4) | (bytes[i + 1] >> 4)]; - base64 += chars[((bytes[i + 1] & 15) << 2) | (bytes[i + 2] >> 6)]; - base64 += chars[bytes[i + 2] & 63]; - } - - if ((len % 3) === 2) { - base64 = base64.substring(0, base64.length - 1) + "="; - } else if (len % 3 === 1) { - base64 = base64.substring(0, base64.length - 2) + "=="; - } - - return base64; - }; - - exports.decode = function(base64) { - var bufferLength = base64.length * 0.75, - len = base64.length, i, p = 0, - encoded1, encoded2, encoded3, encoded4; - - if (base64[base64.length - 1] === "=") { - bufferLength--; - if (base64[base64.length - 2] === "=") { - bufferLength--; - } - } - - var arraybuffer = new ArrayBuffer(bufferLength), - bytes = new Uint8Array(arraybuffer); - - for (i = 0; i < len; i+=4) { - encoded1 = lookup[base64.charCodeAt(i)]; - encoded2 = lookup[base64.charCodeAt(i+1)]; - encoded3 = lookup[base64.charCodeAt(i+2)]; - encoded4 = lookup[base64.charCodeAt(i+3)]; - - bytes[p++] = (encoded1 << 2) | (encoded2 >> 4); - bytes[p++] = ((encoded2 & 15) << 4) | (encoded3 >> 2); - bytes[p++] = ((encoded3 & 3) << 6) | (encoded4 & 63); - } - - return arraybuffer; - }; -})(); - -},{}],73:[function(require,module,exports){ -'use strict' - -exports.byteLength = byteLength -exports.toByteArray = toByteArray -exports.fromByteArray = fromByteArray - -var lookup = [] -var revLookup = [] -var Arr = typeof Uint8Array !== 'undefined' ? Uint8Array : Array - -var code = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/' -for (var i = 0, len = code.length; i < len; ++i) { - lookup[i] = code[i] - revLookup[code.charCodeAt(i)] = i -} - -// Support decoding URL-safe base64 strings, as Node.js does. -// See: https://en.wikipedia.org/wiki/Base64#URL_applications -revLookup['-'.charCodeAt(0)] = 62 -revLookup['_'.charCodeAt(0)] = 63 - -function getLens (b64) { - var len = b64.length - - if (len % 4 > 0) { - throw new Error('Invalid string. Length must be a multiple of 4') - } - - // Trim off extra bytes after placeholder bytes are found - // See: https://github.com/beatgammit/base64-js/issues/42 - var validLen = b64.indexOf('=') - if (validLen === -1) validLen = len - - var placeHoldersLen = validLen === len - ? 0 - : 4 - (validLen % 4) - - return [validLen, placeHoldersLen] -} - -// base64 is 4/3 + up to two characters of the original data -function byteLength (b64) { - var lens = getLens(b64) - var validLen = lens[0] - var placeHoldersLen = lens[1] - return ((validLen + placeHoldersLen) * 3 / 4) - placeHoldersLen -} - -function _byteLength (b64, validLen, placeHoldersLen) { - return ((validLen + placeHoldersLen) * 3 / 4) - placeHoldersLen -} - -function toByteArray (b64) { - var tmp - var lens = getLens(b64) - var validLen = lens[0] - var placeHoldersLen = lens[1] - - var arr = new Arr(_byteLength(b64, validLen, placeHoldersLen)) - - var curByte = 0 - - // if there are placeholders, only get up to the last complete 4 chars - var len = placeHoldersLen > 0 - ? validLen - 4 - : validLen - - var i - for (i = 0; i < len; i += 4) { - tmp = - (revLookup[b64.charCodeAt(i)] << 18) | - (revLookup[b64.charCodeAt(i + 1)] << 12) | - (revLookup[b64.charCodeAt(i + 2)] << 6) | - revLookup[b64.charCodeAt(i + 3)] - arr[curByte++] = (tmp >> 16) & 0xFF - arr[curByte++] = (tmp >> 8) & 0xFF - arr[curByte++] = tmp & 0xFF - } - - if (placeHoldersLen === 2) { - tmp = - (revLookup[b64.charCodeAt(i)] << 2) | - (revLookup[b64.charCodeAt(i + 1)] >> 4) - arr[curByte++] = tmp & 0xFF - } - - if (placeHoldersLen === 1) { - tmp = - (revLookup[b64.charCodeAt(i)] << 10) | - (revLookup[b64.charCodeAt(i + 1)] << 4) | - (revLookup[b64.charCodeAt(i + 2)] >> 2) - arr[curByte++] = (tmp >> 8) & 0xFF - arr[curByte++] = tmp & 0xFF - } - - return arr -} - -function tripletToBase64 (num) { - return lookup[num >> 18 & 0x3F] + - lookup[num >> 12 & 0x3F] + - lookup[num >> 6 & 0x3F] + - lookup[num & 0x3F] -} - -function encodeChunk (uint8, start, end) { - var tmp - var output = [] - for (var i = start; i < end; i += 3) { - tmp = - ((uint8[i] << 16) & 0xFF0000) + - ((uint8[i + 1] << 8) & 0xFF00) + - (uint8[i + 2] & 0xFF) - output.push(tripletToBase64(tmp)) - } - return output.join('') -} - -function fromByteArray (uint8) { - var tmp - var len = uint8.length - var extraBytes = len % 3 // if we have 1 byte left, pad 2 bytes - var parts = [] - var maxChunkLength = 16383 // must be multiple of 3 - - // go through the array every three bytes, we'll deal with trailing stuff later - for (var i = 0, len2 = len - extraBytes; i < len2; i += maxChunkLength) { - parts.push(encodeChunk( - uint8, i, (i + maxChunkLength) > len2 ? len2 : (i + maxChunkLength) - )) - } - - // pad the end with zeros, but make sure to not forget the extra bytes - if (extraBytes === 1) { - tmp = uint8[len - 1] - parts.push( - lookup[tmp >> 2] + - lookup[(tmp << 4) & 0x3F] + - '==' - ) - } else if (extraBytes === 2) { - tmp = (uint8[len - 2] << 8) + uint8[len - 1] - parts.push( - lookup[tmp >> 10] + - lookup[(tmp >> 4) & 0x3F] + - lookup[(tmp << 2) & 0x3F] + - '=' - ) - } - - return parts.join('') -} - -},{}],74:[function(require,module,exports){ -/** - * Create a blob builder even when vendor prefixes exist - */ - -var BlobBuilder = typeof BlobBuilder !== 'undefined' ? BlobBuilder : - typeof WebKitBlobBuilder !== 'undefined' ? WebKitBlobBuilder : - typeof MSBlobBuilder !== 'undefined' ? MSBlobBuilder : - typeof MozBlobBuilder !== 'undefined' ? MozBlobBuilder : - false; - -/** - * Check if Blob constructor is supported - */ - -var blobSupported = (function() { - try { - var a = new Blob(['hi']); - return a.size === 2; - } catch(e) { - return false; - } -})(); - -/** - * Check if Blob constructor supports ArrayBufferViews - * Fails in Safari 6, so we need to map to ArrayBuffers there. - */ - -var blobSupportsArrayBufferView = blobSupported && (function() { - try { - var b = new Blob([new Uint8Array([1,2])]); - return b.size === 2; - } catch(e) { - return false; - } -})(); - -/** - * Check if BlobBuilder is supported - */ - -var blobBuilderSupported = BlobBuilder - && BlobBuilder.prototype.append - && BlobBuilder.prototype.getBlob; - -/** - * Helper function that maps ArrayBufferViews to ArrayBuffers - * Used by BlobBuilder constructor and old browsers that didn't - * support it in the Blob constructor. - */ - -function mapArrayBufferViews(ary) { - return ary.map(function(chunk) { - if (chunk.buffer instanceof ArrayBuffer) { - var buf = chunk.buffer; - - // if this is a subarray, make a copy so we only - // include the subarray region from the underlying buffer - if (chunk.byteLength !== buf.byteLength) { - var copy = new Uint8Array(chunk.byteLength); - copy.set(new Uint8Array(buf, chunk.byteOffset, chunk.byteLength)); - buf = copy.buffer; - } - - return buf; - } - - return chunk; - }); -} - -function BlobBuilderConstructor(ary, options) { - options = options || {}; - - var bb = new BlobBuilder(); - mapArrayBufferViews(ary).forEach(function(part) { - bb.append(part); - }); - - return (options.type) ? bb.getBlob(options.type) : bb.getBlob(); -}; - -function BlobConstructor(ary, options) { - return new Blob(mapArrayBufferViews(ary), options || {}); -}; - -if (typeof Blob !== 'undefined') { - BlobBuilderConstructor.prototype = Blob.prototype; - BlobConstructor.prototype = Blob.prototype; -} - -module.exports = (function() { - if (blobSupported) { - return blobSupportsArrayBufferView ? Blob : BlobConstructor; - } else if (blobBuilderSupported) { - return BlobBuilderConstructor; - } else { - return undefined; - } -})(); - -},{}],75:[function(require,module,exports){ -(function (module, exports) { - 'use strict'; - - // Utils - function assert (val, msg) { - if (!val) throw new Error(msg || 'Assertion failed'); - } - - // Could use `inherits` module, but don't want to move from single file - // architecture yet. - function inherits (ctor, superCtor) { - ctor.super_ = superCtor; - var TempCtor = function () {}; - TempCtor.prototype = superCtor.prototype; - ctor.prototype = new TempCtor(); - ctor.prototype.constructor = ctor; - } - - // BN - - function BN (number, base, endian) { - if (BN.isBN(number)) { - return number; - } - - this.negative = 0; - this.words = null; - this.length = 0; - - // Reduction context - this.red = null; - - if (number !== null) { - if (base === 'le' || base === 'be') { - endian = base; - base = 10; - } - - this._init(number || 0, base || 10, endian || 'be'); - } - } - if (typeof module === 'object') { - module.exports = BN; - } else { - exports.BN = BN; - } - - BN.BN = BN; - BN.wordSize = 26; - - var Buffer; - try { - Buffer = require('buffer').Buffer; - } catch (e) { - } - - BN.isBN = function isBN (num) { - if (num instanceof BN) { - return true; - } - - return num !== null && typeof num === 'object' && - num.constructor.wordSize === BN.wordSize && Array.isArray(num.words); - }; - - BN.max = function max (left, right) { - if (left.cmp(right) > 0) return left; - return right; - }; - - BN.min = function min (left, right) { - if (left.cmp(right) < 0) return left; - return right; - }; - - BN.prototype._init = function init (number, base, endian) { - if (typeof number === 'number') { - return this._initNumber(number, base, endian); - } - - if (typeof number === 'object') { - return this._initArray(number, base, endian); - } - - if (base === 'hex') { - base = 16; - } - assert(base === (base | 0) && base >= 2 && base <= 36); - - number = number.toString().replace(/\s+/g, ''); - var start = 0; - if (number[0] === '-') { - start++; - } - - if (base === 16) { - this._parseHex(number, start); - } else { - this._parseBase(number, base, start); - } - - if (number[0] === '-') { - this.negative = 1; - } - - this._strip(); - - if (endian !== 'le') return; - - this._initArray(this.toArray(), base, endian); - }; - - BN.prototype._initNumber = function _initNumber (number, base, endian) { - if (number < 0) { - this.negative = 1; - number = -number; - } - if (number < 0x4000000) { - this.words = [number & 0x3ffffff]; - this.length = 1; - } else if (number < 0x10000000000000) { - this.words = [ - number & 0x3ffffff, - (number / 0x4000000) & 0x3ffffff - ]; - this.length = 2; - } else { - assert(number < 0x20000000000000); // 2 ^ 53 (unsafe) - this.words = [ - number & 0x3ffffff, - (number / 0x4000000) & 0x3ffffff, - 1 - ]; - this.length = 3; - } - - if (endian !== 'le') return; - - // Reverse the bytes - this._initArray(this.toArray(), base, endian); - }; - - BN.prototype._initArray = function _initArray (number, base, endian) { - // Perhaps a Uint8Array - assert(typeof number.length === 'number'); - if (number.length <= 0) { - this.words = [0]; - this.length = 1; - return this; - } - - this.length = Math.ceil(number.length / 3); - this.words = new Array(this.length); - for (var i = 0; i < this.length; i++) { - this.words[i] = 0; - } - - var j, w; - var off = 0; - if (endian === 'be') { - for (i = number.length - 1, j = 0; i >= 0; i -= 3) { - w = number[i] | (number[i - 1] << 8) | (number[i - 2] << 16); - this.words[j] |= (w << off) & 0x3ffffff; - this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff; - off += 24; - if (off >= 26) { - off -= 26; - j++; - } - } - } else if (endian === 'le') { - for (i = 0, j = 0; i < number.length; i += 3) { - w = number[i] | (number[i + 1] << 8) | (number[i + 2] << 16); - this.words[j] |= (w << off) & 0x3ffffff; - this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff; - off += 24; - if (off >= 26) { - off -= 26; - j++; - } - } - } - return this._strip(); - }; - - function parseHex (str, start, end) { - var r = 0; - var len = Math.min(str.length, end); - var z = 0; - for (var i = start; i < len; i++) { - var c = str.charCodeAt(i) - 48; - - r <<= 4; - - var b; - - // 'a' - 'f' - if (c >= 49 && c <= 54) { - b = c - 49 + 0xa; - - // 'A' - 'F' - } else if (c >= 17 && c <= 22) { - b = c - 17 + 0xa; - - // '0' - '9' - } else { - b = c; - } - - r |= b; - z |= b; - } - - assert(!(z & 0xf0), 'Invalid character in ' + str); - return r; - } - - BN.prototype._parseHex = function _parseHex (number, start) { - // Create possibly bigger array to ensure that it fits the number - this.length = Math.ceil((number.length - start) / 6); - this.words = new Array(this.length); - for (var i = 0; i < this.length; i++) { - this.words[i] = 0; - } - - var j, w; - // Scan 24-bit chunks and add them to the number - var off = 0; - for (i = number.length - 6, j = 0; i >= start; i -= 6) { - w = parseHex(number, i, i + 6); - this.words[j] |= (w << off) & 0x3ffffff; - // NOTE: `0x3fffff` is intentional here, 26bits max shift + 24bit hex limb - this.words[j + 1] |= w >>> (26 - off) & 0x3fffff; - off += 24; - if (off >= 26) { - off -= 26; - j++; - } - } - if (i + 6 !== start) { - w = parseHex(number, start, i + 6); - this.words[j] |= (w << off) & 0x3ffffff; - this.words[j + 1] |= w >>> (26 - off) & 0x3fffff; - } - this._strip(); - }; - - function parseBase (str, start, end, mul) { - var r = 0; - var b = 0; - var len = Math.min(str.length, end); - for (var i = start; i < len; i++) { - var c = str.charCodeAt(i) - 48; - - r *= mul; - - // 'a' - if (c >= 49) { - b = c - 49 + 0xa; - - // 'A' - } else if (c >= 17) { - b = c - 17 + 0xa; - - // '0' - '9' - } else { - b = c; - } - assert(c >= 0 && b < mul, 'Invalid character'); - r += b; - } - return r; - } - - BN.prototype._parseBase = function _parseBase (number, base, start) { - // Initialize as zero - this.words = [0]; - this.length = 1; - - // Find length of limb in base - for (var limbLen = 0, limbPow = 1; limbPow <= 0x3ffffff; limbPow *= base) { - limbLen++; - } - limbLen--; - limbPow = (limbPow / base) | 0; - - var total = number.length - start; - var mod = total % limbLen; - var end = Math.min(total, total - mod) + start; - - var word = 0; - for (var i = start; i < end; i += limbLen) { - word = parseBase(number, i, i + limbLen, base); - - this.imuln(limbPow); - if (this.words[0] + word < 0x4000000) { - this.words[0] += word; - } else { - this._iaddn(word); - } - } - - if (mod !== 0) { - var pow = 1; - word = parseBase(number, i, number.length, base); - - for (i = 0; i < mod; i++) { - pow *= base; - } - - this.imuln(pow); - if (this.words[0] + word < 0x4000000) { - this.words[0] += word; - } else { - this._iaddn(word); - } - } - }; - - BN.prototype.copy = function copy (dest) { - dest.words = new Array(this.length); - for (var i = 0; i < this.length; i++) { - dest.words[i] = this.words[i]; - } - dest.length = this.length; - dest.negative = this.negative; - dest.red = this.red; - }; - - function move (dest, src) { - dest.words = src.words; - dest.length = src.length; - dest.negative = src.negative; - dest.red = src.red; - } - - BN.prototype._move = function _move (dest) { - move(dest, this); - }; - - BN.prototype.clone = function clone () { - var r = new BN(null); - this.copy(r); - return r; - }; - - BN.prototype._expand = function _expand (size) { - while (this.length < size) { - this.words[this.length++] = 0; - } - return this; - }; - - // Remove leading `0` from `this` - BN.prototype._strip = function strip () { - while (this.length > 1 && this.words[this.length - 1] === 0) { - this.length--; - } - return this._normSign(); - }; - - BN.prototype._normSign = function _normSign () { - // -0 = 0 - if (this.length === 1 && this.words[0] === 0) { - this.negative = 0; - } - return this; - }; - - // Check Symbol.for because not everywhere where Symbol defined - // See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Symbol#Browser_compatibility - if (typeof Symbol !== 'undefined' && typeof Symbol.for === 'function') { - BN.prototype[Symbol.for('nodejs.util.inspect.custom')] = inspect; - } else { - BN.prototype.inspect = inspect; - } - - function inspect () { - return (this.red ? ''; - } - - /* - - var zeros = []; - var groupSizes = []; - var groupBases = []; - - var s = ''; - var i = -1; - while (++i < BN.wordSize) { - zeros[i] = s; - s += '0'; - } - groupSizes[0] = 0; - groupSizes[1] = 0; - groupBases[0] = 0; - groupBases[1] = 0; - var base = 2 - 1; - while (++base < 36 + 1) { - var groupSize = 0; - var groupBase = 1; - while (groupBase < (1 << BN.wordSize) / base) { - groupBase *= base; - groupSize += 1; - } - groupSizes[base] = groupSize; - groupBases[base] = groupBase; - } - - */ - - var zeros = [ - '', - '0', - '00', - '000', - '0000', - '00000', - '000000', - '0000000', - '00000000', - '000000000', - '0000000000', - '00000000000', - '000000000000', - '0000000000000', - '00000000000000', - '000000000000000', - '0000000000000000', - '00000000000000000', - '000000000000000000', - '0000000000000000000', - '00000000000000000000', - '000000000000000000000', - '0000000000000000000000', - '00000000000000000000000', - '000000000000000000000000', - '0000000000000000000000000' - ]; - - var groupSizes = [ - 0, 0, - 25, 16, 12, 11, 10, 9, 8, - 8, 7, 7, 7, 7, 6, 6, - 6, 6, 6, 6, 6, 5, 5, - 5, 5, 5, 5, 5, 5, 5, - 5, 5, 5, 5, 5, 5, 5 - ]; - - var groupBases = [ - 0, 0, - 33554432, 43046721, 16777216, 48828125, 60466176, 40353607, 16777216, - 43046721, 10000000, 19487171, 35831808, 62748517, 7529536, 11390625, - 16777216, 24137569, 34012224, 47045881, 64000000, 4084101, 5153632, - 6436343, 7962624, 9765625, 11881376, 14348907, 17210368, 20511149, - 24300000, 28629151, 33554432, 39135393, 45435424, 52521875, 60466176 - ]; - - BN.prototype.toString = function toString (base, padding) { - base = base || 10; - padding = padding | 0 || 1; - - var out; - if (base === 16 || base === 'hex') { - out = ''; - var off = 0; - var carry = 0; - for (var i = 0; i < this.length; i++) { - var w = this.words[i]; - var word = (((w << off) | carry) & 0xffffff).toString(16); - carry = (w >>> (24 - off)) & 0xffffff; - if (carry !== 0 || i !== this.length - 1) { - out = zeros[6 - word.length] + word + out; - } else { - out = word + out; - } - off += 2; - if (off >= 26) { - off -= 26; - i--; - } - } - if (carry !== 0) { - out = carry.toString(16) + out; - } - while (out.length % padding !== 0) { - out = '0' + out; - } - if (this.negative !== 0) { - out = '-' + out; - } - return out; - } - - if (base === (base | 0) && base >= 2 && base <= 36) { - // var groupSize = Math.floor(BN.wordSize * Math.LN2 / Math.log(base)); - var groupSize = groupSizes[base]; - // var groupBase = Math.pow(base, groupSize); - var groupBase = groupBases[base]; - out = ''; - var c = this.clone(); - c.negative = 0; - while (!c.isZero()) { - var r = c.modrn(groupBase).toString(base); - c = c.idivn(groupBase); - - if (!c.isZero()) { - out = zeros[groupSize - r.length] + r + out; - } else { - out = r + out; - } - } - if (this.isZero()) { - out = '0' + out; - } - while (out.length % padding !== 0) { - out = '0' + out; - } - if (this.negative !== 0) { - out = '-' + out; - } - return out; - } - - assert(false, 'Base should be between 2 and 36'); - }; - - BN.prototype.toNumber = function toNumber () { - var ret = this.words[0]; - if (this.length === 2) { - ret += this.words[1] * 0x4000000; - } else if (this.length === 3 && this.words[2] === 0x01) { - // NOTE: at this stage it is known that the top bit is set - ret += 0x10000000000000 + (this.words[1] * 0x4000000); - } else if (this.length > 2) { - assert(false, 'Number can only safely store up to 53 bits'); - } - return (this.negative !== 0) ? -ret : ret; - }; - - BN.prototype.toJSON = function toJSON () { - return this.toString(16, 2); - }; - - if (Buffer) { - BN.prototype.toBuffer = function toBuffer (endian, length) { - return this.toArrayLike(Buffer, endian, length); - }; - } - - BN.prototype.toArray = function toArray (endian, length) { - return this.toArrayLike(Array, endian, length); - }; - - var allocate = function allocate (ArrayType, size) { - if (ArrayType.allocUnsafe) { - return ArrayType.allocUnsafe(size); - } - return new ArrayType(size); - }; - - BN.prototype.toArrayLike = function toArrayLike (ArrayType, endian, length) { - this._strip(); - - var byteLength = this.byteLength(); - var reqLength = length || Math.max(1, byteLength); - assert(byteLength <= reqLength, 'byte array longer than desired length'); - assert(reqLength > 0, 'Requested array length <= 0'); - - var res = allocate(ArrayType, reqLength); - var postfix = endian === 'le' ? 'LE' : 'BE'; - this['_toArrayLike' + postfix](res, byteLength); - return res; - }; - - BN.prototype._toArrayLikeLE = function _toArrayLikeLE (res, byteLength) { - var position = 0; - var carry = 0; - - for (var i = 0, shift = 0; i < this.length; i++) { - var word = (this.words[i] << shift) | carry; - - res[position++] = word & 0xff; - if (position < res.length) { - res[position++] = (word >> 8) & 0xff; - } - if (position < res.length) { - res[position++] = (word >> 16) & 0xff; - } - - if (shift === 6) { - if (position < res.length) { - res[position++] = (word >> 24) & 0xff; - } - carry = 0; - shift = 0; - } else { - carry = word >>> 24; - shift += 2; - } - } - - if (position < res.length) { - res[position++] = carry; - - while (position < res.length) { - res[position++] = 0; - } - } - }; - - BN.prototype._toArrayLikeBE = function _toArrayLikeBE (res, byteLength) { - var position = res.length - 1; - var carry = 0; - - for (var i = 0, shift = 0; i < this.length; i++) { - var word = (this.words[i] << shift) | carry; - - res[position--] = word & 0xff; - if (position >= 0) { - res[position--] = (word >> 8) & 0xff; - } - if (position >= 0) { - res[position--] = (word >> 16) & 0xff; - } - - if (shift === 6) { - if (position >= 0) { - res[position--] = (word >> 24) & 0xff; - } - carry = 0; - shift = 0; - } else { - carry = word >>> 24; - shift += 2; - } - } - - if (position >= 0) { - res[position--] = carry; - - while (position >= 0) { - res[position--] = 0; - } - } - }; - - if (Math.clz32) { - BN.prototype._countBits = function _countBits (w) { - return 32 - Math.clz32(w); - }; - } else { - BN.prototype._countBits = function _countBits (w) { - var t = w; - var r = 0; - if (t >= 0x1000) { - r += 13; - t >>>= 13; - } - if (t >= 0x40) { - r += 7; - t >>>= 7; - } - if (t >= 0x8) { - r += 4; - t >>>= 4; - } - if (t >= 0x02) { - r += 2; - t >>>= 2; - } - return r + t; - }; - } - - BN.prototype._zeroBits = function _zeroBits (w) { - // Short-cut - if (w === 0) return 26; - - var t = w; - var r = 0; - if ((t & 0x1fff) === 0) { - r += 13; - t >>>= 13; - } - if ((t & 0x7f) === 0) { - r += 7; - t >>>= 7; - } - if ((t & 0xf) === 0) { - r += 4; - t >>>= 4; - } - if ((t & 0x3) === 0) { - r += 2; - t >>>= 2; - } - if ((t & 0x1) === 0) { - r++; - } - return r; - }; - - // Return number of used bits in a BN - BN.prototype.bitLength = function bitLength () { - var w = this.words[this.length - 1]; - var hi = this._countBits(w); - return (this.length - 1) * 26 + hi; - }; - - function toBitArray (num) { - var w = new Array(num.bitLength()); - - for (var bit = 0; bit < w.length; bit++) { - var off = (bit / 26) | 0; - var wbit = bit % 26; - - w[bit] = (num.words[off] >>> wbit) & 0x01; - } - - return w; - } - - // Number of trailing zero bits - BN.prototype.zeroBits = function zeroBits () { - if (this.isZero()) return 0; - - var r = 0; - for (var i = 0; i < this.length; i++) { - var b = this._zeroBits(this.words[i]); - r += b; - if (b !== 26) break; - } - return r; - }; - - BN.prototype.byteLength = function byteLength () { - return Math.ceil(this.bitLength() / 8); - }; - - BN.prototype.toTwos = function toTwos (width) { - if (this.negative !== 0) { - return this.abs().inotn(width).iaddn(1); - } - return this.clone(); - }; - - BN.prototype.fromTwos = function fromTwos (width) { - if (this.testn(width - 1)) { - return this.notn(width).iaddn(1).ineg(); - } - return this.clone(); - }; - - BN.prototype.isNeg = function isNeg () { - return this.negative !== 0; - }; - - // Return negative clone of `this` - BN.prototype.neg = function neg () { - return this.clone().ineg(); - }; - - BN.prototype.ineg = function ineg () { - if (!this.isZero()) { - this.negative ^= 1; - } - - return this; - }; - - // Or `num` with `this` in-place - BN.prototype.iuor = function iuor (num) { - while (this.length < num.length) { - this.words[this.length++] = 0; - } - - for (var i = 0; i < num.length; i++) { - this.words[i] = this.words[i] | num.words[i]; - } - - return this._strip(); - }; - - BN.prototype.ior = function ior (num) { - assert((this.negative | num.negative) === 0); - return this.iuor(num); - }; - - // Or `num` with `this` - BN.prototype.or = function or (num) { - if (this.length > num.length) return this.clone().ior(num); - return num.clone().ior(this); - }; - - BN.prototype.uor = function uor (num) { - if (this.length > num.length) return this.clone().iuor(num); - return num.clone().iuor(this); - }; - - // And `num` with `this` in-place - BN.prototype.iuand = function iuand (num) { - // b = min-length(num, this) - var b; - if (this.length > num.length) { - b = num; - } else { - b = this; - } - - for (var i = 0; i < b.length; i++) { - this.words[i] = this.words[i] & num.words[i]; - } - - this.length = b.length; - - return this._strip(); - }; - - BN.prototype.iand = function iand (num) { - assert((this.negative | num.negative) === 0); - return this.iuand(num); - }; - - // And `num` with `this` - BN.prototype.and = function and (num) { - if (this.length > num.length) return this.clone().iand(num); - return num.clone().iand(this); - }; - - BN.prototype.uand = function uand (num) { - if (this.length > num.length) return this.clone().iuand(num); - return num.clone().iuand(this); - }; - - // Xor `num` with `this` in-place - BN.prototype.iuxor = function iuxor (num) { - // a.length > b.length - var a; - var b; - if (this.length > num.length) { - a = this; - b = num; - } else { - a = num; - b = this; - } - - for (var i = 0; i < b.length; i++) { - this.words[i] = a.words[i] ^ b.words[i]; - } - - if (this !== a) { - for (; i < a.length; i++) { - this.words[i] = a.words[i]; - } - } - - this.length = a.length; - - return this._strip(); - }; - - BN.prototype.ixor = function ixor (num) { - assert((this.negative | num.negative) === 0); - return this.iuxor(num); - }; - - // Xor `num` with `this` - BN.prototype.xor = function xor (num) { - if (this.length > num.length) return this.clone().ixor(num); - return num.clone().ixor(this); - }; - - BN.prototype.uxor = function uxor (num) { - if (this.length > num.length) return this.clone().iuxor(num); - return num.clone().iuxor(this); - }; - - // Not ``this`` with ``width`` bitwidth - BN.prototype.inotn = function inotn (width) { - assert(typeof width === 'number' && width >= 0); - - var bytesNeeded = Math.ceil(width / 26) | 0; - var bitsLeft = width % 26; - - // Extend the buffer with leading zeroes - this._expand(bytesNeeded); - - if (bitsLeft > 0) { - bytesNeeded--; - } - - // Handle complete words - for (var i = 0; i < bytesNeeded; i++) { - this.words[i] = ~this.words[i] & 0x3ffffff; - } - - // Handle the residue - if (bitsLeft > 0) { - this.words[i] = ~this.words[i] & (0x3ffffff >> (26 - bitsLeft)); - } - - // And remove leading zeroes - return this._strip(); - }; - - BN.prototype.notn = function notn (width) { - return this.clone().inotn(width); - }; - - // Set `bit` of `this` - BN.prototype.setn = function setn (bit, val) { - assert(typeof bit === 'number' && bit >= 0); - - var off = (bit / 26) | 0; - var wbit = bit % 26; - - this._expand(off + 1); - - if (val) { - this.words[off] = this.words[off] | (1 << wbit); - } else { - this.words[off] = this.words[off] & ~(1 << wbit); - } - - return this._strip(); - }; - - // Add `num` to `this` in-place - BN.prototype.iadd = function iadd (num) { - var r; - - // negative + positive - if (this.negative !== 0 && num.negative === 0) { - this.negative = 0; - r = this.isub(num); - this.negative ^= 1; - return this._normSign(); - - // positive + negative - } else if (this.negative === 0 && num.negative !== 0) { - num.negative = 0; - r = this.isub(num); - num.negative = 1; - return r._normSign(); - } - - // a.length > b.length - var a, b; - if (this.length > num.length) { - a = this; - b = num; - } else { - a = num; - b = this; - } - - var carry = 0; - for (var i = 0; i < b.length; i++) { - r = (a.words[i] | 0) + (b.words[i] | 0) + carry; - this.words[i] = r & 0x3ffffff; - carry = r >>> 26; - } - for (; carry !== 0 && i < a.length; i++) { - r = (a.words[i] | 0) + carry; - this.words[i] = r & 0x3ffffff; - carry = r >>> 26; - } - - this.length = a.length; - if (carry !== 0) { - this.words[this.length] = carry; - this.length++; - // Copy the rest of the words - } else if (a !== this) { - for (; i < a.length; i++) { - this.words[i] = a.words[i]; - } - } - - return this; - }; - - // Add `num` to `this` - BN.prototype.add = function add (num) { - var res; - if (num.negative !== 0 && this.negative === 0) { - num.negative = 0; - res = this.sub(num); - num.negative ^= 1; - return res; - } else if (num.negative === 0 && this.negative !== 0) { - this.negative = 0; - res = num.sub(this); - this.negative = 1; - return res; - } - - if (this.length > num.length) return this.clone().iadd(num); - - return num.clone().iadd(this); - }; - - // Subtract `num` from `this` in-place - BN.prototype.isub = function isub (num) { - // this - (-num) = this + num - if (num.negative !== 0) { - num.negative = 0; - var r = this.iadd(num); - num.negative = 1; - return r._normSign(); - - // -this - num = -(this + num) - } else if (this.negative !== 0) { - this.negative = 0; - this.iadd(num); - this.negative = 1; - return this._normSign(); - } - - // At this point both numbers are positive - var cmp = this.cmp(num); - - // Optimization - zeroify - if (cmp === 0) { - this.negative = 0; - this.length = 1; - this.words[0] = 0; - return this; - } - - // a > b - var a, b; - if (cmp > 0) { - a = this; - b = num; - } else { - a = num; - b = this; - } - - var carry = 0; - for (var i = 0; i < b.length; i++) { - r = (a.words[i] | 0) - (b.words[i] | 0) + carry; - carry = r >> 26; - this.words[i] = r & 0x3ffffff; - } - for (; carry !== 0 && i < a.length; i++) { - r = (a.words[i] | 0) + carry; - carry = r >> 26; - this.words[i] = r & 0x3ffffff; - } - - // Copy rest of the words - if (carry === 0 && i < a.length && a !== this) { - for (; i < a.length; i++) { - this.words[i] = a.words[i]; - } - } - - this.length = Math.max(this.length, i); - - if (a !== this) { - this.negative = 1; - } - - return this._strip(); - }; - - // Subtract `num` from `this` - BN.prototype.sub = function sub (num) { - return this.clone().isub(num); - }; - - function smallMulTo (self, num, out) { - out.negative = num.negative ^ self.negative; - var len = (self.length + num.length) | 0; - out.length = len; - len = (len - 1) | 0; - - // Peel one iteration (compiler can't do it, because of code complexity) - var a = self.words[0] | 0; - var b = num.words[0] | 0; - var r = a * b; - - var lo = r & 0x3ffffff; - var carry = (r / 0x4000000) | 0; - out.words[0] = lo; - - for (var k = 1; k < len; k++) { - // Sum all words with the same `i + j = k` and accumulate `ncarry`, - // note that ncarry could be >= 0x3ffffff - var ncarry = carry >>> 26; - var rword = carry & 0x3ffffff; - var maxJ = Math.min(k, num.length - 1); - for (var j = Math.max(0, k - self.length + 1); j <= maxJ; j++) { - var i = (k - j) | 0; - a = self.words[i] | 0; - b = num.words[j] | 0; - r = a * b + rword; - ncarry += (r / 0x4000000) | 0; - rword = r & 0x3ffffff; - } - out.words[k] = rword | 0; - carry = ncarry | 0; - } - if (carry !== 0) { - out.words[k] = carry | 0; - } else { - out.length--; - } - - return out._strip(); - } - - // TODO(indutny): it may be reasonable to omit it for users who don't need - // to work with 256-bit numbers, otherwise it gives 20% improvement for 256-bit - // multiplication (like elliptic secp256k1). - var comb10MulTo = function comb10MulTo (self, num, out) { - var a = self.words; - var b = num.words; - var o = out.words; - var c = 0; - var lo; - var mid; - var hi; - var a0 = a[0] | 0; - var al0 = a0 & 0x1fff; - var ah0 = a0 >>> 13; - var a1 = a[1] | 0; - var al1 = a1 & 0x1fff; - var ah1 = a1 >>> 13; - var a2 = a[2] | 0; - var al2 = a2 & 0x1fff; - var ah2 = a2 >>> 13; - var a3 = a[3] | 0; - var al3 = a3 & 0x1fff; - var ah3 = a3 >>> 13; - var a4 = a[4] | 0; - var al4 = a4 & 0x1fff; - var ah4 = a4 >>> 13; - var a5 = a[5] | 0; - var al5 = a5 & 0x1fff; - var ah5 = a5 >>> 13; - var a6 = a[6] | 0; - var al6 = a6 & 0x1fff; - var ah6 = a6 >>> 13; - var a7 = a[7] | 0; - var al7 = a7 & 0x1fff; - var ah7 = a7 >>> 13; - var a8 = a[8] | 0; - var al8 = a8 & 0x1fff; - var ah8 = a8 >>> 13; - var a9 = a[9] | 0; - var al9 = a9 & 0x1fff; - var ah9 = a9 >>> 13; - var b0 = b[0] | 0; - var bl0 = b0 & 0x1fff; - var bh0 = b0 >>> 13; - var b1 = b[1] | 0; - var bl1 = b1 & 0x1fff; - var bh1 = b1 >>> 13; - var b2 = b[2] | 0; - var bl2 = b2 & 0x1fff; - var bh2 = b2 >>> 13; - var b3 = b[3] | 0; - var bl3 = b3 & 0x1fff; - var bh3 = b3 >>> 13; - var b4 = b[4] | 0; - var bl4 = b4 & 0x1fff; - var bh4 = b4 >>> 13; - var b5 = b[5] | 0; - var bl5 = b5 & 0x1fff; - var bh5 = b5 >>> 13; - var b6 = b[6] | 0; - var bl6 = b6 & 0x1fff; - var bh6 = b6 >>> 13; - var b7 = b[7] | 0; - var bl7 = b7 & 0x1fff; - var bh7 = b7 >>> 13; - var b8 = b[8] | 0; - var bl8 = b8 & 0x1fff; - var bh8 = b8 >>> 13; - var b9 = b[9] | 0; - var bl9 = b9 & 0x1fff; - var bh9 = b9 >>> 13; - - out.negative = self.negative ^ num.negative; - out.length = 19; - /* k = 0 */ - lo = Math.imul(al0, bl0); - mid = Math.imul(al0, bh0); - mid = (mid + Math.imul(ah0, bl0)) | 0; - hi = Math.imul(ah0, bh0); - var w0 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w0 >>> 26)) | 0; - w0 &= 0x3ffffff; - /* k = 1 */ - lo = Math.imul(al1, bl0); - mid = Math.imul(al1, bh0); - mid = (mid + Math.imul(ah1, bl0)) | 0; - hi = Math.imul(ah1, bh0); - lo = (lo + Math.imul(al0, bl1)) | 0; - mid = (mid + Math.imul(al0, bh1)) | 0; - mid = (mid + Math.imul(ah0, bl1)) | 0; - hi = (hi + Math.imul(ah0, bh1)) | 0; - var w1 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w1 >>> 26)) | 0; - w1 &= 0x3ffffff; - /* k = 2 */ - lo = Math.imul(al2, bl0); - mid = Math.imul(al2, bh0); - mid = (mid + Math.imul(ah2, bl0)) | 0; - hi = Math.imul(ah2, bh0); - lo = (lo + Math.imul(al1, bl1)) | 0; - mid = (mid + Math.imul(al1, bh1)) | 0; - mid = (mid + Math.imul(ah1, bl1)) | 0; - hi = (hi + Math.imul(ah1, bh1)) | 0; - lo = (lo + Math.imul(al0, bl2)) | 0; - mid = (mid + Math.imul(al0, bh2)) | 0; - mid = (mid + Math.imul(ah0, bl2)) | 0; - hi = (hi + Math.imul(ah0, bh2)) | 0; - var w2 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w2 >>> 26)) | 0; - w2 &= 0x3ffffff; - /* k = 3 */ - lo = Math.imul(al3, bl0); - mid = Math.imul(al3, bh0); - mid = (mid + Math.imul(ah3, bl0)) | 0; - hi = Math.imul(ah3, bh0); - lo = (lo + Math.imul(al2, bl1)) | 0; - mid = (mid + Math.imul(al2, bh1)) | 0; - mid = (mid + Math.imul(ah2, bl1)) | 0; - hi = (hi + Math.imul(ah2, bh1)) | 0; - lo = (lo + Math.imul(al1, bl2)) | 0; - mid = (mid + Math.imul(al1, bh2)) | 0; - mid = (mid + Math.imul(ah1, bl2)) | 0; - hi = (hi + Math.imul(ah1, bh2)) | 0; - lo = (lo + Math.imul(al0, bl3)) | 0; - mid = (mid + Math.imul(al0, bh3)) | 0; - mid = (mid + Math.imul(ah0, bl3)) | 0; - hi = (hi + Math.imul(ah0, bh3)) | 0; - var w3 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w3 >>> 26)) | 0; - w3 &= 0x3ffffff; - /* k = 4 */ - lo = Math.imul(al4, bl0); - mid = Math.imul(al4, bh0); - mid = (mid + Math.imul(ah4, bl0)) | 0; - hi = Math.imul(ah4, bh0); - lo = (lo + Math.imul(al3, bl1)) | 0; - mid = (mid + Math.imul(al3, bh1)) | 0; - mid = (mid + Math.imul(ah3, bl1)) | 0; - hi = (hi + Math.imul(ah3, bh1)) | 0; - lo = (lo + Math.imul(al2, bl2)) | 0; - mid = (mid + Math.imul(al2, bh2)) | 0; - mid = (mid + Math.imul(ah2, bl2)) | 0; - hi = (hi + Math.imul(ah2, bh2)) | 0; - lo = (lo + Math.imul(al1, bl3)) | 0; - mid = (mid + Math.imul(al1, bh3)) | 0; - mid = (mid + Math.imul(ah1, bl3)) | 0; - hi = (hi + Math.imul(ah1, bh3)) | 0; - lo = (lo + Math.imul(al0, bl4)) | 0; - mid = (mid + Math.imul(al0, bh4)) | 0; - mid = (mid + Math.imul(ah0, bl4)) | 0; - hi = (hi + Math.imul(ah0, bh4)) | 0; - var w4 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w4 >>> 26)) | 0; - w4 &= 0x3ffffff; - /* k = 5 */ - lo = Math.imul(al5, bl0); - mid = Math.imul(al5, bh0); - mid = (mid + Math.imul(ah5, bl0)) | 0; - hi = Math.imul(ah5, bh0); - lo = (lo + Math.imul(al4, bl1)) | 0; - mid = (mid + Math.imul(al4, bh1)) | 0; - mid = (mid + Math.imul(ah4, bl1)) | 0; - hi = (hi + Math.imul(ah4, bh1)) | 0; - lo = (lo + Math.imul(al3, bl2)) | 0; - mid = (mid + Math.imul(al3, bh2)) | 0; - mid = (mid + Math.imul(ah3, bl2)) | 0; - hi = (hi + Math.imul(ah3, bh2)) | 0; - lo = (lo + Math.imul(al2, bl3)) | 0; - mid = (mid + Math.imul(al2, bh3)) | 0; - mid = (mid + Math.imul(ah2, bl3)) | 0; - hi = (hi + Math.imul(ah2, bh3)) | 0; - lo = (lo + Math.imul(al1, bl4)) | 0; - mid = (mid + Math.imul(al1, bh4)) | 0; - mid = (mid + Math.imul(ah1, bl4)) | 0; - hi = (hi + Math.imul(ah1, bh4)) | 0; - lo = (lo + Math.imul(al0, bl5)) | 0; - mid = (mid + Math.imul(al0, bh5)) | 0; - mid = (mid + Math.imul(ah0, bl5)) | 0; - hi = (hi + Math.imul(ah0, bh5)) | 0; - var w5 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w5 >>> 26)) | 0; - w5 &= 0x3ffffff; - /* k = 6 */ - lo = Math.imul(al6, bl0); - mid = Math.imul(al6, bh0); - mid = (mid + Math.imul(ah6, bl0)) | 0; - hi = Math.imul(ah6, bh0); - lo = (lo + Math.imul(al5, bl1)) | 0; - mid = (mid + Math.imul(al5, bh1)) | 0; - mid = (mid + Math.imul(ah5, bl1)) | 0; - hi = (hi + Math.imul(ah5, bh1)) | 0; - lo = (lo + Math.imul(al4, bl2)) | 0; - mid = (mid + Math.imul(al4, bh2)) | 0; - mid = (mid + Math.imul(ah4, bl2)) | 0; - hi = (hi + Math.imul(ah4, bh2)) | 0; - lo = (lo + Math.imul(al3, bl3)) | 0; - mid = (mid + Math.imul(al3, bh3)) | 0; - mid = (mid + Math.imul(ah3, bl3)) | 0; - hi = (hi + Math.imul(ah3, bh3)) | 0; - lo = (lo + Math.imul(al2, bl4)) | 0; - mid = (mid + Math.imul(al2, bh4)) | 0; - mid = (mid + Math.imul(ah2, bl4)) | 0; - hi = (hi + Math.imul(ah2, bh4)) | 0; - lo = (lo + Math.imul(al1, bl5)) | 0; - mid = (mid + Math.imul(al1, bh5)) | 0; - mid = (mid + Math.imul(ah1, bl5)) | 0; - hi = (hi + Math.imul(ah1, bh5)) | 0; - lo = (lo + Math.imul(al0, bl6)) | 0; - mid = (mid + Math.imul(al0, bh6)) | 0; - mid = (mid + Math.imul(ah0, bl6)) | 0; - hi = (hi + Math.imul(ah0, bh6)) | 0; - var w6 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w6 >>> 26)) | 0; - w6 &= 0x3ffffff; - /* k = 7 */ - lo = Math.imul(al7, bl0); - mid = Math.imul(al7, bh0); - mid = (mid + Math.imul(ah7, bl0)) | 0; - hi = Math.imul(ah7, bh0); - lo = (lo + Math.imul(al6, bl1)) | 0; - mid = (mid + Math.imul(al6, bh1)) | 0; - mid = (mid + Math.imul(ah6, bl1)) | 0; - hi = (hi + Math.imul(ah6, bh1)) | 0; - lo = (lo + Math.imul(al5, bl2)) | 0; - mid = (mid + Math.imul(al5, bh2)) | 0; - mid = (mid + Math.imul(ah5, bl2)) | 0; - hi = (hi + Math.imul(ah5, bh2)) | 0; - lo = (lo + Math.imul(al4, bl3)) | 0; - mid = (mid + Math.imul(al4, bh3)) | 0; - mid = (mid + Math.imul(ah4, bl3)) | 0; - hi = (hi + Math.imul(ah4, bh3)) | 0; - lo = (lo + Math.imul(al3, bl4)) | 0; - mid = (mid + Math.imul(al3, bh4)) | 0; - mid = (mid + Math.imul(ah3, bl4)) | 0; - hi = (hi + Math.imul(ah3, bh4)) | 0; - lo = (lo + Math.imul(al2, bl5)) | 0; - mid = (mid + Math.imul(al2, bh5)) | 0; - mid = (mid + Math.imul(ah2, bl5)) | 0; - hi = (hi + Math.imul(ah2, bh5)) | 0; - lo = (lo + Math.imul(al1, bl6)) | 0; - mid = (mid + Math.imul(al1, bh6)) | 0; - mid = (mid + Math.imul(ah1, bl6)) | 0; - hi = (hi + Math.imul(ah1, bh6)) | 0; - lo = (lo + Math.imul(al0, bl7)) | 0; - mid = (mid + Math.imul(al0, bh7)) | 0; - mid = (mid + Math.imul(ah0, bl7)) | 0; - hi = (hi + Math.imul(ah0, bh7)) | 0; - var w7 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w7 >>> 26)) | 0; - w7 &= 0x3ffffff; - /* k = 8 */ - lo = Math.imul(al8, bl0); - mid = Math.imul(al8, bh0); - mid = (mid + Math.imul(ah8, bl0)) | 0; - hi = Math.imul(ah8, bh0); - lo = (lo + Math.imul(al7, bl1)) | 0; - mid = (mid + Math.imul(al7, bh1)) | 0; - mid = (mid + Math.imul(ah7, bl1)) | 0; - hi = (hi + Math.imul(ah7, bh1)) | 0; - lo = (lo + Math.imul(al6, bl2)) | 0; - mid = (mid + Math.imul(al6, bh2)) | 0; - mid = (mid + Math.imul(ah6, bl2)) | 0; - hi = (hi + Math.imul(ah6, bh2)) | 0; - lo = (lo + Math.imul(al5, bl3)) | 0; - mid = (mid + Math.imul(al5, bh3)) | 0; - mid = (mid + Math.imul(ah5, bl3)) | 0; - hi = (hi + Math.imul(ah5, bh3)) | 0; - lo = (lo + Math.imul(al4, bl4)) | 0; - mid = (mid + Math.imul(al4, bh4)) | 0; - mid = (mid + Math.imul(ah4, bl4)) | 0; - hi = (hi + Math.imul(ah4, bh4)) | 0; - lo = (lo + Math.imul(al3, bl5)) | 0; - mid = (mid + Math.imul(al3, bh5)) | 0; - mid = (mid + Math.imul(ah3, bl5)) | 0; - hi = (hi + Math.imul(ah3, bh5)) | 0; - lo = (lo + Math.imul(al2, bl6)) | 0; - mid = (mid + Math.imul(al2, bh6)) | 0; - mid = (mid + Math.imul(ah2, bl6)) | 0; - hi = (hi + Math.imul(ah2, bh6)) | 0; - lo = (lo + Math.imul(al1, bl7)) | 0; - mid = (mid + Math.imul(al1, bh7)) | 0; - mid = (mid + Math.imul(ah1, bl7)) | 0; - hi = (hi + Math.imul(ah1, bh7)) | 0; - lo = (lo + Math.imul(al0, bl8)) | 0; - mid = (mid + Math.imul(al0, bh8)) | 0; - mid = (mid + Math.imul(ah0, bl8)) | 0; - hi = (hi + Math.imul(ah0, bh8)) | 0; - var w8 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w8 >>> 26)) | 0; - w8 &= 0x3ffffff; - /* k = 9 */ - lo = Math.imul(al9, bl0); - mid = Math.imul(al9, bh0); - mid = (mid + Math.imul(ah9, bl0)) | 0; - hi = Math.imul(ah9, bh0); - lo = (lo + Math.imul(al8, bl1)) | 0; - mid = (mid + Math.imul(al8, bh1)) | 0; - mid = (mid + Math.imul(ah8, bl1)) | 0; - hi = (hi + Math.imul(ah8, bh1)) | 0; - lo = (lo + Math.imul(al7, bl2)) | 0; - mid = (mid + Math.imul(al7, bh2)) | 0; - mid = (mid + Math.imul(ah7, bl2)) | 0; - hi = (hi + Math.imul(ah7, bh2)) | 0; - lo = (lo + Math.imul(al6, bl3)) | 0; - mid = (mid + Math.imul(al6, bh3)) | 0; - mid = (mid + Math.imul(ah6, bl3)) | 0; - hi = (hi + Math.imul(ah6, bh3)) | 0; - lo = (lo + Math.imul(al5, bl4)) | 0; - mid = (mid + Math.imul(al5, bh4)) | 0; - mid = (mid + Math.imul(ah5, bl4)) | 0; - hi = (hi + Math.imul(ah5, bh4)) | 0; - lo = (lo + Math.imul(al4, bl5)) | 0; - mid = (mid + Math.imul(al4, bh5)) | 0; - mid = (mid + Math.imul(ah4, bl5)) | 0; - hi = (hi + Math.imul(ah4, bh5)) | 0; - lo = (lo + Math.imul(al3, bl6)) | 0; - mid = (mid + Math.imul(al3, bh6)) | 0; - mid = (mid + Math.imul(ah3, bl6)) | 0; - hi = (hi + Math.imul(ah3, bh6)) | 0; - lo = (lo + Math.imul(al2, bl7)) | 0; - mid = (mid + Math.imul(al2, bh7)) | 0; - mid = (mid + Math.imul(ah2, bl7)) | 0; - hi = (hi + Math.imul(ah2, bh7)) | 0; - lo = (lo + Math.imul(al1, bl8)) | 0; - mid = (mid + Math.imul(al1, bh8)) | 0; - mid = (mid + Math.imul(ah1, bl8)) | 0; - hi = (hi + Math.imul(ah1, bh8)) | 0; - lo = (lo + Math.imul(al0, bl9)) | 0; - mid = (mid + Math.imul(al0, bh9)) | 0; - mid = (mid + Math.imul(ah0, bl9)) | 0; - hi = (hi + Math.imul(ah0, bh9)) | 0; - var w9 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w9 >>> 26)) | 0; - w9 &= 0x3ffffff; - /* k = 10 */ - lo = Math.imul(al9, bl1); - mid = Math.imul(al9, bh1); - mid = (mid + Math.imul(ah9, bl1)) | 0; - hi = Math.imul(ah9, bh1); - lo = (lo + Math.imul(al8, bl2)) | 0; - mid = (mid + Math.imul(al8, bh2)) | 0; - mid = (mid + Math.imul(ah8, bl2)) | 0; - hi = (hi + Math.imul(ah8, bh2)) | 0; - lo = (lo + Math.imul(al7, bl3)) | 0; - mid = (mid + Math.imul(al7, bh3)) | 0; - mid = (mid + Math.imul(ah7, bl3)) | 0; - hi = (hi + Math.imul(ah7, bh3)) | 0; - lo = (lo + Math.imul(al6, bl4)) | 0; - mid = (mid + Math.imul(al6, bh4)) | 0; - mid = (mid + Math.imul(ah6, bl4)) | 0; - hi = (hi + Math.imul(ah6, bh4)) | 0; - lo = (lo + Math.imul(al5, bl5)) | 0; - mid = (mid + Math.imul(al5, bh5)) | 0; - mid = (mid + Math.imul(ah5, bl5)) | 0; - hi = (hi + Math.imul(ah5, bh5)) | 0; - lo = (lo + Math.imul(al4, bl6)) | 0; - mid = (mid + Math.imul(al4, bh6)) | 0; - mid = (mid + Math.imul(ah4, bl6)) | 0; - hi = (hi + Math.imul(ah4, bh6)) | 0; - lo = (lo + Math.imul(al3, bl7)) | 0; - mid = (mid + Math.imul(al3, bh7)) | 0; - mid = (mid + Math.imul(ah3, bl7)) | 0; - hi = (hi + Math.imul(ah3, bh7)) | 0; - lo = (lo + Math.imul(al2, bl8)) | 0; - mid = (mid + Math.imul(al2, bh8)) | 0; - mid = (mid + Math.imul(ah2, bl8)) | 0; - hi = (hi + Math.imul(ah2, bh8)) | 0; - lo = (lo + Math.imul(al1, bl9)) | 0; - mid = (mid + Math.imul(al1, bh9)) | 0; - mid = (mid + Math.imul(ah1, bl9)) | 0; - hi = (hi + Math.imul(ah1, bh9)) | 0; - var w10 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w10 >>> 26)) | 0; - w10 &= 0x3ffffff; - /* k = 11 */ - lo = Math.imul(al9, bl2); - mid = Math.imul(al9, bh2); - mid = (mid + Math.imul(ah9, bl2)) | 0; - hi = Math.imul(ah9, bh2); - lo = (lo + Math.imul(al8, bl3)) | 0; - mid = (mid + Math.imul(al8, bh3)) | 0; - mid = (mid + Math.imul(ah8, bl3)) | 0; - hi = (hi + Math.imul(ah8, bh3)) | 0; - lo = (lo + Math.imul(al7, bl4)) | 0; - mid = (mid + Math.imul(al7, bh4)) | 0; - mid = (mid + Math.imul(ah7, bl4)) | 0; - hi = (hi + Math.imul(ah7, bh4)) | 0; - lo = (lo + Math.imul(al6, bl5)) | 0; - mid = (mid + Math.imul(al6, bh5)) | 0; - mid = (mid + Math.imul(ah6, bl5)) | 0; - hi = (hi + Math.imul(ah6, bh5)) | 0; - lo = (lo + Math.imul(al5, bl6)) | 0; - mid = (mid + Math.imul(al5, bh6)) | 0; - mid = (mid + Math.imul(ah5, bl6)) | 0; - hi = (hi + Math.imul(ah5, bh6)) | 0; - lo = (lo + Math.imul(al4, bl7)) | 0; - mid = (mid + Math.imul(al4, bh7)) | 0; - mid = (mid + Math.imul(ah4, bl7)) | 0; - hi = (hi + Math.imul(ah4, bh7)) | 0; - lo = (lo + Math.imul(al3, bl8)) | 0; - mid = (mid + Math.imul(al3, bh8)) | 0; - mid = (mid + Math.imul(ah3, bl8)) | 0; - hi = (hi + Math.imul(ah3, bh8)) | 0; - lo = (lo + Math.imul(al2, bl9)) | 0; - mid = (mid + Math.imul(al2, bh9)) | 0; - mid = (mid + Math.imul(ah2, bl9)) | 0; - hi = (hi + Math.imul(ah2, bh9)) | 0; - var w11 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w11 >>> 26)) | 0; - w11 &= 0x3ffffff; - /* k = 12 */ - lo = Math.imul(al9, bl3); - mid = Math.imul(al9, bh3); - mid = (mid + Math.imul(ah9, bl3)) | 0; - hi = Math.imul(ah9, bh3); - lo = (lo + Math.imul(al8, bl4)) | 0; - mid = (mid + Math.imul(al8, bh4)) | 0; - mid = (mid + Math.imul(ah8, bl4)) | 0; - hi = (hi + Math.imul(ah8, bh4)) | 0; - lo = (lo + Math.imul(al7, bl5)) | 0; - mid = (mid + Math.imul(al7, bh5)) | 0; - mid = (mid + Math.imul(ah7, bl5)) | 0; - hi = (hi + Math.imul(ah7, bh5)) | 0; - lo = (lo + Math.imul(al6, bl6)) | 0; - mid = (mid + Math.imul(al6, bh6)) | 0; - mid = (mid + Math.imul(ah6, bl6)) | 0; - hi = (hi + Math.imul(ah6, bh6)) | 0; - lo = (lo + Math.imul(al5, bl7)) | 0; - mid = (mid + Math.imul(al5, bh7)) | 0; - mid = (mid + Math.imul(ah5, bl7)) | 0; - hi = (hi + Math.imul(ah5, bh7)) | 0; - lo = (lo + Math.imul(al4, bl8)) | 0; - mid = (mid + Math.imul(al4, bh8)) | 0; - mid = (mid + Math.imul(ah4, bl8)) | 0; - hi = (hi + Math.imul(ah4, bh8)) | 0; - lo = (lo + Math.imul(al3, bl9)) | 0; - mid = (mid + Math.imul(al3, bh9)) | 0; - mid = (mid + Math.imul(ah3, bl9)) | 0; - hi = (hi + Math.imul(ah3, bh9)) | 0; - var w12 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w12 >>> 26)) | 0; - w12 &= 0x3ffffff; - /* k = 13 */ - lo = Math.imul(al9, bl4); - mid = Math.imul(al9, bh4); - mid = (mid + Math.imul(ah9, bl4)) | 0; - hi = Math.imul(ah9, bh4); - lo = (lo + Math.imul(al8, bl5)) | 0; - mid = (mid + Math.imul(al8, bh5)) | 0; - mid = (mid + Math.imul(ah8, bl5)) | 0; - hi = (hi + Math.imul(ah8, bh5)) | 0; - lo = (lo + Math.imul(al7, bl6)) | 0; - mid = (mid + Math.imul(al7, bh6)) | 0; - mid = (mid + Math.imul(ah7, bl6)) | 0; - hi = (hi + Math.imul(ah7, bh6)) | 0; - lo = (lo + Math.imul(al6, bl7)) | 0; - mid = (mid + Math.imul(al6, bh7)) | 0; - mid = (mid + Math.imul(ah6, bl7)) | 0; - hi = (hi + Math.imul(ah6, bh7)) | 0; - lo = (lo + Math.imul(al5, bl8)) | 0; - mid = (mid + Math.imul(al5, bh8)) | 0; - mid = (mid + Math.imul(ah5, bl8)) | 0; - hi = (hi + Math.imul(ah5, bh8)) | 0; - lo = (lo + Math.imul(al4, bl9)) | 0; - mid = (mid + Math.imul(al4, bh9)) | 0; - mid = (mid + Math.imul(ah4, bl9)) | 0; - hi = (hi + Math.imul(ah4, bh9)) | 0; - var w13 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w13 >>> 26)) | 0; - w13 &= 0x3ffffff; - /* k = 14 */ - lo = Math.imul(al9, bl5); - mid = Math.imul(al9, bh5); - mid = (mid + Math.imul(ah9, bl5)) | 0; - hi = Math.imul(ah9, bh5); - lo = (lo + Math.imul(al8, bl6)) | 0; - mid = (mid + Math.imul(al8, bh6)) | 0; - mid = (mid + Math.imul(ah8, bl6)) | 0; - hi = (hi + Math.imul(ah8, bh6)) | 0; - lo = (lo + Math.imul(al7, bl7)) | 0; - mid = (mid + Math.imul(al7, bh7)) | 0; - mid = (mid + Math.imul(ah7, bl7)) | 0; - hi = (hi + Math.imul(ah7, bh7)) | 0; - lo = (lo + Math.imul(al6, bl8)) | 0; - mid = (mid + Math.imul(al6, bh8)) | 0; - mid = (mid + Math.imul(ah6, bl8)) | 0; - hi = (hi + Math.imul(ah6, bh8)) | 0; - lo = (lo + Math.imul(al5, bl9)) | 0; - mid = (mid + Math.imul(al5, bh9)) | 0; - mid = (mid + Math.imul(ah5, bl9)) | 0; - hi = (hi + Math.imul(ah5, bh9)) | 0; - var w14 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w14 >>> 26)) | 0; - w14 &= 0x3ffffff; - /* k = 15 */ - lo = Math.imul(al9, bl6); - mid = Math.imul(al9, bh6); - mid = (mid + Math.imul(ah9, bl6)) | 0; - hi = Math.imul(ah9, bh6); - lo = (lo + Math.imul(al8, bl7)) | 0; - mid = (mid + Math.imul(al8, bh7)) | 0; - mid = (mid + Math.imul(ah8, bl7)) | 0; - hi = (hi + Math.imul(ah8, bh7)) | 0; - lo = (lo + Math.imul(al7, bl8)) | 0; - mid = (mid + Math.imul(al7, bh8)) | 0; - mid = (mid + Math.imul(ah7, bl8)) | 0; - hi = (hi + Math.imul(ah7, bh8)) | 0; - lo = (lo + Math.imul(al6, bl9)) | 0; - mid = (mid + Math.imul(al6, bh9)) | 0; - mid = (mid + Math.imul(ah6, bl9)) | 0; - hi = (hi + Math.imul(ah6, bh9)) | 0; - var w15 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w15 >>> 26)) | 0; - w15 &= 0x3ffffff; - /* k = 16 */ - lo = Math.imul(al9, bl7); - mid = Math.imul(al9, bh7); - mid = (mid + Math.imul(ah9, bl7)) | 0; - hi = Math.imul(ah9, bh7); - lo = (lo + Math.imul(al8, bl8)) | 0; - mid = (mid + Math.imul(al8, bh8)) | 0; - mid = (mid + Math.imul(ah8, bl8)) | 0; - hi = (hi + Math.imul(ah8, bh8)) | 0; - lo = (lo + Math.imul(al7, bl9)) | 0; - mid = (mid + Math.imul(al7, bh9)) | 0; - mid = (mid + Math.imul(ah7, bl9)) | 0; - hi = (hi + Math.imul(ah7, bh9)) | 0; - var w16 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w16 >>> 26)) | 0; - w16 &= 0x3ffffff; - /* k = 17 */ - lo = Math.imul(al9, bl8); - mid = Math.imul(al9, bh8); - mid = (mid + Math.imul(ah9, bl8)) | 0; - hi = Math.imul(ah9, bh8); - lo = (lo + Math.imul(al8, bl9)) | 0; - mid = (mid + Math.imul(al8, bh9)) | 0; - mid = (mid + Math.imul(ah8, bl9)) | 0; - hi = (hi + Math.imul(ah8, bh9)) | 0; - var w17 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w17 >>> 26)) | 0; - w17 &= 0x3ffffff; - /* k = 18 */ - lo = Math.imul(al9, bl9); - mid = Math.imul(al9, bh9); - mid = (mid + Math.imul(ah9, bl9)) | 0; - hi = Math.imul(ah9, bh9); - var w18 = (((c + lo) | 0) + ((mid & 0x1fff) << 13)) | 0; - c = (((hi + (mid >>> 13)) | 0) + (w18 >>> 26)) | 0; - w18 &= 0x3ffffff; - o[0] = w0; - o[1] = w1; - o[2] = w2; - o[3] = w3; - o[4] = w4; - o[5] = w5; - o[6] = w6; - o[7] = w7; - o[8] = w8; - o[9] = w9; - o[10] = w10; - o[11] = w11; - o[12] = w12; - o[13] = w13; - o[14] = w14; - o[15] = w15; - o[16] = w16; - o[17] = w17; - o[18] = w18; - if (c !== 0) { - o[19] = c; - out.length++; - } - return out; - }; - - // Polyfill comb - if (!Math.imul) { - comb10MulTo = smallMulTo; - } - - function bigMulTo (self, num, out) { - out.negative = num.negative ^ self.negative; - out.length = self.length + num.length; - - var carry = 0; - var hncarry = 0; - for (var k = 0; k < out.length - 1; k++) { - // Sum all words with the same `i + j = k` and accumulate `ncarry`, - // note that ncarry could be >= 0x3ffffff - var ncarry = hncarry; - hncarry = 0; - var rword = carry & 0x3ffffff; - var maxJ = Math.min(k, num.length - 1); - for (var j = Math.max(0, k - self.length + 1); j <= maxJ; j++) { - var i = k - j; - var a = self.words[i] | 0; - var b = num.words[j] | 0; - var r = a * b; - - var lo = r & 0x3ffffff; - ncarry = (ncarry + ((r / 0x4000000) | 0)) | 0; - lo = (lo + rword) | 0; - rword = lo & 0x3ffffff; - ncarry = (ncarry + (lo >>> 26)) | 0; - - hncarry += ncarry >>> 26; - ncarry &= 0x3ffffff; - } - out.words[k] = rword; - carry = ncarry; - ncarry = hncarry; - } - if (carry !== 0) { - out.words[k] = carry; - } else { - out.length--; - } - - return out._strip(); - } - - function jumboMulTo (self, num, out) { - // Temporary disable, see https://github.com/indutny/bn.js/issues/211 - // var fftm = new FFTM(); - // return fftm.mulp(self, num, out); - return bigMulTo(self, num, out); - } - - BN.prototype.mulTo = function mulTo (num, out) { - var res; - var len = this.length + num.length; - if (this.length === 10 && num.length === 10) { - res = comb10MulTo(this, num, out); - } else if (len < 63) { - res = smallMulTo(this, num, out); - } else if (len < 1024) { - res = bigMulTo(this, num, out); - } else { - res = jumboMulTo(this, num, out); - } - - return res; - }; - - // Cooley-Tukey algorithm for FFT - // slightly revisited to rely on looping instead of recursion - - function FFTM (x, y) { - this.x = x; - this.y = y; - } - - FFTM.prototype.makeRBT = function makeRBT (N) { - var t = new Array(N); - var l = BN.prototype._countBits(N) - 1; - for (var i = 0; i < N; i++) { - t[i] = this.revBin(i, l, N); - } - - return t; - }; - - // Returns binary-reversed representation of `x` - FFTM.prototype.revBin = function revBin (x, l, N) { - if (x === 0 || x === N - 1) return x; - - var rb = 0; - for (var i = 0; i < l; i++) { - rb |= (x & 1) << (l - i - 1); - x >>= 1; - } - - return rb; - }; - - // Performs "tweedling" phase, therefore 'emulating' - // behaviour of the recursive algorithm - FFTM.prototype.permute = function permute (rbt, rws, iws, rtws, itws, N) { - for (var i = 0; i < N; i++) { - rtws[i] = rws[rbt[i]]; - itws[i] = iws[rbt[i]]; - } - }; - - FFTM.prototype.transform = function transform (rws, iws, rtws, itws, N, rbt) { - this.permute(rbt, rws, iws, rtws, itws, N); - - for (var s = 1; s < N; s <<= 1) { - var l = s << 1; - - var rtwdf = Math.cos(2 * Math.PI / l); - var itwdf = Math.sin(2 * Math.PI / l); - - for (var p = 0; p < N; p += l) { - var rtwdf_ = rtwdf; - var itwdf_ = itwdf; - - for (var j = 0; j < s; j++) { - var re = rtws[p + j]; - var ie = itws[p + j]; - - var ro = rtws[p + j + s]; - var io = itws[p + j + s]; - - var rx = rtwdf_ * ro - itwdf_ * io; - - io = rtwdf_ * io + itwdf_ * ro; - ro = rx; - - rtws[p + j] = re + ro; - itws[p + j] = ie + io; - - rtws[p + j + s] = re - ro; - itws[p + j + s] = ie - io; - - /* jshint maxdepth : false */ - if (j !== l) { - rx = rtwdf * rtwdf_ - itwdf * itwdf_; - - itwdf_ = rtwdf * itwdf_ + itwdf * rtwdf_; - rtwdf_ = rx; - } - } - } - } - }; - - FFTM.prototype.guessLen13b = function guessLen13b (n, m) { - var N = Math.max(m, n) | 1; - var odd = N & 1; - var i = 0; - for (N = N / 2 | 0; N; N = N >>> 1) { - i++; - } - - return 1 << i + 1 + odd; - }; - - FFTM.prototype.conjugate = function conjugate (rws, iws, N) { - if (N <= 1) return; - - for (var i = 0; i < N / 2; i++) { - var t = rws[i]; - - rws[i] = rws[N - i - 1]; - rws[N - i - 1] = t; - - t = iws[i]; - - iws[i] = -iws[N - i - 1]; - iws[N - i - 1] = -t; - } - }; - - FFTM.prototype.normalize13b = function normalize13b (ws, N) { - var carry = 0; - for (var i = 0; i < N / 2; i++) { - var w = Math.round(ws[2 * i + 1] / N) * 0x2000 + - Math.round(ws[2 * i] / N) + - carry; - - ws[i] = w & 0x3ffffff; - - if (w < 0x4000000) { - carry = 0; - } else { - carry = w / 0x4000000 | 0; - } - } - - return ws; - }; - - FFTM.prototype.convert13b = function convert13b (ws, len, rws, N) { - var carry = 0; - for (var i = 0; i < len; i++) { - carry = carry + (ws[i] | 0); - - rws[2 * i] = carry & 0x1fff; carry = carry >>> 13; - rws[2 * i + 1] = carry & 0x1fff; carry = carry >>> 13; - } - - // Pad with zeroes - for (i = 2 * len; i < N; ++i) { - rws[i] = 0; - } - - assert(carry === 0); - assert((carry & ~0x1fff) === 0); - }; - - FFTM.prototype.stub = function stub (N) { - var ph = new Array(N); - for (var i = 0; i < N; i++) { - ph[i] = 0; - } - - return ph; - }; - - FFTM.prototype.mulp = function mulp (x, y, out) { - var N = 2 * this.guessLen13b(x.length, y.length); - - var rbt = this.makeRBT(N); - - var _ = this.stub(N); - - var rws = new Array(N); - var rwst = new Array(N); - var iwst = new Array(N); - - var nrws = new Array(N); - var nrwst = new Array(N); - var niwst = new Array(N); - - var rmws = out.words; - rmws.length = N; - - this.convert13b(x.words, x.length, rws, N); - this.convert13b(y.words, y.length, nrws, N); - - this.transform(rws, _, rwst, iwst, N, rbt); - this.transform(nrws, _, nrwst, niwst, N, rbt); - - for (var i = 0; i < N; i++) { - var rx = rwst[i] * nrwst[i] - iwst[i] * niwst[i]; - iwst[i] = rwst[i] * niwst[i] + iwst[i] * nrwst[i]; - rwst[i] = rx; - } - - this.conjugate(rwst, iwst, N); - this.transform(rwst, iwst, rmws, _, N, rbt); - this.conjugate(rmws, _, N); - this.normalize13b(rmws, N); - - out.negative = x.negative ^ y.negative; - out.length = x.length + y.length; - return out._strip(); - }; - - // Multiply `this` by `num` - BN.prototype.mul = function mul (num) { - var out = new BN(null); - out.words = new Array(this.length + num.length); - return this.mulTo(num, out); - }; - - // Multiply employing FFT - BN.prototype.mulf = function mulf (num) { - var out = new BN(null); - out.words = new Array(this.length + num.length); - return jumboMulTo(this, num, out); - }; - - // In-place Multiplication - BN.prototype.imul = function imul (num) { - return this.clone().mulTo(num, this); - }; - - BN.prototype.imuln = function imuln (num) { - var isNegNum = num < 0; - if (isNegNum) num = -num; - - assert(typeof num === 'number'); - assert(num < 0x4000000); - - // Carry - var carry = 0; - for (var i = 0; i < this.length; i++) { - var w = (this.words[i] | 0) * num; - var lo = (w & 0x3ffffff) + (carry & 0x3ffffff); - carry >>= 26; - carry += (w / 0x4000000) | 0; - // NOTE: lo is 27bit maximum - carry += lo >>> 26; - this.words[i] = lo & 0x3ffffff; - } - - if (carry !== 0) { - this.words[i] = carry; - this.length++; - } - - return isNegNum ? this.ineg() : this; - }; - - BN.prototype.muln = function muln (num) { - return this.clone().imuln(num); - }; - - // `this` * `this` - BN.prototype.sqr = function sqr () { - return this.mul(this); - }; - - // `this` * `this` in-place - BN.prototype.isqr = function isqr () { - return this.imul(this.clone()); - }; - - // Math.pow(`this`, `num`) - BN.prototype.pow = function pow (num) { - var w = toBitArray(num); - if (w.length === 0) return new BN(1); - - // Skip leading zeroes - var res = this; - for (var i = 0; i < w.length; i++, res = res.sqr()) { - if (w[i] !== 0) break; - } - - if (++i < w.length) { - for (var q = res.sqr(); i < w.length; i++, q = q.sqr()) { - if (w[i] === 0) continue; - - res = res.mul(q); - } - } - - return res; - }; - - // Shift-left in-place - BN.prototype.iushln = function iushln (bits) { - assert(typeof bits === 'number' && bits >= 0); - var r = bits % 26; - var s = (bits - r) / 26; - var carryMask = (0x3ffffff >>> (26 - r)) << (26 - r); - var i; - - if (r !== 0) { - var carry = 0; - - for (i = 0; i < this.length; i++) { - var newCarry = this.words[i] & carryMask; - var c = ((this.words[i] | 0) - newCarry) << r; - this.words[i] = c | carry; - carry = newCarry >>> (26 - r); - } - - if (carry) { - this.words[i] = carry; - this.length++; - } - } - - if (s !== 0) { - for (i = this.length - 1; i >= 0; i--) { - this.words[i + s] = this.words[i]; - } - - for (i = 0; i < s; i++) { - this.words[i] = 0; - } - - this.length += s; - } - - return this._strip(); - }; - - BN.prototype.ishln = function ishln (bits) { - // TODO(indutny): implement me - assert(this.negative === 0); - return this.iushln(bits); - }; - - // Shift-right in-place - // NOTE: `hint` is a lowest bit before trailing zeroes - // NOTE: if `extended` is present - it will be filled with destroyed bits - BN.prototype.iushrn = function iushrn (bits, hint, extended) { - assert(typeof bits === 'number' && bits >= 0); - var h; - if (hint) { - h = (hint - (hint % 26)) / 26; - } else { - h = 0; - } - - var r = bits % 26; - var s = Math.min((bits - r) / 26, this.length); - var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r); - var maskedWords = extended; - - h -= s; - h = Math.max(0, h); - - // Extended mode, copy masked part - if (maskedWords) { - for (var i = 0; i < s; i++) { - maskedWords.words[i] = this.words[i]; - } - maskedWords.length = s; - } - - if (s === 0) { - // No-op, we should not move anything at all - } else if (this.length > s) { - this.length -= s; - for (i = 0; i < this.length; i++) { - this.words[i] = this.words[i + s]; - } - } else { - this.words[0] = 0; - this.length = 1; - } - - var carry = 0; - for (i = this.length - 1; i >= 0 && (carry !== 0 || i >= h); i--) { - var word = this.words[i] | 0; - this.words[i] = (carry << (26 - r)) | (word >>> r); - carry = word & mask; - } - - // Push carried bits as a mask - if (maskedWords && carry !== 0) { - maskedWords.words[maskedWords.length++] = carry; - } - - if (this.length === 0) { - this.words[0] = 0; - this.length = 1; - } - - return this._strip(); - }; - - BN.prototype.ishrn = function ishrn (bits, hint, extended) { - // TODO(indutny): implement me - assert(this.negative === 0); - return this.iushrn(bits, hint, extended); - }; - - // Shift-left - BN.prototype.shln = function shln (bits) { - return this.clone().ishln(bits); - }; - - BN.prototype.ushln = function ushln (bits) { - return this.clone().iushln(bits); - }; - - // Shift-right - BN.prototype.shrn = function shrn (bits) { - return this.clone().ishrn(bits); - }; - - BN.prototype.ushrn = function ushrn (bits) { - return this.clone().iushrn(bits); - }; - - // Test if n bit is set - BN.prototype.testn = function testn (bit) { - assert(typeof bit === 'number' && bit >= 0); - var r = bit % 26; - var s = (bit - r) / 26; - var q = 1 << r; - - // Fast case: bit is much higher than all existing words - if (this.length <= s) return false; - - // Check bit and return - var w = this.words[s]; - - return !!(w & q); - }; - - // Return only lowers bits of number (in-place) - BN.prototype.imaskn = function imaskn (bits) { - assert(typeof bits === 'number' && bits >= 0); - var r = bits % 26; - var s = (bits - r) / 26; - - assert(this.negative === 0, 'imaskn works only with positive numbers'); - - if (this.length <= s) { - return this; - } - - if (r !== 0) { - s++; - } - this.length = Math.min(s, this.length); - - if (r !== 0) { - var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r); - this.words[this.length - 1] &= mask; - } - - return this._strip(); - }; - - // Return only lowers bits of number - BN.prototype.maskn = function maskn (bits) { - return this.clone().imaskn(bits); - }; - - // Add plain number `num` to `this` - BN.prototype.iaddn = function iaddn (num) { - assert(typeof num === 'number'); - assert(num < 0x4000000); - if (num < 0) return this.isubn(-num); - - // Possible sign change - if (this.negative !== 0) { - if (this.length === 1 && (this.words[0] | 0) <= num) { - this.words[0] = num - (this.words[0] | 0); - this.negative = 0; - return this; - } - - this.negative = 0; - this.isubn(num); - this.negative = 1; - return this; - } - - // Add without checks - return this._iaddn(num); - }; - - BN.prototype._iaddn = function _iaddn (num) { - this.words[0] += num; - - // Carry - for (var i = 0; i < this.length && this.words[i] >= 0x4000000; i++) { - this.words[i] -= 0x4000000; - if (i === this.length - 1) { - this.words[i + 1] = 1; - } else { - this.words[i + 1]++; - } - } - this.length = Math.max(this.length, i + 1); - - return this; - }; - - // Subtract plain number `num` from `this` - BN.prototype.isubn = function isubn (num) { - assert(typeof num === 'number'); - assert(num < 0x4000000); - if (num < 0) return this.iaddn(-num); - - if (this.negative !== 0) { - this.negative = 0; - this.iaddn(num); - this.negative = 1; - return this; - } - - this.words[0] -= num; - - if (this.length === 1 && this.words[0] < 0) { - this.words[0] = -this.words[0]; - this.negative = 1; - } else { - // Carry - for (var i = 0; i < this.length && this.words[i] < 0; i++) { - this.words[i] += 0x4000000; - this.words[i + 1] -= 1; - } - } - - return this._strip(); - }; - - BN.prototype.addn = function addn (num) { - return this.clone().iaddn(num); - }; - - BN.prototype.subn = function subn (num) { - return this.clone().isubn(num); - }; - - BN.prototype.iabs = function iabs () { - this.negative = 0; - - return this; - }; - - BN.prototype.abs = function abs () { - return this.clone().iabs(); - }; - - BN.prototype._ishlnsubmul = function _ishlnsubmul (num, mul, shift) { - var len = num.length + shift; - var i; - - this._expand(len); - - var w; - var carry = 0; - for (i = 0; i < num.length; i++) { - w = (this.words[i + shift] | 0) + carry; - var right = (num.words[i] | 0) * mul; - w -= right & 0x3ffffff; - carry = (w >> 26) - ((right / 0x4000000) | 0); - this.words[i + shift] = w & 0x3ffffff; - } - for (; i < this.length - shift; i++) { - w = (this.words[i + shift] | 0) + carry; - carry = w >> 26; - this.words[i + shift] = w & 0x3ffffff; - } - - if (carry === 0) return this._strip(); - - // Subtraction overflow - assert(carry === -1); - carry = 0; - for (i = 0; i < this.length; i++) { - w = -(this.words[i] | 0) + carry; - carry = w >> 26; - this.words[i] = w & 0x3ffffff; - } - this.negative = 1; - - return this._strip(); - }; - - BN.prototype._wordDiv = function _wordDiv (num, mode) { - var shift = this.length - num.length; - - var a = this.clone(); - var b = num; - - // Normalize - var bhi = b.words[b.length - 1] | 0; - var bhiBits = this._countBits(bhi); - shift = 26 - bhiBits; - if (shift !== 0) { - b = b.ushln(shift); - a.iushln(shift); - bhi = b.words[b.length - 1] | 0; - } - - // Initialize quotient - var m = a.length - b.length; - var q; - - if (mode !== 'mod') { - q = new BN(null); - q.length = m + 1; - q.words = new Array(q.length); - for (var i = 0; i < q.length; i++) { - q.words[i] = 0; - } - } - - var diff = a.clone()._ishlnsubmul(b, 1, m); - if (diff.negative === 0) { - a = diff; - if (q) { - q.words[m] = 1; - } - } - - for (var j = m - 1; j >= 0; j--) { - var qj = (a.words[b.length + j] | 0) * 0x4000000 + - (a.words[b.length + j - 1] | 0); - - // NOTE: (qj / bhi) is (0x3ffffff * 0x4000000 + 0x3ffffff) / 0x2000000 max - // (0x7ffffff) - qj = Math.min((qj / bhi) | 0, 0x3ffffff); - - a._ishlnsubmul(b, qj, j); - while (a.negative !== 0) { - qj--; - a.negative = 0; - a._ishlnsubmul(b, 1, j); - if (!a.isZero()) { - a.negative ^= 1; - } - } - if (q) { - q.words[j] = qj; - } - } - if (q) { - q._strip(); - } - a._strip(); - - // Denormalize - if (mode !== 'div' && shift !== 0) { - a.iushrn(shift); - } - - return { - div: q || null, - mod: a - }; - }; - - // NOTE: 1) `mode` can be set to `mod` to request mod only, - // to `div` to request div only, or be absent to - // request both div & mod - // 2) `positive` is true if unsigned mod is requested - BN.prototype.divmod = function divmod (num, mode, positive) { - assert(!num.isZero()); - - if (this.isZero()) { - return { - div: new BN(0), - mod: new BN(0) - }; - } - - var div, mod, res; - if (this.negative !== 0 && num.negative === 0) { - res = this.neg().divmod(num, mode); - - if (mode !== 'mod') { - div = res.div.neg(); - } - - if (mode !== 'div') { - mod = res.mod.neg(); - if (positive && mod.negative !== 0) { - mod.iadd(num); - } - } - - return { - div: div, - mod: mod - }; - } - - if (this.negative === 0 && num.negative !== 0) { - res = this.divmod(num.neg(), mode); - - if (mode !== 'mod') { - div = res.div.neg(); - } - - return { - div: div, - mod: res.mod - }; - } - - if ((this.negative & num.negative) !== 0) { - res = this.neg().divmod(num.neg(), mode); - - if (mode !== 'div') { - mod = res.mod.neg(); - if (positive && mod.negative !== 0) { - mod.isub(num); - } - } - - return { - div: res.div, - mod: mod - }; - } - - // Both numbers are positive at this point - - // Strip both numbers to approximate shift value - if (num.length > this.length || this.cmp(num) < 0) { - return { - div: new BN(0), - mod: this - }; - } - - // Very short reduction - if (num.length === 1) { - if (mode === 'div') { - return { - div: this.divn(num.words[0]), - mod: null - }; - } - - if (mode === 'mod') { - return { - div: null, - mod: new BN(this.modrn(num.words[0])) - }; - } - - return { - div: this.divn(num.words[0]), - mod: new BN(this.modrn(num.words[0])) - }; - } - - return this._wordDiv(num, mode); - }; - - // Find `this` / `num` - BN.prototype.div = function div (num) { - return this.divmod(num, 'div', false).div; - }; - - // Find `this` % `num` - BN.prototype.mod = function mod (num) { - return this.divmod(num, 'mod', false).mod; - }; - - BN.prototype.umod = function umod (num) { - return this.divmod(num, 'mod', true).mod; - }; - - // Find Round(`this` / `num`) - BN.prototype.divRound = function divRound (num) { - var dm = this.divmod(num); - - // Fast case - exact division - if (dm.mod.isZero()) return dm.div; - - var mod = dm.div.negative !== 0 ? dm.mod.isub(num) : dm.mod; - - var half = num.ushrn(1); - var r2 = num.andln(1); - var cmp = mod.cmp(half); - - // Round down - if (cmp < 0 || (r2 === 1 && cmp === 0)) return dm.div; - - // Round up - return dm.div.negative !== 0 ? dm.div.isubn(1) : dm.div.iaddn(1); - }; - - BN.prototype.modrn = function modrn (num) { - var isNegNum = num < 0; - if (isNegNum) num = -num; - - assert(num <= 0x3ffffff); - var p = (1 << 26) % num; - - var acc = 0; - for (var i = this.length - 1; i >= 0; i--) { - acc = (p * acc + (this.words[i] | 0)) % num; - } - - return isNegNum ? -acc : acc; - }; - - // WARNING: DEPRECATED - BN.prototype.modn = function modn (num) { - return this.modrn(num); - }; - - // In-place division by number - BN.prototype.idivn = function idivn (num) { - var isNegNum = num < 0; - if (isNegNum) num = -num; - - assert(num <= 0x3ffffff); - - var carry = 0; - for (var i = this.length - 1; i >= 0; i--) { - var w = (this.words[i] | 0) + carry * 0x4000000; - this.words[i] = (w / num) | 0; - carry = w % num; - } - - this._strip(); - return isNegNum ? this.ineg() : this; - }; - - BN.prototype.divn = function divn (num) { - return this.clone().idivn(num); - }; - - BN.prototype.egcd = function egcd (p) { - assert(p.negative === 0); - assert(!p.isZero()); - - var x = this; - var y = p.clone(); - - if (x.negative !== 0) { - x = x.umod(p); - } else { - x = x.clone(); - } - - // A * x + B * y = x - var A = new BN(1); - var B = new BN(0); - - // C * x + D * y = y - var C = new BN(0); - var D = new BN(1); - - var g = 0; - - while (x.isEven() && y.isEven()) { - x.iushrn(1); - y.iushrn(1); - ++g; - } - - var yp = y.clone(); - var xp = x.clone(); - - while (!x.isZero()) { - for (var i = 0, im = 1; (x.words[0] & im) === 0 && i < 26; ++i, im <<= 1); - if (i > 0) { - x.iushrn(i); - while (i-- > 0) { - if (A.isOdd() || B.isOdd()) { - A.iadd(yp); - B.isub(xp); - } - - A.iushrn(1); - B.iushrn(1); - } - } - - for (var j = 0, jm = 1; (y.words[0] & jm) === 0 && j < 26; ++j, jm <<= 1); - if (j > 0) { - y.iushrn(j); - while (j-- > 0) { - if (C.isOdd() || D.isOdd()) { - C.iadd(yp); - D.isub(xp); - } - - C.iushrn(1); - D.iushrn(1); - } - } - - if (x.cmp(y) >= 0) { - x.isub(y); - A.isub(C); - B.isub(D); - } else { - y.isub(x); - C.isub(A); - D.isub(B); - } - } - - return { - a: C, - b: D, - gcd: y.iushln(g) - }; - }; - - // This is reduced incarnation of the binary EEA - // above, designated to invert members of the - // _prime_ fields F(p) at a maximal speed - BN.prototype._invmp = function _invmp (p) { - assert(p.negative === 0); - assert(!p.isZero()); - - var a = this; - var b = p.clone(); - - if (a.negative !== 0) { - a = a.umod(p); - } else { - a = a.clone(); - } - - var x1 = new BN(1); - var x2 = new BN(0); - - var delta = b.clone(); - - while (a.cmpn(1) > 0 && b.cmpn(1) > 0) { - for (var i = 0, im = 1; (a.words[0] & im) === 0 && i < 26; ++i, im <<= 1); - if (i > 0) { - a.iushrn(i); - while (i-- > 0) { - if (x1.isOdd()) { - x1.iadd(delta); - } - - x1.iushrn(1); - } - } - - for (var j = 0, jm = 1; (b.words[0] & jm) === 0 && j < 26; ++j, jm <<= 1); - if (j > 0) { - b.iushrn(j); - while (j-- > 0) { - if (x2.isOdd()) { - x2.iadd(delta); - } - - x2.iushrn(1); - } - } - - if (a.cmp(b) >= 0) { - a.isub(b); - x1.isub(x2); - } else { - b.isub(a); - x2.isub(x1); - } - } - - var res; - if (a.cmpn(1) === 0) { - res = x1; - } else { - res = x2; - } - - if (res.cmpn(0) < 0) { - res.iadd(p); - } - - return res; - }; - - BN.prototype.gcd = function gcd (num) { - if (this.isZero()) return num.abs(); - if (num.isZero()) return this.abs(); - - var a = this.clone(); - var b = num.clone(); - a.negative = 0; - b.negative = 0; - - // Remove common factor of two - for (var shift = 0; a.isEven() && b.isEven(); shift++) { - a.iushrn(1); - b.iushrn(1); - } - - do { - while (a.isEven()) { - a.iushrn(1); - } - while (b.isEven()) { - b.iushrn(1); - } - - var r = a.cmp(b); - if (r < 0) { - // Swap `a` and `b` to make `a` always bigger than `b` - var t = a; - a = b; - b = t; - } else if (r === 0 || b.cmpn(1) === 0) { - break; - } - - a.isub(b); - } while (true); - - return b.iushln(shift); - }; - - // Invert number in the field F(num) - BN.prototype.invm = function invm (num) { - return this.egcd(num).a.umod(num); - }; - - BN.prototype.isEven = function isEven () { - return (this.words[0] & 1) === 0; - }; - - BN.prototype.isOdd = function isOdd () { - return (this.words[0] & 1) === 1; - }; - - // And first word and num - BN.prototype.andln = function andln (num) { - return this.words[0] & num; - }; - - // Increment at the bit position in-line - BN.prototype.bincn = function bincn (bit) { - assert(typeof bit === 'number'); - var r = bit % 26; - var s = (bit - r) / 26; - var q = 1 << r; - - // Fast case: bit is much higher than all existing words - if (this.length <= s) { - this._expand(s + 1); - this.words[s] |= q; - return this; - } - - // Add bit and propagate, if needed - var carry = q; - for (var i = s; carry !== 0 && i < this.length; i++) { - var w = this.words[i] | 0; - w += carry; - carry = w >>> 26; - w &= 0x3ffffff; - this.words[i] = w; - } - if (carry !== 0) { - this.words[i] = carry; - this.length++; - } - return this; - }; - - BN.prototype.isZero = function isZero () { - return this.length === 1 && this.words[0] === 0; - }; - - BN.prototype.cmpn = function cmpn (num) { - var negative = num < 0; - - if (this.negative !== 0 && !negative) return -1; - if (this.negative === 0 && negative) return 1; - - this._strip(); - - var res; - if (this.length > 1) { - res = 1; - } else { - if (negative) { - num = -num; - } - - assert(num <= 0x3ffffff, 'Number is too big'); - - var w = this.words[0] | 0; - res = w === num ? 0 : w < num ? -1 : 1; - } - if (this.negative !== 0) return -res | 0; - return res; - }; - - // Compare two numbers and return: - // 1 - if `this` > `num` - // 0 - if `this` == `num` - // -1 - if `this` < `num` - BN.prototype.cmp = function cmp (num) { - if (this.negative !== 0 && num.negative === 0) return -1; - if (this.negative === 0 && num.negative !== 0) return 1; - - var res = this.ucmp(num); - if (this.negative !== 0) return -res | 0; - return res; - }; - - // Unsigned comparison - BN.prototype.ucmp = function ucmp (num) { - // At this point both numbers have the same sign - if (this.length > num.length) return 1; - if (this.length < num.length) return -1; - - var res = 0; - for (var i = this.length - 1; i >= 0; i--) { - var a = this.words[i] | 0; - var b = num.words[i] | 0; - - if (a === b) continue; - if (a < b) { - res = -1; - } else if (a > b) { - res = 1; - } - break; - } - return res; - }; - - BN.prototype.gtn = function gtn (num) { - return this.cmpn(num) === 1; - }; - - BN.prototype.gt = function gt (num) { - return this.cmp(num) === 1; - }; - - BN.prototype.gten = function gten (num) { - return this.cmpn(num) >= 0; - }; - - BN.prototype.gte = function gte (num) { - return this.cmp(num) >= 0; - }; - - BN.prototype.ltn = function ltn (num) { - return this.cmpn(num) === -1; - }; - - BN.prototype.lt = function lt (num) { - return this.cmp(num) === -1; - }; - - BN.prototype.lten = function lten (num) { - return this.cmpn(num) <= 0; - }; - - BN.prototype.lte = function lte (num) { - return this.cmp(num) <= 0; - }; - - BN.prototype.eqn = function eqn (num) { - return this.cmpn(num) === 0; - }; - - BN.prototype.eq = function eq (num) { - return this.cmp(num) === 0; - }; - - // - // A reduce context, could be using montgomery or something better, depending - // on the `m` itself. - // - BN.red = function red (num) { - return new Red(num); - }; - - BN.prototype.toRed = function toRed (ctx) { - assert(!this.red, 'Already a number in reduction context'); - assert(this.negative === 0, 'red works only with positives'); - return ctx.convertTo(this)._forceRed(ctx); - }; - - BN.prototype.fromRed = function fromRed () { - assert(this.red, 'fromRed works only with numbers in reduction context'); - return this.red.convertFrom(this); - }; - - BN.prototype._forceRed = function _forceRed (ctx) { - this.red = ctx; - return this; - }; - - BN.prototype.forceRed = function forceRed (ctx) { - assert(!this.red, 'Already a number in reduction context'); - return this._forceRed(ctx); - }; - - BN.prototype.redAdd = function redAdd (num) { - assert(this.red, 'redAdd works only with red numbers'); - return this.red.add(this, num); - }; - - BN.prototype.redIAdd = function redIAdd (num) { - assert(this.red, 'redIAdd works only with red numbers'); - return this.red.iadd(this, num); - }; - - BN.prototype.redSub = function redSub (num) { - assert(this.red, 'redSub works only with red numbers'); - return this.red.sub(this, num); - }; - - BN.prototype.redISub = function redISub (num) { - assert(this.red, 'redISub works only with red numbers'); - return this.red.isub(this, num); - }; - - BN.prototype.redShl = function redShl (num) { - assert(this.red, 'redShl works only with red numbers'); - return this.red.shl(this, num); - }; - - BN.prototype.redMul = function redMul (num) { - assert(this.red, 'redMul works only with red numbers'); - this.red._verify2(this, num); - return this.red.mul(this, num); - }; - - BN.prototype.redIMul = function redIMul (num) { - assert(this.red, 'redMul works only with red numbers'); - this.red._verify2(this, num); - return this.red.imul(this, num); - }; - - BN.prototype.redSqr = function redSqr () { - assert(this.red, 'redSqr works only with red numbers'); - this.red._verify1(this); - return this.red.sqr(this); - }; - - BN.prototype.redISqr = function redISqr () { - assert(this.red, 'redISqr works only with red numbers'); - this.red._verify1(this); - return this.red.isqr(this); - }; - - // Square root over p - BN.prototype.redSqrt = function redSqrt () { - assert(this.red, 'redSqrt works only with red numbers'); - this.red._verify1(this); - return this.red.sqrt(this); - }; - - BN.prototype.redInvm = function redInvm () { - assert(this.red, 'redInvm works only with red numbers'); - this.red._verify1(this); - return this.red.invm(this); - }; - - // Return negative clone of `this` % `red modulo` - BN.prototype.redNeg = function redNeg () { - assert(this.red, 'redNeg works only with red numbers'); - this.red._verify1(this); - return this.red.neg(this); - }; - - BN.prototype.redPow = function redPow (num) { - assert(this.red && !num.red, 'redPow(normalNum)'); - this.red._verify1(this); - return this.red.pow(this, num); - }; - - // Prime numbers with efficient reduction - var primes = { - k256: null, - p224: null, - p192: null, - p25519: null - }; - - // Pseudo-Mersenne prime - function MPrime (name, p) { - // P = 2 ^ N - K - this.name = name; - this.p = new BN(p, 16); - this.n = this.p.bitLength(); - this.k = new BN(1).iushln(this.n).isub(this.p); - - this.tmp = this._tmp(); - } - - MPrime.prototype._tmp = function _tmp () { - var tmp = new BN(null); - tmp.words = new Array(Math.ceil(this.n / 13)); - return tmp; - }; - - MPrime.prototype.ireduce = function ireduce (num) { - // Assumes that `num` is less than `P^2` - // num = HI * (2 ^ N - K) + HI * K + LO = HI * K + LO (mod P) - var r = num; - var rlen; - - do { - this.split(r, this.tmp); - r = this.imulK(r); - r = r.iadd(this.tmp); - rlen = r.bitLength(); - } while (rlen > this.n); - - var cmp = rlen < this.n ? -1 : r.ucmp(this.p); - if (cmp === 0) { - r.words[0] = 0; - r.length = 1; - } else if (cmp > 0) { - r.isub(this.p); - } else { - if (r.strip !== undefined) { - // r is a BN v4 instance - r.strip(); - } else { - // r is a BN v5 instance - r._strip(); - } - } - - return r; - }; - - MPrime.prototype.split = function split (input, out) { - input.iushrn(this.n, 0, out); - }; - - MPrime.prototype.imulK = function imulK (num) { - return num.imul(this.k); - }; - - function K256 () { - MPrime.call( - this, - 'k256', - 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff fffffffe fffffc2f'); - } - inherits(K256, MPrime); - - K256.prototype.split = function split (input, output) { - // 256 = 9 * 26 + 22 - var mask = 0x3fffff; - - var outLen = Math.min(input.length, 9); - for (var i = 0; i < outLen; i++) { - output.words[i] = input.words[i]; - } - output.length = outLen; - - if (input.length <= 9) { - input.words[0] = 0; - input.length = 1; - return; - } - - // Shift by 9 limbs - var prev = input.words[9]; - output.words[output.length++] = prev & mask; - - for (i = 10; i < input.length; i++) { - var next = input.words[i] | 0; - input.words[i - 10] = ((next & mask) << 4) | (prev >>> 22); - prev = next; - } - prev >>>= 22; - input.words[i - 10] = prev; - if (prev === 0 && input.length > 10) { - input.length -= 10; - } else { - input.length -= 9; - } - }; - - K256.prototype.imulK = function imulK (num) { - // K = 0x1000003d1 = [ 0x40, 0x3d1 ] - num.words[num.length] = 0; - num.words[num.length + 1] = 0; - num.length += 2; - - // bounded at: 0x40 * 0x3ffffff + 0x3d0 = 0x100000390 - var lo = 0; - for (var i = 0; i < num.length; i++) { - var w = num.words[i] | 0; - lo += w * 0x3d1; - num.words[i] = lo & 0x3ffffff; - lo = w * 0x40 + ((lo / 0x4000000) | 0); - } - - // Fast length reduction - if (num.words[num.length - 1] === 0) { - num.length--; - if (num.words[num.length - 1] === 0) { - num.length--; - } - } - return num; - }; - - function P224 () { - MPrime.call( - this, - 'p224', - 'ffffffff ffffffff ffffffff ffffffff 00000000 00000000 00000001'); - } - inherits(P224, MPrime); - - function P192 () { - MPrime.call( - this, - 'p192', - 'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff'); - } - inherits(P192, MPrime); - - function P25519 () { - // 2 ^ 255 - 19 - MPrime.call( - this, - '25519', - '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed'); - } - inherits(P25519, MPrime); - - P25519.prototype.imulK = function imulK (num) { - // K = 0x13 - var carry = 0; - for (var i = 0; i < num.length; i++) { - var hi = (num.words[i] | 0) * 0x13 + carry; - var lo = hi & 0x3ffffff; - hi >>>= 26; - - num.words[i] = lo; - carry = hi; - } - if (carry !== 0) { - num.words[num.length++] = carry; - } - return num; - }; - - // Exported mostly for testing purposes, use plain name instead - BN._prime = function prime (name) { - // Cached version of prime - if (primes[name]) return primes[name]; - - var prime; - if (name === 'k256') { - prime = new K256(); - } else if (name === 'p224') { - prime = new P224(); - } else if (name === 'p192') { - prime = new P192(); - } else if (name === 'p25519') { - prime = new P25519(); - } else { - throw new Error('Unknown prime ' + name); - } - primes[name] = prime; - - return prime; - }; - - // - // Base reduction engine - // - function Red (m) { - if (typeof m === 'string') { - var prime = BN._prime(m); - this.m = prime.p; - this.prime = prime; - } else { - assert(m.gtn(1), 'modulus must be greater than 1'); - this.m = m; - this.prime = null; - } - } - - Red.prototype._verify1 = function _verify1 (a) { - assert(a.negative === 0, 'red works only with positives'); - assert(a.red, 'red works only with red numbers'); - }; - - Red.prototype._verify2 = function _verify2 (a, b) { - assert((a.negative | b.negative) === 0, 'red works only with positives'); - assert(a.red && a.red === b.red, - 'red works only with red numbers'); - }; - - Red.prototype.imod = function imod (a) { - if (this.prime) return this.prime.ireduce(a)._forceRed(this); - - move(a, a.umod(this.m)._forceRed(this)); - return a; - }; - - Red.prototype.neg = function neg (a) { - if (a.isZero()) { - return a.clone(); - } - - return this.m.sub(a)._forceRed(this); - }; - - Red.prototype.add = function add (a, b) { - this._verify2(a, b); - - var res = a.add(b); - if (res.cmp(this.m) >= 0) { - res.isub(this.m); - } - return res._forceRed(this); - }; - - Red.prototype.iadd = function iadd (a, b) { - this._verify2(a, b); - - var res = a.iadd(b); - if (res.cmp(this.m) >= 0) { - res.isub(this.m); - } - return res; - }; - - Red.prototype.sub = function sub (a, b) { - this._verify2(a, b); - - var res = a.sub(b); - if (res.cmpn(0) < 0) { - res.iadd(this.m); - } - return res._forceRed(this); - }; - - Red.prototype.isub = function isub (a, b) { - this._verify2(a, b); - - var res = a.isub(b); - if (res.cmpn(0) < 0) { - res.iadd(this.m); - } - return res; - }; - - Red.prototype.shl = function shl (a, num) { - this._verify1(a); - return this.imod(a.ushln(num)); - }; - - Red.prototype.imul = function imul (a, b) { - this._verify2(a, b); - return this.imod(a.imul(b)); - }; - - Red.prototype.mul = function mul (a, b) { - this._verify2(a, b); - return this.imod(a.mul(b)); - }; - - Red.prototype.isqr = function isqr (a) { - return this.imul(a, a.clone()); - }; - - Red.prototype.sqr = function sqr (a) { - return this.mul(a, a); - }; - - Red.prototype.sqrt = function sqrt (a) { - if (a.isZero()) return a.clone(); - - var mod3 = this.m.andln(3); - assert(mod3 % 2 === 1); - - // Fast case - if (mod3 === 3) { - var pow = this.m.add(new BN(1)).iushrn(2); - return this.pow(a, pow); - } - - // Tonelli-Shanks algorithm (Totally unoptimized and slow) - // - // Find Q and S, that Q * 2 ^ S = (P - 1) - var q = this.m.subn(1); - var s = 0; - while (!q.isZero() && q.andln(1) === 0) { - s++; - q.iushrn(1); - } - assert(!q.isZero()); - - var one = new BN(1).toRed(this); - var nOne = one.redNeg(); - - // Find quadratic non-residue - // NOTE: Max is such because of generalized Riemann hypothesis. - var lpow = this.m.subn(1).iushrn(1); - var z = this.m.bitLength(); - z = new BN(2 * z * z).toRed(this); - - while (this.pow(z, lpow).cmp(nOne) !== 0) { - z.redIAdd(nOne); - } - - var c = this.pow(z, q); - var r = this.pow(a, q.addn(1).iushrn(1)); - var t = this.pow(a, q); - var m = s; - while (t.cmp(one) !== 0) { - var tmp = t; - for (var i = 0; tmp.cmp(one) !== 0; i++) { - tmp = tmp.redSqr(); - } - assert(i < m); - var b = this.pow(c, new BN(1).iushln(m - i - 1)); - - r = r.redMul(b); - c = b.redSqr(); - t = t.redMul(c); - m = i; - } - - return r; - }; - - Red.prototype.invm = function invm (a) { - var inv = a._invmp(this.m); - if (inv.negative !== 0) { - inv.negative = 0; - return this.imod(inv).redNeg(); - } else { - return this.imod(inv); - } - }; - - Red.prototype.pow = function pow (a, num) { - if (num.isZero()) return new BN(1).toRed(this); - if (num.cmpn(1) === 0) return a.clone(); - - var windowSize = 4; - var wnd = new Array(1 << windowSize); - wnd[0] = new BN(1).toRed(this); - wnd[1] = a; - for (var i = 2; i < wnd.length; i++) { - wnd[i] = this.mul(wnd[i - 1], a); - } - - var res = wnd[0]; - var current = 0; - var currentLen = 0; - var start = num.bitLength() % 26; - if (start === 0) { - start = 26; - } - - for (i = num.length - 1; i >= 0; i--) { - var word = num.words[i]; - for (var j = start - 1; j >= 0; j--) { - var bit = (word >> j) & 1; - if (res !== wnd[0]) { - res = this.sqr(res); - } - - if (bit === 0 && current === 0) { - currentLen = 0; - continue; - } - - current <<= 1; - current |= bit; - currentLen++; - if (currentLen !== windowSize && (i !== 0 || j !== 0)) continue; - - res = this.mul(res, wnd[current]); - currentLen = 0; - current = 0; - } - start = 26; - } - - return res; - }; - - Red.prototype.convertTo = function convertTo (num) { - var r = num.umod(this.m); - - return r === num ? r.clone() : r; - }; - - Red.prototype.convertFrom = function convertFrom (num) { - var res = num.clone(); - res.red = null; - return res; - }; - - // - // Montgomery method engine - // - - BN.mont = function mont (num) { - return new Mont(num); - }; - - function Mont (m) { - Red.call(this, m); - - this.shift = this.m.bitLength(); - if (this.shift % 26 !== 0) { - this.shift += 26 - (this.shift % 26); - } - - this.r = new BN(1).iushln(this.shift); - this.r2 = this.imod(this.r.sqr()); - this.rinv = this.r._invmp(this.m); - - this.minv = this.rinv.mul(this.r).isubn(1).div(this.m); - this.minv = this.minv.umod(this.r); - this.minv = this.r.sub(this.minv); - } - inherits(Mont, Red); - - Mont.prototype.convertTo = function convertTo (num) { - return this.imod(num.ushln(this.shift)); - }; - - Mont.prototype.convertFrom = function convertFrom (num) { - var r = this.imod(num.mul(this.rinv)); - r.red = null; - return r; - }; - - Mont.prototype.imul = function imul (a, b) { - if (a.isZero() || b.isZero()) { - a.words[0] = 0; - a.length = 1; - return a; - } - - var t = a.imul(b); - var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m); - var u = t.isub(c).iushrn(this.shift); - var res = u; - - if (u.cmp(this.m) >= 0) { - res = u.isub(this.m); - } else if (u.cmpn(0) < 0) { - res = u.iadd(this.m); - } - - return res._forceRed(this); - }; - - Mont.prototype.mul = function mul (a, b) { - if (a.isZero() || b.isZero()) return new BN(0)._forceRed(this); - - var t = a.mul(b); - var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m); - var u = t.isub(c).iushrn(this.shift); - var res = u; - if (u.cmp(this.m) >= 0) { - res = u.isub(this.m); - } else if (u.cmpn(0) < 0) { - res = u.iadd(this.m); - } - - return res._forceRed(this); - }; - - Mont.prototype.invm = function invm (a) { - // (AR)^-1 * R^2 = (A^-1 * R^-1) * R^2 = A^-1 * R - var res = this.imod(a._invmp(this.m).mul(this.r2)); - return res._forceRed(this); - }; -})(typeof module === 'undefined' || module, this); - -},{"buffer":77}],76:[function(require,module,exports){ -var r; - -module.exports = function rand(len) { - if (!r) - r = new Rand(null); - - return r.generate(len); -}; - -function Rand(rand) { - this.rand = rand; -} -module.exports.Rand = Rand; - -Rand.prototype.generate = function generate(len) { - return this._rand(len); -}; - -// Emulate crypto API using randy -Rand.prototype._rand = function _rand(n) { - if (this.rand.getBytes) - return this.rand.getBytes(n); - - var res = new Uint8Array(n); - for (var i = 0; i < res.length; i++) - res[i] = this.rand.getByte(); - return res; -}; - -if (typeof self === 'object') { - if (self.crypto && self.crypto.getRandomValues) { - // Modern browsers - Rand.prototype._rand = function _rand(n) { - var arr = new Uint8Array(n); - self.crypto.getRandomValues(arr); - return arr; - }; - } else if (self.msCrypto && self.msCrypto.getRandomValues) { - // IE - Rand.prototype._rand = function _rand(n) { - var arr = new Uint8Array(n); - self.msCrypto.getRandomValues(arr); - return arr; - }; - - // Safari's WebWorkers do not have `crypto` - } else if (typeof window === 'object') { - // Old junk - Rand.prototype._rand = function() { - throw new Error('Not implemented yet'); - }; - } -} else { - // Node.js or Web worker with no crypto support - try { - var crypto = require('crypto'); - if (typeof crypto.randomBytes !== 'function') - throw new Error('Not supported'); - - Rand.prototype._rand = function _rand(n) { - return crypto.randomBytes(n); - }; - } catch (e) { - } -} - -},{"crypto":77}],77:[function(require,module,exports){ - -},{}],78:[function(require,module,exports){ -// based on the aes implimentation in triple sec -// https://github.com/keybase/triplesec -// which is in turn based on the one from crypto-js -// https://code.google.com/p/crypto-js/ - -var Buffer = require('safe-buffer').Buffer - -function asUInt32Array (buf) { - if (!Buffer.isBuffer(buf)) buf = Buffer.from(buf) - - var len = (buf.length / 4) | 0 - var out = new Array(len) - - for (var i = 0; i < len; i++) { - out[i] = buf.readUInt32BE(i * 4) - } - - return out -} - -function scrubVec (v) { - for (var i = 0; i < v.length; v++) { - v[i] = 0 - } -} - -function cryptBlock (M, keySchedule, SUB_MIX, SBOX, nRounds) { - var SUB_MIX0 = SUB_MIX[0] - var SUB_MIX1 = SUB_MIX[1] - var SUB_MIX2 = SUB_MIX[2] - var SUB_MIX3 = SUB_MIX[3] - - var s0 = M[0] ^ keySchedule[0] - var s1 = M[1] ^ keySchedule[1] - var s2 = M[2] ^ keySchedule[2] - var s3 = M[3] ^ keySchedule[3] - var t0, t1, t2, t3 - var ksRow = 4 - - for (var round = 1; round < nRounds; round++) { - t0 = SUB_MIX0[s0 >>> 24] ^ SUB_MIX1[(s1 >>> 16) & 0xff] ^ SUB_MIX2[(s2 >>> 8) & 0xff] ^ SUB_MIX3[s3 & 0xff] ^ keySchedule[ksRow++] - t1 = SUB_MIX0[s1 >>> 24] ^ SUB_MIX1[(s2 >>> 16) & 0xff] ^ SUB_MIX2[(s3 >>> 8) & 0xff] ^ SUB_MIX3[s0 & 0xff] ^ keySchedule[ksRow++] - t2 = SUB_MIX0[s2 >>> 24] ^ SUB_MIX1[(s3 >>> 16) & 0xff] ^ SUB_MIX2[(s0 >>> 8) & 0xff] ^ SUB_MIX3[s1 & 0xff] ^ keySchedule[ksRow++] - t3 = SUB_MIX0[s3 >>> 24] ^ SUB_MIX1[(s0 >>> 16) & 0xff] ^ SUB_MIX2[(s1 >>> 8) & 0xff] ^ SUB_MIX3[s2 & 0xff] ^ keySchedule[ksRow++] - s0 = t0 - s1 = t1 - s2 = t2 - s3 = t3 - } - - t0 = ((SBOX[s0 >>> 24] << 24) | (SBOX[(s1 >>> 16) & 0xff] << 16) | (SBOX[(s2 >>> 8) & 0xff] << 8) | SBOX[s3 & 0xff]) ^ keySchedule[ksRow++] - t1 = ((SBOX[s1 >>> 24] << 24) | (SBOX[(s2 >>> 16) & 0xff] << 16) | (SBOX[(s3 >>> 8) & 0xff] << 8) | SBOX[s0 & 0xff]) ^ keySchedule[ksRow++] - t2 = ((SBOX[s2 >>> 24] << 24) | (SBOX[(s3 >>> 16) & 0xff] << 16) | (SBOX[(s0 >>> 8) & 0xff] << 8) | SBOX[s1 & 0xff]) ^ keySchedule[ksRow++] - t3 = ((SBOX[s3 >>> 24] << 24) | (SBOX[(s0 >>> 16) & 0xff] << 16) | (SBOX[(s1 >>> 8) & 0xff] << 8) | SBOX[s2 & 0xff]) ^ keySchedule[ksRow++] - t0 = t0 >>> 0 - t1 = t1 >>> 0 - t2 = t2 >>> 0 - t3 = t3 >>> 0 - - return [t0, t1, t2, t3] -} - -// AES constants -var RCON = [0x00, 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36] -var G = (function () { - // Compute double table - var d = new Array(256) - for (var j = 0; j < 256; j++) { - if (j < 128) { - d[j] = j << 1 - } else { - d[j] = (j << 1) ^ 0x11b - } - } - - var SBOX = [] - var INV_SBOX = [] - var SUB_MIX = [[], [], [], []] - var INV_SUB_MIX = [[], [], [], []] - - // Walk GF(2^8) - var x = 0 - var xi = 0 - for (var i = 0; i < 256; ++i) { - // Compute sbox - var sx = xi ^ (xi << 1) ^ (xi << 2) ^ (xi << 3) ^ (xi << 4) - sx = (sx >>> 8) ^ (sx & 0xff) ^ 0x63 - SBOX[x] = sx - INV_SBOX[sx] = x - - // Compute multiplication - var x2 = d[x] - var x4 = d[x2] - var x8 = d[x4] - - // Compute sub bytes, mix columns tables - var t = (d[sx] * 0x101) ^ (sx * 0x1010100) - SUB_MIX[0][x] = (t << 24) | (t >>> 8) - SUB_MIX[1][x] = (t << 16) | (t >>> 16) - SUB_MIX[2][x] = (t << 8) | (t >>> 24) - SUB_MIX[3][x] = t - - // Compute inv sub bytes, inv mix columns tables - t = (x8 * 0x1010101) ^ (x4 * 0x10001) ^ (x2 * 0x101) ^ (x * 0x1010100) - INV_SUB_MIX[0][sx] = (t << 24) | (t >>> 8) - INV_SUB_MIX[1][sx] = (t << 16) | (t >>> 16) - INV_SUB_MIX[2][sx] = (t << 8) | (t >>> 24) - INV_SUB_MIX[3][sx] = t - - if (x === 0) { - x = xi = 1 - } else { - x = x2 ^ d[d[d[x8 ^ x2]]] - xi ^= d[d[xi]] - } - } - - return { - SBOX: SBOX, - INV_SBOX: INV_SBOX, - SUB_MIX: SUB_MIX, - INV_SUB_MIX: INV_SUB_MIX - } -})() - -function AES (key) { - this._key = asUInt32Array(key) - this._reset() -} - -AES.blockSize = 4 * 4 -AES.keySize = 256 / 8 -AES.prototype.blockSize = AES.blockSize -AES.prototype.keySize = AES.keySize -AES.prototype._reset = function () { - var keyWords = this._key - var keySize = keyWords.length - var nRounds = keySize + 6 - var ksRows = (nRounds + 1) * 4 - - var keySchedule = [] - for (var k = 0; k < keySize; k++) { - keySchedule[k] = keyWords[k] - } - - for (k = keySize; k < ksRows; k++) { - var t = keySchedule[k - 1] - - if (k % keySize === 0) { - t = (t << 8) | (t >>> 24) - t = - (G.SBOX[t >>> 24] << 24) | - (G.SBOX[(t >>> 16) & 0xff] << 16) | - (G.SBOX[(t >>> 8) & 0xff] << 8) | - (G.SBOX[t & 0xff]) - - t ^= RCON[(k / keySize) | 0] << 24 - } else if (keySize > 6 && k % keySize === 4) { - t = - (G.SBOX[t >>> 24] << 24) | - (G.SBOX[(t >>> 16) & 0xff] << 16) | - (G.SBOX[(t >>> 8) & 0xff] << 8) | - (G.SBOX[t & 0xff]) - } - - keySchedule[k] = keySchedule[k - keySize] ^ t - } - - var invKeySchedule = [] - for (var ik = 0; ik < ksRows; ik++) { - var ksR = ksRows - ik - var tt = keySchedule[ksR - (ik % 4 ? 0 : 4)] - - if (ik < 4 || ksR <= 4) { - invKeySchedule[ik] = tt - } else { - invKeySchedule[ik] = - G.INV_SUB_MIX[0][G.SBOX[tt >>> 24]] ^ - G.INV_SUB_MIX[1][G.SBOX[(tt >>> 16) & 0xff]] ^ - G.INV_SUB_MIX[2][G.SBOX[(tt >>> 8) & 0xff]] ^ - G.INV_SUB_MIX[3][G.SBOX[tt & 0xff]] - } - } - - this._nRounds = nRounds - this._keySchedule = keySchedule - this._invKeySchedule = invKeySchedule -} - -AES.prototype.encryptBlockRaw = function (M) { - M = asUInt32Array(M) - return cryptBlock(M, this._keySchedule, G.SUB_MIX, G.SBOX, this._nRounds) -} - -AES.prototype.encryptBlock = function (M) { - var out = this.encryptBlockRaw(M) - var buf = Buffer.allocUnsafe(16) - buf.writeUInt32BE(out[0], 0) - buf.writeUInt32BE(out[1], 4) - buf.writeUInt32BE(out[2], 8) - buf.writeUInt32BE(out[3], 12) - return buf -} - -AES.prototype.decryptBlock = function (M) { - M = asUInt32Array(M) - - // swap - var m1 = M[1] - M[1] = M[3] - M[3] = m1 - - var out = cryptBlock(M, this._invKeySchedule, G.INV_SUB_MIX, G.INV_SBOX, this._nRounds) - var buf = Buffer.allocUnsafe(16) - buf.writeUInt32BE(out[0], 0) - buf.writeUInt32BE(out[3], 4) - buf.writeUInt32BE(out[2], 8) - buf.writeUInt32BE(out[1], 12) - return buf -} - -AES.prototype.scrub = function () { - scrubVec(this._keySchedule) - scrubVec(this._invKeySchedule) - scrubVec(this._key) -} - -module.exports.AES = AES - -},{"safe-buffer":269}],79:[function(require,module,exports){ -var aes = require('./aes') -var Buffer = require('safe-buffer').Buffer -var Transform = require('cipher-base') -var inherits = require('inherits') -var GHASH = require('./ghash') -var xor = require('buffer-xor') -var incr32 = require('./incr32') - -function xorTest (a, b) { - var out = 0 - if (a.length !== b.length) out++ - - var len = Math.min(a.length, b.length) - for (var i = 0; i < len; ++i) { - out += (a[i] ^ b[i]) - } - - return out -} - -function calcIv (self, iv, ck) { - if (iv.length === 12) { - self._finID = Buffer.concat([iv, Buffer.from([0, 0, 0, 1])]) - return Buffer.concat([iv, Buffer.from([0, 0, 0, 2])]) - } - var ghash = new GHASH(ck) - var len = iv.length - var toPad = len % 16 - ghash.update(iv) - if (toPad) { - toPad = 16 - toPad - ghash.update(Buffer.alloc(toPad, 0)) - } - ghash.update(Buffer.alloc(8, 0)) - var ivBits = len * 8 - var tail = Buffer.alloc(8) - tail.writeUIntBE(ivBits, 0, 8) - ghash.update(tail) - self._finID = ghash.state - var out = Buffer.from(self._finID) - incr32(out) - return out -} -function StreamCipher (mode, key, iv, decrypt) { - Transform.call(this) - - var h = Buffer.alloc(4, 0) - - this._cipher = new aes.AES(key) - var ck = this._cipher.encryptBlock(h) - this._ghash = new GHASH(ck) - iv = calcIv(this, iv, ck) - - this._prev = Buffer.from(iv) - this._cache = Buffer.allocUnsafe(0) - this._secCache = Buffer.allocUnsafe(0) - this._decrypt = decrypt - this._alen = 0 - this._len = 0 - this._mode = mode - - this._authTag = null - this._called = false -} - -inherits(StreamCipher, Transform) - -StreamCipher.prototype._update = function (chunk) { - if (!this._called && this._alen) { - var rump = 16 - (this._alen % 16) - if (rump < 16) { - rump = Buffer.alloc(rump, 0) - this._ghash.update(rump) - } - } - - this._called = true - var out = this._mode.encrypt(this, chunk) - if (this._decrypt) { - this._ghash.update(chunk) - } else { - this._ghash.update(out) - } - this._len += chunk.length - return out -} - -StreamCipher.prototype._final = function () { - if (this._decrypt && !this._authTag) throw new Error('Unsupported state or unable to authenticate data') - - var tag = xor(this._ghash.final(this._alen * 8, this._len * 8), this._cipher.encryptBlock(this._finID)) - if (this._decrypt && xorTest(tag, this._authTag)) throw new Error('Unsupported state or unable to authenticate data') - - this._authTag = tag - this._cipher.scrub() -} - -StreamCipher.prototype.getAuthTag = function getAuthTag () { - if (this._decrypt || !Buffer.isBuffer(this._authTag)) throw new Error('Attempting to get auth tag in unsupported state') - - return this._authTag -} - -StreamCipher.prototype.setAuthTag = function setAuthTag (tag) { - if (!this._decrypt) throw new Error('Attempting to set auth tag in unsupported state') - - this._authTag = tag -} - -StreamCipher.prototype.setAAD = function setAAD (buf) { - if (this._called) throw new Error('Attempting to set AAD in unsupported state') - - this._ghash.update(buf) - this._alen += buf.length -} - -module.exports = StreamCipher - -},{"./aes":78,"./ghash":83,"./incr32":84,"buffer-xor":123,"cipher-base":125,"inherits":218,"safe-buffer":269}],80:[function(require,module,exports){ -var ciphers = require('./encrypter') -var deciphers = require('./decrypter') -var modes = require('./modes/list.json') - -function getCiphers () { - return Object.keys(modes) -} - -exports.createCipher = exports.Cipher = ciphers.createCipher -exports.createCipheriv = exports.Cipheriv = ciphers.createCipheriv -exports.createDecipher = exports.Decipher = deciphers.createDecipher -exports.createDecipheriv = exports.Decipheriv = deciphers.createDecipheriv -exports.listCiphers = exports.getCiphers = getCiphers - -},{"./decrypter":81,"./encrypter":82,"./modes/list.json":92}],81:[function(require,module,exports){ -var AuthCipher = require('./authCipher') -var Buffer = require('safe-buffer').Buffer -var MODES = require('./modes') -var StreamCipher = require('./streamCipher') -var Transform = require('cipher-base') -var aes = require('./aes') -var ebtk = require('evp_bytestokey') -var inherits = require('inherits') - -function Decipher (mode, key, iv) { - Transform.call(this) - - this._cache = new Splitter() - this._last = void 0 - this._cipher = new aes.AES(key) - this._prev = Buffer.from(iv) - this._mode = mode - this._autopadding = true -} - -inherits(Decipher, Transform) - -Decipher.prototype._update = function (data) { - this._cache.add(data) - var chunk - var thing - var out = [] - while ((chunk = this._cache.get(this._autopadding))) { - thing = this._mode.decrypt(this, chunk) - out.push(thing) - } - return Buffer.concat(out) -} - -Decipher.prototype._final = function () { - var chunk = this._cache.flush() - if (this._autopadding) { - return unpad(this._mode.decrypt(this, chunk)) - } else if (chunk) { - throw new Error('data not multiple of block length') - } -} - -Decipher.prototype.setAutoPadding = function (setTo) { - this._autopadding = !!setTo - return this -} - -function Splitter () { - this.cache = Buffer.allocUnsafe(0) -} - -Splitter.prototype.add = function (data) { - this.cache = Buffer.concat([this.cache, data]) -} - -Splitter.prototype.get = function (autoPadding) { - var out - if (autoPadding) { - if (this.cache.length > 16) { - out = this.cache.slice(0, 16) - this.cache = this.cache.slice(16) - return out - } - } else { - if (this.cache.length >= 16) { - out = this.cache.slice(0, 16) - this.cache = this.cache.slice(16) - return out - } - } - - return null -} - -Splitter.prototype.flush = function () { - if (this.cache.length) return this.cache -} - -function unpad (last) { - var padded = last[15] - if (padded < 1 || padded > 16) { - throw new Error('unable to decrypt data') - } - var i = -1 - while (++i < padded) { - if (last[(i + (16 - padded))] !== padded) { - throw new Error('unable to decrypt data') - } - } - if (padded === 16) return - - return last.slice(0, 16 - padded) -} - -function createDecipheriv (suite, password, iv) { - var config = MODES[suite.toLowerCase()] - if (!config) throw new TypeError('invalid suite type') - - if (typeof iv === 'string') iv = Buffer.from(iv) - if (config.mode !== 'GCM' && iv.length !== config.iv) throw new TypeError('invalid iv length ' + iv.length) - - if (typeof password === 'string') password = Buffer.from(password) - if (password.length !== config.key / 8) throw new TypeError('invalid key length ' + password.length) - - if (config.type === 'stream') { - return new StreamCipher(config.module, password, iv, true) - } else if (config.type === 'auth') { - return new AuthCipher(config.module, password, iv, true) - } - - return new Decipher(config.module, password, iv) -} - -function createDecipher (suite, password) { - var config = MODES[suite.toLowerCase()] - if (!config) throw new TypeError('invalid suite type') - - var keys = ebtk(password, false, config.key, config.iv) - return createDecipheriv(suite, keys.key, keys.iv) -} - -exports.createDecipher = createDecipher -exports.createDecipheriv = createDecipheriv - -},{"./aes":78,"./authCipher":79,"./modes":91,"./streamCipher":94,"cipher-base":125,"evp_bytestokey":182,"inherits":218,"safe-buffer":269}],82:[function(require,module,exports){ -var MODES = require('./modes') -var AuthCipher = require('./authCipher') -var Buffer = require('safe-buffer').Buffer -var StreamCipher = require('./streamCipher') -var Transform = require('cipher-base') -var aes = require('./aes') -var ebtk = require('evp_bytestokey') -var inherits = require('inherits') - -function Cipher (mode, key, iv) { - Transform.call(this) - - this._cache = new Splitter() - this._cipher = new aes.AES(key) - this._prev = Buffer.from(iv) - this._mode = mode - this._autopadding = true -} - -inherits(Cipher, Transform) - -Cipher.prototype._update = function (data) { - this._cache.add(data) - var chunk - var thing - var out = [] - - while ((chunk = this._cache.get())) { - thing = this._mode.encrypt(this, chunk) - out.push(thing) - } - - return Buffer.concat(out) -} - -var PADDING = Buffer.alloc(16, 0x10) - -Cipher.prototype._final = function () { - var chunk = this._cache.flush() - if (this._autopadding) { - chunk = this._mode.encrypt(this, chunk) - this._cipher.scrub() - return chunk - } - - if (!chunk.equals(PADDING)) { - this._cipher.scrub() - throw new Error('data not multiple of block length') - } -} - -Cipher.prototype.setAutoPadding = function (setTo) { - this._autopadding = !!setTo - return this -} - -function Splitter () { - this.cache = Buffer.allocUnsafe(0) -} - -Splitter.prototype.add = function (data) { - this.cache = Buffer.concat([this.cache, data]) -} - -Splitter.prototype.get = function () { - if (this.cache.length > 15) { - var out = this.cache.slice(0, 16) - this.cache = this.cache.slice(16) - return out - } - return null -} - -Splitter.prototype.flush = function () { - var len = 16 - this.cache.length - var padBuff = Buffer.allocUnsafe(len) - - var i = -1 - while (++i < len) { - padBuff.writeUInt8(len, i) - } - - return Buffer.concat([this.cache, padBuff]) -} - -function createCipheriv (suite, password, iv) { - var config = MODES[suite.toLowerCase()] - if (!config) throw new TypeError('invalid suite type') - - if (typeof password === 'string') password = Buffer.from(password) - if (password.length !== config.key / 8) throw new TypeError('invalid key length ' + password.length) - - if (typeof iv === 'string') iv = Buffer.from(iv) - if (config.mode !== 'GCM' && iv.length !== config.iv) throw new TypeError('invalid iv length ' + iv.length) - - if (config.type === 'stream') { - return new StreamCipher(config.module, password, iv) - } else if (config.type === 'auth') { - return new AuthCipher(config.module, password, iv) - } - - return new Cipher(config.module, password, iv) -} - -function createCipher (suite, password) { - var config = MODES[suite.toLowerCase()] - if (!config) throw new TypeError('invalid suite type') - - var keys = ebtk(password, false, config.key, config.iv) - return createCipheriv(suite, keys.key, keys.iv) -} - -exports.createCipheriv = createCipheriv -exports.createCipher = createCipher - -},{"./aes":78,"./authCipher":79,"./modes":91,"./streamCipher":94,"cipher-base":125,"evp_bytestokey":182,"inherits":218,"safe-buffer":269}],83:[function(require,module,exports){ -var Buffer = require('safe-buffer').Buffer -var ZEROES = Buffer.alloc(16, 0) - -function toArray (buf) { - return [ - buf.readUInt32BE(0), - buf.readUInt32BE(4), - buf.readUInt32BE(8), - buf.readUInt32BE(12) - ] -} - -function fromArray (out) { - var buf = Buffer.allocUnsafe(16) - buf.writeUInt32BE(out[0] >>> 0, 0) - buf.writeUInt32BE(out[1] >>> 0, 4) - buf.writeUInt32BE(out[2] >>> 0, 8) - buf.writeUInt32BE(out[3] >>> 0, 12) - return buf -} - -function GHASH (key) { - this.h = key - this.state = Buffer.alloc(16, 0) - this.cache = Buffer.allocUnsafe(0) -} - -// from http://bitwiseshiftleft.github.io/sjcl/doc/symbols/src/core_gcm.js.html -// by Juho Vähä-Herttua -GHASH.prototype.ghash = function (block) { - var i = -1 - while (++i < block.length) { - this.state[i] ^= block[i] - } - this._multiply() -} - -GHASH.prototype._multiply = function () { - var Vi = toArray(this.h) - var Zi = [0, 0, 0, 0] - var j, xi, lsbVi - var i = -1 - while (++i < 128) { - xi = (this.state[~~(i / 8)] & (1 << (7 - (i % 8)))) !== 0 - if (xi) { - // Z_i+1 = Z_i ^ V_i - Zi[0] ^= Vi[0] - Zi[1] ^= Vi[1] - Zi[2] ^= Vi[2] - Zi[3] ^= Vi[3] - } - - // Store the value of LSB(V_i) - lsbVi = (Vi[3] & 1) !== 0 - - // V_i+1 = V_i >> 1 - for (j = 3; j > 0; j--) { - Vi[j] = (Vi[j] >>> 1) | ((Vi[j - 1] & 1) << 31) - } - Vi[0] = Vi[0] >>> 1 - - // If LSB(V_i) is 1, V_i+1 = (V_i >> 1) ^ R - if (lsbVi) { - Vi[0] = Vi[0] ^ (0xe1 << 24) - } - } - this.state = fromArray(Zi) -} - -GHASH.prototype.update = function (buf) { - this.cache = Buffer.concat([this.cache, buf]) - var chunk - while (this.cache.length >= 16) { - chunk = this.cache.slice(0, 16) - this.cache = this.cache.slice(16) - this.ghash(chunk) - } -} - -GHASH.prototype.final = function (abl, bl) { - if (this.cache.length) { - this.ghash(Buffer.concat([this.cache, ZEROES], 16)) - } - - this.ghash(fromArray([0, abl, 0, bl])) - return this.state -} - -module.exports = GHASH - -},{"safe-buffer":269}],84:[function(require,module,exports){ -function incr32 (iv) { - var len = iv.length - var item - while (len--) { - item = iv.readUInt8(len) - if (item === 255) { - iv.writeUInt8(0, len) - } else { - item++ - iv.writeUInt8(item, len) - break - } - } -} -module.exports = incr32 - -},{}],85:[function(require,module,exports){ -var xor = require('buffer-xor') - -exports.encrypt = function (self, block) { - var data = xor(block, self._prev) - - self._prev = self._cipher.encryptBlock(data) - return self._prev -} - -exports.decrypt = function (self, block) { - var pad = self._prev - - self._prev = block - var out = self._cipher.decryptBlock(block) - - return xor(out, pad) -} - -},{"buffer-xor":123}],86:[function(require,module,exports){ -var Buffer = require('safe-buffer').Buffer -var xor = require('buffer-xor') - -function encryptStart (self, data, decrypt) { - var len = data.length - var out = xor(data, self._cache) - self._cache = self._cache.slice(len) - self._prev = Buffer.concat([self._prev, decrypt ? data : out]) - return out -} - -exports.encrypt = function (self, data, decrypt) { - var out = Buffer.allocUnsafe(0) - var len - - while (data.length) { - if (self._cache.length === 0) { - self._cache = self._cipher.encryptBlock(self._prev) - self._prev = Buffer.allocUnsafe(0) - } - - if (self._cache.length <= data.length) { - len = self._cache.length - out = Buffer.concat([out, encryptStart(self, data.slice(0, len), decrypt)]) - data = data.slice(len) - } else { - out = Buffer.concat([out, encryptStart(self, data, decrypt)]) - break - } - } - - return out -} - -},{"buffer-xor":123,"safe-buffer":269}],87:[function(require,module,exports){ -var Buffer = require('safe-buffer').Buffer - -function encryptByte (self, byteParam, decrypt) { - var pad - var i = -1 - var len = 8 - var out = 0 - var bit, value - while (++i < len) { - pad = self._cipher.encryptBlock(self._prev) - bit = (byteParam & (1 << (7 - i))) ? 0x80 : 0 - value = pad[0] ^ bit - out += ((value & 0x80) >> (i % 8)) - self._prev = shiftIn(self._prev, decrypt ? bit : value) - } - return out -} - -function shiftIn (buffer, value) { - var len = buffer.length - var i = -1 - var out = Buffer.allocUnsafe(buffer.length) - buffer = Buffer.concat([buffer, Buffer.from([value])]) - - while (++i < len) { - out[i] = buffer[i] << 1 | buffer[i + 1] >> (7) - } - - return out -} - -exports.encrypt = function (self, chunk, decrypt) { - var len = chunk.length - var out = Buffer.allocUnsafe(len) - var i = -1 - - while (++i < len) { - out[i] = encryptByte(self, chunk[i], decrypt) - } - - return out -} - -},{"safe-buffer":269}],88:[function(require,module,exports){ -var Buffer = require('safe-buffer').Buffer - -function encryptByte (self, byteParam, decrypt) { - var pad = self._cipher.encryptBlock(self._prev) - var out = pad[0] ^ byteParam - - self._prev = Buffer.concat([ - self._prev.slice(1), - Buffer.from([decrypt ? byteParam : out]) - ]) - - return out -} - -exports.encrypt = function (self, chunk, decrypt) { - var len = chunk.length - var out = Buffer.allocUnsafe(len) - var i = -1 - - while (++i < len) { - out[i] = encryptByte(self, chunk[i], decrypt) - } - - return out -} - -},{"safe-buffer":269}],89:[function(require,module,exports){ -var xor = require('buffer-xor') -var Buffer = require('safe-buffer').Buffer -var incr32 = require('../incr32') - -function getBlock (self) { - var out = self._cipher.encryptBlockRaw(self._prev) - incr32(self._prev) - return out -} - -var blockSize = 16 -exports.encrypt = function (self, chunk) { - var chunkNum = Math.ceil(chunk.length / blockSize) - var start = self._cache.length - self._cache = Buffer.concat([ - self._cache, - Buffer.allocUnsafe(chunkNum * blockSize) - ]) - for (var i = 0; i < chunkNum; i++) { - var out = getBlock(self) - var offset = start + i * blockSize - self._cache.writeUInt32BE(out[0], offset + 0) - self._cache.writeUInt32BE(out[1], offset + 4) - self._cache.writeUInt32BE(out[2], offset + 8) - self._cache.writeUInt32BE(out[3], offset + 12) - } - var pad = self._cache.slice(0, chunk.length) - self._cache = self._cache.slice(chunk.length) - return xor(chunk, pad) -} - -},{"../incr32":84,"buffer-xor":123,"safe-buffer":269}],90:[function(require,module,exports){ -exports.encrypt = function (self, block) { - return self._cipher.encryptBlock(block) -} - -exports.decrypt = function (self, block) { - return self._cipher.decryptBlock(block) -} - -},{}],91:[function(require,module,exports){ -var modeModules = { - ECB: require('./ecb'), - CBC: require('./cbc'), - CFB: require('./cfb'), - CFB8: require('./cfb8'), - CFB1: require('./cfb1'), - OFB: require('./ofb'), - CTR: require('./ctr'), - GCM: require('./ctr') -} - -var modes = require('./list.json') - -for (var key in modes) { - modes[key].module = modeModules[modes[key].mode] -} - -module.exports = modes - -},{"./cbc":85,"./cfb":86,"./cfb1":87,"./cfb8":88,"./ctr":89,"./ecb":90,"./list.json":92,"./ofb":93}],92:[function(require,module,exports){ -module.exports={ - "aes-128-ecb": { - "cipher": "AES", - "key": 128, - "iv": 0, - "mode": "ECB", - "type": "block" - }, - "aes-192-ecb": { - "cipher": "AES", - "key": 192, - "iv": 0, - "mode": "ECB", - "type": "block" - }, - "aes-256-ecb": { - "cipher": "AES", - "key": 256, - "iv": 0, - "mode": "ECB", - "type": "block" - }, - "aes-128-cbc": { - "cipher": "AES", - "key": 128, - "iv": 16, - "mode": "CBC", - "type": "block" - }, - "aes-192-cbc": { - "cipher": "AES", - "key": 192, - "iv": 16, - "mode": "CBC", - "type": "block" - }, - "aes-256-cbc": { - "cipher": "AES", - "key": 256, - "iv": 16, - "mode": "CBC", - "type": "block" - }, - "aes128": { - "cipher": "AES", - "key": 128, - "iv": 16, - "mode": "CBC", - "type": "block" - }, - "aes192": { - "cipher": "AES", - "key": 192, - "iv": 16, - "mode": "CBC", - "type": "block" - }, - "aes256": { - "cipher": "AES", - "key": 256, - "iv": 16, - "mode": "CBC", - "type": "block" - }, - "aes-128-cfb": { - "cipher": "AES", - "key": 128, - "iv": 16, - "mode": "CFB", - "type": "stream" - }, - "aes-192-cfb": { - "cipher": "AES", - "key": 192, - "iv": 16, - "mode": "CFB", - "type": "stream" - }, - "aes-256-cfb": { - "cipher": "AES", - "key": 256, - "iv": 16, - "mode": "CFB", - "type": "stream" - }, - "aes-128-cfb8": { - "cipher": "AES", - "key": 128, - "iv": 16, - "mode": "CFB8", - "type": "stream" - }, - "aes-192-cfb8": { - "cipher": "AES", - "key": 192, - "iv": 16, - "mode": "CFB8", - "type": "stream" - }, - "aes-256-cfb8": { - "cipher": "AES", - "key": 256, - "iv": 16, - "mode": "CFB8", - "type": "stream" - }, - "aes-128-cfb1": { - "cipher": "AES", - "key": 128, - "iv": 16, - "mode": "CFB1", - "type": "stream" - }, - "aes-192-cfb1": { - "cipher": "AES", - "key": 192, - "iv": 16, - "mode": "CFB1", - "type": "stream" - }, - "aes-256-cfb1": { - "cipher": "AES", - "key": 256, - "iv": 16, - "mode": "CFB1", - "type": "stream" - }, - "aes-128-ofb": { - "cipher": "AES", - "key": 128, - "iv": 16, - "mode": "OFB", - "type": "stream" - }, - "aes-192-ofb": { - "cipher": "AES", - "key": 192, - "iv": 16, - "mode": "OFB", - "type": "stream" - }, - "aes-256-ofb": { - "cipher": "AES", - "key": 256, - "iv": 16, - "mode": "OFB", - "type": "stream" - }, - "aes-128-ctr": { - "cipher": "AES", - "key": 128, - "iv": 16, - "mode": "CTR", - "type": "stream" - }, - "aes-192-ctr": { - "cipher": "AES", - "key": 192, - "iv": 16, - "mode": "CTR", - "type": "stream" - }, - "aes-256-ctr": { - "cipher": "AES", - "key": 256, - "iv": 16, - "mode": "CTR", - "type": "stream" - }, - "aes-128-gcm": { - "cipher": "AES", - "key": 128, - "iv": 12, - "mode": "GCM", - "type": "auth" - }, - "aes-192-gcm": { - "cipher": "AES", - "key": 192, - "iv": 12, - "mode": "GCM", - "type": "auth" - }, - "aes-256-gcm": { - "cipher": "AES", - "key": 256, - "iv": 12, - "mode": "GCM", - "type": "auth" - } -} - -},{}],93:[function(require,module,exports){ -(function (Buffer){ -var xor = require('buffer-xor') - -function getBlock (self) { - self._prev = self._cipher.encryptBlock(self._prev) - return self._prev -} - -exports.encrypt = function (self, chunk) { - while (self._cache.length < chunk.length) { - self._cache = Buffer.concat([self._cache, getBlock(self)]) - } - - var pad = self._cache.slice(0, chunk.length) - self._cache = self._cache.slice(chunk.length) - return xor(chunk, pad) -} - -}).call(this,require("buffer").Buffer) - -},{"buffer":124,"buffer-xor":123}],94:[function(require,module,exports){ -var aes = require('./aes') -var Buffer = require('safe-buffer').Buffer -var Transform = require('cipher-base') -var inherits = require('inherits') - -function StreamCipher (mode, key, iv, decrypt) { - Transform.call(this) - - this._cipher = new aes.AES(key) - this._prev = Buffer.from(iv) - this._cache = Buffer.allocUnsafe(0) - this._secCache = Buffer.allocUnsafe(0) - this._decrypt = decrypt - this._mode = mode -} - -inherits(StreamCipher, Transform) - -StreamCipher.prototype._update = function (chunk) { - return this._mode.encrypt(this, chunk, this._decrypt) -} - -StreamCipher.prototype._final = function () { - this._cipher.scrub() -} - -module.exports = StreamCipher - -},{"./aes":78,"cipher-base":125,"inherits":218,"safe-buffer":269}],95:[function(require,module,exports){ -var DES = require('browserify-des') -var aes = require('browserify-aes/browser') -var aesModes = require('browserify-aes/modes') -var desModes = require('browserify-des/modes') -var ebtk = require('evp_bytestokey') - -function createCipher (suite, password) { - suite = suite.toLowerCase() - - var keyLen, ivLen - if (aesModes[suite]) { - keyLen = aesModes[suite].key - ivLen = aesModes[suite].iv - } else if (desModes[suite]) { - keyLen = desModes[suite].key * 8 - ivLen = desModes[suite].iv - } else { - throw new TypeError('invalid suite type') - } - - var keys = ebtk(password, false, keyLen, ivLen) - return createCipheriv(suite, keys.key, keys.iv) -} - -function createDecipher (suite, password) { - suite = suite.toLowerCase() - - var keyLen, ivLen - if (aesModes[suite]) { - keyLen = aesModes[suite].key - ivLen = aesModes[suite].iv - } else if (desModes[suite]) { - keyLen = desModes[suite].key * 8 - ivLen = desModes[suite].iv - } else { - throw new TypeError('invalid suite type') - } - - var keys = ebtk(password, false, keyLen, ivLen) - return createDecipheriv(suite, keys.key, keys.iv) -} - -function createCipheriv (suite, key, iv) { - suite = suite.toLowerCase() - if (aesModes[suite]) return aes.createCipheriv(suite, key, iv) - if (desModes[suite]) return new DES({ key: key, iv: iv, mode: suite }) - - throw new TypeError('invalid suite type') -} - -function createDecipheriv (suite, key, iv) { - suite = suite.toLowerCase() - if (aesModes[suite]) return aes.createDecipheriv(suite, key, iv) - if (desModes[suite]) return new DES({ key: key, iv: iv, mode: suite, decrypt: true }) - - throw new TypeError('invalid suite type') -} - -function getCiphers () { - return Object.keys(desModes).concat(aes.getCiphers()) -} - -exports.createCipher = exports.Cipher = createCipher -exports.createCipheriv = exports.Cipheriv = createCipheriv -exports.createDecipher = exports.Decipher = createDecipher -exports.createDecipheriv = exports.Decipheriv = createDecipheriv -exports.listCiphers = exports.getCiphers = getCiphers - -},{"browserify-aes/browser":80,"browserify-aes/modes":91,"browserify-des":96,"browserify-des/modes":97,"evp_bytestokey":182}],96:[function(require,module,exports){ -var CipherBase = require('cipher-base') -var des = require('des.js') -var inherits = require('inherits') -var Buffer = require('safe-buffer').Buffer - -var modes = { - 'des-ede3-cbc': des.CBC.instantiate(des.EDE), - 'des-ede3': des.EDE, - 'des-ede-cbc': des.CBC.instantiate(des.EDE), - 'des-ede': des.EDE, - 'des-cbc': des.CBC.instantiate(des.DES), - 'des-ecb': des.DES -} -modes.des = modes['des-cbc'] -modes.des3 = modes['des-ede3-cbc'] -module.exports = DES -inherits(DES, CipherBase) -function DES (opts) { - CipherBase.call(this) - var modeName = opts.mode.toLowerCase() - var mode = modes[modeName] - var type - if (opts.decrypt) { - type = 'decrypt' - } else { - type = 'encrypt' - } - var key = opts.key - if (!Buffer.isBuffer(key)) { - key = Buffer.from(key) - } - if (modeName === 'des-ede' || modeName === 'des-ede-cbc') { - key = Buffer.concat([key, key.slice(0, 8)]) - } - var iv = opts.iv - if (!Buffer.isBuffer(iv)) { - iv = Buffer.from(iv) - } - this._des = mode.create({ - key: key, - iv: iv, - type: type - }) -} -DES.prototype._update = function (data) { - return Buffer.from(this._des.update(data)) -} -DES.prototype._final = function () { - return Buffer.from(this._des.final()) -} - -},{"cipher-base":125,"des.js":137,"inherits":218,"safe-buffer":269}],97:[function(require,module,exports){ -exports['des-ecb'] = { - key: 8, - iv: 0 -} -exports['des-cbc'] = exports.des = { - key: 8, - iv: 8 -} -exports['des-ede3-cbc'] = exports.des3 = { - key: 24, - iv: 8 -} -exports['des-ede3'] = { - key: 24, - iv: 0 -} -exports['des-ede-cbc'] = { - key: 16, - iv: 8 -} -exports['des-ede'] = { - key: 16, - iv: 0 -} - -},{}],98:[function(require,module,exports){ -(function (Buffer){ -var bn = require('bn.js'); -var randomBytes = require('randombytes'); -module.exports = crt; -function blind(priv) { - var r = getr(priv); - var blinder = r.toRed(bn.mont(priv.modulus)) - .redPow(new bn(priv.publicExponent)).fromRed(); - return { - blinder: blinder, - unblinder:r.invm(priv.modulus) - }; -} -function crt(msg, priv) { - var blinds = blind(priv); - var len = priv.modulus.byteLength(); - var mod = bn.mont(priv.modulus); - var blinded = new bn(msg).mul(blinds.blinder).umod(priv.modulus); - var c1 = blinded.toRed(bn.mont(priv.prime1)); - var c2 = blinded.toRed(bn.mont(priv.prime2)); - var qinv = priv.coefficient; - var p = priv.prime1; - var q = priv.prime2; - var m1 = c1.redPow(priv.exponent1); - var m2 = c2.redPow(priv.exponent2); - m1 = m1.fromRed(); - m2 = m2.fromRed(); - var h = m1.isub(m2).imul(qinv).umod(p); - h.imul(q); - m2.iadd(h); - return new Buffer(m2.imul(blinds.unblinder).umod(priv.modulus).toArray(false, len)); -} -crt.getr = getr; -function getr(priv) { - var len = priv.modulus.byteLength(); - var r = new bn(randomBytes(len)); - while (r.cmp(priv.modulus) >= 0 || !r.umod(priv.prime1) || !r.umod(priv.prime2)) { - r = new bn(randomBytes(len)); - } - return r; -} - -}).call(this,require("buffer").Buffer) - -},{"bn.js":99,"buffer":124,"randombytes":252}],99:[function(require,module,exports){ -arguments[4][70][0].apply(exports,arguments) -},{"buffer":77,"dup":70}],100:[function(require,module,exports){ -module.exports = require('./browser/algorithms.json') - -},{"./browser/algorithms.json":101}],101:[function(require,module,exports){ -module.exports={ - "sha224WithRSAEncryption": { - "sign": "rsa", - "hash": "sha224", - "id": "302d300d06096086480165030402040500041c" - }, - "RSA-SHA224": { - "sign": "ecdsa/rsa", - "hash": "sha224", - "id": "302d300d06096086480165030402040500041c" - }, - "sha256WithRSAEncryption": { - "sign": "rsa", - "hash": "sha256", - "id": "3031300d060960864801650304020105000420" - }, - "RSA-SHA256": { - "sign": "ecdsa/rsa", - "hash": "sha256", - "id": "3031300d060960864801650304020105000420" - }, - "sha384WithRSAEncryption": { - "sign": "rsa", - "hash": "sha384", - "id": "3041300d060960864801650304020205000430" - }, - "RSA-SHA384": { - "sign": "ecdsa/rsa", - "hash": "sha384", - "id": "3041300d060960864801650304020205000430" - }, - "sha512WithRSAEncryption": { - "sign": "rsa", - "hash": "sha512", - "id": "3051300d060960864801650304020305000440" - }, - "RSA-SHA512": { - "sign": "ecdsa/rsa", - "hash": "sha512", - "id": "3051300d060960864801650304020305000440" - }, - "RSA-SHA1": { - "sign": "rsa", - "hash": "sha1", - "id": "3021300906052b0e03021a05000414" - }, - "ecdsa-with-SHA1": { - "sign": "ecdsa", - "hash": "sha1", - "id": "" - }, - "sha256": { - "sign": "ecdsa", - "hash": "sha256", - "id": "" - }, - "sha224": { - "sign": "ecdsa", - "hash": "sha224", - "id": "" - }, - "sha384": { - "sign": "ecdsa", - "hash": "sha384", - "id": "" - }, - "sha512": { - "sign": "ecdsa", - "hash": "sha512", - "id": "" - }, - "DSA-SHA": { - "sign": "dsa", - "hash": "sha1", - "id": "" - }, - "DSA-SHA1": { - "sign": "dsa", - "hash": "sha1", - "id": "" - }, - "DSA": { - "sign": "dsa", - "hash": "sha1", - "id": "" - }, - "DSA-WITH-SHA224": { - "sign": "dsa", - "hash": "sha224", - "id": "" - }, - "DSA-SHA224": { - "sign": "dsa", - "hash": "sha224", - "id": "" - }, - "DSA-WITH-SHA256": { - "sign": "dsa", - "hash": "sha256", - "id": "" - }, - "DSA-SHA256": { - "sign": "dsa", - "hash": "sha256", - "id": "" - }, - "DSA-WITH-SHA384": { - "sign": "dsa", - "hash": "sha384", - "id": "" - }, - "DSA-SHA384": { - "sign": "dsa", - "hash": "sha384", - "id": "" - }, - "DSA-WITH-SHA512": { - "sign": "dsa", - "hash": "sha512", - "id": "" - }, - "DSA-SHA512": { - "sign": "dsa", - "hash": "sha512", - "id": "" - }, - "DSA-RIPEMD160": { - "sign": "dsa", - "hash": "rmd160", - "id": "" - }, - "ripemd160WithRSA": { - "sign": "rsa", - "hash": "rmd160", - "id": "3021300906052b2403020105000414" - }, - "RSA-RIPEMD160": { - "sign": "rsa", - "hash": "rmd160", - "id": "3021300906052b2403020105000414" - }, - "md5WithRSAEncryption": { - "sign": "rsa", - "hash": "md5", - "id": "3020300c06082a864886f70d020505000410" - }, - "RSA-MD5": { - "sign": "rsa", - "hash": "md5", - "id": "3020300c06082a864886f70d020505000410" - } -} - -},{}],102:[function(require,module,exports){ -module.exports={ - "1.3.132.0.10": "secp256k1", - "1.3.132.0.33": "p224", - "1.2.840.10045.3.1.1": "p192", - "1.2.840.10045.3.1.7": "p256", - "1.3.132.0.34": "p384", - "1.3.132.0.35": "p521" -} - -},{}],103:[function(require,module,exports){ -var Buffer = require('safe-buffer').Buffer -var createHash = require('create-hash') -var stream = require('readable-stream') -var inherits = require('inherits') -var sign = require('./sign') -var verify = require('./verify') - -var algorithms = require('./algorithms.json') -Object.keys(algorithms).forEach(function (key) { - algorithms[key].id = Buffer.from(algorithms[key].id, 'hex') - algorithms[key.toLowerCase()] = algorithms[key] -}) - -function Sign (algorithm) { - stream.Writable.call(this) - - var data = algorithms[algorithm] - if (!data) throw new Error('Unknown message digest') - - this._hashType = data.hash - this._hash = createHash(data.hash) - this._tag = data.id - this._signType = data.sign -} -inherits(Sign, stream.Writable) - -Sign.prototype._write = function _write (data, _, done) { - this._hash.update(data) - done() -} - -Sign.prototype.update = function update (data, enc) { - if (typeof data === 'string') data = Buffer.from(data, enc) - - this._hash.update(data) - return this -} - -Sign.prototype.sign = function signMethod (key, enc) { - this.end() - var hash = this._hash.digest() - var sig = sign(hash, key, this._hashType, this._signType, this._tag) - - return enc ? sig.toString(enc) : sig -} - -function Verify (algorithm) { - stream.Writable.call(this) - - var data = algorithms[algorithm] - if (!data) throw new Error('Unknown message digest') - - this._hash = createHash(data.hash) - this._tag = data.id - this._signType = data.sign -} -inherits(Verify, stream.Writable) - -Verify.prototype._write = function _write (data, _, done) { - this._hash.update(data) - done() -} - -Verify.prototype.update = function update (data, enc) { - if (typeof data === 'string') data = Buffer.from(data, enc) - - this._hash.update(data) - return this -} - -Verify.prototype.verify = function verifyMethod (key, sig, enc) { - if (typeof sig === 'string') sig = Buffer.from(sig, enc) - - this.end() - var hash = this._hash.digest() - return verify(sig, hash, key, this._signType, this._tag) -} - -function createSign (algorithm) { - return new Sign(algorithm) -} - -function createVerify (algorithm) { - return new Verify(algorithm) -} - -module.exports = { - Sign: createSign, - Verify: createVerify, - createSign: createSign, - createVerify: createVerify -} - -},{"./algorithms.json":101,"./sign":104,"./verify":105,"create-hash":132,"inherits":106,"readable-stream":121,"safe-buffer":122}],104:[function(require,module,exports){ -// much of this based on https://github.com/indutny/self-signed/blob/gh-pages/lib/rsa.js -var Buffer = require('safe-buffer').Buffer -var createHmac = require('create-hmac') -var crt = require('browserify-rsa') -var EC = require('elliptic').ec -var BN = require('bn.js') -var parseKeys = require('parse-asn1') -var curves = require('./curves.json') - -function sign (hash, key, hashType, signType, tag) { - var priv = parseKeys(key) - if (priv.curve) { - // rsa keys can be interpreted as ecdsa ones in openssl - if (signType !== 'ecdsa' && signType !== 'ecdsa/rsa') throw new Error('wrong private key type') - return ecSign(hash, priv) - } else if (priv.type === 'dsa') { - if (signType !== 'dsa') throw new Error('wrong private key type') - return dsaSign(hash, priv, hashType) - } else { - if (signType !== 'rsa' && signType !== 'ecdsa/rsa') throw new Error('wrong private key type') - } - hash = Buffer.concat([tag, hash]) - var len = priv.modulus.byteLength() - var pad = [0, 1] - while (hash.length + pad.length + 1 < len) pad.push(0xff) - pad.push(0x00) - var i = -1 - while (++i < hash.length) pad.push(hash[i]) - - var out = crt(pad, priv) - return out -} - -function ecSign (hash, priv) { - var curveId = curves[priv.curve.join('.')] - if (!curveId) throw new Error('unknown curve ' + priv.curve.join('.')) - - var curve = new EC(curveId) - var key = curve.keyFromPrivate(priv.privateKey) - var out = key.sign(hash) - - return Buffer.from(out.toDER()) -} - -function dsaSign (hash, priv, algo) { - var x = priv.params.priv_key - var p = priv.params.p - var q = priv.params.q - var g = priv.params.g - var r = new BN(0) - var k - var H = bits2int(hash, q).mod(q) - var s = false - var kv = getKey(x, q, hash, algo) - while (s === false) { - k = makeKey(q, kv, algo) - r = makeR(g, k, p, q) - s = k.invm(q).imul(H.add(x.mul(r))).mod(q) - if (s.cmpn(0) === 0) { - s = false - r = new BN(0) - } - } - return toDER(r, s) -} - -function toDER (r, s) { - r = r.toArray() - s = s.toArray() - - // Pad values - if (r[0] & 0x80) r = [0].concat(r) - if (s[0] & 0x80) s = [0].concat(s) - - var total = r.length + s.length + 4 - var res = [0x30, total, 0x02, r.length] - res = res.concat(r, [0x02, s.length], s) - return Buffer.from(res) -} - -function getKey (x, q, hash, algo) { - x = Buffer.from(x.toArray()) - if (x.length < q.byteLength()) { - var zeros = Buffer.alloc(q.byteLength() - x.length) - x = Buffer.concat([zeros, x]) - } - var hlen = hash.length - var hbits = bits2octets(hash, q) - var v = Buffer.alloc(hlen) - v.fill(1) - var k = Buffer.alloc(hlen) - k = createHmac(algo, k).update(v).update(Buffer.from([0])).update(x).update(hbits).digest() - v = createHmac(algo, k).update(v).digest() - k = createHmac(algo, k).update(v).update(Buffer.from([1])).update(x).update(hbits).digest() - v = createHmac(algo, k).update(v).digest() - return { k: k, v: v } -} - -function bits2int (obits, q) { - var bits = new BN(obits) - var shift = (obits.length << 3) - q.bitLength() - if (shift > 0) bits.ishrn(shift) - return bits -} - -function bits2octets (bits, q) { - bits = bits2int(bits, q) - bits = bits.mod(q) - var out = Buffer.from(bits.toArray()) - if (out.length < q.byteLength()) { - var zeros = Buffer.alloc(q.byteLength() - out.length) - out = Buffer.concat([zeros, out]) - } - return out -} - -function makeKey (q, kv, algo) { - var t - var k - - do { - t = Buffer.alloc(0) - - while (t.length * 8 < q.bitLength()) { - kv.v = createHmac(algo, kv.k).update(kv.v).digest() - t = Buffer.concat([t, kv.v]) - } - - k = bits2int(t, q) - kv.k = createHmac(algo, kv.k).update(kv.v).update(Buffer.from([0])).digest() - kv.v = createHmac(algo, kv.k).update(kv.v).digest() - } while (k.cmp(q) !== -1) - - return k -} - -function makeR (g, k, p, q) { - return g.toRed(BN.mont(p)).redPow(k).fromRed().mod(q) -} - -module.exports = sign -module.exports.getKey = getKey -module.exports.makeKey = makeKey - -},{"./curves.json":102,"bn.js":75,"browserify-rsa":98,"create-hmac":134,"elliptic":148,"parse-asn1":233,"safe-buffer":122}],105:[function(require,module,exports){ -// much of this based on https://github.com/indutny/self-signed/blob/gh-pages/lib/rsa.js -var Buffer = require('safe-buffer').Buffer -var BN = require('bn.js') -var EC = require('elliptic').ec -var parseKeys = require('parse-asn1') -var curves = require('./curves.json') - -function verify (sig, hash, key, signType, tag) { - var pub = parseKeys(key) - if (pub.type === 'ec') { - // rsa keys can be interpreted as ecdsa ones in openssl - if (signType !== 'ecdsa' && signType !== 'ecdsa/rsa') throw new Error('wrong public key type') - return ecVerify(sig, hash, pub) - } else if (pub.type === 'dsa') { - if (signType !== 'dsa') throw new Error('wrong public key type') - return dsaVerify(sig, hash, pub) - } else { - if (signType !== 'rsa' && signType !== 'ecdsa/rsa') throw new Error('wrong public key type') - } - hash = Buffer.concat([tag, hash]) - var len = pub.modulus.byteLength() - var pad = [1] - var padNum = 0 - while (hash.length + pad.length + 2 < len) { - pad.push(0xff) - padNum++ - } - pad.push(0x00) - var i = -1 - while (++i < hash.length) { - pad.push(hash[i]) - } - pad = Buffer.from(pad) - var red = BN.mont(pub.modulus) - sig = new BN(sig).toRed(red) - - sig = sig.redPow(new BN(pub.publicExponent)) - sig = Buffer.from(sig.fromRed().toArray()) - var out = padNum < 8 ? 1 : 0 - len = Math.min(sig.length, pad.length) - if (sig.length !== pad.length) out = 1 - - i = -1 - while (++i < len) out |= sig[i] ^ pad[i] - return out === 0 -} - -function ecVerify (sig, hash, pub) { - var curveId = curves[pub.data.algorithm.curve.join('.')] - if (!curveId) throw new Error('unknown curve ' + pub.data.algorithm.curve.join('.')) - - var curve = new EC(curveId) - var pubkey = pub.data.subjectPrivateKey.data - - return curve.verify(hash, sig, pubkey) -} - -function dsaVerify (sig, hash, pub) { - var p = pub.data.p - var q = pub.data.q - var g = pub.data.g - var y = pub.data.pub_key - var unpacked = parseKeys.signature.decode(sig, 'der') - var s = unpacked.s - var r = unpacked.r - checkValue(s, q) - checkValue(r, q) - var montp = BN.mont(p) - var w = s.invm(q) - var v = g.toRed(montp) - .redPow(new BN(hash).mul(w).mod(q)) - .fromRed() - .mul(y.toRed(montp).redPow(r.mul(w).mod(q)).fromRed()) - .mod(p) - .mod(q) - return v.cmp(r) === 0 -} - -function checkValue (b, q) { - if (b.cmpn(0) <= 0) throw new Error('invalid sig') - if (b.cmp(q) >= q) throw new Error('invalid sig') -} - -module.exports = verify - -},{"./curves.json":102,"bn.js":75,"elliptic":148,"parse-asn1":233,"safe-buffer":122}],106:[function(require,module,exports){ -if (typeof Object.create === 'function') { - // implementation from standard node.js 'util' module - module.exports = function inherits(ctor, superCtor) { - if (superCtor) { - ctor.super_ = superCtor - ctor.prototype = Object.create(superCtor.prototype, { - constructor: { - value: ctor, - enumerable: false, - writable: true, - configurable: true - } - }) - } - }; -} else { - // old school shim for old browsers - module.exports = function inherits(ctor, superCtor) { - if (superCtor) { - ctor.super_ = superCtor - var TempCtor = function () {} - TempCtor.prototype = superCtor.prototype - ctor.prototype = new TempCtor() - ctor.prototype.constructor = ctor - } - } -} - -},{}],107:[function(require,module,exports){ -'use strict'; - -function _inheritsLoose(subClass, superClass) { subClass.prototype = Object.create(superClass.prototype); subClass.prototype.constructor = subClass; subClass.__proto__ = superClass; } - -var codes = {}; - -function createErrorType(code, message, Base) { - if (!Base) { - Base = Error; - } - - function getMessage(arg1, arg2, arg3) { - if (typeof message === 'string') { - return message; - } else { - return message(arg1, arg2, arg3); - } - } - - var NodeError = - /*#__PURE__*/ - function (_Base) { - _inheritsLoose(NodeError, _Base); - - function NodeError(arg1, arg2, arg3) { - return _Base.call(this, getMessage(arg1, arg2, arg3)) || this; - } - - return NodeError; - }(Base); - - NodeError.prototype.name = Base.name; - NodeError.prototype.code = code; - codes[code] = NodeError; -} // https://github.com/nodejs/node/blob/v10.8.0/lib/internal/errors.js - - -function oneOf(expected, thing) { - if (Array.isArray(expected)) { - var len = expected.length; - expected = expected.map(function (i) { - return String(i); - }); - - if (len > 2) { - return "one of ".concat(thing, " ").concat(expected.slice(0, len - 1).join(', '), ", or ") + expected[len - 1]; - } else if (len === 2) { - return "one of ".concat(thing, " ").concat(expected[0], " or ").concat(expected[1]); - } else { - return "of ".concat(thing, " ").concat(expected[0]); - } - } else { - return "of ".concat(thing, " ").concat(String(expected)); - } -} // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/startsWith - - -function startsWith(str, search, pos) { - return str.substr(!pos || pos < 0 ? 0 : +pos, search.length) === search; -} // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/endsWith - - -function endsWith(str, search, this_len) { - if (this_len === undefined || this_len > str.length) { - this_len = str.length; - } - - return str.substring(this_len - search.length, this_len) === search; -} // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/includes - - -function includes(str, search, start) { - if (typeof start !== 'number') { - start = 0; - } - - if (start + search.length > str.length) { - return false; - } else { - return str.indexOf(search, start) !== -1; - } -} - -createErrorType('ERR_INVALID_OPT_VALUE', function (name, value) { - return 'The value "' + value + '" is invalid for option "' + name + '"'; -}, TypeError); -createErrorType('ERR_INVALID_ARG_TYPE', function (name, expected, actual) { - // determiner: 'must be' or 'must not be' - var determiner; - - if (typeof expected === 'string' && startsWith(expected, 'not ')) { - determiner = 'must not be'; - expected = expected.replace(/^not /, ''); - } else { - determiner = 'must be'; - } - - var msg; - - if (endsWith(name, ' argument')) { - // For cases like 'first argument' - msg = "The ".concat(name, " ").concat(determiner, " ").concat(oneOf(expected, 'type')); - } else { - var type = includes(name, '.') ? 'property' : 'argument'; - msg = "The \"".concat(name, "\" ").concat(type, " ").concat(determiner, " ").concat(oneOf(expected, 'type')); - } - - msg += ". Received type ".concat(typeof actual); - return msg; -}, TypeError); -createErrorType('ERR_STREAM_PUSH_AFTER_EOF', 'stream.push() after EOF'); -createErrorType('ERR_METHOD_NOT_IMPLEMENTED', function (name) { - return 'The ' + name + ' method is not implemented'; -}); -createErrorType('ERR_STREAM_PREMATURE_CLOSE', 'Premature close'); -createErrorType('ERR_STREAM_DESTROYED', function (name) { - return 'Cannot call ' + name + ' after a stream was destroyed'; -}); -createErrorType('ERR_MULTIPLE_CALLBACK', 'Callback called multiple times'); -createErrorType('ERR_STREAM_CANNOT_PIPE', 'Cannot pipe, not readable'); -createErrorType('ERR_STREAM_WRITE_AFTER_END', 'write after end'); -createErrorType('ERR_STREAM_NULL_VALUES', 'May not write null values to stream', TypeError); -createErrorType('ERR_UNKNOWN_ENCODING', function (arg) { - return 'Unknown encoding: ' + arg; -}, TypeError); -createErrorType('ERR_STREAM_UNSHIFT_AFTER_END_EVENT', 'stream.unshift() after end event'); -module.exports.codes = codes; - -},{}],108:[function(require,module,exports){ -(function (process){ -// Copyright Joyent, Inc. and other Node contributors. -// -// Permission is hereby granted, free of charge, to any person obtaining a -// copy of this software and associated documentation files (the -// "Software"), to deal in the Software without restriction, including -// without limitation the rights to use, copy, modify, merge, publish, -// distribute, sublicense, and/or sell copies of the Software, and to permit -// persons to whom the Software is furnished to do so, subject to the -// following conditions: -// -// The above copyright notice and this permission notice shall be included -// in all copies or substantial portions of the Software. -// -// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS -// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN -// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, -// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR -// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE -// USE OR OTHER DEALINGS IN THE SOFTWARE. -// a duplex stream is just a stream that is both readable and writable. -// Since JS doesn't have multiple prototypal inheritance, this class -// prototypally inherits from Readable, and then parasitically from -// Writable. -'use strict'; -/**/ - -var objectKeys = Object.keys || function (obj) { - var keys = []; - - for (var key in obj) { - keys.push(key); - } - - return keys; -}; -/**/ - - -module.exports = Duplex; - -var Readable = require('./_stream_readable'); - -var Writable = require('./_stream_writable'); - -require('inherits')(Duplex, Readable); - -{ - // Allow the keys array to be GC'ed. - var keys = objectKeys(Writable.prototype); - - for (var v = 0; v < keys.length; v++) { - var method = keys[v]; - if (!Duplex.prototype[method]) Duplex.prototype[method] = Writable.prototype[method]; - } -} - -function Duplex(options) { - if (!(this instanceof Duplex)) return new Duplex(options); - Readable.call(this, options); - Writable.call(this, options); - this.allowHalfOpen = true; - - if (options) { - if (options.readable === false) this.readable = false; - if (options.writable === false) this.writable = false; - - if (options.allowHalfOpen === false) { - this.allowHalfOpen = false; - this.once('end', onend); - } - } -} - -Object.defineProperty(Duplex.prototype, 'writableHighWaterMark', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - return this._writableState.highWaterMark; - } -}); -Object.defineProperty(Duplex.prototype, 'writableBuffer', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - return this._writableState && this._writableState.getBuffer(); - } -}); -Object.defineProperty(Duplex.prototype, 'writableLength', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - return this._writableState.length; - } -}); // the no-half-open enforcer - -function onend() { - // If the writable side ended, then we're ok. - if (this._writableState.ended) return; // no more data can be written. - // But allow more writes to happen in this tick. - - process.nextTick(onEndNT, this); -} - -function onEndNT(self) { - self.end(); -} - -Object.defineProperty(Duplex.prototype, 'destroyed', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - if (this._readableState === undefined || this._writableState === undefined) { - return false; - } - - return this._readableState.destroyed && this._writableState.destroyed; - }, - set: function set(value) { - // we ignore the value if the stream - // has not been initialized yet - if (this._readableState === undefined || this._writableState === undefined) { - return; - } // backward compatibility, the user is explicitly - // managing destroyed - - - this._readableState.destroyed = value; - this._writableState.destroyed = value; - } -}); -}).call(this,require('_process')) - -},{"./_stream_readable":110,"./_stream_writable":112,"_process":244,"inherits":106}],109:[function(require,module,exports){ -// Copyright Joyent, Inc. and other Node contributors. -// -// Permission is hereby granted, free of charge, to any person obtaining a -// copy of this software and associated documentation files (the -// "Software"), to deal in the Software without restriction, including -// without limitation the rights to use, copy, modify, merge, publish, -// distribute, sublicense, and/or sell copies of the Software, and to permit -// persons to whom the Software is furnished to do so, subject to the -// following conditions: -// -// The above copyright notice and this permission notice shall be included -// in all copies or substantial portions of the Software. -// -// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS -// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN -// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, -// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR -// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE -// USE OR OTHER DEALINGS IN THE SOFTWARE. -// a passthrough stream. -// basically just the most minimal sort of Transform stream. -// Every written chunk gets output as-is. -'use strict'; - -module.exports = PassThrough; - -var Transform = require('./_stream_transform'); - -require('inherits')(PassThrough, Transform); - -function PassThrough(options) { - if (!(this instanceof PassThrough)) return new PassThrough(options); - Transform.call(this, options); -} - -PassThrough.prototype._transform = function (chunk, encoding, cb) { - cb(null, chunk); -}; -},{"./_stream_transform":111,"inherits":106}],110:[function(require,module,exports){ -(function (process,global){ -// Copyright Joyent, Inc. and other Node contributors. -// -// Permission is hereby granted, free of charge, to any person obtaining a -// copy of this software and associated documentation files (the -// "Software"), to deal in the Software without restriction, including -// without limitation the rights to use, copy, modify, merge, publish, -// distribute, sublicense, and/or sell copies of the Software, and to permit -// persons to whom the Software is furnished to do so, subject to the -// following conditions: -// -// The above copyright notice and this permission notice shall be included -// in all copies or substantial portions of the Software. -// -// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS -// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN -// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, -// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR -// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE -// USE OR OTHER DEALINGS IN THE SOFTWARE. -'use strict'; - -module.exports = Readable; -/**/ - -var Duplex; -/**/ - -Readable.ReadableState = ReadableState; -/**/ - -var EE = require('events').EventEmitter; - -var EElistenerCount = function EElistenerCount(emitter, type) { - return emitter.listeners(type).length; -}; -/**/ - -/**/ - - -var Stream = require('./internal/streams/stream'); -/**/ - - -var Buffer = require('buffer').Buffer; - -var OurUint8Array = global.Uint8Array || function () {}; - -function _uint8ArrayToBuffer(chunk) { - return Buffer.from(chunk); -} - -function _isUint8Array(obj) { - return Buffer.isBuffer(obj) || obj instanceof OurUint8Array; -} -/**/ - - -var debugUtil = require('util'); - -var debug; - -if (debugUtil && debugUtil.debuglog) { - debug = debugUtil.debuglog('stream'); -} else { - debug = function debug() {}; -} -/**/ - - -var BufferList = require('./internal/streams/buffer_list'); - -var destroyImpl = require('./internal/streams/destroy'); - -var _require = require('./internal/streams/state'), - getHighWaterMark = _require.getHighWaterMark; - -var _require$codes = require('../errors').codes, - ERR_INVALID_ARG_TYPE = _require$codes.ERR_INVALID_ARG_TYPE, - ERR_STREAM_PUSH_AFTER_EOF = _require$codes.ERR_STREAM_PUSH_AFTER_EOF, - ERR_METHOD_NOT_IMPLEMENTED = _require$codes.ERR_METHOD_NOT_IMPLEMENTED, - ERR_STREAM_UNSHIFT_AFTER_END_EVENT = _require$codes.ERR_STREAM_UNSHIFT_AFTER_END_EVENT; // Lazy loaded to improve the startup performance. - - -var StringDecoder; -var createReadableStreamAsyncIterator; -var from; - -require('inherits')(Readable, Stream); - -var errorOrDestroy = destroyImpl.errorOrDestroy; -var kProxyEvents = ['error', 'close', 'destroy', 'pause', 'resume']; - -function prependListener(emitter, event, fn) { - // Sadly this is not cacheable as some libraries bundle their own - // event emitter implementation with them. - if (typeof emitter.prependListener === 'function') return emitter.prependListener(event, fn); // This is a hack to make sure that our error handler is attached before any - // userland ones. NEVER DO THIS. This is here only because this code needs - // to continue to work with older versions of Node.js that do not include - // the prependListener() method. The goal is to eventually remove this hack. - - if (!emitter._events || !emitter._events[event]) emitter.on(event, fn);else if (Array.isArray(emitter._events[event])) emitter._events[event].unshift(fn);else emitter._events[event] = [fn, emitter._events[event]]; -} - -function ReadableState(options, stream, isDuplex) { - Duplex = Duplex || require('./_stream_duplex'); - options = options || {}; // Duplex streams are both readable and writable, but share - // the same options object. - // However, some cases require setting options to different - // values for the readable and the writable sides of the duplex stream. - // These options can be provided separately as readableXXX and writableXXX. - - if (typeof isDuplex !== 'boolean') isDuplex = stream instanceof Duplex; // object stream flag. Used to make read(n) ignore n and to - // make all the buffer merging and length checks go away - - this.objectMode = !!options.objectMode; - if (isDuplex) this.objectMode = this.objectMode || !!options.readableObjectMode; // the point at which it stops calling _read() to fill the buffer - // Note: 0 is a valid value, means "don't call _read preemptively ever" - - this.highWaterMark = getHighWaterMark(this, options, 'readableHighWaterMark', isDuplex); // A linked list is used to store data chunks instead of an array because the - // linked list can remove elements from the beginning faster than - // array.shift() - - this.buffer = new BufferList(); - this.length = 0; - this.pipes = null; - this.pipesCount = 0; - this.flowing = null; - this.ended = false; - this.endEmitted = false; - this.reading = false; // a flag to be able to tell if the event 'readable'/'data' is emitted - // immediately, or on a later tick. We set this to true at first, because - // any actions that shouldn't happen until "later" should generally also - // not happen before the first read call. - - this.sync = true; // whenever we return null, then we set a flag to say - // that we're awaiting a 'readable' event emission. - - this.needReadable = false; - this.emittedReadable = false; - this.readableListening = false; - this.resumeScheduled = false; - this.paused = true; // Should close be emitted on destroy. Defaults to true. - - this.emitClose = options.emitClose !== false; // Should .destroy() be called after 'end' (and potentially 'finish') - - this.autoDestroy = !!options.autoDestroy; // has it been destroyed - - this.destroyed = false; // Crypto is kind of old and crusty. Historically, its default string - // encoding is 'binary' so we have to make this configurable. - // Everything else in the universe uses 'utf8', though. - - this.defaultEncoding = options.defaultEncoding || 'utf8'; // the number of writers that are awaiting a drain event in .pipe()s - - this.awaitDrain = 0; // if true, a maybeReadMore has been scheduled - - this.readingMore = false; - this.decoder = null; - this.encoding = null; - - if (options.encoding) { - if (!StringDecoder) StringDecoder = require('string_decoder/').StringDecoder; - this.decoder = new StringDecoder(options.encoding); - this.encoding = options.encoding; - } -} - -function Readable(options) { - Duplex = Duplex || require('./_stream_duplex'); - if (!(this instanceof Readable)) return new Readable(options); // Checking for a Stream.Duplex instance is faster here instead of inside - // the ReadableState constructor, at least with V8 6.5 - - var isDuplex = this instanceof Duplex; - this._readableState = new ReadableState(options, this, isDuplex); // legacy - - this.readable = true; - - if (options) { - if (typeof options.read === 'function') this._read = options.read; - if (typeof options.destroy === 'function') this._destroy = options.destroy; - } - - Stream.call(this); -} - -Object.defineProperty(Readable.prototype, 'destroyed', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - if (this._readableState === undefined) { - return false; - } - - return this._readableState.destroyed; - }, - set: function set(value) { - // we ignore the value if the stream - // has not been initialized yet - if (!this._readableState) { - return; - } // backward compatibility, the user is explicitly - // managing destroyed - - - this._readableState.destroyed = value; - } -}); -Readable.prototype.destroy = destroyImpl.destroy; -Readable.prototype._undestroy = destroyImpl.undestroy; - -Readable.prototype._destroy = function (err, cb) { - cb(err); -}; // Manually shove something into the read() buffer. -// This returns true if the highWaterMark has not been hit yet, -// similar to how Writable.write() returns true if you should -// write() some more. - - -Readable.prototype.push = function (chunk, encoding) { - var state = this._readableState; - var skipChunkCheck; - - if (!state.objectMode) { - if (typeof chunk === 'string') { - encoding = encoding || state.defaultEncoding; - - if (encoding !== state.encoding) { - chunk = Buffer.from(chunk, encoding); - encoding = ''; - } - - skipChunkCheck = true; - } - } else { - skipChunkCheck = true; - } - - return readableAddChunk(this, chunk, encoding, false, skipChunkCheck); -}; // Unshift should *always* be something directly out of read() - - -Readable.prototype.unshift = function (chunk) { - return readableAddChunk(this, chunk, null, true, false); -}; - -function readableAddChunk(stream, chunk, encoding, addToFront, skipChunkCheck) { - debug('readableAddChunk', chunk); - var state = stream._readableState; - - if (chunk === null) { - state.reading = false; - onEofChunk(stream, state); - } else { - var er; - if (!skipChunkCheck) er = chunkInvalid(state, chunk); - - if (er) { - errorOrDestroy(stream, er); - } else if (state.objectMode || chunk && chunk.length > 0) { - if (typeof chunk !== 'string' && !state.objectMode && Object.getPrototypeOf(chunk) !== Buffer.prototype) { - chunk = _uint8ArrayToBuffer(chunk); - } - - if (addToFront) { - if (state.endEmitted) errorOrDestroy(stream, new ERR_STREAM_UNSHIFT_AFTER_END_EVENT());else addChunk(stream, state, chunk, true); - } else if (state.ended) { - errorOrDestroy(stream, new ERR_STREAM_PUSH_AFTER_EOF()); - } else if (state.destroyed) { - return false; - } else { - state.reading = false; - - if (state.decoder && !encoding) { - chunk = state.decoder.write(chunk); - if (state.objectMode || chunk.length !== 0) addChunk(stream, state, chunk, false);else maybeReadMore(stream, state); - } else { - addChunk(stream, state, chunk, false); - } - } - } else if (!addToFront) { - state.reading = false; - maybeReadMore(stream, state); - } - } // We can push more data if we are below the highWaterMark. - // Also, if we have no data yet, we can stand some more bytes. - // This is to work around cases where hwm=0, such as the repl. - - - return !state.ended && (state.length < state.highWaterMark || state.length === 0); -} - -function addChunk(stream, state, chunk, addToFront) { - if (state.flowing && state.length === 0 && !state.sync) { - state.awaitDrain = 0; - stream.emit('data', chunk); - } else { - // update the buffer info. - state.length += state.objectMode ? 1 : chunk.length; - if (addToFront) state.buffer.unshift(chunk);else state.buffer.push(chunk); - if (state.needReadable) emitReadable(stream); - } - - maybeReadMore(stream, state); -} - -function chunkInvalid(state, chunk) { - var er; - - if (!_isUint8Array(chunk) && typeof chunk !== 'string' && chunk !== undefined && !state.objectMode) { - er = new ERR_INVALID_ARG_TYPE('chunk', ['string', 'Buffer', 'Uint8Array'], chunk); - } - - return er; -} - -Readable.prototype.isPaused = function () { - return this._readableState.flowing === false; -}; // backwards compatibility. - - -Readable.prototype.setEncoding = function (enc) { - if (!StringDecoder) StringDecoder = require('string_decoder/').StringDecoder; - var decoder = new StringDecoder(enc); - this._readableState.decoder = decoder; // If setEncoding(null), decoder.encoding equals utf8 - - this._readableState.encoding = this._readableState.decoder.encoding; // Iterate over current buffer to convert already stored Buffers: - - var p = this._readableState.buffer.head; - var content = ''; - - while (p !== null) { - content += decoder.write(p.data); - p = p.next; - } - - this._readableState.buffer.clear(); - - if (content !== '') this._readableState.buffer.push(content); - this._readableState.length = content.length; - return this; -}; // Don't raise the hwm > 1GB - - -var MAX_HWM = 0x40000000; - -function computeNewHighWaterMark(n) { - if (n >= MAX_HWM) { - // TODO(ronag): Throw ERR_VALUE_OUT_OF_RANGE. - n = MAX_HWM; - } else { - // Get the next highest power of 2 to prevent increasing hwm excessively in - // tiny amounts - n--; - n |= n >>> 1; - n |= n >>> 2; - n |= n >>> 4; - n |= n >>> 8; - n |= n >>> 16; - n++; - } - - return n; -} // This function is designed to be inlinable, so please take care when making -// changes to the function body. - - -function howMuchToRead(n, state) { - if (n <= 0 || state.length === 0 && state.ended) return 0; - if (state.objectMode) return 1; - - if (n !== n) { - // Only flow one buffer at a time - if (state.flowing && state.length) return state.buffer.head.data.length;else return state.length; - } // If we're asking for more than the current hwm, then raise the hwm. - - - if (n > state.highWaterMark) state.highWaterMark = computeNewHighWaterMark(n); - if (n <= state.length) return n; // Don't have enough - - if (!state.ended) { - state.needReadable = true; - return 0; - } - - return state.length; -} // you can override either this method, or the async _read(n) below. - - -Readable.prototype.read = function (n) { - debug('read', n); - n = parseInt(n, 10); - var state = this._readableState; - var nOrig = n; - if (n !== 0) state.emittedReadable = false; // if we're doing read(0) to trigger a readable event, but we - // already have a bunch of data in the buffer, then just trigger - // the 'readable' event and move on. - - if (n === 0 && state.needReadable && ((state.highWaterMark !== 0 ? state.length >= state.highWaterMark : state.length > 0) || state.ended)) { - debug('read: emitReadable', state.length, state.ended); - if (state.length === 0 && state.ended) endReadable(this);else emitReadable(this); - return null; - } - - n = howMuchToRead(n, state); // if we've ended, and we're now clear, then finish it up. - - if (n === 0 && state.ended) { - if (state.length === 0) endReadable(this); - return null; - } // All the actual chunk generation logic needs to be - // *below* the call to _read. The reason is that in certain - // synthetic stream cases, such as passthrough streams, _read - // may be a completely synchronous operation which may change - // the state of the read buffer, providing enough data when - // before there was *not* enough. - // - // So, the steps are: - // 1. Figure out what the state of things will be after we do - // a read from the buffer. - // - // 2. If that resulting state will trigger a _read, then call _read. - // Note that this may be asynchronous, or synchronous. Yes, it is - // deeply ugly to write APIs this way, but that still doesn't mean - // that the Readable class should behave improperly, as streams are - // designed to be sync/async agnostic. - // Take note if the _read call is sync or async (ie, if the read call - // has returned yet), so that we know whether or not it's safe to emit - // 'readable' etc. - // - // 3. Actually pull the requested chunks out of the buffer and return. - // if we need a readable event, then we need to do some reading. - - - var doRead = state.needReadable; - debug('need readable', doRead); // if we currently have less than the highWaterMark, then also read some - - if (state.length === 0 || state.length - n < state.highWaterMark) { - doRead = true; - debug('length less than watermark', doRead); - } // however, if we've ended, then there's no point, and if we're already - // reading, then it's unnecessary. - - - if (state.ended || state.reading) { - doRead = false; - debug('reading or ended', doRead); - } else if (doRead) { - debug('do read'); - state.reading = true; - state.sync = true; // if the length is currently zero, then we *need* a readable event. - - if (state.length === 0) state.needReadable = true; // call internal read method - - this._read(state.highWaterMark); - - state.sync = false; // If _read pushed data synchronously, then `reading` will be false, - // and we need to re-evaluate how much data we can return to the user. - - if (!state.reading) n = howMuchToRead(nOrig, state); - } - - var ret; - if (n > 0) ret = fromList(n, state);else ret = null; - - if (ret === null) { - state.needReadable = state.length <= state.highWaterMark; - n = 0; - } else { - state.length -= n; - state.awaitDrain = 0; - } - - if (state.length === 0) { - // If we have nothing in the buffer, then we want to know - // as soon as we *do* get something into the buffer. - if (!state.ended) state.needReadable = true; // If we tried to read() past the EOF, then emit end on the next tick. - - if (nOrig !== n && state.ended) endReadable(this); - } - - if (ret !== null) this.emit('data', ret); - return ret; -}; - -function onEofChunk(stream, state) { - debug('onEofChunk'); - if (state.ended) return; - - if (state.decoder) { - var chunk = state.decoder.end(); - - if (chunk && chunk.length) { - state.buffer.push(chunk); - state.length += state.objectMode ? 1 : chunk.length; - } - } - - state.ended = true; - - if (state.sync) { - // if we are sync, wait until next tick to emit the data. - // Otherwise we risk emitting data in the flow() - // the readable code triggers during a read() call - emitReadable(stream); - } else { - // emit 'readable' now to make sure it gets picked up. - state.needReadable = false; - - if (!state.emittedReadable) { - state.emittedReadable = true; - emitReadable_(stream); - } - } -} // Don't emit readable right away in sync mode, because this can trigger -// another read() call => stack overflow. This way, it might trigger -// a nextTick recursion warning, but that's not so bad. - - -function emitReadable(stream) { - var state = stream._readableState; - debug('emitReadable', state.needReadable, state.emittedReadable); - state.needReadable = false; - - if (!state.emittedReadable) { - debug('emitReadable', state.flowing); - state.emittedReadable = true; - process.nextTick(emitReadable_, stream); - } -} - -function emitReadable_(stream) { - var state = stream._readableState; - debug('emitReadable_', state.destroyed, state.length, state.ended); - - if (!state.destroyed && (state.length || state.ended)) { - stream.emit('readable'); - state.emittedReadable = false; - } // The stream needs another readable event if - // 1. It is not flowing, as the flow mechanism will take - // care of it. - // 2. It is not ended. - // 3. It is below the highWaterMark, so we can schedule - // another readable later. - - - state.needReadable = !state.flowing && !state.ended && state.length <= state.highWaterMark; - flow(stream); -} // at this point, the user has presumably seen the 'readable' event, -// and called read() to consume some data. that may have triggered -// in turn another _read(n) call, in which case reading = true if -// it's in progress. -// However, if we're not ended, or reading, and the length < hwm, -// then go ahead and try to read some more preemptively. - - -function maybeReadMore(stream, state) { - if (!state.readingMore) { - state.readingMore = true; - process.nextTick(maybeReadMore_, stream, state); - } -} - -function maybeReadMore_(stream, state) { - // Attempt to read more data if we should. - // - // The conditions for reading more data are (one of): - // - Not enough data buffered (state.length < state.highWaterMark). The loop - // is responsible for filling the buffer with enough data if such data - // is available. If highWaterMark is 0 and we are not in the flowing mode - // we should _not_ attempt to buffer any extra data. We'll get more data - // when the stream consumer calls read() instead. - // - No data in the buffer, and the stream is in flowing mode. In this mode - // the loop below is responsible for ensuring read() is called. Failing to - // call read here would abort the flow and there's no other mechanism for - // continuing the flow if the stream consumer has just subscribed to the - // 'data' event. - // - // In addition to the above conditions to keep reading data, the following - // conditions prevent the data from being read: - // - The stream has ended (state.ended). - // - There is already a pending 'read' operation (state.reading). This is a - // case where the the stream has called the implementation defined _read() - // method, but they are processing the call asynchronously and have _not_ - // called push() with new data. In this case we skip performing more - // read()s. The execution ends in this method again after the _read() ends - // up calling push() with more data. - while (!state.reading && !state.ended && (state.length < state.highWaterMark || state.flowing && state.length === 0)) { - var len = state.length; - debug('maybeReadMore read 0'); - stream.read(0); - if (len === state.length) // didn't get any data, stop spinning. - break; - } - - state.readingMore = false; -} // abstract method. to be overridden in specific implementation classes. -// call cb(er, data) where data is <= n in length. -// for virtual (non-string, non-buffer) streams, "length" is somewhat -// arbitrary, and perhaps not very meaningful. - - -Readable.prototype._read = function (n) { - errorOrDestroy(this, new ERR_METHOD_NOT_IMPLEMENTED('_read()')); -}; - -Readable.prototype.pipe = function (dest, pipeOpts) { - var src = this; - var state = this._readableState; - - switch (state.pipesCount) { - case 0: - state.pipes = dest; - break; - - case 1: - state.pipes = [state.pipes, dest]; - break; - - default: - state.pipes.push(dest); - break; - } - - state.pipesCount += 1; - debug('pipe count=%d opts=%j', state.pipesCount, pipeOpts); - var doEnd = (!pipeOpts || pipeOpts.end !== false) && dest !== process.stdout && dest !== process.stderr; - var endFn = doEnd ? onend : unpipe; - if (state.endEmitted) process.nextTick(endFn);else src.once('end', endFn); - dest.on('unpipe', onunpipe); - - function onunpipe(readable, unpipeInfo) { - debug('onunpipe'); - - if (readable === src) { - if (unpipeInfo && unpipeInfo.hasUnpiped === false) { - unpipeInfo.hasUnpiped = true; - cleanup(); - } - } - } - - function onend() { - debug('onend'); - dest.end(); - } // when the dest drains, it reduces the awaitDrain counter - // on the source. This would be more elegant with a .once() - // handler in flow(), but adding and removing repeatedly is - // too slow. - - - var ondrain = pipeOnDrain(src); - dest.on('drain', ondrain); - var cleanedUp = false; - - function cleanup() { - debug('cleanup'); // cleanup event handlers once the pipe is broken - - dest.removeListener('close', onclose); - dest.removeListener('finish', onfinish); - dest.removeListener('drain', ondrain); - dest.removeListener('error', onerror); - dest.removeListener('unpipe', onunpipe); - src.removeListener('end', onend); - src.removeListener('end', unpipe); - src.removeListener('data', ondata); - cleanedUp = true; // if the reader is waiting for a drain event from this - // specific writer, then it would cause it to never start - // flowing again. - // So, if this is awaiting a drain, then we just call it now. - // If we don't know, then assume that we are waiting for one. - - if (state.awaitDrain && (!dest._writableState || dest._writableState.needDrain)) ondrain(); - } - - src.on('data', ondata); - - function ondata(chunk) { - debug('ondata'); - var ret = dest.write(chunk); - debug('dest.write', ret); - - if (ret === false) { - // If the user unpiped during `dest.write()`, it is possible - // to get stuck in a permanently paused state if that write - // also returned false. - // => Check whether `dest` is still a piping destination. - if ((state.pipesCount === 1 && state.pipes === dest || state.pipesCount > 1 && indexOf(state.pipes, dest) !== -1) && !cleanedUp) { - debug('false write response, pause', state.awaitDrain); - state.awaitDrain++; - } - - src.pause(); - } - } // if the dest has an error, then stop piping into it. - // however, don't suppress the throwing behavior for this. - - - function onerror(er) { - debug('onerror', er); - unpipe(); - dest.removeListener('error', onerror); - if (EElistenerCount(dest, 'error') === 0) errorOrDestroy(dest, er); - } // Make sure our error handler is attached before userland ones. - - - prependListener(dest, 'error', onerror); // Both close and finish should trigger unpipe, but only once. - - function onclose() { - dest.removeListener('finish', onfinish); - unpipe(); - } - - dest.once('close', onclose); - - function onfinish() { - debug('onfinish'); - dest.removeListener('close', onclose); - unpipe(); - } - - dest.once('finish', onfinish); - - function unpipe() { - debug('unpipe'); - src.unpipe(dest); - } // tell the dest that it's being piped to - - - dest.emit('pipe', src); // start the flow if it hasn't been started already. - - if (!state.flowing) { - debug('pipe resume'); - src.resume(); - } - - return dest; -}; - -function pipeOnDrain(src) { - return function pipeOnDrainFunctionResult() { - var state = src._readableState; - debug('pipeOnDrain', state.awaitDrain); - if (state.awaitDrain) state.awaitDrain--; - - if (state.awaitDrain === 0 && EElistenerCount(src, 'data')) { - state.flowing = true; - flow(src); - } - }; -} - -Readable.prototype.unpipe = function (dest) { - var state = this._readableState; - var unpipeInfo = { - hasUnpiped: false - }; // if we're not piping anywhere, then do nothing. - - if (state.pipesCount === 0) return this; // just one destination. most common case. - - if (state.pipesCount === 1) { - // passed in one, but it's not the right one. - if (dest && dest !== state.pipes) return this; - if (!dest) dest = state.pipes; // got a match. - - state.pipes = null; - state.pipesCount = 0; - state.flowing = false; - if (dest) dest.emit('unpipe', this, unpipeInfo); - return this; - } // slow case. multiple pipe destinations. - - - if (!dest) { - // remove all. - var dests = state.pipes; - var len = state.pipesCount; - state.pipes = null; - state.pipesCount = 0; - state.flowing = false; - - for (var i = 0; i < len; i++) { - dests[i].emit('unpipe', this, { - hasUnpiped: false - }); - } - - return this; - } // try to find the right one. - - - var index = indexOf(state.pipes, dest); - if (index === -1) return this; - state.pipes.splice(index, 1); - state.pipesCount -= 1; - if (state.pipesCount === 1) state.pipes = state.pipes[0]; - dest.emit('unpipe', this, unpipeInfo); - return this; -}; // set up data events if they are asked for -// Ensure readable listeners eventually get something - - -Readable.prototype.on = function (ev, fn) { - var res = Stream.prototype.on.call(this, ev, fn); - var state = this._readableState; - - if (ev === 'data') { - // update readableListening so that resume() may be a no-op - // a few lines down. This is needed to support once('readable'). - state.readableListening = this.listenerCount('readable') > 0; // Try start flowing on next tick if stream isn't explicitly paused - - if (state.flowing !== false) this.resume(); - } else if (ev === 'readable') { - if (!state.endEmitted && !state.readableListening) { - state.readableListening = state.needReadable = true; - state.flowing = false; - state.emittedReadable = false; - debug('on readable', state.length, state.reading); - - if (state.length) { - emitReadable(this); - } else if (!state.reading) { - process.nextTick(nReadingNextTick, this); - } - } - } - - return res; -}; - -Readable.prototype.addListener = Readable.prototype.on; - -Readable.prototype.removeListener = function (ev, fn) { - var res = Stream.prototype.removeListener.call(this, ev, fn); - - if (ev === 'readable') { - // We need to check if there is someone still listening to - // readable and reset the state. However this needs to happen - // after readable has been emitted but before I/O (nextTick) to - // support once('readable', fn) cycles. This means that calling - // resume within the same tick will have no - // effect. - process.nextTick(updateReadableListening, this); - } - - return res; -}; - -Readable.prototype.removeAllListeners = function (ev) { - var res = Stream.prototype.removeAllListeners.apply(this, arguments); - - if (ev === 'readable' || ev === undefined) { - // We need to check if there is someone still listening to - // readable and reset the state. However this needs to happen - // after readable has been emitted but before I/O (nextTick) to - // support once('readable', fn) cycles. This means that calling - // resume within the same tick will have no - // effect. - process.nextTick(updateReadableListening, this); - } - - return res; -}; - -function updateReadableListening(self) { - var state = self._readableState; - state.readableListening = self.listenerCount('readable') > 0; - - if (state.resumeScheduled && !state.paused) { - // flowing needs to be set to true now, otherwise - // the upcoming resume will not flow. - state.flowing = true; // crude way to check if we should resume - } else if (self.listenerCount('data') > 0) { - self.resume(); - } -} - -function nReadingNextTick(self) { - debug('readable nexttick read 0'); - self.read(0); -} // pause() and resume() are remnants of the legacy readable stream API -// If the user uses them, then switch into old mode. - - -Readable.prototype.resume = function () { - var state = this._readableState; - - if (!state.flowing) { - debug('resume'); // we flow only if there is no one listening - // for readable, but we still have to call - // resume() - - state.flowing = !state.readableListening; - resume(this, state); - } - - state.paused = false; - return this; -}; - -function resume(stream, state) { - if (!state.resumeScheduled) { - state.resumeScheduled = true; - process.nextTick(resume_, stream, state); - } -} - -function resume_(stream, state) { - debug('resume', state.reading); - - if (!state.reading) { - stream.read(0); - } - - state.resumeScheduled = false; - stream.emit('resume'); - flow(stream); - if (state.flowing && !state.reading) stream.read(0); -} - -Readable.prototype.pause = function () { - debug('call pause flowing=%j', this._readableState.flowing); - - if (this._readableState.flowing !== false) { - debug('pause'); - this._readableState.flowing = false; - this.emit('pause'); - } - - this._readableState.paused = true; - return this; -}; - -function flow(stream) { - var state = stream._readableState; - debug('flow', state.flowing); - - while (state.flowing && stream.read() !== null) { - ; - } -} // wrap an old-style stream as the async data source. -// This is *not* part of the readable stream interface. -// It is an ugly unfortunate mess of history. - - -Readable.prototype.wrap = function (stream) { - var _this = this; - - var state = this._readableState; - var paused = false; - stream.on('end', function () { - debug('wrapped end'); - - if (state.decoder && !state.ended) { - var chunk = state.decoder.end(); - if (chunk && chunk.length) _this.push(chunk); - } - - _this.push(null); - }); - stream.on('data', function (chunk) { - debug('wrapped data'); - if (state.decoder) chunk = state.decoder.write(chunk); // don't skip over falsy values in objectMode - - if (state.objectMode && (chunk === null || chunk === undefined)) return;else if (!state.objectMode && (!chunk || !chunk.length)) return; - - var ret = _this.push(chunk); - - if (!ret) { - paused = true; - stream.pause(); - } - }); // proxy all the other methods. - // important when wrapping filters and duplexes. - - for (var i in stream) { - if (this[i] === undefined && typeof stream[i] === 'function') { - this[i] = function methodWrap(method) { - return function methodWrapReturnFunction() { - return stream[method].apply(stream, arguments); - }; - }(i); - } - } // proxy certain important events. - - - for (var n = 0; n < kProxyEvents.length; n++) { - stream.on(kProxyEvents[n], this.emit.bind(this, kProxyEvents[n])); - } // when we try to consume some more bytes, simply unpause the - // underlying stream. - - - this._read = function (n) { - debug('wrapped _read', n); - - if (paused) { - paused = false; - stream.resume(); - } - }; - - return this; -}; - -if (typeof Symbol === 'function') { - Readable.prototype[Symbol.asyncIterator] = function () { - if (createReadableStreamAsyncIterator === undefined) { - createReadableStreamAsyncIterator = require('./internal/streams/async_iterator'); - } - - return createReadableStreamAsyncIterator(this); - }; -} - -Object.defineProperty(Readable.prototype, 'readableHighWaterMark', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - return this._readableState.highWaterMark; - } -}); -Object.defineProperty(Readable.prototype, 'readableBuffer', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - return this._readableState && this._readableState.buffer; - } -}); -Object.defineProperty(Readable.prototype, 'readableFlowing', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - return this._readableState.flowing; - }, - set: function set(state) { - if (this._readableState) { - this._readableState.flowing = state; - } - } -}); // exposed for testing purposes only. - -Readable._fromList = fromList; -Object.defineProperty(Readable.prototype, 'readableLength', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - return this._readableState.length; - } -}); // Pluck off n bytes from an array of buffers. -// Length is the combined lengths of all the buffers in the list. -// This function is designed to be inlinable, so please take care when making -// changes to the function body. - -function fromList(n, state) { - // nothing buffered - if (state.length === 0) return null; - var ret; - if (state.objectMode) ret = state.buffer.shift();else if (!n || n >= state.length) { - // read it all, truncate the list - if (state.decoder) ret = state.buffer.join('');else if (state.buffer.length === 1) ret = state.buffer.first();else ret = state.buffer.concat(state.length); - state.buffer.clear(); - } else { - // read part of list - ret = state.buffer.consume(n, state.decoder); - } - return ret; -} - -function endReadable(stream) { - var state = stream._readableState; - debug('endReadable', state.endEmitted); - - if (!state.endEmitted) { - state.ended = true; - process.nextTick(endReadableNT, state, stream); - } -} - -function endReadableNT(state, stream) { - debug('endReadableNT', state.endEmitted, state.length); // Check that we didn't get one last unshift. - - if (!state.endEmitted && state.length === 0) { - state.endEmitted = true; - stream.readable = false; - stream.emit('end'); - - if (state.autoDestroy) { - // In case of duplex streams we need a way to detect - // if the writable side is ready for autoDestroy as well - var wState = stream._writableState; - - if (!wState || wState.autoDestroy && wState.finished) { - stream.destroy(); - } - } - } -} - -if (typeof Symbol === 'function') { - Readable.from = function (iterable, opts) { - if (from === undefined) { - from = require('./internal/streams/from'); - } - - return from(Readable, iterable, opts); - }; -} - -function indexOf(xs, x) { - for (var i = 0, l = xs.length; i < l; i++) { - if (xs[i] === x) return i; - } - - return -1; -} -}).call(this,require('_process'),typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) - -},{"../errors":107,"./_stream_duplex":108,"./internal/streams/async_iterator":113,"./internal/streams/buffer_list":114,"./internal/streams/destroy":115,"./internal/streams/from":117,"./internal/streams/state":119,"./internal/streams/stream":120,"_process":244,"buffer":124,"events":181,"inherits":106,"string_decoder/":292,"util":77}],111:[function(require,module,exports){ -// Copyright Joyent, Inc. and other Node contributors. -// -// Permission is hereby granted, free of charge, to any person obtaining a -// copy of this software and associated documentation files (the -// "Software"), to deal in the Software without restriction, including -// without limitation the rights to use, copy, modify, merge, publish, -// distribute, sublicense, and/or sell copies of the Software, and to permit -// persons to whom the Software is furnished to do so, subject to the -// following conditions: -// -// The above copyright notice and this permission notice shall be included -// in all copies or substantial portions of the Software. -// -// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS -// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN -// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, -// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR -// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE -// USE OR OTHER DEALINGS IN THE SOFTWARE. -// a transform stream is a readable/writable stream where you do -// something with the data. Sometimes it's called a "filter", -// but that's not a great name for it, since that implies a thing where -// some bits pass through, and others are simply ignored. (That would -// be a valid example of a transform, of course.) -// -// While the output is causally related to the input, it's not a -// necessarily symmetric or synchronous transformation. For example, -// a zlib stream might take multiple plain-text writes(), and then -// emit a single compressed chunk some time in the future. -// -// Here's how this works: -// -// The Transform stream has all the aspects of the readable and writable -// stream classes. When you write(chunk), that calls _write(chunk,cb) -// internally, and returns false if there's a lot of pending writes -// buffered up. When you call read(), that calls _read(n) until -// there's enough pending readable data buffered up. -// -// In a transform stream, the written data is placed in a buffer. When -// _read(n) is called, it transforms the queued up data, calling the -// buffered _write cb's as it consumes chunks. If consuming a single -// written chunk would result in multiple output chunks, then the first -// outputted bit calls the readcb, and subsequent chunks just go into -// the read buffer, and will cause it to emit 'readable' if necessary. -// -// This way, back-pressure is actually determined by the reading side, -// since _read has to be called to start processing a new chunk. However, -// a pathological inflate type of transform can cause excessive buffering -// here. For example, imagine a stream where every byte of input is -// interpreted as an integer from 0-255, and then results in that many -// bytes of output. Writing the 4 bytes {ff,ff,ff,ff} would result in -// 1kb of data being output. In this case, you could write a very small -// amount of input, and end up with a very large amount of output. In -// such a pathological inflating mechanism, there'd be no way to tell -// the system to stop doing the transform. A single 4MB write could -// cause the system to run out of memory. -// -// However, even in such a pathological case, only a single written chunk -// would be consumed, and then the rest would wait (un-transformed) until -// the results of the previous transformed chunk were consumed. -'use strict'; - -module.exports = Transform; - -var _require$codes = require('../errors').codes, - ERR_METHOD_NOT_IMPLEMENTED = _require$codes.ERR_METHOD_NOT_IMPLEMENTED, - ERR_MULTIPLE_CALLBACK = _require$codes.ERR_MULTIPLE_CALLBACK, - ERR_TRANSFORM_ALREADY_TRANSFORMING = _require$codes.ERR_TRANSFORM_ALREADY_TRANSFORMING, - ERR_TRANSFORM_WITH_LENGTH_0 = _require$codes.ERR_TRANSFORM_WITH_LENGTH_0; - -var Duplex = require('./_stream_duplex'); - -require('inherits')(Transform, Duplex); - -function afterTransform(er, data) { - var ts = this._transformState; - ts.transforming = false; - var cb = ts.writecb; - - if (cb === null) { - return this.emit('error', new ERR_MULTIPLE_CALLBACK()); - } - - ts.writechunk = null; - ts.writecb = null; - if (data != null) // single equals check for both `null` and `undefined` - this.push(data); - cb(er); - var rs = this._readableState; - rs.reading = false; - - if (rs.needReadable || rs.length < rs.highWaterMark) { - this._read(rs.highWaterMark); - } -} - -function Transform(options) { - if (!(this instanceof Transform)) return new Transform(options); - Duplex.call(this, options); - this._transformState = { - afterTransform: afterTransform.bind(this), - needTransform: false, - transforming: false, - writecb: null, - writechunk: null, - writeencoding: null - }; // start out asking for a readable event once data is transformed. - - this._readableState.needReadable = true; // we have implemented the _read method, and done the other things - // that Readable wants before the first _read call, so unset the - // sync guard flag. - - this._readableState.sync = false; - - if (options) { - if (typeof options.transform === 'function') this._transform = options.transform; - if (typeof options.flush === 'function') this._flush = options.flush; - } // When the writable side finishes, then flush out anything remaining. - - - this.on('prefinish', prefinish); -} - -function prefinish() { - var _this = this; - - if (typeof this._flush === 'function' && !this._readableState.destroyed) { - this._flush(function (er, data) { - done(_this, er, data); - }); - } else { - done(this, null, null); - } -} - -Transform.prototype.push = function (chunk, encoding) { - this._transformState.needTransform = false; - return Duplex.prototype.push.call(this, chunk, encoding); -}; // This is the part where you do stuff! -// override this function in implementation classes. -// 'chunk' is an input chunk. -// -// Call `push(newChunk)` to pass along transformed output -// to the readable side. You may call 'push' zero or more times. -// -// Call `cb(err)` when you are done with this chunk. If you pass -// an error, then that'll put the hurt on the whole operation. If you -// never call cb(), then you'll never get another chunk. - - -Transform.prototype._transform = function (chunk, encoding, cb) { - cb(new ERR_METHOD_NOT_IMPLEMENTED('_transform()')); -}; - -Transform.prototype._write = function (chunk, encoding, cb) { - var ts = this._transformState; - ts.writecb = cb; - ts.writechunk = chunk; - ts.writeencoding = encoding; - - if (!ts.transforming) { - var rs = this._readableState; - if (ts.needTransform || rs.needReadable || rs.length < rs.highWaterMark) this._read(rs.highWaterMark); - } -}; // Doesn't matter what the args are here. -// _transform does all the work. -// That we got here means that the readable side wants more data. - - -Transform.prototype._read = function (n) { - var ts = this._transformState; - - if (ts.writechunk !== null && !ts.transforming) { - ts.transforming = true; - - this._transform(ts.writechunk, ts.writeencoding, ts.afterTransform); - } else { - // mark that we need a transform, so that any data that comes in - // will get processed, now that we've asked for it. - ts.needTransform = true; - } -}; - -Transform.prototype._destroy = function (err, cb) { - Duplex.prototype._destroy.call(this, err, function (err2) { - cb(err2); - }); -}; - -function done(stream, er, data) { - if (er) return stream.emit('error', er); - if (data != null) // single equals check for both `null` and `undefined` - stream.push(data); // TODO(BridgeAR): Write a test for these two error cases - // if there's nothing in the write buffer, then that means - // that nothing more will ever be provided - - if (stream._writableState.length) throw new ERR_TRANSFORM_WITH_LENGTH_0(); - if (stream._transformState.transforming) throw new ERR_TRANSFORM_ALREADY_TRANSFORMING(); - return stream.push(null); -} -},{"../errors":107,"./_stream_duplex":108,"inherits":106}],112:[function(require,module,exports){ -(function (process,global){ -// Copyright Joyent, Inc. and other Node contributors. -// -// Permission is hereby granted, free of charge, to any person obtaining a -// copy of this software and associated documentation files (the -// "Software"), to deal in the Software without restriction, including -// without limitation the rights to use, copy, modify, merge, publish, -// distribute, sublicense, and/or sell copies of the Software, and to permit -// persons to whom the Software is furnished to do so, subject to the -// following conditions: -// -// The above copyright notice and this permission notice shall be included -// in all copies or substantial portions of the Software. -// -// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS -// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN -// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, -// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR -// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE -// USE OR OTHER DEALINGS IN THE SOFTWARE. -// A bit simpler than readable streams. -// Implement an async ._write(chunk, encoding, cb), and it'll handle all -// the drain event emission and buffering. -'use strict'; - -module.exports = Writable; -/* */ - -function WriteReq(chunk, encoding, cb) { - this.chunk = chunk; - this.encoding = encoding; - this.callback = cb; - this.next = null; -} // It seems a linked list but it is not -// there will be only 2 of these for each stream - - -function CorkedRequest(state) { - var _this = this; - - this.next = null; - this.entry = null; - - this.finish = function () { - onCorkedFinish(_this, state); - }; -} -/* */ - -/**/ - - -var Duplex; -/**/ - -Writable.WritableState = WritableState; -/**/ - -var internalUtil = { - deprecate: require('util-deprecate') -}; -/**/ - -/**/ - -var Stream = require('./internal/streams/stream'); -/**/ - - -var Buffer = require('buffer').Buffer; - -var OurUint8Array = global.Uint8Array || function () {}; - -function _uint8ArrayToBuffer(chunk) { - return Buffer.from(chunk); -} - -function _isUint8Array(obj) { - return Buffer.isBuffer(obj) || obj instanceof OurUint8Array; -} - -var destroyImpl = require('./internal/streams/destroy'); - -var _require = require('./internal/streams/state'), - getHighWaterMark = _require.getHighWaterMark; - -var _require$codes = require('../errors').codes, - ERR_INVALID_ARG_TYPE = _require$codes.ERR_INVALID_ARG_TYPE, - ERR_METHOD_NOT_IMPLEMENTED = _require$codes.ERR_METHOD_NOT_IMPLEMENTED, - ERR_MULTIPLE_CALLBACK = _require$codes.ERR_MULTIPLE_CALLBACK, - ERR_STREAM_CANNOT_PIPE = _require$codes.ERR_STREAM_CANNOT_PIPE, - ERR_STREAM_DESTROYED = _require$codes.ERR_STREAM_DESTROYED, - ERR_STREAM_NULL_VALUES = _require$codes.ERR_STREAM_NULL_VALUES, - ERR_STREAM_WRITE_AFTER_END = _require$codes.ERR_STREAM_WRITE_AFTER_END, - ERR_UNKNOWN_ENCODING = _require$codes.ERR_UNKNOWN_ENCODING; - -var errorOrDestroy = destroyImpl.errorOrDestroy; - -require('inherits')(Writable, Stream); - -function nop() {} - -function WritableState(options, stream, isDuplex) { - Duplex = Duplex || require('./_stream_duplex'); - options = options || {}; // Duplex streams are both readable and writable, but share - // the same options object. - // However, some cases require setting options to different - // values for the readable and the writable sides of the duplex stream, - // e.g. options.readableObjectMode vs. options.writableObjectMode, etc. - - if (typeof isDuplex !== 'boolean') isDuplex = stream instanceof Duplex; // object stream flag to indicate whether or not this stream - // contains buffers or objects. - - this.objectMode = !!options.objectMode; - if (isDuplex) this.objectMode = this.objectMode || !!options.writableObjectMode; // the point at which write() starts returning false - // Note: 0 is a valid value, means that we always return false if - // the entire buffer is not flushed immediately on write() - - this.highWaterMark = getHighWaterMark(this, options, 'writableHighWaterMark', isDuplex); // if _final has been called - - this.finalCalled = false; // drain event flag. - - this.needDrain = false; // at the start of calling end() - - this.ending = false; // when end() has been called, and returned - - this.ended = false; // when 'finish' is emitted - - this.finished = false; // has it been destroyed - - this.destroyed = false; // should we decode strings into buffers before passing to _write? - // this is here so that some node-core streams can optimize string - // handling at a lower level. - - var noDecode = options.decodeStrings === false; - this.decodeStrings = !noDecode; // Crypto is kind of old and crusty. Historically, its default string - // encoding is 'binary' so we have to make this configurable. - // Everything else in the universe uses 'utf8', though. - - this.defaultEncoding = options.defaultEncoding || 'utf8'; // not an actual buffer we keep track of, but a measurement - // of how much we're waiting to get pushed to some underlying - // socket or file. - - this.length = 0; // a flag to see when we're in the middle of a write. - - this.writing = false; // when true all writes will be buffered until .uncork() call - - this.corked = 0; // a flag to be able to tell if the onwrite cb is called immediately, - // or on a later tick. We set this to true at first, because any - // actions that shouldn't happen until "later" should generally also - // not happen before the first write call. - - this.sync = true; // a flag to know if we're processing previously buffered items, which - // may call the _write() callback in the same tick, so that we don't - // end up in an overlapped onwrite situation. - - this.bufferProcessing = false; // the callback that's passed to _write(chunk,cb) - - this.onwrite = function (er) { - onwrite(stream, er); - }; // the callback that the user supplies to write(chunk,encoding,cb) - - - this.writecb = null; // the amount that is being written when _write is called. - - this.writelen = 0; - this.bufferedRequest = null; - this.lastBufferedRequest = null; // number of pending user-supplied write callbacks - // this must be 0 before 'finish' can be emitted - - this.pendingcb = 0; // emit prefinish if the only thing we're waiting for is _write cbs - // This is relevant for synchronous Transform streams - - this.prefinished = false; // True if the error was already emitted and should not be thrown again - - this.errorEmitted = false; // Should close be emitted on destroy. Defaults to true. - - this.emitClose = options.emitClose !== false; // Should .destroy() be called after 'finish' (and potentially 'end') - - this.autoDestroy = !!options.autoDestroy; // count buffered requests - - this.bufferedRequestCount = 0; // allocate the first CorkedRequest, there is always - // one allocated and free to use, and we maintain at most two - - this.corkedRequestsFree = new CorkedRequest(this); -} - -WritableState.prototype.getBuffer = function getBuffer() { - var current = this.bufferedRequest; - var out = []; - - while (current) { - out.push(current); - current = current.next; - } - - return out; -}; - -(function () { - try { - Object.defineProperty(WritableState.prototype, 'buffer', { - get: internalUtil.deprecate(function writableStateBufferGetter() { - return this.getBuffer(); - }, '_writableState.buffer is deprecated. Use _writableState.getBuffer ' + 'instead.', 'DEP0003') - }); - } catch (_) {} -})(); // Test _writableState for inheritance to account for Duplex streams, -// whose prototype chain only points to Readable. - - -var realHasInstance; - -if (typeof Symbol === 'function' && Symbol.hasInstance && typeof Function.prototype[Symbol.hasInstance] === 'function') { - realHasInstance = Function.prototype[Symbol.hasInstance]; - Object.defineProperty(Writable, Symbol.hasInstance, { - value: function value(object) { - if (realHasInstance.call(this, object)) return true; - if (this !== Writable) return false; - return object && object._writableState instanceof WritableState; - } - }); -} else { - realHasInstance = function realHasInstance(object) { - return object instanceof this; - }; -} - -function Writable(options) { - Duplex = Duplex || require('./_stream_duplex'); // Writable ctor is applied to Duplexes, too. - // `realHasInstance` is necessary because using plain `instanceof` - // would return false, as no `_writableState` property is attached. - // Trying to use the custom `instanceof` for Writable here will also break the - // Node.js LazyTransform implementation, which has a non-trivial getter for - // `_writableState` that would lead to infinite recursion. - // Checking for a Stream.Duplex instance is faster here instead of inside - // the WritableState constructor, at least with V8 6.5 - - var isDuplex = this instanceof Duplex; - if (!isDuplex && !realHasInstance.call(Writable, this)) return new Writable(options); - this._writableState = new WritableState(options, this, isDuplex); // legacy. - - this.writable = true; - - if (options) { - if (typeof options.write === 'function') this._write = options.write; - if (typeof options.writev === 'function') this._writev = options.writev; - if (typeof options.destroy === 'function') this._destroy = options.destroy; - if (typeof options.final === 'function') this._final = options.final; - } - - Stream.call(this); -} // Otherwise people can pipe Writable streams, which is just wrong. - - -Writable.prototype.pipe = function () { - errorOrDestroy(this, new ERR_STREAM_CANNOT_PIPE()); -}; - -function writeAfterEnd(stream, cb) { - var er = new ERR_STREAM_WRITE_AFTER_END(); // TODO: defer error events consistently everywhere, not just the cb - - errorOrDestroy(stream, er); - process.nextTick(cb, er); -} // Checks that a user-supplied chunk is valid, especially for the particular -// mode the stream is in. Currently this means that `null` is never accepted -// and undefined/non-string values are only allowed in object mode. - - -function validChunk(stream, state, chunk, cb) { - var er; - - if (chunk === null) { - er = new ERR_STREAM_NULL_VALUES(); - } else if (typeof chunk !== 'string' && !state.objectMode) { - er = new ERR_INVALID_ARG_TYPE('chunk', ['string', 'Buffer'], chunk); - } - - if (er) { - errorOrDestroy(stream, er); - process.nextTick(cb, er); - return false; - } - - return true; -} - -Writable.prototype.write = function (chunk, encoding, cb) { - var state = this._writableState; - var ret = false; - - var isBuf = !state.objectMode && _isUint8Array(chunk); - - if (isBuf && !Buffer.isBuffer(chunk)) { - chunk = _uint8ArrayToBuffer(chunk); - } - - if (typeof encoding === 'function') { - cb = encoding; - encoding = null; - } - - if (isBuf) encoding = 'buffer';else if (!encoding) encoding = state.defaultEncoding; - if (typeof cb !== 'function') cb = nop; - if (state.ending) writeAfterEnd(this, cb);else if (isBuf || validChunk(this, state, chunk, cb)) { - state.pendingcb++; - ret = writeOrBuffer(this, state, isBuf, chunk, encoding, cb); - } - return ret; -}; - -Writable.prototype.cork = function () { - this._writableState.corked++; -}; - -Writable.prototype.uncork = function () { - var state = this._writableState; - - if (state.corked) { - state.corked--; - if (!state.writing && !state.corked && !state.bufferProcessing && state.bufferedRequest) clearBuffer(this, state); - } -}; - -Writable.prototype.setDefaultEncoding = function setDefaultEncoding(encoding) { - // node::ParseEncoding() requires lower case. - if (typeof encoding === 'string') encoding = encoding.toLowerCase(); - if (!(['hex', 'utf8', 'utf-8', 'ascii', 'binary', 'base64', 'ucs2', 'ucs-2', 'utf16le', 'utf-16le', 'raw'].indexOf((encoding + '').toLowerCase()) > -1)) throw new ERR_UNKNOWN_ENCODING(encoding); - this._writableState.defaultEncoding = encoding; - return this; -}; - -Object.defineProperty(Writable.prototype, 'writableBuffer', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - return this._writableState && this._writableState.getBuffer(); - } -}); - -function decodeChunk(state, chunk, encoding) { - if (!state.objectMode && state.decodeStrings !== false && typeof chunk === 'string') { - chunk = Buffer.from(chunk, encoding); - } - - return chunk; -} - -Object.defineProperty(Writable.prototype, 'writableHighWaterMark', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - return this._writableState.highWaterMark; - } -}); // if we're already writing something, then just put this -// in the queue, and wait our turn. Otherwise, call _write -// If we return false, then we need a drain event, so set that flag. - -function writeOrBuffer(stream, state, isBuf, chunk, encoding, cb) { - if (!isBuf) { - var newChunk = decodeChunk(state, chunk, encoding); - - if (chunk !== newChunk) { - isBuf = true; - encoding = 'buffer'; - chunk = newChunk; - } - } - - var len = state.objectMode ? 1 : chunk.length; - state.length += len; - var ret = state.length < state.highWaterMark; // we must ensure that previous needDrain will not be reset to false. - - if (!ret) state.needDrain = true; - - if (state.writing || state.corked) { - var last = state.lastBufferedRequest; - state.lastBufferedRequest = { - chunk: chunk, - encoding: encoding, - isBuf: isBuf, - callback: cb, - next: null - }; - - if (last) { - last.next = state.lastBufferedRequest; - } else { - state.bufferedRequest = state.lastBufferedRequest; - } - - state.bufferedRequestCount += 1; - } else { - doWrite(stream, state, false, len, chunk, encoding, cb); - } - - return ret; -} - -function doWrite(stream, state, writev, len, chunk, encoding, cb) { - state.writelen = len; - state.writecb = cb; - state.writing = true; - state.sync = true; - if (state.destroyed) state.onwrite(new ERR_STREAM_DESTROYED('write'));else if (writev) stream._writev(chunk, state.onwrite);else stream._write(chunk, encoding, state.onwrite); - state.sync = false; -} - -function onwriteError(stream, state, sync, er, cb) { - --state.pendingcb; - - if (sync) { - // defer the callback if we are being called synchronously - // to avoid piling up things on the stack - process.nextTick(cb, er); // this can emit finish, and it will always happen - // after error - - process.nextTick(finishMaybe, stream, state); - stream._writableState.errorEmitted = true; - errorOrDestroy(stream, er); - } else { - // the caller expect this to happen before if - // it is async - cb(er); - stream._writableState.errorEmitted = true; - errorOrDestroy(stream, er); // this can emit finish, but finish must - // always follow error - - finishMaybe(stream, state); - } -} - -function onwriteStateUpdate(state) { - state.writing = false; - state.writecb = null; - state.length -= state.writelen; - state.writelen = 0; -} - -function onwrite(stream, er) { - var state = stream._writableState; - var sync = state.sync; - var cb = state.writecb; - if (typeof cb !== 'function') throw new ERR_MULTIPLE_CALLBACK(); - onwriteStateUpdate(state); - if (er) onwriteError(stream, state, sync, er, cb);else { - // Check if we're actually ready to finish, but don't emit yet - var finished = needFinish(state) || stream.destroyed; - - if (!finished && !state.corked && !state.bufferProcessing && state.bufferedRequest) { - clearBuffer(stream, state); - } - - if (sync) { - process.nextTick(afterWrite, stream, state, finished, cb); - } else { - afterWrite(stream, state, finished, cb); - } - } -} - -function afterWrite(stream, state, finished, cb) { - if (!finished) onwriteDrain(stream, state); - state.pendingcb--; - cb(); - finishMaybe(stream, state); -} // Must force callback to be called on nextTick, so that we don't -// emit 'drain' before the write() consumer gets the 'false' return -// value, and has a chance to attach a 'drain' listener. - - -function onwriteDrain(stream, state) { - if (state.length === 0 && state.needDrain) { - state.needDrain = false; - stream.emit('drain'); - } -} // if there's something in the buffer waiting, then process it - - -function clearBuffer(stream, state) { - state.bufferProcessing = true; - var entry = state.bufferedRequest; - - if (stream._writev && entry && entry.next) { - // Fast case, write everything using _writev() - var l = state.bufferedRequestCount; - var buffer = new Array(l); - var holder = state.corkedRequestsFree; - holder.entry = entry; - var count = 0; - var allBuffers = true; - - while (entry) { - buffer[count] = entry; - if (!entry.isBuf) allBuffers = false; - entry = entry.next; - count += 1; - } - - buffer.allBuffers = allBuffers; - doWrite(stream, state, true, state.length, buffer, '', holder.finish); // doWrite is almost always async, defer these to save a bit of time - // as the hot path ends with doWrite - - state.pendingcb++; - state.lastBufferedRequest = null; - - if (holder.next) { - state.corkedRequestsFree = holder.next; - holder.next = null; - } else { - state.corkedRequestsFree = new CorkedRequest(state); - } - - state.bufferedRequestCount = 0; - } else { - // Slow case, write chunks one-by-one - while (entry) { - var chunk = entry.chunk; - var encoding = entry.encoding; - var cb = entry.callback; - var len = state.objectMode ? 1 : chunk.length; - doWrite(stream, state, false, len, chunk, encoding, cb); - entry = entry.next; - state.bufferedRequestCount--; // if we didn't call the onwrite immediately, then - // it means that we need to wait until it does. - // also, that means that the chunk and cb are currently - // being processed, so move the buffer counter past them. - - if (state.writing) { - break; - } - } - - if (entry === null) state.lastBufferedRequest = null; - } - - state.bufferedRequest = entry; - state.bufferProcessing = false; -} - -Writable.prototype._write = function (chunk, encoding, cb) { - cb(new ERR_METHOD_NOT_IMPLEMENTED('_write()')); -}; - -Writable.prototype._writev = null; - -Writable.prototype.end = function (chunk, encoding, cb) { - var state = this._writableState; - - if (typeof chunk === 'function') { - cb = chunk; - chunk = null; - encoding = null; - } else if (typeof encoding === 'function') { - cb = encoding; - encoding = null; - } - - if (chunk !== null && chunk !== undefined) this.write(chunk, encoding); // .end() fully uncorks - - if (state.corked) { - state.corked = 1; - this.uncork(); - } // ignore unnecessary end() calls. - - - if (!state.ending) endWritable(this, state, cb); - return this; -}; - -Object.defineProperty(Writable.prototype, 'writableLength', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - return this._writableState.length; - } -}); - -function needFinish(state) { - return state.ending && state.length === 0 && state.bufferedRequest === null && !state.finished && !state.writing; -} - -function callFinal(stream, state) { - stream._final(function (err) { - state.pendingcb--; - - if (err) { - errorOrDestroy(stream, err); - } - - state.prefinished = true; - stream.emit('prefinish'); - finishMaybe(stream, state); - }); -} - -function prefinish(stream, state) { - if (!state.prefinished && !state.finalCalled) { - if (typeof stream._final === 'function' && !state.destroyed) { - state.pendingcb++; - state.finalCalled = true; - process.nextTick(callFinal, stream, state); - } else { - state.prefinished = true; - stream.emit('prefinish'); - } - } -} - -function finishMaybe(stream, state) { - var need = needFinish(state); - - if (need) { - prefinish(stream, state); - - if (state.pendingcb === 0) { - state.finished = true; - stream.emit('finish'); - - if (state.autoDestroy) { - // In case of duplex streams we need a way to detect - // if the readable side is ready for autoDestroy as well - var rState = stream._readableState; - - if (!rState || rState.autoDestroy && rState.endEmitted) { - stream.destroy(); - } - } - } - } - - return need; -} - -function endWritable(stream, state, cb) { - state.ending = true; - finishMaybe(stream, state); - - if (cb) { - if (state.finished) process.nextTick(cb);else stream.once('finish', cb); - } - - state.ended = true; - stream.writable = false; -} - -function onCorkedFinish(corkReq, state, err) { - var entry = corkReq.entry; - corkReq.entry = null; - - while (entry) { - var cb = entry.callback; - state.pendingcb--; - cb(err); - entry = entry.next; - } // reuse the free corkReq. - - - state.corkedRequestsFree.next = corkReq; -} - -Object.defineProperty(Writable.prototype, 'destroyed', { - // making it explicit this property is not enumerable - // because otherwise some prototype manipulation in - // userland will fail - enumerable: false, - get: function get() { - if (this._writableState === undefined) { - return false; - } - - return this._writableState.destroyed; - }, - set: function set(value) { - // we ignore the value if the stream - // has not been initialized yet - if (!this._writableState) { - return; - } // backward compatibility, the user is explicitly - // managing destroyed - - - this._writableState.destroyed = value; - } -}); -Writable.prototype.destroy = destroyImpl.destroy; -Writable.prototype._undestroy = destroyImpl.undestroy; - -Writable.prototype._destroy = function (err, cb) { - cb(err); -}; -}).call(this,require('_process'),typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) - -},{"../errors":107,"./_stream_duplex":108,"./internal/streams/destroy":115,"./internal/streams/state":119,"./internal/streams/stream":120,"_process":244,"buffer":124,"inherits":106,"util-deprecate":295}],113:[function(require,module,exports){ -(function (process){ -'use strict'; - -var _Object$setPrototypeO; - -function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } - -var finished = require('./end-of-stream'); - -var kLastResolve = Symbol('lastResolve'); -var kLastReject = Symbol('lastReject'); -var kError = Symbol('error'); -var kEnded = Symbol('ended'); -var kLastPromise = Symbol('lastPromise'); -var kHandlePromise = Symbol('handlePromise'); -var kStream = Symbol('stream'); - -function createIterResult(value, done) { - return { - value: value, - done: done - }; -} - -function readAndResolve(iter) { - var resolve = iter[kLastResolve]; - - if (resolve !== null) { - var data = iter[kStream].read(); // we defer if data is null - // we can be expecting either 'end' or - // 'error' - - if (data !== null) { - iter[kLastPromise] = null; - iter[kLastResolve] = null; - iter[kLastReject] = null; - resolve(createIterResult(data, false)); - } - } -} - -function onReadable(iter) { - // we wait for the next tick, because it might - // emit an error with process.nextTick - process.nextTick(readAndResolve, iter); -} - -function wrapForNext(lastPromise, iter) { - return function (resolve, reject) { - lastPromise.then(function () { - if (iter[kEnded]) { - resolve(createIterResult(undefined, true)); - return; - } - - iter[kHandlePromise](resolve, reject); - }, reject); - }; -} - -var AsyncIteratorPrototype = Object.getPrototypeOf(function () {}); -var ReadableStreamAsyncIteratorPrototype = Object.setPrototypeOf((_Object$setPrototypeO = { - get stream() { - return this[kStream]; - }, - - next: function next() { - var _this = this; - - // if we have detected an error in the meanwhile - // reject straight away - var error = this[kError]; - - if (error !== null) { - return Promise.reject(error); - } - - if (this[kEnded]) { - return Promise.resolve(createIterResult(undefined, true)); - } - - if (this[kStream].destroyed) { - // We need to defer via nextTick because if .destroy(err) is - // called, the error will be emitted via nextTick, and - // we cannot guarantee that there is no error lingering around - // waiting to be emitted. - return new Promise(function (resolve, reject) { - process.nextTick(function () { - if (_this[kError]) { - reject(_this[kError]); - } else { - resolve(createIterResult(undefined, true)); - } - }); - }); - } // if we have multiple next() calls - // we will wait for the previous Promise to finish - // this logic is optimized to support for await loops, - // where next() is only called once at a time - - - var lastPromise = this[kLastPromise]; - var promise; - - if (lastPromise) { - promise = new Promise(wrapForNext(lastPromise, this)); - } else { - // fast path needed to support multiple this.push() - // without triggering the next() queue - var data = this[kStream].read(); - - if (data !== null) { - return Promise.resolve(createIterResult(data, false)); - } - - promise = new Promise(this[kHandlePromise]); - } - - this[kLastPromise] = promise; - return promise; - } -}, _defineProperty(_Object$setPrototypeO, Symbol.asyncIterator, function () { - return this; -}), _defineProperty(_Object$setPrototypeO, "return", function _return() { - var _this2 = this; - - // destroy(err, cb) is a private API - // we can guarantee we have that here, because we control the - // Readable class this is attached to - return new Promise(function (resolve, reject) { - _this2[kStream].destroy(null, function (err) { - if (err) { - reject(err); - return; - } - - resolve(createIterResult(undefined, true)); - }); - }); -}), _Object$setPrototypeO), AsyncIteratorPrototype); - -var createReadableStreamAsyncIterator = function createReadableStreamAsyncIterator(stream) { - var _Object$create; - - var iterator = Object.create(ReadableStreamAsyncIteratorPrototype, (_Object$create = {}, _defineProperty(_Object$create, kStream, { - value: stream, - writable: true - }), _defineProperty(_Object$create, kLastResolve, { - value: null, - writable: true - }), _defineProperty(_Object$create, kLastReject, { - value: null, - writable: true - }), _defineProperty(_Object$create, kError, { - value: null, - writable: true - }), _defineProperty(_Object$create, kEnded, { - value: stream._readableState.endEmitted, - writable: true - }), _defineProperty(_Object$create, kHandlePromise, { - value: function value(resolve, reject) { - var data = iterator[kStream].read(); - - if (data) { - iterator[kLastPromise] = null; - iterator[kLastResolve] = null; - iterator[kLastReject] = null; - resolve(createIterResult(data, false)); - } else { - iterator[kLastResolve] = resolve; - iterator[kLastReject] = reject; - } - }, - writable: true - }), _Object$create)); - iterator[kLastPromise] = null; - finished(stream, function (err) { - if (err && err.code !== 'ERR_STREAM_PREMATURE_CLOSE') { - var reject = iterator[kLastReject]; // reject if we are waiting for data in the Promise - // returned by next() and store the error - - if (reject !== null) { - iterator[kLastPromise] = null; - iterator[kLastResolve] = null; - iterator[kLastReject] = null; - reject(err); - } - - iterator[kError] = err; - return; - } - - var resolve = iterator[kLastResolve]; - - if (resolve !== null) { - iterator[kLastPromise] = null; - iterator[kLastResolve] = null; - iterator[kLastReject] = null; - resolve(createIterResult(undefined, true)); - } - - iterator[kEnded] = true; - }); - stream.on('readable', onReadable.bind(null, iterator)); - return iterator; -}; - -module.exports = createReadableStreamAsyncIterator; -}).call(this,require('_process')) - -},{"./end-of-stream":116,"_process":244}],114:[function(require,module,exports){ -'use strict'; - -function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } - -function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; } - -function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } - -function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } - -function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } - -function _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); return Constructor; } - -var _require = require('buffer'), - Buffer = _require.Buffer; - -var _require2 = require('util'), - inspect = _require2.inspect; - -var custom = inspect && inspect.custom || 'inspect'; - -function copyBuffer(src, target, offset) { - Buffer.prototype.copy.call(src, target, offset); -} - -module.exports = -/*#__PURE__*/ -function () { - function BufferList() { - _classCallCheck(this, BufferList); - - this.head = null; - this.tail = null; - this.length = 0; - } - - _createClass(BufferList, [{ - key: "push", - value: function push(v) { - var entry = { - data: v, - next: null - }; - if (this.length > 0) this.tail.next = entry;else this.head = entry; - this.tail = entry; - ++this.length; - } - }, { - key: "unshift", - value: function unshift(v) { - var entry = { - data: v, - next: this.head - }; - if (this.length === 0) this.tail = entry; - this.head = entry; - ++this.length; - } - }, { - key: "shift", - value: function shift() { - if (this.length === 0) return; - var ret = this.head.data; - if (this.length === 1) this.head = this.tail = null;else this.head = this.head.next; - --this.length; - return ret; - } - }, { - key: "clear", - value: function clear() { - this.head = this.tail = null; - this.length = 0; - } - }, { - key: "join", - value: function join(s) { - if (this.length === 0) return ''; - var p = this.head; - var ret = '' + p.data; - - while (p = p.next) { - ret += s + p.data; - } - - return ret; - } - }, { - key: "concat", - value: function concat(n) { - if (this.length === 0) return Buffer.alloc(0); - var ret = Buffer.allocUnsafe(n >>> 0); - var p = this.head; - var i = 0; - - while (p) { - copyBuffer(p.data, ret, i); - i += p.data.length; - p = p.next; - } - - return ret; - } // Consumes a specified amount of bytes or characters from the buffered data. - - }, { - key: "consume", - value: function consume(n, hasStrings) { - var ret; - - if (n < this.head.data.length) { - // `slice` is the same for buffers and strings. - ret = this.head.data.slice(0, n); - this.head.data = this.head.data.slice(n); - } else if (n === this.head.data.length) { - // First chunk is a perfect match. - ret = this.shift(); - } else { - // Result spans more than one buffer. - ret = hasStrings ? this._getString(n) : this._getBuffer(n); - } - - return ret; - } - }, { - key: "first", - value: function first() { - return this.head.data; - } // Consumes a specified amount of characters from the buffered data. - - }, { - key: "_getString", - value: function _getString(n) { - var p = this.head; - var c = 1; - var ret = p.data; - n -= ret.length; - - while (p = p.next) { - var str = p.data; - var nb = n > str.length ? str.length : n; - if (nb === str.length) ret += str;else ret += str.slice(0, n); - n -= nb; - - if (n === 0) { - if (nb === str.length) { - ++c; - if (p.next) this.head = p.next;else this.head = this.tail = null; - } else { - this.head = p; - p.data = str.slice(nb); - } - - break; - } - - ++c; - } - - this.length -= c; - return ret; - } // Consumes a specified amount of bytes from the buffered data. - - }, { - key: "_getBuffer", - value: function _getBuffer(n) { - var ret = Buffer.allocUnsafe(n); - var p = this.head; - var c = 1; - p.data.copy(ret); - n -= p.data.length; - - while (p = p.next) { - var buf = p.data; - var nb = n > buf.length ? buf.length : n; - buf.copy(ret, ret.length - n, 0, nb); - n -= nb; - - if (n === 0) { - if (nb === buf.length) { - ++c; - if (p.next) this.head = p.next;else this.head = this.tail = null; - } else { - this.head = p; - p.data = buf.slice(nb); - } - - break; - } - - ++c; - } - - this.length -= c; - return ret; - } // Make sure the linked list only shows the minimal necessary information. - - }, { - key: custom, - value: function value(_, options) { - return inspect(this, _objectSpread({}, options, { - // Only inspect one level. - depth: 0, - // It should not recurse. - customInspect: false - })); - } - }]); - - return BufferList; -}(); -},{"buffer":124,"util":77}],115:[function(require,module,exports){ -(function (process){ -'use strict'; // undocumented cb() API, needed for core, not for public API - -function destroy(err, cb) { - var _this = this; - - var readableDestroyed = this._readableState && this._readableState.destroyed; - var writableDestroyed = this._writableState && this._writableState.destroyed; - - if (readableDestroyed || writableDestroyed) { - if (cb) { - cb(err); - } else if (err) { - if (!this._writableState) { - process.nextTick(emitErrorNT, this, err); - } else if (!this._writableState.errorEmitted) { - this._writableState.errorEmitted = true; - process.nextTick(emitErrorNT, this, err); - } - } - - return this; - } // we set destroyed to true before firing error callbacks in order - // to make it re-entrance safe in case destroy() is called within callbacks - - - if (this._readableState) { - this._readableState.destroyed = true; - } // if this is a duplex stream mark the writable part as destroyed as well - - - if (this._writableState) { - this._writableState.destroyed = true; - } - - this._destroy(err || null, function (err) { - if (!cb && err) { - if (!_this._writableState) { - process.nextTick(emitErrorAndCloseNT, _this, err); - } else if (!_this._writableState.errorEmitted) { - _this._writableState.errorEmitted = true; - process.nextTick(emitErrorAndCloseNT, _this, err); - } else { - process.nextTick(emitCloseNT, _this); - } - } else if (cb) { - process.nextTick(emitCloseNT, _this); - cb(err); - } else { - process.nextTick(emitCloseNT, _this); - } - }); - - return this; -} - -function emitErrorAndCloseNT(self, err) { - emitErrorNT(self, err); - emitCloseNT(self); -} - -function emitCloseNT(self) { - if (self._writableState && !self._writableState.emitClose) return; - if (self._readableState && !self._readableState.emitClose) return; - self.emit('close'); -} - -function undestroy() { - if (this._readableState) { - this._readableState.destroyed = false; - this._readableState.reading = false; - this._readableState.ended = false; - this._readableState.endEmitted = false; - } - - if (this._writableState) { - this._writableState.destroyed = false; - this._writableState.ended = false; - this._writableState.ending = false; - this._writableState.finalCalled = false; - this._writableState.prefinished = false; - this._writableState.finished = false; - this._writableState.errorEmitted = false; - } -} - -function emitErrorNT(self, err) { - self.emit('error', err); -} - -function errorOrDestroy(stream, err) { - // We have tests that rely on errors being emitted - // in the same tick, so changing this is semver major. - // For now when you opt-in to autoDestroy we allow - // the error to be emitted nextTick. In a future - // semver major update we should change the default to this. - var rState = stream._readableState; - var wState = stream._writableState; - if (rState && rState.autoDestroy || wState && wState.autoDestroy) stream.destroy(err);else stream.emit('error', err); -} - -module.exports = { - destroy: destroy, - undestroy: undestroy, - errorOrDestroy: errorOrDestroy -}; -}).call(this,require('_process')) - -},{"_process":244}],116:[function(require,module,exports){ -// Ported from https://github.com/mafintosh/end-of-stream with -// permission from the author, Mathias Buus (@mafintosh). -'use strict'; - -var ERR_STREAM_PREMATURE_CLOSE = require('../../../errors').codes.ERR_STREAM_PREMATURE_CLOSE; - -function once(callback) { - var called = false; - return function () { - if (called) return; - called = true; - - for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { - args[_key] = arguments[_key]; - } - - callback.apply(this, args); - }; -} - -function noop() {} - -function isRequest(stream) { - return stream.setHeader && typeof stream.abort === 'function'; -} - -function eos(stream, opts, callback) { - if (typeof opts === 'function') return eos(stream, null, opts); - if (!opts) opts = {}; - callback = once(callback || noop); - var readable = opts.readable || opts.readable !== false && stream.readable; - var writable = opts.writable || opts.writable !== false && stream.writable; - - var onlegacyfinish = function onlegacyfinish() { - if (!stream.writable) onfinish(); - }; - - var writableEnded = stream._writableState && stream._writableState.finished; - - var onfinish = function onfinish() { - writable = false; - writableEnded = true; - if (!readable) callback.call(stream); - }; - - var readableEnded = stream._readableState && stream._readableState.endEmitted; - - var onend = function onend() { - readable = false; - readableEnded = true; - if (!writable) callback.call(stream); - }; - - var onerror = function onerror(err) { - callback.call(stream, err); - }; - - var onclose = function onclose() { - var err; - - if (readable && !readableEnded) { - if (!stream._readableState || !stream._readableState.ended) err = new ERR_STREAM_PREMATURE_CLOSE(); - return callback.call(stream, err); - } - - if (writable && !writableEnded) { - if (!stream._writableState || !stream._writableState.ended) err = new ERR_STREAM_PREMATURE_CLOSE(); - return callback.call(stream, err); - } - }; - - var onrequest = function onrequest() { - stream.req.on('finish', onfinish); - }; - - if (isRequest(stream)) { - stream.on('complete', onfinish); - stream.on('abort', onclose); - if (stream.req) onrequest();else stream.on('request', onrequest); - } else if (writable && !stream._writableState) { - // legacy streams - stream.on('end', onlegacyfinish); - stream.on('close', onlegacyfinish); - } - - stream.on('end', onend); - stream.on('finish', onfinish); - if (opts.error !== false) stream.on('error', onerror); - stream.on('close', onclose); - return function () { - stream.removeListener('complete', onfinish); - stream.removeListener('abort', onclose); - stream.removeListener('request', onrequest); - if (stream.req) stream.req.removeListener('finish', onfinish); - stream.removeListener('end', onlegacyfinish); - stream.removeListener('close', onlegacyfinish); - stream.removeListener('finish', onfinish); - stream.removeListener('end', onend); - stream.removeListener('error', onerror); - stream.removeListener('close', onclose); - }; -} - -module.exports = eos; -},{"../../../errors":107}],117:[function(require,module,exports){ -module.exports = function () { - throw new Error('Readable.from is not available in the browser') -}; - -},{}],118:[function(require,module,exports){ -// Ported from https://github.com/mafintosh/pump with -// permission from the author, Mathias Buus (@mafintosh). -'use strict'; - -var eos; - -function once(callback) { - var called = false; - return function () { - if (called) return; - called = true; - callback.apply(void 0, arguments); - }; -} - -var _require$codes = require('../../../errors').codes, - ERR_MISSING_ARGS = _require$codes.ERR_MISSING_ARGS, - ERR_STREAM_DESTROYED = _require$codes.ERR_STREAM_DESTROYED; - -function noop(err) { - // Rethrow the error if it exists to avoid swallowing it - if (err) throw err; -} - -function isRequest(stream) { - return stream.setHeader && typeof stream.abort === 'function'; -} - -function destroyer(stream, reading, writing, callback) { - callback = once(callback); - var closed = false; - stream.on('close', function () { - closed = true; - }); - if (eos === undefined) eos = require('./end-of-stream'); - eos(stream, { - readable: reading, - writable: writing - }, function (err) { - if (err) return callback(err); - closed = true; - callback(); - }); - var destroyed = false; - return function (err) { - if (closed) return; - if (destroyed) return; - destroyed = true; // request.destroy just do .end - .abort is what we want - - if (isRequest(stream)) return stream.abort(); - if (typeof stream.destroy === 'function') return stream.destroy(); - callback(err || new ERR_STREAM_DESTROYED('pipe')); - }; -} - -function call(fn) { - fn(); -} - -function pipe(from, to) { - return from.pipe(to); -} - -function popCallback(streams) { - if (!streams.length) return noop; - if (typeof streams[streams.length - 1] !== 'function') return noop; - return streams.pop(); -} - -function pipeline() { - for (var _len = arguments.length, streams = new Array(_len), _key = 0; _key < _len; _key++) { - streams[_key] = arguments[_key]; - } - - var callback = popCallback(streams); - if (Array.isArray(streams[0])) streams = streams[0]; - - if (streams.length < 2) { - throw new ERR_MISSING_ARGS('streams'); - } - - var error; - var destroys = streams.map(function (stream, i) { - var reading = i < streams.length - 1; - var writing = i > 0; - return destroyer(stream, reading, writing, function (err) { - if (!error) error = err; - if (err) destroys.forEach(call); - if (reading) return; - destroys.forEach(call); - callback(error); - }); - }); - return streams.reduce(pipe); -} - -module.exports = pipeline; -},{"../../../errors":107,"./end-of-stream":116}],119:[function(require,module,exports){ -'use strict'; - -var ERR_INVALID_OPT_VALUE = require('../../../errors').codes.ERR_INVALID_OPT_VALUE; - -function highWaterMarkFrom(options, isDuplex, duplexKey) { - return options.highWaterMark != null ? options.highWaterMark : isDuplex ? options[duplexKey] : null; -} - -function getHighWaterMark(state, options, duplexKey, isDuplex) { - var hwm = highWaterMarkFrom(options, isDuplex, duplexKey); - - if (hwm != null) { - if (!(isFinite(hwm) && Math.floor(hwm) === hwm) || hwm < 0) { - var name = isDuplex ? duplexKey : 'highWaterMark'; - throw new ERR_INVALID_OPT_VALUE(name, hwm); - } - - return Math.floor(hwm); - } // Default value - - - return state.objectMode ? 16 : 16 * 1024; -} - -module.exports = { - getHighWaterMark: getHighWaterMark -}; -},{"../../../errors":107}],120:[function(require,module,exports){ -module.exports = require('events').EventEmitter; - -},{"events":181}],121:[function(require,module,exports){ -exports = module.exports = require('./lib/_stream_readable.js'); -exports.Stream = exports; -exports.Readable = exports; -exports.Writable = require('./lib/_stream_writable.js'); -exports.Duplex = require('./lib/_stream_duplex.js'); -exports.Transform = require('./lib/_stream_transform.js'); -exports.PassThrough = require('./lib/_stream_passthrough.js'); -exports.finished = require('./lib/internal/streams/end-of-stream.js'); -exports.pipeline = require('./lib/internal/streams/pipeline.js'); - -},{"./lib/_stream_duplex.js":108,"./lib/_stream_passthrough.js":109,"./lib/_stream_readable.js":110,"./lib/_stream_transform.js":111,"./lib/_stream_writable.js":112,"./lib/internal/streams/end-of-stream.js":116,"./lib/internal/streams/pipeline.js":118}],122:[function(require,module,exports){ -/*! safe-buffer. MIT License. Feross Aboukhadijeh */ -/* eslint-disable node/no-deprecated-api */ -var buffer = require('buffer') -var Buffer = buffer.Buffer - -// alternative to using Object.keys for old browsers -function copyProps (src, dst) { - for (var key in src) { - dst[key] = src[key] - } -} -if (Buffer.from && Buffer.alloc && Buffer.allocUnsafe && Buffer.allocUnsafeSlow) { - module.exports = buffer -} else { - // Copy properties from require('buffer') - copyProps(buffer, exports) - exports.Buffer = SafeBuffer -} - -function SafeBuffer (arg, encodingOrOffset, length) { - return Buffer(arg, encodingOrOffset, length) -} - -SafeBuffer.prototype = Object.create(Buffer.prototype) - -// Copy static methods from Buffer -copyProps(Buffer, SafeBuffer) - -SafeBuffer.from = function (arg, encodingOrOffset, length) { - if (typeof arg === 'number') { - throw new TypeError('Argument must not be a number') - } - return Buffer(arg, encodingOrOffset, length) -} - -SafeBuffer.alloc = function (size, fill, encoding) { - if (typeof size !== 'number') { - throw new TypeError('Argument must be a number') - } - var buf = Buffer(size) - if (fill !== undefined) { - if (typeof encoding === 'string') { - buf.fill(fill, encoding) - } else { - buf.fill(fill) - } - } else { - buf.fill(0) - } - return buf -} - -SafeBuffer.allocUnsafe = function (size) { - if (typeof size !== 'number') { - throw new TypeError('Argument must be a number') - } - return Buffer(size) -} - -SafeBuffer.allocUnsafeSlow = function (size) { - if (typeof size !== 'number') { - throw new TypeError('Argument must be a number') - } - return buffer.SlowBuffer(size) -} - -},{"buffer":124}],123:[function(require,module,exports){ -(function (Buffer){ -module.exports = function xor (a, b) { - var length = Math.min(a.length, b.length) - var buffer = new Buffer(length) - - for (var i = 0; i < length; ++i) { - buffer[i] = a[i] ^ b[i] - } - - return buffer -} - -}).call(this,require("buffer").Buffer) - -},{"buffer":124}],124:[function(require,module,exports){ -(function (Buffer){ -/*! - * The buffer module from node.js, for the browser. - * - * @author Feross Aboukhadijeh - * @license MIT - */ -/* eslint-disable no-proto */ - -'use strict' - -var base64 = require('base64-js') -var ieee754 = require('ieee754') - -exports.Buffer = Buffer -exports.SlowBuffer = SlowBuffer -exports.INSPECT_MAX_BYTES = 50 - -var K_MAX_LENGTH = 0x7fffffff -exports.kMaxLength = K_MAX_LENGTH - -/** - * If `Buffer.TYPED_ARRAY_SUPPORT`: - * === true Use Uint8Array implementation (fastest) - * === false Print warning and recommend using `buffer` v4.x which has an Object - * implementation (most compatible, even IE6) - * - * Browsers that support typed arrays are IE 10+, Firefox 4+, Chrome 7+, Safari 5.1+, - * Opera 11.6+, iOS 4.2+. - * - * We report that the browser does not support typed arrays if the are not subclassable - * using __proto__. Firefox 4-29 lacks support for adding new properties to `Uint8Array` - * (See: https://bugzilla.mozilla.org/show_bug.cgi?id=695438). IE 10 lacks support - * for __proto__ and has a buggy typed array implementation. - */ -Buffer.TYPED_ARRAY_SUPPORT = typedArraySupport() - -if (!Buffer.TYPED_ARRAY_SUPPORT && typeof console !== 'undefined' && - typeof console.error === 'function') { - console.error( - 'This browser lacks typed array (Uint8Array) support which is required by ' + - '`buffer` v5.x. Use `buffer` v4.x if you require old browser support.' - ) -} - -function typedArraySupport () { - // Can typed array instances can be augmented? - try { - var arr = new Uint8Array(1) - arr.__proto__ = { __proto__: Uint8Array.prototype, foo: function () { return 42 } } - return arr.foo() === 42 - } catch (e) { - return false - } -} - -Object.defineProperty(Buffer.prototype, 'parent', { - enumerable: true, - get: function () { - if (!Buffer.isBuffer(this)) return undefined - return this.buffer - } -}) - -Object.defineProperty(Buffer.prototype, 'offset', { - enumerable: true, - get: function () { - if (!Buffer.isBuffer(this)) return undefined - return this.byteOffset - } -}) - -function createBuffer (length) { - if (length > K_MAX_LENGTH) { - throw new RangeError('The value "' + length + '" is invalid for option "size"') - } - // Return an augmented `Uint8Array` instance - var buf = new Uint8Array(length) - buf.__proto__ = Buffer.prototype - return buf -} - -/** - * The Buffer constructor returns instances of `Uint8Array` that have their - * prototype changed to `Buffer.prototype`. Furthermore, `Buffer` is a subclass of - * `Uint8Array`, so the returned instances will have all the node `Buffer` methods - * and the `Uint8Array` methods. Square bracket notation works as expected -- it - * returns a single octet. - * - * The `Uint8Array` prototype remains unmodified. - */ - -function Buffer (arg, encodingOrOffset, length) { - // Common case. - if (typeof arg === 'number') { - if (typeof encodingOrOffset === 'string') { - throw new TypeError( - 'The "string" argument must be of type string. Received type number' - ) - } - return allocUnsafe(arg) - } - return from(arg, encodingOrOffset, length) -} - -// Fix subarray() in ES2016. See: https://github.com/feross/buffer/pull/97 -if (typeof Symbol !== 'undefined' && Symbol.species != null && - Buffer[Symbol.species] === Buffer) { - Object.defineProperty(Buffer, Symbol.species, { - value: null, - configurable: true, - enumerable: false, - writable: false - }) -} - -Buffer.poolSize = 8192 // not used by this implementation - -function from (value, encodingOrOffset, length) { - if (typeof value === 'string') { - return fromString(value, encodingOrOffset) - } - - if (ArrayBuffer.isView(value)) { - return fromArrayLike(value) - } - - if (value == null) { - throw TypeError( - 'The first argument must be one of type string, Buffer, ArrayBuffer, Array, ' + - 'or Array-like Object. Received type ' + (typeof value) - ) - } - - if (isInstance(value, ArrayBuffer) || - (value && isInstance(value.buffer, ArrayBuffer))) { - return fromArrayBuffer(value, encodingOrOffset, length) - } - - if (typeof value === 'number') { - throw new TypeError( - 'The "value" argument must not be of type number. Received type number' - ) - } - - var valueOf = value.valueOf && value.valueOf() - if (valueOf != null && valueOf !== value) { - return Buffer.from(valueOf, encodingOrOffset, length) - } - - var b = fromObject(value) - if (b) return b - - if (typeof Symbol !== 'undefined' && Symbol.toPrimitive != null && - typeof value[Symbol.toPrimitive] === 'function') { - return Buffer.from( - value[Symbol.toPrimitive]('string'), encodingOrOffset, length - ) - } - - throw new TypeError( - 'The first argument must be one of type string, Buffer, ArrayBuffer, Array, ' + - 'or Array-like Object. Received type ' + (typeof value) - ) -} - -/** - * Functionally equivalent to Buffer(arg, encoding) but throws a TypeError - * if value is a number. - * Buffer.from(str[, encoding]) - * Buffer.from(array) - * Buffer.from(buffer) - * Buffer.from(arrayBuffer[, byteOffset[, length]]) - **/ -Buffer.from = function (value, encodingOrOffset, length) { - return from(value, encodingOrOffset, length) -} - -// Note: Change prototype *after* Buffer.from is defined to workaround Chrome bug: -// https://github.com/feross/buffer/pull/148 -Buffer.prototype.__proto__ = Uint8Array.prototype -Buffer.__proto__ = Uint8Array - -function assertSize (size) { - if (typeof size !== 'number') { - throw new TypeError('"size" argument must be of type number') - } else if (size < 0) { - throw new RangeError('The value "' + size + '" is invalid for option "size"') - } -} - -function alloc (size, fill, encoding) { - assertSize(size) - if (size <= 0) { - return createBuffer(size) - } - if (fill !== undefined) { - // Only pay attention to encoding if it's a string. This - // prevents accidentally sending in a number that would - // be interpretted as a start offset. - return typeof encoding === 'string' - ? createBuffer(size).fill(fill, encoding) - : createBuffer(size).fill(fill) - } - return createBuffer(size) -} - -/** - * Creates a new filled Buffer instance. - * alloc(size[, fill[, encoding]]) - **/ -Buffer.alloc = function (size, fill, encoding) { - return alloc(size, fill, encoding) -} - -function allocUnsafe (size) { - assertSize(size) - return createBuffer(size < 0 ? 0 : checked(size) | 0) -} - -/** - * Equivalent to Buffer(num), by default creates a non-zero-filled Buffer instance. - * */ -Buffer.allocUnsafe = function (size) { - return allocUnsafe(size) -} -/** - * Equivalent to SlowBuffer(num), by default creates a non-zero-filled Buffer instance. - */ -Buffer.allocUnsafeSlow = function (size) { - return allocUnsafe(size) -} - -function fromString (string, encoding) { - if (typeof encoding !== 'string' || encoding === '') { - encoding = 'utf8' - } - - if (!Buffer.isEncoding(encoding)) { - throw new TypeError('Unknown encoding: ' + encoding) - } - - var length = byteLength(string, encoding) | 0 - var buf = createBuffer(length) - - var actual = buf.write(string, encoding) - - if (actual !== length) { - // Writing a hex string, for example, that contains invalid characters will - // cause everything after the first invalid character to be ignored. (e.g. - // 'abxxcd' will be treated as 'ab') - buf = buf.slice(0, actual) - } - - return buf -} - -function fromArrayLike (array) { - var length = array.length < 0 ? 0 : checked(array.length) | 0 - var buf = createBuffer(length) - for (var i = 0; i < length; i += 1) { - buf[i] = array[i] & 255 - } - return buf -} - -function fromArrayBuffer (array, byteOffset, length) { - if (byteOffset < 0 || array.byteLength < byteOffset) { - throw new RangeError('"offset" is outside of buffer bounds') - } - - if (array.byteLength < byteOffset + (length || 0)) { - throw new RangeError('"length" is outside of buffer bounds') - } - - var buf - if (byteOffset === undefined && length === undefined) { - buf = new Uint8Array(array) - } else if (length === undefined) { - buf = new Uint8Array(array, byteOffset) - } else { - buf = new Uint8Array(array, byteOffset, length) - } - - // Return an augmented `Uint8Array` instance - buf.__proto__ = Buffer.prototype - return buf -} - -function fromObject (obj) { - if (Buffer.isBuffer(obj)) { - var len = checked(obj.length) | 0 - var buf = createBuffer(len) - - if (buf.length === 0) { - return buf - } - - obj.copy(buf, 0, 0, len) - return buf - } - - if (obj.length !== undefined) { - if (typeof obj.length !== 'number' || numberIsNaN(obj.length)) { - return createBuffer(0) - } - return fromArrayLike(obj) - } - - if (obj.type === 'Buffer' && Array.isArray(obj.data)) { - return fromArrayLike(obj.data) - } -} - -function checked (length) { - // Note: cannot use `length < K_MAX_LENGTH` here because that fails when - // length is NaN (which is otherwise coerced to zero.) - if (length >= K_MAX_LENGTH) { - throw new RangeError('Attempt to allocate Buffer larger than maximum ' + - 'size: 0x' + K_MAX_LENGTH.toString(16) + ' bytes') - } - return length | 0 -} - -function SlowBuffer (length) { - if (+length != length) { // eslint-disable-line eqeqeq - length = 0 - } - return Buffer.alloc(+length) -} - -Buffer.isBuffer = function isBuffer (b) { - return b != null && b._isBuffer === true && - b !== Buffer.prototype // so Buffer.isBuffer(Buffer.prototype) will be false -} - -Buffer.compare = function compare (a, b) { - if (isInstance(a, Uint8Array)) a = Buffer.from(a, a.offset, a.byteLength) - if (isInstance(b, Uint8Array)) b = Buffer.from(b, b.offset, b.byteLength) - if (!Buffer.isBuffer(a) || !Buffer.isBuffer(b)) { - throw new TypeError( - 'The "buf1", "buf2" arguments must be one of type Buffer or Uint8Array' - ) - } - - if (a === b) return 0 - - var x = a.length - var y = b.length - - for (var i = 0, len = Math.min(x, y); i < len; ++i) { - if (a[i] !== b[i]) { - x = a[i] - y = b[i] - break - } - } - - if (x < y) return -1 - if (y < x) return 1 - return 0 -} - -Buffer.isEncoding = function isEncoding (encoding) { - switch (String(encoding).toLowerCase()) { - case 'hex': - case 'utf8': - case 'utf-8': - case 'ascii': - case 'latin1': - case 'binary': - case 'base64': - case 'ucs2': - case 'ucs-2': - case 'utf16le': - case 'utf-16le': - return true - default: - return false - } -} - -Buffer.concat = function concat (list, length) { - if (!Array.isArray(list)) { - throw new TypeError('"list" argument must be an Array of Buffers') - } - - if (list.length === 0) { - return Buffer.alloc(0) - } - - var i - if (length === undefined) { - length = 0 - for (i = 0; i < list.length; ++i) { - length += list[i].length - } - } - - var buffer = Buffer.allocUnsafe(length) - var pos = 0 - for (i = 0; i < list.length; ++i) { - var buf = list[i] - if (isInstance(buf, Uint8Array)) { - buf = Buffer.from(buf) - } - if (!Buffer.isBuffer(buf)) { - throw new TypeError('"list" argument must be an Array of Buffers') - } - buf.copy(buffer, pos) - pos += buf.length - } - return buffer -} - -function byteLength (string, encoding) { - if (Buffer.isBuffer(string)) { - return string.length - } - if (ArrayBuffer.isView(string) || isInstance(string, ArrayBuffer)) { - return string.byteLength - } - if (typeof string !== 'string') { - throw new TypeError( - 'The "string" argument must be one of type string, Buffer, or ArrayBuffer. ' + - 'Received type ' + typeof string - ) - } - - var len = string.length - var mustMatch = (arguments.length > 2 && arguments[2] === true) - if (!mustMatch && len === 0) return 0 - - // Use a for loop to avoid recursion - var loweredCase = false - for (;;) { - switch (encoding) { - case 'ascii': - case 'latin1': - case 'binary': - return len - case 'utf8': - case 'utf-8': - return utf8ToBytes(string).length - case 'ucs2': - case 'ucs-2': - case 'utf16le': - case 'utf-16le': - return len * 2 - case 'hex': - return len >>> 1 - case 'base64': - return base64ToBytes(string).length - default: - if (loweredCase) { - return mustMatch ? -1 : utf8ToBytes(string).length // assume utf8 - } - encoding = ('' + encoding).toLowerCase() - loweredCase = true - } - } -} -Buffer.byteLength = byteLength - -function slowToString (encoding, start, end) { - var loweredCase = false - - // No need to verify that "this.length <= MAX_UINT32" since it's a read-only - // property of a typed array. - - // This behaves neither like String nor Uint8Array in that we set start/end - // to their upper/lower bounds if the value passed is out of range. - // undefined is handled specially as per ECMA-262 6th Edition, - // Section 13.3.3.7 Runtime Semantics: KeyedBindingInitialization. - if (start === undefined || start < 0) { - start = 0 - } - // Return early if start > this.length. Done here to prevent potential uint32 - // coercion fail below. - if (start > this.length) { - return '' - } - - if (end === undefined || end > this.length) { - end = this.length - } - - if (end <= 0) { - return '' - } - - // Force coersion to uint32. This will also coerce falsey/NaN values to 0. - end >>>= 0 - start >>>= 0 - - if (end <= start) { - return '' - } - - if (!encoding) encoding = 'utf8' - - while (true) { - switch (encoding) { - case 'hex': - return hexSlice(this, start, end) - - case 'utf8': - case 'utf-8': - return utf8Slice(this, start, end) - - case 'ascii': - return asciiSlice(this, start, end) - - case 'latin1': - case 'binary': - return latin1Slice(this, start, end) - - case 'base64': - return base64Slice(this, start, end) - - case 'ucs2': - case 'ucs-2': - case 'utf16le': - case 'utf-16le': - return utf16leSlice(this, start, end) - - default: - if (loweredCase) throw new TypeError('Unknown encoding: ' + encoding) - encoding = (encoding + '').toLowerCase() - loweredCase = true - } - } -} - -// This property is used by `Buffer.isBuffer` (and the `is-buffer` npm package) -// to detect a Buffer instance. It's not possible to use `instanceof Buffer` -// reliably in a browserify context because there could be multiple different -// copies of the 'buffer' package in use. This method works even for Buffer -// instances that were created from another copy of the `buffer` package. -// See: https://github.com/feross/buffer/issues/154 -Buffer.prototype._isBuffer = true - -function swap (b, n, m) { - var i = b[n] - b[n] = b[m] - b[m] = i -} - -Buffer.prototype.swap16 = function swap16 () { - var len = this.length - if (len % 2 !== 0) { - throw new RangeError('Buffer size must be a multiple of 16-bits') - } - for (var i = 0; i < len; i += 2) { - swap(this, i, i + 1) - } - return this -} - -Buffer.prototype.swap32 = function swap32 () { - var len = this.length - if (len % 4 !== 0) { - throw new RangeError('Buffer size must be a multiple of 32-bits') - } - for (var i = 0; i < len; i += 4) { - swap(this, i, i + 3) - swap(this, i + 1, i + 2) - } - return this -} - -Buffer.prototype.swap64 = function swap64 () { - var len = this.length - if (len % 8 !== 0) { - throw new RangeError('Buffer size must be a multiple of 64-bits') - } - for (var i = 0; i < len; i += 8) { - swap(this, i, i + 7) - swap(this, i + 1, i + 6) - swap(this, i + 2, i + 5) - swap(this, i + 3, i + 4) - } - return this -} - -Buffer.prototype.toString = function toString () { - var length = this.length - if (length === 0) return '' - if (arguments.length === 0) return utf8Slice(this, 0, length) - return slowToString.apply(this, arguments) -} - -Buffer.prototype.toLocaleString = Buffer.prototype.toString - -Buffer.prototype.equals = function equals (b) { - if (!Buffer.isBuffer(b)) throw new TypeError('Argument must be a Buffer') - if (this === b) return true - return Buffer.compare(this, b) === 0 -} - -Buffer.prototype.inspect = function inspect () { - var str = '' - var max = exports.INSPECT_MAX_BYTES - str = this.toString('hex', 0, max).replace(/(.{2})/g, '$1 ').trim() - if (this.length > max) str += ' ... ' - return '' -} - -Buffer.prototype.compare = function compare (target, start, end, thisStart, thisEnd) { - if (isInstance(target, Uint8Array)) { - target = Buffer.from(target, target.offset, target.byteLength) - } - if (!Buffer.isBuffer(target)) { - throw new TypeError( - 'The "target" argument must be one of type Buffer or Uint8Array. ' + - 'Received type ' + (typeof target) - ) - } - - if (start === undefined) { - start = 0 - } - if (end === undefined) { - end = target ? target.length : 0 - } - if (thisStart === undefined) { - thisStart = 0 - } - if (thisEnd === undefined) { - thisEnd = this.length - } - - if (start < 0 || end > target.length || thisStart < 0 || thisEnd > this.length) { - throw new RangeError('out of range index') - } - - if (thisStart >= thisEnd && start >= end) { - return 0 - } - if (thisStart >= thisEnd) { - return -1 - } - if (start >= end) { - return 1 - } - - start >>>= 0 - end >>>= 0 - thisStart >>>= 0 - thisEnd >>>= 0 - - if (this === target) return 0 - - var x = thisEnd - thisStart - var y = end - start - var len = Math.min(x, y) - - var thisCopy = this.slice(thisStart, thisEnd) - var targetCopy = target.slice(start, end) - - for (var i = 0; i < len; ++i) { - if (thisCopy[i] !== targetCopy[i]) { - x = thisCopy[i] - y = targetCopy[i] - break - } - } - - if (x < y) return -1 - if (y < x) return 1 - return 0 -} - -// Finds either the first index of `val` in `buffer` at offset >= `byteOffset`, -// OR the last index of `val` in `buffer` at offset <= `byteOffset`. -// -// Arguments: -// - buffer - a Buffer to search -// - val - a string, Buffer, or number -// - byteOffset - an index into `buffer`; will be clamped to an int32 -// - encoding - an optional encoding, relevant is val is a string -// - dir - true for indexOf, false for lastIndexOf -function bidirectionalIndexOf (buffer, val, byteOffset, encoding, dir) { - // Empty buffer means no match - if (buffer.length === 0) return -1 - - // Normalize byteOffset - if (typeof byteOffset === 'string') { - encoding = byteOffset - byteOffset = 0 - } else if (byteOffset > 0x7fffffff) { - byteOffset = 0x7fffffff - } else if (byteOffset < -0x80000000) { - byteOffset = -0x80000000 - } - byteOffset = +byteOffset // Coerce to Number. - if (numberIsNaN(byteOffset)) { - // byteOffset: it it's undefined, null, NaN, "foo", etc, search whole buffer - byteOffset = dir ? 0 : (buffer.length - 1) - } - - // Normalize byteOffset: negative offsets start from the end of the buffer - if (byteOffset < 0) byteOffset = buffer.length + byteOffset - if (byteOffset >= buffer.length) { - if (dir) return -1 - else byteOffset = buffer.length - 1 - } else if (byteOffset < 0) { - if (dir) byteOffset = 0 - else return -1 - } - - // Normalize val - if (typeof val === 'string') { - val = Buffer.from(val, encoding) - } - - // Finally, search either indexOf (if dir is true) or lastIndexOf - if (Buffer.isBuffer(val)) { - // Special case: looking for empty string/buffer always fails - if (val.length === 0) { - return -1 - } - return arrayIndexOf(buffer, val, byteOffset, encoding, dir) - } else if (typeof val === 'number') { - val = val & 0xFF // Search for a byte value [0-255] - if (typeof Uint8Array.prototype.indexOf === 'function') { - if (dir) { - return Uint8Array.prototype.indexOf.call(buffer, val, byteOffset) - } else { - return Uint8Array.prototype.lastIndexOf.call(buffer, val, byteOffset) - } - } - return arrayIndexOf(buffer, [ val ], byteOffset, encoding, dir) - } - - throw new TypeError('val must be string, number or Buffer') -} - -function arrayIndexOf (arr, val, byteOffset, encoding, dir) { - var indexSize = 1 - var arrLength = arr.length - var valLength = val.length - - if (encoding !== undefined) { - encoding = String(encoding).toLowerCase() - if (encoding === 'ucs2' || encoding === 'ucs-2' || - encoding === 'utf16le' || encoding === 'utf-16le') { - if (arr.length < 2 || val.length < 2) { - return -1 - } - indexSize = 2 - arrLength /= 2 - valLength /= 2 - byteOffset /= 2 - } - } - - function read (buf, i) { - if (indexSize === 1) { - return buf[i] - } else { - return buf.readUInt16BE(i * indexSize) - } - } - - var i - if (dir) { - var foundIndex = -1 - for (i = byteOffset; i < arrLength; i++) { - if (read(arr, i) === read(val, foundIndex === -1 ? 0 : i - foundIndex)) { - if (foundIndex === -1) foundIndex = i - if (i - foundIndex + 1 === valLength) return foundIndex * indexSize - } else { - if (foundIndex !== -1) i -= i - foundIndex - foundIndex = -1 - } - } - } else { - if (byteOffset + valLength > arrLength) byteOffset = arrLength - valLength - for (i = byteOffset; i >= 0; i--) { - var found = true - for (var j = 0; j < valLength; j++) { - if (read(arr, i + j) !== read(val, j)) { - found = false - break - } - } - if (found) return i - } - } - - return -1 -} - -Buffer.prototype.includes = function includes (val, byteOffset, encoding) { - return this.indexOf(val, byteOffset, encoding) !== -1 -} - -Buffer.prototype.indexOf = function indexOf (val, byteOffset, encoding) { - return bidirectionalIndexOf(this, val, byteOffset, encoding, true) -} - -Buffer.prototype.lastIndexOf = function lastIndexOf (val, byteOffset, encoding) { - return bidirectionalIndexOf(this, val, byteOffset, encoding, false) -} - -function hexWrite (buf, string, offset, length) { - offset = Number(offset) || 0 - var remaining = buf.length - offset - if (!length) { - length = remaining - } else { - length = Number(length) - if (length > remaining) { - length = remaining - } - } - - var strLen = string.length - - if (length > strLen / 2) { - length = strLen / 2 - } - for (var i = 0; i < length; ++i) { - var parsed = parseInt(string.substr(i * 2, 2), 16) - if (numberIsNaN(parsed)) return i - buf[offset + i] = parsed - } - return i -} - -function utf8Write (buf, string, offset, length) { - return blitBuffer(utf8ToBytes(string, buf.length - offset), buf, offset, length) -} - -function asciiWrite (buf, string, offset, length) { - return blitBuffer(asciiToBytes(string), buf, offset, length) -} - -function latin1Write (buf, string, offset, length) { - return asciiWrite(buf, string, offset, length) -} - -function base64Write (buf, string, offset, length) { - return blitBuffer(base64ToBytes(string), buf, offset, length) -} - -function ucs2Write (buf, string, offset, length) { - return blitBuffer(utf16leToBytes(string, buf.length - offset), buf, offset, length) -} - -Buffer.prototype.write = function write (string, offset, length, encoding) { - // Buffer#write(string) - if (offset === undefined) { - encoding = 'utf8' - length = this.length - offset = 0 - // Buffer#write(string, encoding) - } else if (length === undefined && typeof offset === 'string') { - encoding = offset - length = this.length - offset = 0 - // Buffer#write(string, offset[, length][, encoding]) - } else if (isFinite(offset)) { - offset = offset >>> 0 - if (isFinite(length)) { - length = length >>> 0 - if (encoding === undefined) encoding = 'utf8' - } else { - encoding = length - length = undefined - } - } else { - throw new Error( - 'Buffer.write(string, encoding, offset[, length]) is no longer supported' - ) - } - - var remaining = this.length - offset - if (length === undefined || length > remaining) length = remaining - - if ((string.length > 0 && (length < 0 || offset < 0)) || offset > this.length) { - throw new RangeError('Attempt to write outside buffer bounds') - } - - if (!encoding) encoding = 'utf8' - - var loweredCase = false - for (;;) { - switch (encoding) { - case 'hex': - return hexWrite(this, string, offset, length) - - case 'utf8': - case 'utf-8': - return utf8Write(this, string, offset, length) - - case 'ascii': - return asciiWrite(this, string, offset, length) - - case 'latin1': - case 'binary': - return latin1Write(this, string, offset, length) - - case 'base64': - // Warning: maxLength not taken into account in base64Write - return base64Write(this, string, offset, length) - - case 'ucs2': - case 'ucs-2': - case 'utf16le': - case 'utf-16le': - return ucs2Write(this, string, offset, length) - - default: - if (loweredCase) throw new TypeError('Unknown encoding: ' + encoding) - encoding = ('' + encoding).toLowerCase() - loweredCase = true - } - } -} - -Buffer.prototype.toJSON = function toJSON () { - return { - type: 'Buffer', - data: Array.prototype.slice.call(this._arr || this, 0) - } -} - -function base64Slice (buf, start, end) { - if (start === 0 && end === buf.length) { - return base64.fromByteArray(buf) - } else { - return base64.fromByteArray(buf.slice(start, end)) - } -} - -function utf8Slice (buf, start, end) { - end = Math.min(buf.length, end) - var res = [] - - var i = start - while (i < end) { - var firstByte = buf[i] - var codePoint = null - var bytesPerSequence = (firstByte > 0xEF) ? 4 - : (firstByte > 0xDF) ? 3 - : (firstByte > 0xBF) ? 2 - : 1 - - if (i + bytesPerSequence <= end) { - var secondByte, thirdByte, fourthByte, tempCodePoint - - switch (bytesPerSequence) { - case 1: - if (firstByte < 0x80) { - codePoint = firstByte - } - break - case 2: - secondByte = buf[i + 1] - if ((secondByte & 0xC0) === 0x80) { - tempCodePoint = (firstByte & 0x1F) << 0x6 | (secondByte & 0x3F) - if (tempCodePoint > 0x7F) { - codePoint = tempCodePoint - } - } - break - case 3: - secondByte = buf[i + 1] - thirdByte = buf[i + 2] - if ((secondByte & 0xC0) === 0x80 && (thirdByte & 0xC0) === 0x80) { - tempCodePoint = (firstByte & 0xF) << 0xC | (secondByte & 0x3F) << 0x6 | (thirdByte & 0x3F) - if (tempCodePoint > 0x7FF && (tempCodePoint < 0xD800 || tempCodePoint > 0xDFFF)) { - codePoint = tempCodePoint - } - } - break - case 4: - secondByte = buf[i + 1] - thirdByte = buf[i + 2] - fourthByte = buf[i + 3] - if ((secondByte & 0xC0) === 0x80 && (thirdByte & 0xC0) === 0x80 && (fourthByte & 0xC0) === 0x80) { - tempCodePoint = (firstByte & 0xF) << 0x12 | (secondByte & 0x3F) << 0xC | (thirdByte & 0x3F) << 0x6 | (fourthByte & 0x3F) - if (tempCodePoint > 0xFFFF && tempCodePoint < 0x110000) { - codePoint = tempCodePoint - } - } - } - } - - if (codePoint === null) { - // we did not generate a valid codePoint so insert a - // replacement char (U+FFFD) and advance only 1 byte - codePoint = 0xFFFD - bytesPerSequence = 1 - } else if (codePoint > 0xFFFF) { - // encode to utf16 (surrogate pair dance) - codePoint -= 0x10000 - res.push(codePoint >>> 10 & 0x3FF | 0xD800) - codePoint = 0xDC00 | codePoint & 0x3FF - } - - res.push(codePoint) - i += bytesPerSequence - } - - return decodeCodePointsArray(res) -} - -// Based on http://stackoverflow.com/a/22747272/680742, the browser with -// the lowest limit is Chrome, with 0x10000 args. -// We go 1 magnitude less, for safety -var MAX_ARGUMENTS_LENGTH = 0x1000 - -function decodeCodePointsArray (codePoints) { - var len = codePoints.length - if (len <= MAX_ARGUMENTS_LENGTH) { - return String.fromCharCode.apply(String, codePoints) // avoid extra slice() - } - - // Decode in chunks to avoid "call stack size exceeded". - var res = '' - var i = 0 - while (i < len) { - res += String.fromCharCode.apply( - String, - codePoints.slice(i, i += MAX_ARGUMENTS_LENGTH) - ) - } - return res -} - -function asciiSlice (buf, start, end) { - var ret = '' - end = Math.min(buf.length, end) - - for (var i = start; i < end; ++i) { - ret += String.fromCharCode(buf[i] & 0x7F) - } - return ret -} - -function latin1Slice (buf, start, end) { - var ret = '' - end = Math.min(buf.length, end) - - for (var i = start; i < end; ++i) { - ret += String.fromCharCode(buf[i]) - } - return ret -} - -function hexSlice (buf, start, end) { - var len = buf.length - - if (!start || start < 0) start = 0 - if (!end || end < 0 || end > len) end = len - - var out = '' - for (var i = start; i < end; ++i) { - out += toHex(buf[i]) - } - return out -} - -function utf16leSlice (buf, start, end) { - var bytes = buf.slice(start, end) - var res = '' - for (var i = 0; i < bytes.length; i += 2) { - res += String.fromCharCode(bytes[i] + (bytes[i + 1] * 256)) - } - return res -} - -Buffer.prototype.slice = function slice (start, end) { - var len = this.length - start = ~~start - end = end === undefined ? len : ~~end - - if (start < 0) { - start += len - if (start < 0) start = 0 - } else if (start > len) { - start = len - } - - if (end < 0) { - end += len - if (end < 0) end = 0 - } else if (end > len) { - end = len - } - - if (end < start) end = start - - var newBuf = this.subarray(start, end) - // Return an augmented `Uint8Array` instance - newBuf.__proto__ = Buffer.prototype - return newBuf -} - -/* - * Need to make sure that buffer isn't trying to write out of bounds. - */ -function checkOffset (offset, ext, length) { - if ((offset % 1) !== 0 || offset < 0) throw new RangeError('offset is not uint') - if (offset + ext > length) throw new RangeError('Trying to access beyond buffer length') -} - -Buffer.prototype.readUIntLE = function readUIntLE (offset, byteLength, noAssert) { - offset = offset >>> 0 - byteLength = byteLength >>> 0 - if (!noAssert) checkOffset(offset, byteLength, this.length) - - var val = this[offset] - var mul = 1 - var i = 0 - while (++i < byteLength && (mul *= 0x100)) { - val += this[offset + i] * mul - } - - return val -} - -Buffer.prototype.readUIntBE = function readUIntBE (offset, byteLength, noAssert) { - offset = offset >>> 0 - byteLength = byteLength >>> 0 - if (!noAssert) { - checkOffset(offset, byteLength, this.length) - } - - var val = this[offset + --byteLength] - var mul = 1 - while (byteLength > 0 && (mul *= 0x100)) { - val += this[offset + --byteLength] * mul - } - - return val -} - -Buffer.prototype.readUInt8 = function readUInt8 (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 1, this.length) - return this[offset] -} - -Buffer.prototype.readUInt16LE = function readUInt16LE (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 2, this.length) - return this[offset] | (this[offset + 1] << 8) -} - -Buffer.prototype.readUInt16BE = function readUInt16BE (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 2, this.length) - return (this[offset] << 8) | this[offset + 1] -} - -Buffer.prototype.readUInt32LE = function readUInt32LE (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 4, this.length) - - return ((this[offset]) | - (this[offset + 1] << 8) | - (this[offset + 2] << 16)) + - (this[offset + 3] * 0x1000000) -} - -Buffer.prototype.readUInt32BE = function readUInt32BE (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 4, this.length) - - return (this[offset] * 0x1000000) + - ((this[offset + 1] << 16) | - (this[offset + 2] << 8) | - this[offset + 3]) -} - -Buffer.prototype.readIntLE = function readIntLE (offset, byteLength, noAssert) { - offset = offset >>> 0 - byteLength = byteLength >>> 0 - if (!noAssert) checkOffset(offset, byteLength, this.length) - - var val = this[offset] - var mul = 1 - var i = 0 - while (++i < byteLength && (mul *= 0x100)) { - val += this[offset + i] * mul - } - mul *= 0x80 - - if (val >= mul) val -= Math.pow(2, 8 * byteLength) - - return val -} - -Buffer.prototype.readIntBE = function readIntBE (offset, byteLength, noAssert) { - offset = offset >>> 0 - byteLength = byteLength >>> 0 - if (!noAssert) checkOffset(offset, byteLength, this.length) - - var i = byteLength - var mul = 1 - var val = this[offset + --i] - while (i > 0 && (mul *= 0x100)) { - val += this[offset + --i] * mul - } - mul *= 0x80 - - if (val >= mul) val -= Math.pow(2, 8 * byteLength) - - return val -} - -Buffer.prototype.readInt8 = function readInt8 (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 1, this.length) - if (!(this[offset] & 0x80)) return (this[offset]) - return ((0xff - this[offset] + 1) * -1) -} - -Buffer.prototype.readInt16LE = function readInt16LE (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 2, this.length) - var val = this[offset] | (this[offset + 1] << 8) - return (val & 0x8000) ? val | 0xFFFF0000 : val -} - -Buffer.prototype.readInt16BE = function readInt16BE (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 2, this.length) - var val = this[offset + 1] | (this[offset] << 8) - return (val & 0x8000) ? val | 0xFFFF0000 : val -} - -Buffer.prototype.readInt32LE = function readInt32LE (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 4, this.length) - - return (this[offset]) | - (this[offset + 1] << 8) | - (this[offset + 2] << 16) | - (this[offset + 3] << 24) -} - -Buffer.prototype.readInt32BE = function readInt32BE (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 4, this.length) - - return (this[offset] << 24) | - (this[offset + 1] << 16) | - (this[offset + 2] << 8) | - (this[offset + 3]) -} - -Buffer.prototype.readFloatLE = function readFloatLE (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 4, this.length) - return ieee754.read(this, offset, true, 23, 4) -} - -Buffer.prototype.readFloatBE = function readFloatBE (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 4, this.length) - return ieee754.read(this, offset, false, 23, 4) -} - -Buffer.prototype.readDoubleLE = function readDoubleLE (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 8, this.length) - return ieee754.read(this, offset, true, 52, 8) -} - -Buffer.prototype.readDoubleBE = function readDoubleBE (offset, noAssert) { - offset = offset >>> 0 - if (!noAssert) checkOffset(offset, 8, this.length) - return ieee754.read(this, offset, false, 52, 8) -} - -function checkInt (buf, value, offset, ext, max, min) { - if (!Buffer.isBuffer(buf)) throw new TypeError('"buffer" argument must be a Buffer instance') - if (value > max || value < min) throw new RangeError('"value" argument is out of bounds') - if (offset + ext > buf.length) throw new RangeError('Index out of range') -} - -Buffer.prototype.writeUIntLE = function writeUIntLE (value, offset, byteLength, noAssert) { - value = +value - offset = offset >>> 0 - byteLength = byteLength >>> 0 - if (!noAssert) { - var maxBytes = Math.pow(2, 8 * byteLength) - 1 - checkInt(this, value, offset, byteLength, maxBytes, 0) - } - - var mul = 1 - var i = 0 - this[offset] = value & 0xFF - while (++i < byteLength && (mul *= 0x100)) { - this[offset + i] = (value / mul) & 0xFF - } - - return offset + byteLength -} - -Buffer.prototype.writeUIntBE = function writeUIntBE (value, offset, byteLength, noAssert) { - value = +value - offset = offset >>> 0 - byteLength = byteLength >>> 0 - if (!noAssert) { - var maxBytes = Math.pow(2, 8 * byteLength) - 1 - checkInt(this, value, offset, byteLength, maxBytes, 0) - } - - var i = byteLength - 1 - var mul = 1 - this[offset + i] = value & 0xFF - while (--i >= 0 && (mul *= 0x100)) { - this[offset + i] = (value / mul) & 0xFF - } - - return offset + byteLength -} - -Buffer.prototype.writeUInt8 = function writeUInt8 (value, offset, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) checkInt(this, value, offset, 1, 0xff, 0) - this[offset] = (value & 0xff) - return offset + 1 -} - -Buffer.prototype.writeUInt16LE = function writeUInt16LE (value, offset, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) checkInt(this, value, offset, 2, 0xffff, 0) - this[offset] = (value & 0xff) - this[offset + 1] = (value >>> 8) - return offset + 2 -} - -Buffer.prototype.writeUInt16BE = function writeUInt16BE (value, offset, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) checkInt(this, value, offset, 2, 0xffff, 0) - this[offset] = (value >>> 8) - this[offset + 1] = (value & 0xff) - return offset + 2 -} - -Buffer.prototype.writeUInt32LE = function writeUInt32LE (value, offset, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) checkInt(this, value, offset, 4, 0xffffffff, 0) - this[offset + 3] = (value >>> 24) - this[offset + 2] = (value >>> 16) - this[offset + 1] = (value >>> 8) - this[offset] = (value & 0xff) - return offset + 4 -} - -Buffer.prototype.writeUInt32BE = function writeUInt32BE (value, offset, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) checkInt(this, value, offset, 4, 0xffffffff, 0) - this[offset] = (value >>> 24) - this[offset + 1] = (value >>> 16) - this[offset + 2] = (value >>> 8) - this[offset + 3] = (value & 0xff) - return offset + 4 -} - -Buffer.prototype.writeIntLE = function writeIntLE (value, offset, byteLength, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) { - var limit = Math.pow(2, (8 * byteLength) - 1) - - checkInt(this, value, offset, byteLength, limit - 1, -limit) - } - - var i = 0 - var mul = 1 - var sub = 0 - this[offset] = value & 0xFF - while (++i < byteLength && (mul *= 0x100)) { - if (value < 0 && sub === 0 && this[offset + i - 1] !== 0) { - sub = 1 - } - this[offset + i] = ((value / mul) >> 0) - sub & 0xFF - } - - return offset + byteLength -} - -Buffer.prototype.writeIntBE = function writeIntBE (value, offset, byteLength, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) { - var limit = Math.pow(2, (8 * byteLength) - 1) - - checkInt(this, value, offset, byteLength, limit - 1, -limit) - } - - var i = byteLength - 1 - var mul = 1 - var sub = 0 - this[offset + i] = value & 0xFF - while (--i >= 0 && (mul *= 0x100)) { - if (value < 0 && sub === 0 && this[offset + i + 1] !== 0) { - sub = 1 - } - this[offset + i] = ((value / mul) >> 0) - sub & 0xFF - } - - return offset + byteLength -} - -Buffer.prototype.writeInt8 = function writeInt8 (value, offset, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) checkInt(this, value, offset, 1, 0x7f, -0x80) - if (value < 0) value = 0xff + value + 1 - this[offset] = (value & 0xff) - return offset + 1 -} - -Buffer.prototype.writeInt16LE = function writeInt16LE (value, offset, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) checkInt(this, value, offset, 2, 0x7fff, -0x8000) - this[offset] = (value & 0xff) - this[offset + 1] = (value >>> 8) - return offset + 2 -} - -Buffer.prototype.writeInt16BE = function writeInt16BE (value, offset, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) checkInt(this, value, offset, 2, 0x7fff, -0x8000) - this[offset] = (value >>> 8) - this[offset + 1] = (value & 0xff) - return offset + 2 -} - -Buffer.prototype.writeInt32LE = function writeInt32LE (value, offset, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) checkInt(this, value, offset, 4, 0x7fffffff, -0x80000000) - this[offset] = (value & 0xff) - this[offset + 1] = (value >>> 8) - this[offset + 2] = (value >>> 16) - this[offset + 3] = (value >>> 24) - return offset + 4 -} - -Buffer.prototype.writeInt32BE = function writeInt32BE (value, offset, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) checkInt(this, value, offset, 4, 0x7fffffff, -0x80000000) - if (value < 0) value = 0xffffffff + value + 1 - this[offset] = (value >>> 24) - this[offset + 1] = (value >>> 16) - this[offset + 2] = (value >>> 8) - this[offset + 3] = (value & 0xff) - return offset + 4 -} - -function checkIEEE754 (buf, value, offset, ext, max, min) { - if (offset + ext > buf.length) throw new RangeError('Index out of range') - if (offset < 0) throw new RangeError('Index out of range') -} - -function writeFloat (buf, value, offset, littleEndian, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) { - checkIEEE754(buf, value, offset, 4, 3.4028234663852886e+38, -3.4028234663852886e+38) - } - ieee754.write(buf, value, offset, littleEndian, 23, 4) - return offset + 4 -} - -Buffer.prototype.writeFloatLE = function writeFloatLE (value, offset, noAssert) { - return writeFloat(this, value, offset, true, noAssert) -} - -Buffer.prototype.writeFloatBE = function writeFloatBE (value, offset, noAssert) { - return writeFloat(this, value, offset, false, noAssert) -} - -function writeDouble (buf, value, offset, littleEndian, noAssert) { - value = +value - offset = offset >>> 0 - if (!noAssert) { - checkIEEE754(buf, value, offset, 8, 1.7976931348623157E+308, -1.7976931348623157E+308) - } - ieee754.write(buf, value, offset, littleEndian, 52, 8) - return offset + 8 -} - -Buffer.prototype.writeDoubleLE = function writeDoubleLE (value, offset, noAssert) { - return writeDouble(this, value, offset, true, noAssert) -} - -Buffer.prototype.writeDoubleBE = function writeDoubleBE (value, offset, noAssert) { - return writeDouble(this, value, offset, false, noAssert) -} - -// copy(targetBuffer, targetStart=0, sourceStart=0, sourceEnd=buffer.length) -Buffer.prototype.copy = function copy (target, targetStart, start, end) { - if (!Buffer.isBuffer(target)) throw new TypeError('argument should be a Buffer') - if (!start) start = 0 - if (!end && end !== 0) end = this.length - if (targetStart >= target.length) targetStart = target.length - if (!targetStart) targetStart = 0 - if (end > 0 && end < start) end = start - - // Copy 0 bytes; we're done - if (end === start) return 0 - if (target.length === 0 || this.length === 0) return 0 - - // Fatal error conditions - if (targetStart < 0) { - throw new RangeError('targetStart out of bounds') - } - if (start < 0 || start >= this.length) throw new RangeError('Index out of range') - if (end < 0) throw new RangeError('sourceEnd out of bounds') - - // Are we oob? - if (end > this.length) end = this.length - if (target.length - targetStart < end - start) { - end = target.length - targetStart + start - } - - var len = end - start - - if (this === target && typeof Uint8Array.prototype.copyWithin === 'function') { - // Use built-in when available, missing from IE11 - this.copyWithin(targetStart, start, end) - } else if (this === target && start < targetStart && targetStart < end) { - // descending copy from end - for (var i = len - 1; i >= 0; --i) { - target[i + targetStart] = this[i + start] - } - } else { - Uint8Array.prototype.set.call( - target, - this.subarray(start, end), - targetStart - ) - } - - return len -} - -// Usage: -// buffer.fill(number[, offset[, end]]) -// buffer.fill(buffer[, offset[, end]]) -// buffer.fill(string[, offset[, end]][, encoding]) -Buffer.prototype.fill = function fill (val, start, end, encoding) { - // Handle string cases: - if (typeof val === 'string') { - if (typeof start === 'string') { - encoding = start - start = 0 - end = this.length - } else if (typeof end === 'string') { - encoding = end - end = this.length - } - if (encoding !== undefined && typeof encoding !== 'string') { - throw new TypeError('encoding must be a string') - } - if (typeof encoding === 'string' && !Buffer.isEncoding(encoding)) { - throw new TypeError('Unknown encoding: ' + encoding) - } - if (val.length === 1) { - var code = val.charCodeAt(0) - if ((encoding === 'utf8' && code < 128) || - encoding === 'latin1') { - // Fast path: If `val` fits into a single byte, use that numeric value. - val = code - } - } - } else if (typeof val === 'number') { - val = val & 255 - } - - // Invalid ranges are not set to a default, so can range check early. - if (start < 0 || this.length < start || this.length < end) { - throw new RangeError('Out of range index') - } - - if (end <= start) { - return this - } - - start = start >>> 0 - end = end === undefined ? this.length : end >>> 0 - - if (!val) val = 0 - - var i - if (typeof val === 'number') { - for (i = start; i < end; ++i) { - this[i] = val - } - } else { - var bytes = Buffer.isBuffer(val) - ? val - : Buffer.from(val, encoding) - var len = bytes.length - if (len === 0) { - throw new TypeError('The value "' + val + - '" is invalid for argument "value"') - } - for (i = 0; i < end - start; ++i) { - this[i + start] = bytes[i % len] - } - } - - return this -} - -// HELPER FUNCTIONS -// ================ - -var INVALID_BASE64_RE = /[^+/0-9A-Za-z-_]/g - -function base64clean (str) { - // Node takes equal signs as end of the Base64 encoding - str = str.split('=')[0] - // Node strips out invalid characters like \n and \t from the string, base64-js does not - str = str.trim().replace(INVALID_BASE64_RE, '') - // Node converts strings with length < 2 to '' - if (str.length < 2) return '' - // Node allows for non-padded base64 strings (missing trailing ===), base64-js does not - while (str.length % 4 !== 0) { - str = str + '=' - } - return str -} - -function toHex (n) { - if (n < 16) return '0' + n.toString(16) - return n.toString(16) -} - -function utf8ToBytes (string, units) { - units = units || Infinity - var codePoint - var length = string.length - var leadSurrogate = null - var bytes = [] - - for (var i = 0; i < length; ++i) { - codePoint = string.charCodeAt(i) - - // is surrogate component - if (codePoint > 0xD7FF && codePoint < 0xE000) { - // last char was a lead - if (!leadSurrogate) { - // no lead yet - if (codePoint > 0xDBFF) { - // unexpected trail - if ((units -= 3) > -1) bytes.push(0xEF, 0xBF, 0xBD) - continue - } else if (i + 1 === length) { - // unpaired lead - if ((units -= 3) > -1) bytes.push(0xEF, 0xBF, 0xBD) - continue - } - - // valid lead - leadSurrogate = codePoint - - continue - } - - // 2 leads in a row - if (codePoint < 0xDC00) { - if ((units -= 3) > -1) bytes.push(0xEF, 0xBF, 0xBD) - leadSurrogate = codePoint - continue - } - - // valid surrogate pair - codePoint = (leadSurrogate - 0xD800 << 10 | codePoint - 0xDC00) + 0x10000 - } else if (leadSurrogate) { - // valid bmp char, but last char was a lead - if ((units -= 3) > -1) bytes.push(0xEF, 0xBF, 0xBD) - } - - leadSurrogate = null - - // encode utf8 - if (codePoint < 0x80) { - if ((units -= 1) < 0) break - bytes.push(codePoint) - } else if (codePoint < 0x800) { - if ((units -= 2) < 0) break - bytes.push( - codePoint >> 0x6 | 0xC0, - codePoint & 0x3F | 0x80 - ) - } else if (codePoint < 0x10000) { - if ((units -= 3) < 0) break - bytes.push( - codePoint >> 0xC | 0xE0, - codePoint >> 0x6 & 0x3F | 0x80, - codePoint & 0x3F | 0x80 - ) - } else if (codePoint < 0x110000) { - if ((units -= 4) < 0) break - bytes.push( - codePoint >> 0x12 | 0xF0, - codePoint >> 0xC & 0x3F | 0x80, - codePoint >> 0x6 & 0x3F | 0x80, - codePoint & 0x3F | 0x80 - ) - } else { - throw new Error('Invalid code point') - } - } - - return bytes -} - -function asciiToBytes (str) { - var byteArray = [] - for (var i = 0; i < str.length; ++i) { - // Node's code seems to be doing this and not & 0x7F.. - byteArray.push(str.charCodeAt(i) & 0xFF) - } - return byteArray -} - -function utf16leToBytes (str, units) { - var c, hi, lo - var byteArray = [] - for (var i = 0; i < str.length; ++i) { - if ((units -= 2) < 0) break - - c = str.charCodeAt(i) - hi = c >> 8 - lo = c % 256 - byteArray.push(lo) - byteArray.push(hi) - } - - return byteArray -} - -function base64ToBytes (str) { - return base64.toByteArray(base64clean(str)) -} - -function blitBuffer (src, dst, offset, length) { - for (var i = 0; i < length; ++i) { - if ((i + offset >= dst.length) || (i >= src.length)) break - dst[i + offset] = src[i] - } - return i -} - -// ArrayBuffer or Uint8Array objects from other contexts (i.e. iframes) do not pass -// the `instanceof` check but they should be treated as of that type. -// See: https://github.com/feross/buffer/issues/166 -function isInstance (obj, type) { - return obj instanceof type || - (obj != null && obj.constructor != null && obj.constructor.name != null && - obj.constructor.name === type.name) -} -function numberIsNaN (obj) { - // For IE11 support - return obj !== obj // eslint-disable-line no-self-compare -} - -}).call(this,require("buffer").Buffer) - -},{"base64-js":73,"buffer":124,"ieee754":216}],125:[function(require,module,exports){ -var Buffer = require('safe-buffer').Buffer -var Transform = require('stream').Transform -var StringDecoder = require('string_decoder').StringDecoder -var inherits = require('inherits') - -function CipherBase (hashMode) { - Transform.call(this) - this.hashMode = typeof hashMode === 'string' - if (this.hashMode) { - this[hashMode] = this._finalOrDigest - } else { - this.final = this._finalOrDigest - } - if (this._final) { - this.__final = this._final - this._final = null - } - this._decoder = null - this._encoding = null -} -inherits(CipherBase, Transform) - -CipherBase.prototype.update = function (data, inputEnc, outputEnc) { - if (typeof data === 'string') { - data = Buffer.from(data, inputEnc) - } - - var outData = this._update(data) - if (this.hashMode) return this - - if (outputEnc) { - outData = this._toString(outData, outputEnc) - } - - return outData -} - -CipherBase.prototype.setAutoPadding = function () {} -CipherBase.prototype.getAuthTag = function () { - throw new Error('trying to get auth tag in unsupported state') -} - -CipherBase.prototype.setAuthTag = function () { - throw new Error('trying to set auth tag in unsupported state') -} - -CipherBase.prototype.setAAD = function () { - throw new Error('trying to set aad in unsupported state') -} - -CipherBase.prototype._transform = function (data, _, next) { - var err - try { - if (this.hashMode) { - this._update(data) - } else { - this.push(this._update(data)) - } - } catch (e) { - err = e - } finally { - next(err) - } -} -CipherBase.prototype._flush = function (done) { - var err - try { - this.push(this.__final()) - } catch (e) { - err = e - } - - done(err) -} -CipherBase.prototype._finalOrDigest = function (outputEnc) { - var outData = this.__final() || Buffer.alloc(0) - if (outputEnc) { - outData = this._toString(outData, outputEnc, true) - } - return outData -} - -CipherBase.prototype._toString = function (value, enc, fin) { - if (!this._decoder) { - this._decoder = new StringDecoder(enc) - this._encoding = enc - } - - if (this._encoding !== enc) throw new Error('can\'t switch encodings') - - var out = this._decoder.write(value) - if (fin) { - out += this._decoder.end() - } - - return out -} - -module.exports = CipherBase - -},{"inherits":218,"safe-buffer":269,"stream":291,"string_decoder":292}],126:[function(require,module,exports){ -/** - * Slice reference. - */ - -var slice = [].slice; - -/** - * Bind `obj` to `fn`. - * - * @param {Object} obj - * @param {Function|String} fn or string - * @return {Function} - * @api public - */ - -module.exports = function(obj, fn){ - if ('string' == typeof fn) fn = obj[fn]; - if ('function' != typeof fn) throw new Error('bind() requires a function'); - var args = slice.call(arguments, 2); - return function(){ - return fn.apply(obj, args.concat(slice.call(arguments))); - } -}; - -},{}],127:[function(require,module,exports){ - -/** - * Expose `Emitter`. - */ - -if (typeof module !== 'undefined') { - module.exports = Emitter; -} - -/** - * Initialize a new `Emitter`. - * - * @api public - */ - -function Emitter(obj) { - if (obj) return mixin(obj); -}; - -/** - * Mixin the emitter properties. - * - * @param {Object} obj - * @return {Object} - * @api private - */ - -function mixin(obj) { - for (var key in Emitter.prototype) { - obj[key] = Emitter.prototype[key]; - } - return obj; -} - -/** - * Listen on the given `event` with `fn`. - * - * @param {String} event - * @param {Function} fn - * @return {Emitter} - * @api public - */ - -Emitter.prototype.on = -Emitter.prototype.addEventListener = function(event, fn){ - this._callbacks = this._callbacks || {}; - (this._callbacks['$' + event] = this._callbacks['$' + event] || []) - .push(fn); - return this; -}; - -/** - * Adds an `event` listener that will be invoked a single - * time then automatically removed. - * - * @param {String} event - * @param {Function} fn - * @return {Emitter} - * @api public - */ - -Emitter.prototype.once = function(event, fn){ - function on() { - this.off(event, on); - fn.apply(this, arguments); - } - - on.fn = fn; - this.on(event, on); - return this; -}; - -/** - * Remove the given callback for `event` or all - * registered callbacks. - * - * @param {String} event - * @param {Function} fn - * @return {Emitter} - * @api public - */ - -Emitter.prototype.off = -Emitter.prototype.removeListener = -Emitter.prototype.removeAllListeners = -Emitter.prototype.removeEventListener = function(event, fn){ - this._callbacks = this._callbacks || {}; - - // all - if (0 == arguments.length) { - this._callbacks = {}; - return this; - } - - // specific event - var callbacks = this._callbacks['$' + event]; - if (!callbacks) return this; - - // remove all handlers - if (1 == arguments.length) { - delete this._callbacks['$' + event]; - return this; - } - - // remove specific handler - var cb; - for (var i = 0; i < callbacks.length; i++) { - cb = callbacks[i]; - if (cb === fn || cb.fn === fn) { - callbacks.splice(i, 1); - break; - } - } - return this; -}; - -/** - * Emit `event` with the given args. - * - * @param {String} event - * @param {Mixed} ... - * @return {Emitter} - */ - -Emitter.prototype.emit = function(event){ - this._callbacks = this._callbacks || {}; - var args = [].slice.call(arguments, 1) - , callbacks = this._callbacks['$' + event]; - - if (callbacks) { - callbacks = callbacks.slice(0); - for (var i = 0, len = callbacks.length; i < len; ++i) { - callbacks[i].apply(this, args); - } - } - - return this; -}; - -/** - * Return array of callbacks for `event`. - * - * @param {String} event - * @return {Array} - * @api public - */ - -Emitter.prototype.listeners = function(event){ - this._callbacks = this._callbacks || {}; - return this._callbacks['$' + event] || []; -}; - -/** - * Check if this emitter has `event` handlers. - * - * @param {String} event - * @return {Boolean} - * @api public - */ - -Emitter.prototype.hasListeners = function(event){ - return !! this.listeners(event).length; -}; - -},{}],128:[function(require,module,exports){ - -module.exports = function(a, b){ - var fn = function(){}; - fn.prototype = b.prototype; - a.prototype = new fn; - a.prototype.constructor = a; -}; -},{}],129:[function(require,module,exports){ -(function (Buffer){ -// Copyright Joyent, Inc. and other Node contributors. -// -// Permission is hereby granted, free of charge, to any person obtaining a -// copy of this software and associated documentation files (the -// "Software"), to deal in the Software without restriction, including -// without limitation the rights to use, copy, modify, merge, publish, -// distribute, sublicense, and/or sell copies of the Software, and to permit -// persons to whom the Software is furnished to do so, subject to the -// following conditions: -// -// The above copyright notice and this permission notice shall be included -// in all copies or substantial portions of the Software. -// -// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS -// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN -// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, -// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR -// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE -// USE OR OTHER DEALINGS IN THE SOFTWARE. - -// NOTE: These type checking functions intentionally don't use `instanceof` -// because it is fragile and can be easily faked with `Object.create()`. - -function isArray(arg) { - if (Array.isArray) { - return Array.isArray(arg); - } - return objectToString(arg) === '[object Array]'; -} -exports.isArray = isArray; - -function isBoolean(arg) { - return typeof arg === 'boolean'; -} -exports.isBoolean = isBoolean; - -function isNull(arg) { - return arg === null; -} -exports.isNull = isNull; - -function isNullOrUndefined(arg) { - return arg == null; -} -exports.isNullOrUndefined = isNullOrUndefined; - -function isNumber(arg) { - return typeof arg === 'number'; -} -exports.isNumber = isNumber; - -function isString(arg) { - return typeof arg === 'string'; -} -exports.isString = isString; - -function isSymbol(arg) { - return typeof arg === 'symbol'; -} -exports.isSymbol = isSymbol; - -function isUndefined(arg) { - return arg === void 0; -} -exports.isUndefined = isUndefined; - -function isRegExp(re) { - return objectToString(re) === '[object RegExp]'; -} -exports.isRegExp = isRegExp; - -function isObject(arg) { - return typeof arg === 'object' && arg !== null; -} -exports.isObject = isObject; - -function isDate(d) { - return objectToString(d) === '[object Date]'; -} -exports.isDate = isDate; - -function isError(e) { - return (objectToString(e) === '[object Error]' || e instanceof Error); -} -exports.isError = isError; - -function isFunction(arg) { - return typeof arg === 'function'; -} -exports.isFunction = isFunction; - -function isPrimitive(arg) { - return arg === null || - typeof arg === 'boolean' || - typeof arg === 'number' || - typeof arg === 'string' || - typeof arg === 'symbol' || // ES6 symbol - typeof arg === 'undefined'; -} -exports.isPrimitive = isPrimitive; - -exports.isBuffer = Buffer.isBuffer; - -function objectToString(o) { - return Object.prototype.toString.call(o); -} - -}).call(this,{"isBuffer":require("../../is-buffer/index.js")}) - -},{"../../is-buffer/index.js":219}],130:[function(require,module,exports){ -(function (Buffer){ -var elliptic = require('elliptic') -var BN = require('bn.js') - -module.exports = function createECDH (curve) { - return new ECDH(curve) -} - -var aliases = { - secp256k1: { - name: 'secp256k1', - byteLength: 32 - }, - secp224r1: { - name: 'p224', - byteLength: 28 - }, - prime256v1: { - name: 'p256', - byteLength: 32 - }, - prime192v1: { - name: 'p192', - byteLength: 24 - }, - ed25519: { - name: 'ed25519', - byteLength: 32 - }, - secp384r1: { - name: 'p384', - byteLength: 48 - }, - secp521r1: { - name: 'p521', - byteLength: 66 - } -} - -aliases.p224 = aliases.secp224r1 -aliases.p256 = aliases.secp256r1 = aliases.prime256v1 -aliases.p192 = aliases.secp192r1 = aliases.prime192v1 -aliases.p384 = aliases.secp384r1 -aliases.p521 = aliases.secp521r1 - -function ECDH (curve) { - this.curveType = aliases[curve] - if (!this.curveType) { - this.curveType = { - name: curve - } - } - this.curve = new elliptic.ec(this.curveType.name) // eslint-disable-line new-cap - this.keys = void 0 -} - -ECDH.prototype.generateKeys = function (enc, format) { - this.keys = this.curve.genKeyPair() - return this.getPublicKey(enc, format) -} - -ECDH.prototype.computeSecret = function (other, inenc, enc) { - inenc = inenc || 'utf8' - if (!Buffer.isBuffer(other)) { - other = new Buffer(other, inenc) - } - var otherPub = this.curve.keyFromPublic(other).getPublic() - var out = otherPub.mul(this.keys.getPrivate()).getX() - return formatReturnValue(out, enc, this.curveType.byteLength) -} - -ECDH.prototype.getPublicKey = function (enc, format) { - var key = this.keys.getPublic(format === 'compressed', true) - if (format === 'hybrid') { - if (key[key.length - 1] % 2) { - key[0] = 7 - } else { - key[0] = 6 - } - } - return formatReturnValue(key, enc) -} - -ECDH.prototype.getPrivateKey = function (enc) { - return formatReturnValue(this.keys.getPrivate(), enc) -} - -ECDH.prototype.setPublicKey = function (pub, enc) { - enc = enc || 'utf8' - if (!Buffer.isBuffer(pub)) { - pub = new Buffer(pub, enc) - } - this.keys._importPublic(pub) - return this -} - -ECDH.prototype.setPrivateKey = function (priv, enc) { - enc = enc || 'utf8' - if (!Buffer.isBuffer(priv)) { - priv = new Buffer(priv, enc) - } - - var _priv = new BN(priv) - _priv = _priv.toString(16) - this.keys = this.curve.genKeyPair() - this.keys._importPrivate(_priv) - return this -} - -function formatReturnValue (bn, enc, len) { - if (!Array.isArray(bn)) { - bn = bn.toArray() - } - var buf = new Buffer(bn) - if (len && buf.length < len) { - var zeros = new Buffer(len - buf.length) - zeros.fill(0) - buf = Buffer.concat([zeros, buf]) - } - if (!enc) { - return buf - } else { - return buf.toString(enc) - } -} - -}).call(this,require("buffer").Buffer) - -},{"bn.js":131,"buffer":124,"elliptic":148}],131:[function(require,module,exports){ -arguments[4][70][0].apply(exports,arguments) -},{"buffer":77,"dup":70}],132:[function(require,module,exports){ -'use strict' -var inherits = require('inherits') -var MD5 = require('md5.js') -var RIPEMD160 = require('ripemd160') -var sha = require('sha.js') -var Base = require('cipher-base') - -function Hash (hash) { - Base.call(this, 'digest') - - this._hash = hash -} - -inherits(Hash, Base) - -Hash.prototype._update = function (data) { - this._hash.update(data) -} - -Hash.prototype._final = function () { - return this._hash.digest() -} - -module.exports = function createHash (alg) { - alg = alg.toLowerCase() - if (alg === 'md5') return new MD5() - if (alg === 'rmd160' || alg === 'ripemd160') return new RIPEMD160() - - return new Hash(sha(alg)) -} - -},{"cipher-base":125,"inherits":218,"md5.js":223,"ripemd160":268,"sha.js":271}],133:[function(require,module,exports){ -var MD5 = require('md5.js') - -module.exports = function (buffer) { - return new MD5().update(buffer).digest() -} - -},{"md5.js":223}],134:[function(require,module,exports){ -'use strict' -var inherits = require('inherits') -var Legacy = require('./legacy') -var Base = require('cipher-base') -var Buffer = require('safe-buffer').Buffer -var md5 = require('create-hash/md5') -var RIPEMD160 = require('ripemd160') - -var sha = require('sha.js') - -var ZEROS = Buffer.alloc(128) - -function Hmac (alg, key) { - Base.call(this, 'digest') - if (typeof key === 'string') { - key = Buffer.from(key) - } - - var blocksize = (alg === 'sha512' || alg === 'sha384') ? 128 : 64 - - this._alg = alg - this._key = key - if (key.length > blocksize) { - var hash = alg === 'rmd160' ? new RIPEMD160() : sha(alg) - key = hash.update(key).digest() - } else if (key.length < blocksize) { - key = Buffer.concat([key, ZEROS], blocksize) - } - - var ipad = this._ipad = Buffer.allocUnsafe(blocksize) - var opad = this._opad = Buffer.allocUnsafe(blocksize) - - for (var i = 0; i < blocksize; i++) { - ipad[i] = key[i] ^ 0x36 - opad[i] = key[i] ^ 0x5C - } - this._hash = alg === 'rmd160' ? new RIPEMD160() : sha(alg) - this._hash.update(ipad) -} - -inherits(Hmac, Base) - -Hmac.prototype._update = function (data) { - this._hash.update(data) -} - -Hmac.prototype._final = function () { - var h = this._hash.digest() - var hash = this._alg === 'rmd160' ? new RIPEMD160() : sha(this._alg) - return hash.update(this._opad).update(h).digest() -} - -module.exports = function createHmac (alg, key) { - alg = alg.toLowerCase() - if (alg === 'rmd160' || alg === 'ripemd160') { - return new Hmac('rmd160', key) - } - if (alg === 'md5') { - return new Legacy(md5, key) - } - return new Hmac(alg, key) -} - -},{"./legacy":135,"cipher-base":125,"create-hash/md5":133,"inherits":218,"ripemd160":268,"safe-buffer":269,"sha.js":271}],135:[function(require,module,exports){ -'use strict' -var inherits = require('inherits') -var Buffer = require('safe-buffer').Buffer - -var Base = require('cipher-base') - -var ZEROS = Buffer.alloc(128) -var blocksize = 64 - -function Hmac (alg, key) { - Base.call(this, 'digest') - if (typeof key === 'string') { - key = Buffer.from(key) - } - - this._alg = alg - this._key = key - - if (key.length > blocksize) { - key = alg(key) - } else if (key.length < blocksize) { - key = Buffer.concat([key, ZEROS], blocksize) - } - - var ipad = this._ipad = Buffer.allocUnsafe(blocksize) - var opad = this._opad = Buffer.allocUnsafe(blocksize) - - for (var i = 0; i < blocksize; i++) { - ipad[i] = key[i] ^ 0x36 - opad[i] = key[i] ^ 0x5C - } - - this._hash = [ipad] -} - -inherits(Hmac, Base) - -Hmac.prototype._update = function (data) { - this._hash.push(data) -} - -Hmac.prototype._final = function () { - var h = this._alg(Buffer.concat(this._hash)) - return this._alg(Buffer.concat([this._opad, h])) -} -module.exports = Hmac - -},{"cipher-base":125,"inherits":218,"safe-buffer":269}],136:[function(require,module,exports){ -'use strict' - -exports.randomBytes = exports.rng = exports.pseudoRandomBytes = exports.prng = require('randombytes') -exports.createHash = exports.Hash = require('create-hash') -exports.createHmac = exports.Hmac = require('create-hmac') - -var algos = require('browserify-sign/algos') -var algoKeys = Object.keys(algos) -var hashes = ['sha1', 'sha224', 'sha256', 'sha384', 'sha512', 'md5', 'rmd160'].concat(algoKeys) -exports.getHashes = function () { - return hashes -} - -var p = require('pbkdf2') -exports.pbkdf2 = p.pbkdf2 -exports.pbkdf2Sync = p.pbkdf2Sync - -var aes = require('browserify-cipher') - -exports.Cipher = aes.Cipher -exports.createCipher = aes.createCipher -exports.Cipheriv = aes.Cipheriv -exports.createCipheriv = aes.createCipheriv -exports.Decipher = aes.Decipher -exports.createDecipher = aes.createDecipher -exports.Decipheriv = aes.Decipheriv -exports.createDecipheriv = aes.createDecipheriv -exports.getCiphers = aes.getCiphers -exports.listCiphers = aes.listCiphers - -var dh = require('diffie-hellman') - -exports.DiffieHellmanGroup = dh.DiffieHellmanGroup -exports.createDiffieHellmanGroup = dh.createDiffieHellmanGroup -exports.getDiffieHellman = dh.getDiffieHellman -exports.createDiffieHellman = dh.createDiffieHellman -exports.DiffieHellman = dh.DiffieHellman - -var sign = require('browserify-sign') - -exports.createSign = sign.createSign -exports.Sign = sign.Sign -exports.createVerify = sign.createVerify -exports.Verify = sign.Verify - -exports.createECDH = require('create-ecdh') - -var publicEncrypt = require('public-encrypt') - -exports.publicEncrypt = publicEncrypt.publicEncrypt -exports.privateEncrypt = publicEncrypt.privateEncrypt -exports.publicDecrypt = publicEncrypt.publicDecrypt -exports.privateDecrypt = publicEncrypt.privateDecrypt - -// the least I can do is make error messages for the rest of the node.js/crypto api. -// ;[ -// 'createCredentials' -// ].forEach(function (name) { -// exports[name] = function () { -// throw new Error([ -// 'sorry, ' + name + ' is not implemented yet', -// 'we accept pull requests', -// 'https://github.com/crypto-browserify/crypto-browserify' -// ].join('\n')) -// } -// }) - -var rf = require('randomfill') - -exports.randomFill = rf.randomFill -exports.randomFillSync = rf.randomFillSync - -exports.createCredentials = function () { - throw new Error([ - 'sorry, createCredentials is not implemented yet', - 'we accept pull requests', - 'https://github.com/crypto-browserify/crypto-browserify' - ].join('\n')) -} - -exports.constants = { - 'DH_CHECK_P_NOT_SAFE_PRIME': 2, - 'DH_CHECK_P_NOT_PRIME': 1, - 'DH_UNABLE_TO_CHECK_GENERATOR': 4, - 'DH_NOT_SUITABLE_GENERATOR': 8, - 'NPN_ENABLED': 1, - 'ALPN_ENABLED': 1, - 'RSA_PKCS1_PADDING': 1, - 'RSA_SSLV23_PADDING': 2, - 'RSA_NO_PADDING': 3, - 'RSA_PKCS1_OAEP_PADDING': 4, - 'RSA_X931_PADDING': 5, - 'RSA_PKCS1_PSS_PADDING': 6, - 'POINT_CONVERSION_COMPRESSED': 2, - 'POINT_CONVERSION_UNCOMPRESSED': 4, - 'POINT_CONVERSION_HYBRID': 6 -} - -},{"browserify-cipher":95,"browserify-sign":103,"browserify-sign/algos":100,"create-ecdh":130,"create-hash":132,"create-hmac":134,"diffie-hellman":143,"pbkdf2":237,"public-encrypt":245,"randombytes":252,"randomfill":253}],137:[function(require,module,exports){ -'use strict'; - -exports.utils = require('./des/utils'); -exports.Cipher = require('./des/cipher'); -exports.DES = require('./des/des'); -exports.CBC = require('./des/cbc'); -exports.EDE = require('./des/ede'); - -},{"./des/cbc":138,"./des/cipher":139,"./des/des":140,"./des/ede":141,"./des/utils":142}],138:[function(require,module,exports){ -'use strict'; - -var assert = require('minimalistic-assert'); -var inherits = require('inherits'); - -var proto = {}; - -function CBCState(iv) { - assert.equal(iv.length, 8, 'Invalid IV length'); - - this.iv = new Array(8); - for (var i = 0; i < this.iv.length; i++) - this.iv[i] = iv[i]; -} - -function instantiate(Base) { - function CBC(options) { - Base.call(this, options); - this._cbcInit(); - } - inherits(CBC, Base); - - var keys = Object.keys(proto); - for (var i = 0; i < keys.length; i++) { - var key = keys[i]; - CBC.prototype[key] = proto[key]; - } - - CBC.create = function create(options) { - return new CBC(options); - }; - - return CBC; -} - -exports.instantiate = instantiate; - -proto._cbcInit = function _cbcInit() { - var state = new CBCState(this.options.iv); - this._cbcState = state; -}; - -proto._update = function _update(inp, inOff, out, outOff) { - var state = this._cbcState; - var superProto = this.constructor.super_.prototype; - - var iv = state.iv; - if (this.type === 'encrypt') { - for (var i = 0; i < this.blockSize; i++) - iv[i] ^= inp[inOff + i]; - - superProto._update.call(this, iv, 0, out, outOff); - - for (var i = 0; i < this.blockSize; i++) - iv[i] = out[outOff + i]; - } else { - superProto._update.call(this, inp, inOff, out, outOff); - - for (var i = 0; i < this.blockSize; i++) - out[outOff + i] ^= iv[i]; - - for (var i = 0; i < this.blockSize; i++) - iv[i] = inp[inOff + i]; - } -}; - -},{"inherits":218,"minimalistic-assert":226}],139:[function(require,module,exports){ -'use strict'; - -var assert = require('minimalistic-assert'); - -function Cipher(options) { - this.options = options; - - this.type = this.options.type; - this.blockSize = 8; - this._init(); - - this.buffer = new Array(this.blockSize); - this.bufferOff = 0; -} -module.exports = Cipher; - -Cipher.prototype._init = function _init() { - // Might be overrided -}; - -Cipher.prototype.update = function update(data) { - if (data.length === 0) - return []; - - if (this.type === 'decrypt') - return this._updateDecrypt(data); - else - return this._updateEncrypt(data); -}; - -Cipher.prototype._buffer = function _buffer(data, off) { - // Append data to buffer - var min = Math.min(this.buffer.length - this.bufferOff, data.length - off); - for (var i = 0; i < min; i++) - this.buffer[this.bufferOff + i] = data[off + i]; - this.bufferOff += min; - - // Shift next - return min; -}; - -Cipher.prototype._flushBuffer = function _flushBuffer(out, off) { - this._update(this.buffer, 0, out, off); - this.bufferOff = 0; - return this.blockSize; -}; - -Cipher.prototype._updateEncrypt = function _updateEncrypt(data) { - var inputOff = 0; - var outputOff = 0; - - var count = ((this.bufferOff + data.length) / this.blockSize) | 0; - var out = new Array(count * this.blockSize); - - if (this.bufferOff !== 0) { - inputOff += this._buffer(data, inputOff); - - if (this.bufferOff === this.buffer.length) - outputOff += this._flushBuffer(out, outputOff); - } - - // Write blocks - var max = data.length - ((data.length - inputOff) % this.blockSize); - for (; inputOff < max; inputOff += this.blockSize) { - this._update(data, inputOff, out, outputOff); - outputOff += this.blockSize; - } - - // Queue rest - for (; inputOff < data.length; inputOff++, this.bufferOff++) - this.buffer[this.bufferOff] = data[inputOff]; - - return out; -}; - -Cipher.prototype._updateDecrypt = function _updateDecrypt(data) { - var inputOff = 0; - var outputOff = 0; - - var count = Math.ceil((this.bufferOff + data.length) / this.blockSize) - 1; - var out = new Array(count * this.blockSize); - - // TODO(indutny): optimize it, this is far from optimal - for (; count > 0; count--) { - inputOff += this._buffer(data, inputOff); - outputOff += this._flushBuffer(out, outputOff); - } - - // Buffer rest of the input - inputOff += this._buffer(data, inputOff); - - return out; -}; - -Cipher.prototype.final = function final(buffer) { - var first; - if (buffer) - first = this.update(buffer); - - var last; - if (this.type === 'encrypt') - last = this._finalEncrypt(); - else - last = this._finalDecrypt(); - - if (first) - return first.concat(last); - else - return last; -}; - -Cipher.prototype._pad = function _pad(buffer, off) { - if (off === 0) - return false; - - while (off < buffer.length) - buffer[off++] = 0; - - return true; -}; - -Cipher.prototype._finalEncrypt = function _finalEncrypt() { - if (!this._pad(this.buffer, this.bufferOff)) - return []; - - var out = new Array(this.blockSize); - this._update(this.buffer, 0, out, 0); - return out; -}; - -Cipher.prototype._unpad = function _unpad(buffer) { - return buffer; -}; - -Cipher.prototype._finalDecrypt = function _finalDecrypt() { - assert.equal(this.bufferOff, this.blockSize, 'Not enough data to decrypt'); - var out = new Array(this.blockSize); - this._flushBuffer(out, 0); - - return this._unpad(out); -}; - -},{"minimalistic-assert":226}],140:[function(require,module,exports){ -'use strict'; - -var assert = require('minimalistic-assert'); -var inherits = require('inherits'); - -var utils = require('./utils'); -var Cipher = require('./cipher'); - -function DESState() { - this.tmp = new Array(2); - this.keys = null; -} - -function DES(options) { - Cipher.call(this, options); - - var state = new DESState(); - this._desState = state; - - this.deriveKeys(state, options.key); -} -inherits(DES, Cipher); -module.exports = DES; - -DES.create = function create(options) { - return new DES(options); -}; - -var shiftTable = [ - 1, 1, 2, 2, 2, 2, 2, 2, - 1, 2, 2, 2, 2, 2, 2, 1 -]; - -DES.prototype.deriveKeys = function deriveKeys(state, key) { - state.keys = new Array(16 * 2); - - assert.equal(key.length, this.blockSize, 'Invalid key length'); - - var kL = utils.readUInt32BE(key, 0); - var kR = utils.readUInt32BE(key, 4); - - utils.pc1(kL, kR, state.tmp, 0); - kL = state.tmp[0]; - kR = state.tmp[1]; - for (var i = 0; i < state.keys.length; i += 2) { - var shift = shiftTable[i >>> 1]; - kL = utils.r28shl(kL, shift); - kR = utils.r28shl(kR, shift); - utils.pc2(kL, kR, state.keys, i); - } -}; - -DES.prototype._update = function _update(inp, inOff, out, outOff) { - var state = this._desState; - - var l = utils.readUInt32BE(inp, inOff); - var r = utils.readUInt32BE(inp, inOff + 4); - - // Initial Permutation - utils.ip(l, r, state.tmp, 0); - l = state.tmp[0]; - r = state.tmp[1]; - - if (this.type === 'encrypt') - this._encrypt(state, l, r, state.tmp, 0); - else - this._decrypt(state, l, r, state.tmp, 0); - - l = state.tmp[0]; - r = state.tmp[1]; - - utils.writeUInt32BE(out, l, outOff); - utils.writeUInt32BE(out, r, outOff + 4); -}; - -DES.prototype._pad = function _pad(buffer, off) { - var value = buffer.length - off; - for (var i = off; i < buffer.length; i++) - buffer[i] = value; - - return true; -}; - -DES.prototype._unpad = function _unpad(buffer) { - var pad = buffer[buffer.length - 1]; - for (var i = buffer.length - pad; i < buffer.length; i++) - assert.equal(buffer[i], pad); - - return buffer.slice(0, buffer.length - pad); -}; - -DES.prototype._encrypt = function _encrypt(state, lStart, rStart, out, off) { - var l = lStart; - var r = rStart; - - // Apply f() x16 times - for (var i = 0; i < state.keys.length; i += 2) { - var keyL = state.keys[i]; - var keyR = state.keys[i + 1]; - - // f(r, k) - utils.expand(r, state.tmp, 0); - - keyL ^= state.tmp[0]; - keyR ^= state.tmp[1]; - var s = utils.substitute(keyL, keyR); - var f = utils.permute(s); - - var t = r; - r = (l ^ f) >>> 0; - l = t; - } - - // Reverse Initial Permutation - utils.rip(r, l, out, off); -}; - -DES.prototype._decrypt = function _decrypt(state, lStart, rStart, out, off) { - var l = rStart; - var r = lStart; - - // Apply f() x16 times - for (var i = state.keys.length - 2; i >= 0; i -= 2) { - var keyL = state.keys[i]; - var keyR = state.keys[i + 1]; - - // f(r, k) - utils.expand(l, state.tmp, 0); - - keyL ^= state.tmp[0]; - keyR ^= state.tmp[1]; - var s = utils.substitute(keyL, keyR); - var f = utils.permute(s); - - var t = l; - l = (r ^ f) >>> 0; - r = t; - } - - // Reverse Initial Permutation - utils.rip(l, r, out, off); -}; - -},{"./cipher":139,"./utils":142,"inherits":218,"minimalistic-assert":226}],141:[function(require,module,exports){ -'use strict'; - -var assert = require('minimalistic-assert'); -var inherits = require('inherits'); - -var Cipher = require('./cipher'); -var DES = require('./des'); - -function EDEState(type, key) { - assert.equal(key.length, 24, 'Invalid key length'); - - var k1 = key.slice(0, 8); - var k2 = key.slice(8, 16); - var k3 = key.slice(16, 24); - - if (type === 'encrypt') { - this.ciphers = [ - DES.create({ type: 'encrypt', key: k1 }), - DES.create({ type: 'decrypt', key: k2 }), - DES.create({ type: 'encrypt', key: k3 }) - ]; - } else { - this.ciphers = [ - DES.create({ type: 'decrypt', key: k3 }), - DES.create({ type: 'encrypt', key: k2 }), - DES.create({ type: 'decrypt', key: k1 }) - ]; - } -} - -function EDE(options) { - Cipher.call(this, options); - - var state = new EDEState(this.type, this.options.key); - this._edeState = state; -} -inherits(EDE, Cipher); - -module.exports = EDE; - -EDE.create = function create(options) { - return new EDE(options); -}; - -EDE.prototype._update = function _update(inp, inOff, out, outOff) { - var state = this._edeState; - - state.ciphers[0]._update(inp, inOff, out, outOff); - state.ciphers[1]._update(out, outOff, out, outOff); - state.ciphers[2]._update(out, outOff, out, outOff); -}; - -EDE.prototype._pad = DES.prototype._pad; -EDE.prototype._unpad = DES.prototype._unpad; - -},{"./cipher":139,"./des":140,"inherits":218,"minimalistic-assert":226}],142:[function(require,module,exports){ -'use strict'; - -exports.readUInt32BE = function readUInt32BE(bytes, off) { - var res = (bytes[0 + off] << 24) | - (bytes[1 + off] << 16) | - (bytes[2 + off] << 8) | - bytes[3 + off]; - return res >>> 0; -}; - -exports.writeUInt32BE = function writeUInt32BE(bytes, value, off) { - bytes[0 + off] = value >>> 24; - bytes[1 + off] = (value >>> 16) & 0xff; - bytes[2 + off] = (value >>> 8) & 0xff; - bytes[3 + off] = value & 0xff; -}; - -exports.ip = function ip(inL, inR, out, off) { - var outL = 0; - var outR = 0; - - for (var i = 6; i >= 0; i -= 2) { - for (var j = 0; j <= 24; j += 8) { - outL <<= 1; - outL |= (inR >>> (j + i)) & 1; - } - for (var j = 0; j <= 24; j += 8) { - outL <<= 1; - outL |= (inL >>> (j + i)) & 1; - } - } - - for (var i = 6; i >= 0; i -= 2) { - for (var j = 1; j <= 25; j += 8) { - outR <<= 1; - outR |= (inR >>> (j + i)) & 1; - } - for (var j = 1; j <= 25; j += 8) { - outR <<= 1; - outR |= (inL >>> (j + i)) & 1; - } - } - - out[off + 0] = outL >>> 0; - out[off + 1] = outR >>> 0; -}; - -exports.rip = function rip(inL, inR, out, off) { - var outL = 0; - var outR = 0; - - for (var i = 0; i < 4; i++) { - for (var j = 24; j >= 0; j -= 8) { - outL <<= 1; - outL |= (inR >>> (j + i)) & 1; - outL <<= 1; - outL |= (inL >>> (j + i)) & 1; - } - } - for (var i = 4; i < 8; i++) { - for (var j = 24; j >= 0; j -= 8) { - outR <<= 1; - outR |= (inR >>> (j + i)) & 1; - outR <<= 1; - outR |= (inL >>> (j + i)) & 1; - } - } - - out[off + 0] = outL >>> 0; - out[off + 1] = outR >>> 0; -}; - -exports.pc1 = function pc1(inL, inR, out, off) { - var outL = 0; - var outR = 0; - - // 7, 15, 23, 31, 39, 47, 55, 63 - // 6, 14, 22, 30, 39, 47, 55, 63 - // 5, 13, 21, 29, 39, 47, 55, 63 - // 4, 12, 20, 28 - for (var i = 7; i >= 5; i--) { - for (var j = 0; j <= 24; j += 8) { - outL <<= 1; - outL |= (inR >> (j + i)) & 1; - } - for (var j = 0; j <= 24; j += 8) { - outL <<= 1; - outL |= (inL >> (j + i)) & 1; - } - } - for (var j = 0; j <= 24; j += 8) { - outL <<= 1; - outL |= (inR >> (j + i)) & 1; - } - - // 1, 9, 17, 25, 33, 41, 49, 57 - // 2, 10, 18, 26, 34, 42, 50, 58 - // 3, 11, 19, 27, 35, 43, 51, 59 - // 36, 44, 52, 60 - for (var i = 1; i <= 3; i++) { - for (var j = 0; j <= 24; j += 8) { - outR <<= 1; - outR |= (inR >> (j + i)) & 1; - } - for (var j = 0; j <= 24; j += 8) { - outR <<= 1; - outR |= (inL >> (j + i)) & 1; - } - } - for (var j = 0; j <= 24; j += 8) { - outR <<= 1; - outR |= (inL >> (j + i)) & 1; - } - - out[off + 0] = outL >>> 0; - out[off + 1] = outR >>> 0; -}; - -exports.r28shl = function r28shl(num, shift) { - return ((num << shift) & 0xfffffff) | (num >>> (28 - shift)); -}; - -var pc2table = [ - // inL => outL - 14, 11, 17, 4, 27, 23, 25, 0, - 13, 22, 7, 18, 5, 9, 16, 24, - 2, 20, 12, 21, 1, 8, 15, 26, - - // inR => outR - 15, 4, 25, 19, 9, 1, 26, 16, - 5, 11, 23, 8, 12, 7, 17, 0, - 22, 3, 10, 14, 6, 20, 27, 24 -]; - -exports.pc2 = function pc2(inL, inR, out, off) { - var outL = 0; - var outR = 0; - - var len = pc2table.length >>> 1; - for (var i = 0; i < len; i++) { - outL <<= 1; - outL |= (inL >>> pc2table[i]) & 0x1; - } - for (var i = len; i < pc2table.length; i++) { - outR <<= 1; - outR |= (inR >>> pc2table[i]) & 0x1; - } - - out[off + 0] = outL >>> 0; - out[off + 1] = outR >>> 0; -}; - -exports.expand = function expand(r, out, off) { - var outL = 0; - var outR = 0; - - outL = ((r & 1) << 5) | (r >>> 27); - for (var i = 23; i >= 15; i -= 4) { - outL <<= 6; - outL |= (r >>> i) & 0x3f; - } - for (var i = 11; i >= 3; i -= 4) { - outR |= (r >>> i) & 0x3f; - outR <<= 6; - } - outR |= ((r & 0x1f) << 1) | (r >>> 31); - - out[off + 0] = outL >>> 0; - out[off + 1] = outR >>> 0; -}; - -var sTable = [ - 14, 0, 4, 15, 13, 7, 1, 4, 2, 14, 15, 2, 11, 13, 8, 1, - 3, 10, 10, 6, 6, 12, 12, 11, 5, 9, 9, 5, 0, 3, 7, 8, - 4, 15, 1, 12, 14, 8, 8, 2, 13, 4, 6, 9, 2, 1, 11, 7, - 15, 5, 12, 11, 9, 3, 7, 14, 3, 10, 10, 0, 5, 6, 0, 13, - - 15, 3, 1, 13, 8, 4, 14, 7, 6, 15, 11, 2, 3, 8, 4, 14, - 9, 12, 7, 0, 2, 1, 13, 10, 12, 6, 0, 9, 5, 11, 10, 5, - 0, 13, 14, 8, 7, 10, 11, 1, 10, 3, 4, 15, 13, 4, 1, 2, - 5, 11, 8, 6, 12, 7, 6, 12, 9, 0, 3, 5, 2, 14, 15, 9, - - 10, 13, 0, 7, 9, 0, 14, 9, 6, 3, 3, 4, 15, 6, 5, 10, - 1, 2, 13, 8, 12, 5, 7, 14, 11, 12, 4, 11, 2, 15, 8, 1, - 13, 1, 6, 10, 4, 13, 9, 0, 8, 6, 15, 9, 3, 8, 0, 7, - 11, 4, 1, 15, 2, 14, 12, 3, 5, 11, 10, 5, 14, 2, 7, 12, - - 7, 13, 13, 8, 14, 11, 3, 5, 0, 6, 6, 15, 9, 0, 10, 3, - 1, 4, 2, 7, 8, 2, 5, 12, 11, 1, 12, 10, 4, 14, 15, 9, - 10, 3, 6, 15, 9, 0, 0, 6, 12, 10, 11, 1, 7, 13, 13, 8, - 15, 9, 1, 4, 3, 5, 14, 11, 5, 12, 2, 7, 8, 2, 4, 14, - - 2, 14, 12, 11, 4, 2, 1, 12, 7, 4, 10, 7, 11, 13, 6, 1, - 8, 5, 5, 0, 3, 15, 15, 10, 13, 3, 0, 9, 14, 8, 9, 6, - 4, 11, 2, 8, 1, 12, 11, 7, 10, 1, 13, 14, 7, 2, 8, 13, - 15, 6, 9, 15, 12, 0, 5, 9, 6, 10, 3, 4, 0, 5, 14, 3, - - 12, 10, 1, 15, 10, 4, 15, 2, 9, 7, 2, 12, 6, 9, 8, 5, - 0, 6, 13, 1, 3, 13, 4, 14, 14, 0, 7, 11, 5, 3, 11, 8, - 9, 4, 14, 3, 15, 2, 5, 12, 2, 9, 8, 5, 12, 15, 3, 10, - 7, 11, 0, 14, 4, 1, 10, 7, 1, 6, 13, 0, 11, 8, 6, 13, - - 4, 13, 11, 0, 2, 11, 14, 7, 15, 4, 0, 9, 8, 1, 13, 10, - 3, 14, 12, 3, 9, 5, 7, 12, 5, 2, 10, 15, 6, 8, 1, 6, - 1, 6, 4, 11, 11, 13, 13, 8, 12, 1, 3, 4, 7, 10, 14, 7, - 10, 9, 15, 5, 6, 0, 8, 15, 0, 14, 5, 2, 9, 3, 2, 12, - - 13, 1, 2, 15, 8, 13, 4, 8, 6, 10, 15, 3, 11, 7, 1, 4, - 10, 12, 9, 5, 3, 6, 14, 11, 5, 0, 0, 14, 12, 9, 7, 2, - 7, 2, 11, 1, 4, 14, 1, 7, 9, 4, 12, 10, 14, 8, 2, 13, - 0, 15, 6, 12, 10, 9, 13, 0, 15, 3, 3, 5, 5, 6, 8, 11 -]; - -exports.substitute = function substitute(inL, inR) { - var out = 0; - for (var i = 0; i < 4; i++) { - var b = (inL >>> (18 - i * 6)) & 0x3f; - var sb = sTable[i * 0x40 + b]; - - out <<= 4; - out |= sb; - } - for (var i = 0; i < 4; i++) { - var b = (inR >>> (18 - i * 6)) & 0x3f; - var sb = sTable[4 * 0x40 + i * 0x40 + b]; - - out <<= 4; - out |= sb; - } - return out >>> 0; -}; - -var permuteTable = [ - 16, 25, 12, 11, 3, 20, 4, 15, 31, 17, 9, 6, 27, 14, 1, 22, - 30, 24, 8, 18, 0, 5, 29, 23, 13, 19, 2, 26, 10, 21, 28, 7 -]; - -exports.permute = function permute(num) { - var out = 0; - for (var i = 0; i < permuteTable.length; i++) { - out <<= 1; - out |= (num >>> permuteTable[i]) & 0x1; - } - return out >>> 0; -}; - -exports.padSplit = function padSplit(num, size, group) { - var str = num.toString(2); - while (str.length < size) - str = '0' + str; - - var out = []; - for (var i = 0; i < size; i += group) - out.push(str.slice(i, i + group)); - return out.join(' '); -}; - -},{}],143:[function(require,module,exports){ -(function (Buffer){ -var generatePrime = require('./lib/generatePrime') -var primes = require('./lib/primes.json') - -var DH = require('./lib/dh') - -function getDiffieHellman (mod) { - var prime = new Buffer(primes[mod].prime, 'hex') - var gen = new Buffer(primes[mod].gen, 'hex') - - return new DH(prime, gen) -} - -var ENCODINGS = { - 'binary': true, 'hex': true, 'base64': true -} - -function createDiffieHellman (prime, enc, generator, genc) { - if (Buffer.isBuffer(enc) || ENCODINGS[enc] === undefined) { - return createDiffieHellman(prime, 'binary', enc, generator) - } - - enc = enc || 'binary' - genc = genc || 'binary' - generator = generator || new Buffer([2]) - - if (!Buffer.isBuffer(generator)) { - generator = new Buffer(generator, genc) - } - - if (typeof prime === 'number') { - return new DH(generatePrime(prime, generator), generator, true) - } - - if (!Buffer.isBuffer(prime)) { - prime = new Buffer(prime, enc) - } - - return new DH(prime, generator, true) -} - -exports.DiffieHellmanGroup = exports.createDiffieHellmanGroup = exports.getDiffieHellman = getDiffieHellman -exports.createDiffieHellman = exports.DiffieHellman = createDiffieHellman - -}).call(this,require("buffer").Buffer) - -},{"./lib/dh":144,"./lib/generatePrime":145,"./lib/primes.json":146,"buffer":124}],144:[function(require,module,exports){ -(function (Buffer){ -var BN = require('bn.js'); -var MillerRabin = require('miller-rabin'); -var millerRabin = new MillerRabin(); -var TWENTYFOUR = new BN(24); -var ELEVEN = new BN(11); -var TEN = new BN(10); -var THREE = new BN(3); -var SEVEN = new BN(7); -var primes = require('./generatePrime'); -var randomBytes = require('randombytes'); -module.exports = DH; - -function setPublicKey(pub, enc) { - enc = enc || 'utf8'; - if (!Buffer.isBuffer(pub)) { - pub = new Buffer(pub, enc); - } - this._pub = new BN(pub); - return this; -} - -function setPrivateKey(priv, enc) { - enc = enc || 'utf8'; - if (!Buffer.isBuffer(priv)) { - priv = new Buffer(priv, enc); - } - this._priv = new BN(priv); - return this; -} - -var primeCache = {}; -function checkPrime(prime, generator) { - var gen = generator.toString('hex'); - var hex = [gen, prime.toString(16)].join('_'); - if (hex in primeCache) { - return primeCache[hex]; - } - var error = 0; - - if (prime.isEven() || - !primes.simpleSieve || - !primes.fermatTest(prime) || - !millerRabin.test(prime)) { - //not a prime so +1 - error += 1; - - if (gen === '02' || gen === '05') { - // we'd be able to check the generator - // it would fail so +8 - error += 8; - } else { - //we wouldn't be able to test the generator - // so +4 - error += 4; - } - primeCache[hex] = error; - return error; - } - if (!millerRabin.test(prime.shrn(1))) { - //not a safe prime - error += 2; - } - var rem; - switch (gen) { - case '02': - if (prime.mod(TWENTYFOUR).cmp(ELEVEN)) { - // unsuidable generator - error += 8; - } - break; - case '05': - rem = prime.mod(TEN); - if (rem.cmp(THREE) && rem.cmp(SEVEN)) { - // prime mod 10 needs to equal 3 or 7 - error += 8; - } - break; - default: - error += 4; - } - primeCache[hex] = error; - return error; -} - -function DH(prime, generator, malleable) { - this.setGenerator(generator); - this.__prime = new BN(prime); - this._prime = BN.mont(this.__prime); - this._primeLen = prime.length; - this._pub = undefined; - this._priv = undefined; - this._primeCode = undefined; - if (malleable) { - this.setPublicKey = setPublicKey; - this.setPrivateKey = setPrivateKey; - } else { - this._primeCode = 8; - } -} -Object.defineProperty(DH.prototype, 'verifyError', { - enumerable: true, - get: function () { - if (typeof this._primeCode !== 'number') { - this._primeCode = checkPrime(this.__prime, this.__gen); - } - return this._primeCode; - } -}); -DH.prototype.generateKeys = function () { - if (!this._priv) { - this._priv = new BN(randomBytes(this._primeLen)); - } - this._pub = this._gen.toRed(this._prime).redPow(this._priv).fromRed(); - return this.getPublicKey(); -}; - -DH.prototype.computeSecret = function (other) { - other = new BN(other); - other = other.toRed(this._prime); - var secret = other.redPow(this._priv).fromRed(); - var out = new Buffer(secret.toArray()); - var prime = this.getPrime(); - if (out.length < prime.length) { - var front = new Buffer(prime.length - out.length); - front.fill(0); - out = Buffer.concat([front, out]); - } - return out; -}; - -DH.prototype.getPublicKey = function getPublicKey(enc) { - return formatReturnValue(this._pub, enc); -}; - -DH.prototype.getPrivateKey = function getPrivateKey(enc) { - return formatReturnValue(this._priv, enc); -}; - -DH.prototype.getPrime = function (enc) { - return formatReturnValue(this.__prime, enc); -}; - -DH.prototype.getGenerator = function (enc) { - return formatReturnValue(this._gen, enc); -}; - -DH.prototype.setGenerator = function (gen, enc) { - enc = enc || 'utf8'; - if (!Buffer.isBuffer(gen)) { - gen = new Buffer(gen, enc); - } - this.__gen = gen; - this._gen = new BN(gen); - return this; -}; - -function formatReturnValue(bn, enc) { - var buf = new Buffer(bn.toArray()); - if (!enc) { - return buf; - } else { - return buf.toString(enc); - } -} - -}).call(this,require("buffer").Buffer) - -},{"./generatePrime":145,"bn.js":147,"buffer":124,"miller-rabin":224,"randombytes":252}],145:[function(require,module,exports){ -var randomBytes = require('randombytes'); -module.exports = findPrime; -findPrime.simpleSieve = simpleSieve; -findPrime.fermatTest = fermatTest; -var BN = require('bn.js'); -var TWENTYFOUR = new BN(24); -var MillerRabin = require('miller-rabin'); -var millerRabin = new MillerRabin(); -var ONE = new BN(1); -var TWO = new BN(2); -var FIVE = new BN(5); -var SIXTEEN = new BN(16); -var EIGHT = new BN(8); -var TEN = new BN(10); -var THREE = new BN(3); -var SEVEN = new BN(7); -var ELEVEN = new BN(11); -var FOUR = new BN(4); -var TWELVE = new BN(12); -var primes = null; - -function _getPrimes() { - if (primes !== null) - return primes; - - var limit = 0x100000; - var res = []; - res[0] = 2; - for (var i = 1, k = 3; k < limit; k += 2) { - var sqrt = Math.ceil(Math.sqrt(k)); - for (var j = 0; j < i && res[j] <= sqrt; j++) - if (k % res[j] === 0) - break; - - if (i !== j && res[j] <= sqrt) - continue; - - res[i++] = k; - } - primes = res; - return res; -} - -function simpleSieve(p) { - var primes = _getPrimes(); - - for (var i = 0; i < primes.length; i++) - if (p.modn(primes[i]) === 0) { - if (p.cmpn(primes[i]) === 0) { - return true; - } else { - return false; - } - } - - return true; -} - -function fermatTest(p) { - var red = BN.mont(p); - return TWO.toRed(red).redPow(p.subn(1)).fromRed().cmpn(1) === 0; -} - -function findPrime(bits, gen) { - if (bits < 16) { - // this is what openssl does - if (gen === 2 || gen === 5) { - return new BN([0x8c, 0x7b]); - } else { - return new BN([0x8c, 0x27]); - } - } - gen = new BN(gen); - - var num, n2; - - while (true) { - num = new BN(randomBytes(Math.ceil(bits / 8))); - while (num.bitLength() > bits) { - num.ishrn(1); - } - if (num.isEven()) { - num.iadd(ONE); - } - if (!num.testn(1)) { - num.iadd(TWO); - } - if (!gen.cmp(TWO)) { - while (num.mod(TWENTYFOUR).cmp(ELEVEN)) { - num.iadd(FOUR); - } - } else if (!gen.cmp(FIVE)) { - while (num.mod(TEN).cmp(THREE)) { - num.iadd(FOUR); - } - } - n2 = num.shrn(1); - if (simpleSieve(n2) && simpleSieve(num) && - fermatTest(n2) && fermatTest(num) && - millerRabin.test(n2) && millerRabin.test(num)) { - return num; - } - } - -} - -},{"bn.js":147,"miller-rabin":224,"randombytes":252}],146:[function(require,module,exports){ -module.exports={ - "modp1": { - "gen": "02", - "prime": "ffffffffffffffffc90fdaa22168c234c4c6628b80dc1cd129024e088a67cc74020bbea63b139b22514a08798e3404ddef9519b3cd3a431b302b0a6df25f14374fe1356d6d51c245e485b576625e7ec6f44c42e9a63a3620ffffffffffffffff" - }, - "modp2": { - "gen": "02", - "prime": "ffffffffffffffffc90fdaa22168c234c4c6628b80dc1cd129024e088a67cc74020bbea63b139b22514a08798e3404ddef9519b3cd3a431b302b0a6df25f14374fe1356d6d51c245e485b576625e7ec6f44c42e9a637ed6b0bff5cb6f406b7edee386bfb5a899fa5ae9f24117c4b1fe649286651ece65381ffffffffffffffff" - }, - "modp5": { - "gen": "02", - "prime": "ffffffffffffffffc90fdaa22168c234c4c6628b80dc1cd129024e088a67cc74020bbea63b139b22514a08798e3404ddef9519b3cd3a431b302b0a6df25f14374fe1356d6d51c245e485b576625e7ec6f44c42e9a637ed6b0bff5cb6f406b7edee386bfb5a899fa5ae9f24117c4b1fe649286651ece45b3dc2007cb8a163bf0598da48361c55d39a69163fa8fd24cf5f83655d23dca3ad961c62f356208552bb9ed529077096966d670c354e4abc9804f1746c08ca237327ffffffffffffffff" - }, - "modp14": { - "gen": "02", - "prime": "ffffffffffffffffc90fdaa22168c234c4c6628b80dc1cd129024e088a67cc74020bbea63b139b22514a08798e3404ddef9519b3cd3a431b302b0a6df25f14374fe1356d6d51c245e485b576625e7ec6f44c42e9a637ed6b0bff5cb6f406b7edee386bfb5a899fa5ae9f24117c4b1fe649286651ece45b3dc2007cb8a163bf0598da48361c55d39a69163fa8fd24cf5f83655d23dca3ad961c62f356208552bb9ed529077096966d670c354e4abc9804f1746c08ca18217c32905e462e36ce3be39e772c180e86039b2783a2ec07a28fb5c55df06f4c52c9de2bcbf6955817183995497cea956ae515d2261898fa051015728e5a8aacaa68ffffffffffffffff" - }, - "modp15": { - "gen": "02", - "prime": "ffffffffffffffffc90fdaa22168c234c4c6628b80dc1cd129024e088a67cc74020bbea63b139b22514a08798e3404ddef9519b3cd3a431b302b0a6df25f14374fe1356d6d51c245e485b576625e7ec6f44c42e9a637ed6b0bff5cb6f406b7edee386bfb5a899fa5ae9f24117c4b1fe649286651ece45b3dc2007cb8a163bf0598da48361c55d39a69163fa8fd24cf5f83655d23dca3ad961c62f356208552bb9ed529077096966d670c354e4abc9804f1746c08ca18217c32905e462e36ce3be39e772c180e86039b2783a2ec07a28fb5c55df06f4c52c9de2bcbf6955817183995497cea956ae515d2261898fa051015728e5a8aaac42dad33170d04507a33a85521abdf1cba64ecfb850458dbef0a8aea71575d060c7db3970f85a6e1e4c7abf5ae8cdb0933d71e8c94e04a25619dcee3d2261ad2ee6bf12ffa06d98a0864d87602733ec86a64521f2b18177b200cbbe117577a615d6c770988c0bad946e208e24fa074e5ab3143db5bfce0fd108e4b82d120a93ad2caffffffffffffffff" - }, - "modp16": { - "gen": "02", - "prime": "ffffffffffffffffc90fdaa22168c234c4c6628b80dc1cd129024e088a67cc74020bbea63b139b22514a08798e3404ddef9519b3cd3a431b302b0a6df25f14374fe1356d6d51c245e485b576625e7ec6f44c42e9a637ed6b0bff5cb6f406b7edee386bfb5a899fa5ae9f24117c4b1fe649286651ece45b3dc2007cb8a163bf0598da48361c55d39a69163fa8fd24cf5f83655d23dca3ad961c62f356208552bb9ed529077096966d670c354e4abc9804f1746c08ca18217c32905e462e36ce3be39e772c180e86039b2783a2ec07a28fb5c55df06f4c52c9de2bcbf6955817183995497cea956ae515d2261898fa051015728e5a8aaac42dad33170d04507a33a85521abdf1cba64ecfb850458dbef0a8aea71575d060c7db3970f85a6e1e4c7abf5ae8cdb0933d71e8c94e04a25619dcee3d2261ad2ee6bf12ffa06d98a0864d87602733ec86a64521f2b18177b200cbbe117577a615d6c770988c0bad946e208e24fa074e5ab3143db5bfce0fd108e4b82d120a92108011a723c12a787e6d788719a10bdba5b2699c327186af4e23c1a946834b6150bda2583e9ca2ad44ce8dbbbc2db04de8ef92e8efc141fbecaa6287c59474e6bc05d99b2964fa090c3a2233ba186515be7ed1f612970cee2d7afb81bdd762170481cd0069127d5b05aa993b4ea988d8fddc186ffb7dc90a6c08f4df435c934063199ffffffffffffffff" - }, - "modp17": { - "gen": "02", - "prime": "ffffffffffffffffc90fdaa22168c234c4c6628b80dc1cd129024e088a67cc74020bbea63b139b22514a08798e3404ddef9519b3cd3a431b302b0a6df25f14374fe1356d6d51c245e485b576625e7ec6f44c42e9a637ed6b0bff5cb6f406b7edee386bfb5a899fa5ae9f24117c4b1fe649286651ece45b3dc2007cb8a163bf0598da48361c55d39a69163fa8fd24cf5f83655d23dca3ad961c62f356208552bb9ed529077096966d670c354e4abc9804f1746c08ca18217c32905e462e36ce3be39e772c180e86039b2783a2ec07a28fb5c55df06f4c52c9de2bcbf6955817183995497cea956ae515d2261898fa051015728e5a8aaac42dad33170d04507a33a85521abdf1cba64ecfb850458dbef0a8aea71575d060c7db3970f85a6e1e4c7abf5ae8cdb0933d71e8c94e04a25619dcee3d2261ad2ee6bf12ffa06d98a0864d87602733ec86a64521f2b18177b200cbbe117577a615d6c770988c0bad946e208e24fa074e5ab3143db5bfce0fd108e4b82d120a92108011a723c12a787e6d788719a10bdba5b2699c327186af4e23c1a946834b6150bda2583e9ca2ad44ce8dbbbc2db04de8ef92e8efc141fbecaa6287c59474e6bc05d99b2964fa090c3a2233ba186515be7ed1f612970cee2d7afb81bdd762170481cd0069127d5b05aa993b4ea988d8fddc186ffb7dc90a6c08f4df435c93402849236c3fab4d27c7026c1d4dcb2602646dec9751e763dba37bdf8ff9406ad9e530ee5db382f413001aeb06a53ed9027d831179727b0865a8918da3edbebcf9b14ed44ce6cbaced4bb1bdb7f1447e6cc254b332051512bd7af426fb8f401378cd2bf5983ca01c64b92ecf032ea15d1721d03f482d7ce6e74fef6d55e702f46980c82b5a84031900b1c9e59e7c97fbec7e8f323a97a7e36cc88be0f1d45b7ff585ac54bd407b22b4154aacc8f6d7ebf48e1d814cc5ed20f8037e0a79715eef29be32806a1d58bb7c5da76f550aa3d8a1fbff0eb19ccb1a313d55cda56c9ec2ef29632387fe8d76e3c0468043e8f663f4860ee12bf2d5b0b7474d6e694f91e6dcc4024ffffffffffffffff" - }, - "modp18": { - "gen": "02", - "prime": "ffffffffffffffffc90fdaa22168c234c4c6628b80dc1cd129024e088a67cc74020bbea63b139b22514a08798e3404ddef9519b3cd3a431b302b0a6df25f14374fe1356d6d51c245e485b576625e7ec6f44c42e9a637ed6b0bff5cb6f406b7edee386bfb5a899fa5ae9f24117c4b1fe649286651ece45b3dc2007cb8a163bf0598da48361c55d39a69163fa8fd24cf5f83655d23dca3ad961c62f356208552bb9ed529077096966d670c354e4abc9804f1746c08ca18217c32905e462e36ce3be39e772c180e86039b2783a2ec07a28fb5c55df06f4c52c9de2bcbf6955817183995497cea956ae515d2261898fa051015728e5a8aaac42dad33170d04507a33a85521abdf1cba64ecfb850458dbef0a8aea71575d060c7db3970f85a6e1e4c7abf5ae8cdb0933d71e8c94e04a25619dcee3d2261ad2ee6bf12ffa06d98a0864d87602733ec86a64521f2b18177b200cbbe117577a615d6c770988c0bad946e208e24fa074e5ab3143db5bfce0fd108e4b82d120a92108011a723c12a787e6d788719a10bdba5b2699c327186af4e23c1a946834b6150bda2583e9ca2ad44ce8dbbbc2db04de8ef92e8efc141fbecaa6287c59474e6bc05d99b2964fa090c3a2233ba186515be7ed1f612970cee2d7afb81bdd762170481cd0069127d5b05aa993b4ea988d8fddc186ffb7dc90a6c08f4df435c93402849236c3fab4d27c7026c1d4dcb2602646dec9751e763dba37bdf8ff9406ad9e530ee5db382f413001aeb06a53ed9027d831179727b0865a8918da3edbebcf9b14ed44ce6cbaced4bb1bdb7f1447e6cc254b332051512bd7af426fb8f401378cd2bf5983ca01c64b92ecf032ea15d1721d03f482d7ce6e74fef6d55e702f46980c82b5a84031900b1c9e59e7c97fbec7e8f323a97a7e36cc88be0f1d45b7ff585ac54bd407b22b4154aacc8f6d7ebf48e1d814cc5ed20f8037e0a79715eef29be32806a1d58bb7c5da76f550aa3d8a1fbff0eb19ccb1a313d55cda56c9ec2ef29632387fe8d76e3c0468043e8f663f4860ee12bf2d5b0b7474d6e694f91e6dbe115974a3926f12fee5e438777cb6a932df8cd8bec4d073b931ba3bc832b68d9dd300741fa7bf8afc47ed2576f6936ba424663aab639c5ae4f5683423b4742bf1c978238f16cbe39d652de3fdb8befc848ad922222e04a4037c0713eb57a81a23f0c73473fc646cea306b4bcbc8862f8385ddfa9d4b7fa2c087e879683303ed5bdd3a062b3cf5b3a278a66d2a13f83f44f82ddf310ee074ab6a364597e899a0255dc164f31cc50846851df9ab48195ded7ea1b1d510bd7ee74d73faf36bc31ecfa268359046f4eb879f924009438b481c6cd7889a002ed5ee382bc9190da6fc026e479558e4475677e9aa9e3050e2765694dfc81f56e880b96e7160c980dd98edd3dfffffffffffffffff" - } -} -},{}],147:[function(require,module,exports){ -arguments[4][70][0].apply(exports,arguments) -},{"buffer":77,"dup":70}],148:[function(require,module,exports){ -'use strict'; - -var elliptic = exports; - -elliptic.version = require('../package.json').version; -elliptic.utils = require('./elliptic/utils'); -elliptic.rand = require('brorand'); -elliptic.curve = require('./elliptic/curve'); -elliptic.curves = require('./elliptic/curves'); - -// Protocols -elliptic.ec = require('./elliptic/ec'); -elliptic.eddsa = require('./elliptic/eddsa'); - -},{"../package.json":164,"./elliptic/curve":151,"./elliptic/curves":154,"./elliptic/ec":155,"./elliptic/eddsa":158,"./elliptic/utils":162,"brorand":76}],149:[function(require,module,exports){ -'use strict'; - -var BN = require('bn.js'); -var utils = require('../utils'); -var getNAF = utils.getNAF; -var getJSF = utils.getJSF; -var assert = utils.assert; - -function BaseCurve(type, conf) { - this.type = type; - this.p = new BN(conf.p, 16); - - // Use Montgomery, when there is no fast reduction for the prime - this.red = conf.prime ? BN.red(conf.prime) : BN.mont(this.p); - - // Useful for many curves - this.zero = new BN(0).toRed(this.red); - this.one = new BN(1).toRed(this.red); - this.two = new BN(2).toRed(this.red); - - // Curve configuration, optional - this.n = conf.n && new BN(conf.n, 16); - this.g = conf.g && this.pointFromJSON(conf.g, conf.gRed); - - // Temporary arrays - this._wnafT1 = new Array(4); - this._wnafT2 = new Array(4); - this._wnafT3 = new Array(4); - this._wnafT4 = new Array(4); - - this._bitLength = this.n ? this.n.bitLength() : 0; - - // Generalized Greg Maxwell's trick - var adjustCount = this.n && this.p.div(this.n); - if (!adjustCount || adjustCount.cmpn(100) > 0) { - this.redN = null; - } else { - this._maxwellTrick = true; - this.redN = this.n.toRed(this.red); - } -} -module.exports = BaseCurve; - -BaseCurve.prototype.point = function point() { - throw new Error('Not implemented'); -}; - -BaseCurve.prototype.validate = function validate() { - throw new Error('Not implemented'); -}; - -BaseCurve.prototype._fixedNafMul = function _fixedNafMul(p, k) { - assert(p.precomputed); - var doubles = p._getDoubles(); - - var naf = getNAF(k, 1, this._bitLength); - var I = (1 << (doubles.step + 1)) - (doubles.step % 2 === 0 ? 2 : 1); - I /= 3; - - // Translate into more windowed form - var repr = []; - for (var j = 0; j < naf.length; j += doubles.step) { - var nafW = 0; - for (var k = j + doubles.step - 1; k >= j; k--) - nafW = (nafW << 1) + naf[k]; - repr.push(nafW); - } - - var a = this.jpoint(null, null, null); - var b = this.jpoint(null, null, null); - for (var i = I; i > 0; i--) { - for (var j = 0; j < repr.length; j++) { - var nafW = repr[j]; - if (nafW === i) - b = b.mixedAdd(doubles.points[j]); - else if (nafW === -i) - b = b.mixedAdd(doubles.points[j].neg()); - } - a = a.add(b); - } - return a.toP(); -}; - -BaseCurve.prototype._wnafMul = function _wnafMul(p, k) { - var w = 4; - - // Precompute window - var nafPoints = p._getNAFPoints(w); - w = nafPoints.wnd; - var wnd = nafPoints.points; - - // Get NAF form - var naf = getNAF(k, w, this._bitLength); - - // Add `this`*(N+1) for every w-NAF index - var acc = this.jpoint(null, null, null); - for (var i = naf.length - 1; i >= 0; i--) { - // Count zeroes - for (var k = 0; i >= 0 && naf[i] === 0; i--) - k++; - if (i >= 0) - k++; - acc = acc.dblp(k); - - if (i < 0) - break; - var z = naf[i]; - assert(z !== 0); - if (p.type === 'affine') { - // J +- P - if (z > 0) - acc = acc.mixedAdd(wnd[(z - 1) >> 1]); - else - acc = acc.mixedAdd(wnd[(-z - 1) >> 1].neg()); - } else { - // J +- J - if (z > 0) - acc = acc.add(wnd[(z - 1) >> 1]); - else - acc = acc.add(wnd[(-z - 1) >> 1].neg()); - } - } - return p.type === 'affine' ? acc.toP() : acc; -}; - -BaseCurve.prototype._wnafMulAdd = function _wnafMulAdd(defW, - points, - coeffs, - len, - jacobianResult) { - var wndWidth = this._wnafT1; - var wnd = this._wnafT2; - var naf = this._wnafT3; - - // Fill all arrays - var max = 0; - for (var i = 0; i < len; i++) { - var p = points[i]; - var nafPoints = p._getNAFPoints(defW); - wndWidth[i] = nafPoints.wnd; - wnd[i] = nafPoints.points; - } - - // Comb small window NAFs - for (var i = len - 1; i >= 1; i -= 2) { - var a = i - 1; - var b = i; - if (wndWidth[a] !== 1 || wndWidth[b] !== 1) { - naf[a] = getNAF(coeffs[a], wndWidth[a], this._bitLength); - naf[b] = getNAF(coeffs[b], wndWidth[b], this._bitLength); - max = Math.max(naf[a].length, max); - max = Math.max(naf[b].length, max); - continue; - } - - var comb = [ - points[a], /* 1 */ - null, /* 3 */ - null, /* 5 */ - points[b] /* 7 */ - ]; - - // Try to avoid Projective points, if possible - if (points[a].y.cmp(points[b].y) === 0) { - comb[1] = points[a].add(points[b]); - comb[2] = points[a].toJ().mixedAdd(points[b].neg()); - } else if (points[a].y.cmp(points[b].y.redNeg()) === 0) { - comb[1] = points[a].toJ().mixedAdd(points[b]); - comb[2] = points[a].add(points[b].neg()); - } else { - comb[1] = points[a].toJ().mixedAdd(points[b]); - comb[2] = points[a].toJ().mixedAdd(points[b].neg()); - } - - var index = [ - -3, /* -1 -1 */ - -1, /* -1 0 */ - -5, /* -1 1 */ - -7, /* 0 -1 */ - 0, /* 0 0 */ - 7, /* 0 1 */ - 5, /* 1 -1 */ - 1, /* 1 0 */ - 3 /* 1 1 */ - ]; - - var jsf = getJSF(coeffs[a], coeffs[b]); - max = Math.max(jsf[0].length, max); - naf[a] = new Array(max); - naf[b] = new Array(max); - for (var j = 0; j < max; j++) { - var ja = jsf[0][j] | 0; - var jb = jsf[1][j] | 0; - - naf[a][j] = index[(ja + 1) * 3 + (jb + 1)]; - naf[b][j] = 0; - wnd[a] = comb; - } - } - - var acc = this.jpoint(null, null, null); - var tmp = this._wnafT4; - for (var i = max; i >= 0; i--) { - var k = 0; - - while (i >= 0) { - var zero = true; - for (var j = 0; j < len; j++) { - tmp[j] = naf[j][i] | 0; - if (tmp[j] !== 0) - zero = false; - } - if (!zero) - break; - k++; - i--; - } - if (i >= 0) - k++; - acc = acc.dblp(k); - if (i < 0) - break; - - for (var j = 0; j < len; j++) { - var z = tmp[j]; - var p; - if (z === 0) - continue; - else if (z > 0) - p = wnd[j][(z - 1) >> 1]; - else if (z < 0) - p = wnd[j][(-z - 1) >> 1].neg(); - - if (p.type === 'affine') - acc = acc.mixedAdd(p); - else - acc = acc.add(p); - } - } - // Zeroify references - for (var i = 0; i < len; i++) - wnd[i] = null; - - if (jacobianResult) - return acc; - else - return acc.toP(); -}; - -function BasePoint(curve, type) { - this.curve = curve; - this.type = type; - this.precomputed = null; -} -BaseCurve.BasePoint = BasePoint; - -BasePoint.prototype.eq = function eq(/*other*/) { - throw new Error('Not implemented'); -}; - -BasePoint.prototype.validate = function validate() { - return this.curve.validate(this); -}; - -BaseCurve.prototype.decodePoint = function decodePoint(bytes, enc) { - bytes = utils.toArray(bytes, enc); - - var len = this.p.byteLength(); - - // uncompressed, hybrid-odd, hybrid-even - if ((bytes[0] === 0x04 || bytes[0] === 0x06 || bytes[0] === 0x07) && - bytes.length - 1 === 2 * len) { - if (bytes[0] === 0x06) - assert(bytes[bytes.length - 1] % 2 === 0); - else if (bytes[0] === 0x07) - assert(bytes[bytes.length - 1] % 2 === 1); - - var res = this.point(bytes.slice(1, 1 + len), - bytes.slice(1 + len, 1 + 2 * len)); - - return res; - } else if ((bytes[0] === 0x02 || bytes[0] === 0x03) && - bytes.length - 1 === len) { - return this.pointFromX(bytes.slice(1, 1 + len), bytes[0] === 0x03); - } - throw new Error('Unknown point format'); -}; - -BasePoint.prototype.encodeCompressed = function encodeCompressed(enc) { - return this.encode(enc, true); -}; - -BasePoint.prototype._encode = function _encode(compact) { - var len = this.curve.p.byteLength(); - var x = this.getX().toArray('be', len); - - if (compact) - return [ this.getY().isEven() ? 0x02 : 0x03 ].concat(x); - - return [ 0x04 ].concat(x, this.getY().toArray('be', len)) ; -}; - -BasePoint.prototype.encode = function encode(enc, compact) { - return utils.encode(this._encode(compact), enc); -}; - -BasePoint.prototype.precompute = function precompute(power) { - if (this.precomputed) - return this; - - var precomputed = { - doubles: null, - naf: null, - beta: null - }; - precomputed.naf = this._getNAFPoints(8); - precomputed.doubles = this._getDoubles(4, power); - precomputed.beta = this._getBeta(); - this.precomputed = precomputed; - - return this; -}; - -BasePoint.prototype._hasDoubles = function _hasDoubles(k) { - if (!this.precomputed) - return false; - - var doubles = this.precomputed.doubles; - if (!doubles) - return false; - - return doubles.points.length >= Math.ceil((k.bitLength() + 1) / doubles.step); -}; - -BasePoint.prototype._getDoubles = function _getDoubles(step, power) { - if (this.precomputed && this.precomputed.doubles) - return this.precomputed.doubles; - - var doubles = [ this ]; - var acc = this; - for (var i = 0; i < power; i += step) { - for (var j = 0; j < step; j++) - acc = acc.dbl(); - doubles.push(acc); - } - return { - step: step, - points: doubles - }; -}; - -BasePoint.prototype._getNAFPoints = function _getNAFPoints(wnd) { - if (this.precomputed && this.precomputed.naf) - return this.precomputed.naf; - - var res = [ this ]; - var max = (1 << wnd) - 1; - var dbl = max === 1 ? null : this.dbl(); - for (var i = 1; i < max; i++) - res[i] = res[i - 1].add(dbl); - return { - wnd: wnd, - points: res - }; -}; - -BasePoint.prototype._getBeta = function _getBeta() { - return null; -}; - -BasePoint.prototype.dblp = function dblp(k) { - var r = this; - for (var i = 0; i < k; i++) - r = r.dbl(); - return r; -}; - -},{"../utils":162,"bn.js":163}],150:[function(require,module,exports){ -'use strict'; - -var utils = require('../utils'); -var BN = require('bn.js'); -var inherits = require('inherits'); -var Base = require('./base'); - -var assert = utils.assert; - -function EdwardsCurve(conf) { - // NOTE: Important as we are creating point in Base.call() - this.twisted = (conf.a | 0) !== 1; - this.mOneA = this.twisted && (conf.a | 0) === -1; - this.extended = this.mOneA; - - Base.call(this, 'edwards', conf); - - this.a = new BN(conf.a, 16).umod(this.red.m); - this.a = this.a.toRed(this.red); - this.c = new BN(conf.c, 16).toRed(this.red); - this.c2 = this.c.redSqr(); - this.d = new BN(conf.d, 16).toRed(this.red); - this.dd = this.d.redAdd(this.d); - - assert(!this.twisted || this.c.fromRed().cmpn(1) === 0); - this.oneC = (conf.c | 0) === 1; -} -inherits(EdwardsCurve, Base); -module.exports = EdwardsCurve; - -EdwardsCurve.prototype._mulA = function _mulA(num) { - if (this.mOneA) - return num.redNeg(); - else - return this.a.redMul(num); -}; - -EdwardsCurve.prototype._mulC = function _mulC(num) { - if (this.oneC) - return num; - else - return this.c.redMul(num); -}; - -// Just for compatibility with Short curve -EdwardsCurve.prototype.jpoint = function jpoint(x, y, z, t) { - return this.point(x, y, z, t); -}; - -EdwardsCurve.prototype.pointFromX = function pointFromX(x, odd) { - x = new BN(x, 16); - if (!x.red) - x = x.toRed(this.red); - - var x2 = x.redSqr(); - var rhs = this.c2.redSub(this.a.redMul(x2)); - var lhs = this.one.redSub(this.c2.redMul(this.d).redMul(x2)); - - var y2 = rhs.redMul(lhs.redInvm()); - var y = y2.redSqrt(); - if (y.redSqr().redSub(y2).cmp(this.zero) !== 0) - throw new Error('invalid point'); - - var isOdd = y.fromRed().isOdd(); - if (odd && !isOdd || !odd && isOdd) - y = y.redNeg(); - - return this.point(x, y); -}; - -EdwardsCurve.prototype.pointFromY = function pointFromY(y, odd) { - y = new BN(y, 16); - if (!y.red) - y = y.toRed(this.red); - - // x^2 = (y^2 - c^2) / (c^2 d y^2 - a) - var y2 = y.redSqr(); - var lhs = y2.redSub(this.c2); - var rhs = y2.redMul(this.d).redMul(this.c2).redSub(this.a); - var x2 = lhs.redMul(rhs.redInvm()); - - if (x2.cmp(this.zero) === 0) { - if (odd) - throw new Error('invalid point'); - else - return this.point(this.zero, y); - } - - var x = x2.redSqrt(); - if (x.redSqr().redSub(x2).cmp(this.zero) !== 0) - throw new Error('invalid point'); - - if (x.fromRed().isOdd() !== odd) - x = x.redNeg(); - - return this.point(x, y); -}; - -EdwardsCurve.prototype.validate = function validate(point) { - if (point.isInfinity()) - return true; - - // Curve: A * X^2 + Y^2 = C^2 * (1 + D * X^2 * Y^2) - point.normalize(); - - var x2 = point.x.redSqr(); - var y2 = point.y.redSqr(); - var lhs = x2.redMul(this.a).redAdd(y2); - var rhs = this.c2.redMul(this.one.redAdd(this.d.redMul(x2).redMul(y2))); - - return lhs.cmp(rhs) === 0; -}; - -function Point(curve, x, y, z, t) { - Base.BasePoint.call(this, curve, 'projective'); - if (x === null && y === null && z === null) { - this.x = this.curve.zero; - this.y = this.curve.one; - this.z = this.curve.one; - this.t = this.curve.zero; - this.zOne = true; - } else { - this.x = new BN(x, 16); - this.y = new BN(y, 16); - this.z = z ? new BN(z, 16) : this.curve.one; - this.t = t && new BN(t, 16); - if (!this.x.red) - this.x = this.x.toRed(this.curve.red); - if (!this.y.red) - this.y = this.y.toRed(this.curve.red); - if (!this.z.red) - this.z = this.z.toRed(this.curve.red); - if (this.t && !this.t.red) - this.t = this.t.toRed(this.curve.red); - this.zOne = this.z === this.curve.one; - - // Use extended coordinates - if (this.curve.extended && !this.t) { - this.t = this.x.redMul(this.y); - if (!this.zOne) - this.t = this.t.redMul(this.z.redInvm()); - } - } -} -inherits(Point, Base.BasePoint); - -EdwardsCurve.prototype.pointFromJSON = function pointFromJSON(obj) { - return Point.fromJSON(this, obj); -}; - -EdwardsCurve.prototype.point = function point(x, y, z, t) { - return new Point(this, x, y, z, t); -}; - -Point.fromJSON = function fromJSON(curve, obj) { - return new Point(curve, obj[0], obj[1], obj[2]); -}; - -Point.prototype.inspect = function inspect() { - if (this.isInfinity()) - return ''; - return ''; -}; - -Point.prototype.isInfinity = function isInfinity() { - // XXX This code assumes that zero is always zero in red - return this.x.cmpn(0) === 0 && - (this.y.cmp(this.z) === 0 || - (this.zOne && this.y.cmp(this.curve.c) === 0)); -}; - -Point.prototype._extDbl = function _extDbl() { - // hyperelliptic.org/EFD/g1p/auto-twisted-extended-1.html - // #doubling-dbl-2008-hwcd - // 4M + 4S - - // A = X1^2 - var a = this.x.redSqr(); - // B = Y1^2 - var b = this.y.redSqr(); - // C = 2 * Z1^2 - var c = this.z.redSqr(); - c = c.redIAdd(c); - // D = a * A - var d = this.curve._mulA(a); - // E = (X1 + Y1)^2 - A - B - var e = this.x.redAdd(this.y).redSqr().redISub(a).redISub(b); - // G = D + B - var g = d.redAdd(b); - // F = G - C - var f = g.redSub(c); - // H = D - B - var h = d.redSub(b); - // X3 = E * F - var nx = e.redMul(f); - // Y3 = G * H - var ny = g.redMul(h); - // T3 = E * H - var nt = e.redMul(h); - // Z3 = F * G - var nz = f.redMul(g); - return this.curve.point(nx, ny, nz, nt); -}; - -Point.prototype._projDbl = function _projDbl() { - // hyperelliptic.org/EFD/g1p/auto-twisted-projective.html - // #doubling-dbl-2008-bbjlp - // #doubling-dbl-2007-bl - // and others - // Generally 3M + 4S or 2M + 4S - - // B = (X1 + Y1)^2 - var b = this.x.redAdd(this.y).redSqr(); - // C = X1^2 - var c = this.x.redSqr(); - // D = Y1^2 - var d = this.y.redSqr(); - - var nx; - var ny; - var nz; - if (this.curve.twisted) { - // E = a * C - var e = this.curve._mulA(c); - // F = E + D - var f = e.redAdd(d); - if (this.zOne) { - // X3 = (B - C - D) * (F - 2) - nx = b.redSub(c).redSub(d).redMul(f.redSub(this.curve.two)); - // Y3 = F * (E - D) - ny = f.redMul(e.redSub(d)); - // Z3 = F^2 - 2 * F - nz = f.redSqr().redSub(f).redSub(f); - } else { - // H = Z1^2 - var h = this.z.redSqr(); - // J = F - 2 * H - var j = f.redSub(h).redISub(h); - // X3 = (B-C-D)*J - nx = b.redSub(c).redISub(d).redMul(j); - // Y3 = F * (E - D) - ny = f.redMul(e.redSub(d)); - // Z3 = F * J - nz = f.redMul(j); - } - } else { - // E = C + D - var e = c.redAdd(d); - // H = (c * Z1)^2 - var h = this.curve._mulC(this.z).redSqr(); - // J = E - 2 * H - var j = e.redSub(h).redSub(h); - // X3 = c * (B - E) * J - nx = this.curve._mulC(b.redISub(e)).redMul(j); - // Y3 = c * E * (C - D) - ny = this.curve._mulC(e).redMul(c.redISub(d)); - // Z3 = E * J - nz = e.redMul(j); - } - return this.curve.point(nx, ny, nz); -}; - -Point.prototype.dbl = function dbl() { - if (this.isInfinity()) - return this; - - // Double in extended coordinates - if (this.curve.extended) - return this._extDbl(); - else - return this._projDbl(); -}; - -Point.prototype._extAdd = function _extAdd(p) { - // hyperelliptic.org/EFD/g1p/auto-twisted-extended-1.html - // #addition-add-2008-hwcd-3 - // 8M - - // A = (Y1 - X1) * (Y2 - X2) - var a = this.y.redSub(this.x).redMul(p.y.redSub(p.x)); - // B = (Y1 + X1) * (Y2 + X2) - var b = this.y.redAdd(this.x).redMul(p.y.redAdd(p.x)); - // C = T1 * k * T2 - var c = this.t.redMul(this.curve.dd).redMul(p.t); - // D = Z1 * 2 * Z2 - var d = this.z.redMul(p.z.redAdd(p.z)); - // E = B - A - var e = b.redSub(a); - // F = D - C - var f = d.redSub(c); - // G = D + C - var g = d.redAdd(c); - // H = B + A - var h = b.redAdd(a); - // X3 = E * F - var nx = e.redMul(f); - // Y3 = G * H - var ny = g.redMul(h); - // T3 = E * H - var nt = e.redMul(h); - // Z3 = F * G - var nz = f.redMul(g); - return this.curve.point(nx, ny, nz, nt); -}; - -Point.prototype._projAdd = function _projAdd(p) { - // hyperelliptic.org/EFD/g1p/auto-twisted-projective.html - // #addition-add-2008-bbjlp - // #addition-add-2007-bl - // 10M + 1S - - // A = Z1 * Z2 - var a = this.z.redMul(p.z); - // B = A^2 - var b = a.redSqr(); - // C = X1 * X2 - var c = this.x.redMul(p.x); - // D = Y1 * Y2 - var d = this.y.redMul(p.y); - // E = d * C * D - var e = this.curve.d.redMul(c).redMul(d); - // F = B - E - var f = b.redSub(e); - // G = B + E - var g = b.redAdd(e); - // X3 = A * F * ((X1 + Y1) * (X2 + Y2) - C - D) - var tmp = this.x.redAdd(this.y).redMul(p.x.redAdd(p.y)).redISub(c).redISub(d); - var nx = a.redMul(f).redMul(tmp); - var ny; - var nz; - if (this.curve.twisted) { - // Y3 = A * G * (D - a * C) - ny = a.redMul(g).redMul(d.redSub(this.curve._mulA(c))); - // Z3 = F * G - nz = f.redMul(g); - } else { - // Y3 = A * G * (D - C) - ny = a.redMul(g).redMul(d.redSub(c)); - // Z3 = c * F * G - nz = this.curve._mulC(f).redMul(g); - } - return this.curve.point(nx, ny, nz); -}; - -Point.prototype.add = function add(p) { - if (this.isInfinity()) - return p; - if (p.isInfinity()) - return this; - - if (this.curve.extended) - return this._extAdd(p); - else - return this._projAdd(p); -}; - -Point.prototype.mul = function mul(k) { - if (this._hasDoubles(k)) - return this.curve._fixedNafMul(this, k); - else - return this.curve._wnafMul(this, k); -}; - -Point.prototype.mulAdd = function mulAdd(k1, p, k2) { - return this.curve._wnafMulAdd(1, [ this, p ], [ k1, k2 ], 2, false); -}; - -Point.prototype.jmulAdd = function jmulAdd(k1, p, k2) { - return this.curve._wnafMulAdd(1, [ this, p ], [ k1, k2 ], 2, true); -}; - -Point.prototype.normalize = function normalize() { - if (this.zOne) - return this; - - // Normalize coordinates - var zi = this.z.redInvm(); - this.x = this.x.redMul(zi); - this.y = this.y.redMul(zi); - if (this.t) - this.t = this.t.redMul(zi); - this.z = this.curve.one; - this.zOne = true; - return this; -}; - -Point.prototype.neg = function neg() { - return this.curve.point(this.x.redNeg(), - this.y, - this.z, - this.t && this.t.redNeg()); -}; - -Point.prototype.getX = function getX() { - this.normalize(); - return this.x.fromRed(); -}; - -Point.prototype.getY = function getY() { - this.normalize(); - return this.y.fromRed(); -}; - -Point.prototype.eq = function eq(other) { - return this === other || - this.getX().cmp(other.getX()) === 0 && - this.getY().cmp(other.getY()) === 0; -}; - -Point.prototype.eqXToP = function eqXToP(x) { - var rx = x.toRed(this.curve.red).redMul(this.z); - if (this.x.cmp(rx) === 0) - return true; - - var xc = x.clone(); - var t = this.curve.redN.redMul(this.z); - for (;;) { - xc.iadd(this.curve.n); - if (xc.cmp(this.curve.p) >= 0) - return false; - - rx.redIAdd(t); - if (this.x.cmp(rx) === 0) - return true; - } -}; - -// Compatibility with BaseCurve -Point.prototype.toP = Point.prototype.normalize; -Point.prototype.mixedAdd = Point.prototype.add; - -},{"../utils":162,"./base":149,"bn.js":163,"inherits":218}],151:[function(require,module,exports){ -'use strict'; - -var curve = exports; - -curve.base = require('./base'); -curve.short = require('./short'); -curve.mont = require('./mont'); -curve.edwards = require('./edwards'); - -},{"./base":149,"./edwards":150,"./mont":152,"./short":153}],152:[function(require,module,exports){ -'use strict'; - -var BN = require('bn.js'); -var inherits = require('inherits'); -var Base = require('./base'); - -var utils = require('../utils'); - -function MontCurve(conf) { - Base.call(this, 'mont', conf); - - this.a = new BN(conf.a, 16).toRed(this.red); - this.b = new BN(conf.b, 16).toRed(this.red); - this.i4 = new BN(4).toRed(this.red).redInvm(); - this.two = new BN(2).toRed(this.red); - this.a24 = this.i4.redMul(this.a.redAdd(this.two)); -} -inherits(MontCurve, Base); -module.exports = MontCurve; - -MontCurve.prototype.validate = function validate(point) { - var x = point.normalize().x; - var x2 = x.redSqr(); - var rhs = x2.redMul(x).redAdd(x2.redMul(this.a)).redAdd(x); - var y = rhs.redSqrt(); - - return y.redSqr().cmp(rhs) === 0; -}; - -function Point(curve, x, z) { - Base.BasePoint.call(this, curve, 'projective'); - if (x === null && z === null) { - this.x = this.curve.one; - this.z = this.curve.zero; - } else { - this.x = new BN(x, 16); - this.z = new BN(z, 16); - if (!this.x.red) - this.x = this.x.toRed(this.curve.red); - if (!this.z.red) - this.z = this.z.toRed(this.curve.red); - } -} -inherits(Point, Base.BasePoint); - -MontCurve.prototype.decodePoint = function decodePoint(bytes, enc) { - return this.point(utils.toArray(bytes, enc), 1); -}; - -MontCurve.prototype.point = function point(x, z) { - return new Point(this, x, z); -}; - -MontCurve.prototype.pointFromJSON = function pointFromJSON(obj) { - return Point.fromJSON(this, obj); -}; - -Point.prototype.precompute = function precompute() { - // No-op -}; - -Point.prototype._encode = function _encode() { - return this.getX().toArray('be', this.curve.p.byteLength()); -}; - -Point.fromJSON = function fromJSON(curve, obj) { - return new Point(curve, obj[0], obj[1] || curve.one); -}; - -Point.prototype.inspect = function inspect() { - if (this.isInfinity()) - return ''; - return ''; -}; - -Point.prototype.isInfinity = function isInfinity() { - // XXX This code assumes that zero is always zero in red - return this.z.cmpn(0) === 0; -}; - -Point.prototype.dbl = function dbl() { - // http://hyperelliptic.org/EFD/g1p/auto-montgom-xz.html#doubling-dbl-1987-m-3 - // 2M + 2S + 4A - - // A = X1 + Z1 - var a = this.x.redAdd(this.z); - // AA = A^2 - var aa = a.redSqr(); - // B = X1 - Z1 - var b = this.x.redSub(this.z); - // BB = B^2 - var bb = b.redSqr(); - // C = AA - BB - var c = aa.redSub(bb); - // X3 = AA * BB - var nx = aa.redMul(bb); - // Z3 = C * (BB + A24 * C) - var nz = c.redMul(bb.redAdd(this.curve.a24.redMul(c))); - return this.curve.point(nx, nz); -}; - -Point.prototype.add = function add() { - throw new Error('Not supported on Montgomery curve'); -}; - -Point.prototype.diffAdd = function diffAdd(p, diff) { - // http://hyperelliptic.org/EFD/g1p/auto-montgom-xz.html#diffadd-dadd-1987-m-3 - // 4M + 2S + 6A - - // A = X2 + Z2 - var a = this.x.redAdd(this.z); - // B = X2 - Z2 - var b = this.x.redSub(this.z); - // C = X3 + Z3 - var c = p.x.redAdd(p.z); - // D = X3 - Z3 - var d = p.x.redSub(p.z); - // DA = D * A - var da = d.redMul(a); - // CB = C * B - var cb = c.redMul(b); - // X5 = Z1 * (DA + CB)^2 - var nx = diff.z.redMul(da.redAdd(cb).redSqr()); - // Z5 = X1 * (DA - CB)^2 - var nz = diff.x.redMul(da.redISub(cb).redSqr()); - return this.curve.point(nx, nz); -}; - -Point.prototype.mul = function mul(k) { - var t = k.clone(); - var a = this; // (N / 2) * Q + Q - var b = this.curve.point(null, null); // (N / 2) * Q - var c = this; // Q - - for (var bits = []; t.cmpn(0) !== 0; t.iushrn(1)) - bits.push(t.andln(1)); - - for (var i = bits.length - 1; i >= 0; i--) { - if (bits[i] === 0) { - // N * Q + Q = ((N / 2) * Q + Q)) + (N / 2) * Q - a = a.diffAdd(b, c); - // N * Q = 2 * ((N / 2) * Q + Q)) - b = b.dbl(); - } else { - // N * Q = ((N / 2) * Q + Q) + ((N / 2) * Q) - b = a.diffAdd(b, c); - // N * Q + Q = 2 * ((N / 2) * Q + Q) - a = a.dbl(); - } - } - return b; -}; - -Point.prototype.mulAdd = function mulAdd() { - throw new Error('Not supported on Montgomery curve'); -}; - -Point.prototype.jumlAdd = function jumlAdd() { - throw new Error('Not supported on Montgomery curve'); -}; - -Point.prototype.eq = function eq(other) { - return this.getX().cmp(other.getX()) === 0; -}; - -Point.prototype.normalize = function normalize() { - this.x = this.x.redMul(this.z.redInvm()); - this.z = this.curve.one; - return this; -}; - -Point.prototype.getX = function getX() { - // Normalize coordinates - this.normalize(); - - return this.x.fromRed(); -}; - -},{"../utils":162,"./base":149,"bn.js":163,"inherits":218}],153:[function(require,module,exports){ -'use strict'; - -var utils = require('../utils'); -var BN = require('bn.js'); -var inherits = require('inherits'); -var Base = require('./base'); - -var assert = utils.assert; - -function ShortCurve(conf) { - Base.call(this, 'short', conf); - - this.a = new BN(conf.a, 16).toRed(this.red); - this.b = new BN(conf.b, 16).toRed(this.red); - this.tinv = this.two.redInvm(); - - this.zeroA = this.a.fromRed().cmpn(0) === 0; - this.threeA = this.a.fromRed().sub(this.p).cmpn(-3) === 0; - - // If the curve is endomorphic, precalculate beta and lambda - this.endo = this._getEndomorphism(conf); - this._endoWnafT1 = new Array(4); - this._endoWnafT2 = new Array(4); -} -inherits(ShortCurve, Base); -module.exports = ShortCurve; - -ShortCurve.prototype._getEndomorphism = function _getEndomorphism(conf) { - // No efficient endomorphism - if (!this.zeroA || !this.g || !this.n || this.p.modn(3) !== 1) - return; - - // Compute beta and lambda, that lambda * P = (beta * Px; Py) - var beta; - var lambda; - if (conf.beta) { - beta = new BN(conf.beta, 16).toRed(this.red); - } else { - var betas = this._getEndoRoots(this.p); - // Choose the smallest beta - beta = betas[0].cmp(betas[1]) < 0 ? betas[0] : betas[1]; - beta = beta.toRed(this.red); - } - if (conf.lambda) { - lambda = new BN(conf.lambda, 16); - } else { - // Choose the lambda that is matching selected beta - var lambdas = this._getEndoRoots(this.n); - if (this.g.mul(lambdas[0]).x.cmp(this.g.x.redMul(beta)) === 0) { - lambda = lambdas[0]; - } else { - lambda = lambdas[1]; - assert(this.g.mul(lambda).x.cmp(this.g.x.redMul(beta)) === 0); - } - } - - // Get basis vectors, used for balanced length-two representation - var basis; - if (conf.basis) { - basis = conf.basis.map(function(vec) { - return { - a: new BN(vec.a, 16), - b: new BN(vec.b, 16) - }; - }); - } else { - basis = this._getEndoBasis(lambda); - } - - return { - beta: beta, - lambda: lambda, - basis: basis - }; -}; - -ShortCurve.prototype._getEndoRoots = function _getEndoRoots(num) { - // Find roots of for x^2 + x + 1 in F - // Root = (-1 +- Sqrt(-3)) / 2 - // - var red = num === this.p ? this.red : BN.mont(num); - var tinv = new BN(2).toRed(red).redInvm(); - var ntinv = tinv.redNeg(); - - var s = new BN(3).toRed(red).redNeg().redSqrt().redMul(tinv); - - var l1 = ntinv.redAdd(s).fromRed(); - var l2 = ntinv.redSub(s).fromRed(); - return [ l1, l2 ]; -}; - -ShortCurve.prototype._getEndoBasis = function _getEndoBasis(lambda) { - // aprxSqrt >= sqrt(this.n) - var aprxSqrt = this.n.ushrn(Math.floor(this.n.bitLength() / 2)); - - // 3.74 - // Run EGCD, until r(L + 1) < aprxSqrt - var u = lambda; - var v = this.n.clone(); - var x1 = new BN(1); - var y1 = new BN(0); - var x2 = new BN(0); - var y2 = new BN(1); - - // NOTE: all vectors are roots of: a + b * lambda = 0 (mod n) - var a0; - var b0; - // First vector - var a1; - var b1; - // Second vector - var a2; - var b2; - - var prevR; - var i = 0; - var r; - var x; - while (u.cmpn(0) !== 0) { - var q = v.div(u); - r = v.sub(q.mul(u)); - x = x2.sub(q.mul(x1)); - var y = y2.sub(q.mul(y1)); - - if (!a1 && r.cmp(aprxSqrt) < 0) { - a0 = prevR.neg(); - b0 = x1; - a1 = r.neg(); - b1 = x; - } else if (a1 && ++i === 2) { - break; - } - prevR = r; - - v = u; - u = r; - x2 = x1; - x1 = x; - y2 = y1; - y1 = y; - } - a2 = r.neg(); - b2 = x; - - var len1 = a1.sqr().add(b1.sqr()); - var len2 = a2.sqr().add(b2.sqr()); - if (len2.cmp(len1) >= 0) { - a2 = a0; - b2 = b0; - } - - // Normalize signs - if (a1.negative) { - a1 = a1.neg(); - b1 = b1.neg(); - } - if (a2.negative) { - a2 = a2.neg(); - b2 = b2.neg(); - } - - return [ - { a: a1, b: b1 }, - { a: a2, b: b2 } - ]; -}; - -ShortCurve.prototype._endoSplit = function _endoSplit(k) { - var basis = this.endo.basis; - var v1 = basis[0]; - var v2 = basis[1]; - - var c1 = v2.b.mul(k).divRound(this.n); - var c2 = v1.b.neg().mul(k).divRound(this.n); - - var p1 = c1.mul(v1.a); - var p2 = c2.mul(v2.a); - var q1 = c1.mul(v1.b); - var q2 = c2.mul(v2.b); - - // Calculate answer - var k1 = k.sub(p1).sub(p2); - var k2 = q1.add(q2).neg(); - return { k1: k1, k2: k2 }; -}; - -ShortCurve.prototype.pointFromX = function pointFromX(x, odd) { - x = new BN(x, 16); - if (!x.red) - x = x.toRed(this.red); - - var y2 = x.redSqr().redMul(x).redIAdd(x.redMul(this.a)).redIAdd(this.b); - var y = y2.redSqrt(); - if (y.redSqr().redSub(y2).cmp(this.zero) !== 0) - throw new Error('invalid point'); - - // XXX Is there any way to tell if the number is odd without converting it - // to non-red form? - var isOdd = y.fromRed().isOdd(); - if (odd && !isOdd || !odd && isOdd) - y = y.redNeg(); - - return this.point(x, y); -}; - -ShortCurve.prototype.validate = function validate(point) { - if (point.inf) - return true; - - var x = point.x; - var y = point.y; - - var ax = this.a.redMul(x); - var rhs = x.redSqr().redMul(x).redIAdd(ax).redIAdd(this.b); - return y.redSqr().redISub(rhs).cmpn(0) === 0; -}; - -ShortCurve.prototype._endoWnafMulAdd = - function _endoWnafMulAdd(points, coeffs, jacobianResult) { - var npoints = this._endoWnafT1; - var ncoeffs = this._endoWnafT2; - for (var i = 0; i < points.length; i++) { - var split = this._endoSplit(coeffs[i]); - var p = points[i]; - var beta = p._getBeta(); - - if (split.k1.negative) { - split.k1.ineg(); - p = p.neg(true); - } - if (split.k2.negative) { - split.k2.ineg(); - beta = beta.neg(true); - } - - npoints[i * 2] = p; - npoints[i * 2 + 1] = beta; - ncoeffs[i * 2] = split.k1; - ncoeffs[i * 2 + 1] = split.k2; - } - var res = this._wnafMulAdd(1, npoints, ncoeffs, i * 2, jacobianResult); - - // Clean-up references to points and coefficients - for (var j = 0; j < i * 2; j++) { - npoints[j] = null; - ncoeffs[j] = null; - } - return res; -}; - -function Point(curve, x, y, isRed) { - Base.BasePoint.call(this, curve, 'affine'); - if (x === null && y === null) { - this.x = null; - this.y = null; - this.inf = true; - } else { - this.x = new BN(x, 16); - this.y = new BN(y, 16); - // Force redgomery representation when loading from JSON - if (isRed) { - this.x.forceRed(this.curve.red); - this.y.forceRed(this.curve.red); - } - if (!this.x.red) - this.x = this.x.toRed(this.curve.red); - if (!this.y.red) - this.y = this.y.toRed(this.curve.red); - this.inf = false; - } -} -inherits(Point, Base.BasePoint); - -ShortCurve.prototype.point = function point(x, y, isRed) { - return new Point(this, x, y, isRed); -}; - -ShortCurve.prototype.pointFromJSON = function pointFromJSON(obj, red) { - return Point.fromJSON(this, obj, red); -}; - -Point.prototype._getBeta = function _getBeta() { - if (!this.curve.endo) - return; - - var pre = this.precomputed; - if (pre && pre.beta) - return pre.beta; - - var beta = this.curve.point(this.x.redMul(this.curve.endo.beta), this.y); - if (pre) { - var curve = this.curve; - var endoMul = function(p) { - return curve.point(p.x.redMul(curve.endo.beta), p.y); - }; - pre.beta = beta; - beta.precomputed = { - beta: null, - naf: pre.naf && { - wnd: pre.naf.wnd, - points: pre.naf.points.map(endoMul) - }, - doubles: pre.doubles && { - step: pre.doubles.step, - points: pre.doubles.points.map(endoMul) - } - }; - } - return beta; -}; - -Point.prototype.toJSON = function toJSON() { - if (!this.precomputed) - return [ this.x, this.y ]; - - return [ this.x, this.y, this.precomputed && { - doubles: this.precomputed.doubles && { - step: this.precomputed.doubles.step, - points: this.precomputed.doubles.points.slice(1) - }, - naf: this.precomputed.naf && { - wnd: this.precomputed.naf.wnd, - points: this.precomputed.naf.points.slice(1) - } - } ]; -}; - -Point.fromJSON = function fromJSON(curve, obj, red) { - if (typeof obj === 'string') - obj = JSON.parse(obj); - var res = curve.point(obj[0], obj[1], red); - if (!obj[2]) - return res; - - function obj2point(obj) { - return curve.point(obj[0], obj[1], red); - } - - var pre = obj[2]; - res.precomputed = { - beta: null, - doubles: pre.doubles && { - step: pre.doubles.step, - points: [ res ].concat(pre.doubles.points.map(obj2point)) - }, - naf: pre.naf && { - wnd: pre.naf.wnd, - points: [ res ].concat(pre.naf.points.map(obj2point)) - } - }; - return res; -}; - -Point.prototype.inspect = function inspect() { - if (this.isInfinity()) - return ''; - return ''; -}; - -Point.prototype.isInfinity = function isInfinity() { - return this.inf; -}; - -Point.prototype.add = function add(p) { - // O + P = P - if (this.inf) - return p; - - // P + O = P - if (p.inf) - return this; - - // P + P = 2P - if (this.eq(p)) - return this.dbl(); - - // P + (-P) = O - if (this.neg().eq(p)) - return this.curve.point(null, null); - - // P + Q = O - if (this.x.cmp(p.x) === 0) - return this.curve.point(null, null); - - var c = this.y.redSub(p.y); - if (c.cmpn(0) !== 0) - c = c.redMul(this.x.redSub(p.x).redInvm()); - var nx = c.redSqr().redISub(this.x).redISub(p.x); - var ny = c.redMul(this.x.redSub(nx)).redISub(this.y); - return this.curve.point(nx, ny); -}; - -Point.prototype.dbl = function dbl() { - if (this.inf) - return this; - - // 2P = O - var ys1 = this.y.redAdd(this.y); - if (ys1.cmpn(0) === 0) - return this.curve.point(null, null); - - var a = this.curve.a; - - var x2 = this.x.redSqr(); - var dyinv = ys1.redInvm(); - var c = x2.redAdd(x2).redIAdd(x2).redIAdd(a).redMul(dyinv); - - var nx = c.redSqr().redISub(this.x.redAdd(this.x)); - var ny = c.redMul(this.x.redSub(nx)).redISub(this.y); - return this.curve.point(nx, ny); -}; - -Point.prototype.getX = function getX() { - return this.x.fromRed(); -}; - -Point.prototype.getY = function getY() { - return this.y.fromRed(); -}; - -Point.prototype.mul = function mul(k) { - k = new BN(k, 16); - if (this.isInfinity()) - return this; - else if (this._hasDoubles(k)) - return this.curve._fixedNafMul(this, k); - else if (this.curve.endo) - return this.curve._endoWnafMulAdd([ this ], [ k ]); - else - return this.curve._wnafMul(this, k); -}; - -Point.prototype.mulAdd = function mulAdd(k1, p2, k2) { - var points = [ this, p2 ]; - var coeffs = [ k1, k2 ]; - if (this.curve.endo) - return this.curve._endoWnafMulAdd(points, coeffs); - else - return this.curve._wnafMulAdd(1, points, coeffs, 2); -}; - -Point.prototype.jmulAdd = function jmulAdd(k1, p2, k2) { - var points = [ this, p2 ]; - var coeffs = [ k1, k2 ]; - if (this.curve.endo) - return this.curve._endoWnafMulAdd(points, coeffs, true); - else - return this.curve._wnafMulAdd(1, points, coeffs, 2, true); -}; - -Point.prototype.eq = function eq(p) { - return this === p || - this.inf === p.inf && - (this.inf || this.x.cmp(p.x) === 0 && this.y.cmp(p.y) === 0); -}; - -Point.prototype.neg = function neg(_precompute) { - if (this.inf) - return this; - - var res = this.curve.point(this.x, this.y.redNeg()); - if (_precompute && this.precomputed) { - var pre = this.precomputed; - var negate = function(p) { - return p.neg(); - }; - res.precomputed = { - naf: pre.naf && { - wnd: pre.naf.wnd, - points: pre.naf.points.map(negate) - }, - doubles: pre.doubles && { - step: pre.doubles.step, - points: pre.doubles.points.map(negate) - } - }; - } - return res; -}; - -Point.prototype.toJ = function toJ() { - if (this.inf) - return this.curve.jpoint(null, null, null); - - var res = this.curve.jpoint(this.x, this.y, this.curve.one); - return res; -}; - -function JPoint(curve, x, y, z) { - Base.BasePoint.call(this, curve, 'jacobian'); - if (x === null && y === null && z === null) { - this.x = this.curve.one; - this.y = this.curve.one; - this.z = new BN(0); - } else { - this.x = new BN(x, 16); - this.y = new BN(y, 16); - this.z = new BN(z, 16); - } - if (!this.x.red) - this.x = this.x.toRed(this.curve.red); - if (!this.y.red) - this.y = this.y.toRed(this.curve.red); - if (!this.z.red) - this.z = this.z.toRed(this.curve.red); - - this.zOne = this.z === this.curve.one; -} -inherits(JPoint, Base.BasePoint); - -ShortCurve.prototype.jpoint = function jpoint(x, y, z) { - return new JPoint(this, x, y, z); -}; - -JPoint.prototype.toP = function toP() { - if (this.isInfinity()) - return this.curve.point(null, null); - - var zinv = this.z.redInvm(); - var zinv2 = zinv.redSqr(); - var ax = this.x.redMul(zinv2); - var ay = this.y.redMul(zinv2).redMul(zinv); - - return this.curve.point(ax, ay); -}; - -JPoint.prototype.neg = function neg() { - return this.curve.jpoint(this.x, this.y.redNeg(), this.z); -}; - -JPoint.prototype.add = function add(p) { - // O + P = P - if (this.isInfinity()) - return p; - - // P + O = P - if (p.isInfinity()) - return this; - - // 12M + 4S + 7A - var pz2 = p.z.redSqr(); - var z2 = this.z.redSqr(); - var u1 = this.x.redMul(pz2); - var u2 = p.x.redMul(z2); - var s1 = this.y.redMul(pz2.redMul(p.z)); - var s2 = p.y.redMul(z2.redMul(this.z)); - - var h = u1.redSub(u2); - var r = s1.redSub(s2); - if (h.cmpn(0) === 0) { - if (r.cmpn(0) !== 0) - return this.curve.jpoint(null, null, null); - else - return this.dbl(); - } - - var h2 = h.redSqr(); - var h3 = h2.redMul(h); - var v = u1.redMul(h2); - - var nx = r.redSqr().redIAdd(h3).redISub(v).redISub(v); - var ny = r.redMul(v.redISub(nx)).redISub(s1.redMul(h3)); - var nz = this.z.redMul(p.z).redMul(h); - - return this.curve.jpoint(nx, ny, nz); -}; - -JPoint.prototype.mixedAdd = function mixedAdd(p) { - // O + P = P - if (this.isInfinity()) - return p.toJ(); - - // P + O = P - if (p.isInfinity()) - return this; - - // 8M + 3S + 7A - var z2 = this.z.redSqr(); - var u1 = this.x; - var u2 = p.x.redMul(z2); - var s1 = this.y; - var s2 = p.y.redMul(z2).redMul(this.z); - - var h = u1.redSub(u2); - var r = s1.redSub(s2); - if (h.cmpn(0) === 0) { - if (r.cmpn(0) !== 0) - return this.curve.jpoint(null, null, null); - else - return this.dbl(); - } - - var h2 = h.redSqr(); - var h3 = h2.redMul(h); - var v = u1.redMul(h2); - - var nx = r.redSqr().redIAdd(h3).redISub(v).redISub(v); - var ny = r.redMul(v.redISub(nx)).redISub(s1.redMul(h3)); - var nz = this.z.redMul(h); - - return this.curve.jpoint(nx, ny, nz); -}; - -JPoint.prototype.dblp = function dblp(pow) { - if (pow === 0) - return this; - if (this.isInfinity()) - return this; - if (!pow) - return this.dbl(); - - if (this.curve.zeroA || this.curve.threeA) { - var r = this; - for (var i = 0; i < pow; i++) - r = r.dbl(); - return r; - } - - // 1M + 2S + 1A + N * (4S + 5M + 8A) - // N = 1 => 6M + 6S + 9A - var a = this.curve.a; - var tinv = this.curve.tinv; - - var jx = this.x; - var jy = this.y; - var jz = this.z; - var jz4 = jz.redSqr().redSqr(); - - // Reuse results - var jyd = jy.redAdd(jy); - for (var i = 0; i < pow; i++) { - var jx2 = jx.redSqr(); - var jyd2 = jyd.redSqr(); - var jyd4 = jyd2.redSqr(); - var c = jx2.redAdd(jx2).redIAdd(jx2).redIAdd(a.redMul(jz4)); - - var t1 = jx.redMul(jyd2); - var nx = c.redSqr().redISub(t1.redAdd(t1)); - var t2 = t1.redISub(nx); - var dny = c.redMul(t2); - dny = dny.redIAdd(dny).redISub(jyd4); - var nz = jyd.redMul(jz); - if (i + 1 < pow) - jz4 = jz4.redMul(jyd4); - - jx = nx; - jz = nz; - jyd = dny; - } - - return this.curve.jpoint(jx, jyd.redMul(tinv), jz); -}; - -JPoint.prototype.dbl = function dbl() { - if (this.isInfinity()) - return this; - - if (this.curve.zeroA) - return this._zeroDbl(); - else if (this.curve.threeA) - return this._threeDbl(); - else - return this._dbl(); -}; - -JPoint.prototype._zeroDbl = function _zeroDbl() { - var nx; - var ny; - var nz; - // Z = 1 - if (this.zOne) { - // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-0.html - // #doubling-mdbl-2007-bl - // 1M + 5S + 14A - - // XX = X1^2 - var xx = this.x.redSqr(); - // YY = Y1^2 - var yy = this.y.redSqr(); - // YYYY = YY^2 - var yyyy = yy.redSqr(); - // S = 2 * ((X1 + YY)^2 - XX - YYYY) - var s = this.x.redAdd(yy).redSqr().redISub(xx).redISub(yyyy); - s = s.redIAdd(s); - // M = 3 * XX + a; a = 0 - var m = xx.redAdd(xx).redIAdd(xx); - // T = M ^ 2 - 2*S - var t = m.redSqr().redISub(s).redISub(s); - - // 8 * YYYY - var yyyy8 = yyyy.redIAdd(yyyy); - yyyy8 = yyyy8.redIAdd(yyyy8); - yyyy8 = yyyy8.redIAdd(yyyy8); - - // X3 = T - nx = t; - // Y3 = M * (S - T) - 8 * YYYY - ny = m.redMul(s.redISub(t)).redISub(yyyy8); - // Z3 = 2*Y1 - nz = this.y.redAdd(this.y); - } else { - // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-0.html - // #doubling-dbl-2009-l - // 2M + 5S + 13A - - // A = X1^2 - var a = this.x.redSqr(); - // B = Y1^2 - var b = this.y.redSqr(); - // C = B^2 - var c = b.redSqr(); - // D = 2 * ((X1 + B)^2 - A - C) - var d = this.x.redAdd(b).redSqr().redISub(a).redISub(c); - d = d.redIAdd(d); - // E = 3 * A - var e = a.redAdd(a).redIAdd(a); - // F = E^2 - var f = e.redSqr(); - - // 8 * C - var c8 = c.redIAdd(c); - c8 = c8.redIAdd(c8); - c8 = c8.redIAdd(c8); - - // X3 = F - 2 * D - nx = f.redISub(d).redISub(d); - // Y3 = E * (D - X3) - 8 * C - ny = e.redMul(d.redISub(nx)).redISub(c8); - // Z3 = 2 * Y1 * Z1 - nz = this.y.redMul(this.z); - nz = nz.redIAdd(nz); - } - - return this.curve.jpoint(nx, ny, nz); -}; - -JPoint.prototype._threeDbl = function _threeDbl() { - var nx; - var ny; - var nz; - // Z = 1 - if (this.zOne) { - // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-3.html - // #doubling-mdbl-2007-bl - // 1M + 5S + 15A - - // XX = X1^2 - var xx = this.x.redSqr(); - // YY = Y1^2 - var yy = this.y.redSqr(); - // YYYY = YY^2 - var yyyy = yy.redSqr(); - // S = 2 * ((X1 + YY)^2 - XX - YYYY) - var s = this.x.redAdd(yy).redSqr().redISub(xx).redISub(yyyy); - s = s.redIAdd(s); - // M = 3 * XX + a - var m = xx.redAdd(xx).redIAdd(xx).redIAdd(this.curve.a); - // T = M^2 - 2 * S - var t = m.redSqr().redISub(s).redISub(s); - // X3 = T - nx = t; - // Y3 = M * (S - T) - 8 * YYYY - var yyyy8 = yyyy.redIAdd(yyyy); - yyyy8 = yyyy8.redIAdd(yyyy8); - yyyy8 = yyyy8.redIAdd(yyyy8); - ny = m.redMul(s.redISub(t)).redISub(yyyy8); - // Z3 = 2 * Y1 - nz = this.y.redAdd(this.y); - } else { - // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-3.html#doubling-dbl-2001-b - // 3M + 5S - - // delta = Z1^2 - var delta = this.z.redSqr(); - // gamma = Y1^2 - var gamma = this.y.redSqr(); - // beta = X1 * gamma - var beta = this.x.redMul(gamma); - // alpha = 3 * (X1 - delta) * (X1 + delta) - var alpha = this.x.redSub(delta).redMul(this.x.redAdd(delta)); - alpha = alpha.redAdd(alpha).redIAdd(alpha); - // X3 = alpha^2 - 8 * beta - var beta4 = beta.redIAdd(beta); - beta4 = beta4.redIAdd(beta4); - var beta8 = beta4.redAdd(beta4); - nx = alpha.redSqr().redISub(beta8); - // Z3 = (Y1 + Z1)^2 - gamma - delta - nz = this.y.redAdd(this.z).redSqr().redISub(gamma).redISub(delta); - // Y3 = alpha * (4 * beta - X3) - 8 * gamma^2 - var ggamma8 = gamma.redSqr(); - ggamma8 = ggamma8.redIAdd(ggamma8); - ggamma8 = ggamma8.redIAdd(ggamma8); - ggamma8 = ggamma8.redIAdd(ggamma8); - ny = alpha.redMul(beta4.redISub(nx)).redISub(ggamma8); - } - - return this.curve.jpoint(nx, ny, nz); -}; - -JPoint.prototype._dbl = function _dbl() { - var a = this.curve.a; - - // 4M + 6S + 10A - var jx = this.x; - var jy = this.y; - var jz = this.z; - var jz4 = jz.redSqr().redSqr(); - - var jx2 = jx.redSqr(); - var jy2 = jy.redSqr(); - - var c = jx2.redAdd(jx2).redIAdd(jx2).redIAdd(a.redMul(jz4)); - - var jxd4 = jx.redAdd(jx); - jxd4 = jxd4.redIAdd(jxd4); - var t1 = jxd4.redMul(jy2); - var nx = c.redSqr().redISub(t1.redAdd(t1)); - var t2 = t1.redISub(nx); - - var jyd8 = jy2.redSqr(); - jyd8 = jyd8.redIAdd(jyd8); - jyd8 = jyd8.redIAdd(jyd8); - jyd8 = jyd8.redIAdd(jyd8); - var ny = c.redMul(t2).redISub(jyd8); - var nz = jy.redAdd(jy).redMul(jz); - - return this.curve.jpoint(nx, ny, nz); -}; - -JPoint.prototype.trpl = function trpl() { - if (!this.curve.zeroA) - return this.dbl().add(this); - - // hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-0.html#tripling-tpl-2007-bl - // 5M + 10S + ... - - // XX = X1^2 - var xx = this.x.redSqr(); - // YY = Y1^2 - var yy = this.y.redSqr(); - // ZZ = Z1^2 - var zz = this.z.redSqr(); - // YYYY = YY^2 - var yyyy = yy.redSqr(); - // M = 3 * XX + a * ZZ2; a = 0 - var m = xx.redAdd(xx).redIAdd(xx); - // MM = M^2 - var mm = m.redSqr(); - // E = 6 * ((X1 + YY)^2 - XX - YYYY) - MM - var e = this.x.redAdd(yy).redSqr().redISub(xx).redISub(yyyy); - e = e.redIAdd(e); - e = e.redAdd(e).redIAdd(e); - e = e.redISub(mm); - // EE = E^2 - var ee = e.redSqr(); - // T = 16*YYYY - var t = yyyy.redIAdd(yyyy); - t = t.redIAdd(t); - t = t.redIAdd(t); - t = t.redIAdd(t); - // U = (M + E)^2 - MM - EE - T - var u = m.redIAdd(e).redSqr().redISub(mm).redISub(ee).redISub(t); - // X3 = 4 * (X1 * EE - 4 * YY * U) - var yyu4 = yy.redMul(u); - yyu4 = yyu4.redIAdd(yyu4); - yyu4 = yyu4.redIAdd(yyu4); - var nx = this.x.redMul(ee).redISub(yyu4); - nx = nx.redIAdd(nx); - nx = nx.redIAdd(nx); - // Y3 = 8 * Y1 * (U * (T - U) - E * EE) - var ny = this.y.redMul(u.redMul(t.redISub(u)).redISub(e.redMul(ee))); - ny = ny.redIAdd(ny); - ny = ny.redIAdd(ny); - ny = ny.redIAdd(ny); - // Z3 = (Z1 + E)^2 - ZZ - EE - var nz = this.z.redAdd(e).redSqr().redISub(zz).redISub(ee); - - return this.curve.jpoint(nx, ny, nz); -}; - -JPoint.prototype.mul = function mul(k, kbase) { - k = new BN(k, kbase); - - return this.curve._wnafMul(this, k); -}; - -JPoint.prototype.eq = function eq(p) { - if (p.type === 'affine') - return this.eq(p.toJ()); - - if (this === p) - return true; - - // x1 * z2^2 == x2 * z1^2 - var z2 = this.z.redSqr(); - var pz2 = p.z.redSqr(); - if (this.x.redMul(pz2).redISub(p.x.redMul(z2)).cmpn(0) !== 0) - return false; - - // y1 * z2^3 == y2 * z1^3 - var z3 = z2.redMul(this.z); - var pz3 = pz2.redMul(p.z); - return this.y.redMul(pz3).redISub(p.y.redMul(z3)).cmpn(0) === 0; -}; - -JPoint.prototype.eqXToP = function eqXToP(x) { - var zs = this.z.redSqr(); - var rx = x.toRed(this.curve.red).redMul(zs); - if (this.x.cmp(rx) === 0) - return true; - - var xc = x.clone(); - var t = this.curve.redN.redMul(zs); - for (;;) { - xc.iadd(this.curve.n); - if (xc.cmp(this.curve.p) >= 0) - return false; - - rx.redIAdd(t); - if (this.x.cmp(rx) === 0) - return true; - } -}; - -JPoint.prototype.inspect = function inspect() { - if (this.isInfinity()) - return ''; - return ''; -}; - -JPoint.prototype.isInfinity = function isInfinity() { - // XXX This code assumes that zero is always zero in red - return this.z.cmpn(0) === 0; -}; - -},{"../utils":162,"./base":149,"bn.js":163,"inherits":218}],154:[function(require,module,exports){ -'use strict'; - -var curves = exports; - -var hash = require('hash.js'); -var curve = require('./curve'); -var utils = require('./utils'); - -var assert = utils.assert; - -function PresetCurve(options) { - if (options.type === 'short') - this.curve = new curve.short(options); - else if (options.type === 'edwards') - this.curve = new curve.edwards(options); - else - this.curve = new curve.mont(options); - this.g = this.curve.g; - this.n = this.curve.n; - this.hash = options.hash; - - assert(this.g.validate(), 'Invalid curve'); - assert(this.g.mul(this.n).isInfinity(), 'Invalid curve, G*N != O'); -} -curves.PresetCurve = PresetCurve; - -function defineCurve(name, options) { - Object.defineProperty(curves, name, { - configurable: true, - enumerable: true, - get: function() { - var curve = new PresetCurve(options); - Object.defineProperty(curves, name, { - configurable: true, - enumerable: true, - value: curve - }); - return curve; - } - }); -} - -defineCurve('p192', { - type: 'short', - prime: 'p192', - p: 'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff', - a: 'ffffffff ffffffff ffffffff fffffffe ffffffff fffffffc', - b: '64210519 e59c80e7 0fa7e9ab 72243049 feb8deec c146b9b1', - n: 'ffffffff ffffffff ffffffff 99def836 146bc9b1 b4d22831', - hash: hash.sha256, - gRed: false, - g: [ - '188da80e b03090f6 7cbf20eb 43a18800 f4ff0afd 82ff1012', - '07192b95 ffc8da78 631011ed 6b24cdd5 73f977a1 1e794811' - ] -}); - -defineCurve('p224', { - type: 'short', - prime: 'p224', - p: 'ffffffff ffffffff ffffffff ffffffff 00000000 00000000 00000001', - a: 'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff fffffffe', - b: 'b4050a85 0c04b3ab f5413256 5044b0b7 d7bfd8ba 270b3943 2355ffb4', - n: 'ffffffff ffffffff ffffffff ffff16a2 e0b8f03e 13dd2945 5c5c2a3d', - hash: hash.sha256, - gRed: false, - g: [ - 'b70e0cbd 6bb4bf7f 321390b9 4a03c1d3 56c21122 343280d6 115c1d21', - 'bd376388 b5f723fb 4c22dfe6 cd4375a0 5a074764 44d58199 85007e34' - ] -}); - -defineCurve('p256', { - type: 'short', - prime: null, - p: 'ffffffff 00000001 00000000 00000000 00000000 ffffffff ffffffff ffffffff', - a: 'ffffffff 00000001 00000000 00000000 00000000 ffffffff ffffffff fffffffc', - b: '5ac635d8 aa3a93e7 b3ebbd55 769886bc 651d06b0 cc53b0f6 3bce3c3e 27d2604b', - n: 'ffffffff 00000000 ffffffff ffffffff bce6faad a7179e84 f3b9cac2 fc632551', - hash: hash.sha256, - gRed: false, - g: [ - '6b17d1f2 e12c4247 f8bce6e5 63a440f2 77037d81 2deb33a0 f4a13945 d898c296', - '4fe342e2 fe1a7f9b 8ee7eb4a 7c0f9e16 2bce3357 6b315ece cbb64068 37bf51f5' - ] -}); - -defineCurve('p384', { - type: 'short', - prime: null, - p: 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ' + - 'fffffffe ffffffff 00000000 00000000 ffffffff', - a: 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ' + - 'fffffffe ffffffff 00000000 00000000 fffffffc', - b: 'b3312fa7 e23ee7e4 988e056b e3f82d19 181d9c6e fe814112 0314088f ' + - '5013875a c656398d 8a2ed19d 2a85c8ed d3ec2aef', - n: 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff c7634d81 ' + - 'f4372ddf 581a0db2 48b0a77a ecec196a ccc52973', - hash: hash.sha384, - gRed: false, - g: [ - 'aa87ca22 be8b0537 8eb1c71e f320ad74 6e1d3b62 8ba79b98 59f741e0 82542a38 ' + - '5502f25d bf55296c 3a545e38 72760ab7', - '3617de4a 96262c6f 5d9e98bf 9292dc29 f8f41dbd 289a147c e9da3113 b5f0b8c0 ' + - '0a60b1ce 1d7e819d 7a431d7c 90ea0e5f' - ] -}); - -defineCurve('p521', { - type: 'short', - prime: null, - p: '000001ff ffffffff ffffffff ffffffff ffffffff ffffffff ' + - 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ' + - 'ffffffff ffffffff ffffffff ffffffff ffffffff', - a: '000001ff ffffffff ffffffff ffffffff ffffffff ffffffff ' + - 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff ' + - 'ffffffff ffffffff ffffffff ffffffff fffffffc', - b: '00000051 953eb961 8e1c9a1f 929a21a0 b68540ee a2da725b ' + - '99b315f3 b8b48991 8ef109e1 56193951 ec7e937b 1652c0bd ' + - '3bb1bf07 3573df88 3d2c34f1 ef451fd4 6b503f00', - n: '000001ff ffffffff ffffffff ffffffff ffffffff ffffffff ' + - 'ffffffff ffffffff fffffffa 51868783 bf2f966b 7fcc0148 ' + - 'f709a5d0 3bb5c9b8 899c47ae bb6fb71e 91386409', - hash: hash.sha512, - gRed: false, - g: [ - '000000c6 858e06b7 0404e9cd 9e3ecb66 2395b442 9c648139 ' + - '053fb521 f828af60 6b4d3dba a14b5e77 efe75928 fe1dc127 ' + - 'a2ffa8de 3348b3c1 856a429b f97e7e31 c2e5bd66', - '00000118 39296a78 9a3bc004 5c8a5fb4 2c7d1bd9 98f54449 ' + - '579b4468 17afbd17 273e662c 97ee7299 5ef42640 c550b901 ' + - '3fad0761 353c7086 a272c240 88be9476 9fd16650' - ] -}); - -defineCurve('curve25519', { - type: 'mont', - prime: 'p25519', - p: '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed', - a: '76d06', - b: '1', - n: '1000000000000000 0000000000000000 14def9dea2f79cd6 5812631a5cf5d3ed', - hash: hash.sha256, - gRed: false, - g: [ - '9' - ] -}); - -defineCurve('ed25519', { - type: 'edwards', - prime: 'p25519', - p: '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed', - a: '-1', - c: '1', - // -121665 * (121666^(-1)) (mod P) - d: '52036cee2b6ffe73 8cc740797779e898 00700a4d4141d8ab 75eb4dca135978a3', - n: '1000000000000000 0000000000000000 14def9dea2f79cd6 5812631a5cf5d3ed', - hash: hash.sha256, - gRed: false, - g: [ - '216936d3cd6e53fec0a4e231fdd6dc5c692cc7609525a7b2c9562d608f25d51a', - - // 4/5 - '6666666666666666666666666666666666666666666666666666666666666658' - ] -}); - -var pre; -try { - pre = require('./precomputed/secp256k1'); -} catch (e) { - pre = undefined; -} - -defineCurve('secp256k1', { - type: 'short', - prime: 'k256', - p: 'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff fffffffe fffffc2f', - a: '0', - b: '7', - n: 'ffffffff ffffffff ffffffff fffffffe baaedce6 af48a03b bfd25e8c d0364141', - h: '1', - hash: hash.sha256, - - // Precomputed endomorphism - beta: '7ae96a2b657c07106e64479eac3434e99cf0497512f58995c1396c28719501ee', - lambda: '5363ad4cc05c30e0a5261c028812645a122e22ea20816678df02967c1b23bd72', - basis: [ - { - a: '3086d221a7d46bcde86c90e49284eb15', - b: '-e4437ed6010e88286f547fa90abfe4c3' - }, - { - a: '114ca50f7a8e2f3f657c1108d9d44cfd8', - b: '3086d221a7d46bcde86c90e49284eb15' - } - ], - - gRed: false, - g: [ - '79be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798', - '483ada7726a3c4655da4fbfc0e1108a8fd17b448a68554199c47d08ffb10d4b8', - pre - ] -}); - -},{"./curve":151,"./precomputed/secp256k1":161,"./utils":162,"hash.js":203}],155:[function(require,module,exports){ -'use strict'; - -var BN = require('bn.js'); -var HmacDRBG = require('hmac-drbg'); -var utils = require('../utils'); -var curves = require('../curves'); -var rand = require('brorand'); -var assert = utils.assert; - -var KeyPair = require('./key'); -var Signature = require('./signature'); - -function EC(options) { - if (!(this instanceof EC)) - return new EC(options); - - // Shortcut `elliptic.ec(curve-name)` - if (typeof options === 'string') { - assert(curves.hasOwnProperty(options), 'Unknown curve ' + options); - - options = curves[options]; - } - - // Shortcut for `elliptic.ec(elliptic.curves.curveName)` - if (options instanceof curves.PresetCurve) - options = { curve: options }; - - this.curve = options.curve.curve; - this.n = this.curve.n; - this.nh = this.n.ushrn(1); - this.g = this.curve.g; - - // Point on curve - this.g = options.curve.g; - this.g.precompute(options.curve.n.bitLength() + 1); - - // Hash for function for DRBG - this.hash = options.hash || options.curve.hash; -} -module.exports = EC; - -EC.prototype.keyPair = function keyPair(options) { - return new KeyPair(this, options); -}; - -EC.prototype.keyFromPrivate = function keyFromPrivate(priv, enc) { - return KeyPair.fromPrivate(this, priv, enc); -}; - -EC.prototype.keyFromPublic = function keyFromPublic(pub, enc) { - return KeyPair.fromPublic(this, pub, enc); -}; - -EC.prototype.genKeyPair = function genKeyPair(options) { - if (!options) - options = {}; - - // Instantiate Hmac_DRBG - var drbg = new HmacDRBG({ - hash: this.hash, - pers: options.pers, - persEnc: options.persEnc || 'utf8', - entropy: options.entropy || rand(this.hash.hmacStrength), - entropyEnc: options.entropy && options.entropyEnc || 'utf8', - nonce: this.n.toArray() - }); - - var bytes = this.n.byteLength(); - var ns2 = this.n.sub(new BN(2)); - do { - var priv = new BN(drbg.generate(bytes)); - if (priv.cmp(ns2) > 0) - continue; - - priv.iaddn(1); - return this.keyFromPrivate(priv); - } while (true); -}; - -EC.prototype._truncateToN = function truncateToN(msg, truncOnly) { - var delta = msg.byteLength() * 8 - this.n.bitLength(); - if (delta > 0) - msg = msg.ushrn(delta); - if (!truncOnly && msg.cmp(this.n) >= 0) - return msg.sub(this.n); - else - return msg; -}; - -EC.prototype.sign = function sign(msg, key, enc, options) { - if (typeof enc === 'object') { - options = enc; - enc = null; - } - if (!options) - options = {}; - - key = this.keyFromPrivate(key, enc); - msg = this._truncateToN(new BN(msg, 16)); - - // Zero-extend key to provide enough entropy - var bytes = this.n.byteLength(); - var bkey = key.getPrivate().toArray('be', bytes); - - // Zero-extend nonce to have the same byte size as N - var nonce = msg.toArray('be', bytes); - - // Instantiate Hmac_DRBG - var drbg = new HmacDRBG({ - hash: this.hash, - entropy: bkey, - nonce: nonce, - pers: options.pers, - persEnc: options.persEnc || 'utf8' - }); - - // Number of bytes to generate - var ns1 = this.n.sub(new BN(1)); - - for (var iter = 0; true; iter++) { - var k = options.k ? - options.k(iter) : - new BN(drbg.generate(this.n.byteLength())); - k = this._truncateToN(k, true); - if (k.cmpn(1) <= 0 || k.cmp(ns1) >= 0) - continue; - - var kp = this.g.mul(k); - if (kp.isInfinity()) - continue; - - var kpX = kp.getX(); - var r = kpX.umod(this.n); - if (r.cmpn(0) === 0) - continue; - - var s = k.invm(this.n).mul(r.mul(key.getPrivate()).iadd(msg)); - s = s.umod(this.n); - if (s.cmpn(0) === 0) - continue; - - var recoveryParam = (kp.getY().isOdd() ? 1 : 0) | - (kpX.cmp(r) !== 0 ? 2 : 0); - - // Use complement of `s`, if it is > `n / 2` - if (options.canonical && s.cmp(this.nh) > 0) { - s = this.n.sub(s); - recoveryParam ^= 1; - } - - return new Signature({ r: r, s: s, recoveryParam: recoveryParam }); - } -}; - -EC.prototype.verify = function verify(msg, signature, key, enc) { - msg = this._truncateToN(new BN(msg, 16)); - key = this.keyFromPublic(key, enc); - signature = new Signature(signature, 'hex'); - - // Perform primitive values validation - var r = signature.r; - var s = signature.s; - if (r.cmpn(1) < 0 || r.cmp(this.n) >= 0) - return false; - if (s.cmpn(1) < 0 || s.cmp(this.n) >= 0) - return false; - - // Validate signature - var sinv = s.invm(this.n); - var u1 = sinv.mul(msg).umod(this.n); - var u2 = sinv.mul(r).umod(this.n); - - if (!this.curve._maxwellTrick) { - var p = this.g.mulAdd(u1, key.getPublic(), u2); - if (p.isInfinity()) - return false; - - return p.getX().umod(this.n).cmp(r) === 0; - } - - // NOTE: Greg Maxwell's trick, inspired by: - // https://git.io/vad3K - - var p = this.g.jmulAdd(u1, key.getPublic(), u2); - if (p.isInfinity()) - return false; - - // Compare `p.x` of Jacobian point with `r`, - // this will do `p.x == r * p.z^2` instead of multiplying `p.x` by the - // inverse of `p.z^2` - return p.eqXToP(r); -}; - -EC.prototype.recoverPubKey = function(msg, signature, j, enc) { - assert((3 & j) === j, 'The recovery param is more than two bits'); - signature = new Signature(signature, enc); - - var n = this.n; - var e = new BN(msg); - var r = signature.r; - var s = signature.s; - - // A set LSB signifies that the y-coordinate is odd - var isYOdd = j & 1; - var isSecondKey = j >> 1; - if (r.cmp(this.curve.p.umod(this.curve.n)) >= 0 && isSecondKey) - throw new Error('Unable to find sencond key candinate'); - - // 1.1. Let x = r + jn. - if (isSecondKey) - r = this.curve.pointFromX(r.add(this.curve.n), isYOdd); - else - r = this.curve.pointFromX(r, isYOdd); - - var rInv = signature.r.invm(n); - var s1 = n.sub(e).mul(rInv).umod(n); - var s2 = s.mul(rInv).umod(n); - - // 1.6.1 Compute Q = r^-1 (sR - eG) - // Q = r^-1 (sR + -eG) - return this.g.mulAdd(s1, r, s2); -}; - -EC.prototype.getKeyRecoveryParam = function(e, signature, Q, enc) { - signature = new Signature(signature, enc); - if (signature.recoveryParam !== null) - return signature.recoveryParam; - - for (var i = 0; i < 4; i++) { - var Qprime; - try { - Qprime = this.recoverPubKey(e, signature, i); - } catch (e) { - continue; - } - - if (Qprime.eq(Q)) - return i; - } - throw new Error('Unable to find valid recovery factor'); -}; - -},{"../curves":154,"../utils":162,"./key":156,"./signature":157,"bn.js":163,"brorand":76,"hmac-drbg":215}],156:[function(require,module,exports){ -'use strict'; - -var BN = require('bn.js'); -var utils = require('../utils'); -var assert = utils.assert; - -function KeyPair(ec, options) { - this.ec = ec; - this.priv = null; - this.pub = null; - - // KeyPair(ec, { priv: ..., pub: ... }) - if (options.priv) - this._importPrivate(options.priv, options.privEnc); - if (options.pub) - this._importPublic(options.pub, options.pubEnc); -} -module.exports = KeyPair; - -KeyPair.fromPublic = function fromPublic(ec, pub, enc) { - if (pub instanceof KeyPair) - return pub; - - return new KeyPair(ec, { - pub: pub, - pubEnc: enc - }); -}; - -KeyPair.fromPrivate = function fromPrivate(ec, priv, enc) { - if (priv instanceof KeyPair) - return priv; - - return new KeyPair(ec, { - priv: priv, - privEnc: enc - }); -}; - -KeyPair.prototype.validate = function validate() { - var pub = this.getPublic(); - - if (pub.isInfinity()) - return { result: false, reason: 'Invalid public key' }; - if (!pub.validate()) - return { result: false, reason: 'Public key is not a point' }; - if (!pub.mul(this.ec.curve.n).isInfinity()) - return { result: false, reason: 'Public key * N != O' }; - - return { result: true, reason: null }; -}; - -KeyPair.prototype.getPublic = function getPublic(compact, enc) { - // compact is optional argument - if (typeof compact === 'string') { - enc = compact; - compact = null; - } - - if (!this.pub) - this.pub = this.ec.g.mul(this.priv); - - if (!enc) - return this.pub; - - return this.pub.encode(enc, compact); -}; - -KeyPair.prototype.getPrivate = function getPrivate(enc) { - if (enc === 'hex') - return this.priv.toString(16, 2); - else - return this.priv; -}; - -KeyPair.prototype._importPrivate = function _importPrivate(key, enc) { - this.priv = new BN(key, enc || 16); - - // Ensure that the priv won't be bigger than n, otherwise we may fail - // in fixed multiplication method - this.priv = this.priv.umod(this.ec.curve.n); -}; - -KeyPair.prototype._importPublic = function _importPublic(key, enc) { - if (key.x || key.y) { - // Montgomery points only have an `x` coordinate. - // Weierstrass/Edwards points on the other hand have both `x` and - // `y` coordinates. - if (this.ec.curve.type === 'mont') { - assert(key.x, 'Need x coordinate'); - } else if (this.ec.curve.type === 'short' || - this.ec.curve.type === 'edwards') { - assert(key.x && key.y, 'Need both x and y coordinate'); - } - this.pub = this.ec.curve.point(key.x, key.y); - return; - } - this.pub = this.ec.curve.decodePoint(key, enc); -}; - -// ECDH -KeyPair.prototype.derive = function derive(pub) { - return pub.mul(this.priv).getX(); -}; - -// ECDSA -KeyPair.prototype.sign = function sign(msg, enc, options) { - return this.ec.sign(msg, this, enc, options); -}; - -KeyPair.prototype.verify = function verify(msg, signature) { - return this.ec.verify(msg, signature, this); -}; - -KeyPair.prototype.inspect = function inspect() { - return ''; -}; - -},{"../utils":162,"bn.js":163}],157:[function(require,module,exports){ -'use strict'; - -var BN = require('bn.js'); - -var utils = require('../utils'); -var assert = utils.assert; - -function Signature(options, enc) { - if (options instanceof Signature) - return options; - - if (this._importDER(options, enc)) - return; - - assert(options.r && options.s, 'Signature without r or s'); - this.r = new BN(options.r, 16); - this.s = new BN(options.s, 16); - if (options.recoveryParam === undefined) - this.recoveryParam = null; - else - this.recoveryParam = options.recoveryParam; -} -module.exports = Signature; - -function Position() { - this.place = 0; -} - -function getLength(buf, p) { - var initial = buf[p.place++]; - if (!(initial & 0x80)) { - return initial; - } - var octetLen = initial & 0xf; - - // Indefinite length or overflow - if (octetLen === 0 || octetLen > 4) { - return false; - } - - var val = 0; - for (var i = 0, off = p.place; i < octetLen; i++, off++) { - val <<= 8; - val |= buf[off]; - val >>>= 0; - } - - // Leading zeroes - if (val <= 0x7f) { - return false; - } - - p.place = off; - return val; -} - -function rmPadding(buf) { - var i = 0; - var len = buf.length - 1; - while (!buf[i] && !(buf[i + 1] & 0x80) && i < len) { - i++; - } - if (i === 0) { - return buf; - } - return buf.slice(i); -} - -Signature.prototype._importDER = function _importDER(data, enc) { - data = utils.toArray(data, enc); - var p = new Position(); - if (data[p.place++] !== 0x30) { - return false; - } - var len = getLength(data, p); - if (len === false) { - return false; - } - if ((len + p.place) !== data.length) { - return false; - } - if (data[p.place++] !== 0x02) { - return false; - } - var rlen = getLength(data, p); - if (rlen === false) { - return false; - } - var r = data.slice(p.place, rlen + p.place); - p.place += rlen; - if (data[p.place++] !== 0x02) { - return false; - } - var slen = getLength(data, p); - if (slen === false) { - return false; - } - if (data.length !== slen + p.place) { - return false; - } - var s = data.slice(p.place, slen + p.place); - if (r[0] === 0) { - if (r[1] & 0x80) { - r = r.slice(1); - } else { - // Leading zeroes - return false; - } - } - if (s[0] === 0) { - if (s[1] & 0x80) { - s = s.slice(1); - } else { - // Leading zeroes - return false; - } - } - - this.r = new BN(r); - this.s = new BN(s); - this.recoveryParam = null; - - return true; -}; - -function constructLength(arr, len) { - if (len < 0x80) { - arr.push(len); - return; - } - var octets = 1 + (Math.log(len) / Math.LN2 >>> 3); - arr.push(octets | 0x80); - while (--octets) { - arr.push((len >>> (octets << 3)) & 0xff); - } - arr.push(len); -} - -Signature.prototype.toDER = function toDER(enc) { - var r = this.r.toArray(); - var s = this.s.toArray(); - - // Pad values - if (r[0] & 0x80) - r = [ 0 ].concat(r); - // Pad values - if (s[0] & 0x80) - s = [ 0 ].concat(s); - - r = rmPadding(r); - s = rmPadding(s); - - while (!s[0] && !(s[1] & 0x80)) { - s = s.slice(1); - } - var arr = [ 0x02 ]; - constructLength(arr, r.length); - arr = arr.concat(r); - arr.push(0x02); - constructLength(arr, s.length); - var backHalf = arr.concat(s); - var res = [ 0x30 ]; - constructLength(res, backHalf.length); - res = res.concat(backHalf); - return utils.encode(res, enc); -}; - -},{"../utils":162,"bn.js":163}],158:[function(require,module,exports){ -'use strict'; - -var hash = require('hash.js'); -var curves = require('../curves'); -var utils = require('../utils'); -var assert = utils.assert; -var parseBytes = utils.parseBytes; -var KeyPair = require('./key'); -var Signature = require('./signature'); - -function EDDSA(curve) { - assert(curve === 'ed25519', 'only tested with ed25519 so far'); - - if (!(this instanceof EDDSA)) - return new EDDSA(curve); - - var curve = curves[curve].curve; - this.curve = curve; - this.g = curve.g; - this.g.precompute(curve.n.bitLength() + 1); - - this.pointClass = curve.point().constructor; - this.encodingLength = Math.ceil(curve.n.bitLength() / 8); - this.hash = hash.sha512; -} - -module.exports = EDDSA; - -/** -* @param {Array|String} message - message bytes -* @param {Array|String|KeyPair} secret - secret bytes or a keypair -* @returns {Signature} - signature -*/ -EDDSA.prototype.sign = function sign(message, secret) { - message = parseBytes(message); - var key = this.keyFromSecret(secret); - var r = this.hashInt(key.messagePrefix(), message); - var R = this.g.mul(r); - var Rencoded = this.encodePoint(R); - var s_ = this.hashInt(Rencoded, key.pubBytes(), message) - .mul(key.priv()); - var S = r.add(s_).umod(this.curve.n); - return this.makeSignature({ R: R, S: S, Rencoded: Rencoded }); -}; - -/** -* @param {Array} message - message bytes -* @param {Array|String|Signature} sig - sig bytes -* @param {Array|String|Point|KeyPair} pub - public key -* @returns {Boolean} - true if public key matches sig of message -*/ -EDDSA.prototype.verify = function verify(message, sig, pub) { - message = parseBytes(message); - sig = this.makeSignature(sig); - var key = this.keyFromPublic(pub); - var h = this.hashInt(sig.Rencoded(), key.pubBytes(), message); - var SG = this.g.mul(sig.S()); - var RplusAh = sig.R().add(key.pub().mul(h)); - return RplusAh.eq(SG); -}; - -EDDSA.prototype.hashInt = function hashInt() { - var hash = this.hash(); - for (var i = 0; i < arguments.length; i++) - hash.update(arguments[i]); - return utils.intFromLE(hash.digest()).umod(this.curve.n); -}; - -EDDSA.prototype.keyFromPublic = function keyFromPublic(pub) { - return KeyPair.fromPublic(this, pub); -}; - -EDDSA.prototype.keyFromSecret = function keyFromSecret(secret) { - return KeyPair.fromSecret(this, secret); -}; - -EDDSA.prototype.makeSignature = function makeSignature(sig) { - if (sig instanceof Signature) - return sig; - return new Signature(this, sig); -}; - -/** -* * https://tools.ietf.org/html/draft-josefsson-eddsa-ed25519-03#section-5.2 -* -* EDDSA defines methods for encoding and decoding points and integers. These are -* helper convenience methods, that pass along to utility functions implied -* parameters. -* -*/ -EDDSA.prototype.encodePoint = function encodePoint(point) { - var enc = point.getY().toArray('le', this.encodingLength); - enc[this.encodingLength - 1] |= point.getX().isOdd() ? 0x80 : 0; - return enc; -}; - -EDDSA.prototype.decodePoint = function decodePoint(bytes) { - bytes = utils.parseBytes(bytes); - - var lastIx = bytes.length - 1; - var normed = bytes.slice(0, lastIx).concat(bytes[lastIx] & ~0x80); - var xIsOdd = (bytes[lastIx] & 0x80) !== 0; - - var y = utils.intFromLE(normed); - return this.curve.pointFromY(y, xIsOdd); -}; - -EDDSA.prototype.encodeInt = function encodeInt(num) { - return num.toArray('le', this.encodingLength); -}; - -EDDSA.prototype.decodeInt = function decodeInt(bytes) { - return utils.intFromLE(bytes); -}; - -EDDSA.prototype.isPoint = function isPoint(val) { - return val instanceof this.pointClass; -}; - -},{"../curves":154,"../utils":162,"./key":159,"./signature":160,"hash.js":203}],159:[function(require,module,exports){ -'use strict'; - -var utils = require('../utils'); -var assert = utils.assert; -var parseBytes = utils.parseBytes; -var cachedProperty = utils.cachedProperty; - -/** -* @param {EDDSA} eddsa - instance -* @param {Object} params - public/private key parameters -* -* @param {Array} [params.secret] - secret seed bytes -* @param {Point} [params.pub] - public key point (aka `A` in eddsa terms) -* @param {Array} [params.pub] - public key point encoded as bytes -* -*/ -function KeyPair(eddsa, params) { - this.eddsa = eddsa; - this._secret = parseBytes(params.secret); - if (eddsa.isPoint(params.pub)) - this._pub = params.pub; - else - this._pubBytes = parseBytes(params.pub); -} - -KeyPair.fromPublic = function fromPublic(eddsa, pub) { - if (pub instanceof KeyPair) - return pub; - return new KeyPair(eddsa, { pub: pub }); -}; - -KeyPair.fromSecret = function fromSecret(eddsa, secret) { - if (secret instanceof KeyPair) - return secret; - return new KeyPair(eddsa, { secret: secret }); -}; - -KeyPair.prototype.secret = function secret() { - return this._secret; -}; - -cachedProperty(KeyPair, 'pubBytes', function pubBytes() { - return this.eddsa.encodePoint(this.pub()); -}); - -cachedProperty(KeyPair, 'pub', function pub() { - if (this._pubBytes) - return this.eddsa.decodePoint(this._pubBytes); - return this.eddsa.g.mul(this.priv()); -}); - -cachedProperty(KeyPair, 'privBytes', function privBytes() { - var eddsa = this.eddsa; - var hash = this.hash(); - var lastIx = eddsa.encodingLength - 1; - - var a = hash.slice(0, eddsa.encodingLength); - a[0] &= 248; - a[lastIx] &= 127; - a[lastIx] |= 64; - - return a; -}); - -cachedProperty(KeyPair, 'priv', function priv() { - return this.eddsa.decodeInt(this.privBytes()); -}); - -cachedProperty(KeyPair, 'hash', function hash() { - return this.eddsa.hash().update(this.secret()).digest(); -}); - -cachedProperty(KeyPair, 'messagePrefix', function messagePrefix() { - return this.hash().slice(this.eddsa.encodingLength); -}); - -KeyPair.prototype.sign = function sign(message) { - assert(this._secret, 'KeyPair can only verify'); - return this.eddsa.sign(message, this); -}; - -KeyPair.prototype.verify = function verify(message, sig) { - return this.eddsa.verify(message, sig, this); -}; - -KeyPair.prototype.getSecret = function getSecret(enc) { - assert(this._secret, 'KeyPair is public only'); - return utils.encode(this.secret(), enc); -}; - -KeyPair.prototype.getPublic = function getPublic(enc) { - return utils.encode(this.pubBytes(), enc); -}; - -module.exports = KeyPair; - -},{"../utils":162}],160:[function(require,module,exports){ -'use strict'; - -var BN = require('bn.js'); -var utils = require('../utils'); -var assert = utils.assert; -var cachedProperty = utils.cachedProperty; -var parseBytes = utils.parseBytes; - -/** -* @param {EDDSA} eddsa - eddsa instance -* @param {Array|Object} sig - -* @param {Array|Point} [sig.R] - R point as Point or bytes -* @param {Array|bn} [sig.S] - S scalar as bn or bytes -* @param {Array} [sig.Rencoded] - R point encoded -* @param {Array} [sig.Sencoded] - S scalar encoded -*/ -function Signature(eddsa, sig) { - this.eddsa = eddsa; - - if (typeof sig !== 'object') - sig = parseBytes(sig); - - if (Array.isArray(sig)) { - sig = { - R: sig.slice(0, eddsa.encodingLength), - S: sig.slice(eddsa.encodingLength) - }; - } - - assert(sig.R && sig.S, 'Signature without R or S'); - - if (eddsa.isPoint(sig.R)) - this._R = sig.R; - if (sig.S instanceof BN) - this._S = sig.S; - - this._Rencoded = Array.isArray(sig.R) ? sig.R : sig.Rencoded; - this._Sencoded = Array.isArray(sig.S) ? sig.S : sig.Sencoded; -} - -cachedProperty(Signature, 'S', function S() { - return this.eddsa.decodeInt(this.Sencoded()); -}); - -cachedProperty(Signature, 'R', function R() { - return this.eddsa.decodePoint(this.Rencoded()); -}); - -cachedProperty(Signature, 'Rencoded', function Rencoded() { - return this.eddsa.encodePoint(this.R()); -}); - -cachedProperty(Signature, 'Sencoded', function Sencoded() { - return this.eddsa.encodeInt(this.S()); -}); - -Signature.prototype.toBytes = function toBytes() { - return this.Rencoded().concat(this.Sencoded()); -}; - -Signature.prototype.toHex = function toHex() { - return utils.encode(this.toBytes(), 'hex').toUpperCase(); -}; - -module.exports = Signature; - -},{"../utils":162,"bn.js":163}],161:[function(require,module,exports){ -module.exports = { - doubles: { - step: 4, - points: [ - [ - 'e60fce93b59e9ec53011aabc21c23e97b2a31369b87a5ae9c44ee89e2a6dec0a', - 'f7e3507399e595929db99f34f57937101296891e44d23f0be1f32cce69616821' - ], - [ - '8282263212c609d9ea2a6e3e172de238d8c39cabd5ac1ca10646e23fd5f51508', - '11f8a8098557dfe45e8256e830b60ace62d613ac2f7b17bed31b6eaff6e26caf' - ], - [ - '175e159f728b865a72f99cc6c6fc846de0b93833fd2222ed73fce5b551e5b739', - 'd3506e0d9e3c79eba4ef97a51ff71f5eacb5955add24345c6efa6ffee9fed695' - ], - [ - '363d90d447b00c9c99ceac05b6262ee053441c7e55552ffe526bad8f83ff4640', - '4e273adfc732221953b445397f3363145b9a89008199ecb62003c7f3bee9de9' - ], - [ - '8b4b5f165df3c2be8c6244b5b745638843e4a781a15bcd1b69f79a55dffdf80c', - '4aad0a6f68d308b4b3fbd7813ab0da04f9e336546162ee56b3eff0c65fd4fd36' - ], - [ - '723cbaa6e5db996d6bf771c00bd548c7b700dbffa6c0e77bcb6115925232fcda', - '96e867b5595cc498a921137488824d6e2660a0653779494801dc069d9eb39f5f' - ], - [ - 'eebfa4d493bebf98ba5feec812c2d3b50947961237a919839a533eca0e7dd7fa', - '5d9a8ca3970ef0f269ee7edaf178089d9ae4cdc3a711f712ddfd4fdae1de8999' - ], - [ - '100f44da696e71672791d0a09b7bde459f1215a29b3c03bfefd7835b39a48db0', - 'cdd9e13192a00b772ec8f3300c090666b7ff4a18ff5195ac0fbd5cd62bc65a09' - ], - [ - 'e1031be262c7ed1b1dc9227a4a04c017a77f8d4464f3b3852c8acde6e534fd2d', - '9d7061928940405e6bb6a4176597535af292dd419e1ced79a44f18f29456a00d' - ], - [ - 'feea6cae46d55b530ac2839f143bd7ec5cf8b266a41d6af52d5e688d9094696d', - 'e57c6b6c97dce1bab06e4e12bf3ecd5c981c8957cc41442d3155debf18090088' - ], - [ - 'da67a91d91049cdcb367be4be6ffca3cfeed657d808583de33fa978bc1ec6cb1', - '9bacaa35481642bc41f463f7ec9780e5dec7adc508f740a17e9ea8e27a68be1d' - ], - [ - '53904faa0b334cdda6e000935ef22151ec08d0f7bb11069f57545ccc1a37b7c0', - '5bc087d0bc80106d88c9eccac20d3c1c13999981e14434699dcb096b022771c8' - ], - [ - '8e7bcd0bd35983a7719cca7764ca906779b53a043a9b8bcaeff959f43ad86047', - '10b7770b2a3da4b3940310420ca9514579e88e2e47fd68b3ea10047e8460372a' - ], - [ - '385eed34c1cdff21e6d0818689b81bde71a7f4f18397e6690a841e1599c43862', - '283bebc3e8ea23f56701de19e9ebf4576b304eec2086dc8cc0458fe5542e5453' - ], - [ - '6f9d9b803ecf191637c73a4413dfa180fddf84a5947fbc9c606ed86c3fac3a7', - '7c80c68e603059ba69b8e2a30e45c4d47ea4dd2f5c281002d86890603a842160' - ], - [ - '3322d401243c4e2582a2147c104d6ecbf774d163db0f5e5313b7e0e742d0e6bd', - '56e70797e9664ef5bfb019bc4ddaf9b72805f63ea2873af624f3a2e96c28b2a0' - ], - [ - '85672c7d2de0b7da2bd1770d89665868741b3f9af7643397721d74d28134ab83', - '7c481b9b5b43b2eb6374049bfa62c2e5e77f17fcc5298f44c8e3094f790313a6' - ], - [ - '948bf809b1988a46b06c9f1919413b10f9226c60f668832ffd959af60c82a0a', - '53a562856dcb6646dc6b74c5d1c3418c6d4dff08c97cd2bed4cb7f88d8c8e589' - ], - [ - '6260ce7f461801c34f067ce0f02873a8f1b0e44dfc69752accecd819f38fd8e8', - 'bc2da82b6fa5b571a7f09049776a1ef7ecd292238051c198c1a84e95b2b4ae17' - ], - [ - 'e5037de0afc1d8d43d8348414bbf4103043ec8f575bfdc432953cc8d2037fa2d', - '4571534baa94d3b5f9f98d09fb990bddbd5f5b03ec481f10e0e5dc841d755bda' - ], - [ - 'e06372b0f4a207adf5ea905e8f1771b4e7e8dbd1c6a6c5b725866a0ae4fce725', - '7a908974bce18cfe12a27bb2ad5a488cd7484a7787104870b27034f94eee31dd' - ], - [ - '213c7a715cd5d45358d0bbf9dc0ce02204b10bdde2a3f58540ad6908d0559754', - '4b6dad0b5ae462507013ad06245ba190bb4850f5f36a7eeddff2c27534b458f2' - ], - [ - '4e7c272a7af4b34e8dbb9352a5419a87e2838c70adc62cddf0cc3a3b08fbd53c', - '17749c766c9d0b18e16fd09f6def681b530b9614bff7dd33e0b3941817dcaae6' - ], - [ - 'fea74e3dbe778b1b10f238ad61686aa5c76e3db2be43057632427e2840fb27b6', - '6e0568db9b0b13297cf674deccb6af93126b596b973f7b77701d3db7f23cb96f' - ], - [ - '76e64113f677cf0e10a2570d599968d31544e179b760432952c02a4417bdde39', - 'c90ddf8dee4e95cf577066d70681f0d35e2a33d2b56d2032b4b1752d1901ac01' - ], - [ - 'c738c56b03b2abe1e8281baa743f8f9a8f7cc643df26cbee3ab150242bcbb891', - '893fb578951ad2537f718f2eacbfbbbb82314eef7880cfe917e735d9699a84c3' - ], - [ - 'd895626548b65b81e264c7637c972877d1d72e5f3a925014372e9f6588f6c14b', - 'febfaa38f2bc7eae728ec60818c340eb03428d632bb067e179363ed75d7d991f' - ], - [ - 'b8da94032a957518eb0f6433571e8761ceffc73693e84edd49150a564f676e03', - '2804dfa44805a1e4d7c99cc9762808b092cc584d95ff3b511488e4e74efdf6e7' - ], - [ - 'e80fea14441fb33a7d8adab9475d7fab2019effb5156a792f1a11778e3c0df5d', - 'eed1de7f638e00771e89768ca3ca94472d155e80af322ea9fcb4291b6ac9ec78' - ], - [ - 'a301697bdfcd704313ba48e51d567543f2a182031efd6915ddc07bbcc4e16070', - '7370f91cfb67e4f5081809fa25d40f9b1735dbf7c0a11a130c0d1a041e177ea1' - ], - [ - '90ad85b389d6b936463f9d0512678de208cc330b11307fffab7ac63e3fb04ed4', - 'e507a3620a38261affdcbd9427222b839aefabe1582894d991d4d48cb6ef150' - ], - [ - '8f68b9d2f63b5f339239c1ad981f162ee88c5678723ea3351b7b444c9ec4c0da', - '662a9f2dba063986de1d90c2b6be215dbbea2cfe95510bfdf23cbf79501fff82' - ], - [ - 'e4f3fb0176af85d65ff99ff9198c36091f48e86503681e3e6686fd5053231e11', - '1e63633ad0ef4f1c1661a6d0ea02b7286cc7e74ec951d1c9822c38576feb73bc' - ], - [ - '8c00fa9b18ebf331eb961537a45a4266c7034f2f0d4e1d0716fb6eae20eae29e', - 'efa47267fea521a1a9dc343a3736c974c2fadafa81e36c54e7d2a4c66702414b' - ], - [ - 'e7a26ce69dd4829f3e10cec0a9e98ed3143d084f308b92c0997fddfc60cb3e41', - '2a758e300fa7984b471b006a1aafbb18d0a6b2c0420e83e20e8a9421cf2cfd51' - ], - [ - 'b6459e0ee3662ec8d23540c223bcbdc571cbcb967d79424f3cf29eb3de6b80ef', - '67c876d06f3e06de1dadf16e5661db3c4b3ae6d48e35b2ff30bf0b61a71ba45' - ], - [ - 'd68a80c8280bb840793234aa118f06231d6f1fc67e73c5a5deda0f5b496943e8', - 'db8ba9fff4b586d00c4b1f9177b0e28b5b0e7b8f7845295a294c84266b133120' - ], - [ - '324aed7df65c804252dc0270907a30b09612aeb973449cea4095980fc28d3d5d', - '648a365774b61f2ff130c0c35aec1f4f19213b0c7e332843967224af96ab7c84' - ], - [ - '4df9c14919cde61f6d51dfdbe5fee5dceec4143ba8d1ca888e8bd373fd054c96', - '35ec51092d8728050974c23a1d85d4b5d506cdc288490192ebac06cad10d5d' - ], - [ - '9c3919a84a474870faed8a9c1cc66021523489054d7f0308cbfc99c8ac1f98cd', - 'ddb84f0f4a4ddd57584f044bf260e641905326f76c64c8e6be7e5e03d4fc599d' - ], - [ - '6057170b1dd12fdf8de05f281d8e06bb91e1493a8b91d4cc5a21382120a959e5', - '9a1af0b26a6a4807add9a2daf71df262465152bc3ee24c65e899be932385a2a8' - ], - [ - 'a576df8e23a08411421439a4518da31880cef0fba7d4df12b1a6973eecb94266', - '40a6bf20e76640b2c92b97afe58cd82c432e10a7f514d9f3ee8be11ae1b28ec8' - ], - [ - '7778a78c28dec3e30a05fe9629de8c38bb30d1f5cf9a3a208f763889be58ad71', - '34626d9ab5a5b22ff7098e12f2ff580087b38411ff24ac563b513fc1fd9f43ac' - ], - [ - '928955ee637a84463729fd30e7afd2ed5f96274e5ad7e5cb09eda9c06d903ac', - 'c25621003d3f42a827b78a13093a95eeac3d26efa8a8d83fc5180e935bcd091f' - ], - [ - '85d0fef3ec6db109399064f3a0e3b2855645b4a907ad354527aae75163d82751', - '1f03648413a38c0be29d496e582cf5663e8751e96877331582c237a24eb1f962' - ], - [ - 'ff2b0dce97eece97c1c9b6041798b85dfdfb6d8882da20308f5404824526087e', - '493d13fef524ba188af4c4dc54d07936c7b7ed6fb90e2ceb2c951e01f0c29907' - ], - [ - '827fbbe4b1e880ea9ed2b2e6301b212b57f1ee148cd6dd28780e5e2cf856e241', - 'c60f9c923c727b0b71bef2c67d1d12687ff7a63186903166d605b68baec293ec' - ], - [ - 'eaa649f21f51bdbae7be4ae34ce6e5217a58fdce7f47f9aa7f3b58fa2120e2b3', - 'be3279ed5bbbb03ac69a80f89879aa5a01a6b965f13f7e59d47a5305ba5ad93d' - ], - [ - 'e4a42d43c5cf169d9391df6decf42ee541b6d8f0c9a137401e23632dda34d24f', - '4d9f92e716d1c73526fc99ccfb8ad34ce886eedfa8d8e4f13a7f7131deba9414' - ], - [ - '1ec80fef360cbdd954160fadab352b6b92b53576a88fea4947173b9d4300bf19', - 'aeefe93756b5340d2f3a4958a7abbf5e0146e77f6295a07b671cdc1cc107cefd' - ], - [ - '146a778c04670c2f91b00af4680dfa8bce3490717d58ba889ddb5928366642be', - 'b318e0ec3354028add669827f9d4b2870aaa971d2f7e5ed1d0b297483d83efd0' - ], - [ - 'fa50c0f61d22e5f07e3acebb1aa07b128d0012209a28b9776d76a8793180eef9', - '6b84c6922397eba9b72cd2872281a68a5e683293a57a213b38cd8d7d3f4f2811' - ], - [ - 'da1d61d0ca721a11b1a5bf6b7d88e8421a288ab5d5bba5220e53d32b5f067ec2', - '8157f55a7c99306c79c0766161c91e2966a73899d279b48a655fba0f1ad836f1' - ], - [ - 'a8e282ff0c9706907215ff98e8fd416615311de0446f1e062a73b0610d064e13', - '7f97355b8db81c09abfb7f3c5b2515888b679a3e50dd6bd6cef7c73111f4cc0c' - ], - [ - '174a53b9c9a285872d39e56e6913cab15d59b1fa512508c022f382de8319497c', - 'ccc9dc37abfc9c1657b4155f2c47f9e6646b3a1d8cb9854383da13ac079afa73' - ], - [ - '959396981943785c3d3e57edf5018cdbe039e730e4918b3d884fdff09475b7ba', - '2e7e552888c331dd8ba0386a4b9cd6849c653f64c8709385e9b8abf87524f2fd' - ], - [ - 'd2a63a50ae401e56d645a1153b109a8fcca0a43d561fba2dbb51340c9d82b151', - 'e82d86fb6443fcb7565aee58b2948220a70f750af484ca52d4142174dcf89405' - ], - [ - '64587e2335471eb890ee7896d7cfdc866bacbdbd3839317b3436f9b45617e073', - 'd99fcdd5bf6902e2ae96dd6447c299a185b90a39133aeab358299e5e9faf6589' - ], - [ - '8481bde0e4e4d885b3a546d3e549de042f0aa6cea250e7fd358d6c86dd45e458', - '38ee7b8cba5404dd84a25bf39cecb2ca900a79c42b262e556d64b1b59779057e' - ], - [ - '13464a57a78102aa62b6979ae817f4637ffcfed3c4b1ce30bcd6303f6caf666b', - '69be159004614580ef7e433453ccb0ca48f300a81d0942e13f495a907f6ecc27' - ], - [ - 'bc4a9df5b713fe2e9aef430bcc1dc97a0cd9ccede2f28588cada3a0d2d83f366', - 'd3a81ca6e785c06383937adf4b798caa6e8a9fbfa547b16d758d666581f33c1' - ], - [ - '8c28a97bf8298bc0d23d8c749452a32e694b65e30a9472a3954ab30fe5324caa', - '40a30463a3305193378fedf31f7cc0eb7ae784f0451cb9459e71dc73cbef9482' - ], - [ - '8ea9666139527a8c1dd94ce4f071fd23c8b350c5a4bb33748c4ba111faccae0', - '620efabbc8ee2782e24e7c0cfb95c5d735b783be9cf0f8e955af34a30e62b945' - ], - [ - 'dd3625faef5ba06074669716bbd3788d89bdde815959968092f76cc4eb9a9787', - '7a188fa3520e30d461da2501045731ca941461982883395937f68d00c644a573' - ], - [ - 'f710d79d9eb962297e4f6232b40e8f7feb2bc63814614d692c12de752408221e', - 'ea98e67232d3b3295d3b535532115ccac8612c721851617526ae47a9c77bfc82' - ] - ] - }, - naf: { - wnd: 7, - points: [ - [ - 'f9308a019258c31049344f85f89d5229b531c845836f99b08601f113bce036f9', - '388f7b0f632de8140fe337e62a37f3566500a99934c2231b6cb9fd7584b8e672' - ], - [ - '2f8bde4d1a07209355b4a7250a5c5128e88b84bddc619ab7cba8d569b240efe4', - 'd8ac222636e5e3d6d4dba9dda6c9c426f788271bab0d6840dca87d3aa6ac62d6' - ], - [ - '5cbdf0646e5db4eaa398f365f2ea7a0e3d419b7e0330e39ce92bddedcac4f9bc', - '6aebca40ba255960a3178d6d861a54dba813d0b813fde7b5a5082628087264da' - ], - [ - 'acd484e2f0c7f65309ad178a9f559abde09796974c57e714c35f110dfc27ccbe', - 'cc338921b0a7d9fd64380971763b61e9add888a4375f8e0f05cc262ac64f9c37' - ], - [ - '774ae7f858a9411e5ef4246b70c65aac5649980be5c17891bbec17895da008cb', - 'd984a032eb6b5e190243dd56d7b7b365372db1e2dff9d6a8301d74c9c953c61b' - ], - [ - 'f28773c2d975288bc7d1d205c3748651b075fbc6610e58cddeeddf8f19405aa8', - 'ab0902e8d880a89758212eb65cdaf473a1a06da521fa91f29b5cb52db03ed81' - ], - [ - 'd7924d4f7d43ea965a465ae3095ff41131e5946f3c85f79e44adbcf8e27e080e', - '581e2872a86c72a683842ec228cc6defea40af2bd896d3a5c504dc9ff6a26b58' - ], - [ - 'defdea4cdb677750a420fee807eacf21eb9898ae79b9768766e4faa04a2d4a34', - '4211ab0694635168e997b0ead2a93daeced1f4a04a95c0f6cfb199f69e56eb77' - ], - [ - '2b4ea0a797a443d293ef5cff444f4979f06acfebd7e86d277475656138385b6c', - '85e89bc037945d93b343083b5a1c86131a01f60c50269763b570c854e5c09b7a' - ], - [ - '352bbf4a4cdd12564f93fa332ce333301d9ad40271f8107181340aef25be59d5', - '321eb4075348f534d59c18259dda3e1f4a1b3b2e71b1039c67bd3d8bcf81998c' - ], - [ - '2fa2104d6b38d11b0230010559879124e42ab8dfeff5ff29dc9cdadd4ecacc3f', - '2de1068295dd865b64569335bd5dd80181d70ecfc882648423ba76b532b7d67' - ], - [ - '9248279b09b4d68dab21a9b066edda83263c3d84e09572e269ca0cd7f5453714', - '73016f7bf234aade5d1aa71bdea2b1ff3fc0de2a887912ffe54a32ce97cb3402' - ], - [ - 'daed4f2be3a8bf278e70132fb0beb7522f570e144bf615c07e996d443dee8729', - 'a69dce4a7d6c98e8d4a1aca87ef8d7003f83c230f3afa726ab40e52290be1c55' - ], - [ - 'c44d12c7065d812e8acf28d7cbb19f9011ecd9e9fdf281b0e6a3b5e87d22e7db', - '2119a460ce326cdc76c45926c982fdac0e106e861edf61c5a039063f0e0e6482' - ], - [ - '6a245bf6dc698504c89a20cfded60853152b695336c28063b61c65cbd269e6b4', - 'e022cf42c2bd4a708b3f5126f16a24ad8b33ba48d0423b6efd5e6348100d8a82' - ], - [ - '1697ffa6fd9de627c077e3d2fe541084ce13300b0bec1146f95ae57f0d0bd6a5', - 'b9c398f186806f5d27561506e4557433a2cf15009e498ae7adee9d63d01b2396' - ], - [ - '605bdb019981718b986d0f07e834cb0d9deb8360ffb7f61df982345ef27a7479', - '2972d2de4f8d20681a78d93ec96fe23c26bfae84fb14db43b01e1e9056b8c49' - ], - [ - '62d14dab4150bf497402fdc45a215e10dcb01c354959b10cfe31c7e9d87ff33d', - '80fc06bd8cc5b01098088a1950eed0db01aa132967ab472235f5642483b25eaf' - ], - [ - '80c60ad0040f27dade5b4b06c408e56b2c50e9f56b9b8b425e555c2f86308b6f', - '1c38303f1cc5c30f26e66bad7fe72f70a65eed4cbe7024eb1aa01f56430bd57a' - ], - [ - '7a9375ad6167ad54aa74c6348cc54d344cc5dc9487d847049d5eabb0fa03c8fb', - 'd0e3fa9eca8726909559e0d79269046bdc59ea10c70ce2b02d499ec224dc7f7' - ], - [ - 'd528ecd9b696b54c907a9ed045447a79bb408ec39b68df504bb51f459bc3ffc9', - 'eecf41253136e5f99966f21881fd656ebc4345405c520dbc063465b521409933' - ], - [ - '49370a4b5f43412ea25f514e8ecdad05266115e4a7ecb1387231808f8b45963', - '758f3f41afd6ed428b3081b0512fd62a54c3f3afbb5b6764b653052a12949c9a' - ], - [ - '77f230936ee88cbbd73df930d64702ef881d811e0e1498e2f1c13eb1fc345d74', - '958ef42a7886b6400a08266e9ba1b37896c95330d97077cbbe8eb3c7671c60d6' - ], - [ - 'f2dac991cc4ce4b9ea44887e5c7c0bce58c80074ab9d4dbaeb28531b7739f530', - 'e0dedc9b3b2f8dad4da1f32dec2531df9eb5fbeb0598e4fd1a117dba703a3c37' - ], - [ - '463b3d9f662621fb1b4be8fbbe2520125a216cdfc9dae3debcba4850c690d45b', - '5ed430d78c296c3543114306dd8622d7c622e27c970a1de31cb377b01af7307e' - ], - [ - 'f16f804244e46e2a09232d4aff3b59976b98fac14328a2d1a32496b49998f247', - 'cedabd9b82203f7e13d206fcdf4e33d92a6c53c26e5cce26d6579962c4e31df6' - ], - [ - 'caf754272dc84563b0352b7a14311af55d245315ace27c65369e15f7151d41d1', - 'cb474660ef35f5f2a41b643fa5e460575f4fa9b7962232a5c32f908318a04476' - ], - [ - '2600ca4b282cb986f85d0f1709979d8b44a09c07cb86d7c124497bc86f082120', - '4119b88753c15bd6a693b03fcddbb45d5ac6be74ab5f0ef44b0be9475a7e4b40' - ], - [ - '7635ca72d7e8432c338ec53cd12220bc01c48685e24f7dc8c602a7746998e435', - '91b649609489d613d1d5e590f78e6d74ecfc061d57048bad9e76f302c5b9c61' - ], - [ - '754e3239f325570cdbbf4a87deee8a66b7f2b33479d468fbc1a50743bf56cc18', - '673fb86e5bda30fb3cd0ed304ea49a023ee33d0197a695d0c5d98093c536683' - ], - [ - 'e3e6bd1071a1e96aff57859c82d570f0330800661d1c952f9fe2694691d9b9e8', - '59c9e0bba394e76f40c0aa58379a3cb6a5a2283993e90c4167002af4920e37f5' - ], - [ - '186b483d056a033826ae73d88f732985c4ccb1f32ba35f4b4cc47fdcf04aa6eb', - '3b952d32c67cf77e2e17446e204180ab21fb8090895138b4a4a797f86e80888b' - ], - [ - 'df9d70a6b9876ce544c98561f4be4f725442e6d2b737d9c91a8321724ce0963f', - '55eb2dafd84d6ccd5f862b785dc39d4ab157222720ef9da217b8c45cf2ba2417' - ], - [ - '5edd5cc23c51e87a497ca815d5dce0f8ab52554f849ed8995de64c5f34ce7143', - 'efae9c8dbc14130661e8cec030c89ad0c13c66c0d17a2905cdc706ab7399a868' - ], - [ - '290798c2b6476830da12fe02287e9e777aa3fba1c355b17a722d362f84614fba', - 'e38da76dcd440621988d00bcf79af25d5b29c094db2a23146d003afd41943e7a' - ], - [ - 'af3c423a95d9f5b3054754efa150ac39cd29552fe360257362dfdecef4053b45', - 'f98a3fd831eb2b749a93b0e6f35cfb40c8cd5aa667a15581bc2feded498fd9c6' - ], - [ - '766dbb24d134e745cccaa28c99bf274906bb66b26dcf98df8d2fed50d884249a', - '744b1152eacbe5e38dcc887980da38b897584a65fa06cedd2c924f97cbac5996' - ], - [ - '59dbf46f8c94759ba21277c33784f41645f7b44f6c596a58ce92e666191abe3e', - 'c534ad44175fbc300f4ea6ce648309a042ce739a7919798cd85e216c4a307f6e' - ], - [ - 'f13ada95103c4537305e691e74e9a4a8dd647e711a95e73cb62dc6018cfd87b8', - 'e13817b44ee14de663bf4bc808341f326949e21a6a75c2570778419bdaf5733d' - ], - [ - '7754b4fa0e8aced06d4167a2c59cca4cda1869c06ebadfb6488550015a88522c', - '30e93e864e669d82224b967c3020b8fa8d1e4e350b6cbcc537a48b57841163a2' - ], - [ - '948dcadf5990e048aa3874d46abef9d701858f95de8041d2a6828c99e2262519', - 'e491a42537f6e597d5d28a3224b1bc25df9154efbd2ef1d2cbba2cae5347d57e' - ], - [ - '7962414450c76c1689c7b48f8202ec37fb224cf5ac0bfa1570328a8a3d7c77ab', - '100b610ec4ffb4760d5c1fc133ef6f6b12507a051f04ac5760afa5b29db83437' - ], - [ - '3514087834964b54b15b160644d915485a16977225b8847bb0dd085137ec47ca', - 'ef0afbb2056205448e1652c48e8127fc6039e77c15c2378b7e7d15a0de293311' - ], - [ - 'd3cc30ad6b483e4bc79ce2c9dd8bc54993e947eb8df787b442943d3f7b527eaf', - '8b378a22d827278d89c5e9be8f9508ae3c2ad46290358630afb34db04eede0a4' - ], - [ - '1624d84780732860ce1c78fcbfefe08b2b29823db913f6493975ba0ff4847610', - '68651cf9b6da903e0914448c6cd9d4ca896878f5282be4c8cc06e2a404078575' - ], - [ - '733ce80da955a8a26902c95633e62a985192474b5af207da6df7b4fd5fc61cd4', - 'f5435a2bd2badf7d485a4d8b8db9fcce3e1ef8e0201e4578c54673bc1dc5ea1d' - ], - [ - '15d9441254945064cf1a1c33bbd3b49f8966c5092171e699ef258dfab81c045c', - 'd56eb30b69463e7234f5137b73b84177434800bacebfc685fc37bbe9efe4070d' - ], - [ - 'a1d0fcf2ec9de675b612136e5ce70d271c21417c9d2b8aaaac138599d0717940', - 'edd77f50bcb5a3cab2e90737309667f2641462a54070f3d519212d39c197a629' - ], - [ - 'e22fbe15c0af8ccc5780c0735f84dbe9a790badee8245c06c7ca37331cb36980', - 'a855babad5cd60c88b430a69f53a1a7a38289154964799be43d06d77d31da06' - ], - [ - '311091dd9860e8e20ee13473c1155f5f69635e394704eaa74009452246cfa9b3', - '66db656f87d1f04fffd1f04788c06830871ec5a64feee685bd80f0b1286d8374' - ], - [ - '34c1fd04d301be89b31c0442d3e6ac24883928b45a9340781867d4232ec2dbdf', - '9414685e97b1b5954bd46f730174136d57f1ceeb487443dc5321857ba73abee' - ], - [ - 'f219ea5d6b54701c1c14de5b557eb42a8d13f3abbcd08affcc2a5e6b049b8d63', - '4cb95957e83d40b0f73af4544cccf6b1f4b08d3c07b27fb8d8c2962a400766d1' - ], - [ - 'd7b8740f74a8fbaab1f683db8f45de26543a5490bca627087236912469a0b448', - 'fa77968128d9c92ee1010f337ad4717eff15db5ed3c049b3411e0315eaa4593b' - ], - [ - '32d31c222f8f6f0ef86f7c98d3a3335ead5bcd32abdd94289fe4d3091aa824bf', - '5f3032f5892156e39ccd3d7915b9e1da2e6dac9e6f26e961118d14b8462e1661' - ], - [ - '7461f371914ab32671045a155d9831ea8793d77cd59592c4340f86cbc18347b5', - '8ec0ba238b96bec0cbdddcae0aa442542eee1ff50c986ea6b39847b3cc092ff6' - ], - [ - 'ee079adb1df1860074356a25aa38206a6d716b2c3e67453d287698bad7b2b2d6', - '8dc2412aafe3be5c4c5f37e0ecc5f9f6a446989af04c4e25ebaac479ec1c8c1e' - ], - [ - '16ec93e447ec83f0467b18302ee620f7e65de331874c9dc72bfd8616ba9da6b5', - '5e4631150e62fb40d0e8c2a7ca5804a39d58186a50e497139626778e25b0674d' - ], - [ - 'eaa5f980c245f6f038978290afa70b6bd8855897f98b6aa485b96065d537bd99', - 'f65f5d3e292c2e0819a528391c994624d784869d7e6ea67fb18041024edc07dc' - ], - [ - '78c9407544ac132692ee1910a02439958ae04877151342ea96c4b6b35a49f51', - 'f3e0319169eb9b85d5404795539a5e68fa1fbd583c064d2462b675f194a3ddb4' - ], - [ - '494f4be219a1a77016dcd838431aea0001cdc8ae7a6fc688726578d9702857a5', - '42242a969283a5f339ba7f075e36ba2af925ce30d767ed6e55f4b031880d562c' - ], - [ - 'a598a8030da6d86c6bc7f2f5144ea549d28211ea58faa70ebf4c1e665c1fe9b5', - '204b5d6f84822c307e4b4a7140737aec23fc63b65b35f86a10026dbd2d864e6b' - ], - [ - 'c41916365abb2b5d09192f5f2dbeafec208f020f12570a184dbadc3e58595997', - '4f14351d0087efa49d245b328984989d5caf9450f34bfc0ed16e96b58fa9913' - ], - [ - '841d6063a586fa475a724604da03bc5b92a2e0d2e0a36acfe4c73a5514742881', - '73867f59c0659e81904f9a1c7543698e62562d6744c169ce7a36de01a8d6154' - ], - [ - '5e95bb399a6971d376026947f89bde2f282b33810928be4ded112ac4d70e20d5', - '39f23f366809085beebfc71181313775a99c9aed7d8ba38b161384c746012865' - ], - [ - '36e4641a53948fd476c39f8a99fd974e5ec07564b5315d8bf99471bca0ef2f66', - 'd2424b1b1abe4eb8164227b085c9aa9456ea13493fd563e06fd51cf5694c78fc' - ], - [ - '336581ea7bfbbb290c191a2f507a41cf5643842170e914faeab27c2c579f726', - 'ead12168595fe1be99252129b6e56b3391f7ab1410cd1e0ef3dcdcabd2fda224' - ], - [ - '8ab89816dadfd6b6a1f2634fcf00ec8403781025ed6890c4849742706bd43ede', - '6fdcef09f2f6d0a044e654aef624136f503d459c3e89845858a47a9129cdd24e' - ], - [ - '1e33f1a746c9c5778133344d9299fcaa20b0938e8acff2544bb40284b8c5fb94', - '60660257dd11b3aa9c8ed618d24edff2306d320f1d03010e33a7d2057f3b3b6' - ], - [ - '85b7c1dcb3cec1b7ee7f30ded79dd20a0ed1f4cc18cbcfcfa410361fd8f08f31', - '3d98a9cdd026dd43f39048f25a8847f4fcafad1895d7a633c6fed3c35e999511' - ], - [ - '29df9fbd8d9e46509275f4b125d6d45d7fbe9a3b878a7af872a2800661ac5f51', - 'b4c4fe99c775a606e2d8862179139ffda61dc861c019e55cd2876eb2a27d84b' - ], - [ - 'a0b1cae06b0a847a3fea6e671aaf8adfdfe58ca2f768105c8082b2e449fce252', - 'ae434102edde0958ec4b19d917a6a28e6b72da1834aff0e650f049503a296cf2' - ], - [ - '4e8ceafb9b3e9a136dc7ff67e840295b499dfb3b2133e4ba113f2e4c0e121e5', - 'cf2174118c8b6d7a4b48f6d534ce5c79422c086a63460502b827ce62a326683c' - ], - [ - 'd24a44e047e19b6f5afb81c7ca2f69080a5076689a010919f42725c2b789a33b', - '6fb8d5591b466f8fc63db50f1c0f1c69013f996887b8244d2cdec417afea8fa3' - ], - [ - 'ea01606a7a6c9cdd249fdfcfacb99584001edd28abbab77b5104e98e8e3b35d4', - '322af4908c7312b0cfbfe369f7a7b3cdb7d4494bc2823700cfd652188a3ea98d' - ], - [ - 'af8addbf2b661c8a6c6328655eb96651252007d8c5ea31be4ad196de8ce2131f', - '6749e67c029b85f52a034eafd096836b2520818680e26ac8f3dfbcdb71749700' - ], - [ - 'e3ae1974566ca06cc516d47e0fb165a674a3dabcfca15e722f0e3450f45889', - '2aeabe7e4531510116217f07bf4d07300de97e4874f81f533420a72eeb0bd6a4' - ], - [ - '591ee355313d99721cf6993ffed1e3e301993ff3ed258802075ea8ced397e246', - 'b0ea558a113c30bea60fc4775460c7901ff0b053d25ca2bdeee98f1a4be5d196' - ], - [ - '11396d55fda54c49f19aa97318d8da61fa8584e47b084945077cf03255b52984', - '998c74a8cd45ac01289d5833a7beb4744ff536b01b257be4c5767bea93ea57a4' - ], - [ - '3c5d2a1ba39c5a1790000738c9e0c40b8dcdfd5468754b6405540157e017aa7a', - 'b2284279995a34e2f9d4de7396fc18b80f9b8b9fdd270f6661f79ca4c81bd257' - ], - [ - 'cc8704b8a60a0defa3a99a7299f2e9c3fbc395afb04ac078425ef8a1793cc030', - 'bdd46039feed17881d1e0862db347f8cf395b74fc4bcdc4e940b74e3ac1f1b13' - ], - [ - 'c533e4f7ea8555aacd9777ac5cad29b97dd4defccc53ee7ea204119b2889b197', - '6f0a256bc5efdf429a2fb6242f1a43a2d9b925bb4a4b3a26bb8e0f45eb596096' - ], - [ - 'c14f8f2ccb27d6f109f6d08d03cc96a69ba8c34eec07bbcf566d48e33da6593', - 'c359d6923bb398f7fd4473e16fe1c28475b740dd098075e6c0e8649113dc3a38' - ], - [ - 'a6cbc3046bc6a450bac24789fa17115a4c9739ed75f8f21ce441f72e0b90e6ef', - '21ae7f4680e889bb130619e2c0f95a360ceb573c70603139862afd617fa9b9f' - ], - [ - '347d6d9a02c48927ebfb86c1359b1caf130a3c0267d11ce6344b39f99d43cc38', - '60ea7f61a353524d1c987f6ecec92f086d565ab687870cb12689ff1e31c74448' - ], - [ - 'da6545d2181db8d983f7dcb375ef5866d47c67b1bf31c8cf855ef7437b72656a', - '49b96715ab6878a79e78f07ce5680c5d6673051b4935bd897fea824b77dc208a' - ], - [ - 'c40747cc9d012cb1a13b8148309c6de7ec25d6945d657146b9d5994b8feb1111', - '5ca560753be2a12fc6de6caf2cb489565db936156b9514e1bb5e83037e0fa2d4' - ], - [ - '4e42c8ec82c99798ccf3a610be870e78338c7f713348bd34c8203ef4037f3502', - '7571d74ee5e0fb92a7a8b33a07783341a5492144cc54bcc40a94473693606437' - ], - [ - '3775ab7089bc6af823aba2e1af70b236d251cadb0c86743287522a1b3b0dedea', - 'be52d107bcfa09d8bcb9736a828cfa7fac8db17bf7a76a2c42ad961409018cf7' - ], - [ - 'cee31cbf7e34ec379d94fb814d3d775ad954595d1314ba8846959e3e82f74e26', - '8fd64a14c06b589c26b947ae2bcf6bfa0149ef0be14ed4d80f448a01c43b1c6d' - ], - [ - 'b4f9eaea09b6917619f6ea6a4eb5464efddb58fd45b1ebefcdc1a01d08b47986', - '39e5c9925b5a54b07433a4f18c61726f8bb131c012ca542eb24a8ac07200682a' - ], - [ - 'd4263dfc3d2df923a0179a48966d30ce84e2515afc3dccc1b77907792ebcc60e', - '62dfaf07a0f78feb30e30d6295853ce189e127760ad6cf7fae164e122a208d54' - ], - [ - '48457524820fa65a4f8d35eb6930857c0032acc0a4a2de422233eeda897612c4', - '25a748ab367979d98733c38a1fa1c2e7dc6cc07db2d60a9ae7a76aaa49bd0f77' - ], - [ - 'dfeeef1881101f2cb11644f3a2afdfc2045e19919152923f367a1767c11cceda', - 'ecfb7056cf1de042f9420bab396793c0c390bde74b4bbdff16a83ae09a9a7517' - ], - [ - '6d7ef6b17543f8373c573f44e1f389835d89bcbc6062ced36c82df83b8fae859', - 'cd450ec335438986dfefa10c57fea9bcc521a0959b2d80bbf74b190dca712d10' - ], - [ - 'e75605d59102a5a2684500d3b991f2e3f3c88b93225547035af25af66e04541f', - 'f5c54754a8f71ee540b9b48728473e314f729ac5308b06938360990e2bfad125' - ], - [ - 'eb98660f4c4dfaa06a2be453d5020bc99a0c2e60abe388457dd43fefb1ed620c', - '6cb9a8876d9cb8520609af3add26cd20a0a7cd8a9411131ce85f44100099223e' - ], - [ - '13e87b027d8514d35939f2e6892b19922154596941888336dc3563e3b8dba942', - 'fef5a3c68059a6dec5d624114bf1e91aac2b9da568d6abeb2570d55646b8adf1' - ], - [ - 'ee163026e9fd6fe017c38f06a5be6fc125424b371ce2708e7bf4491691e5764a', - '1acb250f255dd61c43d94ccc670d0f58f49ae3fa15b96623e5430da0ad6c62b2' - ], - [ - 'b268f5ef9ad51e4d78de3a750c2dc89b1e626d43505867999932e5db33af3d80', - '5f310d4b3c99b9ebb19f77d41c1dee018cf0d34fd4191614003e945a1216e423' - ], - [ - 'ff07f3118a9df035e9fad85eb6c7bfe42b02f01ca99ceea3bf7ffdba93c4750d', - '438136d603e858a3a5c440c38eccbaddc1d2942114e2eddd4740d098ced1f0d8' - ], - [ - '8d8b9855c7c052a34146fd20ffb658bea4b9f69e0d825ebec16e8c3ce2b526a1', - 'cdb559eedc2d79f926baf44fb84ea4d44bcf50fee51d7ceb30e2e7f463036758' - ], - [ - '52db0b5384dfbf05bfa9d472d7ae26dfe4b851ceca91b1eba54263180da32b63', - 'c3b997d050ee5d423ebaf66a6db9f57b3180c902875679de924b69d84a7b375' - ], - [ - 'e62f9490d3d51da6395efd24e80919cc7d0f29c3f3fa48c6fff543becbd43352', - '6d89ad7ba4876b0b22c2ca280c682862f342c8591f1daf5170e07bfd9ccafa7d' - ], - [ - '7f30ea2476b399b4957509c88f77d0191afa2ff5cb7b14fd6d8e7d65aaab1193', - 'ca5ef7d4b231c94c3b15389a5f6311e9daff7bb67b103e9880ef4bff637acaec' - ], - [ - '5098ff1e1d9f14fb46a210fada6c903fef0fb7b4a1dd1d9ac60a0361800b7a00', - '9731141d81fc8f8084d37c6e7542006b3ee1b40d60dfe5362a5b132fd17ddc0' - ], - [ - '32b78c7de9ee512a72895be6b9cbefa6e2f3c4ccce445c96b9f2c81e2778ad58', - 'ee1849f513df71e32efc3896ee28260c73bb80547ae2275ba497237794c8753c' - ], - [ - 'e2cb74fddc8e9fbcd076eef2a7c72b0ce37d50f08269dfc074b581550547a4f7', - 'd3aa2ed71c9dd2247a62df062736eb0baddea9e36122d2be8641abcb005cc4a4' - ], - [ - '8438447566d4d7bedadc299496ab357426009a35f235cb141be0d99cd10ae3a8', - 'c4e1020916980a4da5d01ac5e6ad330734ef0d7906631c4f2390426b2edd791f' + self._prev = block; + var out = self._cipher.decryptBlock(block); + + return xor(out, pad); + }; + }, + { 'buffer-xor': 123 } ], - [ - '4162d488b89402039b584c6fc6c308870587d9c46f660b878ab65c82c711d67e', - '67163e903236289f776f22c25fb8a3afc1732f2b84b4e95dbda47ae5a0852649' + 86: [ + function (require, module, exports) { + var Buffer = require('safe-buffer').Buffer; + var xor = require('buffer-xor'); + + function encryptStart(self, data, decrypt) { + var len = data.length; + var out = xor(data, self._cache); + self._cache = self._cache.slice(len); + self._prev = Buffer.concat([self._prev, decrypt ? data : out]); + return out; + } + + exports.encrypt = function (self, data, decrypt) { + var out = Buffer.allocUnsafe(0); + var len; + + while (data.length) { + if (self._cache.length === 0) { + self._cache = self._cipher.encryptBlock(self._prev); + self._prev = Buffer.allocUnsafe(0); + } + + if (self._cache.length <= data.length) { + len = self._cache.length; + out = Buffer.concat([out, encryptStart(self, data.slice(0, len), decrypt)]); + data = data.slice(len); + } else { + out = Buffer.concat([out, encryptStart(self, data, decrypt)]); + break; + } + } + + return out; + }; + }, + { 'buffer-xor': 123, 'safe-buffer': 269 } ], - [ - '3fad3fa84caf0f34f0f89bfd2dcf54fc175d767aec3e50684f3ba4a4bf5f683d', - 'cd1bc7cb6cc407bb2f0ca647c718a730cf71872e7d0d2a53fa20efcdfe61826' + 87: [ + function (require, module, exports) { + var Buffer = require('safe-buffer').Buffer; + + function encryptByte(self, byteParam, decrypt) { + var pad; + var i = -1; + var len = 8; + var out = 0; + var bit, value; + while (++i < len) { + pad = self._cipher.encryptBlock(self._prev); + bit = byteParam & (1 << (7 - i)) ? 0x80 : 0; + value = pad[0] ^ bit; + out += (value & 0x80) >> i % 8; + self._prev = shiftIn(self._prev, decrypt ? bit : value); + } + return out; + } + + function shiftIn(buffer, value) { + var len = buffer.length; + var i = -1; + var out = Buffer.allocUnsafe(buffer.length); + buffer = Buffer.concat([buffer, Buffer.from([value])]); + + while (++i < len) { + out[i] = (buffer[i] << 1) | (buffer[i + 1] >> 7); + } + + return out; + } + + exports.encrypt = function (self, chunk, decrypt) { + var len = chunk.length; + var out = Buffer.allocUnsafe(len); + var i = -1; + + while (++i < len) { + out[i] = encryptByte(self, chunk[i], decrypt); + } + + return out; + }; + }, + { 'safe-buffer': 269 } ], - [ - '674f2600a3007a00568c1a7ce05d0816c1fb84bf1370798f1c69532faeb1a86b', - '299d21f9413f33b3edf43b257004580b70db57da0b182259e09eecc69e0d38a5' + 88: [ + function (require, module, exports) { + var Buffer = require('safe-buffer').Buffer; + + function encryptByte(self, byteParam, decrypt) { + var pad = self._cipher.encryptBlock(self._prev); + var out = pad[0] ^ byteParam; + + self._prev = Buffer.concat([self._prev.slice(1), Buffer.from([decrypt ? byteParam : out])]); + + return out; + } + + exports.encrypt = function (self, chunk, decrypt) { + var len = chunk.length; + var out = Buffer.allocUnsafe(len); + var i = -1; + + while (++i < len) { + out[i] = encryptByte(self, chunk[i], decrypt); + } + + return out; + }; + }, + { 'safe-buffer': 269 } ], - [ - 'd32f4da54ade74abb81b815ad1fb3b263d82d6c692714bcff87d29bd5ee9f08f', - 'f9429e738b8e53b968e99016c059707782e14f4535359d582fc416910b3eea87' + 89: [ + function (require, module, exports) { + var xor = require('buffer-xor'); + var Buffer = require('safe-buffer').Buffer; + var incr32 = require('../incr32'); + + function getBlock(self) { + var out = self._cipher.encryptBlockRaw(self._prev); + incr32(self._prev); + return out; + } + + var blockSize = 16; + exports.encrypt = function (self, chunk) { + var chunkNum = Math.ceil(chunk.length / blockSize); + var start = self._cache.length; + self._cache = Buffer.concat([self._cache, Buffer.allocUnsafe(chunkNum * blockSize)]); + for (var i = 0; i < chunkNum; i++) { + var out = getBlock(self); + var offset = start + i * blockSize; + self._cache.writeUInt32BE(out[0], offset + 0); + self._cache.writeUInt32BE(out[1], offset + 4); + self._cache.writeUInt32BE(out[2], offset + 8); + self._cache.writeUInt32BE(out[3], offset + 12); + } + var pad = self._cache.slice(0, chunk.length); + self._cache = self._cache.slice(chunk.length); + return xor(chunk, pad); + }; + }, + { '../incr32': 84, 'buffer-xor': 123, 'safe-buffer': 269 } ], - [ - '30e4e670435385556e593657135845d36fbb6931f72b08cb1ed954f1e3ce3ff6', - '462f9bce619898638499350113bbc9b10a878d35da70740dc695a559eb88db7b' + 90: [ + function (require, module, exports) { + exports.encrypt = function (self, block) { + return self._cipher.encryptBlock(block); + }; + + exports.decrypt = function (self, block) { + return self._cipher.decryptBlock(block); + }; + }, + {} ], - [ - 'be2062003c51cc3004682904330e4dee7f3dcd10b01e580bf1971b04d4cad297', - '62188bc49d61e5428573d48a74e1c655b1c61090905682a0d5558ed72dccb9bc' + 91: [ + function (require, module, exports) { + var modeModules = { + ECB: require('./ecb'), + CBC: require('./cbc'), + CFB: require('./cfb'), + CFB8: require('./cfb8'), + CFB1: require('./cfb1'), + OFB: require('./ofb'), + CTR: require('./ctr'), + GCM: require('./ctr') + }; + + var modes = require('./list.json'); + + for (var key in modes) { + modes[key].module = modeModules[modes[key].mode]; + } + + module.exports = modes; + }, + { './cbc': 85, './cfb': 86, './cfb1': 87, './cfb8': 88, './ctr': 89, './ecb': 90, './list.json': 92, './ofb': 93 } ], - [ - '93144423ace3451ed29e0fb9ac2af211cb6e84a601df5993c419859fff5df04a', - '7c10dfb164c3425f5c71a3f9d7992038f1065224f72bb9d1d902a6d13037b47c' + 92: [ + function (require, module, exports) { + module.exports = { + 'aes-128-ecb': { + cipher: 'AES', + key: 128, + iv: 0, + mode: 'ECB', + type: 'block' + }, + 'aes-192-ecb': { + cipher: 'AES', + key: 192, + iv: 0, + mode: 'ECB', + type: 'block' + }, + 'aes-256-ecb': { + cipher: 'AES', + key: 256, + iv: 0, + mode: 'ECB', + type: 'block' + }, + 'aes-128-cbc': { + cipher: 'AES', + key: 128, + iv: 16, + mode: 'CBC', + type: 'block' + }, + 'aes-192-cbc': { + cipher: 'AES', + key: 192, + iv: 16, + mode: 'CBC', + type: 'block' + }, + 'aes-256-cbc': { + cipher: 'AES', + key: 256, + iv: 16, + mode: 'CBC', + type: 'block' + }, + aes128: { + cipher: 'AES', + key: 128, + iv: 16, + mode: 'CBC', + type: 'block' + }, + aes192: { + cipher: 'AES', + key: 192, + iv: 16, + mode: 'CBC', + type: 'block' + }, + aes256: { + cipher: 'AES', + key: 256, + iv: 16, + mode: 'CBC', + type: 'block' + }, + 'aes-128-cfb': { + cipher: 'AES', + key: 128, + iv: 16, + mode: 'CFB', + type: 'stream' + }, + 'aes-192-cfb': { + cipher: 'AES', + key: 192, + iv: 16, + mode: 'CFB', + type: 'stream' + }, + 'aes-256-cfb': { + cipher: 'AES', + key: 256, + iv: 16, + mode: 'CFB', + type: 'stream' + }, + 'aes-128-cfb8': { + cipher: 'AES', + key: 128, + iv: 16, + mode: 'CFB8', + type: 'stream' + }, + 'aes-192-cfb8': { + cipher: 'AES', + key: 192, + iv: 16, + mode: 'CFB8', + type: 'stream' + }, + 'aes-256-cfb8': { + cipher: 'AES', + key: 256, + iv: 16, + mode: 'CFB8', + type: 'stream' + }, + 'aes-128-cfb1': { + cipher: 'AES', + key: 128, + iv: 16, + mode: 'CFB1', + type: 'stream' + }, + 'aes-192-cfb1': { + cipher: 'AES', + key: 192, + iv: 16, + mode: 'CFB1', + type: 'stream' + }, + 'aes-256-cfb1': { + cipher: 'AES', + key: 256, + iv: 16, + mode: 'CFB1', + type: 'stream' + }, + 'aes-128-ofb': { + cipher: 'AES', + key: 128, + iv: 16, + mode: 'OFB', + type: 'stream' + }, + 'aes-192-ofb': { + cipher: 'AES', + key: 192, + iv: 16, + mode: 'OFB', + type: 'stream' + }, + 'aes-256-ofb': { + cipher: 'AES', + key: 256, + iv: 16, + mode: 'OFB', + type: 'stream' + }, + 'aes-128-ctr': { + cipher: 'AES', + key: 128, + iv: 16, + mode: 'CTR', + type: 'stream' + }, + 'aes-192-ctr': { + cipher: 'AES', + key: 192, + iv: 16, + mode: 'CTR', + type: 'stream' + }, + 'aes-256-ctr': { + cipher: 'AES', + key: 256, + iv: 16, + mode: 'CTR', + type: 'stream' + }, + 'aes-128-gcm': { + cipher: 'AES', + key: 128, + iv: 12, + mode: 'GCM', + type: 'auth' + }, + 'aes-192-gcm': { + cipher: 'AES', + key: 192, + iv: 12, + mode: 'GCM', + type: 'auth' + }, + 'aes-256-gcm': { + cipher: 'AES', + key: 256, + iv: 12, + mode: 'GCM', + type: 'auth' + } + }; + }, + {} ], - [ - 'b015f8044f5fcbdcf21ca26d6c34fb8197829205c7b7d2a7cb66418c157b112c', - 'ab8c1e086d04e813744a655b2df8d5f83b3cdc6faa3088c1d3aea1454e3a1d5f' + 93: [ + function (require, module, exports) { + (function (Buffer) { + var xor = require('buffer-xor'); + + function getBlock(self) { + self._prev = self._cipher.encryptBlock(self._prev); + return self._prev; + } + + exports.encrypt = function (self, chunk) { + while (self._cache.length < chunk.length) { + self._cache = Buffer.concat([self._cache, getBlock(self)]); + } + + var pad = self._cache.slice(0, chunk.length); + self._cache = self._cache.slice(chunk.length); + return xor(chunk, pad); + }; + }).call(this, require('buffer').Buffer); + }, + { buffer: 124, 'buffer-xor': 123 } ], - [ - 'd5e9e1da649d97d89e4868117a465a3a4f8a18de57a140d36b3f2af341a21b52', - '4cb04437f391ed73111a13cc1d4dd0db1693465c2240480d8955e8592f27447a' + 94: [ + function (require, module, exports) { + var aes = require('./aes'); + var Buffer = require('safe-buffer').Buffer; + var Transform = require('cipher-base'); + var inherits = require('inherits'); + + function StreamCipher(mode, key, iv, decrypt) { + Transform.call(this); + + this._cipher = new aes.AES(key); + this._prev = Buffer.from(iv); + this._cache = Buffer.allocUnsafe(0); + this._secCache = Buffer.allocUnsafe(0); + this._decrypt = decrypt; + this._mode = mode; + } + + inherits(StreamCipher, Transform); + + StreamCipher.prototype._update = function (chunk) { + return this._mode.encrypt(this, chunk, this._decrypt); + }; + + StreamCipher.prototype._final = function () { + this._cipher.scrub(); + }; + + module.exports = StreamCipher; + }, + { './aes': 78, 'cipher-base': 125, inherits: 218, 'safe-buffer': 269 } ], - [ - 'd3ae41047dd7ca065dbf8ed77b992439983005cd72e16d6f996a5316d36966bb', - 'bd1aeb21ad22ebb22a10f0303417c6d964f8cdd7df0aca614b10dc14d125ac46' + 95: [ + function (require, module, exports) { + var DES = require('browserify-des'); + var aes = require('browserify-aes/browser'); + var aesModes = require('browserify-aes/modes'); + var desModes = require('browserify-des/modes'); + var ebtk = require('evp_bytestokey'); + + function createCipher(suite, password) { + suite = suite.toLowerCase(); + + var keyLen, ivLen; + if (aesModes[suite]) { + keyLen = aesModes[suite].key; + ivLen = aesModes[suite].iv; + } else if (desModes[suite]) { + keyLen = desModes[suite].key * 8; + ivLen = desModes[suite].iv; + } else { + throw new TypeError('invalid suite type'); + } + + var keys = ebtk(password, false, keyLen, ivLen); + return createCipheriv(suite, keys.key, keys.iv); + } + + function createDecipher(suite, password) { + suite = suite.toLowerCase(); + + var keyLen, ivLen; + if (aesModes[suite]) { + keyLen = aesModes[suite].key; + ivLen = aesModes[suite].iv; + } else if (desModes[suite]) { + keyLen = desModes[suite].key * 8; + ivLen = desModes[suite].iv; + } else { + throw new TypeError('invalid suite type'); + } + + var keys = ebtk(password, false, keyLen, ivLen); + return createDecipheriv(suite, keys.key, keys.iv); + } + + function createCipheriv(suite, key, iv) { + suite = suite.toLowerCase(); + if (aesModes[suite]) return aes.createCipheriv(suite, key, iv); + if (desModes[suite]) return new DES({ key: key, iv: iv, mode: suite }); + + throw new TypeError('invalid suite type'); + } + + function createDecipheriv(suite, key, iv) { + suite = suite.toLowerCase(); + if (aesModes[suite]) return aes.createDecipheriv(suite, key, iv); + if (desModes[suite]) return new DES({ key: key, iv: iv, mode: suite, decrypt: true }); + + throw new TypeError('invalid suite type'); + } + + function getCiphers() { + return Object.keys(desModes).concat(aes.getCiphers()); + } + + exports.createCipher = exports.Cipher = createCipher; + exports.createCipheriv = exports.Cipheriv = createCipheriv; + exports.createDecipher = exports.Decipher = createDecipher; + exports.createDecipheriv = exports.Decipheriv = createDecipheriv; + exports.listCiphers = exports.getCiphers = getCiphers; + }, + { 'browserify-aes/browser': 80, 'browserify-aes/modes': 91, 'browserify-des': 96, 'browserify-des/modes': 97, evp_bytestokey: 182 } ], - [ - '463e2763d885f958fc66cdd22800f0a487197d0a82e377b49f80af87c897b065', - 'bfefacdb0e5d0fd7df3a311a94de062b26b80c61fbc97508b79992671ef7ca7f' + 96: [ + function (require, module, exports) { + var CipherBase = require('cipher-base'); + var des = require('des.js'); + var inherits = require('inherits'); + var Buffer = require('safe-buffer').Buffer; + + var modes = { + 'des-ede3-cbc': des.CBC.instantiate(des.EDE), + 'des-ede3': des.EDE, + 'des-ede-cbc': des.CBC.instantiate(des.EDE), + 'des-ede': des.EDE, + 'des-cbc': des.CBC.instantiate(des.DES), + 'des-ecb': des.DES + }; + modes.des = modes['des-cbc']; + modes.des3 = modes['des-ede3-cbc']; + module.exports = DES; + inherits(DES, CipherBase); + function DES(opts) { + CipherBase.call(this); + var modeName = opts.mode.toLowerCase(); + var mode = modes[modeName]; + var type; + if (opts.decrypt) { + type = 'decrypt'; + } else { + type = 'encrypt'; + } + var key = opts.key; + if (!Buffer.isBuffer(key)) { + key = Buffer.from(key); + } + if (modeName === 'des-ede' || modeName === 'des-ede-cbc') { + key = Buffer.concat([key, key.slice(0, 8)]); + } + var iv = opts.iv; + if (!Buffer.isBuffer(iv)) { + iv = Buffer.from(iv); + } + this._des = mode.create({ + key: key, + iv: iv, + type: type + }); + } + DES.prototype._update = function (data) { + return Buffer.from(this._des.update(data)); + }; + DES.prototype._final = function () { + return Buffer.from(this._des.final()); + }; + }, + { 'cipher-base': 125, 'des.js': 137, inherits: 218, 'safe-buffer': 269 } ], - [ - '7985fdfd127c0567c6f53ec1bb63ec3158e597c40bfe747c83cddfc910641917', - '603c12daf3d9862ef2b25fe1de289aed24ed291e0ec6708703a5bd567f32ed03' + 97: [ + function (require, module, exports) { + exports['des-ecb'] = { + key: 8, + iv: 0 + }; + exports['des-cbc'] = exports.des = { + key: 8, + iv: 8 + }; + exports['des-ede3-cbc'] = exports.des3 = { + key: 24, + iv: 8 + }; + exports['des-ede3'] = { + key: 24, + iv: 0 + }; + exports['des-ede-cbc'] = { + key: 16, + iv: 8 + }; + exports['des-ede'] = { + key: 16, + iv: 0 + }; + }, + {} ], - [ - '74a1ad6b5f76e39db2dd249410eac7f99e74c59cb83d2d0ed5ff1543da7703e9', - 'cc6157ef18c9c63cd6193d83631bbea0093e0968942e8c33d5737fd790e0db08' + 98: [ + function (require, module, exports) { + (function (Buffer) { + var bn = require('bn.js'); + var randomBytes = require('randombytes'); + module.exports = crt; + function blind(priv) { + var r = getr(priv); + var blinder = r.toRed(bn.mont(priv.modulus)).redPow(new bn(priv.publicExponent)).fromRed(); + return { + blinder: blinder, + unblinder: r.invm(priv.modulus) + }; + } + function crt(msg, priv) { + var blinds = blind(priv); + var len = priv.modulus.byteLength(); + var mod = bn.mont(priv.modulus); + var blinded = new bn(msg).mul(blinds.blinder).umod(priv.modulus); + var c1 = blinded.toRed(bn.mont(priv.prime1)); + var c2 = blinded.toRed(bn.mont(priv.prime2)); + var qinv = priv.coefficient; + var p = priv.prime1; + var q = priv.prime2; + var m1 = c1.redPow(priv.exponent1); + var m2 = c2.redPow(priv.exponent2); + m1 = m1.fromRed(); + m2 = m2.fromRed(); + var h = m1.isub(m2).imul(qinv).umod(p); + h.imul(q); + m2.iadd(h); + return new Buffer(m2.imul(blinds.unblinder).umod(priv.modulus).toArray(false, len)); + } + crt.getr = getr; + function getr(priv) { + var len = priv.modulus.byteLength(); + var r = new bn(randomBytes(len)); + while (r.cmp(priv.modulus) >= 0 || !r.umod(priv.prime1) || !r.umod(priv.prime2)) { + r = new bn(randomBytes(len)); + } + return r; + } + }).call(this, require('buffer').Buffer); + }, + { 'bn.js': 99, buffer: 124, randombytes: 252 } ], - [ - '30682a50703375f602d416664ba19b7fc9bab42c72747463a71d0896b22f6da3', - '553e04f6b018b4fa6c8f39e7f311d3176290d0e0f19ca73f17714d9977a22ff8' + 99: [ + function (require, module, exports) { + arguments[4][70][0].apply(exports, arguments); + }, + { buffer: 77, dup: 70 } ], - [ - '9e2158f0d7c0d5f26c3791efefa79597654e7a2b2464f52b1ee6c1347769ef57', - '712fcdd1b9053f09003a3481fa7762e9ffd7c8ef35a38509e2fbf2629008373' + 100: [ + function (require, module, exports) { + module.exports = require('./browser/algorithms.json'); + }, + { './browser/algorithms.json': 101 } ], - [ - '176e26989a43c9cfeba4029c202538c28172e566e3c4fce7322857f3be327d66', - 'ed8cc9d04b29eb877d270b4878dc43c19aefd31f4eee09ee7b47834c1fa4b1c3' + 101: [ + function (require, module, exports) { + module.exports = { + sha224WithRSAEncryption: { + sign: 'rsa', + hash: 'sha224', + id: '302d300d06096086480165030402040500041c' + }, + 'RSA-SHA224': { + sign: 'ecdsa/rsa', + hash: 'sha224', + id: '302d300d06096086480165030402040500041c' + }, + sha256WithRSAEncryption: { + sign: 'rsa', + hash: 'sha256', + id: '3031300d060960864801650304020105000420' + }, + 'RSA-SHA256': { + sign: 'ecdsa/rsa', + hash: 'sha256', + id: '3031300d060960864801650304020105000420' + }, + sha384WithRSAEncryption: { + sign: 'rsa', + hash: 'sha384', + id: '3041300d060960864801650304020205000430' + }, + 'RSA-SHA384': { + sign: 'ecdsa/rsa', + hash: 'sha384', + id: '3041300d060960864801650304020205000430' + }, + sha512WithRSAEncryption: { + sign: 'rsa', + hash: 'sha512', + id: '3051300d060960864801650304020305000440' + }, + 'RSA-SHA512': { + sign: 'ecdsa/rsa', + hash: 'sha512', + id: '3051300d060960864801650304020305000440' + }, + 'RSA-SHA1': { + sign: 'rsa', + hash: 'sha1', + id: '3021300906052b0e03021a05000414' + }, + 'ecdsa-with-SHA1': { + sign: 'ecdsa', + hash: 'sha1', + id: '' + }, + sha256: { + sign: 'ecdsa', + hash: 'sha256', + id: '' + }, + sha224: { + sign: 'ecdsa', + hash: 'sha224', + id: '' + }, + sha384: { + sign: 'ecdsa', + hash: 'sha384', + id: '' + }, + sha512: { + sign: 'ecdsa', + hash: 'sha512', + id: '' + }, + 'DSA-SHA': { + sign: 'dsa', + hash: 'sha1', + id: '' + }, + 'DSA-SHA1': { + sign: 'dsa', + hash: 'sha1', + id: '' + }, + DSA: { + sign: 'dsa', + hash: 'sha1', + id: '' + }, + 'DSA-WITH-SHA224': { + sign: 'dsa', + hash: 'sha224', + id: '' + }, + 'DSA-SHA224': { + sign: 'dsa', + hash: 'sha224', + id: '' + }, + 'DSA-WITH-SHA256': { + sign: 'dsa', + hash: 'sha256', + id: '' + }, + 'DSA-SHA256': { + sign: 'dsa', + hash: 'sha256', + id: '' + }, + 'DSA-WITH-SHA384': { + sign: 'dsa', + hash: 'sha384', + id: '' + }, + 'DSA-SHA384': { + sign: 'dsa', + hash: 'sha384', + id: '' + }, + 'DSA-WITH-SHA512': { + sign: 'dsa', + hash: 'sha512', + id: '' + }, + 'DSA-SHA512': { + sign: 'dsa', + hash: 'sha512', + id: '' + }, + 'DSA-RIPEMD160': { + sign: 'dsa', + hash: 'rmd160', + id: '' + }, + ripemd160WithRSA: { + sign: 'rsa', + hash: 'rmd160', + id: '3021300906052b2403020105000414' + }, + 'RSA-RIPEMD160': { + sign: 'rsa', + hash: 'rmd160', + id: '3021300906052b2403020105000414' + }, + md5WithRSAEncryption: { + sign: 'rsa', + hash: 'md5', + id: '3020300c06082a864886f70d020505000410' + }, + 'RSA-MD5': { + sign: 'rsa', + hash: 'md5', + id: '3020300c06082a864886f70d020505000410' + } + }; + }, + {} ], - [ - '75d46efea3771e6e68abb89a13ad747ecf1892393dfc4f1b7004788c50374da8', - '9852390a99507679fd0b86fd2b39a868d7efc22151346e1a3ca4726586a6bed8' + 102: [ + function (require, module, exports) { + module.exports = { + '1.3.132.0.10': 'secp256k1', + '1.3.132.0.33': 'p224', + '1.2.840.10045.3.1.1': 'p192', + '1.2.840.10045.3.1.7': 'p256', + '1.3.132.0.34': 'p384', + '1.3.132.0.35': 'p521' + }; + }, + {} ], - [ - '809a20c67d64900ffb698c4c825f6d5f2310fb0451c869345b7319f645605721', - '9e994980d9917e22b76b061927fa04143d096ccc54963e6a5ebfa5f3f8e286c1' + 103: [ + function (require, module, exports) { + var Buffer = require('safe-buffer').Buffer; + var createHash = require('create-hash'); + var stream = require('readable-stream'); + var inherits = require('inherits'); + var sign = require('./sign'); + var verify = require('./verify'); + + var algorithms = require('./algorithms.json'); + Object.keys(algorithms).forEach(function (key) { + algorithms[key].id = Buffer.from(algorithms[key].id, 'hex'); + algorithms[key.toLowerCase()] = algorithms[key]; + }); + + function Sign(algorithm) { + stream.Writable.call(this); + + var data = algorithms[algorithm]; + if (!data) throw new Error('Unknown message digest'); + + this._hashType = data.hash; + this._hash = createHash(data.hash); + this._tag = data.id; + this._signType = data.sign; + } + inherits(Sign, stream.Writable); + + Sign.prototype._write = function _write(data, _, done) { + this._hash.update(data); + done(); + }; + + Sign.prototype.update = function update(data, enc) { + if (typeof data === 'string') data = Buffer.from(data, enc); + + this._hash.update(data); + return this; + }; + + Sign.prototype.sign = function signMethod(key, enc) { + this.end(); + var hash = this._hash.digest(); + var sig = sign(hash, key, this._hashType, this._signType, this._tag); + + return enc ? sig.toString(enc) : sig; + }; + + function Verify(algorithm) { + stream.Writable.call(this); + + var data = algorithms[algorithm]; + if (!data) throw new Error('Unknown message digest'); + + this._hash = createHash(data.hash); + this._tag = data.id; + this._signType = data.sign; + } + inherits(Verify, stream.Writable); + + Verify.prototype._write = function _write(data, _, done) { + this._hash.update(data); + done(); + }; + + Verify.prototype.update = function update(data, enc) { + if (typeof data === 'string') data = Buffer.from(data, enc); + + this._hash.update(data); + return this; + }; + + Verify.prototype.verify = function verifyMethod(key, sig, enc) { + if (typeof sig === 'string') sig = Buffer.from(sig, enc); + + this.end(); + var hash = this._hash.digest(); + return verify(sig, hash, key, this._signType, this._tag); + }; + + function createSign(algorithm) { + return new Sign(algorithm); + } + + function createVerify(algorithm) { + return new Verify(algorithm); + } + + module.exports = { + Sign: createSign, + Verify: createVerify, + createSign: createSign, + createVerify: createVerify + }; + }, + { './algorithms.json': 101, './sign': 104, './verify': 105, 'create-hash': 132, inherits: 106, 'readable-stream': 121, 'safe-buffer': 122 } ], - [ - '1b38903a43f7f114ed4500b4eac7083fdefece1cf29c63528d563446f972c180', - '4036edc931a60ae889353f77fd53de4a2708b26b6f5da72ad3394119daf408f9' - ] - ] - } -}; - -},{}],162:[function(require,module,exports){ -'use strict'; - -var utils = exports; -var BN = require('bn.js'); -var minAssert = require('minimalistic-assert'); -var minUtils = require('minimalistic-crypto-utils'); - -utils.assert = minAssert; -utils.toArray = minUtils.toArray; -utils.zero2 = minUtils.zero2; -utils.toHex = minUtils.toHex; -utils.encode = minUtils.encode; - -// Represent num in a w-NAF form -function getNAF(num, w, bits) { - var naf = new Array(Math.max(num.bitLength(), bits) + 1); - naf.fill(0); - - var ws = 1 << (w + 1); - var k = num.clone(); - - for (var i = 0; i < naf.length; i++) { - var z; - var mod = k.andln(ws - 1); - if (k.isOdd()) { - if (mod > (ws >> 1) - 1) - z = (ws >> 1) - mod; - else - z = mod; - k.isubn(z); - } else { - z = 0; - } + 104: [ + function (require, module, exports) { + // much of this based on https://github.com/indutny/self-signed/blob/gh-pages/lib/rsa.js + var Buffer = require('safe-buffer').Buffer; + var createHmac = require('create-hmac'); + var crt = require('browserify-rsa'); + var EC = require('elliptic').ec; + var BN = require('bn.js'); + var parseKeys = require('parse-asn1'); + var curves = require('./curves.json'); + + function sign(hash, key, hashType, signType, tag) { + var priv = parseKeys(key); + if (priv.curve) { + // rsa keys can be interpreted as ecdsa ones in openssl + if (signType !== 'ecdsa' && signType !== 'ecdsa/rsa') throw new Error('wrong private key type'); + return ecSign(hash, priv); + } else if (priv.type === 'dsa') { + if (signType !== 'dsa') throw new Error('wrong private key type'); + return dsaSign(hash, priv, hashType); + } else { + if (signType !== 'rsa' && signType !== 'ecdsa/rsa') throw new Error('wrong private key type'); + } + hash = Buffer.concat([tag, hash]); + var len = priv.modulus.byteLength(); + var pad = [0, 1]; + while (hash.length + pad.length + 1 < len) pad.push(0xff); + pad.push(0x00); + var i = -1; + while (++i < hash.length) pad.push(hash[i]); + + var out = crt(pad, priv); + return out; + } - naf[i] = z; - k.iushrn(1); - } + function ecSign(hash, priv) { + var curveId = curves[priv.curve.join('.')]; + if (!curveId) throw new Error('unknown curve ' + priv.curve.join('.')); - return naf; -} -utils.getNAF = getNAF; - -// Represent k1, k2 in a Joint Sparse Form -function getJSF(k1, k2) { - var jsf = [ - [], - [] - ]; - - k1 = k1.clone(); - k2 = k2.clone(); - var d1 = 0; - var d2 = 0; - while (k1.cmpn(-d1) > 0 || k2.cmpn(-d2) > 0) { - - // First phase - var m14 = (k1.andln(3) + d1) & 3; - var m24 = (k2.andln(3) + d2) & 3; - if (m14 === 3) - m14 = -1; - if (m24 === 3) - m24 = -1; - var u1; - if ((m14 & 1) === 0) { - u1 = 0; - } else { - var m8 = (k1.andln(7) + d1) & 7; - if ((m8 === 3 || m8 === 5) && m24 === 2) - u1 = -m14; - else - u1 = m14; - } - jsf[0].push(u1); + var curve = new EC(curveId); + var key = curve.keyFromPrivate(priv.privateKey); + var out = key.sign(hash); - var u2; - if ((m24 & 1) === 0) { - u2 = 0; - } else { - var m8 = (k2.andln(7) + d2) & 7; - if ((m8 === 3 || m8 === 5) && m14 === 2) - u2 = -m24; - else - u2 = m24; - } - jsf[1].push(u2); - - // Second phase - if (2 * d1 === u1 + 1) - d1 = 1 - d1; - if (2 * d2 === u2 + 1) - d2 = 1 - d2; - k1.iushrn(1); - k2.iushrn(1); - } + return Buffer.from(out.toDER()); + } - return jsf; -} -utils.getJSF = getJSF; + function dsaSign(hash, priv, algo) { + var x = priv.params.priv_key; + var p = priv.params.p; + var q = priv.params.q; + var g = priv.params.g; + var r = new BN(0); + var k; + var H = bits2int(hash, q).mod(q); + var s = false; + var kv = getKey(x, q, hash, algo); + while (s === false) { + k = makeKey(q, kv, algo); + r = makeR(g, k, p, q); + s = k + .invm(q) + .imul(H.add(x.mul(r))) + .mod(q); + if (s.cmpn(0) === 0) { + s = false; + r = new BN(0); + } + } + return toDER(r, s); + } -function cachedProperty(obj, name, computer) { - var key = '_' + name; - obj.prototype[name] = function cachedProperty() { - return this[key] !== undefined ? this[key] : - this[key] = computer.call(this); - }; -} -utils.cachedProperty = cachedProperty; - -function parseBytes(bytes) { - return typeof bytes === 'string' ? utils.toArray(bytes, 'hex') : - bytes; -} -utils.parseBytes = parseBytes; - -function intFromLE(bytes) { - return new BN(bytes, 'hex', 'le'); -} -utils.intFromLE = intFromLE; - - -},{"bn.js":163,"minimalistic-assert":226,"minimalistic-crypto-utils":227}],163:[function(require,module,exports){ -arguments[4][70][0].apply(exports,arguments) -},{"buffer":77,"dup":70}],164:[function(require,module,exports){ -module.exports={ - "_from": "elliptic@^6.5.2", - "_id": "elliptic@6.5.3", - "_inBundle": false, - "_integrity": "sha512-IMqzv5wNQf+E6aHeIqATs0tOLeOTwj1QKbRcS3jBbYkl5oLAserA8yJTT7/VyHUYG91PRmPyeQDObKLPpeS4dw==", - "_location": "/elliptic", - "_phantomChildren": {}, - "_requested": { - "type": "range", - "registry": true, - "raw": "elliptic@^6.5.2", - "name": "elliptic", - "escapedName": "elliptic", - "rawSpec": "^6.5.2", - "saveSpec": null, - "fetchSpec": "^6.5.2" - }, - "_requiredBy": [ - "/browserify-sign", - "/create-ecdh" - ], - "_resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.3.tgz", - "_shasum": "cb59eb2efdaf73a0bd78ccd7015a62ad6e0f93d6", - "_spec": "elliptic@^6.5.2", - "_where": "/Users/peterflockhart/SAIL/jiff/node_modules/browserify-sign", - "author": { - "name": "Fedor Indutny", - "email": "fedor@indutny.com" - }, - "bugs": { - "url": "https://github.com/indutny/elliptic/issues" - }, - "bundleDependencies": false, - "dependencies": { - "bn.js": "^4.4.0", - "brorand": "^1.0.1", - "hash.js": "^1.0.0", - "hmac-drbg": "^1.0.0", - "inherits": "^2.0.1", - "minimalistic-assert": "^1.0.0", - "minimalistic-crypto-utils": "^1.0.0" - }, - "deprecated": false, - "description": "EC cryptography", - "devDependencies": { - "brfs": "^1.4.3", - "coveralls": "^3.0.8", - "grunt": "^1.0.4", - "grunt-browserify": "^5.0.0", - "grunt-cli": "^1.2.0", - "grunt-contrib-connect": "^1.0.0", - "grunt-contrib-copy": "^1.0.0", - "grunt-contrib-uglify": "^1.0.1", - "grunt-mocha-istanbul": "^3.0.1", - "grunt-saucelabs": "^9.0.1", - "istanbul": "^0.4.2", - "jscs": "^3.0.7", - "jshint": "^2.10.3", - "mocha": "^6.2.2" - }, - "files": [ - "lib" - ], - "homepage": "https://github.com/indutny/elliptic", - "keywords": [ - "EC", - "Elliptic", - "curve", - "Cryptography" - ], - "license": "MIT", - "main": "lib/elliptic.js", - "name": "elliptic", - "repository": { - "type": "git", - "url": "git+ssh://git@github.com/indutny/elliptic.git" - }, - "scripts": { - "jscs": "jscs benchmarks/*.js lib/*.js lib/**/*.js lib/**/**/*.js test/index.js", - "jshint": "jscs benchmarks/*.js lib/*.js lib/**/*.js lib/**/**/*.js test/index.js", - "lint": "npm run jscs && npm run jshint", - "test": "npm run lint && npm run unit", - "unit": "istanbul test _mocha --reporter=spec test/index.js", - "version": "grunt dist && git add dist/" - }, - "version": "6.5.3" -} - -},{}],165:[function(require,module,exports){ -module.exports = (function () { - if (typeof self !== 'undefined') { - return self; - } else if (typeof window !== 'undefined') { - return window; - } else { - return Function('return this')(); // eslint-disable-line no-new-func - } -})(); + function toDER(r, s) { + r = r.toArray(); + s = s.toArray(); -},{}],166:[function(require,module,exports){ + // Pad values + if (r[0] & 0x80) r = [0].concat(r); + if (s[0] & 0x80) s = [0].concat(s); -module.exports = require('./socket'); + var total = r.length + s.length + 4; + var res = [0x30, total, 0x02, r.length]; + res = res.concat(r, [0x02, s.length], s); + return Buffer.from(res); + } -/** - * Exports parser - * - * @api public - * - */ -module.exports.parser = require('engine.io-parser'); + function getKey(x, q, hash, algo) { + x = Buffer.from(x.toArray()); + if (x.length < q.byteLength()) { + var zeros = Buffer.alloc(q.byteLength() - x.length); + x = Buffer.concat([zeros, x]); + } + var hlen = hash.length; + var hbits = bits2octets(hash, q); + var v = Buffer.alloc(hlen); + v.fill(1); + var k = Buffer.alloc(hlen); + k = createHmac(algo, k) + .update(v) + .update(Buffer.from([0])) + .update(x) + .update(hbits) + .digest(); + v = createHmac(algo, k).update(v).digest(); + k = createHmac(algo, k) + .update(v) + .update(Buffer.from([1])) + .update(x) + .update(hbits) + .digest(); + v = createHmac(algo, k).update(v).digest(); + return { k: k, v: v }; + } -},{"./socket":167,"engine.io-parser":178}],167:[function(require,module,exports){ -/** - * Module dependencies. - */ + function bits2int(obits, q) { + var bits = new BN(obits); + var shift = (obits.length << 3) - q.bitLength(); + if (shift > 0) bits.ishrn(shift); + return bits; + } -var transports = require('./transports/index'); -var Emitter = require('component-emitter'); -var debug = require('debug')('engine.io-client:socket'); -var index = require('indexof'); -var parser = require('engine.io-parser'); -var parseuri = require('parseuri'); -var parseqs = require('parseqs'); + function bits2octets(bits, q) { + bits = bits2int(bits, q); + bits = bits.mod(q); + var out = Buffer.from(bits.toArray()); + if (out.length < q.byteLength()) { + var zeros = Buffer.alloc(q.byteLength() - out.length); + out = Buffer.concat([zeros, out]); + } + return out; + } -/** - * Module exports. - */ + function makeKey(q, kv, algo) { + var t; + var k; -module.exports = Socket; + do { + t = Buffer.alloc(0); -/** - * Socket constructor. - * - * @param {String|Object} uri or options - * @param {Object} options - * @api public - */ + while (t.length * 8 < q.bitLength()) { + kv.v = createHmac(algo, kv.k).update(kv.v).digest(); + t = Buffer.concat([t, kv.v]); + } -function Socket (uri, opts) { - if (!(this instanceof Socket)) return new Socket(uri, opts); + k = bits2int(t, q); + kv.k = createHmac(algo, kv.k) + .update(kv.v) + .update(Buffer.from([0])) + .digest(); + kv.v = createHmac(algo, kv.k).update(kv.v).digest(); + } while (k.cmp(q) !== -1); - opts = opts || {}; + return k; + } - if (uri && 'object' === typeof uri) { - opts = uri; - uri = null; - } + function makeR(g, k, p, q) { + return g.toRed(BN.mont(p)).redPow(k).fromRed().mod(q); + } - if (uri) { - uri = parseuri(uri); - opts.hostname = uri.host; - opts.secure = uri.protocol === 'https' || uri.protocol === 'wss'; - opts.port = uri.port; - if (uri.query) opts.query = uri.query; - } else if (opts.host) { - opts.hostname = parseuri(opts.host).host; - } + module.exports = sign; + module.exports.getKey = getKey; + module.exports.makeKey = makeKey; + }, + { './curves.json': 102, 'bn.js': 75, 'browserify-rsa': 98, 'create-hmac': 134, elliptic: 148, 'parse-asn1': 233, 'safe-buffer': 122 } + ], + 105: [ + function (require, module, exports) { + // much of this based on https://github.com/indutny/self-signed/blob/gh-pages/lib/rsa.js + var Buffer = require('safe-buffer').Buffer; + var BN = require('bn.js'); + var EC = require('elliptic').ec; + var parseKeys = require('parse-asn1'); + var curves = require('./curves.json'); + + function verify(sig, hash, key, signType, tag) { + var pub = parseKeys(key); + if (pub.type === 'ec') { + // rsa keys can be interpreted as ecdsa ones in openssl + if (signType !== 'ecdsa' && signType !== 'ecdsa/rsa') throw new Error('wrong public key type'); + return ecVerify(sig, hash, pub); + } else if (pub.type === 'dsa') { + if (signType !== 'dsa') throw new Error('wrong public key type'); + return dsaVerify(sig, hash, pub); + } else { + if (signType !== 'rsa' && signType !== 'ecdsa/rsa') throw new Error('wrong public key type'); + } + hash = Buffer.concat([tag, hash]); + var len = pub.modulus.byteLength(); + var pad = [1]; + var padNum = 0; + while (hash.length + pad.length + 2 < len) { + pad.push(0xff); + padNum++; + } + pad.push(0x00); + var i = -1; + while (++i < hash.length) { + pad.push(hash[i]); + } + pad = Buffer.from(pad); + var red = BN.mont(pub.modulus); + sig = new BN(sig).toRed(red); + + sig = sig.redPow(new BN(pub.publicExponent)); + sig = Buffer.from(sig.fromRed().toArray()); + var out = padNum < 8 ? 1 : 0; + len = Math.min(sig.length, pad.length); + if (sig.length !== pad.length) out = 1; + + i = -1; + while (++i < len) out |= sig[i] ^ pad[i]; + return out === 0; + } - this.secure = null != opts.secure ? opts.secure - : (typeof location !== 'undefined' && 'https:' === location.protocol); + function ecVerify(sig, hash, pub) { + var curveId = curves[pub.data.algorithm.curve.join('.')]; + if (!curveId) throw new Error('unknown curve ' + pub.data.algorithm.curve.join('.')); - if (opts.hostname && !opts.port) { - // if no port is specified manually, use the protocol default - opts.port = this.secure ? '443' : '80'; - } + var curve = new EC(curveId); + var pubkey = pub.data.subjectPrivateKey.data; - this.agent = opts.agent || false; - this.hostname = opts.hostname || - (typeof location !== 'undefined' ? location.hostname : 'localhost'); - this.port = opts.port || (typeof location !== 'undefined' && location.port - ? location.port - : (this.secure ? 443 : 80)); - this.query = opts.query || {}; - if ('string' === typeof this.query) this.query = parseqs.decode(this.query); - this.upgrade = false !== opts.upgrade; - this.path = (opts.path || '/engine.io').replace(/\/$/, '') + '/'; - this.forceJSONP = !!opts.forceJSONP; - this.jsonp = false !== opts.jsonp; - this.forceBase64 = !!opts.forceBase64; - this.enablesXDR = !!opts.enablesXDR; - this.withCredentials = false !== opts.withCredentials; - this.timestampParam = opts.timestampParam || 't'; - this.timestampRequests = opts.timestampRequests; - this.transports = opts.transports || ['polling', 'websocket']; - this.transportOptions = opts.transportOptions || {}; - this.readyState = ''; - this.writeBuffer = []; - this.prevBufferLen = 0; - this.policyPort = opts.policyPort || 843; - this.rememberUpgrade = opts.rememberUpgrade || false; - this.binaryType = null; - this.onlyBinaryUpgrades = opts.onlyBinaryUpgrades; - this.perMessageDeflate = false !== opts.perMessageDeflate ? (opts.perMessageDeflate || {}) : false; - - if (true === this.perMessageDeflate) this.perMessageDeflate = {}; - if (this.perMessageDeflate && null == this.perMessageDeflate.threshold) { - this.perMessageDeflate.threshold = 1024; - } + return curve.verify(hash, sig, pubkey); + } - // SSL options for Node.js client - this.pfx = opts.pfx || null; - this.key = opts.key || null; - this.passphrase = opts.passphrase || null; - this.cert = opts.cert || null; - this.ca = opts.ca || null; - this.ciphers = opts.ciphers || null; - this.rejectUnauthorized = opts.rejectUnauthorized === undefined ? true : opts.rejectUnauthorized; - this.forceNode = !!opts.forceNode; - - // detect ReactNative environment - this.isReactNative = (typeof navigator !== 'undefined' && typeof navigator.product === 'string' && navigator.product.toLowerCase() === 'reactnative'); - - // other options for Node.js or ReactNative client - if (typeof self === 'undefined' || this.isReactNative) { - if (opts.extraHeaders && Object.keys(opts.extraHeaders).length > 0) { - this.extraHeaders = opts.extraHeaders; - } + function dsaVerify(sig, hash, pub) { + var p = pub.data.p; + var q = pub.data.q; + var g = pub.data.g; + var y = pub.data.pub_key; + var unpacked = parseKeys.signature.decode(sig, 'der'); + var s = unpacked.s; + var r = unpacked.r; + checkValue(s, q); + checkValue(r, q); + var montp = BN.mont(p); + var w = s.invm(q); + var v = g + .toRed(montp) + .redPow(new BN(hash).mul(w).mod(q)) + .fromRed() + .mul(y.toRed(montp).redPow(r.mul(w).mod(q)).fromRed()) + .mod(p) + .mod(q); + return v.cmp(r) === 0; + } - if (opts.localAddress) { - this.localAddress = opts.localAddress; - } - } + function checkValue(b, q) { + if (b.cmpn(0) <= 0) throw new Error('invalid sig'); + if (b.cmp(q) >= q) throw new Error('invalid sig'); + } - // set on handshake - this.id = null; - this.upgrades = null; - this.pingInterval = null; - this.pingTimeout = null; + module.exports = verify; + }, + { './curves.json': 102, 'bn.js': 75, elliptic: 148, 'parse-asn1': 233, 'safe-buffer': 122 } + ], + 106: [ + function (require, module, exports) { + if (typeof Object.create === 'function') { + // implementation from standard node.js 'util' module + module.exports = function inherits(ctor, superCtor) { + if (superCtor) { + ctor.super_ = superCtor; + ctor.prototype = Object.create(superCtor.prototype, { + constructor: { + value: ctor, + enumerable: false, + writable: true, + configurable: true + } + }); + } + }; + } else { + // old school shim for old browsers + module.exports = function inherits(ctor, superCtor) { + if (superCtor) { + ctor.super_ = superCtor; + var TempCtor = function () {}; + TempCtor.prototype = superCtor.prototype; + ctor.prototype = new TempCtor(); + ctor.prototype.constructor = ctor; + } + }; + } + }, + {} + ], + 107: [ + function (require, module, exports) { + 'use strict'; + + function _inheritsLoose(subClass, superClass) { + subClass.prototype = Object.create(superClass.prototype); + subClass.prototype.constructor = subClass; + subClass.__proto__ = superClass; + } - // set on heartbeat - this.pingIntervalTimer = null; - this.pingTimeoutTimer = null; + var codes = {}; - this.open(); -} + function createErrorType(code, message, Base) { + if (!Base) { + Base = Error; + } -Socket.priorWebsocketSuccess = false; + function getMessage(arg1, arg2, arg3) { + if (typeof message === 'string') { + return message; + } else { + return message(arg1, arg2, arg3); + } + } -/** - * Mix in `Emitter`. - */ + var NodeError = + /*#__PURE__*/ + (function (_Base) { + _inheritsLoose(NodeError, _Base); + + function NodeError(arg1, arg2, arg3) { + return _Base.call(this, getMessage(arg1, arg2, arg3)) || this; + } + + return NodeError; + })(Base); + + NodeError.prototype.name = Base.name; + NodeError.prototype.code = code; + codes[code] = NodeError; + } // https://github.com/nodejs/node/blob/v10.8.0/lib/internal/errors.js + + function oneOf(expected, thing) { + if (Array.isArray(expected)) { + var len = expected.length; + expected = expected.map(function (i) { + return String(i); + }); + + if (len > 2) { + return 'one of '.concat(thing, ' ').concat(expected.slice(0, len - 1).join(', '), ', or ') + expected[len - 1]; + } else if (len === 2) { + return 'one of '.concat(thing, ' ').concat(expected[0], ' or ').concat(expected[1]); + } else { + return 'of '.concat(thing, ' ').concat(expected[0]); + } + } else { + return 'of '.concat(thing, ' ').concat(String(expected)); + } + } // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/startsWith -Emitter(Socket.prototype); + function startsWith(str, search, pos) { + return str.substr(!pos || pos < 0 ? 0 : +pos, search.length) === search; + } // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/endsWith -/** - * Protocol version. - * - * @api public - */ + function endsWith(str, search, this_len) { + if (this_len === undefined || this_len > str.length) { + this_len = str.length; + } -Socket.protocol = parser.protocol; // this is an int + return str.substring(this_len - search.length, this_len) === search; + } // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/includes -/** - * Expose deps for legacy compatibility - * and standalone browser access. - */ + function includes(str, search, start) { + if (typeof start !== 'number') { + start = 0; + } -Socket.Socket = Socket; -Socket.Transport = require('./transport'); -Socket.transports = require('./transports/index'); -Socket.parser = require('engine.io-parser'); + if (start + search.length > str.length) { + return false; + } else { + return str.indexOf(search, start) !== -1; + } + } -/** - * Creates transport of the given type. - * - * @param {String} transport name - * @return {Transport} - * @api private - */ + createErrorType( + 'ERR_INVALID_OPT_VALUE', + function (name, value) { + return 'The value "' + value + '" is invalid for option "' + name + '"'; + }, + TypeError + ); + createErrorType( + 'ERR_INVALID_ARG_TYPE', + function (name, expected, actual) { + // determiner: 'must be' or 'must not be' + var determiner; + + if (typeof expected === 'string' && startsWith(expected, 'not ')) { + determiner = 'must not be'; + expected = expected.replace(/^not /, ''); + } else { + determiner = 'must be'; + } + + var msg; + + if (endsWith(name, ' argument')) { + // For cases like 'first argument' + msg = 'The '.concat(name, ' ').concat(determiner, ' ').concat(oneOf(expected, 'type')); + } else { + var type = includes(name, '.') ? 'property' : 'argument'; + msg = 'The "'.concat(name, '" ').concat(type, ' ').concat(determiner, ' ').concat(oneOf(expected, 'type')); + } + + msg += '. Received type '.concat(typeof actual); + return msg; + }, + TypeError + ); + createErrorType('ERR_STREAM_PUSH_AFTER_EOF', 'stream.push() after EOF'); + createErrorType('ERR_METHOD_NOT_IMPLEMENTED', function (name) { + return 'The ' + name + ' method is not implemented'; + }); + createErrorType('ERR_STREAM_PREMATURE_CLOSE', 'Premature close'); + createErrorType('ERR_STREAM_DESTROYED', function (name) { + return 'Cannot call ' + name + ' after a stream was destroyed'; + }); + createErrorType('ERR_MULTIPLE_CALLBACK', 'Callback called multiple times'); + createErrorType('ERR_STREAM_CANNOT_PIPE', 'Cannot pipe, not readable'); + createErrorType('ERR_STREAM_WRITE_AFTER_END', 'write after end'); + createErrorType('ERR_STREAM_NULL_VALUES', 'May not write null values to stream', TypeError); + createErrorType( + 'ERR_UNKNOWN_ENCODING', + function (arg) { + return 'Unknown encoding: ' + arg; + }, + TypeError + ); + createErrorType('ERR_STREAM_UNSHIFT_AFTER_END_EVENT', 'stream.unshift() after end event'); + module.exports.codes = codes; + }, + {} + ], + 108: [ + function (require, module, exports) { + (function (process) { + // Copyright Joyent, Inc. and other Node contributors. + // + // Permission is hereby granted, free of charge, to any person obtaining a + // copy of this software and associated documentation files (the + // "Software"), to deal in the Software without restriction, including + // without limitation the rights to use, copy, modify, merge, publish, + // distribute, sublicense, and/or sell copies of the Software, and to permit + // persons to whom the Software is furnished to do so, subject to the + // following conditions: + // + // The above copyright notice and this permission notice shall be included + // in all copies or substantial portions of the Software. + // + // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS + // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF + // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN + // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, + // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR + // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE + // USE OR OTHER DEALINGS IN THE SOFTWARE. + // a duplex stream is just a stream that is both readable and writable. + // Since JS doesn't have multiple prototypal inheritance, this class + // prototypally inherits from Readable, and then parasitically from + // Writable. + 'use strict'; + /**/ + + var objectKeys = + Object.keys || + function (obj) { + var keys = []; + + for (var key in obj) { + keys.push(key); + } + + return keys; + }; + /**/ + + module.exports = Duplex; + + var Readable = require('./_stream_readable'); + + var Writable = require('./_stream_writable'); + + require('inherits')(Duplex, Readable); + + { + // Allow the keys array to be GC'ed. + var keys = objectKeys(Writable.prototype); + + for (var v = 0; v < keys.length; v++) { + var method = keys[v]; + if (!Duplex.prototype[method]) Duplex.prototype[method] = Writable.prototype[method]; + } + } -Socket.prototype.createTransport = function (name) { - debug('creating transport "%s"', name); - var query = clone(this.query); - - // append engine.io protocol identifier - query.EIO = parser.protocol; - - // transport name - query.transport = name; - - // per-transport options - var options = this.transportOptions[name] || {}; - - // session id if we already have one - if (this.id) query.sid = this.id; - - var transport = new transports[name]({ - query: query, - socket: this, - agent: options.agent || this.agent, - hostname: options.hostname || this.hostname, - port: options.port || this.port, - secure: options.secure || this.secure, - path: options.path || this.path, - forceJSONP: options.forceJSONP || this.forceJSONP, - jsonp: options.jsonp || this.jsonp, - forceBase64: options.forceBase64 || this.forceBase64, - enablesXDR: options.enablesXDR || this.enablesXDR, - withCredentials: options.withCredentials || this.withCredentials, - timestampRequests: options.timestampRequests || this.timestampRequests, - timestampParam: options.timestampParam || this.timestampParam, - policyPort: options.policyPort || this.policyPort, - pfx: options.pfx || this.pfx, - key: options.key || this.key, - passphrase: options.passphrase || this.passphrase, - cert: options.cert || this.cert, - ca: options.ca || this.ca, - ciphers: options.ciphers || this.ciphers, - rejectUnauthorized: options.rejectUnauthorized || this.rejectUnauthorized, - perMessageDeflate: options.perMessageDeflate || this.perMessageDeflate, - extraHeaders: options.extraHeaders || this.extraHeaders, - forceNode: options.forceNode || this.forceNode, - localAddress: options.localAddress || this.localAddress, - requestTimeout: options.requestTimeout || this.requestTimeout, - protocols: options.protocols || void (0), - isReactNative: this.isReactNative - }); - - return transport; -}; - -function clone (obj) { - var o = {}; - for (var i in obj) { - if (obj.hasOwnProperty(i)) { - o[i] = obj[i]; - } - } - return o; -} + function Duplex(options) { + if (!(this instanceof Duplex)) return new Duplex(options); + Readable.call(this, options); + Writable.call(this, options); + this.allowHalfOpen = true; + + if (options) { + if (options.readable === false) this.readable = false; + if (options.writable === false) this.writable = false; + + if (options.allowHalfOpen === false) { + this.allowHalfOpen = false; + this.once('end', onend); + } + } + } -/** - * Initializes transport to use and starts probe. - * - * @api private - */ -Socket.prototype.open = function () { - var transport; - if (this.rememberUpgrade && Socket.priorWebsocketSuccess && this.transports.indexOf('websocket') !== -1) { - transport = 'websocket'; - } else if (0 === this.transports.length) { - // Emit error on next tick so it can be listened to - var self = this; - setTimeout(function () { - self.emit('error', 'No transports available'); - }, 0); - return; - } else { - transport = this.transports[0]; - } - this.readyState = 'opening'; - - // Retry with the next transport if the transport is disabled (jsonp: false) - try { - transport = this.createTransport(transport); - } catch (e) { - this.transports.shift(); - this.open(); - return; - } + Object.defineProperty(Duplex.prototype, 'writableHighWaterMark', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._writableState.highWaterMark; + } + }); + Object.defineProperty(Duplex.prototype, 'writableBuffer', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._writableState && this._writableState.getBuffer(); + } + }); + Object.defineProperty(Duplex.prototype, 'writableLength', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._writableState.length; + } + }); // the no-half-open enforcer + + function onend() { + // If the writable side ended, then we're ok. + if (this._writableState.ended) return; // no more data can be written. + // But allow more writes to happen in this tick. + + process.nextTick(onEndNT, this); + } - transport.open(); - this.setTransport(transport); -}; + function onEndNT(self) { + self.end(); + } -/** - * Sets the current transport. Disables the existing one (if any). - * - * @api private - */ + Object.defineProperty(Duplex.prototype, 'destroyed', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + if (this._readableState === undefined || this._writableState === undefined) { + return false; + } + + return this._readableState.destroyed && this._writableState.destroyed; + }, + set: function set(value) { + // we ignore the value if the stream + // has not been initialized yet + if (this._readableState === undefined || this._writableState === undefined) { + return; + } // backward compatibility, the user is explicitly + // managing destroyed + + this._readableState.destroyed = value; + this._writableState.destroyed = value; + } + }); + }).call(this, require('_process')); + }, + { './_stream_readable': 110, './_stream_writable': 112, _process: 244, inherits: 106 } + ], + 109: [ + function (require, module, exports) { + // Copyright Joyent, Inc. and other Node contributors. + // + // Permission is hereby granted, free of charge, to any person obtaining a + // copy of this software and associated documentation files (the + // "Software"), to deal in the Software without restriction, including + // without limitation the rights to use, copy, modify, merge, publish, + // distribute, sublicense, and/or sell copies of the Software, and to permit + // persons to whom the Software is furnished to do so, subject to the + // following conditions: + // + // The above copyright notice and this permission notice shall be included + // in all copies or substantial portions of the Software. + // + // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS + // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF + // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN + // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, + // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR + // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE + // USE OR OTHER DEALINGS IN THE SOFTWARE. + // a passthrough stream. + // basically just the most minimal sort of Transform stream. + // Every written chunk gets output as-is. + 'use strict'; + + module.exports = PassThrough; + + var Transform = require('./_stream_transform'); + + require('inherits')(PassThrough, Transform); + + function PassThrough(options) { + if (!(this instanceof PassThrough)) return new PassThrough(options); + Transform.call(this, options); + } -Socket.prototype.setTransport = function (transport) { - debug('setting transport %s', transport.name); - var self = this; + PassThrough.prototype._transform = function (chunk, encoding, cb) { + cb(null, chunk); + }; + }, + { './_stream_transform': 111, inherits: 106 } + ], + 110: [ + function (require, module, exports) { + (function (process, global) { + // Copyright Joyent, Inc. and other Node contributors. + // + // Permission is hereby granted, free of charge, to any person obtaining a + // copy of this software and associated documentation files (the + // "Software"), to deal in the Software without restriction, including + // without limitation the rights to use, copy, modify, merge, publish, + // distribute, sublicense, and/or sell copies of the Software, and to permit + // persons to whom the Software is furnished to do so, subject to the + // following conditions: + // + // The above copyright notice and this permission notice shall be included + // in all copies or substantial portions of the Software. + // + // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS + // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF + // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN + // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, + // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR + // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE + // USE OR OTHER DEALINGS IN THE SOFTWARE. + 'use strict'; + + module.exports = Readable; + /**/ + + var Duplex; + /**/ + + Readable.ReadableState = ReadableState; + /**/ + + var EE = require('events').EventEmitter; + + var EElistenerCount = function EElistenerCount(emitter, type) { + return emitter.listeners(type).length; + }; + /**/ + + /**/ + + var Stream = require('./internal/streams/stream'); + /**/ + + var Buffer = require('buffer').Buffer; + + var OurUint8Array = global.Uint8Array || function () {}; + + function _uint8ArrayToBuffer(chunk) { + return Buffer.from(chunk); + } - if (this.transport) { - debug('clearing existing transport %s', this.transport.name); - this.transport.removeAllListeners(); - } + function _isUint8Array(obj) { + return Buffer.isBuffer(obj) || obj instanceof OurUint8Array; + } + /**/ - // set up transport - this.transport = transport; - - // set up transport listeners - transport - .on('drain', function () { - self.onDrain(); - }) - .on('packet', function (packet) { - self.onPacket(packet); - }) - .on('error', function (e) { - self.onError(e); - }) - .on('close', function () { - self.onClose('transport close'); - }); -}; - -/** - * Probes a transport. - * - * @param {String} transport name - * @api private - */ + var debugUtil = require('util'); -Socket.prototype.probe = function (name) { - debug('probing transport "%s"', name); - var transport = this.createTransport(name, { probe: 1 }); - var failed = false; - var self = this; + var debug; - Socket.priorWebsocketSuccess = false; + if (debugUtil && debugUtil.debuglog) { + debug = debugUtil.debuglog('stream'); + } else { + debug = function debug() {}; + } + /**/ - function onTransportOpen () { - if (self.onlyBinaryUpgrades) { - var upgradeLosesBinary = !this.supportsBinary && self.transport.supportsBinary; - failed = failed || upgradeLosesBinary; - } - if (failed) return; - - debug('probe transport "%s" opened', name); - transport.send([{ type: 'ping', data: 'probe' }]); - transport.once('packet', function (msg) { - if (failed) return; - if ('pong' === msg.type && 'probe' === msg.data) { - debug('probe transport "%s" pong', name); - self.upgrading = true; - self.emit('upgrading', transport); - if (!transport) return; - Socket.priorWebsocketSuccess = 'websocket' === transport.name; - - debug('pausing current transport "%s"', self.transport.name); - self.transport.pause(function () { - if (failed) return; - if ('closed' === self.readyState) return; - debug('changing transport and sending upgrade packet'); - - cleanup(); - - self.setTransport(transport); - transport.send([{ type: 'upgrade' }]); - self.emit('upgrade', transport); - transport = null; - self.upgrading = false; - self.flush(); - }); - } else { - debug('probe transport "%s" failed', name); - var err = new Error('probe error'); - err.transport = transport.name; - self.emit('upgradeError', err); - } - }); - } + var BufferList = require('./internal/streams/buffer_list'); - function freezeTransport () { - if (failed) return; + var destroyImpl = require('./internal/streams/destroy'); - // Any callback called by transport should be ignored since now - failed = true; + var _require = require('./internal/streams/state'), + getHighWaterMark = _require.getHighWaterMark; - cleanup(); + var _require$codes = require('../errors').codes, + ERR_INVALID_ARG_TYPE = _require$codes.ERR_INVALID_ARG_TYPE, + ERR_STREAM_PUSH_AFTER_EOF = _require$codes.ERR_STREAM_PUSH_AFTER_EOF, + ERR_METHOD_NOT_IMPLEMENTED = _require$codes.ERR_METHOD_NOT_IMPLEMENTED, + ERR_STREAM_UNSHIFT_AFTER_END_EVENT = _require$codes.ERR_STREAM_UNSHIFT_AFTER_END_EVENT; // Lazy loaded to improve the startup performance. - transport.close(); - transport = null; - } + var StringDecoder; + var createReadableStreamAsyncIterator; + var from; - // Handle any error that happens while probing - function onerror (err) { - var error = new Error('probe error: ' + err); - error.transport = transport.name; + require('inherits')(Readable, Stream); - freezeTransport(); + var errorOrDestroy = destroyImpl.errorOrDestroy; + var kProxyEvents = ['error', 'close', 'destroy', 'pause', 'resume']; - debug('probe transport "%s" failed because of error: %s', name, err); + function prependListener(emitter, event, fn) { + // Sadly this is not cacheable as some libraries bundle their own + // event emitter implementation with them. + if (typeof emitter.prependListener === 'function') return emitter.prependListener(event, fn); // This is a hack to make sure that our error handler is attached before any + // userland ones. NEVER DO THIS. This is here only because this code needs + // to continue to work with older versions of Node.js that do not include + // the prependListener() method. The goal is to eventually remove this hack. - self.emit('upgradeError', error); - } + if (!emitter._events || !emitter._events[event]) emitter.on(event, fn); + else if (Array.isArray(emitter._events[event])) emitter._events[event].unshift(fn); + else emitter._events[event] = [fn, emitter._events[event]]; + } - function onTransportClose () { - onerror('transport closed'); - } + function ReadableState(options, stream, isDuplex) { + Duplex = Duplex || require('./_stream_duplex'); + options = options || {}; // Duplex streams are both readable and writable, but share + // the same options object. + // However, some cases require setting options to different + // values for the readable and the writable sides of the duplex stream. + // These options can be provided separately as readableXXX and writableXXX. + + if (typeof isDuplex !== 'boolean') isDuplex = stream instanceof Duplex; // object stream flag. Used to make read(n) ignore n and to + // make all the buffer merging and length checks go away + + this.objectMode = !!options.objectMode; + if (isDuplex) this.objectMode = this.objectMode || !!options.readableObjectMode; // the point at which it stops calling _read() to fill the buffer + // Note: 0 is a valid value, means "don't call _read preemptively ever" + + this.highWaterMark = getHighWaterMark(this, options, 'readableHighWaterMark', isDuplex); // A linked list is used to store data chunks instead of an array because the + // linked list can remove elements from the beginning faster than + // array.shift() + + this.buffer = new BufferList(); + this.length = 0; + this.pipes = null; + this.pipesCount = 0; + this.flowing = null; + this.ended = false; + this.endEmitted = false; + this.reading = false; // a flag to be able to tell if the event 'readable'/'data' is emitted + // immediately, or on a later tick. We set this to true at first, because + // any actions that shouldn't happen until "later" should generally also + // not happen before the first read call. + + this.sync = true; // whenever we return null, then we set a flag to say + // that we're awaiting a 'readable' event emission. + + this.needReadable = false; + this.emittedReadable = false; + this.readableListening = false; + this.resumeScheduled = false; + this.paused = true; // Should close be emitted on destroy. Defaults to true. + + this.emitClose = options.emitClose !== false; // Should .destroy() be called after 'end' (and potentially 'finish') + + this.autoDestroy = !!options.autoDestroy; // has it been destroyed + + this.destroyed = false; // Crypto is kind of old and crusty. Historically, its default string + // encoding is 'binary' so we have to make this configurable. + // Everything else in the universe uses 'utf8', though. + + this.defaultEncoding = options.defaultEncoding || 'utf8'; // the number of writers that are awaiting a drain event in .pipe()s + + this.awaitDrain = 0; // if true, a maybeReadMore has been scheduled + + this.readingMore = false; + this.decoder = null; + this.encoding = null; + + if (options.encoding) { + if (!StringDecoder) StringDecoder = require('string_decoder/').StringDecoder; + this.decoder = new StringDecoder(options.encoding); + this.encoding = options.encoding; + } + } - // When the socket is closed while we're probing - function onclose () { - onerror('socket closed'); - } + function Readable(options) { + Duplex = Duplex || require('./_stream_duplex'); + if (!(this instanceof Readable)) return new Readable(options); // Checking for a Stream.Duplex instance is faster here instead of inside + // the ReadableState constructor, at least with V8 6.5 - // When the socket is upgraded while we're probing - function onupgrade (to) { - if (transport && to.name !== transport.name) { - debug('"%s" works - aborting "%s"', to.name, transport.name); - freezeTransport(); - } - } + var isDuplex = this instanceof Duplex; + this._readableState = new ReadableState(options, this, isDuplex); // legacy - // Remove all listeners on the transport and on self - function cleanup () { - transport.removeListener('open', onTransportOpen); - transport.removeListener('error', onerror); - transport.removeListener('close', onTransportClose); - self.removeListener('close', onclose); - self.removeListener('upgrading', onupgrade); - } + this.readable = true; - transport.once('open', onTransportOpen); - transport.once('error', onerror); - transport.once('close', onTransportClose); + if (options) { + if (typeof options.read === 'function') this._read = options.read; + if (typeof options.destroy === 'function') this._destroy = options.destroy; + } - this.once('close', onclose); - this.once('upgrading', onupgrade); + Stream.call(this); + } - transport.open(); -}; + Object.defineProperty(Readable.prototype, 'destroyed', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + if (this._readableState === undefined) { + return false; + } + + return this._readableState.destroyed; + }, + set: function set(value) { + // we ignore the value if the stream + // has not been initialized yet + if (!this._readableState) { + return; + } // backward compatibility, the user is explicitly + // managing destroyed + + this._readableState.destroyed = value; + } + }); + Readable.prototype.destroy = destroyImpl.destroy; + Readable.prototype._undestroy = destroyImpl.undestroy; + + Readable.prototype._destroy = function (err, cb) { + cb(err); + }; // Manually shove something into the read() buffer. + // This returns true if the highWaterMark has not been hit yet, + // similar to how Writable.write() returns true if you should + // write() some more. + + Readable.prototype.push = function (chunk, encoding) { + var state = this._readableState; + var skipChunkCheck; + + if (!state.objectMode) { + if (typeof chunk === 'string') { + encoding = encoding || state.defaultEncoding; + + if (encoding !== state.encoding) { + chunk = Buffer.from(chunk, encoding); + encoding = ''; + } + + skipChunkCheck = true; + } + } else { + skipChunkCheck = true; + } + + return readableAddChunk(this, chunk, encoding, false, skipChunkCheck); + }; // Unshift should *always* be something directly out of read() + + Readable.prototype.unshift = function (chunk) { + return readableAddChunk(this, chunk, null, true, false); + }; + + function readableAddChunk(stream, chunk, encoding, addToFront, skipChunkCheck) { + debug('readableAddChunk', chunk); + var state = stream._readableState; + + if (chunk === null) { + state.reading = false; + onEofChunk(stream, state); + } else { + var er; + if (!skipChunkCheck) er = chunkInvalid(state, chunk); + + if (er) { + errorOrDestroy(stream, er); + } else if (state.objectMode || (chunk && chunk.length > 0)) { + if (typeof chunk !== 'string' && !state.objectMode && Object.getPrototypeOf(chunk) !== Buffer.prototype) { + chunk = _uint8ArrayToBuffer(chunk); + } + + if (addToFront) { + if (state.endEmitted) errorOrDestroy(stream, new ERR_STREAM_UNSHIFT_AFTER_END_EVENT()); + else addChunk(stream, state, chunk, true); + } else if (state.ended) { + errorOrDestroy(stream, new ERR_STREAM_PUSH_AFTER_EOF()); + } else if (state.destroyed) { + return false; + } else { + state.reading = false; + + if (state.decoder && !encoding) { + chunk = state.decoder.write(chunk); + if (state.objectMode || chunk.length !== 0) addChunk(stream, state, chunk, false); + else maybeReadMore(stream, state); + } else { + addChunk(stream, state, chunk, false); + } + } + } else if (!addToFront) { + state.reading = false; + maybeReadMore(stream, state); + } + } // We can push more data if we are below the highWaterMark. + // Also, if we have no data yet, we can stand some more bytes. + // This is to work around cases where hwm=0, such as the repl. + + return !state.ended && (state.length < state.highWaterMark || state.length === 0); + } -/** - * Called when connection is deemed open. - * - * @api public - */ + function addChunk(stream, state, chunk, addToFront) { + if (state.flowing && state.length === 0 && !state.sync) { + state.awaitDrain = 0; + stream.emit('data', chunk); + } else { + // update the buffer info. + state.length += state.objectMode ? 1 : chunk.length; + if (addToFront) state.buffer.unshift(chunk); + else state.buffer.push(chunk); + if (state.needReadable) emitReadable(stream); + } + + maybeReadMore(stream, state); + } -Socket.prototype.onOpen = function () { - debug('socket open'); - this.readyState = 'open'; - Socket.priorWebsocketSuccess = 'websocket' === this.transport.name; - this.emit('open'); - this.flush(); - - // we check for `readyState` in case an `open` - // listener already closed the socket - if ('open' === this.readyState && this.upgrade && this.transport.pause) { - debug('starting upgrade probes'); - for (var i = 0, l = this.upgrades.length; i < l; i++) { - this.probe(this.upgrades[i]); - } - } -}; + function chunkInvalid(state, chunk) { + var er; -/** - * Handles a packet. - * - * @api private - */ + if (!_isUint8Array(chunk) && typeof chunk !== 'string' && chunk !== undefined && !state.objectMode) { + er = new ERR_INVALID_ARG_TYPE('chunk', ['string', 'Buffer', 'Uint8Array'], chunk); + } -Socket.prototype.onPacket = function (packet) { - if ('opening' === this.readyState || 'open' === this.readyState || - 'closing' === this.readyState) { - debug('socket receive: type "%s", data "%s"', packet.type, packet.data); + return er; + } - this.emit('packet', packet); + Readable.prototype.isPaused = function () { + return this._readableState.flowing === false; + }; // backwards compatibility. + + Readable.prototype.setEncoding = function (enc) { + if (!StringDecoder) StringDecoder = require('string_decoder/').StringDecoder; + var decoder = new StringDecoder(enc); + this._readableState.decoder = decoder; // If setEncoding(null), decoder.encoding equals utf8 + + this._readableState.encoding = this._readableState.decoder.encoding; // Iterate over current buffer to convert already stored Buffers: + + var p = this._readableState.buffer.head; + var content = ''; + + while (p !== null) { + content += decoder.write(p.data); + p = p.next; + } + + this._readableState.buffer.clear(); + + if (content !== '') this._readableState.buffer.push(content); + this._readableState.length = content.length; + return this; + }; // Don't raise the hwm > 1GB + + var MAX_HWM = 0x40000000; + + function computeNewHighWaterMark(n) { + if (n >= MAX_HWM) { + // TODO(ronag): Throw ERR_VALUE_OUT_OF_RANGE. + n = MAX_HWM; + } else { + // Get the next highest power of 2 to prevent increasing hwm excessively in + // tiny amounts + n--; + n |= n >>> 1; + n |= n >>> 2; + n |= n >>> 4; + n |= n >>> 8; + n |= n >>> 16; + n++; + } + + return n; + } // This function is designed to be inlinable, so please take care when making + // changes to the function body. + + function howMuchToRead(n, state) { + if (n <= 0 || (state.length === 0 && state.ended)) return 0; + if (state.objectMode) return 1; + + if (n !== n) { + // Only flow one buffer at a time + if (state.flowing && state.length) return state.buffer.head.data.length; + else return state.length; + } // If we're asking for more than the current hwm, then raise the hwm. + + if (n > state.highWaterMark) state.highWaterMark = computeNewHighWaterMark(n); + if (n <= state.length) return n; // Don't have enough + + if (!state.ended) { + state.needReadable = true; + return 0; + } + + return state.length; + } // you can override either this method, or the async _read(n) below. + + Readable.prototype.read = function (n) { + debug('read', n); + n = parseInt(n, 10); + var state = this._readableState; + var nOrig = n; + if (n !== 0) state.emittedReadable = false; // if we're doing read(0) to trigger a readable event, but we + // already have a bunch of data in the buffer, then just trigger + // the 'readable' event and move on. + + if (n === 0 && state.needReadable && ((state.highWaterMark !== 0 ? state.length >= state.highWaterMark : state.length > 0) || state.ended)) { + debug('read: emitReadable', state.length, state.ended); + if (state.length === 0 && state.ended) endReadable(this); + else emitReadable(this); + return null; + } + + n = howMuchToRead(n, state); // if we've ended, and we're now clear, then finish it up. + + if (n === 0 && state.ended) { + if (state.length === 0) endReadable(this); + return null; + } // All the actual chunk generation logic needs to be + // *below* the call to _read. The reason is that in certain + // synthetic stream cases, such as passthrough streams, _read + // may be a completely synchronous operation which may change + // the state of the read buffer, providing enough data when + // before there was *not* enough. + // + // So, the steps are: + // 1. Figure out what the state of things will be after we do + // a read from the buffer. + // + // 2. If that resulting state will trigger a _read, then call _read. + // Note that this may be asynchronous, or synchronous. Yes, it is + // deeply ugly to write APIs this way, but that still doesn't mean + // that the Readable class should behave improperly, as streams are + // designed to be sync/async agnostic. + // Take note if the _read call is sync or async (ie, if the read call + // has returned yet), so that we know whether or not it's safe to emit + // 'readable' etc. + // + // 3. Actually pull the requested chunks out of the buffer and return. + // if we need a readable event, then we need to do some reading. + + var doRead = state.needReadable; + debug('need readable', doRead); // if we currently have less than the highWaterMark, then also read some + + if (state.length === 0 || state.length - n < state.highWaterMark) { + doRead = true; + debug('length less than watermark', doRead); + } // however, if we've ended, then there's no point, and if we're already + // reading, then it's unnecessary. + + if (state.ended || state.reading) { + doRead = false; + debug('reading or ended', doRead); + } else if (doRead) { + debug('do read'); + state.reading = true; + state.sync = true; // if the length is currently zero, then we *need* a readable event. + + if (state.length === 0) state.needReadable = true; // call internal read method + + this._read(state.highWaterMark); + + state.sync = false; // If _read pushed data synchronously, then `reading` will be false, + // and we need to re-evaluate how much data we can return to the user. + + if (!state.reading) n = howMuchToRead(nOrig, state); + } + + var ret; + if (n > 0) ret = fromList(n, state); + else ret = null; + + if (ret === null) { + state.needReadable = state.length <= state.highWaterMark; + n = 0; + } else { + state.length -= n; + state.awaitDrain = 0; + } + + if (state.length === 0) { + // If we have nothing in the buffer, then we want to know + // as soon as we *do* get something into the buffer. + if (!state.ended) state.needReadable = true; // If we tried to read() past the EOF, then emit end on the next tick. + + if (nOrig !== n && state.ended) endReadable(this); + } + + if (ret !== null) this.emit('data', ret); + return ret; + }; + + function onEofChunk(stream, state) { + debug('onEofChunk'); + if (state.ended) return; + + if (state.decoder) { + var chunk = state.decoder.end(); + + if (chunk && chunk.length) { + state.buffer.push(chunk); + state.length += state.objectMode ? 1 : chunk.length; + } + } + + state.ended = true; + + if (state.sync) { + // if we are sync, wait until next tick to emit the data. + // Otherwise we risk emitting data in the flow() + // the readable code triggers during a read() call + emitReadable(stream); + } else { + // emit 'readable' now to make sure it gets picked up. + state.needReadable = false; + + if (!state.emittedReadable) { + state.emittedReadable = true; + emitReadable_(stream); + } + } + } // Don't emit readable right away in sync mode, because this can trigger + // another read() call => stack overflow. This way, it might trigger + // a nextTick recursion warning, but that's not so bad. + + function emitReadable(stream) { + var state = stream._readableState; + debug('emitReadable', state.needReadable, state.emittedReadable); + state.needReadable = false; + + if (!state.emittedReadable) { + debug('emitReadable', state.flowing); + state.emittedReadable = true; + process.nextTick(emitReadable_, stream); + } + } - // Socket is live - any packet counts - this.emit('heartbeat'); + function emitReadable_(stream) { + var state = stream._readableState; + debug('emitReadable_', state.destroyed, state.length, state.ended); + + if (!state.destroyed && (state.length || state.ended)) { + stream.emit('readable'); + state.emittedReadable = false; + } // The stream needs another readable event if + // 1. It is not flowing, as the flow mechanism will take + // care of it. + // 2. It is not ended. + // 3. It is below the highWaterMark, so we can schedule + // another readable later. + + state.needReadable = !state.flowing && !state.ended && state.length <= state.highWaterMark; + flow(stream); + } // at this point, the user has presumably seen the 'readable' event, + // and called read() to consume some data. that may have triggered + // in turn another _read(n) call, in which case reading = true if + // it's in progress. + // However, if we're not ended, or reading, and the length < hwm, + // then go ahead and try to read some more preemptively. + + function maybeReadMore(stream, state) { + if (!state.readingMore) { + state.readingMore = true; + process.nextTick(maybeReadMore_, stream, state); + } + } - switch (packet.type) { - case 'open': - this.onHandshake(JSON.parse(packet.data)); - break; + function maybeReadMore_(stream, state) { + // Attempt to read more data if we should. + // + // The conditions for reading more data are (one of): + // - Not enough data buffered (state.length < state.highWaterMark). The loop + // is responsible for filling the buffer with enough data if such data + // is available. If highWaterMark is 0 and we are not in the flowing mode + // we should _not_ attempt to buffer any extra data. We'll get more data + // when the stream consumer calls read() instead. + // - No data in the buffer, and the stream is in flowing mode. In this mode + // the loop below is responsible for ensuring read() is called. Failing to + // call read here would abort the flow and there's no other mechanism for + // continuing the flow if the stream consumer has just subscribed to the + // 'data' event. + // + // In addition to the above conditions to keep reading data, the following + // conditions prevent the data from being read: + // - The stream has ended (state.ended). + // - There is already a pending 'read' operation (state.reading). This is a + // case where the the stream has called the implementation defined _read() + // method, but they are processing the call asynchronously and have _not_ + // called push() with new data. In this case we skip performing more + // read()s. The execution ends in this method again after the _read() ends + // up calling push() with more data. + while (!state.reading && !state.ended && (state.length < state.highWaterMark || (state.flowing && state.length === 0))) { + var len = state.length; + debug('maybeReadMore read 0'); + stream.read(0); + if (len === state.length) + // didn't get any data, stop spinning. + break; + } + + state.readingMore = false; + } // abstract method. to be overridden in specific implementation classes. + // call cb(er, data) where data is <= n in length. + // for virtual (non-string, non-buffer) streams, "length" is somewhat + // arbitrary, and perhaps not very meaningful. + + Readable.prototype._read = function (n) { + errorOrDestroy(this, new ERR_METHOD_NOT_IMPLEMENTED('_read()')); + }; + + Readable.prototype.pipe = function (dest, pipeOpts) { + var src = this; + var state = this._readableState; + + switch (state.pipesCount) { + case 0: + state.pipes = dest; + break; + + case 1: + state.pipes = [state.pipes, dest]; + break; + + default: + state.pipes.push(dest); + break; + } + + state.pipesCount += 1; + debug('pipe count=%d opts=%j', state.pipesCount, pipeOpts); + var doEnd = (!pipeOpts || pipeOpts.end !== false) && dest !== process.stdout && dest !== process.stderr; + var endFn = doEnd ? onend : unpipe; + if (state.endEmitted) process.nextTick(endFn); + else src.once('end', endFn); + dest.on('unpipe', onunpipe); + + function onunpipe(readable, unpipeInfo) { + debug('onunpipe'); + + if (readable === src) { + if (unpipeInfo && unpipeInfo.hasUnpiped === false) { + unpipeInfo.hasUnpiped = true; + cleanup(); + } + } + } + + function onend() { + debug('onend'); + dest.end(); + } // when the dest drains, it reduces the awaitDrain counter + // on the source. This would be more elegant with a .once() + // handler in flow(), but adding and removing repeatedly is + // too slow. + + var ondrain = pipeOnDrain(src); + dest.on('drain', ondrain); + var cleanedUp = false; + + function cleanup() { + debug('cleanup'); // cleanup event handlers once the pipe is broken + + dest.removeListener('close', onclose); + dest.removeListener('finish', onfinish); + dest.removeListener('drain', ondrain); + dest.removeListener('error', onerror); + dest.removeListener('unpipe', onunpipe); + src.removeListener('end', onend); + src.removeListener('end', unpipe); + src.removeListener('data', ondata); + cleanedUp = true; // if the reader is waiting for a drain event from this + // specific writer, then it would cause it to never start + // flowing again. + // So, if this is awaiting a drain, then we just call it now. + // If we don't know, then assume that we are waiting for one. + + if (state.awaitDrain && (!dest._writableState || dest._writableState.needDrain)) ondrain(); + } + + src.on('data', ondata); + + function ondata(chunk) { + debug('ondata'); + var ret = dest.write(chunk); + debug('dest.write', ret); + + if (ret === false) { + // If the user unpiped during `dest.write()`, it is possible + // to get stuck in a permanently paused state if that write + // also returned false. + // => Check whether `dest` is still a piping destination. + if (((state.pipesCount === 1 && state.pipes === dest) || (state.pipesCount > 1 && indexOf(state.pipes, dest) !== -1)) && !cleanedUp) { + debug('false write response, pause', state.awaitDrain); + state.awaitDrain++; + } + + src.pause(); + } + } // if the dest has an error, then stop piping into it. + // however, don't suppress the throwing behavior for this. + + function onerror(er) { + debug('onerror', er); + unpipe(); + dest.removeListener('error', onerror); + if (EElistenerCount(dest, 'error') === 0) errorOrDestroy(dest, er); + } // Make sure our error handler is attached before userland ones. + + prependListener(dest, 'error', onerror); // Both close and finish should trigger unpipe, but only once. + + function onclose() { + dest.removeListener('finish', onfinish); + unpipe(); + } + + dest.once('close', onclose); + + function onfinish() { + debug('onfinish'); + dest.removeListener('close', onclose); + unpipe(); + } + + dest.once('finish', onfinish); + + function unpipe() { + debug('unpipe'); + src.unpipe(dest); + } // tell the dest that it's being piped to + + dest.emit('pipe', src); // start the flow if it hasn't been started already. + + if (!state.flowing) { + debug('pipe resume'); + src.resume(); + } + + return dest; + }; + + function pipeOnDrain(src) { + return function pipeOnDrainFunctionResult() { + var state = src._readableState; + debug('pipeOnDrain', state.awaitDrain); + if (state.awaitDrain) state.awaitDrain--; + + if (state.awaitDrain === 0 && EElistenerCount(src, 'data')) { + state.flowing = true; + flow(src); + } + }; + } - case 'pong': - this.setPing(); - this.emit('pong'); - break; + Readable.prototype.unpipe = function (dest) { + var state = this._readableState; + var unpipeInfo = { + hasUnpiped: false + }; // if we're not piping anywhere, then do nothing. + + if (state.pipesCount === 0) return this; // just one destination. most common case. + + if (state.pipesCount === 1) { + // passed in one, but it's not the right one. + if (dest && dest !== state.pipes) return this; + if (!dest) dest = state.pipes; // got a match. + + state.pipes = null; + state.pipesCount = 0; + state.flowing = false; + if (dest) dest.emit('unpipe', this, unpipeInfo); + return this; + } // slow case. multiple pipe destinations. + + if (!dest) { + // remove all. + var dests = state.pipes; + var len = state.pipesCount; + state.pipes = null; + state.pipesCount = 0; + state.flowing = false; + + for (var i = 0; i < len; i++) { + dests[i].emit('unpipe', this, { + hasUnpiped: false + }); + } + + return this; + } // try to find the right one. + + var index = indexOf(state.pipes, dest); + if (index === -1) return this; + state.pipes.splice(index, 1); + state.pipesCount -= 1; + if (state.pipesCount === 1) state.pipes = state.pipes[0]; + dest.emit('unpipe', this, unpipeInfo); + return this; + }; // set up data events if they are asked for + // Ensure readable listeners eventually get something + + Readable.prototype.on = function (ev, fn) { + var res = Stream.prototype.on.call(this, ev, fn); + var state = this._readableState; + + if (ev === 'data') { + // update readableListening so that resume() may be a no-op + // a few lines down. This is needed to support once('readable'). + state.readableListening = this.listenerCount('readable') > 0; // Try start flowing on next tick if stream isn't explicitly paused + + if (state.flowing !== false) this.resume(); + } else if (ev === 'readable') { + if (!state.endEmitted && !state.readableListening) { + state.readableListening = state.needReadable = true; + state.flowing = false; + state.emittedReadable = false; + debug('on readable', state.length, state.reading); + + if (state.length) { + emitReadable(this); + } else if (!state.reading) { + process.nextTick(nReadingNextTick, this); + } + } + } + + return res; + }; + + Readable.prototype.addListener = Readable.prototype.on; + + Readable.prototype.removeListener = function (ev, fn) { + var res = Stream.prototype.removeListener.call(this, ev, fn); + + if (ev === 'readable') { + // We need to check if there is someone still listening to + // readable and reset the state. However this needs to happen + // after readable has been emitted but before I/O (nextTick) to + // support once('readable', fn) cycles. This means that calling + // resume within the same tick will have no + // effect. + process.nextTick(updateReadableListening, this); + } + + return res; + }; + + Readable.prototype.removeAllListeners = function (ev) { + var res = Stream.prototype.removeAllListeners.apply(this, arguments); + + if (ev === 'readable' || ev === undefined) { + // We need to check if there is someone still listening to + // readable and reset the state. However this needs to happen + // after readable has been emitted but before I/O (nextTick) to + // support once('readable', fn) cycles. This means that calling + // resume within the same tick will have no + // effect. + process.nextTick(updateReadableListening, this); + } + + return res; + }; + + function updateReadableListening(self) { + var state = self._readableState; + state.readableListening = self.listenerCount('readable') > 0; + + if (state.resumeScheduled && !state.paused) { + // flowing needs to be set to true now, otherwise + // the upcoming resume will not flow. + state.flowing = true; // crude way to check if we should resume + } else if (self.listenerCount('data') > 0) { + self.resume(); + } + } - case 'error': - var err = new Error('server error'); - err.code = packet.data; - this.onError(err); - break; + function nReadingNextTick(self) { + debug('readable nexttick read 0'); + self.read(0); + } // pause() and resume() are remnants of the legacy readable stream API + // If the user uses them, then switch into old mode. + + Readable.prototype.resume = function () { + var state = this._readableState; + + if (!state.flowing) { + debug('resume'); // we flow only if there is no one listening + // for readable, but we still have to call + // resume() + + state.flowing = !state.readableListening; + resume(this, state); + } + + state.paused = false; + return this; + }; + + function resume(stream, state) { + if (!state.resumeScheduled) { + state.resumeScheduled = true; + process.nextTick(resume_, stream, state); + } + } - case 'message': - this.emit('data', packet.data); - this.emit('message', packet.data); - break; - } - } else { - debug('packet received with socket readyState "%s"', this.readyState); - } -}; + function resume_(stream, state) { + debug('resume', state.reading); -/** - * Called upon handshake completion. - * - * @param {Object} handshake obj - * @api private - */ + if (!state.reading) { + stream.read(0); + } -Socket.prototype.onHandshake = function (data) { - this.emit('handshake', data); - this.id = data.sid; - this.transport.query.sid = data.sid; - this.upgrades = this.filterUpgrades(data.upgrades); - this.pingInterval = data.pingInterval; - this.pingTimeout = data.pingTimeout; - this.onOpen(); - // In case open handler closes socket - if ('closed' === this.readyState) return; - this.setPing(); - - // Prolong liveness of socket on heartbeat - this.removeListener('heartbeat', this.onHeartbeat); - this.on('heartbeat', this.onHeartbeat); -}; - -/** - * Resets ping timeout. - * - * @api private - */ + state.resumeScheduled = false; + stream.emit('resume'); + flow(stream); + if (state.flowing && !state.reading) stream.read(0); + } -Socket.prototype.onHeartbeat = function (timeout) { - clearTimeout(this.pingTimeoutTimer); - var self = this; - self.pingTimeoutTimer = setTimeout(function () { - if ('closed' === self.readyState) return; - self.onClose('ping timeout'); - }, timeout || (self.pingInterval + self.pingTimeout)); -}; - -/** - * Pings server every `this.pingInterval` and expects response - * within `this.pingTimeout` or closes connection. - * - * @api private - */ + Readable.prototype.pause = function () { + debug('call pause flowing=%j', this._readableState.flowing); + + if (this._readableState.flowing !== false) { + debug('pause'); + this._readableState.flowing = false; + this.emit('pause'); + } + + this._readableState.paused = true; + return this; + }; + + function flow(stream) { + var state = stream._readableState; + debug('flow', state.flowing); + + while (state.flowing && stream.read() !== null) {} + } // wrap an old-style stream as the async data source. + // This is *not* part of the readable stream interface. + // It is an ugly unfortunate mess of history. + + Readable.prototype.wrap = function (stream) { + var _this = this; + + var state = this._readableState; + var paused = false; + stream.on('end', function () { + debug('wrapped end'); + + if (state.decoder && !state.ended) { + var chunk = state.decoder.end(); + if (chunk && chunk.length) _this.push(chunk); + } + + _this.push(null); + }); + stream.on('data', function (chunk) { + debug('wrapped data'); + if (state.decoder) chunk = state.decoder.write(chunk); // don't skip over falsy values in objectMode + + if (state.objectMode && (chunk === null || chunk === undefined)) return; + else if (!state.objectMode && (!chunk || !chunk.length)) return; + + var ret = _this.push(chunk); + + if (!ret) { + paused = true; + stream.pause(); + } + }); // proxy all the other methods. + // important when wrapping filters and duplexes. + + for (var i in stream) { + if (this[i] === undefined && typeof stream[i] === 'function') { + this[i] = (function methodWrap(method) { + return function methodWrapReturnFunction() { + return stream[method].apply(stream, arguments); + }; + })(i); + } + } // proxy certain important events. + + for (var n = 0; n < kProxyEvents.length; n++) { + stream.on(kProxyEvents[n], this.emit.bind(this, kProxyEvents[n])); + } // when we try to consume some more bytes, simply unpause the + // underlying stream. + + this._read = function (n) { + debug('wrapped _read', n); + + if (paused) { + paused = false; + stream.resume(); + } + }; + + return this; + }; + + if (typeof Symbol === 'function') { + Readable.prototype[Symbol.asyncIterator] = function () { + if (createReadableStreamAsyncIterator === undefined) { + createReadableStreamAsyncIterator = require('./internal/streams/async_iterator'); + } + + return createReadableStreamAsyncIterator(this); + }; + } -Socket.prototype.setPing = function () { - var self = this; - clearTimeout(self.pingIntervalTimer); - self.pingIntervalTimer = setTimeout(function () { - debug('writing ping packet - expecting pong within %sms', self.pingTimeout); - self.ping(); - self.onHeartbeat(self.pingTimeout); - }, self.pingInterval); -}; - -/** -* Sends a ping packet. -* -* @api private -*/ - -Socket.prototype.ping = function () { - var self = this; - this.sendPacket('ping', function () { - self.emit('ping'); - }); -}; - -/** - * Called on `drain` event - * - * @api private - */ + Object.defineProperty(Readable.prototype, 'readableHighWaterMark', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._readableState.highWaterMark; + } + }); + Object.defineProperty(Readable.prototype, 'readableBuffer', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._readableState && this._readableState.buffer; + } + }); + Object.defineProperty(Readable.prototype, 'readableFlowing', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._readableState.flowing; + }, + set: function set(state) { + if (this._readableState) { + this._readableState.flowing = state; + } + } + }); // exposed for testing purposes only. + + Readable._fromList = fromList; + Object.defineProperty(Readable.prototype, 'readableLength', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._readableState.length; + } + }); // Pluck off n bytes from an array of buffers. + // Length is the combined lengths of all the buffers in the list. + // This function is designed to be inlinable, so please take care when making + // changes to the function body. + + function fromList(n, state) { + // nothing buffered + if (state.length === 0) return null; + var ret; + if (state.objectMode) ret = state.buffer.shift(); + else if (!n || n >= state.length) { + // read it all, truncate the list + if (state.decoder) ret = state.buffer.join(''); + else if (state.buffer.length === 1) ret = state.buffer.first(); + else ret = state.buffer.concat(state.length); + state.buffer.clear(); + } else { + // read part of list + ret = state.buffer.consume(n, state.decoder); + } + return ret; + } -Socket.prototype.onDrain = function () { - this.writeBuffer.splice(0, this.prevBufferLen); + function endReadable(stream) { + var state = stream._readableState; + debug('endReadable', state.endEmitted); - // setting prevBufferLen = 0 is very important - // for example, when upgrading, upgrade packet is sent over, - // and a nonzero prevBufferLen could cause problems on `drain` - this.prevBufferLen = 0; + if (!state.endEmitted) { + state.ended = true; + process.nextTick(endReadableNT, state, stream); + } + } - if (0 === this.writeBuffer.length) { - this.emit('drain'); - } else { - this.flush(); - } -}; + function endReadableNT(state, stream) { + debug('endReadableNT', state.endEmitted, state.length); // Check that we didn't get one last unshift. -/** - * Flush write buffers. - * - * @api private - */ + if (!state.endEmitted && state.length === 0) { + state.endEmitted = true; + stream.readable = false; + stream.emit('end'); -Socket.prototype.flush = function () { - if ('closed' !== this.readyState && this.transport.writable && - !this.upgrading && this.writeBuffer.length) { - debug('flushing %d packets in socket', this.writeBuffer.length); - this.transport.send(this.writeBuffer); - // keep track of current length of writeBuffer - // splice writeBuffer and callbackBuffer on `drain` - this.prevBufferLen = this.writeBuffer.length; - this.emit('flush'); - } -}; + if (state.autoDestroy) { + // In case of duplex streams we need a way to detect + // if the writable side is ready for autoDestroy as well + var wState = stream._writableState; + + if (!wState || (wState.autoDestroy && wState.finished)) { + stream.destroy(); + } + } + } + } + + if (typeof Symbol === 'function') { + Readable.from = function (iterable, opts) { + if (from === undefined) { + from = require('./internal/streams/from'); + } + + return from(Readable, iterable, opts); + }; + } + + function indexOf(xs, x) { + for (var i = 0, l = xs.length; i < l; i++) { + if (xs[i] === x) return i; + } + + return -1; + } + }).call(this, require('_process'), typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : typeof window !== 'undefined' ? window : {}); + }, + { + '../errors': 107, + './_stream_duplex': 108, + './internal/streams/async_iterator': 113, + './internal/streams/buffer_list': 114, + './internal/streams/destroy': 115, + './internal/streams/from': 117, + './internal/streams/state': 119, + './internal/streams/stream': 120, + _process: 244, + buffer: 124, + events: 181, + inherits: 106, + 'string_decoder/': 292, + util: 77 + } + ], + 111: [ + function (require, module, exports) { + // Copyright Joyent, Inc. and other Node contributors. + // + // Permission is hereby granted, free of charge, to any person obtaining a + // copy of this software and associated documentation files (the + // "Software"), to deal in the Software without restriction, including + // without limitation the rights to use, copy, modify, merge, publish, + // distribute, sublicense, and/or sell copies of the Software, and to permit + // persons to whom the Software is furnished to do so, subject to the + // following conditions: + // + // The above copyright notice and this permission notice shall be included + // in all copies or substantial portions of the Software. + // + // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS + // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF + // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN + // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, + // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR + // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE + // USE OR OTHER DEALINGS IN THE SOFTWARE. + // a transform stream is a readable/writable stream where you do + // something with the data. Sometimes it's called a "filter", + // but that's not a great name for it, since that implies a thing where + // some bits pass through, and others are simply ignored. (That would + // be a valid example of a transform, of course.) + // + // While the output is causally related to the input, it's not a + // necessarily symmetric or synchronous transformation. For example, + // a zlib stream might take multiple plain-text writes(), and then + // emit a single compressed chunk some time in the future. + // + // Here's how this works: + // + // The Transform stream has all the aspects of the readable and writable + // stream classes. When you write(chunk), that calls _write(chunk,cb) + // internally, and returns false if there's a lot of pending writes + // buffered up. When you call read(), that calls _read(n) until + // there's enough pending readable data buffered up. + // + // In a transform stream, the written data is placed in a buffer. When + // _read(n) is called, it transforms the queued up data, calling the + // buffered _write cb's as it consumes chunks. If consuming a single + // written chunk would result in multiple output chunks, then the first + // outputted bit calls the readcb, and subsequent chunks just go into + // the read buffer, and will cause it to emit 'readable' if necessary. + // + // This way, back-pressure is actually determined by the reading side, + // since _read has to be called to start processing a new chunk. However, + // a pathological inflate type of transform can cause excessive buffering + // here. For example, imagine a stream where every byte of input is + // interpreted as an integer from 0-255, and then results in that many + // bytes of output. Writing the 4 bytes {ff,ff,ff,ff} would result in + // 1kb of data being output. In this case, you could write a very small + // amount of input, and end up with a very large amount of output. In + // such a pathological inflating mechanism, there'd be no way to tell + // the system to stop doing the transform. A single 4MB write could + // cause the system to run out of memory. + // + // However, even in such a pathological case, only a single written chunk + // would be consumed, and then the rest would wait (un-transformed) until + // the results of the previous transformed chunk were consumed. + 'use strict'; + + module.exports = Transform; + + var _require$codes = require('../errors').codes, + ERR_METHOD_NOT_IMPLEMENTED = _require$codes.ERR_METHOD_NOT_IMPLEMENTED, + ERR_MULTIPLE_CALLBACK = _require$codes.ERR_MULTIPLE_CALLBACK, + ERR_TRANSFORM_ALREADY_TRANSFORMING = _require$codes.ERR_TRANSFORM_ALREADY_TRANSFORMING, + ERR_TRANSFORM_WITH_LENGTH_0 = _require$codes.ERR_TRANSFORM_WITH_LENGTH_0; + + var Duplex = require('./_stream_duplex'); + + require('inherits')(Transform, Duplex); + + function afterTransform(er, data) { + var ts = this._transformState; + ts.transforming = false; + var cb = ts.writecb; + + if (cb === null) { + return this.emit('error', new ERR_MULTIPLE_CALLBACK()); + } + + ts.writechunk = null; + ts.writecb = null; + if (data != null) + // single equals check for both `null` and `undefined` + this.push(data); + cb(er); + var rs = this._readableState; + rs.reading = false; + + if (rs.needReadable || rs.length < rs.highWaterMark) { + this._read(rs.highWaterMark); + } + } + + function Transform(options) { + if (!(this instanceof Transform)) return new Transform(options); + Duplex.call(this, options); + this._transformState = { + afterTransform: afterTransform.bind(this), + needTransform: false, + transforming: false, + writecb: null, + writechunk: null, + writeencoding: null + }; // start out asking for a readable event once data is transformed. + + this._readableState.needReadable = true; // we have implemented the _read method, and done the other things + // that Readable wants before the first _read call, so unset the + // sync guard flag. + + this._readableState.sync = false; + + if (options) { + if (typeof options.transform === 'function') this._transform = options.transform; + if (typeof options.flush === 'function') this._flush = options.flush; + } // When the writable side finishes, then flush out anything remaining. + + this.on('prefinish', prefinish); + } + + function prefinish() { + var _this = this; + + if (typeof this._flush === 'function' && !this._readableState.destroyed) { + this._flush(function (er, data) { + done(_this, er, data); + }); + } else { + done(this, null, null); + } + } + + Transform.prototype.push = function (chunk, encoding) { + this._transformState.needTransform = false; + return Duplex.prototype.push.call(this, chunk, encoding); + }; // This is the part where you do stuff! + // override this function in implementation classes. + // 'chunk' is an input chunk. + // + // Call `push(newChunk)` to pass along transformed output + // to the readable side. You may call 'push' zero or more times. + // + // Call `cb(err)` when you are done with this chunk. If you pass + // an error, then that'll put the hurt on the whole operation. If you + // never call cb(), then you'll never get another chunk. + + Transform.prototype._transform = function (chunk, encoding, cb) { + cb(new ERR_METHOD_NOT_IMPLEMENTED('_transform()')); + }; + + Transform.prototype._write = function (chunk, encoding, cb) { + var ts = this._transformState; + ts.writecb = cb; + ts.writechunk = chunk; + ts.writeencoding = encoding; + + if (!ts.transforming) { + var rs = this._readableState; + if (ts.needTransform || rs.needReadable || rs.length < rs.highWaterMark) this._read(rs.highWaterMark); + } + }; // Doesn't matter what the args are here. + // _transform does all the work. + // That we got here means that the readable side wants more data. + + Transform.prototype._read = function (n) { + var ts = this._transformState; + + if (ts.writechunk !== null && !ts.transforming) { + ts.transforming = true; + + this._transform(ts.writechunk, ts.writeencoding, ts.afterTransform); + } else { + // mark that we need a transform, so that any data that comes in + // will get processed, now that we've asked for it. + ts.needTransform = true; + } + }; + + Transform.prototype._destroy = function (err, cb) { + Duplex.prototype._destroy.call(this, err, function (err2) { + cb(err2); + }); + }; + + function done(stream, er, data) { + if (er) return stream.emit('error', er); + if (data != null) + // single equals check for both `null` and `undefined` + stream.push(data); // TODO(BridgeAR): Write a test for these two error cases + // if there's nothing in the write buffer, then that means + // that nothing more will ever be provided + + if (stream._writableState.length) throw new ERR_TRANSFORM_WITH_LENGTH_0(); + if (stream._transformState.transforming) throw new ERR_TRANSFORM_ALREADY_TRANSFORMING(); + return stream.push(null); + } + }, + { '../errors': 107, './_stream_duplex': 108, inherits: 106 } + ], + 112: [ + function (require, module, exports) { + (function (process, global) { + // Copyright Joyent, Inc. and other Node contributors. + // + // Permission is hereby granted, free of charge, to any person obtaining a + // copy of this software and associated documentation files (the + // "Software"), to deal in the Software without restriction, including + // without limitation the rights to use, copy, modify, merge, publish, + // distribute, sublicense, and/or sell copies of the Software, and to permit + // persons to whom the Software is furnished to do so, subject to the + // following conditions: + // + // The above copyright notice and this permission notice shall be included + // in all copies or substantial portions of the Software. + // + // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS + // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF + // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN + // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, + // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR + // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE + // USE OR OTHER DEALINGS IN THE SOFTWARE. + // A bit simpler than readable streams. + // Implement an async ._write(chunk, encoding, cb), and it'll handle all + // the drain event emission and buffering. + 'use strict'; + + module.exports = Writable; + /* */ + + function WriteReq(chunk, encoding, cb) { + this.chunk = chunk; + this.encoding = encoding; + this.callback = cb; + this.next = null; + } // It seems a linked list but it is not + // there will be only 2 of these for each stream + + function CorkedRequest(state) { + var _this = this; + + this.next = null; + this.entry = null; + + this.finish = function () { + onCorkedFinish(_this, state); + }; + } + /* */ + + /**/ + + var Duplex; + /**/ + + Writable.WritableState = WritableState; + /**/ -/** - * Sends a message. - * - * @param {String} message. - * @param {Function} callback function. - * @param {Object} options. - * @return {Socket} for chaining. - * @api public - */ + var internalUtil = { + deprecate: require('util-deprecate') + }; + /**/ -Socket.prototype.write = -Socket.prototype.send = function (msg, options, fn) { - this.sendPacket('message', msg, options, fn); - return this; -}; + /**/ -/** - * Sends a packet. - * - * @param {String} packet type. - * @param {String} data. - * @param {Object} options. - * @param {Function} callback function. - * @api private - */ + var Stream = require('./internal/streams/stream'); + /**/ -Socket.prototype.sendPacket = function (type, data, options, fn) { - if ('function' === typeof data) { - fn = data; - data = undefined; - } + var Buffer = require('buffer').Buffer; - if ('function' === typeof options) { - fn = options; - options = null; - } + var OurUint8Array = global.Uint8Array || function () {}; - if ('closing' === this.readyState || 'closed' === this.readyState) { - return; - } + function _uint8ArrayToBuffer(chunk) { + return Buffer.from(chunk); + } - options = options || {}; - options.compress = false !== options.compress; + function _isUint8Array(obj) { + return Buffer.isBuffer(obj) || obj instanceof OurUint8Array; + } - var packet = { - type: type, - data: data, - options: options - }; - this.emit('packetCreate', packet); - this.writeBuffer.push(packet); - if (fn) this.once('flush', fn); - this.flush(); -}; - -/** - * Closes the connection. - * - * @api private - */ + var destroyImpl = require('./internal/streams/destroy'); -Socket.prototype.close = function () { - if ('opening' === this.readyState || 'open' === this.readyState) { - this.readyState = 'closing'; + var _require = require('./internal/streams/state'), + getHighWaterMark = _require.getHighWaterMark; - var self = this; + var _require$codes = require('../errors').codes, + ERR_INVALID_ARG_TYPE = _require$codes.ERR_INVALID_ARG_TYPE, + ERR_METHOD_NOT_IMPLEMENTED = _require$codes.ERR_METHOD_NOT_IMPLEMENTED, + ERR_MULTIPLE_CALLBACK = _require$codes.ERR_MULTIPLE_CALLBACK, + ERR_STREAM_CANNOT_PIPE = _require$codes.ERR_STREAM_CANNOT_PIPE, + ERR_STREAM_DESTROYED = _require$codes.ERR_STREAM_DESTROYED, + ERR_STREAM_NULL_VALUES = _require$codes.ERR_STREAM_NULL_VALUES, + ERR_STREAM_WRITE_AFTER_END = _require$codes.ERR_STREAM_WRITE_AFTER_END, + ERR_UNKNOWN_ENCODING = _require$codes.ERR_UNKNOWN_ENCODING; - if (this.writeBuffer.length) { - this.once('drain', function () { - if (this.upgrading) { - waitForUpgrade(); - } else { - close(); - } - }); - } else if (this.upgrading) { - waitForUpgrade(); - } else { - close(); - } - } + var errorOrDestroy = destroyImpl.errorOrDestroy; - function close () { - self.onClose('forced close'); - debug('socket closing - telling transport to close'); - self.transport.close(); - } + require('inherits')(Writable, Stream); - function cleanupAndClose () { - self.removeListener('upgrade', cleanupAndClose); - self.removeListener('upgradeError', cleanupAndClose); - close(); - } + function nop() {} - function waitForUpgrade () { - // wait for upgrade to finish since we can't send packets while pausing a transport - self.once('upgrade', cleanupAndClose); - self.once('upgradeError', cleanupAndClose); - } + function WritableState(options, stream, isDuplex) { + Duplex = Duplex || require('./_stream_duplex'); + options = options || {}; // Duplex streams are both readable and writable, but share + // the same options object. + // However, some cases require setting options to different + // values for the readable and the writable sides of the duplex stream, + // e.g. options.readableObjectMode vs. options.writableObjectMode, etc. - return this; -}; + if (typeof isDuplex !== 'boolean') isDuplex = stream instanceof Duplex; // object stream flag to indicate whether or not this stream + // contains buffers or objects. -/** - * Called upon transport error - * - * @api private - */ + this.objectMode = !!options.objectMode; + if (isDuplex) this.objectMode = this.objectMode || !!options.writableObjectMode; // the point at which write() starts returning false + // Note: 0 is a valid value, means that we always return false if + // the entire buffer is not flushed immediately on write() -Socket.prototype.onError = function (err) { - debug('socket error %j', err); - Socket.priorWebsocketSuccess = false; - this.emit('error', err); - this.onClose('transport error', err); -}; + this.highWaterMark = getHighWaterMark(this, options, 'writableHighWaterMark', isDuplex); // if _final has been called -/** - * Called upon transport close. - * - * @api private - */ + this.finalCalled = false; // drain event flag. -Socket.prototype.onClose = function (reason, desc) { - if ('opening' === this.readyState || 'open' === this.readyState || 'closing' === this.readyState) { - debug('socket close with reason: "%s"', reason); - var self = this; + this.needDrain = false; // at the start of calling end() - // clear timers - clearTimeout(this.pingIntervalTimer); - clearTimeout(this.pingTimeoutTimer); + this.ending = false; // when end() has been called, and returned - // stop event from firing again for transport - this.transport.removeAllListeners('close'); + this.ended = false; // when 'finish' is emitted - // ensure transport won't stay open - this.transport.close(); + this.finished = false; // has it been destroyed - // ignore further transport communication - this.transport.removeAllListeners(); + this.destroyed = false; // should we decode strings into buffers before passing to _write? + // this is here so that some node-core streams can optimize string + // handling at a lower level. - // set ready state - this.readyState = 'closed'; + var noDecode = options.decodeStrings === false; + this.decodeStrings = !noDecode; // Crypto is kind of old and crusty. Historically, its default string + // encoding is 'binary' so we have to make this configurable. + // Everything else in the universe uses 'utf8', though. - // clear session id - this.id = null; + this.defaultEncoding = options.defaultEncoding || 'utf8'; // not an actual buffer we keep track of, but a measurement + // of how much we're waiting to get pushed to some underlying + // socket or file. - // emit close event - this.emit('close', reason, desc); + this.length = 0; // a flag to see when we're in the middle of a write. - // clean buffers after, so users can still - // grab the buffers on `close` event - self.writeBuffer = []; - self.prevBufferLen = 0; - } -}; + this.writing = false; // when true all writes will be buffered until .uncork() call -/** - * Filters upgrades, returning only those matching client transports. - * - * @param {Array} server upgrades - * @api private - * - */ + this.corked = 0; // a flag to be able to tell if the onwrite cb is called immediately, + // or on a later tick. We set this to true at first, because any + // actions that shouldn't happen until "later" should generally also + // not happen before the first write call. -Socket.prototype.filterUpgrades = function (upgrades) { - var filteredUpgrades = []; - for (var i = 0, j = upgrades.length; i < j; i++) { - if (~index(this.transports, upgrades[i])) filteredUpgrades.push(upgrades[i]); - } - return filteredUpgrades; -}; + this.sync = true; // a flag to know if we're processing previously buffered items, which + // may call the _write() callback in the same tick, so that we don't + // end up in an overlapped onwrite situation. -},{"./transport":168,"./transports/index":169,"component-emitter":175,"debug":176,"engine.io-parser":178,"indexof":217,"parseqs":234,"parseuri":235}],168:[function(require,module,exports){ -/** - * Module dependencies. - */ + this.bufferProcessing = false; // the callback that's passed to _write(chunk,cb) -var parser = require('engine.io-parser'); -var Emitter = require('component-emitter'); + this.onwrite = function (er) { + onwrite(stream, er); + }; // the callback that the user supplies to write(chunk,encoding,cb) -/** - * Module exports. - */ + this.writecb = null; // the amount that is being written when _write is called. -module.exports = Transport; + this.writelen = 0; + this.bufferedRequest = null; + this.lastBufferedRequest = null; // number of pending user-supplied write callbacks + // this must be 0 before 'finish' can be emitted -/** - * Transport abstract constructor. - * - * @param {Object} options. - * @api private - */ + this.pendingcb = 0; // emit prefinish if the only thing we're waiting for is _write cbs + // This is relevant for synchronous Transform streams -function Transport (opts) { - this.path = opts.path; - this.hostname = opts.hostname; - this.port = opts.port; - this.secure = opts.secure; - this.query = opts.query; - this.timestampParam = opts.timestampParam; - this.timestampRequests = opts.timestampRequests; - this.readyState = ''; - this.agent = opts.agent || false; - this.socket = opts.socket; - this.enablesXDR = opts.enablesXDR; - this.withCredentials = opts.withCredentials; - - // SSL options for Node.js client - this.pfx = opts.pfx; - this.key = opts.key; - this.passphrase = opts.passphrase; - this.cert = opts.cert; - this.ca = opts.ca; - this.ciphers = opts.ciphers; - this.rejectUnauthorized = opts.rejectUnauthorized; - this.forceNode = opts.forceNode; - - // results of ReactNative environment detection - this.isReactNative = opts.isReactNative; - - // other options for Node.js client - this.extraHeaders = opts.extraHeaders; - this.localAddress = opts.localAddress; -} - -/** - * Mix in `Emitter`. - */ + this.prefinished = false; // True if the error was already emitted and should not be thrown again -Emitter(Transport.prototype); + this.errorEmitted = false; // Should close be emitted on destroy. Defaults to true. -/** - * Emits an error. - * - * @param {String} str - * @return {Transport} for chaining - * @api public - */ + this.emitClose = options.emitClose !== false; // Should .destroy() be called after 'finish' (and potentially 'end') -Transport.prototype.onError = function (msg, desc) { - var err = new Error(msg); - err.type = 'TransportError'; - err.description = desc; - this.emit('error', err); - return this; -}; + this.autoDestroy = !!options.autoDestroy; // count buffered requests -/** - * Opens the transport. - * - * @api public - */ + this.bufferedRequestCount = 0; // allocate the first CorkedRequest, there is always + // one allocated and free to use, and we maintain at most two -Transport.prototype.open = function () { - if ('closed' === this.readyState || '' === this.readyState) { - this.readyState = 'opening'; - this.doOpen(); - } + this.corkedRequestsFree = new CorkedRequest(this); + } - return this; -}; + WritableState.prototype.getBuffer = function getBuffer() { + var current = this.bufferedRequest; + var out = []; + + while (current) { + out.push(current); + current = current.next; + } + + return out; + }; + + (function () { + try { + Object.defineProperty(WritableState.prototype, 'buffer', { + get: internalUtil.deprecate( + function writableStateBufferGetter() { + return this.getBuffer(); + }, + '_writableState.buffer is deprecated. Use _writableState.getBuffer ' + 'instead.', + 'DEP0003' + ) + }); + } catch (_) {} + })(); // Test _writableState for inheritance to account for Duplex streams, + // whose prototype chain only points to Readable. + + var realHasInstance; + + if (typeof Symbol === 'function' && Symbol.hasInstance && typeof Function.prototype[Symbol.hasInstance] === 'function') { + realHasInstance = Function.prototype[Symbol.hasInstance]; + Object.defineProperty(Writable, Symbol.hasInstance, { + value: function value(object) { + if (realHasInstance.call(this, object)) return true; + if (this !== Writable) return false; + return object && object._writableState instanceof WritableState; + } + }); + } else { + realHasInstance = function realHasInstance(object) { + return object instanceof this; + }; + } -/** - * Closes the transport. - * - * @api private - */ + function Writable(options) { + Duplex = Duplex || require('./_stream_duplex'); // Writable ctor is applied to Duplexes, too. + // `realHasInstance` is necessary because using plain `instanceof` + // would return false, as no `_writableState` property is attached. + // Trying to use the custom `instanceof` for Writable here will also break the + // Node.js LazyTransform implementation, which has a non-trivial getter for + // `_writableState` that would lead to infinite recursion. + // Checking for a Stream.Duplex instance is faster here instead of inside + // the WritableState constructor, at least with V8 6.5 + + var isDuplex = this instanceof Duplex; + if (!isDuplex && !realHasInstance.call(Writable, this)) return new Writable(options); + this._writableState = new WritableState(options, this, isDuplex); // legacy. + + this.writable = true; + + if (options) { + if (typeof options.write === 'function') this._write = options.write; + if (typeof options.writev === 'function') this._writev = options.writev; + if (typeof options.destroy === 'function') this._destroy = options.destroy; + if (typeof options.final === 'function') this._final = options.final; + } + + Stream.call(this); + } // Otherwise people can pipe Writable streams, which is just wrong. + + Writable.prototype.pipe = function () { + errorOrDestroy(this, new ERR_STREAM_CANNOT_PIPE()); + }; + + function writeAfterEnd(stream, cb) { + var er = new ERR_STREAM_WRITE_AFTER_END(); // TODO: defer error events consistently everywhere, not just the cb + + errorOrDestroy(stream, er); + process.nextTick(cb, er); + } // Checks that a user-supplied chunk is valid, especially for the particular + // mode the stream is in. Currently this means that `null` is never accepted + // and undefined/non-string values are only allowed in object mode. + + function validChunk(stream, state, chunk, cb) { + var er; + + if (chunk === null) { + er = new ERR_STREAM_NULL_VALUES(); + } else if (typeof chunk !== 'string' && !state.objectMode) { + er = new ERR_INVALID_ARG_TYPE('chunk', ['string', 'Buffer'], chunk); + } + + if (er) { + errorOrDestroy(stream, er); + process.nextTick(cb, er); + return false; + } + + return true; + } -Transport.prototype.close = function () { - if ('opening' === this.readyState || 'open' === this.readyState) { - this.doClose(); - this.onClose(); - } + Writable.prototype.write = function (chunk, encoding, cb) { + var state = this._writableState; + var ret = false; + + var isBuf = !state.objectMode && _isUint8Array(chunk); + + if (isBuf && !Buffer.isBuffer(chunk)) { + chunk = _uint8ArrayToBuffer(chunk); + } + + if (typeof encoding === 'function') { + cb = encoding; + encoding = null; + } + + if (isBuf) encoding = 'buffer'; + else if (!encoding) encoding = state.defaultEncoding; + if (typeof cb !== 'function') cb = nop; + if (state.ending) writeAfterEnd(this, cb); + else if (isBuf || validChunk(this, state, chunk, cb)) { + state.pendingcb++; + ret = writeOrBuffer(this, state, isBuf, chunk, encoding, cb); + } + return ret; + }; + + Writable.prototype.cork = function () { + this._writableState.corked++; + }; + + Writable.prototype.uncork = function () { + var state = this._writableState; + + if (state.corked) { + state.corked--; + if (!state.writing && !state.corked && !state.bufferProcessing && state.bufferedRequest) clearBuffer(this, state); + } + }; + + Writable.prototype.setDefaultEncoding = function setDefaultEncoding(encoding) { + // node::ParseEncoding() requires lower case. + if (typeof encoding === 'string') encoding = encoding.toLowerCase(); + if (!(['hex', 'utf8', 'utf-8', 'ascii', 'binary', 'base64', 'ucs2', 'ucs-2', 'utf16le', 'utf-16le', 'raw'].indexOf((encoding + '').toLowerCase()) > -1)) + throw new ERR_UNKNOWN_ENCODING(encoding); + this._writableState.defaultEncoding = encoding; + return this; + }; + + Object.defineProperty(Writable.prototype, 'writableBuffer', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._writableState && this._writableState.getBuffer(); + } + }); + + function decodeChunk(state, chunk, encoding) { + if (!state.objectMode && state.decodeStrings !== false && typeof chunk === 'string') { + chunk = Buffer.from(chunk, encoding); + } + + return chunk; + } - return this; -}; + Object.defineProperty(Writable.prototype, 'writableHighWaterMark', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._writableState.highWaterMark; + } + }); // if we're already writing something, then just put this + // in the queue, and wait our turn. Otherwise, call _write + // If we return false, then we need a drain event, so set that flag. + + function writeOrBuffer(stream, state, isBuf, chunk, encoding, cb) { + if (!isBuf) { + var newChunk = decodeChunk(state, chunk, encoding); + + if (chunk !== newChunk) { + isBuf = true; + encoding = 'buffer'; + chunk = newChunk; + } + } + + var len = state.objectMode ? 1 : chunk.length; + state.length += len; + var ret = state.length < state.highWaterMark; // we must ensure that previous needDrain will not be reset to false. + + if (!ret) state.needDrain = true; + + if (state.writing || state.corked) { + var last = state.lastBufferedRequest; + state.lastBufferedRequest = { + chunk: chunk, + encoding: encoding, + isBuf: isBuf, + callback: cb, + next: null + }; + + if (last) { + last.next = state.lastBufferedRequest; + } else { + state.bufferedRequest = state.lastBufferedRequest; + } + + state.bufferedRequestCount += 1; + } else { + doWrite(stream, state, false, len, chunk, encoding, cb); + } + + return ret; + } -/** - * Sends multiple packets. - * - * @param {Array} packets - * @api private - */ + function doWrite(stream, state, writev, len, chunk, encoding, cb) { + state.writelen = len; + state.writecb = cb; + state.writing = true; + state.sync = true; + if (state.destroyed) state.onwrite(new ERR_STREAM_DESTROYED('write')); + else if (writev) stream._writev(chunk, state.onwrite); + else stream._write(chunk, encoding, state.onwrite); + state.sync = false; + } -Transport.prototype.send = function (packets) { - if ('open' === this.readyState) { - this.write(packets); - } else { - throw new Error('Transport not open'); - } -}; + function onwriteError(stream, state, sync, er, cb) { + --state.pendingcb; + + if (sync) { + // defer the callback if we are being called synchronously + // to avoid piling up things on the stack + process.nextTick(cb, er); // this can emit finish, and it will always happen + // after error + + process.nextTick(finishMaybe, stream, state); + stream._writableState.errorEmitted = true; + errorOrDestroy(stream, er); + } else { + // the caller expect this to happen before if + // it is async + cb(er); + stream._writableState.errorEmitted = true; + errorOrDestroy(stream, er); // this can emit finish, but finish must + // always follow error + + finishMaybe(stream, state); + } + } -/** - * Called upon open - * - * @api private - */ + function onwriteStateUpdate(state) { + state.writing = false; + state.writecb = null; + state.length -= state.writelen; + state.writelen = 0; + } -Transport.prototype.onOpen = function () { - this.readyState = 'open'; - this.writable = true; - this.emit('open'); -}; + function onwrite(stream, er) { + var state = stream._writableState; + var sync = state.sync; + var cb = state.writecb; + if (typeof cb !== 'function') throw new ERR_MULTIPLE_CALLBACK(); + onwriteStateUpdate(state); + if (er) onwriteError(stream, state, sync, er, cb); + else { + // Check if we're actually ready to finish, but don't emit yet + var finished = needFinish(state) || stream.destroyed; + + if (!finished && !state.corked && !state.bufferProcessing && state.bufferedRequest) { + clearBuffer(stream, state); + } + + if (sync) { + process.nextTick(afterWrite, stream, state, finished, cb); + } else { + afterWrite(stream, state, finished, cb); + } + } + } -/** - * Called with data. - * - * @param {String} data - * @api private - */ + function afterWrite(stream, state, finished, cb) { + if (!finished) onwriteDrain(stream, state); + state.pendingcb--; + cb(); + finishMaybe(stream, state); + } // Must force callback to be called on nextTick, so that we don't + // emit 'drain' before the write() consumer gets the 'false' return + // value, and has a chance to attach a 'drain' listener. + + function onwriteDrain(stream, state) { + if (state.length === 0 && state.needDrain) { + state.needDrain = false; + stream.emit('drain'); + } + } // if there's something in the buffer waiting, then process it + + function clearBuffer(stream, state) { + state.bufferProcessing = true; + var entry = state.bufferedRequest; + + if (stream._writev && entry && entry.next) { + // Fast case, write everything using _writev() + var l = state.bufferedRequestCount; + var buffer = new Array(l); + var holder = state.corkedRequestsFree; + holder.entry = entry; + var count = 0; + var allBuffers = true; + + while (entry) { + buffer[count] = entry; + if (!entry.isBuf) allBuffers = false; + entry = entry.next; + count += 1; + } + + buffer.allBuffers = allBuffers; + doWrite(stream, state, true, state.length, buffer, '', holder.finish); // doWrite is almost always async, defer these to save a bit of time + // as the hot path ends with doWrite + + state.pendingcb++; + state.lastBufferedRequest = null; + + if (holder.next) { + state.corkedRequestsFree = holder.next; + holder.next = null; + } else { + state.corkedRequestsFree = new CorkedRequest(state); + } + + state.bufferedRequestCount = 0; + } else { + // Slow case, write chunks one-by-one + while (entry) { + var chunk = entry.chunk; + var encoding = entry.encoding; + var cb = entry.callback; + var len = state.objectMode ? 1 : chunk.length; + doWrite(stream, state, false, len, chunk, encoding, cb); + entry = entry.next; + state.bufferedRequestCount--; // if we didn't call the onwrite immediately, then + // it means that we need to wait until it does. + // also, that means that the chunk and cb are currently + // being processed, so move the buffer counter past them. + + if (state.writing) { + break; + } + } + + if (entry === null) state.lastBufferedRequest = null; + } + + state.bufferedRequest = entry; + state.bufferProcessing = false; + } -Transport.prototype.onData = function (data) { - var packet = parser.decodePacket(data, this.socket.binaryType); - this.onPacket(packet); -}; + Writable.prototype._write = function (chunk, encoding, cb) { + cb(new ERR_METHOD_NOT_IMPLEMENTED('_write()')); + }; + + Writable.prototype._writev = null; + + Writable.prototype.end = function (chunk, encoding, cb) { + var state = this._writableState; + + if (typeof chunk === 'function') { + cb = chunk; + chunk = null; + encoding = null; + } else if (typeof encoding === 'function') { + cb = encoding; + encoding = null; + } + + if (chunk !== null && chunk !== undefined) this.write(chunk, encoding); // .end() fully uncorks + + if (state.corked) { + state.corked = 1; + this.uncork(); + } // ignore unnecessary end() calls. + + if (!state.ending) endWritable(this, state, cb); + return this; + }; + + Object.defineProperty(Writable.prototype, 'writableLength', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + return this._writableState.length; + } + }); + + function needFinish(state) { + return state.ending && state.length === 0 && state.bufferedRequest === null && !state.finished && !state.writing; + } -/** - * Called with a decoded packet. - */ + function callFinal(stream, state) { + stream._final(function (err) { + state.pendingcb--; -Transport.prototype.onPacket = function (packet) { - this.emit('packet', packet); -}; + if (err) { + errorOrDestroy(stream, err); + } -/** - * Called upon close. - * - * @api private - */ + state.prefinished = true; + stream.emit('prefinish'); + finishMaybe(stream, state); + }); + } -Transport.prototype.onClose = function () { - this.readyState = 'closed'; - this.emit('close'); -}; + function prefinish(stream, state) { + if (!state.prefinished && !state.finalCalled) { + if (typeof stream._final === 'function' && !state.destroyed) { + state.pendingcb++; + state.finalCalled = true; + process.nextTick(callFinal, stream, state); + } else { + state.prefinished = true; + stream.emit('prefinish'); + } + } + } -},{"component-emitter":175,"engine.io-parser":178}],169:[function(require,module,exports){ -/** - * Module dependencies - */ + function finishMaybe(stream, state) { + var need = needFinish(state); -var XMLHttpRequest = require('xmlhttprequest-ssl'); -var XHR = require('./polling-xhr'); -var JSONP = require('./polling-jsonp'); -var websocket = require('./websocket'); + if (need) { + prefinish(stream, state); -/** - * Export transports. - */ + if (state.pendingcb === 0) { + state.finished = true; + stream.emit('finish'); -exports.polling = polling; -exports.websocket = websocket; + if (state.autoDestroy) { + // In case of duplex streams we need a way to detect + // if the readable side is ready for autoDestroy as well + var rState = stream._readableState; -/** - * Polling transport polymorphic constructor. - * Decides on xhr vs jsonp based on feature detection. - * - * @api private - */ + if (!rState || (rState.autoDestroy && rState.endEmitted)) { + stream.destroy(); + } + } + } + } -function polling (opts) { - var xhr; - var xd = false; - var xs = false; - var jsonp = false !== opts.jsonp; + return need; + } - if (typeof location !== 'undefined') { - var isSSL = 'https:' === location.protocol; - var port = location.port; + function endWritable(stream, state, cb) { + state.ending = true; + finishMaybe(stream, state); - // some user agents have empty `location.port` - if (!port) { - port = isSSL ? 443 : 80; - } + if (cb) { + if (state.finished) process.nextTick(cb); + else stream.once('finish', cb); + } - xd = opts.hostname !== location.hostname || port !== opts.port; - xs = opts.secure !== isSSL; - } + state.ended = true; + stream.writable = false; + } - opts.xdomain = xd; - opts.xscheme = xs; - xhr = new XMLHttpRequest(opts); + function onCorkedFinish(corkReq, state, err) { + var entry = corkReq.entry; + corkReq.entry = null; - if ('open' in xhr && !opts.forceJSONP) { - return new XHR(opts); - } else { - if (!jsonp) throw new Error('JSONP disabled'); - return new JSONP(opts); - } -} + while (entry) { + var cb = entry.callback; + state.pendingcb--; + cb(err); + entry = entry.next; + } // reuse the free corkReq. -},{"./polling-jsonp":170,"./polling-xhr":171,"./websocket":173,"xmlhttprequest-ssl":174}],170:[function(require,module,exports){ -/** - * Module requirements. - */ + state.corkedRequestsFree.next = corkReq; + } -var Polling = require('./polling'); -var inherit = require('component-inherit'); -var globalThis = require('../globalThis'); + Object.defineProperty(Writable.prototype, 'destroyed', { + // making it explicit this property is not enumerable + // because otherwise some prototype manipulation in + // userland will fail + enumerable: false, + get: function get() { + if (this._writableState === undefined) { + return false; + } + + return this._writableState.destroyed; + }, + set: function set(value) { + // we ignore the value if the stream + // has not been initialized yet + if (!this._writableState) { + return; + } // backward compatibility, the user is explicitly + // managing destroyed + + this._writableState.destroyed = value; + } + }); + Writable.prototype.destroy = destroyImpl.destroy; + Writable.prototype._undestroy = destroyImpl.undestroy; + + Writable.prototype._destroy = function (err, cb) { + cb(err); + }; + }).call(this, require('_process'), typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : typeof window !== 'undefined' ? window : {}); + }, + { + '../errors': 107, + './_stream_duplex': 108, + './internal/streams/destroy': 115, + './internal/streams/state': 119, + './internal/streams/stream': 120, + _process: 244, + buffer: 124, + inherits: 106, + 'util-deprecate': 295 + } + ], + 113: [ + function (require, module, exports) { + (function (process) { + 'use strict'; + + var _Object$setPrototypeO; + + function _defineProperty(obj, key, value) { + if (key in obj) { + Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); + } else { + obj[key] = value; + } + return obj; + } -/** - * Module exports. - */ + var finished = require('./end-of-stream'); + + var kLastResolve = Symbol('lastResolve'); + var kLastReject = Symbol('lastReject'); + var kError = Symbol('error'); + var kEnded = Symbol('ended'); + var kLastPromise = Symbol('lastPromise'); + var kHandlePromise = Symbol('handlePromise'); + var kStream = Symbol('stream'); + + function createIterResult(value, done) { + return { + value: value, + done: done + }; + } -module.exports = JSONPPolling; + function readAndResolve(iter) { + var resolve = iter[kLastResolve]; + + if (resolve !== null) { + var data = iter[kStream].read(); // we defer if data is null + // we can be expecting either 'end' or + // 'error' + + if (data !== null) { + iter[kLastPromise] = null; + iter[kLastResolve] = null; + iter[kLastReject] = null; + resolve(createIterResult(data, false)); + } + } + } -/** - * Cached regular expressions. - */ + function onReadable(iter) { + // we wait for the next tick, because it might + // emit an error with process.nextTick + process.nextTick(readAndResolve, iter); + } -var rNewline = /\n/g; -var rEscapedNewline = /\\n/g; + function wrapForNext(lastPromise, iter) { + return function (resolve, reject) { + lastPromise.then(function () { + if (iter[kEnded]) { + resolve(createIterResult(undefined, true)); + return; + } + + iter[kHandlePromise](resolve, reject); + }, reject); + }; + } -/** - * Global JSONP callbacks. - */ + var AsyncIteratorPrototype = Object.getPrototypeOf(function () {}); + var ReadableStreamAsyncIteratorPrototype = Object.setPrototypeOf( + ((_Object$setPrototypeO = { + get stream() { + return this[kStream]; + }, + + next: function next() { + var _this = this; + + // if we have detected an error in the meanwhile + // reject straight away + var error = this[kError]; + + if (error !== null) { + return Promise.reject(error); + } + + if (this[kEnded]) { + return Promise.resolve(createIterResult(undefined, true)); + } + + if (this[kStream].destroyed) { + // We need to defer via nextTick because if .destroy(err) is + // called, the error will be emitted via nextTick, and + // we cannot guarantee that there is no error lingering around + // waiting to be emitted. + return new Promise(function (resolve, reject) { + process.nextTick(function () { + if (_this[kError]) { + reject(_this[kError]); + } else { + resolve(createIterResult(undefined, true)); + } + }); + }); + } // if we have multiple next() calls + // we will wait for the previous Promise to finish + // this logic is optimized to support for await loops, + // where next() is only called once at a time + + var lastPromise = this[kLastPromise]; + var promise; + + if (lastPromise) { + promise = new Promise(wrapForNext(lastPromise, this)); + } else { + // fast path needed to support multiple this.push() + // without triggering the next() queue + var data = this[kStream].read(); + + if (data !== null) { + return Promise.resolve(createIterResult(data, false)); + } + + promise = new Promise(this[kHandlePromise]); + } + + this[kLastPromise] = promise; + return promise; + } + }), + _defineProperty(_Object$setPrototypeO, Symbol.asyncIterator, function () { + return this; + }), + _defineProperty(_Object$setPrototypeO, 'return', function _return() { + var _this2 = this; + + // destroy(err, cb) is a private API + // we can guarantee we have that here, because we control the + // Readable class this is attached to + return new Promise(function (resolve, reject) { + _this2[kStream].destroy(null, function (err) { + if (err) { + reject(err); + return; + } + + resolve(createIterResult(undefined, true)); + }); + }); + }), + _Object$setPrototypeO), + AsyncIteratorPrototype + ); + + var createReadableStreamAsyncIterator = function createReadableStreamAsyncIterator(stream) { + var _Object$create; + + var iterator = Object.create( + ReadableStreamAsyncIteratorPrototype, + ((_Object$create = {}), + _defineProperty(_Object$create, kStream, { + value: stream, + writable: true + }), + _defineProperty(_Object$create, kLastResolve, { + value: null, + writable: true + }), + _defineProperty(_Object$create, kLastReject, { + value: null, + writable: true + }), + _defineProperty(_Object$create, kError, { + value: null, + writable: true + }), + _defineProperty(_Object$create, kEnded, { + value: stream._readableState.endEmitted, + writable: true + }), + _defineProperty(_Object$create, kHandlePromise, { + value: function value(resolve, reject) { + var data = iterator[kStream].read(); + + if (data) { + iterator[kLastPromise] = null; + iterator[kLastResolve] = null; + iterator[kLastReject] = null; + resolve(createIterResult(data, false)); + } else { + iterator[kLastResolve] = resolve; + iterator[kLastReject] = reject; + } + }, + writable: true + }), + _Object$create) + ); + iterator[kLastPromise] = null; + finished(stream, function (err) { + if (err && err.code !== 'ERR_STREAM_PREMATURE_CLOSE') { + var reject = iterator[kLastReject]; // reject if we are waiting for data in the Promise + // returned by next() and store the error + + if (reject !== null) { + iterator[kLastPromise] = null; + iterator[kLastResolve] = null; + iterator[kLastReject] = null; + reject(err); + } + + iterator[kError] = err; + return; + } + + var resolve = iterator[kLastResolve]; + + if (resolve !== null) { + iterator[kLastPromise] = null; + iterator[kLastResolve] = null; + iterator[kLastReject] = null; + resolve(createIterResult(undefined, true)); + } + + iterator[kEnded] = true; + }); + stream.on('readable', onReadable.bind(null, iterator)); + return iterator; + }; + + module.exports = createReadableStreamAsyncIterator; + }).call(this, require('_process')); + }, + { './end-of-stream': 116, _process: 244 } + ], + 114: [ + function (require, module, exports) { + 'use strict'; + + function ownKeys(object, enumerableOnly) { + var keys = Object.keys(object); + if (Object.getOwnPropertySymbols) { + var symbols = Object.getOwnPropertySymbols(object); + if (enumerableOnly) + symbols = symbols.filter(function (sym) { + return Object.getOwnPropertyDescriptor(object, sym).enumerable; + }); + keys.push.apply(keys, symbols); + } + return keys; + } -var callbacks; + function _objectSpread(target) { + for (var i = 1; i < arguments.length; i++) { + var source = arguments[i] != null ? arguments[i] : {}; + if (i % 2) { + ownKeys(Object(source), true).forEach(function (key) { + _defineProperty(target, key, source[key]); + }); + } else if (Object.getOwnPropertyDescriptors) { + Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); + } else { + ownKeys(Object(source)).forEach(function (key) { + Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); + }); + } + } + return target; + } -/** - * Noop. - */ + function _defineProperty(obj, key, value) { + if (key in obj) { + Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); + } else { + obj[key] = value; + } + return obj; + } -function empty () { } + function _classCallCheck(instance, Constructor) { + if (!(instance instanceof Constructor)) { + throw new TypeError('Cannot call a class as a function'); + } + } -/** - * JSONP Polling constructor. - * - * @param {Object} opts. - * @api public - */ + function _defineProperties(target, props) { + for (var i = 0; i < props.length; i++) { + var descriptor = props[i]; + descriptor.enumerable = descriptor.enumerable || false; + descriptor.configurable = true; + if ('value' in descriptor) descriptor.writable = true; + Object.defineProperty(target, descriptor.key, descriptor); + } + } -function JSONPPolling (opts) { - Polling.call(this, opts); + function _createClass(Constructor, protoProps, staticProps) { + if (protoProps) _defineProperties(Constructor.prototype, protoProps); + if (staticProps) _defineProperties(Constructor, staticProps); + return Constructor; + } - this.query = this.query || {}; + var _require = require('buffer'), + Buffer = _require.Buffer; - // define global callbacks array if not present - // we do this here (lazily) to avoid unneeded global pollution - if (!callbacks) { - // we need to consider multiple engines in the same page - callbacks = globalThis.___eio = (globalThis.___eio || []); - } + var _require2 = require('util'), + inspect = _require2.inspect; - // callback identifier - this.index = callbacks.length; + var custom = (inspect && inspect.custom) || 'inspect'; - // add callback to jsonp global - var self = this; - callbacks.push(function (msg) { - self.onData(msg); - }); + function copyBuffer(src, target, offset) { + Buffer.prototype.copy.call(src, target, offset); + } - // append to query string - this.query.j = this.index; + module.exports = + /*#__PURE__*/ + (function () { + function BufferList() { + _classCallCheck(this, BufferList); + + this.head = null; + this.tail = null; + this.length = 0; + } + + _createClass(BufferList, [ + { + key: 'push', + value: function push(v) { + var entry = { + data: v, + next: null + }; + if (this.length > 0) this.tail.next = entry; + else this.head = entry; + this.tail = entry; + ++this.length; + } + }, + { + key: 'unshift', + value: function unshift(v) { + var entry = { + data: v, + next: this.head + }; + if (this.length === 0) this.tail = entry; + this.head = entry; + ++this.length; + } + }, + { + key: 'shift', + value: function shift() { + if (this.length === 0) return; + var ret = this.head.data; + if (this.length === 1) this.head = this.tail = null; + else this.head = this.head.next; + --this.length; + return ret; + } + }, + { + key: 'clear', + value: function clear() { + this.head = this.tail = null; + this.length = 0; + } + }, + { + key: 'join', + value: function join(s) { + if (this.length === 0) return ''; + var p = this.head; + var ret = '' + p.data; + + while ((p = p.next)) { + ret += s + p.data; + } + + return ret; + } + }, + { + key: 'concat', + value: function concat(n) { + if (this.length === 0) return Buffer.alloc(0); + var ret = Buffer.allocUnsafe(n >>> 0); + var p = this.head; + var i = 0; + + while (p) { + copyBuffer(p.data, ret, i); + i += p.data.length; + p = p.next; + } + + return ret; + } // Consumes a specified amount of bytes or characters from the buffered data. + }, + { + key: 'consume', + value: function consume(n, hasStrings) { + var ret; + + if (n < this.head.data.length) { + // `slice` is the same for buffers and strings. + ret = this.head.data.slice(0, n); + this.head.data = this.head.data.slice(n); + } else if (n === this.head.data.length) { + // First chunk is a perfect match. + ret = this.shift(); + } else { + // Result spans more than one buffer. + ret = hasStrings ? this._getString(n) : this._getBuffer(n); + } + + return ret; + } + }, + { + key: 'first', + value: function first() { + return this.head.data; + } // Consumes a specified amount of characters from the buffered data. + }, + { + key: '_getString', + value: function _getString(n) { + var p = this.head; + var c = 1; + var ret = p.data; + n -= ret.length; + + while ((p = p.next)) { + var str = p.data; + var nb = n > str.length ? str.length : n; + if (nb === str.length) ret += str; + else ret += str.slice(0, n); + n -= nb; + + if (n === 0) { + if (nb === str.length) { + ++c; + if (p.next) this.head = p.next; + else this.head = this.tail = null; + } else { + this.head = p; + p.data = str.slice(nb); + } + + break; + } + + ++c; + } + + this.length -= c; + return ret; + } // Consumes a specified amount of bytes from the buffered data. + }, + { + key: '_getBuffer', + value: function _getBuffer(n) { + var ret = Buffer.allocUnsafe(n); + var p = this.head; + var c = 1; + p.data.copy(ret); + n -= p.data.length; + + while ((p = p.next)) { + var buf = p.data; + var nb = n > buf.length ? buf.length : n; + buf.copy(ret, ret.length - n, 0, nb); + n -= nb; + + if (n === 0) { + if (nb === buf.length) { + ++c; + if (p.next) this.head = p.next; + else this.head = this.tail = null; + } else { + this.head = p; + p.data = buf.slice(nb); + } + + break; + } + + ++c; + } + + this.length -= c; + return ret; + } // Make sure the linked list only shows the minimal necessary information. + }, + { + key: custom, + value: function value(_, options) { + return inspect( + this, + _objectSpread({}, options, { + // Only inspect one level. + depth: 0, + // It should not recurse. + customInspect: false + }) + ); + } + } + ]); + + return BufferList; + })(); + }, + { buffer: 124, util: 77 } + ], + 115: [ + function (require, module, exports) { + (function (process) { + 'use strict'; // undocumented cb() API, needed for core, not for public API + + function destroy(err, cb) { + var _this = this; + + var readableDestroyed = this._readableState && this._readableState.destroyed; + var writableDestroyed = this._writableState && this._writableState.destroyed; + + if (readableDestroyed || writableDestroyed) { + if (cb) { + cb(err); + } else if (err) { + if (!this._writableState) { + process.nextTick(emitErrorNT, this, err); + } else if (!this._writableState.errorEmitted) { + this._writableState.errorEmitted = true; + process.nextTick(emitErrorNT, this, err); + } + } + + return this; + } // we set destroyed to true before firing error callbacks in order + // to make it re-entrance safe in case destroy() is called within callbacks + + if (this._readableState) { + this._readableState.destroyed = true; + } // if this is a duplex stream mark the writable part as destroyed as well + + if (this._writableState) { + this._writableState.destroyed = true; + } + + this._destroy(err || null, function (err) { + if (!cb && err) { + if (!_this._writableState) { + process.nextTick(emitErrorAndCloseNT, _this, err); + } else if (!_this._writableState.errorEmitted) { + _this._writableState.errorEmitted = true; + process.nextTick(emitErrorAndCloseNT, _this, err); + } else { + process.nextTick(emitCloseNT, _this); + } + } else if (cb) { + process.nextTick(emitCloseNT, _this); + cb(err); + } else { + process.nextTick(emitCloseNT, _this); + } + }); + + return this; + } - // prevent spurious errors from being emitted when the window is unloaded - if (typeof addEventListener === 'function') { - addEventListener('beforeunload', function () { - if (self.script) self.script.onerror = empty; - }, false); - } -} + function emitErrorAndCloseNT(self, err) { + emitErrorNT(self, err); + emitCloseNT(self); + } -/** - * Inherits from Polling. - */ + function emitCloseNT(self) { + if (self._writableState && !self._writableState.emitClose) return; + if (self._readableState && !self._readableState.emitClose) return; + self.emit('close'); + } -inherit(JSONPPolling, Polling); + function undestroy() { + if (this._readableState) { + this._readableState.destroyed = false; + this._readableState.reading = false; + this._readableState.ended = false; + this._readableState.endEmitted = false; + } + + if (this._writableState) { + this._writableState.destroyed = false; + this._writableState.ended = false; + this._writableState.ending = false; + this._writableState.finalCalled = false; + this._writableState.prefinished = false; + this._writableState.finished = false; + this._writableState.errorEmitted = false; + } + } -/* - * JSONP only supports binary as base64 encoded strings - */ + function emitErrorNT(self, err) { + self.emit('error', err); + } -JSONPPolling.prototype.supportsBinary = false; + function errorOrDestroy(stream, err) { + // We have tests that rely on errors being emitted + // in the same tick, so changing this is semver major. + // For now when you opt-in to autoDestroy we allow + // the error to be emitted nextTick. In a future + // semver major update we should change the default to this. + var rState = stream._readableState; + var wState = stream._writableState; + if ((rState && rState.autoDestroy) || (wState && wState.autoDestroy)) stream.destroy(err); + else stream.emit('error', err); + } -/** - * Closes the socket. - * - * @api private - */ + module.exports = { + destroy: destroy, + undestroy: undestroy, + errorOrDestroy: errorOrDestroy + }; + }).call(this, require('_process')); + }, + { _process: 244 } + ], + 116: [ + function (require, module, exports) { + // Ported from https://github.com/mafintosh/end-of-stream with + // permission from the author, Mathias Buus (@mafintosh). + 'use strict'; + + var ERR_STREAM_PREMATURE_CLOSE = require('../../../errors').codes.ERR_STREAM_PREMATURE_CLOSE; + + function once(callback) { + var called = false; + return function () { + if (called) return; + called = true; + + for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { + args[_key] = arguments[_key]; + } + + callback.apply(this, args); + }; + } -JSONPPolling.prototype.doClose = function () { - if (this.script) { - this.script.parentNode.removeChild(this.script); - this.script = null; - } + function noop() {} - if (this.form) { - this.form.parentNode.removeChild(this.form); - this.form = null; - this.iframe = null; - } + function isRequest(stream) { + return stream.setHeader && typeof stream.abort === 'function'; + } - Polling.prototype.doClose.call(this); -}; + function eos(stream, opts, callback) { + if (typeof opts === 'function') return eos(stream, null, opts); + if (!opts) opts = {}; + callback = once(callback || noop); + var readable = opts.readable || (opts.readable !== false && stream.readable); + var writable = opts.writable || (opts.writable !== false && stream.writable); + + var onlegacyfinish = function onlegacyfinish() { + if (!stream.writable) onfinish(); + }; + + var writableEnded = stream._writableState && stream._writableState.finished; + + var onfinish = function onfinish() { + writable = false; + writableEnded = true; + if (!readable) callback.call(stream); + }; + + var readableEnded = stream._readableState && stream._readableState.endEmitted; + + var onend = function onend() { + readable = false; + readableEnded = true; + if (!writable) callback.call(stream); + }; + + var onerror = function onerror(err) { + callback.call(stream, err); + }; + + var onclose = function onclose() { + var err; + + if (readable && !readableEnded) { + if (!stream._readableState || !stream._readableState.ended) err = new ERR_STREAM_PREMATURE_CLOSE(); + return callback.call(stream, err); + } + + if (writable && !writableEnded) { + if (!stream._writableState || !stream._writableState.ended) err = new ERR_STREAM_PREMATURE_CLOSE(); + return callback.call(stream, err); + } + }; + + var onrequest = function onrequest() { + stream.req.on('finish', onfinish); + }; + + if (isRequest(stream)) { + stream.on('complete', onfinish); + stream.on('abort', onclose); + if (stream.req) onrequest(); + else stream.on('request', onrequest); + } else if (writable && !stream._writableState) { + // legacy streams + stream.on('end', onlegacyfinish); + stream.on('close', onlegacyfinish); + } -/** - * Starts a poll cycle. - * - * @api private - */ + stream.on('end', onend); + stream.on('finish', onfinish); + if (opts.error !== false) stream.on('error', onerror); + stream.on('close', onclose); + return function () { + stream.removeListener('complete', onfinish); + stream.removeListener('abort', onclose); + stream.removeListener('request', onrequest); + if (stream.req) stream.req.removeListener('finish', onfinish); + stream.removeListener('end', onlegacyfinish); + stream.removeListener('close', onlegacyfinish); + stream.removeListener('finish', onfinish); + stream.removeListener('end', onend); + stream.removeListener('error', onerror); + stream.removeListener('close', onclose); + }; + } -JSONPPolling.prototype.doPoll = function () { - var self = this; - var script = document.createElement('script'); + module.exports = eos; + }, + { '../../../errors': 107 } + ], + 117: [ + function (require, module, exports) { + module.exports = function () { + throw new Error('Readable.from is not available in the browser'); + }; + }, + {} + ], + 118: [ + function (require, module, exports) { + // Ported from https://github.com/mafintosh/pump with + // permission from the author, Mathias Buus (@mafintosh). + 'use strict'; + + var eos; + + function once(callback) { + var called = false; + return function () { + if (called) return; + called = true; + callback.apply(void 0, arguments); + }; + } - if (this.script) { - this.script.parentNode.removeChild(this.script); - this.script = null; - } + var _require$codes = require('../../../errors').codes, + ERR_MISSING_ARGS = _require$codes.ERR_MISSING_ARGS, + ERR_STREAM_DESTROYED = _require$codes.ERR_STREAM_DESTROYED; - script.async = true; - script.src = this.uri(); - script.onerror = function (e) { - self.onError('jsonp poll error', e); - }; + function noop(err) { + // Rethrow the error if it exists to avoid swallowing it + if (err) throw err; + } - var insertAt = document.getElementsByTagName('script')[0]; - if (insertAt) { - insertAt.parentNode.insertBefore(script, insertAt); - } else { - (document.head || document.body).appendChild(script); - } - this.script = script; + function isRequest(stream) { + return stream.setHeader && typeof stream.abort === 'function'; + } - var isUAgecko = 'undefined' !== typeof navigator && /gecko/i.test(navigator.userAgent); + function destroyer(stream, reading, writing, callback) { + callback = once(callback); + var closed = false; + stream.on('close', function () { + closed = true; + }); + if (eos === undefined) eos = require('./end-of-stream'); + eos( + stream, + { + readable: reading, + writable: writing + }, + function (err) { + if (err) return callback(err); + closed = true; + callback(); + } + ); + var destroyed = false; + return function (err) { + if (closed) return; + if (destroyed) return; + destroyed = true; // request.destroy just do .end - .abort is what we want + + if (isRequest(stream)) return stream.abort(); + if (typeof stream.destroy === 'function') return stream.destroy(); + callback(err || new ERR_STREAM_DESTROYED('pipe')); + }; + } - if (isUAgecko) { - setTimeout(function () { - var iframe = document.createElement('iframe'); - document.body.appendChild(iframe); - document.body.removeChild(iframe); - }, 100); - } -}; + function call(fn) { + fn(); + } -/** - * Writes with a hidden iframe. - * - * @param {String} data to send - * @param {Function} called upon flush. - * @api private - */ + function pipe(from, to) { + return from.pipe(to); + } -JSONPPolling.prototype.doWrite = function (data, fn) { - var self = this; - - if (!this.form) { - var form = document.createElement('form'); - var area = document.createElement('textarea'); - var id = this.iframeId = 'eio_iframe_' + this.index; - var iframe; - - form.className = 'socketio'; - form.style.position = 'absolute'; - form.style.top = '-1000px'; - form.style.left = '-1000px'; - form.target = id; - form.method = 'POST'; - form.setAttribute('accept-charset', 'utf-8'); - area.name = 'd'; - form.appendChild(area); - document.body.appendChild(form); - - this.form = form; - this.area = area; - } + function popCallback(streams) { + if (!streams.length) return noop; + if (typeof streams[streams.length - 1] !== 'function') return noop; + return streams.pop(); + } - this.form.action = this.uri(); + function pipeline() { + for (var _len = arguments.length, streams = new Array(_len), _key = 0; _key < _len; _key++) { + streams[_key] = arguments[_key]; + } - function complete () { - initIframe(); - fn(); - } + var callback = popCallback(streams); + if (Array.isArray(streams[0])) streams = streams[0]; - function initIframe () { - if (self.iframe) { - try { - self.form.removeChild(self.iframe); - } catch (e) { - self.onError('jsonp polling iframe removal error', e); - } - } + if (streams.length < 2) { + throw new ERR_MISSING_ARGS('streams'); + } - try { - // ie6 dynamic iframes with target="" support (thanks Chris Lambacher) - var html = '