From 2f5a9d25bfa50f319b1bbe8d93a4e2a52a214124 Mon Sep 17 00:00:00 2001 From: asac Date: Thu, 18 Jul 2019 17:28:36 +0300 Subject: [PATCH] small update --- README.md | 2 +- psknode/bin/pskbuild.js | 0 psknode/bin/psknode.js | 0 psknode/bin/scripts/psknode.js | 16 ++++++++++++++-- psknode/bundles/pskruntime.js | 24 ++++++++++++++++++++---- psknode/bundles/readme.txt | 1 - psknode/core/launcher.js | 11 +++++++---- psknode/core/utils/exitHandler.js | 22 +++++++++++++++++++--- 8 files changed, 61 insertions(+), 15 deletions(-) delete mode 100644 psknode/bin/pskbuild.js delete mode 100644 psknode/bin/psknode.js delete mode 100644 psknode/bundles/readme.txt diff --git a/README.md b/README.md index 0204f0e..27d7468 100644 --- a/README.md +++ b/README.md @@ -12,6 +12,6 @@ When a new version of PrivateSky code will be released the clones of the pskrunt The following scripts are available to run: ``` npm run start ``` will run a PrivateSky node -``` npm run build ``` will build the code for the domain and deply in the current domain +``` npm run build ``` will build the code for the domain and deploy in the current domain ``` npm run watch ``` will automatically build the code for the domain and deploy ``` npm run console ``` will launch a bash or cmd with paths to run pskadmin tools diff --git a/psknode/bin/pskbuild.js b/psknode/bin/pskbuild.js deleted file mode 100644 index e69de29..0000000 diff --git a/psknode/bin/psknode.js b/psknode/bin/psknode.js deleted file mode 100644 index e69de29..0000000 diff --git a/psknode/bin/scripts/psknode.js b/psknode/bin/scripts/psknode.js index 208e070..b1ce287 100644 --- a/psknode/bin/scripts/psknode.js +++ b/psknode/bin/scripts/psknode.js @@ -5,6 +5,8 @@ */ const {spawnSync, fork, spawn} = require('child_process'); +const max_timeout = 10*60*1000; // 10 minutes +const restartDelays = {}; let shouldRestart = true; const forkedProcesses = {}; @@ -16,11 +18,21 @@ function startProcess(filePath) { console.log('SPAWNED ', forkedProcesses[filePath].pid); + function restartWithDelay(filePath){ + let timeout = restartDelays[filePath] || 100; + console.log(`Process will restart in ${timeout} ms ...`); + setTimeout(()=>{ + restartDelays[filePath] = (timeout * 2) % max_timeout; + startProcess(filePath); + }, timeout); + } + function errorHandler(filePath) { + let timeout = 100; return function (error) { console.log(`\x1b[31mException caught on spawning file ${filePath} `, error ? error : "", "\x1b[0m"); //last string is to reset terminal colours if (shouldRestart) { - startProcess(filePath); + restartWithDelay(filePath); } } } @@ -29,7 +41,7 @@ function startProcess(filePath) { return function () { console.log(`\x1b[33mExit caught on spawned file ${filePath}`, "\x1b[0m"); //last string is to reset terminal colours if (shouldRestart) { - startProcess(filePath); + restartWithDelay(filePath); } } } diff --git a/psknode/bundles/pskruntime.js b/psknode/bundles/pskruntime.js index 53ddeec..197fafe 100644 --- a/psknode/bundles/pskruntime.js +++ b/psknode/bundles/pskruntime.js @@ -10940,13 +10940,29 @@ const events = ["exit", "SIGINT", "SIGUSR1", "SIGUSR2", "uncaughtException", "SI module.exports = function manageShutdownProcess(childrenList){ + let shutting = false; function handler(){ - console.log("Handling exit event on", process.pid, "arguments:", arguments); + //console.log("Handling exit event on", process.pid, "arguments:", arguments); var childrenNames = Object.keys(childrenList); for(let j=0; j{ @@ -11508,4 +11524,4 @@ if(typeof global.$$.uidGenerator == "undefined"){ },{"./lib/Combos":"/home/cosmin/Workspace/reorganizing/privatesky/modules/swarmutils/lib/Combos.js","./lib/OwM":"/home/cosmin/Workspace/reorganizing/privatesky/modules/swarmutils/lib/OwM.js","./lib/Queue":"/home/cosmin/Workspace/reorganizing/privatesky/modules/swarmutils/lib/Queue.js","./lib/beesHealer":"/home/cosmin/Workspace/reorganizing/privatesky/modules/swarmutils/lib/beesHealer.js","./lib/pskconsole":"/home/cosmin/Workspace/reorganizing/privatesky/modules/swarmutils/lib/pskconsole.js","./lib/safe-uuid":"/home/cosmin/Workspace/reorganizing/privatesky/modules/swarmutils/lib/safe-uuid.js","./lib/uidGenerator":"/home/cosmin/Workspace/reorganizing/privatesky/modules/swarmutils/lib/uidGenerator.js"}],"utils":[function(require,module,exports){ exports.fsExt = require("./FSExtension").fsExt; },{"./FSExtension":"/home/cosmin/Workspace/reorganizing/privatesky/libraries/utils/FSExtension.js"}]},{},["/home/cosmin/Workspace/reorganizing/privatesky/builds/tmp/pskruntime.js"]) -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","builds/tmp/pskruntime.js","builds/tmp/pskruntime_intermediar.js","libraries/domainBase/domainPubSub.js","libraries/launcher/components.js","libraries/utils/FSExtension.js","modules/callflow/constants.js","modules/callflow/lib/InterceptorRegistry.js","modules/callflow/lib/choreographies/SwarmDebug.js","modules/callflow/lib/choreographies/swarmInstancesManager.js","modules/callflow/lib/choreographies/utilityFunctions/asset.js","modules/callflow/lib/choreographies/utilityFunctions/base.js","modules/callflow/lib/choreographies/utilityFunctions/callflow.js","modules/callflow/lib/choreographies/utilityFunctions/swarm.js","modules/callflow/lib/loadLibrary.js","modules/callflow/lib/overwriteRequire.js","modules/callflow/lib/parallelJoinPoint.js","modules/callflow/lib/serialJoinPoint.js","modules/callflow/lib/swarmDescription.js","modules/double-check/lib/standardAsserts.js","modules/double-check/lib/standardChecks.js","modules/double-check/lib/standardExceptions.js","modules/double-check/lib/standardLogs.js","modules/double-check/lib/testRunner.js","modules/double-check/lib/utils/glob-to-regexp.js","modules/foldermq/lib/folderMQ.js","modules/pskbuffer/lib/PSKBuffer.js","modules/pskcrypto/lib/ECDSA.js","modules/pskcrypto/lib/PskCrypto.js","modules/pskcrypto/lib/asn1/api.js","modules/pskcrypto/lib/asn1/asn1.js","modules/pskcrypto/lib/asn1/base/buffer.js","modules/pskcrypto/lib/asn1/base/index.js","modules/pskcrypto/lib/asn1/base/node.js","modules/pskcrypto/lib/asn1/base/reporter.js","modules/pskcrypto/lib/asn1/bignum/bn.js","modules/pskcrypto/lib/asn1/constants/der.js","modules/pskcrypto/lib/asn1/constants/index.js","modules/pskcrypto/lib/asn1/decoders/der.js","modules/pskcrypto/lib/asn1/decoders/index.js","modules/pskcrypto/lib/asn1/decoders/pem.js","modules/pskcrypto/lib/asn1/encoders/der.js","modules/pskcrypto/lib/asn1/encoders/index.js","modules/pskcrypto/lib/asn1/encoders/pem.js","modules/pskcrypto/lib/keyEncoder.js","modules/pskcrypto/lib/psk-archiver.js","modules/pskcrypto/lib/utils/DuplexStream.js","modules/pskcrypto/lib/utils/PassThroughStream.js","modules/pskcrypto/lib/utils/countFiles.js","modules/pskcrypto/lib/utils/cryptoUtils.js","modules/pskcrypto/lib/utils/isStream.js","modules/pskcrypto/signsensusDS/ssutil.js","modules/soundpubsub/lib/soundPubSub.js","modules/swarmutils/lib/Combos.js","modules/swarmutils/lib/OwM.js","modules/swarmutils/lib/Queue.js","modules/swarmutils/lib/beesHealer.js","modules/swarmutils/lib/pskconsole.js","modules/swarmutils/lib/safe-uuid.js","modules/swarmutils/lib/uidGenerator.js","psknode/core/sandboxes/util/SandBoxManager.js","psknode/core/utils/exitHandler.js","modules/callflow/index.js","modules/dicontainer/lib/container.js","libraries/domainBase/index.js","modules/double-check/lib/checksCore.js","modules/foldermq/index.js","libraries/launcher/index.js","modules/pskbuffer/index.js","modules/pskcrypto/index.js","modules/soundpubsub/index.js","modules/swarmutils/index.js","libraries/utils/index.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AC7bA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACXA;AACA;AACA;;ACFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AC3RA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3HA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChWA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC7OA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AC7pBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5eA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AChDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AC1UA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpHA;AACA;AACA;AACA;AACA;AACA;AACA;;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1lBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9wEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnSA;AACA;AACA;AACA;AACA;;ACJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChRA;AACA;AACA;AACA;AACA;;ACJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AClKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACpaA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACjJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AC/LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3MA;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtHA;AACA;AACA;AACA;;ACHA;AACA;AACA;AACA;AACA;;ACJA;AACA;AACA;AACA;;ACHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACXA;AACA;AACA;;;ACFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACzBA","file":"generated.js","sourceRoot":"","sourcesContent":["(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})()","require(\"../../modules/callflow/lib/overwriteRequire\")\n\nrequire(\"./pskruntime_intermediar\");\n\nrequire(\"callflow\");\n\nconsole.log(\"Loading runtime: callflow module ready\");","global.pskruntimeLoadModules = function(){ \n\t$$.__runtimeModules[\"callflow\"] = require(\"callflow\");\n\t$$.__runtimeModules[\"launcher\"] = require(\"launcher\");\n\t$$.__runtimeModules[\"double-check\"] = require(\"double-check\");\n\t$$.__runtimeModules[\"pskcrypto\"] = require(\"pskcrypto\");\n\t$$.__runtimeModules[\"dicontainer\"] = require(\"dicontainer\");\n\t$$.__runtimeModules[\"swarmutils\"] = require(\"swarmutils\");\n\t$$.__runtimeModules[\"soundpubsub\"] = require(\"soundpubsub\");\n\t$$.__runtimeModules[\"pskbuffer\"] = require(\"pskbuffer\");\n\t$$.__runtimeModules[\"foldermq\"] = require(\"foldermq\");\n\t$$.__runtimeModules[\"domainBase\"] = require(\"domainBase\");\n\t$$.__runtimeModules[\"utils\"] = require(\"utils\");\n}\nif (false) {\n\tpskruntimeLoadModules();\n}; \nglobal.pskruntimeRequire = require;\nif (typeof $$ !== \"undefined\") {            \n    $$.requireBundle(\"pskruntime\");\n};","var pubSub = $$.require(\"soundpubsub\").soundPubSub;\nconst path = require(\"path\");\nconst fs = require(\"fs\");\n\nexports.create = function(folder, codeFolder ){\n\n    $$.PSK_PubSub = pubSub;\n    var sandBoxesRoot = path.join(folder, \"sandboxes\");\n\n    try{\n        fs.mkdirSync(sandBoxesRoot, {recursive: true});\n    }catch(err){\n        console.log(\"Failed to create sandboxes dir structure!\", err);\n        //TODO: maybe it is ok to call process.exit ???\n    }\n\n    $$.SandBoxManager = require(\"../../psknode/core/sandboxes/util/SandBoxManager\").create(sandBoxesRoot, codeFolder, function(err, res){\n        console.log($$.DI_components.sandBoxReady, err, res);\n        $$.container.resolve($$.DI_components.sandBoxReady, true);\n    });\n\n    return pubSub;\n};\n","$$.DI_components = {\n   swarmIsReady:\"SwarmIsReady\",\n   configLoaded:\"configLoaded\",\n   sandBoxReady:\"SandBoxReady\",\n   localNodeAPIs:\"localNodeAPIs\"\n}\n","const fs = require(\"fs\");\nconst path = require(\"path\");\nconst os = require(\"os\");\nconst child_process = require('child_process');\nconst crypto = require('crypto');\n\n// if this is set to true, the logs will be available. Default (false)\nconst DEBUG =  process.env.DEPLOYER_DEBUG || false;\n\nfunction FSExtention(){\n\n    /**\n     * Base path used to resolve all relative paths in the actions bellow.\n     * Default is set to two levels up from the current directory. This can be changed using __setBasePath.\n     * @type {*|string}\n     */\n    var basePath = path.join(__dirname, \"../../\");\n\n    /**\n     * Set the base path to a different absolute directory path.\n     * @param wd {String} absolute directory path.\n     * @private\n     */\n    var __setBasePath = function(wd) {\n        basePath = path.resolve(wd);\n    }\n\n    /**\n     * Resolve path into an absolute path. If filePath is relative, the path is resolved using the basePath as first argument.\n     * @param filePath {String} relative or absolute file path.\n     * @returns {String} absolute path\n     * @private\n     */\n    var __resolvePath = function(filePath) {\n        if(path.isAbsolute(filePath)) {\n            return filePath;\n        }\n\n        return path.resolve(basePath, filePath);\n    }\n\n    /**\n     * If the directory structure does not exist, it is created. Like mkdir -p\n     * @param dir {String} dir path\n     * @private\n     */\n    var __createDir = function(dir) {\n        dir = __resolvePath(dir);\n        if (fs.existsSync(dir)) {\n            log(dir + \" already exist! Continuing ...\")\n            return;\n        }\n\n        var isWin = (os.platform() === 'win32');\n        var cmd = isWin ? \"mkdir \" : \"mkdir -p \";\n\n        child_process.execSync(cmd + \"\\\"\"+dir+\"\\\"\", {stdio:[0,1,2]});\n    }\n\n    /**\n     * Copy a file or directory. The directory can have recursive contents. Like copy -r.\n     * NOTE: If src is a directory it will copy everything inside of the directory, not the entire directory itself.\n     * NOTE: If src is a file, target cannot be a directory.\n     * NOTE: If the destination path structure does not exists, it will be created.\n     * @param src {String} Source file|directory path.\n     * @param dest {String} Destination file|directory path.\n     * @param options {Object} Optional parameters for copy action. Available options:\n     *  - overwrite <Boolean>: overwrite existing file or directory, default is true.\n     *  Note that the copy operation will silently fail if this is set to false and the destination exists.\n     * @param callback {Function}\n     * @private\n     */\n    var __copy = function (src, dest, options, callback) {\n        src = __resolvePath(src);\n        dest = __resolvePath(dest);\n\n        callback = callback || function(){};\n        let rethrow = false;\n\n        try{\n            if (!fs.existsSync(src)) {\n                rethrow = true;\n                throw `Source directory or file \"${src}\" does not exists!`;\n            }\n\n            let srcStat = fs.lstatSync(src);\n            if(srcStat.isDirectory()) {\n                __copyDir(src, dest, options);\n            } else if(srcStat.isFile()) {\n                // destination must be a file too\n                __copyFile(src, dest, options);\n            }\n        } catch (err) {\n            if(rethrow){\n                throw err;\n            }\n            log(err, true);\n            callback(err);\n            return;\n        }\n\n        callback();\n    }\n\n    /**\n     * Copy a directory. The directory can have recursive contents. Like copy -r.\n     * NOTE: Itt will copy everything inside of the directory, not the entire directory itself.\n     * NOTE: If the destination path structure does not exists, it will be created.\n     * @param src {String} Source directory path.\n     * @param dest {String} Destination directory path.\n     * @param options {Object} Optional parameters for copy action. Available options:\n     *  - overwrite <Boolean>: overwrite existing directory, default is true.\n     *  Note that the copy operation will silently fail if this is set to false and the destination exists.\n     * @private\n     */\n    var __copyDir = function(src, dest, options) {\n        src = __resolvePath(src);\n        dest = __resolvePath(dest);\n\n        __createDir(dest);\n\n        var files = fs.readdirSync(src);\n        for(var i = 0; i < files.length; i++) {\n            let current = fs.lstatSync(path.join(src, files[i]));\n            let newSrc = path.join(src, files[i]);\n            let newDest = path.join(dest, files[i]);\n\n            if(current.isDirectory()) {\n                __copyDir(newSrc, newDest, options);\n            } else if(current.isSymbolicLink()) {\n                var symlink = fs.readlinkSync(newSrc);\n                fs.symlinkSync(symlink, newDest);\n            } else {\n                __copyFile(newSrc, newDest, options);\n            }\n        }\n    };\n\n    /**\n     * Copy a file.\n     * NOTE: If src is a file, target cannot be a directory.\n     * NOTE: If the destination path structure does not exists, it will be created.\n     * @param src {String} Source file path.\n     * @param dest {String} Destination file path.\n     * @param options {Object} Optional parameters for copy action. Available options:\n     *  - overwrite <Boolean>: overwrite existing file or directory, default is true.\n     *  Note that the copy operation will silently fail if this is set to false and the destination exists.\n     * @param callback {Function}\n     * @private\n     */\n    var __copyFile = function(src, dest, options) {\n        src = __resolvePath(src);\n        dest = __resolvePath(dest);\n\n        if(options && options.overwrite === false) {\n            if (fs.existsSync(dest)) {\n                // silently fail if overwrite is set to false and the destination exists.\n                let error = `Silent fail - cannot copy. Destination file ${dest} already exists and overwrite option is set to false! Continuing...`;\n                log(error, true);\n                return;\n            }\n        }\n        __createDir(path.dirname(dest));\n\n        var content = fs.readFileSync(src, \"utf8\");\n        fs.writeFileSync(dest, content);\n    }\n\n    /**\n     * Removes a file or directory. The directory can have recursive contents. Like rm -rf\n     * @param src {String} Path\n     * @param callback {Function}\n     * @private\n     */\n    var __remove = function(src, callback) {\n        src = __resolvePath(src);\n\n        callback = callback || function(){};\n\n        log(`Removing ${src}`);\n\n        try{\n            let current = fs.lstatSync(src);\n            if(current.isDirectory()) {\n                __rmDir(src);\n            } else if(current.isFile()) {\n                __rmFile(src);\n            }\n        } catch (err) {\n            if(err.code && err.code === \"ENOENT\"){\n                //ignoring errors like \"file/directory does not exist\"\n                err = null;\n            }else{\n                log(err, true);\n            }\n            callback(err);\n            return;\n        }\n\n        callback();\n    }\n\n    /**\n     * Removes a directory. The directory can have recursive contents. Like rm -rf\n     * @param dir {String} Path\n     * @private\n     */\n    var __rmDir = function (dir) {\n        dir = __resolvePath(dir);\n\n        if (!fs.existsSync(dir)) {\n            log(`Directory ${dir} does not exist!`, true);\n            return;\n        }\n\n        var list = fs.readdirSync(dir);\n        for (var i = 0; i < list.length; i++) {\n            var filename = path.join(dir, list[i]);\n            var stat = fs.lstatSync(filename);\n\n            if (stat.isDirectory()) {\n                __rmDir(filename, null);\n            } else {\n                // rm filename\n                fs.unlinkSync(filename);\n            }\n        }\n\n        fs.rmdirSync(dir);\n    }\n\n    /**\n     * Removes a file.\n     * @param file {String} Path\n     * @private\n     */\n    var __rmFile = function(file) {\n        file = __resolvePath(file);\n        if (!fs.existsSync(file)) {\n            log(`File ${file} does not exist!`, true);\n            return;\n        }\n\n        fs.unlinkSync(file);\n    }\n\n    /**\n     * Writes data to a file, replacing the file if it already exists.\n     * @param file {String} Path.\n     * @param data {String}\n     * @private\n     */\n    var __createFile = function(file, data, options) {\n        file = __resolvePath(file)\n        fs.writeFileSync(file, data, options);\n    }\n\n    /**\n     * Moves a file or directory.\n     * @param src {String} Source path.\n     * @param dest {String} Destination path.\n     * @param options {Object}. Optional parameters for copy action. Available options:\n     *  - overwrite <boolean>: overwrite existing file or directory, default is false. Note that the move operation will silently fail if you set this to true and the destination exists.\n     * @param callback {Function}\n     * @private\n     */\n    var __move = function(src, dest, options, callback) {\n        src = __resolvePath(src);\n        dest = __resolvePath(dest);\n\n        callback = callback || function(){};\n\n        try {\n            if(options && options.overwrite === false) {\n                if (fs.existsSync(dest)) {\n                    // silently fail if overwrite is set to false and the destination exists.\n                    let error = `Silent fail - cannot move. Destination file ${dest} already exists and overwrite option is set to false! Continuing...`;\n                    log(error, true);\n                    callback();\n                    return;\n                }\n            }\n\n            __copy(src, dest, options);\n            __remove(src);\n        }catch(err) {\n            callback(err);\n            return;\n        }\n        callback();\n    }\n\n    /**\n     * Computes checksum to a file or a directory based on their contents only.\n     * If the source is directory, the checksum is a hash of all concatenated file hashes.\n     * @param src {String} Path of a file or directory.\n     * @param algorithm {String} Hashing algorithm(default: md5). The algorithm is dependent on the available algorithms\n     * supported by the version of OpenSSL on the platform. E.g. 'md5', 'sha256', 'sha512'.\n     * @param encoding {String} Hashing encoding (default: 'hex'). The encoding is dependent on the\n     * available digest algorithms. E.g. 'hex', 'latin1' or 'base64'.\n     * @returns {String} Checksum of the file or directory.\n     * @private\n     */\n    var __checksum = function(src, algorithm, encoding) {\n        src = __resolvePath(src);\n\n        if (!fs.existsSync(src)) {\n            throw `Path ${src} does not exists!`;\n        }\n\n        var checksum = \"\";\n        let current = fs.lstatSync(src);\n        if(current.isDirectory()) {\n            let hashDir = __hashDir(src, algorithm, encoding);\n            checksum = hashDir[\"hash\"];\n        } else if(current.isFile()) {\n            checksum = __hashFile(src, algorithm, encoding);\n        }\n\n        return checksum;\n    }\n\n    /**\n     * Computes hash of a string.\n     * @param str {String}\n     * @param algorithm {String} Hashing algorithm(default: md5). The algorithm is dependent on the available algorithms\n     * supported by the version of OpenSSL on the platform. E.g. 'md5', 'sha256', 'sha512'.\n     * @param encoding {String} Hashing encoding (default: 'hex'). The encoding is dependent on the\n     * available digest algorithms. E.g. 'hex', 'latin1' or 'base64'.\n     * @returns {String} Hash of the string.\n     * @private\n     */\n    var __hash =  function(str, algorithm, encoding) {\n        return crypto\n            .createHash(algorithm || 'md5')\n            .update(str, 'utf8')\n            .digest(encoding || 'hex')\n    }\n\n    /**\n     * Computes hash of a file based on its content only.\n     * @param src {String} Path of a file.\n     * @param algorithm {String} Hashing algorithm(default: md5). The algorithm is dependent on the available algorithms\n     * supported by the version of OpenSSL on the platform. E.g. 'md5', 'sha256', 'sha512'.\n     * @param encoding {String} Hashing encoding (default: 'hex'). The encoding is dependent on the\n     * available digest algorithms. E.g. 'hex', 'latin1' or 'base64'.\n     * @returns {String} Hash of the file.\n     * @private\n     */\n    var __hashFile = function(src, algorithm, encoding) {\n        src = __resolvePath(src);\n        if (!fs.existsSync(src)) {\n            throw `${src} does not exist!`;\n        }\n\n        var content = fs.readFileSync(src, \"utf8\");\n        return __hash(content, algorithm, encoding);\n    }\n\n    /**\n     * Computes hash of a directory based on its content only.\n     * If directory has multiple files, the result is a hash of all concatenated file hashes.\n     * @param src {String} Path of a directory.\n     * @param algorithm {String} Hashing algorithm(default: md5). The algorithm is dependent on the available algorithms\n     * supported by the version of OpenSSL on the platform. E.g. 'md5', 'sha256', 'sha512'.\n     * @param encoding {String} Hashing encoding (default: 'hex'). The encoding is dependent on the\n     * available digest algorithms. E.g. 'hex', 'latin1' or 'base64'.\n     * @returns {String} Hash of the directory.\n     * @private\n     */\n    var __hashDir = function(dir, algorithm, encoding) {\n        dir = __resolvePath(dir);\n        if (!fs.existsSync(dir)) {\n            throw `Directory ${dir} does not exist!`;\n        }\n        var hashes = {};\n        var list = fs.readdirSync(dir);\n        for (var i = 0; i < list.length; i++) {\n            var filename = path.join(dir, list[i]);\n            var stat = fs.lstatSync(filename);\n\n            if (stat.isDirectory()) {\n                let tempHashes = __hashDir(filename, algorithm, encoding);\n                hashes = Object.assign(hashes, tempHashes[\"sub-hashes\"]);\n            } else {\n                let tempHash = __hashFile(filename, algorithm, encoding);\n                hashes[filename] = tempHash;\n            }\n        }\n\n        // compute dir hash\n        let dirContent = Object.keys(hashes).reduce(function (previous, key) {\n            return previous += hashes[key];\n        }, \"\");\n\n        let dirHash = __hash(dirContent, algorithm, encoding);\n\n        return {\n            \"hash\": dirHash,\n            \"sub-hashes\": hashes\n        }\n    }\n\n    /**\n     * Generates a guid (global unique identifier).\n     * @returns {String} Guid in the format xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx\n     * @private\n     */\n    var __guid = function guid() {\n        function _make_group(s) {\n            var p = (Math.random().toString(16)+\"000000000\").substr(2,8);\n            return s ? \"-\" + p.substr(0,4) + \"-\" + p.substr(4,4) : p ;\n        }\n        return _make_group() + _make_group(true) + _make_group(true) + _make_group();\n    }\n\n    /**\n     * Logs wrapper.\n     * @param message {String}\n     * @param isError {Boolean}\n     */\n    function log(message, isError) {\n        let logger = isError ? console.error : console.log;\n\n        if(DEBUG) {\n            logger(message);\n        }\n    }\n\n    return {\n        setBasePath: __setBasePath,\n        resolvePath: __resolvePath,\n        createDir: __createDir,\n        copyDir: __copyDir,\n        rmDir: __rmDir,\n        rmFile: __rmFile,\n        createFile: __createFile,\n        copy: __copy,\n        move: __move,\n        remove: __remove,\n        checksum: __checksum,\n        guid: __guid\n    }\n}\n\nmodule.exports.fsExt = new FSExtention();","$$.CONSTANTS = {\n    SWARM_FOR_EXECUTION:\"swarm_for_execution\",\n    INBOUND:\"inbound\",\n    OUTBOUND:\"outbound\",\n    PDS:\"PrivateDataSystem\",\n    CRL:\"CommunicationReplicationLayer\",\n    SWARM_RETURN: 'swarm_return',\n    BEFORE_INTERCEPTOR: 'before',\n    AFTER_INTERCEPTOR: 'after',\n};\n\n","// related to: SwarmSpace.SwarmDescription.createPhase()\n\nfunction InterceptorRegistry() {\n    const rules = new Map();\n\n    // ??? $$.errorHandler Library ???\n    const _CLASS_NAME = 'InterceptorRegistry';\n\n    /************* PRIVATE METHODS *************/\n\n    function _throwError(err, msg) {\n        console.error(err.message, `${_CLASS_NAME} error message:`, msg);\n        throw err;\n    }\n\n    function _warning(msg) {\n        console.warn(`${_CLASS_NAME} warning message:`, msg);\n    }\n\n    const getWhenOptions = (function () {\n        let WHEN_OPTIONS;\n        return function () {\n            if (WHEN_OPTIONS === undefined) {\n                WHEN_OPTIONS = Object.freeze([\n                    $$.CONSTANTS.BEFORE_INTERCEPTOR,\n                    $$.CONSTANTS.AFTER_INTERCEPTOR\n                ]);\n            }\n            return WHEN_OPTIONS;\n        };\n    })();\n\n    function verifyWhenOption(when) {\n        if (!getWhenOptions().includes(when)) {\n            _throwError(new RangeError(`Option '${when}' is wrong!`),\n                `it should be one of: ${getWhenOptions()}`);\n        }\n    }\n\n    function verifyIsFunctionType(fn) {\n        if (typeof fn !== 'function') {\n            _throwError(new TypeError(`Parameter '${fn}' is wrong!`),\n                `it should be a function, not ${typeof fn}!`);\n        }\n    }\n\n    function resolveNamespaceResolution(swarmTypeName) {\n        if (swarmTypeName === '*') {\n            return swarmTypeName;\n        }\n\n        return (swarmTypeName.includes(\".\") ? swarmTypeName : ($$.libraryPrefix + \".\" + swarmTypeName));\n    }\n\n    /**\n     * Transforms an array into a generator with the particularity that done is set to true on the last element,\n     * not after it finished iterating, this is helpful in optimizing some other functions\n     * It is useful if you want call a recursive function over the array elements but without popping the first\n     * element of the Array or sending the index as an extra parameter\n     * @param {Array<*>} arr\n     * @return {IterableIterator<*>}\n     */\n    function* createArrayGenerator(arr) {\n        const len = arr.length;\n\n        for (let i = 0; i < len - 1; ++i) {\n            yield arr[i];\n        }\n\n        return arr[len - 1];\n    }\n\n    /**\n     * Builds a tree like structure over time (if called on the same root node) where internal nodes are instances of\n     * Map containing the name of the children nodes (each child name is the result of calling next on `keysGenerator)\n     * and a reference to them and on leafs it contains an instance of Set where it adds the function given as parameter\n     * (ex: for a keyGenerator that returns in this order (\"key1\", \"key2\") the resulting structure will be:\n     * {\"key1\": {\"key1\": Set([fn])}} - using JSON just for illustration purposes because it's easier to represent)\n     * @param {Map} rulesMap\n     * @param {IterableIterator} keysGenerator - it has the particularity that done is set on last element, not after it\n     * @param {function} fn\n     */\n    function registerRecursiveRule(rulesMap, keysGenerator, fn) {\n        const {value, done} = keysGenerator.next();\n\n        if (!done) { // internal node\n            const nextKey = rulesMap.get(value);\n\n            if (typeof nextKey === 'undefined') { // if value not found in rulesMap\n                rulesMap.set(value, new Map());\n            }\n\n            registerRecursiveRule(rulesMap.get(value), keysGenerator, fn);\n        } else { // reached leaf node\n            if (!rulesMap.has(value)) {\n\n                rulesMap.set(value, new Set([fn]));\n            } else {\n                const set = rulesMap.get(value);\n\n                if (set.has(fn)) {\n                    _warning(`Duplicated interceptor for '${key}'`);\n                }\n\n                set.add(fn);\n            }\n        }\n    }\n\n    /**\n     * Returns the corresponding set of functions for the given key if found\n     * @param {string} key - formatted as a path without the first '/' (ex: swarmType/swarmPhase/before)\n     * @return {Array<Set<function>>}\n     */\n    function getInterceptorsForKey(key) {\n        if (key.startsWith('/')) {\n            _warning(`Interceptor called on key ${key} starting with '/', automatically removing it`);\n            key = key.substring(1);\n        }\n\n        const keyElements = key.split('/');\n        const keysGenerator = createArrayGenerator(keyElements);\n\n        return getValueRecursively([rules], keysGenerator);\n    }\n\n    /**\n     * It works like a BFS search returning the leafs resulting from traversing the internal nodes with corresponding\n     * names given for each level (depth) by `keysGenerator`\n     * @param {Array<Map>} searchableNodes\n     * @param {IterableIterator} keysGenerator - it has the particularity that done is set on last element, not after it\n     * @return {Array<Set<function>>}\n     */\n    function getValueRecursively(searchableNodes, keysGenerator) {\n        const {value: nodeName, done} = keysGenerator.next();\n\n        const nextNodes = [];\n\n        for (const nodeInRules of searchableNodes) {\n            const nextNodeForAll = nodeInRules.get('*');\n            const nextNode = nodeInRules.get(nodeName);\n\n            if (typeof nextNode !== \"undefined\") {\n                nextNodes.push(nextNode);\n            }\n\n            if (typeof nextNodeForAll !== \"undefined\") {\n                nextNodes.push(nextNodeForAll);\n            }\n\n        }\n\n        if (done) {\n            return nextNodes;\n        }\n\n        return getValueRecursively(nextNodes, keysGenerator);\n    }\n\n\n    /************* PUBLIC METHODS *************/\n\n    this.register = function (swarmTypeName, phaseName, when, fn) {\n        verifyWhenOption(when);\n        verifyIsFunctionType(fn);\n\n        const resolvedSwarmTypeName = resolveNamespaceResolution(swarmTypeName);\n        const keys = createArrayGenerator([resolvedSwarmTypeName, phaseName, when]);\n\n        registerRecursiveRule(rules, keys, fn);\n    };\n\n    // this.unregister = function () { }\n\n    this.callInterceptors = function (key, targetObject, args) {\n        const interceptors = getInterceptorsForKey(key);\n\n        if (interceptors) {\n            for (const interceptorSet of interceptors) {\n                for (const fn of interceptorSet) { // interceptors on key '*' are called before those specified by name\n                    fn.apply(targetObject, args);\n                }\n            }\n        }\n    };\n}\n\n\nexports.createInterceptorRegistry = function () {\n    return new InterceptorRegistry();\n};\n","/*\n Initial License: (c) Axiologic Research & Alboaie Sînică.\n Contributors: Axiologic Research , PrivateSky project\n Code License: LGPL or MIT.\n */\n\nvar util = require(\"util\");\nvar fs = require(\"fs\");\ncprint = console.log;\nwprint = console.warn;\ndprint = console.debug;\neprint = console.error;\n\n\n/**\n * Shortcut to JSON.stringify\n * @param obj\n */\nJ = function (obj) {\n    return JSON.stringify(obj);\n}\n\n\n/**\n * Print swarm contexts (Messages) and easier to read compared with J\n * @param obj\n * @return {string}\n */\nexports.cleanDump = function (obj) {\n    var o = obj.valueOf();\n    var meta = {\n        swarmTypeName:o.meta.swarmTypeName\n    };\n    return \"\\t swarmId: \" + o.meta.swarmId + \"{\\n\\t\\tmeta: \"    + J(meta) +\n        \"\\n\\t\\tpublic: \"        + J(o.publicVars) +\n        \"\\n\\t\\tprotected: \"     + J(o.protectedVars) +\n        \"\\n\\t\\tprivate: \"       + J(o.privateVars) + \"\\n\\t}\\n\";\n}\n\n//M = exports.cleanDump;\n/**\n * Experimental functions\n */\n\n\n/*\n\n logger      = monitor.logger;\n assert      = monitor.assert;\n throwing    = monitor.exceptions;\n\n\n var temporaryLogBuffer = [];\n\n var currentSwarmComImpl = null;\n\n logger.record = function(record){\n if(currentSwarmComImpl===null){\n temporaryLogBuffer.push(record);\n } else {\n currentSwarmComImpl.recordLog(record);\n }\n }\n\n var container = require(\"dicontainer\").container;\n\n container.service(\"swarmLoggingMonitor\", [\"swarmingIsWorking\", \"swarmComImpl\"], function(outOfService,swarming, swarmComImpl){\n\n if(outOfService){\n if(!temporaryLogBuffer){\n temporaryLogBuffer = [];\n }\n } else {\n var tmp = temporaryLogBuffer;\n temporaryLogBuffer = [];\n currentSwarmComImpl = swarmComImpl;\n logger.record = function(record){\n currentSwarmComImpl.recordLog(record);\n }\n\n tmp.forEach(function(record){\n logger.record(record);\n });\n }\n })\n\n */\nuncaughtExceptionString = \"\";\nuncaughtExceptionExists = false;\nif(typeof globalVerbosity == 'undefined'){\n    globalVerbosity = false;\n}\n\nvar DEBUG_START_TIME = new Date().getTime();\n\nfunction getDebugDelta(){\n    var currentTime = new Date().getTime();\n    return currentTime - DEBUG_START_TIME;\n}\n\n/**\n * Debug functions, influenced by globalVerbosity global variable\n * @param txt\n */\ndprint = function (txt) {\n    if (globalVerbosity == true) {\n        if (thisAdapter.initilised ) {\n            console.log(\"DEBUG: [\" + thisAdapter.nodeName + \"](\" + getDebugDelta()+ \"):\"+txt);\n        }\n        else {\n            console.log(\"DEBUG: (\" + getDebugDelta()+ \"):\"+txt);\n            console.log(\"DEBUG: \" + txt);\n        }\n    }\n}\n\n/**\n * obsolete!?\n * @param txt\n */\naprint = function (txt) {\n    console.log(\"DEBUG: [\" + thisAdapter.nodeName + \"]: \" + txt);\n}\n\n\n\n/**\n * Utility function usually used in tests, exit current process after a while\n * @param msg\n * @param timeout\n */\ndelayExit = function (msg, retCode,timeout) {\n    if(retCode == undefined){\n        retCode = ExitCodes.UnknownError;\n    }\n\n    if(timeout == undefined){\n        timeout = 100;\n    }\n\n    if(msg == undefined){\n        msg = \"Delaying exit with \"+ timeout + \"ms\";\n    }\n\n    console.log(msg);\n    setTimeout(function () {\n        process.exit(retCode);\n    }, timeout);\n}\n\n\nfunction localLog (logType, message, err) {\n    var time = new Date();\n    var now = time.getDate() + \"-\" + (time.getMonth() + 1) + \",\" + time.getHours() + \":\" + time.getMinutes();\n    var msg;\n\n    msg = '[' + now + '][' + thisAdapter.nodeName + '] ' + message;\n\n    if (err != null && err != undefined) {\n        msg += '\\n     Err: ' + err.toString();\n        if (err.stack && err.stack != undefined)\n            msg += '\\n     Stack: ' + err.stack + '\\n';\n    }\n\n    cprint(msg);\n    if(thisAdapter.initilised){\n        try{\n            fs.appendFileSync(getSwarmFilePath(thisAdapter.config.logsPath + \"/\" + logType), msg);\n        } catch(err){\n            console.log(\"Failing to write logs in \", thisAdapter.config.logsPath );\n        }\n\n    }\n}\n\n\n// printf = function (...params) {\n//     var args = []; // empty array\n//     // copy all other arguments we want to \"pass through\"\n//     for (var i = 0; i < params.length; i++) {\n//         args.push(params[i]);\n//     }\n//     var out = util.format.apply(this, args);\n//     console.log(out);\n// }\n//\n// sprintf = function (...params) {\n//     var args = []; // empty array\n//     for (var i = 0; i < params.length; i++) {\n//         args.push(params[i]);\n//     }\n//     return util.format.apply(this, args);\n// }\n\n","\n\nfunction SwarmsInstancesManager(){\n    var swarmAliveInstances = {\n\n    }\n\n    this.waitForSwarm = function(callback, swarm, keepAliveCheck){\n\n        function doLogic(){\n            var swarmId = swarm.getInnerValue().meta.swarmId;\n            var watcher = swarmAliveInstances[swarmId];\n            if(!watcher){\n                watcher = {\n                    swarm:swarm,\n                    callback:callback,\n                    keepAliveCheck:keepAliveCheck\n                }\n                swarmAliveInstances[swarmId] = watcher;\n            }\n        }\n\n        function filter(){\n            return swarm.getInnerValue().meta.swarmId;\n        }\n\n        //$$.uidGenerator.wait_for_condition(condition,doLogic);\n        swarm.observe(doLogic, null, filter);\n    }\n\n    function cleanSwarmWaiter(swarmSerialisation){ // TODO: add better mechanisms to prevent memory leaks\n        var swarmId = swarmSerialisation.meta.swarmId;\n        var watcher = swarmAliveInstances[swarmId];\n\n        if(!watcher){\n            $$.errorHandler.warning(\"Invalid swarm received: \" + swarmId);\n            return;\n        }\n\n        var args = swarmSerialisation.meta.args;\n        args.push(swarmSerialisation);\n\n        watcher.callback.apply(null, args);\n        if(!watcher.keepAliveCheck()){\n            delete swarmAliveInstances[swarmId];\n        }\n    }\n\n    this.revive_swarm = function(swarmSerialisation){\n\n\n        var swarmId     = swarmSerialisation.meta.swarmId;\n        var swarmType   = swarmSerialisation.meta.swarmTypeName;\n        var instance    = swarmAliveInstances[swarmId];\n\n        var swarm;\n\n        if(instance){\n            swarm = instance.swarm;\n            swarm.update(swarmSerialisation);\n\n        } else {\n            swarm = $$.swarm.start(swarmType);\n            swarm.update(swarmSerialisation);\n            /*swarm = $$.swarm.start(swarmType, swarmSerialisation);*/\n        }\n\n        if (swarmSerialisation.meta.command == \"asyncReturn\") {\n            var co = $$.PSK_PubSub.publish($$.CONSTANTS.SWARM_RETURN, swarmSerialisation);\n            console.log(\"Subscribers listening on\", $$.CONSTANTS.SWARM_RETURN, co);\n            // cleanSwarmWaiter(swarmSerialisation);\n        } else if (swarmSerialisation.meta.command == \"executeSwarmPhase\") {\n            swarm.runPhase(swarmSerialisation.meta.phaseName, swarmSerialisation.meta.args);\n        } else {\n            console.log(\"Unknown command\", swarmSerialisation.meta.command, \"in swarmSerialisation.meta.command\");\n        }\n\n        return swarm;\n    }\n}\n\n\n$$.swarmsInstancesManager = new SwarmsInstancesManager();\n\n\n","exports.createForObject = function(valueObject, thisObject, localId){\n\tvar ret = require(\"./base\").createForObject(valueObject, thisObject, localId);\n\n\tret.swarm           = null;\n\tret.onReturn        = null;\n\tret.onResult        = null;\n\tret.asyncReturn     = null;\n\tret.return          = null;\n\tret.home            = null;\n\tret.isPersisted  \t= function () {\n\t\treturn thisObject.getMetadata('persisted') === true;\n\t};\n\n\treturn ret;\n};","var beesHealer = require(\"swarmutils\").beesHealer;\nvar swarmDebug = require(\"../SwarmDebug\");\n\nexports.createForObject = function(valueObject, thisObject, localId){\n\tvar ret = {};\n\n\tfunction filterForSerialisable (valueObject){\n\t\treturn valueObject.meta.swarmId;\n\t}\n\n\tvar swarmFunction = function(context, phaseName){\n\t\tvar args =[];\n\t\tfor(var i = 2; i < arguments.length; i++){\n\t\t\targs.push(arguments[i]);\n\t\t}\n\n\t\t//make the execution at level 0  (after all pending events) and wait to have a swarmId\n\t\tret.observe(function(){\n\t\t\tbeesHealer.asJSON(valueObject, phaseName, args, function(err,jsMsg){\n\t\t\t\tjsMsg.meta.target = context;\n\t\t\t\tvar subscribersCount = $$.PSK_PubSub.publish($$.CONSTANTS.SWARM_FOR_EXECUTION, jsMsg);\n\t\t\t\tif(!subscribersCount){\n\t\t\t\t\tconsole.log(`Nobody listening for <${$$.CONSTANTS.SWARM_FOR_EXECUTION}>!`);\n\t\t\t\t}\n\t\t\t});\n\t\t},null,filterForSerialisable);\n\n\t\tret.notify();\n\n\n\t\treturn thisObject;\n\t};\n\n\tvar asyncReturn = function(err, result){\n\t\tvar context = valueObject.protectedVars.context;\n\n\t\tif(!context && valueObject.meta.waitStack){\n\t\t\tcontext = valueObject.meta.waitStack.pop();\n\t\t\tvalueObject.protectedVars.context = context;\n\t\t}\n\n\t\tbeesHealer.asJSON(valueObject, \"__return__\", [err, result], function(err,jsMsg){\n\t\t\tjsMsg.meta.command = \"asyncReturn\";\n\t\t\tif(!context){\n\t\t\t\tcontext = valueObject.meta.homeSecurityContext;//TODO: CHECK THIS\n\n\t\t\t}\n\t\t\tjsMsg.meta.target = context;\n\n\t\t\tif(!context){\n\t\t\t\t$$.errorHandler.error(new Error(\"Asynchronous return inside of a swarm that does not wait for results\"));\n\t\t\t} else {\n\t\t\t\t$$.PSK_PubSub.publish($$.CONSTANTS.SWARM_FOR_EXECUTION, jsMsg);\n\t\t\t}\n\t\t});\n\t};\n\n\tfunction home(err, result){\n\t\tbeesHealer.asJSON(valueObject, \"home\", [err, result], function(err,jsMsg){\n\t\t\tvar context = valueObject.meta.homeContext;\n\t\t\tjsMsg.meta.target = context;\n\t\t\t$$.PSK_PubSub.publish($$.CONSTANTS.SWARM_FOR_EXECUTION, jsMsg);\n\t\t});\n\t}\n\n\n\n\tfunction waitResults(callback, keepAliveCheck, swarm){\n\t\tif(!swarm){\n\t\t\tswarm = this;\n\t\t}\n\t\tif(!keepAliveCheck){\n\t\t\tkeepAliveCheck = function(){\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\t\tvar inner = swarm.getInnerValue();\n\t\tif(!inner.meta.waitStack){\n\t\t\tinner.meta.waitStack = [];\n\t\t\tinner.meta.waitStack.push($$.securityContext)\n\t\t}\n\t\t$$.swarmsInstancesManager.waitForSwarm(callback, swarm, keepAliveCheck);\n\t}\n\n\n\tfunction getInnerValue(){\n\t\treturn valueObject;\n\t}\n\n\tfunction runPhase(functName, args){\n\t\tvar func = valueObject.myFunctions[functName];\n\t\tif(func){\n\t\t\tfunc.apply(thisObject, args);\n\t\t} else {\n\t\t\t$$.errorHandler.syntaxError(functName, valueObject, \"Function \" + functName + \" does not exist!\");\n\t\t}\n\n\t}\n\n\tfunction update(serialisation){\n\t\tbeesHealer.jsonToNative(serialisation,valueObject);\n\t}\n\n\n\tfunction valueOf(){\n\t\tvar ret = {};\n\t\tret.meta                = valueObject.meta;\n\t\tret.publicVars          = valueObject.publicVars;\n\t\tret.privateVars         = valueObject.privateVars;\n\t\tret.protectedVars       = valueObject.protectedVars;\n\t\treturn ret;\n\t}\n\n\tfunction toString (){\n\t\treturn swarmDebug.cleanDump(thisObject.valueOf());\n\t}\n\n\n\tfunction createParallel(callback){\n\t\treturn require(\"../../parallelJoinPoint\").createJoinPoint(thisObject, callback, $$.__intern.mkArgs(arguments,1));\n\t}\n\n\tfunction createSerial(callback){\n\t\treturn require(\"../../serialJoinPoint\").createSerialJoinPoint(thisObject, callback, $$.__intern.mkArgs(arguments,1));\n\t}\n\n\tfunction inspect(){\n\t\treturn swarmDebug.cleanDump(thisObject.valueOf());\n\t}\n\n\tfunction constructor(){\n\t\treturn SwarmDescription;\n\t}\n\n\tfunction ensureLocalId(){\n\t\tif(!valueObject.localId){\n\t\t\tvalueObject.localId = valueObject.meta.swarmTypeName + \"-\" + localId;\n\t\t\tlocalId++;\n\t\t}\n\t}\n\n\tfunction observe(callback, waitForMore, filter){\n\t\tif(!waitForMore){\n\t\t\twaitForMore = function (){\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\tensureLocalId();\n\n\t\t$$.PSK_PubSub.subscribe(valueObject.localId, callback, waitForMore, filter);\n\t}\n\n\tfunction toJSON(prop){\n\t\t//preventing max call stack size exceeding on proxy auto referencing\n\t\t//replace {} as result of JSON(Proxy) with the string [Object protected object]\n\t\treturn \"[Object protected object]\";\n\t}\n\n\tfunction getJSON(callback){\n\t\treturn\tbeesHealer.asJSON(valueObject, null, null,callback);\n\t}\n\n\tfunction notify(event){\n\t\tif(!event){\n\t\t\tevent = valueObject;\n\t\t}\n\t\tensureLocalId();\n\t\t$$.PSK_PubSub.publish(valueObject.localId, event);\n\t}\n\n\tfunction getMeta(name){\n\t\treturn valueObject.getMeta(name);\n\t}\n\n\tfunction setMeta(name, value){\n\t\treturn valueObject.setMeta(name, value);\n\t}\n\n\tret.setMeta\t\t\t= setMeta;\n\tret.getMeta\t\t\t= getMeta;\n\tret.swarm           = swarmFunction;\n\tret.notify          = notify;\n\tret.getJSON    \t    = getJSON;\n\tret.toJSON          = toJSON;\n\tret.observe         = observe;\n\tret.inspect         = inspect;\n\tret.join            = createParallel;\n\tret.parallel        = createParallel;\n\tret.serial          = createSerial;\n\tret.valueOf         = valueOf;\n\tret.update          = update;\n\tret.runPhase        = runPhase;\n\tret.onReturn        = waitResults;\n\tret.onResult        = waitResults;\n\tret.asyncReturn     = asyncReturn;\n\tret.return          = asyncReturn;\n\tret.getInnerValue   = getInnerValue;\n\tret.home            = home;\n\tret.toString        = toString;\n\tret.constructor     = constructor;\n\tret.setMetadata\t\t= valueObject.setMeta.bind(valueObject);\n\tret.getMetadata\t\t= valueObject.getMeta.bind(valueObject);\n\n\treturn ret;\n\n};\n","exports.createForObject = function(valueObject, thisObject, localId){\n\tvar ret = require(\"./base\").createForObject(valueObject, thisObject, localId);\n\n\tret.swarm           = null;\n\tret.onReturn        = null;\n\tret.onResult        = null;\n\tret.asyncReturn     = null;\n\tret.return          = null;\n\tret.home            = null;\n\n\treturn ret;\n};","exports.createForObject = function(valueObject, thisObject, localId){\n\treturn require(\"./base\").createForObject(valueObject, thisObject, localId);\n};","/*\nInitial License: (c) Axiologic Research & Alboaie Sînică.\nContributors: Axiologic Research , PrivateSky project\nCode License: LGPL or MIT.\n*/\n\n//var fs = require(\"fs\");\n//var path = require(\"path\");\n\n\nfunction SwarmLibrary(prefixName, folder){\n    var self = this;\n    function wrapCall(original, prefixName){\n        return function(...args){\n            //console.log(\"prefixName\", prefixName)\n            var previousPrefix = $$.libraryPrefix;\n            var previousLibrary = $$.__global.currentLibrary;\n\n            $$.libraryPrefix = prefixName;\n            $$.__global.currentLibrary = self;\n            try{\n                var ret = original.apply(this, args);\n                $$.libraryPrefix = previousPrefix ;\n                $$.__global.currentLibrary = previousLibrary;\n            }catch(err){\n                $$.libraryPrefix = previousPrefix ;\n                $$.__global.currentLibrary = previousLibrary;\n                throw err;\n            }\n            return ret;\n        }\n    }\n\n    $$.libraries[prefixName] = this;\n    var prefixedRequire = wrapCall(function(path){\n        return require(path);\n    }, prefixName);\n\n    function includeAllInRoot(folder) {\n        if(typeof folder != \"string\"){\n            //we assume that it is a library module properly required with require and containing $$.library\n            for(var v in folder){\n                $$.registerSwarmDescription(prefixName,v, prefixName + \".\" + v,  folder[v]);\n            }\n\n            var newNames = $$.__global.requireLibrariesNames[prefixName];\n            for(var v in newNames){\n                self[v] =  newNames[v];\n            }\n            return folder;\n        }\n\n\n        var res = prefixedRequire(folder); // a library is just a module\n        if(typeof res.__autogenerated_privatesky_libraryName != \"undefined\"){\n            var swarms = $$.__global.requireLibrariesNames[res.__autogenerated_privatesky_libraryName];\n        } else {\n            var swarms = $$.__global.requireLibrariesNames[folder];\n        }\n            var existingName;\n            for(var v in swarms){\n                existingName = swarms[v];\n                self[v] = existingName;\n                $$.registerSwarmDescription(prefixName,v, prefixName + \".\" + v,  existingName);\n            }\n        return res;\n    }\n\n    function wrapSwarmRelatedFunctions(space, prefixName){\n        var ret = {};\n        var names = [\"create\", \"describe\", \"start\", \"restart\"];\n        for(var i = 0; i<names.length; i++ ){\n            ret[names[i]] = wrapCall(space[names[i]], prefixName);\n        }\n        return ret;\n    }\n\n    this.callflows        = this.callflow   = wrapSwarmRelatedFunctions($$.callflows, prefixName);\n    this.swarms           = this.swarm      = wrapSwarmRelatedFunctions($$.swarms, prefixName);\n    this.contracts        = this.contract   = wrapSwarmRelatedFunctions($$.contracts, prefixName);\n    includeAllInRoot(folder, prefixName);\n}\n\nexports.loadLibrary = function(prefixName, folder){\n    var existing = $$.libraries[prefixName];\n    if(existing ){\n        if(!(existing instanceof SwarmLibrary)){\n            var sL = new SwarmLibrary(prefixName, folder);\n            for(var prop in existing){\n                sL[prop] = existing[prop];\n            }\n            return sL;\n        }\n        if(folder) {\n            $$.errorHandler.warning(\"Reusing already loaded library \" + prefixName + \"could be an error!\");\n        }\n        return existing;\n    }\n    //var absolutePath = path.resolve(folder);\n    return new SwarmLibrary(prefixName, folder);\n}\n\n","/*\n require and $$.require are overwriting the node.js defaults in loading modules for increasing security,speed and making it work to the privatesky runtime build with browserify.\n The privatesky code for domains should work in node and browsers.\n */\n\n\nif (typeof(window) !== \"undefined\") {\n    global = window;\n}\n\n\nif (typeof(global.$$) == \"undefined\") {\n    global.$$ = {};\n    $$.__global = {};\n}\n\nif (typeof($$.__global) == \"undefined\") {\n    $$.__global = {};\n}\n\nif (typeof($$.__global.requireLibrariesNames) == \"undefined\") {\n    $$.__global.currentLibraryName = null;\n    $$.__global.requireLibrariesNames = {};\n}\n\n\nif (typeof($$.__runtimeModules) == \"undefined\") {\n    $$.__runtimeModules = {};\n}\n\n\nif (typeof(global.functionUndefined) == \"undefined\") {\n    global.functionUndefined = function () {\n        console.log(\"Called of an undefined function!!!!\");\n        throw new Error(\"Called of an undefined function\");\n    };\n    if (typeof(global.webshimsRequire) == \"undefined\") {\n        global.webshimsRequire = global.functionUndefined;\n    }\n\n    if (typeof(global.domainRequire) == \"undefined\") {\n        global.domainRequire = global.functionUndefined;\n    }\n\n    if (typeof(global.pskruntimeRequire) == \"undefined\") {\n        global.pskruntimeRequire = global.functionUndefined;\n    }\n}\n\nif (typeof($$.log) == \"undefined\") {\n    $$.log = function (...args) {\n        console.log(args.join(\" \"));\n    }\n}\n\n\nconst weAreInbrowser = (typeof ($$.browserRuntime) != \"undefined\");\nconst weAreInSandbox = (typeof global.require !== 'undefined');\n\n\nconst pastRequests = {};\n\nfunction preventRecursiveRequire(request) {\n    if (pastRequests[request]) {\n        const err = new Error(\"Preventing recursive require for \" + request);\n        err.type = \"PSKIgnorableError\";\n        throw err;\n    }\n\n}\n\nfunction disableRequire(request) {\n    pastRequests[request] = true;\n}\n\nfunction enableRequire(request) {\n    pastRequests[request] = false;\n}\n\n\nfunction requireFromCache(request) {\n    const existingModule = $$.__runtimeModules[request];\n    return existingModule;\n}\n\nfunction wrapStep(callbackName) {\n    const callback = global[callbackName];\n\n    if (callback === undefined) {\n        return null;\n    }\n\n    if (callback === global.functionUndefined) {\n        return null;\n    }\n\n    return function (request) {\n        const result = callback(request);\n        $$.__runtimeModules[request] = result;\n        return result;\n    }\n}\n\nfunction tryRequireSequence(originalRequire, request) {\n    let arr;\n    if (originalRequire) {\n        arr = $$.__requireFunctionsChain.slice();\n        arr.push(originalRequire);\n    } else {\n        arr = $$.__requireFunctionsChain;\n    }\n\n    preventRecursiveRequire(request);\n    disableRequire(request);\n    let result;\n    const previousRequire = $$.__global.currentLibraryName;\n    let previousRequireChanged = false;\n\n    if (!previousRequire) {\n        // console.log(\"Loading library for require\", request);\n        $$.__global.currentLibraryName = request;\n\n        if (typeof $$.__global.requireLibrariesNames[request] == \"undefined\") {\n            $$.__global.requireLibrariesNames[request] = {};\n            //$$.__global.requireLibrariesDescriptions[request]   = {};\n        }\n        previousRequireChanged = true;\n    }\n    for (let i = 0; i < arr.length; i++) {\n        const func = arr[i];\n        try {\n\n            if (func === global.functionUndefined) continue;\n            result = func(request);\n\n            if (result) {\n                break;\n            }\n\n        } catch (err) {\n            if (err.type !== \"PSKIgnorableError\") {\n                $$.log(\"Require encountered an error while loading \", request, \"\\nCause:\\n\", err.stack);\n            }\n        }\n    }\n\n    if (!result) {\n        $$.log(\"Failed to load module \", request, result);\n    }\n\n    enableRequire(request);\n    if (previousRequireChanged) {\n        //console.log(\"End loading library for require\", request, $$.__global.requireLibrariesNames[request]);\n        $$.__global.currentLibraryName = null;\n    }\n    return result;\n}\n\nif (typeof($$.require) == \"undefined\") {\n\n    $$.__requireList = [\"webshimsRequire\", \"pskruntimeRequire\"];\n    $$.__requireFunctionsChain = [];\n\n    $$.requireBundle = function (name) {\n        name += \"Require\";\n        $$.__requireList.push(name);\n        const arr = [requireFromCache];\n        $$.__requireList.forEach(function (item) {\n            const callback = wrapStep(item);\n            if (callback) {\n                arr.push(callback);\n            }\n        });\n\n        $$.__requireFunctionsChain = arr;\n    };\n\n    $$.requireBundle(\"init\");\n\n    if (weAreInbrowser) {\n        $$.log(\"Defining global require in browser\");\n\n\n        global.require = function (request) {\n\n            ///*[requireFromCache, wrapStep(webshimsRequire), , wrapStep(pskruntimeRequire), wrapStep(domainRequire)*]\n            return tryRequireSequence(null, request);\n        }\n    } else\n        if (weAreInSandbox) {\n        // require should be provided when code is loaded in browserify\n        const bundleRequire = require;\n\n        $$.requireBundle('sandboxBase');\n        // this should be set up by sandbox prior to\n        const sandboxRequire = global.require;\n        global.crypto = require('crypto');\n\n        function newLoader(request) {\n            // console.log(\"newLoader:\", request);\n            //preventRecursiveRequire(request);\n            const self = this;\n\n            // console.log('trying to load ', request);\n\n            function tryBundleRequire(...args) {\n                //return $$.__originalRequire.apply(self,args);\n                //return Module._load.apply(self,args)\n                let res;\n                try {\n                    res = sandboxRequire.apply(self, args);\n                } catch (err) {\n                    if (err.code === \"MODULE_NOT_FOUND\") {\n                        const p = path.join(process.cwd(), request);\n                        res = sandboxRequire.apply(self, [p]);\n                        request = p;\n                    } else {\n                        throw err;\n                    }\n                }\n                return res;\n            }\n\n            let res;\n\n\n            res = tryRequireSequence(tryBundleRequire, request);\n\n\n            return res;\n        }\n\n        global.require = newLoader;\n\n    } else {  //we are in node\n        const path = require(\"path\");\n        $$.__runtimeModules[\"crypto\"] = require(\"crypto\");\n        $$.__runtimeModules[\"util\"] = require(\"util\");\n\n        const Module = require('module');\n        $$.__runtimeModules[\"module\"] = Module;\n\n        $$.log(\"Redefining require for node\");\n\n        $$.__originalRequire = Module._load;\n        const moduleOriginalRequire = Module.prototype.require;\n\n        function newLoader(request) {\n            // console.log(\"newLoader:\", request);\n            //preventRecursiveRequire(request);\n            const self = this;\n\n            function originalRequire(...args) {\n                //return $$.__originalRequire.apply(self,args);\n                //return Module._load.apply(self,args)\n                let res;\n                try {\n                    res = moduleOriginalRequire.apply(self, args);\n                } catch (err) {\n                    if (err.code === \"MODULE_NOT_FOUND\") {\n                        const p = path.join(process.cwd(), request);\n                        res = moduleOriginalRequire.apply(self, [p]);\n                        request = p;\n                    } else {\n                        throw err;\n                    }\n                }\n                return res;\n            }\n\n            function currentFolderRequire(request) {\n                return\n            }\n\n            //[requireFromCache, wrapStep(pskruntimeRequire), wrapStep(domainRequire), originalRequire]\n            return tryRequireSequence(originalRequire, request);\n        }\n\n        Module.prototype.require = newLoader;\n    }\n\n    $$.require = require;\n}\n","\nvar joinCounter = 0;\n\nfunction ParallelJoinPoint(swarm, callback, args){\n    joinCounter++;\n    var channelId = \"ParallelJoinPoint\" + joinCounter;\n    var self = this;\n    var counter = 0;\n    var stopOtherExecution     = false;\n\n    function executionStep(stepFunc, localArgs, stop){\n\n        this.doExecute = function(){\n            if(stopOtherExecution){\n                return false;\n            }\n            try{\n                stepFunc.apply(swarm, localArgs);\n                if(stop){\n                    stopOtherExecution = true;\n                    return false;\n                }\n                return true; //everyting is fine\n            } catch(err){\n                args.unshift(err);\n                sendForSoundExecution(callback, args, true);\n                return false; //stop it, do not call again anything\n            }\n        }\n    }\n\n    if(typeof callback !== \"function\"){\n        $$.errorHandler.syntaxError(\"invalid join\",swarm, \"invalid function at join in swarm\");\n        return;\n    }\n\n    $$.PSK_PubSub.subscribe(channelId,function(forExecution){\n        if(stopOtherExecution){\n            return ;\n        }\n\n        try{\n            if(forExecution.doExecute()){\n                decCounter();\n            } // had an error...\n        } catch(err){\n            //console.log(err);\n            //$$.errorHandler.syntaxError(\"__internal__\",swarm, \"exception in the execution of the join function of a parallel task\");\n        }\n    });\n\n    function incCounter(){\n        if(testIfUnderInspection()){\n            //preventing inspector from increasing counter when reading the values for debug reason\n            //console.log(\"preventing inspection\");\n            return;\n        }\n        counter++;\n    }\n\n    function testIfUnderInspection(){\n        var res = false;\n        var constArgv = process.execArgv.join();\n        if(constArgv.indexOf(\"inspect\")!==-1 || constArgv.indexOf(\"debug\")!==-1){\n            //only when running in debug\n            var callstack = new Error().stack;\n            if(callstack.indexOf(\"DebugCommandProcessor\")!==-1){\n                console.log(\"DebugCommandProcessor detected!\");\n                res = true;\n            }\n        }\n        return res;\n    }\n\n    function sendForSoundExecution(funct, args, stop){\n        var obj = new executionStep(funct, args, stop);\n        $$.PSK_PubSub.publish(channelId, obj); // force execution to be \"sound\"\n    }\n\n    function decCounter(){\n        counter--;\n        if(counter == 0) {\n            args.unshift(null);\n            sendForSoundExecution(callback, args, false);\n        }\n    }\n\n    var inner = swarm.getInnerValue();\n\n    function defaultProgressReport(err, res){\n        if(err) {\n            throw err;\n        }\n        return {\n            text:\"Parallel execution progress event\",\n            swarm:swarm,\n            args:args,\n            currentResult:res\n        };\n    }\n\n    function mkFunction(name){\n        return function(...args){\n            var f = defaultProgressReport;\n            if(name != \"progress\"){\n                f = inner.myFunctions[name];\n            }\n            var args = $$.__intern.mkArgs(args, 0);\n            sendForSoundExecution(f, args, false);\n            return __proxyObject;\n        }\n    }\n\n\n    this.get = function(target, prop, receiver){\n        if(inner.myFunctions.hasOwnProperty(prop) || prop == \"progress\"){\n            incCounter();\n            return mkFunction(prop);\n        }\n        return swarm[prop];\n    };\n\n    var __proxyObject;\n\n    this.__setProxyObject = function(p){\n        __proxyObject = p;\n    }\n}\n\nexports.createJoinPoint = function(swarm, callback, args){\n    var jp = new ParallelJoinPoint(swarm, callback, args);\n    var inner = swarm.getInnerValue();\n    var p = new Proxy(inner, jp);\n    jp.__setProxyObject(p);\n    return p;\n};","\nvar joinCounter = 0;\n\nfunction SerialJoinPoint(swarm, callback, args){\n\n    joinCounter++;\n\n    var self = this;\n    var channelId = \"SerialJoinPoint\" + joinCounter;\n\n    if(typeof callback !== \"function\"){\n        $$.errorHandler.syntaxError(\"unknown\", swarm, \"invalid function given to serial in swarm\");\n        return;\n    }\n\n    var inner = swarm.getInnerValue();\n\n\n    function defaultProgressReport(err, res){\n        if(err) {\n            throw err;\n        }\n        return res;\n    }\n\n\n    var functionCounter     = 0;\n    var executionCounter    = 0;\n\n    var plannedExecutions   = [];\n    var plannedArguments    = {};\n\n    function mkFunction(name, pos){\n        //console.log(\"Creating function \", name, pos);\n        plannedArguments[pos] = undefined;\n\n        function triggetNextStep(){\n            if(plannedExecutions.length == executionCounter || plannedArguments[executionCounter] )  {\n                $$.PSK_PubSub.publish(channelId, self);\n            }\n        }\n\n        var f = function (...args){\n            if(executionCounter != pos) {\n                plannedArguments[pos] = args;\n                //console.log(\"Delaying function:\", executionCounter, pos, plannedArguments, arguments, functionCounter);\n                return __proxy;\n            } else{\n                if(plannedArguments[pos]){\n                    //console.log(\"Executing  function:\", executionCounter, pos, plannedArguments, arguments, functionCounter);\n\t\t\t\t\targs = plannedArguments[pos];\n                } else {\n                    plannedArguments[pos] = args;\n                    triggetNextStep();\n                    return __proxy;\n                }\n            }\n\n            var f = defaultProgressReport;\n            if(name != \"progress\"){\n                f = inner.myFunctions[name];\n            }\n\n\n            try{\n                f.apply(self,args);\n            } catch(err){\n                    args.unshift(err);\n                    callback.apply(swarm,args); //error\n                    $$.PSK_PubSub.unsubscribe(channelId,runNextFunction);\n                return; //terminate execution with an error...!\n            }\n            executionCounter++;\n\n            triggetNextStep();\n\n            return __proxy;\n        };\n\n        plannedExecutions.push(f);\n        functionCounter++;\n        return f;\n    }\n\n     var finished = false;\n\n    function runNextFunction(){\n        if(executionCounter == plannedExecutions.length ){\n            if(!finished){\n                args.unshift(null);\n                callback.apply(swarm,args);\n                finished = true;\n                $$.PSK_PubSub.unsubscribe(channelId,runNextFunction);\n            } else {\n                console.log(\"serial construct is using functions that are called multiple times...\");\n            }\n        } else {\n            plannedExecutions[executionCounter]();\n        }\n    }\n\n    $$.PSK_PubSub.subscribe(channelId,runNextFunction); // force it to be \"sound\"\n\n\n    this.get = function(target, prop, receiver){\n        if(prop == \"progress\" || inner.myFunctions.hasOwnProperty(prop)){\n            return mkFunction(prop, functionCounter);\n        }\n        return swarm[prop];\n    }\n\n    var __proxy;\n    this.setProxyObject = function(p){\n        __proxy = p;\n    }\n}\n\nexports.createSerialJoinPoint = function(swarm, callback, args){\n    var jp = new SerialJoinPoint(swarm, callback, args);\n    var inner = swarm.getInnerValue();\n    var p = new Proxy(inner, jp);\n    jp.setProxyObject(p);\n    return p;\n}","const OwM = require(\"swarmutils\").OwM;\n\nvar swarmDescriptionsRegistry = {};\n\n\n$$.registerSwarmDescription =  function(libraryName, shortName, swarmTypeName, description){\n    if(!$$.libraries[libraryName]){\n        $$.libraries[libraryName] = {};\n    }\n\n    if(!$$.__global.requireLibrariesNames[libraryName]){\n        $$.__global.requireLibrariesNames[libraryName] = {};\n    }\n\n    $$.libraries[libraryName][shortName] = description;\n    //console.log(\"Registering \", libraryName,shortName, $$.__global.currentLibraryName);\n    if($$.__global.currentLibraryName){\n        $$.__global.requireLibrariesNames[$$.__global.currentLibraryName][shortName] = libraryName + \".\" + shortName;\n    }\n\n    $$.__global.requireLibrariesNames[libraryName][shortName] = swarmTypeName;\n\n    if(typeof description == \"string\"){\n        description = swarmDescriptionsRegistry[description];\n    }\n    swarmDescriptionsRegistry[swarmTypeName] = description;\n}\n\n\nvar currentLibraryCounter = 0;\n$$.library = function(callback){\n    currentLibraryCounter++;\n    var previousCurrentLibrary = $$.__global.currentLibraryName;\n    var libraryName = \"___privatesky_library\"+currentLibraryCounter;\n    var ret = $$.__global.requireLibrariesNames[libraryName] = {};\n    $$.__global.currentLibraryName = libraryName;\n    callback();\n    $$.__global.currentLibraryName = previousCurrentLibrary;\n    ret.__autogenerated_privatesky_libraryName = libraryName;\n    return ret;\n}\n\nfunction SwarmSpace(swarmType, utils) {\n\n    var beesHealer = require(\"swarmutils\").beesHealer;\n\n    function getFullName(shortName){\n        var fullName;\n        if(shortName && shortName.includes(\".\")) {\n            fullName = shortName;\n        } else {\n            fullName = $$.libraryPrefix + \".\" + shortName;\n        }\n        return fullName;\n    }\n\n    function VarDescription(desc){\n        return {\n            init:function(){\n                return undefined;\n            },\n            restore:function(jsonString){\n                return JSON.parse(jsonString);\n            },\n            toJsonString:function(x){\n                return JSON.stringify();\n            }\n        };\n    }\n\n    function SwarmDescription(swarmTypeName, description){\n\n        swarmTypeName = getFullName(swarmTypeName);\n\n        var localId = 0;  // unique for each swarm\n\n        function createVars(descr){\n            var members = {};\n            for(var v in descr){\n                members[v] = new VarDescription(descr[v]);\n            }\n            return members;\n        }\n\n        function createMembers(descr){\n            var members = {};\n            for(var v in description){\n\n                if(v != \"public\" && v != \"private\"){\n                    members[v] = description[v];\n                }\n            }\n            return members;\n        }\n\n        var publicVars = createVars(description.public);\n        var privateVars = createVars(description.private);\n        var myFunctions = createMembers(description);\n\n        function createPhase(thisInstance, func, phaseName){\n            var keyBefore = `${swarmTypeName}/${phaseName}/${$$.CONSTANTS.BEFORE_INTERCEPTOR}`;\n            var keyAfter = `${swarmTypeName}/${phaseName}/${$$.CONSTANTS.AFTER_INTERCEPTOR}`;\n\n            var phase = function(...args){\n                var ret;\n                try{\n                    $$.PSK_PubSub.blockCallBacks();\n                    thisInstance.setMetadata('phaseName', phaseName);\n                    $$.interceptor.callInterceptors(keyBefore, thisInstance, args);\n                    ret = func.apply(thisInstance, args);\n                    $$.interceptor.callInterceptors(keyAfter, thisInstance, args);\n                    $$.PSK_PubSub.releaseCallBacks();\n                }catch(err){\n                    $$.PSK_PubSub.releaseCallBacks();\n                    throw err;\n                }\n                return ret;\n            }\n            //dynamic named func in order to improve callstack\n            Object.defineProperty(phase, \"name\", {get: function(){return swarmTypeName+\".\"+func.name}});\n            return phase;\n        }\n\n        this.initialise = function(serialisedValues){\n\n            var result = new OwM({\n                publicVars:{\n\n                },\n                privateVars:{\n\n                },\n                protectedVars:{\n\n                },\n                myFunctions:{\n\n                },\n                utilityFunctions:{\n\n                },\n                meta:{\n                    swarmTypeName:swarmTypeName,\n                    swarmDescription:description\n                }\n            });\n\n\n            for(var v in publicVars){\n                result.publicVars[v] = publicVars[v].init();\n            };\n\n            for(var v in privateVars){\n                result.privateVars[v] = privateVars[v].init();\n            };\n\n\n            if(serialisedValues){\n                beesHealer.jsonToNative(serialisedValues, result);\n            }\n            return result;\n        };\n\n        this.initialiseFunctions = function(valueObject, thisObject){\n\n            for(var v in myFunctions){\n                valueObject.myFunctions[v] = createPhase(thisObject, myFunctions[v], v);\n            };\n\n            localId++;\n            valueObject.utilityFunctions = utils.createForObject(valueObject, thisObject, localId);\n\n        }\n\n        this.get = function(target, property, receiver){\n\n\n            if(publicVars.hasOwnProperty(property))\n            {\n                return target.publicVars[property];\n            }\n\n            if(privateVars.hasOwnProperty(property))\n            {\n                return target.privateVars[property];\n            }\n\n            if(target.utilityFunctions.hasOwnProperty(property))\n            {\n\n                return target.utilityFunctions[property];\n            }\n\n\n            if(myFunctions.hasOwnProperty(property))\n            {\n                return target.myFunctions[property];\n            }\n\n            if(target.protectedVars.hasOwnProperty(property))\n            {\n                return target.protectedVars[property];\n            }\n\n            if(typeof property != \"symbol\") {\n                $$.errorHandler.syntaxError(property, target);\n            }\n            return undefined;\n        }\n\n        this.set = function(target, property, value, receiver){\n\n            if(target.utilityFunctions.hasOwnProperty(property) || target.myFunctions.hasOwnProperty(property)) {\n                $$.errorHandler.syntaxError(property);\n                throw new Error(\"Trying to overwrite immutable member\" + property);\n            }\n\n            if(privateVars.hasOwnProperty(property))\n            {\n                target.privateVars[property] = value;\n            } else\n            if(publicVars.hasOwnProperty(property))\n            {\n                target.publicVars[property] = value;\n            } else {\n                target.protectedVars[property] = value;\n            }\n            return true;\n        }\n\n        this.apply = function(target, thisArg, argumentsList){\n            console.log(\"Proxy apply\");\n            //var func = target[]\n            //swarmGlobals.executionProvider.execute(null, thisArg, func, argumentsList)\n        }\n\n        var self = this;\n\n        this.isExtensible = function(target) {\n            return false;\n        };\n\n        this.has = function(target, prop) {\n            if(target.publicVars[prop] || target.protectedVars[prop]) {\n                return true;\n            }\n            return false;\n        };\n\n        this.ownKeys = function(target) {\n            return Reflect.ownKeys(target.publicVars);\n        };\n\n        return function(serialisedValues){\n            var valueObject = self.initialise(serialisedValues);\n            var result = new Proxy(valueObject,self);\n            self.initialiseFunctions(valueObject,result);\n\t\t\tif(!serialisedValues){\n\t\t\t\tif(!valueObject.getMeta(\"swarmId\")){\n\t\t\t\t\tvalueObject.setMeta(\"swarmId\", $$.uidGenerator.safe_uuid());  //do not overwrite!!!\n\t\t\t\t}\n\t\t\t\tvalueObject.utilityFunctions.notify();\n\t\t\t}\n\t\t\treturn result;\n        }\n    }\n\n\n\n    this.describe = function describeSwarm(swarmTypeName, description){\n        swarmTypeName = getFullName(swarmTypeName);\n\n        var pointPos = swarmTypeName.lastIndexOf('.');\n        var shortName = swarmTypeName.substr( pointPos+ 1);\n        var libraryName = swarmTypeName.substr(0, pointPos);\n        if(!libraryName){\n            libraryName = \"global\";\n        }\n\n        var description = new SwarmDescription(swarmTypeName, description);\n        if(swarmDescriptionsRegistry[swarmTypeName] != undefined){\n            $$.errorHandler.warning(\"Duplicate swarm description \"+ swarmTypeName);\n        }\n\n        //swarmDescriptionsRegistry[swarmTypeName] = description;\n\t\t$$.registerSwarmDescription(libraryName, shortName, swarmTypeName, description);\n\n        return description;\n    }\n\n    this.create = function(){\n        $$.error(\"create function is obsolete. use describe!\");\n    }\n    /* // confusing variant\n    this.create = function createSwarm(swarmTypeName, description, initialValues){\n        swarmTypeName = getFullName(swarmTypeName);\n        try{\n            if(undefined == description){\n                return swarmDescriptionsRegistry[swarmTypeName](initialValues);\n            } else {\n                return this.describe(swarmTypeName, description)(initialValues);\n            }\n        } catch(err){\n            console.log(\"CreateSwarm error\", err);\n            $$.errorHandler.error(err, arguments, \"Wrong name or descriptions\");\n        }\n    }*/\n\n    this.continue = function(swarmTypeName, initialValues){\n        swarmTypeName = getFullName(swarmTypeName);\n        var desc = swarmDescriptionsRegistry[swarmTypeName];\n\n        if(desc){\n            return desc(initialValues);\n        } else {\n            $$.errorHandler.syntaxError(swarmTypeName,initialValues,\n                \"Failed to restart a swarm with type \" + swarmTypeName + \"\\n Maybe different swarm space (used flow instead of swarm!?)\");\n        }\n    }\n\n    this.start = function(swarmTypeName, ctor, ...params){\n        swarmTypeName = getFullName(swarmTypeName);\n        var desc = swarmDescriptionsRegistry[swarmTypeName];\n        if(!desc){\n            $$.errorHandler.syntaxError(null, swarmTypeName);\n            return null;\n        }\n        var res = desc();\n        res.setMetadata(\"homeSecurityContext\", $$.securityContext);\n\n        if(ctor){\n            res[ctor].apply(res, params);\n        }\n\n        return res;\n    }\n}\n\nexports.createSwarmEngine = function(swarmType, utils){\n    if(typeof utils == \"undefined\"){\n        utils = require(\"./choreographies/utilityFunctions/callflow\");\n    }\n    return new SwarmSpace(swarmType, utils);\n};\n","\nmodule.exports.init = function(sf, logger){\n    /**\n     * Registering handler for failed asserts. The handler is doing logging and is throwing an error.\n     * @param explanation {String} - failing reason message.\n     */\n    sf.exceptions.register('assertFail', function(explanation){\n        const message = \"Assert or invariant has failed \" + (explanation ? explanation : \"\");\n        const err = new Error(message);\n        logger.recordAssert('[Fail] ' + message, err, true);\n        throw err;\n    });\n\n    /**\n     * Registering assert for equality. If check fails, the assertFail is invoked.\n     * @param v1 {String|Number|Object} - first value\n     * @param v1 {String|Number|Object} - second value\n     * @param explanation {String} - failing reason message in case the assert fails.\n     */\n    sf.assert.addCheck('equal', function(v1 , v2, explanation){\n        if(v1 !== v2){\n            if(!explanation){\n                explanation =  \"Assertion failed: [\" + v1 + \" !== \" + v2 + \"]\";\n            }\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Registering assert for inequality. If check fails, the assertFail is invoked.\n     * @param v1 {String|Number|Object} - first value\n     * @param v1 {String|Number|Object} - second value\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck('notEqual', function(v1, v2, explanation){\n        if(v1 === v2){\n            if(!explanation){\n                explanation =  \" [\"+ v1 + \" == \" + v2 + \"]\";\n            }\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Registering assert for evaluating an expression to true. If check fails, the assertFail is invoked.\n     * @param b {Boolean} - result of an expression\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck('true', function(b, explanation){\n        if(!b){\n            if(!explanation){\n                explanation =  \" expression is false but is expected to be true\";\n            }\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Registering assert for evaluating an expression to false. If check fails, the assertFail is invoked.\n     * @param b {Boolean} - result of an expression\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck('false', function(b, explanation){\n        if(b){\n            if(!explanation){\n                explanation =  \" expression is true but is expected to be false\";\n            }\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Registering assert for evaluating a value to null. If check fails, the assertFail is invoked.\n     * @param b {Boolean} - result of an expression\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck('isNull', function(v1, explanation){\n        if(v1 !== null){\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Registering assert for evaluating a value to be not null. If check fails, the assertFail is invoked.\n     * @param b {Boolean} - result of an expression\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck('notNull', function(v1 , explanation){\n        if(v1 === null && typeof v1 === \"object\"){\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Checks if all properties of the second object are own properties of the first object.\n     * @param firstObj {Object} - first object\n     * @param secondObj{Object} - second object\n     * @returns {boolean} - returns true, if the check has passed or false otherwise.\n     */\n    function objectHasFields(firstObj, secondObj){\n        for(let field in secondObj) {\n            if (firstObj.hasOwnProperty(field)) {\n                if (firstObj[field] !== secondObj[field]) {\n                    return false;\n                }\n            }\n            else{\n                return false;\n            }\n        }\n        return true;\n    }\n\n    function objectsAreEqual(firstObj, secondObj) {\n        let areEqual = true;\n        if(firstObj !== secondObj) {\n            if(typeof firstObj !== typeof secondObj) {\n                areEqual = false;\n            } else if (Array.isArray(firstObj) && Array.isArray(secondObj)) {\n\t            firstObj.sort();\n\t            secondObj.sort();\n\t\t        if (firstObj.length !== secondObj.length) {\n\t\t\t        areEqual = false;\n\t\t        } else {\n\t\t\t        for (let i = 0; i < firstObj.length; ++i) {\n\t\t\t\t        if (!objectsAreEqual(firstObj[i], secondObj[i])) {\n\t\t\t\t\t        areEqual = false;\n\t\t\t\t\t        break;\n\t\t\t\t        }\n\t\t\t        }\n\t\t        }\n\t        } else if((typeof firstObj === 'function' && typeof secondObj === 'function') ||\n\t\t        (firstObj instanceof Date && secondObj instanceof Date) ||\n\t\t        (firstObj instanceof RegExp && secondObj instanceof RegExp) ||\n\t\t        (firstObj instanceof String && secondObj instanceof String) ||\n\t\t        (firstObj instanceof Number && secondObj instanceof Number)) {\n                    areEqual = firstObj.toString() === secondObj.toString();\n            } else if(typeof firstObj === 'object' && typeof secondObj === 'object') {\n                areEqual = objectHasFields(firstObj, secondObj);\n            // isNaN(undefined) returns true\n            } else if(isNaN(firstObj) && isNaN(secondObj) && typeof firstObj === 'number' && typeof secondObj === 'number') {\n                areEqual = true;\n            } else {\n                areEqual = false;\n            }\n        }\n\n        return areEqual;\n    }\n\n    /**\n     * Registering assert for evaluating if all properties of the second object are own properties of the first object.\n     * If check fails, the assertFail is invoked.\n     * @param firstObj {Object} - first object\n     * @param secondObj{Object} - second object\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck(\"objectHasFields\", function(firstObj, secondObj, explanation){\n        if(!objectHasFields(firstObj, secondObj)) {\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Registering assert for evaluating if all element from the second array are present in the first array.\n     * Deep comparison between the elements of the array is used.\n     * If check fails, the assertFail is invoked.\n     * @param firstArray {Array}- first array\n     * @param secondArray {Array} - second array\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck(\"arraysMatch\", function(firstArray, secondArray, explanation){\n        if(firstArray.length !== secondArray.length){\n            sf.exceptions.assertFail(explanation);\n        }\n        else {\n            const result = objectsAreEqual(firstArray, secondArray);\n            // const arraysDontMatch = secondArray.every(element => firstArray.indexOf(element) !== -1);\n            // let arraysDontMatch = secondArray.some(function (expectedElement) {\n            //     let found = firstArray.some(function(resultElement){\n            //         return objectHasFields(resultElement,expectedElement);\n            //     });\n            //     return found === false;\n            // });\n\n            if(!result){\n                sf.exceptions.assertFail(explanation);\n            }\n        }\n    });\n\n    // added mainly for test purposes, better test frameworks like mocha could be much better\n\n    /**\n     * Registering assert for checking if a function is failing.\n     * If the function is throwing an exception, the test is passed or failed otherwise.\n     * @param testName {String} - test name or description\n     * @param func {Function} - function to be invoked\n     */\n    sf.assert.addCheck('fail', function(testName, func){\n        try{\n            func();\n            logger.recordAssert(\"[Fail] \" + testName);\n        } catch(err){\n            logger.recordAssert(\"[Pass] \" + testName);\n        }\n    });\n\n    /**\n     * Registering assert for checking if a function is executed with no exceptions.\n     * If the function is not throwing any exception, the test is passed or failed otherwise.\n     * @param testName {String} - test name or description\n     * @param func {Function} - function to be invoked\n     */\n    sf.assert.addCheck('pass', function(testName, func){\n        try{\n            func();\n            logger.recordAssert(\"[Pass] \" + testName);\n        } catch(err){\n            logger.recordAssert(\"[Fail] \" + testName, err.stack);\n        }\n    });\n\n    /**\n     * Alias for the pass assert.\n     */\n    sf.assert.alias('test', 'pass');\n\n    /**\n     * Registering assert for checking if a callback function is executed before timeout is reached without any exceptions.\n     * If the function is throwing any exception or the timeout is reached, the test is failed or passed otherwise.\n     * @param testName {String} - test name or description\n     * @param func {Function} - function to be invoked\n     * @param timeout {Number} - number of milliseconds for the timeout check. Default to 500ms.\n     */\n    sf.assert.addCheck('callback', function(testName, func, timeout){\n\n        if(!func || typeof func != \"function\"){\n            throw new Error(\"Wrong usage of assert.callback!\");\n        }\n\n        if(!timeout){\n            timeout = 500;\n        }\n\n        var passed = false;\n        function callback(){\n            if(!passed){\n                passed = true;\n                logger.recordAssert(\"[Pass] \" + testName);\n                successTest();\n            } else {\n                logger.recordAssert(\"[Fail (multiple calls)] \" + testName);\n            }\n        }\n        \n        try{\n            func(callback);\n        } catch(err){\n            logger.recordAssert(\"[Fail] \" + testName,  err, true);\n        }\n\n        function successTest(force){\n            if(!passed){\n                logger.recordAssert(\"[Fail Timeout] \" + testName );\n            }\n        }\n\n        setTimeout(successTest, timeout);\n    });\n\n    /**\n     * Registering assert for checking if an array of callback functions are executed in a waterfall manner,\n     * before timeout is reached without any exceptions.\n     * If any of the functions is throwing any exception or the timeout is reached, the test is failed or passed otherwise.\n     * @param testName {String} - test name or description\n     * @param func {Function} - function to be invoked\n     * @param timeout {Number} - number of milliseconds for the timeout check. Default to 500ms.\n     */\n    sf.assert.addCheck('steps', function(testName, arr, timeout){\n        if(!timeout){\n            timeout = 500;\n        }\n\n        var currentStep = 0;\n        var passed = false;\n\n        function next(){\n            if(currentStep === arr.length){\n                passed = true;\n                logger.recordAssert(\"[Pass] \" + testName );\n                return;\n            }\n\n            var func = arr[currentStep];\n            currentStep++;\n            try{\n                func(next);\n            } catch(err){\n                logger.recordAssert(\"[Fail] \" + testName  + \" [at step \" + currentStep + \"]\", err);\n            }\n        }\n\n        function successTest(force){\n            if(!passed){\n                logger.recordAssert(\"[Fail Timeout] \" + testName  + \" [at step \" + currentStep + \"]\");\n            }\n        }\n\n        setTimeout(successTest, timeout);\n        next();\n    });\n\n    /**\n     * Alias for the steps assert.\n     */\n    sf.assert.alias('waterfall', 'steps');\n\n    /**\n     * Registering assert for asynchronously printing all execution summary from logger.dumpWhys.\n     * @param message {String} - message to be recorded\n     * @param timeout {Number} - number of milliseconds for the timeout check. Default to 500ms.\n     */\n    sf.assert.addCheck('end', function(timeout, silence){\n        if(!timeout){\n            timeout = 1000;\n        }\n\n        function handler() {\n            logger.dumpWhys().forEach(function(c){\n                const executionSummary = c.getExecutionSummary();\n                console.log(JSON.stringify(executionSummary, null, 4));\n            });\n\n            if(!silence){\n                console.log(\"Forcing exit after\", timeout, \"ms\");\n            }\n            process.exit(0);\n        }\n\n        setTimeout(handler, timeout);\n    });\n\n    /**\n     * Registering assert for printing a message and asynchronously printing all logs from logger.dumpWhys.\n     * @param message {String} - message to be recorded\n     * @param timeout {Number} - number of milliseconds for the timeout check. Default to 500ms.\n     */\n    sf.assert.addCheck('begin', function(message, timeout){\n        logger.recordAssert(message);\n        sf.assert.end(timeout, true);\n    });\n};","/*\n    checks are like asserts but are intended to be used in production code to help debugging and signaling wrong behaviours\n\n */\n\nexports.init = function(sf){\n    sf.exceptions.register('checkFail', function(explanation, err){\n        var stack;\n        if(err){\n            stack = err.stack;\n        }\n        console.log(\"Check failed \", explanation, stack);\n    });\n\n    sf.check.addCheck('equal', function(v1 , v2, explanation){\n\n        if(v1 !== v2){\n            if(!explanation){\n                explanation =  \" [\"+ v1 + \" != \" + v2 + \"]\";\n            }\n\n            sf.exceptions.checkFail(explanation);\n        }\n    });\n\n\n    sf.check.addCheck('true', function(b, explanation){\n        if(!b){\n            if(!explanation){\n                explanation =  \" expression is false but is expected to be true\";\n            }\n\n            sf.exceptions.checkFail(explanation);\n        }\n    });\n\n\n    sf.check.addCheck('false', function(b, explanation){\n        if(b){\n            if(!explanation){\n                explanation =  \" expression is true but is expected to be false\";\n            }\n\n            sf.exceptions.checkFail(explanation);\n        }\n    });\n\n    sf.check.addCheck('notequal', function(v1 , v2, explanation){\n        if(v1 == v2){\n            if(!explanation){\n                explanation =  \" [\"+ v1 + \" == \" + v2 + \"]\";\n            }\n            sf.exceptions.checkFail(explanation);\n        }\n    });\n\n\n    /*\n        added mainly for test purposes, better test frameworks like mocha could be much better :)\n    */\n    sf.check.addCheck('fail', function(testName ,func){\n        try{\n            func();\n            console.log(\"[Fail] \" + testName );\n        } catch(err){\n            console.log(\"[Pass] \" + testName );\n        }\n    });\n\n\n    sf.check.addCheck('pass', function(testName ,func){\n        try{\n            func();\n            console.log(\"[Pass] \" + testName );\n        } catch(err){\n            console.log(\"[Fail] \" + testName  ,  err.stack);\n        }\n    });\n\n\n    sf.check.alias('test','pass');\n\n\n    sf.check.addCheck('callback', function(testName ,func, timeout){\n        if(!timeout){\n            timeout = 500;\n        }\n        var passed = false;\n        function callback(){\n            if(!passed){\n                passed = true;\n                console.log(\"[Pass] \" + testName );\n                SuccessTest();\n            } else {\n                console.log(\"[Fail (multiple calls)] \" + testName );\n            }\n        }\n        try{\n            func(callback);\n        } catch(err){\n            console.log(\"[Fail] \" + testName  ,  err.stack);\n        }\n\n        function SuccessTest(force){\n            if(!passed){\n                console.log(\"[Fail Timeout] \" + testName );\n            }\n        }\n\n        setTimeout(SuccessTest, timeout);\n    });\n\n\n    sf.check.addCheck('steps', function(testName , arr, timeout){\n        var  currentStep = 0;\n        var passed = false;\n        if(!timeout){\n            timeout = 500;\n        }\n\n        function next(){\n            if(currentStep === arr.length){\n                passed = true;\n                console.log(\"[Pass] \" + testName );\n                return ;\n            }\n            var func = arr[currentStep];\n            currentStep++;\n            try{\n                func(next);\n            } catch(err){\n                console.log(\"[Fail] \" + testName  ,\"\\n\\t\" , err.stack + \"\\n\\t\" , \" [at step \", currentStep + \"]\");\n            }\n        }\n\n        function SuccessTest(force){\n            if(!passed){\n                console.log(\"[Fail Timeout] \" + testName + \"\\n\\t\" , \" [at step \", currentStep+ \"]\");\n            }\n        }\n\n        setTimeout(SuccessTest, timeout);\n        next();\n    });\n\n    sf.check.alias('waterfall','steps');\n    sf.check.alias('notEqual','notequal');\n\n    sf.check.addCheck('end', function(timeOut, silence){\n        if(!timeOut){\n            timeOut = 1000;\n        }\n\n        setTimeout(function(){\n            if(!silence){\n                console.log(\"Forcing exit after\", timeOut, \"ms\");\n            }\n            process.exit(0);\n        }, timeOut);\n    });\n\n\n    sf.check.addCheck('begin', function(message, timeOut){\n        console.log(message);\n        sf.check.end(timeOut, true);\n    });\n\n\n};","exports.init = function(sf){\n    /**\n     * Registering unknown exception handler.\n     */\n    sf.exceptions.register('unknown', function(explanation){\n        explanation = explanation || \"\";\n        const message = \"Unknown exception\" + explanation;\n        throw(message);\n    });\n\n    /**\n     * Registering resend exception handler.\n     */\n    sf.exceptions.register('resend', function(exceptions){\n        throw(exceptions);\n    });\n\n    /**\n     * Registering notImplemented exception handler.\n     */\n    sf.exceptions.register('notImplemented', function(explanation){\n        explanation = explanation || \"\";\n        const message = \"notImplemented exception\" + explanation;\n        throw(message);\n    });\n\n    /**\n     * Registering security exception handler.\n     */\n    sf.exceptions.register('security', function(explanation){\n        explanation = explanation || \"\";\n        const message = \"security exception\" + explanation;\n        throw(message);\n    });\n\n    /**\n     * Registering duplicateDependency exception handler.\n     */\n    sf.exceptions.register('duplicateDependency', function(variable){\n        variable = variable || \"\";\n        const message = \"duplicateDependency exception\" + variable;\n        throw(message);\n    });\n};","const LOG_LEVELS = {\n    HARD_ERROR:     0,  // system level critical error: hardError\n    ERROR:          1,  // potentially causing user's data loosing error: error\n    LOG_ERROR:      2,  // minor annoyance, recoverable error:   logError\n    UX_ERROR:       3,  // user experience causing issues error:  uxError\n    WARN:           4,  // warning,possible isues but somehow unclear behaviour: warn\n    INFO:           5,  // store general info about the system working: info\n    DEBUG:          6,  // system level debug: debug\n    LOCAL_DEBUG:    7,  // local node/service debug: ldebug\n    USER_DEBUG:     8,  // user level debug; udebug\n    DEV_DEBUG:      9,  // development time debug: ddebug\n    WHYS:            10, // whyLog for code reasoning\n    TEST_RESULT:    11, // testResult to log running tests\n};\n\nexports.init = function(sf){\n\n    /**\n     * Records log messages from various use cases.\n     * @param record {String} - log message.\n     */\n    sf.logger.record = function(record){\n        var displayOnConsole = true;\n        if(process.send) {\n            process.send(record);\n            displayOnConsole = false;\n        }\n\n        if(displayOnConsole) {\n            const prettyLog = JSON.stringify(record, null, 2);\n            console.log(prettyLog);\n        }\n    };\n\n    /**\n     * Adding case for logging system level critical errors.\n     */\n    sf.logger.addCase('hardError', function(message, exception, args, pos, data){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.HARD_ERROR, 'systemError', message, exception, true, args, pos, data));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging potentially causing user's data loosing errors.\n     */\n    sf.logger.addCase('error', function(message, exception, args, pos, data){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.ERROR, 'error', message, exception, true, args, pos, data));\n    }, [\n        {\n            'message':'explanation'\n        },\n        {\n            'exception':'exception'\n        }\n    ]);\n\n    /**\n     * Adding case for logging minor annoyance, recoverable errors.\n     */\n    sf.logger.addCase('logError', function(message, exception, args, pos, data){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.LOG_ERROR, 'logError', message, exception, true, args, pos, data));\n    }, [\n        {\n            'message':'explanation'\n        },\n        {\n            'exception':'exception'\n        }\n    ]);\n\n    /**\n     * Adding case for logging user experience causing issues errors.\n     */\n    sf.logger.addCase('uxError', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.UX_ERROR, 'uxError', message, null, false));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging throttling messages.\n     */\n    sf.logger.addCase('throttling', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.WARN, 'throttling', message, null, false));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging warning, possible issues, but somehow unclear behaviours.\n     */\n    sf.logger.addCase('warning', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.WARN, 'warning', message,null, false, arguments, 0));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n    \n    sf.logger.alias('warn', 'warning');\n\n    /**\n     * Adding case for logging general info about the system working.\n     */\n    sf.logger.addCase('info', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.INFO, 'info', message,null, false, arguments, 0));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging system level debug messages.\n     */\n    sf.logger.addCase('debug', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.DEBUG, 'debug', message,null, false, arguments, 0));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n\n    /**\n     * Adding case for logging local node/service debug messages.\n     */\n    sf.logger.addCase('ldebug', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.LOCAL_DEBUG, 'ldebug', message, null, false, arguments, 0));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging user level debug messages.\n     */\n    sf.logger.addCase('udebug', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.USER_DEBUG, 'udebug', message ,null, false, arguments, 0));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging development debug messages.\n     */\n    sf.logger.addCase('devel', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.DEV_DEBUG, 'devel', message, null, false, arguments, 0));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging \"whys\" reasoning messages.\n     */\n    sf.logger.addCase(\"logWhy\", function(logOnlyCurrentWhyContext){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.WHYS, 'logwhy', undefined, undefined, undefined, undefined, undefined, undefined, logOnlyCurrentWhyContext));\n    });\n\n    /**\n     * Adding case for logging asserts messages to running tests.\n     */\n    sf.logger.addCase(\"recordAssert\", function (message, error,showStack){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.TEST_RESULT, 'assert', message, error, showStack));\n    });\n\n    /**\n     * Generic method to create structured debug records based on the log level.\n     * @param level {Number} - number from 1-11, used to identify the level of attention that a log entry should get from operations point of view\n     * @param type {String} - identifier name for log type\n     * @param message {String} - description of the debug record\n     * @param exception {String} - exception details if any\n     * @param saveStack {Boolean} - if set to true, the exception call stack will be added to the debug record\n     * @param args {Array} - arguments of the caller function\n     * @param pos {Number} - position\n     * @param data {String|Number|Array|Object} - payload information\n     * @param logOnlyCurrentWhyContext - if whys is enabled, only the current context will be logged\n     * @returns Debug record model {Object} with the following fields:\n     * [required]: level: *, type: *, timestamp: number, message: *, data: * and\n     * [optional]: stack: *, exception: *, args: *, whyLog: *\n     */\n    function createDebugRecord(level, type, message, exception, saveStack, args, pos, data, logOnlyCurrentWhyContext){\n\n        var ret = {\n            level: level,\n            type: type,\n            timestamp: (new Date()).getTime(),\n            message: message,\n            data: data\n        };\n\n        if(saveStack){\n            var stack = '';\n            if(exception){\n                stack = exception.stack;\n            } else {\n                stack  = (new Error()).stack;\n            }\n            ret.stack = stack;\n        }\n\n        if(exception){\n            ret.exception = exception.message;\n        }\n\n        if(args){\n            ret.args = JSON.parse(JSON.stringify(args));\n        }\n\n        if(process.env.RUN_WITH_WHYS){\n            var why = require('whys');\n            if(logOnlyCurrentWhyContext) {\n                ret['whyLog'] = why.getGlobalCurrentContext().getExecutionSummary();\n            }else{\n                ret['whyLog'] = why.getAllContexts().map(function (context) {\n                    return context.getExecutionSummary();\n                });\n            }\n        }\n\n        return ret;\n    }\n\n};\n\n","const fs = require(\"fs\");\nconst path = require(\"path\");\nconst forker = require('child_process');\n\nconst DEFAULT_TIMEOUT = 2000;\n\nvar globToRegExp =  require(\"./utils/glob-to-regexp\");\n\nvar defaultConfig = {\n    confFileName: \"double-check.json\",      // name of the conf file\n    fileExt: \".js\",                         // test file supported by extension\n    matchDirs: [ 'test', 'tests' ],           // dirs names for tests - case insensitive (used in discovery process)\n    testsDir: process.cwd(),                // path to the root tests location\n    reports: {\n        basePath: process.cwd(),            // path where the reports will be saved\n        prefix: \"Report-\",                  // prefix for report files, filename pattern: [prefix]-{timestamp}{ext}\n        ext: \".txt\"                         // report file extension\n    }\n};\n\nconst TAG = \"[TEST_RUNNER]\";\nconst MAX_WORKERS = process.env['DOUBLE_CHECK_POOL_SIZE'] || 10;\nconst DEBUG = typeof v8debug === 'object';\n\nconst TEST_STATES = {\n    READY: 'ready',\n    RUNNING: 'running',\n    FINISHED: 'finished',\n    TIMEOUT: 'timeout'\n};\n\n// Session object\nvar defaultSession = {\n    testCount: 0,\n    currentTestIndex: 0,\n    debugPort: process.debugPort,   // current process debug port. The child process will be increased from this port\n    workers: {\n        running: 0,\n        terminated: 0\n    }\n};\n\n// Template structure for test reports.\nvar reportFileStructure = {\n    count: 0,\n    suites: {\n        count: 0,\n        items: []\n    },\n    passed: {\n        count: 0,\n        items: []\n    },\n    failed: {\n        count: 0,\n        items: []\n    },\n};\n\nexports.init = function(sf){\n    sf.testRunner = {\n        /**\n         * Initialization of the test runner.\n         * @param config {Object} - settings object that will be merged with the default one\n         * @private\n         */\n        __init: function(config) {\n            this.config = this.__extend(defaultConfig, config);\n            this.testTree = {};\n            this.testList = [];\n\n            this.session = defaultSession;\n\n            // create reports directory if not exist\n            if (!fs.existsSync(this.config.reports.basePath)){\n                fs.mkdirSync(this.config.reports.basePath);\n            }\n        },\n        /**\n         * Main entry point. It will start the flow runner flow.\n         * @param config {Object} - object containing settings such as conf file name, test dir.\n         * @param callback {Function} - handler(error, result) invoked when an error occurred or the runner has completed all jobs.\n         */\n        start: function(config, callback) {\n\n            // wrapper for provided callback, if any\n            this.callback = function(err, result) {\n                if(err) {\n                    this.__debugInfo(err.message || err);\n                }\n\n                if(callback) {\n                    return callback(err, result);\n                }\n            };\n\n            this.__init(config);\n\n            this.__consoleLog(\"Discovering tests ...\");\n            this.testTree = this.__discoverTestFiles(this.config.testsDir, config);\n            this.testList = this.__toTestTreeToList(this.testTree);\n            this.__launchTests();\n        },\n        /**\n         * Reads configuration settings from a json file.\n         * @param confPath {String} - absolute path to the configuration file.\n         * @returns {Object} - configuration object {{}}\n         * @private\n         */\n        __readConf: function(confPath) {\n            var config = {};\n            try{\n                config = require(confPath);\n            } catch(error) {\n                console.error(error);\n            }\n\n            return config;\n        },\n        /**\n         * Discovers test files recursively starting from a path. The dir is the root of the test files. It can contains\n         * test files and test sub directories. It will create a tree structure with the test files discovered.\n         * Notes: Only the config.matchDirs will be taken into consideration. Also, based on the conf (double-check.json)\n         * it will include the test files or not.\n         * @param dir {String} - path where the discovery process starts\n         * @param parentConf {String} - configuration object (double-check.json) from the parent directory\n         * @returns The root node object of the file structure tree. E.g. {*|{__meta, data, result, items}}\n         * @private\n         */\n        __discoverTestFiles: function(dir, parentConf) {\n            const stat = fs.statSync(dir);\n            if(!stat.isDirectory()){\n                throw new Error(dir + \" is not a directory!\");\n            }\n\n            let currentConf = parentConf;\n\n            let currentNode = this.__getDefaultNodeStructure();\n            currentNode.__meta.parent = path.dirname(dir);\n            currentNode.__meta.isDirectory = true;\n\n            let files = fs.readdirSync(dir);\n            // first look for conf file\n            if(files.indexOf(this.config.confFileName) !== -1) {\n                let fd = path.join(dir, this.config.confFileName);\n                let conf = this.__readConf(fd);\n                if(conf) {\n                    currentNode.__meta.conf = conf;\n                    currentConf = conf;\n                }\n            }\n\n            currentNode.data.name = path.basename(dir);\n            currentNode.data.path = dir;\n            currentNode.items = [];\n\n            for(let i = 0, len = files.length; i < len; i++) {\n                let item = files[i];\n\n                let fd = path.join(dir, item);\n                let stat = fs.statSync(fd);\n                let isDir = stat.isDirectory();\n                let isTestDir = this.__isTestDir(fd);\n\n                if(isDir && !isTestDir) {\n                    continue; // ignore dirs that does not follow the naming rule for test dirs\n                }\n\n                if(!isDir && item.match(this.config.confFileName)){\n                    continue; // already processed\n                }\n\n                // exclude files based on glob patterns\n                if(currentConf) {\n                    // currentConf['ignore'] - array of regExp\n                    if(currentConf['ignore']) {\n                        const isMatch = this.__isAnyMatch(currentConf['ignore'], item);\n                        if(isMatch) {continue;}\n                    }\n                }\n\n                let childNode = this.__getDefaultNodeStructure();\n                childNode.__meta.conf = {};\n                childNode.__meta.isDirectory = isDir;\n                childNode.__meta.parent = path.dirname(fd);\n\n                if (isDir) {\n                    let tempChildNode = this.__discoverTestFiles(fd, currentConf);\n                    childNode = Object.assign(childNode, tempChildNode);\n                    currentNode.items.push(childNode);\n                }\n                else if(path.extname(fd) ===  this.config.fileExt){\n                    childNode.__meta.conf.runs = currentConf['runs'] || 1;\n                    childNode.__meta.conf.silent = currentConf['silent'];\n                    childNode.__meta.conf.timeout = currentConf['timeout'] || DEFAULT_TIMEOUT;\n\n                    childNode.data.name = item;\n                    childNode.data.path = fd;\n\n                    currentNode.items.push(childNode);\n                }\n            }\n\n            return currentNode;\n        },\n        /**\n         * Launch collected tests. Initialises session variables, that are specific for the current launch.\n         * @private\n         */\n        __launchTests: function() {\n            this.__consoleLog(\"Launching tests ...\");\n            this.session.testCount = this.testList.length;\n            this.session.processedTestCount = 0;\n            this.session.workers.running = 0;\n            this.session.workers.terminated = 0;\n\n            if(this.session.testCount > 0) {\n                this.__scheduleWork();\n            } else {\n                this.__doTestReports();\n            }\n        },\n        /**\n         * Schedules work based on the MAX available workers, and based on the number of runs of a test.\n         * If a test has multiple runs as a option, it will be started in multiple workers. Once all runs are completed,\n         * the test is considered as processed.\n         * @private\n         */\n        __scheduleWork: function() {\n            while(this.session.workers.running < MAX_WORKERS && this.session.currentTestIndex < this.session.testCount){\n                let test = this.testList[this.session.currentTestIndex];\n                if(test.result.runs < test.__meta.conf.runs) {\n                    test.result.runs++;\n                    this.__launchTest(test);\n                } else {\n                    this.session.currentTestIndex++;\n                }\n            }\n        },\n        /**\n         * Launch a test into a separate worker (child process).\n         * Each worker has handlers for message, exit and error events. Once the exit or error event is invoked,\n         * new work is scheduled and session object is updated.\n         * Notes: On debug mode, the workers will receive a debug port, that is increased incrementally.\n         * @param test {Object} - test object\n         * @private\n         */\n        __launchTest: function(test) {\n            this.session.workers.running++;\n\n            test.result.state = TEST_STATES.RUNNING;\n            test.result.pass = true;\n            test.result.asserts[test.result.runs] = [];\n            test.result.messages[test.result.runs] = [];\n\n            let env = process.env;\n\n            let execArgv = [];\n            if(DEBUG) {\n                const debugPort = ++defaultSession.debugPort;\n                const debugFlag = '--debug=' + debugPort;\n                execArgv.push(debugFlag);\n            }\n\n            const cwd = test.__meta.parent;\n\n            let worker = forker.fork(test.data.path, [], {'cwd': cwd, 'env': env, 'execArgv': execArgv, stdio: [ 'inherit', \"pipe\", 'inherit', 'ipc' ], silent:false });\n\n            this.__debugInfo(`Launching test ${test.data.name}, run[${test.result.runs}], on worker pid[${worker.pid}] `+new Date().getTime());\n\n            worker.on(\"message\", onMessageEventHandlerWrapper(test));\n            worker.on(\"exit\", onExitEventHandlerWrapper(test));\n            worker.on(\"error\", onErrorEventHandlerWrapper(test));\n\n            worker.terminated = false;\n\n            worker.stdout.on('data', function (chunk) {\n                let content = new Buffer(chunk).toString('utf8'); //TODO: replace with PSKBUFFER\n                if(test.__meta.conf.silent) {\n                    this.__consoleLog(content);\n                }\n            }.bind(this));\n\n            var self = this;\n            function onMessageEventHandlerWrapper(test) {\n                const currentRun = test.result.runs;\n                return function(log) {\n                    if(log.type === 'assert'){\n                        if(log.message.includes(\"[Fail\")) {\n                            test.result.pass = false;\n                        }\n                        test.result.asserts[currentRun].push(log);\n                    } else {\n                        test.result.messages[currentRun].push(log);\n                    }\n                };\n            }\n\n            function onExitEventHandlerWrapper(test) {\n                return function(code, signal) {\n                    clearTimeout(worker.timerVar);\n                    self.__debugInfo(`Worker ${worker.pid} - exit event. Code ${code}, signal ${signal} `+new Date().getTime());\n\n                    worker.terminated = true;\n\n                    test.result.state = TEST_STATES.FINISHED;\n                    if(code !== null && code!==0 /*&& typeof test.result.pass === 'undefined'*/){\n                        test.result.pass = false;\n                        test.result.messages[test.result.runs].push( {message: \"Process finished with errors!\", \"Exit code\":code, \"Signal\":signal});\n                    }\n\n                    self.session.workers.running--;\n                    self.session.workers.terminated++;\n\n                    self.__scheduleWork();\n                    self.__checkWorkersStatus();\n                };\n            }\n\n            // this handler can be triggered when:\n            // 1. The process could not be spawned, or\n            // 2. The process could not be killed, or\n            // 3. Sending a message to the child process failed.\n            // IMPORTANT: The 'exit' event may or may not fire after an error has occurred!\n            function onErrorEventHandlerWrapper(test) {\n                return function(error) {\n                    self.__debugInfo(`Worker ${worker.pid} - error event.`, test);\n                    self.__debugError(error);\n\n                    self.session.workers.running--;\n                    self.session.workers.terminated++;\n                };\n            }\n\n            // Note: on debug, the timeout is reached before exit event is called\n            // when kill is called, the exit event is raised\n            worker.timerVar = setTimeout(()=>{\n                if(!worker.terminated){\n                    this.__consoleLog(`worker pid [${worker.pid}] - timeout event`,new Date().getTime(),  test);\n\n                    if(test.result.state !== TEST_STATES.FINISHED){\n                        test.result.pass = false;\n                    }\n                    worker.kill();\n                    test.result.state = TEST_STATES.TIMEOUT;\n                }else{\n                    console.log(\"Got something, but don't know what...\", test);\n                }\n            }, test.__meta.conf.timeout);\n\n                self.__debugInfo(`Worker ${worker.pid} - set timeout event at `+new Date().getTime() + \" for \"+test.__meta.conf.timeout);\n\n        },\n        /**\n         * Checks if all workers completed their job (finished or have been terminated).\n         * If true, then the reporting steps can be started.\n         * @private\n         */\n        __checkWorkersStatus: function() {\n            if(this.session.workers.running === 0) {\n                this.__doTestReports();\n            }\n        },\n        /**\n         * Creates test reports object (JSON) that will be saved in the test report.\n         * Filename of the report is using the following pattern: {prefix}-{timestamp}{ext}\n         * The file will be saved in config.reports.basePath.\n         * @private\n         */\n        __doTestReports: function() {\n            this.__consoleLog(\"Doing reports ...\");\n            reportFileStructure.count = this.testList.length;\n\n            // pass/failed tests\n            for(let i = 0, len = this.testList.length; i < len; i++) {\n                let test = this.testList[i];\n\n                let testPath = this.__toRelativePath(test.data.path);\n                let item = {path: testPath};\n                if(test.result.pass) {\n                    item.reason = this.__getFirstFailReasonPerRun(test);\n                    reportFileStructure.passed.items.push(item);\n                } else {\n                    item.reason = this.__getFirstFailReasonPerRun(test);\n                    reportFileStructure.failed.items.push(item);\n                }\n            }\n            reportFileStructure.passed.count = reportFileStructure.passed.items.length;\n            reportFileStructure.failed.count = reportFileStructure.failed.items.length;\n\n            // suites (first level of directories)\n            for(let i = 0, len = this.testTree.items.length; i < len; i++) {\n                let item = this.testTree.items[i];\n                if(item.__meta.isDirectory) {\n                    let suitePath = this.__toRelativePath(item.data.path);\n                    reportFileStructure.suites.items.push(suitePath);\n                }\n            }\n            reportFileStructure.suites.count = reportFileStructure.suites.items.length;\n\n            let numberOfReports = 2;\n\n            let finishReports = (err, res) => {\n                if(numberOfReports > 1){\n                    numberOfReports -= 1;\n                    return;\n                }\n                if(reportFileStructure.failed.count === 0){\n                    this.__consoleLog(\"\\nEverything went well! No failed tests.\\n\\n\");\n                }else{\n                    this.__consoleLog(\"\\nSome tests failed. Check report files!\\n\\n\");\n                }\n\n                this.callback(err, \"Done\");\n            };\n\n\n            this.__consoleLog(this.config.reports.prefix);\n            const fileName = `${this.config.reports.prefix}latest${this.config.reports.ext}`;\n            const filePath = path.join(this.config.reports.basePath, fileName);\n            this.__saveReportToFile(reportFileStructure, filePath, finishReports);\n\n            const timestamp = new Date().getTime().toString();\n            const htmlFileName = `${this.config.reports.prefix}latest.html`;\n            const htmlFilePath = path.join(this.config.reports.basePath, htmlFileName);\n            this.__saveHtmlReportToFile(reportFileStructure, htmlFilePath, timestamp, finishReports);\n        },\n        /**\n         * Saves test reports object (JSON) in the specified path.\n         * @param reportFileStructure {Object} - test reports object (JSON)\n         * @param destination {String} - path of the file report (the base path MUST exist)\n         * @private\n         */\n        __saveReportToFile: function(reportFileStructure, destination, callback) {\n\n            var content = JSON.stringify(reportFileStructure, null, 4);\n            fs.writeFile(destination, content, 'utf8', function (err) {\n                if (err) {\n                    const message = \"An error occurred while writing the report file, with the following error: \" + JSON.stringify(err);\n                    this.__debugInfo(message);\n                    throw err;\n                } else{\n                    const message = `Finished writing report to ${destination}`;\n                    this.__consoleLog(message);\n                }\n                callback();\n            }.bind(this));\n        },\n        /**\n         * Saves test reports as HTML in the specified path.\n         * @param reportFileStructure {Object} - test reports object (JSON)\n         * @param destination {String} - path of the file report (the base path MUST exist)\n         * @param timestamp {String} - timestamp to be injected in html template\n         * @private\n         */\n        __saveHtmlReportToFile: function (reportFileStructure, destination, timestamp, callback) {\n            var folderName = path.resolve(__dirname);\n            fs.readFile(path.join(folderName,'/utils/reportTemplate.html'), 'utf8', (err, res) => {\n                if (err) {\n                    const message = 'An error occurred while reading the html report template file, with the following error: ' + JSON.stringify(err);\n                    this.__debugInfo(message);\n                    throw err;\n                }\n\n                fs.writeFile(destination, res + `<script>init(${JSON.stringify(reportFileStructure)}, ${timestamp});</script>`, 'utf8', (err) => {\n                    if (err) {\n                        const message = 'An error occurred while writing the html report file, with the following error: ' + JSON.stringify(err);\n                        this.__debugInfo(message);\n                        throw err;\n                    }\n\n                    const message = `Finished writing report to ${destination}`;\n                    this.__consoleLog(message);\n\n                    callback();\n                });\n            });\n        },\n        /**\n         * Converts absolute file path to relative path.\n         * @param absolutePath {String} - absolute path\n         * @returns {string | void | *} - relative path\n         * @private\n         */\n        __toRelativePath: function(absolutePath) {\n            const basePath = path.join(this.config.testsDir, \"/\");\n            const relativePath = absolutePath.replace(basePath, \"\");\n            return relativePath;\n        },\n        /**\n         * Checks if a directory is a test dir, by matching its name against config.matchDirs array.\n         * @param dir {String} - directory name\n         * @returns {boolean} - returns true if there is a match and false otherwise.\n         * @private\n         */\n        __isTestDir: function(dir) {\n            if(!this.config || !this.config.matchDirs ) {\n                throw `matchDirs is not defined on config ${JSON.stringify(this.config)} does not exist!`;\n            }\n\n            var isTestDir = this.config.matchDirs.some(function(item) {\n                return dir.toLowerCase().includes(item.toLowerCase());\n            });\n\n            return isTestDir;\n        },\n        /**\n         * For a failed test, it returns only the first fail reason per each run.\n         * @param test {Object} - test object\n         * @returns {Array} - an array of reasons per each test run.\n         * @private\n         */\n        __getFirstFailReasonPerRun: function(test) {\n            const reason = [];\n            for(let i = 1; i <= test.result.runs; i++) {\n                if(test.result.asserts[i] && test.result.asserts[i].length > 0) {\n                    addReason(i, test.result.asserts[i][0]);\n                }\n\n                if(test.result.messages[i] && test.result.messages[i].length > 0) {\n                    addReason(i, test.result.messages[i][0]);\n                }\n\n                function addReason(run, log) {\n                    const message = {\n                        run: run,\n                        log: log\n                    };\n\n                    reason.push(message);\n                }\n            }\n\n            return reason;\n        },\n        /**\n         * Described default tree node structure.\n         * @returns {{__meta: {conf: null, parent: null, isDirectory: boolean}, data: {name: null, path: null}, result: {state: string, pass: null, executionTime: number, runs: number, asserts: {}, messages: {}}, items: null}}\n         * @private\n         */\n        __getDefaultNodeStructure: function() {\n            return  {\n                __meta: {\n                    conf: null,\n                    parent: null,\n                    isDirectory: false\n                },\n                data: {\n                    name: null,\n                    path: null,\n                },\n                result: {\n                    state: TEST_STATES.READY, // ready | running | terminated | timeout\n                    pass: null,\n                    executionTime: 0,\n                    runs: 0,\n                    asserts: {},\n                    messages: {}\n                },\n                items: null\n            };\n        },\n        /**\n         * Match a test file path to a UNIX glob expression array. If its any match returns true, otherwise returns false.\n         * @param globExpArray {Array} - an array with glob expression (UNIX style)\n         * @param str {String} - the string to be matched\n         * @returns {boolean} - returns true if there is any match and false otherwise.\n         * @private\n         */\n        __isAnyMatch: function(globExpArray, str) {\n            const hasMatch = function(globExp) {\n                const regex = globToRegExp(globExp);\n                return regex.test(str);\n            };\n\n            return globExpArray.some(hasMatch);\n        },\n        /**\n         * Converts a tree structure into an array list of test nodes. The tree traversal is DFS (Deep-First-Search).\n         * @param rootNode {Object} - root node of the test tree.\n         * @returns {Array} - List of test nodes.\n         * @private\n         */\n        __toTestTreeToList: function(rootNode) {\n            var testList = [];\n\n            traverse(rootNode);\n\n            function traverse(node) {\n                if(!node.__meta.isDirectory || !node.items) {\n                    return;\n                }\n\n                for(let i = 0, len = node.items.length; i < len; i++) {\n                    const item = node.items[i];\n                    if(item.__meta.isDirectory) {\n                        traverse(item);\n                    } else {\n                        testList.push(item);\n                    }\n                }\n            }\n\n            return testList;\n        },\n        /**\n         * Logging to console wrapper.\n         * @param log {String|Object|Number} - log message\n         * @private\n         */\n        __consoleLog: function(log) {\n            console.log(TAG, log);\n        },\n        /**\n         * Logging debugging info messages wrapper.\n         * Logger: console.info\n         * @param log {String|Object|Number} - log message\n         * @private\n         */\n        __debugInfo: function(log) {\n            this.__debug(console.info, log);\n        },\n        /**\n         * Logging debugging error messages wrapper.\n         * Logger: console.error\n         * @param log {String|Object|Number} - log message\n         * @private\n         */\n        __debugError: function(log) {\n            this.__debug(console.error, log);\n        },\n        /**\n         *  Logging debugging messages wrapper. One debug mode, the logging is silent.\n         * @param logger {Function} - handler for logging\n         * @param log {String|Object|Number} - log message\n         * @private\n         */\n        __debug: function(logger, log) {\n            if(!DEBUG) {return;}\n\n            // let prettyLog = JSON.stringify(log, null, 2);\n            logger(\"DEBUG\", log);\n        },\n        /**\n         * Deep extend one object with properties of another object.\n         * If the property exists in both objects the property from the first object is overridden.\n         * @param first {Object} - the first object\n         * @param second {Object} - the second object\n         * @returns {Object} - an object with both properties from the first and second object.\n         * @private\n         */\n        __extend: function (first, second) {\n            for (const key in second) {\n                if (!first.hasOwnProperty(key)) {\n                    first[key] = second[key];\n                } else {\n                    let val = second[key];\n                    if(typeof first[key] === 'object') {\n                        val = this.__extend(first[key], second[key]);\n                    }\n\n                    first[key] = val;\n                }\n            }\n\n            return first;\n        }\n    };\n};\n","\n// globToRegExp turns a UNIX glob expression into a RegEx expression.\n//  Supports all simple glob patterns. Examples: *.ext, /foo/*, ../../path, ^foo.*\n// - single character matching, matching ranges of characters etc. group matching are no supported\n// - flags are not supported\nvar globToRegExp = function (globExp) {\n    if (typeof globExp !== 'string') {\n        throw new TypeError('Glob Expression must be a string!');\n    }\n\n    var regExp = \"\";\n\n    for (let i = 0, len = globExp.length; i < len; i++) {\n        let c = globExp[i];\n\n        switch (c) {\n            case \"/\":\n            case \"$\":\n            case \"^\":\n            case \"+\":\n            case \".\":\n            case \"(\":\n            case \")\":\n            case \"=\":\n            case \"!\":\n            case \"|\":\n                regExp += \"\\\\\" + c;\n                break;\n\n            case \"*\":\n                // treat any number of \"*\" as one\n                while(globExp[i + 1] === \"*\") {\n                    i++;\n                }\n                regExp += \".*\";\n                break;\n\n            default:\n                regExp += c;\n        }\n    }\n\n    // set the regular expression with ^ & $\n    regExp = \"^\" + regExp + \"$\";\n\n    return new RegExp(regExp);\n};\n\nmodule.exports = globToRegExp;","const utils = require(\"swarmutils\");\nconst OwM = utils.OwM;\nvar beesHealer = utils.beesHealer;\nvar fs = require(\"fs\");\nvar path = require(\"path\");\n\n\n//TODO: prevent a class of race condition type of errors by signaling with files metadata to the watcher when it is safe to consume\n\nfunction FolderMQ(folder, callback = () => {}){\n\n\tif(typeof callback !== \"function\"){\n\t\tthrow new Error(\"Second parameter should be a callback function\");\n\t}\n\n\tfolder = path.normalize(folder);\n\n\tfs.mkdir(folder, {recursive: true}, function(err, res){\n\t\tfs.exists(folder, function(exists) {\n\t\t\tif (exists) {\n\t\t\t\treturn callback(null, folder);\n\t\t\t} else {\n\t\t\t\treturn callback(err);\n\t\t\t}\n\t\t});\n\t});\n\n\tfunction mkFileName(swarmRaw){\n\t\tlet meta = OwM.prototype.getMetaFrom(swarmRaw);\n\t\tlet name = `${folder}${path.sep}${meta.swarmId}.${meta.swarmTypeName}`;\n\t\tconst unique = meta.phaseId || $$.uidGenerator.safe_uuid();\n\n\t\tname = name+`.${unique}`;\n\t\treturn path.normalize(name);\n\t}\n\n\tthis.getHandler = function(){\n\t\tif(producer){\n\t\t\tthrow new Error(\"Only one consumer is allowed!\");\n\t\t}\n\t\tproducer = true;\n\t\treturn {\n\t\t\tsendSwarmSerialization: function(serialization, callback){\n\t\t\t\tif(typeof callback !== \"function\"){\n\t\t\t\t\tthrow new Error(\"Second parameter should be a callback function\");\n\t\t\t\t}\n\t\t\t\twriteFile(mkFileName(JSON.parse(serialization)), serialization, callback);\n\t\t\t},\n\t\t\taddStream : function(stream, callback){\n\t\t\t\tif(typeof callback !== \"function\"){\n\t\t\t\t\tthrow new Error(\"Second parameter should be a callback function\");\n\t\t\t\t}\n\n\t\t\t\tif(!stream || !stream.pipe || typeof stream.pipe !== \"function\"){\n\t\t\t\t\treturn callback(new Error(\"Something wrong happened\"));\n\t\t\t\t}\n\n\t\t\t\tlet swarm = \"\";\n\t\t\t\tstream.on('data', (chunk) =>{\n\t\t\t\t\tswarm += chunk;\n\t\t\t\t});\n\n\t\t\t\tstream.on(\"end\", () => {\n\t\t\t\t\twriteFile(mkFileName(JSON.parse(swarm)), swarm, callback);\n\t\t\t\t});\n\n\t\t\t\tstream.on(\"error\", (err) =>{\n\t\t\t\t\tcallback(err);\n\t\t\t\t});\n\t\t\t},\n\t\t\taddSwarm : function(swarm, callback){\n\t\t\t\tif(!callback){\n\t\t\t\t\tcallback = $$.defaultErrorHandlingImplementation;\n\t\t\t\t}else if(typeof callback !== \"function\"){\n\t\t\t\t\tthrow new Error(\"Second parameter should be a callback function\");\n\t\t\t\t}\n\n\t\t\t\tbeesHealer.asJSON(swarm,null, null, function(err, res){\n\t\t\t\t\tif (err) {\n\t\t\t\t\t\tconsole.log(err);\n\t\t\t\t\t}\n\t\t\t\t\twriteFile(mkFileName(res), J(res), callback);\n\t\t\t\t});\n\t\t\t},\n\t\t\tsendSwarmForExecution: function(swarm, callback){\n\t\t\t\tif(!callback){\n\t\t\t\t\tcallback = $$.defaultErrorHandlingImplementation;\n\t\t\t\t}else if(typeof callback !== \"function\"){\n\t\t\t\t\tthrow new Error(\"Second parameter should be a callback function\");\n\t\t\t\t}\n\n\t\t\t\tbeesHealer.asJSON(swarm, OwM.prototype.getMetaFrom(swarm, \"phaseName\"), OwM.prototype.getMetaFrom(swarm, \"args\"), function(err, res){\n\t\t\t\t\tif (err) {\n\t\t\t\t\t\tconsole.log(err);\n\t\t\t\t\t}\n\t\t\t\t\tvar file = mkFileName(res);\n\t\t\t\t\tvar content = JSON.stringify(res);\n\n\t\t\t\t\t//if there are no more FD's for files to be written we retry.\n\t\t\t\t\tfunction wrapper(error, result){\n\t\t\t\t\t\tif(error){\n\t\t\t\t\t\t\tconsole.log(`Caught an write error. Retry to write file [${file}]`);\n\t\t\t\t\t\t\tsetTimeout(()=>{\n\t\t\t\t\t\t\t\twriteFile(file, content, wrapper);\n\t\t\t\t\t\t\t}, 10);\n\t\t\t\t\t\t}else{\n\t\t\t\t\t\t\treturn callback(error, result);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\twriteFile(file, content, wrapper);\n\t\t\t\t});\n\t\t\t}\n\t\t};\n\t};\n\n\tvar recipient;\n\tthis.setIPCChannel = function(processChannel){\n\t\tif(processChannel && !processChannel.send || (typeof processChannel.send) != \"function\"){\n\t\t\tthrow new Error(\"Recipient is not instance of process/child_process or it was not spawned with IPC channel!\");\n\t\t}\n\t\trecipient = processChannel;\n\t\tif(consumer){\n\t\t\tconsole.log(`Channel updated`);\n\t\t\t(recipient || process).on(\"message\", receiveEnvelope);\n\t\t}\n\t};\n\n\n\tvar consumedMessages = {};\n\n\tfunction checkIfConsummed(name, message){\n\t\tconst shortName = path.basename(name);\n\t\tconst previousSaved = consumedMessages[shortName];\n\t\tlet result = false;\n\t\tif(previousSaved && !previousSaved.localeCompare(message)){\n\t\t\tresult = true;\n\t\t}\n\t\treturn result;\n\t}\n\n\tfunction save2History(envelope){\n\t\tconsumedMessages[path.basename(envelope.name)] = envelope.message;\n\t}\n\n\tfunction buildEnvelopeConfirmation(envelope, saveHistory){\n\t\tif(saveHistory){\n\t\t\tsave2History(envelope);\n\t\t}\n\t\treturn `Confirm envelope ${envelope.timestamp} sent to ${envelope.dest}`;\n\t}\n\n\tfunction buildEnvelope(name, message){\n\t\treturn {\n\t\t\tdest: folder,\n\t\t\tsrc: process.pid,\n\t\t\ttimestamp: new Date().getTime(),\n\t\t\tmessage: message,\n\t\t\tname: name\n\t\t};\n\t}\n\n\tfunction receiveEnvelope(envelope){\n\t\tif(!envelope || typeof envelope !== \"object\"){\n\t\t\treturn;\n\t\t}\n\t\t//console.log(\"received envelope\", envelope, folder);\n\n\t\tif(envelope.dest !== folder && folder.indexOf(envelope.dest)!== -1 && folder.length === envelope.dest+1){\n\t\t\tconsole.log(\"This envelope is not for me!\");\n\t\t\treturn;\n\t\t}\n\n\t\tlet message = envelope.message;\n\n\t\tif(callback){\n\t\t\t//console.log(\"Sending confirmation\", process.pid);\n\t\t\trecipient.send(buildEnvelopeConfirmation(envelope, true));\n\t\t\tconsumer(null, JSON.parse(message));\n\t\t}\n\t}\n\n\tthis.registerAsIPCConsumer = function(callback){\n\t\tif(typeof callback !== \"function\"){\n\t\t\tthrow new Error(\"The argument should be a callback function\");\n\t\t}\n\t\tregisteredAsIPCConsumer = true;\n\t\t//will register as normal consumer in order to consume all existing messages but without setting the watcher\n\t\tthis.registerConsumer(callback, true, (watcher) => !watcher);\n\n\t\t//console.log(\"Registered as IPC Consummer\", );\n\t\t(recipient || process).on(\"message\", receiveEnvelope);\n\t};\n\n\tthis.registerConsumer = function (callback, shouldDeleteAfterRead = true, shouldWaitForMore = (watcher) => true) {\n\t\tif(typeof callback !== \"function\"){\n\t\t\tthrow new Error(\"First parameter should be a callback function\");\n\t\t}\n\t\tif (consumer) {\n\t\t\tthrow new Error(\"Only one consumer is allowed! \" + folder);\n\t\t}\n\n\t\tconsumer = callback;\n\n\t\tfs.mkdir(folder, {recursive: true}, function (err, res) {\n\t\t\tif (err && (err.code !== 'EEXIST')) {\n\t\t\t\tconsole.log(err);\n\t\t\t}\n\t\t\tconsumeAllExisting(shouldDeleteAfterRead, shouldWaitForMore);\n\t\t});\n\t};\n\n\tthis.writeMessage = writeFile;\n\n\tthis.unlinkContent = function (messageId, callback) {\n\t\tconst messagePath = path.join(folder, messageId);\n\n\t\tfs.unlink(messagePath, (err) => {\n\t\t\tcallback(err);\n\t\t});\n\t};\n\n\tthis.dispose = function(force){\n\t\tif(typeof folder != \"undefined\"){\n\t\t\tvar files;\n\t\t\ttry{\n\t\t\t\tfiles = fs.readdirSync(folder);\n\t\t\t}catch(error){\n\t\t\t\t//..\n\t\t\t}\n\n\t\t\tif(files && files.length > 0 && !force){\n\t\t\t\tconsole.log(\"Disposing a channel that still has messages! Dir will not be removed!\");\n\t\t\t\treturn false;\n\t\t\t}else{\n\t\t\t\ttry{\n\t\t\t\t\tfs.rmdirSync(folder);\n\t\t\t\t}catch(err){\n\t\t\t\t\t//..\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tfolder = null;\n\t\t}\n\n\t\tif(producer){\n\t\t\t//no need to do anything else\n\t\t}\n\n\t\tif(typeof consumer != \"undefined\"){\n\t\t\tconsumer = () => {};\n\t\t}\n\n\t\tif(watcher){\n\t\t\twatcher.close();\n\t\t\twatcher = null;\n\t\t}\n\n\t\treturn true;\n\t};\n\n\n\t/* ---------------- protected  functions */\n\tvar consumer = null;\n\tvar registeredAsIPCConsumer = false;\n\tvar producer = null;\n\n\tfunction buildPathForFile(filename){\n\t\treturn path.normalize(path.join(folder, filename));\n\t}\n\n\tfunction consumeMessage(filename, shouldDeleteAfterRead, callback) {\n\t\tvar fullPath = buildPathForFile(filename);\n\n\t\tfs.readFile(fullPath, \"utf8\", function (err, data) {\n\t\t\tif (!err) {\n\t\t\t\tif (data !== \"\") {\n\t\t\t\t\ttry {\n\t\t\t\t\t\tvar message = JSON.parse(data);\n\t\t\t\t\t} catch (error) {\n\t\t\t\t\t\tconsole.log(\"Parsing error\", error);\n\t\t\t\t\t\terr = error;\n\t\t\t\t\t}\n\n\t\t\t\t\tif(checkIfConsummed(fullPath, data)){\n\t\t\t\t\t\t//console.log(`message already consumed [${filename}]`);\n\t\t\t\t\t\treturn ;\n\t\t\t\t\t}\n\n\t\t\t\t\tif (shouldDeleteAfterRead) {\n\n\t\t\t\t\t\tfs.unlink(fullPath, function (err, res) {\n\t\t\t\t\t\t\tif (err) {throw err;};\n\t\t\t\t\t\t});\n\n\t\t\t\t\t}\n\t\t\t\t\treturn callback(err, message);\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tconsole.log(\"Consume error\", err);\n\t\t\t\treturn callback(err);\n\t\t\t}\n\t\t});\n\t}\n\n\tfunction consumeAllExisting(shouldDeleteAfterRead, shouldWaitForMore) {\n\n\t\tlet currentFiles = [];\n\n\t\tfs.readdir(folder, 'utf8', function (err, files) {\n\t\t\tif (err) {\n\t\t\t\t$$.errorHandler.error(err);\n\t\t\t\treturn;\n\t\t\t}\n\t\t\tcurrentFiles = files;\n\t\t\titerateAndConsume(files);\n\n\t\t});\n\n\t\tfunction startWatching(){\n\t\t\tif (shouldWaitForMore(true)) {\n\t\t\t\twatchFolder(shouldDeleteAfterRead, shouldWaitForMore);\n\t\t\t}\n\t\t}\n\n\t\tfunction iterateAndConsume(files, currentIndex = 0) {\n\t\t\tif (currentIndex === files.length) {\n\t\t\t\t//console.log(\"start watching\", new Date().getTime());\n\t\t\t\tstartWatching();\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif (path.extname(files[currentIndex]) !== in_progress) {\n\t\t\t\tconsumeMessage(files[currentIndex], shouldDeleteAfterRead, (err, data) => {\n\t\t\t\t\tif (err) {\n\t\t\t\t\t\titerateAndConsume(files, ++currentIndex);\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\t\t\t\t\tconsumer(null, data, path.basename(files[currentIndex]));\n\t\t\t\t\tif (shouldWaitForMore()) {\n\t\t\t\t\t\titerateAndConsume(files, ++currentIndex);\n\t\t\t\t\t}\n\t\t\t\t});\n\t\t\t} else {\n\t\t\t\titerateAndConsume(files, ++currentIndex);\n\t\t\t}\n\t\t}\n\t}\n\n\tfunction writeFile(filename, content, callback){\n\t\tif(recipient){\n\t\t\tvar envelope = buildEnvelope(filename, content);\n\t\t\t//console.log(\"Sending to\", recipient.pid, recipient.ppid, \"envelope\", envelope);\n\t\t\trecipient.send(envelope);\n\t\t\tvar confirmationReceived = false;\n\n\t\t\tfunction receiveConfirmation(message){\n\t\t\t\tif(message === buildEnvelopeConfirmation(envelope)){\n\t\t\t\t\t//console.log(\"Received confirmation\", recipient.pid);\n\t\t\t\t\tconfirmationReceived = true;\n\t\t\t\t\ttry{\n\t\t\t\t\t\trecipient.off(\"message\", receiveConfirmation);\n\t\t\t\t\t}catch(err){\n\t\t\t\t\t\t//...\n\t\t\t\t\t}\n\n\t\t\t\t}\n\t\t\t}\n\n\t\t\trecipient.on(\"message\", receiveConfirmation);\n\n\t\t\tsetTimeout(()=>{\n\t\t\t\tif(!confirmationReceived){\n\t\t\t\t\t//console.log(\"No confirmation...\", process.pid);\n\t\t\t\t\thidden_writeFile(filename, content, callback);\n\t\t\t\t}else{\n\t\t\t\t\tif(callback){\n\t\t\t\t\t\treturn callback(null, content);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}, 200);\n\t\t}else{\n\t\t\thidden_writeFile(filename, content, callback);\n\t\t}\n\t}\n\n\tconst in_progress = \".in_progress\";\n\tfunction hidden_writeFile(filename, content, callback){\n\t\tvar tmpFilename = filename+in_progress;\n\t\ttry{\n\t\t\tif(fs.existsSync(tmpFilename) || fs.existsSync(filename)){\n\t\t\t\tconsole.log(new Error(`Overwriting file ${filename}`));\n\t\t\t}\n\t\t\tfs.writeFileSync(tmpFilename, content);\n\t\t\tfs.renameSync(tmpFilename, filename);\n\t\t}catch(err){\n\t\t\treturn callback(err);\n\t\t}\n\t\tcallback(null, content);\n\t}\n\n\tvar alreadyKnownChanges = {};\n\n\tfunction alreadyFiredChanges(filename, change){\n\t\tvar res = false;\n\t\tif(alreadyKnownChanges[filename]){\n\t\t\tres = true;\n\t\t}else{\n\t\t\talreadyKnownChanges[filename] = change;\n\t\t}\n\n\t\treturn res;\n\t}\n\n\tfunction watchFolder(shouldDeleteAfterRead, shouldWaitForMore){\n\n\t\tsetTimeout(function(){\n\t\t\tfs.readdir(folder, 'utf8', function (err, files) {\n\t\t\t\tif (err) {\n\t\t\t\t\t$$.errorHandler.error(err);\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tfor(var i=0; i<files.length; i++){\n\t\t\t\t\twatchFilesHandler(\"change\", files[i]);\n\t\t\t\t}\n\t\t\t});\n\t\t}, 1000);\n\n\t\tfunction watchFilesHandler(eventType, filename){\n\t\t\t//console.log(`Got ${eventType} on ${filename}`);\n\n\t\t\tif(!filename || path.extname(filename) === in_progress){\n\t\t\t\t//caught a delete event of a file\n\t\t\t\t//or\n\t\t\t\t//file not ready to be consumed (in progress)\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tvar f = buildPathForFile(filename);\n\t\t\tif(!fs.existsSync(f)){\n\t\t\t\t//console.log(\"File not found\", f);\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t//console.log(`Preparing to consume ${filename}`);\n\t\t\tif(!alreadyFiredChanges(filename, eventType)){\n\t\t\t\tconsumeMessage(filename, shouldDeleteAfterRead, (err, data) => {\n\t\t\t\t\t//allow a read a the file\n\t\t\t\t\talreadyKnownChanges[filename] = undefined;\n\n\t\t\t\t\tif (err) {\n\t\t\t\t\t\t// ??\n\t\t\t\t\t\tconsole.log(\"\\nCaught an error\", err);\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\n\t\t\t\t\tconsumer(null, data, filename);\n\n\n\t\t\t\t\tif (!shouldWaitForMore()) {\n\t\t\t\t\t\twatcher.close();\n\t\t\t\t\t}\n\t\t\t\t});\n\t\t\t}else{\n\t\t\t\tconsole.log(\"Something happens...\", filename);\n\t\t\t}\n\t\t}\n\n\n\t\tconst watcher = fs.watch(folder, watchFilesHandler);\n\n\t\tconst intervalTimer = setInterval(()=>{\n\t\t\tfs.readdir(folder, 'utf8', function (err, files) {\n\t\t\t\tif (err) {\n\t\t\t\t\t$$.errorHandler.error(err);\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tif(files.length > 0){\n\t\t\t\t\tconsole.log(`\\n\\nFound ${files.length} files not consumed yet in ${folder}`, new Date().getTime(),\"\\n\\n\");\n\t\t\t\t\t//faking a rename event trigger\n\t\t\t\t\twatchFilesHandler(\"rename\", files[0]);\n\t\t\t\t}\n\t\t\t});\n\t\t}, 5000);\n\t}\n}\n\nexports.getFolderQueue = function(folder, callback){\n\treturn new FolderMQ(folder, callback);\n};\n","function PSKBuffer() {}\n\nfunction getArrayBufferInterface () {\n    if(typeof SharedArrayBuffer === 'undefined') {\n        return ArrayBuffer;\n    } else {\n        return SharedArrayBuffer;\n    }\n}\n\nPSKBuffer.from = function (source) {\n    const ArrayBufferInterface = getArrayBufferInterface();\n\n    const buffer = new Uint8Array(new ArrayBufferInterface(source.length));\n    buffer.set(source, 0);\n\n    return buffer;\n};\n\nPSKBuffer.concat = function ([ ...params ], totalLength) {\n    const ArrayBufferInterface = getArrayBufferInterface();\n\n    if (!totalLength && totalLength !== 0) {\n        totalLength = 0;\n        for (const buffer of params) {\n            totalLength += buffer.length;\n        }\n    }\n\n    const buffer = new Uint8Array(new ArrayBufferInterface(totalLength));\n    let offset = 0;\n\n    for (const buf of params) {\n        const len = buf.length;\n\n        const nextOffset = offset + len;\n        if (nextOffset > totalLength) {\n            const remainingSpace = totalLength - offset;\n            for (let i = 0; i < remainingSpace; ++i) {\n                buffer[offset + i] = buf[i];\n            }\n        } else {\n            buffer.set(buf, offset);\n        }\n\n        offset = nextOffset;\n    }\n\n    return buffer;\n};\n\nPSKBuffer.isBuffer = function (pskBuffer) {\n    return !!ArrayBuffer.isView(pskBuffer);\n};\n\nPSKBuffer.alloc = function(size) {\n    const ArrayBufferInterface = getArrayBufferInterface();\n\n    return new Uint8Array(new ArrayBufferInterface(size));\n};\n\nmodule.exports = PSKBuffer;","const crypto = require('crypto');\nconst KeyEncoder = require('./keyEncoder');\n\nfunction ECDSA(curveName){\n    this.curve = curveName || 'secp256k1';\n    const self = this;\n\n    this.generateKeyPair = function() {\n        const result     = {};\n        const ec         = crypto.createECDH(self.curve);\n        result.public  = ec.generateKeys('hex');\n        result.private = ec.getPrivateKey('hex');\n        return keysToPEM(result);\n    };\n\n    function keysToPEM(keys){\n        const result                  = {};\n        const ECPrivateKeyASN         = KeyEncoder.ECPrivateKeyASN;\n        const SubjectPublicKeyInfoASN = KeyEncoder.SubjectPublicKeyInfoASN;\n        const keyEncoder              = new KeyEncoder(self.curve);\n\n        const privateKeyObject        = keyEncoder.privateKeyObject(keys.private,keys.public);\n        const publicKeyObject         = keyEncoder.publicKeyObject(keys.public);\n\n        result.private              = ECPrivateKeyASN.encode(privateKeyObject, 'pem', privateKeyObject.pemOptions);\n        result.public               = SubjectPublicKeyInfoASN.encode(publicKeyObject, 'pem', publicKeyObject.pemOptions);\n\n        return result;\n\n    }\n\n    this.sign = function (privateKey,digest) {\n        const sign = crypto.createSign(\"sha256\");\n        sign.update(digest);\n\n        return sign.sign(privateKey,'hex');\n    };\n\n    this.verify = function (publicKey,signature,digest) {\n        const verify = crypto.createVerify('sha256');\n        verify.update(digest);\n\n        return verify.verify(publicKey,signature,'hex');\n    }\n}\n\nexports.createECDSA = function (curve){\n    return new ECDSA(curve);\n};","const crypto = require('crypto');\nconst fs = require('fs');\nconst path = require('path');\nconst os = require('os');\n\nconst utils = require(\"./utils/cryptoUtils\");\nconst PskArchiver = require(\"./psk-archiver\");\nconst PassThroughStream = require('./utils/PassThroughStream');\n\nconst EventEmitter = require('events');\nconst tempFolder = os.tmpdir();\n\nfunction PskCrypto() {\n\n    const self = this;\n\n    const event = new EventEmitter();\n\n    this.on = event.on;\n    this.off = event.removeListener;\n    this.removeAllListeners = event.removeAllListeners;\n    this.emit = event.emit;\n\n    /*--------------------------------------------- ECDSA functions ------------------------------------------*/\n    const ecdsa = require(\"./ECDSA\").createECDSA();\n    this.generateECDSAKeyPair = function () {\n        return ecdsa.generateKeyPair();\n    };\n\n    this.sign = function (privateKey, digest) {\n        return ecdsa.sign(privateKey, digest);\n    };\n\n    this.verify = function (publicKey, signature, digest) {\n        return ecdsa.verify(publicKey, signature, digest);\n    };\n\n    /*---------------------------------------------Encryption functions -------------------------------------*/\n\n    this.encryptStream = function (inputPath, destinationPath, password, callback) {\n        const archiver = new PskArchiver();\n\n        archiver.on('progress', (progress) => {\n            self.emit('progress', progress);\n        });\n\n        fs.open(destinationPath, \"wx\", function (err, fd) {\n            if (err) {\n                callback(err);\n                return;\n            }\n\n            fs.close(fd, function (err) {\n                if (err) {\n                    return callback(err);\n                }\n\n                const ws = fs.createWriteStream(destinationPath, {autoClose: false});\n                const keySalt = crypto.randomBytes(32);\n                const key = crypto.pbkdf2Sync(password, keySalt, utils.iterations_number, 32, 'sha512');\n\n                const aadSalt = crypto.randomBytes(32);\n                const aad = crypto.pbkdf2Sync(password, aadSalt, utils.iterations_number, 32, 'sha512');\n\n                const salt = Buffer.concat([keySalt, aadSalt]);\n                const iv = crypto.pbkdf2Sync(password, salt, utils.iterations_number, 12, 'sha512');\n\n                const cipher = crypto.createCipheriv(utils.algorithm, key, iv);\n                cipher.setAAD(aad);\n                archiver.zipStream(inputPath, cipher, function (err, cipherStream) {\n\n                    if (err) {\n                        return callback(err);\n                    }\n\n                    cipherStream.on(\"data\", function (chunk) {\n                        ws.write(chunk);\n                    });\n                    cipherStream.on('end', function () {\n                        const tag = cipher.getAuthTag();\n                        const dataToAppend = Buffer.concat([salt, tag]);\n                        ws.end(dataToAppend, function (err) {\n                            if (err) {\n                                return callback(err);\n                            }\n\n                            callback();\n                        })\n                    });\n                });\n            });\n        });\n    };\n\n    this.decryptStream = function (encryptedInputPath, outputFolder, password, callback) {\n\n        const archiver = new PskArchiver();\n\n        decryptFile(encryptedInputPath, tempFolder, password, function (err, tempArchivePath) {\n            if (err) {\n                return callback(err);\n            }\n\n            archiver.on('progress', (progress) => {\n                self.emit('progress', 10 + 0.9 * progress);\n            });\n\n\n            archiver.unzipStream(tempArchivePath, outputFolder, function (err, unzippedFileNames) {\n                if (err) {\n                    return callback(err);\n                }\n\n                utils.deleteRecursively(tempArchivePath, function (err) {\n                    if (err) {\n                        return callback(err);\n                    }\n\n                    callback(undefined, unzippedFileNames);\n                });\n\n            });\n        })\n    };\n\n    this.encryptObject = function (inputObj, dseed, depth, callback) {\n        const archiver = new PskArchiver();\n\n        archiver.zipInMemory(inputObj, depth, function (err, zippedObj) {\n            if (err) {\n                return callback(err);\n            }\n            const cipherText = utils.encrypt(zippedObj, dseed);\n            callback(null, cipherText);\n        })\n    };\n\n    this.decryptObject = function (encryptedData, dseed, callback) {\n        const archiver = new PskArchiver();\n\n        const zippedObject = utils.decrypt(encryptedData, dseed);\n        archiver.unzipInMemory(zippedObject, function (err, obj) {\n            if (err) {\n                return callback(err);\n            }\n            callback(null, obj);\n        })\n    };\n\n    this.pskHash = function (data) {\n        if (Buffer.isBuffer(data)) {\n            return utils.createPskHash(data);\n        }\n        if (data instanceof Object) {\n            return utils.createPskHash(JSON.stringify(data));\n        }\n        return utils.createPskHash(data);\n    };\n\n    this.pskHashStream = function (readStream, callback) {\n        const pskHash = new utils.PskHash();\n\n        readStream.on('data', (chunk) => {\n            pskHash.update(chunk);\n        });\n\n\n        readStream.on('end', () => {\n            callback(null, pskHash.digest());\n        })\n    };\n\n\n    this.saveData = function (data, password, path, callback) {\n        const encryptionKey = this.deriveKey(password, null, null);\n        const iv = crypto.randomBytes(16);\n        const cipher = crypto.createCipheriv('aes-256-cfb', encryptionKey, iv);\n        let encryptedDSeed = cipher.update(data, 'binary');\n        const final = Buffer.from(cipher.final('binary'), 'binary');\n        encryptedDSeed = Buffer.concat([iv, encryptedDSeed, final]);\n        fs.writeFile(path, encryptedDSeed, function (err) {\n            callback(err);\n        });\n    };\n\n\n    this.loadData = function (password, path, callback) {\n\n        fs.readFile(path, null, (err, encryptedData) => {\n            if (err) {\n                callback(err);\n            } else {\n                const iv = encryptedData.slice(0, 16);\n                const encryptedDseed = encryptedData.slice(16);\n                const encryptionKey = this.deriveKey(password, null, null);\n                const decipher = crypto.createDecipheriv('aes-256-cfb', encryptionKey, iv);\n                let dseed = Buffer.from(decipher.update(encryptedDseed, 'binary'), 'binary');\n                const final = Buffer.from(decipher.final('binary'), 'binary');\n                dseed = Buffer.concat([dseed, final]);\n                callback(null, dseed);\n            }\n        });\n    };\n\n\n    this.generateSafeUid = function (password, additionalData) {\n        password = password || Buffer.alloc(0);\n        if (!additionalData) {\n            additionalData = Buffer.alloc(0);\n        }\n\n        if (!Buffer.isBuffer(additionalData)) {\n            additionalData = Buffer.from(additionalData);\n        }\n\n        return utils.encode(this.pskHash(Buffer.concat([password, additionalData])));\n    };\n\n    this.deriveKey = function deriveKey(password, iterations, dkLen) {\n        iterations = iterations || 1000;\n        dkLen = dkLen || 32;\n        const salt = utils.generateSalt(password, 32);\n        const dk = crypto.pbkdf2Sync(password, salt, iterations, dkLen, 'sha512');\n        return Buffer.from(dk);\n    };\n\n    this.randomBytes = crypto.randomBytes;\n    this.PskHash = utils.PskHash;\n\n    //-------------------------- Internal functions -----------------------------------\n    function decryptFile(encryptedInputPath, tempFolder, password, callback) {\n        fs.stat(encryptedInputPath, function (err, stats) {\n            if (err) {\n                return callback(err, null);\n            }\n\n            const fileSizeInBytes = stats.size;\n\n            fs.open(encryptedInputPath, \"r\", function (err, fd) {\n                if (err) {\n                    callback(err, null);\n                } else {\n                    const encryptedAuthData = Buffer.alloc(80);\n\n                    fs.read(fd, encryptedAuthData, 0, 80, fileSizeInBytes - 80, function (err, bytesRead) {\n                        const salt = encryptedAuthData.slice(0, 64);\n                        const keySalt = salt.slice(0, 32);\n                        const aadSalt = salt.slice(-32);\n\n                        const iv = crypto.pbkdf2Sync(password, salt, utils.iterations_number, 12, 'sha512');\n                        const key = crypto.pbkdf2Sync(password, keySalt, utils.iterations_number, 32, 'sha512');\n                        const aad = crypto.pbkdf2Sync(password, aadSalt, utils.iterations_number, 32, 'sha512');\n                        const tag = encryptedAuthData.slice(-16);\n\n                        const decipher = crypto.createDecipheriv(utils.algorithm, key, iv);\n\n                        decipher.setAAD(aad);\n                        decipher.setAuthTag(tag);\n                        const rs = fs.createReadStream(encryptedInputPath, {start: 0, end: fileSizeInBytes - 81});\n                        fs.mkdir(tempFolder, {recursive: true}, function (err) {\n\n                            if (err) {\n                                return callback(err);\n                            }\n                            const tempArchivePath = path.join(tempFolder, path.basename(encryptedInputPath) + \".zip\");\n\n                            fs.open(tempArchivePath, \"w\", function (err, fd) {\n                                if (err) {\n                                    callback(err);\n                                    return;\n                                }\n\n                                fs.close(fd, function (err) {\n\n                                    if (err) {\n                                        return callback(err);\n                                    }\n\n                                    const ptStream = new PassThroughStream();\n\n                                    const ws = fs.createWriteStream(tempArchivePath, {autoClose: false});\n                                    ws.on(\"finish\", function () {\n                                        callback(null, tempArchivePath);\n                                    });\n\n\n                                    let progressLength = 0;\n                                    let totalLength = 0;\n\n                                    /**\n                                     * TODO review this\n                                     * In browser, piping will block the event loop and the stack queue is not called.\n                                     */\n                                    rs.on(\"data\", (chunk) => {\n                                        progressLength += chunk.length;\n                                        totalLength += chunk.length;\n\n                                        if (progressLength > 300000) {\n                                            progressLength = 0;\n                                            emitProgress(fileSizeInBytes, totalLength)\n                                        }\n                                    });\n\n                                    rs.pipe(decipher).pipe(ptStream).pipe(ws);\n\n                                });\n                            });\n                        });\n\n                    });\n\n                }\n            });\n        });\n    }\n\n    function emitProgress(total, processed) {\n\n\n        if (processed > total) {\n            processed = total;\n        }\n\n        const progress = (100 * processed) / total;\n        self.emit('progress', parseInt(progress));\n    }\n\n}\n\nmodule.exports = new PskCrypto();\n","var asn1 = require('./asn1');\nvar inherits = require('util').inherits;\n\nvar api = exports;\n\napi.define = function define(name, body) {\n  return new Entity(name, body);\n};\n\nfunction Entity(name, body) {\n  this.name = name;\n  this.body = body;\n\n  this.decoders = {};\n  this.encoders = {};\n};\n\nEntity.prototype._createNamed = function createNamed(base) {\n  var named;\n  try {\n    named = require('vm').runInThisContext(\n      '(function ' + this.name + '(entity) {\\n' +\n      '  this._initNamed(entity);\\n' +\n      '})'\n    );\n  } catch (e) {\n    named = function (entity) {\n      this._initNamed(entity);\n    };\n  }\n  inherits(named, base);\n  named.prototype._initNamed = function initnamed(entity) {\n    base.call(this, entity);\n  };\n\n  return new named(this);\n};\n\nEntity.prototype._getDecoder = function _getDecoder(enc) {\n  // Lazily create decoder\n  if (!this.decoders.hasOwnProperty(enc))\n    this.decoders[enc] = this._createNamed(asn1.decoders[enc]);\n  return this.decoders[enc];\n};\n\nEntity.prototype.decode = function decode(data, enc, options) {\n  return this._getDecoder(enc).decode(data, options);\n};\n\nEntity.prototype._getEncoder = function _getEncoder(enc) {\n  // Lazily create encoder\n  if (!this.encoders.hasOwnProperty(enc))\n    this.encoders[enc] = this._createNamed(asn1.encoders[enc]);\n  return this.encoders[enc];\n};\n\nEntity.prototype.encode = function encode(data, enc, /* internal */ reporter) {\n  return this._getEncoder(enc).encode(data, reporter);\n};\n","var asn1 = exports;\n\nasn1.bignum = require('./bignum/bn');\n\nasn1.define = require('./api').define;\nasn1.base = require('./base/index');\nasn1.constants = require('./constants/index');\nasn1.decoders = require('./decoders/index');\nasn1.encoders = require('./encoders/index');\n","var inherits = require('util').inherits;\nvar Reporter = require('../base').Reporter;\nvar Buffer = require('buffer').Buffer;\n\nfunction DecoderBuffer(base, options) {\n  Reporter.call(this, options);\n  if (!Buffer.isBuffer(base)) {\n    this.error('Input not Buffer');\n    return;\n  }\n\n  this.base = base;\n  this.offset = 0;\n  this.length = base.length;\n}\ninherits(DecoderBuffer, Reporter);\nexports.DecoderBuffer = DecoderBuffer;\n\nDecoderBuffer.prototype.save = function save() {\n  return { offset: this.offset, reporter: Reporter.prototype.save.call(this) };\n};\n\nDecoderBuffer.prototype.restore = function restore(save) {\n  // Return skipped data\n  var res = new DecoderBuffer(this.base);\n  res.offset = save.offset;\n  res.length = this.offset;\n\n  this.offset = save.offset;\n  Reporter.prototype.restore.call(this, save.reporter);\n\n  return res;\n};\n\nDecoderBuffer.prototype.isEmpty = function isEmpty() {\n  return this.offset === this.length;\n};\n\nDecoderBuffer.prototype.readUInt8 = function readUInt8(fail) {\n  if (this.offset + 1 <= this.length)\n    return this.base.readUInt8(this.offset++, true);\n  else\n    return this.error(fail || 'DecoderBuffer overrun');\n}\n\nDecoderBuffer.prototype.skip = function skip(bytes, fail) {\n  if (!(this.offset + bytes <= this.length))\n    return this.error(fail || 'DecoderBuffer overrun');\n\n  var res = new DecoderBuffer(this.base);\n\n  // Share reporter state\n  res._reporterState = this._reporterState;\n\n  res.offset = this.offset;\n  res.length = this.offset + bytes;\n  this.offset += bytes;\n  return res;\n}\n\nDecoderBuffer.prototype.raw = function raw(save) {\n  return this.base.slice(save ? save.offset : this.offset, this.length);\n}\n\nfunction EncoderBuffer(value, reporter) {\n  if (Array.isArray(value)) {\n    this.length = 0;\n    this.value = value.map(function(item) {\n      if (!(item instanceof EncoderBuffer))\n        item = new EncoderBuffer(item, reporter);\n      this.length += item.length;\n      return item;\n    }, this);\n  } else if (typeof value === 'number') {\n    if (!(0 <= value && value <= 0xff))\n      return reporter.error('non-byte EncoderBuffer value');\n    this.value = value;\n    this.length = 1;\n  } else if (typeof value === 'string') {\n    this.value = value;\n    this.length = Buffer.byteLength(value);\n  } else if (Buffer.isBuffer(value)) {\n    this.value = value;\n    this.length = value.length;\n  } else {\n    return reporter.error('Unsupported type: ' + typeof value);\n  }\n}\nexports.EncoderBuffer = EncoderBuffer;\n\nEncoderBuffer.prototype.join = function join(out, offset) {\n  if (!out)\n    out = new Buffer(this.length);\n  if (!offset)\n    offset = 0;\n\n  if (this.length === 0)\n    return out;\n\n  if (Array.isArray(this.value)) {\n    this.value.forEach(function(item) {\n      item.join(out, offset);\n      offset += item.length;\n    });\n  } else {\n    if (typeof this.value === 'number')\n      out[offset] = this.value;\n    else if (typeof this.value === 'string')\n      out.write(this.value, offset);\n    else if (Buffer.isBuffer(this.value))\n      this.value.copy(out, offset);\n    offset += this.length;\n  }\n\n  return out;\n};\n","var base = exports;\n\nbase.Reporter = require('./reporter').Reporter;\nbase.DecoderBuffer = require('./buffer').DecoderBuffer;\nbase.EncoderBuffer = require('./buffer').EncoderBuffer;\nbase.Node = require('./node');\n","var Reporter = require('../base').Reporter;\nvar EncoderBuffer = require('../base').EncoderBuffer;\n//var assert = require('double-check').assert;\n\n// Supported tags\nvar tags = [\n  'seq', 'seqof', 'set', 'setof', 'octstr', 'bitstr', 'objid', 'bool',\n  'gentime', 'utctime', 'null_', 'enum', 'int', 'ia5str', 'utf8str'\n];\n\n// Public methods list\nvar methods = [\n  'key', 'obj', 'use', 'optional', 'explicit', 'implicit', 'def', 'choice',\n  'any'\n].concat(tags);\n\n// Overrided methods list\nvar overrided = [\n  '_peekTag', '_decodeTag', '_use',\n  '_decodeStr', '_decodeObjid', '_decodeTime',\n  '_decodeNull', '_decodeInt', '_decodeBool', '_decodeList',\n\n  '_encodeComposite', '_encodeStr', '_encodeObjid', '_encodeTime',\n  '_encodeNull', '_encodeInt', '_encodeBool'\n];\n\nfunction Node(enc, parent) {\n  var state = {};\n  this._baseState = state;\n\n  state.enc = enc;\n\n  state.parent = parent || null;\n  state.children = null;\n\n  // State\n  state.tag = null;\n  state.args = null;\n  state.reverseArgs = null;\n  state.choice = null;\n  state.optional = false;\n  state.any = false;\n  state.obj = false;\n  state.use = null;\n  state.useDecoder = null;\n  state.key = null;\n  state['default'] = null;\n  state.explicit = null;\n  state.implicit = null;\n\n  // Should create new instance on each method\n  if (!state.parent) {\n    state.children = [];\n    this._wrap();\n  }\n}\nmodule.exports = Node;\n\nvar stateProps = [\n  'enc', 'parent', 'children', 'tag', 'args', 'reverseArgs', 'choice',\n  'optional', 'any', 'obj', 'use', 'alteredUse', 'key', 'default', 'explicit',\n  'implicit'\n];\n\nNode.prototype.clone = function clone() {\n  var state = this._baseState;\n  var cstate = {};\n  stateProps.forEach(function(prop) {\n    cstate[prop] = state[prop];\n  });\n  var res = new this.constructor(cstate.parent);\n  res._baseState = cstate;\n  return res;\n};\n\nNode.prototype._wrap = function wrap() {\n  var state = this._baseState;\n  methods.forEach(function(method) {\n    this[method] = function _wrappedMethod() {\n      var clone = new this.constructor(this);\n      state.children.push(clone);\n      return clone[method].apply(clone, arguments);\n    };\n  }, this);\n};\n\nNode.prototype._init = function init(body) {\n  var state = this._baseState;\n\n  //assert.equal(state.parent,null,'state.parent should be null');\n  body.call(this);\n\n  // Filter children\n  state.children = state.children.filter(function(child) {\n    return child._baseState.parent === this;\n  }, this);\n  // assert.equal(state.children.length, 1, 'Root node can have only one child');\n};\n\nNode.prototype._useArgs = function useArgs(args) {\n  var state = this._baseState;\n\n  // Filter children and args\n  var children = args.filter(function(arg) {\n    return arg instanceof this.constructor;\n  }, this);\n  args = args.filter(function(arg) {\n    return !(arg instanceof this.constructor);\n  }, this);\n\n  if (children.length !== 0) {\n    // assert.equal(state.children, null, 'state.children should be null');\n    state.children = children;\n\n    // Replace parent to maintain backward link\n    children.forEach(function(child) {\n      child._baseState.parent = this;\n    }, this);\n  }\n  if (args.length !== 0) {\n    // assert.equal(state.args, null, 'state.args should be null');\n    state.args = args;\n    state.reverseArgs = args.map(function(arg) {\n      if (typeof arg !== 'object' || arg.constructor !== Object)\n        return arg;\n\n      var res = {};\n      Object.keys(arg).forEach(function(key) {\n        if (key == (key | 0))\n          key |= 0;\n        var value = arg[key];\n        res[value] = key;\n      });\n      return res;\n    });\n  }\n};\n\n//\n// Overrided methods\n//\n\noverrided.forEach(function(method) {\n  Node.prototype[method] = function _overrided() {\n    var state = this._baseState;\n    throw new Error(method + ' not implemented for encoding: ' + state.enc);\n  };\n});\n\n//\n// Public methods\n//\n\ntags.forEach(function(tag) {\n  Node.prototype[tag] = function _tagMethod() {\n    var state = this._baseState;\n    var args = Array.prototype.slice.call(arguments);\n\n    // assert.equal(state.tag, null, 'state.tag should be null');\n    state.tag = tag;\n\n    this._useArgs(args);\n\n    return this;\n  };\n});\n\nNode.prototype.use = function use(item) {\n  var state = this._baseState;\n\n  // assert.equal(state.use, null, 'state.use should be null');\n  state.use = item;\n\n  return this;\n};\n\nNode.prototype.optional = function optional() {\n  var state = this._baseState;\n\n  state.optional = true;\n\n  return this;\n};\n\nNode.prototype.def = function def(val) {\n  var state = this._baseState;\n\n  // assert.equal(state['default'], null, \"state['default'] should be null\");\n  state['default'] = val;\n  state.optional = true;\n\n  return this;\n};\n\nNode.prototype.explicit = function explicit(num) {\n  var state = this._baseState;\n\n  // assert.equal(state.explicit,null, 'state.explicit should be null');\n  // assert.equal(state.implicit,null, 'state.implicit should be null');\n\n  state.explicit = num;\n\n  return this;\n};\n\nNode.prototype.implicit = function implicit(num) {\n  var state = this._baseState;\n\n    // assert.equal(state.explicit,null, 'state.explicit should be null');\n    // assert.equal(state.implicit,null, 'state.implicit should be null');\n\n    state.implicit = num;\n\n  return this;\n};\n\nNode.prototype.obj = function obj() {\n  var state = this._baseState;\n  var args = Array.prototype.slice.call(arguments);\n\n  state.obj = true;\n\n  if (args.length !== 0)\n    this._useArgs(args);\n\n  return this;\n};\n\nNode.prototype.key = function key(newKey) {\n  var state = this._baseState;\n\n  // assert.equal(state.key, null, 'state.key should be null');\n  state.key = newKey;\n\n  return this;\n};\n\nNode.prototype.any = function any() {\n  var state = this._baseState;\n\n  state.any = true;\n\n  return this;\n};\n\nNode.prototype.choice = function choice(obj) {\n  var state = this._baseState;\n\n  // assert.equal(state.choice, null,'state.choice should be null');\n  state.choice = obj;\n  this._useArgs(Object.keys(obj).map(function(key) {\n    return obj[key];\n  }));\n\n  return this;\n};\n\n//\n// Decoding\n//\n\nNode.prototype._decode = function decode(input) {\n  var state = this._baseState;\n\n  // Decode root node\n  if (state.parent === null)\n    return input.wrapResult(state.children[0]._decode(input));\n\n  var result = state['default'];\n  var present = true;\n\n  var prevKey;\n  if (state.key !== null)\n    prevKey = input.enterKey(state.key);\n\n  // Check if tag is there\n  if (state.optional) {\n    var tag = null;\n    if (state.explicit !== null)\n      tag = state.explicit;\n    else if (state.implicit !== null)\n      tag = state.implicit;\n    else if (state.tag !== null)\n      tag = state.tag;\n\n    if (tag === null && !state.any) {\n      // Trial and Error\n      var save = input.save();\n      try {\n        if (state.choice === null)\n          this._decodeGeneric(state.tag, input);\n        else\n          this._decodeChoice(input);\n        present = true;\n      } catch (e) {\n        present = false;\n      }\n      input.restore(save);\n    } else {\n      present = this._peekTag(input, tag, state.any);\n\n      if (input.isError(present))\n        return present;\n    }\n  }\n\n  // Push object on stack\n  var prevObj;\n  if (state.obj && present)\n    prevObj = input.enterObject();\n\n  if (present) {\n    // Unwrap explicit values\n    if (state.explicit !== null) {\n      var explicit = this._decodeTag(input, state.explicit);\n      if (input.isError(explicit))\n        return explicit;\n      input = explicit;\n    }\n\n    // Unwrap implicit and normal values\n    if (state.use === null && state.choice === null) {\n      if (state.any)\n        var save = input.save();\n      var body = this._decodeTag(\n        input,\n        state.implicit !== null ? state.implicit : state.tag,\n        state.any\n      );\n      if (input.isError(body))\n        return body;\n\n      if (state.any)\n        result = input.raw(save);\n      else\n        input = body;\n    }\n\n    // Select proper method for tag\n    if (state.any)\n      result = result;\n    else if (state.choice === null)\n      result = this._decodeGeneric(state.tag, input);\n    else\n      result = this._decodeChoice(input);\n\n    if (input.isError(result))\n      return result;\n\n    // Decode children\n    if (!state.any && state.choice === null && state.children !== null) {\n      var fail = state.children.some(function decodeChildren(child) {\n        // NOTE: We are ignoring errors here, to let parser continue with other\n        // parts of encoded data\n        child._decode(input);\n      });\n      if (fail)\n        return err;\n    }\n  }\n\n  // Pop object\n  if (state.obj && present)\n    result = input.leaveObject(prevObj);\n\n  // Set key\n  if (state.key !== null && (result !== null || present === true))\n    input.leaveKey(prevKey, state.key, result);\n\n  return result;\n};\n\nNode.prototype._decodeGeneric = function decodeGeneric(tag, input) {\n  var state = this._baseState;\n\n  if (tag === 'seq' || tag === 'set')\n    return null;\n  if (tag === 'seqof' || tag === 'setof')\n    return this._decodeList(input, tag, state.args[0]);\n  else if (tag === 'octstr' || tag === 'bitstr')\n    return this._decodeStr(input, tag);\n  else if (tag === 'ia5str' || tag === 'utf8str')\n    return this._decodeStr(input, tag);\n  else if (tag === 'objid' && state.args)\n    return this._decodeObjid(input, state.args[0], state.args[1]);\n  else if (tag === 'objid')\n    return this._decodeObjid(input, null, null);\n  else if (tag === 'gentime' || tag === 'utctime')\n    return this._decodeTime(input, tag);\n  else if (tag === 'null_')\n    return this._decodeNull(input);\n  else if (tag === 'bool')\n    return this._decodeBool(input);\n  else if (tag === 'int' || tag === 'enum')\n    return this._decodeInt(input, state.args && state.args[0]);\n  else if (state.use !== null)\n    return this._getUse(state.use, input._reporterState.obj)._decode(input);\n  else\n    return input.error('unknown tag: ' + tag);\n\n  return null;\n};\n\nNode.prototype._getUse = function _getUse(entity, obj) {\n\n  var state = this._baseState;\n  // Create altered use decoder if implicit is set\n  state.useDecoder = this._use(entity, obj);\n  // assert.equal(state.useDecoder._baseState.parent, null, 'state.useDecoder._baseState.parent should be null');\n  state.useDecoder = state.useDecoder._baseState.children[0];\n  if (state.implicit !== state.useDecoder._baseState.implicit) {\n    state.useDecoder = state.useDecoder.clone();\n    state.useDecoder._baseState.implicit = state.implicit;\n  }\n  return state.useDecoder;\n};\n\nNode.prototype._decodeChoice = function decodeChoice(input) {\n  var state = this._baseState;\n  var result = null;\n  var match = false;\n\n  Object.keys(state.choice).some(function(key) {\n    var save = input.save();\n    var node = state.choice[key];\n    try {\n      var value = node._decode(input);\n      if (input.isError(value))\n        return false;\n\n      result = { type: key, value: value };\n      match = true;\n    } catch (e) {\n      input.restore(save);\n      return false;\n    }\n    return true;\n  }, this);\n\n  if (!match)\n    return input.error('Choice not matched');\n\n  return result;\n};\n\n//\n// Encoding\n//\n\nNode.prototype._createEncoderBuffer = function createEncoderBuffer(data) {\n  return new EncoderBuffer(data, this.reporter);\n};\n\nNode.prototype._encode = function encode(data, reporter, parent) {\n  var state = this._baseState;\n  if (state['default'] !== null && state['default'] === data)\n    return;\n\n  var result = this._encodeValue(data, reporter, parent);\n  if (result === undefined)\n    return;\n\n  if (this._skipDefault(result, reporter, parent))\n    return;\n\n  return result;\n};\n\nNode.prototype._encodeValue = function encode(data, reporter, parent) {\n  var state = this._baseState;\n\n  // Decode root node\n  if (state.parent === null)\n    return state.children[0]._encode(data, reporter || new Reporter());\n\n  var result = null;\n  var present = true;\n\n  // Set reporter to share it with a child class\n  this.reporter = reporter;\n\n  // Check if data is there\n  if (state.optional && data === undefined) {\n    if (state['default'] !== null)\n      data = state['default']\n    else\n      return;\n  }\n\n  // For error reporting\n  var prevKey;\n\n  // Encode children first\n  var content = null;\n  var primitive = false;\n  if (state.any) {\n    // Anything that was given is translated to buffer\n    result = this._createEncoderBuffer(data);\n  } else if (state.choice) {\n    result = this._encodeChoice(data, reporter);\n  } else if (state.children) {\n    content = state.children.map(function(child) {\n      if (child._baseState.tag === 'null_')\n        return child._encode(null, reporter, data);\n\n      if (child._baseState.key === null)\n        return reporter.error('Child should have a key');\n      var prevKey = reporter.enterKey(child._baseState.key);\n\n      if (typeof data !== 'object')\n        return reporter.error('Child expected, but input is not object');\n\n      var res = child._encode(data[child._baseState.key], reporter, data);\n      reporter.leaveKey(prevKey);\n\n      return res;\n    }, this).filter(function(child) {\n      return child;\n    });\n\n    content = this._createEncoderBuffer(content);\n  } else {\n    if (state.tag === 'seqof' || state.tag === 'setof') {\n      // TODO(indutny): this should be thrown on DSL level\n      if (!(state.args && state.args.length === 1))\n        return reporter.error('Too many args for : ' + state.tag);\n\n      if (!Array.isArray(data))\n        return reporter.error('seqof/setof, but data is not Array');\n\n      var child = this.clone();\n      child._baseState.implicit = null;\n      content = this._createEncoderBuffer(data.map(function(item) {\n        var state = this._baseState;\n\n        return this._getUse(state.args[0], data)._encode(item, reporter);\n      }, child));\n    } else if (state.use !== null) {\n      result = this._getUse(state.use, parent)._encode(data, reporter);\n    } else {\n      content = this._encodePrimitive(state.tag, data);\n      primitive = true;\n    }\n  }\n\n  // Encode data itself\n  var result;\n  if (!state.any && state.choice === null) {\n    var tag = state.implicit !== null ? state.implicit : state.tag;\n    var cls = state.implicit === null ? 'universal' : 'context';\n\n    if (tag === null) {\n      if (state.use === null)\n        reporter.error('Tag could be ommited only for .use()');\n    } else {\n      if (state.use === null)\n        result = this._encodeComposite(tag, primitive, cls, content);\n    }\n  }\n\n  // Wrap in explicit\n  if (state.explicit !== null)\n    result = this._encodeComposite(state.explicit, false, 'context', result);\n\n  return result;\n};\n\nNode.prototype._encodeChoice = function encodeChoice(data, reporter) {\n  var state = this._baseState;\n\n  var node = state.choice[data.type];\n  // if (!node) {\n  //   assert(\n  //       false,\n  //       data.type + ' not found in ' +\n  //           JSON.stringify(Object.keys(state.choice)));\n  // }\n  return node._encode(data.value, reporter);\n};\n\nNode.prototype._encodePrimitive = function encodePrimitive(tag, data) {\n  var state = this._baseState;\n\n  if (tag === 'octstr' || tag === 'bitstr' || tag === 'ia5str')\n    return this._encodeStr(data, tag);\n  else if (tag === 'utf8str')\n    return this._encodeStr(data, tag);\n  else if (tag === 'objid' && state.args)\n    return this._encodeObjid(data, state.reverseArgs[0], state.args[1]);\n  else if (tag === 'objid')\n    return this._encodeObjid(data, null, null);\n  else if (tag === 'gentime' || tag === 'utctime')\n    return this._encodeTime(data, tag);\n  else if (tag === 'null_')\n    return this._encodeNull();\n  else if (tag === 'int' || tag === 'enum')\n    return this._encodeInt(data, state.args && state.reverseArgs[0]);\n  else if (tag === 'bool')\n    return this._encodeBool(data);\n  else\n    throw new Error('Unsupported tag: ' + tag);\n};\n","var inherits = require('util').inherits;\n\nfunction Reporter(options) {\n  this._reporterState = {\n    obj: null,\n    path: [],\n    options: options || {},\n    errors: []\n  };\n}\nexports.Reporter = Reporter;\n\nReporter.prototype.isError = function isError(obj) {\n  return obj instanceof ReporterError;\n};\n\nReporter.prototype.save = function save() {\n  var state = this._reporterState;\n\n  return { obj: state.obj, pathLen: state.path.length };\n};\n\nReporter.prototype.restore = function restore(data) {\n  var state = this._reporterState;\n\n  state.obj = data.obj;\n  state.path = state.path.slice(0, data.pathLen);\n};\n\nReporter.prototype.enterKey = function enterKey(key) {\n  return this._reporterState.path.push(key);\n};\n\nReporter.prototype.leaveKey = function leaveKey(index, key, value) {\n  var state = this._reporterState;\n\n  state.path = state.path.slice(0, index - 1);\n  if (state.obj !== null)\n    state.obj[key] = value;\n};\n\nReporter.prototype.enterObject = function enterObject() {\n  var state = this._reporterState;\n\n  var prev = state.obj;\n  state.obj = {};\n  return prev;\n};\n\nReporter.prototype.leaveObject = function leaveObject(prev) {\n  var state = this._reporterState;\n\n  var now = state.obj;\n  state.obj = prev;\n  return now;\n};\n\nReporter.prototype.error = function error(msg) {\n  var err;\n  var state = this._reporterState;\n\n  var inherited = msg instanceof ReporterError;\n  if (inherited) {\n    err = msg;\n  } else {\n    err = new ReporterError(state.path.map(function(elem) {\n      return '[' + JSON.stringify(elem) + ']';\n    }).join(''), msg.message || msg, msg.stack);\n  }\n\n  if (!state.options.partial)\n    throw err;\n\n  if (!inherited)\n    state.errors.push(err);\n\n  return err;\n};\n\nReporter.prototype.wrapResult = function wrapResult(result) {\n  var state = this._reporterState;\n  if (!state.options.partial)\n    return result;\n\n  return {\n    result: this.isError(result) ? null : result,\n    errors: state.errors\n  };\n};\n\nfunction ReporterError(path, msg) {\n  this.path = path;\n  this.rethrow(msg);\n};\ninherits(ReporterError, Error);\n\nReporterError.prototype.rethrow = function rethrow(msg) {\n  this.message = msg + ' at: ' + (this.path || '(shallow)');\n  Error.captureStackTrace(this, ReporterError);\n\n  return this;\n};\n","(function (module, exports) {\n\n'use strict';\n\n// Utils\n\nfunction assert(val, msg) {\n  if (!val)\n    throw new Error(msg || 'Assertion failed');\n}\n\n// Could use `inherits` module, but don't want to move from single file\n// architecture yet.\nfunction inherits(ctor, superCtor) {\n  ctor.super_ = superCtor;\n  var TempCtor = function () {};\n  TempCtor.prototype = superCtor.prototype;\n  ctor.prototype = new TempCtor();\n  ctor.prototype.constructor = ctor;\n}\n\n// BN\n\nfunction BN(number, base, endian) {\n  // May be `new BN(bn)` ?\n  if (number !== null &&\n      typeof number === 'object' &&\n      Array.isArray(number.words)) {\n    return number;\n  }\n\n  this.sign = false;\n  this.words = null;\n  this.length = 0;\n\n  // Reduction context\n  this.red = null;\n\n  if (base === 'le' || base === 'be') {\n    endian = base;\n    base = 10;\n  }\n\n  if (number !== null)\n    this._init(number || 0, base || 10, endian || 'be');\n}\nif (typeof module === 'object')\n  module.exports = BN;\nelse\n  exports.BN = BN;\n\nBN.BN = BN;\nBN.wordSize = 26;\n\nBN.prototype._init = function init(number, base, endian) {\n  if (typeof number === 'number') {\n    return this._initNumber(number, base, endian);\n  } else if (typeof number === 'object') {\n    return this._initArray(number, base, endian);\n  }\n  if (base === 'hex')\n    base = 16;\n  assert(base === (base | 0) && base >= 2 && base <= 36);\n\n  number = number.toString().replace(/\\s+/g, '');\n  var start = 0;\n  if (number[0] === '-')\n    start++;\n\n  if (base === 16)\n    this._parseHex(number, start);\n  else\n    this._parseBase(number, base, start);\n\n  if (number[0] === '-')\n    this.sign = true;\n\n  this.strip();\n\n  if (endian !== 'le')\n    return;\n\n  this._initArray(this.toArray(), base, endian);\n};\n\nBN.prototype._initNumber = function _initNumber(number, base, endian) {\n  if (number < 0) {\n    this.sign = true;\n    number = -number;\n  }\n  if (number < 0x4000000) {\n    this.words = [ number & 0x3ffffff ];\n    this.length = 1;\n  } else if (number < 0x10000000000000) {\n    this.words = [\n      number & 0x3ffffff,\n      (number / 0x4000000) & 0x3ffffff\n    ];\n    this.length = 2;\n  } else {\n    assert(number < 0x20000000000000); // 2 ^ 53 (unsafe)\n    this.words = [\n      number & 0x3ffffff,\n      (number / 0x4000000) & 0x3ffffff,\n      1\n    ];\n    this.length = 3;\n  }\n\n  if (endian !== 'le')\n    return;\n\n  // Reverse the bytes\n  this._initArray(this.toArray(), base, endian);\n};\n\nBN.prototype._initArray = function _initArray(number, base, endian) {\n  // Perhaps a Uint8Array\n  assert(typeof number.length === 'number');\n  if (number.length <= 0) {\n    this.words = [ 0 ];\n    this.length = 1;\n    return this;\n  }\n\n  this.length = Math.ceil(number.length / 3);\n  this.words = new Array(this.length);\n  for (var i = 0; i < this.length; i++)\n    this.words[i] = 0;\n\n  var off = 0;\n  if (endian === 'be') {\n    for (var i = number.length - 1, j = 0; i >= 0; i -= 3) {\n      var w = number[i] | (number[i - 1] << 8) | (number[i - 2] << 16);\n      this.words[j] |= (w << off) & 0x3ffffff;\n      this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff;\n      off += 24;\n      if (off >= 26) {\n        off -= 26;\n        j++;\n      }\n    }\n  } else if (endian === 'le') {\n    for (var i = 0, j = 0; i < number.length; i += 3) {\n      var w = number[i] | (number[i + 1] << 8) | (number[i + 2] << 16);\n      this.words[j] |= (w << off) & 0x3ffffff;\n      this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff;\n      off += 24;\n      if (off >= 26) {\n        off -= 26;\n        j++;\n      }\n    }\n  }\n  return this.strip();\n};\n\nfunction parseHex(str, start, end) {\n  var r = 0;\n  var len = Math.min(str.length, end);\n  for (var i = start; i < len; i++) {\n    var c = str.charCodeAt(i) - 48;\n\n    r <<= 4;\n\n    // 'a' - 'f'\n    if (c >= 49 && c <= 54)\n      r |= c - 49 + 0xa;\n\n    // 'A' - 'F'\n    else if (c >= 17 && c <= 22)\n      r |= c - 17 + 0xa;\n\n    // '0' - '9'\n    else\n      r |= c & 0xf;\n  }\n  return r;\n}\n\nBN.prototype._parseHex = function _parseHex(number, start) {\n  // Create possibly bigger array to ensure that it fits the number\n  this.length = Math.ceil((number.length - start) / 6);\n  this.words = new Array(this.length);\n  for (var i = 0; i < this.length; i++)\n    this.words[i] = 0;\n\n  // Scan 24-bit chunks and add them to the number\n  var off = 0;\n  for (var i = number.length - 6, j = 0; i >= start; i -= 6) {\n    var w = parseHex(number, i, i + 6);\n    this.words[j] |= (w << off) & 0x3ffffff;\n    this.words[j + 1] |= w >>> (26 - off) & 0x3fffff;\n    off += 24;\n    if (off >= 26) {\n      off -= 26;\n      j++;\n    }\n  }\n  if (i + 6 !== start) {\n    var w = parseHex(number, start, i + 6);\n    this.words[j] |= (w << off) & 0x3ffffff;\n    this.words[j + 1] |= w >>> (26 - off) & 0x3fffff;\n  }\n  this.strip();\n};\n\nfunction parseBase(str, start, end, mul) {\n  var r = 0;\n  var len = Math.min(str.length, end);\n  for (var i = start; i < len; i++) {\n    var c = str.charCodeAt(i) - 48;\n\n    r *= mul;\n\n    // 'a'\n    if (c >= 49)\n      r += c - 49 + 0xa;\n\n    // 'A'\n    else if (c >= 17)\n      r += c - 17 + 0xa;\n\n    // '0' - '9'\n    else\n      r += c;\n  }\n  return r;\n}\n\nBN.prototype._parseBase = function _parseBase(number, base, start) {\n  // Initialize as zero\n  this.words = [ 0 ];\n  this.length = 1;\n\n  // Find length of limb in base\n  for (var limbLen = 0, limbPow = 1; limbPow <= 0x3ffffff; limbPow *= base)\n    limbLen++;\n  limbLen--;\n  limbPow = (limbPow / base) | 0;\n\n  var total = number.length - start;\n  var mod = total % limbLen;\n  var end = Math.min(total, total - mod) + start;\n\n  var word = 0;\n  for (var i = start; i < end; i += limbLen) {\n    word = parseBase(number, i, i + limbLen, base);\n\n    this.imuln(limbPow);\n    if (this.words[0] + word < 0x4000000)\n      this.words[0] += word;\n    else\n      this._iaddn(word);\n  }\n\n  if (mod !== 0) {\n    var pow = 1;\n    var word = parseBase(number, i, number.length, base);\n\n    for (var i = 0; i < mod; i++)\n      pow *= base;\n    this.imuln(pow);\n    if (this.words[0] + word < 0x4000000)\n      this.words[0] += word;\n    else\n      this._iaddn(word);\n  }\n};\n\nBN.prototype.copy = function copy(dest) {\n  dest.words = new Array(this.length);\n  for (var i = 0; i < this.length; i++)\n    dest.words[i] = this.words[i];\n  dest.length = this.length;\n  dest.sign = this.sign;\n  dest.red = this.red;\n};\n\nBN.prototype.clone = function clone() {\n  var r = new BN(null);\n  this.copy(r);\n  return r;\n};\n\n// Remove leading `0` from `this`\nBN.prototype.strip = function strip() {\n  while (this.length > 1 && this.words[this.length - 1] === 0)\n    this.length--;\n  return this._normSign();\n};\n\nBN.prototype._normSign = function _normSign() {\n  // -0 = 0\n  if (this.length === 1 && this.words[0] === 0)\n    this.sign = false;\n  return this;\n};\n\nBN.prototype.inspect = function inspect() {\n  return (this.red ? '<BN-R: ' : '<BN: ') + this.toString(16) + '>';\n};\n\n/*\n\nvar zeros = [];\nvar groupSizes = [];\nvar groupBases = [];\n\nvar s = '';\nvar i = -1;\nwhile (++i < BN.wordSize) {\n  zeros[i] = s;\n  s += '0';\n}\ngroupSizes[0] = 0;\ngroupSizes[1] = 0;\ngroupBases[0] = 0;\ngroupBases[1] = 0;\nvar base = 2 - 1;\nwhile (++base < 36 + 1) {\n  var groupSize = 0;\n  var groupBase = 1;\n  while (groupBase < (1 << BN.wordSize) / base) {\n    groupBase *= base;\n    groupSize += 1;\n  }\n  groupSizes[base] = groupSize;\n  groupBases[base] = groupBase;\n}\n\n*/\n\nvar zeros = [\n  '',\n  '0',\n  '00',\n  '000',\n  '0000',\n  '00000',\n  '000000',\n  '0000000',\n  '00000000',\n  '000000000',\n  '0000000000',\n  '00000000000',\n  '000000000000',\n  '0000000000000',\n  '00000000000000',\n  '000000000000000',\n  '0000000000000000',\n  '00000000000000000',\n  '000000000000000000',\n  '0000000000000000000',\n  '00000000000000000000',\n  '000000000000000000000',\n  '0000000000000000000000',\n  '00000000000000000000000',\n  '000000000000000000000000',\n  '0000000000000000000000000'\n];\n\nvar groupSizes = [\n  0, 0,\n  25, 16, 12, 11, 10, 9, 8,\n  8, 7, 7, 7, 7, 6, 6,\n  6, 6, 6, 6, 6, 5, 5,\n  5, 5, 5, 5, 5, 5, 5,\n  5, 5, 5, 5, 5, 5, 5\n];\n\nvar groupBases = [\n  0, 0,\n  33554432, 43046721, 16777216, 48828125, 60466176, 40353607, 16777216,\n  43046721, 10000000, 19487171, 35831808, 62748517, 7529536, 11390625,\n  16777216, 24137569, 34012224, 47045881, 64000000, 4084101, 5153632,\n  6436343, 7962624, 9765625, 11881376, 14348907, 17210368, 20511149,\n  24300000, 28629151, 33554432, 39135393, 45435424, 52521875, 60466176\n];\n\nBN.prototype.toString = function toString(base, padding) {\n  base = base || 10;\n  if (base === 16 || base === 'hex') {\n    var out = '';\n    var off = 0;\n    var padding = padding | 0 || 1;\n    var carry = 0;\n    for (var i = 0; i < this.length; i++) {\n      var w = this.words[i];\n      var word = (((w << off) | carry) & 0xffffff).toString(16);\n      carry = (w >>> (24 - off)) & 0xffffff;\n      if (carry !== 0 || i !== this.length - 1)\n        out = zeros[6 - word.length] + word + out;\n      else\n        out = word + out;\n      off += 2;\n      if (off >= 26) {\n        off -= 26;\n        i--;\n      }\n    }\n    if (carry !== 0)\n      out = carry.toString(16) + out;\n    while (out.length % padding !== 0)\n      out = '0' + out;\n    if (this.sign)\n      out = '-' + out;\n    return out;\n  } else if (base === (base | 0) && base >= 2 && base <= 36) {\n    // var groupSize = Math.floor(BN.wordSize * Math.LN2 / Math.log(base));\n    var groupSize = groupSizes[base];\n    // var groupBase = Math.pow(base, groupSize);\n    var groupBase = groupBases[base];\n    var out = '';\n    var c = this.clone();\n    c.sign = false;\n    while (c.cmpn(0) !== 0) {\n      var r = c.modn(groupBase).toString(base);\n      c = c.idivn(groupBase);\n\n      if (c.cmpn(0) !== 0)\n        out = zeros[groupSize - r.length] + r + out;\n      else\n        out = r + out;\n    }\n    if (this.cmpn(0) === 0)\n      out = '0' + out;\n    if (this.sign)\n      out = '-' + out;\n    return out;\n  } else {\n    assert(false, 'Base should be between 2 and 36');\n  }\n};\n\nBN.prototype.toJSON = function toJSON() {\n  return this.toString(16);\n};\n\nBN.prototype.toArray = function toArray(endian) {\n  this.strip();\n  var res = new Array(this.byteLength());\n  res[0] = 0;\n\n  var q = this.clone();\n  if (endian !== 'le') {\n    // Assume big-endian\n    for (var i = 0; q.cmpn(0) !== 0; i++) {\n      var b = q.andln(0xff);\n      q.ishrn(8);\n\n      res[res.length - i - 1] = b;\n    }\n  } else {\n    // Assume little-endian\n    for (var i = 0; q.cmpn(0) !== 0; i++) {\n      var b = q.andln(0xff);\n      q.ishrn(8);\n\n      res[i] = b;\n    }\n  }\n\n  return res;\n};\n\nif (Math.clz32) {\n  BN.prototype._countBits = function _countBits(w) {\n    return 32 - Math.clz32(w);\n  };\n} else {\n  BN.prototype._countBits = function _countBits(w) {\n    var t = w;\n    var r = 0;\n    if (t >= 0x1000) {\n      r += 13;\n      t >>>= 13;\n    }\n    if (t >= 0x40) {\n      r += 7;\n      t >>>= 7;\n    }\n    if (t >= 0x8) {\n      r += 4;\n      t >>>= 4;\n    }\n    if (t >= 0x02) {\n      r += 2;\n      t >>>= 2;\n    }\n    return r + t;\n  };\n}\n\nBN.prototype._zeroBits = function _zeroBits(w) {\n  // Short-cut\n  if (w === 0)\n    return 26;\n\n  var t = w;\n  var r = 0;\n  if ((t & 0x1fff) === 0) {\n    r += 13;\n    t >>>= 13;\n  }\n  if ((t & 0x7f) === 0) {\n    r += 7;\n    t >>>= 7;\n  }\n  if ((t & 0xf) === 0) {\n    r += 4;\n    t >>>= 4;\n  }\n  if ((t & 0x3) === 0) {\n    r += 2;\n    t >>>= 2;\n  }\n  if ((t & 0x1) === 0)\n    r++;\n  return r;\n};\n\n// Return number of used bits in a BN\nBN.prototype.bitLength = function bitLength() {\n  var hi = 0;\n  var w = this.words[this.length - 1];\n  var hi = this._countBits(w);\n  return (this.length - 1) * 26 + hi;\n};\n\n// Number of trailing zero bits\nBN.prototype.zeroBits = function zeroBits() {\n  if (this.cmpn(0) === 0)\n    return 0;\n\n  var r = 0;\n  for (var i = 0; i < this.length; i++) {\n    var b = this._zeroBits(this.words[i]);\n    r += b;\n    if (b !== 26)\n      break;\n  }\n  return r;\n};\n\nBN.prototype.byteLength = function byteLength() {\n  return Math.ceil(this.bitLength() / 8);\n};\n\n// Return negative clone of `this`\nBN.prototype.neg = function neg() {\n  if (this.cmpn(0) === 0)\n    return this.clone();\n\n  var r = this.clone();\n  r.sign = !this.sign;\n  return r;\n};\n\n\n// Or `num` with `this` in-place\nBN.prototype.ior = function ior(num) {\n  this.sign = this.sign || num.sign;\n\n  while (this.length < num.length)\n    this.words[this.length++] = 0;\n\n  for (var i = 0; i < num.length; i++)\n    this.words[i] = this.words[i] | num.words[i];\n\n  return this.strip();\n};\n\n\n// Or `num` with `this`\nBN.prototype.or = function or(num) {\n  if (this.length > num.length)\n    return this.clone().ior(num);\n  else\n    return num.clone().ior(this);\n};\n\n\n// And `num` with `this` in-place\nBN.prototype.iand = function iand(num) {\n  this.sign = this.sign && num.sign;\n\n  // b = min-length(num, this)\n  var b;\n  if (this.length > num.length)\n    b = num;\n  else\n    b = this;\n\n  for (var i = 0; i < b.length; i++)\n    this.words[i] = this.words[i] & num.words[i];\n\n  this.length = b.length;\n\n  return this.strip();\n};\n\n\n// And `num` with `this`\nBN.prototype.and = function and(num) {\n  if (this.length > num.length)\n    return this.clone().iand(num);\n  else\n    return num.clone().iand(this);\n};\n\n\n// Xor `num` with `this` in-place\nBN.prototype.ixor = function ixor(num) {\n  this.sign = this.sign || num.sign;\n\n  // a.length > b.length\n  var a;\n  var b;\n  if (this.length > num.length) {\n    a = this;\n    b = num;\n  } else {\n    a = num;\n    b = this;\n  }\n\n  for (var i = 0; i < b.length; i++)\n    this.words[i] = a.words[i] ^ b.words[i];\n\n  if (this !== a)\n    for (; i < a.length; i++)\n      this.words[i] = a.words[i];\n\n  this.length = a.length;\n\n  return this.strip();\n};\n\n\n// Xor `num` with `this`\nBN.prototype.xor = function xor(num) {\n  if (this.length > num.length)\n    return this.clone().ixor(num);\n  else\n    return num.clone().ixor(this);\n};\n\n\n// Set `bit` of `this`\nBN.prototype.setn = function setn(bit, val) {\n  assert(typeof bit === 'number' && bit >= 0);\n\n  var off = (bit / 26) | 0;\n  var wbit = bit % 26;\n\n  while (this.length <= off)\n    this.words[this.length++] = 0;\n\n  if (val)\n    this.words[off] = this.words[off] | (1 << wbit);\n  else\n    this.words[off] = this.words[off] & ~(1 << wbit);\n\n  return this.strip();\n};\n\n\n// Add `num` to `this` in-place\nBN.prototype.iadd = function iadd(num) {\n  // negative + positive\n  if (this.sign && !num.sign) {\n    this.sign = false;\n    var r = this.isub(num);\n    this.sign = !this.sign;\n    return this._normSign();\n\n  // positive + negative\n  } else if (!this.sign && num.sign) {\n    num.sign = false;\n    var r = this.isub(num);\n    num.sign = true;\n    return r._normSign();\n  }\n\n  // a.length > b.length\n  var a;\n  var b;\n  if (this.length > num.length) {\n    a = this;\n    b = num;\n  } else {\n    a = num;\n    b = this;\n  }\n\n  var carry = 0;\n  for (var i = 0; i < b.length; i++) {\n    var r = a.words[i] + b.words[i] + carry;\n    this.words[i] = r & 0x3ffffff;\n    carry = r >>> 26;\n  }\n  for (; carry !== 0 && i < a.length; i++) {\n    var r = a.words[i] + carry;\n    this.words[i] = r & 0x3ffffff;\n    carry = r >>> 26;\n  }\n\n  this.length = a.length;\n  if (carry !== 0) {\n    this.words[this.length] = carry;\n    this.length++;\n  // Copy the rest of the words\n  } else if (a !== this) {\n    for (; i < a.length; i++)\n      this.words[i] = a.words[i];\n  }\n\n  return this;\n};\n\n// Add `num` to `this`\nBN.prototype.add = function add(num) {\n  if (num.sign && !this.sign) {\n    num.sign = false;\n    var res = this.sub(num);\n    num.sign = true;\n    return res;\n  } else if (!num.sign && this.sign) {\n    this.sign = false;\n    var res = num.sub(this);\n    this.sign = true;\n    return res;\n  }\n\n  if (this.length > num.length)\n    return this.clone().iadd(num);\n  else\n    return num.clone().iadd(this);\n};\n\n// Subtract `num` from `this` in-place\nBN.prototype.isub = function isub(num) {\n  // this - (-num) = this + num\n  if (num.sign) {\n    num.sign = false;\n    var r = this.iadd(num);\n    num.sign = true;\n    return r._normSign();\n\n  // -this - num = -(this + num)\n  } else if (this.sign) {\n    this.sign = false;\n    this.iadd(num);\n    this.sign = true;\n    return this._normSign();\n  }\n\n  // At this point both numbers are positive\n  var cmp = this.cmp(num);\n\n  // Optimization - zeroify\n  if (cmp === 0) {\n    this.sign = false;\n    this.length = 1;\n    this.words[0] = 0;\n    return this;\n  }\n\n  // a > b\n  var a;\n  var b;\n  if (cmp > 0) {\n    a = this;\n    b = num;\n  } else {\n    a = num;\n    b = this;\n  }\n\n  var carry = 0;\n  for (var i = 0; i < b.length; i++) {\n    var r = a.words[i] - b.words[i] + carry;\n    carry = r >> 26;\n    this.words[i] = r & 0x3ffffff;\n  }\n  for (; carry !== 0 && i < a.length; i++) {\n    var r = a.words[i] + carry;\n    carry = r >> 26;\n    this.words[i] = r & 0x3ffffff;\n  }\n\n  // Copy rest of the words\n  if (carry === 0 && i < a.length && a !== this)\n    for (; i < a.length; i++)\n      this.words[i] = a.words[i];\n  this.length = Math.max(this.length, i);\n\n  if (a !== this)\n    this.sign = true;\n\n  return this.strip();\n};\n\n// Subtract `num` from `this`\nBN.prototype.sub = function sub(num) {\n  return this.clone().isub(num);\n};\n\n/*\n// NOTE: This could be potentionally used to generate loop-less multiplications\nfunction _genCombMulTo(alen, blen) {\n  var len = alen + blen - 1;\n  var src = [\n    'var a = this.words, b = num.words, o = out.words, c = 0, w, ' +\n        'mask = 0x3ffffff, shift = 0x4000000;',\n    'out.length = ' + len + ';'\n  ];\n  for (var k = 0; k < len; k++) {\n    var minJ = Math.max(0, k - alen + 1);\n    var maxJ = Math.min(k, blen - 1);\n\n    for (var j = minJ; j <= maxJ; j++) {\n      var i = k - j;\n      var mul = 'a[' + i + '] * b[' + j + ']';\n\n      if (j === minJ) {\n        src.push('w = ' + mul + ' + c;');\n        src.push('c = (w / shift) | 0;');\n      } else {\n        src.push('w += ' + mul + ';');\n        src.push('c += (w / shift) | 0;');\n      }\n      src.push('w &= mask;');\n    }\n    src.push('o[' + k + '] = w;');\n  }\n  src.push('if (c !== 0) {',\n           '  o[' + k + '] = c;',\n           '  out.length++;',\n           '}',\n           'return out;');\n\n  return src.join('\\n');\n}\n*/\n\nBN.prototype._smallMulTo = function _smallMulTo(num, out) {\n  out.sign = num.sign !== this.sign;\n  out.length = this.length + num.length;\n\n  var carry = 0;\n  for (var k = 0; k < out.length - 1; k++) {\n    // Sum all words with the same `i + j = k` and accumulate `ncarry`,\n    // note that ncarry could be >= 0x3ffffff\n    var ncarry = carry >>> 26;\n    var rword = carry & 0x3ffffff;\n    var maxJ = Math.min(k, num.length - 1);\n    for (var j = Math.max(0, k - this.length + 1); j <= maxJ; j++) {\n      var i = k - j;\n      var a = this.words[i] | 0;\n      var b = num.words[j] | 0;\n      var r = a * b;\n\n      var lo = r & 0x3ffffff;\n      ncarry = (ncarry + ((r / 0x4000000) | 0)) | 0;\n      lo = (lo + rword) | 0;\n      rword = lo & 0x3ffffff;\n      ncarry = (ncarry + (lo >>> 26)) | 0;\n    }\n    out.words[k] = rword;\n    carry = ncarry;\n  }\n  if (carry !== 0) {\n    out.words[k] = carry;\n  } else {\n    out.length--;\n  }\n\n  return out.strip();\n};\n\nBN.prototype._bigMulTo = function _bigMulTo(num, out) {\n  out.sign = num.sign !== this.sign;\n  out.length = this.length + num.length;\n\n  var carry = 0;\n  var hncarry = 0;\n  for (var k = 0; k < out.length - 1; k++) {\n    // Sum all words with the same `i + j = k` and accumulate `ncarry`,\n    // note that ncarry could be >= 0x3ffffff\n    var ncarry = hncarry;\n    hncarry = 0;\n    var rword = carry & 0x3ffffff;\n    var maxJ = Math.min(k, num.length - 1);\n    for (var j = Math.max(0, k - this.length + 1); j <= maxJ; j++) {\n      var i = k - j;\n      var a = this.words[i] | 0;\n      var b = num.words[j] | 0;\n      var r = a * b;\n\n      var lo = r & 0x3ffffff;\n      ncarry = (ncarry + ((r / 0x4000000) | 0)) | 0;\n      lo = (lo + rword) | 0;\n      rword = lo & 0x3ffffff;\n      ncarry = (ncarry + (lo >>> 26)) | 0;\n\n      hncarry += ncarry >>> 26;\n      ncarry &= 0x3ffffff;\n    }\n    out.words[k] = rword;\n    carry = ncarry;\n    ncarry = hncarry;\n  }\n  if (carry !== 0) {\n    out.words[k] = carry;\n  } else {\n    out.length--;\n  }\n\n  return out.strip();\n};\n\nBN.prototype.mulTo = function mulTo(num, out) {\n  var res;\n  if (this.length + num.length < 63)\n    res = this._smallMulTo(num, out);\n  else\n    res = this._bigMulTo(num, out);\n  return res;\n};\n\n// Multiply `this` by `num`\nBN.prototype.mul = function mul(num) {\n  var out = new BN(null);\n  out.words = new Array(this.length + num.length);\n  return this.mulTo(num, out);\n};\n\n// In-place Multiplication\nBN.prototype.imul = function imul(num) {\n  if (this.cmpn(0) === 0 || num.cmpn(0) === 0) {\n    this.words[0] = 0;\n    this.length = 1;\n    return this;\n  }\n\n  var tlen = this.length;\n  var nlen = num.length;\n\n  this.sign = num.sign !== this.sign;\n  this.length = this.length + num.length;\n  this.words[this.length - 1] = 0;\n\n  for (var k = this.length - 2; k >= 0; k--) {\n    // Sum all words with the same `i + j = k` and accumulate `carry`,\n    // note that carry could be >= 0x3ffffff\n    var carry = 0;\n    var rword = 0;\n    var maxJ = Math.min(k, nlen - 1);\n    for (var j = Math.max(0, k - tlen + 1); j <= maxJ; j++) {\n      var i = k - j;\n      var a = this.words[i];\n      var b = num.words[j];\n      var r = a * b;\n\n      var lo = r & 0x3ffffff;\n      carry += (r / 0x4000000) | 0;\n      lo += rword;\n      rword = lo & 0x3ffffff;\n      carry += lo >>> 26;\n    }\n    this.words[k] = rword;\n    this.words[k + 1] += carry;\n    carry = 0;\n  }\n\n  // Propagate overflows\n  var carry = 0;\n  for (var i = 1; i < this.length; i++) {\n    var w = this.words[i] + carry;\n    this.words[i] = w & 0x3ffffff;\n    carry = w >>> 26;\n  }\n\n  return this.strip();\n};\n\nBN.prototype.imuln = function imuln(num) {\n  assert(typeof num === 'number');\n\n  // Carry\n  var carry = 0;\n  for (var i = 0; i < this.length; i++) {\n    var w = this.words[i] * num;\n    var lo = (w & 0x3ffffff) + (carry & 0x3ffffff);\n    carry >>= 26;\n    carry += (w / 0x4000000) | 0;\n    // NOTE: lo is 27bit maximum\n    carry += lo >>> 26;\n    this.words[i] = lo & 0x3ffffff;\n  }\n\n  if (carry !== 0) {\n    this.words[i] = carry;\n    this.length++;\n  }\n\n  return this;\n};\n\nBN.prototype.muln = function muln(num) {\n  return this.clone().imuln(num);\n};\n\n// `this` * `this`\nBN.prototype.sqr = function sqr() {\n  return this.mul(this);\n};\n\n// `this` * `this` in-place\nBN.prototype.isqr = function isqr() {\n  return this.mul(this);\n};\n\n// Shift-left in-place\nBN.prototype.ishln = function ishln(bits) {\n  assert(typeof bits === 'number' && bits >= 0);\n  var r = bits % 26;\n  var s = (bits - r) / 26;\n  var carryMask = (0x3ffffff >>> (26 - r)) << (26 - r);\n\n  if (r !== 0) {\n    var carry = 0;\n    for (var i = 0; i < this.length; i++) {\n      var newCarry = this.words[i] & carryMask;\n      var c = (this.words[i] - newCarry) << r;\n      this.words[i] = c | carry;\n      carry = newCarry >>> (26 - r);\n    }\n    if (carry) {\n      this.words[i] = carry;\n      this.length++;\n    }\n  }\n\n  if (s !== 0) {\n    for (var i = this.length - 1; i >= 0; i--)\n      this.words[i + s] = this.words[i];\n    for (var i = 0; i < s; i++)\n      this.words[i] = 0;\n    this.length += s;\n  }\n\n  return this.strip();\n};\n\n// Shift-right in-place\n// NOTE: `hint` is a lowest bit before trailing zeroes\n// NOTE: if `extended` is present - it will be filled with destroyed bits\nBN.prototype.ishrn = function ishrn(bits, hint, extended) {\n  assert(typeof bits === 'number' && bits >= 0);\n  var h;\n  if (hint)\n    h = (hint - (hint % 26)) / 26;\n  else\n    h = 0;\n\n  var r = bits % 26;\n  var s = Math.min((bits - r) / 26, this.length);\n  var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r);\n  var maskedWords = extended;\n\n  h -= s;\n  h = Math.max(0, h);\n\n  // Extended mode, copy masked part\n  if (maskedWords) {\n    for (var i = 0; i < s; i++)\n      maskedWords.words[i] = this.words[i];\n    maskedWords.length = s;\n  }\n\n  if (s === 0) {\n    // No-op, we should not move anything at all\n  } else if (this.length > s) {\n    this.length -= s;\n    for (var i = 0; i < this.length; i++)\n      this.words[i] = this.words[i + s];\n  } else {\n    this.words[0] = 0;\n    this.length = 1;\n  }\n\n  var carry = 0;\n  for (var i = this.length - 1; i >= 0 && (carry !== 0 || i >= h); i--) {\n    var word = this.words[i];\n    this.words[i] = (carry << (26 - r)) | (word >>> r);\n    carry = word & mask;\n  }\n\n  // Push carried bits as a mask\n  if (maskedWords && carry !== 0)\n    maskedWords.words[maskedWords.length++] = carry;\n\n  if (this.length === 0) {\n    this.words[0] = 0;\n    this.length = 1;\n  }\n\n  this.strip();\n\n  return this;\n};\n\n// Shift-left\nBN.prototype.shln = function shln(bits) {\n  return this.clone().ishln(bits);\n};\n\n// Shift-right\nBN.prototype.shrn = function shrn(bits) {\n  return this.clone().ishrn(bits);\n};\n\n// Test if n bit is set\nBN.prototype.testn = function testn(bit) {\n  assert(typeof bit === 'number' && bit >= 0);\n  var r = bit % 26;\n  var s = (bit - r) / 26;\n  var q = 1 << r;\n\n  // Fast case: bit is much higher than all existing words\n  if (this.length <= s) {\n    return false;\n  }\n\n  // Check bit and return\n  var w = this.words[s];\n\n  return !!(w & q);\n};\n\n// Return only lowers bits of number (in-place)\nBN.prototype.imaskn = function imaskn(bits) {\n  assert(typeof bits === 'number' && bits >= 0);\n  var r = bits % 26;\n  var s = (bits - r) / 26;\n\n  assert(!this.sign, 'imaskn works only with positive numbers');\n\n  if (r !== 0)\n    s++;\n  this.length = Math.min(s, this.length);\n\n  if (r !== 0) {\n    var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r);\n    this.words[this.length - 1] &= mask;\n  }\n\n  return this.strip();\n};\n\n// Return only lowers bits of number\nBN.prototype.maskn = function maskn(bits) {\n  return this.clone().imaskn(bits);\n};\n\n// Add plain number `num` to `this`\nBN.prototype.iaddn = function iaddn(num) {\n  assert(typeof num === 'number');\n  if (num < 0)\n    return this.isubn(-num);\n\n  // Possible sign change\n  if (this.sign) {\n    if (this.length === 1 && this.words[0] < num) {\n      this.words[0] = num - this.words[0];\n      this.sign = false;\n      return this;\n    }\n\n    this.sign = false;\n    this.isubn(num);\n    this.sign = true;\n    return this;\n  }\n\n  // Add without checks\n  return this._iaddn(num);\n};\n\nBN.prototype._iaddn = function _iaddn(num) {\n  this.words[0] += num;\n\n  // Carry\n  for (var i = 0; i < this.length && this.words[i] >= 0x4000000; i++) {\n    this.words[i] -= 0x4000000;\n    if (i === this.length - 1)\n      this.words[i + 1] = 1;\n    else\n      this.words[i + 1]++;\n  }\n  this.length = Math.max(this.length, i + 1);\n\n  return this;\n};\n\n// Subtract plain number `num` from `this`\nBN.prototype.isubn = function isubn(num) {\n  assert(typeof num === 'number');\n  if (num < 0)\n    return this.iaddn(-num);\n\n  if (this.sign) {\n    this.sign = false;\n    this.iaddn(num);\n    this.sign = true;\n    return this;\n  }\n\n  this.words[0] -= num;\n\n  // Carry\n  for (var i = 0; i < this.length && this.words[i] < 0; i++) {\n    this.words[i] += 0x4000000;\n    this.words[i + 1] -= 1;\n  }\n\n  return this.strip();\n};\n\nBN.prototype.addn = function addn(num) {\n  return this.clone().iaddn(num);\n};\n\nBN.prototype.subn = function subn(num) {\n  return this.clone().isubn(num);\n};\n\nBN.prototype.iabs = function iabs() {\n  this.sign = false;\n\n  return this;\n};\n\nBN.prototype.abs = function abs() {\n  return this.clone().iabs();\n};\n\nBN.prototype._ishlnsubmul = function _ishlnsubmul(num, mul, shift) {\n  // Bigger storage is needed\n  var len = num.length + shift;\n  var i;\n  if (this.words.length < len) {\n    var t = new Array(len);\n    for (var i = 0; i < this.length; i++)\n      t[i] = this.words[i];\n    this.words = t;\n  } else {\n    i = this.length;\n  }\n\n  // Zeroify rest\n  this.length = Math.max(this.length, len);\n  for (; i < this.length; i++)\n    this.words[i] = 0;\n\n  var carry = 0;\n  for (var i = 0; i < num.length; i++) {\n    var w = this.words[i + shift] + carry;\n    var right = num.words[i] * mul;\n    w -= right & 0x3ffffff;\n    carry = (w >> 26) - ((right / 0x4000000) | 0);\n    this.words[i + shift] = w & 0x3ffffff;\n  }\n  for (; i < this.length - shift; i++) {\n    var w = this.words[i + shift] + carry;\n    carry = w >> 26;\n    this.words[i + shift] = w & 0x3ffffff;\n  }\n\n  if (carry === 0)\n    return this.strip();\n\n  // Subtraction overflow\n  assert(carry === -1);\n  carry = 0;\n  for (var i = 0; i < this.length; i++) {\n    var w = -this.words[i] + carry;\n    carry = w >> 26;\n    this.words[i] = w & 0x3ffffff;\n  }\n  this.sign = true;\n\n  return this.strip();\n};\n\nBN.prototype._wordDiv = function _wordDiv(num, mode) {\n  var shift = this.length - num.length;\n\n  var a = this.clone();\n  var b = num;\n\n  // Normalize\n  var bhi = b.words[b.length - 1];\n  var bhiBits = this._countBits(bhi);\n  shift = 26 - bhiBits;\n  if (shift !== 0) {\n    b = b.shln(shift);\n    a.ishln(shift);\n    bhi = b.words[b.length - 1];\n  }\n\n  // Initialize quotient\n  var m = a.length - b.length;\n  var q;\n\n  if (mode !== 'mod') {\n    q = new BN(null);\n    q.length = m + 1;\n    q.words = new Array(q.length);\n    for (var i = 0; i < q.length; i++)\n      q.words[i] = 0;\n  }\n\n  var diff = a.clone()._ishlnsubmul(b, 1, m);\n  if (!diff.sign) {\n    a = diff;\n    if (q)\n      q.words[m] = 1;\n  }\n\n  for (var j = m - 1; j >= 0; j--) {\n    var qj = a.words[b.length + j] * 0x4000000 + a.words[b.length + j - 1];\n\n    // NOTE: (qj / bhi) is (0x3ffffff * 0x4000000 + 0x3ffffff) / 0x2000000 max\n    // (0x7ffffff)\n    qj = Math.min((qj / bhi) | 0, 0x3ffffff);\n\n    a._ishlnsubmul(b, qj, j);\n    while (a.sign) {\n      qj--;\n      a.sign = false;\n      a._ishlnsubmul(b, 1, j);\n      if (a.cmpn(0) !== 0)\n        a.sign = !a.sign;\n    }\n    if (q)\n      q.words[j] = qj;\n  }\n  if (q)\n    q.strip();\n  a.strip();\n\n  // Denormalize\n  if (mode !== 'div' && shift !== 0)\n    a.ishrn(shift);\n  return { div: q ? q : null, mod: a };\n};\n\nBN.prototype.divmod = function divmod(num, mode) {\n  assert(num.cmpn(0) !== 0);\n\n  if (this.sign && !num.sign) {\n    var res = this.neg().divmod(num, mode);\n    var div;\n    var mod;\n    if (mode !== 'mod')\n      div = res.div.neg();\n    if (mode !== 'div')\n      mod = res.mod.cmpn(0) === 0 ? res.mod : num.sub(res.mod);\n    return {\n      div: div,\n      mod: mod\n    };\n  } else if (!this.sign && num.sign) {\n    var res = this.divmod(num.neg(), mode);\n    var div;\n    if (mode !== 'mod')\n      div = res.div.neg();\n    return { div: div, mod: res.mod };\n  } else if (this.sign && num.sign) {\n    return this.neg().divmod(num.neg(), mode);\n  }\n\n  // Both numbers are positive at this point\n\n  // Strip both numbers to approximate shift value\n  if (num.length > this.length || this.cmp(num) < 0)\n    return { div: new BN(0), mod: this };\n\n  // Very short reduction\n  if (num.length === 1) {\n    if (mode === 'div')\n      return { div: this.divn(num.words[0]), mod: null };\n    else if (mode === 'mod')\n      return { div: null, mod: new BN(this.modn(num.words[0])) };\n    return {\n      div: this.divn(num.words[0]),\n      mod: new BN(this.modn(num.words[0]))\n    };\n  }\n\n  return this._wordDiv(num, mode);\n};\n\n// Find `this` / `num`\nBN.prototype.div = function div(num) {\n  return this.divmod(num, 'div').div;\n};\n\n// Find `this` % `num`\nBN.prototype.mod = function mod(num) {\n  return this.divmod(num, 'mod').mod;\n};\n\n// Find Round(`this` / `num`)\nBN.prototype.divRound = function divRound(num) {\n  var dm = this.divmod(num);\n\n  // Fast case - exact division\n  if (dm.mod.cmpn(0) === 0)\n    return dm.div;\n\n  var mod = dm.div.sign ? dm.mod.isub(num) : dm.mod;\n\n  var half = num.shrn(1);\n  var r2 = num.andln(1);\n  var cmp = mod.cmp(half);\n\n  // Round down\n  if (cmp < 0 || r2 === 1 && cmp === 0)\n    return dm.div;\n\n  // Round up\n  return dm.div.sign ? dm.div.isubn(1) : dm.div.iaddn(1);\n};\n\nBN.prototype.modn = function modn(num) {\n  assert(num <= 0x3ffffff);\n  var p = (1 << 26) % num;\n\n  var acc = 0;\n  for (var i = this.length - 1; i >= 0; i--)\n    acc = (p * acc + this.words[i]) % num;\n\n  return acc;\n};\n\n// In-place division by number\nBN.prototype.idivn = function idivn(num) {\n  assert(num <= 0x3ffffff);\n\n  var carry = 0;\n  for (var i = this.length - 1; i >= 0; i--) {\n    var w = this.words[i] + carry * 0x4000000;\n    this.words[i] = (w / num) | 0;\n    carry = w % num;\n  }\n\n  return this.strip();\n};\n\nBN.prototype.divn = function divn(num) {\n  return this.clone().idivn(num);\n};\n\nBN.prototype.egcd = function egcd(p) {\n  assert(!p.sign);\n  assert(p.cmpn(0) !== 0);\n\n  var x = this;\n  var y = p.clone();\n\n  if (x.sign)\n    x = x.mod(p);\n  else\n    x = x.clone();\n\n  // A * x + B * y = x\n  var A = new BN(1);\n  var B = new BN(0);\n\n  // C * x + D * y = y\n  var C = new BN(0);\n  var D = new BN(1);\n\n  var g = 0;\n\n  while (x.isEven() && y.isEven()) {\n    x.ishrn(1);\n    y.ishrn(1);\n    ++g;\n  }\n\n  var yp = y.clone();\n  var xp = x.clone();\n\n  while (x.cmpn(0) !== 0) {\n    while (x.isEven()) {\n      x.ishrn(1);\n      if (A.isEven() && B.isEven()) {\n        A.ishrn(1);\n        B.ishrn(1);\n      } else {\n        A.iadd(yp).ishrn(1);\n        B.isub(xp).ishrn(1);\n      }\n    }\n\n    while (y.isEven()) {\n      y.ishrn(1);\n      if (C.isEven() && D.isEven()) {\n        C.ishrn(1);\n        D.ishrn(1);\n      } else {\n        C.iadd(yp).ishrn(1);\n        D.isub(xp).ishrn(1);\n      }\n    }\n\n    if (x.cmp(y) >= 0) {\n      x.isub(y);\n      A.isub(C);\n      B.isub(D);\n    } else {\n      y.isub(x);\n      C.isub(A);\n      D.isub(B);\n    }\n  }\n\n  return {\n    a: C,\n    b: D,\n    gcd: y.ishln(g)\n  };\n};\n\n// This is reduced incarnation of the binary EEA\n// above, designated to invert members of the\n// _prime_ fields F(p) at a maximal speed\nBN.prototype._invmp = function _invmp(p) {\n  assert(!p.sign);\n  assert(p.cmpn(0) !== 0);\n\n  var a = this;\n  var b = p.clone();\n\n  if (a.sign)\n    a = a.mod(p);\n  else\n    a = a.clone();\n\n  var x1 = new BN(1);\n  var x2 = new BN(0);\n\n  var delta = b.clone();\n\n  while (a.cmpn(1) > 0 && b.cmpn(1) > 0) {\n    while (a.isEven()) {\n      a.ishrn(1);\n      if (x1.isEven())\n        x1.ishrn(1);\n      else\n        x1.iadd(delta).ishrn(1);\n    }\n    while (b.isEven()) {\n      b.ishrn(1);\n      if (x2.isEven())\n        x2.ishrn(1);\n      else\n        x2.iadd(delta).ishrn(1);\n    }\n    if (a.cmp(b) >= 0) {\n      a.isub(b);\n      x1.isub(x2);\n    } else {\n      b.isub(a);\n      x2.isub(x1);\n    }\n  }\n  if (a.cmpn(1) === 0)\n    return x1;\n  else\n    return x2;\n};\n\nBN.prototype.gcd = function gcd(num) {\n  if (this.cmpn(0) === 0)\n    return num.clone();\n  if (num.cmpn(0) === 0)\n    return this.clone();\n\n  var a = this.clone();\n  var b = num.clone();\n  a.sign = false;\n  b.sign = false;\n\n  // Remove common factor of two\n  for (var shift = 0; a.isEven() && b.isEven(); shift++) {\n    a.ishrn(1);\n    b.ishrn(1);\n  }\n\n  do {\n    while (a.isEven())\n      a.ishrn(1);\n    while (b.isEven())\n      b.ishrn(1);\n\n    var r = a.cmp(b);\n    if (r < 0) {\n      // Swap `a` and `b` to make `a` always bigger than `b`\n      var t = a;\n      a = b;\n      b = t;\n    } else if (r === 0 || b.cmpn(1) === 0) {\n      break;\n    }\n\n    a.isub(b);\n  } while (true);\n\n  return b.ishln(shift);\n};\n\n// Invert number in the field F(num)\nBN.prototype.invm = function invm(num) {\n  return this.egcd(num).a.mod(num);\n};\n\nBN.prototype.isEven = function isEven() {\n  return (this.words[0] & 1) === 0;\n};\n\nBN.prototype.isOdd = function isOdd() {\n  return (this.words[0] & 1) === 1;\n};\n\n// And first word and num\nBN.prototype.andln = function andln(num) {\n  return this.words[0] & num;\n};\n\n// Increment at the bit position in-line\nBN.prototype.bincn = function bincn(bit) {\n  assert(typeof bit === 'number');\n  var r = bit % 26;\n  var s = (bit - r) / 26;\n  var q = 1 << r;\n\n  // Fast case: bit is much higher than all existing words\n  if (this.length <= s) {\n    for (var i = this.length; i < s + 1; i++)\n      this.words[i] = 0;\n    this.words[s] |= q;\n    this.length = s + 1;\n    return this;\n  }\n\n  // Add bit and propagate, if needed\n  var carry = q;\n  for (var i = s; carry !== 0 && i < this.length; i++) {\n    var w = this.words[i];\n    w += carry;\n    carry = w >>> 26;\n    w &= 0x3ffffff;\n    this.words[i] = w;\n  }\n  if (carry !== 0) {\n    this.words[i] = carry;\n    this.length++;\n  }\n  return this;\n};\n\nBN.prototype.cmpn = function cmpn(num) {\n  var sign = num < 0;\n  if (sign)\n    num = -num;\n\n  if (this.sign && !sign)\n    return -1;\n  else if (!this.sign && sign)\n    return 1;\n\n  num &= 0x3ffffff;\n  this.strip();\n\n  var res;\n  if (this.length > 1) {\n    res = 1;\n  } else {\n    var w = this.words[0];\n    res = w === num ? 0 : w < num ? -1 : 1;\n  }\n  if (this.sign)\n    res = -res;\n  return res;\n};\n\n// Compare two numbers and return:\n// 1 - if `this` > `num`\n// 0 - if `this` == `num`\n// -1 - if `this` < `num`\nBN.prototype.cmp = function cmp(num) {\n  if (this.sign && !num.sign)\n    return -1;\n  else if (!this.sign && num.sign)\n    return 1;\n\n  var res = this.ucmp(num);\n  if (this.sign)\n    return -res;\n  else\n    return res;\n};\n\n// Unsigned comparison\nBN.prototype.ucmp = function ucmp(num) {\n  // At this point both numbers have the same sign\n  if (this.length > num.length)\n    return 1;\n  else if (this.length < num.length)\n    return -1;\n\n  var res = 0;\n  for (var i = this.length - 1; i >= 0; i--) {\n    var a = this.words[i];\n    var b = num.words[i];\n\n    if (a === b)\n      continue;\n    if (a < b)\n      res = -1;\n    else if (a > b)\n      res = 1;\n    break;\n  }\n  return res;\n};\n\n//\n// A reduce context, could be using montgomery or something better, depending\n// on the `m` itself.\n//\nBN.red = function red(num) {\n  return new Red(num);\n};\n\nBN.prototype.toRed = function toRed(ctx) {\n  assert(!this.red, 'Already a number in reduction context');\n  assert(!this.sign, 'red works only with positives');\n  return ctx.convertTo(this)._forceRed(ctx);\n};\n\nBN.prototype.fromRed = function fromRed() {\n  assert(this.red, 'fromRed works only with numbers in reduction context');\n  return this.red.convertFrom(this);\n};\n\nBN.prototype._forceRed = function _forceRed(ctx) {\n  this.red = ctx;\n  return this;\n};\n\nBN.prototype.forceRed = function forceRed(ctx) {\n  assert(!this.red, 'Already a number in reduction context');\n  return this._forceRed(ctx);\n};\n\nBN.prototype.redAdd = function redAdd(num) {\n  assert(this.red, 'redAdd works only with red numbers');\n  return this.red.add(this, num);\n};\n\nBN.prototype.redIAdd = function redIAdd(num) {\n  assert(this.red, 'redIAdd works only with red numbers');\n  return this.red.iadd(this, num);\n};\n\nBN.prototype.redSub = function redSub(num) {\n  assert(this.red, 'redSub works only with red numbers');\n  return this.red.sub(this, num);\n};\n\nBN.prototype.redISub = function redISub(num) {\n  assert(this.red, 'redISub works only with red numbers');\n  return this.red.isub(this, num);\n};\n\nBN.prototype.redShl = function redShl(num) {\n  assert(this.red, 'redShl works only with red numbers');\n  return this.red.shl(this, num);\n};\n\nBN.prototype.redMul = function redMul(num) {\n  assert(this.red, 'redMul works only with red numbers');\n  this.red._verify2(this, num);\n  return this.red.mul(this, num);\n};\n\nBN.prototype.redIMul = function redIMul(num) {\n  assert(this.red, 'redMul works only with red numbers');\n  this.red._verify2(this, num);\n  return this.red.imul(this, num);\n};\n\nBN.prototype.redSqr = function redSqr() {\n  assert(this.red, 'redSqr works only with red numbers');\n  this.red._verify1(this);\n  return this.red.sqr(this);\n};\n\nBN.prototype.redISqr = function redISqr() {\n  assert(this.red, 'redISqr works only with red numbers');\n  this.red._verify1(this);\n  return this.red.isqr(this);\n};\n\n// Square root over p\nBN.prototype.redSqrt = function redSqrt() {\n  assert(this.red, 'redSqrt works only with red numbers');\n  this.red._verify1(this);\n  return this.red.sqrt(this);\n};\n\nBN.prototype.redInvm = function redInvm() {\n  assert(this.red, 'redInvm works only with red numbers');\n  this.red._verify1(this);\n  return this.red.invm(this);\n};\n\n// Return negative clone of `this` % `red modulo`\nBN.prototype.redNeg = function redNeg() {\n  assert(this.red, 'redNeg works only with red numbers');\n  this.red._verify1(this);\n  return this.red.neg(this);\n};\n\nBN.prototype.redPow = function redPow(num) {\n  assert(this.red && !num.red, 'redPow(normalNum)');\n  this.red._verify1(this);\n  return this.red.pow(this, num);\n};\n\n// Prime numbers with efficient reduction\nvar primes = {\n  k256: null,\n  p224: null,\n  p192: null,\n  p25519: null\n};\n\n// Pseudo-Mersenne prime\nfunction MPrime(name, p) {\n  // P = 2 ^ N - K\n  this.name = name;\n  this.p = new BN(p, 16);\n  this.n = this.p.bitLength();\n  this.k = new BN(1).ishln(this.n).isub(this.p);\n\n  this.tmp = this._tmp();\n}\n\nMPrime.prototype._tmp = function _tmp() {\n  var tmp = new BN(null);\n  tmp.words = new Array(Math.ceil(this.n / 13));\n  return tmp;\n};\n\nMPrime.prototype.ireduce = function ireduce(num) {\n  // Assumes that `num` is less than `P^2`\n  // num = HI * (2 ^ N - K) + HI * K + LO = HI * K + LO (mod P)\n  var r = num;\n  var rlen;\n\n  do {\n    this.split(r, this.tmp);\n    r = this.imulK(r);\n    r = r.iadd(this.tmp);\n    rlen = r.bitLength();\n  } while (rlen > this.n);\n\n  var cmp = rlen < this.n ? -1 : r.ucmp(this.p);\n  if (cmp === 0) {\n    r.words[0] = 0;\n    r.length = 1;\n  } else if (cmp > 0) {\n    r.isub(this.p);\n  } else {\n    r.strip();\n  }\n\n  return r;\n};\n\nMPrime.prototype.split = function split(input, out) {\n  input.ishrn(this.n, 0, out);\n};\n\nMPrime.prototype.imulK = function imulK(num) {\n  return num.imul(this.k);\n};\n\nfunction K256() {\n  MPrime.call(\n    this,\n    'k256',\n    'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff fffffffe fffffc2f');\n}\ninherits(K256, MPrime);\n\nK256.prototype.split = function split(input, output) {\n  // 256 = 9 * 26 + 22\n  var mask = 0x3fffff;\n\n  var outLen = Math.min(input.length, 9);\n  for (var i = 0; i < outLen; i++)\n    output.words[i] = input.words[i];\n  output.length = outLen;\n\n  if (input.length <= 9) {\n    input.words[0] = 0;\n    input.length = 1;\n    return;\n  }\n\n  // Shift by 9 limbs\n  var prev = input.words[9];\n  output.words[output.length++] = prev & mask;\n\n  for (var i = 10; i < input.length; i++) {\n    var next = input.words[i];\n    input.words[i - 10] = ((next & mask) << 4) | (prev >>> 22);\n    prev = next;\n  }\n  input.words[i - 10] = prev >>> 22;\n  input.length -= 9;\n};\n\nK256.prototype.imulK = function imulK(num) {\n  // K = 0x1000003d1 = [ 0x40, 0x3d1 ]\n  num.words[num.length] = 0;\n  num.words[num.length + 1] = 0;\n  num.length += 2;\n\n  // bounded at: 0x40 * 0x3ffffff + 0x3d0 = 0x100000390\n  var hi;\n  var lo = 0;\n  for (var i = 0; i < num.length; i++) {\n    var w = num.words[i];\n    hi = w * 0x40;\n    lo += w * 0x3d1;\n    hi += (lo / 0x4000000) | 0;\n    lo &= 0x3ffffff;\n\n    num.words[i] = lo;\n\n    lo = hi;\n  }\n\n  // Fast length reduction\n  if (num.words[num.length - 1] === 0) {\n    num.length--;\n    if (num.words[num.length - 1] === 0)\n      num.length--;\n  }\n  return num;\n};\n\nfunction P224() {\n  MPrime.call(\n    this,\n    'p224',\n    'ffffffff ffffffff ffffffff ffffffff 00000000 00000000 00000001');\n}\ninherits(P224, MPrime);\n\nfunction P192() {\n  MPrime.call(\n    this,\n    'p192',\n    'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff');\n}\ninherits(P192, MPrime);\n\nfunction P25519() {\n  // 2 ^ 255 - 19\n  MPrime.call(\n    this,\n    '25519',\n    '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed');\n}\ninherits(P25519, MPrime);\n\nP25519.prototype.imulK = function imulK(num) {\n  // K = 0x13\n  var carry = 0;\n  for (var i = 0; i < num.length; i++) {\n    var hi = num.words[i] * 0x13 + carry;\n    var lo = hi & 0x3ffffff;\n    hi >>>= 26;\n\n    num.words[i] = lo;\n    carry = hi;\n  }\n  if (carry !== 0)\n    num.words[num.length++] = carry;\n  return num;\n};\n\n// Exported mostly for testing purposes, use plain name instead\nBN._prime = function prime(name) {\n  // Cached version of prime\n  if (primes[name])\n    return primes[name];\n\n  var prime;\n  if (name === 'k256')\n    prime = new K256();\n  else if (name === 'p224')\n    prime = new P224();\n  else if (name === 'p192')\n    prime = new P192();\n  else if (name === 'p25519')\n    prime = new P25519();\n  else\n    throw new Error('Unknown prime ' + name);\n  primes[name] = prime;\n\n  return prime;\n};\n\n//\n// Base reduction engine\n//\nfunction Red(m) {\n  if (typeof m === 'string') {\n    var prime = BN._prime(m);\n    this.m = prime.p;\n    this.prime = prime;\n  } else {\n    this.m = m;\n    this.prime = null;\n  }\n}\n\nRed.prototype._verify1 = function _verify1(a) {\n  assert(!a.sign, 'red works only with positives');\n  assert(a.red, 'red works only with red numbers');\n};\n\nRed.prototype._verify2 = function _verify2(a, b) {\n  assert(!a.sign && !b.sign, 'red works only with positives');\n  assert(a.red && a.red === b.red,\n         'red works only with red numbers');\n};\n\nRed.prototype.imod = function imod(a) {\n  if (this.prime)\n    return this.prime.ireduce(a)._forceRed(this);\n  return a.mod(this.m)._forceRed(this);\n};\n\nRed.prototype.neg = function neg(a) {\n  var r = a.clone();\n  r.sign = !r.sign;\n  return r.iadd(this.m)._forceRed(this);\n};\n\nRed.prototype.add = function add(a, b) {\n  this._verify2(a, b);\n\n  var res = a.add(b);\n  if (res.cmp(this.m) >= 0)\n    res.isub(this.m);\n  return res._forceRed(this);\n};\n\nRed.prototype.iadd = function iadd(a, b) {\n  this._verify2(a, b);\n\n  var res = a.iadd(b);\n  if (res.cmp(this.m) >= 0)\n    res.isub(this.m);\n  return res;\n};\n\nRed.prototype.sub = function sub(a, b) {\n  this._verify2(a, b);\n\n  var res = a.sub(b);\n  if (res.cmpn(0) < 0)\n    res.iadd(this.m);\n  return res._forceRed(this);\n};\n\nRed.prototype.isub = function isub(a, b) {\n  this._verify2(a, b);\n\n  var res = a.isub(b);\n  if (res.cmpn(0) < 0)\n    res.iadd(this.m);\n  return res;\n};\n\nRed.prototype.shl = function shl(a, num) {\n  this._verify1(a);\n  return this.imod(a.shln(num));\n};\n\nRed.prototype.imul = function imul(a, b) {\n  this._verify2(a, b);\n  return this.imod(a.imul(b));\n};\n\nRed.prototype.mul = function mul(a, b) {\n  this._verify2(a, b);\n  return this.imod(a.mul(b));\n};\n\nRed.prototype.isqr = function isqr(a) {\n  return this.imul(a, a);\n};\n\nRed.prototype.sqr = function sqr(a) {\n  return this.mul(a, a);\n};\n\nRed.prototype.sqrt = function sqrt(a) {\n  if (a.cmpn(0) === 0)\n    return a.clone();\n\n  var mod3 = this.m.andln(3);\n  assert(mod3 % 2 === 1);\n\n  // Fast case\n  if (mod3 === 3) {\n    var pow = this.m.add(new BN(1)).ishrn(2);\n    var r = this.pow(a, pow);\n    return r;\n  }\n\n  // Tonelli-Shanks algorithm (Totally unoptimized and slow)\n  //\n  // Find Q and S, that Q * 2 ^ S = (P - 1)\n  var q = this.m.subn(1);\n  var s = 0;\n  while (q.cmpn(0) !== 0 && q.andln(1) === 0) {\n    s++;\n    q.ishrn(1);\n  }\n  assert(q.cmpn(0) !== 0);\n\n  var one = new BN(1).toRed(this);\n  var nOne = one.redNeg();\n\n  // Find quadratic non-residue\n  // NOTE: Max is such because of generalized Riemann hypothesis.\n  var lpow = this.m.subn(1).ishrn(1);\n  var z = this.m.bitLength();\n  z = new BN(2 * z * z).toRed(this);\n  while (this.pow(z, lpow).cmp(nOne) !== 0)\n    z.redIAdd(nOne);\n\n  var c = this.pow(z, q);\n  var r = this.pow(a, q.addn(1).ishrn(1));\n  var t = this.pow(a, q);\n  var m = s;\n  while (t.cmp(one) !== 0) {\n    var tmp = t;\n    for (var i = 0; tmp.cmp(one) !== 0; i++)\n      tmp = tmp.redSqr();\n    assert(i < m);\n    var b = this.pow(c, new BN(1).ishln(m - i - 1));\n\n    r = r.redMul(b);\n    c = b.redSqr();\n    t = t.redMul(c);\n    m = i;\n  }\n\n  return r;\n};\n\nRed.prototype.invm = function invm(a) {\n  var inv = a._invmp(this.m);\n  if (inv.sign) {\n    inv.sign = false;\n    return this.imod(inv).redNeg();\n  } else {\n    return this.imod(inv);\n  }\n};\n\nRed.prototype.pow = function pow(a, num) {\n  var w = [];\n\n  if (num.cmpn(0) === 0)\n    return new BN(1);\n\n  var q = num.clone();\n\n  while (q.cmpn(0) !== 0) {\n    w.push(q.andln(1));\n    q.ishrn(1);\n  }\n\n  // Skip leading zeroes\n  var res = a;\n  for (var i = 0; i < w.length; i++, res = this.sqr(res))\n    if (w[i] !== 0)\n      break;\n\n  if (++i < w.length) {\n    for (var q = this.sqr(res); i < w.length; i++, q = this.sqr(q)) {\n      if (w[i] === 0)\n        continue;\n      res = this.mul(res, q);\n    }\n  }\n\n  return res;\n};\n\nRed.prototype.convertTo = function convertTo(num) {\n  var r = num.mod(this.m);\n  if (r === num)\n    return r.clone();\n  else\n    return r;\n};\n\nRed.prototype.convertFrom = function convertFrom(num) {\n  var res = num.clone();\n  res.red = null;\n  return res;\n};\n\n//\n// Montgomery method engine\n//\n\nBN.mont = function mont(num) {\n  return new Mont(num);\n};\n\nfunction Mont(m) {\n  Red.call(this, m);\n\n  this.shift = this.m.bitLength();\n  if (this.shift % 26 !== 0)\n    this.shift += 26 - (this.shift % 26);\n  this.r = new BN(1).ishln(this.shift);\n  this.r2 = this.imod(this.r.sqr());\n  this.rinv = this.r._invmp(this.m);\n\n  this.minv = this.rinv.mul(this.r).isubn(1).div(this.m);\n  this.minv.sign = true;\n  this.minv = this.minv.mod(this.r);\n}\ninherits(Mont, Red);\n\nMont.prototype.convertTo = function convertTo(num) {\n  return this.imod(num.shln(this.shift));\n};\n\nMont.prototype.convertFrom = function convertFrom(num) {\n  var r = this.imod(num.mul(this.rinv));\n  r.red = null;\n  return r;\n};\n\nMont.prototype.imul = function imul(a, b) {\n  if (a.cmpn(0) === 0 || b.cmpn(0) === 0) {\n    a.words[0] = 0;\n    a.length = 1;\n    return a;\n  }\n\n  var t = a.imul(b);\n  var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m);\n  var u = t.isub(c).ishrn(this.shift);\n  var res = u;\n  if (u.cmp(this.m) >= 0)\n    res = u.isub(this.m);\n  else if (u.cmpn(0) < 0)\n    res = u.iadd(this.m);\n\n  return res._forceRed(this);\n};\n\nMont.prototype.mul = function mul(a, b) {\n  if (a.cmpn(0) === 0 || b.cmpn(0) === 0)\n    return new BN(0)._forceRed(this);\n\n  var t = a.mul(b);\n  var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m);\n  var u = t.isub(c).ishrn(this.shift);\n  var res = u;\n  if (u.cmp(this.m) >= 0)\n    res = u.isub(this.m);\n  else if (u.cmpn(0) < 0)\n    res = u.iadd(this.m);\n\n  return res._forceRed(this);\n};\n\nMont.prototype.invm = function invm(a) {\n  // (AR)^-1 * R^2 = (A^-1 * R^-1) * R^2 = A^-1 * R\n  var res = this.imod(a._invmp(this.m).mul(this.r2));\n  return res._forceRed(this);\n};\n\n})(typeof module === 'undefined' || module, this);\n","var constants = require('../constants');\n\nexports.tagClass = {\n  0: 'universal',\n  1: 'application',\n  2: 'context',\n  3: 'private'\n};\nexports.tagClassByName = constants._reverse(exports.tagClass);\n\nexports.tag = {\n  0x00: 'end',\n  0x01: 'bool',\n  0x02: 'int',\n  0x03: 'bitstr',\n  0x04: 'octstr',\n  0x05: 'null_',\n  0x06: 'objid',\n  0x07: 'objDesc',\n  0x08: 'external',\n  0x09: 'real',\n  0x0a: 'enum',\n  0x0b: 'embed',\n  0x0c: 'utf8str',\n  0x0d: 'relativeOid',\n  0x10: 'seq',\n  0x11: 'set',\n  0x12: 'numstr',\n  0x13: 'printstr',\n  0x14: 't61str',\n  0x15: 'videostr',\n  0x16: 'ia5str',\n  0x17: 'utctime',\n  0x18: 'gentime',\n  0x19: 'graphstr',\n  0x1a: 'iso646str',\n  0x1b: 'genstr',\n  0x1c: 'unistr',\n  0x1d: 'charstr',\n  0x1e: 'bmpstr'\n};\nexports.tagByName = constants._reverse(exports.tag);\n","var constants = exports;\n\n// Helper\nconstants._reverse = function reverse(map) {\n  var res = {};\n\n  Object.keys(map).forEach(function(key) {\n    // Convert key to integer if it is stringified\n    if ((key | 0) == key)\n      key = key | 0;\n\n    var value = map[key];\n    res[value] = key;\n  });\n\n  return res;\n};\n\nconstants.der = require('./der');\n","var inherits = require('util').inherits;\n\nvar asn1 = require('../asn1');\nvar base = asn1.base;\nvar bignum = asn1.bignum;\n\n// Import DER constants\nvar der = asn1.constants.der;\n\nfunction DERDecoder(entity) {\n  this.enc = 'der';\n  this.name = entity.name;\n  this.entity = entity;\n\n  // Construct base tree\n  this.tree = new DERNode();\n  this.tree._init(entity.body);\n};\nmodule.exports = DERDecoder;\n\nDERDecoder.prototype.decode = function decode(data, options) {\n  if (!(data instanceof base.DecoderBuffer))\n    data = new base.DecoderBuffer(data, options);\n\n  return this.tree._decode(data, options);\n};\n\n// Tree methods\n\nfunction DERNode(parent) {\n  base.Node.call(this, 'der', parent);\n}\ninherits(DERNode, base.Node);\n\nDERNode.prototype._peekTag = function peekTag(buffer, tag, any) {\n  if (buffer.isEmpty())\n    return false;\n\n  var state = buffer.save();\n  var decodedTag = derDecodeTag(buffer, 'Failed to peek tag: \"' + tag + '\"');\n  if (buffer.isError(decodedTag))\n    return decodedTag;\n\n  buffer.restore(state);\n\n  return decodedTag.tag === tag || decodedTag.tagStr === tag || any;\n};\n\nDERNode.prototype._decodeTag = function decodeTag(buffer, tag, any) {\n  var decodedTag = derDecodeTag(buffer,\n                                'Failed to decode tag of \"' + tag + '\"');\n  if (buffer.isError(decodedTag))\n    return decodedTag;\n\n  var len = derDecodeLen(buffer,\n                         decodedTag.primitive,\n                         'Failed to get length of \"' + tag + '\"');\n\n  // Failure\n  if (buffer.isError(len))\n    return len;\n\n  if (!any &&\n      decodedTag.tag !== tag &&\n      decodedTag.tagStr !== tag &&\n      decodedTag.tagStr + 'of' !== tag) {\n    return buffer.error('Failed to match tag: \"' + tag + '\"');\n  }\n\n  if (decodedTag.primitive || len !== null)\n    return buffer.skip(len, 'Failed to match body of: \"' + tag + '\"');\n\n  // Indefinite length... find END tag\n  var state = buffer.save();\n  var res = this._skipUntilEnd(\n      buffer,\n      'Failed to skip indefinite length body: \"' + this.tag + '\"');\n  if (buffer.isError(res))\n    return res;\n\n  len = buffer.offset - state.offset;\n  buffer.restore(state);\n  return buffer.skip(len, 'Failed to match body of: \"' + tag + '\"');\n};\n\nDERNode.prototype._skipUntilEnd = function skipUntilEnd(buffer, fail) {\n  while (true) {\n    var tag = derDecodeTag(buffer, fail);\n    if (buffer.isError(tag))\n      return tag;\n    var len = derDecodeLen(buffer, tag.primitive, fail);\n    if (buffer.isError(len))\n      return len;\n\n    var res;\n    if (tag.primitive || len !== null)\n      res = buffer.skip(len)\n    else\n      res = this._skipUntilEnd(buffer, fail);\n\n    // Failure\n    if (buffer.isError(res))\n      return res;\n\n    if (tag.tagStr === 'end')\n      break;\n  }\n};\n\nDERNode.prototype._decodeList = function decodeList(buffer, tag, decoder) {\n  var result = [];\n  while (!buffer.isEmpty()) {\n    var possibleEnd = this._peekTag(buffer, 'end');\n    if (buffer.isError(possibleEnd))\n      return possibleEnd;\n\n    var res = decoder.decode(buffer, 'der');\n    if (buffer.isError(res) && possibleEnd)\n      break;\n    result.push(res);\n  }\n  return result;\n};\n\nDERNode.prototype._decodeStr = function decodeStr(buffer, tag) {\n  if (tag === 'octstr') {\n    return buffer.raw();\n  } else if (tag === 'bitstr') {\n    var unused = buffer.readUInt8();\n    if (buffer.isError(unused))\n      return unused;\n\n    return { unused: unused, data: buffer.raw() };\n  } else if (tag === 'ia5str' || tag === 'utf8str') {\n    return buffer.raw().toString();\n  } else {\n    return this.error('Decoding of string type: ' + tag + ' unsupported');\n  }\n};\n\nDERNode.prototype._decodeObjid = function decodeObjid(buffer, values, relative) {\n  var identifiers = [];\n  var ident = 0;\n  while (!buffer.isEmpty()) {\n    var subident = buffer.readUInt8();\n    ident <<= 7;\n    ident |= subident & 0x7f;\n    if ((subident & 0x80) === 0) {\n      identifiers.push(ident);\n      ident = 0;\n    }\n  }\n  if (subident & 0x80)\n    identifiers.push(ident);\n\n  var first = (identifiers[0] / 40) | 0;\n  var second = identifiers[0] % 40;\n\n  if (relative)\n    result = identifiers;\n  else\n    result = [first, second].concat(identifiers.slice(1));\n\n  if (values)\n    result = values[result.join(' ')];\n\n  return result;\n};\n\nDERNode.prototype._decodeTime = function decodeTime(buffer, tag) {\n  var str = buffer.raw().toString();\n  if (tag === 'gentime') {\n    var year = str.slice(0, 4) | 0;\n    var mon = str.slice(4, 6) | 0;\n    var day = str.slice(6, 8) | 0;\n    var hour = str.slice(8, 10) | 0;\n    var min = str.slice(10, 12) | 0;\n    var sec = str.slice(12, 14) | 0;\n  } else if (tag === 'utctime') {\n    var year = str.slice(0, 2) | 0;\n    var mon = str.slice(2, 4) | 0;\n    var day = str.slice(4, 6) | 0;\n    var hour = str.slice(6, 8) | 0;\n    var min = str.slice(8, 10) | 0;\n    var sec = str.slice(10, 12) | 0;\n    if (year < 70)\n      year = 2000 + year;\n    else\n      year = 1900 + year;\n  } else {\n    return this.error('Decoding ' + tag + ' time is not supported yet');\n  }\n\n  return Date.UTC(year, mon - 1, day, hour, min, sec, 0);\n};\n\nDERNode.prototype._decodeNull = function decodeNull(buffer) {\n  return null;\n};\n\nDERNode.prototype._decodeBool = function decodeBool(buffer) {\n  var res = buffer.readUInt8();\n  if (buffer.isError(res))\n    return res;\n  else\n    return res !== 0;\n};\n\nDERNode.prototype._decodeInt = function decodeInt(buffer, values) {\n  // Bigint, return as it is (assume big endian)\n  var raw = buffer.raw();\n  var res = new bignum(raw);\n\n  if (values)\n    res = values[res.toString(10)] || res;\n\n  return res;\n};\n\nDERNode.prototype._use = function use(entity, obj) {\n  if (typeof entity === 'function')\n    entity = entity(obj);\n  return entity._getDecoder('der').tree;\n};\n\n// Utility methods\n\nfunction derDecodeTag(buf, fail) {\n  var tag = buf.readUInt8(fail);\n  if (buf.isError(tag))\n    return tag;\n\n  var cls = der.tagClass[tag >> 6];\n  var primitive = (tag & 0x20) === 0;\n\n  // Multi-octet tag - load\n  if ((tag & 0x1f) === 0x1f) {\n    var oct = tag;\n    tag = 0;\n    while ((oct & 0x80) === 0x80) {\n      oct = buf.readUInt8(fail);\n      if (buf.isError(oct))\n        return oct;\n\n      tag <<= 7;\n      tag |= oct & 0x7f;\n    }\n  } else {\n    tag &= 0x1f;\n  }\n  var tagStr = der.tag[tag];\n\n  return {\n    cls: cls,\n    primitive: primitive,\n    tag: tag,\n    tagStr: tagStr\n  };\n}\n\nfunction derDecodeLen(buf, primitive, fail) {\n  var len = buf.readUInt8(fail);\n  if (buf.isError(len))\n    return len;\n\n  // Indefinite form\n  if (!primitive && len === 0x80)\n    return null;\n\n  // Definite form\n  if ((len & 0x80) === 0) {\n    // Short form\n    return len;\n  }\n\n  // Long form\n  var num = len & 0x7f;\n  if (num >= 4)\n    return buf.error('length octect is too long');\n\n  len = 0;\n  for (var i = 0; i < num; i++) {\n    len <<= 8;\n    var j = buf.readUInt8(fail);\n    if (buf.isError(j))\n      return j;\n    len |= j;\n  }\n\n  return len;\n}\n","var decoders = exports;\n\ndecoders.der = require('./der');\ndecoders.pem = require('./pem');\n","var inherits = require('util').inherits;\nvar Buffer = require('buffer').Buffer;\n\nvar asn1 = require('../asn1');\nvar DERDecoder = require('./der');\n\nfunction PEMDecoder(entity) {\n  DERDecoder.call(this, entity);\n  this.enc = 'pem';\n};\ninherits(PEMDecoder, DERDecoder);\nmodule.exports = PEMDecoder;\n\nPEMDecoder.prototype.decode = function decode(data, options) {\n  var lines = data.toString().split(/[\\r\\n]+/g);\n\n  var label = options.label.toUpperCase();\n\n  var re = /^-----(BEGIN|END) ([^-]+)-----$/;\n  var start = -1;\n  var end = -1;\n  for (var i = 0; i < lines.length; i++) {\n    var match = lines[i].match(re);\n    if (match === null)\n      continue;\n\n    if (match[2] !== label)\n      continue;\n\n    if (start === -1) {\n      if (match[1] !== 'BEGIN')\n        break;\n      start = i;\n    } else {\n      if (match[1] !== 'END')\n        break;\n      end = i;\n      break;\n    }\n  }\n  if (start === -1 || end === -1)\n    throw new Error('PEM section not found for: ' + label);\n\n  var base64 = lines.slice(start + 1, end).join('');\n  // Remove excessive symbols\n  base64.replace(/[^a-z0-9\\+\\/=]+/gi, '');\n\n  var input = new Buffer(base64, 'base64');\n  return DERDecoder.prototype.decode.call(this, input, options);\n};\n","var inherits = require('util').inherits;\nvar Buffer = require('buffer').Buffer;\n\nvar asn1 = require('../asn1');\nvar base = asn1.base;\nvar bignum = asn1.bignum;\n\n// Import DER constants\nvar der = asn1.constants.der;\n\nfunction DEREncoder(entity) {\n  this.enc = 'der';\n  this.name = entity.name;\n  this.entity = entity;\n\n  // Construct base tree\n  this.tree = new DERNode();\n  this.tree._init(entity.body);\n};\nmodule.exports = DEREncoder;\n\nDEREncoder.prototype.encode = function encode(data, reporter) {\n  return this.tree._encode(data, reporter).join();\n};\n\n// Tree methods\n\nfunction DERNode(parent) {\n  base.Node.call(this, 'der', parent);\n}\ninherits(DERNode, base.Node);\n\nDERNode.prototype._encodeComposite = function encodeComposite(tag,\n                                                              primitive,\n                                                              cls,\n                                                              content) {\n  var encodedTag = encodeTag(tag, primitive, cls, this.reporter);\n\n  // Short form\n  if (content.length < 0x80) {\n    var header = new Buffer(2);\n    header[0] = encodedTag;\n    header[1] = content.length;\n    return this._createEncoderBuffer([ header, content ]);\n  }\n\n  // Long form\n  // Count octets required to store length\n  var lenOctets = 1;\n  for (var i = content.length; i >= 0x100; i >>= 8)\n    lenOctets++;\n\n  var header = new Buffer(1 + 1 + lenOctets);\n  header[0] = encodedTag;\n  header[1] = 0x80 | lenOctets;\n\n  for (var i = 1 + lenOctets, j = content.length; j > 0; i--, j >>= 8)\n    header[i] = j & 0xff;\n\n  return this._createEncoderBuffer([ header, content ]);\n};\n\nDERNode.prototype._encodeStr = function encodeStr(str, tag) {\n  if (tag === 'octstr')\n    return this._createEncoderBuffer(str);\n  else if (tag === 'bitstr')\n    return this._createEncoderBuffer([ str.unused | 0, str.data ]);\n  else if (tag === 'ia5str' || tag === 'utf8str')\n    return this._createEncoderBuffer(str);\n  return this.reporter.error('Encoding of string type: ' + tag +\n                             ' unsupported');\n};\n\nDERNode.prototype._encodeObjid = function encodeObjid(id, values, relative) {\n  if (typeof id === 'string') {\n    if (!values)\n      return this.reporter.error('string objid given, but no values map found');\n    if (!values.hasOwnProperty(id))\n      return this.reporter.error('objid not found in values map');\n    id = values[id].split(/[\\s\\.]+/g);\n    for (var i = 0; i < id.length; i++)\n      id[i] |= 0;\n  } else if (Array.isArray(id)) {\n    id = id.slice();\n    for (var i = 0; i < id.length; i++)\n      id[i] |= 0;\n  }\n\n  if (!Array.isArray(id)) {\n    return this.reporter.error('objid() should be either array or string, ' +\n                               'got: ' + JSON.stringify(id));\n  }\n\n  if (!relative) {\n    if (id[1] >= 40)\n      return this.reporter.error('Second objid identifier OOB');\n    id.splice(0, 2, id[0] * 40 + id[1]);\n  }\n\n  // Count number of octets\n  var size = 0;\n  for (var i = 0; i < id.length; i++) {\n    var ident = id[i];\n    for (size++; ident >= 0x80; ident >>= 7)\n      size++;\n  }\n\n  var objid = new Buffer(size);\n  var offset = objid.length - 1;\n  for (var i = id.length - 1; i >= 0; i--) {\n    var ident = id[i];\n    objid[offset--] = ident & 0x7f;\n    while ((ident >>= 7) > 0)\n      objid[offset--] = 0x80 | (ident & 0x7f);\n  }\n\n  return this._createEncoderBuffer(objid);\n};\n\nfunction two(num) {\n  if (num < 10)\n    return '0' + num;\n  else\n    return num;\n}\n\nDERNode.prototype._encodeTime = function encodeTime(time, tag) {\n  var str;\n  var date = new Date(time);\n\n  if (tag === 'gentime') {\n    str = [\n      two(date.getFullYear()),\n      two(date.getUTCMonth() + 1),\n      two(date.getUTCDate()),\n      two(date.getUTCHours()),\n      two(date.getUTCMinutes()),\n      two(date.getUTCSeconds()),\n      'Z'\n    ].join('');\n  } else if (tag === 'utctime') {\n    str = [\n      two(date.getFullYear() % 100),\n      two(date.getUTCMonth() + 1),\n      two(date.getUTCDate()),\n      two(date.getUTCHours()),\n      two(date.getUTCMinutes()),\n      two(date.getUTCSeconds()),\n      'Z'\n    ].join('');\n  } else {\n    this.reporter.error('Encoding ' + tag + ' time is not supported yet');\n  }\n\n  return this._encodeStr(str, 'octstr');\n};\n\nDERNode.prototype._encodeNull = function encodeNull() {\n  return this._createEncoderBuffer('');\n};\n\nDERNode.prototype._encodeInt = function encodeInt(num, values) {\n  if (typeof num === 'string') {\n    if (!values)\n      return this.reporter.error('String int or enum given, but no values map');\n    if (!values.hasOwnProperty(num)) {\n      return this.reporter.error('Values map doesn\\'t contain: ' +\n                                 JSON.stringify(num));\n    }\n    num = values[num];\n  }\n\n  // Bignum, assume big endian\n  if (typeof num !== 'number' && !Buffer.isBuffer(num)) {\n    var numArray = num.toArray();\n    if (num.sign === false && numArray[0] & 0x80) {\n      numArray.unshift(0);\n    }\n    num = new Buffer(numArray);\n  }\n\n  if (Buffer.isBuffer(num)) {\n    var size = num.length;\n    if (num.length === 0)\n      size++;\n\n    var out = new Buffer(size);\n    num.copy(out);\n    if (num.length === 0)\n      out[0] = 0\n    return this._createEncoderBuffer(out);\n  }\n\n  if (num < 0x80)\n    return this._createEncoderBuffer(num);\n\n  if (num < 0x100)\n    return this._createEncoderBuffer([0, num]);\n\n  var size = 1;\n  for (var i = num; i >= 0x100; i >>= 8)\n    size++;\n\n  var out = new Array(size);\n  for (var i = out.length - 1; i >= 0; i--) {\n    out[i] = num & 0xff;\n    num >>= 8;\n  }\n  if(out[0] & 0x80) {\n    out.unshift(0);\n  }\n\n  return this._createEncoderBuffer(new Buffer(out));\n};\n\nDERNode.prototype._encodeBool = function encodeBool(value) {\n  return this._createEncoderBuffer(value ? 0xff : 0);\n};\n\nDERNode.prototype._use = function use(entity, obj) {\n  if (typeof entity === 'function')\n    entity = entity(obj);\n  return entity._getEncoder('der').tree;\n};\n\nDERNode.prototype._skipDefault = function skipDefault(dataBuffer, reporter, parent) {\n  var state = this._baseState;\n  var i;\n  if (state['default'] === null)\n    return false;\n\n  var data = dataBuffer.join();\n  if (state.defaultBuffer === undefined)\n    state.defaultBuffer = this._encodeValue(state['default'], reporter, parent).join();\n\n  if (data.length !== state.defaultBuffer.length)\n    return false;\n\n  for (i=0; i < data.length; i++)\n    if (data[i] !== state.defaultBuffer[i])\n      return false;\n\n  return true;\n};\n\n// Utility methods\n\nfunction encodeTag(tag, primitive, cls, reporter) {\n  var res;\n\n  if (tag === 'seqof')\n    tag = 'seq';\n  else if (tag === 'setof')\n    tag = 'set';\n\n  if (der.tagByName.hasOwnProperty(tag))\n    res = der.tagByName[tag];\n  else if (typeof tag === 'number' && (tag | 0) === tag)\n    res = tag;\n  else\n    return reporter.error('Unknown tag: ' + tag);\n\n  if (res >= 0x1f)\n    return reporter.error('Multi-octet tag encoding unsupported');\n\n  if (!primitive)\n    res |= 0x20;\n\n  res |= (der.tagClassByName[cls || 'universal'] << 6);\n\n  return res;\n}\n","var encoders = exports;\n\nencoders.der = require('./der');\nencoders.pem = require('./pem');\n","var inherits = require('util').inherits;\nvar Buffer = require('buffer').Buffer;\n\nvar asn1 = require('../asn1');\nvar DEREncoder = require('./der');\n\nfunction PEMEncoder(entity) {\n  DEREncoder.call(this, entity);\n  this.enc = 'pem';\n};\ninherits(PEMEncoder, DEREncoder);\nmodule.exports = PEMEncoder;\n\nPEMEncoder.prototype.encode = function encode(data, options) {\n  var buf = DEREncoder.prototype.encode.call(this, data);\n\n  var p = buf.toString('base64');\n  var out = [ '-----BEGIN ' + options.label + '-----' ];\n  for (var i = 0; i < p.length; i += 64)\n    out.push(p.slice(i, i + 64));\n  out.push('-----END ' + options.label + '-----');\n  return out.join('\\n');\n};\n","'use strict'\n\nvar asn1 = require('./asn1/asn1');\nvar BN = require('./asn1/bignum/bn');\n\nvar ECPrivateKeyASN = asn1.define('ECPrivateKey', function() {\n    this.seq().obj(\n        this.key('version').int(),\n        this.key('privateKey').octstr(),\n        this.key('parameters').explicit(0).objid().optional(),\n        this.key('publicKey').explicit(1).bitstr().optional()\n    )\n})\n\nvar SubjectPublicKeyInfoASN = asn1.define('SubjectPublicKeyInfo', function() {\n    this.seq().obj(\n        this.key('algorithm').seq().obj(\n            this.key(\"id\").objid(),\n            this.key(\"curve\").objid()\n        ),\n        this.key('pub').bitstr()\n    )\n})\n\nvar curves = {\n    secp256k1: {\n        curveParameters: [1, 3, 132, 0, 10],\n        privatePEMOptions: {label: 'EC PRIVATE KEY'},\n        publicPEMOptions: {label: 'PUBLIC KEY'}\n    }\n}\n\nfunction assert(val, msg) {\n    if (!val) {\n        throw new Error(msg || 'Assertion failed')\n    }\n}\n\nfunction KeyEncoder(options) {\n    if (typeof options === 'string') {\n        assert(curves.hasOwnProperty(options), 'Unknown curve ' + options);\n        options = curves[options]\n    }\n    this.options = options;\n    this.algorithmID = [1, 2, 840, 10045, 2, 1]\n}\n\nKeyEncoder.ECPrivateKeyASN = ECPrivateKeyASN;\nKeyEncoder.SubjectPublicKeyInfoASN = SubjectPublicKeyInfoASN;\n\nKeyEncoder.prototype.privateKeyObject = function(rawPrivateKey, rawPublicKey) {\n    var privateKeyObject = {\n        version: new BN(1),\n        privateKey: new Buffer(rawPrivateKey, 'hex'),\n        parameters: this.options.curveParameters,\n        pemOptions: {label:\"EC PRIVATE KEY\"}\n    };\n\n    if (rawPublicKey) {\n        privateKeyObject.publicKey = {\n            unused: 0,\n            data: new Buffer(rawPublicKey, 'hex')\n        }\n    }\n\n    return privateKeyObject\n};\n\nKeyEncoder.prototype.publicKeyObject = function(rawPublicKey) {\n    return {\n        algorithm: {\n            id: this.algorithmID,\n            curve: this.options.curveParameters\n        },\n        pub: {\n            unused: 0,\n            data: new Buffer(rawPublicKey, 'hex')\n        },\n        pemOptions: { label :\"PUBLIC KEY\"}\n    }\n}\n\nKeyEncoder.prototype.encodePrivate = function(privateKey, originalFormat, destinationFormat) {\n    var privateKeyObject\n\n    /* Parse the incoming private key and convert it to a private key object */\n    if (originalFormat === 'raw') {\n        if (!typeof privateKey === 'string') {\n            throw 'private key must be a string'\n        }\n        var privateKeyObject = this.options.curve.keyFromPrivate(privateKey, 'hex'),\n            rawPublicKey = privateKeyObject.getPublic('hex')\n        privateKeyObject = this.privateKeyObject(privateKey, rawPublicKey)\n    } else if (originalFormat === 'der') {\n        if (typeof privateKey === 'buffer') {\n            // do nothing\n        } else if (typeof privateKey === 'string') {\n            privateKey = new Buffer(privateKey, 'hex')\n        } else {\n            throw 'private key must be a buffer or a string'\n        }\n        privateKeyObject = ECPrivateKeyASN.decode(privateKey, 'der')\n    } else if (originalFormat === 'pem') {\n        if (!typeof privateKey === 'string') {\n            throw 'private key must be a string'\n        }\n        privateKeyObject = ECPrivateKeyASN.decode(privateKey, 'pem', this.options.privatePEMOptions)\n    } else {\n        throw 'invalid private key format'\n    }\n\n    /* Export the private key object to the desired format */\n    if (destinationFormat === 'raw') {\n        return privateKeyObject.privateKey.toString('hex')\n    } else if (destinationFormat === 'der') {\n        return ECPrivateKeyASN.encode(privateKeyObject, 'der').toString('hex')\n    } else if (destinationFormat === 'pem') {\n        return ECPrivateKeyASN.encode(privateKeyObject, 'pem', this.options.privatePEMOptions)\n    } else {\n        throw 'invalid destination format for private key'\n    }\n}\n\nKeyEncoder.prototype.encodePublic = function(publicKey, originalFormat, destinationFormat) {\n    var publicKeyObject\n\n    /* Parse the incoming public key and convert it to a public key object */\n    if (originalFormat === 'raw') {\n        if (!typeof publicKey === 'string') {\n            throw 'public key must be a string'\n        }\n        publicKeyObject = this.publicKeyObject(publicKey)\n    } else if (originalFormat === 'der') {\n        if (typeof publicKey === 'buffer') {\n            // do nothing\n        } else if (typeof publicKey === 'string') {\n            publicKey = new Buffer(publicKey, 'hex')\n        } else {\n            throw 'public key must be a buffer or a string'\n        }\n        publicKeyObject = SubjectPublicKeyInfoASN.decode(publicKey, 'der')\n    } else if (originalFormat === 'pem') {\n        if (!typeof publicKey === 'string') {\n            throw 'public key must be a string'\n        }\n        publicKeyObject = SubjectPublicKeyInfoASN.decode(publicKey, 'pem', this.options.publicPEMOptions)\n    } else {\n        throw 'invalid public key format'\n    }\n\n    /* Export the private key object to the desired format */\n    if (destinationFormat === 'raw') {\n        return publicKeyObject.pub.data.toString('hex')\n    } else if (destinationFormat === 'der') {\n        return SubjectPublicKeyInfoASN.encode(publicKeyObject, 'der').toString('hex')\n    } else if (destinationFormat === 'pem') {\n        return SubjectPublicKeyInfoASN.encode(publicKeyObject, 'pem', this.options.publicPEMOptions)\n    } else {\n        throw 'invalid destination format for public key'\n    }\n}\n\nmodule.exports = KeyEncoder;","const path = require(\"path\");\nconst yazl = require(\"yazl\");\nconst yauzl = require(\"yauzl\");\nconst fs = require(\"fs\");\nconst DuplexStream = require(\"./utils/DuplexStream\");\nconst PassThroughStream = require(\"./utils/PassThroughStream\");\nconst isStream = require(\"./utils/isStream\");\n\nconst EventEmitter = require('events');\n\nconst countFiles = require('./utils/countFiles');\n\nfunction PskArchiver() {\n\n    const self = this;\n\n    const event = new EventEmitter();\n\n    this.on = event.on;\n    this.off = event.off;\n    this.emit = event.emit;\n\n    this.zipStream = function (inputPath, output, callback) {\n        let ext = \"\";\n        const zipFile = new yazl.ZipFile();\n        const ptStream = new PassThroughStream();\n\n        countFiles.computeSize(inputPath, (err, totalSize) => {\n            if (err) {\n                return callback(err);\n            }\n\n            __addToArchiveRecursively(zipFile, inputPath, \"\", (err) => {\n                if (err) {\n                    return callback(err);\n                }\n\n                zipFile.end();\n                const filename = path.basename(inputPath);\n                const splitFilename = filename.split(\".\");\n                if (splitFilename.length >= 2) {\n                    ext = \".\" + splitFilename[splitFilename.length - 1];\n                }\n                const myStream = zipFile.outputStream.pipe(ptStream);\n\n                let progressLength = 0;\n                let totalLength = 0;\n\n                /**\n                 * TODO review this\n                 * In browser, piping will block the event loop and the stack queue is not called.\n                 */\n                myStream.on(\"data\", (chunk) => {\n                    progressLength += chunk.length;\n                    totalLength += chunk.length;\n\n                    if (progressLength > 300000) {\n                        myStream.pause();\n                        progressLength = 0;\n                        setTimeout(function () {\n                            myStream.resume();\n                        }, 10);\n                        emitProgress(totalSize, totalLength)\n                    }\n                });\n\n                myStream.on('end', () => {\n                    emitProgress(totalSize, totalSize);\n                    emitTotalSize(totalSize);\n                });\n                if (isStream.isWritable(output)) {\n                    callback(null, myStream.pipe(output));\n                } else if (typeof output === \"string\") {\n                   fs.mkdir(output, {recursive: true}, () => {\n                        const destinationPath = path.join(output, path.basename(inputPath, ext) + \".zip\");\n                        myStream.pipe(fs.createWriteStream(destinationPath));\n                    });\n                }\n            });\n\n            function __addToArchiveRecursively(zipFile, inputPath, root = '', callback) {\n                root = root || '';\n                fs.stat(inputPath, (err, stats) => {\n                    if (err) {\n                        return callback(err);\n                    }\n                    if (stats.isFile()) {\n                        zipFile.addFile(inputPath, path.join(root, path.basename(inputPath)));\n                        callback(null);\n\n                    } else {\n                        fs.readdir(inputPath, (err, files) => {\n                            if (err) {\n                                return callback(err);\n                            }\n                            const f_length = files.length;\n                            let f_add_index = 0;\n\n                            const checkStatus = () => {\n                                if (f_length === f_add_index) {\n                                    callback(null);\n                                    return true;\n                                }\n                                return false;\n                            };\n\n                            if (!checkStatus()) {\n                                files.forEach(file => {\n                                    const tempPath = path.join(inputPath, file);\n                                    __addToArchiveRecursively(zipFile, tempPath, path.join(root, path.basename(inputPath)), (err) => {\n                                        if (err) {\n                                            return callback(err);\n                                        }\n                                        f_add_index++;\n                                        checkStatus();\n                                    })\n                                });\n                            }\n                        })\n                    }\n                });\n            }\n\n        });\n\n    };\n\n    this.unzipStream = function (input, outputPath, callback) {\n\n        let size = 0;\n\n        fs.stat(input, (err, stats) => {\n            if (err) {\n                return callback(err);\n            }\n\n            let totalSize = stats.size;\n\n\n            yauzl.open(input, {lazyEntries: true}, (err, zipFile) => {\n                if (err) {\n                    return callback(err);\n                }\n\n                let progressLength = 0;\n                let totalLength = 0;\n\n                const fileNames = [];\n                zipFile.readEntry();\n                zipFile.once(\"end\", () => {\n                    emitProgress(totalSize, totalSize);\n                    callback(null, fileNames);\n                });\n                zipFile.on(\"entry\", (entry) => {\n                    if (entry.fileName.endsWith(path.sep)) {\n                        zipFile.readEntry();\n                    } else {\n                        let folder = path.dirname(entry.fileName);\n                        fs.mkdir(path.join(outputPath, folder), {recursive: true}, () => {\n                            zipFile.openReadStream(entry, (err, readStream) => {\n                                if (err) {\n                                    return callback(err);\n                                }\n\n                                /**\n                                 * TODO review this\n                                 * In browser, piping will block the event loop and the stack queue is not called.\n                                 */\n\n                                readStream.on(\"data\", (chunk) => {\n                                    progressLength += chunk.length;\n                                    totalLength += chunk.length;\n\n                                    if (progressLength > 300000) {\n                                        readStream.pause();\n                                        progressLength = 0;\n                                        setTimeout(function () {\n                                            readStream.resume();\n                                        }, 30);\n                                        emitProgress(totalSize, totalLength)\n                                    }\n                                });\n\n\n                                readStream.on(\"end\", () => {\n                                    zipFile.readEntry();\n                                });\n                                const ptStream = new PassThroughStream();\n                                let fileName = path.join(outputPath, entry.fileName);\n                                let folder = path.dirname(fileName);\n                                const tempStream = readStream.pipe(ptStream);\n\n                                fs.mkdir(folder, {recursive: true}, (err) => {\n                                    if (err) {\n                                        return callback(err);\n                                    }\n\n                                    size += ptStream.getSize();\n                                    let output = fs.createWriteStream(fileName);\n                                    fileNames.push(fileName);\n                                    tempStream.pipe(output);\n                                });\n                            });\n                        });\n                    }\n                });\n            });\n\n        });\n\n    };\n\n    this.zipInMemory = function (inputObj, depth, callback) {\n        const zipFile = new yazl.ZipFile();\n        const ds = new DuplexStream();\n        zipRecursively(zipFile, inputObj, \"\", depth, (err) => {\n            if (err) {\n                return callback(err);\n            }\n            zipFile.end();\n            let buffer = Buffer.alloc(0);\n            ds.on('data', (chunk) => {\n                buffer = Buffer.concat([buffer, chunk]);\n            });\n\n            zipFile.outputStream.pipe(ds).on(\"finish\", (err) => {\n                if (err) {\n                    return callback(err);\n                }\n                callback(null, buffer);\n            });\n        })\n    };\n\n    this.unzipInMemory = function (inputZip, callback) {\n\n        function unzipInput(zipFile) {\n            zipFile.readEntry();\n            const obj = {};\n            zipFile.once(\"end\", () => {\n                callback(null, obj);\n            });\n\n            zipFile.on(\"entry\", (entry) => {\n                zipFile.openReadStream(entry, (err, readStream) => {\n                    const ds = new DuplexStream();\n                    let str = '';\n                    if (err) {\n                        return callback(err);\n                    }\n                    readStream.on(\"end\", () => {\n                        zipFile.readEntry();\n                    });\n                    ds.on(\"data\", (chunk) => {\n                        str += chunk.toString();\n                    });\n\n                    readStream.pipe(ds).on(\"finish\", (err) => {\n                        if (err) {\n                            return callback(err);\n                        }\n                        const splitEntry = entry.fileName.split(\"/\");\n                        const type = splitEntry.pop();\n                        addPropsRecursively(obj, splitEntry, type, new Buffer(str));\n                    });\n\n                });\n            })\n        }\n\n        if (Buffer.isBuffer(inputZip)) {\n            yauzl.fromBuffer(inputZip, {lazyEntries: true}, (err, zipFile) => {\n                if (err) {\n                    return callback(err);\n                }\n                unzipInput(zipFile)\n            });\n        } else {\n            return callback(new Error(\"input should be a buffer\"));\n        }\n\n    };\n\n    function zipRecursively(zipFile, obj, root, depth, callback) {\n        if (depth === 0) {\n            zipFile.addBuffer(new Buffer(JSON.stringify(obj)), root + \"/stringify\");\n            return;\n        }\n\n        if (typeof obj === 'undefined') {\n            zipFile.addBuffer(Buffer.alloc(0), root + \"/undefined\");\n        } else if (typeof obj === 'number') {\n            zipFile.addBuffer(new Buffer(obj.toString()), root + \"/number\");\n        } else if (typeof obj === 'string') {\n            zipFile.addBuffer(new Buffer(obj), root + \"/string\")\n        } else if (obj === null) {\n            zipFile.addBuffer(Buffer.alloc(0), root + \"/null\");\n        } else if (Buffer.isBuffer(obj)) {\n            zipFile.addBuffer(obj, root + \"/buffer\");\n        } else if (isStream.isReadable(obj)) {\n            zipFile.addReadStream(obj, root + \"/stream\");\n        } else if (Array.isArray(obj)) {\n            for (let i = 0; i < obj.length; i++) {\n                if (obj.length === 0) {\n                    zipFile.addBuffer(Buffer.alloc(0), root + \"/array\")\n                } else {\n                    zipRecursively(zipFile, obj[i], root + \"/array/\" + i, depth, (err) => {\n                        if (err) {\n                            return callback(err);\n                        }\n                    });\n                }\n            }\n        } else if (obj && typeof obj === 'object') {\n            let keys = Object.keys(obj);\n            if (keys.length === 0 && obj.constructor === Object) {\n                zipFile.addBuffer(Buffer.alloc(0), root + \"/object\");\n            } else {\n                const encodedObj = {};\n                Object.entries(obj).forEach(([key, value]) => {\n                    encodedObj[encodeURIComponent(key)] = value;\n                });\n                obj = encodedObj;\n                keys = Object.keys(obj);\n                keys.forEach(key => {\n                    let entryName;\n                    if (root === \"\") {\n                        entryName = key;\n                    } else {\n                        entryName = root + \"/\" + key;\n                    }\n                    zipRecursively(zipFile, obj[key], entryName, depth - 1, (err) => {\n                        if (err) {\n                            return callback(err);\n                        }\n                    });\n                });\n            }\n        } else {\n            throw new Error('Should never reach this');\n        }\n        callback(null);\n    }\n\n    function addPropsRecursively(obj, splitName, type, data) {\n        if (splitName.length >= 1) {\n            const prop = decodeURIComponent(splitName.shift());\n\n            if (splitName.length === 0) {\n                switch (type) {\n                    case 'undefined':\n                        obj[prop] = undefined;\n                        break;\n                    case 'null':\n                        obj[prop] = null;\n                        break;\n                    case 'number':\n                        obj[prop] = parseInt(data.toString());\n                        break;\n                    case 'string':\n                        obj[prop] = data.toString();\n                        break;\n                    case 'stream':\n                        obj[prop] = bufferToStream(data);\n                        break;\n                    case 'array':\n                        obj[prop] = [];\n                        break;\n                    case 'object':\n                        obj[prop] = {};\n                        break;\n                    case 'stringify':\n                        obj[prop] = JSON.parse(data.toString());\n                        break;\n                    default:\n                        throw new Error('Should never reach this');\n                }\n            } else {\n                if (splitName[0] === 'array') {\n                    if (!obj.hasOwnProperty(prop)) {\n                        obj[prop] = [];\n                    }\n                    splitName.shift();\n                    addPropsRecursively(obj[prop], splitName, type, data);\n                } else {\n                    if (!obj.hasOwnProperty(prop)) {\n                        obj[prop] = {};\n                    }\n                    addPropsRecursively(obj[prop], splitName, type, data);\n                }\n            }\n        }\n    }\n\n\n    function bufferToStream(buffer) {\n        let stream = new require('stream').Readable();\n        stream.push(buffer);\n        stream.push(null);\n        return stream;\n    }\n\n    function emitProgress(total, processed) {\n\n\n        if (processed > total) {\n            processed = total;\n        }\n\n        const progress = (100 * processed) / total;\n        self.emit('progress', progress);\n    }\n\n    function emitTotalSize(total) {\n        self.emit('total', total);\n    }\n\n\n}\n\nmodule.exports = PskArchiver;","const stream = require('stream');\nconst util = require('util');\n\nconst Duplex = stream.Duplex;\n\nfunction DuplexStream(options) {\n\tif (!(this instanceof DuplexStream)) {\n\t\treturn new DuplexStream(options);\n\t}\n\tDuplex.call(this, options);\n}\nutil.inherits(DuplexStream, Duplex);\n\nDuplexStream.prototype._write = function (chunk, enc, cb) {\n\tthis.push(chunk);\n\tcb();\n};\n\n\nDuplexStream.prototype._read = function (n) {\n\n};\n\nmodule.exports = DuplexStream;","const stream = require('stream');\nconst util = require('util');\n\nconst PassThrough = stream.PassThrough;\n\nfunction PassThroughStream(options) {\n    if (!(this instanceof PassThroughStream)) {\n        return new PassThroughStream(options);\n    }\n    PassThrough.call(this, options);\n\n    let size = 0;\n\n    this.addToSize = function (amount) {\n        size += amount;\n    };\n\n    this.getSize = function () {\n        return size;\n    }\n}\n\nutil.inherits(PassThroughStream, PassThrough);\n\nPassThroughStream.prototype._write = function (chunk, enc, cb) {\n    this.addToSize(chunk.length);\n    this.push(chunk);\n    cb();\n};\n\n\nPassThroughStream.prototype._read = function (n) {\n\n};\n\nmodule.exports = PassThroughStream;","const fs = require('fs');\nconst path = require('path');\nconst yauzl = require('yauzl');\n\nfunction countFiles(inputPath, callback) {\n    let total = 0;\n\n    fs.stat(inputPath, (err, stats) => {\n        if (err) {\n            return callback(err);\n        }\n\n        if (stats.isFile()) {\n            return callback(undefined, 1);\n        }\n\n        fs.readdir(inputPath, (err, files) => {\n            if (err) {\n                return callback(err);\n            }\n\n\n            total = files.length;\n            let count = files.length;\n\n            if (total === 0) {\n                return callback(undefined, 0);\n            }\n\n            files.forEach(file => {\n                fs.stat(path.join(inputPath, file), (err, stats) => {\n                    if (err) {\n                        return callback(err);\n                    }\n\n                    if (stats.isDirectory()) {\n                        --total;\n                        countFiles(path.join(inputPath, file), (err, filesNumber) => {\n                            if (err) {\n                                return callback(err);\n                            }\n\n                            total += filesNumber;\n\n\n                            if (--count === 0) {\n                                callback(undefined, total);\n                            }\n                        });\n                    } else {\n                        if (!stats.isFile()) {\n                            --total;\n                        }\n\n                        if (--count === 0) {\n                            callback(undefined, total);\n                        }\n                    }\n                });\n            })\n        });\n    });\n}\n\nfunction countZipEntries(inputPath, callback) {\n    let processed = 0;\n\n    yauzl.open(inputPath, {lazyEntries: true}, (err, zipFile) => {\n        if (err) {\n            return callback(err);\n        }\n\n        zipFile.readEntry();\n        zipFile.once(\"end\", () => {\n            callback(null, processed);\n        });\n\n        zipFile.on(\"entry\", (entry) => {\n            ++processed;\n\n            zipFile.readEntry();\n        });\n    });\n}\n\nfunction computeSize(inputPath, callback) {\n    let totalSize = 0;\n    fs.stat(inputPath, (err, stats) => {\n        if (err) {\n            return callback(err);\n        }\n\n        if (stats.isFile()) {\n            return callback(undefined, stats.size);\n        }\n\n        fs.readdir(inputPath, (err, files) => {\n            if (err) {\n                return callback(err);\n            }\n\n\n            let count = files.length;\n\n            if (count === 0) {\n                return callback(undefined, 0);\n            }\n\n            files.forEach(file => {\n                fs.stat(path.join(inputPath, file), (err, stats) => {\n                    if (err) {\n                        return callback(err);\n                    }\n\n                    if (stats.isDirectory()) {\n                        computeSize(path.join(inputPath, file), (err, filesSize) => {\n                            if (err) {\n                                return callback(err);\n                            }\n\n                            totalSize += filesSize;\n\n                            if (--count === 0) {\n                                callback(undefined, totalSize);\n                            }\n                        });\n                    } else {\n\n                        totalSize += stats.size;\n\n                        if (--count === 0) {\n                            callback(undefined, totalSize);\n                        }\n                    }\n                });\n            })\n        });\n    });\n}\n\nmodule.exports = {\n    countFiles,\n    countZipEntries,\n    computeSize\n};\n","const crypto = require('crypto');\nconst fs = require('fs');\nconst path = require(\"path\");\nconst PskArchiver = require(\"../psk-archiver\");\nconst algorithm = 'aes-256-gcm';\n\n\nconst iterations_number = 1000;\n\nfunction encode(buffer) {\n\treturn buffer.toString('base64')\n\t\t.replace(/\\+/g, '')\n\t\t.replace(/\\//g, '')\n\t\t.replace(/=+$/, '');\n}\n\nfunction deleteRecursively(inputPath, callback) {\n\n\tfs.stat(inputPath, function (err, stats) {\n\t\tif (err) {\n\t\t\tcallback(err, stats);\n\t\t\treturn;\n\t\t}\n\t\tif (stats.isFile()) {\n\t\t\tfs.unlink(inputPath, function (err) {\n\t\t\t\tif (err) {\n\t\t\t\t\tcallback(err, null);\n\t\t\t\t} else {\n\t\t\t\t\tcallback(null, true);\n\t\t\t\t}\n\t\t\t});\n\t\t} else if (stats.isDirectory()) {\n\t\t\tfs.readdir(inputPath, function (err, files) {\n\t\t\t\tif (err) {\n\t\t\t\t\tcallback(err, null);\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t\tconst f_length = files.length;\n\t\t\t\tlet f_delete_index = 0;\n\n\t\t\t\tconst checkStatus = function () {\n\t\t\t\t\tif (f_length === f_delete_index) {\n\t\t\t\t\t\tfs.rmdir(inputPath, function (err) {\n\t\t\t\t\t\t\tif (err) {\n\t\t\t\t\t\t\t\tcallback(err, null);\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tcallback(null, true);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t});\n\t\t\t\t\t\treturn true;\n\t\t\t\t\t}\n\t\t\t\t\treturn false;\n\t\t\t\t};\n\t\t\t\tif (!checkStatus()) {\n\t\t\t\t\tfiles.forEach(function (file) {\n\t\t\t\t\t\tconst tempPath = path.join(inputPath, file);\n\t\t\t\t\t\tdeleteRecursively(tempPath, function removeRecursiveCB(err, status) {\n\t\t\t\t\t\t\tif (!err) {\n\t\t\t\t\t\t\t\tf_delete_index++;\n\t\t\t\t\t\t\t\tcheckStatus();\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tcallback(err, null);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t});\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t});\n\t\t}\n\t});\n}\n\n\n\n\n\nfunction createPskHash(data) {\n\tconst pskHash = new PskHash();\n\tpskHash.update(data);\n\treturn pskHash.digest();\n}\n\nfunction PskHash() {\n\tconst sha512 = crypto.createHash('sha512');\n\tconst sha256 = crypto.createHash('sha256');\n\n\tfunction update(data) {\n\t\tsha512.update(data);\n\t}\n\n\tfunction digest() {\n\t\tsha256.update(sha512.digest());\n\t\treturn sha256.digest();\n\t}\n\n\treturn {\n\t\tupdate,\n\t\tdigest\n\t}\n}\n\n\nfunction generateSalt(inputData, saltLen) {\n\tconst hash = crypto.createHash('sha512');\n\thash.update(inputData);\n\tconst digest = Buffer.from(hash.digest('hex'), 'binary');\n\n\treturn digest.slice(0, saltLen);\n}\n\nfunction encrypt(data, password) {\n\tconst keySalt = crypto.randomBytes(32);\n\tconst key = crypto.pbkdf2Sync(password, keySalt, iterations_number, 32, 'sha512');\n\n\tconst aadSalt = crypto.randomBytes(32);\n\tconst aad = crypto.pbkdf2Sync(password, aadSalt, iterations_number, 32, 'sha512');\n\n\tconst salt = Buffer.concat([keySalt, aadSalt]);\n\tconst iv = crypto.pbkdf2Sync(password, salt, iterations_number, 12, 'sha512');\n\n\tconst cipher = crypto.createCipheriv(algorithm, key, iv);\n\tcipher.setAAD(aad);\n\tlet encryptedText = cipher.update(data, 'binary');\n\tconst final = Buffer.from(cipher.final('binary'), 'binary');\n\tconst tag = cipher.getAuthTag();\n\n\tencryptedText = Buffer.concat([encryptedText, final]);\n\n\treturn Buffer.concat([salt, encryptedText, tag]);\n}\n\nfunction decrypt(encryptedData, password) {\n\tconst salt = encryptedData.slice(0, 64);\n\tconst keySalt = salt.slice(0, 32);\n\tconst aadSalt = salt.slice(-32);\n\n\tconst iv = crypto.pbkdf2Sync(password, salt, iterations_number, 12, 'sha512');\n\tconst key = crypto.pbkdf2Sync(password, keySalt, iterations_number, 32, 'sha512');\n\tconst aad = crypto.pbkdf2Sync(password, aadSalt, iterations_number, 32, 'sha512');\n\n\tconst ciphertext = encryptedData.slice(64, encryptedData.length - 16);\n\tconst tag = encryptedData.slice(-16);\n\n\tconst decipher = crypto.createDecipheriv(algorithm, key, iv);\n\tdecipher.setAuthTag(tag);\n\tdecipher.setAAD(aad);\n\n\tlet plaintext = Buffer.from(decipher.update(ciphertext, 'binary'), 'binary');\n\tconst final = Buffer.from(decipher.final('binary'), 'binary');\n\tplaintext = Buffer.concat([plaintext, final]);\n\treturn plaintext;\n}\n\nfunction encryptObjectInMemory(inputObj, password, depth, callback) {\n\tconst archiver = new PskArchiver();\n\n\tarchiver.zipInMemory(inputObj, depth, function (err, zippedObj) {\n\t\tif (err) {\n\t\t\treturn callback(err);\n\t\t}\n\t\tconst cipherText = encrypt(zippedObj, password);\n\t\tcallback(null, cipherText);\n\t})\n}\n\nfunction decryptObjectInMemory(encryptedObject, password, callback) {\n\tconst archiver = new PskArchiver();\n\n\tconst zippedObject = decrypt(encryptedObject, password);\n\tarchiver.unzipInMemory(zippedObject, function (err, obj) {\n\t\tif (err) {\n\t\t\treturn callback(err);\n\t\t}\n\t\tcallback(null, obj);\n\t})\n}\n\n\nmodule.exports = {\n\tcreatePskHash,\n\tencrypt,\n\tencryptObjectInMemory,\n\tdecrypt,\n\tdecryptObjectInMemory,\n\tdeleteRecursively,\n\tencode,\n\tgenerateSalt,\n\titerations_number,\n\talgorithm,\n\tPskHash\n};\n\n","const stream = require('stream');\n\n\nfunction isStream (obj) {\n\treturn obj instanceof stream.Stream || obj instanceof stream.Duplex;\n}\n\n\nfunction isReadable (obj) {\n\treturn isStream(obj) && typeof obj._read === 'function' && typeof obj._readableState === 'object'\n}\n\n\nfunction isWritable (obj) {\n\treturn isStream(obj) && typeof obj._write === 'function' && typeof obj._writableState === 'object'\n}\n\n\nfunction isDuplex (obj) {\n\treturn isReadable(obj) && isWritable(obj)\n}\n\n\nmodule.exports            = isStream;\nmodule.exports.isReadable = isReadable;\nmodule.exports.isWritable = isWritable;\nmodule.exports.isDuplex   = isDuplex;","/*\n SignSens helper functions\n */\nconst crypto = require('crypto');\n\nexports.wipeOutsidePayload = function wipeOutsidePayload(hashStringHexa, pos, size){\n    var result;\n    var sz = hashStringHexa.length;\n\n    var end = (pos + size) % sz;\n\n    if(pos < end){\n        result = '0'.repeat(pos) +  hashStringHexa.substring(pos, end) + '0'.repeat(sz - end);\n    }\n    else {\n        result = hashStringHexa.substring(0, end) + '0'.repeat(pos - end) + hashStringHexa.substring(pos, sz);\n    }\n    return result;\n}\n\n\n\nexports.extractPayload = function extractPayload(hashStringHexa, pos, size){\n    var result;\n\n    var sz = hashStringHexa.length;\n    var end = (pos + size) % sz;\n\n    if( pos < end){\n        result = hashStringHexa.substring(pos, pos + size);\n    } else{\n\n        if(0 != end){\n            result = hashStringHexa.substring(0, end)\n        }  else {\n            result = \"\";\n        }\n        result += hashStringHexa.substring(pos, sz);\n    }\n    return result;\n}\n\n\n\nexports.fillPayload = function fillPayload(payload, pos, size){\n    var sz = 64;\n    var result = \"\";\n\n    var end = (pos + size) % sz;\n\n    if( pos < end){\n        result = '0'.repeat(pos) + payload + '0'.repeat(sz - end);\n    } else{\n        result = payload.substring(0,end);\n        result += '0'.repeat(pos - end);\n        result += payload.substring(end);\n    }\n    return result;\n}\n\n\n\nexports.generatePosHashXTimes = function generatePosHashXTimes(buffer, pos, size, count){ //generate positional hash\n    var result  = buffer.toString(\"hex\");\n\n    /*if(pos != -1 )\n        result[pos] = 0; */\n\n    for(var i = 0; i < count; i++){\n        var hash = crypto.createHash('sha256');\n        result = exports.wipeOutsidePayload(result, pos, size);\n        hash.update(result);\n        result = hash.digest('hex');\n    }\n    return exports.wipeOutsidePayload(result, pos, size);\n}\n\nexports.hashStringArray = function (counter, arr, payloadSize){\n\n    const hash = crypto.createHash('sha256');\n    var result = counter.toString(16);\n\n    for(var i = 0 ; i < 64; i++){\n        result += exports.extractPayload(arr[i],i, payloadSize);\n    }\n\n    hash.update(result);\n    var result = hash.digest('hex');\n    return result;\n}\n\n\n\n\n\n\nfunction dumpMember(obj){\n    var type = Array.isArray(obj) ? \"array\" : typeof obj;\n    if(obj === null){\n        return \"null\";\n    }\n    if(obj === undefined){\n        return \"undefined\";\n    }\n\n    switch(type){\n        case \"number\":\n        case \"string\":return obj.toString(); break;\n        case \"object\": return exports.dumpObjectForHashing(obj); break;\n        case \"boolean\": return  obj? \"true\": \"false\"; break;\n        case \"array\":\n            var result = \"\";\n            for(var i=0; i < obj.length; i++){\n                result += exports.dumpObjectForHashing(obj[i]);\n            }\n            return result;\n            break;\n        default:\n            throw new Error(\"Type \" +  type + \" cannot be cryptographically digested\");\n    }\n\n}\n\n\nexports.dumpObjectForHashing = function(obj){\n    var result = \"\";\n\n    if(obj === null){\n        return \"null\";\n    }\n    if(obj === undefined){\n        return \"undefined\";\n    }\n\n    var basicTypes = {\n        \"array\"     : true,\n        \"number\"    : true,\n        \"boolean\"   : true,\n        \"string\"    : true,\n        \"object\"    : false\n    }\n\n    var type = Array.isArray(obj) ? \"array\" : typeof obj;\n    if( basicTypes[type]){\n        return dumpMember(obj);\n    }\n\n    var keys = Object.keys(obj);\n    keys.sort();\n\n\n    for(var i=0; i < keys.length; i++){\n        result += dumpMember(keys[i]);\n        result += dumpMember(obj[keys[i]]);\n    }\n\n    return result;\n}\n\n\nexports.hashValues  = function (values){\n    const hash = crypto.createHash('sha256');\n    var result = exports.dumpObjectForHashing(values);\n    hash.update(result);\n    return hash.digest('hex');\n};\n\nexports.getJSONFromSignature = function getJSONFromSignature(signature, size){\n    var result = {\n        proof:[]\n    };\n    var a = signature.split(\":\");\n    result.agent        = a[0];\n    result.counter      =  parseInt(a[1], \"hex\");\n    result.nextPublic   =  a[2];\n\n    var proof = a[3]\n\n\n    if(proof.length/size != 64) {\n        throw new Error(\"Invalid signature \" + proof);\n    }\n\n    for(var i = 0; i < 64; i++){\n        result.proof.push(exports.fillPayload(proof.substring(i * size,(i+1) * size ), i, size))\n    }\n\n    return result;\n}\n\nexports.createSignature = function (agent,counter, nextPublic, arr, size){\n    var result = \"\";\n\n    for(var i = 0; i < arr.length; i++){\n        result += exports.extractPayload(arr[i], i , size);\n    }\n\n    return agent + \":\" + counter + \":\" + nextPublic + \":\" + result;\n}","/*\nInitial License: (c) Axiologic Research & Alboaie Sînică.\nContributors: Axiologic Research , PrivateSky project\nCode License: LGPL or MIT.\n*/\n\n\n/**\n *   Usually an event could cause execution of other callback events . We say that is a level 1 event if is causeed by a level 0 event and so on\n *\n *      SoundPubSub provides intuitive results regarding to asynchronous calls of callbacks and computed values/expressions:\n *   we prevent immediate execution of event callbacks to ensure the intuitive final result is guaranteed as level 0 execution\n *   we guarantee that any callback function is \"re-entrant\"\n *   we are also trying to reduce the number of callback execution by looking in queues at new messages published by\n *   trying to compact those messages (removing duplicate messages, modifying messages, or adding in the history of another event ,etc)\n *\n *      Example of what can be wrong without non-sound asynchronous calls:\n *\n *  Step 0: Initial state:\n *   a = 0;\n *   b = 0;\n *\n *  Step 1: Initial operations:\n *   a = 1;\n *   b = -1;\n *\n *  // an observer reacts to changes in a and b and compute CORRECT like this:\n *   if( a + b == 0) {\n *       CORRECT = false;\n *       notify(...); // act or send a notification somewhere..\n *   } else {\n *      CORRECT = false;\n *   }\n *\n *    Notice that: CORRECT will be true in the end , but meantime, after a notification was sent and CORRECT was wrongly, temporarily false!\n *    soundPubSub guarantee that this does not happen because the syncronous call will before any observer (bot asignation on a and b)\n *\n *   More:\n *   you can use blockCallBacks and releaseCallBacks in a function that change a lot a collection or bindable objects and all\n *   the notifications will be sent compacted and properly\n */\n\n// TODO: optimisation!? use a more efficient queue instead of arrays with push and shift!?\n// TODO: see how big those queues can be in real applications\n// for a few hundreds items, queues made from array should be enough\n//*   Potential TODOs:\n//    *     prevent any form of problem by calling callbacks in the expected order !?\n//*     preventing infinite loops execution cause by events!?\n//*\n//*\n// TODO: detect infinite loops (or very deep propagation) It is possible!?\n\nconst Queue = require('swarmutils').Queue;\n\nfunction SoundPubSub(){\n\n\t/**\n\t * publish\n\t *      Publish a message {Object} to a list of subscribers on a specific topic\n\t *\n\t * @params {String|Number} target,  {Object} message\n\t * @return number of channel subscribers that will be notified\n\t */\n\tthis.publish = function(target, message){\n\t\tif(!invalidChannelName(target) && !invalidMessageType(message) && (typeof channelSubscribers[target] != 'undefined')){\n\t\t\tcompactAndStore(target, message);\n\t\t\tsetTimeout(dispatchNext, 0);\n\t\t\treturn channelSubscribers[target].length;\n\t\t} else{\n\t\t\treturn null;\n\t\t}\n\t};\n\n\t/**\n\t * subscribe\n\t *      Subscribe / add a {Function} callBack on a {String|Number}target channel subscribers list in order to receive\n\t *      messages published if the conditions defined by {Function}waitForMore and {Function}filter are passed.\n\t *\n\t * @params {String|Number}target, {Function}callBack, {Function}waitForMore, {Function}filter\n\t *\n\t *          target      - channel name to subscribe\n\t *          callback    - function to be called when a message was published on the channel\n\t *          waitForMore - a intermediary function that will be called after a successfuly message delivery in order\n\t *                          to decide if a new messages is expected...\n\t *          filter      - a function that receives the message before invocation of callback function in order to allow\n\t *                          relevant message before entering in normal callback flow\n\t * @return\n\t */\n\tthis.subscribe = function(target, callBack, waitForMore, filter){\n\t\tif(!invalidChannelName(target) && !invalidFunction(callBack)){\n\t\t\tvar subscriber = {\"callBack\":callBack, \"waitForMore\":waitForMore, \"filter\":filter};\n\t\t\tvar arr = channelSubscribers[target];\n\t\t\tif(typeof arr == 'undefined'){\n\t\t\t\tarr = [];\n\t\t\t\tchannelSubscribers[target] = arr;\n\t\t\t}\n\t\t\tarr.push(subscriber);\n\t\t}\n\t};\n\n\t/**\n\t * unsubscribe\n\t *      Unsubscribe/remove {Function} callBack from the list of subscribers of the {String|Number} target channel\n\t *\n\t * @params {String|Number} target, {Function} callBack, {Function} filter\n\t *\n\t *          target      - channel name to unsubscribe\n\t *          callback    - reference of the original function that was used as subscribe\n\t *          filter      - reference of the original filter function\n\t * @return\n\t */\n\tthis.unsubscribe = function(target, callBack, filter){\n\t\tif(!invalidFunction(callBack)){\n\t\t\tvar gotit = false;\n\t\t\tif(channelSubscribers[target]){\n\t\t\t\tfor(var i = 0; i < channelSubscribers[target].length;i++){\n\t\t\t\t\tvar subscriber =  channelSubscribers[target][i];\n\t\t\t\t\tif(subscriber.callBack === callBack && ( typeof filter === 'undefined' || subscriber.filter === filter )){\n\t\t\t\t\t\tgotit = true;\n\t\t\t\t\t\tsubscriber.forDelete = true;\n\t\t\t\t\t\tsubscriber.callBack = undefined;\n\t\t\t\t\t\tsubscriber.filter = undefined;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t\tif(!gotit){\n\t\t\t\twprint(\"Unable to unsubscribe a callback that was not subscribed!\");\n\t\t\t}\n\t\t}\n\t};\n\n\t/**\n\t * blockCallBacks\n\t *\n\t * @params\n\t * @return\n\t */\n\tthis.blockCallBacks = function(){\n\t\tlevel++;\n\t};\n\n\t/**\n\t * releaseCallBacks\n\t *\n\t * @params\n\t * @return\n\t */\n\tthis.releaseCallBacks = function(){\n\t\tlevel--;\n\t\t//hack/optimisation to not fill the stack in extreme cases (many events caused by loops in collections,etc)\n\t\twhile(level === 0 && dispatchNext(true)){\n\t\t\t//nothing\n\t\t}\n\n\t\twhile(level === 0 && callAfterAllEvents()){\n            //nothing\n\t\t}\n\t};\n\n\t/**\n\t * afterAllEvents\n\t *\n\t * @params {Function} callback\n\t *\n\t *          callback - function that needs to be invoked once all events are delivered\n\t * @return\n\t */\n\tthis.afterAllEvents = function(callBack){\n\t\tif(!invalidFunction(callBack)){\n\t\t\tafterEventsCalls.push(callBack);\n\t\t}\n\t\tthis.blockCallBacks();\n\t\tthis.releaseCallBacks();\n\t};\n\n\t/**\n\t * hasChannel\n\t *\n\t * @params {String|Number} channel\n\t *\n\t *          channel - name of the channel that need to be tested if present\n\t * @return\n\t */\n\tthis.hasChannel = function(channel){\n\t\treturn !invalidChannelName(channel) && (typeof channelSubscribers[channel] != 'undefined') ? true : false;\n\t};\n\n\t/**\n\t * addChannel\n\t *\n\t * @params {String} channel\n\t *\n\t *          channel - name of a channel that needs to be created and added to soundpubsub repository\n\t * @return\n\t */\n\tthis.addChannel = function(channel){\n\t\tif(!invalidChannelName(channel) && !this.hasChannel(channel)){\n\t\t\tchannelSubscribers[channel] = [];\n\t\t}\n\t};\n\n\t/* ---------------------------------------- protected stuff ---------------------------------------- */\n\tvar self = this;\n\t// map channelName (object local id) -> array with subscribers\n\tvar channelSubscribers = {};\n\n\t// map channelName (object local id) -> queue with waiting messages\n\tvar channelsStorage = {};\n\n\t// object\n\tvar typeCompactor = {};\n\n\t// channel names\n\tvar executionQueue = new Queue();\n\tvar level = 0;\n\n\n\n\t/**\n\t * registerCompactor\n\t *\n\t *       An compactor takes a newEvent and and oldEvent and return the one that survives (oldEvent if\n\t *  it can compact the new one or the newEvent if can't be compacted)\n\t *\n\t * @params {String} type, {Function} callBack\n\t *\n\t *          type        - channel name to unsubscribe\n\t *          callBack    - handler function for that specific event type\n\t * @return\n\t */\n\tthis.registerCompactor = function(type, callBack) {\n\t\tif(!invalidFunction(callBack)){\n\t\t\ttypeCompactor[type] = callBack;\n\t\t}\n\t};\n\n\t/**\n\t * dispatchNext\n\t *\n\t * @param fromReleaseCallBacks: hack to prevent too many recursive calls on releaseCallBacks\n\t * @return {Boolean}\n\t */\n\tfunction dispatchNext(fromReleaseCallBacks){\n\t\tif(level > 0) {\n\t\t\treturn false;\n\t\t}\n\t\tconst channelName = executionQueue.front();\n\t\tif(typeof channelName != 'undefined'){\n\t\t\tself.blockCallBacks();\n\t\t\ttry{\n\t\t\t\tlet message;\n\t\t\t\tif(!channelsStorage[channelName].isEmpty()) {\n\t\t\t\t\tmessage = channelsStorage[channelName].front();\n\t\t\t\t}\n\t\t\t\tif(typeof message == 'undefined'){\n\t\t\t\t\tif(!channelsStorage[channelName].isEmpty()){\n\t\t\t\t\t\twprint(\"Can't use as message in a pub/sub channel this object: \" + message);\n\t\t\t\t\t}\n\t\t\t\t\texecutionQueue.pop();\n\t\t\t\t} else {\n\t\t\t\t\tif(typeof message.__transmisionIndex == 'undefined'){\n\t\t\t\t\t\tmessage.__transmisionIndex = 0;\n\t\t\t\t\t\tfor(var i = channelSubscribers[channelName].length-1; i >= 0 ; i--){\n\t\t\t\t\t\t\tvar subscriber =  channelSubscribers[channelName][i];\n\t\t\t\t\t\t\tif(subscriber.forDelete === true){\n\t\t\t\t\t\t\t\tchannelSubscribers[channelName].splice(i,1);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t} else{\n\t\t\t\t\t\tmessage.__transmisionIndex++;\n\t\t\t\t\t}\n\t\t\t\t\t//TODO: for immutable objects it will not work also, fix for shape models\n\t\t\t\t\tif(typeof message.__transmisionIndex == 'undefined'){\n\t\t\t\t\t\twprint(\"Can't use as message in a pub/sub channel this object: \" + message);\n\t\t\t\t\t}\n\t\t\t\t\tsubscriber = channelSubscribers[channelName][message.__transmisionIndex];\n\t\t\t\t\tif(typeof subscriber == 'undefined'){\n\t\t\t\t\t\tdelete message.__transmisionIndex;\n\t\t\t\t\t\tchannelsStorage[channelName].pop();\n\t\t\t\t\t} else{\n\t\t\t\t\t\tif(subscriber.filter === null || typeof subscriber.filter === \"undefined\" || (!invalidFunction(subscriber.filter) && subscriber.filter(message))){\n\t\t\t\t\t\t\tif(!subscriber.forDelete){\n\t\t\t\t\t\t\t\tsubscriber.callBack(message);\n\t\t\t\t\t\t\t\tif(subscriber.waitForMore && !invalidFunction(subscriber.waitForMore) && !subscriber.waitForMore(message)){\n\t\t\t\t\t\t\t\t\tsubscriber.forDelete = true;\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} catch(err){\n\t\t\t\twprint(\"Event callback failed: \"+ subscriber.callBack +\"error: \" + err.stack);\n\t\t\t}\n\t\t\t//\n\t\t\tif(fromReleaseCallBacks){\n\t\t\t\tlevel--;\n\t\t\t} else{\n\t\t\t\tself.releaseCallBacks();\n\t\t\t}\n\t\t\treturn true;\n\t\t} else{\n\t\t\treturn false;\n\t\t}\n\t}\n\n\tfunction compactAndStore(target, message){\n\t\tvar gotCompacted = false;\n\t\tvar arr = channelsStorage[target];\n\t\tif(typeof arr == 'undefined'){\n\t\t\tarr = new Queue();\n\t\t\tchannelsStorage[target] = arr;\n\t\t}\n\n\t\tif(message && typeof message.type != 'undefined'){\n\t\t\tvar typeCompactorCallBack = typeCompactor[message.type];\n\n\t\t\tif(typeof typeCompactorCallBack != 'undefined'){\n\t\t\t\tfor(let channel of arr) {\n\t\t\t\t\tif(typeCompactorCallBack(message, channel) === channel) {\n\t\t\t\t\t\tif(typeof channel.__transmisionIndex == 'undefined') {\n\t\t\t\t\t\t\tgotCompacted = true;\n\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tif(!gotCompacted && message){\n\t\t\tarr.push(message);\n\t\t\texecutionQueue.push(target);\n\t\t}\n\t}\n\n\tvar afterEventsCalls = new Queue();\n\tfunction callAfterAllEvents (){\n\t\tif(!afterEventsCalls.isEmpty()){\n\t\t\tvar callBack = afterEventsCalls.pop();\n\t\t\t//do not catch exceptions here..\n\t\t\tcallBack();\n\t\t}\n\t\treturn !afterEventsCalls.isEmpty();\n\t}\n\n\tfunction invalidChannelName(name){\n\t\tvar result = false;\n\t\tif(!name || (typeof name != \"string\" && typeof name != \"number\")){\n\t\t\tresult = true;\n\t\t\twprint(\"Invalid channel name: \" + name);\n\t\t}\n\n\t\treturn result;\n\t}\n\n\tfunction invalidMessageType(message){\n\t\tvar result = false;\n\t\tif(!message || typeof message != \"object\"){\n\t\t\tresult = true;\n\t\t\twprint(\"Invalid messages types: \" + message);\n\t\t}\n\t\treturn result;\n\t}\n\n\tfunction invalidFunction(callback){\n\t\tvar result = false;\n\t\tif(!callback || typeof callback != \"function\"){\n\t\t\tresult = true;\n\t\t\twprint(\"Expected to be function but is: \" + callback);\n\t\t}\n\t\treturn result;\n\t}\n}\n\nexports.soundPubSub = new SoundPubSub();","function product(args) {\n    if(!args.length){\n        return [ [] ];\n    }\n    var prod = product(args.slice(1)), r = [];\n    args[0].forEach(function(x) {\n        prod.forEach(function(p) {\n            r.push([ x ].concat(p));\n        });\n    });\n    return r;\n}\n\nfunction objectProduct(obj) {\n    var keys = Object.keys(obj),\n        values = keys.map(function(x) { return obj[x]; });\n\n    return product(values).map(function(p) {\n        var e = {};\n        keys.forEach(function(k, n) { e[k] = p[n]; });\n        return e;\n    });\n}\n\nmodule.exports = objectProduct;","var meta = \"meta\";\n\nfunction OwM(serialized){\n\n    if(serialized){\n        return OwM.prototype.convert(serialized);\n    }\n\n    Object.defineProperty(this, meta, {\n        writable: false,\n        enumerable: true,\n        value: {}\n    });\n\n    Object.defineProperty(this, \"setMeta\", {\n        writable: false,\n        enumerable: false,\n        configurable:false,\n        value: function(prop, value){\n            if(typeof prop == \"object\" && typeof value == \"undefined\"){\n                for(var p in prop){\n                    this[meta][p] = prop[p];\n                }\n                return prop;\n            }\n            this[meta][prop] = value;\n            return value;\n        }\n    });\n\n    Object.defineProperty(this, \"getMeta\", {\n        writable: false,\n        value: function(prop){\n            return this[meta][prop];\n        }\n    });\n}\n\nfunction testOwMSerialization(obj){\n    let res = false;\n\n    if(obj){\n        res = typeof obj[meta] != \"undefined\" && !(obj instanceof OwM);\n    }\n\n    return res;\n}\n\nOwM.prototype.convert = function(serialized){\n    const owm = new OwM();\n\n    for(var metaProp in serialized.meta){\n        if(!testOwMSerialization(serialized[metaProp])) {\n            owm.setMeta(metaProp, serialized.meta[metaProp]);\n        }else{\n            owm.setMeta(metaProp, OwM.prototype.convert(serialized.meta[metaProp]));\n        }\n    }\n\n    for(var simpleProp in serialized){\n        if(simpleProp === meta) {\n            continue;\n        }\n\n        if(!testOwMSerialization(serialized[simpleProp])){\n            owm[simpleProp] = serialized[simpleProp];\n        }else{\n            owm[simpleProp] = OwM.prototype.convert(serialized[simpleProp]);\n        }\n    }\n\n    return owm;\n};\n\nOwM.prototype.getMetaFrom = function(obj, name){\n    var res;\n    if(!name){\n        res = obj[meta];\n    }else{\n        res = obj[meta][name];\n    }\n    return res;\n};\n\nOwM.prototype.setMetaFor = function(obj, name, value){\n    obj[meta][name] = value;\n    return obj[meta][name];\n};\n\nmodule.exports = OwM;","function QueueElement(content) {\n\tthis.content = content;\n\tthis.next = null;\n}\n\nfunction Queue() {\n\tthis.head = null;\n\tthis.tail = null;\n\tthis.length = 0;\n\tthis.push = function (value) {\n\t\tconst newElement = new QueueElement(value);\n\t\tif (!this.head) {\n\t\t\tthis.head = newElement;\n\t\t\tthis.tail = newElement;\n\t\t} else {\n\t\t\tthis.tail.next = newElement;\n\t\t\tthis.tail = newElement;\n\t\t}\n\t\tthis.length++;\n\t};\n\n\tthis.pop = function () {\n\t\tif (!this.head) {\n\t\t\treturn null;\n\t\t}\n\t\tconst headCopy = this.head;\n\t\tthis.head = this.head.next;\n\t\tthis.length--;\n\n\t\t//fix???????\n\t\tif(this.length === 0){\n            this.tail = null;\n\t\t}\n\n\t\treturn headCopy.content;\n\t};\n\n\tthis.front = function () {\n\t\treturn this.head ? this.head.content : undefined;\n\t};\n\n\tthis.isEmpty = function () {\n\t\treturn this.head === null;\n\t};\n\n\tthis[Symbol.iterator] = function* () {\n\t\tlet head = this.head;\n\t\twhile(head !== null) {\n\t\t\tyield head.content;\n\t\t\thead = head.next;\n\t\t}\n\t}.bind(this);\n}\n\nQueue.prototype.toString = function () {\n\tlet stringifiedQueue = '';\n\tlet iterator = this.head;\n\twhile (iterator) {\n\t\tstringifiedQueue += `${JSON.stringify(iterator.content)} `;\n\t\titerator = iterator.next;\n\t}\n\treturn stringifiedQueue;\n};\n\nQueue.prototype.inspect = Queue.prototype.toString;\n\nmodule.exports = Queue;","const OwM = require(\"./OwM\");\n\n/*\n    Prepare the state of a swarm to be serialised\n*/\n\nexports.asJSON = function(valueObj, phaseName, args, callback){\n\n        let valueObject = valueObj.valueOf();\n        let res = new OwM();\n        res.publicVars          = valueObject.publicVars;\n        res.privateVars         = valueObject.privateVars;\n\n        res.setMeta(\"swarmTypeName\", OwM.prototype.getMetaFrom(valueObject, \"swarmTypeName\"));\n        res.setMeta(\"swarmId\",       OwM.prototype.getMetaFrom(valueObject, \"swarmId\"));\n        res.setMeta(\"target\",        OwM.prototype.getMetaFrom(valueObject, \"target\"));\n        res.setMeta(\"homeSecurityContext\",        OwM.prototype.getMetaFrom(valueObject, \"homeSecurityContext\"));\n        res.setMeta(\"requestId\",        OwM.prototype.getMetaFrom(valueObject, \"requestId\"));\n\n        if(!phaseName){\n            res.setMeta(\"command\", \"stored\");\n        } else {\n            res.setMeta(\"phaseName\", phaseName);\n            res.setMeta(\"phaseId\", $$.uidGenerator.safe_uuid());\n            res.setMeta(\"args\", args);\n            res.setMeta(\"command\", OwM.prototype.getMetaFrom(valueObject, \"command\") || \"executeSwarmPhase\");\n        }\n\n        res.setMeta(\"waitStack\", valueObject.meta.waitStack); //TODO: think if is not better to be deep cloned and not referenced!!!\n\n        if(callback){\n            return callback(null, res);\n        }\n        //console.log(\"asJSON:\", res, valueObject);\n        return res;\n};\n\nexports.jsonToNative = function(serialisedValues, result){\n\n    for(let v in serialisedValues.publicVars){\n        result.publicVars[v] = serialisedValues.publicVars[v];\n\n    };\n    for(let l in serialisedValues.privateVars){\n        result.privateVars[l] = serialisedValues.privateVars[l];\n    };\n\n    for(let i in OwM.prototype.getMetaFrom(serialisedValues)){\n        OwM.prototype.setMetaFor(result, i, OwM.prototype.getMetaFrom(serialisedValues, i));\n    };\n\n};","var commands = {};\nvar commands_help = {};\n\n//global function addCommand\naddCommand = function addCommand(verb, adverbe, funct, helpLine){\n    var cmdId;\n    if(!helpLine){\n        helpLine = \" \";\n    } else {\n        helpLine = \" \" + helpLine;\n    }\n    if(adverbe){\n        cmdId = verb + \" \" +  adverbe;\n        helpLine = verb + \" \" +  adverbe + helpLine;\n    } else {\n        cmdId = verb;\n        helpLine = verb + helpLine;\n    }\n    commands[cmdId] = funct;\n        commands_help[cmdId] = helpLine;\n};\n\nfunction doHelp(){\n    console.log(\"List of commands:\");\n    for(var l in commands_help){\n        console.log(\"\\t\", commands_help[l]);\n    }\n}\n\naddCommand(\"-h\", null, doHelp, \"\\t\\t\\t\\t\\t\\t |just print the help\");\naddCommand(\"/?\", null, doHelp, \"\\t\\t\\t\\t\\t\\t |just print the help\");\naddCommand(\"help\", null, doHelp, \"\\t\\t\\t\\t\\t\\t |just print the help\");\n\n\nfunction runCommand(){\n  var argv = Object.assign([], process.argv);\n  var cmdId = null;\n  var cmd = null;\n  argv.shift();\n  argv.shift();\n\n  if(argv.length >=1){\n      cmdId = argv[0];\n      cmd = commands[cmdId];\n      argv.shift();\n  }\n\n\n  if(!cmd && argv.length >=1){\n      cmdId = cmdId + \" \" + argv[0];\n      cmd = commands[cmdId];\n      argv.shift();\n  }\n\n  if(!cmd){\n    if(cmdId){\n        console.log(\"Unknown command: \", cmdId);\n    }\n    cmd = doHelp;\n  }\n\n  cmd.apply(null,argv);\n\n}\n\nmodule.exports = {\n    runCommand\n};\n\n","\nfunction encode(buffer) {\n    return buffer.toString('base64')\n        .replace(/\\+/g, '')\n        .replace(/\\//g, '')\n        .replace(/=+$/, '');\n};\n\nfunction stampWithTime(buf, salt, msalt){\n    if(!salt){\n        salt = 1;\n    }\n    if(!msalt){\n        msalt = 1;\n    }\n    var date = new Date;\n    var ct = Math.floor(date.getTime() / salt);\n    var counter = 0;\n    while(ct > 0 ){\n        //console.log(\"Counter\", counter, ct);\n        buf[counter*msalt] = Math.floor(ct % 256);\n        ct = Math.floor(ct / 256);\n        counter++;\n    }\n}\n\n/*\n    The uid contains around 256 bits of randomness and are unique at the level of seconds. This UUID should by cryptographically safe (can not be guessed)\n\n    We generate a safe UID that is guaranteed unique (by usage of a PRNG to geneate 256 bits) and time stamping with the number of seconds at the moment when is generated\n    This method should be safe to use at the level of very large distributed systems.\n    The UUID is stamped with time (seconds): does it open a way to guess the UUID? It depends how safe is \"crypto\" PRNG, but it should be no problem...\n\n */\n\nvar generateUid = null;\n\n\nexports.init = function(externalGenerator){\n    generateUid = externalGenerator.generateUid;\n    return module.exports;\n};\n\nexports.safe_uuid = function() {\n    var buf = generateUid(32);\n    stampWithTime(buf, 1000, 3);\n    return encode(buf);\n};\n\n\n\n/*\n    Try to generate a small UID that is unique against chance in the same millisecond second and in a specific context (eg in the same choreography execution)\n    The id contains around 6*8 = 48  bits of randomness and are unique at the level of milliseconds\n    This method is safe on a single computer but should be used with care otherwise\n    This UUID is not cryptographically safe (can be guessed)\n */\nexports.short_uuid = function(callback) {\n    require('crypto').randomBytes(12, function (err, buf) {\n        if (err) {\n            callback(err);\n            return;\n        }\n        stampWithTime(buf,1,2);\n        callback(null, encode(buf));\n    });\n};","const crypto = require('crypto');\nconst Queue = require(\"./Queue\");\nvar PSKBuffer = typeof $$ !== \"undefined\" && $$.PSKBuffer ? $$.PSKBuffer : Buffer;\n\nfunction UidGenerator(minBuffers, buffersSize) {\n\tvar buffers = new Queue();\n\tvar lowLimit = .2;\n\n\tfunction fillBuffers(size){\n\t\t//notifyObserver();\n\t\tconst sz = size || minBuffers;\n\t\tif(buffers.length < Math.floor(minBuffers*lowLimit)){\n\t\t\tfor(var i=0+buffers.length; i < sz; i++){\n\t\t\t\tgenerateOneBuffer(null);\n\t\t\t}\n\t\t}\n\t}\n\n\tfillBuffers();\n\n\tfunction generateOneBuffer(b){\n\t\tif(!b){\n\t\t\tb = PSKBuffer.alloc(0);\n\t\t}\n\t\tconst sz = buffersSize - b.length;\n\t\t/*crypto.randomBytes(sz, function (err, res) {\n\t\t\tbuffers.push(Buffer.concat([res, b]));\n\t\t\tnotifyObserver();\n\t\t});*/\n\t\tbuffers.push(PSKBuffer.concat([ crypto.randomBytes(sz), b ]));\n\t\tnotifyObserver();\n\t}\n\n\tfunction extractN(n){\n\t\tvar sz = Math.floor(n / buffersSize);\n\t\tvar ret = [];\n\n\t\tfor(var i=0; i<sz; i++){\n\t\t\tret.push(buffers.pop());\n\t\t\tsetTimeout(generateOneBuffer, 1);\n\t\t}\n\n\n\n\t\tvar remainder = n % buffersSize;\n\t\tif(remainder > 0){\n\t\t\tvar front = buffers.pop();\n\t\t\tret.push(front.slice(0,remainder));\n\t\t\t//generateOneBuffer(front.slice(remainder));\n\t\t\tsetTimeout(function(){\n\t\t\t\tgenerateOneBuffer(front.slice(remainder));\n\t\t\t},1);\n\t\t}\n\n\t\t//setTimeout(fillBuffers, 1);\n\n\t\treturn Buffer.concat(ret);\n\t}\n\n\tvar fillInProgress = false;\n\n\tthis.generateUid = function(n){\n\t\tvar totalSize = buffers.length * buffersSize;\n\t\tif(n <= totalSize){\n\t\t\treturn extractN(n);\n\t\t} else {\n\t\t\tif(!fillInProgress){\n\t\t\t\tfillInProgress = true;\n\t\t\t\tsetTimeout(function(){\n\t\t\t\t\tfillBuffers(Math.floor(minBuffers*2.5));\n\t\t\t\t\tfillInProgress = false;\n\t\t\t\t}, 1);\n\t\t\t}\n\t\t\treturn crypto.randomBytes(n);\n\t\t}\n\t};\n\n\tvar observer;\n\tthis.registerObserver = function(obs){\n\t\tif(observer){\n\t\t\tconsole.error(new Error(\"One observer allowed!\"));\n\t\t}else{\n\t\t\tif(typeof obs == \"function\"){\n\t\t\t\tobserver = obs;\n\t\t\t\t//notifyObserver();\n\t\t\t}\n\t\t}\n\t};\n\n\tfunction notifyObserver(){\n\t\tif(observer){\n\t\t\tvar valueToReport = buffers.length*buffersSize;\n\t\t\tsetTimeout(function(){\n\t\t\t\tobserver(null, {\"size\": valueToReport});\n\t\t\t}, 10);\n\t\t}\n\t}\n}\n\nmodule.exports.createUidGenerator = function (minBuffers, bufferSize) {\n\treturn new UidGenerator(minBuffers, bufferSize);\n};\n","var mq = $$.require(\"foldermq\");\n\nconst path = require('path');\nconst child_process = require(\"child_process\");\nconst fs = require('fs');\n\nconst RESTART_TIMEOUT = 500;\nconst RESTART_TIMEOUT_LIMIT = 50000;\n\nvar sandboxes = {};\nvar exitHandler = require(\"../../utils/exitHandler\")(sandboxes);\n\nvar bootSandBox = $$.flow.describe(\"PrivateSky.swarm.engine.bootInLauncher\", {\n    boot:function(sandBox, spaceName, folder, codeFolder, callback){\n        // console.log(\"Booting in \", folder, \" context \", spaceName);\n\n        this.callback   = callback;\n        this.folder     = folder;\n        this.spaceName  = spaceName;\n        this.sandBox    = sandBox;\n        this.codeFolder    = codeFolder;\n        this.timeoutMultiplier = 1;\n\n        var task = this.serial(this.ensureFoldersExists);\n\n        task.folderShouldExist(path.join(this.folder, \"mq\"),    task.progress);\n        task.folderShouldExist(path.join(this.folder, \"code\"),  task.progress);\n        task.folderShouldExist(path.join(this.folder, \"tmp\"),   task.progress);\n    },\n    folderShouldExist:  function(path, progress){\n        fs.mkdir(path, {recursive: true}, progress);\n    },\n    copyFolder: function(sourcePath, targetPath, callback){\n        let fsExt = require(\"utils\").fsExt;\n        fsExt.copy(sourcePath, targetPath, {overwrite: true}, callback);\n    },\n    ensureFoldersExists: function(err, res){\n        if(err){\n            console.log(err);\n        } else {\n            var task = this.parallel(this.runCode);\n            task.copyFolder(path.join(this.codeFolder, \"bundles\"), path.join(this.folder, \"bundles\"), task.progress);\n            this.sandBox.inbound = mq.createQue(path.join(this.folder, \"mq/inbound\"), task.progress);\n            this.sandBox.outbound = mq.createQue(path.join(this.folder, \"mq/outbound\"), task.progress);\n        }\n\n    },\n    runCode: function(err, res){\n        if(!err){\n            var mainFile = path.join(process.env.PRIVATESKY_ROOT_FOLDER, \"core\", \"sandboxes\", \"agentSandbox.js\");\n            var args = [this.spaceName, process.env.PRIVATESKY_ROOT_FOLDER, path.resolve(process.env.PRIVATESKY_DOMAIN_BUILD)];\n            var opts = {stdio: [0, 1, 2, \"ipc\"]};\n\n            var startChild = (mainFile, args, opts) => {\n\t\t\t\tconsole.log(\"Running: \", mainFile, args, opts);\n\t\t\t\tvar child = child_process.fork(mainFile, args);\n\t\t\t\tsandboxes[this.spaceName] = child;\n\n\t\t\t\tthis.sandBox.inbound.setIPCChannel(child);\n\t\t\t\tthis.sandBox.outbound.setIPCChannel(child);\n\n\t\t\t\tchild.on(\"exit\", (code, signal)=>{\n\t\t\t\t    if(code === 0){\n\t\t\t\t        console.log(`Sandbox <${this.spaceName}> shutting down.`);\n\t\t\t\t        return;\n                    }\n\t\t\t\t    let timeout = (this.timeoutMultiplier*RESTART_TIMEOUT) % RESTART_TIMEOUT_LIMIT;\n\t\t\t\t    console.log(`Sandbox <${this.spaceName}> exits with code ${code}. Restarting in ${timeout} ms.`);\n\t\t\t\t\tsetTimeout(()=>{\n\t\t\t\t\t\tstartChild(mainFile, args, opts);\n                        this.timeoutMultiplier *= 1.5;\n                    }, timeout);\n\t\t\t\t});\n\n\t\t\t\treturn child;\n            };\n\n            this.callback(null, startChild(mainFile, args, opts));\n        } else {\n            console.log(\"Error executing sandbox!:\", err);\n            this.callback(err, null);\n        }\n    }\n\n});\n\nfunction SandBoxHandler(spaceName, folder, codeFolder, resultCallBack){\n\n    var self = this;\n    var mqHandler;\n\n\n    bootSandBox().boot(this, spaceName,folder, codeFolder, function(err, childProcess){\n        if(!err){\n            self.childProcess = childProcess;\n\n\n            /*self.outbound.registerConsumer(function(err, swarm){\n                $$.PSK_PubSub.publish($$.CONSTANTS.SWARM_FOR_EXECUTION, swarm);\n            });*/\n\n            self.outbound.registerAsIPCConsumer(function(err, swarm){\n                $$.PSK_PubSub.publish($$.CONSTANTS.SWARM_FOR_EXECUTION, swarm);\n            });\n\n            mqHandler = self.inbound.getHandler();\n            if(pendingMessages.length){\n                pendingMessages.map(function(item){\n                    self.send(item);\n                });\n                pendingMessages = null;\n            }\n        }\n    });\n\n    var pendingMessages = [];\n\n    this.send = function (swarm, callback) {\n        if(mqHandler){\n            mqHandler.sendSwarmForExecution(swarm, callback);\n        } else {\n            pendingMessages.push(swarm); //TODO: well, a deep clone will not be a better idea?\n        }\n    }\n\n}\n\n\nfunction SandBoxManager(sandboxesFolder, codeFolder, callback){\n    var self = this;\n\n    var sandBoxes = {\n\n    };\n    function belongsToReplicatedSpace(){\n        return true;\n    }\n\n    //console.log(\"Subscribing to:\", $$.CONSTANTS.SWARM_FOR_EXECUTION);\n    $$.PSK_PubSub.subscribe($$.CONSTANTS.SWARM_FOR_EXECUTION, function(swarm){\n        console.log(\"Executing in sandbox towards: \", swarm.meta.target);\n\n        if(swarm.meta.target == \"system\" || swarm.meta.command == \"asyncReturn\"){\n            $$.swarmsInstancesManager.revive_swarm(swarm);\n            //$$.swarms.restart(swarm.meta.swarmTypeName, swarm);\n        } else\n        if(swarm.meta.target == \"pds\"){\n            //\n        } else\n        if(belongsToReplicatedSpace(swarm.meta.target)){\n            self.pushToSpaceASwarm(swarm.meta.target, swarm);\n        } else {\n            //TODO: send towards network\n        }\n\n    });\n\n\n    function startSandBox(spaceName){\n        var sandBox = new SandBoxHandler(spaceName, path.join(sandboxesFolder, spaceName), codeFolder);\n        sandBoxes[spaceName] = sandBox;\n        return sandBox;\n    }\n\n\n    this.pushToSpaceASwarm = function(spaceName, swarm, callback){\n\n        console.log(\"pushToSpaceASwarm \" , spaceName);\n        var sandbox = sandBoxes[spaceName];\n        if(!sandbox){\n            sandbox = sandBoxes[spaceName] = startSandBox(spaceName);\n        }\n        sandbox.send(swarm, callback);\n    }\n\n    callback(null, this);\n}\n\n\nexports.create = function(folder, codeFolder, callback){\n    new SandBoxManager(folder, codeFolder, callback);\n};\n\n\n","const events = [\"exit\", \"SIGINT\", \"SIGUSR1\", \"SIGUSR2\", \"uncaughtException\", \"SIGTERM\", \"SIGHUP\"];\n\nmodule.exports = function manageShutdownProcess(childrenList){\n\n    function handler(){\n        console.log(\"Handling exit event on\", process.pid, \"arguments:\", arguments);\n        var childrenNames = Object.keys(childrenList);\n        for(let j=0; j<childrenNames.length; j++){\n            var child = childrenList[childrenNames[j]];\n            console.log(`[${process.pid}]`, \"Sending kill signal to PID:\", child.pid);\n            process.kill(child.pid);\n        }\n\n        setTimeout(()=>{\n            process.exit(0);\n        }, 0);\n    }\n\n    process.stdin.resume();\n    for(let i=0; i<events.length; i++){\n        var eventType = events[i];\n        process.on(eventType, handler);\n    }\n    //console.log(\"Exit handler setup!\", `[${process.pid}]`);\n};","\n//var path = require(\"path\");\nfunction defaultErrorHandlingImplementation(err, res){\n\t//console.log(err.stack);\n\tif(err) throw err;\n\treturn res;\n}\n\nrequire(\"./lib/overwriteRequire\");\nconst PSKBuffer = require('pskbuffer');\n$$.PSKBuffer = PSKBuffer;\n\n\n$$.errorHandler = {\n        error:function(err, args, msg){\n            console.log(err, \"Unknown error from function call with arguments:\", args, \"Message:\", msg);\n        },\n        throwError:function(err, args, msg){\n            console.log(err, \"Unknown error from function call with arguments:\", args, \"Message:\", msg);\n            throw err;\n        },\n        ignorePossibleError: function(name){\n            console.log(name);\n        },\n        syntaxError:function(property, swarm, text){\n            //throw new Error(\"Misspelled member name or other internal error!\");\n            var swarmName;\n            try{\n                if(typeof swarm == \"string\"){\n                    swarmName = swarm;\n                } else\n                if(swarm && swarm.meta){\n                    swarmName  = swarm.meta.swarmTypeName;\n                } else {\n                    swarmName = swarm.getInnerValue().meta.swarmTypeName;\n                }\n            } catch(err){\n                swarmName = err.toString();\n            }\n            if(property){\n                console.log(\"Wrong member name \", property,  \" in swarm \", swarmName);\n                if(text) {\n                    console.log(text);\n                }\n            } else {\n                console.log(\"Unknown swarm\", swarmName);\n            }\n\n        },\n        warning:function(msg){\n            console.log(msg);\n        }\n    };\n\n\n\n$$.safeErrorHandling = function(callback){\n        if(callback){\n            return callback;\n        } else{\n            return defaultErrorHandlingImplementation;\n        }\n    };\n\n$$.__intern = {\n        mkArgs:function(args,pos){\n            var argsArray = [];\n            for(var i = pos; i < args.length; i++){\n                argsArray.push(args[i]);\n            }\n            return argsArray;\n        }\n    };\n\n\n\nvar swarmUtils = require(\"./lib/choreographies/utilityFunctions/swarm\");\nvar assetUtils = require(\"./lib/choreographies/utilityFunctions/asset\");\n$$.defaultErrorHandlingImplementation = defaultErrorHandlingImplementation;\n\nvar callflowModule = require(\"./lib/swarmDescription\");\n$$.callflows        = callflowModule.createSwarmEngine(\"callflow\");\n$$.callflow         = $$.callflows;\n$$.flow             = $$.callflows;\n$$.flows            = $$.callflows;\n\n$$.swarms           = callflowModule.createSwarmEngine(\"swarm\", swarmUtils);\n$$.swarm            = $$.swarms;\n$$.contracts        = callflowModule.createSwarmEngine(\"contract\", swarmUtils);\n$$.contract         = $$.contracts;\n$$.assets           = callflowModule.createSwarmEngine(\"asset\", assetUtils);\n$$.asset            = $$.assets;\n$$.transactions     = callflowModule.createSwarmEngine(\"transaction\", swarmUtils);\n$$.transaction      = $$.transactions;\n\n\n$$.PSK_PubSub = require(\"soundpubsub\").soundPubSub;\n\n$$.securityContext = \"system\";\n$$.libraryPrefix = \"global\";\n$$.libraries = {\n    global:{\n\n    }\n};\n\n$$.interceptor = require(\"./lib/InterceptorRegistry\").createInterceptorRegistry();\n\n$$.loadLibrary = require(\"./lib/loadLibrary\").loadLibrary;\n\nrequireLibrary = function(name){\n    //var absolutePath = path.resolve(  $$.__global.__loadLibraryRoot + name);\n    return $$.loadLibrary(name,name);\n};\n\nrequire(\"./constants\");\n\n/*//TODO: SHOULD be moved in $$.__globals\n$$.ensureFolderExists = function (folder, callback) {\n    const flow = $$.flow.start(\"utils.mkDirRec\");\n    flow.make(folder, callback);\n};\n\n$$.ensureLinkExists = function (existingPath, newPath, callback) {\n    const flow = $$.flow.start(\"utils.mkDirRec\");\n    flow.makeLink(existingPath, newPath, callback);\n};*/\n\n$$.pathNormalize = function (pathToNormalize) {\n    const path = require(\"path\");\n    pathToNormalize = path.normalize(pathToNormalize);\n\n    return pathToNormalize.replace(/[\\/\\\\]/g, path.sep);\n};\n\nmodule.exports = {\n    \t\t\t\tcreateSwarmEngine: require(\"./lib/swarmDescription\").createSwarmEngine,\n                    createJoinPoint: require(\"./lib/parallelJoinPoint\").createJoinPoint,\n                    createSerialJoinPoint: require(\"./lib/serialJoinPoint\").createSerialJoinPoint,\n                    swarmInstanceManager: require(\"./lib/choreographies/swarmInstancesManager\"),\n                    enableInternalSwarmRouting: function(){\n                        function dummyVM(name){\n                            function solveSwarm(swarm){\n                                $$.swarmsInstancesManager.revive_swarm(swarm);\n                            }\n\n                            $$.PSK_PubSub.subscribe(name, solveSwarm);\n                            console.log(\"Creating a fake execution context...\");\n                        }\n                        dummyVM($$.CONSTANTS.SWARM_FOR_EXECUTION);\n                    }\n\t\t\t\t};\n","if(typeof singleton_container_module_workaround_for_wired_node_js_caching == 'undefined') {\n    singleton_container_module_workaround_for_wired_node_js_caching   = module;\n} else {\n    module.exports = singleton_container_module_workaround_for_wired_node_js_caching .exports;\n    return module;\n}\n\n/**\n * Created by salboaie on 4/27/15.\n */\nfunction Container(errorHandler){\n    var things = {};        //the actual values for our services, things\n    var immediate = {};     //how dependencies were declared\n    var callbacks = {};     //callback that should be called for each dependency declaration\n    var depsCounter = {};   //count dependencies\n    var reversedTree = {};  //reversed dependencies, opposite of immediate object\n\n     this.dump = function(){\n         console.log(\"Conatiner dump\\n Things:\", things, \"\\nDeps counter: \", depsCounter, \"\\nStright:\", immediate, \"\\nReversed:\", reversedTree);\n     };\n\n    function incCounter(name){\n        if(!depsCounter[name]){\n            depsCounter[name] = 1;\n        } else {\n            depsCounter[name]++;\n        }\n    }\n\n    function insertDependencyinRT(nodeName, dependencies){\n        dependencies.forEach(function(itemName){\n            var l = reversedTree[itemName];\n            if(!l){\n                l = reversedTree[itemName] = {};\n            }\n            l[nodeName] = nodeName;\n        });\n    }\n\n\n    function discoverUpNodes(nodeName){\n        var res = {};\n\n        function DFS(nn){\n            var l = reversedTree[nn];\n            for(var i in l){\n                if(!res[i]){\n                    res[i] = true;\n                    DFS(i);\n                }\n            }\n        }\n\n        DFS(nodeName);\n        return Object.keys(res);\n    }\n\n    function resetCounter(name){\n        var dependencyArray = immediate[name];\n        var counter = 0;\n        if(dependencyArray){\n            dependencyArray.forEach(function(dep){\n                if(things[dep] == null){\n                    incCounter(name);\n                    counter++;\n                }\n            });\n        }\n        depsCounter[name] = counter;\n        //console.log(\"Counter for \", name, ' is ', counter);\n        return counter;\n    }\n\n    /* returns those that are ready to be resolved*/\n    function resetUpCounters(name){\n        var ret = [];\n        //console.log('Reseting up counters for ', name, \"Reverse:\", reversedTree[name]);\n        var ups = reversedTree[name];\n        for(var v in ups){\n            if(resetCounter(v) === 0){\n                ret.push(v);\n            }\n        }\n        return ret;\n    }\n\n    /*\n         The first argument is a name for a service, variable,a  thing that should be initialised, recreated, etc\n         The second argument is an array with dependencies\n         the last argument is a function(err,...) that is called when dependencies are ready or recalled when are not ready (stop was called)\n         If err is not undefined it means that one or any undefined variables are not ready and the callback will be called again later\n         All the other arguments are the corresponding arguments of the callback will be the actual values of the corresponding dependency\n         The callback functions should return the current value (or null)\n     */\n    this.declareDependency = function(name, dependencyArray, callback){\n        if(callbacks[name]){\n            errorHandler.ignorePossibleError(\"Duplicate dependency:\" + name);\n        } else {\n            callbacks[name] = callback;\n            immediate[name]   = dependencyArray;\n            insertDependencyinRT(name, dependencyArray);\n            things[name] = null;\n        }\n\n        var unsatisfiedCounter = resetCounter(name);\n        if(unsatisfiedCounter === 0 ){\n            callForThing(name, false);\n        } else {\n            callForThing(name, true);\n        }\n    };\n\n\n    /*\n        create a service\n     */\n    this.service = function(name, dependencyArray, constructor){\n        this.declareDependency(name, dependencyArray, constructor);\n    };\n\n\n    var subsystemCounter = 0;\n    /*\n     create a anonymous subsystem\n     */\n    this.subsystem = function(dependencyArray, constructor){\n        subsystemCounter++;\n        this.declareDependency(\"dicontainer_subsystem_placeholder\" + subsystemCounter, dependencyArray, constructor);\n    };\n\n    /* not documented.. limbo state*/\n    this.factory = function(name, dependencyArray, constructor){\n        this.declareDependency(name, dependencyArray, function(){\n            return new constructor();\n        });\n    };\n\n    function callForThing(name, outOfService){\n        var args = immediate[name].map(function(item){\n            return things[item];\n        });\n        args.unshift(outOfService);\n        try{\n            var value = callbacks[name].apply({},args);\n        } catch(err){\n            errorHandler.throwError(err);\n        }\n\n\n        if(outOfService || value===null){   //enable returning a temporary dependency resolution!\n            if(things[name]){\n                things[name] = null;\n                resetUpCounters(name);\n            }\n        } else {\n            //console.log(\"Success resolving \", name, \":\", value, \"Other ready:\", otherReady);\n            if(!value){\n                value =  {\"placeholder\": name};\n            }\n            things[name] = value;\n            var otherReady = resetUpCounters(name);\n            otherReady.forEach(function(item){\n                callForThing(item, false);\n            });\n        }\n    }\n\n    /*\n        Declare that a name is ready, resolved and should try to resolve all other waiting for it\n     */\n    this.resolve    = function(name, value){\n        things[name] = value;\n        var otherReady = resetUpCounters(name);\n\n        otherReady.forEach(function(item){\n            callForThing(item, false);\n        });\n    };\n\n\n\n    this.instanceFactory = function(name, dependencyArray, constructor){\n        errorHandler.notImplemented(\"instanceFactory is planned but not implemented\");\n    };\n\n    /*\n        Declare that a service or feature is not working properly. All services depending on this will get notified\n     */\n    this.outOfService    = function(name){\n        things[name] = null;\n        var upNodes = discoverUpNodes(name);\n        upNodes.forEach(function(node){\n            things[name] = null;\n            callForThing(node, true);\n        });\n    };\n}\n\n\nexports.newContainer    = function(checksLibrary){\n    return new Container(checksLibrary);\n};\n\n//exports.container = new Container($$.errorHandler);","exports.domainPubSub = require(\"./domainPubSub\");","\n/**\n * Generic function used to registers methods such as asserts, logging, etc. on the current context.\n * @param name {String)} - name of the method (use case) to be registered.\n * @param func {Function} - handler to be invoked.\n * @param paramsDescription {Object} - parameters descriptions\n * @param after {Function} - callback function to be called after the function has been executed.\n */\nfunction addUseCase(name, func, paramsDescription, after){\n    var newFunc = func;\n    if(typeof after === \"function\") {\n        newFunc = function(){\n            const args = Array.from(arguments);\n            func.apply(this, args);\n            after();\n        };\n    }\n\n    // some properties should not be overridden\n    const protectedProperties = [ 'addCheck', 'addCase', 'register' ];\n    if(protectedProperties.indexOf(name) === -1){\n        this[name] = newFunc;\n    } else {\n        throw new Error('Cant overwrite ' + name);\n    }\n\n    if(paramsDescription){\n        this.params[name] = paramsDescription;\n    }\n}\n\n/**\n * Creates an alias to an existing function.\n * @param name1 {String} - New function name.\n * @param name2 {String} - Existing function name.\n */\nfunction alias(name1, name2){\n    this[name1] = this[name2];\n}\n\n/**\n * Singleton for adding various functions for use cases regarding logging.\n * @constructor\n */\nfunction LogsCore(){\n    this.params = {};\n}\n\n/**\n * Singleton for adding your various functions for asserts.\n * @constructor\n */\nfunction AssertCore(){\n    this.params = {};\n}\n\n/**\n * Singleton for adding your various functions for checks.\n * @constructor\n */\nfunction CheckCore(){\n    this.params = {};\n}\n\n/**\n * Singleton for adding your various functions for generating exceptions.\n * @constructor\n */\nfunction ExceptionsCore(){\n    this.params = {};\n}\n\n/**\n * Singleton for adding your various functions for running tests.\n * @constructor\n */\nfunction TestRunnerCore(){\n}\n\nLogsCore.prototype.addCase           = addUseCase;\nAssertCore.prototype.addCheck        = addUseCase;\nCheckCore.prototype.addCheck         = addUseCase;\nExceptionsCore.prototype.register    = addUseCase;\n\nLogsCore.prototype.alias             = alias;\nAssertCore.prototype.alias           = alias;\nCheckCore.prototype.alias            = alias;\nExceptionsCore.prototype.alias       = alias;\n\n// Create modules\nvar assertObj       = new AssertCore();\nvar checkObj        = new CheckCore();\nvar exceptionsObj   = new ExceptionsCore();\nvar loggerObj       = new LogsCore();\nvar testRunnerObj   = new TestRunnerCore();\n\n// Export modules\nexports.assert      = assertObj;\nexports.check       = checkObj;\nexports.exceptions  = exceptionsObj;\nexports.logger      = loggerObj;\nexports.testRunner  = testRunnerObj;\n\n// Initialise modules\nrequire(\"./standardAsserts.js\").init(exports, loggerObj);\nrequire(\"./standardLogs.js\").init(exports);\nrequire(\"./standardExceptions.js\").init(exports);\nrequire(\"./standardChecks.js\").init(exports);\nrequire(\"./testRunner.js\").init(exports);\n\n// Global Uncaught Exception handler.\nif(process.on)\n{\n    process.on('uncaughtException', function (err) {\n\t\tconst tag = \"uncaughtException\";\n\t\tconsole.log(tag, err);\n\t\tconsole.log(tag, err.stack);\n\t});\n}","module.exports = {\n\t\t\t\t\tcreateQue: require(\"./lib/folderMQ\").getFolderQueue\n\t\t\t\t\t//folderMQ: require(\"./lib/folderMQ\")\n};","//console.log(require.resolve(\"./components.js\"));\nmodule.exports = $$.library(function(){\n\trequire(\"./components.js\");\n\t/*require(\"./mkDirRec.js\");*/\n})","const PSKBuffer = require('./lib/PSKBuffer');\n\nmodule.exports = PSKBuffer;\n","const PskCrypto = require(\"./lib/PskCrypto\");\nconst ssutil = require(\"./signsensusDS/ssutil\");\n\nmodule.exports = PskCrypto;\n\nmodule.exports.hashValues = ssutil.hashValues;\n\nmodule.exports.PskArchiver = require(\"./lib/psk-archiver\");\n\nmodule.exports.DuplexStream = require(\"./lib/utils/DuplexStream\");\n\nmodule.exports.isStream = require(\"./lib/utils/isStream\");","module.exports = {\n\t\t\t\t\tsoundPubSub: require(\"./lib/soundPubSub\").soundPubSub\n};","module.exports.OwM = require(\"./lib/OwM\");\nmodule.exports.beesHealer = require(\"./lib/beesHealer\");\n\nconst uidGenerator = require(\"./lib/uidGenerator\").createUidGenerator(200, 32);\n\nmodule.exports.safe_uuid = require(\"./lib/safe-uuid\").init(uidGenerator);\n\nmodule.exports.Queue = require(\"./lib/Queue\");\nmodule.exports.combos = require(\"./lib/Combos\");\n\nmodule.exports.uidGenerator = uidGenerator;\nmodule.exports.generateUid = uidGenerator.generateUid;\n\nmodule.exports.createPskConsole = function () {\n  return require('./lib/pskconsole');\n};\n\n\nif(typeof global.$$ == \"undefined\"){\n  global.$$ = {};\n}\n\nif(typeof global.$$.uidGenerator == \"undefined\"){\n    $$.uidGenerator = module.exports.safe_uuid;\n}\n","exports.fsExt = require(\"./FSExtension\").fsExt;"]} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","builds/tmp/pskruntime.js","builds/tmp/pskruntime_intermediar.js","libraries/domainBase/domainPubSub.js","libraries/launcher/components.js","libraries/utils/FSExtension.js","modules/callflow/constants.js","modules/callflow/lib/InterceptorRegistry.js","modules/callflow/lib/choreographies/SwarmDebug.js","modules/callflow/lib/choreographies/swarmInstancesManager.js","modules/callflow/lib/choreographies/utilityFunctions/asset.js","modules/callflow/lib/choreographies/utilityFunctions/base.js","modules/callflow/lib/choreographies/utilityFunctions/callflow.js","modules/callflow/lib/choreographies/utilityFunctions/swarm.js","modules/callflow/lib/loadLibrary.js","modules/callflow/lib/overwriteRequire.js","modules/callflow/lib/parallelJoinPoint.js","modules/callflow/lib/serialJoinPoint.js","modules/callflow/lib/swarmDescription.js","modules/double-check/lib/standardAsserts.js","modules/double-check/lib/standardChecks.js","modules/double-check/lib/standardExceptions.js","modules/double-check/lib/standardLogs.js","modules/double-check/lib/testRunner.js","modules/double-check/lib/utils/glob-to-regexp.js","modules/foldermq/lib/folderMQ.js","modules/pskbuffer/lib/PSKBuffer.js","modules/pskcrypto/lib/ECDSA.js","modules/pskcrypto/lib/PskCrypto.js","modules/pskcrypto/lib/asn1/api.js","modules/pskcrypto/lib/asn1/asn1.js","modules/pskcrypto/lib/asn1/base/buffer.js","modules/pskcrypto/lib/asn1/base/index.js","modules/pskcrypto/lib/asn1/base/node.js","modules/pskcrypto/lib/asn1/base/reporter.js","modules/pskcrypto/lib/asn1/bignum/bn.js","modules/pskcrypto/lib/asn1/constants/der.js","modules/pskcrypto/lib/asn1/constants/index.js","modules/pskcrypto/lib/asn1/decoders/der.js","modules/pskcrypto/lib/asn1/decoders/index.js","modules/pskcrypto/lib/asn1/decoders/pem.js","modules/pskcrypto/lib/asn1/encoders/der.js","modules/pskcrypto/lib/asn1/encoders/index.js","modules/pskcrypto/lib/asn1/encoders/pem.js","modules/pskcrypto/lib/keyEncoder.js","modules/pskcrypto/lib/psk-archiver.js","modules/pskcrypto/lib/utils/DuplexStream.js","modules/pskcrypto/lib/utils/PassThroughStream.js","modules/pskcrypto/lib/utils/countFiles.js","modules/pskcrypto/lib/utils/cryptoUtils.js","modules/pskcrypto/lib/utils/isStream.js","modules/pskcrypto/signsensusDS/ssutil.js","modules/soundpubsub/lib/soundPubSub.js","modules/swarmutils/lib/Combos.js","modules/swarmutils/lib/OwM.js","modules/swarmutils/lib/Queue.js","modules/swarmutils/lib/beesHealer.js","modules/swarmutils/lib/pskconsole.js","modules/swarmutils/lib/safe-uuid.js","modules/swarmutils/lib/uidGenerator.js","psknode/core/sandboxes/util/SandBoxManager.js","psknode/core/utils/exitHandler.js","modules/callflow/index.js","modules/dicontainer/lib/container.js","libraries/domainBase/index.js","modules/double-check/lib/checksCore.js","modules/foldermq/index.js","libraries/launcher/index.js","modules/pskbuffer/index.js","modules/pskcrypto/index.js","modules/soundpubsub/index.js","modules/swarmutils/index.js","libraries/utils/index.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AC7bA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACXA;AACA;AACA;;ACFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AC3RA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3HA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChWA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC7OA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AC7pBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5eA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AChDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AC1UA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpHA;AACA;AACA;AACA;AACA;AACA;AACA;;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1lBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9wEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnSA;AACA;AACA;AACA;AACA;;ACJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChRA;AACA;AACA;AACA;AACA;;ACJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AClKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACpaA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACjJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AC/LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3MA;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtHA;AACA;AACA;AACA;;ACHA;AACA;AACA;AACA;AACA;;ACJA;AACA;AACA;AACA;;ACHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACXA;AACA;AACA;;;ACFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACzBA","file":"generated.js","sourceRoot":"","sourcesContent":["(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})()","require(\"../../modules/callflow/lib/overwriteRequire\")\n\nrequire(\"./pskruntime_intermediar\");\n\nrequire(\"callflow\");\n\nconsole.log(\"Loading runtime: callflow module ready\");","global.pskruntimeLoadModules = function(){ \n\t$$.__runtimeModules[\"callflow\"] = require(\"callflow\");\n\t$$.__runtimeModules[\"launcher\"] = require(\"launcher\");\n\t$$.__runtimeModules[\"double-check\"] = require(\"double-check\");\n\t$$.__runtimeModules[\"pskcrypto\"] = require(\"pskcrypto\");\n\t$$.__runtimeModules[\"dicontainer\"] = require(\"dicontainer\");\n\t$$.__runtimeModules[\"swarmutils\"] = require(\"swarmutils\");\n\t$$.__runtimeModules[\"soundpubsub\"] = require(\"soundpubsub\");\n\t$$.__runtimeModules[\"pskbuffer\"] = require(\"pskbuffer\");\n\t$$.__runtimeModules[\"foldermq\"] = require(\"foldermq\");\n\t$$.__runtimeModules[\"domainBase\"] = require(\"domainBase\");\n\t$$.__runtimeModules[\"utils\"] = require(\"utils\");\n}\nif (false) {\n\tpskruntimeLoadModules();\n}; \nglobal.pskruntimeRequire = require;\nif (typeof $$ !== \"undefined\") {            \n    $$.requireBundle(\"pskruntime\");\n};","var pubSub = $$.require(\"soundpubsub\").soundPubSub;\nconst path = require(\"path\");\nconst fs = require(\"fs\");\n\nexports.create = function(folder, codeFolder ){\n\n    $$.PSK_PubSub = pubSub;\n    var sandBoxesRoot = path.join(folder, \"sandboxes\");\n\n    try{\n        fs.mkdirSync(sandBoxesRoot, {recursive: true});\n    }catch(err){\n        console.log(\"Failed to create sandboxes dir structure!\", err);\n        //TODO: maybe it is ok to call process.exit ???\n    }\n\n    $$.SandBoxManager = require(\"../../psknode/core/sandboxes/util/SandBoxManager\").create(sandBoxesRoot, codeFolder, function(err, res){\n        console.log($$.DI_components.sandBoxReady, err, res);\n        $$.container.resolve($$.DI_components.sandBoxReady, true);\n    });\n\n    return pubSub;\n};\n","$$.DI_components = {\n   swarmIsReady:\"SwarmIsReady\",\n   configLoaded:\"configLoaded\",\n   sandBoxReady:\"SandBoxReady\",\n   localNodeAPIs:\"localNodeAPIs\"\n}\n","const fs = require(\"fs\");\nconst path = require(\"path\");\nconst os = require(\"os\");\nconst child_process = require('child_process');\nconst crypto = require('crypto');\n\n// if this is set to true, the logs will be available. Default (false)\nconst DEBUG =  process.env.DEPLOYER_DEBUG || false;\n\nfunction FSExtention(){\n\n    /**\n     * Base path used to resolve all relative paths in the actions bellow.\n     * Default is set to two levels up from the current directory. This can be changed using __setBasePath.\n     * @type {*|string}\n     */\n    var basePath = path.join(__dirname, \"../../\");\n\n    /**\n     * Set the base path to a different absolute directory path.\n     * @param wd {String} absolute directory path.\n     * @private\n     */\n    var __setBasePath = function(wd) {\n        basePath = path.resolve(wd);\n    }\n\n    /**\n     * Resolve path into an absolute path. If filePath is relative, the path is resolved using the basePath as first argument.\n     * @param filePath {String} relative or absolute file path.\n     * @returns {String} absolute path\n     * @private\n     */\n    var __resolvePath = function(filePath) {\n        if(path.isAbsolute(filePath)) {\n            return filePath;\n        }\n\n        return path.resolve(basePath, filePath);\n    }\n\n    /**\n     * If the directory structure does not exist, it is created. Like mkdir -p\n     * @param dir {String} dir path\n     * @private\n     */\n    var __createDir = function(dir) {\n        dir = __resolvePath(dir);\n        if (fs.existsSync(dir)) {\n            log(dir + \" already exist! Continuing ...\")\n            return;\n        }\n\n        var isWin = (os.platform() === 'win32');\n        var cmd = isWin ? \"mkdir \" : \"mkdir -p \";\n\n        child_process.execSync(cmd + \"\\\"\"+dir+\"\\\"\", {stdio:[0,1,2]});\n    }\n\n    /**\n     * Copy a file or directory. The directory can have recursive contents. Like copy -r.\n     * NOTE: If src is a directory it will copy everything inside of the directory, not the entire directory itself.\n     * NOTE: If src is a file, target cannot be a directory.\n     * NOTE: If the destination path structure does not exists, it will be created.\n     * @param src {String} Source file|directory path.\n     * @param dest {String} Destination file|directory path.\n     * @param options {Object} Optional parameters for copy action. Available options:\n     *  - overwrite <Boolean>: overwrite existing file or directory, default is true.\n     *  Note that the copy operation will silently fail if this is set to false and the destination exists.\n     * @param callback {Function}\n     * @private\n     */\n    var __copy = function (src, dest, options, callback) {\n        src = __resolvePath(src);\n        dest = __resolvePath(dest);\n\n        callback = callback || function(){};\n        let rethrow = false;\n\n        try{\n            if (!fs.existsSync(src)) {\n                rethrow = true;\n                throw `Source directory or file \"${src}\" does not exists!`;\n            }\n\n            let srcStat = fs.lstatSync(src);\n            if(srcStat.isDirectory()) {\n                __copyDir(src, dest, options);\n            } else if(srcStat.isFile()) {\n                // destination must be a file too\n                __copyFile(src, dest, options);\n            }\n        } catch (err) {\n            if(rethrow){\n                throw err;\n            }\n            log(err, true);\n            callback(err);\n            return;\n        }\n\n        callback();\n    }\n\n    /**\n     * Copy a directory. The directory can have recursive contents. Like copy -r.\n     * NOTE: Itt will copy everything inside of the directory, not the entire directory itself.\n     * NOTE: If the destination path structure does not exists, it will be created.\n     * @param src {String} Source directory path.\n     * @param dest {String} Destination directory path.\n     * @param options {Object} Optional parameters for copy action. Available options:\n     *  - overwrite <Boolean>: overwrite existing directory, default is true.\n     *  Note that the copy operation will silently fail if this is set to false and the destination exists.\n     * @private\n     */\n    var __copyDir = function(src, dest, options) {\n        src = __resolvePath(src);\n        dest = __resolvePath(dest);\n\n        __createDir(dest);\n\n        var files = fs.readdirSync(src);\n        for(var i = 0; i < files.length; i++) {\n            let current = fs.lstatSync(path.join(src, files[i]));\n            let newSrc = path.join(src, files[i]);\n            let newDest = path.join(dest, files[i]);\n\n            if(current.isDirectory()) {\n                __copyDir(newSrc, newDest, options);\n            } else if(current.isSymbolicLink()) {\n                var symlink = fs.readlinkSync(newSrc);\n                fs.symlinkSync(symlink, newDest);\n            } else {\n                __copyFile(newSrc, newDest, options);\n            }\n        }\n    };\n\n    /**\n     * Copy a file.\n     * NOTE: If src is a file, target cannot be a directory.\n     * NOTE: If the destination path structure does not exists, it will be created.\n     * @param src {String} Source file path.\n     * @param dest {String} Destination file path.\n     * @param options {Object} Optional parameters for copy action. Available options:\n     *  - overwrite <Boolean>: overwrite existing file or directory, default is true.\n     *  Note that the copy operation will silently fail if this is set to false and the destination exists.\n     * @param callback {Function}\n     * @private\n     */\n    var __copyFile = function(src, dest, options) {\n        src = __resolvePath(src);\n        dest = __resolvePath(dest);\n\n        if(options && options.overwrite === false) {\n            if (fs.existsSync(dest)) {\n                // silently fail if overwrite is set to false and the destination exists.\n                let error = `Silent fail - cannot copy. Destination file ${dest} already exists and overwrite option is set to false! Continuing...`;\n                log(error, true);\n                return;\n            }\n        }\n        __createDir(path.dirname(dest));\n\n        var content = fs.readFileSync(src, \"utf8\");\n        fs.writeFileSync(dest, content);\n    }\n\n    /**\n     * Removes a file or directory. The directory can have recursive contents. Like rm -rf\n     * @param src {String} Path\n     * @param callback {Function}\n     * @private\n     */\n    var __remove = function(src, callback) {\n        src = __resolvePath(src);\n\n        callback = callback || function(){};\n\n        log(`Removing ${src}`);\n\n        try{\n            let current = fs.lstatSync(src);\n            if(current.isDirectory()) {\n                __rmDir(src);\n            } else if(current.isFile()) {\n                __rmFile(src);\n            }\n        } catch (err) {\n            if(err.code && err.code === \"ENOENT\"){\n                //ignoring errors like \"file/directory does not exist\"\n                err = null;\n            }else{\n                log(err, true);\n            }\n            callback(err);\n            return;\n        }\n\n        callback();\n    }\n\n    /**\n     * Removes a directory. The directory can have recursive contents. Like rm -rf\n     * @param dir {String} Path\n     * @private\n     */\n    var __rmDir = function (dir) {\n        dir = __resolvePath(dir);\n\n        if (!fs.existsSync(dir)) {\n            log(`Directory ${dir} does not exist!`, true);\n            return;\n        }\n\n        var list = fs.readdirSync(dir);\n        for (var i = 0; i < list.length; i++) {\n            var filename = path.join(dir, list[i]);\n            var stat = fs.lstatSync(filename);\n\n            if (stat.isDirectory()) {\n                __rmDir(filename, null);\n            } else {\n                // rm filename\n                fs.unlinkSync(filename);\n            }\n        }\n\n        fs.rmdirSync(dir);\n    }\n\n    /**\n     * Removes a file.\n     * @param file {String} Path\n     * @private\n     */\n    var __rmFile = function(file) {\n        file = __resolvePath(file);\n        if (!fs.existsSync(file)) {\n            log(`File ${file} does not exist!`, true);\n            return;\n        }\n\n        fs.unlinkSync(file);\n    }\n\n    /**\n     * Writes data to a file, replacing the file if it already exists.\n     * @param file {String} Path.\n     * @param data {String}\n     * @private\n     */\n    var __createFile = function(file, data, options) {\n        file = __resolvePath(file)\n        fs.writeFileSync(file, data, options);\n    }\n\n    /**\n     * Moves a file or directory.\n     * @param src {String} Source path.\n     * @param dest {String} Destination path.\n     * @param options {Object}. Optional parameters for copy action. Available options:\n     *  - overwrite <boolean>: overwrite existing file or directory, default is false. Note that the move operation will silently fail if you set this to true and the destination exists.\n     * @param callback {Function}\n     * @private\n     */\n    var __move = function(src, dest, options, callback) {\n        src = __resolvePath(src);\n        dest = __resolvePath(dest);\n\n        callback = callback || function(){};\n\n        try {\n            if(options && options.overwrite === false) {\n                if (fs.existsSync(dest)) {\n                    // silently fail if overwrite is set to false and the destination exists.\n                    let error = `Silent fail - cannot move. Destination file ${dest} already exists and overwrite option is set to false! Continuing...`;\n                    log(error, true);\n                    callback();\n                    return;\n                }\n            }\n\n            __copy(src, dest, options);\n            __remove(src);\n        }catch(err) {\n            callback(err);\n            return;\n        }\n        callback();\n    }\n\n    /**\n     * Computes checksum to a file or a directory based on their contents only.\n     * If the source is directory, the checksum is a hash of all concatenated file hashes.\n     * @param src {String} Path of a file or directory.\n     * @param algorithm {String} Hashing algorithm(default: md5). The algorithm is dependent on the available algorithms\n     * supported by the version of OpenSSL on the platform. E.g. 'md5', 'sha256', 'sha512'.\n     * @param encoding {String} Hashing encoding (default: 'hex'). The encoding is dependent on the\n     * available digest algorithms. E.g. 'hex', 'latin1' or 'base64'.\n     * @returns {String} Checksum of the file or directory.\n     * @private\n     */\n    var __checksum = function(src, algorithm, encoding) {\n        src = __resolvePath(src);\n\n        if (!fs.existsSync(src)) {\n            throw `Path ${src} does not exists!`;\n        }\n\n        var checksum = \"\";\n        let current = fs.lstatSync(src);\n        if(current.isDirectory()) {\n            let hashDir = __hashDir(src, algorithm, encoding);\n            checksum = hashDir[\"hash\"];\n        } else if(current.isFile()) {\n            checksum = __hashFile(src, algorithm, encoding);\n        }\n\n        return checksum;\n    }\n\n    /**\n     * Computes hash of a string.\n     * @param str {String}\n     * @param algorithm {String} Hashing algorithm(default: md5). The algorithm is dependent on the available algorithms\n     * supported by the version of OpenSSL on the platform. E.g. 'md5', 'sha256', 'sha512'.\n     * @param encoding {String} Hashing encoding (default: 'hex'). The encoding is dependent on the\n     * available digest algorithms. E.g. 'hex', 'latin1' or 'base64'.\n     * @returns {String} Hash of the string.\n     * @private\n     */\n    var __hash =  function(str, algorithm, encoding) {\n        return crypto\n            .createHash(algorithm || 'md5')\n            .update(str, 'utf8')\n            .digest(encoding || 'hex')\n    }\n\n    /**\n     * Computes hash of a file based on its content only.\n     * @param src {String} Path of a file.\n     * @param algorithm {String} Hashing algorithm(default: md5). The algorithm is dependent on the available algorithms\n     * supported by the version of OpenSSL on the platform. E.g. 'md5', 'sha256', 'sha512'.\n     * @param encoding {String} Hashing encoding (default: 'hex'). The encoding is dependent on the\n     * available digest algorithms. E.g. 'hex', 'latin1' or 'base64'.\n     * @returns {String} Hash of the file.\n     * @private\n     */\n    var __hashFile = function(src, algorithm, encoding) {\n        src = __resolvePath(src);\n        if (!fs.existsSync(src)) {\n            throw `${src} does not exist!`;\n        }\n\n        var content = fs.readFileSync(src, \"utf8\");\n        return __hash(content, algorithm, encoding);\n    }\n\n    /**\n     * Computes hash of a directory based on its content only.\n     * If directory has multiple files, the result is a hash of all concatenated file hashes.\n     * @param src {String} Path of a directory.\n     * @param algorithm {String} Hashing algorithm(default: md5). The algorithm is dependent on the available algorithms\n     * supported by the version of OpenSSL on the platform. E.g. 'md5', 'sha256', 'sha512'.\n     * @param encoding {String} Hashing encoding (default: 'hex'). The encoding is dependent on the\n     * available digest algorithms. E.g. 'hex', 'latin1' or 'base64'.\n     * @returns {String} Hash of the directory.\n     * @private\n     */\n    var __hashDir = function(dir, algorithm, encoding) {\n        dir = __resolvePath(dir);\n        if (!fs.existsSync(dir)) {\n            throw `Directory ${dir} does not exist!`;\n        }\n        var hashes = {};\n        var list = fs.readdirSync(dir);\n        for (var i = 0; i < list.length; i++) {\n            var filename = path.join(dir, list[i]);\n            var stat = fs.lstatSync(filename);\n\n            if (stat.isDirectory()) {\n                let tempHashes = __hashDir(filename, algorithm, encoding);\n                hashes = Object.assign(hashes, tempHashes[\"sub-hashes\"]);\n            } else {\n                let tempHash = __hashFile(filename, algorithm, encoding);\n                hashes[filename] = tempHash;\n            }\n        }\n\n        // compute dir hash\n        let dirContent = Object.keys(hashes).reduce(function (previous, key) {\n            return previous += hashes[key];\n        }, \"\");\n\n        let dirHash = __hash(dirContent, algorithm, encoding);\n\n        return {\n            \"hash\": dirHash,\n            \"sub-hashes\": hashes\n        }\n    }\n\n    /**\n     * Generates a guid (global unique identifier).\n     * @returns {String} Guid in the format xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx\n     * @private\n     */\n    var __guid = function guid() {\n        function _make_group(s) {\n            var p = (Math.random().toString(16)+\"000000000\").substr(2,8);\n            return s ? \"-\" + p.substr(0,4) + \"-\" + p.substr(4,4) : p ;\n        }\n        return _make_group() + _make_group(true) + _make_group(true) + _make_group();\n    }\n\n    /**\n     * Logs wrapper.\n     * @param message {String}\n     * @param isError {Boolean}\n     */\n    function log(message, isError) {\n        let logger = isError ? console.error : console.log;\n\n        if(DEBUG) {\n            logger(message);\n        }\n    }\n\n    return {\n        setBasePath: __setBasePath,\n        resolvePath: __resolvePath,\n        createDir: __createDir,\n        copyDir: __copyDir,\n        rmDir: __rmDir,\n        rmFile: __rmFile,\n        createFile: __createFile,\n        copy: __copy,\n        move: __move,\n        remove: __remove,\n        checksum: __checksum,\n        guid: __guid\n    }\n}\n\nmodule.exports.fsExt = new FSExtention();","$$.CONSTANTS = {\n    SWARM_FOR_EXECUTION:\"swarm_for_execution\",\n    INBOUND:\"inbound\",\n    OUTBOUND:\"outbound\",\n    PDS:\"PrivateDataSystem\",\n    CRL:\"CommunicationReplicationLayer\",\n    SWARM_RETURN: 'swarm_return',\n    BEFORE_INTERCEPTOR: 'before',\n    AFTER_INTERCEPTOR: 'after',\n};\n\n","// related to: SwarmSpace.SwarmDescription.createPhase()\n\nfunction InterceptorRegistry() {\n    const rules = new Map();\n\n    // ??? $$.errorHandler Library ???\n    const _CLASS_NAME = 'InterceptorRegistry';\n\n    /************* PRIVATE METHODS *************/\n\n    function _throwError(err, msg) {\n        console.error(err.message, `${_CLASS_NAME} error message:`, msg);\n        throw err;\n    }\n\n    function _warning(msg) {\n        console.warn(`${_CLASS_NAME} warning message:`, msg);\n    }\n\n    const getWhenOptions = (function () {\n        let WHEN_OPTIONS;\n        return function () {\n            if (WHEN_OPTIONS === undefined) {\n                WHEN_OPTIONS = Object.freeze([\n                    $$.CONSTANTS.BEFORE_INTERCEPTOR,\n                    $$.CONSTANTS.AFTER_INTERCEPTOR\n                ]);\n            }\n            return WHEN_OPTIONS;\n        };\n    })();\n\n    function verifyWhenOption(when) {\n        if (!getWhenOptions().includes(when)) {\n            _throwError(new RangeError(`Option '${when}' is wrong!`),\n                `it should be one of: ${getWhenOptions()}`);\n        }\n    }\n\n    function verifyIsFunctionType(fn) {\n        if (typeof fn !== 'function') {\n            _throwError(new TypeError(`Parameter '${fn}' is wrong!`),\n                `it should be a function, not ${typeof fn}!`);\n        }\n    }\n\n    function resolveNamespaceResolution(swarmTypeName) {\n        if (swarmTypeName === '*') {\n            return swarmTypeName;\n        }\n\n        return (swarmTypeName.includes(\".\") ? swarmTypeName : ($$.libraryPrefix + \".\" + swarmTypeName));\n    }\n\n    /**\n     * Transforms an array into a generator with the particularity that done is set to true on the last element,\n     * not after it finished iterating, this is helpful in optimizing some other functions\n     * It is useful if you want call a recursive function over the array elements but without popping the first\n     * element of the Array or sending the index as an extra parameter\n     * @param {Array<*>} arr\n     * @return {IterableIterator<*>}\n     */\n    function* createArrayGenerator(arr) {\n        const len = arr.length;\n\n        for (let i = 0; i < len - 1; ++i) {\n            yield arr[i];\n        }\n\n        return arr[len - 1];\n    }\n\n    /**\n     * Builds a tree like structure over time (if called on the same root node) where internal nodes are instances of\n     * Map containing the name of the children nodes (each child name is the result of calling next on `keysGenerator)\n     * and a reference to them and on leafs it contains an instance of Set where it adds the function given as parameter\n     * (ex: for a keyGenerator that returns in this order (\"key1\", \"key2\") the resulting structure will be:\n     * {\"key1\": {\"key1\": Set([fn])}} - using JSON just for illustration purposes because it's easier to represent)\n     * @param {Map} rulesMap\n     * @param {IterableIterator} keysGenerator - it has the particularity that done is set on last element, not after it\n     * @param {function} fn\n     */\n    function registerRecursiveRule(rulesMap, keysGenerator, fn) {\n        const {value, done} = keysGenerator.next();\n\n        if (!done) { // internal node\n            const nextKey = rulesMap.get(value);\n\n            if (typeof nextKey === 'undefined') { // if value not found in rulesMap\n                rulesMap.set(value, new Map());\n            }\n\n            registerRecursiveRule(rulesMap.get(value), keysGenerator, fn);\n        } else { // reached leaf node\n            if (!rulesMap.has(value)) {\n\n                rulesMap.set(value, new Set([fn]));\n            } else {\n                const set = rulesMap.get(value);\n\n                if (set.has(fn)) {\n                    _warning(`Duplicated interceptor for '${key}'`);\n                }\n\n                set.add(fn);\n            }\n        }\n    }\n\n    /**\n     * Returns the corresponding set of functions for the given key if found\n     * @param {string} key - formatted as a path without the first '/' (ex: swarmType/swarmPhase/before)\n     * @return {Array<Set<function>>}\n     */\n    function getInterceptorsForKey(key) {\n        if (key.startsWith('/')) {\n            _warning(`Interceptor called on key ${key} starting with '/', automatically removing it`);\n            key = key.substring(1);\n        }\n\n        const keyElements = key.split('/');\n        const keysGenerator = createArrayGenerator(keyElements);\n\n        return getValueRecursively([rules], keysGenerator);\n    }\n\n    /**\n     * It works like a BFS search returning the leafs resulting from traversing the internal nodes with corresponding\n     * names given for each level (depth) by `keysGenerator`\n     * @param {Array<Map>} searchableNodes\n     * @param {IterableIterator} keysGenerator - it has the particularity that done is set on last element, not after it\n     * @return {Array<Set<function>>}\n     */\n    function getValueRecursively(searchableNodes, keysGenerator) {\n        const {value: nodeName, done} = keysGenerator.next();\n\n        const nextNodes = [];\n\n        for (const nodeInRules of searchableNodes) {\n            const nextNodeForAll = nodeInRules.get('*');\n            const nextNode = nodeInRules.get(nodeName);\n\n            if (typeof nextNode !== \"undefined\") {\n                nextNodes.push(nextNode);\n            }\n\n            if (typeof nextNodeForAll !== \"undefined\") {\n                nextNodes.push(nextNodeForAll);\n            }\n\n        }\n\n        if (done) {\n            return nextNodes;\n        }\n\n        return getValueRecursively(nextNodes, keysGenerator);\n    }\n\n\n    /************* PUBLIC METHODS *************/\n\n    this.register = function (swarmTypeName, phaseName, when, fn) {\n        verifyWhenOption(when);\n        verifyIsFunctionType(fn);\n\n        const resolvedSwarmTypeName = resolveNamespaceResolution(swarmTypeName);\n        const keys = createArrayGenerator([resolvedSwarmTypeName, phaseName, when]);\n\n        registerRecursiveRule(rules, keys, fn);\n    };\n\n    // this.unregister = function () { }\n\n    this.callInterceptors = function (key, targetObject, args) {\n        const interceptors = getInterceptorsForKey(key);\n\n        if (interceptors) {\n            for (const interceptorSet of interceptors) {\n                for (const fn of interceptorSet) { // interceptors on key '*' are called before those specified by name\n                    fn.apply(targetObject, args);\n                }\n            }\n        }\n    };\n}\n\n\nexports.createInterceptorRegistry = function () {\n    return new InterceptorRegistry();\n};\n","/*\n Initial License: (c) Axiologic Research & Alboaie Sînică.\n Contributors: Axiologic Research , PrivateSky project\n Code License: LGPL or MIT.\n */\n\nvar util = require(\"util\");\nvar fs = require(\"fs\");\ncprint = console.log;\nwprint = console.warn;\ndprint = console.debug;\neprint = console.error;\n\n\n/**\n * Shortcut to JSON.stringify\n * @param obj\n */\nJ = function (obj) {\n    return JSON.stringify(obj);\n}\n\n\n/**\n * Print swarm contexts (Messages) and easier to read compared with J\n * @param obj\n * @return {string}\n */\nexports.cleanDump = function (obj) {\n    var o = obj.valueOf();\n    var meta = {\n        swarmTypeName:o.meta.swarmTypeName\n    };\n    return \"\\t swarmId: \" + o.meta.swarmId + \"{\\n\\t\\tmeta: \"    + J(meta) +\n        \"\\n\\t\\tpublic: \"        + J(o.publicVars) +\n        \"\\n\\t\\tprotected: \"     + J(o.protectedVars) +\n        \"\\n\\t\\tprivate: \"       + J(o.privateVars) + \"\\n\\t}\\n\";\n}\n\n//M = exports.cleanDump;\n/**\n * Experimental functions\n */\n\n\n/*\n\n logger      = monitor.logger;\n assert      = monitor.assert;\n throwing    = monitor.exceptions;\n\n\n var temporaryLogBuffer = [];\n\n var currentSwarmComImpl = null;\n\n logger.record = function(record){\n if(currentSwarmComImpl===null){\n temporaryLogBuffer.push(record);\n } else {\n currentSwarmComImpl.recordLog(record);\n }\n }\n\n var container = require(\"dicontainer\").container;\n\n container.service(\"swarmLoggingMonitor\", [\"swarmingIsWorking\", \"swarmComImpl\"], function(outOfService,swarming, swarmComImpl){\n\n if(outOfService){\n if(!temporaryLogBuffer){\n temporaryLogBuffer = [];\n }\n } else {\n var tmp = temporaryLogBuffer;\n temporaryLogBuffer = [];\n currentSwarmComImpl = swarmComImpl;\n logger.record = function(record){\n currentSwarmComImpl.recordLog(record);\n }\n\n tmp.forEach(function(record){\n logger.record(record);\n });\n }\n })\n\n */\nuncaughtExceptionString = \"\";\nuncaughtExceptionExists = false;\nif(typeof globalVerbosity == 'undefined'){\n    globalVerbosity = false;\n}\n\nvar DEBUG_START_TIME = new Date().getTime();\n\nfunction getDebugDelta(){\n    var currentTime = new Date().getTime();\n    return currentTime - DEBUG_START_TIME;\n}\n\n/**\n * Debug functions, influenced by globalVerbosity global variable\n * @param txt\n */\ndprint = function (txt) {\n    if (globalVerbosity == true) {\n        if (thisAdapter.initilised ) {\n            console.log(\"DEBUG: [\" + thisAdapter.nodeName + \"](\" + getDebugDelta()+ \"):\"+txt);\n        }\n        else {\n            console.log(\"DEBUG: (\" + getDebugDelta()+ \"):\"+txt);\n            console.log(\"DEBUG: \" + txt);\n        }\n    }\n}\n\n/**\n * obsolete!?\n * @param txt\n */\naprint = function (txt) {\n    console.log(\"DEBUG: [\" + thisAdapter.nodeName + \"]: \" + txt);\n}\n\n\n\n/**\n * Utility function usually used in tests, exit current process after a while\n * @param msg\n * @param timeout\n */\ndelayExit = function (msg, retCode,timeout) {\n    if(retCode == undefined){\n        retCode = ExitCodes.UnknownError;\n    }\n\n    if(timeout == undefined){\n        timeout = 100;\n    }\n\n    if(msg == undefined){\n        msg = \"Delaying exit with \"+ timeout + \"ms\";\n    }\n\n    console.log(msg);\n    setTimeout(function () {\n        process.exit(retCode);\n    }, timeout);\n}\n\n\nfunction localLog (logType, message, err) {\n    var time = new Date();\n    var now = time.getDate() + \"-\" + (time.getMonth() + 1) + \",\" + time.getHours() + \":\" + time.getMinutes();\n    var msg;\n\n    msg = '[' + now + '][' + thisAdapter.nodeName + '] ' + message;\n\n    if (err != null && err != undefined) {\n        msg += '\\n     Err: ' + err.toString();\n        if (err.stack && err.stack != undefined)\n            msg += '\\n     Stack: ' + err.stack + '\\n';\n    }\n\n    cprint(msg);\n    if(thisAdapter.initilised){\n        try{\n            fs.appendFileSync(getSwarmFilePath(thisAdapter.config.logsPath + \"/\" + logType), msg);\n        } catch(err){\n            console.log(\"Failing to write logs in \", thisAdapter.config.logsPath );\n        }\n\n    }\n}\n\n\n// printf = function (...params) {\n//     var args = []; // empty array\n//     // copy all other arguments we want to \"pass through\"\n//     for (var i = 0; i < params.length; i++) {\n//         args.push(params[i]);\n//     }\n//     var out = util.format.apply(this, args);\n//     console.log(out);\n// }\n//\n// sprintf = function (...params) {\n//     var args = []; // empty array\n//     for (var i = 0; i < params.length; i++) {\n//         args.push(params[i]);\n//     }\n//     return util.format.apply(this, args);\n// }\n\n","\n\nfunction SwarmsInstancesManager(){\n    var swarmAliveInstances = {\n\n    }\n\n    this.waitForSwarm = function(callback, swarm, keepAliveCheck){\n\n        function doLogic(){\n            var swarmId = swarm.getInnerValue().meta.swarmId;\n            var watcher = swarmAliveInstances[swarmId];\n            if(!watcher){\n                watcher = {\n                    swarm:swarm,\n                    callback:callback,\n                    keepAliveCheck:keepAliveCheck\n                }\n                swarmAliveInstances[swarmId] = watcher;\n            }\n        }\n\n        function filter(){\n            return swarm.getInnerValue().meta.swarmId;\n        }\n\n        //$$.uidGenerator.wait_for_condition(condition,doLogic);\n        swarm.observe(doLogic, null, filter);\n    }\n\n    function cleanSwarmWaiter(swarmSerialisation){ // TODO: add better mechanisms to prevent memory leaks\n        var swarmId = swarmSerialisation.meta.swarmId;\n        var watcher = swarmAliveInstances[swarmId];\n\n        if(!watcher){\n            $$.errorHandler.warning(\"Invalid swarm received: \" + swarmId);\n            return;\n        }\n\n        var args = swarmSerialisation.meta.args;\n        args.push(swarmSerialisation);\n\n        watcher.callback.apply(null, args);\n        if(!watcher.keepAliveCheck()){\n            delete swarmAliveInstances[swarmId];\n        }\n    }\n\n    this.revive_swarm = function(swarmSerialisation){\n\n\n        var swarmId     = swarmSerialisation.meta.swarmId;\n        var swarmType   = swarmSerialisation.meta.swarmTypeName;\n        var instance    = swarmAliveInstances[swarmId];\n\n        var swarm;\n\n        if(instance){\n            swarm = instance.swarm;\n            swarm.update(swarmSerialisation);\n\n        } else {\n            swarm = $$.swarm.start(swarmType);\n            swarm.update(swarmSerialisation);\n            /*swarm = $$.swarm.start(swarmType, swarmSerialisation);*/\n        }\n\n        if (swarmSerialisation.meta.command == \"asyncReturn\") {\n            var co = $$.PSK_PubSub.publish($$.CONSTANTS.SWARM_RETURN, swarmSerialisation);\n            console.log(\"Subscribers listening on\", $$.CONSTANTS.SWARM_RETURN, co);\n            // cleanSwarmWaiter(swarmSerialisation);\n        } else if (swarmSerialisation.meta.command == \"executeSwarmPhase\") {\n            swarm.runPhase(swarmSerialisation.meta.phaseName, swarmSerialisation.meta.args);\n        } else {\n            console.log(\"Unknown command\", swarmSerialisation.meta.command, \"in swarmSerialisation.meta.command\");\n        }\n\n        return swarm;\n    }\n}\n\n\n$$.swarmsInstancesManager = new SwarmsInstancesManager();\n\n\n","exports.createForObject = function(valueObject, thisObject, localId){\n\tvar ret = require(\"./base\").createForObject(valueObject, thisObject, localId);\n\n\tret.swarm           = null;\n\tret.onReturn        = null;\n\tret.onResult        = null;\n\tret.asyncReturn     = null;\n\tret.return          = null;\n\tret.home            = null;\n\tret.isPersisted  \t= function () {\n\t\treturn thisObject.getMetadata('persisted') === true;\n\t};\n\n\treturn ret;\n};","var beesHealer = require(\"swarmutils\").beesHealer;\nvar swarmDebug = require(\"../SwarmDebug\");\n\nexports.createForObject = function(valueObject, thisObject, localId){\n\tvar ret = {};\n\n\tfunction filterForSerialisable (valueObject){\n\t\treturn valueObject.meta.swarmId;\n\t}\n\n\tvar swarmFunction = function(context, phaseName){\n\t\tvar args =[];\n\t\tfor(var i = 2; i < arguments.length; i++){\n\t\t\targs.push(arguments[i]);\n\t\t}\n\n\t\t//make the execution at level 0  (after all pending events) and wait to have a swarmId\n\t\tret.observe(function(){\n\t\t\tbeesHealer.asJSON(valueObject, phaseName, args, function(err,jsMsg){\n\t\t\t\tjsMsg.meta.target = context;\n\t\t\t\tvar subscribersCount = $$.PSK_PubSub.publish($$.CONSTANTS.SWARM_FOR_EXECUTION, jsMsg);\n\t\t\t\tif(!subscribersCount){\n\t\t\t\t\tconsole.log(`Nobody listening for <${$$.CONSTANTS.SWARM_FOR_EXECUTION}>!`);\n\t\t\t\t}\n\t\t\t});\n\t\t},null,filterForSerialisable);\n\n\t\tret.notify();\n\n\n\t\treturn thisObject;\n\t};\n\n\tvar asyncReturn = function(err, result){\n\t\tvar context = valueObject.protectedVars.context;\n\n\t\tif(!context && valueObject.meta.waitStack){\n\t\t\tcontext = valueObject.meta.waitStack.pop();\n\t\t\tvalueObject.protectedVars.context = context;\n\t\t}\n\n\t\tbeesHealer.asJSON(valueObject, \"__return__\", [err, result], function(err,jsMsg){\n\t\t\tjsMsg.meta.command = \"asyncReturn\";\n\t\t\tif(!context){\n\t\t\t\tcontext = valueObject.meta.homeSecurityContext;//TODO: CHECK THIS\n\n\t\t\t}\n\t\t\tjsMsg.meta.target = context;\n\n\t\t\tif(!context){\n\t\t\t\t$$.errorHandler.error(new Error(\"Asynchronous return inside of a swarm that does not wait for results\"));\n\t\t\t} else {\n\t\t\t\t$$.PSK_PubSub.publish($$.CONSTANTS.SWARM_FOR_EXECUTION, jsMsg);\n\t\t\t}\n\t\t});\n\t};\n\n\tfunction home(err, result){\n\t\tbeesHealer.asJSON(valueObject, \"home\", [err, result], function(err,jsMsg){\n\t\t\tvar context = valueObject.meta.homeContext;\n\t\t\tjsMsg.meta.target = context;\n\t\t\t$$.PSK_PubSub.publish($$.CONSTANTS.SWARM_FOR_EXECUTION, jsMsg);\n\t\t});\n\t}\n\n\n\n\tfunction waitResults(callback, keepAliveCheck, swarm){\n\t\tif(!swarm){\n\t\t\tswarm = this;\n\t\t}\n\t\tif(!keepAliveCheck){\n\t\t\tkeepAliveCheck = function(){\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\t\tvar inner = swarm.getInnerValue();\n\t\tif(!inner.meta.waitStack){\n\t\t\tinner.meta.waitStack = [];\n\t\t\tinner.meta.waitStack.push($$.securityContext)\n\t\t}\n\t\t$$.swarmsInstancesManager.waitForSwarm(callback, swarm, keepAliveCheck);\n\t}\n\n\n\tfunction getInnerValue(){\n\t\treturn valueObject;\n\t}\n\n\tfunction runPhase(functName, args){\n\t\tvar func = valueObject.myFunctions[functName];\n\t\tif(func){\n\t\t\tfunc.apply(thisObject, args);\n\t\t} else {\n\t\t\t$$.errorHandler.syntaxError(functName, valueObject, \"Function \" + functName + \" does not exist!\");\n\t\t}\n\n\t}\n\n\tfunction update(serialisation){\n\t\tbeesHealer.jsonToNative(serialisation,valueObject);\n\t}\n\n\n\tfunction valueOf(){\n\t\tvar ret = {};\n\t\tret.meta                = valueObject.meta;\n\t\tret.publicVars          = valueObject.publicVars;\n\t\tret.privateVars         = valueObject.privateVars;\n\t\tret.protectedVars       = valueObject.protectedVars;\n\t\treturn ret;\n\t}\n\n\tfunction toString (){\n\t\treturn swarmDebug.cleanDump(thisObject.valueOf());\n\t}\n\n\n\tfunction createParallel(callback){\n\t\treturn require(\"../../parallelJoinPoint\").createJoinPoint(thisObject, callback, $$.__intern.mkArgs(arguments,1));\n\t}\n\n\tfunction createSerial(callback){\n\t\treturn require(\"../../serialJoinPoint\").createSerialJoinPoint(thisObject, callback, $$.__intern.mkArgs(arguments,1));\n\t}\n\n\tfunction inspect(){\n\t\treturn swarmDebug.cleanDump(thisObject.valueOf());\n\t}\n\n\tfunction constructor(){\n\t\treturn SwarmDescription;\n\t}\n\n\tfunction ensureLocalId(){\n\t\tif(!valueObject.localId){\n\t\t\tvalueObject.localId = valueObject.meta.swarmTypeName + \"-\" + localId;\n\t\t\tlocalId++;\n\t\t}\n\t}\n\n\tfunction observe(callback, waitForMore, filter){\n\t\tif(!waitForMore){\n\t\t\twaitForMore = function (){\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\tensureLocalId();\n\n\t\t$$.PSK_PubSub.subscribe(valueObject.localId, callback, waitForMore, filter);\n\t}\n\n\tfunction toJSON(prop){\n\t\t//preventing max call stack size exceeding on proxy auto referencing\n\t\t//replace {} as result of JSON(Proxy) with the string [Object protected object]\n\t\treturn \"[Object protected object]\";\n\t}\n\n\tfunction getJSON(callback){\n\t\treturn\tbeesHealer.asJSON(valueObject, null, null,callback);\n\t}\n\n\tfunction notify(event){\n\t\tif(!event){\n\t\t\tevent = valueObject;\n\t\t}\n\t\tensureLocalId();\n\t\t$$.PSK_PubSub.publish(valueObject.localId, event);\n\t}\n\n\tfunction getMeta(name){\n\t\treturn valueObject.getMeta(name);\n\t}\n\n\tfunction setMeta(name, value){\n\t\treturn valueObject.setMeta(name, value);\n\t}\n\n\tret.setMeta\t\t\t= setMeta;\n\tret.getMeta\t\t\t= getMeta;\n\tret.swarm           = swarmFunction;\n\tret.notify          = notify;\n\tret.getJSON    \t    = getJSON;\n\tret.toJSON          = toJSON;\n\tret.observe         = observe;\n\tret.inspect         = inspect;\n\tret.join            = createParallel;\n\tret.parallel        = createParallel;\n\tret.serial          = createSerial;\n\tret.valueOf         = valueOf;\n\tret.update          = update;\n\tret.runPhase        = runPhase;\n\tret.onReturn        = waitResults;\n\tret.onResult        = waitResults;\n\tret.asyncReturn     = asyncReturn;\n\tret.return          = asyncReturn;\n\tret.getInnerValue   = getInnerValue;\n\tret.home            = home;\n\tret.toString        = toString;\n\tret.constructor     = constructor;\n\tret.setMetadata\t\t= valueObject.setMeta.bind(valueObject);\n\tret.getMetadata\t\t= valueObject.getMeta.bind(valueObject);\n\n\treturn ret;\n\n};\n","exports.createForObject = function(valueObject, thisObject, localId){\n\tvar ret = require(\"./base\").createForObject(valueObject, thisObject, localId);\n\n\tret.swarm           = null;\n\tret.onReturn        = null;\n\tret.onResult        = null;\n\tret.asyncReturn     = null;\n\tret.return          = null;\n\tret.home            = null;\n\n\treturn ret;\n};","exports.createForObject = function(valueObject, thisObject, localId){\n\treturn require(\"./base\").createForObject(valueObject, thisObject, localId);\n};","/*\nInitial License: (c) Axiologic Research & Alboaie Sînică.\nContributors: Axiologic Research , PrivateSky project\nCode License: LGPL or MIT.\n*/\n\n//var fs = require(\"fs\");\n//var path = require(\"path\");\n\n\nfunction SwarmLibrary(prefixName, folder){\n    var self = this;\n    function wrapCall(original, prefixName){\n        return function(...args){\n            //console.log(\"prefixName\", prefixName)\n            var previousPrefix = $$.libraryPrefix;\n            var previousLibrary = $$.__global.currentLibrary;\n\n            $$.libraryPrefix = prefixName;\n            $$.__global.currentLibrary = self;\n            try{\n                var ret = original.apply(this, args);\n                $$.libraryPrefix = previousPrefix ;\n                $$.__global.currentLibrary = previousLibrary;\n            }catch(err){\n                $$.libraryPrefix = previousPrefix ;\n                $$.__global.currentLibrary = previousLibrary;\n                throw err;\n            }\n            return ret;\n        }\n    }\n\n    $$.libraries[prefixName] = this;\n    var prefixedRequire = wrapCall(function(path){\n        return require(path);\n    }, prefixName);\n\n    function includeAllInRoot(folder) {\n        if(typeof folder != \"string\"){\n            //we assume that it is a library module properly required with require and containing $$.library\n            for(var v in folder){\n                $$.registerSwarmDescription(prefixName,v, prefixName + \".\" + v,  folder[v]);\n            }\n\n            var newNames = $$.__global.requireLibrariesNames[prefixName];\n            for(var v in newNames){\n                self[v] =  newNames[v];\n            }\n            return folder;\n        }\n\n\n        var res = prefixedRequire(folder); // a library is just a module\n        if(typeof res.__autogenerated_privatesky_libraryName != \"undefined\"){\n            var swarms = $$.__global.requireLibrariesNames[res.__autogenerated_privatesky_libraryName];\n        } else {\n            var swarms = $$.__global.requireLibrariesNames[folder];\n        }\n            var existingName;\n            for(var v in swarms){\n                existingName = swarms[v];\n                self[v] = existingName;\n                $$.registerSwarmDescription(prefixName,v, prefixName + \".\" + v,  existingName);\n            }\n        return res;\n    }\n\n    function wrapSwarmRelatedFunctions(space, prefixName){\n        var ret = {};\n        var names = [\"create\", \"describe\", \"start\", \"restart\"];\n        for(var i = 0; i<names.length; i++ ){\n            ret[names[i]] = wrapCall(space[names[i]], prefixName);\n        }\n        return ret;\n    }\n\n    this.callflows        = this.callflow   = wrapSwarmRelatedFunctions($$.callflows, prefixName);\n    this.swarms           = this.swarm      = wrapSwarmRelatedFunctions($$.swarms, prefixName);\n    this.contracts        = this.contract   = wrapSwarmRelatedFunctions($$.contracts, prefixName);\n    includeAllInRoot(folder, prefixName);\n}\n\nexports.loadLibrary = function(prefixName, folder){\n    var existing = $$.libraries[prefixName];\n    if(existing ){\n        if(!(existing instanceof SwarmLibrary)){\n            var sL = new SwarmLibrary(prefixName, folder);\n            for(var prop in existing){\n                sL[prop] = existing[prop];\n            }\n            return sL;\n        }\n        if(folder) {\n            $$.errorHandler.warning(\"Reusing already loaded library \" + prefixName + \"could be an error!\");\n        }\n        return existing;\n    }\n    //var absolutePath = path.resolve(folder);\n    return new SwarmLibrary(prefixName, folder);\n}\n\n","/*\n require and $$.require are overwriting the node.js defaults in loading modules for increasing security,speed and making it work to the privatesky runtime build with browserify.\n The privatesky code for domains should work in node and browsers.\n */\n\n\nif (typeof(window) !== \"undefined\") {\n    global = window;\n}\n\n\nif (typeof(global.$$) == \"undefined\") {\n    global.$$ = {};\n    $$.__global = {};\n}\n\nif (typeof($$.__global) == \"undefined\") {\n    $$.__global = {};\n}\n\nif (typeof($$.__global.requireLibrariesNames) == \"undefined\") {\n    $$.__global.currentLibraryName = null;\n    $$.__global.requireLibrariesNames = {};\n}\n\n\nif (typeof($$.__runtimeModules) == \"undefined\") {\n    $$.__runtimeModules = {};\n}\n\n\nif (typeof(global.functionUndefined) == \"undefined\") {\n    global.functionUndefined = function () {\n        console.log(\"Called of an undefined function!!!!\");\n        throw new Error(\"Called of an undefined function\");\n    };\n    if (typeof(global.webshimsRequire) == \"undefined\") {\n        global.webshimsRequire = global.functionUndefined;\n    }\n\n    if (typeof(global.domainRequire) == \"undefined\") {\n        global.domainRequire = global.functionUndefined;\n    }\n\n    if (typeof(global.pskruntimeRequire) == \"undefined\") {\n        global.pskruntimeRequire = global.functionUndefined;\n    }\n}\n\nif (typeof($$.log) == \"undefined\") {\n    $$.log = function (...args) {\n        console.log(args.join(\" \"));\n    }\n}\n\n\nconst weAreInbrowser = (typeof ($$.browserRuntime) != \"undefined\");\nconst weAreInSandbox = (typeof global.require !== 'undefined');\n\n\nconst pastRequests = {};\n\nfunction preventRecursiveRequire(request) {\n    if (pastRequests[request]) {\n        const err = new Error(\"Preventing recursive require for \" + request);\n        err.type = \"PSKIgnorableError\";\n        throw err;\n    }\n\n}\n\nfunction disableRequire(request) {\n    pastRequests[request] = true;\n}\n\nfunction enableRequire(request) {\n    pastRequests[request] = false;\n}\n\n\nfunction requireFromCache(request) {\n    const existingModule = $$.__runtimeModules[request];\n    return existingModule;\n}\n\nfunction wrapStep(callbackName) {\n    const callback = global[callbackName];\n\n    if (callback === undefined) {\n        return null;\n    }\n\n    if (callback === global.functionUndefined) {\n        return null;\n    }\n\n    return function (request) {\n        const result = callback(request);\n        $$.__runtimeModules[request] = result;\n        return result;\n    }\n}\n\nfunction tryRequireSequence(originalRequire, request) {\n    let arr;\n    if (originalRequire) {\n        arr = $$.__requireFunctionsChain.slice();\n        arr.push(originalRequire);\n    } else {\n        arr = $$.__requireFunctionsChain;\n    }\n\n    preventRecursiveRequire(request);\n    disableRequire(request);\n    let result;\n    const previousRequire = $$.__global.currentLibraryName;\n    let previousRequireChanged = false;\n\n    if (!previousRequire) {\n        // console.log(\"Loading library for require\", request);\n        $$.__global.currentLibraryName = request;\n\n        if (typeof $$.__global.requireLibrariesNames[request] == \"undefined\") {\n            $$.__global.requireLibrariesNames[request] = {};\n            //$$.__global.requireLibrariesDescriptions[request]   = {};\n        }\n        previousRequireChanged = true;\n    }\n    for (let i = 0; i < arr.length; i++) {\n        const func = arr[i];\n        try {\n\n            if (func === global.functionUndefined) continue;\n            result = func(request);\n\n            if (result) {\n                break;\n            }\n\n        } catch (err) {\n            if (err.type !== \"PSKIgnorableError\") {\n                $$.log(\"Require encountered an error while loading \", request, \"\\nCause:\\n\", err.stack);\n            }\n        }\n    }\n\n    if (!result) {\n        $$.log(\"Failed to load module \", request, result);\n    }\n\n    enableRequire(request);\n    if (previousRequireChanged) {\n        //console.log(\"End loading library for require\", request, $$.__global.requireLibrariesNames[request]);\n        $$.__global.currentLibraryName = null;\n    }\n    return result;\n}\n\nif (typeof($$.require) == \"undefined\") {\n\n    $$.__requireList = [\"webshimsRequire\", \"pskruntimeRequire\"];\n    $$.__requireFunctionsChain = [];\n\n    $$.requireBundle = function (name) {\n        name += \"Require\";\n        $$.__requireList.push(name);\n        const arr = [requireFromCache];\n        $$.__requireList.forEach(function (item) {\n            const callback = wrapStep(item);\n            if (callback) {\n                arr.push(callback);\n            }\n        });\n\n        $$.__requireFunctionsChain = arr;\n    };\n\n    $$.requireBundle(\"init\");\n\n    if (weAreInbrowser) {\n        $$.log(\"Defining global require in browser\");\n\n\n        global.require = function (request) {\n\n            ///*[requireFromCache, wrapStep(webshimsRequire), , wrapStep(pskruntimeRequire), wrapStep(domainRequire)*]\n            return tryRequireSequence(null, request);\n        }\n    } else\n        if (weAreInSandbox) {\n        // require should be provided when code is loaded in browserify\n        const bundleRequire = require;\n\n        $$.requireBundle('sandboxBase');\n        // this should be set up by sandbox prior to\n        const sandboxRequire = global.require;\n        global.crypto = require('crypto');\n\n        function newLoader(request) {\n            // console.log(\"newLoader:\", request);\n            //preventRecursiveRequire(request);\n            const self = this;\n\n            // console.log('trying to load ', request);\n\n            function tryBundleRequire(...args) {\n                //return $$.__originalRequire.apply(self,args);\n                //return Module._load.apply(self,args)\n                let res;\n                try {\n                    res = sandboxRequire.apply(self, args);\n                } catch (err) {\n                    if (err.code === \"MODULE_NOT_FOUND\") {\n                        const p = path.join(process.cwd(), request);\n                        res = sandboxRequire.apply(self, [p]);\n                        request = p;\n                    } else {\n                        throw err;\n                    }\n                }\n                return res;\n            }\n\n            let res;\n\n\n            res = tryRequireSequence(tryBundleRequire, request);\n\n\n            return res;\n        }\n\n        global.require = newLoader;\n\n    } else {  //we are in node\n        const path = require(\"path\");\n        $$.__runtimeModules[\"crypto\"] = require(\"crypto\");\n        $$.__runtimeModules[\"util\"] = require(\"util\");\n\n        const Module = require('module');\n        $$.__runtimeModules[\"module\"] = Module;\n\n        $$.log(\"Redefining require for node\");\n\n        $$.__originalRequire = Module._load;\n        const moduleOriginalRequire = Module.prototype.require;\n\n        function newLoader(request) {\n            // console.log(\"newLoader:\", request);\n            //preventRecursiveRequire(request);\n            const self = this;\n\n            function originalRequire(...args) {\n                //return $$.__originalRequire.apply(self,args);\n                //return Module._load.apply(self,args)\n                let res;\n                try {\n                    res = moduleOriginalRequire.apply(self, args);\n                } catch (err) {\n                    if (err.code === \"MODULE_NOT_FOUND\") {\n                        const p = path.join(process.cwd(), request);\n                        res = moduleOriginalRequire.apply(self, [p]);\n                        request = p;\n                    } else {\n                        throw err;\n                    }\n                }\n                return res;\n            }\n\n            function currentFolderRequire(request) {\n                return\n            }\n\n            //[requireFromCache, wrapStep(pskruntimeRequire), wrapStep(domainRequire), originalRequire]\n            return tryRequireSequence(originalRequire, request);\n        }\n\n        Module.prototype.require = newLoader;\n    }\n\n    $$.require = require;\n}\n","\nvar joinCounter = 0;\n\nfunction ParallelJoinPoint(swarm, callback, args){\n    joinCounter++;\n    var channelId = \"ParallelJoinPoint\" + joinCounter;\n    var self = this;\n    var counter = 0;\n    var stopOtherExecution     = false;\n\n    function executionStep(stepFunc, localArgs, stop){\n\n        this.doExecute = function(){\n            if(stopOtherExecution){\n                return false;\n            }\n            try{\n                stepFunc.apply(swarm, localArgs);\n                if(stop){\n                    stopOtherExecution = true;\n                    return false;\n                }\n                return true; //everyting is fine\n            } catch(err){\n                args.unshift(err);\n                sendForSoundExecution(callback, args, true);\n                return false; //stop it, do not call again anything\n            }\n        }\n    }\n\n    if(typeof callback !== \"function\"){\n        $$.errorHandler.syntaxError(\"invalid join\",swarm, \"invalid function at join in swarm\");\n        return;\n    }\n\n    $$.PSK_PubSub.subscribe(channelId,function(forExecution){\n        if(stopOtherExecution){\n            return ;\n        }\n\n        try{\n            if(forExecution.doExecute()){\n                decCounter();\n            } // had an error...\n        } catch(err){\n            //console.log(err);\n            //$$.errorHandler.syntaxError(\"__internal__\",swarm, \"exception in the execution of the join function of a parallel task\");\n        }\n    });\n\n    function incCounter(){\n        if(testIfUnderInspection()){\n            //preventing inspector from increasing counter when reading the values for debug reason\n            //console.log(\"preventing inspection\");\n            return;\n        }\n        counter++;\n    }\n\n    function testIfUnderInspection(){\n        var res = false;\n        var constArgv = process.execArgv.join();\n        if(constArgv.indexOf(\"inspect\")!==-1 || constArgv.indexOf(\"debug\")!==-1){\n            //only when running in debug\n            var callstack = new Error().stack;\n            if(callstack.indexOf(\"DebugCommandProcessor\")!==-1){\n                console.log(\"DebugCommandProcessor detected!\");\n                res = true;\n            }\n        }\n        return res;\n    }\n\n    function sendForSoundExecution(funct, args, stop){\n        var obj = new executionStep(funct, args, stop);\n        $$.PSK_PubSub.publish(channelId, obj); // force execution to be \"sound\"\n    }\n\n    function decCounter(){\n        counter--;\n        if(counter == 0) {\n            args.unshift(null);\n            sendForSoundExecution(callback, args, false);\n        }\n    }\n\n    var inner = swarm.getInnerValue();\n\n    function defaultProgressReport(err, res){\n        if(err) {\n            throw err;\n        }\n        return {\n            text:\"Parallel execution progress event\",\n            swarm:swarm,\n            args:args,\n            currentResult:res\n        };\n    }\n\n    function mkFunction(name){\n        return function(...args){\n            var f = defaultProgressReport;\n            if(name != \"progress\"){\n                f = inner.myFunctions[name];\n            }\n            var args = $$.__intern.mkArgs(args, 0);\n            sendForSoundExecution(f, args, false);\n            return __proxyObject;\n        }\n    }\n\n\n    this.get = function(target, prop, receiver){\n        if(inner.myFunctions.hasOwnProperty(prop) || prop == \"progress\"){\n            incCounter();\n            return mkFunction(prop);\n        }\n        return swarm[prop];\n    };\n\n    var __proxyObject;\n\n    this.__setProxyObject = function(p){\n        __proxyObject = p;\n    }\n}\n\nexports.createJoinPoint = function(swarm, callback, args){\n    var jp = new ParallelJoinPoint(swarm, callback, args);\n    var inner = swarm.getInnerValue();\n    var p = new Proxy(inner, jp);\n    jp.__setProxyObject(p);\n    return p;\n};","\nvar joinCounter = 0;\n\nfunction SerialJoinPoint(swarm, callback, args){\n\n    joinCounter++;\n\n    var self = this;\n    var channelId = \"SerialJoinPoint\" + joinCounter;\n\n    if(typeof callback !== \"function\"){\n        $$.errorHandler.syntaxError(\"unknown\", swarm, \"invalid function given to serial in swarm\");\n        return;\n    }\n\n    var inner = swarm.getInnerValue();\n\n\n    function defaultProgressReport(err, res){\n        if(err) {\n            throw err;\n        }\n        return res;\n    }\n\n\n    var functionCounter     = 0;\n    var executionCounter    = 0;\n\n    var plannedExecutions   = [];\n    var plannedArguments    = {};\n\n    function mkFunction(name, pos){\n        //console.log(\"Creating function \", name, pos);\n        plannedArguments[pos] = undefined;\n\n        function triggetNextStep(){\n            if(plannedExecutions.length == executionCounter || plannedArguments[executionCounter] )  {\n                $$.PSK_PubSub.publish(channelId, self);\n            }\n        }\n\n        var f = function (...args){\n            if(executionCounter != pos) {\n                plannedArguments[pos] = args;\n                //console.log(\"Delaying function:\", executionCounter, pos, plannedArguments, arguments, functionCounter);\n                return __proxy;\n            } else{\n                if(plannedArguments[pos]){\n                    //console.log(\"Executing  function:\", executionCounter, pos, plannedArguments, arguments, functionCounter);\n\t\t\t\t\targs = plannedArguments[pos];\n                } else {\n                    plannedArguments[pos] = args;\n                    triggetNextStep();\n                    return __proxy;\n                }\n            }\n\n            var f = defaultProgressReport;\n            if(name != \"progress\"){\n                f = inner.myFunctions[name];\n            }\n\n\n            try{\n                f.apply(self,args);\n            } catch(err){\n                    args.unshift(err);\n                    callback.apply(swarm,args); //error\n                    $$.PSK_PubSub.unsubscribe(channelId,runNextFunction);\n                return; //terminate execution with an error...!\n            }\n            executionCounter++;\n\n            triggetNextStep();\n\n            return __proxy;\n        };\n\n        plannedExecutions.push(f);\n        functionCounter++;\n        return f;\n    }\n\n     var finished = false;\n\n    function runNextFunction(){\n        if(executionCounter == plannedExecutions.length ){\n            if(!finished){\n                args.unshift(null);\n                callback.apply(swarm,args);\n                finished = true;\n                $$.PSK_PubSub.unsubscribe(channelId,runNextFunction);\n            } else {\n                console.log(\"serial construct is using functions that are called multiple times...\");\n            }\n        } else {\n            plannedExecutions[executionCounter]();\n        }\n    }\n\n    $$.PSK_PubSub.subscribe(channelId,runNextFunction); // force it to be \"sound\"\n\n\n    this.get = function(target, prop, receiver){\n        if(prop == \"progress\" || inner.myFunctions.hasOwnProperty(prop)){\n            return mkFunction(prop, functionCounter);\n        }\n        return swarm[prop];\n    }\n\n    var __proxy;\n    this.setProxyObject = function(p){\n        __proxy = p;\n    }\n}\n\nexports.createSerialJoinPoint = function(swarm, callback, args){\n    var jp = new SerialJoinPoint(swarm, callback, args);\n    var inner = swarm.getInnerValue();\n    var p = new Proxy(inner, jp);\n    jp.setProxyObject(p);\n    return p;\n}","const OwM = require(\"swarmutils\").OwM;\n\nvar swarmDescriptionsRegistry = {};\n\n\n$$.registerSwarmDescription =  function(libraryName, shortName, swarmTypeName, description){\n    if(!$$.libraries[libraryName]){\n        $$.libraries[libraryName] = {};\n    }\n\n    if(!$$.__global.requireLibrariesNames[libraryName]){\n        $$.__global.requireLibrariesNames[libraryName] = {};\n    }\n\n    $$.libraries[libraryName][shortName] = description;\n    //console.log(\"Registering \", libraryName,shortName, $$.__global.currentLibraryName);\n    if($$.__global.currentLibraryName){\n        $$.__global.requireLibrariesNames[$$.__global.currentLibraryName][shortName] = libraryName + \".\" + shortName;\n    }\n\n    $$.__global.requireLibrariesNames[libraryName][shortName] = swarmTypeName;\n\n    if(typeof description == \"string\"){\n        description = swarmDescriptionsRegistry[description];\n    }\n    swarmDescriptionsRegistry[swarmTypeName] = description;\n}\n\n\nvar currentLibraryCounter = 0;\n$$.library = function(callback){\n    currentLibraryCounter++;\n    var previousCurrentLibrary = $$.__global.currentLibraryName;\n    var libraryName = \"___privatesky_library\"+currentLibraryCounter;\n    var ret = $$.__global.requireLibrariesNames[libraryName] = {};\n    $$.__global.currentLibraryName = libraryName;\n    callback();\n    $$.__global.currentLibraryName = previousCurrentLibrary;\n    ret.__autogenerated_privatesky_libraryName = libraryName;\n    return ret;\n}\n\nfunction SwarmSpace(swarmType, utils) {\n\n    var beesHealer = require(\"swarmutils\").beesHealer;\n\n    function getFullName(shortName){\n        var fullName;\n        if(shortName && shortName.includes(\".\")) {\n            fullName = shortName;\n        } else {\n            fullName = $$.libraryPrefix + \".\" + shortName;\n        }\n        return fullName;\n    }\n\n    function VarDescription(desc){\n        return {\n            init:function(){\n                return undefined;\n            },\n            restore:function(jsonString){\n                return JSON.parse(jsonString);\n            },\n            toJsonString:function(x){\n                return JSON.stringify();\n            }\n        };\n    }\n\n    function SwarmDescription(swarmTypeName, description){\n\n        swarmTypeName = getFullName(swarmTypeName);\n\n        var localId = 0;  // unique for each swarm\n\n        function createVars(descr){\n            var members = {};\n            for(var v in descr){\n                members[v] = new VarDescription(descr[v]);\n            }\n            return members;\n        }\n\n        function createMembers(descr){\n            var members = {};\n            for(var v in description){\n\n                if(v != \"public\" && v != \"private\"){\n                    members[v] = description[v];\n                }\n            }\n            return members;\n        }\n\n        var publicVars = createVars(description.public);\n        var privateVars = createVars(description.private);\n        var myFunctions = createMembers(description);\n\n        function createPhase(thisInstance, func, phaseName){\n            var keyBefore = `${swarmTypeName}/${phaseName}/${$$.CONSTANTS.BEFORE_INTERCEPTOR}`;\n            var keyAfter = `${swarmTypeName}/${phaseName}/${$$.CONSTANTS.AFTER_INTERCEPTOR}`;\n\n            var phase = function(...args){\n                var ret;\n                try{\n                    $$.PSK_PubSub.blockCallBacks();\n                    thisInstance.setMetadata('phaseName', phaseName);\n                    $$.interceptor.callInterceptors(keyBefore, thisInstance, args);\n                    ret = func.apply(thisInstance, args);\n                    $$.interceptor.callInterceptors(keyAfter, thisInstance, args);\n                    $$.PSK_PubSub.releaseCallBacks();\n                }catch(err){\n                    $$.PSK_PubSub.releaseCallBacks();\n                    throw err;\n                }\n                return ret;\n            }\n            //dynamic named func in order to improve callstack\n            Object.defineProperty(phase, \"name\", {get: function(){return swarmTypeName+\".\"+func.name}});\n            return phase;\n        }\n\n        this.initialise = function(serialisedValues){\n\n            var result = new OwM({\n                publicVars:{\n\n                },\n                privateVars:{\n\n                },\n                protectedVars:{\n\n                },\n                myFunctions:{\n\n                },\n                utilityFunctions:{\n\n                },\n                meta:{\n                    swarmTypeName:swarmTypeName,\n                    swarmDescription:description\n                }\n            });\n\n\n            for(var v in publicVars){\n                result.publicVars[v] = publicVars[v].init();\n            };\n\n            for(var v in privateVars){\n                result.privateVars[v] = privateVars[v].init();\n            };\n\n\n            if(serialisedValues){\n                beesHealer.jsonToNative(serialisedValues, result);\n            }\n            return result;\n        };\n\n        this.initialiseFunctions = function(valueObject, thisObject){\n\n            for(var v in myFunctions){\n                valueObject.myFunctions[v] = createPhase(thisObject, myFunctions[v], v);\n            };\n\n            localId++;\n            valueObject.utilityFunctions = utils.createForObject(valueObject, thisObject, localId);\n\n        }\n\n        this.get = function(target, property, receiver){\n\n\n            if(publicVars.hasOwnProperty(property))\n            {\n                return target.publicVars[property];\n            }\n\n            if(privateVars.hasOwnProperty(property))\n            {\n                return target.privateVars[property];\n            }\n\n            if(target.utilityFunctions.hasOwnProperty(property))\n            {\n\n                return target.utilityFunctions[property];\n            }\n\n\n            if(myFunctions.hasOwnProperty(property))\n            {\n                return target.myFunctions[property];\n            }\n\n            if(target.protectedVars.hasOwnProperty(property))\n            {\n                return target.protectedVars[property];\n            }\n\n            if(typeof property != \"symbol\") {\n                $$.errorHandler.syntaxError(property, target);\n            }\n            return undefined;\n        }\n\n        this.set = function(target, property, value, receiver){\n\n            if(target.utilityFunctions.hasOwnProperty(property) || target.myFunctions.hasOwnProperty(property)) {\n                $$.errorHandler.syntaxError(property);\n                throw new Error(\"Trying to overwrite immutable member\" + property);\n            }\n\n            if(privateVars.hasOwnProperty(property))\n            {\n                target.privateVars[property] = value;\n            } else\n            if(publicVars.hasOwnProperty(property))\n            {\n                target.publicVars[property] = value;\n            } else {\n                target.protectedVars[property] = value;\n            }\n            return true;\n        }\n\n        this.apply = function(target, thisArg, argumentsList){\n            console.log(\"Proxy apply\");\n            //var func = target[]\n            //swarmGlobals.executionProvider.execute(null, thisArg, func, argumentsList)\n        }\n\n        var self = this;\n\n        this.isExtensible = function(target) {\n            return false;\n        };\n\n        this.has = function(target, prop) {\n            if(target.publicVars[prop] || target.protectedVars[prop]) {\n                return true;\n            }\n            return false;\n        };\n\n        this.ownKeys = function(target) {\n            return Reflect.ownKeys(target.publicVars);\n        };\n\n        return function(serialisedValues){\n            var valueObject = self.initialise(serialisedValues);\n            var result = new Proxy(valueObject,self);\n            self.initialiseFunctions(valueObject,result);\n\t\t\tif(!serialisedValues){\n\t\t\t\tif(!valueObject.getMeta(\"swarmId\")){\n\t\t\t\t\tvalueObject.setMeta(\"swarmId\", $$.uidGenerator.safe_uuid());  //do not overwrite!!!\n\t\t\t\t}\n\t\t\t\tvalueObject.utilityFunctions.notify();\n\t\t\t}\n\t\t\treturn result;\n        }\n    }\n\n\n\n    this.describe = function describeSwarm(swarmTypeName, description){\n        swarmTypeName = getFullName(swarmTypeName);\n\n        var pointPos = swarmTypeName.lastIndexOf('.');\n        var shortName = swarmTypeName.substr( pointPos+ 1);\n        var libraryName = swarmTypeName.substr(0, pointPos);\n        if(!libraryName){\n            libraryName = \"global\";\n        }\n\n        var description = new SwarmDescription(swarmTypeName, description);\n        if(swarmDescriptionsRegistry[swarmTypeName] != undefined){\n            $$.errorHandler.warning(\"Duplicate swarm description \"+ swarmTypeName);\n        }\n\n        //swarmDescriptionsRegistry[swarmTypeName] = description;\n\t\t$$.registerSwarmDescription(libraryName, shortName, swarmTypeName, description);\n\n        return description;\n    }\n\n    this.create = function(){\n        $$.error(\"create function is obsolete. use describe!\");\n    }\n    /* // confusing variant\n    this.create = function createSwarm(swarmTypeName, description, initialValues){\n        swarmTypeName = getFullName(swarmTypeName);\n        try{\n            if(undefined == description){\n                return swarmDescriptionsRegistry[swarmTypeName](initialValues);\n            } else {\n                return this.describe(swarmTypeName, description)(initialValues);\n            }\n        } catch(err){\n            console.log(\"CreateSwarm error\", err);\n            $$.errorHandler.error(err, arguments, \"Wrong name or descriptions\");\n        }\n    }*/\n\n    this.continue = function(swarmTypeName, initialValues){\n        swarmTypeName = getFullName(swarmTypeName);\n        var desc = swarmDescriptionsRegistry[swarmTypeName];\n\n        if(desc){\n            return desc(initialValues);\n        } else {\n            $$.errorHandler.syntaxError(swarmTypeName,initialValues,\n                \"Failed to restart a swarm with type \" + swarmTypeName + \"\\n Maybe different swarm space (used flow instead of swarm!?)\");\n        }\n    }\n\n    this.start = function(swarmTypeName, ctor, ...params){\n        swarmTypeName = getFullName(swarmTypeName);\n        var desc = swarmDescriptionsRegistry[swarmTypeName];\n        if(!desc){\n            $$.errorHandler.syntaxError(null, swarmTypeName);\n            return null;\n        }\n        var res = desc();\n        res.setMetadata(\"homeSecurityContext\", $$.securityContext);\n\n        if(ctor){\n            res[ctor].apply(res, params);\n        }\n\n        return res;\n    }\n}\n\nexports.createSwarmEngine = function(swarmType, utils){\n    if(typeof utils == \"undefined\"){\n        utils = require(\"./choreographies/utilityFunctions/callflow\");\n    }\n    return new SwarmSpace(swarmType, utils);\n};\n","\nmodule.exports.init = function(sf, logger){\n    /**\n     * Registering handler for failed asserts. The handler is doing logging and is throwing an error.\n     * @param explanation {String} - failing reason message.\n     */\n    sf.exceptions.register('assertFail', function(explanation){\n        const message = \"Assert or invariant has failed \" + (explanation ? explanation : \"\");\n        const err = new Error(message);\n        logger.recordAssert('[Fail] ' + message, err, true);\n        throw err;\n    });\n\n    /**\n     * Registering assert for equality. If check fails, the assertFail is invoked.\n     * @param v1 {String|Number|Object} - first value\n     * @param v1 {String|Number|Object} - second value\n     * @param explanation {String} - failing reason message in case the assert fails.\n     */\n    sf.assert.addCheck('equal', function(v1 , v2, explanation){\n        if(v1 !== v2){\n            if(!explanation){\n                explanation =  \"Assertion failed: [\" + v1 + \" !== \" + v2 + \"]\";\n            }\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Registering assert for inequality. If check fails, the assertFail is invoked.\n     * @param v1 {String|Number|Object} - first value\n     * @param v1 {String|Number|Object} - second value\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck('notEqual', function(v1, v2, explanation){\n        if(v1 === v2){\n            if(!explanation){\n                explanation =  \" [\"+ v1 + \" == \" + v2 + \"]\";\n            }\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Registering assert for evaluating an expression to true. If check fails, the assertFail is invoked.\n     * @param b {Boolean} - result of an expression\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck('true', function(b, explanation){\n        if(!b){\n            if(!explanation){\n                explanation =  \" expression is false but is expected to be true\";\n            }\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Registering assert for evaluating an expression to false. If check fails, the assertFail is invoked.\n     * @param b {Boolean} - result of an expression\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck('false', function(b, explanation){\n        if(b){\n            if(!explanation){\n                explanation =  \" expression is true but is expected to be false\";\n            }\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Registering assert for evaluating a value to null. If check fails, the assertFail is invoked.\n     * @param b {Boolean} - result of an expression\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck('isNull', function(v1, explanation){\n        if(v1 !== null){\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Registering assert for evaluating a value to be not null. If check fails, the assertFail is invoked.\n     * @param b {Boolean} - result of an expression\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck('notNull', function(v1 , explanation){\n        if(v1 === null && typeof v1 === \"object\"){\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Checks if all properties of the second object are own properties of the first object.\n     * @param firstObj {Object} - first object\n     * @param secondObj{Object} - second object\n     * @returns {boolean} - returns true, if the check has passed or false otherwise.\n     */\n    function objectHasFields(firstObj, secondObj){\n        for(let field in secondObj) {\n            if (firstObj.hasOwnProperty(field)) {\n                if (firstObj[field] !== secondObj[field]) {\n                    return false;\n                }\n            }\n            else{\n                return false;\n            }\n        }\n        return true;\n    }\n\n    function objectsAreEqual(firstObj, secondObj) {\n        let areEqual = true;\n        if(firstObj !== secondObj) {\n            if(typeof firstObj !== typeof secondObj) {\n                areEqual = false;\n            } else if (Array.isArray(firstObj) && Array.isArray(secondObj)) {\n\t            firstObj.sort();\n\t            secondObj.sort();\n\t\t        if (firstObj.length !== secondObj.length) {\n\t\t\t        areEqual = false;\n\t\t        } else {\n\t\t\t        for (let i = 0; i < firstObj.length; ++i) {\n\t\t\t\t        if (!objectsAreEqual(firstObj[i], secondObj[i])) {\n\t\t\t\t\t        areEqual = false;\n\t\t\t\t\t        break;\n\t\t\t\t        }\n\t\t\t        }\n\t\t        }\n\t        } else if((typeof firstObj === 'function' && typeof secondObj === 'function') ||\n\t\t        (firstObj instanceof Date && secondObj instanceof Date) ||\n\t\t        (firstObj instanceof RegExp && secondObj instanceof RegExp) ||\n\t\t        (firstObj instanceof String && secondObj instanceof String) ||\n\t\t        (firstObj instanceof Number && secondObj instanceof Number)) {\n                    areEqual = firstObj.toString() === secondObj.toString();\n            } else if(typeof firstObj === 'object' && typeof secondObj === 'object') {\n                areEqual = objectHasFields(firstObj, secondObj);\n            // isNaN(undefined) returns true\n            } else if(isNaN(firstObj) && isNaN(secondObj) && typeof firstObj === 'number' && typeof secondObj === 'number') {\n                areEqual = true;\n            } else {\n                areEqual = false;\n            }\n        }\n\n        return areEqual;\n    }\n\n    /**\n     * Registering assert for evaluating if all properties of the second object are own properties of the first object.\n     * If check fails, the assertFail is invoked.\n     * @param firstObj {Object} - first object\n     * @param secondObj{Object} - second object\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck(\"objectHasFields\", function(firstObj, secondObj, explanation){\n        if(!objectHasFields(firstObj, secondObj)) {\n            sf.exceptions.assertFail(explanation);\n        }\n    });\n\n    /**\n     * Registering assert for evaluating if all element from the second array are present in the first array.\n     * Deep comparison between the elements of the array is used.\n     * If check fails, the assertFail is invoked.\n     * @param firstArray {Array}- first array\n     * @param secondArray {Array} - second array\n     * @param explanation {String} - failing reason message in case the assert fails\n     */\n    sf.assert.addCheck(\"arraysMatch\", function(firstArray, secondArray, explanation){\n        if(firstArray.length !== secondArray.length){\n            sf.exceptions.assertFail(explanation);\n        }\n        else {\n            const result = objectsAreEqual(firstArray, secondArray);\n            // const arraysDontMatch = secondArray.every(element => firstArray.indexOf(element) !== -1);\n            // let arraysDontMatch = secondArray.some(function (expectedElement) {\n            //     let found = firstArray.some(function(resultElement){\n            //         return objectHasFields(resultElement,expectedElement);\n            //     });\n            //     return found === false;\n            // });\n\n            if(!result){\n                sf.exceptions.assertFail(explanation);\n            }\n        }\n    });\n\n    // added mainly for test purposes, better test frameworks like mocha could be much better\n\n    /**\n     * Registering assert for checking if a function is failing.\n     * If the function is throwing an exception, the test is passed or failed otherwise.\n     * @param testName {String} - test name or description\n     * @param func {Function} - function to be invoked\n     */\n    sf.assert.addCheck('fail', function(testName, func){\n        try{\n            func();\n            logger.recordAssert(\"[Fail] \" + testName);\n        } catch(err){\n            logger.recordAssert(\"[Pass] \" + testName);\n        }\n    });\n\n    /**\n     * Registering assert for checking if a function is executed with no exceptions.\n     * If the function is not throwing any exception, the test is passed or failed otherwise.\n     * @param testName {String} - test name or description\n     * @param func {Function} - function to be invoked\n     */\n    sf.assert.addCheck('pass', function(testName, func){\n        try{\n            func();\n            logger.recordAssert(\"[Pass] \" + testName);\n        } catch(err){\n            logger.recordAssert(\"[Fail] \" + testName, err.stack);\n        }\n    });\n\n    /**\n     * Alias for the pass assert.\n     */\n    sf.assert.alias('test', 'pass');\n\n    /**\n     * Registering assert for checking if a callback function is executed before timeout is reached without any exceptions.\n     * If the function is throwing any exception or the timeout is reached, the test is failed or passed otherwise.\n     * @param testName {String} - test name or description\n     * @param func {Function} - function to be invoked\n     * @param timeout {Number} - number of milliseconds for the timeout check. Default to 500ms.\n     */\n    sf.assert.addCheck('callback', function(testName, func, timeout){\n\n        if(!func || typeof func != \"function\"){\n            throw new Error(\"Wrong usage of assert.callback!\");\n        }\n\n        if(!timeout){\n            timeout = 500;\n        }\n\n        var passed = false;\n        function callback(){\n            if(!passed){\n                passed = true;\n                logger.recordAssert(\"[Pass] \" + testName);\n                successTest();\n            } else {\n                logger.recordAssert(\"[Fail (multiple calls)] \" + testName);\n            }\n        }\n        \n        try{\n            func(callback);\n        } catch(err){\n            logger.recordAssert(\"[Fail] \" + testName,  err, true);\n        }\n\n        function successTest(force){\n            if(!passed){\n                logger.recordAssert(\"[Fail Timeout] \" + testName );\n            }\n        }\n\n        setTimeout(successTest, timeout);\n    });\n\n    /**\n     * Registering assert for checking if an array of callback functions are executed in a waterfall manner,\n     * before timeout is reached without any exceptions.\n     * If any of the functions is throwing any exception or the timeout is reached, the test is failed or passed otherwise.\n     * @param testName {String} - test name or description\n     * @param func {Function} - function to be invoked\n     * @param timeout {Number} - number of milliseconds for the timeout check. Default to 500ms.\n     */\n    sf.assert.addCheck('steps', function(testName, arr, timeout){\n        if(!timeout){\n            timeout = 500;\n        }\n\n        var currentStep = 0;\n        var passed = false;\n\n        function next(){\n            if(currentStep === arr.length){\n                passed = true;\n                logger.recordAssert(\"[Pass] \" + testName );\n                return;\n            }\n\n            var func = arr[currentStep];\n            currentStep++;\n            try{\n                func(next);\n            } catch(err){\n                logger.recordAssert(\"[Fail] \" + testName  + \" [at step \" + currentStep + \"]\", err);\n            }\n        }\n\n        function successTest(force){\n            if(!passed){\n                logger.recordAssert(\"[Fail Timeout] \" + testName  + \" [at step \" + currentStep + \"]\");\n            }\n        }\n\n        setTimeout(successTest, timeout);\n        next();\n    });\n\n    /**\n     * Alias for the steps assert.\n     */\n    sf.assert.alias('waterfall', 'steps');\n\n    /**\n     * Registering assert for asynchronously printing all execution summary from logger.dumpWhys.\n     * @param message {String} - message to be recorded\n     * @param timeout {Number} - number of milliseconds for the timeout check. Default to 500ms.\n     */\n    sf.assert.addCheck('end', function(timeout, silence){\n        if(!timeout){\n            timeout = 1000;\n        }\n\n        function handler() {\n            logger.dumpWhys().forEach(function(c){\n                const executionSummary = c.getExecutionSummary();\n                console.log(JSON.stringify(executionSummary, null, 4));\n            });\n\n            if(!silence){\n                console.log(\"Forcing exit after\", timeout, \"ms\");\n            }\n            process.exit(0);\n        }\n\n        setTimeout(handler, timeout);\n    });\n\n    /**\n     * Registering assert for printing a message and asynchronously printing all logs from logger.dumpWhys.\n     * @param message {String} - message to be recorded\n     * @param timeout {Number} - number of milliseconds for the timeout check. Default to 500ms.\n     */\n    sf.assert.addCheck('begin', function(message, timeout){\n        logger.recordAssert(message);\n        sf.assert.end(timeout, true);\n    });\n};","/*\n    checks are like asserts but are intended to be used in production code to help debugging and signaling wrong behaviours\n\n */\n\nexports.init = function(sf){\n    sf.exceptions.register('checkFail', function(explanation, err){\n        var stack;\n        if(err){\n            stack = err.stack;\n        }\n        console.log(\"Check failed \", explanation, stack);\n    });\n\n    sf.check.addCheck('equal', function(v1 , v2, explanation){\n\n        if(v1 !== v2){\n            if(!explanation){\n                explanation =  \" [\"+ v1 + \" != \" + v2 + \"]\";\n            }\n\n            sf.exceptions.checkFail(explanation);\n        }\n    });\n\n\n    sf.check.addCheck('true', function(b, explanation){\n        if(!b){\n            if(!explanation){\n                explanation =  \" expression is false but is expected to be true\";\n            }\n\n            sf.exceptions.checkFail(explanation);\n        }\n    });\n\n\n    sf.check.addCheck('false', function(b, explanation){\n        if(b){\n            if(!explanation){\n                explanation =  \" expression is true but is expected to be false\";\n            }\n\n            sf.exceptions.checkFail(explanation);\n        }\n    });\n\n    sf.check.addCheck('notequal', function(v1 , v2, explanation){\n        if(v1 == v2){\n            if(!explanation){\n                explanation =  \" [\"+ v1 + \" == \" + v2 + \"]\";\n            }\n            sf.exceptions.checkFail(explanation);\n        }\n    });\n\n\n    /*\n        added mainly for test purposes, better test frameworks like mocha could be much better :)\n    */\n    sf.check.addCheck('fail', function(testName ,func){\n        try{\n            func();\n            console.log(\"[Fail] \" + testName );\n        } catch(err){\n            console.log(\"[Pass] \" + testName );\n        }\n    });\n\n\n    sf.check.addCheck('pass', function(testName ,func){\n        try{\n            func();\n            console.log(\"[Pass] \" + testName );\n        } catch(err){\n            console.log(\"[Fail] \" + testName  ,  err.stack);\n        }\n    });\n\n\n    sf.check.alias('test','pass');\n\n\n    sf.check.addCheck('callback', function(testName ,func, timeout){\n        if(!timeout){\n            timeout = 500;\n        }\n        var passed = false;\n        function callback(){\n            if(!passed){\n                passed = true;\n                console.log(\"[Pass] \" + testName );\n                SuccessTest();\n            } else {\n                console.log(\"[Fail (multiple calls)] \" + testName );\n            }\n        }\n        try{\n            func(callback);\n        } catch(err){\n            console.log(\"[Fail] \" + testName  ,  err.stack);\n        }\n\n        function SuccessTest(force){\n            if(!passed){\n                console.log(\"[Fail Timeout] \" + testName );\n            }\n        }\n\n        setTimeout(SuccessTest, timeout);\n    });\n\n\n    sf.check.addCheck('steps', function(testName , arr, timeout){\n        var  currentStep = 0;\n        var passed = false;\n        if(!timeout){\n            timeout = 500;\n        }\n\n        function next(){\n            if(currentStep === arr.length){\n                passed = true;\n                console.log(\"[Pass] \" + testName );\n                return ;\n            }\n            var func = arr[currentStep];\n            currentStep++;\n            try{\n                func(next);\n            } catch(err){\n                console.log(\"[Fail] \" + testName  ,\"\\n\\t\" , err.stack + \"\\n\\t\" , \" [at step \", currentStep + \"]\");\n            }\n        }\n\n        function SuccessTest(force){\n            if(!passed){\n                console.log(\"[Fail Timeout] \" + testName + \"\\n\\t\" , \" [at step \", currentStep+ \"]\");\n            }\n        }\n\n        setTimeout(SuccessTest, timeout);\n        next();\n    });\n\n    sf.check.alias('waterfall','steps');\n    sf.check.alias('notEqual','notequal');\n\n    sf.check.addCheck('end', function(timeOut, silence){\n        if(!timeOut){\n            timeOut = 1000;\n        }\n\n        setTimeout(function(){\n            if(!silence){\n                console.log(\"Forcing exit after\", timeOut, \"ms\");\n            }\n            process.exit(0);\n        }, timeOut);\n    });\n\n\n    sf.check.addCheck('begin', function(message, timeOut){\n        console.log(message);\n        sf.check.end(timeOut, true);\n    });\n\n\n};","exports.init = function(sf){\n    /**\n     * Registering unknown exception handler.\n     */\n    sf.exceptions.register('unknown', function(explanation){\n        explanation = explanation || \"\";\n        const message = \"Unknown exception\" + explanation;\n        throw(message);\n    });\n\n    /**\n     * Registering resend exception handler.\n     */\n    sf.exceptions.register('resend', function(exceptions){\n        throw(exceptions);\n    });\n\n    /**\n     * Registering notImplemented exception handler.\n     */\n    sf.exceptions.register('notImplemented', function(explanation){\n        explanation = explanation || \"\";\n        const message = \"notImplemented exception\" + explanation;\n        throw(message);\n    });\n\n    /**\n     * Registering security exception handler.\n     */\n    sf.exceptions.register('security', function(explanation){\n        explanation = explanation || \"\";\n        const message = \"security exception\" + explanation;\n        throw(message);\n    });\n\n    /**\n     * Registering duplicateDependency exception handler.\n     */\n    sf.exceptions.register('duplicateDependency', function(variable){\n        variable = variable || \"\";\n        const message = \"duplicateDependency exception\" + variable;\n        throw(message);\n    });\n};","const LOG_LEVELS = {\n    HARD_ERROR:     0,  // system level critical error: hardError\n    ERROR:          1,  // potentially causing user's data loosing error: error\n    LOG_ERROR:      2,  // minor annoyance, recoverable error:   logError\n    UX_ERROR:       3,  // user experience causing issues error:  uxError\n    WARN:           4,  // warning,possible isues but somehow unclear behaviour: warn\n    INFO:           5,  // store general info about the system working: info\n    DEBUG:          6,  // system level debug: debug\n    LOCAL_DEBUG:    7,  // local node/service debug: ldebug\n    USER_DEBUG:     8,  // user level debug; udebug\n    DEV_DEBUG:      9,  // development time debug: ddebug\n    WHYS:            10, // whyLog for code reasoning\n    TEST_RESULT:    11, // testResult to log running tests\n};\n\nexports.init = function(sf){\n\n    /**\n     * Records log messages from various use cases.\n     * @param record {String} - log message.\n     */\n    sf.logger.record = function(record){\n        var displayOnConsole = true;\n        if(process.send) {\n            process.send(record);\n            displayOnConsole = false;\n        }\n\n        if(displayOnConsole) {\n            const prettyLog = JSON.stringify(record, null, 2);\n            console.log(prettyLog);\n        }\n    };\n\n    /**\n     * Adding case for logging system level critical errors.\n     */\n    sf.logger.addCase('hardError', function(message, exception, args, pos, data){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.HARD_ERROR, 'systemError', message, exception, true, args, pos, data));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging potentially causing user's data loosing errors.\n     */\n    sf.logger.addCase('error', function(message, exception, args, pos, data){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.ERROR, 'error', message, exception, true, args, pos, data));\n    }, [\n        {\n            'message':'explanation'\n        },\n        {\n            'exception':'exception'\n        }\n    ]);\n\n    /**\n     * Adding case for logging minor annoyance, recoverable errors.\n     */\n    sf.logger.addCase('logError', function(message, exception, args, pos, data){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.LOG_ERROR, 'logError', message, exception, true, args, pos, data));\n    }, [\n        {\n            'message':'explanation'\n        },\n        {\n            'exception':'exception'\n        }\n    ]);\n\n    /**\n     * Adding case for logging user experience causing issues errors.\n     */\n    sf.logger.addCase('uxError', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.UX_ERROR, 'uxError', message, null, false));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging throttling messages.\n     */\n    sf.logger.addCase('throttling', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.WARN, 'throttling', message, null, false));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging warning, possible issues, but somehow unclear behaviours.\n     */\n    sf.logger.addCase('warning', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.WARN, 'warning', message,null, false, arguments, 0));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n    \n    sf.logger.alias('warn', 'warning');\n\n    /**\n     * Adding case for logging general info about the system working.\n     */\n    sf.logger.addCase('info', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.INFO, 'info', message,null, false, arguments, 0));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging system level debug messages.\n     */\n    sf.logger.addCase('debug', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.DEBUG, 'debug', message,null, false, arguments, 0));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n\n    /**\n     * Adding case for logging local node/service debug messages.\n     */\n    sf.logger.addCase('ldebug', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.LOCAL_DEBUG, 'ldebug', message, null, false, arguments, 0));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging user level debug messages.\n     */\n    sf.logger.addCase('udebug', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.USER_DEBUG, 'udebug', message ,null, false, arguments, 0));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging development debug messages.\n     */\n    sf.logger.addCase('devel', function(message){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.DEV_DEBUG, 'devel', message, null, false, arguments, 0));\n    }, [\n        {\n            'message':'explanation'\n        }\n    ]);\n\n    /**\n     * Adding case for logging \"whys\" reasoning messages.\n     */\n    sf.logger.addCase(\"logWhy\", function(logOnlyCurrentWhyContext){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.WHYS, 'logwhy', undefined, undefined, undefined, undefined, undefined, undefined, logOnlyCurrentWhyContext));\n    });\n\n    /**\n     * Adding case for logging asserts messages to running tests.\n     */\n    sf.logger.addCase(\"recordAssert\", function (message, error,showStack){\n        sf.logger.record(createDebugRecord(LOG_LEVELS.TEST_RESULT, 'assert', message, error, showStack));\n    });\n\n    /**\n     * Generic method to create structured debug records based on the log level.\n     * @param level {Number} - number from 1-11, used to identify the level of attention that a log entry should get from operations point of view\n     * @param type {String} - identifier name for log type\n     * @param message {String} - description of the debug record\n     * @param exception {String} - exception details if any\n     * @param saveStack {Boolean} - if set to true, the exception call stack will be added to the debug record\n     * @param args {Array} - arguments of the caller function\n     * @param pos {Number} - position\n     * @param data {String|Number|Array|Object} - payload information\n     * @param logOnlyCurrentWhyContext - if whys is enabled, only the current context will be logged\n     * @returns Debug record model {Object} with the following fields:\n     * [required]: level: *, type: *, timestamp: number, message: *, data: * and\n     * [optional]: stack: *, exception: *, args: *, whyLog: *\n     */\n    function createDebugRecord(level, type, message, exception, saveStack, args, pos, data, logOnlyCurrentWhyContext){\n\n        var ret = {\n            level: level,\n            type: type,\n            timestamp: (new Date()).getTime(),\n            message: message,\n            data: data\n        };\n\n        if(saveStack){\n            var stack = '';\n            if(exception){\n                stack = exception.stack;\n            } else {\n                stack  = (new Error()).stack;\n            }\n            ret.stack = stack;\n        }\n\n        if(exception){\n            ret.exception = exception.message;\n        }\n\n        if(args){\n            ret.args = JSON.parse(JSON.stringify(args));\n        }\n\n        if(process.env.RUN_WITH_WHYS){\n            var why = require('whys');\n            if(logOnlyCurrentWhyContext) {\n                ret['whyLog'] = why.getGlobalCurrentContext().getExecutionSummary();\n            }else{\n                ret['whyLog'] = why.getAllContexts().map(function (context) {\n                    return context.getExecutionSummary();\n                });\n            }\n        }\n\n        return ret;\n    }\n\n};\n\n","const fs = require(\"fs\");\nconst path = require(\"path\");\nconst forker = require('child_process');\n\nconst DEFAULT_TIMEOUT = 2000;\n\nvar globToRegExp =  require(\"./utils/glob-to-regexp\");\n\nvar defaultConfig = {\n    confFileName: \"double-check.json\",      // name of the conf file\n    fileExt: \".js\",                         // test file supported by extension\n    matchDirs: [ 'test', 'tests' ],           // dirs names for tests - case insensitive (used in discovery process)\n    testsDir: process.cwd(),                // path to the root tests location\n    reports: {\n        basePath: process.cwd(),            // path where the reports will be saved\n        prefix: \"Report-\",                  // prefix for report files, filename pattern: [prefix]-{timestamp}{ext}\n        ext: \".txt\"                         // report file extension\n    }\n};\n\nconst TAG = \"[TEST_RUNNER]\";\nconst MAX_WORKERS = process.env['DOUBLE_CHECK_POOL_SIZE'] || 10;\nconst DEBUG = typeof v8debug === 'object';\n\nconst TEST_STATES = {\n    READY: 'ready',\n    RUNNING: 'running',\n    FINISHED: 'finished',\n    TIMEOUT: 'timeout'\n};\n\n// Session object\nvar defaultSession = {\n    testCount: 0,\n    currentTestIndex: 0,\n    debugPort: process.debugPort,   // current process debug port. The child process will be increased from this port\n    workers: {\n        running: 0,\n        terminated: 0\n    }\n};\n\n// Template structure for test reports.\nvar reportFileStructure = {\n    count: 0,\n    suites: {\n        count: 0,\n        items: []\n    },\n    passed: {\n        count: 0,\n        items: []\n    },\n    failed: {\n        count: 0,\n        items: []\n    },\n};\n\nexports.init = function(sf){\n    sf.testRunner = {\n        /**\n         * Initialization of the test runner.\n         * @param config {Object} - settings object that will be merged with the default one\n         * @private\n         */\n        __init: function(config) {\n            this.config = this.__extend(defaultConfig, config);\n            this.testTree = {};\n            this.testList = [];\n\n            this.session = defaultSession;\n\n            // create reports directory if not exist\n            if (!fs.existsSync(this.config.reports.basePath)){\n                fs.mkdirSync(this.config.reports.basePath);\n            }\n        },\n        /**\n         * Main entry point. It will start the flow runner flow.\n         * @param config {Object} - object containing settings such as conf file name, test dir.\n         * @param callback {Function} - handler(error, result) invoked when an error occurred or the runner has completed all jobs.\n         */\n        start: function(config, callback) {\n\n            // wrapper for provided callback, if any\n            this.callback = function(err, result) {\n                if(err) {\n                    this.__debugInfo(err.message || err);\n                }\n\n                if(callback) {\n                    return callback(err, result);\n                }\n            };\n\n            this.__init(config);\n\n            this.__consoleLog(\"Discovering tests ...\");\n            this.testTree = this.__discoverTestFiles(this.config.testsDir, config);\n            this.testList = this.__toTestTreeToList(this.testTree);\n            this.__launchTests();\n        },\n        /**\n         * Reads configuration settings from a json file.\n         * @param confPath {String} - absolute path to the configuration file.\n         * @returns {Object} - configuration object {{}}\n         * @private\n         */\n        __readConf: function(confPath) {\n            var config = {};\n            try{\n                config = require(confPath);\n            } catch(error) {\n                console.error(error);\n            }\n\n            return config;\n        },\n        /**\n         * Discovers test files recursively starting from a path. The dir is the root of the test files. It can contains\n         * test files and test sub directories. It will create a tree structure with the test files discovered.\n         * Notes: Only the config.matchDirs will be taken into consideration. Also, based on the conf (double-check.json)\n         * it will include the test files or not.\n         * @param dir {String} - path where the discovery process starts\n         * @param parentConf {String} - configuration object (double-check.json) from the parent directory\n         * @returns The root node object of the file structure tree. E.g. {*|{__meta, data, result, items}}\n         * @private\n         */\n        __discoverTestFiles: function(dir, parentConf) {\n            const stat = fs.statSync(dir);\n            if(!stat.isDirectory()){\n                throw new Error(dir + \" is not a directory!\");\n            }\n\n            let currentConf = parentConf;\n\n            let currentNode = this.__getDefaultNodeStructure();\n            currentNode.__meta.parent = path.dirname(dir);\n            currentNode.__meta.isDirectory = true;\n\n            let files = fs.readdirSync(dir);\n            // first look for conf file\n            if(files.indexOf(this.config.confFileName) !== -1) {\n                let fd = path.join(dir, this.config.confFileName);\n                let conf = this.__readConf(fd);\n                if(conf) {\n                    currentNode.__meta.conf = conf;\n                    currentConf = conf;\n                }\n            }\n\n            currentNode.data.name = path.basename(dir);\n            currentNode.data.path = dir;\n            currentNode.items = [];\n\n            for(let i = 0, len = files.length; i < len; i++) {\n                let item = files[i];\n\n                let fd = path.join(dir, item);\n                let stat = fs.statSync(fd);\n                let isDir = stat.isDirectory();\n                let isTestDir = this.__isTestDir(fd);\n\n                if(isDir && !isTestDir) {\n                    continue; // ignore dirs that does not follow the naming rule for test dirs\n                }\n\n                if(!isDir && item.match(this.config.confFileName)){\n                    continue; // already processed\n                }\n\n                // exclude files based on glob patterns\n                if(currentConf) {\n                    // currentConf['ignore'] - array of regExp\n                    if(currentConf['ignore']) {\n                        const isMatch = this.__isAnyMatch(currentConf['ignore'], item);\n                        if(isMatch) {continue;}\n                    }\n                }\n\n                let childNode = this.__getDefaultNodeStructure();\n                childNode.__meta.conf = {};\n                childNode.__meta.isDirectory = isDir;\n                childNode.__meta.parent = path.dirname(fd);\n\n                if (isDir) {\n                    let tempChildNode = this.__discoverTestFiles(fd, currentConf);\n                    childNode = Object.assign(childNode, tempChildNode);\n                    currentNode.items.push(childNode);\n                }\n                else if(path.extname(fd) ===  this.config.fileExt){\n                    childNode.__meta.conf.runs = currentConf['runs'] || 1;\n                    childNode.__meta.conf.silent = currentConf['silent'];\n                    childNode.__meta.conf.timeout = currentConf['timeout'] || DEFAULT_TIMEOUT;\n\n                    childNode.data.name = item;\n                    childNode.data.path = fd;\n\n                    currentNode.items.push(childNode);\n                }\n            }\n\n            return currentNode;\n        },\n        /**\n         * Launch collected tests. Initialises session variables, that are specific for the current launch.\n         * @private\n         */\n        __launchTests: function() {\n            this.__consoleLog(\"Launching tests ...\");\n            this.session.testCount = this.testList.length;\n            this.session.processedTestCount = 0;\n            this.session.workers.running = 0;\n            this.session.workers.terminated = 0;\n\n            if(this.session.testCount > 0) {\n                this.__scheduleWork();\n            } else {\n                this.__doTestReports();\n            }\n        },\n        /**\n         * Schedules work based on the MAX available workers, and based on the number of runs of a test.\n         * If a test has multiple runs as a option, it will be started in multiple workers. Once all runs are completed,\n         * the test is considered as processed.\n         * @private\n         */\n        __scheduleWork: function() {\n            while(this.session.workers.running < MAX_WORKERS && this.session.currentTestIndex < this.session.testCount){\n                let test = this.testList[this.session.currentTestIndex];\n                if(test.result.runs < test.__meta.conf.runs) {\n                    test.result.runs++;\n                    this.__launchTest(test);\n                } else {\n                    this.session.currentTestIndex++;\n                }\n            }\n        },\n        /**\n         * Launch a test into a separate worker (child process).\n         * Each worker has handlers for message, exit and error events. Once the exit or error event is invoked,\n         * new work is scheduled and session object is updated.\n         * Notes: On debug mode, the workers will receive a debug port, that is increased incrementally.\n         * @param test {Object} - test object\n         * @private\n         */\n        __launchTest: function(test) {\n            this.session.workers.running++;\n\n            test.result.state = TEST_STATES.RUNNING;\n            test.result.pass = true;\n            test.result.asserts[test.result.runs] = [];\n            test.result.messages[test.result.runs] = [];\n\n            let env = process.env;\n\n            let execArgv = [];\n            if(DEBUG) {\n                const debugPort = ++defaultSession.debugPort;\n                const debugFlag = '--debug=' + debugPort;\n                execArgv.push(debugFlag);\n            }\n\n            const cwd = test.__meta.parent;\n\n            let worker = forker.fork(test.data.path, [], {'cwd': cwd, 'env': env, 'execArgv': execArgv, stdio: [ 'inherit', \"pipe\", 'inherit', 'ipc' ], silent:false });\n\n            this.__debugInfo(`Launching test ${test.data.name}, run[${test.result.runs}], on worker pid[${worker.pid}] `+new Date().getTime());\n\n            worker.on(\"message\", onMessageEventHandlerWrapper(test));\n            worker.on(\"exit\", onExitEventHandlerWrapper(test));\n            worker.on(\"error\", onErrorEventHandlerWrapper(test));\n\n            worker.terminated = false;\n\n            worker.stdout.on('data', function (chunk) {\n                let content = new Buffer(chunk).toString('utf8'); //TODO: replace with PSKBUFFER\n                if(test.__meta.conf.silent) {\n                    this.__consoleLog(content);\n                }\n            }.bind(this));\n\n            var self = this;\n            function onMessageEventHandlerWrapper(test) {\n                const currentRun = test.result.runs;\n                return function(log) {\n                    if(log.type === 'assert'){\n                        if(log.message.includes(\"[Fail\")) {\n                            test.result.pass = false;\n                        }\n                        test.result.asserts[currentRun].push(log);\n                    } else {\n                        test.result.messages[currentRun].push(log);\n                    }\n                };\n            }\n\n            function onExitEventHandlerWrapper(test) {\n                return function(code, signal) {\n                    clearTimeout(worker.timerVar);\n                    self.__debugInfo(`Worker ${worker.pid} - exit event. Code ${code}, signal ${signal} `+new Date().getTime());\n\n                    worker.terminated = true;\n\n                    test.result.state = TEST_STATES.FINISHED;\n                    if(code !== null && code!==0 /*&& typeof test.result.pass === 'undefined'*/){\n                        test.result.pass = false;\n                        test.result.messages[test.result.runs].push( {message: \"Process finished with errors!\", \"Exit code\":code, \"Signal\":signal});\n                    }\n\n                    self.session.workers.running--;\n                    self.session.workers.terminated++;\n\n                    self.__scheduleWork();\n                    self.__checkWorkersStatus();\n                };\n            }\n\n            // this handler can be triggered when:\n            // 1. The process could not be spawned, or\n            // 2. The process could not be killed, or\n            // 3. Sending a message to the child process failed.\n            // IMPORTANT: The 'exit' event may or may not fire after an error has occurred!\n            function onErrorEventHandlerWrapper(test) {\n                return function(error) {\n                    self.__debugInfo(`Worker ${worker.pid} - error event.`, test);\n                    self.__debugError(error);\n\n                    self.session.workers.running--;\n                    self.session.workers.terminated++;\n                };\n            }\n\n            // Note: on debug, the timeout is reached before exit event is called\n            // when kill is called, the exit event is raised\n            worker.timerVar = setTimeout(()=>{\n                if(!worker.terminated){\n                    this.__consoleLog(`worker pid [${worker.pid}] - timeout event`,new Date().getTime(),  test);\n\n                    if(test.result.state !== TEST_STATES.FINISHED){\n                        test.result.pass = false;\n                    }\n                    worker.kill();\n                    test.result.state = TEST_STATES.TIMEOUT;\n                }else{\n                    console.log(\"Got something, but don't know what...\", test);\n                }\n            }, test.__meta.conf.timeout);\n\n                self.__debugInfo(`Worker ${worker.pid} - set timeout event at `+new Date().getTime() + \" for \"+test.__meta.conf.timeout);\n\n        },\n        /**\n         * Checks if all workers completed their job (finished or have been terminated).\n         * If true, then the reporting steps can be started.\n         * @private\n         */\n        __checkWorkersStatus: function() {\n            if(this.session.workers.running === 0) {\n                this.__doTestReports();\n            }\n        },\n        /**\n         * Creates test reports object (JSON) that will be saved in the test report.\n         * Filename of the report is using the following pattern: {prefix}-{timestamp}{ext}\n         * The file will be saved in config.reports.basePath.\n         * @private\n         */\n        __doTestReports: function() {\n            this.__consoleLog(\"Doing reports ...\");\n            reportFileStructure.count = this.testList.length;\n\n            // pass/failed tests\n            for(let i = 0, len = this.testList.length; i < len; i++) {\n                let test = this.testList[i];\n\n                let testPath = this.__toRelativePath(test.data.path);\n                let item = {path: testPath};\n                if(test.result.pass) {\n                    item.reason = this.__getFirstFailReasonPerRun(test);\n                    reportFileStructure.passed.items.push(item);\n                } else {\n                    item.reason = this.__getFirstFailReasonPerRun(test);\n                    reportFileStructure.failed.items.push(item);\n                }\n            }\n            reportFileStructure.passed.count = reportFileStructure.passed.items.length;\n            reportFileStructure.failed.count = reportFileStructure.failed.items.length;\n\n            // suites (first level of directories)\n            for(let i = 0, len = this.testTree.items.length; i < len; i++) {\n                let item = this.testTree.items[i];\n                if(item.__meta.isDirectory) {\n                    let suitePath = this.__toRelativePath(item.data.path);\n                    reportFileStructure.suites.items.push(suitePath);\n                }\n            }\n            reportFileStructure.suites.count = reportFileStructure.suites.items.length;\n\n            let numberOfReports = 2;\n\n            let finishReports = (err, res) => {\n                if(numberOfReports > 1){\n                    numberOfReports -= 1;\n                    return;\n                }\n                if(reportFileStructure.failed.count === 0){\n                    this.__consoleLog(\"\\nEverything went well! No failed tests.\\n\\n\");\n                }else{\n                    this.__consoleLog(\"\\nSome tests failed. Check report files!\\n\\n\");\n                }\n\n                this.callback(err, \"Done\");\n            };\n\n\n            this.__consoleLog(this.config.reports.prefix);\n            const fileName = `${this.config.reports.prefix}latest${this.config.reports.ext}`;\n            const filePath = path.join(this.config.reports.basePath, fileName);\n            this.__saveReportToFile(reportFileStructure, filePath, finishReports);\n\n            const timestamp = new Date().getTime().toString();\n            const htmlFileName = `${this.config.reports.prefix}latest.html`;\n            const htmlFilePath = path.join(this.config.reports.basePath, htmlFileName);\n            this.__saveHtmlReportToFile(reportFileStructure, htmlFilePath, timestamp, finishReports);\n        },\n        /**\n         * Saves test reports object (JSON) in the specified path.\n         * @param reportFileStructure {Object} - test reports object (JSON)\n         * @param destination {String} - path of the file report (the base path MUST exist)\n         * @private\n         */\n        __saveReportToFile: function(reportFileStructure, destination, callback) {\n\n            var content = JSON.stringify(reportFileStructure, null, 4);\n            fs.writeFile(destination, content, 'utf8', function (err) {\n                if (err) {\n                    const message = \"An error occurred while writing the report file, with the following error: \" + JSON.stringify(err);\n                    this.__debugInfo(message);\n                    throw err;\n                } else{\n                    const message = `Finished writing report to ${destination}`;\n                    this.__consoleLog(message);\n                }\n                callback();\n            }.bind(this));\n        },\n        /**\n         * Saves test reports as HTML in the specified path.\n         * @param reportFileStructure {Object} - test reports object (JSON)\n         * @param destination {String} - path of the file report (the base path MUST exist)\n         * @param timestamp {String} - timestamp to be injected in html template\n         * @private\n         */\n        __saveHtmlReportToFile: function (reportFileStructure, destination, timestamp, callback) {\n            var folderName = path.resolve(__dirname);\n            fs.readFile(path.join(folderName,'/utils/reportTemplate.html'), 'utf8', (err, res) => {\n                if (err) {\n                    const message = 'An error occurred while reading the html report template file, with the following error: ' + JSON.stringify(err);\n                    this.__debugInfo(message);\n                    throw err;\n                }\n\n                fs.writeFile(destination, res + `<script>init(${JSON.stringify(reportFileStructure)}, ${timestamp});</script>`, 'utf8', (err) => {\n                    if (err) {\n                        const message = 'An error occurred while writing the html report file, with the following error: ' + JSON.stringify(err);\n                        this.__debugInfo(message);\n                        throw err;\n                    }\n\n                    const message = `Finished writing report to ${destination}`;\n                    this.__consoleLog(message);\n\n                    callback();\n                });\n            });\n        },\n        /**\n         * Converts absolute file path to relative path.\n         * @param absolutePath {String} - absolute path\n         * @returns {string | void | *} - relative path\n         * @private\n         */\n        __toRelativePath: function(absolutePath) {\n            const basePath = path.join(this.config.testsDir, \"/\");\n            const relativePath = absolutePath.replace(basePath, \"\");\n            return relativePath;\n        },\n        /**\n         * Checks if a directory is a test dir, by matching its name against config.matchDirs array.\n         * @param dir {String} - directory name\n         * @returns {boolean} - returns true if there is a match and false otherwise.\n         * @private\n         */\n        __isTestDir: function(dir) {\n            if(!this.config || !this.config.matchDirs ) {\n                throw `matchDirs is not defined on config ${JSON.stringify(this.config)} does not exist!`;\n            }\n\n            var isTestDir = this.config.matchDirs.some(function(item) {\n                return dir.toLowerCase().includes(item.toLowerCase());\n            });\n\n            return isTestDir;\n        },\n        /**\n         * For a failed test, it returns only the first fail reason per each run.\n         * @param test {Object} - test object\n         * @returns {Array} - an array of reasons per each test run.\n         * @private\n         */\n        __getFirstFailReasonPerRun: function(test) {\n            const reason = [];\n            for(let i = 1; i <= test.result.runs; i++) {\n                if(test.result.asserts[i] && test.result.asserts[i].length > 0) {\n                    addReason(i, test.result.asserts[i][0]);\n                }\n\n                if(test.result.messages[i] && test.result.messages[i].length > 0) {\n                    addReason(i, test.result.messages[i][0]);\n                }\n\n                function addReason(run, log) {\n                    const message = {\n                        run: run,\n                        log: log\n                    };\n\n                    reason.push(message);\n                }\n            }\n\n            return reason;\n        },\n        /**\n         * Described default tree node structure.\n         * @returns {{__meta: {conf: null, parent: null, isDirectory: boolean}, data: {name: null, path: null}, result: {state: string, pass: null, executionTime: number, runs: number, asserts: {}, messages: {}}, items: null}}\n         * @private\n         */\n        __getDefaultNodeStructure: function() {\n            return  {\n                __meta: {\n                    conf: null,\n                    parent: null,\n                    isDirectory: false\n                },\n                data: {\n                    name: null,\n                    path: null,\n                },\n                result: {\n                    state: TEST_STATES.READY, // ready | running | terminated | timeout\n                    pass: null,\n                    executionTime: 0,\n                    runs: 0,\n                    asserts: {},\n                    messages: {}\n                },\n                items: null\n            };\n        },\n        /**\n         * Match a test file path to a UNIX glob expression array. If its any match returns true, otherwise returns false.\n         * @param globExpArray {Array} - an array with glob expression (UNIX style)\n         * @param str {String} - the string to be matched\n         * @returns {boolean} - returns true if there is any match and false otherwise.\n         * @private\n         */\n        __isAnyMatch: function(globExpArray, str) {\n            const hasMatch = function(globExp) {\n                const regex = globToRegExp(globExp);\n                return regex.test(str);\n            };\n\n            return globExpArray.some(hasMatch);\n        },\n        /**\n         * Converts a tree structure into an array list of test nodes. The tree traversal is DFS (Deep-First-Search).\n         * @param rootNode {Object} - root node of the test tree.\n         * @returns {Array} - List of test nodes.\n         * @private\n         */\n        __toTestTreeToList: function(rootNode) {\n            var testList = [];\n\n            traverse(rootNode);\n\n            function traverse(node) {\n                if(!node.__meta.isDirectory || !node.items) {\n                    return;\n                }\n\n                for(let i = 0, len = node.items.length; i < len; i++) {\n                    const item = node.items[i];\n                    if(item.__meta.isDirectory) {\n                        traverse(item);\n                    } else {\n                        testList.push(item);\n                    }\n                }\n            }\n\n            return testList;\n        },\n        /**\n         * Logging to console wrapper.\n         * @param log {String|Object|Number} - log message\n         * @private\n         */\n        __consoleLog: function(log) {\n            console.log(TAG, log);\n        },\n        /**\n         * Logging debugging info messages wrapper.\n         * Logger: console.info\n         * @param log {String|Object|Number} - log message\n         * @private\n         */\n        __debugInfo: function(log) {\n            this.__debug(console.info, log);\n        },\n        /**\n         * Logging debugging error messages wrapper.\n         * Logger: console.error\n         * @param log {String|Object|Number} - log message\n         * @private\n         */\n        __debugError: function(log) {\n            this.__debug(console.error, log);\n        },\n        /**\n         *  Logging debugging messages wrapper. One debug mode, the logging is silent.\n         * @param logger {Function} - handler for logging\n         * @param log {String|Object|Number} - log message\n         * @private\n         */\n        __debug: function(logger, log) {\n            if(!DEBUG) {return;}\n\n            // let prettyLog = JSON.stringify(log, null, 2);\n            logger(\"DEBUG\", log);\n        },\n        /**\n         * Deep extend one object with properties of another object.\n         * If the property exists in both objects the property from the first object is overridden.\n         * @param first {Object} - the first object\n         * @param second {Object} - the second object\n         * @returns {Object} - an object with both properties from the first and second object.\n         * @private\n         */\n        __extend: function (first, second) {\n            for (const key in second) {\n                if (!first.hasOwnProperty(key)) {\n                    first[key] = second[key];\n                } else {\n                    let val = second[key];\n                    if(typeof first[key] === 'object') {\n                        val = this.__extend(first[key], second[key]);\n                    }\n\n                    first[key] = val;\n                }\n            }\n\n            return first;\n        }\n    };\n};\n","\n// globToRegExp turns a UNIX glob expression into a RegEx expression.\n//  Supports all simple glob patterns. Examples: *.ext, /foo/*, ../../path, ^foo.*\n// - single character matching, matching ranges of characters etc. group matching are no supported\n// - flags are not supported\nvar globToRegExp = function (globExp) {\n    if (typeof globExp !== 'string') {\n        throw new TypeError('Glob Expression must be a string!');\n    }\n\n    var regExp = \"\";\n\n    for (let i = 0, len = globExp.length; i < len; i++) {\n        let c = globExp[i];\n\n        switch (c) {\n            case \"/\":\n            case \"$\":\n            case \"^\":\n            case \"+\":\n            case \".\":\n            case \"(\":\n            case \")\":\n            case \"=\":\n            case \"!\":\n            case \"|\":\n                regExp += \"\\\\\" + c;\n                break;\n\n            case \"*\":\n                // treat any number of \"*\" as one\n                while(globExp[i + 1] === \"*\") {\n                    i++;\n                }\n                regExp += \".*\";\n                break;\n\n            default:\n                regExp += c;\n        }\n    }\n\n    // set the regular expression with ^ & $\n    regExp = \"^\" + regExp + \"$\";\n\n    return new RegExp(regExp);\n};\n\nmodule.exports = globToRegExp;","const utils = require(\"swarmutils\");\nconst OwM = utils.OwM;\nvar beesHealer = utils.beesHealer;\nvar fs = require(\"fs\");\nvar path = require(\"path\");\n\n\n//TODO: prevent a class of race condition type of errors by signaling with files metadata to the watcher when it is safe to consume\n\nfunction FolderMQ(folder, callback = () => {}){\n\n\tif(typeof callback !== \"function\"){\n\t\tthrow new Error(\"Second parameter should be a callback function\");\n\t}\n\n\tfolder = path.normalize(folder);\n\n\tfs.mkdir(folder, {recursive: true}, function(err, res){\n\t\tfs.exists(folder, function(exists) {\n\t\t\tif (exists) {\n\t\t\t\treturn callback(null, folder);\n\t\t\t} else {\n\t\t\t\treturn callback(err);\n\t\t\t}\n\t\t});\n\t});\n\n\tfunction mkFileName(swarmRaw){\n\t\tlet meta = OwM.prototype.getMetaFrom(swarmRaw);\n\t\tlet name = `${folder}${path.sep}${meta.swarmId}.${meta.swarmTypeName}`;\n\t\tconst unique = meta.phaseId || $$.uidGenerator.safe_uuid();\n\n\t\tname = name+`.${unique}`;\n\t\treturn path.normalize(name);\n\t}\n\n\tthis.getHandler = function(){\n\t\tif(producer){\n\t\t\tthrow new Error(\"Only one consumer is allowed!\");\n\t\t}\n\t\tproducer = true;\n\t\treturn {\n\t\t\tsendSwarmSerialization: function(serialization, callback){\n\t\t\t\tif(typeof callback !== \"function\"){\n\t\t\t\t\tthrow new Error(\"Second parameter should be a callback function\");\n\t\t\t\t}\n\t\t\t\twriteFile(mkFileName(JSON.parse(serialization)), serialization, callback);\n\t\t\t},\n\t\t\taddStream : function(stream, callback){\n\t\t\t\tif(typeof callback !== \"function\"){\n\t\t\t\t\tthrow new Error(\"Second parameter should be a callback function\");\n\t\t\t\t}\n\n\t\t\t\tif(!stream || !stream.pipe || typeof stream.pipe !== \"function\"){\n\t\t\t\t\treturn callback(new Error(\"Something wrong happened\"));\n\t\t\t\t}\n\n\t\t\t\tlet swarm = \"\";\n\t\t\t\tstream.on('data', (chunk) =>{\n\t\t\t\t\tswarm += chunk;\n\t\t\t\t});\n\n\t\t\t\tstream.on(\"end\", () => {\n\t\t\t\t\twriteFile(mkFileName(JSON.parse(swarm)), swarm, callback);\n\t\t\t\t});\n\n\t\t\t\tstream.on(\"error\", (err) =>{\n\t\t\t\t\tcallback(err);\n\t\t\t\t});\n\t\t\t},\n\t\t\taddSwarm : function(swarm, callback){\n\t\t\t\tif(!callback){\n\t\t\t\t\tcallback = $$.defaultErrorHandlingImplementation;\n\t\t\t\t}else if(typeof callback !== \"function\"){\n\t\t\t\t\tthrow new Error(\"Second parameter should be a callback function\");\n\t\t\t\t}\n\n\t\t\t\tbeesHealer.asJSON(swarm,null, null, function(err, res){\n\t\t\t\t\tif (err) {\n\t\t\t\t\t\tconsole.log(err);\n\t\t\t\t\t}\n\t\t\t\t\twriteFile(mkFileName(res), J(res), callback);\n\t\t\t\t});\n\t\t\t},\n\t\t\tsendSwarmForExecution: function(swarm, callback){\n\t\t\t\tif(!callback){\n\t\t\t\t\tcallback = $$.defaultErrorHandlingImplementation;\n\t\t\t\t}else if(typeof callback !== \"function\"){\n\t\t\t\t\tthrow new Error(\"Second parameter should be a callback function\");\n\t\t\t\t}\n\n\t\t\t\tbeesHealer.asJSON(swarm, OwM.prototype.getMetaFrom(swarm, \"phaseName\"), OwM.prototype.getMetaFrom(swarm, \"args\"), function(err, res){\n\t\t\t\t\tif (err) {\n\t\t\t\t\t\tconsole.log(err);\n\t\t\t\t\t}\n\t\t\t\t\tvar file = mkFileName(res);\n\t\t\t\t\tvar content = JSON.stringify(res);\n\n\t\t\t\t\t//if there are no more FD's for files to be written we retry.\n\t\t\t\t\tfunction wrapper(error, result){\n\t\t\t\t\t\tif(error){\n\t\t\t\t\t\t\tconsole.log(`Caught an write error. Retry to write file [${file}]`);\n\t\t\t\t\t\t\tsetTimeout(()=>{\n\t\t\t\t\t\t\t\twriteFile(file, content, wrapper);\n\t\t\t\t\t\t\t}, 10);\n\t\t\t\t\t\t}else{\n\t\t\t\t\t\t\treturn callback(error, result);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\twriteFile(file, content, wrapper);\n\t\t\t\t});\n\t\t\t}\n\t\t};\n\t};\n\n\tvar recipient;\n\tthis.setIPCChannel = function(processChannel){\n\t\tif(processChannel && !processChannel.send || (typeof processChannel.send) != \"function\"){\n\t\t\tthrow new Error(\"Recipient is not instance of process/child_process or it was not spawned with IPC channel!\");\n\t\t}\n\t\trecipient = processChannel;\n\t\tif(consumer){\n\t\t\tconsole.log(`Channel updated`);\n\t\t\t(recipient || process).on(\"message\", receiveEnvelope);\n\t\t}\n\t};\n\n\n\tvar consumedMessages = {};\n\n\tfunction checkIfConsummed(name, message){\n\t\tconst shortName = path.basename(name);\n\t\tconst previousSaved = consumedMessages[shortName];\n\t\tlet result = false;\n\t\tif(previousSaved && !previousSaved.localeCompare(message)){\n\t\t\tresult = true;\n\t\t}\n\t\treturn result;\n\t}\n\n\tfunction save2History(envelope){\n\t\tconsumedMessages[path.basename(envelope.name)] = envelope.message;\n\t}\n\n\tfunction buildEnvelopeConfirmation(envelope, saveHistory){\n\t\tif(saveHistory){\n\t\t\tsave2History(envelope);\n\t\t}\n\t\treturn `Confirm envelope ${envelope.timestamp} sent to ${envelope.dest}`;\n\t}\n\n\tfunction buildEnvelope(name, message){\n\t\treturn {\n\t\t\tdest: folder,\n\t\t\tsrc: process.pid,\n\t\t\ttimestamp: new Date().getTime(),\n\t\t\tmessage: message,\n\t\t\tname: name\n\t\t};\n\t}\n\n\tfunction receiveEnvelope(envelope){\n\t\tif(!envelope || typeof envelope !== \"object\"){\n\t\t\treturn;\n\t\t}\n\t\t//console.log(\"received envelope\", envelope, folder);\n\n\t\tif(envelope.dest !== folder && folder.indexOf(envelope.dest)!== -1 && folder.length === envelope.dest+1){\n\t\t\tconsole.log(\"This envelope is not for me!\");\n\t\t\treturn;\n\t\t}\n\n\t\tlet message = envelope.message;\n\n\t\tif(callback){\n\t\t\t//console.log(\"Sending confirmation\", process.pid);\n\t\t\trecipient.send(buildEnvelopeConfirmation(envelope, true));\n\t\t\tconsumer(null, JSON.parse(message));\n\t\t}\n\t}\n\n\tthis.registerAsIPCConsumer = function(callback){\n\t\tif(typeof callback !== \"function\"){\n\t\t\tthrow new Error(\"The argument should be a callback function\");\n\t\t}\n\t\tregisteredAsIPCConsumer = true;\n\t\t//will register as normal consumer in order to consume all existing messages but without setting the watcher\n\t\tthis.registerConsumer(callback, true, (watcher) => !watcher);\n\n\t\t//console.log(\"Registered as IPC Consummer\", );\n\t\t(recipient || process).on(\"message\", receiveEnvelope);\n\t};\n\n\tthis.registerConsumer = function (callback, shouldDeleteAfterRead = true, shouldWaitForMore = (watcher) => true) {\n\t\tif(typeof callback !== \"function\"){\n\t\t\tthrow new Error(\"First parameter should be a callback function\");\n\t\t}\n\t\tif (consumer) {\n\t\t\tthrow new Error(\"Only one consumer is allowed! \" + folder);\n\t\t}\n\n\t\tconsumer = callback;\n\n\t\tfs.mkdir(folder, {recursive: true}, function (err, res) {\n\t\t\tif (err && (err.code !== 'EEXIST')) {\n\t\t\t\tconsole.log(err);\n\t\t\t}\n\t\t\tconsumeAllExisting(shouldDeleteAfterRead, shouldWaitForMore);\n\t\t});\n\t};\n\n\tthis.writeMessage = writeFile;\n\n\tthis.unlinkContent = function (messageId, callback) {\n\t\tconst messagePath = path.join(folder, messageId);\n\n\t\tfs.unlink(messagePath, (err) => {\n\t\t\tcallback(err);\n\t\t});\n\t};\n\n\tthis.dispose = function(force){\n\t\tif(typeof folder != \"undefined\"){\n\t\t\tvar files;\n\t\t\ttry{\n\t\t\t\tfiles = fs.readdirSync(folder);\n\t\t\t}catch(error){\n\t\t\t\t//..\n\t\t\t}\n\n\t\t\tif(files && files.length > 0 && !force){\n\t\t\t\tconsole.log(\"Disposing a channel that still has messages! Dir will not be removed!\");\n\t\t\t\treturn false;\n\t\t\t}else{\n\t\t\t\ttry{\n\t\t\t\t\tfs.rmdirSync(folder);\n\t\t\t\t}catch(err){\n\t\t\t\t\t//..\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tfolder = null;\n\t\t}\n\n\t\tif(producer){\n\t\t\t//no need to do anything else\n\t\t}\n\n\t\tif(typeof consumer != \"undefined\"){\n\t\t\tconsumer = () => {};\n\t\t}\n\n\t\tif(watcher){\n\t\t\twatcher.close();\n\t\t\twatcher = null;\n\t\t}\n\n\t\treturn true;\n\t};\n\n\n\t/* ---------------- protected  functions */\n\tvar consumer = null;\n\tvar registeredAsIPCConsumer = false;\n\tvar producer = null;\n\n\tfunction buildPathForFile(filename){\n\t\treturn path.normalize(path.join(folder, filename));\n\t}\n\n\tfunction consumeMessage(filename, shouldDeleteAfterRead, callback) {\n\t\tvar fullPath = buildPathForFile(filename);\n\n\t\tfs.readFile(fullPath, \"utf8\", function (err, data) {\n\t\t\tif (!err) {\n\t\t\t\tif (data !== \"\") {\n\t\t\t\t\ttry {\n\t\t\t\t\t\tvar message = JSON.parse(data);\n\t\t\t\t\t} catch (error) {\n\t\t\t\t\t\tconsole.log(\"Parsing error\", error);\n\t\t\t\t\t\terr = error;\n\t\t\t\t\t}\n\n\t\t\t\t\tif(checkIfConsummed(fullPath, data)){\n\t\t\t\t\t\t//console.log(`message already consumed [${filename}]`);\n\t\t\t\t\t\treturn ;\n\t\t\t\t\t}\n\n\t\t\t\t\tif (shouldDeleteAfterRead) {\n\n\t\t\t\t\t\tfs.unlink(fullPath, function (err, res) {\n\t\t\t\t\t\t\tif (err) {throw err;};\n\t\t\t\t\t\t});\n\n\t\t\t\t\t}\n\t\t\t\t\treturn callback(err, message);\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tconsole.log(\"Consume error\", err);\n\t\t\t\treturn callback(err);\n\t\t\t}\n\t\t});\n\t}\n\n\tfunction consumeAllExisting(shouldDeleteAfterRead, shouldWaitForMore) {\n\n\t\tlet currentFiles = [];\n\n\t\tfs.readdir(folder, 'utf8', function (err, files) {\n\t\t\tif (err) {\n\t\t\t\t$$.errorHandler.error(err);\n\t\t\t\treturn;\n\t\t\t}\n\t\t\tcurrentFiles = files;\n\t\t\titerateAndConsume(files);\n\n\t\t});\n\n\t\tfunction startWatching(){\n\t\t\tif (shouldWaitForMore(true)) {\n\t\t\t\twatchFolder(shouldDeleteAfterRead, shouldWaitForMore);\n\t\t\t}\n\t\t}\n\n\t\tfunction iterateAndConsume(files, currentIndex = 0) {\n\t\t\tif (currentIndex === files.length) {\n\t\t\t\t//console.log(\"start watching\", new Date().getTime());\n\t\t\t\tstartWatching();\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif (path.extname(files[currentIndex]) !== in_progress) {\n\t\t\t\tconsumeMessage(files[currentIndex], shouldDeleteAfterRead, (err, data) => {\n\t\t\t\t\tif (err) {\n\t\t\t\t\t\titerateAndConsume(files, ++currentIndex);\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\t\t\t\t\tconsumer(null, data, path.basename(files[currentIndex]));\n\t\t\t\t\tif (shouldWaitForMore()) {\n\t\t\t\t\t\titerateAndConsume(files, ++currentIndex);\n\t\t\t\t\t}\n\t\t\t\t});\n\t\t\t} else {\n\t\t\t\titerateAndConsume(files, ++currentIndex);\n\t\t\t}\n\t\t}\n\t}\n\n\tfunction writeFile(filename, content, callback){\n\t\tif(recipient){\n\t\t\tvar envelope = buildEnvelope(filename, content);\n\t\t\t//console.log(\"Sending to\", recipient.pid, recipient.ppid, \"envelope\", envelope);\n\t\t\trecipient.send(envelope);\n\t\t\tvar confirmationReceived = false;\n\n\t\t\tfunction receiveConfirmation(message){\n\t\t\t\tif(message === buildEnvelopeConfirmation(envelope)){\n\t\t\t\t\t//console.log(\"Received confirmation\", recipient.pid);\n\t\t\t\t\tconfirmationReceived = true;\n\t\t\t\t\ttry{\n\t\t\t\t\t\trecipient.off(\"message\", receiveConfirmation);\n\t\t\t\t\t}catch(err){\n\t\t\t\t\t\t//...\n\t\t\t\t\t}\n\n\t\t\t\t}\n\t\t\t}\n\n\t\t\trecipient.on(\"message\", receiveConfirmation);\n\n\t\t\tsetTimeout(()=>{\n\t\t\t\tif(!confirmationReceived){\n\t\t\t\t\t//console.log(\"No confirmation...\", process.pid);\n\t\t\t\t\thidden_writeFile(filename, content, callback);\n\t\t\t\t}else{\n\t\t\t\t\tif(callback){\n\t\t\t\t\t\treturn callback(null, content);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}, 200);\n\t\t}else{\n\t\t\thidden_writeFile(filename, content, callback);\n\t\t}\n\t}\n\n\tconst in_progress = \".in_progress\";\n\tfunction hidden_writeFile(filename, content, callback){\n\t\tvar tmpFilename = filename+in_progress;\n\t\ttry{\n\t\t\tif(fs.existsSync(tmpFilename) || fs.existsSync(filename)){\n\t\t\t\tconsole.log(new Error(`Overwriting file ${filename}`));\n\t\t\t}\n\t\t\tfs.writeFileSync(tmpFilename, content);\n\t\t\tfs.renameSync(tmpFilename, filename);\n\t\t}catch(err){\n\t\t\treturn callback(err);\n\t\t}\n\t\tcallback(null, content);\n\t}\n\n\tvar alreadyKnownChanges = {};\n\n\tfunction alreadyFiredChanges(filename, change){\n\t\tvar res = false;\n\t\tif(alreadyKnownChanges[filename]){\n\t\t\tres = true;\n\t\t}else{\n\t\t\talreadyKnownChanges[filename] = change;\n\t\t}\n\n\t\treturn res;\n\t}\n\n\tfunction watchFolder(shouldDeleteAfterRead, shouldWaitForMore){\n\n\t\tsetTimeout(function(){\n\t\t\tfs.readdir(folder, 'utf8', function (err, files) {\n\t\t\t\tif (err) {\n\t\t\t\t\t$$.errorHandler.error(err);\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tfor(var i=0; i<files.length; i++){\n\t\t\t\t\twatchFilesHandler(\"change\", files[i]);\n\t\t\t\t}\n\t\t\t});\n\t\t}, 1000);\n\n\t\tfunction watchFilesHandler(eventType, filename){\n\t\t\t//console.log(`Got ${eventType} on ${filename}`);\n\n\t\t\tif(!filename || path.extname(filename) === in_progress){\n\t\t\t\t//caught a delete event of a file\n\t\t\t\t//or\n\t\t\t\t//file not ready to be consumed (in progress)\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tvar f = buildPathForFile(filename);\n\t\t\tif(!fs.existsSync(f)){\n\t\t\t\t//console.log(\"File not found\", f);\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t//console.log(`Preparing to consume ${filename}`);\n\t\t\tif(!alreadyFiredChanges(filename, eventType)){\n\t\t\t\tconsumeMessage(filename, shouldDeleteAfterRead, (err, data) => {\n\t\t\t\t\t//allow a read a the file\n\t\t\t\t\talreadyKnownChanges[filename] = undefined;\n\n\t\t\t\t\tif (err) {\n\t\t\t\t\t\t// ??\n\t\t\t\t\t\tconsole.log(\"\\nCaught an error\", err);\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\n\t\t\t\t\tconsumer(null, data, filename);\n\n\n\t\t\t\t\tif (!shouldWaitForMore()) {\n\t\t\t\t\t\twatcher.close();\n\t\t\t\t\t}\n\t\t\t\t});\n\t\t\t}else{\n\t\t\t\tconsole.log(\"Something happens...\", filename);\n\t\t\t}\n\t\t}\n\n\n\t\tconst watcher = fs.watch(folder, watchFilesHandler);\n\n\t\tconst intervalTimer = setInterval(()=>{\n\t\t\tfs.readdir(folder, 'utf8', function (err, files) {\n\t\t\t\tif (err) {\n\t\t\t\t\t$$.errorHandler.error(err);\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tif(files.length > 0){\n\t\t\t\t\tconsole.log(`\\n\\nFound ${files.length} files not consumed yet in ${folder}`, new Date().getTime(),\"\\n\\n\");\n\t\t\t\t\t//faking a rename event trigger\n\t\t\t\t\twatchFilesHandler(\"rename\", files[0]);\n\t\t\t\t}\n\t\t\t});\n\t\t}, 5000);\n\t}\n}\n\nexports.getFolderQueue = function(folder, callback){\n\treturn new FolderMQ(folder, callback);\n};\n","function PSKBuffer() {}\n\nfunction getArrayBufferInterface () {\n    if(typeof SharedArrayBuffer === 'undefined') {\n        return ArrayBuffer;\n    } else {\n        return SharedArrayBuffer;\n    }\n}\n\nPSKBuffer.from = function (source) {\n    const ArrayBufferInterface = getArrayBufferInterface();\n\n    const buffer = new Uint8Array(new ArrayBufferInterface(source.length));\n    buffer.set(source, 0);\n\n    return buffer;\n};\n\nPSKBuffer.concat = function ([ ...params ], totalLength) {\n    const ArrayBufferInterface = getArrayBufferInterface();\n\n    if (!totalLength && totalLength !== 0) {\n        totalLength = 0;\n        for (const buffer of params) {\n            totalLength += buffer.length;\n        }\n    }\n\n    const buffer = new Uint8Array(new ArrayBufferInterface(totalLength));\n    let offset = 0;\n\n    for (const buf of params) {\n        const len = buf.length;\n\n        const nextOffset = offset + len;\n        if (nextOffset > totalLength) {\n            const remainingSpace = totalLength - offset;\n            for (let i = 0; i < remainingSpace; ++i) {\n                buffer[offset + i] = buf[i];\n            }\n        } else {\n            buffer.set(buf, offset);\n        }\n\n        offset = nextOffset;\n    }\n\n    return buffer;\n};\n\nPSKBuffer.isBuffer = function (pskBuffer) {\n    return !!ArrayBuffer.isView(pskBuffer);\n};\n\nPSKBuffer.alloc = function(size) {\n    const ArrayBufferInterface = getArrayBufferInterface();\n\n    return new Uint8Array(new ArrayBufferInterface(size));\n};\n\nmodule.exports = PSKBuffer;","const crypto = require('crypto');\nconst KeyEncoder = require('./keyEncoder');\n\nfunction ECDSA(curveName){\n    this.curve = curveName || 'secp256k1';\n    const self = this;\n\n    this.generateKeyPair = function() {\n        const result     = {};\n        const ec         = crypto.createECDH(self.curve);\n        result.public  = ec.generateKeys('hex');\n        result.private = ec.getPrivateKey('hex');\n        return keysToPEM(result);\n    };\n\n    function keysToPEM(keys){\n        const result                  = {};\n        const ECPrivateKeyASN         = KeyEncoder.ECPrivateKeyASN;\n        const SubjectPublicKeyInfoASN = KeyEncoder.SubjectPublicKeyInfoASN;\n        const keyEncoder              = new KeyEncoder(self.curve);\n\n        const privateKeyObject        = keyEncoder.privateKeyObject(keys.private,keys.public);\n        const publicKeyObject         = keyEncoder.publicKeyObject(keys.public);\n\n        result.private              = ECPrivateKeyASN.encode(privateKeyObject, 'pem', privateKeyObject.pemOptions);\n        result.public               = SubjectPublicKeyInfoASN.encode(publicKeyObject, 'pem', publicKeyObject.pemOptions);\n\n        return result;\n\n    }\n\n    this.sign = function (privateKey,digest) {\n        const sign = crypto.createSign(\"sha256\");\n        sign.update(digest);\n\n        return sign.sign(privateKey,'hex');\n    };\n\n    this.verify = function (publicKey,signature,digest) {\n        const verify = crypto.createVerify('sha256');\n        verify.update(digest);\n\n        return verify.verify(publicKey,signature,'hex');\n    }\n}\n\nexports.createECDSA = function (curve){\n    return new ECDSA(curve);\n};","const crypto = require('crypto');\nconst fs = require('fs');\nconst path = require('path');\nconst os = require('os');\n\nconst utils = require(\"./utils/cryptoUtils\");\nconst PskArchiver = require(\"./psk-archiver\");\nconst PassThroughStream = require('./utils/PassThroughStream');\n\nconst EventEmitter = require('events');\nconst tempFolder = os.tmpdir();\n\nfunction PskCrypto() {\n\n    const self = this;\n\n    const event = new EventEmitter();\n\n    this.on = event.on;\n    this.off = event.removeListener;\n    this.removeAllListeners = event.removeAllListeners;\n    this.emit = event.emit;\n\n    /*--------------------------------------------- ECDSA functions ------------------------------------------*/\n    const ecdsa = require(\"./ECDSA\").createECDSA();\n    this.generateECDSAKeyPair = function () {\n        return ecdsa.generateKeyPair();\n    };\n\n    this.sign = function (privateKey, digest) {\n        return ecdsa.sign(privateKey, digest);\n    };\n\n    this.verify = function (publicKey, signature, digest) {\n        return ecdsa.verify(publicKey, signature, digest);\n    };\n\n    /*---------------------------------------------Encryption functions -------------------------------------*/\n\n    this.encryptStream = function (inputPath, destinationPath, password, callback) {\n        const archiver = new PskArchiver();\n\n        archiver.on('progress', (progress) => {\n            self.emit('progress', progress);\n        });\n\n        fs.open(destinationPath, \"wx\", function (err, fd) {\n            if (err) {\n                callback(err);\n                return;\n            }\n\n            fs.close(fd, function (err) {\n                if (err) {\n                    return callback(err);\n                }\n\n                const ws = fs.createWriteStream(destinationPath, {autoClose: false});\n                const keySalt = crypto.randomBytes(32);\n                const key = crypto.pbkdf2Sync(password, keySalt, utils.iterations_number, 32, 'sha512');\n\n                const aadSalt = crypto.randomBytes(32);\n                const aad = crypto.pbkdf2Sync(password, aadSalt, utils.iterations_number, 32, 'sha512');\n\n                const salt = Buffer.concat([keySalt, aadSalt]);\n                const iv = crypto.pbkdf2Sync(password, salt, utils.iterations_number, 12, 'sha512');\n\n                const cipher = crypto.createCipheriv(utils.algorithm, key, iv);\n                cipher.setAAD(aad);\n                archiver.zipStream(inputPath, cipher, function (err, cipherStream) {\n\n                    if (err) {\n                        return callback(err);\n                    }\n\n                    cipherStream.on(\"data\", function (chunk) {\n                        ws.write(chunk);\n                    });\n                    cipherStream.on('end', function () {\n                        const tag = cipher.getAuthTag();\n                        const dataToAppend = Buffer.concat([salt, tag]);\n                        ws.end(dataToAppend, function (err) {\n                            if (err) {\n                                return callback(err);\n                            }\n\n                            callback();\n                        })\n                    });\n                });\n            });\n        });\n    };\n\n    this.decryptStream = function (encryptedInputPath, outputFolder, password, callback) {\n\n        const archiver = new PskArchiver();\n\n        decryptFile(encryptedInputPath, tempFolder, password, function (err, tempArchivePath) {\n            if (err) {\n                return callback(err);\n            }\n\n            archiver.on('progress', (progress) => {\n                self.emit('progress', 10 + 0.9 * progress);\n            });\n\n\n            archiver.unzipStream(tempArchivePath, outputFolder, function (err, unzippedFileNames) {\n                if (err) {\n                    return callback(err);\n                }\n\n                utils.deleteRecursively(tempArchivePath, function (err) {\n                    if (err) {\n                        return callback(err);\n                    }\n\n                    callback(undefined, unzippedFileNames);\n                });\n\n            });\n        })\n    };\n\n    this.encryptObject = function (inputObj, dseed, depth, callback) {\n        const archiver = new PskArchiver();\n\n        archiver.zipInMemory(inputObj, depth, function (err, zippedObj) {\n            if (err) {\n                return callback(err);\n            }\n            const cipherText = utils.encrypt(zippedObj, dseed);\n            callback(null, cipherText);\n        })\n    };\n\n    this.decryptObject = function (encryptedData, dseed, callback) {\n        const archiver = new PskArchiver();\n\n        const zippedObject = utils.decrypt(encryptedData, dseed);\n        archiver.unzipInMemory(zippedObject, function (err, obj) {\n            if (err) {\n                return callback(err);\n            }\n            callback(null, obj);\n        })\n    };\n\n    this.pskHash = function (data) {\n        if (Buffer.isBuffer(data)) {\n            return utils.createPskHash(data);\n        }\n        if (data instanceof Object) {\n            return utils.createPskHash(JSON.stringify(data));\n        }\n        return utils.createPskHash(data);\n    };\n\n    this.pskHashStream = function (readStream, callback) {\n        const pskHash = new utils.PskHash();\n\n        readStream.on('data', (chunk) => {\n            pskHash.update(chunk);\n        });\n\n\n        readStream.on('end', () => {\n            callback(null, pskHash.digest());\n        })\n    };\n\n\n    this.saveData = function (data, password, path, callback) {\n        const encryptionKey = this.deriveKey(password, null, null);\n        const iv = crypto.randomBytes(16);\n        const cipher = crypto.createCipheriv('aes-256-cfb', encryptionKey, iv);\n        let encryptedDSeed = cipher.update(data, 'binary');\n        const final = Buffer.from(cipher.final('binary'), 'binary');\n        encryptedDSeed = Buffer.concat([iv, encryptedDSeed, final]);\n        fs.writeFile(path, encryptedDSeed, function (err) {\n            callback(err);\n        });\n    };\n\n\n    this.loadData = function (password, path, callback) {\n\n        fs.readFile(path, null, (err, encryptedData) => {\n            if (err) {\n                callback(err);\n            } else {\n                const iv = encryptedData.slice(0, 16);\n                const encryptedDseed = encryptedData.slice(16);\n                const encryptionKey = this.deriveKey(password, null, null);\n                const decipher = crypto.createDecipheriv('aes-256-cfb', encryptionKey, iv);\n                let dseed = Buffer.from(decipher.update(encryptedDseed, 'binary'), 'binary');\n                const final = Buffer.from(decipher.final('binary'), 'binary');\n                dseed = Buffer.concat([dseed, final]);\n                callback(null, dseed);\n            }\n        });\n    };\n\n\n    this.generateSafeUid = function (password, additionalData) {\n        password = password || Buffer.alloc(0);\n        if (!additionalData) {\n            additionalData = Buffer.alloc(0);\n        }\n\n        if (!Buffer.isBuffer(additionalData)) {\n            additionalData = Buffer.from(additionalData);\n        }\n\n        return utils.encode(this.pskHash(Buffer.concat([password, additionalData])));\n    };\n\n    this.deriveKey = function deriveKey(password, iterations, dkLen) {\n        iterations = iterations || 1000;\n        dkLen = dkLen || 32;\n        const salt = utils.generateSalt(password, 32);\n        const dk = crypto.pbkdf2Sync(password, salt, iterations, dkLen, 'sha512');\n        return Buffer.from(dk);\n    };\n\n    this.randomBytes = crypto.randomBytes;\n    this.PskHash = utils.PskHash;\n\n    //-------------------------- Internal functions -----------------------------------\n    function decryptFile(encryptedInputPath, tempFolder, password, callback) {\n        fs.stat(encryptedInputPath, function (err, stats) {\n            if (err) {\n                return callback(err, null);\n            }\n\n            const fileSizeInBytes = stats.size;\n\n            fs.open(encryptedInputPath, \"r\", function (err, fd) {\n                if (err) {\n                    callback(err, null);\n                } else {\n                    const encryptedAuthData = Buffer.alloc(80);\n\n                    fs.read(fd, encryptedAuthData, 0, 80, fileSizeInBytes - 80, function (err, bytesRead) {\n                        const salt = encryptedAuthData.slice(0, 64);\n                        const keySalt = salt.slice(0, 32);\n                        const aadSalt = salt.slice(-32);\n\n                        const iv = crypto.pbkdf2Sync(password, salt, utils.iterations_number, 12, 'sha512');\n                        const key = crypto.pbkdf2Sync(password, keySalt, utils.iterations_number, 32, 'sha512');\n                        const aad = crypto.pbkdf2Sync(password, aadSalt, utils.iterations_number, 32, 'sha512');\n                        const tag = encryptedAuthData.slice(-16);\n\n                        const decipher = crypto.createDecipheriv(utils.algorithm, key, iv);\n\n                        decipher.setAAD(aad);\n                        decipher.setAuthTag(tag);\n                        const rs = fs.createReadStream(encryptedInputPath, {start: 0, end: fileSizeInBytes - 81});\n                        fs.mkdir(tempFolder, {recursive: true}, function (err) {\n\n                            if (err) {\n                                return callback(err);\n                            }\n                            const tempArchivePath = path.join(tempFolder, path.basename(encryptedInputPath) + \".zip\");\n\n                            fs.open(tempArchivePath, \"w\", function (err, fd) {\n                                if (err) {\n                                    callback(err);\n                                    return;\n                                }\n\n                                fs.close(fd, function (err) {\n\n                                    if (err) {\n                                        return callback(err);\n                                    }\n\n                                    const ptStream = new PassThroughStream();\n\n                                    const ws = fs.createWriteStream(tempArchivePath, {autoClose: false});\n                                    ws.on(\"finish\", function () {\n                                        callback(null, tempArchivePath);\n                                    });\n\n\n                                    let progressLength = 0;\n                                    let totalLength = 0;\n\n                                    /**\n                                     * TODO review this\n                                     * In browser, piping will block the event loop and the stack queue is not called.\n                                     */\n                                    rs.on(\"data\", (chunk) => {\n                                        progressLength += chunk.length;\n                                        totalLength += chunk.length;\n\n                                        if (progressLength > 300000) {\n                                            progressLength = 0;\n                                            emitProgress(fileSizeInBytes, totalLength)\n                                        }\n                                    });\n\n                                    rs.pipe(decipher).pipe(ptStream).pipe(ws);\n\n                                });\n                            });\n                        });\n\n                    });\n\n                }\n            });\n        });\n    }\n\n    function emitProgress(total, processed) {\n\n\n        if (processed > total) {\n            processed = total;\n        }\n\n        const progress = (100 * processed) / total;\n        self.emit('progress', parseInt(progress));\n    }\n\n}\n\nmodule.exports = new PskCrypto();\n","var asn1 = require('./asn1');\nvar inherits = require('util').inherits;\n\nvar api = exports;\n\napi.define = function define(name, body) {\n  return new Entity(name, body);\n};\n\nfunction Entity(name, body) {\n  this.name = name;\n  this.body = body;\n\n  this.decoders = {};\n  this.encoders = {};\n};\n\nEntity.prototype._createNamed = function createNamed(base) {\n  var named;\n  try {\n    named = require('vm').runInThisContext(\n      '(function ' + this.name + '(entity) {\\n' +\n      '  this._initNamed(entity);\\n' +\n      '})'\n    );\n  } catch (e) {\n    named = function (entity) {\n      this._initNamed(entity);\n    };\n  }\n  inherits(named, base);\n  named.prototype._initNamed = function initnamed(entity) {\n    base.call(this, entity);\n  };\n\n  return new named(this);\n};\n\nEntity.prototype._getDecoder = function _getDecoder(enc) {\n  // Lazily create decoder\n  if (!this.decoders.hasOwnProperty(enc))\n    this.decoders[enc] = this._createNamed(asn1.decoders[enc]);\n  return this.decoders[enc];\n};\n\nEntity.prototype.decode = function decode(data, enc, options) {\n  return this._getDecoder(enc).decode(data, options);\n};\n\nEntity.prototype._getEncoder = function _getEncoder(enc) {\n  // Lazily create encoder\n  if (!this.encoders.hasOwnProperty(enc))\n    this.encoders[enc] = this._createNamed(asn1.encoders[enc]);\n  return this.encoders[enc];\n};\n\nEntity.prototype.encode = function encode(data, enc, /* internal */ reporter) {\n  return this._getEncoder(enc).encode(data, reporter);\n};\n","var asn1 = exports;\n\nasn1.bignum = require('./bignum/bn');\n\nasn1.define = require('./api').define;\nasn1.base = require('./base/index');\nasn1.constants = require('./constants/index');\nasn1.decoders = require('./decoders/index');\nasn1.encoders = require('./encoders/index');\n","var inherits = require('util').inherits;\nvar Reporter = require('../base').Reporter;\nvar Buffer = require('buffer').Buffer;\n\nfunction DecoderBuffer(base, options) {\n  Reporter.call(this, options);\n  if (!Buffer.isBuffer(base)) {\n    this.error('Input not Buffer');\n    return;\n  }\n\n  this.base = base;\n  this.offset = 0;\n  this.length = base.length;\n}\ninherits(DecoderBuffer, Reporter);\nexports.DecoderBuffer = DecoderBuffer;\n\nDecoderBuffer.prototype.save = function save() {\n  return { offset: this.offset, reporter: Reporter.prototype.save.call(this) };\n};\n\nDecoderBuffer.prototype.restore = function restore(save) {\n  // Return skipped data\n  var res = new DecoderBuffer(this.base);\n  res.offset = save.offset;\n  res.length = this.offset;\n\n  this.offset = save.offset;\n  Reporter.prototype.restore.call(this, save.reporter);\n\n  return res;\n};\n\nDecoderBuffer.prototype.isEmpty = function isEmpty() {\n  return this.offset === this.length;\n};\n\nDecoderBuffer.prototype.readUInt8 = function readUInt8(fail) {\n  if (this.offset + 1 <= this.length)\n    return this.base.readUInt8(this.offset++, true);\n  else\n    return this.error(fail || 'DecoderBuffer overrun');\n}\n\nDecoderBuffer.prototype.skip = function skip(bytes, fail) {\n  if (!(this.offset + bytes <= this.length))\n    return this.error(fail || 'DecoderBuffer overrun');\n\n  var res = new DecoderBuffer(this.base);\n\n  // Share reporter state\n  res._reporterState = this._reporterState;\n\n  res.offset = this.offset;\n  res.length = this.offset + bytes;\n  this.offset += bytes;\n  return res;\n}\n\nDecoderBuffer.prototype.raw = function raw(save) {\n  return this.base.slice(save ? save.offset : this.offset, this.length);\n}\n\nfunction EncoderBuffer(value, reporter) {\n  if (Array.isArray(value)) {\n    this.length = 0;\n    this.value = value.map(function(item) {\n      if (!(item instanceof EncoderBuffer))\n        item = new EncoderBuffer(item, reporter);\n      this.length += item.length;\n      return item;\n    }, this);\n  } else if (typeof value === 'number') {\n    if (!(0 <= value && value <= 0xff))\n      return reporter.error('non-byte EncoderBuffer value');\n    this.value = value;\n    this.length = 1;\n  } else if (typeof value === 'string') {\n    this.value = value;\n    this.length = Buffer.byteLength(value);\n  } else if (Buffer.isBuffer(value)) {\n    this.value = value;\n    this.length = value.length;\n  } else {\n    return reporter.error('Unsupported type: ' + typeof value);\n  }\n}\nexports.EncoderBuffer = EncoderBuffer;\n\nEncoderBuffer.prototype.join = function join(out, offset) {\n  if (!out)\n    out = new Buffer(this.length);\n  if (!offset)\n    offset = 0;\n\n  if (this.length === 0)\n    return out;\n\n  if (Array.isArray(this.value)) {\n    this.value.forEach(function(item) {\n      item.join(out, offset);\n      offset += item.length;\n    });\n  } else {\n    if (typeof this.value === 'number')\n      out[offset] = this.value;\n    else if (typeof this.value === 'string')\n      out.write(this.value, offset);\n    else if (Buffer.isBuffer(this.value))\n      this.value.copy(out, offset);\n    offset += this.length;\n  }\n\n  return out;\n};\n","var base = exports;\n\nbase.Reporter = require('./reporter').Reporter;\nbase.DecoderBuffer = require('./buffer').DecoderBuffer;\nbase.EncoderBuffer = require('./buffer').EncoderBuffer;\nbase.Node = require('./node');\n","var Reporter = require('../base').Reporter;\nvar EncoderBuffer = require('../base').EncoderBuffer;\n//var assert = require('double-check').assert;\n\n// Supported tags\nvar tags = [\n  'seq', 'seqof', 'set', 'setof', 'octstr', 'bitstr', 'objid', 'bool',\n  'gentime', 'utctime', 'null_', 'enum', 'int', 'ia5str', 'utf8str'\n];\n\n// Public methods list\nvar methods = [\n  'key', 'obj', 'use', 'optional', 'explicit', 'implicit', 'def', 'choice',\n  'any'\n].concat(tags);\n\n// Overrided methods list\nvar overrided = [\n  '_peekTag', '_decodeTag', '_use',\n  '_decodeStr', '_decodeObjid', '_decodeTime',\n  '_decodeNull', '_decodeInt', '_decodeBool', '_decodeList',\n\n  '_encodeComposite', '_encodeStr', '_encodeObjid', '_encodeTime',\n  '_encodeNull', '_encodeInt', '_encodeBool'\n];\n\nfunction Node(enc, parent) {\n  var state = {};\n  this._baseState = state;\n\n  state.enc = enc;\n\n  state.parent = parent || null;\n  state.children = null;\n\n  // State\n  state.tag = null;\n  state.args = null;\n  state.reverseArgs = null;\n  state.choice = null;\n  state.optional = false;\n  state.any = false;\n  state.obj = false;\n  state.use = null;\n  state.useDecoder = null;\n  state.key = null;\n  state['default'] = null;\n  state.explicit = null;\n  state.implicit = null;\n\n  // Should create new instance on each method\n  if (!state.parent) {\n    state.children = [];\n    this._wrap();\n  }\n}\nmodule.exports = Node;\n\nvar stateProps = [\n  'enc', 'parent', 'children', 'tag', 'args', 'reverseArgs', 'choice',\n  'optional', 'any', 'obj', 'use', 'alteredUse', 'key', 'default', 'explicit',\n  'implicit'\n];\n\nNode.prototype.clone = function clone() {\n  var state = this._baseState;\n  var cstate = {};\n  stateProps.forEach(function(prop) {\n    cstate[prop] = state[prop];\n  });\n  var res = new this.constructor(cstate.parent);\n  res._baseState = cstate;\n  return res;\n};\n\nNode.prototype._wrap = function wrap() {\n  var state = this._baseState;\n  methods.forEach(function(method) {\n    this[method] = function _wrappedMethod() {\n      var clone = new this.constructor(this);\n      state.children.push(clone);\n      return clone[method].apply(clone, arguments);\n    };\n  }, this);\n};\n\nNode.prototype._init = function init(body) {\n  var state = this._baseState;\n\n  //assert.equal(state.parent,null,'state.parent should be null');\n  body.call(this);\n\n  // Filter children\n  state.children = state.children.filter(function(child) {\n    return child._baseState.parent === this;\n  }, this);\n  // assert.equal(state.children.length, 1, 'Root node can have only one child');\n};\n\nNode.prototype._useArgs = function useArgs(args) {\n  var state = this._baseState;\n\n  // Filter children and args\n  var children = args.filter(function(arg) {\n    return arg instanceof this.constructor;\n  }, this);\n  args = args.filter(function(arg) {\n    return !(arg instanceof this.constructor);\n  }, this);\n\n  if (children.length !== 0) {\n    // assert.equal(state.children, null, 'state.children should be null');\n    state.children = children;\n\n    // Replace parent to maintain backward link\n    children.forEach(function(child) {\n      child._baseState.parent = this;\n    }, this);\n  }\n  if (args.length !== 0) {\n    // assert.equal(state.args, null, 'state.args should be null');\n    state.args = args;\n    state.reverseArgs = args.map(function(arg) {\n      if (typeof arg !== 'object' || arg.constructor !== Object)\n        return arg;\n\n      var res = {};\n      Object.keys(arg).forEach(function(key) {\n        if (key == (key | 0))\n          key |= 0;\n        var value = arg[key];\n        res[value] = key;\n      });\n      return res;\n    });\n  }\n};\n\n//\n// Overrided methods\n//\n\noverrided.forEach(function(method) {\n  Node.prototype[method] = function _overrided() {\n    var state = this._baseState;\n    throw new Error(method + ' not implemented for encoding: ' + state.enc);\n  };\n});\n\n//\n// Public methods\n//\n\ntags.forEach(function(tag) {\n  Node.prototype[tag] = function _tagMethod() {\n    var state = this._baseState;\n    var args = Array.prototype.slice.call(arguments);\n\n    // assert.equal(state.tag, null, 'state.tag should be null');\n    state.tag = tag;\n\n    this._useArgs(args);\n\n    return this;\n  };\n});\n\nNode.prototype.use = function use(item) {\n  var state = this._baseState;\n\n  // assert.equal(state.use, null, 'state.use should be null');\n  state.use = item;\n\n  return this;\n};\n\nNode.prototype.optional = function optional() {\n  var state = this._baseState;\n\n  state.optional = true;\n\n  return this;\n};\n\nNode.prototype.def = function def(val) {\n  var state = this._baseState;\n\n  // assert.equal(state['default'], null, \"state['default'] should be null\");\n  state['default'] = val;\n  state.optional = true;\n\n  return this;\n};\n\nNode.prototype.explicit = function explicit(num) {\n  var state = this._baseState;\n\n  // assert.equal(state.explicit,null, 'state.explicit should be null');\n  // assert.equal(state.implicit,null, 'state.implicit should be null');\n\n  state.explicit = num;\n\n  return this;\n};\n\nNode.prototype.implicit = function implicit(num) {\n  var state = this._baseState;\n\n    // assert.equal(state.explicit,null, 'state.explicit should be null');\n    // assert.equal(state.implicit,null, 'state.implicit should be null');\n\n    state.implicit = num;\n\n  return this;\n};\n\nNode.prototype.obj = function obj() {\n  var state = this._baseState;\n  var args = Array.prototype.slice.call(arguments);\n\n  state.obj = true;\n\n  if (args.length !== 0)\n    this._useArgs(args);\n\n  return this;\n};\n\nNode.prototype.key = function key(newKey) {\n  var state = this._baseState;\n\n  // assert.equal(state.key, null, 'state.key should be null');\n  state.key = newKey;\n\n  return this;\n};\n\nNode.prototype.any = function any() {\n  var state = this._baseState;\n\n  state.any = true;\n\n  return this;\n};\n\nNode.prototype.choice = function choice(obj) {\n  var state = this._baseState;\n\n  // assert.equal(state.choice, null,'state.choice should be null');\n  state.choice = obj;\n  this._useArgs(Object.keys(obj).map(function(key) {\n    return obj[key];\n  }));\n\n  return this;\n};\n\n//\n// Decoding\n//\n\nNode.prototype._decode = function decode(input) {\n  var state = this._baseState;\n\n  // Decode root node\n  if (state.parent === null)\n    return input.wrapResult(state.children[0]._decode(input));\n\n  var result = state['default'];\n  var present = true;\n\n  var prevKey;\n  if (state.key !== null)\n    prevKey = input.enterKey(state.key);\n\n  // Check if tag is there\n  if (state.optional) {\n    var tag = null;\n    if (state.explicit !== null)\n      tag = state.explicit;\n    else if (state.implicit !== null)\n      tag = state.implicit;\n    else if (state.tag !== null)\n      tag = state.tag;\n\n    if (tag === null && !state.any) {\n      // Trial and Error\n      var save = input.save();\n      try {\n        if (state.choice === null)\n          this._decodeGeneric(state.tag, input);\n        else\n          this._decodeChoice(input);\n        present = true;\n      } catch (e) {\n        present = false;\n      }\n      input.restore(save);\n    } else {\n      present = this._peekTag(input, tag, state.any);\n\n      if (input.isError(present))\n        return present;\n    }\n  }\n\n  // Push object on stack\n  var prevObj;\n  if (state.obj && present)\n    prevObj = input.enterObject();\n\n  if (present) {\n    // Unwrap explicit values\n    if (state.explicit !== null) {\n      var explicit = this._decodeTag(input, state.explicit);\n      if (input.isError(explicit))\n        return explicit;\n      input = explicit;\n    }\n\n    // Unwrap implicit and normal values\n    if (state.use === null && state.choice === null) {\n      if (state.any)\n        var save = input.save();\n      var body = this._decodeTag(\n        input,\n        state.implicit !== null ? state.implicit : state.tag,\n        state.any\n      );\n      if (input.isError(body))\n        return body;\n\n      if (state.any)\n        result = input.raw(save);\n      else\n        input = body;\n    }\n\n    // Select proper method for tag\n    if (state.any)\n      result = result;\n    else if (state.choice === null)\n      result = this._decodeGeneric(state.tag, input);\n    else\n      result = this._decodeChoice(input);\n\n    if (input.isError(result))\n      return result;\n\n    // Decode children\n    if (!state.any && state.choice === null && state.children !== null) {\n      var fail = state.children.some(function decodeChildren(child) {\n        // NOTE: We are ignoring errors here, to let parser continue with other\n        // parts of encoded data\n        child._decode(input);\n      });\n      if (fail)\n        return err;\n    }\n  }\n\n  // Pop object\n  if (state.obj && present)\n    result = input.leaveObject(prevObj);\n\n  // Set key\n  if (state.key !== null && (result !== null || present === true))\n    input.leaveKey(prevKey, state.key, result);\n\n  return result;\n};\n\nNode.prototype._decodeGeneric = function decodeGeneric(tag, input) {\n  var state = this._baseState;\n\n  if (tag === 'seq' || tag === 'set')\n    return null;\n  if (tag === 'seqof' || tag === 'setof')\n    return this._decodeList(input, tag, state.args[0]);\n  else if (tag === 'octstr' || tag === 'bitstr')\n    return this._decodeStr(input, tag);\n  else if (tag === 'ia5str' || tag === 'utf8str')\n    return this._decodeStr(input, tag);\n  else if (tag === 'objid' && state.args)\n    return this._decodeObjid(input, state.args[0], state.args[1]);\n  else if (tag === 'objid')\n    return this._decodeObjid(input, null, null);\n  else if (tag === 'gentime' || tag === 'utctime')\n    return this._decodeTime(input, tag);\n  else if (tag === 'null_')\n    return this._decodeNull(input);\n  else if (tag === 'bool')\n    return this._decodeBool(input);\n  else if (tag === 'int' || tag === 'enum')\n    return this._decodeInt(input, state.args && state.args[0]);\n  else if (state.use !== null)\n    return this._getUse(state.use, input._reporterState.obj)._decode(input);\n  else\n    return input.error('unknown tag: ' + tag);\n\n  return null;\n};\n\nNode.prototype._getUse = function _getUse(entity, obj) {\n\n  var state = this._baseState;\n  // Create altered use decoder if implicit is set\n  state.useDecoder = this._use(entity, obj);\n  // assert.equal(state.useDecoder._baseState.parent, null, 'state.useDecoder._baseState.parent should be null');\n  state.useDecoder = state.useDecoder._baseState.children[0];\n  if (state.implicit !== state.useDecoder._baseState.implicit) {\n    state.useDecoder = state.useDecoder.clone();\n    state.useDecoder._baseState.implicit = state.implicit;\n  }\n  return state.useDecoder;\n};\n\nNode.prototype._decodeChoice = function decodeChoice(input) {\n  var state = this._baseState;\n  var result = null;\n  var match = false;\n\n  Object.keys(state.choice).some(function(key) {\n    var save = input.save();\n    var node = state.choice[key];\n    try {\n      var value = node._decode(input);\n      if (input.isError(value))\n        return false;\n\n      result = { type: key, value: value };\n      match = true;\n    } catch (e) {\n      input.restore(save);\n      return false;\n    }\n    return true;\n  }, this);\n\n  if (!match)\n    return input.error('Choice not matched');\n\n  return result;\n};\n\n//\n// Encoding\n//\n\nNode.prototype._createEncoderBuffer = function createEncoderBuffer(data) {\n  return new EncoderBuffer(data, this.reporter);\n};\n\nNode.prototype._encode = function encode(data, reporter, parent) {\n  var state = this._baseState;\n  if (state['default'] !== null && state['default'] === data)\n    return;\n\n  var result = this._encodeValue(data, reporter, parent);\n  if (result === undefined)\n    return;\n\n  if (this._skipDefault(result, reporter, parent))\n    return;\n\n  return result;\n};\n\nNode.prototype._encodeValue = function encode(data, reporter, parent) {\n  var state = this._baseState;\n\n  // Decode root node\n  if (state.parent === null)\n    return state.children[0]._encode(data, reporter || new Reporter());\n\n  var result = null;\n  var present = true;\n\n  // Set reporter to share it with a child class\n  this.reporter = reporter;\n\n  // Check if data is there\n  if (state.optional && data === undefined) {\n    if (state['default'] !== null)\n      data = state['default']\n    else\n      return;\n  }\n\n  // For error reporting\n  var prevKey;\n\n  // Encode children first\n  var content = null;\n  var primitive = false;\n  if (state.any) {\n    // Anything that was given is translated to buffer\n    result = this._createEncoderBuffer(data);\n  } else if (state.choice) {\n    result = this._encodeChoice(data, reporter);\n  } else if (state.children) {\n    content = state.children.map(function(child) {\n      if (child._baseState.tag === 'null_')\n        return child._encode(null, reporter, data);\n\n      if (child._baseState.key === null)\n        return reporter.error('Child should have a key');\n      var prevKey = reporter.enterKey(child._baseState.key);\n\n      if (typeof data !== 'object')\n        return reporter.error('Child expected, but input is not object');\n\n      var res = child._encode(data[child._baseState.key], reporter, data);\n      reporter.leaveKey(prevKey);\n\n      return res;\n    }, this).filter(function(child) {\n      return child;\n    });\n\n    content = this._createEncoderBuffer(content);\n  } else {\n    if (state.tag === 'seqof' || state.tag === 'setof') {\n      // TODO(indutny): this should be thrown on DSL level\n      if (!(state.args && state.args.length === 1))\n        return reporter.error('Too many args for : ' + state.tag);\n\n      if (!Array.isArray(data))\n        return reporter.error('seqof/setof, but data is not Array');\n\n      var child = this.clone();\n      child._baseState.implicit = null;\n      content = this._createEncoderBuffer(data.map(function(item) {\n        var state = this._baseState;\n\n        return this._getUse(state.args[0], data)._encode(item, reporter);\n      }, child));\n    } else if (state.use !== null) {\n      result = this._getUse(state.use, parent)._encode(data, reporter);\n    } else {\n      content = this._encodePrimitive(state.tag, data);\n      primitive = true;\n    }\n  }\n\n  // Encode data itself\n  var result;\n  if (!state.any && state.choice === null) {\n    var tag = state.implicit !== null ? state.implicit : state.tag;\n    var cls = state.implicit === null ? 'universal' : 'context';\n\n    if (tag === null) {\n      if (state.use === null)\n        reporter.error('Tag could be ommited only for .use()');\n    } else {\n      if (state.use === null)\n        result = this._encodeComposite(tag, primitive, cls, content);\n    }\n  }\n\n  // Wrap in explicit\n  if (state.explicit !== null)\n    result = this._encodeComposite(state.explicit, false, 'context', result);\n\n  return result;\n};\n\nNode.prototype._encodeChoice = function encodeChoice(data, reporter) {\n  var state = this._baseState;\n\n  var node = state.choice[data.type];\n  // if (!node) {\n  //   assert(\n  //       false,\n  //       data.type + ' not found in ' +\n  //           JSON.stringify(Object.keys(state.choice)));\n  // }\n  return node._encode(data.value, reporter);\n};\n\nNode.prototype._encodePrimitive = function encodePrimitive(tag, data) {\n  var state = this._baseState;\n\n  if (tag === 'octstr' || tag === 'bitstr' || tag === 'ia5str')\n    return this._encodeStr(data, tag);\n  else if (tag === 'utf8str')\n    return this._encodeStr(data, tag);\n  else if (tag === 'objid' && state.args)\n    return this._encodeObjid(data, state.reverseArgs[0], state.args[1]);\n  else if (tag === 'objid')\n    return this._encodeObjid(data, null, null);\n  else if (tag === 'gentime' || tag === 'utctime')\n    return this._encodeTime(data, tag);\n  else if (tag === 'null_')\n    return this._encodeNull();\n  else if (tag === 'int' || tag === 'enum')\n    return this._encodeInt(data, state.args && state.reverseArgs[0]);\n  else if (tag === 'bool')\n    return this._encodeBool(data);\n  else\n    throw new Error('Unsupported tag: ' + tag);\n};\n","var inherits = require('util').inherits;\n\nfunction Reporter(options) {\n  this._reporterState = {\n    obj: null,\n    path: [],\n    options: options || {},\n    errors: []\n  };\n}\nexports.Reporter = Reporter;\n\nReporter.prototype.isError = function isError(obj) {\n  return obj instanceof ReporterError;\n};\n\nReporter.prototype.save = function save() {\n  var state = this._reporterState;\n\n  return { obj: state.obj, pathLen: state.path.length };\n};\n\nReporter.prototype.restore = function restore(data) {\n  var state = this._reporterState;\n\n  state.obj = data.obj;\n  state.path = state.path.slice(0, data.pathLen);\n};\n\nReporter.prototype.enterKey = function enterKey(key) {\n  return this._reporterState.path.push(key);\n};\n\nReporter.prototype.leaveKey = function leaveKey(index, key, value) {\n  var state = this._reporterState;\n\n  state.path = state.path.slice(0, index - 1);\n  if (state.obj !== null)\n    state.obj[key] = value;\n};\n\nReporter.prototype.enterObject = function enterObject() {\n  var state = this._reporterState;\n\n  var prev = state.obj;\n  state.obj = {};\n  return prev;\n};\n\nReporter.prototype.leaveObject = function leaveObject(prev) {\n  var state = this._reporterState;\n\n  var now = state.obj;\n  state.obj = prev;\n  return now;\n};\n\nReporter.prototype.error = function error(msg) {\n  var err;\n  var state = this._reporterState;\n\n  var inherited = msg instanceof ReporterError;\n  if (inherited) {\n    err = msg;\n  } else {\n    err = new ReporterError(state.path.map(function(elem) {\n      return '[' + JSON.stringify(elem) + ']';\n    }).join(''), msg.message || msg, msg.stack);\n  }\n\n  if (!state.options.partial)\n    throw err;\n\n  if (!inherited)\n    state.errors.push(err);\n\n  return err;\n};\n\nReporter.prototype.wrapResult = function wrapResult(result) {\n  var state = this._reporterState;\n  if (!state.options.partial)\n    return result;\n\n  return {\n    result: this.isError(result) ? null : result,\n    errors: state.errors\n  };\n};\n\nfunction ReporterError(path, msg) {\n  this.path = path;\n  this.rethrow(msg);\n};\ninherits(ReporterError, Error);\n\nReporterError.prototype.rethrow = function rethrow(msg) {\n  this.message = msg + ' at: ' + (this.path || '(shallow)');\n  Error.captureStackTrace(this, ReporterError);\n\n  return this;\n};\n","(function (module, exports) {\n\n'use strict';\n\n// Utils\n\nfunction assert(val, msg) {\n  if (!val)\n    throw new Error(msg || 'Assertion failed');\n}\n\n// Could use `inherits` module, but don't want to move from single file\n// architecture yet.\nfunction inherits(ctor, superCtor) {\n  ctor.super_ = superCtor;\n  var TempCtor = function () {};\n  TempCtor.prototype = superCtor.prototype;\n  ctor.prototype = new TempCtor();\n  ctor.prototype.constructor = ctor;\n}\n\n// BN\n\nfunction BN(number, base, endian) {\n  // May be `new BN(bn)` ?\n  if (number !== null &&\n      typeof number === 'object' &&\n      Array.isArray(number.words)) {\n    return number;\n  }\n\n  this.sign = false;\n  this.words = null;\n  this.length = 0;\n\n  // Reduction context\n  this.red = null;\n\n  if (base === 'le' || base === 'be') {\n    endian = base;\n    base = 10;\n  }\n\n  if (number !== null)\n    this._init(number || 0, base || 10, endian || 'be');\n}\nif (typeof module === 'object')\n  module.exports = BN;\nelse\n  exports.BN = BN;\n\nBN.BN = BN;\nBN.wordSize = 26;\n\nBN.prototype._init = function init(number, base, endian) {\n  if (typeof number === 'number') {\n    return this._initNumber(number, base, endian);\n  } else if (typeof number === 'object') {\n    return this._initArray(number, base, endian);\n  }\n  if (base === 'hex')\n    base = 16;\n  assert(base === (base | 0) && base >= 2 && base <= 36);\n\n  number = number.toString().replace(/\\s+/g, '');\n  var start = 0;\n  if (number[0] === '-')\n    start++;\n\n  if (base === 16)\n    this._parseHex(number, start);\n  else\n    this._parseBase(number, base, start);\n\n  if (number[0] === '-')\n    this.sign = true;\n\n  this.strip();\n\n  if (endian !== 'le')\n    return;\n\n  this._initArray(this.toArray(), base, endian);\n};\n\nBN.prototype._initNumber = function _initNumber(number, base, endian) {\n  if (number < 0) {\n    this.sign = true;\n    number = -number;\n  }\n  if (number < 0x4000000) {\n    this.words = [ number & 0x3ffffff ];\n    this.length = 1;\n  } else if (number < 0x10000000000000) {\n    this.words = [\n      number & 0x3ffffff,\n      (number / 0x4000000) & 0x3ffffff\n    ];\n    this.length = 2;\n  } else {\n    assert(number < 0x20000000000000); // 2 ^ 53 (unsafe)\n    this.words = [\n      number & 0x3ffffff,\n      (number / 0x4000000) & 0x3ffffff,\n      1\n    ];\n    this.length = 3;\n  }\n\n  if (endian !== 'le')\n    return;\n\n  // Reverse the bytes\n  this._initArray(this.toArray(), base, endian);\n};\n\nBN.prototype._initArray = function _initArray(number, base, endian) {\n  // Perhaps a Uint8Array\n  assert(typeof number.length === 'number');\n  if (number.length <= 0) {\n    this.words = [ 0 ];\n    this.length = 1;\n    return this;\n  }\n\n  this.length = Math.ceil(number.length / 3);\n  this.words = new Array(this.length);\n  for (var i = 0; i < this.length; i++)\n    this.words[i] = 0;\n\n  var off = 0;\n  if (endian === 'be') {\n    for (var i = number.length - 1, j = 0; i >= 0; i -= 3) {\n      var w = number[i] | (number[i - 1] << 8) | (number[i - 2] << 16);\n      this.words[j] |= (w << off) & 0x3ffffff;\n      this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff;\n      off += 24;\n      if (off >= 26) {\n        off -= 26;\n        j++;\n      }\n    }\n  } else if (endian === 'le') {\n    for (var i = 0, j = 0; i < number.length; i += 3) {\n      var w = number[i] | (number[i + 1] << 8) | (number[i + 2] << 16);\n      this.words[j] |= (w << off) & 0x3ffffff;\n      this.words[j + 1] = (w >>> (26 - off)) & 0x3ffffff;\n      off += 24;\n      if (off >= 26) {\n        off -= 26;\n        j++;\n      }\n    }\n  }\n  return this.strip();\n};\n\nfunction parseHex(str, start, end) {\n  var r = 0;\n  var len = Math.min(str.length, end);\n  for (var i = start; i < len; i++) {\n    var c = str.charCodeAt(i) - 48;\n\n    r <<= 4;\n\n    // 'a' - 'f'\n    if (c >= 49 && c <= 54)\n      r |= c - 49 + 0xa;\n\n    // 'A' - 'F'\n    else if (c >= 17 && c <= 22)\n      r |= c - 17 + 0xa;\n\n    // '0' - '9'\n    else\n      r |= c & 0xf;\n  }\n  return r;\n}\n\nBN.prototype._parseHex = function _parseHex(number, start) {\n  // Create possibly bigger array to ensure that it fits the number\n  this.length = Math.ceil((number.length - start) / 6);\n  this.words = new Array(this.length);\n  for (var i = 0; i < this.length; i++)\n    this.words[i] = 0;\n\n  // Scan 24-bit chunks and add them to the number\n  var off = 0;\n  for (var i = number.length - 6, j = 0; i >= start; i -= 6) {\n    var w = parseHex(number, i, i + 6);\n    this.words[j] |= (w << off) & 0x3ffffff;\n    this.words[j + 1] |= w >>> (26 - off) & 0x3fffff;\n    off += 24;\n    if (off >= 26) {\n      off -= 26;\n      j++;\n    }\n  }\n  if (i + 6 !== start) {\n    var w = parseHex(number, start, i + 6);\n    this.words[j] |= (w << off) & 0x3ffffff;\n    this.words[j + 1] |= w >>> (26 - off) & 0x3fffff;\n  }\n  this.strip();\n};\n\nfunction parseBase(str, start, end, mul) {\n  var r = 0;\n  var len = Math.min(str.length, end);\n  for (var i = start; i < len; i++) {\n    var c = str.charCodeAt(i) - 48;\n\n    r *= mul;\n\n    // 'a'\n    if (c >= 49)\n      r += c - 49 + 0xa;\n\n    // 'A'\n    else if (c >= 17)\n      r += c - 17 + 0xa;\n\n    // '0' - '9'\n    else\n      r += c;\n  }\n  return r;\n}\n\nBN.prototype._parseBase = function _parseBase(number, base, start) {\n  // Initialize as zero\n  this.words = [ 0 ];\n  this.length = 1;\n\n  // Find length of limb in base\n  for (var limbLen = 0, limbPow = 1; limbPow <= 0x3ffffff; limbPow *= base)\n    limbLen++;\n  limbLen--;\n  limbPow = (limbPow / base) | 0;\n\n  var total = number.length - start;\n  var mod = total % limbLen;\n  var end = Math.min(total, total - mod) + start;\n\n  var word = 0;\n  for (var i = start; i < end; i += limbLen) {\n    word = parseBase(number, i, i + limbLen, base);\n\n    this.imuln(limbPow);\n    if (this.words[0] + word < 0x4000000)\n      this.words[0] += word;\n    else\n      this._iaddn(word);\n  }\n\n  if (mod !== 0) {\n    var pow = 1;\n    var word = parseBase(number, i, number.length, base);\n\n    for (var i = 0; i < mod; i++)\n      pow *= base;\n    this.imuln(pow);\n    if (this.words[0] + word < 0x4000000)\n      this.words[0] += word;\n    else\n      this._iaddn(word);\n  }\n};\n\nBN.prototype.copy = function copy(dest) {\n  dest.words = new Array(this.length);\n  for (var i = 0; i < this.length; i++)\n    dest.words[i] = this.words[i];\n  dest.length = this.length;\n  dest.sign = this.sign;\n  dest.red = this.red;\n};\n\nBN.prototype.clone = function clone() {\n  var r = new BN(null);\n  this.copy(r);\n  return r;\n};\n\n// Remove leading `0` from `this`\nBN.prototype.strip = function strip() {\n  while (this.length > 1 && this.words[this.length - 1] === 0)\n    this.length--;\n  return this._normSign();\n};\n\nBN.prototype._normSign = function _normSign() {\n  // -0 = 0\n  if (this.length === 1 && this.words[0] === 0)\n    this.sign = false;\n  return this;\n};\n\nBN.prototype.inspect = function inspect() {\n  return (this.red ? '<BN-R: ' : '<BN: ') + this.toString(16) + '>';\n};\n\n/*\n\nvar zeros = [];\nvar groupSizes = [];\nvar groupBases = [];\n\nvar s = '';\nvar i = -1;\nwhile (++i < BN.wordSize) {\n  zeros[i] = s;\n  s += '0';\n}\ngroupSizes[0] = 0;\ngroupSizes[1] = 0;\ngroupBases[0] = 0;\ngroupBases[1] = 0;\nvar base = 2 - 1;\nwhile (++base < 36 + 1) {\n  var groupSize = 0;\n  var groupBase = 1;\n  while (groupBase < (1 << BN.wordSize) / base) {\n    groupBase *= base;\n    groupSize += 1;\n  }\n  groupSizes[base] = groupSize;\n  groupBases[base] = groupBase;\n}\n\n*/\n\nvar zeros = [\n  '',\n  '0',\n  '00',\n  '000',\n  '0000',\n  '00000',\n  '000000',\n  '0000000',\n  '00000000',\n  '000000000',\n  '0000000000',\n  '00000000000',\n  '000000000000',\n  '0000000000000',\n  '00000000000000',\n  '000000000000000',\n  '0000000000000000',\n  '00000000000000000',\n  '000000000000000000',\n  '0000000000000000000',\n  '00000000000000000000',\n  '000000000000000000000',\n  '0000000000000000000000',\n  '00000000000000000000000',\n  '000000000000000000000000',\n  '0000000000000000000000000'\n];\n\nvar groupSizes = [\n  0, 0,\n  25, 16, 12, 11, 10, 9, 8,\n  8, 7, 7, 7, 7, 6, 6,\n  6, 6, 6, 6, 6, 5, 5,\n  5, 5, 5, 5, 5, 5, 5,\n  5, 5, 5, 5, 5, 5, 5\n];\n\nvar groupBases = [\n  0, 0,\n  33554432, 43046721, 16777216, 48828125, 60466176, 40353607, 16777216,\n  43046721, 10000000, 19487171, 35831808, 62748517, 7529536, 11390625,\n  16777216, 24137569, 34012224, 47045881, 64000000, 4084101, 5153632,\n  6436343, 7962624, 9765625, 11881376, 14348907, 17210368, 20511149,\n  24300000, 28629151, 33554432, 39135393, 45435424, 52521875, 60466176\n];\n\nBN.prototype.toString = function toString(base, padding) {\n  base = base || 10;\n  if (base === 16 || base === 'hex') {\n    var out = '';\n    var off = 0;\n    var padding = padding | 0 || 1;\n    var carry = 0;\n    for (var i = 0; i < this.length; i++) {\n      var w = this.words[i];\n      var word = (((w << off) | carry) & 0xffffff).toString(16);\n      carry = (w >>> (24 - off)) & 0xffffff;\n      if (carry !== 0 || i !== this.length - 1)\n        out = zeros[6 - word.length] + word + out;\n      else\n        out = word + out;\n      off += 2;\n      if (off >= 26) {\n        off -= 26;\n        i--;\n      }\n    }\n    if (carry !== 0)\n      out = carry.toString(16) + out;\n    while (out.length % padding !== 0)\n      out = '0' + out;\n    if (this.sign)\n      out = '-' + out;\n    return out;\n  } else if (base === (base | 0) && base >= 2 && base <= 36) {\n    // var groupSize = Math.floor(BN.wordSize * Math.LN2 / Math.log(base));\n    var groupSize = groupSizes[base];\n    // var groupBase = Math.pow(base, groupSize);\n    var groupBase = groupBases[base];\n    var out = '';\n    var c = this.clone();\n    c.sign = false;\n    while (c.cmpn(0) !== 0) {\n      var r = c.modn(groupBase).toString(base);\n      c = c.idivn(groupBase);\n\n      if (c.cmpn(0) !== 0)\n        out = zeros[groupSize - r.length] + r + out;\n      else\n        out = r + out;\n    }\n    if (this.cmpn(0) === 0)\n      out = '0' + out;\n    if (this.sign)\n      out = '-' + out;\n    return out;\n  } else {\n    assert(false, 'Base should be between 2 and 36');\n  }\n};\n\nBN.prototype.toJSON = function toJSON() {\n  return this.toString(16);\n};\n\nBN.prototype.toArray = function toArray(endian) {\n  this.strip();\n  var res = new Array(this.byteLength());\n  res[0] = 0;\n\n  var q = this.clone();\n  if (endian !== 'le') {\n    // Assume big-endian\n    for (var i = 0; q.cmpn(0) !== 0; i++) {\n      var b = q.andln(0xff);\n      q.ishrn(8);\n\n      res[res.length - i - 1] = b;\n    }\n  } else {\n    // Assume little-endian\n    for (var i = 0; q.cmpn(0) !== 0; i++) {\n      var b = q.andln(0xff);\n      q.ishrn(8);\n\n      res[i] = b;\n    }\n  }\n\n  return res;\n};\n\nif (Math.clz32) {\n  BN.prototype._countBits = function _countBits(w) {\n    return 32 - Math.clz32(w);\n  };\n} else {\n  BN.prototype._countBits = function _countBits(w) {\n    var t = w;\n    var r = 0;\n    if (t >= 0x1000) {\n      r += 13;\n      t >>>= 13;\n    }\n    if (t >= 0x40) {\n      r += 7;\n      t >>>= 7;\n    }\n    if (t >= 0x8) {\n      r += 4;\n      t >>>= 4;\n    }\n    if (t >= 0x02) {\n      r += 2;\n      t >>>= 2;\n    }\n    return r + t;\n  };\n}\n\nBN.prototype._zeroBits = function _zeroBits(w) {\n  // Short-cut\n  if (w === 0)\n    return 26;\n\n  var t = w;\n  var r = 0;\n  if ((t & 0x1fff) === 0) {\n    r += 13;\n    t >>>= 13;\n  }\n  if ((t & 0x7f) === 0) {\n    r += 7;\n    t >>>= 7;\n  }\n  if ((t & 0xf) === 0) {\n    r += 4;\n    t >>>= 4;\n  }\n  if ((t & 0x3) === 0) {\n    r += 2;\n    t >>>= 2;\n  }\n  if ((t & 0x1) === 0)\n    r++;\n  return r;\n};\n\n// Return number of used bits in a BN\nBN.prototype.bitLength = function bitLength() {\n  var hi = 0;\n  var w = this.words[this.length - 1];\n  var hi = this._countBits(w);\n  return (this.length - 1) * 26 + hi;\n};\n\n// Number of trailing zero bits\nBN.prototype.zeroBits = function zeroBits() {\n  if (this.cmpn(0) === 0)\n    return 0;\n\n  var r = 0;\n  for (var i = 0; i < this.length; i++) {\n    var b = this._zeroBits(this.words[i]);\n    r += b;\n    if (b !== 26)\n      break;\n  }\n  return r;\n};\n\nBN.prototype.byteLength = function byteLength() {\n  return Math.ceil(this.bitLength() / 8);\n};\n\n// Return negative clone of `this`\nBN.prototype.neg = function neg() {\n  if (this.cmpn(0) === 0)\n    return this.clone();\n\n  var r = this.clone();\n  r.sign = !this.sign;\n  return r;\n};\n\n\n// Or `num` with `this` in-place\nBN.prototype.ior = function ior(num) {\n  this.sign = this.sign || num.sign;\n\n  while (this.length < num.length)\n    this.words[this.length++] = 0;\n\n  for (var i = 0; i < num.length; i++)\n    this.words[i] = this.words[i] | num.words[i];\n\n  return this.strip();\n};\n\n\n// Or `num` with `this`\nBN.prototype.or = function or(num) {\n  if (this.length > num.length)\n    return this.clone().ior(num);\n  else\n    return num.clone().ior(this);\n};\n\n\n// And `num` with `this` in-place\nBN.prototype.iand = function iand(num) {\n  this.sign = this.sign && num.sign;\n\n  // b = min-length(num, this)\n  var b;\n  if (this.length > num.length)\n    b = num;\n  else\n    b = this;\n\n  for (var i = 0; i < b.length; i++)\n    this.words[i] = this.words[i] & num.words[i];\n\n  this.length = b.length;\n\n  return this.strip();\n};\n\n\n// And `num` with `this`\nBN.prototype.and = function and(num) {\n  if (this.length > num.length)\n    return this.clone().iand(num);\n  else\n    return num.clone().iand(this);\n};\n\n\n// Xor `num` with `this` in-place\nBN.prototype.ixor = function ixor(num) {\n  this.sign = this.sign || num.sign;\n\n  // a.length > b.length\n  var a;\n  var b;\n  if (this.length > num.length) {\n    a = this;\n    b = num;\n  } else {\n    a = num;\n    b = this;\n  }\n\n  for (var i = 0; i < b.length; i++)\n    this.words[i] = a.words[i] ^ b.words[i];\n\n  if (this !== a)\n    for (; i < a.length; i++)\n      this.words[i] = a.words[i];\n\n  this.length = a.length;\n\n  return this.strip();\n};\n\n\n// Xor `num` with `this`\nBN.prototype.xor = function xor(num) {\n  if (this.length > num.length)\n    return this.clone().ixor(num);\n  else\n    return num.clone().ixor(this);\n};\n\n\n// Set `bit` of `this`\nBN.prototype.setn = function setn(bit, val) {\n  assert(typeof bit === 'number' && bit >= 0);\n\n  var off = (bit / 26) | 0;\n  var wbit = bit % 26;\n\n  while (this.length <= off)\n    this.words[this.length++] = 0;\n\n  if (val)\n    this.words[off] = this.words[off] | (1 << wbit);\n  else\n    this.words[off] = this.words[off] & ~(1 << wbit);\n\n  return this.strip();\n};\n\n\n// Add `num` to `this` in-place\nBN.prototype.iadd = function iadd(num) {\n  // negative + positive\n  if (this.sign && !num.sign) {\n    this.sign = false;\n    var r = this.isub(num);\n    this.sign = !this.sign;\n    return this._normSign();\n\n  // positive + negative\n  } else if (!this.sign && num.sign) {\n    num.sign = false;\n    var r = this.isub(num);\n    num.sign = true;\n    return r._normSign();\n  }\n\n  // a.length > b.length\n  var a;\n  var b;\n  if (this.length > num.length) {\n    a = this;\n    b = num;\n  } else {\n    a = num;\n    b = this;\n  }\n\n  var carry = 0;\n  for (var i = 0; i < b.length; i++) {\n    var r = a.words[i] + b.words[i] + carry;\n    this.words[i] = r & 0x3ffffff;\n    carry = r >>> 26;\n  }\n  for (; carry !== 0 && i < a.length; i++) {\n    var r = a.words[i] + carry;\n    this.words[i] = r & 0x3ffffff;\n    carry = r >>> 26;\n  }\n\n  this.length = a.length;\n  if (carry !== 0) {\n    this.words[this.length] = carry;\n    this.length++;\n  // Copy the rest of the words\n  } else if (a !== this) {\n    for (; i < a.length; i++)\n      this.words[i] = a.words[i];\n  }\n\n  return this;\n};\n\n// Add `num` to `this`\nBN.prototype.add = function add(num) {\n  if (num.sign && !this.sign) {\n    num.sign = false;\n    var res = this.sub(num);\n    num.sign = true;\n    return res;\n  } else if (!num.sign && this.sign) {\n    this.sign = false;\n    var res = num.sub(this);\n    this.sign = true;\n    return res;\n  }\n\n  if (this.length > num.length)\n    return this.clone().iadd(num);\n  else\n    return num.clone().iadd(this);\n};\n\n// Subtract `num` from `this` in-place\nBN.prototype.isub = function isub(num) {\n  // this - (-num) = this + num\n  if (num.sign) {\n    num.sign = false;\n    var r = this.iadd(num);\n    num.sign = true;\n    return r._normSign();\n\n  // -this - num = -(this + num)\n  } else if (this.sign) {\n    this.sign = false;\n    this.iadd(num);\n    this.sign = true;\n    return this._normSign();\n  }\n\n  // At this point both numbers are positive\n  var cmp = this.cmp(num);\n\n  // Optimization - zeroify\n  if (cmp === 0) {\n    this.sign = false;\n    this.length = 1;\n    this.words[0] = 0;\n    return this;\n  }\n\n  // a > b\n  var a;\n  var b;\n  if (cmp > 0) {\n    a = this;\n    b = num;\n  } else {\n    a = num;\n    b = this;\n  }\n\n  var carry = 0;\n  for (var i = 0; i < b.length; i++) {\n    var r = a.words[i] - b.words[i] + carry;\n    carry = r >> 26;\n    this.words[i] = r & 0x3ffffff;\n  }\n  for (; carry !== 0 && i < a.length; i++) {\n    var r = a.words[i] + carry;\n    carry = r >> 26;\n    this.words[i] = r & 0x3ffffff;\n  }\n\n  // Copy rest of the words\n  if (carry === 0 && i < a.length && a !== this)\n    for (; i < a.length; i++)\n      this.words[i] = a.words[i];\n  this.length = Math.max(this.length, i);\n\n  if (a !== this)\n    this.sign = true;\n\n  return this.strip();\n};\n\n// Subtract `num` from `this`\nBN.prototype.sub = function sub(num) {\n  return this.clone().isub(num);\n};\n\n/*\n// NOTE: This could be potentionally used to generate loop-less multiplications\nfunction _genCombMulTo(alen, blen) {\n  var len = alen + blen - 1;\n  var src = [\n    'var a = this.words, b = num.words, o = out.words, c = 0, w, ' +\n        'mask = 0x3ffffff, shift = 0x4000000;',\n    'out.length = ' + len + ';'\n  ];\n  for (var k = 0; k < len; k++) {\n    var minJ = Math.max(0, k - alen + 1);\n    var maxJ = Math.min(k, blen - 1);\n\n    for (var j = minJ; j <= maxJ; j++) {\n      var i = k - j;\n      var mul = 'a[' + i + '] * b[' + j + ']';\n\n      if (j === minJ) {\n        src.push('w = ' + mul + ' + c;');\n        src.push('c = (w / shift) | 0;');\n      } else {\n        src.push('w += ' + mul + ';');\n        src.push('c += (w / shift) | 0;');\n      }\n      src.push('w &= mask;');\n    }\n    src.push('o[' + k + '] = w;');\n  }\n  src.push('if (c !== 0) {',\n           '  o[' + k + '] = c;',\n           '  out.length++;',\n           '}',\n           'return out;');\n\n  return src.join('\\n');\n}\n*/\n\nBN.prototype._smallMulTo = function _smallMulTo(num, out) {\n  out.sign = num.sign !== this.sign;\n  out.length = this.length + num.length;\n\n  var carry = 0;\n  for (var k = 0; k < out.length - 1; k++) {\n    // Sum all words with the same `i + j = k` and accumulate `ncarry`,\n    // note that ncarry could be >= 0x3ffffff\n    var ncarry = carry >>> 26;\n    var rword = carry & 0x3ffffff;\n    var maxJ = Math.min(k, num.length - 1);\n    for (var j = Math.max(0, k - this.length + 1); j <= maxJ; j++) {\n      var i = k - j;\n      var a = this.words[i] | 0;\n      var b = num.words[j] | 0;\n      var r = a * b;\n\n      var lo = r & 0x3ffffff;\n      ncarry = (ncarry + ((r / 0x4000000) | 0)) | 0;\n      lo = (lo + rword) | 0;\n      rword = lo & 0x3ffffff;\n      ncarry = (ncarry + (lo >>> 26)) | 0;\n    }\n    out.words[k] = rword;\n    carry = ncarry;\n  }\n  if (carry !== 0) {\n    out.words[k] = carry;\n  } else {\n    out.length--;\n  }\n\n  return out.strip();\n};\n\nBN.prototype._bigMulTo = function _bigMulTo(num, out) {\n  out.sign = num.sign !== this.sign;\n  out.length = this.length + num.length;\n\n  var carry = 0;\n  var hncarry = 0;\n  for (var k = 0; k < out.length - 1; k++) {\n    // Sum all words with the same `i + j = k` and accumulate `ncarry`,\n    // note that ncarry could be >= 0x3ffffff\n    var ncarry = hncarry;\n    hncarry = 0;\n    var rword = carry & 0x3ffffff;\n    var maxJ = Math.min(k, num.length - 1);\n    for (var j = Math.max(0, k - this.length + 1); j <= maxJ; j++) {\n      var i = k - j;\n      var a = this.words[i] | 0;\n      var b = num.words[j] | 0;\n      var r = a * b;\n\n      var lo = r & 0x3ffffff;\n      ncarry = (ncarry + ((r / 0x4000000) | 0)) | 0;\n      lo = (lo + rword) | 0;\n      rword = lo & 0x3ffffff;\n      ncarry = (ncarry + (lo >>> 26)) | 0;\n\n      hncarry += ncarry >>> 26;\n      ncarry &= 0x3ffffff;\n    }\n    out.words[k] = rword;\n    carry = ncarry;\n    ncarry = hncarry;\n  }\n  if (carry !== 0) {\n    out.words[k] = carry;\n  } else {\n    out.length--;\n  }\n\n  return out.strip();\n};\n\nBN.prototype.mulTo = function mulTo(num, out) {\n  var res;\n  if (this.length + num.length < 63)\n    res = this._smallMulTo(num, out);\n  else\n    res = this._bigMulTo(num, out);\n  return res;\n};\n\n// Multiply `this` by `num`\nBN.prototype.mul = function mul(num) {\n  var out = new BN(null);\n  out.words = new Array(this.length + num.length);\n  return this.mulTo(num, out);\n};\n\n// In-place Multiplication\nBN.prototype.imul = function imul(num) {\n  if (this.cmpn(0) === 0 || num.cmpn(0) === 0) {\n    this.words[0] = 0;\n    this.length = 1;\n    return this;\n  }\n\n  var tlen = this.length;\n  var nlen = num.length;\n\n  this.sign = num.sign !== this.sign;\n  this.length = this.length + num.length;\n  this.words[this.length - 1] = 0;\n\n  for (var k = this.length - 2; k >= 0; k--) {\n    // Sum all words with the same `i + j = k` and accumulate `carry`,\n    // note that carry could be >= 0x3ffffff\n    var carry = 0;\n    var rword = 0;\n    var maxJ = Math.min(k, nlen - 1);\n    for (var j = Math.max(0, k - tlen + 1); j <= maxJ; j++) {\n      var i = k - j;\n      var a = this.words[i];\n      var b = num.words[j];\n      var r = a * b;\n\n      var lo = r & 0x3ffffff;\n      carry += (r / 0x4000000) | 0;\n      lo += rword;\n      rword = lo & 0x3ffffff;\n      carry += lo >>> 26;\n    }\n    this.words[k] = rword;\n    this.words[k + 1] += carry;\n    carry = 0;\n  }\n\n  // Propagate overflows\n  var carry = 0;\n  for (var i = 1; i < this.length; i++) {\n    var w = this.words[i] + carry;\n    this.words[i] = w & 0x3ffffff;\n    carry = w >>> 26;\n  }\n\n  return this.strip();\n};\n\nBN.prototype.imuln = function imuln(num) {\n  assert(typeof num === 'number');\n\n  // Carry\n  var carry = 0;\n  for (var i = 0; i < this.length; i++) {\n    var w = this.words[i] * num;\n    var lo = (w & 0x3ffffff) + (carry & 0x3ffffff);\n    carry >>= 26;\n    carry += (w / 0x4000000) | 0;\n    // NOTE: lo is 27bit maximum\n    carry += lo >>> 26;\n    this.words[i] = lo & 0x3ffffff;\n  }\n\n  if (carry !== 0) {\n    this.words[i] = carry;\n    this.length++;\n  }\n\n  return this;\n};\n\nBN.prototype.muln = function muln(num) {\n  return this.clone().imuln(num);\n};\n\n// `this` * `this`\nBN.prototype.sqr = function sqr() {\n  return this.mul(this);\n};\n\n// `this` * `this` in-place\nBN.prototype.isqr = function isqr() {\n  return this.mul(this);\n};\n\n// Shift-left in-place\nBN.prototype.ishln = function ishln(bits) {\n  assert(typeof bits === 'number' && bits >= 0);\n  var r = bits % 26;\n  var s = (bits - r) / 26;\n  var carryMask = (0x3ffffff >>> (26 - r)) << (26 - r);\n\n  if (r !== 0) {\n    var carry = 0;\n    for (var i = 0; i < this.length; i++) {\n      var newCarry = this.words[i] & carryMask;\n      var c = (this.words[i] - newCarry) << r;\n      this.words[i] = c | carry;\n      carry = newCarry >>> (26 - r);\n    }\n    if (carry) {\n      this.words[i] = carry;\n      this.length++;\n    }\n  }\n\n  if (s !== 0) {\n    for (var i = this.length - 1; i >= 0; i--)\n      this.words[i + s] = this.words[i];\n    for (var i = 0; i < s; i++)\n      this.words[i] = 0;\n    this.length += s;\n  }\n\n  return this.strip();\n};\n\n// Shift-right in-place\n// NOTE: `hint` is a lowest bit before trailing zeroes\n// NOTE: if `extended` is present - it will be filled with destroyed bits\nBN.prototype.ishrn = function ishrn(bits, hint, extended) {\n  assert(typeof bits === 'number' && bits >= 0);\n  var h;\n  if (hint)\n    h = (hint - (hint % 26)) / 26;\n  else\n    h = 0;\n\n  var r = bits % 26;\n  var s = Math.min((bits - r) / 26, this.length);\n  var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r);\n  var maskedWords = extended;\n\n  h -= s;\n  h = Math.max(0, h);\n\n  // Extended mode, copy masked part\n  if (maskedWords) {\n    for (var i = 0; i < s; i++)\n      maskedWords.words[i] = this.words[i];\n    maskedWords.length = s;\n  }\n\n  if (s === 0) {\n    // No-op, we should not move anything at all\n  } else if (this.length > s) {\n    this.length -= s;\n    for (var i = 0; i < this.length; i++)\n      this.words[i] = this.words[i + s];\n  } else {\n    this.words[0] = 0;\n    this.length = 1;\n  }\n\n  var carry = 0;\n  for (var i = this.length - 1; i >= 0 && (carry !== 0 || i >= h); i--) {\n    var word = this.words[i];\n    this.words[i] = (carry << (26 - r)) | (word >>> r);\n    carry = word & mask;\n  }\n\n  // Push carried bits as a mask\n  if (maskedWords && carry !== 0)\n    maskedWords.words[maskedWords.length++] = carry;\n\n  if (this.length === 0) {\n    this.words[0] = 0;\n    this.length = 1;\n  }\n\n  this.strip();\n\n  return this;\n};\n\n// Shift-left\nBN.prototype.shln = function shln(bits) {\n  return this.clone().ishln(bits);\n};\n\n// Shift-right\nBN.prototype.shrn = function shrn(bits) {\n  return this.clone().ishrn(bits);\n};\n\n// Test if n bit is set\nBN.prototype.testn = function testn(bit) {\n  assert(typeof bit === 'number' && bit >= 0);\n  var r = bit % 26;\n  var s = (bit - r) / 26;\n  var q = 1 << r;\n\n  // Fast case: bit is much higher than all existing words\n  if (this.length <= s) {\n    return false;\n  }\n\n  // Check bit and return\n  var w = this.words[s];\n\n  return !!(w & q);\n};\n\n// Return only lowers bits of number (in-place)\nBN.prototype.imaskn = function imaskn(bits) {\n  assert(typeof bits === 'number' && bits >= 0);\n  var r = bits % 26;\n  var s = (bits - r) / 26;\n\n  assert(!this.sign, 'imaskn works only with positive numbers');\n\n  if (r !== 0)\n    s++;\n  this.length = Math.min(s, this.length);\n\n  if (r !== 0) {\n    var mask = 0x3ffffff ^ ((0x3ffffff >>> r) << r);\n    this.words[this.length - 1] &= mask;\n  }\n\n  return this.strip();\n};\n\n// Return only lowers bits of number\nBN.prototype.maskn = function maskn(bits) {\n  return this.clone().imaskn(bits);\n};\n\n// Add plain number `num` to `this`\nBN.prototype.iaddn = function iaddn(num) {\n  assert(typeof num === 'number');\n  if (num < 0)\n    return this.isubn(-num);\n\n  // Possible sign change\n  if (this.sign) {\n    if (this.length === 1 && this.words[0] < num) {\n      this.words[0] = num - this.words[0];\n      this.sign = false;\n      return this;\n    }\n\n    this.sign = false;\n    this.isubn(num);\n    this.sign = true;\n    return this;\n  }\n\n  // Add without checks\n  return this._iaddn(num);\n};\n\nBN.prototype._iaddn = function _iaddn(num) {\n  this.words[0] += num;\n\n  // Carry\n  for (var i = 0; i < this.length && this.words[i] >= 0x4000000; i++) {\n    this.words[i] -= 0x4000000;\n    if (i === this.length - 1)\n      this.words[i + 1] = 1;\n    else\n      this.words[i + 1]++;\n  }\n  this.length = Math.max(this.length, i + 1);\n\n  return this;\n};\n\n// Subtract plain number `num` from `this`\nBN.prototype.isubn = function isubn(num) {\n  assert(typeof num === 'number');\n  if (num < 0)\n    return this.iaddn(-num);\n\n  if (this.sign) {\n    this.sign = false;\n    this.iaddn(num);\n    this.sign = true;\n    return this;\n  }\n\n  this.words[0] -= num;\n\n  // Carry\n  for (var i = 0; i < this.length && this.words[i] < 0; i++) {\n    this.words[i] += 0x4000000;\n    this.words[i + 1] -= 1;\n  }\n\n  return this.strip();\n};\n\nBN.prototype.addn = function addn(num) {\n  return this.clone().iaddn(num);\n};\n\nBN.prototype.subn = function subn(num) {\n  return this.clone().isubn(num);\n};\n\nBN.prototype.iabs = function iabs() {\n  this.sign = false;\n\n  return this;\n};\n\nBN.prototype.abs = function abs() {\n  return this.clone().iabs();\n};\n\nBN.prototype._ishlnsubmul = function _ishlnsubmul(num, mul, shift) {\n  // Bigger storage is needed\n  var len = num.length + shift;\n  var i;\n  if (this.words.length < len) {\n    var t = new Array(len);\n    for (var i = 0; i < this.length; i++)\n      t[i] = this.words[i];\n    this.words = t;\n  } else {\n    i = this.length;\n  }\n\n  // Zeroify rest\n  this.length = Math.max(this.length, len);\n  for (; i < this.length; i++)\n    this.words[i] = 0;\n\n  var carry = 0;\n  for (var i = 0; i < num.length; i++) {\n    var w = this.words[i + shift] + carry;\n    var right = num.words[i] * mul;\n    w -= right & 0x3ffffff;\n    carry = (w >> 26) - ((right / 0x4000000) | 0);\n    this.words[i + shift] = w & 0x3ffffff;\n  }\n  for (; i < this.length - shift; i++) {\n    var w = this.words[i + shift] + carry;\n    carry = w >> 26;\n    this.words[i + shift] = w & 0x3ffffff;\n  }\n\n  if (carry === 0)\n    return this.strip();\n\n  // Subtraction overflow\n  assert(carry === -1);\n  carry = 0;\n  for (var i = 0; i < this.length; i++) {\n    var w = -this.words[i] + carry;\n    carry = w >> 26;\n    this.words[i] = w & 0x3ffffff;\n  }\n  this.sign = true;\n\n  return this.strip();\n};\n\nBN.prototype._wordDiv = function _wordDiv(num, mode) {\n  var shift = this.length - num.length;\n\n  var a = this.clone();\n  var b = num;\n\n  // Normalize\n  var bhi = b.words[b.length - 1];\n  var bhiBits = this._countBits(bhi);\n  shift = 26 - bhiBits;\n  if (shift !== 0) {\n    b = b.shln(shift);\n    a.ishln(shift);\n    bhi = b.words[b.length - 1];\n  }\n\n  // Initialize quotient\n  var m = a.length - b.length;\n  var q;\n\n  if (mode !== 'mod') {\n    q = new BN(null);\n    q.length = m + 1;\n    q.words = new Array(q.length);\n    for (var i = 0; i < q.length; i++)\n      q.words[i] = 0;\n  }\n\n  var diff = a.clone()._ishlnsubmul(b, 1, m);\n  if (!diff.sign) {\n    a = diff;\n    if (q)\n      q.words[m] = 1;\n  }\n\n  for (var j = m - 1; j >= 0; j--) {\n    var qj = a.words[b.length + j] * 0x4000000 + a.words[b.length + j - 1];\n\n    // NOTE: (qj / bhi) is (0x3ffffff * 0x4000000 + 0x3ffffff) / 0x2000000 max\n    // (0x7ffffff)\n    qj = Math.min((qj / bhi) | 0, 0x3ffffff);\n\n    a._ishlnsubmul(b, qj, j);\n    while (a.sign) {\n      qj--;\n      a.sign = false;\n      a._ishlnsubmul(b, 1, j);\n      if (a.cmpn(0) !== 0)\n        a.sign = !a.sign;\n    }\n    if (q)\n      q.words[j] = qj;\n  }\n  if (q)\n    q.strip();\n  a.strip();\n\n  // Denormalize\n  if (mode !== 'div' && shift !== 0)\n    a.ishrn(shift);\n  return { div: q ? q : null, mod: a };\n};\n\nBN.prototype.divmod = function divmod(num, mode) {\n  assert(num.cmpn(0) !== 0);\n\n  if (this.sign && !num.sign) {\n    var res = this.neg().divmod(num, mode);\n    var div;\n    var mod;\n    if (mode !== 'mod')\n      div = res.div.neg();\n    if (mode !== 'div')\n      mod = res.mod.cmpn(0) === 0 ? res.mod : num.sub(res.mod);\n    return {\n      div: div,\n      mod: mod\n    };\n  } else if (!this.sign && num.sign) {\n    var res = this.divmod(num.neg(), mode);\n    var div;\n    if (mode !== 'mod')\n      div = res.div.neg();\n    return { div: div, mod: res.mod };\n  } else if (this.sign && num.sign) {\n    return this.neg().divmod(num.neg(), mode);\n  }\n\n  // Both numbers are positive at this point\n\n  // Strip both numbers to approximate shift value\n  if (num.length > this.length || this.cmp(num) < 0)\n    return { div: new BN(0), mod: this };\n\n  // Very short reduction\n  if (num.length === 1) {\n    if (mode === 'div')\n      return { div: this.divn(num.words[0]), mod: null };\n    else if (mode === 'mod')\n      return { div: null, mod: new BN(this.modn(num.words[0])) };\n    return {\n      div: this.divn(num.words[0]),\n      mod: new BN(this.modn(num.words[0]))\n    };\n  }\n\n  return this._wordDiv(num, mode);\n};\n\n// Find `this` / `num`\nBN.prototype.div = function div(num) {\n  return this.divmod(num, 'div').div;\n};\n\n// Find `this` % `num`\nBN.prototype.mod = function mod(num) {\n  return this.divmod(num, 'mod').mod;\n};\n\n// Find Round(`this` / `num`)\nBN.prototype.divRound = function divRound(num) {\n  var dm = this.divmod(num);\n\n  // Fast case - exact division\n  if (dm.mod.cmpn(0) === 0)\n    return dm.div;\n\n  var mod = dm.div.sign ? dm.mod.isub(num) : dm.mod;\n\n  var half = num.shrn(1);\n  var r2 = num.andln(1);\n  var cmp = mod.cmp(half);\n\n  // Round down\n  if (cmp < 0 || r2 === 1 && cmp === 0)\n    return dm.div;\n\n  // Round up\n  return dm.div.sign ? dm.div.isubn(1) : dm.div.iaddn(1);\n};\n\nBN.prototype.modn = function modn(num) {\n  assert(num <= 0x3ffffff);\n  var p = (1 << 26) % num;\n\n  var acc = 0;\n  for (var i = this.length - 1; i >= 0; i--)\n    acc = (p * acc + this.words[i]) % num;\n\n  return acc;\n};\n\n// In-place division by number\nBN.prototype.idivn = function idivn(num) {\n  assert(num <= 0x3ffffff);\n\n  var carry = 0;\n  for (var i = this.length - 1; i >= 0; i--) {\n    var w = this.words[i] + carry * 0x4000000;\n    this.words[i] = (w / num) | 0;\n    carry = w % num;\n  }\n\n  return this.strip();\n};\n\nBN.prototype.divn = function divn(num) {\n  return this.clone().idivn(num);\n};\n\nBN.prototype.egcd = function egcd(p) {\n  assert(!p.sign);\n  assert(p.cmpn(0) !== 0);\n\n  var x = this;\n  var y = p.clone();\n\n  if (x.sign)\n    x = x.mod(p);\n  else\n    x = x.clone();\n\n  // A * x + B * y = x\n  var A = new BN(1);\n  var B = new BN(0);\n\n  // C * x + D * y = y\n  var C = new BN(0);\n  var D = new BN(1);\n\n  var g = 0;\n\n  while (x.isEven() && y.isEven()) {\n    x.ishrn(1);\n    y.ishrn(1);\n    ++g;\n  }\n\n  var yp = y.clone();\n  var xp = x.clone();\n\n  while (x.cmpn(0) !== 0) {\n    while (x.isEven()) {\n      x.ishrn(1);\n      if (A.isEven() && B.isEven()) {\n        A.ishrn(1);\n        B.ishrn(1);\n      } else {\n        A.iadd(yp).ishrn(1);\n        B.isub(xp).ishrn(1);\n      }\n    }\n\n    while (y.isEven()) {\n      y.ishrn(1);\n      if (C.isEven() && D.isEven()) {\n        C.ishrn(1);\n        D.ishrn(1);\n      } else {\n        C.iadd(yp).ishrn(1);\n        D.isub(xp).ishrn(1);\n      }\n    }\n\n    if (x.cmp(y) >= 0) {\n      x.isub(y);\n      A.isub(C);\n      B.isub(D);\n    } else {\n      y.isub(x);\n      C.isub(A);\n      D.isub(B);\n    }\n  }\n\n  return {\n    a: C,\n    b: D,\n    gcd: y.ishln(g)\n  };\n};\n\n// This is reduced incarnation of the binary EEA\n// above, designated to invert members of the\n// _prime_ fields F(p) at a maximal speed\nBN.prototype._invmp = function _invmp(p) {\n  assert(!p.sign);\n  assert(p.cmpn(0) !== 0);\n\n  var a = this;\n  var b = p.clone();\n\n  if (a.sign)\n    a = a.mod(p);\n  else\n    a = a.clone();\n\n  var x1 = new BN(1);\n  var x2 = new BN(0);\n\n  var delta = b.clone();\n\n  while (a.cmpn(1) > 0 && b.cmpn(1) > 0) {\n    while (a.isEven()) {\n      a.ishrn(1);\n      if (x1.isEven())\n        x1.ishrn(1);\n      else\n        x1.iadd(delta).ishrn(1);\n    }\n    while (b.isEven()) {\n      b.ishrn(1);\n      if (x2.isEven())\n        x2.ishrn(1);\n      else\n        x2.iadd(delta).ishrn(1);\n    }\n    if (a.cmp(b) >= 0) {\n      a.isub(b);\n      x1.isub(x2);\n    } else {\n      b.isub(a);\n      x2.isub(x1);\n    }\n  }\n  if (a.cmpn(1) === 0)\n    return x1;\n  else\n    return x2;\n};\n\nBN.prototype.gcd = function gcd(num) {\n  if (this.cmpn(0) === 0)\n    return num.clone();\n  if (num.cmpn(0) === 0)\n    return this.clone();\n\n  var a = this.clone();\n  var b = num.clone();\n  a.sign = false;\n  b.sign = false;\n\n  // Remove common factor of two\n  for (var shift = 0; a.isEven() && b.isEven(); shift++) {\n    a.ishrn(1);\n    b.ishrn(1);\n  }\n\n  do {\n    while (a.isEven())\n      a.ishrn(1);\n    while (b.isEven())\n      b.ishrn(1);\n\n    var r = a.cmp(b);\n    if (r < 0) {\n      // Swap `a` and `b` to make `a` always bigger than `b`\n      var t = a;\n      a = b;\n      b = t;\n    } else if (r === 0 || b.cmpn(1) === 0) {\n      break;\n    }\n\n    a.isub(b);\n  } while (true);\n\n  return b.ishln(shift);\n};\n\n// Invert number in the field F(num)\nBN.prototype.invm = function invm(num) {\n  return this.egcd(num).a.mod(num);\n};\n\nBN.prototype.isEven = function isEven() {\n  return (this.words[0] & 1) === 0;\n};\n\nBN.prototype.isOdd = function isOdd() {\n  return (this.words[0] & 1) === 1;\n};\n\n// And first word and num\nBN.prototype.andln = function andln(num) {\n  return this.words[0] & num;\n};\n\n// Increment at the bit position in-line\nBN.prototype.bincn = function bincn(bit) {\n  assert(typeof bit === 'number');\n  var r = bit % 26;\n  var s = (bit - r) / 26;\n  var q = 1 << r;\n\n  // Fast case: bit is much higher than all existing words\n  if (this.length <= s) {\n    for (var i = this.length; i < s + 1; i++)\n      this.words[i] = 0;\n    this.words[s] |= q;\n    this.length = s + 1;\n    return this;\n  }\n\n  // Add bit and propagate, if needed\n  var carry = q;\n  for (var i = s; carry !== 0 && i < this.length; i++) {\n    var w = this.words[i];\n    w += carry;\n    carry = w >>> 26;\n    w &= 0x3ffffff;\n    this.words[i] = w;\n  }\n  if (carry !== 0) {\n    this.words[i] = carry;\n    this.length++;\n  }\n  return this;\n};\n\nBN.prototype.cmpn = function cmpn(num) {\n  var sign = num < 0;\n  if (sign)\n    num = -num;\n\n  if (this.sign && !sign)\n    return -1;\n  else if (!this.sign && sign)\n    return 1;\n\n  num &= 0x3ffffff;\n  this.strip();\n\n  var res;\n  if (this.length > 1) {\n    res = 1;\n  } else {\n    var w = this.words[0];\n    res = w === num ? 0 : w < num ? -1 : 1;\n  }\n  if (this.sign)\n    res = -res;\n  return res;\n};\n\n// Compare two numbers and return:\n// 1 - if `this` > `num`\n// 0 - if `this` == `num`\n// -1 - if `this` < `num`\nBN.prototype.cmp = function cmp(num) {\n  if (this.sign && !num.sign)\n    return -1;\n  else if (!this.sign && num.sign)\n    return 1;\n\n  var res = this.ucmp(num);\n  if (this.sign)\n    return -res;\n  else\n    return res;\n};\n\n// Unsigned comparison\nBN.prototype.ucmp = function ucmp(num) {\n  // At this point both numbers have the same sign\n  if (this.length > num.length)\n    return 1;\n  else if (this.length < num.length)\n    return -1;\n\n  var res = 0;\n  for (var i = this.length - 1; i >= 0; i--) {\n    var a = this.words[i];\n    var b = num.words[i];\n\n    if (a === b)\n      continue;\n    if (a < b)\n      res = -1;\n    else if (a > b)\n      res = 1;\n    break;\n  }\n  return res;\n};\n\n//\n// A reduce context, could be using montgomery or something better, depending\n// on the `m` itself.\n//\nBN.red = function red(num) {\n  return new Red(num);\n};\n\nBN.prototype.toRed = function toRed(ctx) {\n  assert(!this.red, 'Already a number in reduction context');\n  assert(!this.sign, 'red works only with positives');\n  return ctx.convertTo(this)._forceRed(ctx);\n};\n\nBN.prototype.fromRed = function fromRed() {\n  assert(this.red, 'fromRed works only with numbers in reduction context');\n  return this.red.convertFrom(this);\n};\n\nBN.prototype._forceRed = function _forceRed(ctx) {\n  this.red = ctx;\n  return this;\n};\n\nBN.prototype.forceRed = function forceRed(ctx) {\n  assert(!this.red, 'Already a number in reduction context');\n  return this._forceRed(ctx);\n};\n\nBN.prototype.redAdd = function redAdd(num) {\n  assert(this.red, 'redAdd works only with red numbers');\n  return this.red.add(this, num);\n};\n\nBN.prototype.redIAdd = function redIAdd(num) {\n  assert(this.red, 'redIAdd works only with red numbers');\n  return this.red.iadd(this, num);\n};\n\nBN.prototype.redSub = function redSub(num) {\n  assert(this.red, 'redSub works only with red numbers');\n  return this.red.sub(this, num);\n};\n\nBN.prototype.redISub = function redISub(num) {\n  assert(this.red, 'redISub works only with red numbers');\n  return this.red.isub(this, num);\n};\n\nBN.prototype.redShl = function redShl(num) {\n  assert(this.red, 'redShl works only with red numbers');\n  return this.red.shl(this, num);\n};\n\nBN.prototype.redMul = function redMul(num) {\n  assert(this.red, 'redMul works only with red numbers');\n  this.red._verify2(this, num);\n  return this.red.mul(this, num);\n};\n\nBN.prototype.redIMul = function redIMul(num) {\n  assert(this.red, 'redMul works only with red numbers');\n  this.red._verify2(this, num);\n  return this.red.imul(this, num);\n};\n\nBN.prototype.redSqr = function redSqr() {\n  assert(this.red, 'redSqr works only with red numbers');\n  this.red._verify1(this);\n  return this.red.sqr(this);\n};\n\nBN.prototype.redISqr = function redISqr() {\n  assert(this.red, 'redISqr works only with red numbers');\n  this.red._verify1(this);\n  return this.red.isqr(this);\n};\n\n// Square root over p\nBN.prototype.redSqrt = function redSqrt() {\n  assert(this.red, 'redSqrt works only with red numbers');\n  this.red._verify1(this);\n  return this.red.sqrt(this);\n};\n\nBN.prototype.redInvm = function redInvm() {\n  assert(this.red, 'redInvm works only with red numbers');\n  this.red._verify1(this);\n  return this.red.invm(this);\n};\n\n// Return negative clone of `this` % `red modulo`\nBN.prototype.redNeg = function redNeg() {\n  assert(this.red, 'redNeg works only with red numbers');\n  this.red._verify1(this);\n  return this.red.neg(this);\n};\n\nBN.prototype.redPow = function redPow(num) {\n  assert(this.red && !num.red, 'redPow(normalNum)');\n  this.red._verify1(this);\n  return this.red.pow(this, num);\n};\n\n// Prime numbers with efficient reduction\nvar primes = {\n  k256: null,\n  p224: null,\n  p192: null,\n  p25519: null\n};\n\n// Pseudo-Mersenne prime\nfunction MPrime(name, p) {\n  // P = 2 ^ N - K\n  this.name = name;\n  this.p = new BN(p, 16);\n  this.n = this.p.bitLength();\n  this.k = new BN(1).ishln(this.n).isub(this.p);\n\n  this.tmp = this._tmp();\n}\n\nMPrime.prototype._tmp = function _tmp() {\n  var tmp = new BN(null);\n  tmp.words = new Array(Math.ceil(this.n / 13));\n  return tmp;\n};\n\nMPrime.prototype.ireduce = function ireduce(num) {\n  // Assumes that `num` is less than `P^2`\n  // num = HI * (2 ^ N - K) + HI * K + LO = HI * K + LO (mod P)\n  var r = num;\n  var rlen;\n\n  do {\n    this.split(r, this.tmp);\n    r = this.imulK(r);\n    r = r.iadd(this.tmp);\n    rlen = r.bitLength();\n  } while (rlen > this.n);\n\n  var cmp = rlen < this.n ? -1 : r.ucmp(this.p);\n  if (cmp === 0) {\n    r.words[0] = 0;\n    r.length = 1;\n  } else if (cmp > 0) {\n    r.isub(this.p);\n  } else {\n    r.strip();\n  }\n\n  return r;\n};\n\nMPrime.prototype.split = function split(input, out) {\n  input.ishrn(this.n, 0, out);\n};\n\nMPrime.prototype.imulK = function imulK(num) {\n  return num.imul(this.k);\n};\n\nfunction K256() {\n  MPrime.call(\n    this,\n    'k256',\n    'ffffffff ffffffff ffffffff ffffffff ffffffff ffffffff fffffffe fffffc2f');\n}\ninherits(K256, MPrime);\n\nK256.prototype.split = function split(input, output) {\n  // 256 = 9 * 26 + 22\n  var mask = 0x3fffff;\n\n  var outLen = Math.min(input.length, 9);\n  for (var i = 0; i < outLen; i++)\n    output.words[i] = input.words[i];\n  output.length = outLen;\n\n  if (input.length <= 9) {\n    input.words[0] = 0;\n    input.length = 1;\n    return;\n  }\n\n  // Shift by 9 limbs\n  var prev = input.words[9];\n  output.words[output.length++] = prev & mask;\n\n  for (var i = 10; i < input.length; i++) {\n    var next = input.words[i];\n    input.words[i - 10] = ((next & mask) << 4) | (prev >>> 22);\n    prev = next;\n  }\n  input.words[i - 10] = prev >>> 22;\n  input.length -= 9;\n};\n\nK256.prototype.imulK = function imulK(num) {\n  // K = 0x1000003d1 = [ 0x40, 0x3d1 ]\n  num.words[num.length] = 0;\n  num.words[num.length + 1] = 0;\n  num.length += 2;\n\n  // bounded at: 0x40 * 0x3ffffff + 0x3d0 = 0x100000390\n  var hi;\n  var lo = 0;\n  for (var i = 0; i < num.length; i++) {\n    var w = num.words[i];\n    hi = w * 0x40;\n    lo += w * 0x3d1;\n    hi += (lo / 0x4000000) | 0;\n    lo &= 0x3ffffff;\n\n    num.words[i] = lo;\n\n    lo = hi;\n  }\n\n  // Fast length reduction\n  if (num.words[num.length - 1] === 0) {\n    num.length--;\n    if (num.words[num.length - 1] === 0)\n      num.length--;\n  }\n  return num;\n};\n\nfunction P224() {\n  MPrime.call(\n    this,\n    'p224',\n    'ffffffff ffffffff ffffffff ffffffff 00000000 00000000 00000001');\n}\ninherits(P224, MPrime);\n\nfunction P192() {\n  MPrime.call(\n    this,\n    'p192',\n    'ffffffff ffffffff ffffffff fffffffe ffffffff ffffffff');\n}\ninherits(P192, MPrime);\n\nfunction P25519() {\n  // 2 ^ 255 - 19\n  MPrime.call(\n    this,\n    '25519',\n    '7fffffffffffffff ffffffffffffffff ffffffffffffffff ffffffffffffffed');\n}\ninherits(P25519, MPrime);\n\nP25519.prototype.imulK = function imulK(num) {\n  // K = 0x13\n  var carry = 0;\n  for (var i = 0; i < num.length; i++) {\n    var hi = num.words[i] * 0x13 + carry;\n    var lo = hi & 0x3ffffff;\n    hi >>>= 26;\n\n    num.words[i] = lo;\n    carry = hi;\n  }\n  if (carry !== 0)\n    num.words[num.length++] = carry;\n  return num;\n};\n\n// Exported mostly for testing purposes, use plain name instead\nBN._prime = function prime(name) {\n  // Cached version of prime\n  if (primes[name])\n    return primes[name];\n\n  var prime;\n  if (name === 'k256')\n    prime = new K256();\n  else if (name === 'p224')\n    prime = new P224();\n  else if (name === 'p192')\n    prime = new P192();\n  else if (name === 'p25519')\n    prime = new P25519();\n  else\n    throw new Error('Unknown prime ' + name);\n  primes[name] = prime;\n\n  return prime;\n};\n\n//\n// Base reduction engine\n//\nfunction Red(m) {\n  if (typeof m === 'string') {\n    var prime = BN._prime(m);\n    this.m = prime.p;\n    this.prime = prime;\n  } else {\n    this.m = m;\n    this.prime = null;\n  }\n}\n\nRed.prototype._verify1 = function _verify1(a) {\n  assert(!a.sign, 'red works only with positives');\n  assert(a.red, 'red works only with red numbers');\n};\n\nRed.prototype._verify2 = function _verify2(a, b) {\n  assert(!a.sign && !b.sign, 'red works only with positives');\n  assert(a.red && a.red === b.red,\n         'red works only with red numbers');\n};\n\nRed.prototype.imod = function imod(a) {\n  if (this.prime)\n    return this.prime.ireduce(a)._forceRed(this);\n  return a.mod(this.m)._forceRed(this);\n};\n\nRed.prototype.neg = function neg(a) {\n  var r = a.clone();\n  r.sign = !r.sign;\n  return r.iadd(this.m)._forceRed(this);\n};\n\nRed.prototype.add = function add(a, b) {\n  this._verify2(a, b);\n\n  var res = a.add(b);\n  if (res.cmp(this.m) >= 0)\n    res.isub(this.m);\n  return res._forceRed(this);\n};\n\nRed.prototype.iadd = function iadd(a, b) {\n  this._verify2(a, b);\n\n  var res = a.iadd(b);\n  if (res.cmp(this.m) >= 0)\n    res.isub(this.m);\n  return res;\n};\n\nRed.prototype.sub = function sub(a, b) {\n  this._verify2(a, b);\n\n  var res = a.sub(b);\n  if (res.cmpn(0) < 0)\n    res.iadd(this.m);\n  return res._forceRed(this);\n};\n\nRed.prototype.isub = function isub(a, b) {\n  this._verify2(a, b);\n\n  var res = a.isub(b);\n  if (res.cmpn(0) < 0)\n    res.iadd(this.m);\n  return res;\n};\n\nRed.prototype.shl = function shl(a, num) {\n  this._verify1(a);\n  return this.imod(a.shln(num));\n};\n\nRed.prototype.imul = function imul(a, b) {\n  this._verify2(a, b);\n  return this.imod(a.imul(b));\n};\n\nRed.prototype.mul = function mul(a, b) {\n  this._verify2(a, b);\n  return this.imod(a.mul(b));\n};\n\nRed.prototype.isqr = function isqr(a) {\n  return this.imul(a, a);\n};\n\nRed.prototype.sqr = function sqr(a) {\n  return this.mul(a, a);\n};\n\nRed.prototype.sqrt = function sqrt(a) {\n  if (a.cmpn(0) === 0)\n    return a.clone();\n\n  var mod3 = this.m.andln(3);\n  assert(mod3 % 2 === 1);\n\n  // Fast case\n  if (mod3 === 3) {\n    var pow = this.m.add(new BN(1)).ishrn(2);\n    var r = this.pow(a, pow);\n    return r;\n  }\n\n  // Tonelli-Shanks algorithm (Totally unoptimized and slow)\n  //\n  // Find Q and S, that Q * 2 ^ S = (P - 1)\n  var q = this.m.subn(1);\n  var s = 0;\n  while (q.cmpn(0) !== 0 && q.andln(1) === 0) {\n    s++;\n    q.ishrn(1);\n  }\n  assert(q.cmpn(0) !== 0);\n\n  var one = new BN(1).toRed(this);\n  var nOne = one.redNeg();\n\n  // Find quadratic non-residue\n  // NOTE: Max is such because of generalized Riemann hypothesis.\n  var lpow = this.m.subn(1).ishrn(1);\n  var z = this.m.bitLength();\n  z = new BN(2 * z * z).toRed(this);\n  while (this.pow(z, lpow).cmp(nOne) !== 0)\n    z.redIAdd(nOne);\n\n  var c = this.pow(z, q);\n  var r = this.pow(a, q.addn(1).ishrn(1));\n  var t = this.pow(a, q);\n  var m = s;\n  while (t.cmp(one) !== 0) {\n    var tmp = t;\n    for (var i = 0; tmp.cmp(one) !== 0; i++)\n      tmp = tmp.redSqr();\n    assert(i < m);\n    var b = this.pow(c, new BN(1).ishln(m - i - 1));\n\n    r = r.redMul(b);\n    c = b.redSqr();\n    t = t.redMul(c);\n    m = i;\n  }\n\n  return r;\n};\n\nRed.prototype.invm = function invm(a) {\n  var inv = a._invmp(this.m);\n  if (inv.sign) {\n    inv.sign = false;\n    return this.imod(inv).redNeg();\n  } else {\n    return this.imod(inv);\n  }\n};\n\nRed.prototype.pow = function pow(a, num) {\n  var w = [];\n\n  if (num.cmpn(0) === 0)\n    return new BN(1);\n\n  var q = num.clone();\n\n  while (q.cmpn(0) !== 0) {\n    w.push(q.andln(1));\n    q.ishrn(1);\n  }\n\n  // Skip leading zeroes\n  var res = a;\n  for (var i = 0; i < w.length; i++, res = this.sqr(res))\n    if (w[i] !== 0)\n      break;\n\n  if (++i < w.length) {\n    for (var q = this.sqr(res); i < w.length; i++, q = this.sqr(q)) {\n      if (w[i] === 0)\n        continue;\n      res = this.mul(res, q);\n    }\n  }\n\n  return res;\n};\n\nRed.prototype.convertTo = function convertTo(num) {\n  var r = num.mod(this.m);\n  if (r === num)\n    return r.clone();\n  else\n    return r;\n};\n\nRed.prototype.convertFrom = function convertFrom(num) {\n  var res = num.clone();\n  res.red = null;\n  return res;\n};\n\n//\n// Montgomery method engine\n//\n\nBN.mont = function mont(num) {\n  return new Mont(num);\n};\n\nfunction Mont(m) {\n  Red.call(this, m);\n\n  this.shift = this.m.bitLength();\n  if (this.shift % 26 !== 0)\n    this.shift += 26 - (this.shift % 26);\n  this.r = new BN(1).ishln(this.shift);\n  this.r2 = this.imod(this.r.sqr());\n  this.rinv = this.r._invmp(this.m);\n\n  this.minv = this.rinv.mul(this.r).isubn(1).div(this.m);\n  this.minv.sign = true;\n  this.minv = this.minv.mod(this.r);\n}\ninherits(Mont, Red);\n\nMont.prototype.convertTo = function convertTo(num) {\n  return this.imod(num.shln(this.shift));\n};\n\nMont.prototype.convertFrom = function convertFrom(num) {\n  var r = this.imod(num.mul(this.rinv));\n  r.red = null;\n  return r;\n};\n\nMont.prototype.imul = function imul(a, b) {\n  if (a.cmpn(0) === 0 || b.cmpn(0) === 0) {\n    a.words[0] = 0;\n    a.length = 1;\n    return a;\n  }\n\n  var t = a.imul(b);\n  var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m);\n  var u = t.isub(c).ishrn(this.shift);\n  var res = u;\n  if (u.cmp(this.m) >= 0)\n    res = u.isub(this.m);\n  else if (u.cmpn(0) < 0)\n    res = u.iadd(this.m);\n\n  return res._forceRed(this);\n};\n\nMont.prototype.mul = function mul(a, b) {\n  if (a.cmpn(0) === 0 || b.cmpn(0) === 0)\n    return new BN(0)._forceRed(this);\n\n  var t = a.mul(b);\n  var c = t.maskn(this.shift).mul(this.minv).imaskn(this.shift).mul(this.m);\n  var u = t.isub(c).ishrn(this.shift);\n  var res = u;\n  if (u.cmp(this.m) >= 0)\n    res = u.isub(this.m);\n  else if (u.cmpn(0) < 0)\n    res = u.iadd(this.m);\n\n  return res._forceRed(this);\n};\n\nMont.prototype.invm = function invm(a) {\n  // (AR)^-1 * R^2 = (A^-1 * R^-1) * R^2 = A^-1 * R\n  var res = this.imod(a._invmp(this.m).mul(this.r2));\n  return res._forceRed(this);\n};\n\n})(typeof module === 'undefined' || module, this);\n","var constants = require('../constants');\n\nexports.tagClass = {\n  0: 'universal',\n  1: 'application',\n  2: 'context',\n  3: 'private'\n};\nexports.tagClassByName = constants._reverse(exports.tagClass);\n\nexports.tag = {\n  0x00: 'end',\n  0x01: 'bool',\n  0x02: 'int',\n  0x03: 'bitstr',\n  0x04: 'octstr',\n  0x05: 'null_',\n  0x06: 'objid',\n  0x07: 'objDesc',\n  0x08: 'external',\n  0x09: 'real',\n  0x0a: 'enum',\n  0x0b: 'embed',\n  0x0c: 'utf8str',\n  0x0d: 'relativeOid',\n  0x10: 'seq',\n  0x11: 'set',\n  0x12: 'numstr',\n  0x13: 'printstr',\n  0x14: 't61str',\n  0x15: 'videostr',\n  0x16: 'ia5str',\n  0x17: 'utctime',\n  0x18: 'gentime',\n  0x19: 'graphstr',\n  0x1a: 'iso646str',\n  0x1b: 'genstr',\n  0x1c: 'unistr',\n  0x1d: 'charstr',\n  0x1e: 'bmpstr'\n};\nexports.tagByName = constants._reverse(exports.tag);\n","var constants = exports;\n\n// Helper\nconstants._reverse = function reverse(map) {\n  var res = {};\n\n  Object.keys(map).forEach(function(key) {\n    // Convert key to integer if it is stringified\n    if ((key | 0) == key)\n      key = key | 0;\n\n    var value = map[key];\n    res[value] = key;\n  });\n\n  return res;\n};\n\nconstants.der = require('./der');\n","var inherits = require('util').inherits;\n\nvar asn1 = require('../asn1');\nvar base = asn1.base;\nvar bignum = asn1.bignum;\n\n// Import DER constants\nvar der = asn1.constants.der;\n\nfunction DERDecoder(entity) {\n  this.enc = 'der';\n  this.name = entity.name;\n  this.entity = entity;\n\n  // Construct base tree\n  this.tree = new DERNode();\n  this.tree._init(entity.body);\n};\nmodule.exports = DERDecoder;\n\nDERDecoder.prototype.decode = function decode(data, options) {\n  if (!(data instanceof base.DecoderBuffer))\n    data = new base.DecoderBuffer(data, options);\n\n  return this.tree._decode(data, options);\n};\n\n// Tree methods\n\nfunction DERNode(parent) {\n  base.Node.call(this, 'der', parent);\n}\ninherits(DERNode, base.Node);\n\nDERNode.prototype._peekTag = function peekTag(buffer, tag, any) {\n  if (buffer.isEmpty())\n    return false;\n\n  var state = buffer.save();\n  var decodedTag = derDecodeTag(buffer, 'Failed to peek tag: \"' + tag + '\"');\n  if (buffer.isError(decodedTag))\n    return decodedTag;\n\n  buffer.restore(state);\n\n  return decodedTag.tag === tag || decodedTag.tagStr === tag || any;\n};\n\nDERNode.prototype._decodeTag = function decodeTag(buffer, tag, any) {\n  var decodedTag = derDecodeTag(buffer,\n                                'Failed to decode tag of \"' + tag + '\"');\n  if (buffer.isError(decodedTag))\n    return decodedTag;\n\n  var len = derDecodeLen(buffer,\n                         decodedTag.primitive,\n                         'Failed to get length of \"' + tag + '\"');\n\n  // Failure\n  if (buffer.isError(len))\n    return len;\n\n  if (!any &&\n      decodedTag.tag !== tag &&\n      decodedTag.tagStr !== tag &&\n      decodedTag.tagStr + 'of' !== tag) {\n    return buffer.error('Failed to match tag: \"' + tag + '\"');\n  }\n\n  if (decodedTag.primitive || len !== null)\n    return buffer.skip(len, 'Failed to match body of: \"' + tag + '\"');\n\n  // Indefinite length... find END tag\n  var state = buffer.save();\n  var res = this._skipUntilEnd(\n      buffer,\n      'Failed to skip indefinite length body: \"' + this.tag + '\"');\n  if (buffer.isError(res))\n    return res;\n\n  len = buffer.offset - state.offset;\n  buffer.restore(state);\n  return buffer.skip(len, 'Failed to match body of: \"' + tag + '\"');\n};\n\nDERNode.prototype._skipUntilEnd = function skipUntilEnd(buffer, fail) {\n  while (true) {\n    var tag = derDecodeTag(buffer, fail);\n    if (buffer.isError(tag))\n      return tag;\n    var len = derDecodeLen(buffer, tag.primitive, fail);\n    if (buffer.isError(len))\n      return len;\n\n    var res;\n    if (tag.primitive || len !== null)\n      res = buffer.skip(len)\n    else\n      res = this._skipUntilEnd(buffer, fail);\n\n    // Failure\n    if (buffer.isError(res))\n      return res;\n\n    if (tag.tagStr === 'end')\n      break;\n  }\n};\n\nDERNode.prototype._decodeList = function decodeList(buffer, tag, decoder) {\n  var result = [];\n  while (!buffer.isEmpty()) {\n    var possibleEnd = this._peekTag(buffer, 'end');\n    if (buffer.isError(possibleEnd))\n      return possibleEnd;\n\n    var res = decoder.decode(buffer, 'der');\n    if (buffer.isError(res) && possibleEnd)\n      break;\n    result.push(res);\n  }\n  return result;\n};\n\nDERNode.prototype._decodeStr = function decodeStr(buffer, tag) {\n  if (tag === 'octstr') {\n    return buffer.raw();\n  } else if (tag === 'bitstr') {\n    var unused = buffer.readUInt8();\n    if (buffer.isError(unused))\n      return unused;\n\n    return { unused: unused, data: buffer.raw() };\n  } else if (tag === 'ia5str' || tag === 'utf8str') {\n    return buffer.raw().toString();\n  } else {\n    return this.error('Decoding of string type: ' + tag + ' unsupported');\n  }\n};\n\nDERNode.prototype._decodeObjid = function decodeObjid(buffer, values, relative) {\n  var identifiers = [];\n  var ident = 0;\n  while (!buffer.isEmpty()) {\n    var subident = buffer.readUInt8();\n    ident <<= 7;\n    ident |= subident & 0x7f;\n    if ((subident & 0x80) === 0) {\n      identifiers.push(ident);\n      ident = 0;\n    }\n  }\n  if (subident & 0x80)\n    identifiers.push(ident);\n\n  var first = (identifiers[0] / 40) | 0;\n  var second = identifiers[0] % 40;\n\n  if (relative)\n    result = identifiers;\n  else\n    result = [first, second].concat(identifiers.slice(1));\n\n  if (values)\n    result = values[result.join(' ')];\n\n  return result;\n};\n\nDERNode.prototype._decodeTime = function decodeTime(buffer, tag) {\n  var str = buffer.raw().toString();\n  if (tag === 'gentime') {\n    var year = str.slice(0, 4) | 0;\n    var mon = str.slice(4, 6) | 0;\n    var day = str.slice(6, 8) | 0;\n    var hour = str.slice(8, 10) | 0;\n    var min = str.slice(10, 12) | 0;\n    var sec = str.slice(12, 14) | 0;\n  } else if (tag === 'utctime') {\n    var year = str.slice(0, 2) | 0;\n    var mon = str.slice(2, 4) | 0;\n    var day = str.slice(4, 6) | 0;\n    var hour = str.slice(6, 8) | 0;\n    var min = str.slice(8, 10) | 0;\n    var sec = str.slice(10, 12) | 0;\n    if (year < 70)\n      year = 2000 + year;\n    else\n      year = 1900 + year;\n  } else {\n    return this.error('Decoding ' + tag + ' time is not supported yet');\n  }\n\n  return Date.UTC(year, mon - 1, day, hour, min, sec, 0);\n};\n\nDERNode.prototype._decodeNull = function decodeNull(buffer) {\n  return null;\n};\n\nDERNode.prototype._decodeBool = function decodeBool(buffer) {\n  var res = buffer.readUInt8();\n  if (buffer.isError(res))\n    return res;\n  else\n    return res !== 0;\n};\n\nDERNode.prototype._decodeInt = function decodeInt(buffer, values) {\n  // Bigint, return as it is (assume big endian)\n  var raw = buffer.raw();\n  var res = new bignum(raw);\n\n  if (values)\n    res = values[res.toString(10)] || res;\n\n  return res;\n};\n\nDERNode.prototype._use = function use(entity, obj) {\n  if (typeof entity === 'function')\n    entity = entity(obj);\n  return entity._getDecoder('der').tree;\n};\n\n// Utility methods\n\nfunction derDecodeTag(buf, fail) {\n  var tag = buf.readUInt8(fail);\n  if (buf.isError(tag))\n    return tag;\n\n  var cls = der.tagClass[tag >> 6];\n  var primitive = (tag & 0x20) === 0;\n\n  // Multi-octet tag - load\n  if ((tag & 0x1f) === 0x1f) {\n    var oct = tag;\n    tag = 0;\n    while ((oct & 0x80) === 0x80) {\n      oct = buf.readUInt8(fail);\n      if (buf.isError(oct))\n        return oct;\n\n      tag <<= 7;\n      tag |= oct & 0x7f;\n    }\n  } else {\n    tag &= 0x1f;\n  }\n  var tagStr = der.tag[tag];\n\n  return {\n    cls: cls,\n    primitive: primitive,\n    tag: tag,\n    tagStr: tagStr\n  };\n}\n\nfunction derDecodeLen(buf, primitive, fail) {\n  var len = buf.readUInt8(fail);\n  if (buf.isError(len))\n    return len;\n\n  // Indefinite form\n  if (!primitive && len === 0x80)\n    return null;\n\n  // Definite form\n  if ((len & 0x80) === 0) {\n    // Short form\n    return len;\n  }\n\n  // Long form\n  var num = len & 0x7f;\n  if (num >= 4)\n    return buf.error('length octect is too long');\n\n  len = 0;\n  for (var i = 0; i < num; i++) {\n    len <<= 8;\n    var j = buf.readUInt8(fail);\n    if (buf.isError(j))\n      return j;\n    len |= j;\n  }\n\n  return len;\n}\n","var decoders = exports;\n\ndecoders.der = require('./der');\ndecoders.pem = require('./pem');\n","var inherits = require('util').inherits;\nvar Buffer = require('buffer').Buffer;\n\nvar asn1 = require('../asn1');\nvar DERDecoder = require('./der');\n\nfunction PEMDecoder(entity) {\n  DERDecoder.call(this, entity);\n  this.enc = 'pem';\n};\ninherits(PEMDecoder, DERDecoder);\nmodule.exports = PEMDecoder;\n\nPEMDecoder.prototype.decode = function decode(data, options) {\n  var lines = data.toString().split(/[\\r\\n]+/g);\n\n  var label = options.label.toUpperCase();\n\n  var re = /^-----(BEGIN|END) ([^-]+)-----$/;\n  var start = -1;\n  var end = -1;\n  for (var i = 0; i < lines.length; i++) {\n    var match = lines[i].match(re);\n    if (match === null)\n      continue;\n\n    if (match[2] !== label)\n      continue;\n\n    if (start === -1) {\n      if (match[1] !== 'BEGIN')\n        break;\n      start = i;\n    } else {\n      if (match[1] !== 'END')\n        break;\n      end = i;\n      break;\n    }\n  }\n  if (start === -1 || end === -1)\n    throw new Error('PEM section not found for: ' + label);\n\n  var base64 = lines.slice(start + 1, end).join('');\n  // Remove excessive symbols\n  base64.replace(/[^a-z0-9\\+\\/=]+/gi, '');\n\n  var input = new Buffer(base64, 'base64');\n  return DERDecoder.prototype.decode.call(this, input, options);\n};\n","var inherits = require('util').inherits;\nvar Buffer = require('buffer').Buffer;\n\nvar asn1 = require('../asn1');\nvar base = asn1.base;\nvar bignum = asn1.bignum;\n\n// Import DER constants\nvar der = asn1.constants.der;\n\nfunction DEREncoder(entity) {\n  this.enc = 'der';\n  this.name = entity.name;\n  this.entity = entity;\n\n  // Construct base tree\n  this.tree = new DERNode();\n  this.tree._init(entity.body);\n};\nmodule.exports = DEREncoder;\n\nDEREncoder.prototype.encode = function encode(data, reporter) {\n  return this.tree._encode(data, reporter).join();\n};\n\n// Tree methods\n\nfunction DERNode(parent) {\n  base.Node.call(this, 'der', parent);\n}\ninherits(DERNode, base.Node);\n\nDERNode.prototype._encodeComposite = function encodeComposite(tag,\n                                                              primitive,\n                                                              cls,\n                                                              content) {\n  var encodedTag = encodeTag(tag, primitive, cls, this.reporter);\n\n  // Short form\n  if (content.length < 0x80) {\n    var header = new Buffer(2);\n    header[0] = encodedTag;\n    header[1] = content.length;\n    return this._createEncoderBuffer([ header, content ]);\n  }\n\n  // Long form\n  // Count octets required to store length\n  var lenOctets = 1;\n  for (var i = content.length; i >= 0x100; i >>= 8)\n    lenOctets++;\n\n  var header = new Buffer(1 + 1 + lenOctets);\n  header[0] = encodedTag;\n  header[1] = 0x80 | lenOctets;\n\n  for (var i = 1 + lenOctets, j = content.length; j > 0; i--, j >>= 8)\n    header[i] = j & 0xff;\n\n  return this._createEncoderBuffer([ header, content ]);\n};\n\nDERNode.prototype._encodeStr = function encodeStr(str, tag) {\n  if (tag === 'octstr')\n    return this._createEncoderBuffer(str);\n  else if (tag === 'bitstr')\n    return this._createEncoderBuffer([ str.unused | 0, str.data ]);\n  else if (tag === 'ia5str' || tag === 'utf8str')\n    return this._createEncoderBuffer(str);\n  return this.reporter.error('Encoding of string type: ' + tag +\n                             ' unsupported');\n};\n\nDERNode.prototype._encodeObjid = function encodeObjid(id, values, relative) {\n  if (typeof id === 'string') {\n    if (!values)\n      return this.reporter.error('string objid given, but no values map found');\n    if (!values.hasOwnProperty(id))\n      return this.reporter.error('objid not found in values map');\n    id = values[id].split(/[\\s\\.]+/g);\n    for (var i = 0; i < id.length; i++)\n      id[i] |= 0;\n  } else if (Array.isArray(id)) {\n    id = id.slice();\n    for (var i = 0; i < id.length; i++)\n      id[i] |= 0;\n  }\n\n  if (!Array.isArray(id)) {\n    return this.reporter.error('objid() should be either array or string, ' +\n                               'got: ' + JSON.stringify(id));\n  }\n\n  if (!relative) {\n    if (id[1] >= 40)\n      return this.reporter.error('Second objid identifier OOB');\n    id.splice(0, 2, id[0] * 40 + id[1]);\n  }\n\n  // Count number of octets\n  var size = 0;\n  for (var i = 0; i < id.length; i++) {\n    var ident = id[i];\n    for (size++; ident >= 0x80; ident >>= 7)\n      size++;\n  }\n\n  var objid = new Buffer(size);\n  var offset = objid.length - 1;\n  for (var i = id.length - 1; i >= 0; i--) {\n    var ident = id[i];\n    objid[offset--] = ident & 0x7f;\n    while ((ident >>= 7) > 0)\n      objid[offset--] = 0x80 | (ident & 0x7f);\n  }\n\n  return this._createEncoderBuffer(objid);\n};\n\nfunction two(num) {\n  if (num < 10)\n    return '0' + num;\n  else\n    return num;\n}\n\nDERNode.prototype._encodeTime = function encodeTime(time, tag) {\n  var str;\n  var date = new Date(time);\n\n  if (tag === 'gentime') {\n    str = [\n      two(date.getFullYear()),\n      two(date.getUTCMonth() + 1),\n      two(date.getUTCDate()),\n      two(date.getUTCHours()),\n      two(date.getUTCMinutes()),\n      two(date.getUTCSeconds()),\n      'Z'\n    ].join('');\n  } else if (tag === 'utctime') {\n    str = [\n      two(date.getFullYear() % 100),\n      two(date.getUTCMonth() + 1),\n      two(date.getUTCDate()),\n      two(date.getUTCHours()),\n      two(date.getUTCMinutes()),\n      two(date.getUTCSeconds()),\n      'Z'\n    ].join('');\n  } else {\n    this.reporter.error('Encoding ' + tag + ' time is not supported yet');\n  }\n\n  return this._encodeStr(str, 'octstr');\n};\n\nDERNode.prototype._encodeNull = function encodeNull() {\n  return this._createEncoderBuffer('');\n};\n\nDERNode.prototype._encodeInt = function encodeInt(num, values) {\n  if (typeof num === 'string') {\n    if (!values)\n      return this.reporter.error('String int or enum given, but no values map');\n    if (!values.hasOwnProperty(num)) {\n      return this.reporter.error('Values map doesn\\'t contain: ' +\n                                 JSON.stringify(num));\n    }\n    num = values[num];\n  }\n\n  // Bignum, assume big endian\n  if (typeof num !== 'number' && !Buffer.isBuffer(num)) {\n    var numArray = num.toArray();\n    if (num.sign === false && numArray[0] & 0x80) {\n      numArray.unshift(0);\n    }\n    num = new Buffer(numArray);\n  }\n\n  if (Buffer.isBuffer(num)) {\n    var size = num.length;\n    if (num.length === 0)\n      size++;\n\n    var out = new Buffer(size);\n    num.copy(out);\n    if (num.length === 0)\n      out[0] = 0\n    return this._createEncoderBuffer(out);\n  }\n\n  if (num < 0x80)\n    return this._createEncoderBuffer(num);\n\n  if (num < 0x100)\n    return this._createEncoderBuffer([0, num]);\n\n  var size = 1;\n  for (var i = num; i >= 0x100; i >>= 8)\n    size++;\n\n  var out = new Array(size);\n  for (var i = out.length - 1; i >= 0; i--) {\n    out[i] = num & 0xff;\n    num >>= 8;\n  }\n  if(out[0] & 0x80) {\n    out.unshift(0);\n  }\n\n  return this._createEncoderBuffer(new Buffer(out));\n};\n\nDERNode.prototype._encodeBool = function encodeBool(value) {\n  return this._createEncoderBuffer(value ? 0xff : 0);\n};\n\nDERNode.prototype._use = function use(entity, obj) {\n  if (typeof entity === 'function')\n    entity = entity(obj);\n  return entity._getEncoder('der').tree;\n};\n\nDERNode.prototype._skipDefault = function skipDefault(dataBuffer, reporter, parent) {\n  var state = this._baseState;\n  var i;\n  if (state['default'] === null)\n    return false;\n\n  var data = dataBuffer.join();\n  if (state.defaultBuffer === undefined)\n    state.defaultBuffer = this._encodeValue(state['default'], reporter, parent).join();\n\n  if (data.length !== state.defaultBuffer.length)\n    return false;\n\n  for (i=0; i < data.length; i++)\n    if (data[i] !== state.defaultBuffer[i])\n      return false;\n\n  return true;\n};\n\n// Utility methods\n\nfunction encodeTag(tag, primitive, cls, reporter) {\n  var res;\n\n  if (tag === 'seqof')\n    tag = 'seq';\n  else if (tag === 'setof')\n    tag = 'set';\n\n  if (der.tagByName.hasOwnProperty(tag))\n    res = der.tagByName[tag];\n  else if (typeof tag === 'number' && (tag | 0) === tag)\n    res = tag;\n  else\n    return reporter.error('Unknown tag: ' + tag);\n\n  if (res >= 0x1f)\n    return reporter.error('Multi-octet tag encoding unsupported');\n\n  if (!primitive)\n    res |= 0x20;\n\n  res |= (der.tagClassByName[cls || 'universal'] << 6);\n\n  return res;\n}\n","var encoders = exports;\n\nencoders.der = require('./der');\nencoders.pem = require('./pem');\n","var inherits = require('util').inherits;\nvar Buffer = require('buffer').Buffer;\n\nvar asn1 = require('../asn1');\nvar DEREncoder = require('./der');\n\nfunction PEMEncoder(entity) {\n  DEREncoder.call(this, entity);\n  this.enc = 'pem';\n};\ninherits(PEMEncoder, DEREncoder);\nmodule.exports = PEMEncoder;\n\nPEMEncoder.prototype.encode = function encode(data, options) {\n  var buf = DEREncoder.prototype.encode.call(this, data);\n\n  var p = buf.toString('base64');\n  var out = [ '-----BEGIN ' + options.label + '-----' ];\n  for (var i = 0; i < p.length; i += 64)\n    out.push(p.slice(i, i + 64));\n  out.push('-----END ' + options.label + '-----');\n  return out.join('\\n');\n};\n","'use strict'\n\nvar asn1 = require('./asn1/asn1');\nvar BN = require('./asn1/bignum/bn');\n\nvar ECPrivateKeyASN = asn1.define('ECPrivateKey', function() {\n    this.seq().obj(\n        this.key('version').int(),\n        this.key('privateKey').octstr(),\n        this.key('parameters').explicit(0).objid().optional(),\n        this.key('publicKey').explicit(1).bitstr().optional()\n    )\n})\n\nvar SubjectPublicKeyInfoASN = asn1.define('SubjectPublicKeyInfo', function() {\n    this.seq().obj(\n        this.key('algorithm').seq().obj(\n            this.key(\"id\").objid(),\n            this.key(\"curve\").objid()\n        ),\n        this.key('pub').bitstr()\n    )\n})\n\nvar curves = {\n    secp256k1: {\n        curveParameters: [1, 3, 132, 0, 10],\n        privatePEMOptions: {label: 'EC PRIVATE KEY'},\n        publicPEMOptions: {label: 'PUBLIC KEY'}\n    }\n}\n\nfunction assert(val, msg) {\n    if (!val) {\n        throw new Error(msg || 'Assertion failed')\n    }\n}\n\nfunction KeyEncoder(options) {\n    if (typeof options === 'string') {\n        assert(curves.hasOwnProperty(options), 'Unknown curve ' + options);\n        options = curves[options]\n    }\n    this.options = options;\n    this.algorithmID = [1, 2, 840, 10045, 2, 1]\n}\n\nKeyEncoder.ECPrivateKeyASN = ECPrivateKeyASN;\nKeyEncoder.SubjectPublicKeyInfoASN = SubjectPublicKeyInfoASN;\n\nKeyEncoder.prototype.privateKeyObject = function(rawPrivateKey, rawPublicKey) {\n    var privateKeyObject = {\n        version: new BN(1),\n        privateKey: new Buffer(rawPrivateKey, 'hex'),\n        parameters: this.options.curveParameters,\n        pemOptions: {label:\"EC PRIVATE KEY\"}\n    };\n\n    if (rawPublicKey) {\n        privateKeyObject.publicKey = {\n            unused: 0,\n            data: new Buffer(rawPublicKey, 'hex')\n        }\n    }\n\n    return privateKeyObject\n};\n\nKeyEncoder.prototype.publicKeyObject = function(rawPublicKey) {\n    return {\n        algorithm: {\n            id: this.algorithmID,\n            curve: this.options.curveParameters\n        },\n        pub: {\n            unused: 0,\n            data: new Buffer(rawPublicKey, 'hex')\n        },\n        pemOptions: { label :\"PUBLIC KEY\"}\n    }\n}\n\nKeyEncoder.prototype.encodePrivate = function(privateKey, originalFormat, destinationFormat) {\n    var privateKeyObject\n\n    /* Parse the incoming private key and convert it to a private key object */\n    if (originalFormat === 'raw') {\n        if (!typeof privateKey === 'string') {\n            throw 'private key must be a string'\n        }\n        var privateKeyObject = this.options.curve.keyFromPrivate(privateKey, 'hex'),\n            rawPublicKey = privateKeyObject.getPublic('hex')\n        privateKeyObject = this.privateKeyObject(privateKey, rawPublicKey)\n    } else if (originalFormat === 'der') {\n        if (typeof privateKey === 'buffer') {\n            // do nothing\n        } else if (typeof privateKey === 'string') {\n            privateKey = new Buffer(privateKey, 'hex')\n        } else {\n            throw 'private key must be a buffer or a string'\n        }\n        privateKeyObject = ECPrivateKeyASN.decode(privateKey, 'der')\n    } else if (originalFormat === 'pem') {\n        if (!typeof privateKey === 'string') {\n            throw 'private key must be a string'\n        }\n        privateKeyObject = ECPrivateKeyASN.decode(privateKey, 'pem', this.options.privatePEMOptions)\n    } else {\n        throw 'invalid private key format'\n    }\n\n    /* Export the private key object to the desired format */\n    if (destinationFormat === 'raw') {\n        return privateKeyObject.privateKey.toString('hex')\n    } else if (destinationFormat === 'der') {\n        return ECPrivateKeyASN.encode(privateKeyObject, 'der').toString('hex')\n    } else if (destinationFormat === 'pem') {\n        return ECPrivateKeyASN.encode(privateKeyObject, 'pem', this.options.privatePEMOptions)\n    } else {\n        throw 'invalid destination format for private key'\n    }\n}\n\nKeyEncoder.prototype.encodePublic = function(publicKey, originalFormat, destinationFormat) {\n    var publicKeyObject\n\n    /* Parse the incoming public key and convert it to a public key object */\n    if (originalFormat === 'raw') {\n        if (!typeof publicKey === 'string') {\n            throw 'public key must be a string'\n        }\n        publicKeyObject = this.publicKeyObject(publicKey)\n    } else if (originalFormat === 'der') {\n        if (typeof publicKey === 'buffer') {\n            // do nothing\n        } else if (typeof publicKey === 'string') {\n            publicKey = new Buffer(publicKey, 'hex')\n        } else {\n            throw 'public key must be a buffer or a string'\n        }\n        publicKeyObject = SubjectPublicKeyInfoASN.decode(publicKey, 'der')\n    } else if (originalFormat === 'pem') {\n        if (!typeof publicKey === 'string') {\n            throw 'public key must be a string'\n        }\n        publicKeyObject = SubjectPublicKeyInfoASN.decode(publicKey, 'pem', this.options.publicPEMOptions)\n    } else {\n        throw 'invalid public key format'\n    }\n\n    /* Export the private key object to the desired format */\n    if (destinationFormat === 'raw') {\n        return publicKeyObject.pub.data.toString('hex')\n    } else if (destinationFormat === 'der') {\n        return SubjectPublicKeyInfoASN.encode(publicKeyObject, 'der').toString('hex')\n    } else if (destinationFormat === 'pem') {\n        return SubjectPublicKeyInfoASN.encode(publicKeyObject, 'pem', this.options.publicPEMOptions)\n    } else {\n        throw 'invalid destination format for public key'\n    }\n}\n\nmodule.exports = KeyEncoder;","const path = require(\"path\");\nconst yazl = require(\"yazl\");\nconst yauzl = require(\"yauzl\");\nconst fs = require(\"fs\");\nconst DuplexStream = require(\"./utils/DuplexStream\");\nconst PassThroughStream = require(\"./utils/PassThroughStream\");\nconst isStream = require(\"./utils/isStream\");\n\nconst EventEmitter = require('events');\n\nconst countFiles = require('./utils/countFiles');\n\nfunction PskArchiver() {\n\n    const self = this;\n\n    const event = new EventEmitter();\n\n    this.on = event.on;\n    this.off = event.off;\n    this.emit = event.emit;\n\n    this.zipStream = function (inputPath, output, callback) {\n        let ext = \"\";\n        const zipFile = new yazl.ZipFile();\n        const ptStream = new PassThroughStream();\n\n        countFiles.computeSize(inputPath, (err, totalSize) => {\n            if (err) {\n                return callback(err);\n            }\n\n            __addToArchiveRecursively(zipFile, inputPath, \"\", (err) => {\n                if (err) {\n                    return callback(err);\n                }\n\n                zipFile.end();\n                const filename = path.basename(inputPath);\n                const splitFilename = filename.split(\".\");\n                if (splitFilename.length >= 2) {\n                    ext = \".\" + splitFilename[splitFilename.length - 1];\n                }\n                const myStream = zipFile.outputStream.pipe(ptStream);\n\n                let progressLength = 0;\n                let totalLength = 0;\n\n                /**\n                 * TODO review this\n                 * In browser, piping will block the event loop and the stack queue is not called.\n                 */\n                myStream.on(\"data\", (chunk) => {\n                    progressLength += chunk.length;\n                    totalLength += chunk.length;\n\n                    if (progressLength > 300000) {\n                        myStream.pause();\n                        progressLength = 0;\n                        setTimeout(function () {\n                            myStream.resume();\n                        }, 10);\n                        emitProgress(totalSize, totalLength)\n                    }\n                });\n\n                myStream.on('end', () => {\n                    emitProgress(totalSize, totalSize);\n                    emitTotalSize(totalSize);\n                });\n                if (isStream.isWritable(output)) {\n                    callback(null, myStream.pipe(output));\n                } else if (typeof output === \"string\") {\n                   fs.mkdir(output, {recursive: true}, () => {\n                        const destinationPath = path.join(output, path.basename(inputPath, ext) + \".zip\");\n                        myStream.pipe(fs.createWriteStream(destinationPath));\n                    });\n                }\n            });\n\n            function __addToArchiveRecursively(zipFile, inputPath, root = '', callback) {\n                root = root || '';\n                fs.stat(inputPath, (err, stats) => {\n                    if (err) {\n                        return callback(err);\n                    }\n                    if (stats.isFile()) {\n                        zipFile.addFile(inputPath, path.join(root, path.basename(inputPath)));\n                        callback(null);\n\n                    } else {\n                        fs.readdir(inputPath, (err, files) => {\n                            if (err) {\n                                return callback(err);\n                            }\n                            const f_length = files.length;\n                            let f_add_index = 0;\n\n                            const checkStatus = () => {\n                                if (f_length === f_add_index) {\n                                    callback(null);\n                                    return true;\n                                }\n                                return false;\n                            };\n\n                            if (!checkStatus()) {\n                                files.forEach(file => {\n                                    const tempPath = path.join(inputPath, file);\n                                    __addToArchiveRecursively(zipFile, tempPath, path.join(root, path.basename(inputPath)), (err) => {\n                                        if (err) {\n                                            return callback(err);\n                                        }\n                                        f_add_index++;\n                                        checkStatus();\n                                    })\n                                });\n                            }\n                        })\n                    }\n                });\n            }\n\n        });\n\n    };\n\n    this.unzipStream = function (input, outputPath, callback) {\n\n        let size = 0;\n\n        fs.stat(input, (err, stats) => {\n            if (err) {\n                return callback(err);\n            }\n\n            let totalSize = stats.size;\n\n\n            yauzl.open(input, {lazyEntries: true}, (err, zipFile) => {\n                if (err) {\n                    return callback(err);\n                }\n\n                let progressLength = 0;\n                let totalLength = 0;\n\n                const fileNames = [];\n                zipFile.readEntry();\n                zipFile.once(\"end\", () => {\n                    emitProgress(totalSize, totalSize);\n                    callback(null, fileNames);\n                });\n                zipFile.on(\"entry\", (entry) => {\n                    if (entry.fileName.endsWith(path.sep)) {\n                        zipFile.readEntry();\n                    } else {\n                        let folder = path.dirname(entry.fileName);\n                        fs.mkdir(path.join(outputPath, folder), {recursive: true}, () => {\n                            zipFile.openReadStream(entry, (err, readStream) => {\n                                if (err) {\n                                    return callback(err);\n                                }\n\n                                /**\n                                 * TODO review this\n                                 * In browser, piping will block the event loop and the stack queue is not called.\n                                 */\n\n                                readStream.on(\"data\", (chunk) => {\n                                    progressLength += chunk.length;\n                                    totalLength += chunk.length;\n\n                                    if (progressLength > 300000) {\n                                        readStream.pause();\n                                        progressLength = 0;\n                                        setTimeout(function () {\n                                            readStream.resume();\n                                        }, 30);\n                                        emitProgress(totalSize, totalLength)\n                                    }\n                                });\n\n\n                                readStream.on(\"end\", () => {\n                                    zipFile.readEntry();\n                                });\n                                const ptStream = new PassThroughStream();\n                                let fileName = path.join(outputPath, entry.fileName);\n                                let folder = path.dirname(fileName);\n                                const tempStream = readStream.pipe(ptStream);\n\n                                fs.mkdir(folder, {recursive: true}, (err) => {\n                                    if (err) {\n                                        return callback(err);\n                                    }\n\n                                    size += ptStream.getSize();\n                                    let output = fs.createWriteStream(fileName);\n                                    fileNames.push(fileName);\n                                    tempStream.pipe(output);\n                                });\n                            });\n                        });\n                    }\n                });\n            });\n\n        });\n\n    };\n\n    this.zipInMemory = function (inputObj, depth, callback) {\n        const zipFile = new yazl.ZipFile();\n        const ds = new DuplexStream();\n        zipRecursively(zipFile, inputObj, \"\", depth, (err) => {\n            if (err) {\n                return callback(err);\n            }\n            zipFile.end();\n            let buffer = Buffer.alloc(0);\n            ds.on('data', (chunk) => {\n                buffer = Buffer.concat([buffer, chunk]);\n            });\n\n            zipFile.outputStream.pipe(ds).on(\"finish\", (err) => {\n                if (err) {\n                    return callback(err);\n                }\n                callback(null, buffer);\n            });\n        })\n    };\n\n    this.unzipInMemory = function (inputZip, callback) {\n\n        function unzipInput(zipFile) {\n            zipFile.readEntry();\n            const obj = {};\n            zipFile.once(\"end\", () => {\n                callback(null, obj);\n            });\n\n            zipFile.on(\"entry\", (entry) => {\n                zipFile.openReadStream(entry, (err, readStream) => {\n                    const ds = new DuplexStream();\n                    let str = '';\n                    if (err) {\n                        return callback(err);\n                    }\n                    readStream.on(\"end\", () => {\n                        zipFile.readEntry();\n                    });\n                    ds.on(\"data\", (chunk) => {\n                        str += chunk.toString();\n                    });\n\n                    readStream.pipe(ds).on(\"finish\", (err) => {\n                        if (err) {\n                            return callback(err);\n                        }\n                        const splitEntry = entry.fileName.split(\"/\");\n                        const type = splitEntry.pop();\n                        addPropsRecursively(obj, splitEntry, type, new Buffer(str));\n                    });\n\n                });\n            })\n        }\n\n        if (Buffer.isBuffer(inputZip)) {\n            yauzl.fromBuffer(inputZip, {lazyEntries: true}, (err, zipFile) => {\n                if (err) {\n                    return callback(err);\n                }\n                unzipInput(zipFile)\n            });\n        } else {\n            return callback(new Error(\"input should be a buffer\"));\n        }\n\n    };\n\n    function zipRecursively(zipFile, obj, root, depth, callback) {\n        if (depth === 0) {\n            zipFile.addBuffer(new Buffer(JSON.stringify(obj)), root + \"/stringify\");\n            return;\n        }\n\n        if (typeof obj === 'undefined') {\n            zipFile.addBuffer(Buffer.alloc(0), root + \"/undefined\");\n        } else if (typeof obj === 'number') {\n            zipFile.addBuffer(new Buffer(obj.toString()), root + \"/number\");\n        } else if (typeof obj === 'string') {\n            zipFile.addBuffer(new Buffer(obj), root + \"/string\")\n        } else if (obj === null) {\n            zipFile.addBuffer(Buffer.alloc(0), root + \"/null\");\n        } else if (Buffer.isBuffer(obj)) {\n            zipFile.addBuffer(obj, root + \"/buffer\");\n        } else if (isStream.isReadable(obj)) {\n            zipFile.addReadStream(obj, root + \"/stream\");\n        } else if (Array.isArray(obj)) {\n            for (let i = 0; i < obj.length; i++) {\n                if (obj.length === 0) {\n                    zipFile.addBuffer(Buffer.alloc(0), root + \"/array\")\n                } else {\n                    zipRecursively(zipFile, obj[i], root + \"/array/\" + i, depth, (err) => {\n                        if (err) {\n                            return callback(err);\n                        }\n                    });\n                }\n            }\n        } else if (obj && typeof obj === 'object') {\n            let keys = Object.keys(obj);\n            if (keys.length === 0 && obj.constructor === Object) {\n                zipFile.addBuffer(Buffer.alloc(0), root + \"/object\");\n            } else {\n                const encodedObj = {};\n                Object.entries(obj).forEach(([key, value]) => {\n                    encodedObj[encodeURIComponent(key)] = value;\n                });\n                obj = encodedObj;\n                keys = Object.keys(obj);\n                keys.forEach(key => {\n                    let entryName;\n                    if (root === \"\") {\n                        entryName = key;\n                    } else {\n                        entryName = root + \"/\" + key;\n                    }\n                    zipRecursively(zipFile, obj[key], entryName, depth - 1, (err) => {\n                        if (err) {\n                            return callback(err);\n                        }\n                    });\n                });\n            }\n        } else {\n            throw new Error('Should never reach this');\n        }\n        callback(null);\n    }\n\n    function addPropsRecursively(obj, splitName, type, data) {\n        if (splitName.length >= 1) {\n            const prop = decodeURIComponent(splitName.shift());\n\n            if (splitName.length === 0) {\n                switch (type) {\n                    case 'undefined':\n                        obj[prop] = undefined;\n                        break;\n                    case 'null':\n                        obj[prop] = null;\n                        break;\n                    case 'number':\n                        obj[prop] = parseInt(data.toString());\n                        break;\n                    case 'string':\n                        obj[prop] = data.toString();\n                        break;\n                    case 'stream':\n                        obj[prop] = bufferToStream(data);\n                        break;\n                    case 'array':\n                        obj[prop] = [];\n                        break;\n                    case 'object':\n                        obj[prop] = {};\n                        break;\n                    case 'stringify':\n                        obj[prop] = JSON.parse(data.toString());\n                        break;\n                    default:\n                        throw new Error('Should never reach this');\n                }\n            } else {\n                if (splitName[0] === 'array') {\n                    if (!obj.hasOwnProperty(prop)) {\n                        obj[prop] = [];\n                    }\n                    splitName.shift();\n                    addPropsRecursively(obj[prop], splitName, type, data);\n                } else {\n                    if (!obj.hasOwnProperty(prop)) {\n                        obj[prop] = {};\n                    }\n                    addPropsRecursively(obj[prop], splitName, type, data);\n                }\n            }\n        }\n    }\n\n\n    function bufferToStream(buffer) {\n        let stream = new require('stream').Readable();\n        stream.push(buffer);\n        stream.push(null);\n        return stream;\n    }\n\n    function emitProgress(total, processed) {\n\n\n        if (processed > total) {\n            processed = total;\n        }\n\n        const progress = (100 * processed) / total;\n        self.emit('progress', progress);\n    }\n\n    function emitTotalSize(total) {\n        self.emit('total', total);\n    }\n\n\n}\n\nmodule.exports = PskArchiver;","const stream = require('stream');\nconst util = require('util');\n\nconst Duplex = stream.Duplex;\n\nfunction DuplexStream(options) {\n\tif (!(this instanceof DuplexStream)) {\n\t\treturn new DuplexStream(options);\n\t}\n\tDuplex.call(this, options);\n}\nutil.inherits(DuplexStream, Duplex);\n\nDuplexStream.prototype._write = function (chunk, enc, cb) {\n\tthis.push(chunk);\n\tcb();\n};\n\n\nDuplexStream.prototype._read = function (n) {\n\n};\n\nmodule.exports = DuplexStream;","const stream = require('stream');\nconst util = require('util');\n\nconst PassThrough = stream.PassThrough;\n\nfunction PassThroughStream(options) {\n    if (!(this instanceof PassThroughStream)) {\n        return new PassThroughStream(options);\n    }\n    PassThrough.call(this, options);\n\n    let size = 0;\n\n    this.addToSize = function (amount) {\n        size += amount;\n    };\n\n    this.getSize = function () {\n        return size;\n    }\n}\n\nutil.inherits(PassThroughStream, PassThrough);\n\nPassThroughStream.prototype._write = function (chunk, enc, cb) {\n    this.addToSize(chunk.length);\n    this.push(chunk);\n    cb();\n};\n\n\nPassThroughStream.prototype._read = function (n) {\n\n};\n\nmodule.exports = PassThroughStream;","const fs = require('fs');\nconst path = require('path');\nconst yauzl = require('yauzl');\n\nfunction countFiles(inputPath, callback) {\n    let total = 0;\n\n    fs.stat(inputPath, (err, stats) => {\n        if (err) {\n            return callback(err);\n        }\n\n        if (stats.isFile()) {\n            return callback(undefined, 1);\n        }\n\n        fs.readdir(inputPath, (err, files) => {\n            if (err) {\n                return callback(err);\n            }\n\n\n            total = files.length;\n            let count = files.length;\n\n            if (total === 0) {\n                return callback(undefined, 0);\n            }\n\n            files.forEach(file => {\n                fs.stat(path.join(inputPath, file), (err, stats) => {\n                    if (err) {\n                        return callback(err);\n                    }\n\n                    if (stats.isDirectory()) {\n                        --total;\n                        countFiles(path.join(inputPath, file), (err, filesNumber) => {\n                            if (err) {\n                                return callback(err);\n                            }\n\n                            total += filesNumber;\n\n\n                            if (--count === 0) {\n                                callback(undefined, total);\n                            }\n                        });\n                    } else {\n                        if (!stats.isFile()) {\n                            --total;\n                        }\n\n                        if (--count === 0) {\n                            callback(undefined, total);\n                        }\n                    }\n                });\n            })\n        });\n    });\n}\n\nfunction countZipEntries(inputPath, callback) {\n    let processed = 0;\n\n    yauzl.open(inputPath, {lazyEntries: true}, (err, zipFile) => {\n        if (err) {\n            return callback(err);\n        }\n\n        zipFile.readEntry();\n        zipFile.once(\"end\", () => {\n            callback(null, processed);\n        });\n\n        zipFile.on(\"entry\", (entry) => {\n            ++processed;\n\n            zipFile.readEntry();\n        });\n    });\n}\n\nfunction computeSize(inputPath, callback) {\n    let totalSize = 0;\n    fs.stat(inputPath, (err, stats) => {\n        if (err) {\n            return callback(err);\n        }\n\n        if (stats.isFile()) {\n            return callback(undefined, stats.size);\n        }\n\n        fs.readdir(inputPath, (err, files) => {\n            if (err) {\n                return callback(err);\n            }\n\n\n            let count = files.length;\n\n            if (count === 0) {\n                return callback(undefined, 0);\n            }\n\n            files.forEach(file => {\n                fs.stat(path.join(inputPath, file), (err, stats) => {\n                    if (err) {\n                        return callback(err);\n                    }\n\n                    if (stats.isDirectory()) {\n                        computeSize(path.join(inputPath, file), (err, filesSize) => {\n                            if (err) {\n                                return callback(err);\n                            }\n\n                            totalSize += filesSize;\n\n                            if (--count === 0) {\n                                callback(undefined, totalSize);\n                            }\n                        });\n                    } else {\n\n                        totalSize += stats.size;\n\n                        if (--count === 0) {\n                            callback(undefined, totalSize);\n                        }\n                    }\n                });\n            })\n        });\n    });\n}\n\nmodule.exports = {\n    countFiles,\n    countZipEntries,\n    computeSize\n};\n","const crypto = require('crypto');\nconst fs = require('fs');\nconst path = require(\"path\");\nconst PskArchiver = require(\"../psk-archiver\");\nconst algorithm = 'aes-256-gcm';\n\n\nconst iterations_number = 1000;\n\nfunction encode(buffer) {\n\treturn buffer.toString('base64')\n\t\t.replace(/\\+/g, '')\n\t\t.replace(/\\//g, '')\n\t\t.replace(/=+$/, '');\n}\n\nfunction deleteRecursively(inputPath, callback) {\n\n\tfs.stat(inputPath, function (err, stats) {\n\t\tif (err) {\n\t\t\tcallback(err, stats);\n\t\t\treturn;\n\t\t}\n\t\tif (stats.isFile()) {\n\t\t\tfs.unlink(inputPath, function (err) {\n\t\t\t\tif (err) {\n\t\t\t\t\tcallback(err, null);\n\t\t\t\t} else {\n\t\t\t\t\tcallback(null, true);\n\t\t\t\t}\n\t\t\t});\n\t\t} else if (stats.isDirectory()) {\n\t\t\tfs.readdir(inputPath, function (err, files) {\n\t\t\t\tif (err) {\n\t\t\t\t\tcallback(err, null);\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t\tconst f_length = files.length;\n\t\t\t\tlet f_delete_index = 0;\n\n\t\t\t\tconst checkStatus = function () {\n\t\t\t\t\tif (f_length === f_delete_index) {\n\t\t\t\t\t\tfs.rmdir(inputPath, function (err) {\n\t\t\t\t\t\t\tif (err) {\n\t\t\t\t\t\t\t\tcallback(err, null);\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tcallback(null, true);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t});\n\t\t\t\t\t\treturn true;\n\t\t\t\t\t}\n\t\t\t\t\treturn false;\n\t\t\t\t};\n\t\t\t\tif (!checkStatus()) {\n\t\t\t\t\tfiles.forEach(function (file) {\n\t\t\t\t\t\tconst tempPath = path.join(inputPath, file);\n\t\t\t\t\t\tdeleteRecursively(tempPath, function removeRecursiveCB(err, status) {\n\t\t\t\t\t\t\tif (!err) {\n\t\t\t\t\t\t\t\tf_delete_index++;\n\t\t\t\t\t\t\t\tcheckStatus();\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tcallback(err, null);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t});\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t});\n\t\t}\n\t});\n}\n\n\n\n\n\nfunction createPskHash(data) {\n\tconst pskHash = new PskHash();\n\tpskHash.update(data);\n\treturn pskHash.digest();\n}\n\nfunction PskHash() {\n\tconst sha512 = crypto.createHash('sha512');\n\tconst sha256 = crypto.createHash('sha256');\n\n\tfunction update(data) {\n\t\tsha512.update(data);\n\t}\n\n\tfunction digest() {\n\t\tsha256.update(sha512.digest());\n\t\treturn sha256.digest();\n\t}\n\n\treturn {\n\t\tupdate,\n\t\tdigest\n\t}\n}\n\n\nfunction generateSalt(inputData, saltLen) {\n\tconst hash = crypto.createHash('sha512');\n\thash.update(inputData);\n\tconst digest = Buffer.from(hash.digest('hex'), 'binary');\n\n\treturn digest.slice(0, saltLen);\n}\n\nfunction encrypt(data, password) {\n\tconst keySalt = crypto.randomBytes(32);\n\tconst key = crypto.pbkdf2Sync(password, keySalt, iterations_number, 32, 'sha512');\n\n\tconst aadSalt = crypto.randomBytes(32);\n\tconst aad = crypto.pbkdf2Sync(password, aadSalt, iterations_number, 32, 'sha512');\n\n\tconst salt = Buffer.concat([keySalt, aadSalt]);\n\tconst iv = crypto.pbkdf2Sync(password, salt, iterations_number, 12, 'sha512');\n\n\tconst cipher = crypto.createCipheriv(algorithm, key, iv);\n\tcipher.setAAD(aad);\n\tlet encryptedText = cipher.update(data, 'binary');\n\tconst final = Buffer.from(cipher.final('binary'), 'binary');\n\tconst tag = cipher.getAuthTag();\n\n\tencryptedText = Buffer.concat([encryptedText, final]);\n\n\treturn Buffer.concat([salt, encryptedText, tag]);\n}\n\nfunction decrypt(encryptedData, password) {\n\tconst salt = encryptedData.slice(0, 64);\n\tconst keySalt = salt.slice(0, 32);\n\tconst aadSalt = salt.slice(-32);\n\n\tconst iv = crypto.pbkdf2Sync(password, salt, iterations_number, 12, 'sha512');\n\tconst key = crypto.pbkdf2Sync(password, keySalt, iterations_number, 32, 'sha512');\n\tconst aad = crypto.pbkdf2Sync(password, aadSalt, iterations_number, 32, 'sha512');\n\n\tconst ciphertext = encryptedData.slice(64, encryptedData.length - 16);\n\tconst tag = encryptedData.slice(-16);\n\n\tconst decipher = crypto.createDecipheriv(algorithm, key, iv);\n\tdecipher.setAuthTag(tag);\n\tdecipher.setAAD(aad);\n\n\tlet plaintext = Buffer.from(decipher.update(ciphertext, 'binary'), 'binary');\n\tconst final = Buffer.from(decipher.final('binary'), 'binary');\n\tplaintext = Buffer.concat([plaintext, final]);\n\treturn plaintext;\n}\n\nfunction encryptObjectInMemory(inputObj, password, depth, callback) {\n\tconst archiver = new PskArchiver();\n\n\tarchiver.zipInMemory(inputObj, depth, function (err, zippedObj) {\n\t\tif (err) {\n\t\t\treturn callback(err);\n\t\t}\n\t\tconst cipherText = encrypt(zippedObj, password);\n\t\tcallback(null, cipherText);\n\t})\n}\n\nfunction decryptObjectInMemory(encryptedObject, password, callback) {\n\tconst archiver = new PskArchiver();\n\n\tconst zippedObject = decrypt(encryptedObject, password);\n\tarchiver.unzipInMemory(zippedObject, function (err, obj) {\n\t\tif (err) {\n\t\t\treturn callback(err);\n\t\t}\n\t\tcallback(null, obj);\n\t})\n}\n\n\nmodule.exports = {\n\tcreatePskHash,\n\tencrypt,\n\tencryptObjectInMemory,\n\tdecrypt,\n\tdecryptObjectInMemory,\n\tdeleteRecursively,\n\tencode,\n\tgenerateSalt,\n\titerations_number,\n\talgorithm,\n\tPskHash\n};\n\n","const stream = require('stream');\n\n\nfunction isStream (obj) {\n\treturn obj instanceof stream.Stream || obj instanceof stream.Duplex;\n}\n\n\nfunction isReadable (obj) {\n\treturn isStream(obj) && typeof obj._read === 'function' && typeof obj._readableState === 'object'\n}\n\n\nfunction isWritable (obj) {\n\treturn isStream(obj) && typeof obj._write === 'function' && typeof obj._writableState === 'object'\n}\n\n\nfunction isDuplex (obj) {\n\treturn isReadable(obj) && isWritable(obj)\n}\n\n\nmodule.exports            = isStream;\nmodule.exports.isReadable = isReadable;\nmodule.exports.isWritable = isWritable;\nmodule.exports.isDuplex   = isDuplex;","/*\n SignSens helper functions\n */\nconst crypto = require('crypto');\n\nexports.wipeOutsidePayload = function wipeOutsidePayload(hashStringHexa, pos, size){\n    var result;\n    var sz = hashStringHexa.length;\n\n    var end = (pos + size) % sz;\n\n    if(pos < end){\n        result = '0'.repeat(pos) +  hashStringHexa.substring(pos, end) + '0'.repeat(sz - end);\n    }\n    else {\n        result = hashStringHexa.substring(0, end) + '0'.repeat(pos - end) + hashStringHexa.substring(pos, sz);\n    }\n    return result;\n}\n\n\n\nexports.extractPayload = function extractPayload(hashStringHexa, pos, size){\n    var result;\n\n    var sz = hashStringHexa.length;\n    var end = (pos + size) % sz;\n\n    if( pos < end){\n        result = hashStringHexa.substring(pos, pos + size);\n    } else{\n\n        if(0 != end){\n            result = hashStringHexa.substring(0, end)\n        }  else {\n            result = \"\";\n        }\n        result += hashStringHexa.substring(pos, sz);\n    }\n    return result;\n}\n\n\n\nexports.fillPayload = function fillPayload(payload, pos, size){\n    var sz = 64;\n    var result = \"\";\n\n    var end = (pos + size) % sz;\n\n    if( pos < end){\n        result = '0'.repeat(pos) + payload + '0'.repeat(sz - end);\n    } else{\n        result = payload.substring(0,end);\n        result += '0'.repeat(pos - end);\n        result += payload.substring(end);\n    }\n    return result;\n}\n\n\n\nexports.generatePosHashXTimes = function generatePosHashXTimes(buffer, pos, size, count){ //generate positional hash\n    var result  = buffer.toString(\"hex\");\n\n    /*if(pos != -1 )\n        result[pos] = 0; */\n\n    for(var i = 0; i < count; i++){\n        var hash = crypto.createHash('sha256');\n        result = exports.wipeOutsidePayload(result, pos, size);\n        hash.update(result);\n        result = hash.digest('hex');\n    }\n    return exports.wipeOutsidePayload(result, pos, size);\n}\n\nexports.hashStringArray = function (counter, arr, payloadSize){\n\n    const hash = crypto.createHash('sha256');\n    var result = counter.toString(16);\n\n    for(var i = 0 ; i < 64; i++){\n        result += exports.extractPayload(arr[i],i, payloadSize);\n    }\n\n    hash.update(result);\n    var result = hash.digest('hex');\n    return result;\n}\n\n\n\n\n\n\nfunction dumpMember(obj){\n    var type = Array.isArray(obj) ? \"array\" : typeof obj;\n    if(obj === null){\n        return \"null\";\n    }\n    if(obj === undefined){\n        return \"undefined\";\n    }\n\n    switch(type){\n        case \"number\":\n        case \"string\":return obj.toString(); break;\n        case \"object\": return exports.dumpObjectForHashing(obj); break;\n        case \"boolean\": return  obj? \"true\": \"false\"; break;\n        case \"array\":\n            var result = \"\";\n            for(var i=0; i < obj.length; i++){\n                result += exports.dumpObjectForHashing(obj[i]);\n            }\n            return result;\n            break;\n        default:\n            throw new Error(\"Type \" +  type + \" cannot be cryptographically digested\");\n    }\n\n}\n\n\nexports.dumpObjectForHashing = function(obj){\n    var result = \"\";\n\n    if(obj === null){\n        return \"null\";\n    }\n    if(obj === undefined){\n        return \"undefined\";\n    }\n\n    var basicTypes = {\n        \"array\"     : true,\n        \"number\"    : true,\n        \"boolean\"   : true,\n        \"string\"    : true,\n        \"object\"    : false\n    }\n\n    var type = Array.isArray(obj) ? \"array\" : typeof obj;\n    if( basicTypes[type]){\n        return dumpMember(obj);\n    }\n\n    var keys = Object.keys(obj);\n    keys.sort();\n\n\n    for(var i=0; i < keys.length; i++){\n        result += dumpMember(keys[i]);\n        result += dumpMember(obj[keys[i]]);\n    }\n\n    return result;\n}\n\n\nexports.hashValues  = function (values){\n    const hash = crypto.createHash('sha256');\n    var result = exports.dumpObjectForHashing(values);\n    hash.update(result);\n    return hash.digest('hex');\n};\n\nexports.getJSONFromSignature = function getJSONFromSignature(signature, size){\n    var result = {\n        proof:[]\n    };\n    var a = signature.split(\":\");\n    result.agent        = a[0];\n    result.counter      =  parseInt(a[1], \"hex\");\n    result.nextPublic   =  a[2];\n\n    var proof = a[3]\n\n\n    if(proof.length/size != 64) {\n        throw new Error(\"Invalid signature \" + proof);\n    }\n\n    for(var i = 0; i < 64; i++){\n        result.proof.push(exports.fillPayload(proof.substring(i * size,(i+1) * size ), i, size))\n    }\n\n    return result;\n}\n\nexports.createSignature = function (agent,counter, nextPublic, arr, size){\n    var result = \"\";\n\n    for(var i = 0; i < arr.length; i++){\n        result += exports.extractPayload(arr[i], i , size);\n    }\n\n    return agent + \":\" + counter + \":\" + nextPublic + \":\" + result;\n}","/*\nInitial License: (c) Axiologic Research & Alboaie Sînică.\nContributors: Axiologic Research , PrivateSky project\nCode License: LGPL or MIT.\n*/\n\n\n/**\n *   Usually an event could cause execution of other callback events . We say that is a level 1 event if is causeed by a level 0 event and so on\n *\n *      SoundPubSub provides intuitive results regarding to asynchronous calls of callbacks and computed values/expressions:\n *   we prevent immediate execution of event callbacks to ensure the intuitive final result is guaranteed as level 0 execution\n *   we guarantee that any callback function is \"re-entrant\"\n *   we are also trying to reduce the number of callback execution by looking in queues at new messages published by\n *   trying to compact those messages (removing duplicate messages, modifying messages, or adding in the history of another event ,etc)\n *\n *      Example of what can be wrong without non-sound asynchronous calls:\n *\n *  Step 0: Initial state:\n *   a = 0;\n *   b = 0;\n *\n *  Step 1: Initial operations:\n *   a = 1;\n *   b = -1;\n *\n *  // an observer reacts to changes in a and b and compute CORRECT like this:\n *   if( a + b == 0) {\n *       CORRECT = false;\n *       notify(...); // act or send a notification somewhere..\n *   } else {\n *      CORRECT = false;\n *   }\n *\n *    Notice that: CORRECT will be true in the end , but meantime, after a notification was sent and CORRECT was wrongly, temporarily false!\n *    soundPubSub guarantee that this does not happen because the syncronous call will before any observer (bot asignation on a and b)\n *\n *   More:\n *   you can use blockCallBacks and releaseCallBacks in a function that change a lot a collection or bindable objects and all\n *   the notifications will be sent compacted and properly\n */\n\n// TODO: optimisation!? use a more efficient queue instead of arrays with push and shift!?\n// TODO: see how big those queues can be in real applications\n// for a few hundreds items, queues made from array should be enough\n//*   Potential TODOs:\n//    *     prevent any form of problem by calling callbacks in the expected order !?\n//*     preventing infinite loops execution cause by events!?\n//*\n//*\n// TODO: detect infinite loops (or very deep propagation) It is possible!?\n\nconst Queue = require('swarmutils').Queue;\n\nfunction SoundPubSub(){\n\n\t/**\n\t * publish\n\t *      Publish a message {Object} to a list of subscribers on a specific topic\n\t *\n\t * @params {String|Number} target,  {Object} message\n\t * @return number of channel subscribers that will be notified\n\t */\n\tthis.publish = function(target, message){\n\t\tif(!invalidChannelName(target) && !invalidMessageType(message) && (typeof channelSubscribers[target] != 'undefined')){\n\t\t\tcompactAndStore(target, message);\n\t\t\tsetTimeout(dispatchNext, 0);\n\t\t\treturn channelSubscribers[target].length;\n\t\t} else{\n\t\t\treturn null;\n\t\t}\n\t};\n\n\t/**\n\t * subscribe\n\t *      Subscribe / add a {Function} callBack on a {String|Number}target channel subscribers list in order to receive\n\t *      messages published if the conditions defined by {Function}waitForMore and {Function}filter are passed.\n\t *\n\t * @params {String|Number}target, {Function}callBack, {Function}waitForMore, {Function}filter\n\t *\n\t *          target      - channel name to subscribe\n\t *          callback    - function to be called when a message was published on the channel\n\t *          waitForMore - a intermediary function that will be called after a successfuly message delivery in order\n\t *                          to decide if a new messages is expected...\n\t *          filter      - a function that receives the message before invocation of callback function in order to allow\n\t *                          relevant message before entering in normal callback flow\n\t * @return\n\t */\n\tthis.subscribe = function(target, callBack, waitForMore, filter){\n\t\tif(!invalidChannelName(target) && !invalidFunction(callBack)){\n\t\t\tvar subscriber = {\"callBack\":callBack, \"waitForMore\":waitForMore, \"filter\":filter};\n\t\t\tvar arr = channelSubscribers[target];\n\t\t\tif(typeof arr == 'undefined'){\n\t\t\t\tarr = [];\n\t\t\t\tchannelSubscribers[target] = arr;\n\t\t\t}\n\t\t\tarr.push(subscriber);\n\t\t}\n\t};\n\n\t/**\n\t * unsubscribe\n\t *      Unsubscribe/remove {Function} callBack from the list of subscribers of the {String|Number} target channel\n\t *\n\t * @params {String|Number} target, {Function} callBack, {Function} filter\n\t *\n\t *          target      - channel name to unsubscribe\n\t *          callback    - reference of the original function that was used as subscribe\n\t *          filter      - reference of the original filter function\n\t * @return\n\t */\n\tthis.unsubscribe = function(target, callBack, filter){\n\t\tif(!invalidFunction(callBack)){\n\t\t\tvar gotit = false;\n\t\t\tif(channelSubscribers[target]){\n\t\t\t\tfor(var i = 0; i < channelSubscribers[target].length;i++){\n\t\t\t\t\tvar subscriber =  channelSubscribers[target][i];\n\t\t\t\t\tif(subscriber.callBack === callBack && ( typeof filter === 'undefined' || subscriber.filter === filter )){\n\t\t\t\t\t\tgotit = true;\n\t\t\t\t\t\tsubscriber.forDelete = true;\n\t\t\t\t\t\tsubscriber.callBack = undefined;\n\t\t\t\t\t\tsubscriber.filter = undefined;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t\tif(!gotit){\n\t\t\t\twprint(\"Unable to unsubscribe a callback that was not subscribed!\");\n\t\t\t}\n\t\t}\n\t};\n\n\t/**\n\t * blockCallBacks\n\t *\n\t * @params\n\t * @return\n\t */\n\tthis.blockCallBacks = function(){\n\t\tlevel++;\n\t};\n\n\t/**\n\t * releaseCallBacks\n\t *\n\t * @params\n\t * @return\n\t */\n\tthis.releaseCallBacks = function(){\n\t\tlevel--;\n\t\t//hack/optimisation to not fill the stack in extreme cases (many events caused by loops in collections,etc)\n\t\twhile(level === 0 && dispatchNext(true)){\n\t\t\t//nothing\n\t\t}\n\n\t\twhile(level === 0 && callAfterAllEvents()){\n            //nothing\n\t\t}\n\t};\n\n\t/**\n\t * afterAllEvents\n\t *\n\t * @params {Function} callback\n\t *\n\t *          callback - function that needs to be invoked once all events are delivered\n\t * @return\n\t */\n\tthis.afterAllEvents = function(callBack){\n\t\tif(!invalidFunction(callBack)){\n\t\t\tafterEventsCalls.push(callBack);\n\t\t}\n\t\tthis.blockCallBacks();\n\t\tthis.releaseCallBacks();\n\t};\n\n\t/**\n\t * hasChannel\n\t *\n\t * @params {String|Number} channel\n\t *\n\t *          channel - name of the channel that need to be tested if present\n\t * @return\n\t */\n\tthis.hasChannel = function(channel){\n\t\treturn !invalidChannelName(channel) && (typeof channelSubscribers[channel] != 'undefined') ? true : false;\n\t};\n\n\t/**\n\t * addChannel\n\t *\n\t * @params {String} channel\n\t *\n\t *          channel - name of a channel that needs to be created and added to soundpubsub repository\n\t * @return\n\t */\n\tthis.addChannel = function(channel){\n\t\tif(!invalidChannelName(channel) && !this.hasChannel(channel)){\n\t\t\tchannelSubscribers[channel] = [];\n\t\t}\n\t};\n\n\t/* ---------------------------------------- protected stuff ---------------------------------------- */\n\tvar self = this;\n\t// map channelName (object local id) -> array with subscribers\n\tvar channelSubscribers = {};\n\n\t// map channelName (object local id) -> queue with waiting messages\n\tvar channelsStorage = {};\n\n\t// object\n\tvar typeCompactor = {};\n\n\t// channel names\n\tvar executionQueue = new Queue();\n\tvar level = 0;\n\n\n\n\t/**\n\t * registerCompactor\n\t *\n\t *       An compactor takes a newEvent and and oldEvent and return the one that survives (oldEvent if\n\t *  it can compact the new one or the newEvent if can't be compacted)\n\t *\n\t * @params {String} type, {Function} callBack\n\t *\n\t *          type        - channel name to unsubscribe\n\t *          callBack    - handler function for that specific event type\n\t * @return\n\t */\n\tthis.registerCompactor = function(type, callBack) {\n\t\tif(!invalidFunction(callBack)){\n\t\t\ttypeCompactor[type] = callBack;\n\t\t}\n\t};\n\n\t/**\n\t * dispatchNext\n\t *\n\t * @param fromReleaseCallBacks: hack to prevent too many recursive calls on releaseCallBacks\n\t * @return {Boolean}\n\t */\n\tfunction dispatchNext(fromReleaseCallBacks){\n\t\tif(level > 0) {\n\t\t\treturn false;\n\t\t}\n\t\tconst channelName = executionQueue.front();\n\t\tif(typeof channelName != 'undefined'){\n\t\t\tself.blockCallBacks();\n\t\t\ttry{\n\t\t\t\tlet message;\n\t\t\t\tif(!channelsStorage[channelName].isEmpty()) {\n\t\t\t\t\tmessage = channelsStorage[channelName].front();\n\t\t\t\t}\n\t\t\t\tif(typeof message == 'undefined'){\n\t\t\t\t\tif(!channelsStorage[channelName].isEmpty()){\n\t\t\t\t\t\twprint(\"Can't use as message in a pub/sub channel this object: \" + message);\n\t\t\t\t\t}\n\t\t\t\t\texecutionQueue.pop();\n\t\t\t\t} else {\n\t\t\t\t\tif(typeof message.__transmisionIndex == 'undefined'){\n\t\t\t\t\t\tmessage.__transmisionIndex = 0;\n\t\t\t\t\t\tfor(var i = channelSubscribers[channelName].length-1; i >= 0 ; i--){\n\t\t\t\t\t\t\tvar subscriber =  channelSubscribers[channelName][i];\n\t\t\t\t\t\t\tif(subscriber.forDelete === true){\n\t\t\t\t\t\t\t\tchannelSubscribers[channelName].splice(i,1);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t} else{\n\t\t\t\t\t\tmessage.__transmisionIndex++;\n\t\t\t\t\t}\n\t\t\t\t\t//TODO: for immutable objects it will not work also, fix for shape models\n\t\t\t\t\tif(typeof message.__transmisionIndex == 'undefined'){\n\t\t\t\t\t\twprint(\"Can't use as message in a pub/sub channel this object: \" + message);\n\t\t\t\t\t}\n\t\t\t\t\tsubscriber = channelSubscribers[channelName][message.__transmisionIndex];\n\t\t\t\t\tif(typeof subscriber == 'undefined'){\n\t\t\t\t\t\tdelete message.__transmisionIndex;\n\t\t\t\t\t\tchannelsStorage[channelName].pop();\n\t\t\t\t\t} else{\n\t\t\t\t\t\tif(subscriber.filter === null || typeof subscriber.filter === \"undefined\" || (!invalidFunction(subscriber.filter) && subscriber.filter(message))){\n\t\t\t\t\t\t\tif(!subscriber.forDelete){\n\t\t\t\t\t\t\t\tsubscriber.callBack(message);\n\t\t\t\t\t\t\t\tif(subscriber.waitForMore && !invalidFunction(subscriber.waitForMore) && !subscriber.waitForMore(message)){\n\t\t\t\t\t\t\t\t\tsubscriber.forDelete = true;\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} catch(err){\n\t\t\t\twprint(\"Event callback failed: \"+ subscriber.callBack +\"error: \" + err.stack);\n\t\t\t}\n\t\t\t//\n\t\t\tif(fromReleaseCallBacks){\n\t\t\t\tlevel--;\n\t\t\t} else{\n\t\t\t\tself.releaseCallBacks();\n\t\t\t}\n\t\t\treturn true;\n\t\t} else{\n\t\t\treturn false;\n\t\t}\n\t}\n\n\tfunction compactAndStore(target, message){\n\t\tvar gotCompacted = false;\n\t\tvar arr = channelsStorage[target];\n\t\tif(typeof arr == 'undefined'){\n\t\t\tarr = new Queue();\n\t\t\tchannelsStorage[target] = arr;\n\t\t}\n\n\t\tif(message && typeof message.type != 'undefined'){\n\t\t\tvar typeCompactorCallBack = typeCompactor[message.type];\n\n\t\t\tif(typeof typeCompactorCallBack != 'undefined'){\n\t\t\t\tfor(let channel of arr) {\n\t\t\t\t\tif(typeCompactorCallBack(message, channel) === channel) {\n\t\t\t\t\t\tif(typeof channel.__transmisionIndex == 'undefined') {\n\t\t\t\t\t\t\tgotCompacted = true;\n\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tif(!gotCompacted && message){\n\t\t\tarr.push(message);\n\t\t\texecutionQueue.push(target);\n\t\t}\n\t}\n\n\tvar afterEventsCalls = new Queue();\n\tfunction callAfterAllEvents (){\n\t\tif(!afterEventsCalls.isEmpty()){\n\t\t\tvar callBack = afterEventsCalls.pop();\n\t\t\t//do not catch exceptions here..\n\t\t\tcallBack();\n\t\t}\n\t\treturn !afterEventsCalls.isEmpty();\n\t}\n\n\tfunction invalidChannelName(name){\n\t\tvar result = false;\n\t\tif(!name || (typeof name != \"string\" && typeof name != \"number\")){\n\t\t\tresult = true;\n\t\t\twprint(\"Invalid channel name: \" + name);\n\t\t}\n\n\t\treturn result;\n\t}\n\n\tfunction invalidMessageType(message){\n\t\tvar result = false;\n\t\tif(!message || typeof message != \"object\"){\n\t\t\tresult = true;\n\t\t\twprint(\"Invalid messages types: \" + message);\n\t\t}\n\t\treturn result;\n\t}\n\n\tfunction invalidFunction(callback){\n\t\tvar result = false;\n\t\tif(!callback || typeof callback != \"function\"){\n\t\t\tresult = true;\n\t\t\twprint(\"Expected to be function but is: \" + callback);\n\t\t}\n\t\treturn result;\n\t}\n}\n\nexports.soundPubSub = new SoundPubSub();","function product(args) {\n    if(!args.length){\n        return [ [] ];\n    }\n    var prod = product(args.slice(1)), r = [];\n    args[0].forEach(function(x) {\n        prod.forEach(function(p) {\n            r.push([ x ].concat(p));\n        });\n    });\n    return r;\n}\n\nfunction objectProduct(obj) {\n    var keys = Object.keys(obj),\n        values = keys.map(function(x) { return obj[x]; });\n\n    return product(values).map(function(p) {\n        var e = {};\n        keys.forEach(function(k, n) { e[k] = p[n]; });\n        return e;\n    });\n}\n\nmodule.exports = objectProduct;","var meta = \"meta\";\n\nfunction OwM(serialized){\n\n    if(serialized){\n        return OwM.prototype.convert(serialized);\n    }\n\n    Object.defineProperty(this, meta, {\n        writable: false,\n        enumerable: true,\n        value: {}\n    });\n\n    Object.defineProperty(this, \"setMeta\", {\n        writable: false,\n        enumerable: false,\n        configurable:false,\n        value: function(prop, value){\n            if(typeof prop == \"object\" && typeof value == \"undefined\"){\n                for(var p in prop){\n                    this[meta][p] = prop[p];\n                }\n                return prop;\n            }\n            this[meta][prop] = value;\n            return value;\n        }\n    });\n\n    Object.defineProperty(this, \"getMeta\", {\n        writable: false,\n        value: function(prop){\n            return this[meta][prop];\n        }\n    });\n}\n\nfunction testOwMSerialization(obj){\n    let res = false;\n\n    if(obj){\n        res = typeof obj[meta] != \"undefined\" && !(obj instanceof OwM);\n    }\n\n    return res;\n}\n\nOwM.prototype.convert = function(serialized){\n    const owm = new OwM();\n\n    for(var metaProp in serialized.meta){\n        if(!testOwMSerialization(serialized[metaProp])) {\n            owm.setMeta(metaProp, serialized.meta[metaProp]);\n        }else{\n            owm.setMeta(metaProp, OwM.prototype.convert(serialized.meta[metaProp]));\n        }\n    }\n\n    for(var simpleProp in serialized){\n        if(simpleProp === meta) {\n            continue;\n        }\n\n        if(!testOwMSerialization(serialized[simpleProp])){\n            owm[simpleProp] = serialized[simpleProp];\n        }else{\n            owm[simpleProp] = OwM.prototype.convert(serialized[simpleProp]);\n        }\n    }\n\n    return owm;\n};\n\nOwM.prototype.getMetaFrom = function(obj, name){\n    var res;\n    if(!name){\n        res = obj[meta];\n    }else{\n        res = obj[meta][name];\n    }\n    return res;\n};\n\nOwM.prototype.setMetaFor = function(obj, name, value){\n    obj[meta][name] = value;\n    return obj[meta][name];\n};\n\nmodule.exports = OwM;","function QueueElement(content) {\n\tthis.content = content;\n\tthis.next = null;\n}\n\nfunction Queue() {\n\tthis.head = null;\n\tthis.tail = null;\n\tthis.length = 0;\n\tthis.push = function (value) {\n\t\tconst newElement = new QueueElement(value);\n\t\tif (!this.head) {\n\t\t\tthis.head = newElement;\n\t\t\tthis.tail = newElement;\n\t\t} else {\n\t\t\tthis.tail.next = newElement;\n\t\t\tthis.tail = newElement;\n\t\t}\n\t\tthis.length++;\n\t};\n\n\tthis.pop = function () {\n\t\tif (!this.head) {\n\t\t\treturn null;\n\t\t}\n\t\tconst headCopy = this.head;\n\t\tthis.head = this.head.next;\n\t\tthis.length--;\n\n\t\t//fix???????\n\t\tif(this.length === 0){\n            this.tail = null;\n\t\t}\n\n\t\treturn headCopy.content;\n\t};\n\n\tthis.front = function () {\n\t\treturn this.head ? this.head.content : undefined;\n\t};\n\n\tthis.isEmpty = function () {\n\t\treturn this.head === null;\n\t};\n\n\tthis[Symbol.iterator] = function* () {\n\t\tlet head = this.head;\n\t\twhile(head !== null) {\n\t\t\tyield head.content;\n\t\t\thead = head.next;\n\t\t}\n\t}.bind(this);\n}\n\nQueue.prototype.toString = function () {\n\tlet stringifiedQueue = '';\n\tlet iterator = this.head;\n\twhile (iterator) {\n\t\tstringifiedQueue += `${JSON.stringify(iterator.content)} `;\n\t\titerator = iterator.next;\n\t}\n\treturn stringifiedQueue;\n};\n\nQueue.prototype.inspect = Queue.prototype.toString;\n\nmodule.exports = Queue;","const OwM = require(\"./OwM\");\n\n/*\n    Prepare the state of a swarm to be serialised\n*/\n\nexports.asJSON = function(valueObj, phaseName, args, callback){\n\n        let valueObject = valueObj.valueOf();\n        let res = new OwM();\n        res.publicVars          = valueObject.publicVars;\n        res.privateVars         = valueObject.privateVars;\n\n        res.setMeta(\"swarmTypeName\", OwM.prototype.getMetaFrom(valueObject, \"swarmTypeName\"));\n        res.setMeta(\"swarmId\",       OwM.prototype.getMetaFrom(valueObject, \"swarmId\"));\n        res.setMeta(\"target\",        OwM.prototype.getMetaFrom(valueObject, \"target\"));\n        res.setMeta(\"homeSecurityContext\",        OwM.prototype.getMetaFrom(valueObject, \"homeSecurityContext\"));\n        res.setMeta(\"requestId\",        OwM.prototype.getMetaFrom(valueObject, \"requestId\"));\n\n        if(!phaseName){\n            res.setMeta(\"command\", \"stored\");\n        } else {\n            res.setMeta(\"phaseName\", phaseName);\n            res.setMeta(\"phaseId\", $$.uidGenerator.safe_uuid());\n            res.setMeta(\"args\", args);\n            res.setMeta(\"command\", OwM.prototype.getMetaFrom(valueObject, \"command\") || \"executeSwarmPhase\");\n        }\n\n        res.setMeta(\"waitStack\", valueObject.meta.waitStack); //TODO: think if is not better to be deep cloned and not referenced!!!\n\n        if(callback){\n            return callback(null, res);\n        }\n        //console.log(\"asJSON:\", res, valueObject);\n        return res;\n};\n\nexports.jsonToNative = function(serialisedValues, result){\n\n    for(let v in serialisedValues.publicVars){\n        result.publicVars[v] = serialisedValues.publicVars[v];\n\n    };\n    for(let l in serialisedValues.privateVars){\n        result.privateVars[l] = serialisedValues.privateVars[l];\n    };\n\n    for(let i in OwM.prototype.getMetaFrom(serialisedValues)){\n        OwM.prototype.setMetaFor(result, i, OwM.prototype.getMetaFrom(serialisedValues, i));\n    };\n\n};","var commands = {};\nvar commands_help = {};\n\n//global function addCommand\naddCommand = function addCommand(verb, adverbe, funct, helpLine){\n    var cmdId;\n    if(!helpLine){\n        helpLine = \" \";\n    } else {\n        helpLine = \" \" + helpLine;\n    }\n    if(adverbe){\n        cmdId = verb + \" \" +  adverbe;\n        helpLine = verb + \" \" +  adverbe + helpLine;\n    } else {\n        cmdId = verb;\n        helpLine = verb + helpLine;\n    }\n    commands[cmdId] = funct;\n        commands_help[cmdId] = helpLine;\n};\n\nfunction doHelp(){\n    console.log(\"List of commands:\");\n    for(var l in commands_help){\n        console.log(\"\\t\", commands_help[l]);\n    }\n}\n\naddCommand(\"-h\", null, doHelp, \"\\t\\t\\t\\t\\t\\t |just print the help\");\naddCommand(\"/?\", null, doHelp, \"\\t\\t\\t\\t\\t\\t |just print the help\");\naddCommand(\"help\", null, doHelp, \"\\t\\t\\t\\t\\t\\t |just print the help\");\n\n\nfunction runCommand(){\n  var argv = Object.assign([], process.argv);\n  var cmdId = null;\n  var cmd = null;\n  argv.shift();\n  argv.shift();\n\n  if(argv.length >=1){\n      cmdId = argv[0];\n      cmd = commands[cmdId];\n      argv.shift();\n  }\n\n\n  if(!cmd && argv.length >=1){\n      cmdId = cmdId + \" \" + argv[0];\n      cmd = commands[cmdId];\n      argv.shift();\n  }\n\n  if(!cmd){\n    if(cmdId){\n        console.log(\"Unknown command: \", cmdId);\n    }\n    cmd = doHelp;\n  }\n\n  cmd.apply(null,argv);\n\n}\n\nmodule.exports = {\n    runCommand\n};\n\n","\nfunction encode(buffer) {\n    return buffer.toString('base64')\n        .replace(/\\+/g, '')\n        .replace(/\\//g, '')\n        .replace(/=+$/, '');\n};\n\nfunction stampWithTime(buf, salt, msalt){\n    if(!salt){\n        salt = 1;\n    }\n    if(!msalt){\n        msalt = 1;\n    }\n    var date = new Date;\n    var ct = Math.floor(date.getTime() / salt);\n    var counter = 0;\n    while(ct > 0 ){\n        //console.log(\"Counter\", counter, ct);\n        buf[counter*msalt] = Math.floor(ct % 256);\n        ct = Math.floor(ct / 256);\n        counter++;\n    }\n}\n\n/*\n    The uid contains around 256 bits of randomness and are unique at the level of seconds. This UUID should by cryptographically safe (can not be guessed)\n\n    We generate a safe UID that is guaranteed unique (by usage of a PRNG to geneate 256 bits) and time stamping with the number of seconds at the moment when is generated\n    This method should be safe to use at the level of very large distributed systems.\n    The UUID is stamped with time (seconds): does it open a way to guess the UUID? It depends how safe is \"crypto\" PRNG, but it should be no problem...\n\n */\n\nvar generateUid = null;\n\n\nexports.init = function(externalGenerator){\n    generateUid = externalGenerator.generateUid;\n    return module.exports;\n};\n\nexports.safe_uuid = function() {\n    var buf = generateUid(32);\n    stampWithTime(buf, 1000, 3);\n    return encode(buf);\n};\n\n\n\n/*\n    Try to generate a small UID that is unique against chance in the same millisecond second and in a specific context (eg in the same choreography execution)\n    The id contains around 6*8 = 48  bits of randomness and are unique at the level of milliseconds\n    This method is safe on a single computer but should be used with care otherwise\n    This UUID is not cryptographically safe (can be guessed)\n */\nexports.short_uuid = function(callback) {\n    require('crypto').randomBytes(12, function (err, buf) {\n        if (err) {\n            callback(err);\n            return;\n        }\n        stampWithTime(buf,1,2);\n        callback(null, encode(buf));\n    });\n};","const crypto = require('crypto');\nconst Queue = require(\"./Queue\");\nvar PSKBuffer = typeof $$ !== \"undefined\" && $$.PSKBuffer ? $$.PSKBuffer : Buffer;\n\nfunction UidGenerator(minBuffers, buffersSize) {\n\tvar buffers = new Queue();\n\tvar lowLimit = .2;\n\n\tfunction fillBuffers(size){\n\t\t//notifyObserver();\n\t\tconst sz = size || minBuffers;\n\t\tif(buffers.length < Math.floor(minBuffers*lowLimit)){\n\t\t\tfor(var i=0+buffers.length; i < sz; i++){\n\t\t\t\tgenerateOneBuffer(null);\n\t\t\t}\n\t\t}\n\t}\n\n\tfillBuffers();\n\n\tfunction generateOneBuffer(b){\n\t\tif(!b){\n\t\t\tb = PSKBuffer.alloc(0);\n\t\t}\n\t\tconst sz = buffersSize - b.length;\n\t\t/*crypto.randomBytes(sz, function (err, res) {\n\t\t\tbuffers.push(Buffer.concat([res, b]));\n\t\t\tnotifyObserver();\n\t\t});*/\n\t\tbuffers.push(PSKBuffer.concat([ crypto.randomBytes(sz), b ]));\n\t\tnotifyObserver();\n\t}\n\n\tfunction extractN(n){\n\t\tvar sz = Math.floor(n / buffersSize);\n\t\tvar ret = [];\n\n\t\tfor(var i=0; i<sz; i++){\n\t\t\tret.push(buffers.pop());\n\t\t\tsetTimeout(generateOneBuffer, 1);\n\t\t}\n\n\n\n\t\tvar remainder = n % buffersSize;\n\t\tif(remainder > 0){\n\t\t\tvar front = buffers.pop();\n\t\t\tret.push(front.slice(0,remainder));\n\t\t\t//generateOneBuffer(front.slice(remainder));\n\t\t\tsetTimeout(function(){\n\t\t\t\tgenerateOneBuffer(front.slice(remainder));\n\t\t\t},1);\n\t\t}\n\n\t\t//setTimeout(fillBuffers, 1);\n\n\t\treturn Buffer.concat(ret);\n\t}\n\n\tvar fillInProgress = false;\n\n\tthis.generateUid = function(n){\n\t\tvar totalSize = buffers.length * buffersSize;\n\t\tif(n <= totalSize){\n\t\t\treturn extractN(n);\n\t\t} else {\n\t\t\tif(!fillInProgress){\n\t\t\t\tfillInProgress = true;\n\t\t\t\tsetTimeout(function(){\n\t\t\t\t\tfillBuffers(Math.floor(minBuffers*2.5));\n\t\t\t\t\tfillInProgress = false;\n\t\t\t\t}, 1);\n\t\t\t}\n\t\t\treturn crypto.randomBytes(n);\n\t\t}\n\t};\n\n\tvar observer;\n\tthis.registerObserver = function(obs){\n\t\tif(observer){\n\t\t\tconsole.error(new Error(\"One observer allowed!\"));\n\t\t}else{\n\t\t\tif(typeof obs == \"function\"){\n\t\t\t\tobserver = obs;\n\t\t\t\t//notifyObserver();\n\t\t\t}\n\t\t}\n\t};\n\n\tfunction notifyObserver(){\n\t\tif(observer){\n\t\t\tvar valueToReport = buffers.length*buffersSize;\n\t\t\tsetTimeout(function(){\n\t\t\t\tobserver(null, {\"size\": valueToReport});\n\t\t\t}, 10);\n\t\t}\n\t}\n}\n\nmodule.exports.createUidGenerator = function (minBuffers, bufferSize) {\n\treturn new UidGenerator(minBuffers, bufferSize);\n};\n","var mq = $$.require(\"foldermq\");\n\nconst path = require('path');\nconst child_process = require(\"child_process\");\nconst fs = require('fs');\n\nconst RESTART_TIMEOUT = 500;\nconst RESTART_TIMEOUT_LIMIT = 50000;\n\nvar sandboxes = {};\nvar exitHandler = require(\"../../utils/exitHandler\")(sandboxes);\n\nvar bootSandBox = $$.flow.describe(\"PrivateSky.swarm.engine.bootInLauncher\", {\n    boot:function(sandBox, spaceName, folder, codeFolder, callback){\n        // console.log(\"Booting in \", folder, \" context \", spaceName);\n\n        this.callback   = callback;\n        this.folder     = folder;\n        this.spaceName  = spaceName;\n        this.sandBox    = sandBox;\n        this.codeFolder    = codeFolder;\n        this.timeoutMultiplier = 1;\n\n        var task = this.serial(this.ensureFoldersExists);\n\n        task.folderShouldExist(path.join(this.folder, \"mq\"),    task.progress);\n        task.folderShouldExist(path.join(this.folder, \"code\"),  task.progress);\n        task.folderShouldExist(path.join(this.folder, \"tmp\"),   task.progress);\n    },\n    folderShouldExist:  function(path, progress){\n        fs.mkdir(path, {recursive: true}, progress);\n    },\n    copyFolder: function(sourcePath, targetPath, callback){\n        let fsExt = require(\"utils\").fsExt;\n        fsExt.copy(sourcePath, targetPath, {overwrite: true}, callback);\n    },\n    ensureFoldersExists: function(err, res){\n        if(err){\n            console.log(err);\n        } else {\n            var task = this.parallel(this.runCode);\n            task.copyFolder(path.join(this.codeFolder, \"bundles\"), path.join(this.folder, \"bundles\"), task.progress);\n            this.sandBox.inbound = mq.createQue(path.join(this.folder, \"mq/inbound\"), task.progress);\n            this.sandBox.outbound = mq.createQue(path.join(this.folder, \"mq/outbound\"), task.progress);\n        }\n\n    },\n    runCode: function(err, res){\n        if(!err){\n            var mainFile = path.join(process.env.PRIVATESKY_ROOT_FOLDER, \"core\", \"sandboxes\", \"agentSandbox.js\");\n            var args = [this.spaceName, process.env.PRIVATESKY_ROOT_FOLDER, path.resolve(process.env.PRIVATESKY_DOMAIN_BUILD)];\n            var opts = {stdio: [0, 1, 2, \"ipc\"]};\n\n            var startChild = (mainFile, args, opts) => {\n\t\t\t\tconsole.log(\"Running: \", mainFile, args, opts);\n\t\t\t\tvar child = child_process.fork(mainFile, args);\n\t\t\t\tsandboxes[this.spaceName] = child;\n\n\t\t\t\tthis.sandBox.inbound.setIPCChannel(child);\n\t\t\t\tthis.sandBox.outbound.setIPCChannel(child);\n\n\t\t\t\tchild.on(\"exit\", (code, signal)=>{\n\t\t\t\t    if(code === 0){\n\t\t\t\t        console.log(`Sandbox <${this.spaceName}> shutting down.`);\n\t\t\t\t        return;\n                    }\n\t\t\t\t    let timeout = (this.timeoutMultiplier*RESTART_TIMEOUT) % RESTART_TIMEOUT_LIMIT;\n\t\t\t\t    console.log(`Sandbox <${this.spaceName}> exits with code ${code}. Restarting in ${timeout} ms.`);\n\t\t\t\t\tsetTimeout(()=>{\n\t\t\t\t\t\tstartChild(mainFile, args, opts);\n                        this.timeoutMultiplier *= 1.5;\n                    }, timeout);\n\t\t\t\t});\n\n\t\t\t\treturn child;\n            };\n\n            this.callback(null, startChild(mainFile, args, opts));\n        } else {\n            console.log(\"Error executing sandbox!:\", err);\n            this.callback(err, null);\n        }\n    }\n\n});\n\nfunction SandBoxHandler(spaceName, folder, codeFolder, resultCallBack){\n\n    var self = this;\n    var mqHandler;\n\n\n    bootSandBox().boot(this, spaceName,folder, codeFolder, function(err, childProcess){\n        if(!err){\n            self.childProcess = childProcess;\n\n\n            /*self.outbound.registerConsumer(function(err, swarm){\n                $$.PSK_PubSub.publish($$.CONSTANTS.SWARM_FOR_EXECUTION, swarm);\n            });*/\n\n            self.outbound.registerAsIPCConsumer(function(err, swarm){\n                $$.PSK_PubSub.publish($$.CONSTANTS.SWARM_FOR_EXECUTION, swarm);\n            });\n\n            mqHandler = self.inbound.getHandler();\n            if(pendingMessages.length){\n                pendingMessages.map(function(item){\n                    self.send(item);\n                });\n                pendingMessages = null;\n            }\n        }\n    });\n\n    var pendingMessages = [];\n\n    this.send = function (swarm, callback) {\n        if(mqHandler){\n            mqHandler.sendSwarmForExecution(swarm, callback);\n        } else {\n            pendingMessages.push(swarm); //TODO: well, a deep clone will not be a better idea?\n        }\n    }\n\n}\n\n\nfunction SandBoxManager(sandboxesFolder, codeFolder, callback){\n    var self = this;\n\n    var sandBoxes = {\n\n    };\n    function belongsToReplicatedSpace(){\n        return true;\n    }\n\n    //console.log(\"Subscribing to:\", $$.CONSTANTS.SWARM_FOR_EXECUTION);\n    $$.PSK_PubSub.subscribe($$.CONSTANTS.SWARM_FOR_EXECUTION, function(swarm){\n        console.log(\"Executing in sandbox towards: \", swarm.meta.target);\n\n        if(swarm.meta.target == \"system\" || swarm.meta.command == \"asyncReturn\"){\n            $$.swarmsInstancesManager.revive_swarm(swarm);\n            //$$.swarms.restart(swarm.meta.swarmTypeName, swarm);\n        } else\n        if(swarm.meta.target == \"pds\"){\n            //\n        } else\n        if(belongsToReplicatedSpace(swarm.meta.target)){\n            self.pushToSpaceASwarm(swarm.meta.target, swarm);\n        } else {\n            //TODO: send towards network\n        }\n\n    });\n\n\n    function startSandBox(spaceName){\n        var sandBox = new SandBoxHandler(spaceName, path.join(sandboxesFolder, spaceName), codeFolder);\n        sandBoxes[spaceName] = sandBox;\n        return sandBox;\n    }\n\n\n    this.pushToSpaceASwarm = function(spaceName, swarm, callback){\n\n        console.log(\"pushToSpaceASwarm \" , spaceName);\n        var sandbox = sandBoxes[spaceName];\n        if(!sandbox){\n            sandbox = sandBoxes[spaceName] = startSandBox(spaceName);\n        }\n        sandbox.send(swarm, callback);\n    }\n\n    callback(null, this);\n}\n\n\nexports.create = function(folder, codeFolder, callback){\n    new SandBoxManager(folder, codeFolder, callback);\n};\n\n\n","const events = [\"exit\", \"SIGINT\", \"SIGUSR1\", \"SIGUSR2\", \"uncaughtException\", \"SIGTERM\", \"SIGHUP\"];\n\nmodule.exports = function manageShutdownProcess(childrenList){\n\n    let shutting = false;\n    function handler(){\n        //console.log(\"Handling exit event on\", process.pid, \"arguments:\", arguments);\n        var childrenNames = Object.keys(childrenList);\n        for(let j=0; j<childrenNames.length; j++){\n            var child = childrenList[childrenNames[j]];\n            //console.log(`[${process.pid}]`, \"Sending kill signal to PID:\", child.pid);\n            try{\n                process.kill(child.pid);\n            }catch(err){\n                //...\n            }\n        }\n\n        if(!shutting){\n            try{\n                process.stdout.cursorTo(0);\n                process.stdout.write(`[PID: ${process.pid}] [Timestamp: ${new Date().getTime()}] [Process argv: ${process.argv}]- Shutting down...\\n`);\n            }catch(err)\n            {\n                //...\n            }\n            shutting = true;\n        }\n\n        setTimeout(()=>{\n            process.exit(0);\n        }, 0);\n    }\n\n    process.stdin.resume();\n    for(let i=0; i<events.length; i++){\n        var eventType = events[i];\n        process.on(eventType, handler);\n    }\n    //console.log(\"Exit handler setup!\", `[${process.pid}]`);\n};","\n//var path = require(\"path\");\nfunction defaultErrorHandlingImplementation(err, res){\n\t//console.log(err.stack);\n\tif(err) throw err;\n\treturn res;\n}\n\nrequire(\"./lib/overwriteRequire\");\nconst PSKBuffer = require('pskbuffer');\n$$.PSKBuffer = PSKBuffer;\n\n\n$$.errorHandler = {\n        error:function(err, args, msg){\n            console.log(err, \"Unknown error from function call with arguments:\", args, \"Message:\", msg);\n        },\n        throwError:function(err, args, msg){\n            console.log(err, \"Unknown error from function call with arguments:\", args, \"Message:\", msg);\n            throw err;\n        },\n        ignorePossibleError: function(name){\n            console.log(name);\n        },\n        syntaxError:function(property, swarm, text){\n            //throw new Error(\"Misspelled member name or other internal error!\");\n            var swarmName;\n            try{\n                if(typeof swarm == \"string\"){\n                    swarmName = swarm;\n                } else\n                if(swarm && swarm.meta){\n                    swarmName  = swarm.meta.swarmTypeName;\n                } else {\n                    swarmName = swarm.getInnerValue().meta.swarmTypeName;\n                }\n            } catch(err){\n                swarmName = err.toString();\n            }\n            if(property){\n                console.log(\"Wrong member name \", property,  \" in swarm \", swarmName);\n                if(text) {\n                    console.log(text);\n                }\n            } else {\n                console.log(\"Unknown swarm\", swarmName);\n            }\n\n        },\n        warning:function(msg){\n            console.log(msg);\n        }\n    };\n\n\n\n$$.safeErrorHandling = function(callback){\n        if(callback){\n            return callback;\n        } else{\n            return defaultErrorHandlingImplementation;\n        }\n    };\n\n$$.__intern = {\n        mkArgs:function(args,pos){\n            var argsArray = [];\n            for(var i = pos; i < args.length; i++){\n                argsArray.push(args[i]);\n            }\n            return argsArray;\n        }\n    };\n\n\n\nvar swarmUtils = require(\"./lib/choreographies/utilityFunctions/swarm\");\nvar assetUtils = require(\"./lib/choreographies/utilityFunctions/asset\");\n$$.defaultErrorHandlingImplementation = defaultErrorHandlingImplementation;\n\nvar callflowModule = require(\"./lib/swarmDescription\");\n$$.callflows        = callflowModule.createSwarmEngine(\"callflow\");\n$$.callflow         = $$.callflows;\n$$.flow             = $$.callflows;\n$$.flows            = $$.callflows;\n\n$$.swarms           = callflowModule.createSwarmEngine(\"swarm\", swarmUtils);\n$$.swarm            = $$.swarms;\n$$.contracts        = callflowModule.createSwarmEngine(\"contract\", swarmUtils);\n$$.contract         = $$.contracts;\n$$.assets           = callflowModule.createSwarmEngine(\"asset\", assetUtils);\n$$.asset            = $$.assets;\n$$.transactions     = callflowModule.createSwarmEngine(\"transaction\", swarmUtils);\n$$.transaction      = $$.transactions;\n\n\n$$.PSK_PubSub = require(\"soundpubsub\").soundPubSub;\n\n$$.securityContext = \"system\";\n$$.libraryPrefix = \"global\";\n$$.libraries = {\n    global:{\n\n    }\n};\n\n$$.interceptor = require(\"./lib/InterceptorRegistry\").createInterceptorRegistry();\n\n$$.loadLibrary = require(\"./lib/loadLibrary\").loadLibrary;\n\nrequireLibrary = function(name){\n    //var absolutePath = path.resolve(  $$.__global.__loadLibraryRoot + name);\n    return $$.loadLibrary(name,name);\n};\n\nrequire(\"./constants\");\n\n/*//TODO: SHOULD be moved in $$.__globals\n$$.ensureFolderExists = function (folder, callback) {\n    const flow = $$.flow.start(\"utils.mkDirRec\");\n    flow.make(folder, callback);\n};\n\n$$.ensureLinkExists = function (existingPath, newPath, callback) {\n    const flow = $$.flow.start(\"utils.mkDirRec\");\n    flow.makeLink(existingPath, newPath, callback);\n};*/\n\n$$.pathNormalize = function (pathToNormalize) {\n    const path = require(\"path\");\n    pathToNormalize = path.normalize(pathToNormalize);\n\n    return pathToNormalize.replace(/[\\/\\\\]/g, path.sep);\n};\n\nmodule.exports = {\n    \t\t\t\tcreateSwarmEngine: require(\"./lib/swarmDescription\").createSwarmEngine,\n                    createJoinPoint: require(\"./lib/parallelJoinPoint\").createJoinPoint,\n                    createSerialJoinPoint: require(\"./lib/serialJoinPoint\").createSerialJoinPoint,\n                    swarmInstanceManager: require(\"./lib/choreographies/swarmInstancesManager\"),\n                    enableInternalSwarmRouting: function(){\n                        function dummyVM(name){\n                            function solveSwarm(swarm){\n                                $$.swarmsInstancesManager.revive_swarm(swarm);\n                            }\n\n                            $$.PSK_PubSub.subscribe(name, solveSwarm);\n                            console.log(\"Creating a fake execution context...\");\n                        }\n                        dummyVM($$.CONSTANTS.SWARM_FOR_EXECUTION);\n                    }\n\t\t\t\t};\n","if(typeof singleton_container_module_workaround_for_wired_node_js_caching == 'undefined') {\n    singleton_container_module_workaround_for_wired_node_js_caching   = module;\n} else {\n    module.exports = singleton_container_module_workaround_for_wired_node_js_caching .exports;\n    return module;\n}\n\n/**\n * Created by salboaie on 4/27/15.\n */\nfunction Container(errorHandler){\n    var things = {};        //the actual values for our services, things\n    var immediate = {};     //how dependencies were declared\n    var callbacks = {};     //callback that should be called for each dependency declaration\n    var depsCounter = {};   //count dependencies\n    var reversedTree = {};  //reversed dependencies, opposite of immediate object\n\n     this.dump = function(){\n         console.log(\"Conatiner dump\\n Things:\", things, \"\\nDeps counter: \", depsCounter, \"\\nStright:\", immediate, \"\\nReversed:\", reversedTree);\n     };\n\n    function incCounter(name){\n        if(!depsCounter[name]){\n            depsCounter[name] = 1;\n        } else {\n            depsCounter[name]++;\n        }\n    }\n\n    function insertDependencyinRT(nodeName, dependencies){\n        dependencies.forEach(function(itemName){\n            var l = reversedTree[itemName];\n            if(!l){\n                l = reversedTree[itemName] = {};\n            }\n            l[nodeName] = nodeName;\n        });\n    }\n\n\n    function discoverUpNodes(nodeName){\n        var res = {};\n\n        function DFS(nn){\n            var l = reversedTree[nn];\n            for(var i in l){\n                if(!res[i]){\n                    res[i] = true;\n                    DFS(i);\n                }\n            }\n        }\n\n        DFS(nodeName);\n        return Object.keys(res);\n    }\n\n    function resetCounter(name){\n        var dependencyArray = immediate[name];\n        var counter = 0;\n        if(dependencyArray){\n            dependencyArray.forEach(function(dep){\n                if(things[dep] == null){\n                    incCounter(name);\n                    counter++;\n                }\n            });\n        }\n        depsCounter[name] = counter;\n        //console.log(\"Counter for \", name, ' is ', counter);\n        return counter;\n    }\n\n    /* returns those that are ready to be resolved*/\n    function resetUpCounters(name){\n        var ret = [];\n        //console.log('Reseting up counters for ', name, \"Reverse:\", reversedTree[name]);\n        var ups = reversedTree[name];\n        for(var v in ups){\n            if(resetCounter(v) === 0){\n                ret.push(v);\n            }\n        }\n        return ret;\n    }\n\n    /*\n         The first argument is a name for a service, variable,a  thing that should be initialised, recreated, etc\n         The second argument is an array with dependencies\n         the last argument is a function(err,...) that is called when dependencies are ready or recalled when are not ready (stop was called)\n         If err is not undefined it means that one or any undefined variables are not ready and the callback will be called again later\n         All the other arguments are the corresponding arguments of the callback will be the actual values of the corresponding dependency\n         The callback functions should return the current value (or null)\n     */\n    this.declareDependency = function(name, dependencyArray, callback){\n        if(callbacks[name]){\n            errorHandler.ignorePossibleError(\"Duplicate dependency:\" + name);\n        } else {\n            callbacks[name] = callback;\n            immediate[name]   = dependencyArray;\n            insertDependencyinRT(name, dependencyArray);\n            things[name] = null;\n        }\n\n        var unsatisfiedCounter = resetCounter(name);\n        if(unsatisfiedCounter === 0 ){\n            callForThing(name, false);\n        } else {\n            callForThing(name, true);\n        }\n    };\n\n\n    /*\n        create a service\n     */\n    this.service = function(name, dependencyArray, constructor){\n        this.declareDependency(name, dependencyArray, constructor);\n    };\n\n\n    var subsystemCounter = 0;\n    /*\n     create a anonymous subsystem\n     */\n    this.subsystem = function(dependencyArray, constructor){\n        subsystemCounter++;\n        this.declareDependency(\"dicontainer_subsystem_placeholder\" + subsystemCounter, dependencyArray, constructor);\n    };\n\n    /* not documented.. limbo state*/\n    this.factory = function(name, dependencyArray, constructor){\n        this.declareDependency(name, dependencyArray, function(){\n            return new constructor();\n        });\n    };\n\n    function callForThing(name, outOfService){\n        var args = immediate[name].map(function(item){\n            return things[item];\n        });\n        args.unshift(outOfService);\n        try{\n            var value = callbacks[name].apply({},args);\n        } catch(err){\n            errorHandler.throwError(err);\n        }\n\n\n        if(outOfService || value===null){   //enable returning a temporary dependency resolution!\n            if(things[name]){\n                things[name] = null;\n                resetUpCounters(name);\n            }\n        } else {\n            //console.log(\"Success resolving \", name, \":\", value, \"Other ready:\", otherReady);\n            if(!value){\n                value =  {\"placeholder\": name};\n            }\n            things[name] = value;\n            var otherReady = resetUpCounters(name);\n            otherReady.forEach(function(item){\n                callForThing(item, false);\n            });\n        }\n    }\n\n    /*\n        Declare that a name is ready, resolved and should try to resolve all other waiting for it\n     */\n    this.resolve    = function(name, value){\n        things[name] = value;\n        var otherReady = resetUpCounters(name);\n\n        otherReady.forEach(function(item){\n            callForThing(item, false);\n        });\n    };\n\n\n\n    this.instanceFactory = function(name, dependencyArray, constructor){\n        errorHandler.notImplemented(\"instanceFactory is planned but not implemented\");\n    };\n\n    /*\n        Declare that a service or feature is not working properly. All services depending on this will get notified\n     */\n    this.outOfService    = function(name){\n        things[name] = null;\n        var upNodes = discoverUpNodes(name);\n        upNodes.forEach(function(node){\n            things[name] = null;\n            callForThing(node, true);\n        });\n    };\n}\n\n\nexports.newContainer    = function(checksLibrary){\n    return new Container(checksLibrary);\n};\n\n//exports.container = new Container($$.errorHandler);","exports.domainPubSub = require(\"./domainPubSub\");","\n/**\n * Generic function used to registers methods such as asserts, logging, etc. on the current context.\n * @param name {String)} - name of the method (use case) to be registered.\n * @param func {Function} - handler to be invoked.\n * @param paramsDescription {Object} - parameters descriptions\n * @param after {Function} - callback function to be called after the function has been executed.\n */\nfunction addUseCase(name, func, paramsDescription, after){\n    var newFunc = func;\n    if(typeof after === \"function\") {\n        newFunc = function(){\n            const args = Array.from(arguments);\n            func.apply(this, args);\n            after();\n        };\n    }\n\n    // some properties should not be overridden\n    const protectedProperties = [ 'addCheck', 'addCase', 'register' ];\n    if(protectedProperties.indexOf(name) === -1){\n        this[name] = newFunc;\n    } else {\n        throw new Error('Cant overwrite ' + name);\n    }\n\n    if(paramsDescription){\n        this.params[name] = paramsDescription;\n    }\n}\n\n/**\n * Creates an alias to an existing function.\n * @param name1 {String} - New function name.\n * @param name2 {String} - Existing function name.\n */\nfunction alias(name1, name2){\n    this[name1] = this[name2];\n}\n\n/**\n * Singleton for adding various functions for use cases regarding logging.\n * @constructor\n */\nfunction LogsCore(){\n    this.params = {};\n}\n\n/**\n * Singleton for adding your various functions for asserts.\n * @constructor\n */\nfunction AssertCore(){\n    this.params = {};\n}\n\n/**\n * Singleton for adding your various functions for checks.\n * @constructor\n */\nfunction CheckCore(){\n    this.params = {};\n}\n\n/**\n * Singleton for adding your various functions for generating exceptions.\n * @constructor\n */\nfunction ExceptionsCore(){\n    this.params = {};\n}\n\n/**\n * Singleton for adding your various functions for running tests.\n * @constructor\n */\nfunction TestRunnerCore(){\n}\n\nLogsCore.prototype.addCase           = addUseCase;\nAssertCore.prototype.addCheck        = addUseCase;\nCheckCore.prototype.addCheck         = addUseCase;\nExceptionsCore.prototype.register    = addUseCase;\n\nLogsCore.prototype.alias             = alias;\nAssertCore.prototype.alias           = alias;\nCheckCore.prototype.alias            = alias;\nExceptionsCore.prototype.alias       = alias;\n\n// Create modules\nvar assertObj       = new AssertCore();\nvar checkObj        = new CheckCore();\nvar exceptionsObj   = new ExceptionsCore();\nvar loggerObj       = new LogsCore();\nvar testRunnerObj   = new TestRunnerCore();\n\n// Export modules\nexports.assert      = assertObj;\nexports.check       = checkObj;\nexports.exceptions  = exceptionsObj;\nexports.logger      = loggerObj;\nexports.testRunner  = testRunnerObj;\n\n// Initialise modules\nrequire(\"./standardAsserts.js\").init(exports, loggerObj);\nrequire(\"./standardLogs.js\").init(exports);\nrequire(\"./standardExceptions.js\").init(exports);\nrequire(\"./standardChecks.js\").init(exports);\nrequire(\"./testRunner.js\").init(exports);\n\n// Global Uncaught Exception handler.\nif(process.on)\n{\n    process.on('uncaughtException', function (err) {\n\t\tconst tag = \"uncaughtException\";\n\t\tconsole.log(tag, err);\n\t\tconsole.log(tag, err.stack);\n\t});\n}","module.exports = {\n\t\t\t\t\tcreateQue: require(\"./lib/folderMQ\").getFolderQueue\n\t\t\t\t\t//folderMQ: require(\"./lib/folderMQ\")\n};","//console.log(require.resolve(\"./components.js\"));\nmodule.exports = $$.library(function(){\n\trequire(\"./components.js\");\n\t/*require(\"./mkDirRec.js\");*/\n})","const PSKBuffer = require('./lib/PSKBuffer');\n\nmodule.exports = PSKBuffer;\n","const PskCrypto = require(\"./lib/PskCrypto\");\nconst ssutil = require(\"./signsensusDS/ssutil\");\n\nmodule.exports = PskCrypto;\n\nmodule.exports.hashValues = ssutil.hashValues;\n\nmodule.exports.PskArchiver = require(\"./lib/psk-archiver\");\n\nmodule.exports.DuplexStream = require(\"./lib/utils/DuplexStream\");\n\nmodule.exports.isStream = require(\"./lib/utils/isStream\");","module.exports = {\n\t\t\t\t\tsoundPubSub: require(\"./lib/soundPubSub\").soundPubSub\n};","module.exports.OwM = require(\"./lib/OwM\");\nmodule.exports.beesHealer = require(\"./lib/beesHealer\");\n\nconst uidGenerator = require(\"./lib/uidGenerator\").createUidGenerator(200, 32);\n\nmodule.exports.safe_uuid = require(\"./lib/safe-uuid\").init(uidGenerator);\n\nmodule.exports.Queue = require(\"./lib/Queue\");\nmodule.exports.combos = require(\"./lib/Combos\");\n\nmodule.exports.uidGenerator = uidGenerator;\nmodule.exports.generateUid = uidGenerator.generateUid;\n\nmodule.exports.createPskConsole = function () {\n  return require('./lib/pskconsole');\n};\n\n\nif(typeof global.$$ == \"undefined\"){\n  global.$$ = {};\n}\n\nif(typeof global.$$.uidGenerator == \"undefined\"){\n    $$.uidGenerator = module.exports.safe_uuid;\n}\n","exports.fsExt = require(\"./FSExtension\").fsExt;"]} diff --git a/psknode/bundles/readme.txt b/psknode/bundles/readme.txt deleted file mode 100644 index 9f2f528..0000000 --- a/psknode/bundles/readme.txt +++ /dev/null @@ -1 +0,0 @@ -privatesky standard bundles here \ No newline at end of file diff --git a/psknode/core/launcher.js b/psknode/core/launcher.js index 9af8238..147b92e 100644 --- a/psknode/core/launcher.js +++ b/psknode/core/launcher.js @@ -65,12 +65,15 @@ function launchDomainSandbox(name, configuration) { child_env.config = JSON.stringify(env.config); child_env.PRIVATESKY_TMP = process.env.PRIVATESKY_TMP; child_env.PRIVATESKY_ROOT_FOLDER = process.env.PRIVATESKY_ROOT_FOLDER; + const child = childProcess.fork('sandboxes/domainSandbox.js', [name], {cwd: __dirname, env: child_env}); child.on('exit', (code, signal) => { - console.log(`DomainSandbox [${name}] got an error code ${code}. Restarting...`); - delete domainSandboxes[name]; - launchDomainSandbox(name, configuration); - }); + setTimeout(()=>{ + console.log(`DomainSandbox [${name}] got an error code ${code}. Restarting...`); + delete domainSandboxes[name]; + launchDomainSandbox(name, configuration); + }, 100); + }); domainSandboxes[name] = child; } else { diff --git a/psknode/core/utils/exitHandler.js b/psknode/core/utils/exitHandler.js index 57f68fb..d08ce47 100644 --- a/psknode/core/utils/exitHandler.js +++ b/psknode/core/utils/exitHandler.js @@ -2,13 +2,29 @@ const events = ["exit", "SIGINT", "SIGUSR1", "SIGUSR2", "uncaughtException", "SI module.exports = function manageShutdownProcess(childrenList){ + let shutting = false; function handler(){ - console.log("Handling exit event on", process.pid, "arguments:", arguments); + //console.log("Handling exit event on", process.pid, "arguments:", arguments); var childrenNames = Object.keys(childrenList); for(let j=0; j{