diff --git a/.eslintignore b/.eslintignore
index 0d06625c..db694a0e 100644
--- a/.eslintignore
+++ b/.eslintignore
@@ -4,3 +4,4 @@ cache
 coverage
 typechain
 dist
+forks
diff --git a/.gitmodules b/.gitmodules
index 888d42dc..ed35e1f7 100644
--- a/.gitmodules
+++ b/.gitmodules
@@ -1,3 +1,9 @@
 [submodule "lib/forge-std"]
 	path = lib/forge-std
 	url = https://github.com/foundry-rs/forge-std
+[submodule "passport"]
+	path = passport
+	url = https://github.com/immutable/wallet-contracts
+[submodule "forks/seaport"]
+	path = forks/seaport
+	url = https://github.com/immutable/seaport
diff --git a/.prettierignore b/.prettierignore
index f268596e..c328d919 100644
--- a/.prettierignore
+++ b/.prettierignore
@@ -3,3 +3,4 @@ artifacts
 cache
 coverage*
 gasReporterOutput.json
+
diff --git a/.solhintignore b/.solhintignore
index 3c3629e6..793e901b 100644
--- a/.solhintignore
+++ b/.solhintignore
@@ -1 +1,4 @@
 node_modules
+contracts/passport
+contracts/seaport
+forks
\ No newline at end of file
diff --git a/contracts/passport/Factory.sol b/contracts/passport/Factory.sol
new file mode 100644
index 00000000..40f5c6ae
--- /dev/null
+++ b/contracts/passport/Factory.sol
@@ -0,0 +1,59 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+import '@openzeppelin/contracts/access/AccessControl.sol';
+import "./Wallet.sol";
+
+/**
+ * @title Factory
+ * @notice Factory contract to retrieve counterfactual wallet addresses and
+ * deploy new Sequence wallet instances to those addresses
+ */
+contract Factory is AccessControl {
+  // Role to deploy new wallets
+  bytes32 public constant DEPLOYER_ROLE = keccak256('DEPLOYER_ROLE');
+
+  event WalletDeployed(address indexed wallet, address indexed mainModule, bytes32 salt);
+
+  constructor(address _admin, address _deployer) {
+    _grantRole(DEFAULT_ADMIN_ROLE, _admin);
+    _grantRole(DEPLOYER_ROLE, _deployer);
+  }
+
+  /**
+   * @notice Returns a deterministic contract address given a salt
+   * @param _mainModule Address of the main module to be used by the wallet
+   * @param _salt Salt used to generate the address
+   * @return _address The deterministic address
+   */
+  function getAddress(address _mainModule, bytes32 _salt) external view returns (address _address) {
+    bytes32 _hash = keccak256(
+      abi.encodePacked(
+        bytes1(0xff),
+        address(this),
+        _salt,
+        keccak256(abi.encodePacked(Wallet.creationCode, uint256(uint160(_mainModule))))
+      )
+    );
+    return address(uint160(uint256(_hash)));
+  }
+
+  /**
+   * @notice Will deploy a new wallet instance using create2
+   * @param _mainModule Address of the main module to be used by the wallet
+   * @param _salt Salt used to generate the wallet, which is the imageHash
+   *       of the wallet's configuration.
+   * @dev It is recommended to not have more than 200 signers as opcode repricing
+   *      could make transactions impossible to execute as all the signers must be
+   *      passed for each transaction.
+   */
+  function deploy(address _mainModule, bytes32 _salt) external payable onlyRole(DEPLOYER_ROLE) returns (address _contract) {
+    bytes memory code = abi.encodePacked(Wallet.creationCode, uint256(uint160(_mainModule)));
+    assembly {
+      _contract := create2(callvalue(), add(code, 32), mload(code), _salt)
+    }
+    // check deployment success
+    require(_contract != address(0), 'WalletFactory: deployment failed');
+    // emit event, increases gas cost by ~2k
+    emit WalletDeployed(_contract, _mainModule, _salt);
+  }
+}
diff --git a/contracts/passport/IWalletProxy.sol b/contracts/passport/IWalletProxy.sol
new file mode 100644
index 00000000..02d88473
--- /dev/null
+++ b/contracts/passport/IWalletProxy.sol
@@ -0,0 +1,11 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity ^0.8.3;
+
+/**
+ * Interface that WalletProxy.yul implements.
+ */
+interface IWalletProxy {
+    /// @dev Retrieve current implementation contract used by proxy
+    function PROXY_getImplementation() external view returns (address implementation);
+}
\ No newline at end of file
diff --git a/contracts/passport/MultiCallDeploy.sol b/contracts/passport/MultiCallDeploy.sol
new file mode 100644
index 00000000..63a88450
--- /dev/null
+++ b/contracts/passport/MultiCallDeploy.sol
@@ -0,0 +1,72 @@
+// Copyright (c) Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./modules/commons/interfaces/IModuleCalls.sol";
+import '@openzeppelin/contracts/access/AccessControl.sol';
+import "./interfaces/IFactory.sol";
+
+/**
+ * @title MultiCallDeploy
+ * @notice This contract is bundles the wallet deployment and the users first write transaction into a single transaction.
+ *         Contract usage is intended for the submitter inside the relayer service, which will call either of the functions.
+ */
+contract MultiCallDeploy is AccessControl {
+    // Role to execute functions
+    bytes32 public constant EXECUTOR_ROLE = keccak256('EXECUTOR_ROLE');
+
+    constructor(address _admin, address _executor) {
+        _grantRole(DEFAULT_ADMIN_ROLE, _admin);
+        _grantRole(EXECUTOR_ROLE, _executor);
+    }
+
+    /*
+    * @dev Grants EXECUTOR_ROLE to an user.
+    * @param _executor Address that will be allowed to execute functions
+    */
+    function grantExecutorRole(address _executor) external onlyRole(DEFAULT_ADMIN_ROLE) {
+        _grantRole(EXECUTOR_ROLE, _executor);
+    }
+
+    /*
+    * @dev Deploy wallet and execute transaction.
+    * @param _mainModule Address of the main module to be used by the wallet
+    * @param _salt Salt used to generate the address
+    * @param factory address of the factory contract
+    * @param _txs transaction to execute
+    * @param _nonce nonce of the wallet
+    * @param _signature transaction signature from wallet
+    */
+    function deployExecute(address _mainModule, bytes32 _salt, address factory,  IModuleCalls.Transaction[] calldata _txs, uint256 _nonce, bytes calldata _signature) external onlyRole(EXECUTOR_ROLE) {
+        address ret = IFactory(factory).deploy(_mainModule, _salt);
+        IModuleCalls(ret).execute(_txs, _nonce, _signature);
+    }
+
+    /*
+    * @dev Handles deployment of wallet and transaction execution for both cases
+    * @param cfa counter factual address of the wallet
+    * @param _mainModule Address of the main module to be used by the wallet
+    * @param _salt Salt used to generate the address
+    * @param factory address of the factory contract
+    * @param _txs transaction to execute
+    * @param _nonce nonce of the wallet
+    * @param _signature transaction signature from wallet
+    */
+    function deployAndExecute(address cfa, address _mainModule, bytes32 _salt, address factory,  IModuleCalls.Transaction[] calldata _txs, uint256 _nonce, bytes calldata _signature) external onlyRole(EXECUTOR_ROLE){
+        // Get code size at CFA
+        uint32 size;
+        assembly {
+            size := extcodesize(cfa)
+        }
+
+        // If size is 0, deploy the proxy and execute write tx
+        // Else, execute the users transaction
+        if (size == 0) {
+            address ret = IFactory(factory).deploy(_mainModule, _salt);
+            require(cfa == ret, "MultiCallDeploy: deployed address does not match CFA");
+            IModuleCalls(ret).execute(_txs, _nonce, _signature);
+        } else {
+            IModuleCalls(cfa).execute(_txs, _nonce, _signature);
+        }
+    }
+}
\ No newline at end of file
diff --git a/contracts/passport/README.md b/contracts/passport/README.md
new file mode 100644
index 00000000..e74e141d
--- /dev/null
+++ b/contracts/passport/README.md
@@ -0,0 +1 @@
+DO NOT MODIFY THESE CONTRACTS DIRECTLY. This folder has been automatically extracted from https://github.com/immutable/wallet-contracts via a submodule in this repository's forks directory. See the upstream repository for full context.
\ No newline at end of file
diff --git a/contracts/passport/Wallet.sol b/contracts/passport/Wallet.sol
new file mode 100644
index 00000000..65532d0e
--- /dev/null
+++ b/contracts/passport/Wallet.sol
@@ -0,0 +1,10 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+// Holds the creation code of the WalletProxy.yul used by smart contract wallet instances.
+// Generate this bytecode using ./compileWalletProxyYul.sh
+library Wallet {
+    // This bytecode must precisely match that in tests/utils/helpers.ts
+    bytes internal constant creationCode = hex"6054600f3d396034805130553df3fe63906111273d3560e01c14602b57363d3d373d3d3d3d369030545af43d82803e156027573d90f35b3d90fd5b30543d5260203df3";
+}
diff --git a/contracts/passport/WalletProxy.yul b/contracts/passport/WalletProxy.yul
new file mode 100644
index 00000000..2bf1f8e9
--- /dev/null
+++ b/contracts/passport/WalletProxy.yul
@@ -0,0 +1,81 @@
+// Copyright (c) Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+//
+// This Yul code creates a minimalist transparent proxy with a function to fetch
+// the address of the contract being proxied to using the interface described in
+// IWalletProxy.sol .
+//
+object "ProxyGetImplYul" {
+    // This is the initcode of the contract.
+    code {
+        // Copy the runtime code plus the address of the implementation 
+        // parameter (32 bytes) which is appended to the end to memory.
+        // copy s bytes from code at position f to mem at position t
+        // codecopy(t, f, s)
+        // This will turn into a memory->memory copy for Ewasm and
+        // a codecopy for EVM
+        // The constant 0x54 is datasize("runtime") + 32. The solc compiler is
+        // unable to do constant addition as part of the compilation process, hence
+        // the constant.
+        // If the runtime code is to be updated, uncomment the following line, and comment
+        // out the following line, so that datasize("runtime") can be determined. It will
+        // be the byte following the 0x60 push1 opcode.
+//        datacopy(returndatasize(), dataoffset("runtime"), add(datasize("runtime"), 32))
+        datacopy(returndatasize(), dataoffset("runtime"), 0x54)
+
+        // Store the implementation address at the storage slot which is 
+        // equivalent to the deployed address of this contract.
+        let implAddress := mload(datasize("runtime"))
+        sstore(address(), implAddress)
+
+        // now return the runtime object (the currently
+        // executing code is the constructor code)
+        return(returndatasize(), datasize("runtime"))
+    }
+
+
+    // Code for deployed contract
+    object "runtime" {
+        code {
+            // Load the function selector (the first four bytes of calldata) by shifting the 
+            // word to the right. 
+            let selector := shr(224, calldataload(returndatasize()))
+
+            if eq(selector, 0x90611127) /* Function selector for "PROXY_getImplementation()" */ {
+                let impl := sload(address())
+                mstore(returndatasize(), impl)
+                return(returndatasize(), 0x20)
+            }
+
+            // Load calldata to memory location 0.
+            // Copy s bytes from calldata at position f to mem at position t
+            // calldatacopy(t, f, s)
+            calldatacopy(returndatasize(), returndatasize(), calldatasize())
+
+            // Use returndatasize to load zero.
+            let zero := returndatasize()
+
+            // Execute delegate call. Have outsize set to zero, to indicate
+            // don't return any data automatically.
+            // Call contract at address a with input mem[in…(in+insize)) 
+            // providing g gas and v wei and output area 
+            // mem[out…(out+outsize)) returning 0 on error 
+            // (eg. out of gas) and 1 on success
+            // delegatecall(g, a, in, insize, out, outsize)
+            // Use sload(address()) to load the implemntation address.
+            let success := delegatecall(gas(), sload(address()), returndatasize(), calldatasize(), returndatasize(), returndatasize())
+
+            // Copy the return result to memory location 0.
+            // Copy s bytes from returndata at position f to mem at position t
+            // returndatacopy(t, f, s)
+            returndatacopy(zero, zero, returndatasize())
+
+            // Return or revert: memory location 0 contains either the return value
+            // or the revert information.
+            if iszero(success) {
+                revert (zero,returndatasize())
+            }
+            return (zero,returndatasize())
+        }
+    }
+}
\ No newline at end of file
diff --git a/contracts/passport/interfaces/IERC1271Wallet.sol b/contracts/passport/interfaces/IERC1271Wallet.sol
new file mode 100644
index 00000000..794798d8
--- /dev/null
+++ b/contracts/passport/interfaces/IERC1271Wallet.sol
@@ -0,0 +1,38 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IERC1271Wallet {
+
+  /**
+   * @notice Verifies whether the provided signature is valid with respect to the provided data
+   * @dev MUST return the correct magic value if the signature provided is valid for the provided data
+   *   > The bytes4 magic value to return when signature is valid is 0x20c13b0b : bytes4(keccak256("isValidSignature(bytes,bytes)")
+   *   > This function MAY modify Ethereum's state
+   * @param _data       Arbitrary length data signed on the behalf of address(this)
+   * @param _signature  Signature byte array associated with _data
+   * @return magicValue Magic value 0x20c13b0b if the signature is valid and 0x0 otherwise
+   */
+  function isValidSignature(
+    bytes calldata _data,
+    bytes calldata _signature)
+    external
+    view
+    returns (bytes4 magicValue);
+
+  /**
+   * @notice Verifies whether the provided signature is valid with respect to the provided hash
+   * @dev MUST return the correct magic value if the signature provided is valid for the provided hash
+   *   > The bytes4 magic value to return when signature is valid is 0x20c13b0b : bytes4(keccak256("isValidSignature(bytes,bytes)")
+   *   > This function MAY modify Ethereum's state
+   * @param _hash       keccak256 hash that was signed
+   * @param _signature  Signature byte array associated with _data
+   * @return magicValue Magic value 0x20c13b0b if the signature is valid and 0x0 otherwise
+   */
+  function isValidSignature(
+    bytes32 _hash,
+    bytes calldata _signature)
+    external
+    view
+    returns (bytes4 magicValue);
+}
\ No newline at end of file
diff --git a/contracts/passport/interfaces/IFactory.sol b/contracts/passport/interfaces/IFactory.sol
new file mode 100644
index 00000000..94258ad0
--- /dev/null
+++ b/contracts/passport/interfaces/IFactory.sol
@@ -0,0 +1,29 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+/**
+ * @title IFactory
+ * @notice Factory interface to interact with wallet factory
+ */
+interface IFactory {
+  event WalletDeployed(address indexed wallet, address indexed mainModule, bytes32 salt);
+
+  /**
+   * @notice Returns a deterministic contract address given a salt
+   * @param _mainModule Address of the main module to be used by the wallet
+   * @param _salt Salt used to generate the address
+   * @return _address The deterministic address
+   */
+  function getAddress(address _mainModule, bytes32 _salt) external view returns (address);
+
+  /**
+   * @notice Will deploy a new wallet instance using create2
+   * @param _mainModule Address of the main module to be used by the wallet
+   * @param _salt Salt used to generate the wallet, which is the imageHash
+   *       of the wallet's configuration.
+   * @dev It is recommended to not have more than 200 signers as opcode repricing
+   *      could make transactions impossible to execute as all the signers must be
+   *      passed for each transaction.
+   */
+  function deploy(address _mainModule, bytes32 _salt) external payable returns (address);
+}
diff --git a/contracts/passport/interfaces/receivers/IERC1155Receiver.sol b/contracts/passport/interfaces/receivers/IERC1155Receiver.sol
new file mode 100644
index 00000000..f61d6466
--- /dev/null
+++ b/contracts/passport/interfaces/receivers/IERC1155Receiver.sol
@@ -0,0 +1,8 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IERC1155Receiver {
+  function onERC1155Received(address, address, uint256, uint256, bytes calldata) external returns (bytes4);
+  function onERC1155BatchReceived(address, address, uint256[] calldata, uint256[] calldata, bytes calldata) external returns (bytes4);
+}
diff --git a/contracts/passport/interfaces/receivers/IERC223Receiver.sol b/contracts/passport/interfaces/receivers/IERC223Receiver.sol
new file mode 100644
index 00000000..dc092c69
--- /dev/null
+++ b/contracts/passport/interfaces/receivers/IERC223Receiver.sol
@@ -0,0 +1,7 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IERC223Receiver {
+  function tokenFallback(address, uint256, bytes calldata) external;
+}
diff --git a/contracts/passport/interfaces/receivers/IERC721Receiver.sol b/contracts/passport/interfaces/receivers/IERC721Receiver.sol
new file mode 100644
index 00000000..495be6b1
--- /dev/null
+++ b/contracts/passport/interfaces/receivers/IERC721Receiver.sol
@@ -0,0 +1,7 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IERC721Receiver {
+  function onERC721Received(address, address, uint256, bytes calldata) external returns (bytes4);
+}
diff --git a/contracts/passport/migrations/Migrations.sol b/contracts/passport/migrations/Migrations.sol
new file mode 100644
index 00000000..03215be8
--- /dev/null
+++ b/contracts/passport/migrations/Migrations.sol
@@ -0,0 +1,26 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract Migrations {
+  address public owner;
+  uint public last_completed_migration;
+
+  constructor() public {
+    owner = msg.sender;
+  }
+
+  modifier restricted() {
+    if (msg.sender == owner)
+      _;
+  }
+
+  function setCompleted(uint completed) public restricted {
+    last_completed_migration = completed;
+  }
+
+  function upgrade(address new_address) public restricted {
+    Migrations upgraded = Migrations(new_address);
+    upgraded.setCompleted(last_completed_migration);
+  }
+}
diff --git a/contracts/passport/modules/GuestModule.sol b/contracts/passport/modules/GuestModule.sol
new file mode 100644
index 00000000..972bf598
--- /dev/null
+++ b/contracts/passport/modules/GuestModule.sol
@@ -0,0 +1,117 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../utils/SignatureValidator.sol";
+
+import "./commons/Implementation.sol";
+import "./commons/ModuleAuth.sol";
+import "./commons/ModuleHooks.sol";
+import "./commons/ModuleCalls.sol";
+import "./commons/ModuleUpdate.sol";
+import "./commons/ModuleCreator.sol";
+
+import "../interfaces/receivers/IERC1155Receiver.sol";
+import "../interfaces/receivers/IERC721Receiver.sol";
+
+import "../interfaces/IERC1271Wallet.sol";
+
+
+/**
+ * GuestModule implements an Arcadeum wallet without signatures, nonce or replay protection.
+ * executing transactions using this wallet is not an authenticated process, and can be done by any address.
+ *
+ * @notice This contract is completely public with no security, designed to execute pre-signed transactions
+ *   and use Arcadeum tools without using the wallets.
+ */
+contract GuestModule is
+  ModuleAuth,
+  ModuleCalls,
+  ModuleCreator
+{
+  /**
+   * @notice Allow any caller to execute an action
+   * @param _txs Transactions to process
+   */
+  function execute(
+    Transaction[] memory _txs,
+    uint256,
+    bytes memory
+  ) public override {
+    // Hash transaction bundle
+    bytes32 txHash = _subDigest(keccak256(abi.encode('guest:', _txs)));
+
+    // Execute the transactions
+    _executeGuest(txHash, _txs);
+  }
+
+  /**
+   * @notice Allow any caller to execute an action
+   * @param _txs Transactions to process
+   */
+  function selfExecute(
+    Transaction[] memory _txs
+  ) public override {
+    // Hash transaction bundle
+    bytes32 txHash = _subDigest(keccak256(abi.encode('self:', _txs)));
+
+    // Execute the transactions
+    _executeGuest(txHash, _txs);
+  }
+
+  /**
+   * @notice Executes a list of transactions
+   * @param _txHash  Hash of the batch of transactions
+   * @param _txs  Transactions to execute
+   */
+  function _executeGuest(
+    bytes32 _txHash,
+    Transaction[] memory _txs
+  ) private {
+    // Execute transaction
+    for (uint256 i = 0; i < _txs.length; i++) {
+      Transaction memory transaction = _txs[i];
+
+      bool success;
+      bytes memory result;
+
+      require(!transaction.delegateCall, 'GuestModule#_executeGuest: delegateCall not allowed');
+      require(gasleft() >= transaction.gasLimit, "GuestModule#_executeGuest: NOT_ENOUGH_GAS");
+
+      // solhint-disable
+      (success, result) = transaction.target.call{
+        value: transaction.value,
+        gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+      }(transaction.data);
+      // solhint-enable
+
+      if (success) {
+        emit TxExecuted(_txHash);
+      } else {
+        _revertBytes(transaction, _txHash, result);
+      }
+    }
+  }
+
+  /**
+   * @notice Validates any signature image, because the wallet is public and has now owner.
+   * @return true, all signatures are valid, false, no updates required
+   */
+  function _isValidImage(bytes32) internal override view returns (bool, bool) {
+    return (true, false);
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(
+    bytes4 _interfaceID
+  ) public override (
+    ModuleAuth,
+    ModuleCalls,
+    ModuleCreator
+  ) pure returns (bool) {
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/MainModule.sol b/contracts/passport/modules/MainModule.sol
new file mode 100644
index 00000000..e1b6f4dd
--- /dev/null
+++ b/contracts/passport/modules/MainModule.sol
@@ -0,0 +1,54 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../utils/SignatureValidator.sol";
+
+import "./commons/Implementation.sol";
+import "./commons/ModuleAuthFixed.sol";
+import "./commons/ModuleHooks.sol";
+import "./commons/ModuleCalls.sol";
+import "./commons/ModuleUpdate.sol";
+import "./commons/ModuleCreator.sol";
+
+import "../interfaces/receivers/IERC1155Receiver.sol";
+import "../interfaces/receivers/IERC721Receiver.sol";
+
+import "../interfaces/IERC1271Wallet.sol";
+
+
+/**
+ * @notice Contains the core functionality arcadeum wallets will inherit.
+ * @dev If using a new main module, developpers must ensure that all inherited
+ *      contracts by the mainmodule don't conflict and are accounted for to be
+ *      supported by the supportsInterface method.
+ */
+contract MainModule is
+  ModuleAuthFixed,
+  ModuleCalls,
+  ModuleUpdate,
+  ModuleHooks,
+  ModuleCreator
+{
+  constructor(
+    address _factory
+  ) public ModuleAuthFixed(
+    _factory
+  ) { }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(
+    bytes4 _interfaceID
+  ) public override(
+    ModuleAuth,
+    ModuleCalls,
+    ModuleUpdate,
+    ModuleHooks,
+    ModuleCreator
+  ) pure returns (bool) {
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/MainModuleDynamicAuth.sol b/contracts/passport/modules/MainModuleDynamicAuth.sol
new file mode 100644
index 00000000..149ce806
--- /dev/null
+++ b/contracts/passport/modules/MainModuleDynamicAuth.sol
@@ -0,0 +1,52 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./commons/ModuleAuthDynamic.sol";
+import "./commons/ModuleReceivers.sol";
+import "./commons/ModuleCalls.sol";
+import "./commons/ModuleUpdate.sol";
+
+
+/**
+ * TODO Peter update docs
+ * @notice Contains the core functionality arcadeum wallets will inherit with
+ *         the added functionality that the main-module can be changed.
+ * @dev If using a new main module, developpers must ensure that all inherited
+ *      contracts by the mainmodule don't conflict and are accounted for to be
+ *      supported by the supportsInterface method.
+ */
+contract MainModuleDynamicAuth is
+  ModuleAuthDynamic,
+  ModuleCalls,
+  ModuleReceivers,
+  ModuleUpdate
+{
+
+  // solhint-disable-next-line no-empty-blocks
+  constructor(address _factory, address _startup) ModuleAuthDynamic (_factory, _startup) { }
+
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @dev If using a new main module, developpers must ensure that all inherited
+   *      contracts by the mainmodule don't conflict and are accounted for to be
+   *      supported by the supportsInterface method.
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(
+    bytes4 _interfaceID
+  ) public override(
+    ModuleAuthUpgradable,
+    ModuleCalls,
+    ModuleReceivers,
+    ModuleUpdate
+  ) pure returns (bool) {
+    return super.supportsInterface(_interfaceID);
+  }
+
+  function version() external pure virtual returns (uint256) {
+    return 1;
+  }
+}
diff --git a/contracts/passport/modules/MainModuleGasEstimation.sol b/contracts/passport/modules/MainModuleGasEstimation.sol
new file mode 100644
index 00000000..f1fddbab
--- /dev/null
+++ b/contracts/passport/modules/MainModuleGasEstimation.sol
@@ -0,0 +1,93 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./commons/ModuleIgnoreAuthUpgradable.sol";
+import "./commons/ModuleIgnoreNonceCalls.sol";
+import "./commons/ModuleHooks.sol";
+import "./commons/ModuleUpdate.sol";
+import "./commons/ModuleCreator.sol";
+
+
+/**
+ * @notice Contains an alternative implementation of the MainModules that skips validation of
+ *   signatures, this implementation SHOULD NOT be used directly on a wallet.
+ *
+ *   Intended to be used only for gas estimation, using eth_call and overrides.
+ */
+contract MainModuleGasEstimation is
+  ModuleIgnoreAuthUpgradable,
+  ModuleIgnoreNonceCalls,
+  ModuleUpdate,
+  ModuleHooks,
+  ModuleCreator
+{
+  struct SimulateResult {
+    bool executed;
+    bool succeeded;
+    bytes result;
+    uint256 gasUsed;
+  }
+
+  /**
+   * @notice Simulate each transaction in a bundle for gas usage and execution result
+   * @param _txs Transactions to process
+   * @return The gas used and execution result for each transaction in the bundle
+   */
+  function simulateExecute(Transaction[] calldata _txs) public virtual returns (SimulateResult[] memory) {
+    SimulateResult[] memory results = new SimulateResult[](_txs.length);
+
+    // Execute transaction
+    for (uint256 i = 0; i < _txs.length; i++) {
+      Transaction memory transaction = _txs[i];
+
+      require(gasleft() >= transaction.gasLimit, "MainModuleGasEstimation#simulateExecute: NOT_ENOUGH_GAS");
+
+      results[i].executed = true;
+
+      if (transaction.delegateCall) {
+        uint256 initialGas = gasleft();
+
+        (results[i].succeeded, results[i].result) = transaction.target.delegatecall{
+          gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+        }(transaction.data);
+
+        results[i].gasUsed = initialGas - gasleft();
+      } else {
+        uint256 initialGas = gasleft();
+
+        (results[i].succeeded, results[i].result) = transaction.target.call{
+          value: transaction.value,
+          gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+        }(transaction.data);
+
+        results[i].gasUsed = initialGas - gasleft();
+      }
+
+      if (!results[i].succeeded && _txs[i].revertOnError) {
+        break;
+      }
+    }
+
+    return results;
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @dev If using a new main module, developpers must ensure that all inherited
+   *      contracts by the mainmodule don't conflict and are accounted for to be
+   *      supported by the supportsInterface method.
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(
+    bytes4 _interfaceID
+  ) public override(
+    ModuleIgnoreAuthUpgradable,
+    ModuleIgnoreNonceCalls,
+    ModuleUpdate,
+    ModuleHooks,
+    ModuleCreator
+  ) pure returns (bool) {
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/MainModuleUpgradable.sol b/contracts/passport/modules/MainModuleUpgradable.sol
new file mode 100644
index 00000000..588682da
--- /dev/null
+++ b/contracts/passport/modules/MainModuleUpgradable.sol
@@ -0,0 +1,44 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./commons/ModuleAuthUpgradable.sol";
+import "./commons/ModuleHooks.sol";
+import "./commons/ModuleCalls.sol";
+import "./commons/ModuleUpdate.sol";
+import "./commons/ModuleCreator.sol";
+
+
+/**
+ * @notice Contains the core functionality arcadeum wallets will inherit with
+ *         the added functionality that the main-module can be changed.
+ * @dev If using a new main module, developpers must ensure that all inherited
+ *      contracts by the mainmodule don't conflict and are accounted for to be
+ *      supported by the supportsInterface method.
+ */
+contract MainModuleUpgradable is
+  ModuleAuthUpgradable,
+  ModuleCalls,
+  ModuleUpdate,
+  ModuleHooks,
+  ModuleCreator
+{
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @dev If using a new main module, developpers must ensure that all inherited
+   *      contracts by the mainmodule don't conflict and are accounted for to be
+   *      supported by the supportsInterface method.
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(
+    bytes4 _interfaceID
+  ) public override(
+    ModuleAuthUpgradable,
+    ModuleCalls,
+    ModuleUpdate,
+    ModuleHooks,
+    ModuleCreator
+  ) pure returns (bool) {
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/commons/ImageHashKey.sol b/contracts/passport/modules/commons/ImageHashKey.sol
new file mode 100644
index 00000000..e1d9ec44
--- /dev/null
+++ b/contracts/passport/modules/commons/ImageHashKey.sol
@@ -0,0 +1,8 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+library ImageHashKey {
+  // Randomly generated to avoid collisions, with:
+  // xxd -len 32 -plain -cols 32 /dev/urandom
+  bytes32 internal constant IMAGE_HASH_KEY = bytes32(0xad348b32c79cd46ad46d61aede26d38affaee58f9a122f91eb271e08720464bf);
+}
diff --git a/contracts/passport/modules/commons/Implementation.sol b/contracts/passport/modules/commons/Implementation.sol
new file mode 100644
index 00000000..a1c65df9
--- /dev/null
+++ b/contracts/passport/modules/commons/Implementation.sol
@@ -0,0 +1,31 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+/**
+ * @dev Allows modules to access the implementation slot
+ */
+contract Implementation {
+  /**
+   * @notice Updates the Wallet implementation
+   * @param _imp New implementation address
+   * @dev The wallet implementation is stored on the storage slot
+   *   defined by the address of the wallet itself
+   *   WARNING updating this value may break the wallet and users
+   *   must be confident that the new implementation is safe.
+   */
+  function _setImplementation(address _imp) internal {
+    assembly {
+      sstore(address(), _imp)
+    }
+  }
+
+  /**
+   * @notice Returns the Wallet implementation
+   * @return _imp The address of the current Wallet implementation
+   */
+  function _getImplementation() internal view returns (address _imp) {
+    assembly {
+      _imp := sload(address())
+    }
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleAuth.sol b/contracts/passport/modules/commons/ModuleAuth.sol
new file mode 100644
index 00000000..25cdc1d9
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleAuth.sol
@@ -0,0 +1,228 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../../utils/LibBytes.sol";
+import "../../utils/SignatureValidator.sol";
+import "../../interfaces/IERC1271Wallet.sol";
+
+import "./interfaces/IModuleAuth.sol";
+
+import "./ModuleERC165.sol";
+
+
+abstract contract ModuleAuth is IModuleAuth, ModuleERC165, SignatureValidator, IERC1271Wallet {
+  using LibBytes for bytes;
+
+  uint256 private constant FLAG_SIGNATURE = 0;
+  uint256 private constant FLAG_ADDRESS = 1;
+  uint256 private constant FLAG_DYNAMIC_SIGNATURE = 2;
+
+  bytes4 private constant SELECTOR_ERC1271_BYTES_BYTES = 0x20c13b0b;
+  bytes4 private constant SELECTOR_ERC1271_BYTES32_BYTES = 0x1626ba7e;
+
+  /**
+   * @notice Verify if signer is default wallet owner
+   * @param _hash       Hashed signed message
+   * @param _signature  Array of signatures with signers ordered
+   *                    like the the keys in the multisig configs
+   *
+   * @dev The signature must be solidity packed and contain the total number of owners,
+   *      the threshold, the weight and either the address or a signature for each owner.
+   *
+   *      Each weight & (address or signature) pair is prefixed by a flag that signals if such pair
+   *      contains an address or a signature. The aggregated weight of the signatures must surpass the threshold.
+   *
+   *      Flag types:
+   *        0x00 - Signature
+   *        0x01 - Address
+   *
+   *      E.g:
+   *      abi.encodePacked(
+   *        uint16 threshold,
+   *        uint8 01,  uint8 weight_1, address signer_1,
+   *        uint8 00, uint8 weight_2, bytes signature_2,
+   *        ...
+   *        uint8 01,  uint8 weight_5, address signer_5
+   *      )
+   */
+  function _signatureValidation(
+    bytes32 _hash,
+    bytes memory _signature
+  )
+    internal override returns (bool)
+  {
+    (bool verified, bool needsUpdate, bytes32 imageHash) = _signatureValidationWithUpdateCheck(_hash, _signature);
+    if (needsUpdate) {
+      updateImageHashInternal(imageHash);
+    }
+    return verified;
+  }
+
+  function _signatureValidationInternal(
+    bytes32 _hash,
+    bytes memory _signature
+  )
+    internal view returns (bool)
+  {
+    (bool verified, , ) = _signatureValidationWithUpdateCheck(_hash, _signature);
+    return verified;
+  }
+
+
+
+  function _signatureValidationWithUpdateCheck(
+    bytes32 _hash,
+    bytes memory _signature
+  )
+    internal view returns (bool, bool, bytes32)
+  {
+    (
+      uint16 threshold,  // required threshold signature
+      uint256 rindex     // read index
+    ) = _signature.readFirstUint16();
+
+    // Start image hash generation
+    bytes32 imageHash = bytes32(uint256(threshold));
+
+    // Acumulated weight of signatures
+    uint256 totalWeight;
+
+    // Iterate until the image is completed
+    while (rindex < _signature.length) {
+      // Read next item type and addrWeight
+      uint256 flag; uint256 addrWeight; address addr;
+      (flag, addrWeight, rindex) = _signature.readUint8Uint8(rindex);
+
+      if (flag == FLAG_ADDRESS) {
+        // Read plain address
+        (addr, rindex) = _signature.readAddress(rindex);
+      } else if (flag == FLAG_SIGNATURE) {
+        // Read single signature and recover signer
+        bytes memory signature;
+        (signature, rindex) = _signature.readBytes66(rindex);
+        addr = recoverSigner(_hash, signature);
+
+        // Acumulate total weight of the signature
+        totalWeight += addrWeight;
+      } else if (flag == FLAG_DYNAMIC_SIGNATURE) {
+        // Read signer
+        (addr, rindex) = _signature.readAddress(rindex);
+
+        // Read signature size
+        uint256 size;
+        (size, rindex) = _signature.readUint16(rindex);
+
+        // Read dynamic size signature
+        bytes memory signature;
+        (signature, rindex) = _signature.readBytes(rindex, size);
+        require(isValidSignature(_hash, addr, signature), "ModuleAuth#_signatureValidation: INVALID_SIGNATURE");
+
+        // Acumulate total weight of the signature
+        totalWeight += addrWeight;
+      } else {
+        revert("ModuleAuth#_signatureValidation INVALID_FLAG");
+      }
+
+      // Write weight and address to image
+      imageHash = keccak256(abi.encode(imageHash, addrWeight, addr));
+    }
+
+    (bool verified, bool needsUpdate) = _isValidImage(imageHash);
+    return ((totalWeight >= threshold && verified), needsUpdate, imageHash);
+  }
+
+
+  /**
+   * @notice Validates the signature image
+   * @param _imageHash Hashed image of signature
+   * @return true if the signature image is valid, and true if the image hash should be updated.
+   */
+  function _isValidImage(bytes32 _imageHash) internal view virtual returns (bool, bool);
+
+  /**
+   * @notice Will hash _data to be signed (similar to EIP-712)
+   * @param _digest Pre-final digest
+   * @return hashed data for this wallet
+   */
+  function _subDigest(bytes32 _digest) internal override view returns (bytes32) {
+    uint256 chainId; assembly { chainId := chainid() }
+    return keccak256(
+      abi.encodePacked(
+        "\x19\x01",
+        chainId,
+        address(this),
+        _digest
+      )
+    );
+  }
+
+  /**
+   * @notice Updates the signers configuration of the wallet
+   * @param _imageHash New required image hash of the signature
+   * @dev It is recommended to not have more than 200 signers as opcode repricing
+   *      could make transactions impossible to execute as all the signers must be
+   *      passed for each transaction.
+   */
+  // solhint-disable-next-line no-empty-blocks
+  function updateImageHashInternal(bytes32 _imageHash) internal virtual {
+    // Default implementation does nothing
+  }
+  
+
+
+  /**
+   * @notice Verifies whether the provided signature is valid with respect to the provided data
+   * @dev MUST return the correct magic value if the signature provided is valid for the provided data
+   *   > The bytes4 magic value to return when signature is valid is 0x20c13b0b : bytes4(keccak256("isValidSignature(bytes,bytes)"))
+   * @param _data       Arbitrary length data signed on the behalf of address(this)
+   * @param _signatures Signature byte array associated with _data.
+   *                    Encoded as abi.encode(Signature[], Configs)
+   * @return magicValue Magic value 0x20c13b0b if the signature is valid and 0x0 otherwise
+   */
+  function isValidSignature(
+    bytes calldata _data,
+    bytes calldata _signatures
+  ) external override view returns (bytes4) {
+    // Validate signatures
+    if (_signatureValidationInternal(_subDigest(keccak256(_data)), _signatures)) {
+      return SELECTOR_ERC1271_BYTES_BYTES;
+    }
+    return 0;
+  }
+
+  /**
+   * @notice Verifies whether the provided signature is valid with respect to the provided hash
+   * @dev MUST return the correct magic value if the signature provided is valid for the provided hash
+   *   > The bytes4 magic value to return when signature is valid is 0x1626ba7e : bytes4(keccak256("isValidSignature(bytes32,bytes)"))
+   * @param _hash       keccak256 hash that was signed
+   * @param _signatures Signature byte array associated with _data.
+   *                    Encoded as abi.encode(Signature[], Configs)
+   * @return magicValue Magic value 0x1626ba7e if the signature is valid and 0x0 otherwise
+   */
+  function isValidSignature(
+    bytes32 _hash,
+    bytes calldata _signatures
+  ) external override view returns (bytes4) {
+    // Validate signatures
+    if (_signatureValidationInternal(_subDigest(_hash), _signatures)) {
+      return SELECTOR_ERC1271_BYTES32_BYTES;
+    }
+    return 0;
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (
+      _interfaceID == type(IModuleAuth).interfaceId ||
+      _interfaceID == type(IERC1271Wallet).interfaceId
+    ) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleAuthDynamic.sol b/contracts/passport/modules/commons/ModuleAuthDynamic.sol
new file mode 100644
index 00000000..505c4ed8
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleAuthDynamic.sol
@@ -0,0 +1,57 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./ModuleAuthUpgradable.sol";
+import "./ImageHashKey.sol";
+import "../../Wallet.sol";
+
+
+abstract contract ModuleAuthDynamic is ModuleAuthUpgradable {
+  bytes32 public immutable INIT_CODE_HASH;
+  address public immutable FACTORY;
+
+  constructor(address _factory, address _startupWalletImpl) {
+    // Build init code hash of the deployed wallets using that module
+    bytes32 initCodeHash = keccak256(abi.encodePacked(Wallet.creationCode, uint256(uint160(_startupWalletImpl))));
+
+    INIT_CODE_HASH = initCodeHash;
+    FACTORY = _factory;
+  }
+
+  /**
+   * @notice Validates the signature image with the salt used to deploy the contract
+   *         if there is no stored image hash. This will happen prior to the first meta 
+   *         transaction. Subsequently, validate the 
+   *         signature image with a valid image hash defined in the contract storage
+   * @param _imageHash Hash image of signature
+   * @return true if the signature image is valid, and true if the image hash needs to be updated
+   */
+  function _isValidImage(bytes32 _imageHash) internal view override returns (bool, bool) {
+    bytes32 storedImageHash = ModuleStorage.readBytes32(ImageHashKey.IMAGE_HASH_KEY);
+    if (storedImageHash == 0) {
+      // No image hash stored. Check that the image hash was used as the salt when 
+      // deploying the wallet proxy contract.
+      bool authenticated = address(
+        uint160(uint256(
+          keccak256(
+            abi.encodePacked(
+              bytes1(0xff),
+              FACTORY,
+              _imageHash,
+              INIT_CODE_HASH
+            )
+          )
+        ))
+      ) == address(this);
+      // Indicate need to update = true. This will trigger a call to store the image hash
+      return (authenticated, true);
+    }
+
+    // Image hash has been stored. 
+    return ((_imageHash != bytes32(0) && _imageHash == storedImageHash), false);
+  }
+}
+
+
+
diff --git a/contracts/passport/modules/commons/ModuleAuthFixed.sol b/contracts/passport/modules/commons/ModuleAuthFixed.sol
new file mode 100644
index 00000000..22cf41a4
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleAuthFixed.sol
@@ -0,0 +1,45 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./ModuleAuth.sol";
+import "../../Wallet.sol";
+
+/**
+ *  Implements ModuleAuth by validating the signature image against
+ *  the salt used to deploy the contract
+ *
+ *  This module allows wallets to be deployed with a default configuration
+ *  without using any aditional contract storage
+ */
+abstract contract ModuleAuthFixed is ModuleAuth {
+  bytes32 public immutable INIT_CODE_HASH;
+  address public immutable FACTORY;
+
+  constructor(address _factory) {
+    // Build init code hash of the deployed wallets using that module
+    bytes32 initCodeHash = keccak256(abi.encodePacked(Wallet.creationCode, uint256(uint160(address(this)))));
+
+    INIT_CODE_HASH = initCodeHash;
+    FACTORY = _factory;
+  }
+
+  /**
+   * @notice Validates the signature image with the salt used to deploy the contract
+   * @param _imageHash Hash image of signature
+   * @return true if the signature image is valid, and always false, indicating no updates required
+   */
+  function _isValidImage(bytes32 _imageHash) internal view override returns (bool, bool) {
+    return ((address(
+      uint160(uint256(
+        keccak256(
+          abi.encodePacked(
+            bytes1(0xff),
+            FACTORY,
+            _imageHash,
+            INIT_CODE_HASH
+          )
+        )
+      ))
+    ) == address(this)), false);
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleAuthUpgradable.sol b/contracts/passport/modules/commons/ModuleAuthUpgradable.sol
new file mode 100644
index 00000000..c3a80e95
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleAuthUpgradable.sol
@@ -0,0 +1,68 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./interfaces/IModuleAuthUpgradable.sol";
+
+import "./ModuleSelfAuth.sol";
+import "./ModuleAuth.sol";
+import "./ModuleStorage.sol";
+import "./ImageHashKey.sol";
+
+
+abstract contract ModuleAuthUpgradable is IModuleAuthUpgradable, ModuleAuth, ModuleSelfAuth {
+  event ImageHashUpdated(bytes32 newImageHash);
+
+  /**
+   * @notice Updates the signers configuration of the wallet
+   * @param _imageHash New required image hash of the signature
+   * @dev It is recommended to not have more than 200 signers as opcode repricing
+   *      could make transactions impossible to execute as all the signers must be
+   *      passed for each transaction.
+   */
+  function updateImageHash(bytes32 _imageHash) external override onlySelf {
+    updateImageHashInternal(_imageHash);
+  }
+
+  /**
+   * @notice Returns the current image hash of the wallet
+   */
+  function imageHash() external override view returns (bytes32) {
+    return ModuleStorage.readBytes32(ImageHashKey.IMAGE_HASH_KEY);
+  }
+
+  /**
+   * @notice Validates the signature image with a valid image hash defined
+   *   in the contract storage
+   * @param _imageHash Hash image of signature
+   * @return true if the signature image is valid, and always false indicating no updates required
+   */
+  function _isValidImage(bytes32 _imageHash) internal virtual view override returns (bool, bool) {
+    return ((_imageHash != bytes32(0) && _imageHash == ModuleStorage.readBytes32(ImageHashKey.IMAGE_HASH_KEY)), false);
+  }
+
+  /**
+   * @notice Updates the signers configuration of the wallet
+   * @param _imageHash New required image hash of the signature
+   * @dev It is recommended to not have more than 200 signers as opcode repricing
+   *      could make transactions impossible to execute as all the signers must be
+   *      passed for each transaction.
+   */
+  function updateImageHashInternal(bytes32 _imageHash) internal override {
+    require(_imageHash != bytes32(0), "ModuleAuthUpgradable#updateImageHash INVALID_IMAGE_HASH");
+    ModuleStorage.writeBytes32(ImageHashKey.IMAGE_HASH_KEY, _imageHash);
+    emit ImageHashUpdated(_imageHash);
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (_interfaceID == type(IModuleAuthUpgradable).interfaceId) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleCalls.sol b/contracts/passport/modules/commons/ModuleCalls.sol
new file mode 100644
index 00000000..3574c99e
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleCalls.sol
@@ -0,0 +1,192 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./ModuleSelfAuth.sol";
+import "./ModuleStorage.sol";
+import "./ModuleERC165.sol";
+import "./NonceKey.sol";
+
+import "./interfaces/IModuleCalls.sol";
+import "./interfaces/IModuleAuth.sol";
+
+
+abstract contract ModuleCalls is IModuleCalls, IModuleAuth, ModuleERC165, ModuleSelfAuth {
+  uint256 private constant NONCE_BITS = 96;
+  bytes32 private constant NONCE_MASK = bytes32((1 << NONCE_BITS) - 1);
+
+  /**
+   * @notice Returns the next nonce of the default nonce space
+   * @dev The default nonce space is 0x00
+   * @return The next nonce
+   */
+  function nonce() external override virtual view returns (uint256) {
+    return readNonce(0);
+  }
+
+  /**
+   * @notice Returns the next nonce of the given nonce space
+   * @param _space Nonce space, each space keeps an independent nonce count
+   * @return The next nonce
+   */
+  function readNonce(uint256 _space) public override virtual view returns (uint256) {
+    return uint256(ModuleStorage.readBytes32Map(NonceKey.NONCE_KEY, bytes32(_space)));
+  }
+
+  /**
+   * @notice Changes the next nonce of the given nonce space
+   * @param _space Nonce space, each space keeps an independent nonce count
+   * @param _nonce Nonce to write on the space
+   */
+  function _writeNonce(uint256 _space, uint256 _nonce) private {
+    ModuleStorage.writeBytes32Map(NonceKey.NONCE_KEY, bytes32(_space), bytes32(_nonce));
+  }
+
+  /**
+   * @notice Allow wallet owner to execute an action
+   * @dev Relayers must ensure that the gasLimit specified for each transaction
+   *      is acceptable to them. A user could specify large enough that it could
+   *      consume all the gas available.
+   * @param _txs        Transactions to process
+   * @param _nonce      Signature nonce (may contain an encoded space)
+   * @param _signature  Encoded signature
+   */
+  function execute(
+    Transaction[] memory _txs,
+    uint256 _nonce,
+    bytes memory _signature
+  ) public override virtual {
+    // Validate and update nonce
+    _validateNonce(_nonce);
+
+    // Hash transaction bundle
+    bytes32 txHash = _subDigest(keccak256(abi.encode(_nonce, _txs)));
+
+    // Verify that signatures are valid
+    require(
+      _signatureValidation(txHash, _signature),
+      "ModuleCalls#execute: INVALID_SIGNATURE"
+    );
+
+    // Execute the transactions
+    _execute(txHash, _txs);
+  }
+
+  /**
+   * @notice Allow wallet to execute an action
+   *   without signing the message
+   * @param _txs  Transactions to execute
+   */
+  function selfExecute(
+    Transaction[] memory _txs
+  ) public override virtual onlySelf {
+    // Hash transaction bundle
+    bytes32 txHash = _subDigest(keccak256(abi.encode('self:', _txs)));
+
+    // Execute the transactions
+    _execute(txHash, _txs);
+  }
+
+  /**
+   * @notice Executes a list of transactions
+   * @param _txHash  Hash of the batch of transactions
+   * @param _txs  Transactions to execute
+   */
+  function _execute(
+    bytes32 _txHash,
+    Transaction[] memory _txs
+  ) private {
+    // Execute transaction
+    for (uint256 i = 0; i < _txs.length; i++) {
+      Transaction memory transaction = _txs[i];
+
+      bool success;
+      bytes memory result;
+
+      require(gasleft() >= transaction.gasLimit, "ModuleCalls#_execute: NOT_ENOUGH_GAS");
+
+      if (transaction.delegateCall) {
+        (success, result) = transaction.target.delegatecall{
+          gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+        }(transaction.data);
+      } else {
+        (success, result) = transaction.target.call{
+          value: transaction.value,
+          gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+        }(transaction.data);
+      }
+
+      if (success) {
+        emit TxExecuted(_txHash);
+      } else {
+        _revertBytes(transaction, _txHash, result);
+      }
+    }
+  }
+
+  /**
+   * @notice Verify if a nonce is valid
+   * @param _rawNonce Nonce to validate (may contain an encoded space)
+   * @dev A valid nonce must be above the last one used
+   *   with a maximum delta of 100
+   */
+  function _validateNonce(uint256 _rawNonce) private {
+    // Retrieve current nonce for this wallet
+    (uint256 space, uint256 providedNonce) = _decodeNonce(_rawNonce);
+    uint256 currentNonce = readNonce(space);
+
+    // Verify if nonce is valid
+    require(
+      providedNonce == currentNonce,
+      "MainModule#_auth: INVALID_NONCE"
+    );
+
+    // Update signature nonce
+    uint256 newNonce = providedNonce + 1;
+    _writeNonce(space, newNonce);
+    emit NonceChange(space, newNonce);
+  }
+
+  /**
+   * @notice Logs a failed transaction, reverts if the transaction is not optional
+   * @param _tx      Transaction that is reverting
+   * @param _txHash  Hash of the transaction
+   * @param _reason  Encoded revert message
+   */
+  function _revertBytes(
+    Transaction memory _tx,
+    bytes32 _txHash,
+    bytes memory _reason
+  ) internal {
+    if (_tx.revertOnError) {
+      assembly { revert(add(_reason, 0x20), mload(_reason)) }
+    } else {
+      emit TxFailed(_txHash, _reason);
+    }
+  }
+
+  /**
+   * @notice Decodes a raw nonce
+   * @dev A raw nonce is encoded using the first 160 bits for the space
+   *  and the last 96 bits for the nonce
+   * @param _rawNonce Nonce to be decoded
+   * @return _space The nonce space of the raw nonce
+   * @return _nonce The nonce of the raw nonce
+   */
+  function _decodeNonce(uint256 _rawNonce) private pure returns (uint256 _space, uint256 _nonce) {
+    _nonce = uint256(bytes32(_rawNonce) & NONCE_MASK);
+    _space = _rawNonce >> NONCE_BITS;
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (_interfaceID == type(IModuleCalls).interfaceId) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleCreator.sol b/contracts/passport/modules/commons/ModuleCreator.sol
new file mode 100644
index 00000000..28b21784
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleCreator.sol
@@ -0,0 +1,36 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./interfaces/IModuleCreator.sol";
+
+import "./ModuleSelfAuth.sol";
+import "./ModuleERC165.sol";
+
+
+contract ModuleCreator is IModuleCreator, ModuleERC165, ModuleSelfAuth {
+  event CreatedContract(address _contract);
+
+  /**
+   * @notice Creates a contract forwarding eth value
+   * @param _code Creation code of the contract
+   * @return addr The address of the created contract
+   */
+  function createContract(bytes memory _code) public override payable onlySelf returns (address addr) {
+    assembly { addr := create(callvalue(), add(_code, 32), mload(_code)) }
+    require(addr != address(0), 'ModuleCreator: creation failed');
+    emit CreatedContract(addr);
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (_interfaceID == type(IModuleCreator).interfaceId) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleERC165.sol b/contracts/passport/modules/commons/ModuleERC165.sol
new file mode 100644
index 00000000..af033f1c
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleERC165.sol
@@ -0,0 +1,18 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+abstract contract ModuleERC165 {
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @dev Adding new hooks will not lead to them being reported by this function
+   *      without upgrading the wallet. In addition, developpers must ensure that 
+   *      all inherited contracts by the mainmodule don't conflict and are accounted
+   *      to be supported by the supportsInterface method.
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) virtual public pure returns (bool) {
+    return _interfaceID == this.supportsInterface.selector;
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleHooks.sol b/contracts/passport/modules/commons/ModuleHooks.sol
new file mode 100644
index 00000000..d3cb9859
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleHooks.sol
@@ -0,0 +1,146 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./interfaces/IModuleHooks.sol";
+
+import "./ModuleSelfAuth.sol";
+import "./ModuleStorage.sol";
+import "./ModuleERC165.sol";
+
+import "../../interfaces/receivers/IERC1155Receiver.sol";
+import "../../interfaces/receivers/IERC721Receiver.sol";
+import "../../interfaces/receivers/IERC223Receiver.sol";
+
+
+contract ModuleHooks is IERC1155Receiver, IERC721Receiver, IModuleHooks, ModuleERC165, ModuleSelfAuth {
+  // Randomly generated to avoid collisions, with:
+  // xxd -len 32 -plain -cols 32 /dev/urandom
+  bytes32 private constant HOOKS_KEY = bytes32(0x5f198cb61cfcc209f357b4ede4ad2218c53e3b4cb7e8fa1e8b0ec5e3951acbaa);
+
+  /**
+   * @notice Reads the implementation hook of a signature
+   * @param _signature Signature function
+   * @return The address of the implementation hook, address(0) if none
+  */
+  function readHook(bytes4 _signature) external override view returns (address) {
+    return _readHook(_signature);
+  }
+
+  /**
+   * @notice Adds a new hook to handle a given function selector
+   * @param _signature Signature function linked to the hook
+   * @param _implementation Hook implementation contract
+   * @dev Can't overwrite hooks that are part of the mainmodule (those defined below)
+   */
+  function addHook(bytes4 _signature, address _implementation) external override onlySelf {
+    require(_readHook(_signature) == address(0), "ModuleHooks#addHook: HOOK_ALREADY_REGISTERED");
+    _writeHook(_signature, _implementation);
+  }
+
+  /**
+   * @notice Removes a registered hook
+   * @param _signature Signature function linked to the hook
+   * @dev Can't remove hooks that are part of the mainmodule (those defined below) 
+   *      without upgrading the wallet
+   */
+  function removeHook(bytes4 _signature) external override onlySelf {
+    require(_readHook(_signature) != address(0), "ModuleHooks#removeHook: HOOK_NOT_REGISTERED");
+    _writeHook(_signature, address(0));
+  }
+
+  /**
+   * @notice Reads the implementation hook of a signature
+   * @param _signature Signature function
+   * @return The address of the implementation hook, address(0) if none
+  */
+  function _readHook(bytes4 _signature) private view returns (address) {
+    return address(uint160(uint256(ModuleStorage.readBytes32Map(HOOKS_KEY, _signature))));
+  }
+
+  /**
+   * @notice Writes the implementation hook of a signature
+   * @param _signature Signature function
+   * @param _implementation Hook implementation contract
+  */
+  function _writeHook(bytes4 _signature, address _implementation) private {
+    ModuleStorage.writeBytes32Map(HOOKS_KEY, _signature, bytes32(uint256(uint160(_implementation))));
+  }
+
+  /**
+   * @notice Handle the receipt of a single ERC1155 token type.
+   * @return `bytes4(keccak256("onERC1155Received(address,address,uint256,uint256,bytes)"))`
+   */
+  function onERC1155Received(
+    address,
+    address,
+    uint256,
+    uint256,
+    bytes calldata
+  ) external override returns (bytes4) {
+    return ModuleHooks.onERC1155Received.selector;
+  }
+
+  /**
+   * @notice Handle the receipt of multiple ERC1155 token types.
+   * @return `bytes4(keccak256("onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)"))`
+   */
+  function onERC1155BatchReceived(
+    address,
+    address,
+    uint256[] calldata,
+    uint256[] calldata,
+    bytes calldata
+  ) external override returns (bytes4) {
+    return ModuleHooks.onERC1155BatchReceived.selector;
+  }
+
+  /**
+   * @notice Handle the receipt of a single ERC721 token.
+   * @return `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))`
+   */
+  function onERC721Received(address, address, uint256, bytes calldata) external override returns (bytes4) {
+    return ModuleHooks.onERC721Received.selector;
+  }
+
+  /**
+   * @notice Routes fallback calls through hooks
+   */
+  fallback() external payable {
+      if (msg.data.length >= 4) {
+      address target = _readHook(msg.sig);
+      if (target != address(0)) {
+        (bool success, bytes memory result) = target.delegatecall(msg.data);
+        assembly {
+          if iszero(success)  {
+            revert(add(result, 0x20), mload(result))
+          }
+
+          return(add(result, 0x20), mload(result))
+        }
+      }
+    }
+  }
+
+  /**
+   * @notice Allows the wallet to receive ETH
+   */
+  receive() external payable { }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (
+      _interfaceID == type(IModuleHooks).interfaceId ||
+      _interfaceID == type(IERC1155Receiver).interfaceId ||
+      _interfaceID == type(IERC721Receiver).interfaceId ||
+      _interfaceID == type(IERC223Receiver).interfaceId
+    ) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleIgnoreAuthUpgradable.sol b/contracts/passport/modules/commons/ModuleIgnoreAuthUpgradable.sol
new file mode 100644
index 00000000..7198e42e
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleIgnoreAuthUpgradable.sol
@@ -0,0 +1,61 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./interfaces/IModuleAuthUpgradable.sol";
+
+import "./ModuleSelfAuth.sol";
+import "./ModuleAuth.sol";
+import "./ImageHashKey.sol";
+import "./ModuleStorage.sol";
+
+
+/**
+  @notice Implements ModuleAuthUpgradable but ignores the validity of the signature
+    should only be used during gas estimation.
+*/
+abstract contract ModuleIgnoreAuthUpgradable is IModuleAuthUpgradable, ModuleAuth, ModuleSelfAuth {
+  event ImageHashUpdated(bytes32 newImageHash);
+
+  /**
+   * @notice Updates the signers configuration of the wallet
+   * @param _imageHash New required image hash of the signature
+   * @dev It is recommended to not have more than 200 signers as opcode repricing
+   *      could make transactions impossible to execute as all the signers must be
+   *      passed for each transaction.
+   */
+  function updateImageHash(bytes32 _imageHash) external override virtual onlySelf {
+    require(_imageHash != bytes32(0), "ModuleAuthUpgradable#updateImageHash INVALID_IMAGE_HASH");
+    ModuleStorage.writeBytes32(ImageHashKey.IMAGE_HASH_KEY, _imageHash);
+    emit ImageHashUpdated(_imageHash);
+  }
+
+  /**
+   * @notice Returns the current image hash of the wallet
+   */
+  function imageHash() external override virtual view returns (bytes32) {
+    return ModuleStorage.readBytes32(ImageHashKey.IMAGE_HASH_KEY);
+  }
+
+  /**
+   * @notice Removes the signature validation from the module, by returning true for any _imageHash
+   * @param _imageHash Hash image of signature
+   * @return true always, hence always valid, and false always, indicating no update required
+   */
+  function _isValidImage(bytes32 _imageHash) internal override view returns (bool, bool) {
+    // Still validates the imageHash using the original mechanism for a more acurate estimation
+    return (true, false);
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (_interfaceID == type(IModuleAuthUpgradable).interfaceId) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleIgnoreNonceCalls.sol b/contracts/passport/modules/commons/ModuleIgnoreNonceCalls.sol
new file mode 100644
index 00000000..e3cbdb07
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleIgnoreNonceCalls.sol
@@ -0,0 +1,199 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./ModuleSelfAuth.sol";
+import "./ModuleStorage.sol";
+import "./ModuleERC165.sol";
+import "./NonceKey.sol";
+
+import "./interfaces/IModuleCalls.sol";
+import "./interfaces/IModuleAuth.sol";
+
+/**
+  @notice Implements ModuleCalls but ignores the validity of the nonce
+    should only be used during gas estimation.
+*/
+abstract contract ModuleIgnoreNonceCalls is IModuleCalls, IModuleAuth, ModuleERC165, ModuleSelfAuth {
+  uint256 private constant NONCE_BITS = 96;
+  bytes32 private constant NONCE_MASK = bytes32((1 << NONCE_BITS) - 1);
+
+  /**
+   * @notice Returns the next nonce of the default nonce space
+   * @dev The default nonce space is 0x00
+   * @return The next nonce
+   */
+  function nonce() external override virtual view returns (uint256) {
+    return readNonce(0);
+  }
+
+  /**
+   * @notice Returns the next nonce of the given nonce space
+   * @param _space Nonce space, each space keeps an independent nonce count
+   * @return The next nonce
+   */
+  function readNonce(uint256 _space) public override virtual view returns (uint256) {
+    return uint256(ModuleStorage.readBytes32Map(NonceKey.NONCE_KEY, bytes32(_space)));
+  }
+
+  /**
+   * @notice Changes the next nonce of the given nonce space
+   * @param _space Nonce space, each space keeps an independent nonce count
+   * @param _nonce Nonce to write on the space
+   */
+  function _writeNonce(uint256 _space, uint256 _nonce) private {
+    ModuleStorage.writeBytes32Map(NonceKey.NONCE_KEY, bytes32(_space), bytes32(_nonce));
+  }
+
+  /**
+   * @notice Allow wallet owner to execute an action
+   * @dev Relayers must ensure that the gasLimit specified for each transaction
+   *      is acceptable to them. A user could specify large enough that it could
+   *      consume all the gas available.
+   * @param _txs        Transactions to process
+   * @param _nonce      Signature nonce (may contain an encoded space)
+   * @param _signature  Encoded signature
+   */
+  function execute(
+    Transaction[] memory _txs,
+    uint256 _nonce,
+    bytes memory _signature
+  ) public override virtual {
+    // Validate and update nonce
+    _validateNonce(_nonce);
+
+    // Hash transaction bundle
+    bytes32 txHash = _subDigest(keccak256(abi.encode(_nonce, _txs)));
+
+    // Verify that signatures are valid
+    require(
+      _signatureValidation(txHash, _signature),
+      "ModuleCalls#execute: INVALID_SIGNATURE"
+    );
+
+    // Execute the transactions
+    _execute(txHash, _txs);
+  }
+
+  /**
+   * @notice Allow wallet to execute an action
+   *   without signing the message
+   * @param _txs  Transactions to execute
+   */
+  function selfExecute(
+    Transaction[] memory _txs
+  ) public override virtual onlySelf {
+    // Hash transaction bundle
+    bytes32 txHash = _subDigest(keccak256(abi.encode('self:', _txs)));
+
+    // Execute the transactions
+    _execute(txHash, _txs);
+  }
+
+  /**
+   * @notice Executes a list of transactions
+   * @param _txHash  Hash of the batch of transactions
+   * @param _txs  Transactions to execute
+   */
+  function _execute(
+    bytes32 _txHash,
+    Transaction[] memory _txs
+  ) private {
+    // Execute transaction
+    for (uint256 i = 0; i < _txs.length; i++) {
+      Transaction memory transaction = _txs[i];
+
+      bool success;
+      bytes memory result;
+
+      require(gasleft() >= transaction.gasLimit, "ModuleCalls#_execute: NOT_ENOUGH_GAS");
+
+      if (transaction.delegateCall) {
+        (success, result) = transaction.target.delegatecall{
+          gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+        }(transaction.data);
+      } else {
+        (success, result) = transaction.target.call{
+          value: transaction.value,
+          gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+        }(transaction.data);
+      }
+
+      if (success) {
+        emit TxExecuted(_txHash);
+      } else {
+        _revertBytes(transaction, _txHash, result);
+      }
+    }
+  }
+
+  /**
+   * @notice Verify if a nonce is valid
+   * @param _rawNonce Nonce to validate (may contain an encoded space)
+   * @dev A valid nonce must be above the last one used
+   *   with a maximum delta of 100
+   */
+  function _validateNonce(uint256 _rawNonce) private {
+    // Retrieve current nonce for this wallet
+    (uint256 space, uint256 providedNonce) = _decodeNonce(_rawNonce);
+    uint256 currentNonce = readNonce(space);
+
+    // Verify if nonce is valid
+    // 
+    // Skip nonce validation for gas estimation, but keep the statement to
+    // arrive at a closer gas expenditure when compared with the version in
+    // ModuleCalls.sol
+    require(
+      (providedNonce == currentNonce) || true,
+      "MainModule#_auth: INVALID_NONCE"
+    );
+
+    // Update signature nonce
+    uint256 newNonce = providedNonce + 1;
+    _writeNonce(space, newNonce);
+    emit NonceChange(space, newNonce);
+  }
+
+  /**
+   * @notice Logs a failed transaction, reverts if the transaction is not optional
+   * @param _tx      Transaction that is reverting
+   * @param _txHash  Hash of the transaction
+   * @param _reason  Encoded revert message
+   */
+  function _revertBytes(
+    Transaction memory _tx,
+    bytes32 _txHash,
+    bytes memory _reason
+  ) internal {
+    if (_tx.revertOnError) {
+      assembly { revert(add(_reason, 0x20), mload(_reason)) }
+    } else {
+      emit TxFailed(_txHash, _reason);
+    }
+  }
+
+  /**
+   * @notice Decodes a raw nonce
+   * @dev A raw nonce is encoded using the first 160 bits for the space
+   *  and the last 96 bits for the nonce
+   * @param _rawNonce Nonce to be decoded
+   * @return _space The nonce space of the raw nonce
+   * @return _nonce The nonce of the raw nonce
+   */
+  function _decodeNonce(uint256 _rawNonce) private pure returns (uint256 _space, uint256 _nonce) {
+    _nonce = uint256(bytes32(_rawNonce) & NONCE_MASK);
+    _space = _rawNonce >> NONCE_BITS;
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (_interfaceID == type(IModuleCalls).interfaceId) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleReceivers.sol b/contracts/passport/modules/commons/ModuleReceivers.sol
new file mode 100644
index 00000000..27788146
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleReceivers.sol
@@ -0,0 +1,68 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./ModuleERC165.sol";
+
+import "../../interfaces/receivers/IERC1155Receiver.sol";
+import "../../interfaces/receivers/IERC721Receiver.sol";
+import "../../interfaces/receivers/IERC223Receiver.sol";
+
+contract ModuleReceivers is IERC1155Receiver, IERC721Receiver, ModuleERC165 {
+  /**
+   * @notice Handle the receipt of a single ERC1155 token type.
+   * @return `bytes4(keccak256("onERC1155Received(address,address,uint256,uint256,bytes)"))`
+   */
+  function onERC1155Received(
+    address,
+    address,
+    uint256,
+    uint256,
+    bytes calldata
+  ) external override returns (bytes4) {
+    return ModuleReceivers.onERC1155Received.selector;
+  }
+
+  /**
+   * @notice Handle the receipt of multiple ERC1155 token types.
+   * @return `bytes4(keccak256("onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)"))`
+   */
+  function onERC1155BatchReceived(
+    address,
+    address,
+    uint256[] calldata,
+    uint256[] calldata,
+    bytes calldata
+  ) external override returns (bytes4) {
+    return ModuleReceivers.onERC1155BatchReceived.selector;
+  }
+
+  /**
+   * @notice Handle the receipt of a single ERC721 token.
+   * @return `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))`
+   */
+  function onERC721Received(address, address, uint256, bytes calldata) external override returns (bytes4) {
+    return ModuleReceivers.onERC721Received.selector;
+  }
+
+  /**
+   * @notice Allows the wallet to receive ETH
+   */
+  receive() external payable { }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (
+      _interfaceID == type(IERC1155Receiver).interfaceId ||
+      _interfaceID == type(IERC721Receiver).interfaceId ||
+      _interfaceID == type(IERC223Receiver).interfaceId
+    ) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleSelfAuth.sol b/contracts/passport/modules/commons/ModuleSelfAuth.sol
new file mode 100644
index 00000000..9a59d7b2
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleSelfAuth.sol
@@ -0,0 +1,10 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract ModuleSelfAuth {
+  modifier onlySelf() {
+    require(msg.sender == address(this), "ModuleSelfAuth#onlySelf: NOT_AUTHORIZED");
+    _;
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleStorage.sol b/contracts/passport/modules/commons/ModuleStorage.sol
new file mode 100644
index 00000000..ad98b9b4
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleStorage.sol
@@ -0,0 +1,23 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+library ModuleStorage {
+  function writeBytes32(bytes32 _key, bytes32 _val) internal {
+    assembly { sstore(_key, _val) }
+  }
+
+  function readBytes32(bytes32 _key) internal view returns (bytes32 val) {
+    assembly { val := sload(_key) }
+  }
+
+  function writeBytes32Map(bytes32 _key, bytes32 _subKey, bytes32 _val) internal {
+    bytes32 key = keccak256(abi.encode(_key, _subKey));
+    assembly { sstore(key, _val) }
+  }
+
+  function readBytes32Map(bytes32 _key, bytes32 _subKey) internal view returns (bytes32 val) {
+    bytes32 key = keccak256(abi.encode(_key, _subKey));
+    assembly { val := sload(key) }
+  }
+}
diff --git a/contracts/passport/modules/commons/ModuleUpdate.sol b/contracts/passport/modules/commons/ModuleUpdate.sol
new file mode 100644
index 00000000..044e3ef4
--- /dev/null
+++ b/contracts/passport/modules/commons/ModuleUpdate.sol
@@ -0,0 +1,41 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./interfaces/IModuleUpdate.sol";
+
+import "./Implementation.sol";
+import "./ModuleSelfAuth.sol";
+import "./ModuleERC165.sol";
+
+import "../../utils/LibAddress.sol";
+
+
+contract ModuleUpdate is IModuleUpdate, ModuleERC165, ModuleSelfAuth, Implementation {
+  using LibAddress for address;
+
+  event ImplementationUpdated(address newImplementation);
+
+  /**
+   * @notice Updates the implementation of the base wallet
+   * @param _implementation New main module implementation
+   * @dev WARNING Updating the implementation can brick the wallet
+   */
+  function updateImplementation(address _implementation) external override onlySelf {
+    require(_implementation.isContract(), "ModuleUpdate#updateImplementation: INVALID_IMPLEMENTATION");
+    _setImplementation(_implementation);
+    emit ImplementationUpdated(_implementation);
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (_interfaceID == type(IModuleUpdate).interfaceId) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/contracts/passport/modules/commons/NonceKey.sol b/contracts/passport/modules/commons/NonceKey.sol
new file mode 100644
index 00000000..64539afe
--- /dev/null
+++ b/contracts/passport/modules/commons/NonceKey.sol
@@ -0,0 +1,9 @@
+
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+library NonceKey {
+  // Randomly generated to avoid collisions, with:
+  // xxd -len 32 -plain -cols 32 /dev/urandom
+  bytes32 internal constant NONCE_KEY = bytes32(0xc40e2218089ef03fc40794d84d38778f688da53b98c9236b084936bfafc9a601);
+}
diff --git a/contracts/passport/modules/commons/interfaces/IModuleAuth.sol b/contracts/passport/modules/commons/interfaces/IModuleAuth.sol
new file mode 100644
index 00000000..233ac2c7
--- /dev/null
+++ b/contracts/passport/modules/commons/interfaces/IModuleAuth.sol
@@ -0,0 +1,25 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+abstract contract IModuleAuth {
+  /**
+   * @notice Hashed _data to be signed
+   * @param _digest Pre-final digest
+   * @return hashed data for this wallet
+   */
+  function _subDigest(
+    bytes32 _digest
+  ) internal virtual view returns (bytes32);
+
+  /**
+   * @notice Verify if signer is default wallet owner
+   * @param _hash Hashed signed message
+   * @param _signature Encoded signature
+   * @return True is the signature is valid
+   */
+  function _signatureValidation(
+    bytes32 _hash,
+    bytes memory _signature
+  ) internal virtual returns (bool);
+}
diff --git a/contracts/passport/modules/commons/interfaces/IModuleAuthUpgradable.sol b/contracts/passport/modules/commons/interfaces/IModuleAuthUpgradable.sol
new file mode 100644
index 00000000..f26ce97f
--- /dev/null
+++ b/contracts/passport/modules/commons/interfaces/IModuleAuthUpgradable.sol
@@ -0,0 +1,16 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IModuleAuthUpgradable {
+  /**
+   * @notice Updates the signers configuration of the wallet
+   * @param _imageHash New required image hash of the signature
+   */
+  function updateImageHash(bytes32 _imageHash) external;
+
+  /**
+   * @notice Returns the current image hash of the wallet
+   */
+  function imageHash() external view returns (bytes32);
+}
diff --git a/contracts/passport/modules/commons/interfaces/IModuleCalls.sol b/contracts/passport/modules/commons/interfaces/IModuleCalls.sol
new file mode 100644
index 00000000..ad0e78f4
--- /dev/null
+++ b/contracts/passport/modules/commons/interfaces/IModuleCalls.sol
@@ -0,0 +1,55 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IModuleCalls {
+  // Events
+  event NonceChange(uint256 _space, uint256 _newNonce);
+  event TxFailed(bytes32 _tx, bytes _reason);
+  event TxExecuted(bytes32 _tx) anonymous;
+
+  // Transaction structure
+  struct Transaction {
+    bool delegateCall;   // Performs delegatecall
+    bool revertOnError;  // Reverts transaction bundle if tx fails
+    uint256 gasLimit;    // Maximum gas to be forwarded
+    address target;      // Address of the contract to call
+    uint256 value;       // Amount of ETH to pass with the call
+    bytes data;          // calldata to pass
+  }
+
+  /**
+   * @notice Returns the next nonce of the default nonce space
+   * @dev The default nonce space is 0x00
+   * @return The next nonce
+   */
+  function nonce() external view returns (uint256);
+
+  /**
+   * @notice Returns the next nonce of the given nonce space
+   * @param _space Nonce space, each space keeps an independent nonce count
+   * @return The next nonce
+   */
+  function readNonce(uint256 _space) external view returns (uint256);
+
+  /**
+   * @notice Allow wallet owner to execute an action
+   * @param _txs        Transactions to process
+   * @param _nonce      Signature nonce (may contain an encoded space)
+   * @param _signature  Encoded signature
+   */
+  function execute(
+    Transaction[] calldata _txs,
+    uint256 _nonce,
+    bytes calldata _signature
+  ) external;
+
+  /**
+   * @notice Allow wallet to execute an action
+   *   without signing the message
+   * @param _txs  Transactions to execute
+   */
+  function selfExecute(
+    Transaction[] calldata _txs
+  ) external;
+}
diff --git a/contracts/passport/modules/commons/interfaces/IModuleCreator.sol b/contracts/passport/modules/commons/interfaces/IModuleCreator.sol
new file mode 100644
index 00000000..63c490ec
--- /dev/null
+++ b/contracts/passport/modules/commons/interfaces/IModuleCreator.sol
@@ -0,0 +1,12 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IModuleCreator {
+  /**
+   * @notice Creates a contract forwarding eth value
+   * @param _code Creation code of the contract
+   * @return addr The address of the created contract
+   */
+  function createContract(bytes calldata _code) external payable returns (address addr);
+}
diff --git a/contracts/passport/modules/commons/interfaces/IModuleHooks.sol b/contracts/passport/modules/commons/interfaces/IModuleHooks.sol
new file mode 100644
index 00000000..5ab2ae56
--- /dev/null
+++ b/contracts/passport/modules/commons/interfaces/IModuleHooks.sol
@@ -0,0 +1,25 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IModuleHooks {
+  /**
+   * @notice Reads the implementation hook of a signature
+   * @param _signature Signature function
+   * @return The address of the implementation hook, address(0) if none
+  */
+  function readHook(bytes4 _signature) external view returns (address);
+
+  /**
+   * @notice Adds a new hook to handle a given function selector
+   * @param _signature Signature function linked to the hook
+   * @param _implementation Hook implementation contract
+   */
+  function addHook(bytes4 _signature, address _implementation) external;
+
+  /**
+   * @notice Removes a registered hook
+   * @param _signature Signature function linked to the hook
+   */
+  function removeHook(bytes4 _signature) external;
+}
diff --git a/contracts/passport/modules/commons/interfaces/IModuleUpdate.sol b/contracts/passport/modules/commons/interfaces/IModuleUpdate.sol
new file mode 100644
index 00000000..ed06243a
--- /dev/null
+++ b/contracts/passport/modules/commons/interfaces/IModuleUpdate.sol
@@ -0,0 +1,12 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IModuleUpdate {
+  /**
+   * @notice Updates the implementation of the base wallet
+   * @param _implementation New main module implementation
+   * @dev WARNING Updating the implementation can brick the wallet
+   */
+  function updateImplementation(address _implementation) external;
+}
diff --git a/contracts/passport/modules/utils/GasEstimator.sol b/contracts/passport/modules/utils/GasEstimator.sol
new file mode 100644
index 00000000..4447868e
--- /dev/null
+++ b/contracts/passport/modules/utils/GasEstimator.sol
@@ -0,0 +1,16 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract GasEstimator {
+  function estimate(
+    address _to,
+    bytes calldata _data
+  ) external returns (bool success, bytes memory result, uint256 gas) {
+    // solhint-disable
+    uint256 initialGas = gasleft();
+    (success, result) = _to.call(_data);
+    gas = initialGas - gasleft();
+    // solhint-enable
+  }
+}
diff --git a/contracts/passport/modules/utils/MultiCallUtils.sol b/contracts/passport/modules/utils/MultiCallUtils.sol
new file mode 100644
index 00000000..bb7d3174
--- /dev/null
+++ b/contracts/passport/modules/utils/MultiCallUtils.sol
@@ -0,0 +1,99 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../commons/interfaces/IModuleCalls.sol";
+
+
+contract MultiCallUtils {
+  function multiCall(
+    IModuleCalls.Transaction[] memory _txs
+  ) public payable returns (
+    bool[] memory _successes,
+    bytes[] memory _results
+  ) {
+    _successes = new bool[](_txs.length);
+    _results = new bytes[](_txs.length);
+
+    for (uint256 i = 0; i < _txs.length; i++) {
+      IModuleCalls.Transaction memory transaction = _txs[i];
+
+      require(!transaction.delegateCall, 'MultiCallUtils#multiCall: delegateCall not allowed');
+      require(gasleft() >= transaction.gasLimit, "MultiCallUtils#multiCall: NOT_ENOUGH_GAS");
+
+      // solhint-disable
+      (_successes[i], _results[i]) = transaction.target.call{
+        value: transaction.value,
+        gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+      }(transaction.data);
+      // solhint-enable
+
+      require(_successes[i] || !_txs[i].revertOnError, 'MultiCallUtils#multiCall: CALL_REVERTED');
+    }
+  }
+
+  // ///
+  // Globals
+  // ///
+
+  function callBlockhash(uint256 _i) external view returns (bytes32) {
+    return blockhash(_i);
+  }
+
+  function callCoinbase() external view returns (address) {
+    return block.coinbase;
+  }
+
+  function callDifficulty() external view returns (uint256) {
+    return block.difficulty;
+  }
+
+  function callGasLimit() external view returns (uint256) {
+    return block.gaslimit;
+  }
+
+  function callBlockNumber() external view returns (uint256) {
+    return block.number;
+  }
+
+  function callTimestamp() external view returns (uint256) {
+    return block.timestamp;
+  }
+
+  function callGasLeft() external view returns (uint256) {
+    return gasleft();
+  }
+
+  function callGasPrice() external view returns (uint256) {
+    return tx.gasprice;
+  }
+
+  function callOrigin() external view returns (address) {
+    return tx.origin;
+  }
+
+  function callBalanceOf(address _addr) external view returns (uint256) {
+    return _addr.balance;
+  }
+
+  function callCodeSize(address _addr) external view returns (uint256 size) {
+    assembly { size := extcodesize(_addr) }
+  }
+
+  function callCode(address _addr) external view returns (bytes memory code) {
+    assembly {
+      let size := extcodesize(_addr)
+      code := mload(0x40)
+      mstore(0x40, add(code, and(add(add(size, 0x20), 0x1f), not(0x1f))))
+      mstore(code, size)
+      extcodecopy(_addr, add(code, 0x20), 0, size)
+    }
+  }
+
+  function callCodeHash(address _addr) external view returns (bytes32 codeHash) {
+    assembly { codeHash := extcodehash(_addr) }
+  }
+
+  function callChainId() external view returns (uint256 id) {
+    assembly { id := chainid() }
+  }
+}
diff --git a/contracts/passport/modules/utils/RequireUtils.sol b/contracts/passport/modules/utils/RequireUtils.sol
new file mode 100644
index 00000000..fa285d19
--- /dev/null
+++ b/contracts/passport/modules/utils/RequireUtils.sol
@@ -0,0 +1,281 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../commons/interfaces/IModuleCalls.sol";
+import "../commons/interfaces/IModuleAuthUpgradable.sol";
+import "../../interfaces/IERC1271Wallet.sol";
+import "../../utils/SignatureValidator.sol";
+import "../../utils/LibBytes.sol";
+import "../../Wallet.sol";
+
+contract RequireUtils is SignatureValidator {
+  using LibBytes for bytes;
+
+  uint256 private constant NONCE_BITS = 96;
+  bytes32 private constant NONCE_MASK = bytes32((1 << NONCE_BITS) - 1);
+
+  uint256 private constant FLAG_SIGNATURE = 0;
+  uint256 private constant FLAG_ADDRESS = 1;
+  uint256 private constant FLAG_DYNAMIC_SIGNATURE = 2;
+
+  bytes32 private immutable INIT_CODE_HASH;
+  address private immutable FACTORY;
+
+  struct Member {
+    uint256 weight;
+    address signer;
+  }
+
+  event RequiredConfig(
+    address indexed _wallet,
+    bytes32 indexed _imageHash,
+    uint256 _threshold,
+    bytes _signers
+  );
+
+  event RequiredSigner(
+    address indexed _wallet,
+    address indexed _signer
+  );
+
+  mapping(address => uint256) public lastSignerUpdate;
+  mapping(address => uint256) public lastWalletUpdate;
+  mapping(address => bytes32) public knownImageHashes;
+  mapping(bytes32 => uint256) public lastImageHashUpdate;
+
+  constructor(address _factory, address _mainModule) public {
+    FACTORY = _factory;
+    INIT_CODE_HASH = keccak256(abi.encodePacked(Wallet.creationCode, uint256(uint160(_mainModule))));
+  }
+
+  /**
+   * @notice Publishes the current configuration of a Sequence wallets using logs
+   * @dev Used for fast lookup of a wallet configuration based on its image-hash, compatible with updated and counter-factual wallets.
+   *
+   * @param _wallet      Sequence wallet
+   * @param _threshold   Thershold of the current configuration
+   * @param _members     Members of the current configuration
+   * @param _index       True if an index in contract-storage is desired 
+   */
+  function publishConfig(
+    address _wallet,
+    uint256 _threshold,
+    Member[] calldata _members,
+    bool _index
+  ) external {
+    // Compute expected imageHash
+    bytes32 imageHash = bytes32(uint256(_threshold));
+    for (uint256 i = 0; i < _members.length; i++) {
+      imageHash = keccak256(abi.encode(imageHash, _members[i].weight, _members[i].signer));
+    }
+
+    // Check against wallet imageHash
+    (bool succeed, bytes memory data) = _wallet.call(abi.encodePacked(IModuleAuthUpgradable(_wallet).imageHash.selector));
+    if (succeed && data.length == 32) {
+      // Check contract defined
+      bytes32 currentImageHash = abi.decode(data, (bytes32));
+      require(currentImageHash == imageHash, "RequireUtils#publishConfig: UNEXPECTED_IMAGE_HASH");
+    } else {
+      // Check counter-factual
+      require(address(
+        uint160(uint256(
+          keccak256(
+            abi.encodePacked(
+              bytes1(0xff),
+              FACTORY,
+              imageHash,
+              INIT_CODE_HASH
+            )
+          )
+        ))
+      ) == _wallet, "RequireUtils#publishConfig: UNEXPECTED_COUNTERFACTUAL_IMAGE_HASH");
+
+      // Register known image-hash for counter-factual wallet
+      if (_index) knownImageHashes[_wallet] = imageHash;
+    }
+
+    // Emit event for easy config retrieval
+    emit RequiredConfig(_wallet, imageHash, _threshold, abi.encode(_members));
+
+    if (_index) {
+      // Register last event for given wallet
+      lastWalletUpdate[_wallet] = block.number;
+
+      // Register last event for image-hash
+      lastImageHashUpdate[imageHash] = block.number;
+    }
+  }
+
+  /**
+   * @notice Publishes the configuration and set of signers for a counter-factual Sequence wallets using logs
+   * @dev Used for fast lookup of a wallet based on its signer members, only signing members are included in the logs
+   *   as a mechanism to avoid poisoning of the directory of wallets.
+   *
+   *   Only the initial counter-factual configuration can be published, to publish updated configurations see `publishConfig`.
+   *
+   * @param _wallet      Sequence wallet
+   * @param _hash        Any hash signed by the wallet
+   * @param _sizeMembers Number of members on the counter-factual configuration
+   * @param _signature   Signature for the given hash
+   * @param _index       True if an index in contract-storage is desired 
+   */
+  function publishInitialSigners(
+    address _wallet,
+    bytes32 _hash,
+    uint256 _sizeMembers,
+    bytes memory _signature,
+    bool _index
+  ) external {
+    // Decode and index signature
+    (
+      uint16 threshold,  // required threshold signature
+      uint256 rindex     // read index
+    ) = _signature.readFirstUint16();
+
+    // Generate sub-digest
+    bytes32 subDigest; {
+      uint256 chainId; assembly { chainId := chainid() }
+      subDigest = keccak256(
+        abi.encodePacked(
+          "\x19\x01",
+          chainId,
+          _wallet,
+          _hash
+        )
+      );
+    }
+
+    // Recover signature
+    bytes32 imageHash = bytes32(uint256(threshold));
+
+    Member[] memory members = new Member[](_sizeMembers);
+    uint256 membersIndex = 0;
+
+    while (rindex < _signature.length) {
+      // Read next item type and addrWeight
+      uint256 flag; uint256 addrWeight; address addr;
+      (flag, addrWeight, rindex) = _signature.readUint8Uint8(rindex);
+
+      if (flag == FLAG_ADDRESS) {
+        // Read plain address
+        (addr, rindex) = _signature.readAddress(rindex);
+      } else if (flag == FLAG_SIGNATURE) {
+        // Read single signature and recover signer
+        bytes memory signature;
+        (signature, rindex) = _signature.readBytes66(rindex);
+        addr = recoverSigner(subDigest, signature);
+
+        // Publish signer
+        _publishSigner(_wallet, addr, _index);
+      } else if (flag == FLAG_DYNAMIC_SIGNATURE) {
+        // Read signer
+        (addr, rindex) = _signature.readAddress(rindex);
+
+        {
+          // Read signature size
+          uint256 size;
+          (size, rindex) = _signature.readUint16(rindex);
+
+          // Read dynamic size signature
+          bytes memory signature;
+          (signature, rindex) = _signature.readBytes(rindex, size);
+          require(isValidSignature(subDigest, addr, signature), "ModuleAuth#_signatureValidation: INVALID_SIGNATURE");
+        }
+
+        // Publish signer
+        _publishSigner(_wallet, addr, _index);
+      } else {
+        revert("RequireUtils#publishInitialSigners: INVALID_SIGNATURE_FLAG");
+      }
+
+      // Store member on array
+      members[membersIndex] = Member(addrWeight, addr);
+      membersIndex++;
+
+      // Write weight and address to image
+      imageHash = keccak256(abi.encode(imageHash, addrWeight, addr));
+    }
+
+    require(membersIndex == _sizeMembers, "RequireUtils#publishInitialSigners: INVALID_MEMBERS_COUNT");
+
+    // Check against counter-factual imageHash
+    require(address(
+      uint160(uint256(
+        keccak256(
+          abi.encodePacked(
+            bytes1(0xff),
+            FACTORY,
+            imageHash,
+            INIT_CODE_HASH
+          )
+        )
+      ))
+    ) == _wallet, "RequireUtils#publishInitialSigners: UNEXPECTED_COUNTERFACTUAL_IMAGE_HASH");
+
+    // Emit event for easy config retrieval
+    emit RequiredConfig(_wallet, imageHash, threshold, abi.encode(members));
+
+    if (_index) {
+      // Register last event for given wallet
+      lastWalletUpdate[_wallet] = block.number;
+
+      // Register last event for image-hash
+      lastImageHashUpdate[imageHash] = block.number;
+
+      // Register known image-hash for counter-factual wallet
+      knownImageHashes[_wallet] = imageHash;
+    }
+  }
+
+  /**
+   * @notice Validates that a given expiration hasn't expired
+   * @dev Used as an optional transaction on a Sequence batch, to create expirable transactions.
+   *
+   * @param _expiration  Expiration to check
+   */
+  function requireNonExpired(uint256 _expiration) external view {
+    require(block.timestamp <= _expiration, "RequireUtils#requireNonExpired: EXPIRED");
+  }
+
+  /**
+   * @notice Validates that a given wallet has reached a given nonce
+   * @dev Used as an optional transaction on a Sequence batch, to define transaction execution order
+   *
+   * @param _wallet Sequence wallet
+   * @param _nonce  Required nonce
+   */
+  function requireMinNonce(address _wallet, uint256 _nonce) external view {
+    (uint256 space, uint256 nonce) = _decodeNonce(_nonce);
+    uint256 currentNonce = IModuleCalls(_wallet).readNonce(space);
+    require(currentNonce >= nonce, "RequireUtils#requireMinNonce: NONCE_BELOW_REQUIRED");
+  }
+
+  /**
+   * @notice Decodes a raw nonce
+   * @dev A raw nonce is encoded using the first 160 bits for the space
+   *  and the last 96 bits for the nonce
+   * @param _rawNonce Nonce to be decoded
+   * @return _space The nonce space of the raw nonce
+   * @return _nonce The nonce of the raw nonce
+   */
+  function _decodeNonce(uint256 _rawNonce) private pure returns (uint256 _space, uint256 _nonce) {
+    _nonce = uint256(bytes32(_rawNonce) & NONCE_MASK);
+    _space = _rawNonce >> NONCE_BITS;
+  }
+
+  /**
+   * @notice Publishes a signer that was validated to sign for a particular wallet
+   * @param _wallet Address of the wallet
+   * @param _signer Address of the signer
+   * @param _index True if an index on contract storage is desired
+   */
+  function _publishSigner(address _wallet, address _signer, bool _index) private {
+    // Required signer event
+    emit RequiredSigner(_wallet, _signer);
+
+    if (_index) {
+      // Register last event for given signer
+      lastSignerUpdate[_signer] = block.number;
+    }
+  }
+}
diff --git a/contracts/passport/modules/utils/SequenceUtils.sol b/contracts/passport/modules/utils/SequenceUtils.sol
new file mode 100644
index 00000000..5e2ec872
--- /dev/null
+++ b/contracts/passport/modules/utils/SequenceUtils.sol
@@ -0,0 +1,19 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./MultiCallUtils.sol";
+import "./RequireUtils.sol";
+
+
+contract SequenceUtils is 
+  MultiCallUtils,
+  RequireUtils
+{
+  constructor(
+    address _factory,
+    address _mainModule
+  ) RequireUtils(
+    _factory,
+    _mainModule
+  ) {}
+}
diff --git a/contracts/passport/modules/utils/libs/RequireFreshSigner.sol b/contracts/passport/modules/utils/libs/RequireFreshSigner.sol
new file mode 100644
index 00000000..e0df82a6
--- /dev/null
+++ b/contracts/passport/modules/utils/libs/RequireFreshSigner.sol
@@ -0,0 +1,17 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../RequireUtils.sol";
+
+
+contract RequireFreshSigner {
+  RequireUtils public immutable REQUIRE_UTILS;
+
+  constructor (RequireUtils _requireUtils) {
+    REQUIRE_UTILS = _requireUtils;
+  }
+
+  function requireFreshSigner(address _signer) external {
+    require(REQUIRE_UTILS.lastSignerUpdate(_signer) == 0, "RequireFreshSigner#requireFreshSigner: DUPLICATED_SIGNER");
+  }
+}
diff --git a/contracts/passport/signer/ImmutableSigner.sol b/contracts/passport/signer/ImmutableSigner.sol
new file mode 100644
index 00000000..b0e43700
--- /dev/null
+++ b/contracts/passport/signer/ImmutableSigner.sol
@@ -0,0 +1,114 @@
+// Copyright Immutable Pty Ltd 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import {SignatureValidator} from '../utils/SignatureValidator.sol';
+import {AccessControl} from '@openzeppelin/contracts/access/AccessControl.sol';
+
+/**
+ * @title ImmutableSigner
+ *
+ * @notice ERC-1271 Wallet implementation. Verifies signatures using a public
+ * key in storage matching an offchain private key. The key can be upgraded by
+ * the role SIGNER_ADMIN_ROLE, set during initialization.
+ */
+contract ImmutableSigner is SignatureValidator, AccessControl {
+  // @dev A signer only valid until the validUntil unix timestmap
+  struct ExpirableSigner {
+    address signer;
+    uint256 validUntil;
+  }
+
+  address public primarySigner;
+  ExpirableSigner public rolloverSigner;
+
+  bytes32 public constant SIGNER_ADMIN_ROLE = keccak256('SIGNER_ADMIN_ROLE');
+
+  /*
+   * @notice Emitted whenever the main authorized signer is updated.
+   */
+  event PrimarySignerUpdated(address indexed _previousSigner, address indexed _newSigner);
+  /*
+   * @notice Emitted whenever a temporary signer is enabled.
+   */
+  event RolloverSignerEnabled(address indexed _signer, uint256 _validUntil);
+
+
+  constructor(address _rootAdmin, address _signerAdmin, address _signer) {
+    _grantRole(DEFAULT_ADMIN_ROLE, _rootAdmin);
+    _grantRole(SIGNER_ADMIN_ROLE, _signerAdmin);
+
+    primarySigner = _signer;
+
+    // We do not need a rollover signer initially as there's no rollover
+    // happening, so validUntil must be in the past. Luckily that aligns with
+    // its default initial value.
+    //
+    // rolloverSigner.validUntil = 0;
+  }
+
+  /*
+   * @dev Grants SIGNER_ADMIN_ROLE to an user.
+   * @param _signerAdmin Address that will be allowed to update the wallet signer.
+   */
+  function grantSignerRole(address _signerAdmin) external onlyRole(DEFAULT_ADMIN_ROLE) {
+    _grantRole(SIGNER_ADMIN_ROLE, _signerAdmin);
+  }
+
+  /*
+   * @dev Updates the authorized public key address. Any old keys are expired effective immediately.
+   * @param _newSigner The address of the new authorized signer.
+   */
+  function updateSigner(address _newSigner) public onlyRole(SIGNER_ADMIN_ROLE) {
+    address previousSigner = primarySigner;
+    primarySigner = _newSigner;
+
+    // Make sure any rollover is immediately finished.
+    rolloverSigner.validUntil = 0;
+
+    emit PrimarySignerUpdated(previousSigner, _newSigner);
+  }
+
+  /*
+   * @dev Updates the authorized public key address. Allows the previous public
+   * key to remain valid for a specified rollover period.
+   *
+   * @param _newSigner The address of the new authorized signer.
+   * @param _rolloverPeriod Period for which the previous key will still be valid.
+   */
+  function updateSignerWithRolloverPeriod(
+    address _newSigner,
+    uint256 _rolloverPeriod
+  ) public onlyRole(SIGNER_ADMIN_ROLE) {
+    address previousSigner = primarySigner;
+    primarySigner = _newSigner;
+
+    rolloverSigner.signer = previousSigner;
+    rolloverSigner.validUntil = block.timestamp + _rolloverPeriod;
+
+    emit PrimarySignerUpdated(previousSigner, _newSigner);
+    emit RolloverSignerEnabled(rolloverSigner.signer, rolloverSigner.validUntil);
+  }
+
+  /**
+   * @notice Verifies whether the provided signature is valid with respect to the provided hash
+   * @dev MUST return the correct magic value if the signature provided is valid for the provided hash
+   *   > The bytes4 magic value to return when signature is valid is 0x1626ba7e
+   * @param _hash       keccak256 hash that was signed
+   * @param _signature  Signature byte array associated with _data
+   * @return magicValue Magic value 0x1626ba7e if the signature is valid and 0x0 otherwise
+   */
+  function isValidSignature(bytes32 _hash, bytes memory _signature) external view returns (bytes4) {
+    if (recoverSigner(_hash, _signature) == primarySigner) {
+      return ERC1271_MAGICVALUE_BYTES32;
+    }
+
+    if (block.timestamp <= rolloverSigner.validUntil) {
+      if (recoverSigner(_hash, _signature) == rolloverSigner.signer) {
+        return ERC1271_MAGICVALUE_BYTES32;
+      }
+    }
+
+    return 0;
+  }
+}
diff --git a/contracts/passport/startup/ILatestWalletImplLocator.sol b/contracts/passport/startup/ILatestWalletImplLocator.sol
new file mode 100644
index 00000000..6cb4985b
--- /dev/null
+++ b/contracts/passport/startup/ILatestWalletImplLocator.sol
@@ -0,0 +1,15 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+/**
+ * @title ILatestWalletImplLocator
+ * @notice Interface for using the latest wallet implementation locator contract.
+ */
+interface ILatestWalletImplLocator {
+    /**
+     * Return the address of the latest wallet implementation contract.
+     */
+    function latestWalletImplementation() external returns (address);
+}
\ No newline at end of file
diff --git a/contracts/passport/startup/LatestWalletImplLocator.sol b/contracts/passport/startup/LatestWalletImplLocator.sol
new file mode 100644
index 00000000..54644cca
--- /dev/null
+++ b/contracts/passport/startup/LatestWalletImplLocator.sol
@@ -0,0 +1,38 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import '@openzeppelin/contracts/access/AccessControl.sol';
+import "./ILatestWalletImplLocator.sol";
+
+/**
+ * @title LatestWalletImplLocator
+ * @notice Contract to return the address of the latest wallet implementation contract.
+ */
+contract LatestWalletImplLocator is ILatestWalletImplLocator, AccessControl {
+    // Role to change the implementation contract address.
+    bytes32 public constant IMPLCHANGER_ROLE = keccak256('IMPLCHANGER_ROLE');
+
+    address public latestWalletImplementation;
+
+    event ImplChanged(address indexed _whoBy, address indexed _newImpl);
+
+    /**
+     * @param _admin Role that can grant / revoke roles: DEFAULT_ADMIN and IMPLCHANGER.
+     * @param _implChanger Initial address that can change the latest wallet implementation address.
+     */
+    constructor(address _admin, address _implChanger) {
+        _grantRole(DEFAULT_ADMIN_ROLE, _admin);
+        _grantRole(IMPLCHANGER_ROLE, _implChanger);
+    }
+
+
+    /**
+     * Change the address of the latest wallet implementation contract.
+     * @param _newImpl Address of the main module to be used by the wallet.
+     */
+    function changeWalletImplementation(address _newImpl) external onlyRole(IMPLCHANGER_ROLE) {
+        latestWalletImplementation = _newImpl;
+        emit ImplChanged(msg.sender, _newImpl);
+    }
+}
\ No newline at end of file
diff --git a/contracts/passport/startup/StartupWalletImpl.sol b/contracts/passport/startup/StartupWalletImpl.sol
new file mode 100644
index 00000000..33b8b37e
--- /dev/null
+++ b/contracts/passport/startup/StartupWalletImpl.sol
@@ -0,0 +1,46 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./ILatestWalletImplLocator.sol";
+
+/**
+ * @title StartupWalletImpl
+ * @notice Initial wallet implementation contract used to setup the proxy with the latest
+ *        implementation contract and ensure that the minimal proxy deployment data never
+ *        needs to change.
+ *        Note that this code executed in the context of the WalletProxy.yul. As such,
+ *        the address of the Latest Wallet Implementation Locator contract needs to be
+ *        inserted directly into the code of the contract. This is achieved by using an 
+ *        immutable variable.
+ */
+contract StartupWalletImpl {
+    address public immutable walletImplementationLocator;
+
+    constructor (address _walletImplementationLocator) {
+        walletImplementationLocator = _walletImplementationLocator;
+    }
+
+
+    fallback() external payable {
+        // Get the address of the latest version of the wallet implementation.
+        ILatestWalletImplLocator locator = ILatestWalletImplLocator(walletImplementationLocator);
+        address latestImplAddr = locator.latestWalletImplementation();
+
+        // solhint-disable-next-line no-inline-assembly
+        assembly{
+            // Store the address of the implementation to use in the storage slot defined by the
+            // address of this contract.
+            sstore(address(), latestImplAddr)
+
+            // Now do a standard delegate call to the wallet implementation and return the
+            // results / errors.
+            calldatacopy(0, 0, calldatasize())
+            let result := delegatecall(gas(), latestImplAddr, 0, calldatasize(), 0, 0)
+            returndatacopy(0, 0, returndatasize())
+            switch result
+            case 0 {revert(0, returndatasize())}
+            default {return (0, returndatasize())}
+        }
+    }
+}
diff --git a/contracts/passport/utils/LibAddress.sol b/contracts/passport/utils/LibAddress.sol
new file mode 100644
index 00000000..7508365f
--- /dev/null
+++ b/contracts/passport/utils/LibAddress.sol
@@ -0,0 +1,18 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+library LibAddress {
+  /**
+   * @notice Will return true if provided address is a contract
+   * @param account Address to verify if contract or not
+   * @dev This contract will return false if called within the constructor of
+   *      a contract's deployment, as the code is not yet stored on-chain.
+   */
+  function isContract(address account) internal view returns (bool) {
+    uint256 csize;
+    // solhint-disable-next-line no-inline-assembly
+    assembly { csize := extcodesize(account) }
+    return csize != 0;
+  }
+}
diff --git a/contracts/passport/utils/LibBytes.sol b/contracts/passport/utils/LibBytes.sol
new file mode 100644
index 00000000..b8ffc64c
--- /dev/null
+++ b/contracts/passport/utils/LibBytes.sol
@@ -0,0 +1,195 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+library LibBytes {
+  using LibBytes for bytes;
+
+  /***********************************|
+  |        Read Bytes Functions       |
+  |__________________________________*/
+
+  /**
+   * @dev Read firsts uint16 value.
+   * @param data Byte array to be read.
+   * @return a uint16 value of data at index zero.
+   * @return newIndex Updated index after reading the values.
+   */
+  function readFirstUint16(
+    bytes memory data
+  ) internal pure returns (
+    uint16 a,
+    uint256 newIndex
+  ) {
+    assembly {
+      let word := mload(add(32, data))
+      a := shr(240, word)
+      newIndex := 2
+    }
+    require(2 <= data.length, "LibBytes#readFirstUint16: OUT_OF_BOUNDS");
+  }
+
+  /**
+   * @dev Reads consecutive bool (8 bits) and uint8 values.
+   * @param data Byte array to be read.
+   * @param index Index in byte array of uint8 and uint8 values.
+   * @return a uint8 value of data at given index.
+   * @return b uint8 value of data at given index + 8.
+   * @return newIndex Updated index after reading the values.
+   */
+  function readUint8Uint8(
+    bytes memory data,
+    uint256 index
+  ) internal pure returns (
+    uint8 a,
+    uint8 b,
+    uint256 newIndex
+  ) {
+    assembly {
+      let word := mload(add(index, add(32, data)))
+      a := shr(248, word)
+      b := and(shr(240, word), 0xff)
+      newIndex := add(index, 2)
+    }
+    assert(newIndex > index);
+    require(newIndex <= data.length, "LibBytes#readUint8Uint8: OUT_OF_BOUNDS");
+  }
+
+  /**
+   * @dev Reads an address value from a position in a byte array.
+   * @param data Byte array to be read.
+   * @param index Index in byte array of address value.
+   * @return a address value of data at given index.
+   * @return newIndex Updated index after reading the value.
+   */
+  function readAddress(
+    bytes memory data,
+    uint256 index
+  ) internal pure returns (
+    address a,
+    uint256 newIndex
+  ) {
+    assembly {
+      let word := mload(add(index, add(32, data)))
+      a := and(shr(96, word), 0xffffffffffffffffffffffffffffffffffffffff)
+      newIndex := add(index, 20)
+    }
+    assert(newIndex > index);
+    require(newIndex <= data.length, "LibBytes#readAddress: OUT_OF_BOUNDS");
+  }
+
+  /**
+   * @dev Reads 66 bytes from a position in a byte array.
+   * @param data Byte array to be read.
+   * @param index Index in byte array of 66 bytes value.
+   * @return a 66 bytes bytes array value of data at given index.
+   * @return newIndex Updated index after reading the value.
+   */
+  function readBytes66(
+    bytes memory data,
+    uint256 index
+  ) internal pure returns (
+    bytes memory a,
+    uint256 newIndex
+  ) {
+    a = new bytes(66);
+    assembly {
+      let offset := add(32, add(data, index))
+      mstore(add(a, 32), mload(offset))
+      mstore(add(a, 64), mload(add(offset, 32)))
+      mstore(add(a, 66), mload(add(offset, 34)))
+      newIndex := add(index, 66)
+    }
+    assert(newIndex > index);
+    require(newIndex <= data.length, "LibBytes#readBytes66: OUT_OF_BOUNDS");
+  }
+
+  /**
+   * @dev Reads a bytes32 value from a position in a byte array.
+   * @param b Byte array containing a bytes32 value.
+   * @param index Index in byte array of bytes32 value.
+   * @return result bytes32 value from byte array.
+   */
+  function readBytes32(
+    bytes memory b,
+    uint256 index
+  )
+    internal
+    pure
+    returns (bytes32 result)
+  {
+    require(
+      b.length >= index + 32,
+      "LibBytes#readBytes32: GREATER_OR_EQUAL_TO_32_LENGTH_REQUIRED"
+    );
+
+    // Arrays are prefixed by a 256 bit length parameter
+    uint256 pos = index + 32;
+
+    // Read the bytes32 from array memory
+    assembly {
+      result := mload(add(b, pos))
+    }
+    return result;
+  }
+
+  /**
+   * @dev Reads an uint16 value from a position in a byte array.
+   * @param data Byte array to be read.
+   * @param index Index in byte array of uint16 value.
+   * @return a uint16 value of data at given index.
+   * @return newIndex Updated index after reading the value.
+   */
+  function readUint16(
+    bytes memory data,
+    uint256 index
+  ) internal pure returns (uint16 a, uint256 newIndex) {
+    assembly {
+      let word := mload(add(index, add(32, data)))
+      a := and(shr(240, word), 0xffff)
+      newIndex := add(index, 2)
+    }
+    assert(newIndex > index);
+    require(newIndex <= data.length, "LibBytes#readUint16: OUT_OF_BOUNDS");
+  }
+
+  /**
+   * @dev Reads bytes from a position in a byte array.
+   * @param data Byte array to be read.
+   * @param index Index in byte array of bytes value.
+   * @param size Number of bytes to read.
+   * @return a bytes bytes array value of data at given index.
+   * @return newIndex Updated index after reading the value.
+   */
+  function readBytes(
+    bytes memory data,
+    uint256 index,
+    uint256 size
+  ) internal pure returns (bytes memory a, uint256 newIndex) {
+    a = new bytes(size);
+
+    assembly {
+      let offset := add(32, add(data, index))
+
+      let i := 0 let n := 32
+      // Copy each word, except last one
+      for { } lt(n, size) { i := n n := add(n, 32) } {
+        mstore(add(a, n), mload(add(offset, i)))
+      }
+
+      // Load word after new array
+      let suffix := add(a, add(32, size))
+      let suffixWord := mload(suffix)
+
+      // Copy last word, overwrites after array 
+      mstore(add(a, n), mload(add(offset, i)))
+
+      // Restore after array
+      mstore(suffix, suffixWord)
+
+      newIndex := add(index, size)
+    }
+
+    assert(newIndex >= index);
+    require(newIndex <= data.length, "LibBytes#readBytes: OUT_OF_BOUNDS");
+  }
+}
diff --git a/contracts/passport/utils/SignatureValidator.sol b/contracts/passport/utils/SignatureValidator.sol
new file mode 100644
index 00000000..255b820a
--- /dev/null
+++ b/contracts/passport/utils/SignatureValidator.sol
@@ -0,0 +1,143 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../interfaces/IERC1271Wallet.sol";
+
+import "./LibBytes.sol";
+
+/**
+ * @dev Contains logic for signature validation.
+ * Signatures from wallet contracts assume ERC-1271 support (https://github.com/ethereum/EIPs/blob/master/EIPS/eip-1271.md)
+ * Notes: Methods are strongly inspired by contracts in https://github.com/0xProject/0x-monorepo/blob/development/
+ */
+contract SignatureValidator {
+  using LibBytes for bytes;
+
+  /***********************************|
+  |             Variables             |
+  |__________________________________*/
+
+  // bytes4(keccak256("isValidSignature(bytes,bytes)"))
+  bytes4 constant internal ERC1271_MAGICVALUE = 0x20c13b0b;
+
+  // bytes4(keccak256("isValidSignature(bytes32,bytes)"))
+  bytes4 constant internal ERC1271_MAGICVALUE_BYTES32 = 0x1626ba7e;
+
+  // Allowed signature types.
+  uint256 private constant SIG_TYPE_EIP712 = 1;
+  uint256 private constant SIG_TYPE_ETH_SIGN = 2;
+  uint256 private constant SIG_TYPE_WALLET_BYTES32 = 3;
+
+  /***********************************|
+  |        Signature Functions        |
+  |__________________________________*/
+
+ /**
+   * @notice Recover the signer of hash, assuming it's an EOA account
+   * @dev Only for SignatureType.EIP712 and SignatureType.EthSign signatures
+   * @param _hash      Hash that was signed
+   *   encoded as (bytes32 r, bytes32 s, uint8 v, ... , SignatureType sigType)
+   */
+  function recoverSigner(
+    bytes32 _hash,
+    bytes memory _signature
+  ) internal pure returns (address signer) {
+    require(_signature.length == 66, "SignatureValidator#recoverSigner: invalid signature length");
+    uint256 signatureType = uint8(_signature[_signature.length - 1]);
+
+    // Variables are not scoped in Solidity.
+    uint8 v = uint8(_signature[64]);
+    bytes32 r = _signature.readBytes32(0);
+    bytes32 s = _signature.readBytes32(32);
+
+    // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature
+    // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines
+    // the valid range for s in (281): 0 < s < secp256k1n ÷ 2 + 1, and for v in (282): v ∈ {27, 28}. Most
+    // signatures from current libraries generate a unique signature with an s-value in the lower half order.
+    //
+    // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value
+    // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or
+    // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept
+    // these malleable signatures as well.
+    //
+    // Source OpenZeppelin
+    // https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/cryptography/ECDSA.sol
+
+    if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) {
+      revert("SignatureValidator#recoverSigner: invalid signature 's' value");
+    }
+
+    if (v != 27 && v != 28) {
+      revert("SignatureValidator#recoverSigner: invalid signature 'v' value");
+    }
+
+    // Signature using EIP712
+    if (signatureType == SIG_TYPE_EIP712) {
+      signer = ecrecover(_hash, v, r, s);
+
+    // Signed using web3.eth_sign() or Ethers wallet.signMessage()
+    } else if (signatureType == SIG_TYPE_ETH_SIGN) {
+      signer = ecrecover(
+        keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", _hash)),
+        v,
+        r,
+        s
+      );
+
+    } else {
+      // Anything other signature types are illegal (We do not return false because
+      // the signature may actually be valid, just not in a format
+      // that we currently support. In this case returning false
+      // may lead the caller to incorrectly believe that the
+      // signature was invalid.)
+      revert("SignatureValidator#recoverSigner: UNSUPPORTED_SIGNATURE_TYPE");
+    }
+
+    // Prevent signer from being 0x0
+    require(
+      signer != address(0x0),
+      "SignatureValidator#recoverSigner: INVALID_SIGNER"
+    );
+
+    return signer;
+  }
+
+ /**
+   * @notice Returns true if the provided signature is valid for the given signer.
+   * @dev Supports SignatureType.EIP712, SignatureType.EthSign, and ERC1271 signatures
+   * @param _hash      Hash that was signed
+   * @param _signer    Address of the signer candidate
+   * @param _signature Signature byte array
+   */
+  function isValidSignature(
+    bytes32 _hash,
+    address _signer,
+    bytes memory _signature
+  ) internal view returns (bool valid) {
+    // Check for valid signature length
+    if (_signature.length == 0) {
+      revert("SignatureValidator#isValidSignature: signature is empty");
+    }
+    
+    uint256 signatureType = uint8(_signature[_signature.length - 1]);
+
+    if (signatureType == SIG_TYPE_EIP712 || signatureType == SIG_TYPE_ETH_SIGN) {
+      // Recover signer and compare with provided
+      valid = recoverSigner(_hash, _signature) == _signer;
+
+    } else if (signatureType == SIG_TYPE_WALLET_BYTES32) {
+      // Remove signature type before calling ERC1271, restore after call
+      uint256 prevSize; assembly { prevSize := mload(_signature) mstore(_signature, sub(prevSize, 1)) }
+      valid = ERC1271_MAGICVALUE_BYTES32 == IERC1271Wallet(_signer).isValidSignature(_hash, _signature);
+      assembly { mstore(_signature, prevSize) }
+
+    } else {
+      // Anything other signature types are illegal (We do not return false because
+      // the signature may actually be valid, just not in a format
+      // that we currently support. In this case returning false
+      // may lead the caller to incorrectly believe that the
+      // signature was invalid.)
+      revert("SignatureValidator#isValidSignature: UNSUPPORTED_SIGNATURE_TYPE");
+    }
+  }
+}
diff --git a/contracts/seaport/README.md b/contracts/seaport/README.md
new file mode 100644
index 00000000..bc3ab72d
--- /dev/null
+++ b/contracts/seaport/README.md
@@ -0,0 +1 @@
+DO NOT MODIFY THESE CONTRACTS DIRECTLY. This folder has been automatically extracted from https://github.com/immutable/seaport via a submodule in this repository's forks directory. See the upstream repository for full context.
\ No newline at end of file
diff --git a/contracts/seaport/Seaport.sol b/contracts/seaport/Seaport.sol
new file mode 100644
index 00000000..bea2dc2a
--- /dev/null
+++ b/contracts/seaport/Seaport.sol
@@ -0,0 +1,121 @@
+// SPDX-License-Identifier: MIT
+pragma solidity 0.8.17;
+
+import { Consideration } from "seaport-core/src/lib/Consideration.sol";
+
+/**
+ * @title Seaport
+ * @custom:version 1.5
+ * @author 0age (0age.eth)
+ * @custom:coauthor d1ll0n (d1ll0n.eth)
+ * @custom:coauthor transmissions11 (t11s.eth)
+ * @custom:coauthor James Wenzel (emo.eth)
+ * @custom:contributor Kartik (slokh.eth)
+ * @custom:contributor LeFevre (lefevre.eth)
+ * @custom:contributor Joseph Schiarizzi (CupOJoseph.eth)
+ * @custom:contributor Aspyn Palatnick (stuckinaboot.eth)
+ * @custom:contributor Stephan Min (stephanm.eth)
+ * @custom:contributor Ryan Ghods (ralxz.eth)
+ * @custom:contributor Daniel Viau (snotrocket.eth)
+ * @custom:contributor hack3r-0m (hack3r-0m.eth)
+ * @custom:contributor Diego Estevez (antidiego.eth)
+ * @custom:contributor Chomtana (chomtana.eth)
+ * @custom:contributor Saw-mon and Natalie (sawmonandnatalie.eth)
+ * @custom:contributor 0xBeans (0xBeans.eth)
+ * @custom:contributor 0x4non (punkdev.eth)
+ * @custom:contributor Laurence E. Day (norsefire.eth)
+ * @custom:contributor vectorized.eth (vectorized.eth)
+ * @custom:contributor karmacoma (karmacoma.eth)
+ * @custom:contributor horsefacts (horsefacts.eth)
+ * @custom:contributor UncarvedBlock (uncarvedblock.eth)
+ * @custom:contributor Zoraiz Mahmood (zorz.eth)
+ * @custom:contributor William Poulin (wpoulin.eth)
+ * @custom:contributor Rajiv Patel-O'Connor (rajivpoc.eth)
+ * @custom:contributor tserg (tserg.eth)
+ * @custom:contributor cygaar (cygaar.eth)
+ * @custom:contributor Meta0xNull (meta0xnull.eth)
+ * @custom:contributor gpersoon (gpersoon.eth)
+ * @custom:contributor Matt Solomon (msolomon.eth)
+ * @custom:contributor Weikang Song (weikangs.eth)
+ * @custom:contributor zer0dot (zer0dot.eth)
+ * @custom:contributor Mudit Gupta (mudit.eth)
+ * @custom:contributor leonardoalt (leoalt.eth)
+ * @custom:contributor cmichel (cmichel.eth)
+ * @custom:contributor PraneshASP (pranesh.eth)
+ * @custom:contributor JasperAlexander (jasperalexander.eth)
+ * @custom:contributor Ellahi (ellahi.eth)
+ * @custom:contributor zaz (1zaz1.eth)
+ * @custom:contributor berndartmueller (berndartmueller.eth)
+ * @custom:contributor dmfxyz (dmfxyz.eth)
+ * @custom:contributor daltoncoder (dontkillrobots.eth)
+ * @custom:contributor 0xf4ce (0xf4ce.eth)
+ * @custom:contributor phaze (phaze.eth)
+ * @custom:contributor hrkrshnn (hrkrshnn.eth)
+ * @custom:contributor axic (axic.eth)
+ * @custom:contributor leastwood (leastwood.eth)
+ * @custom:contributor 0xsanson (sanson.eth)
+ * @custom:contributor blockdev (blockd3v.eth)
+ * @custom:contributor fiveoutofnine (fiveoutofnine.eth)
+ * @custom:contributor shuklaayush (shuklaayush.eth)
+ * @custom:contributor dravee (dravee.eth)
+ * @custom:contributor 0xPatissier
+ * @custom:contributor pcaversaccio
+ * @custom:contributor David Eiber
+ * @custom:contributor csanuragjain
+ * @custom:contributor sach1r0
+ * @custom:contributor twojoy0
+ * @custom:contributor ori_dabush
+ * @custom:contributor Daniel Gelfand
+ * @custom:contributor okkothejawa
+ * @custom:contributor FlameHorizon
+ * @custom:contributor vdrg
+ * @custom:contributor dmitriia
+ * @custom:contributor bokeh-eth
+ * @custom:contributor asutorufos
+ * @custom:contributor rfart(rfa)
+ * @custom:contributor Riley Holterhus
+ * @custom:contributor big-tech-sux
+ * @notice Seaport is a generalized native token/ERC20/ERC721/ERC1155
+ *         marketplace with lightweight methods for common routes as well as
+ *         more flexible methods for composing advanced orders or groups of
+ *         orders. Each order contains an arbitrary number of items that may be
+ *         spent (the "offer") along with an arbitrary number of items that must
+ *         be received back by the indicated recipients (the "consideration").
+ */
+contract Seaport is Consideration {
+    /**
+     * @notice Derive and set hashes, reference chainId, and associated domain
+     *         separator during deployment.
+     *
+     * @param conduitController A contract that deploys conduits, or proxies
+     *                          that may optionally be used to transfer approved
+     *                          ERC20/721/1155 tokens.
+     */
+    constructor(address conduitController) Consideration(conduitController) {}
+
+    /**
+     * @dev Internal pure function to retrieve and return the name of this
+     *      contract.
+     *
+     * @return The name of this contract.
+     */
+    function _name() internal pure override returns (string memory) {
+        // Return the name of the contract.
+        assembly {
+            mstore(0x20, 0x20)
+            mstore(0x47, 0x07536561706f7274)
+            return(0x20, 0x60)
+        }
+    }
+
+    /**
+     * @dev Internal pure function to retrieve the name of this contract as a
+     *      string that will be used to derive the name hash in the constructor.
+     *
+     * @return The name of this contract as a string.
+     */
+    function _nameString() internal pure override returns (string memory) {
+        // Return the name of the contract.
+        return "Seaport";
+    }
+}
diff --git a/contracts/seaport/conduit/Conduit.sol b/contracts/seaport/conduit/Conduit.sol
new file mode 100644
index 00000000..75fc3ea0
--- /dev/null
+++ b/contracts/seaport/conduit/Conduit.sol
@@ -0,0 +1,18 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import { Conduit as CoreConduit } from "seaport-core/src/conduit/Conduit.sol";
+
+/**
+ * @title Conduit
+ * @author 0age
+ * @notice This contract serves as an originator for "proxied" transfers. Each
+ *         conduit is deployed and controlled by a "conduit controller" that can
+ *         add and remove "channels" or contracts that can instruct the conduit
+ *         to transfer approved ERC20/721/1155 tokens. *IMPORTANT NOTE: each
+ *         conduit has an owner that can arbitrarily add or remove channels, and
+ *         a malicious or negligent owner can add a channel that allows for any
+ *         approved ERC20/721/1155 tokens to be taken immediately — be extremely
+ *         cautious with what conduits you give token approvals to!*
+ */
+contract LocalConduit is CoreConduit {}
diff --git a/contracts/seaport/conduit/ConduitController.sol b/contracts/seaport/conduit/ConduitController.sol
new file mode 100644
index 00000000..27e2b60e
--- /dev/null
+++ b/contracts/seaport/conduit/ConduitController.sol
@@ -0,0 +1,13 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import { ConduitController as CoreConduitController } from "seaport-core/src/conduit/ConduitController.sol";
+
+/**
+ * @title ConduitController
+ * @author 0age
+ * @notice ConduitController enables deploying and managing new conduits, or
+ *         contracts that allow registered callers (or open "channels") to
+ *         transfer approved ERC20/721/1155 tokens on their behalf.
+ */
+contract LocalConduitController is CoreConduitController {}
diff --git a/contracts/seaport/helpers/ArrayHelpers.sol b/contracts/seaport/helpers/ArrayHelpers.sol
new file mode 100644
index 00000000..99bb3bfa
--- /dev/null
+++ b/contracts/seaport/helpers/ArrayHelpers.sol
@@ -0,0 +1,705 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import "seaport-types/src/helpers/PointerLibraries.sol";
+
+/**
+ * @author d1ll0n
+ * @custom:coauthor Most of the natspec is cribbed from the TypeScript
+ *                  documentation
+ */
+library ArrayHelpers {
+    // Has to be out of place to silence a linter warning
+    function reduceWithArg(
+        MemoryPointer array,
+        /* function (uint256 currentResult, uint256 element, uint256 arg) */
+        /* returns (uint256 newResult) */
+        function(uint256, uint256, MemoryPointer) internal returns (uint256) fn,
+        uint256 initialValue,
+        MemoryPointer arg
+    ) internal returns (uint256 result) {
+        unchecked {
+            uint256 length = array.readUint256();
+
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+
+            result = initialValue;
+            while (srcPosition.lt(srcEnd)) {
+                result = fn(result, srcPosition.readUint256(), arg);
+                srcPosition = srcPosition.next();
+            }
+        }
+    }
+
+    function flatten(
+        MemoryPointer array1,
+        MemoryPointer array2
+    ) internal view returns (MemoryPointer newArray) {
+        unchecked {
+            uint256 arrayLength1 = array1.readUint256();
+            uint256 arrayLength2 = array2.readUint256();
+            uint256 array1HeadSize = arrayLength1 * 32;
+            uint256 array2HeadSize = arrayLength2 * 32;
+
+            newArray = malloc(array1HeadSize + array2HeadSize + 32);
+            newArray.write(arrayLength1 + arrayLength2);
+
+            MemoryPointer dst = newArray.next();
+            if (arrayLength1 > 0) {
+                array1.next().copy(dst, array1HeadSize);
+            }
+            if (arrayLength2 > 0) {
+                array2.next().copy(dst.offset(array1HeadSize), array2HeadSize);
+            }
+        }
+    }
+
+    function flattenThree(
+        MemoryPointer array1,
+        MemoryPointer array2,
+        MemoryPointer array3
+    ) internal view returns (MemoryPointer newArray) {
+        unchecked {
+            uint256 arrayLength1 = array1.readUint256();
+            uint256 arrayLength2 = array2.readUint256();
+            uint256 arrayLength3 = array3.readUint256();
+            uint256 array1HeadSize = arrayLength1 * 32;
+            uint256 array2HeadSize = arrayLength2 * 32;
+            uint256 array3HeadSize = arrayLength3 * 32;
+
+            newArray = malloc(
+                array1HeadSize + array2HeadSize + array3HeadSize + 32
+            );
+            newArray.write(arrayLength1 + arrayLength2 + arrayLength3);
+
+            MemoryPointer dst = newArray.next();
+            if (arrayLength1 > 0) {
+                array1.next().copy(dst, array1HeadSize);
+            }
+            if (arrayLength2 > 0) {
+                array2.next().copy(dst.offset(array1HeadSize), array2HeadSize);
+            }
+            if (arrayLength3 > 0) {
+                array3.next().copy(
+                    dst.offset(array1HeadSize + array2HeadSize),
+                    array3HeadSize
+                );
+            }
+        }
+    }
+
+    // =====================================================================//
+    //            map with (element) => (newElement) callback               //
+    // =====================================================================//
+
+    /**
+     * @dev map calls a defined callback function on each element of an array
+     *      and returns an array that contains the results
+     *
+     * @param array   the array to map
+     * @param fn      a function that accepts each element in the array and
+     *                returns a new value to put in its place in the new array
+     *
+     * @return newArray the new array created with the results from calling
+     *         fn with each element
+     */
+    function map(
+        MemoryPointer array,
+        /* function (uint256 value) returns (uint256 newValue) */
+        function(uint256) internal pure returns (uint256) fn
+    ) internal pure returns (MemoryPointer newArray) {
+        unchecked {
+            uint256 length = array.readUint256();
+
+            newArray = malloc((length + 1) * 32);
+            newArray.write(length);
+
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+            MemoryPointer dstPosition = newArray.next();
+
+            while (srcPosition.lt(srcEnd)) {
+                dstPosition.write(fn(srcPosition.readUint256()));
+                srcPosition = srcPosition.next();
+                dstPosition = dstPosition.next();
+            }
+        }
+    }
+
+    // =====================================================================//
+    //         filterMap with (element) => (newElement) callback            //
+    // =====================================================================//
+
+    /**
+     * @dev filterMap calls a defined callback function on each element of an
+     *      array and returns an array that contains only the non-zero results
+     *
+     * @param array   the array to map
+     * @param fn      a function that accepts each element in the array and
+     *                returns a new value to put in its place in the new array
+     *                or a zero value to indicate that the element should not
+     *                be included in the new array
+     *
+     * @return newArray the new array created with the results from calling
+     *                  fn with each element
+     */
+    function filterMap(
+        MemoryPointer array,
+        /* function (uint256 value) returns (uint256 newValue) */
+        function(MemoryPointer) internal pure returns (MemoryPointer) fn
+    ) internal pure returns (MemoryPointer newArray) {
+        unchecked {
+            uint256 length = array.readUint256();
+
+            newArray = malloc((length + 1) * 32);
+
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+            MemoryPointer dstPosition = newArray.next();
+
+            length = 0;
+
+            while (srcPosition.lt(srcEnd)) {
+                MemoryPointer result = fn(srcPosition.readMemoryPointer());
+                if (!result.isNull()) {
+                    dstPosition.write(result);
+                    dstPosition = dstPosition.next();
+                    length += 1;
+                }
+                srcPosition = srcPosition.next();
+            }
+            newArray.write(length);
+        }
+    }
+
+    // =====================================================================//
+    //      filterMap with (element, arg) => (newElement) callback          //
+    // =====================================================================//
+
+    /**
+     * @dev filterMap calls a defined callback function on each element of an
+     *      array and returns an array that contains only the non-zero results
+     *
+     *        filterMapWithArg = (arr, callback, arg) => arr.map(
+     *          (element) => callback(element, arg)
+     *        ).filter(result => result != 0)
+     *
+     * @param array   the array to map
+     * @param fn      a function that accepts each element in the array and
+     *                returns a new value to put in its place in the new array
+     *                or a zero value to indicate that the element should not
+     *                be included in the new array
+     * @param arg     an arbitrary value provided in each call to fn
+     *
+     * @return newArray the new array created with the results from calling
+     *                  fn with each element
+     */
+    function filterMapWithArg(
+        MemoryPointer array,
+        /* function (MemoryPointer element, MemoryPointer arg) */
+        /* returns (uint256 newValue) */
+        function(MemoryPointer, MemoryPointer)
+            internal
+            pure
+            returns (MemoryPointer) fn,
+        MemoryPointer arg
+    ) internal pure returns (MemoryPointer newArray) {
+        unchecked {
+            uint256 length = array.readUint256();
+
+            newArray = malloc((length + 1) * 32);
+
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+            MemoryPointer dstPosition = newArray.next();
+
+            length = 0;
+
+            while (srcPosition.lt(srcEnd)) {
+                MemoryPointer result = fn(srcPosition.readMemoryPointer(), arg);
+                if (!result.isNull()) {
+                    dstPosition.write(result);
+                    dstPosition = dstPosition.next();
+                    length += 1;
+                }
+                srcPosition = srcPosition.next();
+            }
+            newArray.write(length);
+        }
+    }
+
+    // ====================================================================//
+    //         filter  with (element, arg) => (bool) predicate             //
+    // ====================================================================//
+
+    /**
+     * @dev filter calls a defined callback function on each element of an array
+     *      and returns an array that contains only the elements which the
+     *      callback returned true for
+     *
+     * @param array   the array to map
+     * @param fn      a function that accepts each element in the array and
+     *                returns a boolean that indicates whether the element
+     *                should be included in the new array
+     * @param arg     an arbitrary value provided in each call to fn
+     *
+     * @return newArray the new array created with the elements which the
+     *                  callback returned true for
+     */
+    function filterWithArg(
+        MemoryPointer array,
+        /* function (uint256 value, uint256 arg) returns (bool) */
+        function(MemoryPointer, MemoryPointer) internal pure returns (bool) fn,
+        MemoryPointer arg
+    ) internal pure returns (MemoryPointer newArray) {
+        unchecked {
+            uint256 length = array.readUint256();
+
+            newArray = malloc((length + 1) * 32);
+
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+            MemoryPointer dstPosition = newArray.next();
+
+            length = 0;
+
+            while (srcPosition.lt(srcEnd)) {
+                MemoryPointer element = srcPosition.readMemoryPointer();
+                if (fn(element, arg)) {
+                    dstPosition.write(element);
+                    dstPosition = dstPosition.next();
+                    length += 1;
+                }
+                srcPosition = srcPosition.next();
+            }
+            newArray.write(length);
+        }
+    }
+
+    // ====================================================================//
+    //            filter  with (element) => (bool) predicate               //
+    // ====================================================================//
+
+    /**
+     * @dev filter calls a defined callback function on each element of an array
+     *      and returns an array that contains only the elements which the
+     *      callback returned true for
+     *
+     * @param array   the array to map
+     * @param fn      a function that accepts each element in the array and
+     *                returns a boolean that indicates whether the element
+     *                should be included in the new array
+     *
+     * @return newArray the new array created with the elements which the
+     *                  callback returned true for
+     */
+    function filter(
+        MemoryPointer array,
+        /* function (uint256 value) returns (bool) */
+        function(MemoryPointer) internal pure returns (bool) fn
+    ) internal pure returns (MemoryPointer newArray) {
+        unchecked {
+            uint256 length = array.readUint256();
+
+            newArray = malloc((length + 1) * 32);
+
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+            MemoryPointer dstPosition = newArray.next();
+
+            length = 0;
+
+            while (srcPosition.lt(srcEnd)) {
+                MemoryPointer element = srcPosition.readMemoryPointer();
+                if (fn(element)) {
+                    dstPosition.write(element);
+                    dstPosition = dstPosition.next();
+                    length += 1;
+                }
+                srcPosition = srcPosition.next();
+            }
+            newArray.write(length);
+        }
+    }
+
+    /**
+     * @dev mapWithIndex calls a defined callback function with each element of
+     *      an array and its index and returns an array that contains the
+     *      results
+     *
+     * @param array   the array to map
+     * @param fn      a function that accepts each element in the array and
+     *                its index and returns a new value to put in its place
+     *                in the new array
+     *
+     * @return newArray the new array created with the results from calling
+     *         fn with each element
+     */
+    function mapWithIndex(
+        MemoryPointer array,
+        /* function (uint256 value, uint256 index) returns (uint256 newValue) */
+        function(uint256, uint256) internal pure returns (uint256) fn
+    ) internal pure returns (MemoryPointer newArray) {
+        unchecked {
+            uint256 length = array.readUint256();
+
+            newArray = malloc((length + 1) * 32);
+            newArray.write(length);
+
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+            MemoryPointer dstPosition = newArray.next();
+
+            uint256 index;
+            while (srcPosition.lt(srcEnd)) {
+                dstPosition.write(fn(srcPosition.readUint256(), index++));
+                srcPosition = srcPosition.next();
+                dstPosition = dstPosition.next();
+            }
+        }
+    }
+
+    /**
+     * @dev map calls a defined callback function on each element of an array
+     *      and returns an array that contains the results
+     *
+     * @param array   the array to map
+     * @param fn      a function that accepts each element in the array and
+     *                the `arg` value provided in the call to map and returns
+     *                a new value to put in its place in the new array
+     * @param arg     an arbitrary value provided in each call to fn
+     *
+     * @return newArray the new array created with the results from calling
+     *         fn with each element
+     */
+    function mapWithArg(
+        MemoryPointer array,
+        /* function (uint256 value, uint256 arg) returns (uint256 newValue) */
+        function(MemoryPointer, MemoryPointer)
+            internal
+            pure
+            returns (MemoryPointer) fn,
+        MemoryPointer arg
+    ) internal pure returns (MemoryPointer newArray) {
+        unchecked {
+            uint256 length = array.readUint256();
+
+            newArray = malloc((length + 1) * 32);
+            newArray.write(length);
+
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+            MemoryPointer dstPosition = newArray.next();
+
+            while (srcPosition.lt(srcEnd)) {
+                dstPosition.write(fn(srcPosition.readMemoryPointer(), arg));
+                srcPosition = srcPosition.next();
+                dstPosition = dstPosition.next();
+            }
+        }
+    }
+
+    function mapWithIndex(
+        MemoryPointer array,
+        /* function (uint256 value, uint256 index, uint256 arg) */
+        /* returns (uint256 newValue) */
+        function(uint256, uint256, uint256) internal pure returns (uint256) fn,
+        uint256 arg
+    ) internal pure returns (MemoryPointer newArray) {
+        unchecked {
+            uint256 length = array.readUint256();
+
+            newArray = malloc((length + 1) * 32);
+            newArray.write(length);
+
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+            MemoryPointer dstPosition = newArray.next();
+
+            uint256 index;
+            while (srcPosition.lt(srcEnd)) {
+                dstPosition.write(fn(srcPosition.readUint256(), index++, arg));
+                srcPosition = srcPosition.next();
+                dstPosition = dstPosition.next();
+            }
+        }
+    }
+
+    function reduce(
+        MemoryPointer array,
+        /* function (uint256 currentResult, uint256 element) */
+        /* returns (uint256 newResult) */
+        function(uint256, uint256) internal pure returns (uint256) fn,
+        uint256 initialValue
+    ) internal pure returns (uint256 result) {
+        unchecked {
+            uint256 length = array.readUint256();
+
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+
+            result = initialValue;
+            while (srcPosition.lt(srcEnd)) {
+                result = fn(result, srcPosition.readUint256());
+                srcPosition = srcPosition.next();
+            }
+        }
+    }
+
+    // This was the previous home of `reduceWithArg`. It can now be found near
+    // the top of this file.
+
+    function forEach(
+        MemoryPointer array,
+        /* function (MemoryPointer element, MemoryPointer arg) */
+        function(MemoryPointer, MemoryPointer) internal pure fn,
+        MemoryPointer arg
+    ) internal pure {
+        unchecked {
+            uint256 length = array.readUint256();
+
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+
+            while (srcPosition.lt(srcEnd)) {
+                fn(srcPosition.readMemoryPointer(), arg);
+                srcPosition = srcPosition.next();
+            }
+        }
+    }
+
+    function forEach(
+        MemoryPointer array,
+        /* function (MemoryPointer element) */
+        function(MemoryPointer) internal pure fn
+    ) internal pure {
+        unchecked {
+            uint256 length = array.readUint256();
+
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+
+            while (srcPosition.lt(srcEnd)) {
+                fn(srcPosition.readMemoryPointer());
+                srcPosition = srcPosition.next();
+            }
+        }
+    }
+
+    // =====================================================================//
+    //     find with function(uint256 element, uint256 arg) predicate       //
+    // =====================================================================//
+
+    /**
+     * @dev calls `predicate` once for each element of the array, in ascending
+     *      order, until it finds one where predicate returns true. If such an
+     *      element is found, find immediately returns that element value.
+     *      Otherwise, find returns 0.
+     *
+     * @param array     array to search
+     * @param predicate function that checks whether each element meets the
+     *                  search filter.
+     * @param arg       second input to `predicate`
+     *
+     * @return          the value of the first element in the array where
+     *                  predicate is true and 0 otherwise.
+     */
+    function find(
+        MemoryPointer array,
+        function(uint256, uint256) internal pure returns (bool) predicate,
+        uint256 arg
+    ) internal pure returns (uint256) {
+        unchecked {
+            uint256 length = array.readUint256();
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+            while (srcPosition.lt(srcEnd)) {
+                uint256 value = srcPosition.readUint256();
+                if (predicate(value, arg)) return value;
+                srcPosition = srcPosition.next();
+            }
+            return 0;
+        }
+    }
+
+    // =====================================================================//
+    //            find with function(uint256 element) predicate             //
+    // =====================================================================//
+
+    /**
+     * @dev calls `predicate` once for each element of the array, in ascending
+     *      order, until it finds one where predicate returns true. If such an
+     *      element is found, find immediately returns that element value.
+     *      Otherwise, find returns 0.
+     *
+     * @param array     array to search
+     * @param predicate function that checks whether each element meets the
+     *                  search filter.
+     * @param fromIndex index to start search at
+     *
+     * @custom:return   the value of the first element in the array where
+     *                  predicate is trueand 0 otherwise.
+     */
+    function find(
+        MemoryPointer array,
+        function(uint256) internal pure returns (bool) predicate,
+        uint256 fromIndex
+    ) internal pure returns (uint256) {
+        unchecked {
+            uint256 length = array.readUint256();
+            MemoryPointer srcPosition = array.next().offset(fromIndex * 0x20);
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+            while (srcPosition.lt(srcEnd)) {
+                uint256 value = srcPosition.readUint256();
+                if (predicate(value)) return value;
+                srcPosition = srcPosition.next();
+            }
+            return 0;
+        }
+    }
+
+    // =====================================================================//
+    //            find with function(uint256 element) predicate             //
+    // =====================================================================//
+
+    /**
+     * @dev calls `predicate` once for each element of the array, in ascending
+     *      order, until it finds one where predicate returns true. If such an
+     *      element is found, find immediately returns that element value.
+     *      Otherwise, find returns 0.
+     *
+     * @param array     array to search
+     * @param predicate function that checks whether each element meets the
+     *                  search filter.
+     *
+     * @return          the value of the first element in the array where
+     *                  predicate is true and 0 otherwise.
+     */
+    function find(
+        MemoryPointer array,
+        function(uint256) internal pure returns (bool) predicate
+    ) internal pure returns (uint256) {
+        unchecked {
+            uint256 length = array.readUint256();
+            MemoryPointer srcPosition = array.next();
+            MemoryPointer srcEnd = srcPosition.offset(length * 0x20);
+            while (srcPosition.lt(srcEnd)) {
+                uint256 value = srcPosition.readUint256();
+                if (predicate(value)) return value;
+                srcPosition = srcPosition.next();
+            }
+            return 0;
+        }
+    }
+
+    // =====================================================================//
+    //                               indexOf                                //
+    // =====================================================================//
+
+    /**
+     * @dev Returns the index of the first occurrence of a value in an array,
+     *      or -1 if it is not present.
+     *
+     * @param array         array to search
+     * @param searchElement the value to locate in the array.
+     */
+    function indexOf(
+        MemoryPointer array,
+        uint256 searchElement
+    ) internal pure returns (int256 index) {
+        unchecked {
+            int256 length = array.readInt256();
+            MemoryPointer src = array;
+            int256 reachedEnd;
+            while (
+                ((reachedEnd = toInt(index == length)) |
+                    toInt((src = src.next()).readUint256() == searchElement)) ==
+                0
+            ) {
+                index += 1;
+            }
+            return (reachedEnd * -1) | index;
+        }
+    }
+
+    function toInt(bool a) internal pure returns (int256 b) {
+        assembly {
+            b := a
+        }
+    }
+
+    // =====================================================================//
+    //                     findIndex with one argument                      //
+    // =====================================================================//
+
+    function findIndexWithArg(
+        MemoryPointer array,
+        function(uint256, uint256) internal pure returns (bool) predicate,
+        uint256 arg
+    ) internal pure returns (int256 index) {
+        unchecked {
+            int256 length = array.readInt256();
+            MemoryPointer src = array;
+            while (index < length) {
+                if (predicate((src = src.next()).readUint256(), arg)) {
+                    return index;
+                }
+                index += 1;
+            }
+            return -1;
+        }
+    }
+
+    // =====================================================================//
+    //                     findIndex from start index                       //
+    // =====================================================================//
+
+    function findIndexFrom(
+        MemoryPointer array,
+        function(MemoryPointer) internal pure returns (bool) predicate,
+        uint256 fromIndex
+    ) internal pure returns (int256 index) {
+        unchecked {
+            index = int256(fromIndex);
+            int256 length = array.readInt256();
+            MemoryPointer src = array.offset(fromIndex * 0x20);
+            while (index < length) {
+                if (predicate((src = src.next()).readMemoryPointer())) {
+                    return index;
+                }
+                index += 1;
+            }
+            return -1;
+        }
+    }
+
+    function countFrom(
+        MemoryPointer array,
+        function(MemoryPointer) internal pure returns (bool) predicate,
+        uint256 fromIndex
+    ) internal pure returns (int256 count) {
+        unchecked {
+            uint256 index = fromIndex;
+            uint256 length = array.readUint256();
+            MemoryPointer src = array.offset(fromIndex * 0x20);
+            while (index < length) {
+                if (predicate((src = src.next()).readMemoryPointer())) {
+                    count += 1;
+                }
+                index += 1;
+            }
+        }
+    }
+
+    // =====================================================================//
+    //                      includes with one argument                      //
+    // =====================================================================//
+
+    function includes(
+        MemoryPointer array,
+        uint256 value
+    ) internal pure returns (bool) {
+        return indexOf(array, value) != -1;
+    }
+}
diff --git a/contracts/seaport/helpers/SeaportRouter.sol b/contracts/seaport/helpers/SeaportRouter.sol
new file mode 100644
index 00000000..9b192069
--- /dev/null
+++ b/contracts/seaport/helpers/SeaportRouter.sol
@@ -0,0 +1,256 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import {
+    SeaportRouterInterface
+} from "seaport-types/src/interfaces/SeaportRouterInterface.sol";
+
+import {
+    SeaportInterface
+} from "seaport-types/src/interfaces/SeaportInterface.sol";
+
+import { ReentrancyGuard } from "seaport-core/src/lib/ReentrancyGuard.sol";
+
+import {
+    AdvancedOrder,
+    CriteriaResolver,
+    Execution,
+    FulfillmentComponent
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+/**
+ * @title  SeaportRouter
+ * @author Ryan Ghods (ralxz.eth), 0age (0age.eth), James Wenzel (emo.eth)
+ * @notice A utility contract for fulfilling orders with multiple
+ *         Seaport versions. DISCLAIMER: This contract only works when
+ *         all consideration items across all listings are native tokens.
+ */
+contract SeaportRouter is SeaportRouterInterface, ReentrancyGuard {
+    /// @dev The allowed v1.4 contract usable through this router.
+    address private immutable _SEAPORT_V1_4;
+    /// @dev The allowed v1.5 contract usable through this router.
+    address private immutable _SEAPORT_V1_5;
+
+    /**
+     * @dev Deploy contract with the supported Seaport contracts.
+     *
+     * @param seaportV1point4 The address of the Seaport v1.4 contract.
+     * @param seaportV1point5 The address of the Seaport v1.5 contract.
+     */
+    constructor(address seaportV1point4, address seaportV1point5) {
+        _SEAPORT_V1_4 = seaportV1point4;
+        _SEAPORT_V1_5 = seaportV1point5;
+    }
+
+    /**
+     * @dev Fallback function to receive excess ether, in case total amount of
+     *      ether sent is more than the amount required to fulfill the order.
+     */
+    receive() external payable override {
+        // Ensure we only receive ether from Seaport.
+        _assertSeaportAllowed(msg.sender);
+    }
+
+    /**
+     * @notice Fulfill available advanced orders through multiple Seaport
+     *         versions.
+     *         See {SeaportInterface-fulfillAvailableAdvancedOrders}
+     *
+     * @param params The parameters for fulfilling available advanced orders.
+     */
+    function fulfillAvailableAdvancedOrders(
+        FulfillAvailableAdvancedOrdersParams calldata params
+    )
+        external
+        payable
+        override
+        returns (
+            bool[][] memory availableOrders,
+            Execution[][] memory executions
+        )
+    {
+        // Ensure this function cannot be triggered during a reentrant call.
+        _setReentrancyGuard(true);
+
+        // Put the number of Seaport contracts on the stack.
+        uint256 seaportContractsLength = params.seaportContracts.length;
+
+        // Set the availableOrders and executions arrays to the correct length.
+        availableOrders = new bool[][](seaportContractsLength);
+        executions = new Execution[][](seaportContractsLength);
+
+        // Track the number of order fulfillments left.
+        uint256 fulfillmentsLeft = params.maximumFulfilled;
+
+        // To help avoid stack too deep errors, we format the calldata
+        // params in a struct and put it on the stack.
+        AdvancedOrder[] memory emptyAdvancedOrders;
+        CriteriaResolver[] memory emptyCriteriaResolvers;
+        FulfillmentComponent[][] memory emptyFulfillmentComponents;
+        CalldataParams memory calldataParams = CalldataParams({
+            advancedOrders: emptyAdvancedOrders,
+            criteriaResolvers: emptyCriteriaResolvers,
+            offerFulfillments: emptyFulfillmentComponents,
+            considerationFulfillments: emptyFulfillmentComponents,
+            fulfillerConduitKey: params.fulfillerConduitKey,
+            recipient: params.recipient,
+            maximumFulfilled: fulfillmentsLeft
+        });
+
+        // If recipient is not provided assign to msg.sender.
+        if (calldataParams.recipient == address(0)) {
+            calldataParams.recipient = msg.sender;
+        }
+
+        // Iterate through the provided Seaport contracts.
+        for (uint256 i = 0; i < params.seaportContracts.length; ) {
+            // Ensure the provided Seaport contract is allowed.
+            _assertSeaportAllowed(params.seaportContracts[i]);
+
+            // Put the order params on the stack.
+            AdvancedOrderParams calldata orderParams = params
+                .advancedOrderParams[i];
+
+            // Assign the variables to the calldata params.
+            calldataParams.advancedOrders = orderParams.advancedOrders;
+            calldataParams.criteriaResolvers = orderParams.criteriaResolvers;
+            calldataParams.offerFulfillments = orderParams.offerFulfillments;
+            calldataParams.considerationFulfillments = orderParams
+                .considerationFulfillments;
+
+            // Execute the orders, collecting availableOrders and executions.
+            // This is wrapped in a try/catch in case a single order is
+            // executed that is no longer available, leading to a revert
+            // with `NoSpecifiedOrdersAvailable()` that can be ignored.
+            try
+                SeaportInterface(params.seaportContracts[i])
+                    .fulfillAvailableAdvancedOrders{
+                    value: orderParams.etherValue
+                }(
+                    calldataParams.advancedOrders,
+                    calldataParams.criteriaResolvers,
+                    calldataParams.offerFulfillments,
+                    calldataParams.considerationFulfillments,
+                    calldataParams.fulfillerConduitKey,
+                    calldataParams.recipient,
+                    calldataParams.maximumFulfilled
+                )
+            returns (
+                bool[] memory newAvailableOrders,
+                Execution[] memory newExecutions
+            ) {
+                availableOrders[i] = newAvailableOrders;
+                executions[i] = newExecutions;
+
+                // Subtract the number of orders fulfilled.
+                uint256 newAvailableOrdersLength = newAvailableOrders.length;
+                for (uint256 j = 0; j < newAvailableOrdersLength; ) {
+                    if (newAvailableOrders[j]) {
+                        unchecked {
+                            --fulfillmentsLeft;
+                            ++j;
+                        }
+                    }
+                }
+
+                // Break if the maximum number of executions has been reached.
+                if (fulfillmentsLeft == 0) {
+                    break;
+                }
+            } catch (bytes memory data) {
+                // Set initial value of first four bytes of revert data
+                // to the mask.
+                bytes4 customErrorSelector = bytes4(0xffffffff);
+
+                // Utilize assembly to read first four bytes
+                // (if present) directly.
+                assembly {
+                    // Combine original mask with first four bytes of
+                    // revert data.
+                    customErrorSelector := and(
+                        // Data begins after length offset.
+                        mload(add(data, 0x20)),
+                        customErrorSelector
+                    )
+                }
+
+                // Pass through the custom error if the error is
+                // not NoSpecifiedOrdersAvailable()
+                if (
+                    customErrorSelector != NoSpecifiedOrdersAvailable.selector
+                ) {
+                    assembly {
+                        revert(add(data, 32), mload(data))
+                    }
+                }
+            }
+
+            // Update fulfillments left.
+            calldataParams.maximumFulfilled = fulfillmentsLeft;
+
+            unchecked {
+                ++i;
+            }
+        }
+
+        // Throw an error if no orders were fulfilled.
+        if (fulfillmentsLeft == params.maximumFulfilled) {
+            revert NoSpecifiedOrdersAvailable();
+        }
+
+        // Return excess ether that may not have been used or was sent back.
+        if (address(this).balance > 0) {
+            _returnExcessEther();
+        }
+
+        // Clear the reentrancy guard.
+        _clearReentrancyGuard();
+    }
+
+    /**
+     * @notice Returns the Seaport contracts allowed to be used through this
+     *         router.
+     */
+    function getAllowedSeaportContracts()
+        external
+        view
+        override
+        returns (address[] memory seaportContracts)
+    {
+        seaportContracts = new address[](2);
+        seaportContracts[0] = _SEAPORT_V1_4;
+        seaportContracts[1] = _SEAPORT_V1_5;
+    }
+
+    /**
+     * @dev Reverts if the provided Seaport contract is not allowed.
+     */
+    function _assertSeaportAllowed(address seaport) internal view {
+        if (
+            _cast(seaport == _SEAPORT_V1_4) | _cast(seaport == _SEAPORT_V1_5) ==
+            0
+        ) {
+            revert SeaportNotAllowed(seaport);
+        }
+    }
+
+    /**
+     * @dev Function to return excess ether, in case total amount of
+     *      ether sent is more than the amount required to fulfill the order.
+     */
+    function _returnExcessEther() private {
+        // Send received funds back to msg.sender.
+        (bool success, bytes memory data) = payable(msg.sender).call{
+            value: address(this).balance
+        }("");
+
+        // Revert with an error if the ether transfer failed.
+        if (!success) {
+            revert EtherReturnTransferFailed(
+                msg.sender,
+                address(this).balance,
+                data
+            );
+        }
+    }
+}
diff --git a/contracts/seaport/helpers/TransferHelper.sol b/contracts/seaport/helpers/TransferHelper.sol
new file mode 100644
index 00000000..dba38197
--- /dev/null
+++ b/contracts/seaport/helpers/TransferHelper.sol
@@ -0,0 +1,348 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import { IERC721Receiver } from "seaport-types/src/interfaces/IERC721Receiver.sol";
+
+import {
+    TransferHelperItem,
+    TransferHelperItemsWithRecipient
+} from "seaport-types/src/helpers/TransferHelperStructs.sol";
+
+import { ConduitItemType } from "seaport-types/src/conduit/lib/ConduitEnums.sol";
+
+import {
+    ConduitInterface
+} from "seaport-types/src/interfaces/ConduitInterface.sol";
+
+import {
+    ConduitControllerInterface
+} from "seaport-types/src/interfaces/ConduitControllerInterface.sol";
+
+import { ConduitTransfer } from "seaport-types/src/conduit/lib/ConduitStructs.sol";
+
+import {
+    TransferHelperInterface
+} from "seaport-types/src/interfaces/TransferHelperInterface.sol";
+
+import {
+    TransferHelperErrors
+} from "seaport-types/src/interfaces/TransferHelperErrors.sol";
+
+/**
+ * @title TransferHelper
+ * @author stephankmin, stuckinaboot, ryanio
+ * @notice TransferHelper is a utility contract for transferring
+ *         ERC20/ERC721/ERC1155 items in bulk to specific recipients.
+ */
+contract TransferHelper is TransferHelperInterface, TransferHelperErrors {
+    // Allow for interaction with the conduit controller.
+    ConduitControllerInterface internal immutable _CONDUIT_CONTROLLER;
+
+    // Set conduit creation code and runtime code hashes as immutable arguments.
+    bytes32 internal immutable _CONDUIT_CREATION_CODE_HASH;
+    bytes32 internal immutable _CONDUIT_RUNTIME_CODE_HASH;
+
+    /**
+     * @dev Set the supplied conduit controller and retrieve its
+     *      conduit creation code hash.
+     *
+     *
+     * @param conduitController A contract that deploys conduits, or proxies
+     *                          that may optionally be used to transfer approved
+     *                          ERC20/721/1155 tokens.
+     */
+    constructor(address conduitController) {
+        // Get the conduit creation code and runtime code hashes from the
+        // supplied conduit controller and set them as an immutable.
+        ConduitControllerInterface controller = ConduitControllerInterface(
+            conduitController
+        );
+        (_CONDUIT_CREATION_CODE_HASH, _CONDUIT_RUNTIME_CODE_HASH) = controller
+            .getConduitCodeHashes();
+
+        // Set the supplied conduit controller as an immutable.
+        _CONDUIT_CONTROLLER = controller;
+    }
+
+    /**
+     * @notice Transfer multiple ERC20/ERC721/ERC1155 items to
+     *         specified recipients.
+     *
+     * @param items      The items to transfer to an intended recipient.
+     * @param conduitKey A mandatory conduit key referring to a conduit through
+     *                   which the bulk transfer should occur.
+     *
+     * @return magicValue A value indicating that the transfers were successful.
+     */
+    function bulkTransfer(
+        TransferHelperItemsWithRecipient[] calldata items,
+        bytes32 conduitKey
+    ) external override returns (bytes4 magicValue) {
+        // Ensure that a conduit key has been supplied.
+        if (conduitKey == bytes32(0)) {
+            revert InvalidConduit(conduitKey, address(0));
+        }
+
+        // Use conduit derived from supplied conduit key to perform transfers.
+        _performTransfersWithConduit(items, conduitKey);
+
+        // Return a magic value indicating that the transfers were performed.
+        magicValue = this.bulkTransfer.selector;
+    }
+
+    /**
+     * @notice Perform multiple transfers to specified recipients via the
+     *         conduit derived from the provided conduit key.
+     *
+     * @param transfers  The items to transfer.
+     * @param conduitKey The conduit key referring to the conduit through
+     *                   which the bulk transfer should occur.
+     */
+    function _performTransfersWithConduit(
+        TransferHelperItemsWithRecipient[] calldata transfers,
+        bytes32 conduitKey
+    ) internal {
+        // Retrieve total number of transfers and place on stack.
+        uint256 numTransfers = transfers.length;
+
+        // Derive the conduit address from the deployer, conduit key
+        // and creation code hash.
+        address conduit = address(
+            uint160(
+                uint256(
+                    keccak256(
+                        abi.encodePacked(
+                            bytes1(0xff),
+                            address(_CONDUIT_CONTROLLER),
+                            conduitKey,
+                            _CONDUIT_CREATION_CODE_HASH
+                        )
+                    )
+                )
+            )
+        );
+
+        // Declare a variable to store the sum of all items across transfers.
+        uint256 sumOfItemsAcrossAllTransfers;
+
+        // Skip overflow checks: all for loops are indexed starting at zero.
+        unchecked {
+            // Iterate over each transfer.
+            for (uint256 i = 0; i < numTransfers; ++i) {
+                // Retrieve the transfer in question.
+                TransferHelperItemsWithRecipient calldata transfer = transfers[
+                    i
+                ];
+
+                // Increment totalItems by the number of items in the transfer.
+                sumOfItemsAcrossAllTransfers += transfer.items.length;
+            }
+        }
+
+        // Declare a new array in memory with length totalItems to populate with
+        // each conduit transfer.
+        ConduitTransfer[] memory conduitTransfers = new ConduitTransfer[](
+            sumOfItemsAcrossAllTransfers
+        );
+
+        // Declare an index for storing ConduitTransfers in conduitTransfers.
+        uint256 itemIndex;
+
+        // Skip overflow checks: all for loops are indexed starting at zero.
+        unchecked {
+            // Iterate over each transfer.
+            for (uint256 i = 0; i < numTransfers; ++i) {
+                // Retrieve the transfer in question.
+                TransferHelperItemsWithRecipient calldata transfer = transfers[
+                    i
+                ];
+
+                // Retrieve the items of the transfer in question.
+                TransferHelperItem[] calldata transferItems = transfer.items;
+
+                // Ensure recipient is not the zero address.
+                _checkRecipientIsNotZeroAddress(transfer.recipient);
+
+                // Create a boolean indicating whether validateERC721Receiver
+                // is true and recipient is a contract.
+                bool callERC721Receiver = transfer.validateERC721Receiver &&
+                    transfer.recipient.code.length != 0;
+
+                // Retrieve the total number of items in the transfer and
+                // place on stack.
+                uint256 numItemsInTransfer = transferItems.length;
+
+                // Iterate over each item in the transfer to create a
+                // corresponding ConduitTransfer.
+                for (uint256 j = 0; j < numItemsInTransfer; ++j) {
+                    // Retrieve the item from the transfer.
+                    TransferHelperItem calldata item = transferItems[j];
+
+                    if (item.itemType == ConduitItemType.ERC20) {
+                        // Ensure that the identifier of an ERC20 token is 0.
+                        if (item.identifier != 0) {
+                            revert InvalidERC20Identifier();
+                        }
+                    }
+
+                    // If the item is an ERC721 token and
+                    // callERC721Receiver is true...
+                    if (item.itemType == ConduitItemType.ERC721) {
+                        if (callERC721Receiver) {
+                            // Check if the recipient implements
+                            // onERC721Received for the given tokenId.
+                            _checkERC721Receiver(
+                                conduit,
+                                transfer.recipient,
+                                item.identifier
+                            );
+                        }
+                    }
+
+                    // Create a ConduitTransfer corresponding to each
+                    // TransferHelperItem.
+                    conduitTransfers[itemIndex] = ConduitTransfer(
+                        item.itemType,
+                        item.token,
+                        msg.sender,
+                        transfer.recipient,
+                        item.identifier,
+                        item.amount
+                    );
+
+                    // Increment the index for storing ConduitTransfers.
+                    ++itemIndex;
+                }
+            }
+        }
+
+        // Attempt the external call to transfer tokens via the derived conduit.
+        try ConduitInterface(conduit).execute(conduitTransfers) returns (
+            bytes4 conduitMagicValue
+        ) {
+            // Check if the value returned from the external call matches
+            // the conduit `execute` selector.
+            if (conduitMagicValue != ConduitInterface.execute.selector) {
+                // If the external call fails, revert with the conduit key
+                // and conduit address.
+                revert InvalidConduit(conduitKey, conduit);
+            }
+        } catch Error(string memory reason) {
+            // Catch reverts with a provided reason string and
+            // revert with the reason, conduit key and conduit address.
+            revert ConduitErrorRevertString(reason, conduitKey, conduit);
+        } catch (bytes memory data) {
+            // Conduits will throw a custom error when attempting to transfer
+            // native token item types or an ERC721 item amount other than 1.
+            // Bubble up these custom errors when encountered. Note that the
+            // conduit itself will bubble up revert reasons from transfers as
+            // well, meaning that these errors are not necessarily indicative of
+            // an issue with the item type or amount in cases where the same
+            // custom error signature is encountered during a conduit transfer.
+
+            // Set initial value of first four bytes of revert data to the mask.
+            bytes4 customErrorSelector = bytes4(0xffffffff);
+
+            // Utilize assembly to read first four bytes (if present) directly.
+            assembly {
+                // Combine original mask with first four bytes of revert data.
+                customErrorSelector := and(
+                    mload(add(data, 0x20)), // Data begins after length offset.
+                    customErrorSelector
+                )
+            }
+
+            // Pass through the custom error in question if the revert data is
+            // the correct length and matches an expected custom error selector.
+            if (
+                data.length == 4 &&
+                customErrorSelector == InvalidItemType.selector
+            ) {
+                // "Bubble up" the revert reason.
+                assembly {
+                    revert(add(data, 0x20), 0x04)
+                }
+            } else if (
+                data.length == 36 &&
+                customErrorSelector == InvalidERC721TransferAmount.selector
+            ) {
+                // "Bubble up" the revert reason.
+                assembly {
+                    revert(add(data, 0x20), 0x24)
+                }
+            }
+
+            // Catch all other reverts from the external call to the conduit and
+            // include the conduit's raw revert reason as a data argument to a
+            // new custom error.
+            revert ConduitErrorRevertBytes(data, conduitKey, conduit);
+        }
+    }
+
+    /**
+     * @notice An internal function to check if a recipient address implements
+     *         onERC721Received for a given tokenId. Note that this check does
+     *         not adhere to the safe transfer specification and is only meant
+     *         to provide an additional layer of assurance that the recipient
+     *         can receive the tokens — any hooks or post-transfer checks will
+     *         fail and the caller will be the transfer helper rather than the
+     *         ERC721 contract. Note that the conduit is set as the operator, as
+     *         it will be the caller once the transfer is performed.
+     *
+     * @param conduit   The conduit to provide as the operator when calling
+     *                  onERC721Received.
+     * @param recipient The ERC721 recipient on which to call onERC721Received.
+     * @param tokenId   The ERC721 tokenId of the token being transferred.
+     */
+    function _checkERC721Receiver(
+        address conduit,
+        address recipient,
+        uint256 tokenId
+    ) internal {
+        // Check if recipient can receive ERC721 tokens.
+        try
+            IERC721Receiver(recipient).onERC721Received(
+                conduit,
+                msg.sender,
+                tokenId,
+                ""
+            )
+        returns (bytes4 selector) {
+            // Check if onERC721Received selector is valid.
+            if (selector != IERC721Receiver.onERC721Received.selector) {
+                // Revert if recipient cannot accept
+                // ERC721 tokens.
+                revert InvalidERC721Recipient(recipient);
+            }
+        } catch (bytes memory data) {
+            // "Bubble up" recipient's revert reason.
+            revert ERC721ReceiverErrorRevertBytes(
+                data,
+                recipient,
+                msg.sender,
+                tokenId
+            );
+        } catch Error(string memory reason) {
+            // "Bubble up" recipient's revert reason.
+            revert ERC721ReceiverErrorRevertString(
+                reason,
+                recipient,
+                msg.sender,
+                tokenId
+            );
+        }
+    }
+
+    /**
+     * @notice An internal function that reverts if the passed-in recipient
+     *         is the zero address.
+     *
+     * @param recipient The recipient on which to perform the check.
+     */
+    function _checkRecipientIsNotZeroAddress(address recipient) internal pure {
+        // Revert if the recipient is the zero address.
+        if (recipient == address(0x0)) {
+            revert RecipientCannotBeZeroAddress();
+        }
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/SeaportNavigator.sol b/contracts/seaport/helpers/navigator/SeaportNavigator.sol
new file mode 100644
index 00000000..9997b33a
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/SeaportNavigator.sol
@@ -0,0 +1,131 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    ConsiderationInterface
+} from "seaport-types/src/interfaces/ConsiderationInterface.sol";
+import {
+    AdvancedOrder,
+    CriteriaResolver
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    SeaportValidatorInterface
+} from "../order-validator/SeaportValidator.sol";
+
+import { NavigatorContextLib } from "./lib/NavigatorContextLib.sol";
+
+import { CriteriaHelperLib } from "./lib/CriteriaHelperLib.sol";
+
+import {
+    NavigatorContext,
+    NavigatorRequest,
+    NavigatorResponse
+} from "./lib/SeaportNavigatorTypes.sol";
+
+import { SeaportNavigatorInterface } from "./lib/SeaportNavigatorInterface.sol";
+
+import { HelperInterface } from "./lib/HelperInterface.sol";
+
+/**
+ * @notice SeaportNavigator is a helper contract that generates additional
+ *         information useful for fulfilling Seaport orders. Given an array of
+ *         orders and external parameters like caller, recipient, and native
+ *         tokens supplied, SeaportNavigator will validate the orders and
+ *         return associated errors and warnings, recommend a fulfillment
+ *         method, suggest fulfillments, provide execution and order details,
+ *         and optionally generate criteria resolvers from provided token IDs.
+ */
+contract SeaportNavigator is SeaportNavigatorInterface {
+    using NavigatorContextLib for NavigatorContext;
+    using CriteriaHelperLib for uint256[];
+
+    HelperInterface public immutable requestValidator;
+    HelperInterface public immutable criteriaHelper;
+    HelperInterface public immutable validatorHelper;
+    HelperInterface public immutable orderDetailsHelper;
+    HelperInterface public immutable fulfillmentsHelper;
+    HelperInterface public immutable suggestedActionHelper;
+    HelperInterface public immutable executionsHelper;
+
+    HelperInterface[] public helpers;
+
+    constructor(
+        address _requestValidator,
+        address _criteriaHelper,
+        address _validatorHelper,
+        address _orderDetailsHelper,
+        address _fulfillmentsHelper,
+        address _suggestedActionHelper,
+        address _executionsHelper
+    ) {
+        requestValidator = HelperInterface(_requestValidator);
+        helpers.push(requestValidator);
+
+        criteriaHelper = HelperInterface(_criteriaHelper);
+        helpers.push(criteriaHelper);
+
+        validatorHelper = HelperInterface(_validatorHelper);
+        helpers.push(validatorHelper);
+
+        orderDetailsHelper = HelperInterface(_orderDetailsHelper);
+        helpers.push(orderDetailsHelper);
+
+        fulfillmentsHelper = HelperInterface(_fulfillmentsHelper);
+        helpers.push(fulfillmentsHelper);
+
+        suggestedActionHelper = HelperInterface(_suggestedActionHelper);
+        helpers.push(suggestedActionHelper);
+
+        executionsHelper = HelperInterface(_executionsHelper);
+        helpers.push(executionsHelper);
+    }
+
+    function prepare(
+        NavigatorRequest calldata request
+    ) public view returns (NavigatorResponse memory) {
+        NavigatorContext memory context = NavigatorContextLib
+            .from(request)
+            .withEmptyResponse();
+
+        for (uint256 i; i < helpers.length; i++) {
+            context = helpers[i].prepare(context);
+        }
+
+        return context.response;
+    }
+
+    /**
+     * @notice Generate a criteria merkle root from an array of `tokenIds`. Use
+     *         this helper to construct an order item's `identifierOrCriteria`.
+     *
+     * @param tokenIds An array of integer token IDs to be converted to a merkle
+     *                 root.
+     *
+     * @return The bytes32 merkle root of a criteria tree containing the given
+     *         token IDs.
+     */
+    function criteriaRoot(
+        uint256[] memory tokenIds
+    ) external pure returns (bytes32) {
+        return tokenIds.criteriaRoot();
+    }
+
+    /**
+     * @notice Generate a criteria merkle proof that `id` is a member of
+     *        `tokenIds`. Reverts if `id` is not a member of `tokenIds`. Use
+     *         this helper to construct proof data for criteria resolvers.
+     *
+     * @param tokenIds An array of integer token IDs.
+     * @param id       The integer token ID to generate a proof for.
+     *
+     * @return Merkle proof that the given token ID is  amember of the criteria
+     *         tree containing the given token IDs.
+     */
+    function criteriaProof(
+        uint256[] memory tokenIds,
+        uint256 id
+    ) external pure returns (bytes32[] memory) {
+        return tokenIds.criteriaProof(id);
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/CriteriaHelper.sol b/contracts/seaport/helpers/navigator/lib/CriteriaHelper.sol
new file mode 100644
index 00000000..d132a674
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/CriteriaHelper.sol
@@ -0,0 +1,24 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    CriteriaResolver
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    NavigatorCriteriaResolverLib
+} from "./NavigatorCriteriaResolverLib.sol";
+
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+
+import { HelperInterface } from "./HelperInterface.sol";
+
+contract CriteriaHelper is HelperInterface {
+    using NavigatorCriteriaResolverLib for NavigatorContext;
+
+    function prepare(
+        NavigatorContext memory context
+    ) public pure returns (NavigatorContext memory) {
+        return context.withCriteria();
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/CriteriaHelperLib.sol b/contracts/seaport/helpers/navigator/lib/CriteriaHelperLib.sol
new file mode 100644
index 00000000..a50a69ac
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/CriteriaHelperLib.sol
@@ -0,0 +1,131 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { MerkleLib } from "./MerkleLib.sol";
+
+error TokenIdNotFound();
+
+struct HashAndIntTuple {
+    uint256 num;
+    bytes32 hash;
+}
+
+/**
+ * @notice Helper library for calculating criteria resolver Merkle roots and
+ *         proofs from integer token ids.
+ */
+library CriteriaHelperLib {
+    error CannotDeriveRootForSingleTokenId();
+    error CannotDeriveProofForSingleTokenId();
+
+    /**
+     * @notice Calculate the Merkle root of a criteria tree containing the given
+     *         integer token ids.
+     */
+    function criteriaRoot(
+        uint256[] memory tokenIds
+    ) internal pure returns (bytes32) {
+        if (tokenIds.length == 0) {
+            return bytes32(0);
+        } else if (tokenIds.length == 1) {
+            revert CannotDeriveRootForSingleTokenId();
+        } else {
+            return
+                MerkleLib.getRoot(
+                    toSortedHashes(tokenIds),
+                    MerkleLib.merkleHash
+                );
+        }
+    }
+
+    /**
+     * @notice Calculate the Merkle proof that the given token id is a member of
+     *         the criteria tree containing the provided tokenIds.
+     */
+    function criteriaProof(
+        uint256[] memory tokenIds,
+        uint256 id
+    ) internal pure returns (bytes32[] memory) {
+        if (tokenIds.length == 0) {
+            return new bytes32[](0);
+        } else if (tokenIds.length == 1) {
+            revert CannotDeriveProofForSingleTokenId();
+        } else {
+            bytes32 idHash = keccak256(abi.encode(id));
+            uint256 idx;
+            bool found;
+            bytes32[] memory idHashes = toSortedHashes(tokenIds);
+            for (; idx < idHashes.length; idx++) {
+                if (idHashes[idx] == idHash) {
+                    found = true;
+                    break;
+                }
+            }
+            if (!found) revert TokenIdNotFound();
+            return MerkleLib.getProof(idHashes, idx, MerkleLib.merkleHash);
+        }
+    }
+
+    /**
+     * @notice Sort an array of integer token ids by their hashed values.
+     */
+    function sortByHash(
+        uint256[] memory tokenIds
+    ) internal pure returns (uint256[] memory sortedIds) {
+        HashAndIntTuple[] memory toSort = new HashAndIntTuple[](
+            tokenIds.length
+        );
+        for (uint256 i = 0; i < tokenIds.length; i++) {
+            toSort[i] = HashAndIntTuple(
+                tokenIds[i],
+                keccak256(abi.encode(tokenIds[i]))
+            );
+        }
+
+        _quickSort(toSort, 0, int256(toSort.length - 1));
+
+        sortedIds = new uint256[](tokenIds.length);
+        for (uint256 i = 0; i < tokenIds.length; i++) {
+            sortedIds[i] = toSort[i].num;
+        }
+    }
+
+    /**
+     * @notice Convert an array of integer token ids to a sorted array of
+     *         their hashed values.
+     */
+    function toSortedHashes(
+        uint256[] memory tokenIds
+    ) internal pure returns (bytes32[] memory hashes) {
+        hashes = new bytes32[](tokenIds.length);
+        uint256[] memory ids = sortByHash(tokenIds);
+        for (uint256 i; i < ids.length; ++i) {
+            hashes[i] = keccak256(abi.encode(ids[i]));
+        }
+    }
+
+    function _quickSort(
+        HashAndIntTuple[] memory arr,
+        int256 left,
+        int256 right
+    ) internal pure {
+        int256 i = left;
+        int256 j = right;
+        if (i == j) return;
+        bytes32 pivot = arr[uint256(left + (right - left) / 2)].hash;
+        while (i <= j) {
+            while (arr[uint256(i)].hash < pivot) i++;
+            while (pivot < arr[uint256(j)].hash) j--;
+            if (i <= j) {
+                (arr[uint256(i)], arr[uint256(j)]) = (
+                    arr[uint256(j)],
+                    arr[uint256(i)]
+                );
+                i++;
+                j--;
+            }
+        }
+        if (left < j) _quickSort(arr, left, j);
+        if (i < right) _quickSort(arr, i, right);
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/ExecutionsHelper.sol b/contracts/seaport/helpers/navigator/lib/ExecutionsHelper.sol
new file mode 100644
index 00000000..cdfc8841
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/ExecutionsHelper.sol
@@ -0,0 +1,18 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { NavigatorExecutionsLib } from "./NavigatorExecutionsLib.sol";
+
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+
+import { HelperInterface } from "./HelperInterface.sol";
+
+contract ExecutionsHelper is HelperInterface {
+    using NavigatorExecutionsLib for NavigatorContext;
+
+    function prepare(
+        NavigatorContext memory context
+    ) public pure returns (NavigatorContext memory) {
+        return context.withExecutions();
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/FulfillmentsHelper.sol b/contracts/seaport/helpers/navigator/lib/FulfillmentsHelper.sol
new file mode 100644
index 00000000..0109cdee
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/FulfillmentsHelper.sol
@@ -0,0 +1,18 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { NavigatorFulfillmentsLib } from "./NavigatorFulfillmentsLib.sol";
+
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+
+import { HelperInterface } from "./HelperInterface.sol";
+
+contract FulfillmentsHelper is HelperInterface {
+    using NavigatorFulfillmentsLib for NavigatorContext;
+
+    function prepare(
+        NavigatorContext memory context
+    ) public pure returns (NavigatorContext memory) {
+        return context.withFulfillments();
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/HelperInterface.sol b/contracts/seaport/helpers/navigator/lib/HelperInterface.sol
new file mode 100644
index 00000000..34f4f831
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/HelperInterface.sol
@@ -0,0 +1,10 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+
+interface HelperInterface {
+    function prepare(
+        NavigatorContext memory context
+    ) external view returns (NavigatorContext memory);
+}
diff --git a/contracts/seaport/helpers/navigator/lib/HelperItemLib.sol b/contracts/seaport/helpers/navigator/lib/HelperItemLib.sol
new file mode 100644
index 00000000..824057f8
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/HelperItemLib.sol
@@ -0,0 +1,145 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { ItemType } from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    NavigatorOfferItem,
+    NavigatorConsiderationItem
+} from "./SeaportNavigatorTypes.sol";
+
+library HelperItemLib {
+    error InvalidIdentifier(uint256 identifier, uint256[] candidateIdentifiers);
+    error InvalidItemTypeForCandidateIdentifiers();
+
+    /**
+     * @dev Internal error: Could not convert item type.
+     */
+    error UnknownItemType();
+
+    function normalizeType(
+        NavigatorOfferItem memory item
+    ) internal pure returns (ItemType) {
+        ItemType itemType = item.itemType;
+        if (hasCriteria(item)) {
+            if (
+                itemType == ItemType.ERC721 ||
+                itemType == ItemType.ERC721_WITH_CRITERIA
+            ) {
+                return ItemType.ERC721_WITH_CRITERIA;
+            } else if (
+                itemType == ItemType.ERC1155 ||
+                itemType == ItemType.ERC1155_WITH_CRITERIA
+            ) {
+                return ItemType.ERC1155_WITH_CRITERIA;
+            } else {
+                revert UnknownItemType();
+            }
+        } else {
+            return itemType;
+        }
+    }
+
+    function normalizeType(
+        NavigatorConsiderationItem memory item
+    ) internal pure returns (ItemType) {
+        ItemType itemType = item.itemType;
+        if (hasCriteria(item)) {
+            if (
+                itemType == ItemType.ERC721 ||
+                itemType == ItemType.ERC721_WITH_CRITERIA
+            ) {
+                return ItemType.ERC721_WITH_CRITERIA;
+            } else if (
+                itemType == ItemType.ERC1155 ||
+                itemType == ItemType.ERC1155_WITH_CRITERIA
+            ) {
+                return ItemType.ERC1155_WITH_CRITERIA;
+            } else {
+                revert UnknownItemType();
+            }
+        } else {
+            return itemType;
+        }
+    }
+
+    function hasCriteria(
+        NavigatorOfferItem memory item
+    ) internal pure returns (bool) {
+        return item.candidateIdentifiers.length > 0;
+    }
+
+    function hasCriteria(
+        NavigatorConsiderationItem memory item
+    ) internal pure returns (bool) {
+        return item.candidateIdentifiers.length > 0;
+    }
+
+    function validate(NavigatorOfferItem memory item) internal pure {
+        ItemType itemType = item.itemType;
+        if (itemType == ItemType.ERC20 || itemType == ItemType.NATIVE) {
+            if (item.candidateIdentifiers.length > 0) {
+                revert InvalidItemTypeForCandidateIdentifiers();
+            } else {
+                return;
+            }
+        }
+        // If the item has candidate identifiers, the item identifier must be
+        // zero for wildcard or one of the candidates.
+        if (item.candidateIdentifiers.length == 0 && item.identifier == 0) {
+            revert InvalidIdentifier(
+                item.identifier,
+                item.candidateIdentifiers
+            );
+        }
+        if (item.candidateIdentifiers.length > 0) {
+            bool identifierFound;
+            for (uint256 i; i < item.candidateIdentifiers.length; i++) {
+                if (item.candidateIdentifiers[i] == item.identifier) {
+                    identifierFound = true;
+                    break;
+                }
+            }
+            if (!identifierFound && item.identifier != 0) {
+                revert InvalidIdentifier(
+                    item.identifier,
+                    item.candidateIdentifiers
+                );
+            }
+        }
+    }
+
+    function validate(NavigatorConsiderationItem memory item) internal pure {
+        ItemType itemType = item.itemType;
+        if (itemType == ItemType.ERC20 || itemType == ItemType.NATIVE) {
+            if (item.candidateIdentifiers.length > 0) {
+                revert InvalidItemTypeForCandidateIdentifiers();
+            } else {
+                return;
+            }
+        }
+        // If the item has candidate identifiers, the item identifier must be
+        // zero for wildcard or one of the candidates.
+        if (item.candidateIdentifiers.length == 0 && item.identifier == 0) {
+            revert InvalidIdentifier(
+                item.identifier,
+                item.candidateIdentifiers
+            );
+        }
+        if (item.candidateIdentifiers.length > 0) {
+            bool identifierFound;
+            for (uint256 i; i < item.candidateIdentifiers.length; i++) {
+                if (item.candidateIdentifiers[i] == item.identifier) {
+                    identifierFound = true;
+                    break;
+                }
+            }
+            if (!identifierFound && item.identifier != 0) {
+                revert InvalidIdentifier(
+                    item.identifier,
+                    item.candidateIdentifiers
+                );
+            }
+        }
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/MerkleLib.sol b/contracts/seaport/helpers/navigator/lib/MerkleLib.sol
new file mode 100644
index 00000000..f0be29e0
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/MerkleLib.sol
@@ -0,0 +1,218 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+/**
+ * @notice A pure library for generating Merkle trees/proofs, adapted from
+ * Murky: https://github.com/dmfxyz/murky
+ */
+library MerkleLib {
+    function merkleHash(
+        bytes32 left,
+        bytes32 right
+    ) internal pure returns (bytes32 _hash) {
+        assembly {
+            switch lt(left, right)
+            case 0 {
+                mstore(0x0, right)
+                mstore(0x20, left)
+            }
+            default {
+                mstore(0x0, left)
+                mstore(0x20, right)
+            }
+            _hash := keccak256(0x0, 0x40)
+        }
+    }
+
+    function xorkleHash(
+        bytes32 left,
+        bytes32 right
+    ) internal pure returns (bytes32 _hash) {
+        assembly {
+            mstore(0x0, xor(left, right))
+            _hash := keccak256(0x0, 0x20)
+        }
+    }
+
+    function verifyProof(
+        bytes32 root,
+        bytes32[] memory proof,
+        bytes32 valueToProve,
+        function(bytes32, bytes32) internal pure returns (bytes32) hashLeafPairs
+    ) internal pure returns (bool) {
+        // proof length must be less than max array size
+        bytes32 rollingHash = valueToProve;
+        uint256 length = proof.length;
+        unchecked {
+            for (uint i = 0; i < length; ++i) {
+                rollingHash = hashLeafPairs(rollingHash, proof[i]);
+            }
+        }
+        return root == rollingHash;
+    }
+
+    function getRoot(
+        bytes32[] memory data,
+        function(bytes32, bytes32) internal pure returns (bytes32) hashLeafPairs
+    ) internal pure returns (bytes32) {
+        require(data.length > 1, "won't generate root for single leaf");
+        while (data.length > 1) {
+            data = hashLevel(data, hashLeafPairs);
+        }
+        return data[0];
+    }
+
+    function getProof(
+        bytes32[] memory data,
+        uint256 node,
+        function(bytes32, bytes32) internal pure returns (bytes32) hashLeafPairs
+    ) internal pure returns (bytes32[] memory) {
+        require(data.length > 1, "won't generate proof for single leaf");
+        // The size of the proof is equal to the ceiling of log2(numLeaves)
+        bytes32[] memory result = new bytes32[](log2ceilBitMagic(data.length));
+        uint256 pos = 0;
+
+        // Two overflow risks: node, pos
+        // node: max array size is 2**256-1. Largest index in the array will be 1 less than that. Also,
+        // for dynamic arrays, size is limited to 2**64-1
+        // pos: pos is bounded by log2(data.length), which should be less than type(uint256).max
+        while (data.length > 1) {
+            unchecked {
+                if (node & 0x1 == 1) {
+                    result[pos] = data[node - 1];
+                } else if (node + 1 == data.length) {
+                    result[pos] = bytes32(0);
+                } else {
+                    result[pos] = data[node + 1];
+                }
+                ++pos;
+                node /= 2;
+            }
+            data = hashLevel(data, hashLeafPairs);
+        }
+        return result;
+    }
+
+    ///@dev function is private to prevent unsafe data from being passed
+    function hashLevel(
+        bytes32[] memory data,
+        function(bytes32, bytes32) internal pure returns (bytes32) hashLeafPairs
+    ) private pure returns (bytes32[] memory) {
+        bytes32[] memory result;
+
+        // Function is private, and all internal callers check that data.length >=2.
+        // Underflow is not possible as lowest possible value for data/result index is 1
+        // overflow should be safe as length is / 2 always.
+        unchecked {
+            uint256 length = data.length;
+            if (length & 0x1 == 1) {
+                result = new bytes32[](length / 2 + 1);
+                result[result.length - 1] = hashLeafPairs(
+                    data[length - 1],
+                    bytes32(0)
+                );
+            } else {
+                result = new bytes32[](length / 2);
+            }
+            // pos is upper bounded by data.length / 2, so safe even if array is at max size
+            uint256 pos = 0;
+            for (uint256 i = 0; i < length - 1; i += 2) {
+                result[pos] = hashLeafPairs(data[i], data[i + 1]);
+                ++pos;
+            }
+        }
+        return result;
+    }
+
+    /******************
+     * MATH "LIBRARY" *
+     ******************/
+
+    /// @dev  Note that x is assumed > 0
+    function log2ceil(uint256 x) internal pure returns (uint256) {
+        uint256 ceil = 0;
+        uint pOf2;
+        // If x is a power of 2, then this function will return a ceiling
+        // that is 1 greater than the actual ceiling. So we need to check if
+        // x is a power of 2, and subtract one from ceil if so.
+        assembly {
+            // we check by seeing if x == (~x + 1) & x. This applies a mask
+            // to find the lowest set bit of x and then checks it for equality
+            // with x. If they are equal, then x is a power of 2.
+
+            /* Example
+                x has single bit set
+                x := 0000_1000
+                (~x + 1) = (1111_0111) + 1 = 1111_1000
+                (1111_1000 & 0000_1000) = 0000_1000 == x
+
+                x has multiple bits set
+                x := 1001_0010
+                (~x + 1) = (0110_1101 + 1) = 0110_1110
+                (0110_1110 & x) = 0000_0010 != x
+            */
+
+            // we do some assembly magic to treat the bool as an integer later on
+            pOf2 := eq(and(add(not(x), 1), x), x)
+        }
+
+        // if x == type(uint256).max, than ceil is capped at 256
+        // if x == 0, then pO2 == 0, so ceil won't underflow
+        unchecked {
+            while (x > 0) {
+                x >>= 1;
+                ceil++;
+            }
+            ceil -= pOf2; // see above
+        }
+        return ceil;
+    }
+
+    /// Original bitmagic adapted from https://github.com/paulrberg/prb-math/blob/main/contracts/PRBMath.sol
+    /// @dev Note that x assumed > 1
+    function log2ceilBitMagic(uint256 x) internal pure returns (uint256) {
+        if (x <= 1) {
+            return 0;
+        }
+        uint256 msb = 0;
+        uint256 _x = x;
+        if (x >= 2 ** 128) {
+            x >>= 128;
+            msb += 128;
+        }
+        if (x >= 2 ** 64) {
+            x >>= 64;
+            msb += 64;
+        }
+        if (x >= 2 ** 32) {
+            x >>= 32;
+            msb += 32;
+        }
+        if (x >= 2 ** 16) {
+            x >>= 16;
+            msb += 16;
+        }
+        if (x >= 2 ** 8) {
+            x >>= 8;
+            msb += 8;
+        }
+        if (x >= 2 ** 4) {
+            x >>= 4;
+            msb += 4;
+        }
+        if (x >= 2 ** 2) {
+            x >>= 2;
+            msb += 2;
+        }
+        if (x >= 2 ** 1) {
+            msb += 1;
+        }
+
+        uint256 lsb = (~_x + 1) & _x;
+        if ((lsb == _x) && (msb > 0)) {
+            return msb;
+        } else {
+            return msb + 1;
+        }
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/NavigatorAdvancedOrderLib.sol b/contracts/seaport/helpers/navigator/lib/NavigatorAdvancedOrderLib.sol
new file mode 100644
index 00000000..2f99fc17
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/NavigatorAdvancedOrderLib.sol
@@ -0,0 +1,246 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    AdvancedOrder,
+    CriteriaResolver,
+    OfferItem,
+    ConsiderationItem,
+    OrderParameters
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import { Side } from "seaport-types/src/lib/ConsiderationEnums.sol";
+
+import { CriteriaHelperLib } from "./CriteriaHelperLib.sol";
+import { HelperItemLib } from "./HelperItemLib.sol";
+import {
+    NavigatorAdvancedOrder,
+    NavigatorOrderParameters,
+    NavigatorOfferItem,
+    NavigatorConsiderationItem
+} from "./SeaportNavigatorTypes.sol";
+
+library NavigatorAdvancedOrderLib {
+    using CriteriaHelperLib for uint256[];
+    using HelperItemLib for NavigatorOfferItem;
+    using HelperItemLib for NavigatorConsiderationItem;
+
+    function fromAdvancedOrders(
+        AdvancedOrder[] memory orders
+    ) internal pure returns (NavigatorAdvancedOrder[] memory) {
+        NavigatorAdvancedOrder[]
+            memory helperOrders = new NavigatorAdvancedOrder[](orders.length);
+        for (uint256 i; i < orders.length; i++) {
+            helperOrders[i] = fromAdvancedOrder(orders[i]);
+        }
+        return helperOrders;
+    }
+
+    function fromAdvancedOrder(
+        AdvancedOrder memory order
+    ) internal pure returns (NavigatorAdvancedOrder memory) {
+        NavigatorOfferItem[] memory offerItems = new NavigatorOfferItem[](
+            order.parameters.offer.length
+        );
+        for (uint256 i; i < order.parameters.offer.length; i++) {
+            OfferItem memory item = order.parameters.offer[i];
+            offerItems[i] = NavigatorOfferItem({
+                itemType: item.itemType,
+                token: item.token,
+                identifier: item.identifierOrCriteria,
+                startAmount: item.startAmount,
+                endAmount: item.endAmount,
+                candidateIdentifiers: new uint256[](0)
+            });
+        }
+        NavigatorConsiderationItem[]
+            memory considerationItems = new NavigatorConsiderationItem[](
+                order.parameters.consideration.length
+            );
+        for (uint256 i; i < order.parameters.consideration.length; i++) {
+            ConsiderationItem memory item = order.parameters.consideration[i];
+            considerationItems[i] = NavigatorConsiderationItem({
+                itemType: item.itemType,
+                token: item.token,
+                identifier: item.identifierOrCriteria,
+                startAmount: item.startAmount,
+                endAmount: item.endAmount,
+                recipient: item.recipient,
+                candidateIdentifiers: new uint256[](0)
+            });
+        }
+        return
+            NavigatorAdvancedOrder({
+                parameters: NavigatorOrderParameters({
+                    offerer: order.parameters.offerer,
+                    zone: order.parameters.zone,
+                    offer: offerItems,
+                    consideration: considerationItems,
+                    orderType: order.parameters.orderType,
+                    startTime: order.parameters.startTime,
+                    endTime: order.parameters.endTime,
+                    zoneHash: order.parameters.zoneHash,
+                    salt: order.parameters.salt,
+                    conduitKey: order.parameters.conduitKey,
+                    totalOriginalConsiderationItems: order
+                        .parameters
+                        .totalOriginalConsiderationItems
+                }),
+                numerator: order.numerator,
+                denominator: order.denominator,
+                signature: order.signature,
+                extraData: order.extraData
+            });
+    }
+
+    function toAdvancedOrder(
+        NavigatorAdvancedOrder memory order,
+        uint256 orderIndex
+    ) internal pure returns (AdvancedOrder memory, CriteriaResolver[] memory) {
+        CriteriaResolver[] memory criteriaResolvers = new CriteriaResolver[](
+            order.parameters.offer.length +
+                order.parameters.consideration.length
+        );
+        uint256 criteriaResolverLen;
+        OfferItem[] memory offer = new OfferItem[](
+            order.parameters.offer.length
+        );
+        for (uint256 i; i < order.parameters.offer.length; i++) {
+            NavigatorOfferItem memory item = order.parameters.offer[i];
+            if (item.hasCriteria()) {
+                item.validate();
+                offer[i] = OfferItem({
+                    itemType: item.normalizeType(),
+                    token: item.token,
+                    identifierOrCriteria: uint256(
+                        item.candidateIdentifiers.criteriaRoot()
+                    ),
+                    startAmount: item.startAmount,
+                    endAmount: item.endAmount
+                });
+                criteriaResolvers[criteriaResolverLen] = CriteriaResolver({
+                    orderIndex: orderIndex,
+                    side: Side.OFFER,
+                    index: i,
+                    identifier: item.identifier,
+                    criteriaProof: item.candidateIdentifiers.criteriaProof(
+                        item.identifier
+                    )
+                });
+                criteriaResolverLen++;
+            } else {
+                offer[i] = OfferItem({
+                    itemType: item.itemType,
+                    token: item.token,
+                    identifierOrCriteria: item.identifier,
+                    startAmount: item.startAmount,
+                    endAmount: item.endAmount
+                });
+            }
+        }
+        ConsiderationItem[] memory consideration = new ConsiderationItem[](
+            order.parameters.consideration.length
+        );
+        for (uint256 i; i < order.parameters.consideration.length; i++) {
+            NavigatorConsiderationItem memory item = order
+                .parameters
+                .consideration[i];
+            if (item.hasCriteria()) {
+                item.validate();
+                consideration[i] = ConsiderationItem({
+                    itemType: item.normalizeType(),
+                    token: item.token,
+                    identifierOrCriteria: uint256(
+                        item.candidateIdentifiers.criteriaRoot()
+                    ),
+                    startAmount: item.startAmount,
+                    endAmount: item.endAmount,
+                    recipient: item.recipient
+                });
+                criteriaResolvers[criteriaResolverLen] = CriteriaResolver({
+                    orderIndex: orderIndex,
+                    side: Side.CONSIDERATION,
+                    index: i,
+                    identifier: item.identifier,
+                    criteriaProof: item.candidateIdentifiers.criteriaProof(
+                        item.identifier
+                    )
+                });
+                criteriaResolverLen++;
+            } else {
+                consideration[i] = ConsiderationItem({
+                    itemType: item.itemType,
+                    token: item.token,
+                    identifierOrCriteria: item.identifier,
+                    startAmount: item.startAmount,
+                    endAmount: item.endAmount,
+                    recipient: item.recipient
+                });
+            }
+        }
+        assembly {
+            mstore(criteriaResolvers, criteriaResolverLen)
+        }
+        return (
+            AdvancedOrder({
+                parameters: OrderParameters({
+                    offerer: order.parameters.offerer,
+                    zone: order.parameters.zone,
+                    offer: offer,
+                    consideration: consideration,
+                    orderType: order.parameters.orderType,
+                    startTime: order.parameters.startTime,
+                    endTime: order.parameters.endTime,
+                    zoneHash: order.parameters.zoneHash,
+                    salt: order.parameters.salt,
+                    conduitKey: order.parameters.conduitKey,
+                    totalOriginalConsiderationItems: order
+                        .parameters
+                        .totalOriginalConsiderationItems
+                }),
+                numerator: order.numerator,
+                denominator: order.denominator,
+                signature: order.signature,
+                extraData: order.extraData
+            }),
+            criteriaResolvers
+        );
+    }
+
+    function toAdvancedOrders(
+        NavigatorAdvancedOrder[] memory orders
+    )
+        internal
+        pure
+        returns (AdvancedOrder[] memory, CriteriaResolver[] memory)
+    {
+        AdvancedOrder[] memory advancedOrders = new AdvancedOrder[](
+            orders.length
+        );
+        uint256 maxCriteriaResolvers;
+        for (uint256 i; i < orders.length; i++) {
+            NavigatorOrderParameters memory parameters = orders[i].parameters;
+            maxCriteriaResolvers += (parameters.offer.length +
+                parameters.consideration.length);
+        }
+        uint256 criteriaResolverIndex;
+        CriteriaResolver[] memory criteriaResolvers = new CriteriaResolver[](
+            maxCriteriaResolvers
+        );
+        for (uint256 i = 0; i < orders.length; i++) {
+            (
+                AdvancedOrder memory order,
+                CriteriaResolver[] memory orderResolvers
+            ) = toAdvancedOrder(orders[i], i);
+            advancedOrders[i] = order;
+            for (uint256 j; j < orderResolvers.length; j++) {
+                criteriaResolvers[criteriaResolverIndex] = orderResolvers[j];
+                criteriaResolverIndex++;
+            }
+        }
+        assembly {
+            mstore(criteriaResolvers, criteriaResolverIndex)
+        }
+        return (advancedOrders, criteriaResolvers);
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/NavigatorContextLib.sol b/contracts/seaport/helpers/navigator/lib/NavigatorContextLib.sol
new file mode 100644
index 00000000..a00ea9f0
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/NavigatorContextLib.sol
@@ -0,0 +1,66 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    ConsiderationInterface
+} from "seaport-types/src/interfaces/ConsiderationInterface.sol";
+
+import {
+    MatchComponent
+} from "seaport-sol/src/lib/types/MatchComponentType.sol";
+import { OrderDetails } from "seaport-sol/src/fulfillments/lib/Structs.sol";
+
+import {
+    AdvancedOrder,
+    Execution,
+    Fulfillment,
+    FulfillmentComponent,
+    CriteriaResolver
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    NavigatorRequest,
+    NavigatorResponse,
+    NavigatorContext,
+    NavigatorOfferItem,
+    NavigatorConsiderationItem,
+    NavigatorOrderParameters,
+    NavigatorAdvancedOrder
+} from "./SeaportNavigatorTypes.sol";
+
+import {
+    SeaportValidatorInterface,
+    ErrorsAndWarnings
+} from "../../order-validator/SeaportValidator.sol";
+
+library NavigatorContextLib {
+    function from(
+        NavigatorRequest memory request
+    ) internal pure returns (NavigatorContext memory context) {
+        context.request = request;
+    }
+
+    function withEmptyResponse(
+        NavigatorContext memory context
+    ) internal pure returns (NavigatorContext memory) {
+        context.response = NavigatorResponse({
+            orders: new AdvancedOrder[](0),
+            criteriaResolvers: new CriteriaResolver[](0),
+            suggestedActionName: "",
+            suggestedCallData: hex"",
+            validationErrors: new ErrorsAndWarnings[](0),
+            orderDetails: new OrderDetails[](0),
+            offerFulfillments: new FulfillmentComponent[][](0),
+            considerationFulfillments: new FulfillmentComponent[][](0),
+            fulfillments: new Fulfillment[](0),
+            unspentOfferComponents: new MatchComponent[](0),
+            unmetConsiderationComponents: new MatchComponent[](0),
+            explicitExecutions: new Execution[](0),
+            implicitExecutions: new Execution[](0),
+            implicitExecutionsPre: new Execution[](0),
+            implicitExecutionsPost: new Execution[](0),
+            nativeTokensReturned: 0
+        });
+        return context;
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/NavigatorCriteriaResolverLib.sol b/contracts/seaport/helpers/navigator/lib/NavigatorCriteriaResolverLib.sol
new file mode 100644
index 00000000..4198760a
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/NavigatorCriteriaResolverLib.sol
@@ -0,0 +1,44 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    AdvancedOrder,
+    CriteriaResolver
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    NavigatorAdvancedOrder,
+    NavigatorContext
+} from "./SeaportNavigatorTypes.sol";
+
+import { NavigatorAdvancedOrderLib } from "./NavigatorAdvancedOrderLib.sol";
+
+library NavigatorCriteriaResolverLib {
+    using NavigatorAdvancedOrderLib for NavigatorAdvancedOrder[];
+
+    /**
+     * @dev Calculate criteria resolvers, merkle proofs, and criteria merkle
+     *      roots for the provided orders and criteria constraints. Modifies
+     *      orders in place to add criteria merkle roots to the appropriate
+     *      offer/consdieration items. Adds calculated criteria resolvers to
+     *      the NavigatorResponse.
+     */
+    function withCriteria(
+        NavigatorContext memory context
+    ) internal pure returns (NavigatorContext memory) {
+        (
+            AdvancedOrder[] memory orders,
+            CriteriaResolver[] memory resolvers
+        ) = context.request.orders.toAdvancedOrders();
+        context.response.orders = orders;
+        if (context.request.criteriaResolvers.length > 0) {
+            context.response.criteriaResolvers = context
+                .request
+                .criteriaResolvers;
+            return context;
+        } else {
+            context.response.criteriaResolvers = resolvers;
+            return context;
+        }
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/NavigatorDetailsLib.sol b/contracts/seaport/helpers/navigator/lib/NavigatorDetailsLib.sol
new file mode 100644
index 00000000..3047208c
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/NavigatorDetailsLib.sol
@@ -0,0 +1,37 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { UnavailableReason } from "seaport-sol/src/SpaceEnums.sol";
+import { AdvancedOrder } from "seaport-types/src/lib/ConsiderationStructs.sol";
+import { AdvancedOrderLib } from "seaport-sol/src/lib/AdvancedOrderLib.sol";
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+import { OrderAvailabilityLib } from "./OrderAvailabilityLib.sol";
+
+library NavigatorDetailsLib {
+    using AdvancedOrderLib for AdvancedOrder[];
+    using OrderAvailabilityLib for AdvancedOrder[];
+
+    /**
+     * @dev Calculate OrderDetails for each order and add them to the NavigatorResponse.
+     */
+    function withDetails(
+        NavigatorContext memory context
+    ) internal view returns (NavigatorContext memory) {
+        UnavailableReason[] memory unavailableReasons = context
+            .response
+            .orders
+            .unavailableReasons(
+                context.request.maximumFulfilled,
+                context.request.seaport
+            );
+        bytes32[] memory orderHashes = context.response.orders.getOrderHashes(
+            address(context.request.seaport)
+        );
+        context.response.orderDetails = context.response.orders.getOrderDetails(
+            context.response.criteriaResolvers,
+            orderHashes,
+            unavailableReasons
+        );
+        return context;
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/NavigatorExecutionsLib.sol b/contracts/seaport/helpers/navigator/lib/NavigatorExecutionsLib.sol
new file mode 100644
index 00000000..5941f082
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/NavigatorExecutionsLib.sol
@@ -0,0 +1,118 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    ConsiderationInterface
+} from "seaport-types/src/interfaces/ConsiderationInterface.sol";
+
+import {
+    AdvancedOrder,
+    Execution,
+    SpentItem,
+    ReceivedItem
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import { ItemType } from "seaport-types/src/lib/ConsiderationEnums.sol";
+
+import {
+    FulfillmentDetails
+} from "seaport-sol/src/fulfillments/lib/Structs.sol";
+
+import {
+    ExecutionHelper
+} from "seaport-sol/src/executions/ExecutionHelper.sol";
+
+import { UnavailableReason } from "seaport-sol/src/SpaceEnums.sol";
+
+import { OrderDetails } from "seaport-sol/src/fulfillments/lib/Structs.sol";
+
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+
+library NavigatorExecutionsLib {
+    using ExecutionHelper for FulfillmentDetails;
+
+    /**
+     * @dev Internal error: Could not select a fulfillment method for the provided
+     *      orders.
+     */
+    error UnknownAction();
+
+    /**
+     * @dev Calculate executions for the provided orders and add them to the
+     *      NavigatorResponse.
+     */
+    function withExecutions(
+        NavigatorContext memory context
+    ) internal pure returns (NavigatorContext memory) {
+        bytes memory callData = context.response.suggestedCallData;
+        bytes4 _suggestedAction = bytes4(callData);
+        FulfillmentDetails memory fulfillmentDetails = FulfillmentDetails({
+            orders: context.response.orderDetails,
+            recipient: payable(context.request.recipient),
+            fulfiller: payable(context.request.caller),
+            nativeTokensSupplied: context.request.nativeTokensSupplied,
+            fulfillerConduitKey: context.request.fulfillerConduitKey,
+            seaport: address(context.request.seaport)
+        });
+
+        Execution[] memory explicitExecutions;
+        Execution[] memory implicitExecutions;
+        Execution[] memory implicitExecutionsPre;
+        Execution[] memory implicitExecutionsPost;
+        uint256 nativeTokensReturned;
+
+        if (
+            _suggestedAction ==
+            ConsiderationInterface.fulfillAvailableOrders.selector ||
+            _suggestedAction ==
+            ConsiderationInterface.fulfillAvailableAdvancedOrders.selector
+        ) {
+            (
+                explicitExecutions,
+                implicitExecutionsPre,
+                implicitExecutionsPost,
+                nativeTokensReturned
+            ) = fulfillmentDetails.getFulfillAvailableExecutions(
+                context.response.offerFulfillments,
+                context.response.considerationFulfillments,
+                context.response.orderDetails
+            );
+        } else if (
+            _suggestedAction == ConsiderationInterface.matchOrders.selector ||
+            _suggestedAction ==
+            ConsiderationInterface.matchAdvancedOrders.selector
+        ) {
+            (
+                explicitExecutions,
+                implicitExecutionsPre,
+                implicitExecutionsPost,
+                nativeTokensReturned
+            ) = fulfillmentDetails.getMatchExecutions(
+                context.response.fulfillments
+            );
+        } else if (
+            _suggestedAction == ConsiderationInterface.fulfillOrder.selector ||
+            _suggestedAction ==
+            ConsiderationInterface.fulfillAdvancedOrder.selector
+        ) {
+            (implicitExecutions, nativeTokensReturned) = fulfillmentDetails
+                .getStandardExecutions();
+        } else if (
+            _suggestedAction ==
+            ConsiderationInterface.fulfillBasicOrder.selector ||
+            _suggestedAction ==
+            ConsiderationInterface.fulfillBasicOrder_efficient_6GL6yc.selector
+        ) {
+            (implicitExecutions, nativeTokensReturned) = fulfillmentDetails
+                .getBasicExecutions();
+        } else {
+            revert UnknownAction();
+        }
+        context.response.explicitExecutions = explicitExecutions;
+        context.response.implicitExecutions = implicitExecutions;
+        context.response.implicitExecutionsPre = implicitExecutionsPre;
+        context.response.implicitExecutionsPost = implicitExecutionsPost;
+        context.response.nativeTokensReturned = nativeTokensReturned;
+        return context;
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/NavigatorFulfillmentsLib.sol b/contracts/seaport/helpers/navigator/lib/NavigatorFulfillmentsLib.sol
new file mode 100644
index 00000000..1cc18a60
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/NavigatorFulfillmentsLib.sol
@@ -0,0 +1,51 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { Fulfillment } from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    FulfillmentGeneratorLib
+} from "seaport-sol/src/fulfillments/lib/FulfillmentLib.sol";
+
+import {
+    FulfillmentComponent,
+    MatchComponent,
+    OrderDetails
+} from "seaport-sol/src/fulfillments/lib/Structs.sol";
+
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+
+library NavigatorFulfillmentsLib {
+    using FulfillmentGeneratorLib for OrderDetails[];
+
+    /**
+     * @dev Calculate fulfillments and match components for the provided orders
+     *      and add them to the NavigatorResponse.
+     */
+    function withFulfillments(
+        NavigatorContext memory context
+    ) internal pure returns (NavigatorContext memory) {
+        (
+            ,
+            FulfillmentComponent[][] memory offerFulfillments,
+            FulfillmentComponent[][] memory considerationFulfillments,
+            Fulfillment[] memory fulfillments,
+            MatchComponent[] memory unspentOfferComponents,
+            MatchComponent[] memory unmetConsiderationComponents
+        ) = context.response.orderDetails.getFulfillments(
+                context.request.fulfillmentStrategy,
+                context.request.recipient,
+                context.request.caller,
+                context.request.seed
+            );
+
+        context.response.offerFulfillments = offerFulfillments;
+        context.response.considerationFulfillments = considerationFulfillments;
+        context.response.fulfillments = fulfillments;
+        context.response.unspentOfferComponents = unspentOfferComponents;
+        context
+            .response
+            .unmetConsiderationComponents = unmetConsiderationComponents;
+        return context;
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/NavigatorRequestValidatorLib.sol b/contracts/seaport/helpers/navigator/lib/NavigatorRequestValidatorLib.sol
new file mode 100644
index 00000000..77922d9e
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/NavigatorRequestValidatorLib.sol
@@ -0,0 +1,43 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { AdvancedOrder } from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import { Type, OrderStructureLib } from "./OrderStructureLib.sol";
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+
+library NavigatorRequestValidatorLib {
+    using OrderStructureLib for AdvancedOrder;
+
+    /**
+     * @dev Bad request error: provided orders include at least one contract order.
+     *      The order helper does not currently support contract orders.
+     */
+    error ContractOrdersNotSupported();
+
+    /**
+     * @dev Validate the provided orders. Checks that none of the provided orders
+     *      are contract orders and applies basic criteria constraint validations.
+     */
+    function validate(
+        NavigatorContext memory context
+    ) internal pure returns (NavigatorContext memory) {
+        validateNoContractOrders(context);
+        return context;
+    }
+
+    /**
+     * @dev Checks that none of the provided orders are contract orders.
+     */
+    function validateNoContractOrders(
+        NavigatorContext memory context
+    ) internal pure returns (NavigatorContext memory) {
+        for (uint256 i; i < context.response.orders.length; i++) {
+            AdvancedOrder memory order = context.response.orders[i];
+            if (order.getType() == Type.CONTRACT) {
+                revert ContractOrdersNotSupported();
+            }
+        }
+        return context;
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/NavigatorSeaportValidatorLib.sol b/contracts/seaport/helpers/navigator/lib/NavigatorSeaportValidatorLib.sol
new file mode 100644
index 00000000..5bbe7779
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/NavigatorSeaportValidatorLib.sol
@@ -0,0 +1,37 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { AdvancedOrder } from "seaport-types/src/lib/ConsiderationStructs.sol";
+import { AdvancedOrderLib } from "seaport-sol/src/lib/AdvancedOrderLib.sol";
+
+import { ErrorsAndWarnings } from "../../order-validator/SeaportValidator.sol";
+
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+
+library NavigatorSeaportValidatorLib {
+    using AdvancedOrderLib for AdvancedOrder;
+
+    error ValidatorReverted();
+
+    /**
+     * @dev Validate each order using SeaportValidator and add the results to
+     *      the NavigatorResponse.
+     */
+    function withErrors(
+        NavigatorContext memory context
+    ) internal view returns (NavigatorContext memory) {
+        AdvancedOrder[] memory orders = context.response.orders;
+
+        ErrorsAndWarnings[] memory errors = new ErrorsAndWarnings[](
+            orders.length
+        );
+        for (uint256 i; i < orders.length; i++) {
+            errors[i] = context.request.validator.isValidOrder(
+                orders[i].toOrder(),
+                address(context.request.seaport)
+            );
+        }
+        context.response.validationErrors = errors;
+        return context;
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/NavigatorSuggestedActionLib.sol b/contracts/seaport/helpers/navigator/lib/NavigatorSuggestedActionLib.sol
new file mode 100644
index 00000000..cd30c344
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/NavigatorSuggestedActionLib.sol
@@ -0,0 +1,362 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    ConsiderationInterface
+} from "seaport-types/src/interfaces/ConsiderationInterface.sol";
+
+import {
+    AdvancedOrder,
+    Execution,
+    SpentItem,
+    ReceivedItem
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import { ItemType } from "seaport-types/src/lib/ConsiderationEnums.sol";
+
+import { UnavailableReason } from "seaport-sol/src/SpaceEnums.sol";
+
+import { OrderDetails } from "seaport-sol/src/fulfillments/lib/Structs.sol";
+
+import { AdvancedOrderLib } from "seaport-sol/src/lib/AdvancedOrderLib.sol";
+
+import { Family, Structure, OrderStructureLib } from "./OrderStructureLib.sol";
+
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+
+library NavigatorSuggestedActionLib {
+    using OrderStructureLib for AdvancedOrder;
+    using OrderStructureLib for AdvancedOrder[];
+    using AdvancedOrderLib for AdvancedOrder;
+    using AdvancedOrderLib for AdvancedOrder[];
+
+    /**
+     * @dev Bad request error: provided orders cannot be fulfilled.
+     */
+    error CannotFulfillProvidedCombinedOrder();
+
+    /**
+     * @dev Bad request error: provided orders include an invalid combination of
+     *      native tokens and unavailable orders.
+     */
+    error InvalidNativeTokenUnavailableCombination();
+
+    /**
+     * @dev Internal error: Could not find selector for the suggested action.
+     */
+    error UnknownSelector();
+
+    /**
+     * @dev Choose a suggested fulfillment method based on the structure of the
+     *      orders and add it to the NavigatorResponse.
+     */
+    function withSuggestedAction(
+        NavigatorContext memory context
+    ) internal view returns (NavigatorContext memory) {
+        bytes memory callData = suggestedCallData(context);
+        bytes4 selector = bytes4(callData);
+        context.response.suggestedActionName = actionName(selector);
+        context.response.suggestedCallData = callData;
+        return context;
+    }
+
+    /**
+     * @dev Add the human-readable name of the selected fulfillment method to
+     *      the NavigatorResponse.
+     */
+    function actionName(bytes4 selector) internal pure returns (string memory) {
+        if (selector == 0xe7acab24) return "fulfillAdvancedOrder";
+        if (selector == 0x87201b41) return "fulfillAvailableAdvancedOrders";
+        if (selector == 0xed98a574) return "fulfillAvailableOrders";
+        if (selector == 0xfb0f3ee1) return "fulfillBasicOrder";
+        if (selector == 0x00000000) return "fulfillBasicOrder_efficient_6GL6yc";
+        if (selector == 0xb3a34c4c) return "fulfillOrder";
+        if (selector == 0xf2d12b12) return "matchAdvancedOrders";
+        if (selector == 0xa8174404) return "matchOrders";
+
+        revert UnknownSelector();
+    }
+
+    /**
+     * @dev Choose a suggested fulfillment method based on the structure of the
+     *      orders.
+     */
+    function suggestedCallData(
+        NavigatorContext memory context
+    ) internal view returns (bytes memory) {
+        Family family = context.response.orders.getFamily();
+
+        bool invalidOfferItemsLocated = mustUseMatch(context);
+
+        Structure structure = context.response.orders.getStructure(
+            address(context.request.seaport)
+        );
+
+        bool hasUnavailable = context.request.maximumFulfilled <
+            context.response.orders.length;
+        for (uint256 i = 0; i < context.response.orderDetails.length; ++i) {
+            if (
+                context.response.orderDetails[i].unavailableReason !=
+                UnavailableReason.AVAILABLE
+            ) {
+                hasUnavailable = true;
+                break;
+            }
+        }
+
+        if (hasUnavailable) {
+            if (invalidOfferItemsLocated) {
+                revert InvalidNativeTokenUnavailableCombination();
+            }
+
+            if (structure == Structure.ADVANCED) {
+                return
+                    abi.encodeCall(
+                        ConsiderationInterface.fulfillAvailableAdvancedOrders,
+                        (
+                            context.response.orders,
+                            context.response.criteriaResolvers,
+                            context.response.offerFulfillments,
+                            context.response.considerationFulfillments,
+                            context.request.fulfillerConduitKey,
+                            context.request.recipient,
+                            context.request.maximumFulfilled
+                        )
+                    );
+            } else {
+                return
+                    abi.encodeCall(
+                        ConsiderationInterface.fulfillAvailableOrders,
+                        (
+                            context.response.orders.toOrders(),
+                            context.response.offerFulfillments,
+                            context.response.considerationFulfillments,
+                            context.request.fulfillerConduitKey,
+                            context.request.maximumFulfilled
+                        )
+                    );
+            }
+        }
+
+        if (family == Family.SINGLE && !invalidOfferItemsLocated) {
+            if (structure == Structure.BASIC) {
+                AdvancedOrder memory order = context.response.orders[0];
+                return
+                    abi.encodeCall(
+                        ConsiderationInterface
+                            .fulfillBasicOrder_efficient_6GL6yc,
+                        (
+                            order.toBasicOrderParameters(
+                                order.getBasicOrderType()
+                            )
+                        )
+                    );
+            }
+
+            if (structure == Structure.STANDARD) {
+                return
+                    abi.encodeCall(
+                        ConsiderationInterface.fulfillOrder,
+                        (
+                            context.response.orders[0].toOrder(),
+                            context.request.fulfillerConduitKey
+                        )
+                    );
+            }
+
+            if (structure == Structure.ADVANCED) {
+                return
+                    abi.encodeCall(
+                        ConsiderationInterface.fulfillAdvancedOrder,
+                        (
+                            context.response.orders[0],
+                            context.response.criteriaResolvers,
+                            context.request.fulfillerConduitKey,
+                            context.request.recipient
+                        )
+                    );
+            }
+        }
+
+        bool cannotMatch = (context
+            .response
+            .unmetConsiderationComponents
+            .length !=
+            0 ||
+            hasUnavailable);
+
+        if (cannotMatch && invalidOfferItemsLocated) {
+            revert CannotFulfillProvidedCombinedOrder();
+        }
+
+        if (cannotMatch) {
+            if (structure == Structure.ADVANCED) {
+                return
+                    abi.encodeCall(
+                        ConsiderationInterface.fulfillAvailableAdvancedOrders,
+                        (
+                            context.response.orders,
+                            context.response.criteriaResolvers,
+                            context.response.offerFulfillments,
+                            context.response.considerationFulfillments,
+                            context.request.fulfillerConduitKey,
+                            context.request.recipient,
+                            context.request.maximumFulfilled
+                        )
+                    );
+            } else {
+                return
+                    abi.encodeCall(
+                        ConsiderationInterface.fulfillAvailableOrders,
+                        (
+                            context.response.orders.toOrders(),
+                            context.response.offerFulfillments,
+                            context.response.considerationFulfillments,
+                            context.request.fulfillerConduitKey,
+                            context.request.maximumFulfilled
+                        )
+                    );
+            }
+        } else if (invalidOfferItemsLocated) {
+            if (structure == Structure.ADVANCED) {
+                return
+                    abi.encodeCall(
+                        ConsiderationInterface.fulfillAvailableAdvancedOrders,
+                        (
+                            context.response.orders,
+                            context.response.criteriaResolvers,
+                            context.response.offerFulfillments,
+                            context.response.considerationFulfillments,
+                            context.request.fulfillerConduitKey,
+                            context.request.recipient,
+                            context.request.maximumFulfilled
+                        )
+                    );
+            } else {
+                return
+                    abi.encodeCall(
+                        ConsiderationInterface.matchOrders,
+                        (
+                            context.response.orders.toOrders(),
+                            context.response.fulfillments
+                        )
+                    );
+            }
+        } else {
+            if (structure == Structure.ADVANCED) {
+                if (context.request.preferMatch) {
+                    return
+                        abi.encodeCall(
+                            ConsiderationInterface.matchAdvancedOrders,
+                            (
+                                context.response.orders,
+                                context.response.criteriaResolvers,
+                                context.response.fulfillments,
+                                context.request.recipient
+                            )
+                        );
+                } else {
+                    return
+                        abi.encodeCall(
+                            ConsiderationInterface
+                                .fulfillAvailableAdvancedOrders,
+                            (
+                                context.response.orders,
+                                context.response.criteriaResolvers,
+                                context.response.offerFulfillments,
+                                context.response.considerationFulfillments,
+                                context.request.fulfillerConduitKey,
+                                context.request.recipient,
+                                context.request.maximumFulfilled
+                            )
+                        );
+                }
+            } else {
+                if (context.request.preferMatch) {
+                    return
+                        abi.encodeCall(
+                            ConsiderationInterface.matchOrders,
+                            (
+                                context.response.orders.toOrders(),
+                                context.response.fulfillments
+                            )
+                        );
+                } else {
+                    return
+                        abi.encodeCall(
+                            ConsiderationInterface.fulfillAvailableOrders,
+                            (
+                                context.response.orders.toOrders(),
+                                context.response.offerFulfillments,
+                                context.response.considerationFulfillments,
+                                context.request.fulfillerConduitKey,
+                                context.request.maximumFulfilled
+                            )
+                        );
+                }
+            }
+        }
+    }
+
+    /**
+     * @dev Return whether the provided orders must be matched using matchOrders
+     *      or matchAdvancedOrders.
+     */
+    function mustUseMatch(
+        NavigatorContext memory context
+    ) internal pure returns (bool) {
+        OrderDetails[] memory orders = context.response.orderDetails;
+
+        for (uint256 i = 0; i < orders.length; ++i) {
+            OrderDetails memory order = orders[i];
+
+            if (order.isContract) {
+                continue;
+            }
+
+            for (uint256 j = 0; j < order.offer.length; ++j) {
+                if (order.offer[j].itemType == ItemType.NATIVE) {
+                    return true;
+                }
+            }
+        }
+
+        if (context.request.caller == context.request.recipient) {
+            return false;
+        }
+
+        for (uint256 i = 0; i < orders.length; ++i) {
+            OrderDetails memory order = orders[i];
+
+            for (uint256 j = 0; j < order.offer.length; ++j) {
+                SpentItem memory item = order.offer[j];
+
+                if (item.itemType != ItemType.ERC721) {
+                    continue;
+                }
+
+                for (uint256 k = 0; k < orders.length; ++k) {
+                    OrderDetails memory comparisonOrder = orders[k];
+                    for (
+                        uint256 l = 0;
+                        l < comparisonOrder.consideration.length;
+                        ++l
+                    ) {
+                        ReceivedItem memory considerationItem = comparisonOrder
+                            .consideration[l];
+
+                        if (
+                            considerationItem.itemType == ItemType.ERC721 &&
+                            considerationItem.identifier == item.identifier &&
+                            considerationItem.token == item.token
+                        ) {
+                            return true;
+                        }
+                    }
+                }
+            }
+        }
+
+        return false;
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/OrderAvailabilityLib.sol b/contracts/seaport/helpers/navigator/lib/OrderAvailabilityLib.sol
new file mode 100644
index 00000000..02f9aebf
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/OrderAvailabilityLib.sol
@@ -0,0 +1,80 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    ConsiderationInterface
+} from "seaport-types/src/interfaces/ConsiderationInterface.sol";
+
+import { AdvancedOrder } from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import { UnavailableReason } from "seaport-sol/src/SpaceEnums.sol";
+
+import { OrderStructureLib, State } from "./OrderStructureLib.sol";
+
+/**
+ * @notice Helper library for determining order availability.
+ */
+library OrderAvailabilityLib {
+    using OrderStructureLib for AdvancedOrder;
+
+    /**
+     * @notice Returns true if the order is available for fulfillment.
+     */
+    function isAvailable(
+        AdvancedOrder memory order,
+        ConsiderationInterface seaport
+    ) internal view returns (bool) {
+        return unavailableReason(order, seaport) == UnavailableReason.AVAILABLE;
+    }
+
+    /**
+     * @notice Returns the order's UnavailableReason. Available orders will
+     *         return UnavailableReason.AVAILABLE to indicate that they are
+     *         available for fulfillment.
+     */
+    function unavailableReason(
+        AdvancedOrder memory order,
+        ConsiderationInterface seaport
+    ) internal view returns (UnavailableReason) {
+        if (order.parameters.endTime <= block.timestamp) {
+            return UnavailableReason.EXPIRED;
+        }
+        if (order.parameters.startTime > block.timestamp) {
+            return UnavailableReason.STARTS_IN_FUTURE;
+        }
+        if (order.getState(seaport) == State.CANCELLED) {
+            return UnavailableReason.CANCELLED;
+        }
+        if (order.getState(seaport) == State.FULLY_FILLED) {
+            return UnavailableReason.ALREADY_FULFILLED;
+        }
+        return UnavailableReason.AVAILABLE;
+    }
+
+    /**
+     * @notice Return an array of UnavailableReasons for the provided orders.
+     */
+    function unavailableReasons(
+        AdvancedOrder[] memory orders,
+        uint256 maximumFulfilled,
+        ConsiderationInterface seaport
+    ) internal view returns (UnavailableReason[] memory) {
+        UnavailableReason[] memory reasons = new UnavailableReason[](
+            orders.length
+        );
+        uint256 totalAvailable;
+        UnavailableReason reason;
+        for (uint256 i = 0; i < orders.length; i++) {
+            if (totalAvailable < maximumFulfilled) {
+                reason = unavailableReason(orders[i], seaport);
+            } else {
+                reason = UnavailableReason.MAX_FULFILLED_SATISFIED;
+            }
+            reasons[i] = reason;
+            if (reason == UnavailableReason.AVAILABLE) {
+                totalAvailable++;
+            }
+        }
+        return reasons;
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/OrderDetailsHelper.sol b/contracts/seaport/helpers/navigator/lib/OrderDetailsHelper.sol
new file mode 100644
index 00000000..b2c108dc
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/OrderDetailsHelper.sol
@@ -0,0 +1,18 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { NavigatorDetailsLib } from "./NavigatorDetailsLib.sol";
+
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+
+import { HelperInterface } from "./HelperInterface.sol";
+
+contract OrderDetailsHelper is HelperInterface {
+    using NavigatorDetailsLib for NavigatorContext;
+
+    function prepare(
+        NavigatorContext memory context
+    ) public view returns (NavigatorContext memory) {
+        return context.withDetails();
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/OrderStructureLib.sol b/contracts/seaport/helpers/navigator/lib/OrderStructureLib.sol
new file mode 100644
index 00000000..736bcd9d
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/OrderStructureLib.sol
@@ -0,0 +1,516 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    AdvancedOrderLib,
+    ConsiderationItemLib,
+    MatchComponent,
+    MatchComponentType,
+    OfferItemLib,
+    OrderComponentsLib,
+    OrderLib,
+    OrderParametersLib
+} from "seaport-sol/src/SeaportSol.sol";
+
+import {
+    AdvancedOrder,
+    CriteriaResolver,
+    Execution,
+    Order,
+    OrderComponents,
+    OrderParameters,
+    ConsiderationItem,
+    OfferItem,
+    ReceivedItem,
+    SpentItem,
+    Fulfillment,
+    FulfillmentComponent
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    BasicOrderRouteType,
+    BasicOrderType,
+    ItemType,
+    OrderType,
+    Side
+} from "seaport-types/src/lib/ConsiderationEnums.sol";
+
+import {
+    ConsiderationInterface
+} from "seaport-types/src/interfaces/ConsiderationInterface.sol";
+
+error TypeNotFound();
+
+/**
+ * @dev The "structure" of the order.
+ *      - BASIC: adheres to basic construction rules.
+ *      - STANDARD: does not adhere to basic construction rules.
+ *      - ADVANCED: requires criteria resolution, partial fulfillment, and/or
+ *        extraData.
+ */
+enum Structure {
+    BASIC,
+    STANDARD,
+    ADVANCED
+}
+
+/**
+ * @dev The "type" of the order.
+ *      - OPEN: FULL_OPEN or PARTIAL_OPEN orders.
+ *      - RESTRICTED: FULL_RESTRICTED or PARTIAL_RESTRICTED orders.
+ *      - CONTRACT: CONTRACT orders
+ */
+enum Type {
+    OPEN,
+    RESTRICTED,
+    CONTRACT
+}
+
+/**
+ * @dev The "family" of method that can fulfill the order.
+ *      - SINGLE: methods that accept a single order.
+ *        (fulfillOrder, fulfillAdvancedOrder, fulfillBasicOrder,
+ *        fulfillBasicOrder_efficient_6GL6yc)
+ *      - COMBINED: methods that accept multiple orders.
+ *        (fulfillAvailableOrders, fulfillAvailableAdvancedOrders, matchOrders,
+ *        matchAdvancedOrders, cancel, validate)
+ */
+enum Family {
+    SINGLE,
+    COMBINED
+}
+
+/**
+ * @dev The "state" of the order.
+ *      - UNUSED: New, not validated, cancelled, or partially/fully filled.
+ *      - VALIDATED: Order has been validated, but not cancelled or filled.
+ *      - CANCELLED: Order has been cancelled.
+ *      - PARTIALLY_FILLED: Order is partially filled.
+ *      - FULLY_FILLED: Order is fully filled.
+ */
+enum State {
+    UNUSED,
+    VALIDATED,
+    CANCELLED,
+    PARTIALLY_FILLED,
+    FULLY_FILLED
+}
+
+/**
+ * @notice Helper library for classifying an order's structure. This is helpful
+ *         for determining which fulfillment method to use.
+ */
+library OrderStructureLib {
+    using OrderLib for Order;
+    using OrderComponentsLib for OrderComponents;
+    using OrderParametersLib for OrderParameters;
+    using OfferItemLib for OfferItem[];
+    using ConsiderationItemLib for ConsiderationItem[];
+    using AdvancedOrderLib for AdvancedOrder;
+    using AdvancedOrderLib for AdvancedOrder[];
+
+    function getQuantity(
+        AdvancedOrder[] memory orders
+    ) internal pure returns (uint256) {
+        return orders.length;
+    }
+
+    function getFamily(
+        AdvancedOrder[] memory orders
+    ) internal pure returns (Family) {
+        uint256 quantity = getQuantity(orders);
+        if (quantity > 1) {
+            return Family.COMBINED;
+        }
+        return Family.SINGLE;
+    }
+
+    function getState(
+        AdvancedOrder memory order,
+        ConsiderationInterface seaport
+    ) internal view returns (State) {
+        uint256 counter = seaport.getCounter(order.parameters.offerer);
+        bytes32 orderHash = seaport.getOrderHash(
+            order.parameters.toOrderComponents(counter)
+        );
+        (
+            bool isValidated,
+            bool isCancelled,
+            uint256 totalFilled,
+            uint256 totalSize
+        ) = seaport.getOrderStatus(orderHash);
+
+        if (totalFilled != 0 && totalSize != 0 && totalFilled == totalSize)
+            return State.FULLY_FILLED;
+        if (totalFilled != 0 && totalSize != 0) return State.PARTIALLY_FILLED;
+        if (isCancelled) return State.CANCELLED;
+        if (isValidated) return State.VALIDATED;
+        return State.UNUSED;
+    }
+
+    function getType(AdvancedOrder memory order) internal pure returns (Type) {
+        OrderType orderType = order.parameters.orderType;
+        if (
+            orderType == OrderType.FULL_OPEN ||
+            orderType == OrderType.PARTIAL_OPEN
+        ) {
+            return Type.OPEN;
+        } else if (
+            orderType == OrderType.FULL_RESTRICTED ||
+            orderType == OrderType.PARTIAL_RESTRICTED
+        ) {
+            return Type.RESTRICTED;
+        } else if (orderType == OrderType.CONTRACT) {
+            return Type.CONTRACT;
+        } else {
+            revert TypeNotFound();
+        }
+    }
+
+    function getStructure(
+        AdvancedOrder memory order,
+        address seaport
+    ) internal view returns (Structure) {
+        // If the order has extraData, it's advanced
+        if (order.extraData.length > 0) return Structure.ADVANCED;
+
+        // If the order has numerator or denominator, it's advanced
+        if (order.numerator != 0 || order.denominator != 0) {
+            if (order.numerator < order.denominator) {
+                return Structure.ADVANCED;
+            }
+        }
+
+        (bool hasCriteria, bool hasNonzeroCriteria) = _checkCriteria(order);
+        bool isContractOrder = order.parameters.orderType == OrderType.CONTRACT;
+
+        // If any non-contract item has criteria, it's advanced,
+        if (hasCriteria) {
+            // Unless it's a contract order
+            if (isContractOrder) {
+                // And the contract order's critera are all zero
+                if (hasNonzeroCriteria) {
+                    return Structure.ADVANCED;
+                }
+            } else {
+                return Structure.ADVANCED;
+            }
+        }
+
+        if (getBasicOrderTypeEligibility(order, seaport)) {
+            return Structure.BASIC;
+        }
+
+        return Structure.STANDARD;
+    }
+
+    function getStructure(
+        AdvancedOrder[] memory orders,
+        address seaport
+    ) internal view returns (Structure) {
+        if (orders.length == 1) {
+            return getStructure(orders[0], seaport);
+        }
+
+        for (uint256 i; i < orders.length; i++) {
+            Structure structure = getStructure(orders[i], seaport);
+            if (structure == Structure.ADVANCED) {
+                return Structure.ADVANCED;
+            }
+        }
+
+        return Structure.STANDARD;
+    }
+
+    function getBasicOrderTypeEligibility(
+        AdvancedOrder memory order,
+        address seaport
+    ) internal view returns (bool) {
+        uint256 i;
+        ConsiderationItem[] memory consideration = order
+            .parameters
+            .consideration;
+        OfferItem[] memory offer = order.parameters.offer;
+
+        // Order must contain exactly one offer item and one or more
+        // consideration items.
+        if (offer.length != 1) {
+            return false;
+        }
+        if (
+            consideration.length == 0 ||
+            order.parameters.totalOriginalConsiderationItems == 0
+        ) {
+            return false;
+        }
+
+        // The order cannot have a contract order type.
+        if (order.parameters.orderType == OrderType.CONTRACT) {
+            return false;
+
+            // Note: the order type is combined with the “route” into a single
+            // BasicOrderType with a value between 0 and 23; there are 4
+            // supported order types (full open, partial open, full restricted,
+            // partial restricted) and 6 routes (ETH ⇒ ERC721, ETH ⇒ ERC1155,
+            // ERC20 ⇒ ERC721, ERC20 ⇒ ERC1155, ERC721 ⇒ ERC20, ERC1155 ⇒ ERC20)
+        }
+
+        // Order cannot specify a partial fraction to fill.
+        if (order.denominator > 1 && (order.numerator < order.denominator)) {
+            return false;
+        }
+
+        // Order cannot be partially filled.
+        ConsiderationInterface seaportInterface = ConsiderationInterface(
+            seaport
+        );
+        uint256 counter = seaportInterface.getCounter(order.parameters.offerer);
+        OrderComponents memory orderComponents = order
+            .parameters
+            .toOrderComponents(counter);
+        bytes32 orderHash = seaportInterface.getOrderHash(orderComponents);
+        (, , uint256 totalFilled, uint256 totalSize) = seaportInterface
+            .getOrderStatus(orderHash);
+
+        if (totalFilled != totalSize) {
+            return false;
+        }
+
+        // Order cannot contain any criteria-based items.
+        for (i = 0; i < consideration.length; ++i) {
+            if (
+                consideration[i].itemType == ItemType.ERC721_WITH_CRITERIA ||
+                consideration[i].itemType == ItemType.ERC1155_WITH_CRITERIA
+            ) {
+                return false;
+            }
+        }
+
+        if (
+            offer[0].itemType == ItemType.ERC721_WITH_CRITERIA ||
+            offer[0].itemType == ItemType.ERC1155_WITH_CRITERIA
+        ) {
+            return false;
+        }
+
+        // Order cannot contain any extraData.
+        if (order.extraData.length != 0) {
+            return false;
+        }
+
+        // Order must contain exactly one NFT item.
+        uint256 totalNFTs;
+        if (
+            offer[0].itemType == ItemType.ERC721 ||
+            offer[0].itemType == ItemType.ERC1155
+        ) {
+            totalNFTs += 1;
+        }
+        for (i = 0; i < consideration.length; ++i) {
+            if (
+                consideration[i].itemType == ItemType.ERC721 ||
+                consideration[i].itemType == ItemType.ERC1155
+            ) {
+                totalNFTs += 1;
+            }
+        }
+
+        if (totalNFTs != 1) {
+            return false;
+        }
+
+        // The one NFT must appear either as the offer item or as the first
+        // consideration item.
+        if (
+            offer[0].itemType != ItemType.ERC721 &&
+            offer[0].itemType != ItemType.ERC1155 &&
+            consideration[0].itemType != ItemType.ERC721 &&
+            consideration[0].itemType != ItemType.ERC1155
+        ) {
+            return false;
+        }
+
+        // All items that are not the NFT must share the same item type and
+        // token (and the identifier must be zero).
+        if (
+            offer[0].itemType == ItemType.ERC721 ||
+            offer[0].itemType == ItemType.ERC1155
+        ) {
+            ItemType expectedItemType = consideration[0].itemType;
+            address expectedToken = consideration[0].token;
+
+            for (i = 0; i < consideration.length; ++i) {
+                if (consideration[i].itemType != expectedItemType) {
+                    return false;
+                }
+
+                if (consideration[i].token != expectedToken) {
+                    return false;
+                }
+
+                if (consideration[i].identifierOrCriteria != 0) {
+                    return false;
+                }
+            }
+        }
+
+        if (
+            consideration[0].itemType == ItemType.ERC721 ||
+            consideration[0].itemType == ItemType.ERC1155
+        ) {
+            if (consideration.length >= 2) {
+                ItemType expectedItemType = offer[0].itemType;
+                address expectedToken = offer[0].token;
+                for (i = 1; i < consideration.length; ++i) {
+                    if (consideration[i].itemType != expectedItemType) {
+                        return false;
+                    }
+
+                    if (consideration[i].token != expectedToken) {
+                        return false;
+                    }
+
+                    if (consideration[i].identifierOrCriteria != 0) {
+                        return false;
+                    }
+                }
+            }
+        }
+
+        // The offerer must be the recipient of the first consideration item.
+        if (consideration[0].recipient != order.parameters.offerer) {
+            return false;
+        }
+
+        // If the NFT is the first consideration item, the sum of the amounts of
+        // all the other consideration items cannot exceed the amount of the
+        // offer item.
+        if (
+            consideration[0].itemType == ItemType.ERC721 ||
+            consideration[0].itemType == ItemType.ERC1155
+        ) {
+            uint256 totalConsiderationAmount;
+            for (i = 1; i < consideration.length; ++i) {
+                totalConsiderationAmount += consideration[i].startAmount;
+            }
+
+            if (totalConsiderationAmount > offer[0].startAmount) {
+                return false;
+            }
+
+            // Note: these cases represent a “bid” for an NFT, and the non-NFT
+            // consideration items (i.e. the “payment tokens”) are sent directly
+            // from the offerer to each recipient; this means that the fulfiller
+            // accepting the bid does not need to have approval set for the
+            // payment tokens.
+        }
+
+        // All items must have startAmount == endAmount
+        if (offer[0].startAmount != offer[0].endAmount) {
+            return false;
+        }
+        for (i = 0; i < consideration.length; ++i) {
+            if (consideration[i].startAmount != consideration[i].endAmount) {
+                return false;
+            }
+        }
+
+        // The offer item cannot have a native token type.
+        if (offer[0].itemType == ItemType.NATIVE) {
+            return false;
+        }
+
+        return true;
+    }
+
+    function getBasicOrderType(
+        AdvancedOrder memory order
+    ) internal pure returns (BasicOrderType basicOrderType) {
+        // Get the route (ETH ⇒ ERC721, etc.) for the order.
+        BasicOrderRouteType route = getBasicOrderRouteType(order);
+
+        // Get the order type (restricted, etc.) for the order.
+        OrderType orderType = order.parameters.orderType;
+
+        // Multiply the route by 4 and add the order type to get the
+        // BasicOrderType.
+        assembly {
+            basicOrderType := add(orderType, mul(route, 4))
+        }
+    }
+
+    function getBasicOrderRouteType(
+        AdvancedOrder memory order
+    ) internal pure returns (BasicOrderRouteType route) {
+        // Get the route (ETH ⇒ ERC721, etc.) for the order.
+        ItemType providingItemType = order.parameters.consideration[0].itemType;
+        ItemType offeredItemType = order.parameters.offer[0].itemType;
+
+        if (providingItemType == ItemType.NATIVE) {
+            if (offeredItemType == ItemType.ERC721) {
+                route = BasicOrderRouteType.ETH_TO_ERC721;
+            } else if (offeredItemType == ItemType.ERC1155) {
+                route = BasicOrderRouteType.ETH_TO_ERC1155;
+            }
+        } else if (providingItemType == ItemType.ERC20) {
+            if (offeredItemType == ItemType.ERC721) {
+                route = BasicOrderRouteType.ERC20_TO_ERC721;
+            } else if (offeredItemType == ItemType.ERC1155) {
+                route = BasicOrderRouteType.ERC20_TO_ERC1155;
+            }
+        } else if (providingItemType == ItemType.ERC721) {
+            if (offeredItemType == ItemType.ERC20) {
+                route = BasicOrderRouteType.ERC721_TO_ERC20;
+            }
+        } else if (providingItemType == ItemType.ERC1155) {
+            if (offeredItemType == ItemType.ERC20) {
+                route = BasicOrderRouteType.ERC1155_TO_ERC20;
+            }
+        }
+    }
+
+    /**
+     * @dev Check all offer and consideration items for criteria.
+     *
+     * @param order The advanced order.
+     *
+     * @return hasCriteria        Whether any offer or consideration item has
+     *                            criteria.
+     * @return hasNonzeroCriteria Whether any offer or consideration item has
+     *                            nonzero criteria.
+     */
+    function _checkCriteria(
+        AdvancedOrder memory order
+    ) internal pure returns (bool hasCriteria, bool hasNonzeroCriteria) {
+        // Check if any offer item has criteria
+        OfferItem[] memory offer = order.parameters.offer;
+        for (uint256 i; i < offer.length; ++i) {
+            OfferItem memory offerItem = offer[i];
+            ItemType itemType = offerItem.itemType;
+            hasCriteria = (itemType == ItemType.ERC721_WITH_CRITERIA ||
+                itemType == ItemType.ERC1155_WITH_CRITERIA);
+            if (hasCriteria) {
+                return (hasCriteria, offerItem.identifierOrCriteria != 0);
+            }
+        }
+
+        // Check if any consideration item has criteria
+        ConsiderationItem[] memory consideration = order
+            .parameters
+            .consideration;
+        for (uint256 i; i < consideration.length; ++i) {
+            ConsiderationItem memory considerationItem = consideration[i];
+            ItemType itemType = considerationItem.itemType;
+            hasCriteria = (itemType == ItemType.ERC721_WITH_CRITERIA ||
+                itemType == ItemType.ERC1155_WITH_CRITERIA);
+            if (hasCriteria) {
+                return (
+                    hasCriteria,
+                    considerationItem.identifierOrCriteria != 0
+                );
+            }
+        }
+
+        return (false, false);
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/RequestValidator.sol b/contracts/seaport/helpers/navigator/lib/RequestValidator.sol
new file mode 100644
index 00000000..a72b0bf2
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/RequestValidator.sol
@@ -0,0 +1,23 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    NavigatorRequestValidatorLib
+} from "./NavigatorRequestValidatorLib.sol";
+
+import {
+    NavigatorContext,
+    NavigatorResponse
+} from "./SeaportNavigatorTypes.sol";
+
+import { HelperInterface } from "./HelperInterface.sol";
+
+contract RequestValidator is HelperInterface {
+    using NavigatorRequestValidatorLib for NavigatorContext;
+
+    function prepare(
+        NavigatorContext memory context
+    ) public pure returns (NavigatorContext memory) {
+        return context.validate();
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/SeaportNavigatorInterface.sol b/contracts/seaport/helpers/navigator/lib/SeaportNavigatorInterface.sol
new file mode 100644
index 00000000..013e2d5d
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/SeaportNavigatorInterface.sol
@@ -0,0 +1,48 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    AdvancedOrder,
+    CriteriaResolver
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    NavigatorRequest,
+    NavigatorResponse
+} from "./SeaportNavigatorTypes.sol";
+
+interface SeaportNavigatorInterface {
+    function prepare(
+        NavigatorRequest memory request
+    ) external view returns (NavigatorResponse memory);
+
+    /**
+     * @notice Generate a criteria merkle root from an array of `tokenIds`. Use
+     *         this helper to construct an order item's `identifierOrCriteria`.
+     *
+     * @param tokenIds An array of integer token IDs to be converted to a merkle
+     *                 root.
+     *
+     * @return The bytes32 merkle root of a criteria tree containing the given
+     *         token IDs.
+     */
+    function criteriaRoot(
+        uint256[] memory tokenIds
+    ) external pure returns (bytes32);
+
+    /**
+     * @notice Generate a criteria merkle proof that `id` is a member of
+     *        `tokenIds`. Reverts if `id` is not a member of `tokenIds`. Use
+     *         this helper to construct proof data for criteria resolvers.
+     *
+     * @param tokenIds An array of integer token IDs.
+     * @param id       The integer token ID to generate a proof for.
+     *
+     * @return Merkle proof that the given token ID is  amember of the criteria
+     *         tree containing the given token IDs.
+     */
+    function criteriaProof(
+        uint256[] memory tokenIds,
+        uint256 id
+    ) external pure returns (bytes32[] memory);
+}
diff --git a/contracts/seaport/helpers/navigator/lib/SeaportNavigatorTypes.sol b/contracts/seaport/helpers/navigator/lib/SeaportNavigatorTypes.sol
new file mode 100644
index 00000000..2314f8e3
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/SeaportNavigatorTypes.sol
@@ -0,0 +1,171 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    SeaportValidatorInterface,
+    ErrorsAndWarnings
+} from "../../order-validator/SeaportValidator.sol";
+
+import {
+    ConsiderationInterface
+} from "seaport-types/src/interfaces/ConsiderationInterface.sol";
+
+import {
+    AdvancedOrder,
+    CriteriaResolver,
+    Execution,
+    Order,
+    OrderComponents,
+    OrderParameters,
+    ConsiderationItem,
+    OfferItem,
+    ReceivedItem,
+    SpentItem,
+    Fulfillment,
+    FulfillmentComponent
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    ItemType,
+    Side,
+    OrderType
+} from "seaport-types/src/lib/ConsiderationEnums.sol";
+
+import {
+    MatchComponent
+} from "seaport-sol/src/lib/types/MatchComponentType.sol";
+
+import {
+    FulfillmentDetails,
+    OrderDetails
+} from "seaport-sol/src/fulfillments/lib/Structs.sol";
+
+import {
+    FulfillmentStrategy
+} from "seaport-sol/src/fulfillments/lib/FulfillmentLib.sol";
+
+struct NavigatorAdvancedOrder {
+    NavigatorOrderParameters parameters;
+    uint120 numerator;
+    uint120 denominator;
+    bytes signature;
+    bytes extraData;
+}
+
+struct NavigatorOrderParameters {
+    address offerer;
+    address zone;
+    NavigatorOfferItem[] offer;
+    NavigatorConsiderationItem[] consideration;
+    OrderType orderType;
+    uint256 startTime;
+    uint256 endTime;
+    bytes32 zoneHash;
+    uint256 salt;
+    bytes32 conduitKey;
+    uint256 totalOriginalConsiderationItems;
+}
+
+struct NavigatorOfferItem {
+    ItemType itemType;
+    address token;
+    uint256 identifier;
+    uint256 startAmount;
+    uint256 endAmount;
+    uint256[] candidateIdentifiers;
+}
+
+struct NavigatorConsiderationItem {
+    ItemType itemType;
+    address token;
+    uint256 identifier;
+    uint256 startAmount;
+    uint256 endAmount;
+    address payable recipient;
+    uint256[] candidateIdentifiers;
+}
+
+/**
+ * @dev Context struct for NavigatorContextLib. Includes context information
+ *      necessary for fulfillment, like the caller and recipient addresses,
+ *      and Seaport and SeaportValidator interfaces.
+ */
+struct NavigatorContext {
+    NavigatorRequest request;
+    NavigatorResponse response;
+}
+
+struct NavigatorRequest {
+    ConsiderationInterface seaport;
+    SeaportValidatorInterface validator;
+    NavigatorAdvancedOrder[] orders;
+    address caller;
+    address recipient;
+    uint256 nativeTokensSupplied;
+    uint256 maximumFulfilled;
+    bytes32 fulfillerConduitKey;
+    uint256 seed;
+    FulfillmentStrategy fulfillmentStrategy;
+    CriteriaResolver[] criteriaResolvers;
+    bool preferMatch;
+}
+
+struct NavigatorResponse {
+    /**
+     * @dev The provided orders. If the caller provides explicit criteria
+     *      resolvers, the orders will not be modified. If the caller provides
+     *      criteria constraints, the returned offer/consideration items will be
+     *      updated with calculated merkle roots as their `identifierOrCriteria`
+     */
+    AdvancedOrder[] orders;
+    /**
+     * @dev The provided or calculated criteria resolvers. If the caller
+     *      provides criteria constraints rather than explicit criteria
+     *      resolvers, criteria resolvers and merkle proofs will be calculated
+     *      based on provided criteria constraints.
+     */
+    CriteriaResolver[] criteriaResolvers;
+    /**
+     * @dev Human-readable name of the suggested Seaport fulfillment method for
+     *      the provided orders.
+     */
+    string suggestedActionName;
+    /**
+     * @dev Encoded calldata for the suggested Seaport fulfillment method,
+     *      provided orders, and context args.
+     */
+    bytes suggestedCallData;
+    /**
+     * @dev Array of errors and warnings returned by SeaportValidator for the
+     *      provided orders, by order index in the orders array.
+     */
+    ErrorsAndWarnings[] validationErrors;
+    /**
+     * @dev Calculated OrderDetails structs for the provided orders, by order
+     *      index. Includes offerer, conduit key, spent and received items,
+     *      order hash, and unavailable reason.
+     */
+    OrderDetails[] orderDetails;
+    /**
+     * @dev Calculated fulfillment components and combined Fullfiilments.
+     */
+    FulfillmentComponent[][] offerFulfillments;
+    FulfillmentComponent[][] considerationFulfillments;
+    Fulfillment[] fulfillments;
+    /**
+     * @dev Calculated match components for matchable orders.
+     */
+    MatchComponent[] unspentOfferComponents;
+    MatchComponent[] unmetConsiderationComponents;
+    /**
+     * @dev Calculated explicit and implicit executions.
+     */
+    Execution[] explicitExecutions;
+    Execution[] implicitExecutions;
+    Execution[] implicitExecutionsPre;
+    Execution[] implicitExecutionsPost;
+    /**
+     * @dev Quantity of native tokens returned to caller.
+     */
+    uint256 nativeTokensReturned;
+}
diff --git a/contracts/seaport/helpers/navigator/lib/SuggestedActionHelper.sol b/contracts/seaport/helpers/navigator/lib/SuggestedActionHelper.sol
new file mode 100644
index 00000000..858ce6c3
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/SuggestedActionHelper.sol
@@ -0,0 +1,18 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { NavigatorSuggestedActionLib } from "./NavigatorSuggestedActionLib.sol";
+
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+
+import { HelperInterface } from "./HelperInterface.sol";
+
+contract SuggestedActionHelper is HelperInterface {
+    using NavigatorSuggestedActionLib for NavigatorContext;
+
+    function prepare(
+        NavigatorContext memory context
+    ) public view returns (NavigatorContext memory) {
+        return context.withSuggestedAction();
+    }
+}
diff --git a/contracts/seaport/helpers/navigator/lib/ValidatorHelper.sol b/contracts/seaport/helpers/navigator/lib/ValidatorHelper.sol
new file mode 100644
index 00000000..5a3d813c
--- /dev/null
+++ b/contracts/seaport/helpers/navigator/lib/ValidatorHelper.sol
@@ -0,0 +1,26 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import {
+    NavigatorSeaportValidatorLib
+} from "./NavigatorSeaportValidatorLib.sol";
+
+import { NavigatorContext } from "./SeaportNavigatorTypes.sol";
+
+import { HelperInterface } from "./HelperInterface.sol";
+
+import { Order } from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    SeaportValidatorInterface
+} from "../../order-validator/SeaportValidator.sol";
+
+contract ValidatorHelper is HelperInterface {
+    using NavigatorSeaportValidatorLib for NavigatorContext;
+
+    function prepare(
+        NavigatorContext memory context
+    ) public view returns (NavigatorContext memory) {
+        return context.withErrors();
+    }
+}
diff --git a/contracts/seaport/helpers/order-validator/SeaportValidator.sol b/contracts/seaport/helpers/order-validator/SeaportValidator.sol
new file mode 100644
index 00000000..2ec7997c
--- /dev/null
+++ b/contracts/seaport/helpers/order-validator/SeaportValidator.sol
@@ -0,0 +1,1141 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { ItemType } from "seaport-types/src/lib/ConsiderationEnums.sol";
+import {
+    Order,
+    OrderParameters,
+    BasicOrderParameters,
+    OfferItem,
+    ConsiderationItem,
+    Schema,
+    ZoneParameters
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+import { ConsiderationTypeHashes } from "./lib/ConsiderationTypeHashes.sol";
+import {
+    ConsiderationInterface
+} from "seaport-types/src/interfaces/ConsiderationInterface.sol";
+import {
+    ConduitControllerInterface
+} from "seaport-types/src/interfaces/ConduitControllerInterface.sol";
+import {
+    ContractOffererInterface
+} from "seaport-types/src/interfaces/ContractOffererInterface.sol";
+import { ZoneInterface } from "seaport-types/src/interfaces/ZoneInterface.sol";
+import {
+    GettersAndDerivers
+} from "seaport-core/src/lib/GettersAndDerivers.sol";
+import { SeaportValidatorInterface } from "./lib/SeaportValidatorInterface.sol";
+import { ZoneInterface } from "seaport-types/src/interfaces/ZoneInterface.sol";
+import {
+    ERC20Interface,
+    ERC721Interface,
+    ERC1155Interface
+} from "seaport-types/src/interfaces/AbridgedTokenInterfaces.sol";
+import { IERC165 } from "@openzeppelin/contracts/seaport/interfaces/IERC165.sol";
+import { IERC2981 } from "@openzeppelin/contracts/interfaces/IERC2981.sol";
+import {
+    ErrorsAndWarnings,
+    ErrorsAndWarningsLib
+} from "./lib/ErrorsAndWarnings.sol";
+import { SafeStaticCall } from "./lib/SafeStaticCall.sol";
+import {
+    IssueParser,
+    ValidationConfiguration,
+    TimeIssue,
+    StatusIssue,
+    OfferIssue,
+    ContractOffererIssue,
+    ConsiderationIssue,
+    PrimaryFeeIssue,
+    ERC721Issue,
+    ERC1155Issue,
+    ERC20Issue,
+    NativeIssue,
+    ZoneIssue,
+    ConduitIssue,
+    CreatorFeeIssue,
+    SignatureIssue,
+    GenericIssue,
+    ConsiderationItemConfiguration
+} from "./lib/SeaportValidatorTypes.sol";
+import { Verifiers } from "seaport-core/src/lib/Verifiers.sol";
+import { ReadOnlyOrderValidator } from "./lib/ReadOnlyOrderValidator.sol";
+import { SeaportValidatorHelper } from "./lib/SeaportValidatorHelper.sol";
+
+/**
+ * @title SeaportValidator
+ * @author OpenSea Protocol Team
+ * @notice SeaportValidator provides advanced validation to seaport orders.
+ */
+contract SeaportValidator is
+    SeaportValidatorInterface,
+    ConsiderationTypeHashes
+{
+    using ErrorsAndWarningsLib for ErrorsAndWarnings;
+    using SafeStaticCall for address;
+    using IssueParser for *;
+
+    /// @notice Cross-chain conduit controller Address
+    ConduitControllerInterface private immutable _conduitController;
+
+    SeaportValidatorHelper private immutable _helper;
+
+    ReadOnlyOrderValidator private immutable _readOnlyOrderValidator;
+
+    bytes4 public constant ERC20_INTERFACE_ID = 0x36372b07;
+
+    bytes4 public constant ERC721_INTERFACE_ID = 0x80ac58cd;
+
+    bytes4 public constant ERC1155_INTERFACE_ID = 0xd9b67a26;
+
+    bytes4 public constant CONTRACT_OFFERER_INTERFACE_ID = 0x1be900b1;
+
+    bytes4 public constant ZONE_INTERFACE_ID = 0x3839be19;
+
+    constructor(
+        address readOnlyOrderValidatorAddress,
+        address seaportValidatorHelperAddress,
+        address conduitControllerAddress
+    ) {
+        _readOnlyOrderValidator = ReadOnlyOrderValidator(
+            readOnlyOrderValidatorAddress
+        );
+        _helper = SeaportValidatorHelper(seaportValidatorHelperAddress);
+        _conduitController = ConduitControllerInterface(
+            conduitControllerAddress
+        );
+    }
+
+    /**
+     * @notice Conduct a comprehensive validation of the given order.
+     *    `isValidOrder` validates simple orders that adhere to a set of rules defined below:
+     *    - The order is either a listing or an offer order (one NFT to buy or one NFT to sell).
+     *    - The first consideration is the primary consideration.
+     *    - The order pays up to two fees in the fungible token currency. First fee is primary fee, second is creator fee.
+     *    - In private orders, the last consideration specifies a recipient for the offer item.
+     *    - Offer items must be owned and properly approved by the offerer.
+     *    - There must be one offer item
+     *    - Consideration items must exist.
+     *    - The signature must be valid, or the order must be already validated on chain
+     * @param order The order to validate.
+     * @return errorsAndWarnings The errors and warnings found in the order.
+     */
+    function isValidOrder(
+        Order calldata order,
+        address seaportAddress
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        return
+            isValidOrderWithConfiguration(
+                ValidationConfiguration(
+                    seaportAddress,
+                    address(0),
+                    0,
+                    false,
+                    false,
+                    30 minutes,
+                    26 weeks
+                ),
+                order
+            );
+    }
+
+    /**
+     * @notice Same as `isValidOrder` but allows for more configuration related to fee validation.
+     *    If `skipStrictValidation` is set order logic validation is not carried out: fees are not
+     *       checked and there may be more than one offer item as well as any number of consideration items.
+     */
+    function isValidOrderWithConfiguration(
+        ValidationConfiguration memory validationConfiguration,
+        Order memory order
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // Concatenates errorsAndWarnings with the returned errorsAndWarnings
+        errorsAndWarnings.concat(
+            validateTime(
+                order.parameters,
+                validationConfiguration.shortOrderDuration,
+                validationConfiguration.distantOrderExpiration
+            )
+        );
+        errorsAndWarnings.concat(
+            validateOrderStatus(
+                order.parameters,
+                validationConfiguration.seaport
+            )
+        );
+        errorsAndWarnings.concat(
+            validateOfferItems(
+                order.parameters,
+                validationConfiguration.seaport
+            )
+        );
+        errorsAndWarnings.concat(
+            validateConsiderationItems(
+                order.parameters,
+                validationConfiguration.seaport
+            )
+        );
+        errorsAndWarnings.concat(isValidZone(order.parameters));
+        errorsAndWarnings.concat(
+            validateSignature(order, validationConfiguration.seaport)
+        );
+
+        // Skip strict validation if requested
+        if (!validationConfiguration.skipStrictValidation) {
+            errorsAndWarnings.concat(
+                validateStrictLogic(
+                    order.parameters,
+                    validationConfiguration.primaryFeeRecipient,
+                    validationConfiguration.primaryFeeBips,
+                    validationConfiguration.checkCreatorFee
+                )
+            );
+        }
+    }
+
+    /**
+     * @notice Strict validation operates under tight assumptions. It validates primary
+     *    fee, creator fee, private sale consideration, and overall order format.
+     * @dev Only checks first fee recipient provided by CreatorFeeEngine.
+     *    Order of consideration items must be as follows:
+     *    1. Primary consideration
+     *    2. Primary fee
+     *    3. Creator fee
+     *    4. Private sale consideration
+     * @param orderParameters The parameters for the order to validate.
+     * @param primaryFeeRecipient The primary fee recipient. Set to null address for no primary fee.
+     * @param primaryFeeBips The primary fee in BIPs.
+     * @param checkCreatorFee Should check for creator fee. If true, creator fee must be present as
+     *    according to creator fee engine. If false, must not have creator fee.
+     * @return errorsAndWarnings The errors and warnings.
+     */
+    function validateStrictLogic(
+        OrderParameters memory orderParameters,
+        address primaryFeeRecipient,
+        uint256 primaryFeeBips,
+        bool checkCreatorFee
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        return
+            _helper.validateStrictLogic(
+                orderParameters,
+                primaryFeeRecipient,
+                primaryFeeBips,
+                checkCreatorFee
+            );
+    }
+
+    /**
+     * @notice Checks if a conduit key is valid.
+     * @param conduitKey The conduit key to check.
+     * @return errorsAndWarnings The errors and warnings
+     */
+    function isValidConduit(
+        bytes32 conduitKey,
+        address seaportAddress
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        (, errorsAndWarnings) = getApprovalAddress(conduitKey, seaportAddress);
+    }
+
+    /**
+     * @notice Checks if the zone of an order is set and implements the EIP165
+     *         zone interface
+     * @dev To validate the zone call for an order, see validateOrderWithZone
+     * @param orderParameters The order parameters to check.
+     * @return errorsAndWarnings The errors and warnings
+     */
+    function isValidZone(
+        OrderParameters memory orderParameters
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // If not restricted, zone isn't checked
+        if (
+            uint8(orderParameters.orderType) < 2 ||
+            uint8(orderParameters.orderType) == 4
+        ) {
+            return errorsAndWarnings;
+        }
+
+        if (orderParameters.zone == address(0)) {
+            // Zone is not set
+            errorsAndWarnings.addError(ZoneIssue.NotSet.parseInt());
+            return errorsAndWarnings;
+        }
+
+        // Warn if zone is an EOA
+        if (address(orderParameters.zone).code.length == 0) {
+            errorsAndWarnings.addWarning(ZoneIssue.EOAZone.parseInt());
+            return errorsAndWarnings;
+        }
+
+        // Check the EIP165 zone interface
+        if (!checkInterface(orderParameters.zone, ZONE_INTERFACE_ID)) {
+            errorsAndWarnings.addWarning(ZoneIssue.InvalidZone.parseInt());
+            return errorsAndWarnings;
+        }
+
+        // Check if the zone implements SIP-5
+        try ZoneInterface(orderParameters.zone).getSeaportMetadata() {} catch {
+            errorsAndWarnings.addWarning(ZoneIssue.InvalidZone.parseInt());
+        }
+    }
+
+    /**
+     * @notice Gets the approval address for the given conduit key
+     * @param conduitKey Conduit key to get approval address for
+     * @param seaportAddress The Seaport address
+     * @return approvalAddress The address to use for approvals
+     * @return errorsAndWarnings An ErrorsAndWarnings structs with results
+     */
+    function getApprovalAddress(
+        bytes32 conduitKey,
+        address seaportAddress
+    )
+        public
+        view
+        returns (address, ErrorsAndWarnings memory errorsAndWarnings)
+    {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // Zero conduit key corresponds to seaport
+        if (conduitKey == 0) return (seaportAddress, errorsAndWarnings);
+
+        // Pull conduit info from conduitController
+        (address conduitAddress, bool exists) = _conduitController.getConduit(
+            conduitKey
+        );
+
+        // Conduit does not exist
+        if (!exists) {
+            errorsAndWarnings.addError(ConduitIssue.KeyInvalid.parseInt());
+            conduitAddress = address(0); // Don't return invalid conduit
+        }
+
+        // Approval address does not have Seaport added as a channel
+        if (
+            exists &&
+            !_conduitController.getChannelStatus(conduitAddress, seaportAddress)
+        ) {
+            errorsAndWarnings.addError(
+                ConduitIssue.MissingSeaportChannel.parseInt()
+            );
+        }
+
+        return (conduitAddress, errorsAndWarnings);
+    }
+
+    /**
+     * @notice Validates the signature for the order using the offerer's current counter
+     * @dev Will also check if order is validated on chain.
+     */
+    function validateSignature(
+        Order memory order,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        // Pull current counter from seaport
+        uint256 currentCounter = ConsiderationInterface(seaportAddress)
+            .getCounter(order.parameters.offerer);
+
+        return
+            validateSignatureWithCounter(order, currentCounter, seaportAddress);
+    }
+
+    /**
+     * @notice Validates the signature for the order using the given counter
+     * @dev Will also check if order is validated on chain.
+     */
+    function validateSignatureWithCounter(
+        Order memory order,
+        uint256 counter,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // Typecast Seaport address to ConsiderationInterface
+        ConsiderationInterface seaport = ConsiderationInterface(seaportAddress);
+
+        // Contract orders do not have signatures
+        if (uint8(order.parameters.orderType) == 4) {
+            errorsAndWarnings.addWarning(
+                SignatureIssue.ContractOrder.parseInt()
+            );
+        }
+
+        // Get current counter for context
+        uint256 currentCounter = seaport.getCounter(order.parameters.offerer);
+
+        if (currentCounter > counter) {
+            // Counter strictly increases
+            errorsAndWarnings.addError(SignatureIssue.LowCounter.parseInt());
+            return errorsAndWarnings;
+        } else if (currentCounter < counter) {
+            // Counter is incremented by random large number
+            errorsAndWarnings.addError(SignatureIssue.HighCounter.parseInt());
+            return errorsAndWarnings;
+        }
+
+        bytes32 orderHash = _deriveOrderHash(order.parameters, counter);
+
+        // Check if order is validated on chain
+        (bool isValid, , , ) = seaport.getOrderStatus(orderHash);
+
+        if (isValid) {
+            // Shortcut success, valid on chain
+            return errorsAndWarnings;
+        }
+
+        // Create memory array to pass into validate
+        Order[] memory orderArray = new Order[](1);
+
+        // Store order in array
+        orderArray[0] = order;
+
+        try
+            // Call validate on Seaport
+            _readOnlyOrderValidator.canValidate(seaportAddress, orderArray)
+        returns (bool success) {
+            if (!success) {
+                // Call was unsuccessful, so signature is invalid
+                errorsAndWarnings.addError(SignatureIssue.Invalid.parseInt());
+            }
+        } catch {
+            if (
+                order.parameters.consideration.length !=
+                order.parameters.totalOriginalConsiderationItems
+            ) {
+                // May help diagnose signature issues
+                errorsAndWarnings.addWarning(
+                    SignatureIssue.OriginalConsiderationItems.parseInt()
+                );
+            }
+            // Call reverted, so signature is invalid
+            errorsAndWarnings.addError(SignatureIssue.Invalid.parseInt());
+        }
+    }
+
+    /**
+     * @notice Check that a contract offerer implements the EIP165
+     *         contract offerer interface
+     * @param contractOfferer The address of the contract offerer
+     * @return errorsAndWarnings The errors and warnings
+     */
+    function validateContractOfferer(
+        address contractOfferer
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // Check the EIP165 contract offerer interface
+        if (!checkInterface(contractOfferer, CONTRACT_OFFERER_INTERFACE_ID)) {
+            errorsAndWarnings.addWarning(
+                ContractOffererIssue.InvalidContractOfferer.parseInt()
+            );
+        }
+
+        // Check if the contract offerer implements SIP-5
+        try
+            ContractOffererInterface(contractOfferer).getSeaportMetadata()
+        {} catch {
+            errorsAndWarnings.addWarning(
+                ContractOffererIssue.InvalidContractOfferer.parseInt()
+            );
+        }
+
+        return errorsAndWarnings;
+    }
+
+    /**
+     * @notice Check the time validity of an order
+     * @param orderParameters The parameters for the order to validate
+     * @param shortOrderDuration The duration of which an order is considered short
+     * @param distantOrderExpiration Distant order expiration delta in seconds.
+     * @return errorsAndWarnings The errors and warnings
+     */
+    function validateTime(
+        OrderParameters memory orderParameters,
+        uint256 shortOrderDuration,
+        uint256 distantOrderExpiration
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        if (orderParameters.endTime <= orderParameters.startTime) {
+            // Order duration is zero
+            errorsAndWarnings.addError(
+                TimeIssue.EndTimeBeforeStartTime.parseInt()
+            );
+            return errorsAndWarnings;
+        }
+
+        if (orderParameters.endTime < block.timestamp) {
+            // Order is expired
+            errorsAndWarnings.addError(TimeIssue.Expired.parseInt());
+            return errorsAndWarnings;
+        } else if (
+            orderParameters.endTime > block.timestamp + distantOrderExpiration
+        ) {
+            // Order expires in a long time
+            errorsAndWarnings.addWarning(
+                TimeIssue.DistantExpiration.parseInt()
+            );
+        }
+
+        if (orderParameters.startTime > block.timestamp) {
+            // Order is not active
+            errorsAndWarnings.addWarning(TimeIssue.NotActive.parseInt());
+        }
+
+        if (
+            orderParameters.endTime -
+                (
+                    orderParameters.startTime > block.timestamp
+                        ? orderParameters.startTime
+                        : block.timestamp
+                ) <
+            shortOrderDuration
+        ) {
+            // Order has a short duration
+            errorsAndWarnings.addWarning(TimeIssue.ShortOrder.parseInt());
+        }
+    }
+
+    /**
+     * @notice Validate the status of an order
+     * @param orderParameters The parameters for the order to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateOrderStatus(
+        OrderParameters memory orderParameters,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // Typecast Seaport address to ConsiderationInterface
+        ConsiderationInterface seaport = ConsiderationInterface(seaportAddress);
+
+        // Cannot validate status of contract order
+        if (uint8(orderParameters.orderType) == 4) {
+            errorsAndWarnings.addWarning(StatusIssue.ContractOrder.parseInt());
+        }
+
+        // Pull current counter from seaport
+        uint256 currentOffererCounter = seaport.getCounter(
+            orderParameters.offerer
+        );
+        // Derive order hash using orderParameters and currentOffererCounter
+        bytes32 orderHash = _deriveOrderHash(
+            orderParameters,
+            currentOffererCounter
+        );
+        // Get order status from seaport
+        (, bool isCancelled, uint256 totalFilled, uint256 totalSize) = seaport
+            .getOrderStatus(orderHash);
+
+        if (isCancelled) {
+            // Order is cancelled
+            errorsAndWarnings.addError(StatusIssue.Cancelled.parseInt());
+        }
+
+        if (totalSize > 0 && totalFilled == totalSize) {
+            // Order is fully filled
+            errorsAndWarnings.addError(StatusIssue.FullyFilled.parseInt());
+        }
+    }
+
+    /**
+     * @notice Validate all offer items for an order. Ensures that
+     *         offerer has sufficient balance and approval for each item.
+     * @dev Amounts are not summed and verified, just the individual amounts.
+     * @param orderParameters The parameters for the order to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateOfferItems(
+        OrderParameters memory orderParameters,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // Iterate over each offer item and validate it
+        for (uint256 i = 0; i < orderParameters.offer.length; i++) {
+            errorsAndWarnings.concat(
+                validateOfferItem(orderParameters, i, seaportAddress)
+            );
+
+            // Check for duplicate offer item
+            OfferItem memory offerItem1 = orderParameters.offer[i];
+
+            for (uint256 j = i + 1; j < orderParameters.offer.length; j++) {
+                // Iterate over each remaining offer item
+                // (previous items already check with this item)
+                OfferItem memory offerItem2 = orderParameters.offer[j];
+
+                // Check if token and id are the same
+                if (
+                    offerItem1.token == offerItem2.token &&
+                    offerItem1.identifierOrCriteria ==
+                    offerItem2.identifierOrCriteria
+                ) {
+                    errorsAndWarnings.addError(
+                        OfferIssue.DuplicateItem.parseInt()
+                    );
+                }
+            }
+        }
+
+        // You must have an offer item
+        if (orderParameters.offer.length == 0) {
+            errorsAndWarnings.addWarning(OfferIssue.ZeroItems.parseInt());
+        }
+
+        // Warning if there is more than one offer item
+        if (orderParameters.offer.length > 1) {
+            errorsAndWarnings.addWarning(OfferIssue.MoreThanOneItem.parseInt());
+        }
+    }
+
+    /**
+     * @notice Validates an offer item
+     * @param orderParameters The parameters for the order to validate
+     * @param offerItemIndex The index of the offerItem in offer array to validate
+     * @return errorsAndWarnings An ErrorsAndWarnings structs with results
+     */
+    function validateOfferItem(
+        OrderParameters memory orderParameters,
+        uint256 offerItemIndex,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        // First validate the parameters (correct amount, contract, etc)
+        errorsAndWarnings = validateOfferItemParameters(
+            orderParameters,
+            offerItemIndex,
+            seaportAddress
+        );
+        if (errorsAndWarnings.hasErrors()) {
+            // Only validate approvals and balances if parameters are valid
+            return errorsAndWarnings;
+        }
+
+        // Validate approvals and balances for the offer item
+        errorsAndWarnings.concat(
+            validateOfferItemApprovalAndBalance(
+                orderParameters,
+                offerItemIndex,
+                seaportAddress
+            )
+        );
+    }
+
+    /**
+     * @notice Validates the OfferItem parameters. This includes token contract validation
+     * @dev OfferItems with criteria are currently not allowed
+     * @param orderParameters The parameters for the order to validate
+     * @param offerItemIndex The index of the offerItem in offer array to validate
+     * @return errorsAndWarnings An ErrorsAndWarnings structs with results
+     */
+    function validateOfferItemParameters(
+        OrderParameters memory orderParameters,
+        uint256 offerItemIndex,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // Get the offer item at offerItemIndex
+        OfferItem memory offerItem = orderParameters.offer[offerItemIndex];
+
+        // Check if start amount and end amount are zero
+        if (offerItem.startAmount == 0 && offerItem.endAmount == 0) {
+            errorsAndWarnings.addError(OfferIssue.AmountZero.parseInt());
+            return errorsAndWarnings;
+        }
+
+        // Check that amount velocity is not too high.
+        if (
+            offerItem.startAmount != offerItem.endAmount &&
+            orderParameters.endTime > orderParameters.startTime
+        ) {
+            // Assign larger and smaller amount values
+            (uint256 maxAmount, uint256 minAmount) = offerItem.startAmount >
+                offerItem.endAmount
+                ? (offerItem.startAmount, offerItem.endAmount)
+                : (offerItem.endAmount, offerItem.startAmount);
+
+            uint256 amountDelta = maxAmount - minAmount;
+            // delta of time that order exists for
+            uint256 timeDelta = orderParameters.endTime -
+                orderParameters.startTime;
+
+            // Velocity scaled by 1e10 for precision
+            uint256 velocity = (amountDelta * 1e10) / timeDelta;
+            // gives velocity percentage in hundredth of a basis points per second in terms of larger value
+            uint256 velocityPercentage = velocity / (maxAmount * 1e4);
+
+            // 278 * 60 * 30 ~= 500,000
+            if (velocityPercentage > 278) {
+                // Over 50% change per 30 min
+                errorsAndWarnings.addError(
+                    OfferIssue.AmountVelocityHigh.parseInt()
+                );
+            }
+            // Over 50% change per 30 min
+            else if (velocityPercentage > 28) {
+                // Over 5% change per 30 min
+                errorsAndWarnings.addWarning(
+                    OfferIssue.AmountVelocityHigh.parseInt()
+                );
+            }
+
+            // Check for large amount steps
+            if (minAmount <= 1e15) {
+                errorsAndWarnings.addWarning(
+                    OfferIssue.AmountStepLarge.parseInt()
+                );
+            }
+        }
+
+        if (offerItem.itemType == ItemType.ERC721) {
+            // ERC721 type requires amounts to be 1
+            if (offerItem.startAmount != 1 || offerItem.endAmount != 1) {
+                errorsAndWarnings.addError(ERC721Issue.AmountNotOne.parseInt());
+            }
+
+            // Check the EIP165 token interface
+            if (!checkInterface(offerItem.token, ERC721_INTERFACE_ID)) {
+                errorsAndWarnings.addError(ERC721Issue.InvalidToken.parseInt());
+            }
+        } else if (offerItem.itemType == ItemType.ERC721_WITH_CRITERIA) {
+            // Check the EIP165 token interface
+            if (!checkInterface(offerItem.token, ERC721_INTERFACE_ID)) {
+                errorsAndWarnings.addError(ERC721Issue.InvalidToken.parseInt());
+            }
+
+            if (offerItem.startAmount > 1 || offerItem.endAmount > 1) {
+                // Require partial fill enabled. Even orderTypes are full
+                if (uint8(orderParameters.orderType) % 2 == 0) {
+                    errorsAndWarnings.addError(
+                        ERC721Issue.CriteriaNotPartialFill.parseInt()
+                    );
+                }
+            }
+        } else if (
+            offerItem.itemType == ItemType.ERC1155 ||
+            offerItem.itemType == ItemType.ERC1155_WITH_CRITERIA
+        ) {
+            // Check the EIP165 token interface
+            if (!checkInterface(offerItem.token, ERC1155_INTERFACE_ID)) {
+                errorsAndWarnings.addError(
+                    ERC1155Issue.InvalidToken.parseInt()
+                );
+            }
+        } else if (offerItem.itemType == ItemType.ERC20) {
+            // ERC20 must have `identifierOrCriteria` be zero
+            if (offerItem.identifierOrCriteria != 0) {
+                errorsAndWarnings.addError(
+                    ERC20Issue.IdentifierNonZero.parseInt()
+                );
+            }
+
+            // Validate contract, should return an uint256 if its an ERC20
+            if (
+                !offerItem.token.safeStaticCallUint256(
+                    abi.encodeWithSelector(
+                        ERC20Interface.allowance.selector,
+                        seaportAddress,
+                        seaportAddress
+                    ),
+                    0
+                )
+            ) {
+                errorsAndWarnings.addError(ERC20Issue.InvalidToken.parseInt());
+            }
+        } else {
+            // Must be native
+            // NATIVE must have `token` be zero address
+            if (offerItem.token != address(0)) {
+                errorsAndWarnings.addError(NativeIssue.TokenAddress.parseInt());
+            }
+
+            // NATIVE must have `identifierOrCriteria` be zero
+            if (offerItem.identifierOrCriteria != 0) {
+                errorsAndWarnings.addError(
+                    NativeIssue.IdentifierNonZero.parseInt()
+                );
+            }
+        }
+    }
+
+    /**
+     * @notice Validates the OfferItem approvals and balances
+     * @param orderParameters The parameters for the order to validate
+     * @param offerItemIndex The index of the offerItem in offer array to validate
+     * @return errorsAndWarnings An ErrorsAndWarnings structs with results
+     */
+    function validateOfferItemApprovalAndBalance(
+        OrderParameters memory orderParameters,
+        uint256 offerItemIndex,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        // Note: If multiple items are of the same token, token amounts are not summed for validation
+
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // Get the approval address for the given conduit key
+        (
+            address approvalAddress,
+            ErrorsAndWarnings memory ew
+        ) = getApprovalAddress(orderParameters.conduitKey, seaportAddress);
+        errorsAndWarnings.concat(ew);
+
+        if (ew.hasErrors()) {
+            // Approval address is invalid
+            return errorsAndWarnings;
+        }
+
+        // Get the offer item at offerItemIndex
+        OfferItem memory offerItem = orderParameters.offer[offerItemIndex];
+
+        if (offerItem.itemType == ItemType.ERC721) {
+            ERC721Interface token = ERC721Interface(offerItem.token);
+
+            // Check that offerer owns token
+            if (
+                !address(token).safeStaticCallAddress(
+                    abi.encodeWithSelector(
+                        ERC721Interface.ownerOf.selector,
+                        offerItem.identifierOrCriteria
+                    ),
+                    orderParameters.offerer
+                )
+            ) {
+                errorsAndWarnings.addError(ERC721Issue.NotOwner.parseInt());
+            }
+
+            // Check for approval via `getApproved`
+            if (
+                !address(token).safeStaticCallAddress(
+                    abi.encodeWithSelector(
+                        ERC721Interface.getApproved.selector,
+                        offerItem.identifierOrCriteria
+                    ),
+                    approvalAddress
+                )
+            ) {
+                // Fallback to `isApprovalForAll`
+                if (
+                    !address(token).safeStaticCallBool(
+                        abi.encodeWithSelector(
+                            ERC721Interface.isApprovedForAll.selector,
+                            orderParameters.offerer,
+                            approvalAddress
+                        ),
+                        true
+                    )
+                ) {
+                    // Not approved
+                    errorsAndWarnings.addError(
+                        ERC721Issue.NotApproved.parseInt()
+                    );
+                }
+            }
+        } else if (offerItem.itemType == ItemType.ERC721_WITH_CRITERIA) {
+            ERC721Interface token = ERC721Interface(offerItem.token);
+
+            // Check for approval
+            if (
+                !address(token).safeStaticCallBool(
+                    abi.encodeWithSelector(
+                        ERC721Interface.isApprovedForAll.selector,
+                        orderParameters.offerer,
+                        approvalAddress
+                    ),
+                    true
+                )
+            ) {
+                // Not approved
+                errorsAndWarnings.addError(ERC721Issue.NotApproved.parseInt());
+            }
+        } else if (offerItem.itemType == ItemType.ERC1155) {
+            ERC1155Interface token = ERC1155Interface(offerItem.token);
+
+            // Check for approval
+            if (
+                !address(token).safeStaticCallBool(
+                    abi.encodeWithSelector(
+                        ERC1155Interface.isApprovedForAll.selector,
+                        orderParameters.offerer,
+                        approvalAddress
+                    ),
+                    true
+                )
+            ) {
+                errorsAndWarnings.addError(ERC1155Issue.NotApproved.parseInt());
+            }
+
+            // Get min required balance (max(startAmount, endAmount))
+            uint256 minBalance = offerItem.startAmount < offerItem.endAmount
+                ? offerItem.startAmount
+                : offerItem.endAmount;
+
+            // Check for sufficient balance
+            if (
+                !address(token).safeStaticCallUint256(
+                    abi.encodeWithSelector(
+                        ERC1155Interface.balanceOf.selector,
+                        orderParameters.offerer,
+                        offerItem.identifierOrCriteria
+                    ),
+                    minBalance
+                )
+            ) {
+                // Insufficient balance
+                errorsAndWarnings.addError(
+                    ERC1155Issue.InsufficientBalance.parseInt()
+                );
+            }
+        } else if (offerItem.itemType == ItemType.ERC1155_WITH_CRITERIA) {
+            ERC1155Interface token = ERC1155Interface(offerItem.token);
+
+            // Check for approval
+            if (
+                !address(token).safeStaticCallBool(
+                    abi.encodeWithSelector(
+                        ERC1155Interface.isApprovedForAll.selector,
+                        orderParameters.offerer,
+                        approvalAddress
+                    ),
+                    true
+                )
+            ) {
+                errorsAndWarnings.addError(ERC1155Issue.NotApproved.parseInt());
+            }
+        } else if (offerItem.itemType == ItemType.ERC20) {
+            ERC20Interface token = ERC20Interface(offerItem.token);
+
+            // Get min required balance and approval (max(startAmount, endAmount))
+            uint256 minBalanceAndAllowance = offerItem.startAmount <
+                offerItem.endAmount
+                ? offerItem.startAmount
+                : offerItem.endAmount;
+
+            // Check allowance
+            if (
+                !address(token).safeStaticCallUint256(
+                    abi.encodeWithSelector(
+                        ERC20Interface.allowance.selector,
+                        orderParameters.offerer,
+                        approvalAddress
+                    ),
+                    minBalanceAndAllowance
+                )
+            ) {
+                errorsAndWarnings.addError(
+                    ERC20Issue.InsufficientAllowance.parseInt()
+                );
+            }
+
+            // Check balance
+            if (
+                !address(token).safeStaticCallUint256(
+                    abi.encodeWithSelector(
+                        ERC20Interface.balanceOf.selector,
+                        orderParameters.offerer
+                    ),
+                    minBalanceAndAllowance
+                )
+            ) {
+                errorsAndWarnings.addError(
+                    ERC20Issue.InsufficientBalance.parseInt()
+                );
+            }
+        } else {
+            // Must be native
+            // Get min required balance (max(startAmount, endAmount))
+            uint256 minBalance = offerItem.startAmount < offerItem.endAmount
+                ? offerItem.startAmount
+                : offerItem.endAmount;
+
+            // Check for sufficient balance
+            if (orderParameters.offerer.balance < minBalance) {
+                errorsAndWarnings.addError(
+                    NativeIssue.InsufficientBalance.parseInt()
+                );
+            }
+
+            // Native items can not be pulled so warn
+            errorsAndWarnings.addWarning(OfferIssue.NativeItem.parseInt());
+        }
+    }
+
+    /**
+     * @notice Validate all consideration items for an order
+     * @param orderParameters The parameters for the order to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateConsiderationItems(
+        OrderParameters memory orderParameters,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        return
+            _helper.validateConsiderationItems(orderParameters, seaportAddress);
+    }
+
+    /**
+     * @notice Validate a consideration item
+     * @param orderParameters The parameters for the order to validate
+     * @param considerationItemIndex The index of the consideration item to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateConsiderationItem(
+        OrderParameters memory orderParameters,
+        uint256 considerationItemIndex,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        return
+            _helper.validateConsiderationItem(
+                orderParameters,
+                considerationItemIndex,
+                seaportAddress
+            );
+    }
+
+    /**
+     * @notice Validates the parameters of a consideration item including contract validation
+     * @param orderParameters The parameters for the order to validate
+     * @param considerationItemIndex The index of the consideration item to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateConsiderationItemParameters(
+        OrderParameters memory orderParameters,
+        uint256 considerationItemIndex,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        return
+            _helper.validateConsiderationItemParameters(
+                orderParameters,
+                considerationItemIndex,
+                seaportAddress
+            );
+    }
+
+    /**
+     * @notice Validates the zone call for an order
+     * @param orderParameters The order parameters for the order to validate
+     * @param zoneParameters The zone parameters for the order to validate
+     * @return errorsAndWarnings An ErrorsAndWarnings structs with results
+     */
+    function validateOrderWithZone(
+        OrderParameters memory orderParameters,
+        ZoneParameters memory zoneParameters
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // Call isValidZone to check if zone is set and implements EIP165
+        errorsAndWarnings.concat(isValidZone(orderParameters));
+
+        // Call zone function `validateOrder` with the supplied ZoneParameters
+        if (
+            !orderParameters.zone.safeStaticCallBytes4(
+                abi.encodeWithSelector(
+                    ZoneInterface.validateOrder.selector,
+                    zoneParameters
+                ),
+                ZoneInterface.validateOrder.selector
+            )
+        ) {
+            // Call to validateOrder reverted or returned invalid magic value
+            errorsAndWarnings.addWarning(ZoneIssue.RejectedOrder.parseInt());
+        }
+    }
+
+    /**
+     * @notice Safely check that a contract implements an interface
+     * @param token The token address to check
+     * @param interfaceHash The interface hash to check
+     */
+    function checkInterface(
+        address token,
+        bytes4 interfaceHash
+    ) public view returns (bool) {
+        return
+            token.safeStaticCallBool(
+                abi.encodeWithSelector(
+                    IERC165.supportsInterface.selector,
+                    interfaceHash
+                ),
+                true
+            );
+    }
+
+    function isPaymentToken(ItemType itemType) public pure returns (bool) {
+        return itemType == ItemType.NATIVE || itemType == ItemType.ERC20;
+    }
+
+    /*//////////////////////////////////////////////////////////////
+                        Merkle Helpers
+    //////////////////////////////////////////////////////////////*/
+
+    /**
+     * @notice Sorts an array of token ids by the keccak256 hash of the id. Required ordering of ids
+     *    for other merkle operations.
+     * @param includedTokens An array of included token ids.
+     * @return sortedTokens The sorted `includedTokens` array.
+     */
+    function sortMerkleTokens(
+        uint256[] memory includedTokens
+    ) public view returns (uint256[] memory sortedTokens) {
+        // Sort token ids by the keccak256 hash of the id
+        return _helper.sortMerkleTokens(includedTokens);
+    }
+
+    /**
+     * @notice Creates a merkle root for includedTokens.
+     * @dev `includedTokens` must be sorting in strictly ascending order according to the keccak256 hash of the value.
+     * @return merkleRoot The merkle root
+     * @return errorsAndWarnings Errors and warnings from the operation
+     */
+    function getMerkleRoot(
+        uint256[] memory includedTokens
+    )
+        public
+        view
+        returns (bytes32 merkleRoot, ErrorsAndWarnings memory errorsAndWarnings)
+    {
+        return _helper.getMerkleRoot(includedTokens);
+    }
+
+    /**
+     * @notice Creates a merkle proof for the the targetIndex contained in includedTokens.
+     * @dev `targetIndex` is referring to the index of an element in `includedTokens`.
+     *    `includedTokens` must be sorting in ascending order according to the keccak256 hash of the value.
+     * @return merkleProof The merkle proof
+     * @return errorsAndWarnings Errors and warnings from the operation
+     */
+    function getMerkleProof(
+        uint256[] memory includedTokens,
+        uint256 targetIndex
+    )
+        public
+        view
+        returns (
+            bytes32[] memory merkleProof,
+            ErrorsAndWarnings memory errorsAndWarnings
+        )
+    {
+        return _helper.getMerkleProof(includedTokens, targetIndex);
+    }
+
+    /**
+     * @notice Verifies a merkle proof for the value to prove and given root and proof.
+     * @dev The `valueToProve` is hashed prior to executing the proof verification.
+     * @param merkleRoot The root of the merkle tree
+     * @param merkleProof The merkle proof
+     * @param valueToProve The value to prove
+     * @return whether proof is valid
+     */
+    function verifyMerkleProof(
+        bytes32 merkleRoot,
+        bytes32[] memory merkleProof,
+        uint256 valueToProve
+    ) public view returns (bool) {
+        return _helper.verifyMerkleProof(merkleRoot, merkleProof, valueToProve);
+    }
+}
diff --git a/contracts/seaport/helpers/order-validator/lib/ConsiderationTypeHashes.sol b/contracts/seaport/helpers/order-validator/lib/ConsiderationTypeHashes.sol
new file mode 100644
index 00000000..fc266232
--- /dev/null
+++ b/contracts/seaport/helpers/order-validator/lib/ConsiderationTypeHashes.sol
@@ -0,0 +1,270 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.10;
+
+import "seaport-types/src/lib/ConsiderationStructs.sol";
+
+uint256 constant EIP712_Order_size = 0x180;
+uint256 constant EIP712_OfferItem_size = 0xc0;
+uint256 constant EIP712_ConsiderationItem_size = 0xe0;
+uint256 constant EIP712_DomainSeparator_offset = 0x02;
+uint256 constant EIP712_OrderHash_offset = 0x22;
+uint256 constant EIP712_DigestPayload_size = 0x42;
+uint256 constant EIP_712_PREFIX = (
+    0x1901000000000000000000000000000000000000000000000000000000000000
+);
+
+contract ConsiderationTypeHashes {
+    bytes32 internal immutable _NAME_HASH;
+    bytes32 internal immutable _VERSION_HASH;
+    bytes32 internal immutable _EIP_712_DOMAIN_TYPEHASH;
+    bytes32 internal immutable _OFFER_ITEM_TYPEHASH;
+    bytes32 internal immutable _CONSIDERATION_ITEM_TYPEHASH;
+    bytes32 internal immutable _ORDER_TYPEHASH;
+    bytes32 internal immutable _DOMAIN_SEPARATOR;
+    address internal constant seaportAddress =
+        address(0x00000000000006c7676171937C444f6BDe3D6282);
+
+    constructor() {
+        // Derive hash of the name of the contract.
+        _NAME_HASH = keccak256(bytes("Seaport"));
+
+        // Derive hash of the version string of the contract.
+        _VERSION_HASH = keccak256(bytes("1.2"));
+
+        bytes memory offerItemTypeString = abi.encodePacked(
+            "OfferItem(",
+            "uint8 itemType,",
+            "address token,",
+            "uint256 identifierOrCriteria,",
+            "uint256 startAmount,",
+            "uint256 endAmount",
+            ")"
+        );
+
+        // Construct the ConsiderationItem type string.
+        // prettier-ignore
+        bytes memory considerationItemTypeString = abi.encodePacked(
+            "ConsiderationItem(",
+                "uint8 itemType,",
+                "address token,",
+                "uint256 identifierOrCriteria,",
+                "uint256 startAmount,",
+                "uint256 endAmount,",
+                "address recipient",
+            ")"
+        );
+
+        // Construct the OrderComponents type string, not including the above.
+        // prettier-ignore
+        bytes memory orderComponentsPartialTypeString = abi.encodePacked(
+            "OrderComponents(",
+                "address offerer,",
+                "address zone,",
+                "OfferItem[] offer,",
+                "ConsiderationItem[] consideration,",
+                "uint8 orderType,",
+                "uint256 startTime,",
+                "uint256 endTime,",
+                "bytes32 zoneHash,",
+                "uint256 salt,",
+                "bytes32 conduitKey,",
+                "uint256 counter",
+            ")"
+        );
+        // Derive the OfferItem type hash using the corresponding type string.
+        bytes32 offerItemTypehash = keccak256(offerItemTypeString);
+
+        // Derive ConsiderationItem type hash using corresponding type string.
+        bytes32 considerationItemTypehash = keccak256(
+            considerationItemTypeString
+        );
+
+        // Construct the primary EIP-712 domain type string.
+        // prettier-ignore
+        _EIP_712_DOMAIN_TYPEHASH = keccak256(
+            abi.encodePacked(
+                "EIP712Domain(",
+                    "string name,",
+                    "string version,",
+                    "uint256 chainId,",
+                    "address verifyingContract",
+                ")"
+            )
+        );
+
+        _OFFER_ITEM_TYPEHASH = offerItemTypehash;
+        _CONSIDERATION_ITEM_TYPEHASH = considerationItemTypehash;
+
+        // Derive OrderItem type hash via combination of relevant type strings.
+        _ORDER_TYPEHASH = keccak256(
+            abi.encodePacked(
+                orderComponentsPartialTypeString,
+                considerationItemTypeString,
+                offerItemTypeString
+            )
+        );
+
+        _DOMAIN_SEPARATOR = _deriveDomainSeparator();
+    }
+
+    /**
+     * @dev Internal view function to derive the EIP-712 domain separator.
+     *
+     * @return The derived domain separator.
+     */
+    function _deriveDomainSeparator() internal view returns (bytes32) {
+        // prettier-ignore
+        return keccak256(
+            abi.encode(
+                _EIP_712_DOMAIN_TYPEHASH,
+                _NAME_HASH,
+                _VERSION_HASH,
+                block.chainid,
+                seaportAddress
+            )
+        );
+    }
+
+    /**
+     * @dev Internal pure function to efficiently derive an digest to sign for
+     *      an order in accordance with EIP-712.
+     *
+     * @param orderHash       The order hash.
+     *
+     * @return value The hash.
+     */
+    function _deriveEIP712Digest(
+        bytes32 orderHash
+    ) internal view returns (bytes32 value) {
+        bytes32 domainSeparator = _DOMAIN_SEPARATOR;
+        // Leverage scratch space to perform an efficient hash.
+        assembly {
+            // Place the EIP-712 prefix at the start of scratch space.
+            mstore(0, EIP_712_PREFIX)
+
+            // Place the domain separator in the next region of scratch space.
+            mstore(EIP712_DomainSeparator_offset, domainSeparator)
+
+            // Place the order hash in scratch space, spilling into the first
+            // two bytes of the free memory pointer — this should never be set
+            // as memory cannot be expanded to that size, and will be zeroed out
+            // after the hash is performed.
+            mstore(EIP712_OrderHash_offset, orderHash)
+
+            // Hash the relevant region (65 bytes).
+            value := keccak256(0, EIP712_DigestPayload_size)
+
+            // Clear out the dirtied bits in the memory pointer.
+            mstore(EIP712_OrderHash_offset, 0)
+        }
+    }
+
+    /**
+     * @dev Internal view function to derive the EIP-712 hash for an offer item.
+     *
+     * @param offerItem The offered item to hash.
+     *
+     * @return The hash.
+     */
+    function _hashOfferItem(
+        OfferItem memory offerItem
+    ) internal view returns (bytes32) {
+        return
+            keccak256(
+                abi.encode(
+                    _OFFER_ITEM_TYPEHASH,
+                    offerItem.itemType,
+                    offerItem.token,
+                    offerItem.identifierOrCriteria,
+                    offerItem.startAmount,
+                    offerItem.endAmount
+                )
+            );
+    }
+
+    /**
+     * @dev Internal view function to derive the EIP-712 hash for a consideration item.
+     *
+     * @param considerationItem The consideration item to hash.
+     *
+     * @return The hash.
+     */
+    function _hashConsiderationItem(
+        ConsiderationItem memory considerationItem
+    ) internal view returns (bytes32) {
+        return
+            keccak256(
+                abi.encode(
+                    _CONSIDERATION_ITEM_TYPEHASH,
+                    considerationItem.itemType,
+                    considerationItem.token,
+                    considerationItem.identifierOrCriteria,
+                    considerationItem.startAmount,
+                    considerationItem.endAmount,
+                    considerationItem.recipient
+                )
+            );
+    }
+
+    /**
+     * @dev Internal view function to derive the order hash for a given order.
+     *      Note that only the original consideration items are included in the
+     *      order hash, as additional consideration items may be supplied by the
+     *      caller.
+     *
+     * @param orderParameters The parameters of the order to hash.
+     * @param counter           The counter of the order to hash.
+     *
+     * @return orderHash The hash.
+     */
+    function _deriveOrderHash(
+        OrderParameters memory orderParameters,
+        uint256 counter
+    ) internal view returns (bytes32 orderHash) {
+        // Designate new memory regions for offer and consideration item hashes.
+        bytes32[] memory offerHashes = new bytes32[](
+            orderParameters.offer.length
+        );
+        bytes32[] memory considerationHashes = new bytes32[](
+            orderParameters.totalOriginalConsiderationItems
+        );
+
+        // Iterate over each offer on the order.
+        for (uint256 i = 0; i < orderParameters.offer.length; ++i) {
+            // Hash the offer and place the result into memory.
+            offerHashes[i] = _hashOfferItem(orderParameters.offer[i]);
+        }
+
+        // Iterate over each consideration on the order.
+        for (
+            uint256 i = 0;
+            i < orderParameters.totalOriginalConsiderationItems;
+            ++i
+        ) {
+            // Hash the consideration and place the result into memory.
+            considerationHashes[i] = _hashConsiderationItem(
+                orderParameters.consideration[i]
+            );
+        }
+
+        // Derive and return the order hash as specified by EIP-712.
+
+        return
+            keccak256(
+                abi.encode(
+                    _ORDER_TYPEHASH,
+                    orderParameters.offerer,
+                    orderParameters.zone,
+                    keccak256(abi.encodePacked(offerHashes)),
+                    keccak256(abi.encodePacked(considerationHashes)),
+                    orderParameters.orderType,
+                    orderParameters.startTime,
+                    orderParameters.endTime,
+                    orderParameters.zoneHash,
+                    orderParameters.salt,
+                    orderParameters.conduitKey,
+                    counter
+                )
+            );
+    }
+}
diff --git a/contracts/seaport/helpers/order-validator/lib/ErrorsAndWarnings.sol b/contracts/seaport/helpers/order-validator/lib/ErrorsAndWarnings.sol
new file mode 100644
index 00000000..e9934f44
--- /dev/null
+++ b/contracts/seaport/helpers/order-validator/lib/ErrorsAndWarnings.sol
@@ -0,0 +1,284 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.10;
+
+import {
+    ConduitIssue,
+    ConsiderationIssue,
+    ERC20Issue,
+    ERC721Issue,
+    ERC1155Issue,
+    GenericIssue,
+    OfferIssue,
+    SignatureIssue,
+    StatusIssue,
+    TimeIssue,
+    NativeIssue,
+    IssueParser
+} from "./SeaportValidatorTypes.sol";
+
+struct ErrorsAndWarnings {
+    uint16[] errors;
+    uint16[] warnings;
+}
+
+library ErrorsAndWarningsLib {
+    using IssueParser for ConduitIssue;
+    using IssueParser for ConsiderationIssue;
+    using IssueParser for ERC20Issue;
+    using IssueParser for ERC721Issue;
+    using IssueParser for ERC1155Issue;
+    using IssueParser for GenericIssue;
+    using IssueParser for OfferIssue;
+    using IssueParser for SignatureIssue;
+    using IssueParser for StatusIssue;
+    using IssueParser for TimeIssue;
+    using IssueParser for NativeIssue;
+
+    function concat(
+        ErrorsAndWarnings memory ew1,
+        ErrorsAndWarnings memory ew2
+    ) internal pure {
+        ew1.errors = concatMemory(ew1.errors, ew2.errors);
+        ew1.warnings = concatMemory(ew1.warnings, ew2.warnings);
+    }
+
+    function empty() internal pure returns (ErrorsAndWarnings memory) {
+        return ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+    }
+
+    function addError(
+        uint16 err
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        ErrorsAndWarnings memory ew = ErrorsAndWarnings(
+            new uint16[](0),
+            new uint16[](0)
+        );
+        ew.errors = pushMemory(ew.errors, err);
+        return ew;
+    }
+
+    function addError(
+        ErrorsAndWarnings memory ew,
+        uint16 err
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        ew.errors = pushMemory(ew.errors, err);
+        return ew;
+    }
+
+    function addError(
+        ErrorsAndWarnings memory ew,
+        GenericIssue err
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addError(ew, err.parseInt());
+    }
+
+    function addError(
+        ErrorsAndWarnings memory ew,
+        ERC20Issue err
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addError(ew, err.parseInt());
+    }
+
+    function addError(
+        ErrorsAndWarnings memory ew,
+        ERC721Issue err
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addError(ew, err.parseInt());
+    }
+
+    function addError(
+        ErrorsAndWarnings memory ew,
+        ERC1155Issue err
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addError(ew, err.parseInt());
+    }
+
+    function addError(
+        ErrorsAndWarnings memory ew,
+        ConsiderationIssue err
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addError(ew, err.parseInt());
+    }
+
+    function addError(
+        ErrorsAndWarnings memory ew,
+        OfferIssue err
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addError(ew, err.parseInt());
+    }
+
+    function addError(
+        ErrorsAndWarnings memory ew,
+        SignatureIssue err
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addError(ew, err.parseInt());
+    }
+
+    function addError(
+        ErrorsAndWarnings memory ew,
+        TimeIssue err
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addError(ew, err.parseInt());
+    }
+
+    function addError(
+        ErrorsAndWarnings memory ew,
+        ConduitIssue err
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addError(ew, err.parseInt());
+    }
+
+    function addError(
+        ErrorsAndWarnings memory ew,
+        StatusIssue err
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addError(ew, err.parseInt());
+    }
+
+    function addWarning(
+        uint16 warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        ErrorsAndWarnings memory ew = ErrorsAndWarnings(
+            new uint16[](0),
+            new uint16[](0)
+        );
+        ew.warnings = pushMemory(ew.warnings, warn);
+        return ew;
+    }
+
+    function addWarning(
+        ErrorsAndWarnings memory ew,
+        uint16 warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        ew.warnings = pushMemory(ew.warnings, warn);
+        return ew;
+    }
+
+    function addWarning(
+        ErrorsAndWarnings memory ew,
+        GenericIssue warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addWarning(ew, warn.parseInt());
+    }
+
+    function addWarning(
+        ErrorsAndWarnings memory ew,
+        ERC20Issue warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addWarning(ew, warn.parseInt());
+    }
+
+    function addWarning(
+        ErrorsAndWarnings memory ew,
+        ERC721Issue warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addWarning(ew, warn.parseInt());
+    }
+
+    function addWarning(
+        ErrorsAndWarnings memory ew,
+        ERC1155Issue warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addWarning(ew, warn.parseInt());
+    }
+
+    function addWarning(
+        ErrorsAndWarnings memory ew,
+        OfferIssue warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addWarning(ew, warn.parseInt());
+    }
+
+    function addWarning(
+        ErrorsAndWarnings memory ew,
+        ConsiderationIssue warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addWarning(ew, warn.parseInt());
+    }
+
+    function addWarning(
+        ErrorsAndWarnings memory ew,
+        SignatureIssue warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addWarning(ew, warn.parseInt());
+    }
+
+    function addWarning(
+        ErrorsAndWarnings memory ew,
+        TimeIssue warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addWarning(ew, warn.parseInt());
+    }
+
+    function addWarning(
+        ErrorsAndWarnings memory ew,
+        ConduitIssue warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addWarning(ew, warn.parseInt());
+    }
+
+    function addWarning(
+        ErrorsAndWarnings memory ew,
+        StatusIssue warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addWarning(ew, warn.parseInt());
+    }
+
+    function addWarning(
+        ErrorsAndWarnings memory ew,
+        NativeIssue warn
+    ) internal pure returns (ErrorsAndWarnings memory) {
+        return addWarning(ew, warn.parseInt());
+    }
+
+    function hasErrors(
+        ErrorsAndWarnings memory ew
+    ) internal pure returns (bool) {
+        return ew.errors.length != 0;
+    }
+
+    function hasWarnings(
+        ErrorsAndWarnings memory ew
+    ) internal pure returns (bool) {
+        return ew.warnings.length != 0;
+    }
+
+    // Helper Functions
+    function concatMemory(
+        uint16[] memory array1,
+        uint16[] memory array2
+    ) private pure returns (uint16[] memory) {
+        if (array1.length == 0) {
+            return array2;
+        } else if (array2.length == 0) {
+            return array1;
+        }
+
+        uint16[] memory returnValue = new uint16[](
+            array1.length + array2.length
+        );
+
+        for (uint256 i = 0; i < array1.length; i++) {
+            returnValue[i] = array1[i];
+        }
+        for (uint256 i = 0; i < array2.length; i++) {
+            returnValue[i + array1.length] = array2[i];
+        }
+
+        return returnValue;
+    }
+
+    function pushMemory(
+        uint16[] memory uint16Array,
+        uint16 newValue
+    ) internal pure returns (uint16[] memory) {
+        uint16[] memory returnValue = new uint16[](uint16Array.length + 1);
+
+        for (uint256 i = 0; i < uint16Array.length; i++) {
+            returnValue[i] = uint16Array[i];
+        }
+        returnValue[uint16Array.length] = newValue;
+
+        return returnValue;
+    }
+}
diff --git a/contracts/seaport/helpers/order-validator/lib/Murky.sol b/contracts/seaport/helpers/order-validator/lib/Murky.sol
new file mode 100644
index 00000000..d49675a3
--- /dev/null
+++ b/contracts/seaport/helpers/order-validator/lib/Murky.sol
@@ -0,0 +1,400 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.10;
+
+import {
+    ErrorsAndWarnings,
+    ErrorsAndWarningsLib
+} from "./ErrorsAndWarnings.sol";
+
+import { IssueParser, MerkleIssue } from "./SeaportValidatorTypes.sol";
+
+contract Murky {
+    using ErrorsAndWarningsLib for ErrorsAndWarnings;
+    using IssueParser for MerkleIssue;
+
+    bool internal constant HASH_ODD_WITH_ZERO = false;
+
+    function _verifyProof(
+        bytes32 root,
+        bytes32[] memory proof,
+        bytes32 valueToProve
+    ) internal pure returns (bool) {
+        // proof length must be less than max array size
+        bytes32 rollingHash = valueToProve;
+        uint256 length = proof.length;
+        unchecked {
+            for (uint256 i = 0; i < length; ++i) {
+                rollingHash = _hashLeafPairs(rollingHash, proof[i]);
+            }
+        }
+        return root == rollingHash;
+    }
+
+    /********************
+     * HASHING FUNCTION *
+     ********************/
+
+    /// ascending sort and concat prior to hashing
+    function _hashLeafPairs(bytes32 left, bytes32 right)
+        internal
+        pure
+        returns (bytes32 _hash)
+    {
+        assembly {
+            switch lt(left, right)
+            case 0 {
+                mstore(0x0, right)
+                mstore(0x20, left)
+            }
+            default {
+                mstore(0x0, left)
+                mstore(0x20, right)
+            }
+            _hash := keccak256(0x0, 0x40)
+        }
+    }
+
+    /********************
+     * PROOF GENERATION *
+     ********************/
+
+    function _getRoot(uint256[] memory data)
+        internal
+        pure
+        returns (bytes32 result, ErrorsAndWarnings memory errorsAndWarnings)
+    {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        if (data.length < 2) {
+            errorsAndWarnings.addError(MerkleIssue.SingleLeaf.parseInt());
+            return (0, errorsAndWarnings);
+        }
+
+        bool hashOddWithZero = HASH_ODD_WITH_ZERO;
+
+        if (!_processInput(data)) {
+            errorsAndWarnings.addError(MerkleIssue.Unsorted.parseInt());
+            return (0, errorsAndWarnings);
+        }
+
+        assembly {
+            function hashLeafPairs(left, right) -> _hash {
+                switch lt(left, right)
+                case 0 {
+                    mstore(0x0, right)
+                    mstore(0x20, left)
+                }
+                default {
+                    mstore(0x0, left)
+                    mstore(0x20, right)
+                }
+                _hash := keccak256(0x0, 0x40)
+            }
+            function hashLevel(_data, length, _hashOddWithZero) -> newLength {
+                // we will be modifying data in-place, so set result pointer to data pointer
+                let _result := _data
+                // get length of original data array
+                // let length := mload(_data)
+                // bool to track if we need to hash the last element of an odd-length array with zero
+                let oddLength
+
+                // if length is odd, we need to hash the last element with zero
+                switch and(length, 1)
+                case 1 {
+                    // if length is odd, add 1 so division by 2 will round up
+                    newLength := add(1, div(length, 2))
+                    oddLength := 1
+                }
+                default {
+                    newLength := div(length, 2)
+                }
+                // todo: necessary?
+                // mstore(_data, newLength)
+                let resultIndexPointer := add(0x20, _data)
+                let dataIndexPointer := resultIndexPointer
+
+                // stop iterating over for loop at length-1
+                let stopIteration := add(_data, mul(length, 0x20))
+                // write result array in-place over data array
+                for {
+
+                } lt(dataIndexPointer, stopIteration) {
+
+                } {
+                    // get next two elements from data, hash them together
+                    let data1 := mload(dataIndexPointer)
+                    let data2 := mload(add(dataIndexPointer, 0x20))
+                    let hashedPair := hashLeafPairs(data1, data2)
+                    // overwrite an element of data array with
+                    mstore(resultIndexPointer, hashedPair)
+                    // increment result pointer by 1 slot
+                    resultIndexPointer := add(0x20, resultIndexPointer)
+                    // increment data pointer by 2 slot
+                    dataIndexPointer := add(0x40, dataIndexPointer)
+                }
+                // we did not yet hash last index if odd-length
+                if oddLength {
+                    let data1 := mload(dataIndexPointer)
+                    let nextValue
+                    switch _hashOddWithZero
+                    case 0 {
+                        nextValue := data1
+                    }
+                    default {
+                        nextValue := hashLeafPairs(data1, 0)
+                    }
+                    mstore(resultIndexPointer, nextValue)
+                }
+            }
+
+            let dataLength := mload(data)
+            for {
+
+            } gt(dataLength, 1) {
+
+            } {
+                dataLength := hashLevel(data, dataLength, hashOddWithZero)
+            }
+            result := mload(add(0x20, data))
+        }
+    }
+
+    function _getProof(uint256[] memory data, uint256 node)
+        internal
+        pure
+        returns (
+            bytes32[] memory result,
+            ErrorsAndWarnings memory errorsAndWarnings
+        )
+    {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        if (data.length < 2) {
+            errorsAndWarnings.addError(MerkleIssue.SingleLeaf.parseInt());
+            return (new bytes32[](0), errorsAndWarnings);
+        }
+
+        bool hashOddWithZero = HASH_ODD_WITH_ZERO;
+
+        if (!_processInput(data)) {
+            errorsAndWarnings.addError(MerkleIssue.Unsorted.parseInt());
+            return (new bytes32[](0), errorsAndWarnings);
+        }
+
+        // The size of the proof is equal to the ceiling of log2(numLeaves)
+        // Two overflow risks: node, pos
+        // node: max array size is 2**256-1. Largest index in the array will be 1 less than that. Also,
+        // for dynamic arrays, size is limited to 2**64-1
+        // pos: pos is bounded by log2(data.length), which should be less than type(uint256).max
+        assembly {
+            function hashLeafPairs(left, right) -> _hash {
+                switch lt(left, right)
+                case 0 {
+                    mstore(0x0, right)
+                    mstore(0x20, left)
+                }
+                default {
+                    mstore(0x0, left)
+                    mstore(0x20, right)
+                }
+                _hash := keccak256(0x0, 0x40)
+            }
+            function hashLevel(_data, length, _hashOddWithZero) -> newLength {
+                // we will be modifying data in-place, so set result pointer to data pointer
+                let _result := _data
+                // get length of original data array
+                // let length := mload(_data)
+                // bool to track if we need to hash the last element of an odd-length array with zero
+                let oddLength
+
+                // if length is odd, we'll need to hash the last element with zero
+                switch and(length, 1)
+                case 1 {
+                    // if length is odd, add 1 so division by 2 will round up
+                    newLength := add(1, div(length, 2))
+                    oddLength := 1
+                }
+                default {
+                    newLength := div(length, 2)
+                }
+                // todo: necessary?
+                // mstore(_data, newLength)
+                let resultIndexPointer := add(0x20, _data)
+                let dataIndexPointer := resultIndexPointer
+
+                // stop iterating over for loop at length-1
+                let stopIteration := add(_data, mul(length, 0x20))
+                // write result array in-place over data array
+                for {
+
+                } lt(dataIndexPointer, stopIteration) {
+
+                } {
+                    // get next two elements from data, hash them together
+                    let data1 := mload(dataIndexPointer)
+                    let data2 := mload(add(dataIndexPointer, 0x20))
+                    let hashedPair := hashLeafPairs(data1, data2)
+                    // overwrite an element of data array with
+                    mstore(resultIndexPointer, hashedPair)
+                    // increment result pointer by 1 slot
+                    resultIndexPointer := add(0x20, resultIndexPointer)
+                    // increment data pointer by 2 slot
+                    dataIndexPointer := add(0x40, dataIndexPointer)
+                }
+                // we did not yet hash last index if odd-length
+                if oddLength {
+                    let data1 := mload(dataIndexPointer)
+                    let nextValue
+                    switch _hashOddWithZero
+                    case 0 {
+                        nextValue := data1
+                    }
+                    default {
+                        nextValue := hashLeafPairs(data1, 0)
+                    }
+                    mstore(resultIndexPointer, nextValue)
+                }
+            }
+
+            // set result pointer to free memory
+            result := mload(0x40)
+            // get pointer to first index of result
+            let resultIndexPtr := add(0x20, result)
+            // declare so we can use later
+            let newLength
+            // put length of data onto stack
+            let dataLength := mload(data)
+            for {
+                // repeat until only one element is left
+            } gt(dataLength, 1) {
+
+            } {
+                // bool if node is odd
+                let oddNodeIndex := and(node, 1)
+                // bool if node is last
+                let lastNodeIndex := eq(dataLength, add(1, node))
+                // store both bools in one value so we can switch on it
+                let switchVal := or(shl(1, lastNodeIndex), oddNodeIndex)
+                switch switchVal
+                // 00 - neither odd nor last
+                case 0 {
+                    // store data[node+1] at result[i]
+                    // get pointer to result[node+1] by adding 2 to node and multiplying by 0x20
+                    // to account for the fact that result points to array length, not first index
+                    mstore(
+                        resultIndexPtr,
+                        mload(add(data, mul(0x20, add(2, node))))
+                    )
+                }
+                // 10 - node is last
+                case 2 {
+                    // store 0 at result[i]
+                    mstore(resultIndexPtr, 0)
+                }
+                // 01 or 11 - node is odd (and possibly also last)
+                default {
+                    // store data[node-1] at result[i]
+                    mstore(resultIndexPtr, mload(add(data, mul(0x20, node))))
+                }
+                // increment result index
+                resultIndexPtr := add(0x20, resultIndexPtr)
+
+                // get new node index
+                node := div(node, 2)
+                // keep track of how long result array is
+                newLength := add(1, newLength)
+                // compute the next hash level, overwriting data, and get the new length
+                dataLength := hashLevel(data, dataLength, hashOddWithZero)
+            }
+            // store length of result array at pointer
+            mstore(result, newLength)
+            // set free mem pointer to word after end of result array
+            mstore(0x40, resultIndexPtr)
+        }
+    }
+
+    /**
+     * Hashes each element of the input array in place using keccak256
+     */
+    function _processInput(uint256[] memory data)
+        private
+        pure
+        returns (bool sorted)
+    {
+        sorted = true;
+
+        // Hash inputs with keccak256
+        for (uint256 i = 0; i < data.length; ++i) {
+            assembly {
+                mstore(
+                    add(data, mul(0x20, add(1, i))),
+                    keccak256(add(data, mul(0x20, add(1, i))), 0x20)
+                )
+                // for every element after the first, hashed value must be greater than the last one
+                if and(
+                    gt(i, 0),
+                    iszero(
+                        gt(
+                            mload(add(data, mul(0x20, add(1, i)))),
+                            mload(add(data, mul(0x20, add(1, sub(i, 1)))))
+                        )
+                    )
+                ) {
+                    sorted := 0 // Elements not ordered by hash
+                }
+            }
+        }
+    }
+
+    // Sort uint256 in order of the keccak256 hashes
+    struct HashAndIntTuple {
+        uint256 num;
+        bytes32 hash;
+    }
+
+    function _sortUint256ByHash(uint256[] memory values)
+        internal
+        pure
+        returns (uint256[] memory sortedValues)
+    {
+        HashAndIntTuple[] memory toSort = new HashAndIntTuple[](values.length);
+        for (uint256 i = 0; i < values.length; i++) {
+            toSort[i] = HashAndIntTuple(
+                values[i],
+                keccak256(abi.encode(values[i]))
+            );
+        }
+
+        _quickSort(toSort, 0, int256(toSort.length - 1));
+
+        sortedValues = new uint256[](values.length);
+        for (uint256 i = 0; i < values.length; i++) {
+            sortedValues[i] = toSort[i].num;
+        }
+    }
+
+    function _quickSort(
+        HashAndIntTuple[] memory arr,
+        int256 left,
+        int256 right
+    ) internal pure {
+        int256 i = left;
+        int256 j = right;
+        if (i == j) return;
+        bytes32 pivot = arr[uint256(left + (right - left) / 2)].hash;
+        while (i <= j) {
+            while (arr[uint256(i)].hash < pivot) i++;
+            while (pivot < arr[uint256(j)].hash) j--;
+            if (i <= j) {
+                (arr[uint256(i)], arr[uint256(j)]) = (
+                    arr[uint256(j)],
+                    arr[uint256(i)]
+                );
+                i++;
+                j--;
+            }
+        }
+        if (left < j) _quickSort(arr, left, j);
+        if (i < right) _quickSort(arr, i, right);
+    }
+}
diff --git a/contracts/seaport/helpers/order-validator/lib/ReadOnlyOrderValidator.sol b/contracts/seaport/helpers/order-validator/lib/ReadOnlyOrderValidator.sol
new file mode 100644
index 00000000..c0d23124
--- /dev/null
+++ b/contracts/seaport/helpers/order-validator/lib/ReadOnlyOrderValidator.sol
@@ -0,0 +1,722 @@
+// SPDX-License-Identifier: MIT
+pragma solidity 0.8.17;
+
+import { OrderType } from "seaport-types/src/lib/ConsiderationEnums.sol";
+
+import {
+    Order,
+    OrderComponents,
+    OrderParameters,
+    OrderStatus
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    _revertConsiderationLengthNotEqualToTotalOriginal,
+    _revertMissingOriginalConsiderationItems
+} from "seaport-types/src/lib/ConsiderationErrors.sol";
+
+import {
+    SignatureVerification
+} from "seaport-core/src/lib/SignatureVerification.sol";
+
+import {
+    _revertOrderIsCancelled,
+    _revertOrderPartiallyFilled,
+    _revertOrderAlreadyFilled
+} from "seaport-types/src/lib/ConsiderationErrors.sol";
+
+import { SeaportInterface } from "seaport-sol/src/SeaportInterface.sol";
+
+import {
+    EIP_712_PREFIX,
+    EIP712_DigestPayload_size,
+    EIP712_DomainSeparator_offset,
+    EIP712_OrderHash_offset,
+    BulkOrderProof_keyShift,
+    BulkOrderProof_keySize,
+    BulkOrderProof_lengthAdjustmentBeforeMask,
+    BulkOrderProof_lengthRangeAfterMask,
+    BulkOrderProof_minSize,
+    BulkOrderProof_rangeSize,
+    ECDSA_MaxLength,
+    OneWordShift,
+    ThirtyOneBytes,
+    OneWord,
+    TwoWords,
+    BulkOrder_Typehash_Height_One,
+    BulkOrder_Typehash_Height_Two,
+    BulkOrder_Typehash_Height_Three,
+    BulkOrder_Typehash_Height_Four,
+    BulkOrder_Typehash_Height_Five,
+    BulkOrder_Typehash_Height_Six,
+    BulkOrder_Typehash_Height_Seven,
+    BulkOrder_Typehash_Height_Eight,
+    BulkOrder_Typehash_Height_Nine,
+    BulkOrder_Typehash_Height_Ten,
+    BulkOrder_Typehash_Height_Eleven,
+    BulkOrder_Typehash_Height_Twelve,
+    BulkOrder_Typehash_Height_Thirteen,
+    BulkOrder_Typehash_Height_Fourteen,
+    BulkOrder_Typehash_Height_Fifteen,
+    BulkOrder_Typehash_Height_Sixteen,
+    BulkOrder_Typehash_Height_Seventeen,
+    BulkOrder_Typehash_Height_Eighteen,
+    BulkOrder_Typehash_Height_Nineteen,
+    BulkOrder_Typehash_Height_Twenty,
+    BulkOrder_Typehash_Height_TwentyOne,
+    BulkOrder_Typehash_Height_TwentyTwo,
+    BulkOrder_Typehash_Height_TwentyThree,
+    BulkOrder_Typehash_Height_TwentyFour
+} from "seaport-types/src/lib/ConsiderationConstants.sol";
+
+contract ReadOnlyOrderValidator is SignatureVerification {
+    function canValidate(
+        address seaport,
+        Order[] memory orders
+    ) external view returns (bool) {
+        return _validate(orders, SeaportInterface(seaport));
+    }
+
+    /**
+     * @dev Internal function to validate an arbitrary number of orders, thereby
+     *      registering their signatures as valid and allowing the fulfiller to
+     *      skip signature verification on fulfillment. Note that validated
+     *      orders may still be unfulfillable due to invalid item amounts or
+     *      other factors; callers should determine whether validated orders are
+     *      fulfillable by simulating the fulfillment call prior to execution.
+     *      Also note that anyone can validate a signed order, but only the
+     *      offerer can validate an order without supplying a signature.
+     *
+     * @param orders The orders to validate.
+     *
+     * @return validated A boolean indicating whether the supplied orders were
+     *                   successfully validated.
+     */
+    function _validate(
+        Order[] memory orders,
+        SeaportInterface seaport
+    ) internal view returns (bool validated) {
+        (, bytes32 domainSeparator, ) = seaport.information();
+
+        // Declare variables outside of the loop.
+        OrderStatus memory orderStatus;
+        bytes32 orderHash;
+        address offerer;
+
+        // Skip overflow check as for loop is indexed starting at zero.
+        unchecked {
+            // Read length of the orders array from memory and place on stack.
+            uint256 totalOrders = orders.length;
+
+            // Iterate over each order.
+            for (uint256 i = 0; i < totalOrders; ++i) {
+                // Retrieve the order.
+                Order memory order = orders[i];
+
+                // Retrieve the order parameters.
+                OrderParameters memory orderParameters = order.parameters;
+
+                // Skip contract orders.
+                if (orderParameters.orderType == OrderType.CONTRACT) {
+                    continue;
+                }
+
+                // Move offerer from memory to the stack.
+                offerer = orderParameters.offerer;
+
+                // Get current counter & use it w/ params to derive order hash.
+                orderHash = _assertConsiderationLengthAndGetOrderHash(
+                    orderParameters,
+                    seaport
+                );
+
+                {
+                    // Retrieve the order status using the derived order hash.
+                    (
+                        bool isValidated,
+                        bool isCancelled,
+                        uint256 totalFilled,
+                        uint256 totalSize
+                    ) = seaport.getOrderStatus(orderHash);
+                    orderStatus = OrderStatus(
+                        isValidated,
+                        isCancelled,
+                        uint120(totalFilled),
+                        uint120(totalSize)
+                    );
+                }
+
+                // Ensure order is fillable and retrieve the filled amount.
+                _verifyOrderStatus(
+                    orderHash,
+                    orderStatus,
+                    false, // Signifies that partially filled orders are valid.
+                    true // Signifies to revert if the order is invalid.
+                );
+
+                // If the order has not already been validated...
+                if (!orderStatus.isValidated) {
+                    // Ensure that consideration array length is equal to the
+                    // total original consideration items value.
+                    if (
+                        orderParameters.consideration.length !=
+                        orderParameters.totalOriginalConsiderationItems
+                    ) {
+                        _revertConsiderationLengthNotEqualToTotalOriginal();
+                    }
+
+                    // Verify the supplied signature.
+                    _verifySignature(
+                        offerer,
+                        orderHash,
+                        order.signature,
+                        domainSeparator
+                    );
+
+                    // Update order status to mark the order as valid.
+                    // orderStatus.isValidated = true;
+
+                    // Emit an event signifying the order has been validated.
+                    // emit OrderValidated(orderHash, orderParameters);
+                }
+            }
+        }
+
+        // Return a boolean indicating that orders were successfully validated.
+        validated = true;
+    }
+
+    /**
+     * @dev Internal view function to validate that a given order is fillable
+     *      and not cancelled based on the order status.
+     *
+     * @param orderHash       The order hash.
+     * @param orderStatus     The status of the order, including whether it has
+     *                        been cancelled and the fraction filled.
+     * @param onlyAllowUnused A boolean flag indicating whether partial fills
+     *                        are supported by the calling function.
+     * @param revertOnInvalid A boolean indicating whether to revert if the
+     *                        order has been cancelled or filled beyond the
+     *                        allowable amount.
+     *
+     * @return valid A boolean indicating whether the order is valid.
+     */
+    function _verifyOrderStatus(
+        bytes32 orderHash,
+        OrderStatus memory orderStatus,
+        bool onlyAllowUnused,
+        bool revertOnInvalid
+    ) internal pure returns (bool valid) {
+        // Ensure that the order has not been cancelled.
+        if (orderStatus.isCancelled) {
+            // Only revert if revertOnInvalid has been supplied as true.
+            if (revertOnInvalid) {
+                _revertOrderIsCancelled(orderHash);
+            }
+
+            // Return false as the order status is invalid.
+            return false;
+        }
+
+        // Read order status numerator and place on stack.
+        uint256 orderStatusNumerator = orderStatus.numerator;
+
+        // If the order is not entirely unused...
+        if (orderStatusNumerator != 0) {
+            // ensure the order has not been partially filled when not allowed.
+            if (onlyAllowUnused) {
+                // Always revert on partial fills when onlyAllowUnused is true.
+                _revertOrderPartiallyFilled(orderHash);
+            }
+            // Otherwise, ensure that order has not been entirely filled.
+            else if (orderStatusNumerator >= orderStatus.denominator) {
+                // Only revert if revertOnInvalid has been supplied as true.
+                if (revertOnInvalid) {
+                    _revertOrderAlreadyFilled(orderHash);
+                }
+
+                // Return false as the order status is invalid.
+                return false;
+            }
+        }
+
+        // Return true as the order status is valid.
+        valid = true;
+    }
+
+    /**
+     * @dev Internal view function to verify the signature of an order. An
+     *      ERC-1271 fallback will be attempted if either the signature length
+     *      is not 64 or 65 bytes or if the recovered signer does not match the
+     *      supplied offerer. Note that in cases where a 64 or 65 byte signature
+     *      is supplied, only standard ECDSA signatures that recover to a
+     *      non-zero address are supported.
+     *
+     * @param offerer   The offerer for the order.
+     * @param orderHash The order hash.
+     * @param signature A signature from the offerer indicating that the order
+     *                  has been approved.
+     */
+    function _verifySignature(
+        address offerer,
+        bytes32 orderHash,
+        bytes memory signature,
+        bytes32 domainSeparator
+    ) internal view {
+        // Determine whether the offerer is the caller.
+        bool offererIsCaller;
+        assembly {
+            offererIsCaller := eq(offerer, caller())
+        }
+
+        // Skip signature verification if the offerer is the caller.
+        if (offererIsCaller) {
+            return;
+        }
+
+        // Derive original EIP-712 digest using domain separator and order hash.
+        bytes32 originalDigest = _deriveEIP712Digest(
+            domainSeparator,
+            orderHash
+        );
+
+        // Read the length of the signature from memory and place on the stack.
+        uint256 originalSignatureLength = signature.length;
+
+        // Determine effective digest if signature has a valid bulk order size.
+        bytes32 digest;
+        if (_isValidBulkOrderSize(originalSignatureLength)) {
+            // Rederive order hash and digest using bulk order proof.
+            (orderHash) = _computeBulkOrderProof(signature, orderHash);
+            digest = _deriveEIP712Digest(domainSeparator, orderHash);
+        } else {
+            // Supply the original digest as the effective digest.
+            digest = originalDigest;
+        }
+
+        // Ensure that the signature for the digest is valid for the offerer.
+        _assertValidSignature(
+            offerer,
+            digest,
+            originalDigest,
+            originalSignatureLength,
+            signature
+        );
+    }
+
+    /**
+     * @dev Determines whether the specified bulk order size is valid.
+     *
+     * @param signatureLength The signature length of the bulk order to check.
+     *
+     * @return validLength True if bulk order size is valid, false otherwise.
+     */
+    function _isValidBulkOrderSize(
+        uint256 signatureLength
+    ) internal pure returns (bool validLength) {
+        // Utilize assembly to validate the length; the equivalent logic is
+        // (64 + x) + 3 + 32y where (0 <= x <= 1) and (1 <= y <= 24).
+        assembly {
+            validLength := and(
+                lt(
+                    sub(signatureLength, BulkOrderProof_minSize),
+                    BulkOrderProof_rangeSize
+                ),
+                lt(
+                    and(
+                        add(
+                            signatureLength,
+                            BulkOrderProof_lengthAdjustmentBeforeMask
+                        ),
+                        ThirtyOneBytes
+                    ),
+                    BulkOrderProof_lengthRangeAfterMask
+                )
+            )
+        }
+    }
+
+    /**
+     * @dev Computes the bulk order hash for the specified proof and leaf. Note
+     *      that if an index that exceeds the number of orders in the bulk order
+     *      payload will instead "wrap around" and refer to an earlier index.
+     *
+     * @param proofAndSignature The proof and signature of the bulk order.
+     * @param leaf              The leaf of the bulk order tree.
+     *
+     * @return bulkOrderHash The bulk order hash.
+     */
+    function _computeBulkOrderProof(
+        bytes memory proofAndSignature,
+        bytes32 leaf
+    ) internal pure returns (bytes32 bulkOrderHash) {
+        // Declare arguments for the root hash and the height of the proof.
+        bytes32 root;
+        uint256 height;
+
+        // Utilize assembly to efficiently derive the root hash using the proof.
+        assembly {
+            // Retrieve the length of the proof, key, and signature combined.
+            let fullLength := mload(proofAndSignature)
+
+            // If proofAndSignature has odd length, it is a compact signature
+            // with 64 bytes.
+            let signatureLength := sub(ECDSA_MaxLength, and(fullLength, 1))
+
+            // Derive height (or depth of tree) with signature and proof length.
+            height := shr(OneWordShift, sub(fullLength, signatureLength))
+
+            // Update the length in memory to only include the signature.
+            mstore(proofAndSignature, signatureLength)
+
+            // Derive the pointer for the key using the signature length.
+            let keyPtr := add(proofAndSignature, add(OneWord, signatureLength))
+
+            // Retrieve the three-byte key using the derived pointer.
+            let key := shr(BulkOrderProof_keyShift, mload(keyPtr))
+
+            /// Retrieve pointer to first proof element by applying a constant
+            // for the key size to the derived key pointer.
+            let proof := add(keyPtr, BulkOrderProof_keySize)
+
+            // Compute level 1.
+            let scratchPtr1 := shl(OneWordShift, and(key, 1))
+            mstore(scratchPtr1, leaf)
+            mstore(xor(scratchPtr1, OneWord), mload(proof))
+
+            // Compute remaining proofs.
+            for {
+                let i := 1
+            } lt(i, height) {
+                i := add(i, 1)
+            } {
+                proof := add(proof, OneWord)
+                let scratchPtr := shl(OneWordShift, and(shr(i, key), 1))
+                mstore(scratchPtr, keccak256(0, TwoWords))
+                mstore(xor(scratchPtr, OneWord), mload(proof))
+            }
+
+            // Compute root hash.
+            root := keccak256(0, TwoWords)
+        }
+
+        // Retrieve appropriate typehash constant based on height.
+        bytes32 rootTypeHash = _lookupBulkOrderTypehash(height);
+
+        // Use the typehash and the root hash to derive final bulk order hash.
+        assembly {
+            mstore(0, rootTypeHash)
+            mstore(OneWord, root)
+            bulkOrderHash := keccak256(0, TwoWords)
+        }
+    }
+
+    /**
+     * @dev Internal pure function to look up one of twenty-four potential bulk
+     *      order typehash constants based on the height of the bulk order tree.
+     *      Note that values between one and twenty-four are supported, which is
+     *      enforced by _isValidBulkOrderSize.
+     *
+     * @param _treeHeight The height of the bulk order tree. The value must be
+     *                    between one and twenty-four.
+     *
+     * @return _typeHash The EIP-712 typehash for the bulk order type with the
+     *                   given height.
+     */
+    function _lookupBulkOrderTypehash(
+        uint256 _treeHeight
+    ) internal pure returns (bytes32 _typeHash) {
+        // Utilize assembly to efficiently retrieve correct bulk order typehash.
+        assembly {
+            // Use a Yul function to enable use of the `leave` keyword
+            // to stop searching once the appropriate type hash is found.
+            function lookupTypeHash(treeHeight) -> typeHash {
+                // Handle tree heights one through eight.
+                if lt(treeHeight, 9) {
+                    // Handle tree heights one through four.
+                    if lt(treeHeight, 5) {
+                        // Handle tree heights one and two.
+                        if lt(treeHeight, 3) {
+                            // Utilize branchless logic to determine typehash.
+                            typeHash := ternary(
+                                eq(treeHeight, 1),
+                                BulkOrder_Typehash_Height_One,
+                                BulkOrder_Typehash_Height_Two
+                            )
+
+                            // Exit the function once typehash has been located.
+                            leave
+                        }
+
+                        // Handle height three and four via branchless logic.
+                        typeHash := ternary(
+                            eq(treeHeight, 3),
+                            BulkOrder_Typehash_Height_Three,
+                            BulkOrder_Typehash_Height_Four
+                        )
+
+                        // Exit the function once typehash has been located.
+                        leave
+                    }
+
+                    // Handle tree height five and six.
+                    if lt(treeHeight, 7) {
+                        // Utilize branchless logic to determine typehash.
+                        typeHash := ternary(
+                            eq(treeHeight, 5),
+                            BulkOrder_Typehash_Height_Five,
+                            BulkOrder_Typehash_Height_Six
+                        )
+
+                        // Exit the function once typehash has been located.
+                        leave
+                    }
+
+                    // Handle height seven and eight via branchless logic.
+                    typeHash := ternary(
+                        eq(treeHeight, 7),
+                        BulkOrder_Typehash_Height_Seven,
+                        BulkOrder_Typehash_Height_Eight
+                    )
+
+                    // Exit the function once typehash has been located.
+                    leave
+                }
+
+                // Handle tree height nine through sixteen.
+                if lt(treeHeight, 17) {
+                    // Handle tree height nine through twelve.
+                    if lt(treeHeight, 13) {
+                        // Handle tree height nine and ten.
+                        if lt(treeHeight, 11) {
+                            // Utilize branchless logic to determine typehash.
+                            typeHash := ternary(
+                                eq(treeHeight, 9),
+                                BulkOrder_Typehash_Height_Nine,
+                                BulkOrder_Typehash_Height_Ten
+                            )
+
+                            // Exit the function once typehash has been located.
+                            leave
+                        }
+
+                        // Handle height eleven and twelve via branchless logic.
+                        typeHash := ternary(
+                            eq(treeHeight, 11),
+                            BulkOrder_Typehash_Height_Eleven,
+                            BulkOrder_Typehash_Height_Twelve
+                        )
+
+                        // Exit the function once typehash has been located.
+                        leave
+                    }
+
+                    // Handle tree height thirteen and fourteen.
+                    if lt(treeHeight, 15) {
+                        // Utilize branchless logic to determine typehash.
+                        typeHash := ternary(
+                            eq(treeHeight, 13),
+                            BulkOrder_Typehash_Height_Thirteen,
+                            BulkOrder_Typehash_Height_Fourteen
+                        )
+
+                        // Exit the function once typehash has been located.
+                        leave
+                    }
+                    // Handle height fifteen and sixteen via branchless logic.
+                    typeHash := ternary(
+                        eq(treeHeight, 15),
+                        BulkOrder_Typehash_Height_Fifteen,
+                        BulkOrder_Typehash_Height_Sixteen
+                    )
+
+                    // Exit the function once typehash has been located.
+                    leave
+                }
+
+                // Handle tree height seventeen through twenty.
+                if lt(treeHeight, 21) {
+                    // Handle tree height seventeen and eighteen.
+                    if lt(treeHeight, 19) {
+                        // Utilize branchless logic to determine typehash.
+                        typeHash := ternary(
+                            eq(treeHeight, 17),
+                            BulkOrder_Typehash_Height_Seventeen,
+                            BulkOrder_Typehash_Height_Eighteen
+                        )
+
+                        // Exit the function once typehash has been located.
+                        leave
+                    }
+
+                    // Handle height nineteen and twenty via branchless logic.
+                    typeHash := ternary(
+                        eq(treeHeight, 19),
+                        BulkOrder_Typehash_Height_Nineteen,
+                        BulkOrder_Typehash_Height_Twenty
+                    )
+
+                    // Exit the function once typehash has been located.
+                    leave
+                }
+
+                // Handle tree height twenty-one and twenty-two.
+                if lt(treeHeight, 23) {
+                    // Utilize branchless logic to determine typehash.
+                    typeHash := ternary(
+                        eq(treeHeight, 21),
+                        BulkOrder_Typehash_Height_TwentyOne,
+                        BulkOrder_Typehash_Height_TwentyTwo
+                    )
+
+                    // Exit the function once typehash has been located.
+                    leave
+                }
+
+                // Handle height twenty-three & twenty-four w/ branchless logic.
+                typeHash := ternary(
+                    eq(treeHeight, 23),
+                    BulkOrder_Typehash_Height_TwentyThree,
+                    BulkOrder_Typehash_Height_TwentyFour
+                )
+
+                // Exit the function once typehash has been located.
+                leave
+            }
+
+            // Implement ternary conditional using branchless logic.
+            function ternary(cond, ifTrue, ifFalse) -> c {
+                c := xor(ifFalse, mul(cond, xor(ifFalse, ifTrue)))
+            }
+
+            // Look up the typehash using the supplied tree height.
+            _typeHash := lookupTypeHash(_treeHeight)
+        }
+    }
+
+    /**
+     * @dev Internal view function to ensure that the supplied consideration
+     *      array length on a given set of order parameters is not less than the
+     *      original consideration array length for that order and to retrieve
+     *      the current counter for a given order's offerer and zone and use it
+     *      to derive the order hash.
+     *
+     * @param orderParameters The parameters of the order to hash.
+     *
+     * @return The hash.
+     */
+    function _assertConsiderationLengthAndGetOrderHash(
+        OrderParameters memory orderParameters,
+        SeaportInterface seaport
+    ) internal view returns (bytes32) {
+        // Ensure supplied consideration array length is not less than original.
+        _assertConsiderationLengthIsNotLessThanOriginalConsiderationLength(
+            orderParameters.consideration.length,
+            orderParameters.totalOriginalConsiderationItems
+        );
+
+        // Derive and return order hash using current counter for the offerer.
+        return
+            _deriveOrderHash(
+                orderParameters,
+                _getCounter(seaport, orderParameters.offerer),
+                seaport
+            );
+    }
+
+    function _getCounter(
+        SeaportInterface seaport,
+        address offerer
+    ) internal view returns (uint256) {
+        return seaport.getCounter(offerer);
+    }
+
+    function _deriveOrderHash(
+        OrderParameters memory orderParameters,
+        uint256 counter,
+        SeaportInterface seaport
+    ) internal view returns (bytes32 orderHash) {
+        return
+            seaport.getOrderHash(_toOrderComponents(orderParameters, counter));
+    }
+
+    /**
+     * @dev Converts an OrderParameters struct into an OrderComponents struct.
+     *
+     * @param parameters the OrderParameters struct to convert
+     * @param counter    the counter to use for the OrderComponents struct
+     *
+     * @return components the OrderComponents struct
+     */
+    function _toOrderComponents(
+        OrderParameters memory parameters,
+        uint256 counter
+    ) internal pure returns (OrderComponents memory components) {
+        components.offerer = parameters.offerer;
+        components.zone = parameters.zone;
+        components.offer = parameters.offer;
+        components.consideration = parameters.consideration;
+        components.orderType = parameters.orderType;
+        components.startTime = parameters.startTime;
+        components.endTime = parameters.endTime;
+        components.zoneHash = parameters.zoneHash;
+        components.salt = parameters.salt;
+        components.conduitKey = parameters.conduitKey;
+        components.counter = counter;
+    }
+
+    /**
+     * @dev Internal pure function to efficiently derive an digest to sign for
+     *      an order in accordance with EIP-712.
+     *
+     * @param domainSeparator The domain separator.
+     * @param orderHash       The order hash.
+     *
+     * @return value The hash.
+     */
+    function _deriveEIP712Digest(
+        bytes32 domainSeparator,
+        bytes32 orderHash
+    ) internal pure returns (bytes32 value) {
+        // Leverage scratch space to perform an efficient hash.
+        assembly {
+            // Place the EIP-712 prefix at the start of scratch space.
+            mstore(0, EIP_712_PREFIX)
+
+            // Place the domain separator in the next region of scratch space.
+            mstore(EIP712_DomainSeparator_offset, domainSeparator)
+
+            // Place the order hash in scratch space, spilling into the first
+            // two bytes of the free memory pointer — this should never be set
+            // as memory cannot be expanded to that size, and will be zeroed out
+            // after the hash is performed.
+            mstore(EIP712_OrderHash_offset, orderHash)
+
+            // Hash the relevant region (65 bytes).
+            value := keccak256(0, EIP712_DigestPayload_size)
+
+            // Clear out the dirtied bits in the memory pointer.
+            mstore(EIP712_OrderHash_offset, 0)
+        }
+    }
+
+    /**
+     * @dev Internal pure function to ensure that the supplied consideration
+     *      array length for an order to be fulfilled is not less than the
+     *      original consideration array length for that order.
+     *
+     * @param suppliedConsiderationItemTotal The number of consideration items
+     *                                       supplied when fulfilling the order.
+     * @param originalConsiderationItemTotal The number of consideration items
+     *                                       supplied on initial order creation.
+     */
+    function _assertConsiderationLengthIsNotLessThanOriginalConsiderationLength(
+        uint256 suppliedConsiderationItemTotal,
+        uint256 originalConsiderationItemTotal
+    ) internal pure {
+        // Ensure supplied consideration array length is not less than original.
+        if (suppliedConsiderationItemTotal < originalConsiderationItemTotal) {
+            _revertMissingOriginalConsiderationItems();
+        }
+    }
+}
diff --git a/contracts/seaport/helpers/order-validator/lib/SafeStaticCall.sol b/contracts/seaport/helpers/order-validator/lib/SafeStaticCall.sol
new file mode 100644
index 00000000..1963e01a
--- /dev/null
+++ b/contracts/seaport/helpers/order-validator/lib/SafeStaticCall.sol
@@ -0,0 +1,77 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.10;
+
+library SafeStaticCall {
+    function safeStaticCallBool(
+        address target,
+        bytes memory callData,
+        bool expectedReturn
+    ) internal view returns (bool) {
+        (bool success, bytes memory res) = target.staticcall(callData);
+        if (!success) return false;
+        if (res.length != 32) return false;
+
+        if (
+            bytes32(res) &
+                0x0000000000000000000000000000000000000000000000000000000000000001 !=
+            bytes32(res)
+        ) {
+            return false;
+        }
+
+        return expectedReturn ? res[31] == 0x01 : res[31] == 0;
+    }
+
+    function safeStaticCallAddress(
+        address target,
+        bytes memory callData,
+        address expectedReturn
+    ) internal view returns (bool) {
+        (bool success, bytes memory res) = target.staticcall(callData);
+        if (!success) return false;
+        if (res.length != 32) return false;
+
+        if (
+            bytes32(res) &
+                0x000000000000000000000000FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF !=
+            bytes32(res)
+        ) {
+            // Ensure only 20 bytes used
+            return false;
+        }
+
+        return abi.decode(res, (address)) == expectedReturn;
+    }
+
+    function safeStaticCallUint256(
+        address target,
+        bytes memory callData,
+        uint256 minExpectedReturn
+    ) internal view returns (bool) {
+        (bool success, bytes memory res) = target.staticcall(callData);
+        if (!success) return false;
+        if (res.length != 32) return false;
+
+        return abi.decode(res, (uint256)) >= minExpectedReturn;
+    }
+
+    function safeStaticCallBytes4(
+        address target,
+        bytes memory callData,
+        bytes4 expectedReturn
+    ) internal view returns (bool) {
+        (bool success, bytes memory res) = target.staticcall(callData);
+        if (!success) return false;
+        if (res.length != 32) return false;
+        if (
+            bytes32(res) &
+                0xFFFFFFFF00000000000000000000000000000000000000000000000000000000 !=
+            bytes32(res)
+        ) {
+            // Ensure only 4 bytes used
+            return false;
+        }
+
+        return abi.decode(res, (bytes4)) == expectedReturn;
+    }
+}
diff --git a/contracts/seaport/helpers/order-validator/lib/SeaportValidatorHelper.sol b/contracts/seaport/helpers/order-validator/lib/SeaportValidatorHelper.sol
new file mode 100644
index 00000000..b3eef36e
--- /dev/null
+++ b/contracts/seaport/helpers/order-validator/lib/SeaportValidatorHelper.sol
@@ -0,0 +1,958 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { ItemType } from "seaport-types/src/lib/ConsiderationEnums.sol";
+import {
+    Order,
+    OrderParameters,
+    BasicOrderParameters,
+    OfferItem,
+    ConsiderationItem,
+    Schema,
+    ZoneParameters
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+import { ConsiderationTypeHashes } from "./ConsiderationTypeHashes.sol";
+import {
+    ConsiderationInterface
+} from "seaport-types/src/interfaces/ConsiderationInterface.sol";
+import {
+    ConduitControllerInterface
+} from "seaport-types/src/interfaces/ConduitControllerInterface.sol";
+import {
+    ContractOffererInterface
+} from "seaport-types/src/interfaces/ContractOffererInterface.sol";
+import { ZoneInterface } from "seaport-types/src/interfaces/ZoneInterface.sol";
+import { GettersAndDerivers } from "seaport-core/src/lib/GettersAndDerivers.sol";
+import {
+    SeaportValidatorInterface
+} from "../lib/SeaportValidatorInterface.sol";
+import { ZoneInterface } from "seaport-types/src/interfaces/ZoneInterface.sol";
+import {
+    ERC20Interface,
+    ERC721Interface,
+    ERC1155Interface
+} from "seaport-types/src/interfaces/AbridgedTokenInterfaces.sol";
+import { IERC165 } from "@openzeppelin/contracts/seaport/interfaces/IERC165.sol";
+import { IERC2981 } from "@openzeppelin/contracts/interfaces/IERC2981.sol";
+import {
+    ErrorsAndWarnings,
+    ErrorsAndWarningsLib
+} from "../lib/ErrorsAndWarnings.sol";
+import { SafeStaticCall } from "../lib/SafeStaticCall.sol";
+import { Murky } from "../lib/Murky.sol";
+import {
+    IssueParser,
+    ValidationConfiguration,
+    TimeIssue,
+    StatusIssue,
+    OfferIssue,
+    ContractOffererIssue,
+    ConsiderationIssue,
+    PrimaryFeeIssue,
+    ERC721Issue,
+    ERC1155Issue,
+    ERC20Issue,
+    NativeIssue,
+    ZoneIssue,
+    ConduitIssue,
+    CreatorFeeIssue,
+    SignatureIssue,
+    GenericIssue,
+    ConsiderationItemConfiguration
+} from "./SeaportValidatorTypes.sol";
+import { Verifiers } from "seaport-core/src/lib/Verifiers.sol";
+
+/**
+ * @title SeaportValidator
+ * @author OpenSea Protocol Team
+ * @notice SeaportValidatorHelper assists in advanced validation to seaport orders.
+ */
+contract SeaportValidatorHelper is Murky {
+    using ErrorsAndWarningsLib for ErrorsAndWarnings;
+    using SafeStaticCall for address;
+    using IssueParser for *;
+
+    /// @notice Ethereum creator fee engine address
+    CreatorFeeEngineInterface public immutable creatorFeeEngine;
+
+    bytes4 public constant ERC20_INTERFACE_ID = 0x36372b07;
+
+    bytes4 public constant ERC721_INTERFACE_ID = 0x80ac58cd;
+
+    bytes4 public constant ERC1155_INTERFACE_ID = 0xd9b67a26;
+
+    constructor() {
+        address creatorFeeEngineAddress;
+        if (block.chainid == 1 || block.chainid == 31337) {
+            creatorFeeEngineAddress = 0x0385603ab55642cb4Dd5De3aE9e306809991804f;
+        } else if (block.chainid == 3) {
+            // Ropsten
+            creatorFeeEngineAddress = 0xFf5A6F7f36764aAD301B7C9E85A5277614Df5E26;
+        } else if (block.chainid == 4) {
+            // Rinkeby
+            creatorFeeEngineAddress = 0x8d17687ea9a6bb6efA24ec11DcFab01661b2ddcd;
+        } else if (block.chainid == 5) {
+            // Goerli
+            creatorFeeEngineAddress = 0xe7c9Cb6D966f76f3B5142167088927Bf34966a1f;
+        } else if (block.chainid == 42) {
+            // Kovan
+            creatorFeeEngineAddress = 0x54D88324cBedfFe1e62c9A59eBb310A11C295198;
+        } else if (block.chainid == 137) {
+            // Polygon
+            creatorFeeEngineAddress = 0x28EdFcF0Be7E86b07493466e7631a213bDe8eEF2;
+        } else if (block.chainid == 80001) {
+            // Mumbai
+            creatorFeeEngineAddress = 0x0a01E11887f727D1b1Cd81251eeEE9BEE4262D07;
+        } else {
+            // No creator fee engine for this chain
+            creatorFeeEngineAddress = address(0);
+        }
+
+        creatorFeeEngine = CreatorFeeEngineInterface(creatorFeeEngineAddress);
+    }
+
+    /**
+     * @notice Strict validation operates under tight assumptions. It validates primary
+     *    fee, creator fee, private sale consideration, and overall order format.
+     * @dev Only checks first fee recipient provided by CreatorFeeEngine.
+     *    Order of consideration items must be as follows:
+     *    1. Primary consideration
+     *    2. Primary fee
+     *    3. Creator fee
+     *    4. Private sale consideration
+     * @param orderParameters The parameters for the order to validate.
+     * @param primaryFeeRecipient The primary fee recipient. Set to null address for no primary fee.
+     * @param primaryFeeBips The primary fee in BIPs.
+     * @param checkCreatorFee Should check for creator fee. If true, creator fee must be present as
+     *    according to creator fee engine. If false, must not have creator fee.
+     * @return errorsAndWarnings The errors and warnings.
+     */
+    function validateStrictLogic(
+        OrderParameters memory orderParameters,
+        address primaryFeeRecipient,
+        uint256 primaryFeeBips,
+        bool checkCreatorFee
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // Check that order matches the required format (listing or offer)
+        {
+            bool canCheckFee = true;
+            // Single offer item and at least one consideration
+            if (
+                orderParameters.offer.length != 1 ||
+                orderParameters.consideration.length == 0
+            ) {
+                // Not listing or offer, can't check fees
+                canCheckFee = false;
+            } else if (
+                // Can't have both items be fungible
+                isPaymentToken(orderParameters.offer[0].itemType) &&
+                isPaymentToken(orderParameters.consideration[0].itemType)
+            ) {
+                // Not listing or offer, can't check fees
+                canCheckFee = false;
+            } else if (
+                // Can't have both items be non-fungible
+                !isPaymentToken(orderParameters.offer[0].itemType) &&
+                !isPaymentToken(orderParameters.consideration[0].itemType)
+            ) {
+                // Not listing or offer, can't check fees
+                canCheckFee = false;
+            }
+            if (!canCheckFee) {
+                // Does not match required format
+                errorsAndWarnings.addError(
+                    GenericIssue.InvalidOrderFormat.parseInt()
+                );
+                return errorsAndWarnings;
+            }
+        }
+
+        // Validate secondary consideration items (fees)
+        (
+            uint256 tertiaryConsiderationIndex,
+            ErrorsAndWarnings memory errorsAndWarningsLocal
+        ) = _validateSecondaryConsiderationItems(
+                orderParameters,
+                ConsiderationItemConfiguration({
+                    primaryFeeRecipient: primaryFeeRecipient,
+                    primaryFeeBips: primaryFeeBips,
+                    checkCreatorFee: checkCreatorFee
+                })
+            );
+
+        errorsAndWarnings.concat(errorsAndWarningsLocal);
+
+        // Validate tertiary consideration items if not 0 (0 indicates error).
+        // Only if no prior errors
+        if (tertiaryConsiderationIndex != 0) {
+            errorsAndWarnings.concat(
+                _validateTertiaryConsiderationItems(
+                    orderParameters,
+                    tertiaryConsiderationIndex
+                )
+            );
+        }
+    }
+
+    /**
+     * @notice Validate all consideration items for an order
+     * @param orderParameters The parameters for the order to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateConsiderationItems(
+        OrderParameters memory orderParameters,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // You must have a consideration item
+        if (orderParameters.consideration.length == 0) {
+            errorsAndWarnings.addWarning(
+                ConsiderationIssue.ZeroItems.parseInt()
+            );
+            return errorsAndWarnings;
+        }
+
+        // Declare a boolean to check if offerer is receiving at least
+        // one consideration item
+        bool offererReceivingAtLeastOneItem = false;
+
+        // Iterate over each consideration item
+        for (uint256 i = 0; i < orderParameters.consideration.length; i++) {
+            // Validate consideration item
+            errorsAndWarnings.concat(
+                validateConsiderationItem(orderParameters, i, seaportAddress)
+            );
+
+            ConsiderationItem memory considerationItem1 = orderParameters
+                .consideration[i];
+
+            // Check if the offerer is the recipient
+            if (!offererReceivingAtLeastOneItem) {
+                if (considerationItem1.recipient == orderParameters.offerer) {
+                    offererReceivingAtLeastOneItem = true;
+                }
+            }
+
+            // Check for duplicate consideration items
+            for (
+                uint256 j = i + 1;
+                j < orderParameters.consideration.length;
+                j++
+            ) {
+                // Iterate over each remaining consideration item
+                // (previous items already check with this item)
+                ConsiderationItem memory considerationItem2 = orderParameters
+                    .consideration[j];
+
+                // Check if itemType, token, id, and recipient are the same
+                if (
+                    considerationItem2.itemType ==
+                    considerationItem1.itemType &&
+                    considerationItem2.token == considerationItem1.token &&
+                    considerationItem2.identifierOrCriteria ==
+                    considerationItem1.identifierOrCriteria &&
+                    considerationItem2.recipient == considerationItem1.recipient
+                ) {
+                    errorsAndWarnings.addWarning(
+                        // Duplicate consideration item, warning
+                        ConsiderationIssue.DuplicateItem.parseInt()
+                    );
+                }
+            }
+        }
+
+        if (!offererReceivingAtLeastOneItem) {
+            // Offerer is not receiving at least one consideration item
+            errorsAndWarnings.addWarning(
+                ConsiderationIssue.OffererNotReceivingAtLeastOneItem.parseInt()
+            );
+        }
+    }
+
+    /**
+     * @notice Validate a consideration item
+     * @param orderParameters The parameters for the order to validate
+     * @param considerationItemIndex The index of the consideration item to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateConsiderationItem(
+        OrderParameters memory orderParameters,
+        uint256 considerationItemIndex,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        // Validate the consideration item at considerationItemIndex
+        errorsAndWarnings.concat(
+            validateConsiderationItemParameters(
+                orderParameters,
+                considerationItemIndex,
+                seaportAddress
+            )
+        );
+    }
+
+    /**
+     * @notice Validates the parameters of a consideration item including contract validation
+     * @param orderParameters The parameters for the order to validate
+     * @param considerationItemIndex The index of the consideration item to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateConsiderationItemParameters(
+        OrderParameters memory orderParameters,
+        uint256 considerationItemIndex,
+        address seaportAddress
+    ) public view returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        ConsiderationItem memory considerationItem = orderParameters
+            .consideration[considerationItemIndex];
+
+        // Check if startAmount and endAmount are zero
+        if (
+            considerationItem.startAmount == 0 &&
+            considerationItem.endAmount == 0
+        ) {
+            errorsAndWarnings.addError(
+                ConsiderationIssue.AmountZero.parseInt()
+            );
+            return errorsAndWarnings;
+        }
+
+        // Check if the recipient is the null address
+        if (considerationItem.recipient == address(0)) {
+            errorsAndWarnings.addError(
+                ConsiderationIssue.NullRecipient.parseInt()
+            );
+        }
+
+        if (
+            considerationItem.startAmount != considerationItem.endAmount &&
+            orderParameters.endTime > orderParameters.startTime
+        ) {
+            // Check that amount velocity is not too high.
+            // Assign larger and smaller amount values
+            (uint256 maxAmount, uint256 minAmount) = considerationItem
+                .startAmount > considerationItem.endAmount
+                ? (considerationItem.startAmount, considerationItem.endAmount)
+                : (considerationItem.endAmount, considerationItem.startAmount);
+
+            uint256 amountDelta = maxAmount - minAmount;
+            // delta of time that order exists for
+            uint256 timeDelta = orderParameters.endTime -
+                orderParameters.startTime;
+
+            // Velocity scaled by 1e10 for precision
+            uint256 velocity = (amountDelta * 1e10) / timeDelta;
+            // gives velocity percentage in hundredth of a basis points per second in terms of larger value
+            uint256 velocityPercentage = velocity / (maxAmount * 1e4);
+
+            // 278 * 60 * 30 ~= 500,000
+            if (velocityPercentage > 278) {
+                // Over 50% change per 30 min
+                errorsAndWarnings.addError(
+                    ConsiderationIssue.AmountVelocityHigh.parseInt()
+                );
+            }
+            // 28 * 60 * 30 ~= 50,000
+            else if (velocityPercentage > 28) {
+                // Over 5% change per 30 min
+                errorsAndWarnings.addWarning(
+                    ConsiderationIssue.AmountVelocityHigh.parseInt()
+                );
+            }
+
+            // Check for large amount steps
+            if (minAmount <= 1e15) {
+                errorsAndWarnings.addWarning(
+                    ConsiderationIssue.AmountStepLarge.parseInt()
+                );
+            }
+        }
+
+        if (considerationItem.itemType == ItemType.ERC721) {
+            // ERC721 type requires amounts to be 1
+            if (
+                considerationItem.startAmount != 1 ||
+                considerationItem.endAmount != 1
+            ) {
+                errorsAndWarnings.addError(ERC721Issue.AmountNotOne.parseInt());
+            }
+
+            // Check EIP165 interface
+            if (!checkInterface(considerationItem.token, ERC721_INTERFACE_ID)) {
+                errorsAndWarnings.addError(ERC721Issue.InvalidToken.parseInt());
+                return errorsAndWarnings;
+            }
+
+            // Check that token exists
+            if (
+                !considerationItem.token.safeStaticCallUint256(
+                    abi.encodeWithSelector(
+                        ERC721Interface.ownerOf.selector,
+                        considerationItem.identifierOrCriteria
+                    ),
+                    1
+                )
+            ) {
+                // Token does not exist
+                errorsAndWarnings.addError(
+                    ERC721Issue.IdentifierDNE.parseInt()
+                );
+            }
+        } else if (
+            considerationItem.itemType == ItemType.ERC721_WITH_CRITERIA
+        ) {
+            // Check EIP165 interface
+            if (!checkInterface(considerationItem.token, ERC721_INTERFACE_ID)) {
+                // Does not implement required interface
+                errorsAndWarnings.addError(ERC721Issue.InvalidToken.parseInt());
+            }
+        } else if (
+            considerationItem.itemType == ItemType.ERC1155 ||
+            considerationItem.itemType == ItemType.ERC1155_WITH_CRITERIA
+        ) {
+            // Check EIP165 interface
+            if (
+                !checkInterface(considerationItem.token, ERC1155_INTERFACE_ID)
+            ) {
+                // Does not implement required interface
+                errorsAndWarnings.addError(
+                    ERC1155Issue.InvalidToken.parseInt()
+                );
+            }
+        } else if (considerationItem.itemType == ItemType.ERC20) {
+            // ERC20 must have `identifierOrCriteria` be zero
+            if (considerationItem.identifierOrCriteria != 0) {
+                errorsAndWarnings.addError(
+                    ERC20Issue.IdentifierNonZero.parseInt()
+                );
+            }
+
+            // Check that it is an ERC20 token. ERC20 will return a uint256
+            if (
+                !considerationItem.token.safeStaticCallUint256(
+                    abi.encodeWithSelector(
+                        ERC20Interface.allowance.selector,
+                        seaportAddress,
+                        seaportAddress
+                    ),
+                    0
+                )
+            ) {
+                // Not an ERC20 token
+                errorsAndWarnings.addError(ERC20Issue.InvalidToken.parseInt());
+            }
+        } else {
+            // Must be native
+            // NATIVE must have `token` be zero address
+            if (considerationItem.token != address(0)) {
+                errorsAndWarnings.addError(NativeIssue.TokenAddress.parseInt());
+            }
+            // NATIVE must have `identifierOrCriteria` be zero
+            if (considerationItem.identifierOrCriteria != 0) {
+                errorsAndWarnings.addError(
+                    NativeIssue.IdentifierNonZero.parseInt()
+                );
+            }
+        }
+    }
+
+    function _validateSecondaryConsiderationItems(
+        OrderParameters memory orderParameters,
+        ConsiderationItemConfiguration memory config
+    )
+        internal
+        view
+        returns (
+            uint256 /* tertiaryConsiderationIndex */,
+            ErrorsAndWarnings memory /* errorsAndWarnings */
+        )
+    {
+        ErrorsAndWarnings memory errorsAndWarnings = ErrorsAndWarnings(
+            new uint16[](0),
+            new uint16[](0)
+        );
+
+        // Consideration item to hold expected creator fee info
+        ConsiderationItem memory creatorFeeConsideration;
+
+        bool primaryFeePresent;
+
+        {
+            // non-fungible item address
+            address itemAddress;
+            // non-fungible item identifier
+            uint256 itemIdentifier;
+            // fungible item start amount
+            uint256 transactionAmountStart;
+            // fungible item end amount
+            uint256 transactionAmountEnd;
+
+            if (isPaymentToken(orderParameters.offer[0].itemType)) {
+                // Offer is an offer. Offer item is fungible and used for fees
+                creatorFeeConsideration.itemType = orderParameters
+                    .offer[0]
+                    .itemType;
+                creatorFeeConsideration.token = orderParameters.offer[0].token;
+                transactionAmountStart = orderParameters.offer[0].startAmount;
+                transactionAmountEnd = orderParameters.offer[0].endAmount;
+
+                // Set non-fungible information for calculating creator fee
+                itemAddress = orderParameters.consideration[0].token;
+                itemIdentifier = orderParameters
+                    .consideration[0]
+                    .identifierOrCriteria;
+            } else {
+                // Offer is an offer. Consideration item is fungible and used for fees
+                creatorFeeConsideration.itemType = orderParameters
+                    .consideration[0]
+                    .itemType;
+                creatorFeeConsideration.token = orderParameters
+                    .consideration[0]
+                    .token;
+                transactionAmountStart = orderParameters
+                    .consideration[0]
+                    .startAmount;
+                transactionAmountEnd = orderParameters
+                    .consideration[0]
+                    .endAmount;
+
+                // Set non-fungible information for calculating creator fees
+                itemAddress = orderParameters.offer[0].token;
+                itemIdentifier = orderParameters.offer[0].identifierOrCriteria;
+            }
+
+            // Store flag if primary fee is present
+            primaryFeePresent = false;
+            {
+                // Calculate primary fee start and end amounts
+                uint256 primaryFeeStartAmount = (transactionAmountStart *
+                    config.primaryFeeBips) / 10000;
+                uint256 primaryFeeEndAmount = (transactionAmountEnd *
+                    config.primaryFeeBips) / 10000;
+
+                // Check if primary fee check is desired. Skip if calculated amount is zero.
+                if (
+                    config.primaryFeeRecipient != address(0) &&
+                    (primaryFeeStartAmount > 0 || primaryFeeEndAmount > 0)
+                ) {
+                    // Ensure primary fee is present
+                    if (orderParameters.consideration.length < 2) {
+                        errorsAndWarnings.addError(
+                            PrimaryFeeIssue.Missing.parseInt()
+                        );
+                        return (0, errorsAndWarnings);
+                    }
+                    primaryFeePresent = true;
+
+                    ConsiderationItem memory primaryFeeItem = orderParameters
+                        .consideration[1];
+
+                    // Check item type
+                    if (
+                        primaryFeeItem.itemType !=
+                        creatorFeeConsideration.itemType
+                    ) {
+                        errorsAndWarnings.addError(
+                            PrimaryFeeIssue.ItemType.parseInt()
+                        );
+                        return (0, errorsAndWarnings);
+                    }
+                    // Check token
+                    if (primaryFeeItem.token != creatorFeeConsideration.token) {
+                        errorsAndWarnings.addError(
+                            PrimaryFeeIssue.Token.parseInt()
+                        );
+                    }
+                    // Check start amount
+                    if (primaryFeeItem.startAmount < primaryFeeStartAmount) {
+                        errorsAndWarnings.addError(
+                            PrimaryFeeIssue.StartAmount.parseInt()
+                        );
+                    }
+                    // Check end amount
+                    if (primaryFeeItem.endAmount < primaryFeeEndAmount) {
+                        errorsAndWarnings.addError(
+                            PrimaryFeeIssue.EndAmount.parseInt()
+                        );
+                    }
+                    // Check recipient
+                    if (
+                        primaryFeeItem.recipient != config.primaryFeeRecipient
+                    ) {
+                        errorsAndWarnings.addError(
+                            PrimaryFeeIssue.Recipient.parseInt()
+                        );
+                    }
+                }
+            }
+
+            // Check creator fee
+            (
+                creatorFeeConsideration.recipient,
+                creatorFeeConsideration.startAmount,
+                creatorFeeConsideration.endAmount
+            ) = getCreatorFeeInfo(
+                itemAddress,
+                itemIdentifier,
+                transactionAmountStart,
+                transactionAmountEnd
+            );
+        }
+
+        // Flag indicating if creator fee is present in considerations
+        bool creatorFeePresent = false;
+
+        // Determine if should check for creator fee
+        if (
+            creatorFeeConsideration.recipient != address(0) &&
+            config.checkCreatorFee &&
+            (creatorFeeConsideration.startAmount > 0 ||
+                creatorFeeConsideration.endAmount > 0)
+        ) {
+            // Calculate index of creator fee consideration item
+            uint16 creatorFeeConsiderationIndex = primaryFeePresent ? 2 : 1; // 2 if primary fee, ow 1
+
+            // Check that creator fee consideration item exists
+            if (
+                orderParameters.consideration.length - 1 <
+                creatorFeeConsiderationIndex
+            ) {
+                errorsAndWarnings.addError(CreatorFeeIssue.Missing.parseInt());
+                return (0, errorsAndWarnings);
+            }
+
+            ConsiderationItem memory creatorFeeItem = orderParameters
+                .consideration[creatorFeeConsiderationIndex];
+
+            creatorFeePresent = true;
+
+            // Check type
+            if (creatorFeeItem.itemType != creatorFeeConsideration.itemType) {
+                errorsAndWarnings.addError(CreatorFeeIssue.ItemType.parseInt());
+                return (0, errorsAndWarnings);
+            }
+            // Check token
+            if (creatorFeeItem.token != creatorFeeConsideration.token) {
+                errorsAndWarnings.addError(CreatorFeeIssue.Token.parseInt());
+            }
+            // Check start amount
+            if (
+                creatorFeeItem.startAmount < creatorFeeConsideration.startAmount
+            ) {
+                errorsAndWarnings.addError(
+                    CreatorFeeIssue.StartAmount.parseInt()
+                );
+            }
+            // Check end amount
+            if (creatorFeeItem.endAmount < creatorFeeConsideration.endAmount) {
+                errorsAndWarnings.addError(
+                    CreatorFeeIssue.EndAmount.parseInt()
+                );
+            }
+            // Check recipient
+            if (creatorFeeItem.recipient != creatorFeeConsideration.recipient) {
+                errorsAndWarnings.addError(
+                    CreatorFeeIssue.Recipient.parseInt()
+                );
+            }
+        }
+
+        // Calculate index of first tertiary consideration item
+        uint256 tertiaryConsiderationIndex = 1 +
+            (primaryFeePresent ? 1 : 0) +
+            (creatorFeePresent ? 1 : 0);
+
+        return (tertiaryConsiderationIndex, errorsAndWarnings);
+    }
+
+    /**
+     * @notice Internal function for validating all consideration items after the fee items.
+     *    Only additional acceptable consideration is private sale.
+     */
+    function _validateTertiaryConsiderationItems(
+        OrderParameters memory orderParameters,
+        uint256 considerationItemIndex
+    ) internal pure returns (ErrorsAndWarnings memory errorsAndWarnings) {
+        errorsAndWarnings = ErrorsAndWarnings(new uint16[](0), new uint16[](0));
+
+        if (orderParameters.consideration.length <= considerationItemIndex) {
+            // No more consideration items
+            return errorsAndWarnings;
+        }
+
+        ConsiderationItem memory privateSaleConsideration = orderParameters
+            .consideration[considerationItemIndex];
+
+        // Check if offer is payment token. Private sale not possible if so.
+        if (isPaymentToken(orderParameters.offer[0].itemType)) {
+            errorsAndWarnings.addError(
+                ConsiderationIssue.ExtraItems.parseInt()
+            );
+            return errorsAndWarnings;
+        }
+
+        // Check if private sale to self
+        if (privateSaleConsideration.recipient == orderParameters.offerer) {
+            errorsAndWarnings.addError(
+                ConsiderationIssue.PrivateSaleToSelf.parseInt()
+            );
+            return errorsAndWarnings;
+        }
+
+        // Ensure that private sale parameters match offer item.
+        if (
+            privateSaleConsideration.itemType !=
+            orderParameters.offer[0].itemType ||
+            privateSaleConsideration.token != orderParameters.offer[0].token ||
+            orderParameters.offer[0].startAmount !=
+            privateSaleConsideration.startAmount ||
+            orderParameters.offer[0].endAmount !=
+            privateSaleConsideration.endAmount ||
+            orderParameters.offer[0].identifierOrCriteria !=
+            privateSaleConsideration.identifierOrCriteria
+        ) {
+            // Invalid private sale, say extra consideration item
+            errorsAndWarnings.addError(
+                ConsiderationIssue.ExtraItems.parseInt()
+            );
+            return errorsAndWarnings;
+        }
+
+        errorsAndWarnings.addWarning(ConsiderationIssue.PrivateSale.parseInt());
+
+        // Should not be any additional consideration items
+        if (orderParameters.consideration.length - 1 > considerationItemIndex) {
+            // Extra consideration items
+            errorsAndWarnings.addError(
+                ConsiderationIssue.ExtraItems.parseInt()
+            );
+            return errorsAndWarnings;
+        }
+    }
+
+    /**
+     * @notice Fetches the on chain creator fees.
+     * @dev Uses the creatorFeeEngine when available, otherwise fallback to `IERC2981`.
+     * @param token The token address
+     * @param tokenId The token identifier
+     * @param transactionAmountStart The transaction start amount
+     * @param transactionAmountEnd The transaction end amount
+     * @return recipient creator fee recipient
+     * @return creatorFeeAmountStart creator fee start amount
+     * @return creatorFeeAmountEnd creator fee end amount
+     */
+    function getCreatorFeeInfo(
+        address token,
+        uint256 tokenId,
+        uint256 transactionAmountStart,
+        uint256 transactionAmountEnd
+    )
+        public
+        view
+        returns (
+            address payable recipient,
+            uint256 creatorFeeAmountStart,
+            uint256 creatorFeeAmountEnd
+        )
+    {
+        // Check if creator fee engine is on this chain
+        if (address(creatorFeeEngine) != address(0)) {
+            // Creator fee engine may revert if no creator fees are present.
+            try
+                creatorFeeEngine.getRoyaltyView(
+                    token,
+                    tokenId,
+                    transactionAmountStart
+                )
+            returns (
+                address payable[] memory creatorFeeRecipients,
+                uint256[] memory creatorFeeAmountsStart
+            ) {
+                if (creatorFeeRecipients.length != 0) {
+                    // Use first recipient and amount
+                    recipient = creatorFeeRecipients[0];
+                    creatorFeeAmountStart = creatorFeeAmountsStart[0];
+                }
+            } catch {
+                // Creator fee not found
+            }
+
+            // If fees found for start amount, check end amount
+            if (recipient != address(0)) {
+                // Creator fee engine may revert if no creator fees are present.
+                try
+                    creatorFeeEngine.getRoyaltyView(
+                        token,
+                        tokenId,
+                        transactionAmountEnd
+                    )
+                returns (
+                    address payable[] memory,
+                    uint256[] memory creatorFeeAmountsEnd
+                ) {
+                    creatorFeeAmountEnd = creatorFeeAmountsEnd[0];
+                } catch {}
+            }
+        } else {
+            // Fallback to ERC2981
+            {
+                // Static call to token using ERC2981
+                (bool success, bytes memory res) = token.staticcall(
+                    abi.encodeWithSelector(
+                        IERC2981.royaltyInfo.selector,
+                        tokenId,
+                        transactionAmountStart
+                    )
+                );
+                // Check if call succeeded
+                if (success) {
+                    // Ensure 64 bytes returned
+                    if (res.length == 64) {
+                        // Decode result and assign recipient and start amount
+                        (recipient, creatorFeeAmountStart) = abi.decode(
+                            res,
+                            (address, uint256)
+                        );
+                    }
+                }
+            }
+
+            // Only check end amount if start amount found
+            if (recipient != address(0)) {
+                // Static call to token using ERC2981
+                (bool success, bytes memory res) = token.staticcall(
+                    abi.encodeWithSelector(
+                        IERC2981.royaltyInfo.selector,
+                        tokenId,
+                        transactionAmountEnd
+                    )
+                );
+                // Check if call succeeded
+                if (success) {
+                    // Ensure 64 bytes returned
+                    if (res.length == 64) {
+                        // Decode result and assign end amount
+                        (, creatorFeeAmountEnd) = abi.decode(
+                            res,
+                            (address, uint256)
+                        );
+                    }
+                }
+            }
+        }
+    }
+
+    /**
+     * @notice Safely check that a contract implements an interface
+     * @param token The token address to check
+     * @param interfaceHash The interface hash to check
+     */
+    function checkInterface(
+        address token,
+        bytes4 interfaceHash
+    ) public view returns (bool) {
+        return
+            token.safeStaticCallBool(
+                abi.encodeWithSelector(
+                    IERC165.supportsInterface.selector,
+                    interfaceHash
+                ),
+                true
+            );
+    }
+
+    /*//////////////////////////////////////////////////////////////
+                        Merkle Helpers
+    //////////////////////////////////////////////////////////////*/
+
+    /**
+     * @notice Sorts an array of token ids by the keccak256 hash of the id. Required ordering of ids
+     *    for other merkle operations.
+     * @param includedTokens An array of included token ids.
+     * @return sortedTokens The sorted `includedTokens` array.
+     */
+    function sortMerkleTokens(
+        uint256[] memory includedTokens
+    ) public pure returns (uint256[] memory sortedTokens) {
+        // Sort token ids by the keccak256 hash of the id
+        return _sortUint256ByHash(includedTokens);
+    }
+
+    /**
+     * @notice Creates a merkle root for includedTokens.
+     * @dev `includedTokens` must be sorting in strictly ascending order according to the keccak256 hash of the value.
+     * @return merkleRoot The merkle root
+     * @return errorsAndWarnings Errors and warnings from the operation
+     */
+    function getMerkleRoot(
+        uint256[] memory includedTokens
+    )
+        public
+        pure
+        returns (bytes32 merkleRoot, ErrorsAndWarnings memory errorsAndWarnings)
+    {
+        (merkleRoot, errorsAndWarnings) = _getRoot(includedTokens);
+    }
+
+    /**
+     * @notice Creates a merkle proof for the the targetIndex contained in includedTokens.
+     * @dev `targetIndex` is referring to the index of an element in `includedTokens`.
+     *    `includedTokens` must be sorting in ascending order according to the keccak256 hash of the value.
+     * @return merkleProof The merkle proof
+     * @return errorsAndWarnings Errors and warnings from the operation
+     */
+    function getMerkleProof(
+        uint256[] memory includedTokens,
+        uint256 targetIndex
+    )
+        public
+        pure
+        returns (
+            bytes32[] memory merkleProof,
+            ErrorsAndWarnings memory errorsAndWarnings
+        )
+    {
+        (merkleProof, errorsAndWarnings) = _getProof(
+            includedTokens,
+            targetIndex
+        );
+    }
+
+    /**
+     * @notice Verifies a merkle proof for the value to prove and given root and proof.
+     * @dev The `valueToProve` is hashed prior to executing the proof verification.
+     * @param merkleRoot The root of the merkle tree
+     * @param merkleProof The merkle proof
+     * @param valueToProve The value to prove
+     * @return whether proof is valid
+     */
+    function verifyMerkleProof(
+        bytes32 merkleRoot,
+        bytes32[] memory merkleProof,
+        uint256 valueToProve
+    ) public pure returns (bool) {
+        bytes32 hashedValue = keccak256(abi.encode(valueToProve));
+
+        return _verifyProof(merkleRoot, merkleProof, hashedValue);
+    }
+
+    function isPaymentToken(ItemType itemType) public pure returns (bool) {
+        return itemType == ItemType.NATIVE || itemType == ItemType.ERC20;
+    }
+}
+
+interface CreatorFeeEngineInterface {
+    function getRoyaltyView(
+        address tokenAddress,
+        uint256 tokenId,
+        uint256 value
+    )
+        external
+        view
+        returns (address payable[] memory recipients, uint256[] memory amounts);
+}
diff --git a/contracts/seaport/helpers/order-validator/lib/SeaportValidatorInterface.sol b/contracts/seaport/helpers/order-validator/lib/SeaportValidatorInterface.sol
new file mode 100644
index 00000000..3d0abd32
--- /dev/null
+++ b/contracts/seaport/helpers/order-validator/lib/SeaportValidatorInterface.sol
@@ -0,0 +1,289 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.10;
+
+import { ItemType } from "seaport-types/src/lib/ConsiderationEnums.sol";
+import {
+    Order,
+    OrderParameters,
+    ZoneParameters
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+import { ErrorsAndWarnings } from "./ErrorsAndWarnings.sol";
+import { ValidationConfiguration } from "./SeaportValidatorTypes.sol";
+
+/**
+ * @title SeaportValidator
+ * @notice SeaportValidator validates simple orders that adhere to a set of rules defined below:
+ *    - The order is either a listing or an offer order (one NFT to buy or one NFT to sell).
+ *    - The first consideration is the primary consideration.
+ *    - The order pays up to two fees in the fungible token currency. First fee is primary fee, second is creator fee.
+ *    - In private orders, the last consideration specifies a recipient for the offer item.
+ *    - Offer items must be owned and properly approved by the offerer.
+ *    - Consideration items must exist.
+ */
+interface SeaportValidatorInterface {
+    /**
+     * @notice Conduct a comprehensive validation of the given order.
+     * @param order The order to validate.
+     * @return errorsAndWarnings The errors and warnings found in the order.
+     */
+    function isValidOrder(
+        Order calldata order,
+        address seaportAddress
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Same as `isValidOrder` but allows for more configuration related to fee validation.
+     */
+    function isValidOrderWithConfiguration(
+        ValidationConfiguration memory validationConfiguration,
+        Order memory order
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Checks if a conduit key is valid.
+     * @param conduitKey The conduit key to check.
+     * @return errorsAndWarnings The errors and warnings
+     */
+    function isValidConduit(
+        bytes32 conduitKey,
+        address seaportAddress
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    // TODO: Need to add support for order with extra data
+    /**
+     * @notice Checks that the zone of an order implements the required interface
+     * @param orderParameters The parameters for the order to validate
+     * @return errorsAndWarnings An ErrorsAndWarnings structs with results
+     */
+    function isValidZone(
+        OrderParameters memory orderParameters
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    function validateSignature(
+        Order memory order,
+        address seaportAddress
+    ) external returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    function validateSignatureWithCounter(
+        Order memory order,
+        uint256 counter,
+        address seaportAddress
+    ) external returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Check the time validity of an order
+     * @param orderParameters The parameters for the order to validate
+     * @param shortOrderDuration The duration of which an order is considered short
+     * @param distantOrderExpiration Distant order expiration delta in seconds.
+     * @return errorsAndWarnings The Issues and warnings
+     */
+    function validateTime(
+        OrderParameters memory orderParameters,
+        uint256 shortOrderDuration,
+        uint256 distantOrderExpiration
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Validate the status of an order
+     * @param orderParameters The parameters for the order to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateOrderStatus(
+        OrderParameters memory orderParameters,
+        address seaportAddress
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Validate all offer items for an order
+     * @param orderParameters The parameters for the order to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateOfferItems(
+        OrderParameters memory orderParameters,
+        address seaportAddress
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Validate all consideration items for an order
+     * @param orderParameters The parameters for the order to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateConsiderationItems(
+        OrderParameters memory orderParameters,
+        address seaportAddress
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Strict validation operates under tight assumptions. It validates primary
+     *    fee, creator fee, private sale consideration, and overall order format.
+     * @dev Only checks first fee recipient provided by CreatorFeeRegistry.
+     *    Order of consideration items must be as follows:
+     *    1. Primary consideration
+     *    2. Primary fee
+     *    3. Creator Fee
+     *    4. Private sale consideration
+     * @param orderParameters The parameters for the order to validate.
+     * @param primaryFeeRecipient The primary fee recipient. Set to null address for no primary fee.
+     * @param primaryFeeBips The primary fee in BIPs.
+     * @param checkCreatorFee Should check for creator fee. If true, creator fee must be present as
+     *    according to creator fee engine. If false, must not have creator fee.
+     * @return errorsAndWarnings The errors and warnings.
+     */
+    function validateStrictLogic(
+        OrderParameters memory orderParameters,
+        address primaryFeeRecipient,
+        uint256 primaryFeeBips,
+        bool checkCreatorFee
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Validate a consideration item
+     * @param orderParameters The parameters for the order to validate
+     * @param considerationItemIndex The index of the consideration item to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateConsiderationItem(
+        OrderParameters memory orderParameters,
+        uint256 considerationItemIndex,
+        address seaportAddress
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Validates the parameters of a consideration item including contract validation
+     * @param orderParameters The parameters for the order to validate
+     * @param considerationItemIndex The index of the consideration item to validate
+     * @return errorsAndWarnings  The errors and warnings
+     */
+    function validateConsiderationItemParameters(
+        OrderParameters memory orderParameters,
+        uint256 considerationItemIndex,
+        address seaportAddress
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Validates an offer item
+     * @param orderParameters The parameters for the order to validate
+     * @param offerItemIndex The index of the offerItem in offer array to validate
+     * @return errorsAndWarnings An ErrorsAndWarnings structs with results
+     */
+    function validateOfferItem(
+        OrderParameters memory orderParameters,
+        uint256 offerItemIndex,
+        address seaportAddress
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Validates the OfferItem parameters. This includes token contract validation
+     * @dev OfferItems with criteria are currently not allowed
+     * @param orderParameters The parameters for the order to validate
+     * @param offerItemIndex The index of the offerItem in offer array to validate
+     * @return errorsAndWarnings An ErrorsAndWarnings structs with results
+     */
+    function validateOfferItemParameters(
+        OrderParameters memory orderParameters,
+        uint256 offerItemIndex,
+        address seaportAddress
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Validates the OfferItem approvals and balances
+     * @param orderParameters The parameters for the order to validate
+     * @param offerItemIndex The index of the offerItem in offer array to validate
+     * @return errorsAndWarnings An ErrorsAndWarnings structs with results
+     */
+    function validateOfferItemApprovalAndBalance(
+        OrderParameters memory orderParameters,
+        uint256 offerItemIndex,
+        address seaportAddress
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Calls validateOrder on the order's zone with the given zoneParameters
+     * @param orderParameters The parameters for the order to validate
+     * @param zoneParameters The parameters for the zone to validate
+     * @return errorsAndWarnings An ErrorsAndWarnings structs with results
+     */
+    function validateOrderWithZone(
+        OrderParameters memory orderParameters,
+        ZoneParameters memory zoneParameters
+    ) external view returns (ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Gets the approval address for the given conduit key
+     * @param conduitKey Conduit key to get approval address for
+     * @return errorsAndWarnings An ErrorsAndWarnings structs with results
+     */
+    function getApprovalAddress(
+        bytes32 conduitKey,
+        address seaportAddress
+    )
+        external
+        view
+        returns (address, ErrorsAndWarnings memory errorsAndWarnings);
+
+    /**
+     * @notice Safely check that a contract implements an interface
+     * @param token The token address to check
+     * @param interfaceHash The interface hash to check
+     */
+    function checkInterface(
+        address token,
+        bytes4 interfaceHash
+    ) external view returns (bool);
+
+    function isPaymentToken(ItemType itemType) external pure returns (bool);
+
+    /*//////////////////////////////////////////////////////////////
+                        Merkle Helpers
+    //////////////////////////////////////////////////////////////*/
+
+    /**
+     * @notice Sorts an array of token ids by the keccak256 hash of the id. Required ordering of ids
+     *    for other merkle operations.
+     * @param includedTokens An array of included token ids.
+     * @return sortedTokens The sorted `includedTokens` array.
+     */
+    function sortMerkleTokens(
+        uint256[] memory includedTokens
+    ) external view returns (uint256[] memory sortedTokens);
+
+    /**
+     * @notice Creates a merkle root for includedTokens.
+     * @dev `includedTokens` must be sorting in strictly ascending order according to the keccak256 hash of the value.
+     * @return merkleRoot The merkle root
+     * @return errorsAndWarnings Errors and warnings from the operation
+     */
+    function getMerkleRoot(
+        uint256[] memory includedTokens
+    )
+        external
+        view
+        returns (
+            bytes32 merkleRoot,
+            ErrorsAndWarnings memory errorsAndWarnings
+        );
+
+    /**
+     * @notice Creates a merkle proof for the the targetIndex contained in includedTokens.
+     * @dev `targetIndex` is referring to the index of an element in `includedTokens`.
+     *    `includedTokens` must be sorting in ascending order according to the keccak256 hash of the value.
+     * @return merkleProof The merkle proof
+     * @return errorsAndWarnings Errors and warnings from the operation
+     */
+    function getMerkleProof(
+        uint256[] memory includedTokens,
+        uint256 targetIndex
+    )
+        external
+        view
+        returns (
+            bytes32[] memory merkleProof,
+            ErrorsAndWarnings memory errorsAndWarnings
+        );
+
+    function verifyMerkleProof(
+        bytes32 merkleRoot,
+        bytes32[] memory merkleProof,
+        uint256 valueToProve
+    ) external view returns (bool);
+}
diff --git a/contracts/seaport/helpers/order-validator/lib/SeaportValidatorTypes.sol b/contracts/seaport/helpers/order-validator/lib/SeaportValidatorTypes.sol
new file mode 100644
index 00000000..065e8933
--- /dev/null
+++ b/contracts/seaport/helpers/order-validator/lib/SeaportValidatorTypes.sol
@@ -0,0 +1,508 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.10;
+
+struct ValidationConfiguration {
+    /// @notice The seaport address.
+    address seaport;
+    /// @notice Recipient for primary fee payments.
+    address primaryFeeRecipient;
+    /// @notice Bips for primary fee payments.
+    uint256 primaryFeeBips;
+    /// @notice Should creator fees be checked?
+    bool checkCreatorFee;
+    /// @notice Should strict validation be skipped?
+    bool skipStrictValidation;
+    /// @notice Short order duration in seconds
+    uint256 shortOrderDuration;
+    /// @notice Distant order expiration delta in seconds. Warning if order expires in longer than this.
+    uint256 distantOrderExpiration;
+}
+
+struct ConsiderationItemConfiguration {
+    address primaryFeeRecipient;
+    uint256 primaryFeeBips;
+    bool checkCreatorFee;
+}
+
+enum GenericIssue {
+    InvalidOrderFormat // 100
+}
+
+enum ERC20Issue {
+    IdentifierNonZero, // 200
+    InvalidToken, // 201
+    InsufficientAllowance, // 202
+    InsufficientBalance // 203
+}
+
+enum ERC721Issue {
+    AmountNotOne, // 300
+    InvalidToken, // 301
+    IdentifierDNE, // 302
+    NotOwner, // 303
+    NotApproved, // 304
+    CriteriaNotPartialFill // 305
+}
+
+enum ERC1155Issue {
+    InvalidToken, // 400
+    NotApproved, // 401
+    InsufficientBalance // 402
+}
+
+enum ConsiderationIssue {
+    AmountZero, // 500
+    NullRecipient, // 501
+    ExtraItems, // 502
+    PrivateSaleToSelf, // 503
+    ZeroItems, // 504
+    DuplicateItem, // 505
+    OffererNotReceivingAtLeastOneItem, // 506
+    PrivateSale, // 507
+    AmountVelocityHigh, // 508
+    AmountStepLarge // 509
+}
+
+enum OfferIssue {
+    ZeroItems, // 600
+    AmountZero, // 601
+    MoreThanOneItem, // 602
+    NativeItem, // 603
+    DuplicateItem, // 604
+    AmountVelocityHigh, // 605
+    AmountStepLarge // 606
+}
+
+enum PrimaryFeeIssue {
+    Missing, // 700
+    ItemType, // 701
+    Token, // 702
+    StartAmount, // 703
+    EndAmount, // 704
+    Recipient // 705
+}
+
+enum StatusIssue {
+    Cancelled, // 800
+    FullyFilled, // 801
+    ContractOrder // 802
+}
+
+enum TimeIssue {
+    EndTimeBeforeStartTime, // 900
+    Expired, // 901
+    DistantExpiration, // 902
+    NotActive, // 903
+    ShortOrder // 904
+}
+
+enum ConduitIssue {
+    KeyInvalid, // 1000
+    MissingSeaportChannel // 1001
+}
+
+enum SignatureIssue {
+    Invalid, // 1100
+    ContractOrder, // 1101
+    LowCounter, // 1102
+    HighCounter, // 1103
+    OriginalConsiderationItems // 1104
+}
+
+enum CreatorFeeIssue {
+    Missing, // 1200
+    ItemType, // 1201
+    Token, // 1202
+    StartAmount, // 1203
+    EndAmount, // 1204
+    Recipient // 1205
+}
+
+enum NativeIssue {
+    TokenAddress, // 1300
+    IdentifierNonZero, // 1301
+    InsufficientBalance // 1302
+}
+
+enum ZoneIssue {
+    InvalidZone, // 1400
+    RejectedOrder, // 1401
+    NotSet, // 1402
+    EOAZone // 1403
+}
+
+enum MerkleIssue {
+    SingleLeaf, // 1500
+    Unsorted // 1501
+}
+
+enum ContractOffererIssue {
+    InvalidContractOfferer // 1600
+}
+
+/**
+ * @title IssueParser - parse issues into integers
+ * @notice Implements a `parseInt` function for each issue type.
+ *    offsets the enum value to place within the issue range.
+ */
+library IssueParser {
+    function parseInt(GenericIssue err) internal pure returns (uint16) {
+        return uint16(err) + 100;
+    }
+
+    function parseInt(ERC20Issue err) internal pure returns (uint16) {
+        return uint16(err) + 200;
+    }
+
+    function parseInt(ERC721Issue err) internal pure returns (uint16) {
+        return uint16(err) + 300;
+    }
+
+    function parseInt(ERC1155Issue err) internal pure returns (uint16) {
+        return uint16(err) + 400;
+    }
+
+    function parseInt(ConsiderationIssue err) internal pure returns (uint16) {
+        return uint16(err) + 500;
+    }
+
+    function parseInt(OfferIssue err) internal pure returns (uint16) {
+        return uint16(err) + 600;
+    }
+
+    function parseInt(PrimaryFeeIssue err) internal pure returns (uint16) {
+        return uint16(err) + 700;
+    }
+
+    function parseInt(StatusIssue err) internal pure returns (uint16) {
+        return uint16(err) + 800;
+    }
+
+    function parseInt(TimeIssue err) internal pure returns (uint16) {
+        return uint16(err) + 900;
+    }
+
+    function parseInt(ConduitIssue err) internal pure returns (uint16) {
+        return uint16(err) + 1000;
+    }
+
+    function parseInt(SignatureIssue err) internal pure returns (uint16) {
+        return uint16(err) + 1100;
+    }
+
+    function parseInt(CreatorFeeIssue err) internal pure returns (uint16) {
+        return uint16(err) + 1200;
+    }
+
+    function parseInt(NativeIssue err) internal pure returns (uint16) {
+        return uint16(err) + 1300;
+    }
+
+    function parseInt(ZoneIssue err) internal pure returns (uint16) {
+        return uint16(err) + 1400;
+    }
+
+    function parseInt(MerkleIssue err) internal pure returns (uint16) {
+        return uint16(err) + 1500;
+    }
+
+    function parseInt(ContractOffererIssue err) internal pure returns (uint16) {
+        return uint16(err) + 1600;
+    }
+}
+
+library IssueStringHelpers {
+    function toString(GenericIssue id) internal pure returns (string memory) {
+        string memory code;
+        if (id == GenericIssue.InvalidOrderFormat) {
+            code = "InvalidOrderFormat";
+        }
+        return string.concat("GenericIssue: ", code);
+    }
+
+    function toString(ERC20Issue id) internal pure returns (string memory) {
+        string memory code;
+        if (id == ERC20Issue.IdentifierNonZero) {
+            code = "IdentifierNonZero";
+        } else if (id == ERC20Issue.InvalidToken) {
+            code = "InvalidToken";
+        } else if (id == ERC20Issue.InsufficientAllowance) {
+            code = "InsufficientAllowance";
+        } else if (id == ERC20Issue.InsufficientBalance) {
+            code = "InsufficientBalance";
+        }
+        return string.concat("ERC20Issue: ", code);
+    }
+
+    function toString(ERC721Issue id) internal pure returns (string memory) {
+        string memory code;
+        if (id == ERC721Issue.AmountNotOne) {
+            code = "AmountNotOne";
+        } else if (id == ERC721Issue.InvalidToken) {
+            code = "InvalidToken";
+        } else if (id == ERC721Issue.IdentifierDNE) {
+            code = "IdentifierDNE";
+        } else if (id == ERC721Issue.NotOwner) {
+            code = "NotOwner";
+        } else if (id == ERC721Issue.NotApproved) {
+            code = "NotApproved";
+        } else if (id == ERC721Issue.CriteriaNotPartialFill) {
+            code = "CriteriaNotPartialFill";
+        }
+        return string.concat("ERC721Issue: ", code);
+    }
+
+    function toString(ERC1155Issue id) internal pure returns (string memory) {
+        string memory code;
+        if (id == ERC1155Issue.InvalidToken) {
+            code = "InvalidToken";
+        } else if (id == ERC1155Issue.NotApproved) {
+            code = "NotApproved";
+        } else if (id == ERC1155Issue.InsufficientBalance) {
+            code = "InsufficientBalance";
+        }
+        return string.concat("ERC1155Issue: ", code);
+    }
+
+    function toString(
+        ConsiderationIssue id
+    ) internal pure returns (string memory) {
+        string memory code;
+        if (id == ConsiderationIssue.AmountZero) {
+            code = "AmountZero";
+        } else if (id == ConsiderationIssue.NullRecipient) {
+            code = "NullRecipient";
+        } else if (id == ConsiderationIssue.ExtraItems) {
+            code = "ExtraItems";
+        } else if (id == ConsiderationIssue.PrivateSaleToSelf) {
+            code = "PrivateSaleToSelf";
+        } else if (id == ConsiderationIssue.ZeroItems) {
+            code = "ZeroItems";
+        } else if (id == ConsiderationIssue.DuplicateItem) {
+            code = "DuplicateItem";
+        } else if (id == ConsiderationIssue.OffererNotReceivingAtLeastOneItem) {
+            code = "OffererNotReceivingAtLeastOneItem";
+        } else if (id == ConsiderationIssue.PrivateSale) {
+            code = "PrivateSale";
+        } else if (id == ConsiderationIssue.AmountVelocityHigh) {
+            code = "AmountVelocityHigh";
+        } else if (id == ConsiderationIssue.AmountStepLarge) {
+            code = "AmountStepLarge";
+        }
+        return string.concat("ConsiderationIssue: ", code);
+    }
+
+    function toString(OfferIssue id) internal pure returns (string memory) {
+        string memory code;
+        if (id == OfferIssue.ZeroItems) {
+            code = "ZeroItems";
+        } else if (id == OfferIssue.AmountZero) {
+            code = "AmountZero";
+        } else if (id == OfferIssue.MoreThanOneItem) {
+            code = "MoreThanOneItem";
+        } else if (id == OfferIssue.NativeItem) {
+            code = "NativeItem";
+        } else if (id == OfferIssue.DuplicateItem) {
+            code = "DuplicateItem";
+        } else if (id == OfferIssue.AmountVelocityHigh) {
+            code = "AmountVelocityHigh";
+        } else if (id == OfferIssue.AmountStepLarge) {
+            code = "AmountStepLarge";
+        }
+        return string.concat("OfferIssue: ", code);
+    }
+
+    function toString(
+        PrimaryFeeIssue id
+    ) internal pure returns (string memory) {
+        string memory code;
+        if (id == PrimaryFeeIssue.Missing) {
+            code = "Missing";
+        } else if (id == PrimaryFeeIssue.ItemType) {
+            code = "ItemType";
+        } else if (id == PrimaryFeeIssue.Token) {
+            code = "Token";
+        } else if (id == PrimaryFeeIssue.StartAmount) {
+            code = "StartAmount";
+        } else if (id == PrimaryFeeIssue.EndAmount) {
+            code = "EndAmount";
+        } else if (id == PrimaryFeeIssue.Recipient) {
+            code = "Recipient";
+        }
+        return string.concat("PrimaryFeeIssue: ", code);
+    }
+
+    function toString(StatusIssue id) internal pure returns (string memory) {
+        string memory code;
+        if (id == StatusIssue.Cancelled) {
+            code = "Cancelled";
+        } else if (id == StatusIssue.FullyFilled) {
+            code = "FullyFilled";
+        } else if (id == StatusIssue.ContractOrder) {
+            code = "ContractOrder";
+        }
+        return string.concat("StatusIssue: ", code);
+    }
+
+    function toString(TimeIssue id) internal pure returns (string memory) {
+        string memory code;
+        if (id == TimeIssue.EndTimeBeforeStartTime) {
+            code = "EndTimeBeforeStartTime";
+        } else if (id == TimeIssue.Expired) {
+            code = "Expired";
+        } else if (id == TimeIssue.DistantExpiration) {
+            code = "DistantExpiration";
+        } else if (id == TimeIssue.NotActive) {
+            code = "NotActive";
+        } else if (id == TimeIssue.ShortOrder) {
+            code = "ShortOrder";
+        }
+        return string.concat("TimeIssue: ", code);
+    }
+
+    function toString(ConduitIssue id) internal pure returns (string memory) {
+        string memory code;
+        if (id == ConduitIssue.KeyInvalid) {
+            code = "KeyInvalid";
+        } else if (id == ConduitIssue.MissingSeaportChannel) {
+            code = "MissingSeaportChannel";
+        }
+        return string.concat("ConduitIssue: ", code);
+    }
+
+    function toString(SignatureIssue id) internal pure returns (string memory) {
+        string memory code;
+        if (id == SignatureIssue.Invalid) {
+            code = "Invalid";
+        } else if (id == SignatureIssue.ContractOrder) {
+            code = "ContractOrder";
+        } else if (id == SignatureIssue.LowCounter) {
+            code = "LowCounter";
+        } else if (id == SignatureIssue.HighCounter) {
+            code = "HighCounter";
+        } else if (id == SignatureIssue.OriginalConsiderationItems) {
+            code = "OriginalConsiderationItems";
+        }
+        return string.concat("SignatureIssue: ", code);
+    }
+
+    function toString(
+        CreatorFeeIssue id
+    ) internal pure returns (string memory) {
+        string memory code;
+        if (id == CreatorFeeIssue.Missing) {
+            code = "Missing";
+        } else if (id == CreatorFeeIssue.ItemType) {
+            code = "ItemType";
+        } else if (id == CreatorFeeIssue.Token) {
+            code = "Token";
+        } else if (id == CreatorFeeIssue.StartAmount) {
+            code = "StartAmount";
+        } else if (id == CreatorFeeIssue.EndAmount) {
+            code = "EndAmount";
+        } else if (id == CreatorFeeIssue.Recipient) {
+            code = "Recipient";
+        }
+        return string.concat("CreatorFeeIssue: ", code);
+    }
+
+    function toString(NativeIssue id) internal pure returns (string memory) {
+        string memory code;
+        if (id == NativeIssue.TokenAddress) {
+            code = "TokenAddress";
+        } else if (id == NativeIssue.IdentifierNonZero) {
+            code = "IdentifierNonZero";
+        } else if (id == NativeIssue.InsufficientBalance) {
+            code = "InsufficientBalance";
+        }
+        return string.concat("NativeIssue: ", code);
+    }
+
+    function toString(ZoneIssue id) internal pure returns (string memory) {
+        string memory code;
+        if (id == ZoneIssue.InvalidZone) {
+            code = "InvalidZone";
+        } else if (id == ZoneIssue.RejectedOrder) {
+            code = "RejectedOrder";
+        } else if (id == ZoneIssue.NotSet) {
+            code = "NotSet";
+        } else if (id == ZoneIssue.EOAZone) {
+            code = "EOAZone";
+        }
+        return string.concat("ZoneIssue: ", code);
+    }
+
+    function toString(MerkleIssue id) internal pure returns (string memory) {
+        string memory code;
+        if (id == MerkleIssue.SingleLeaf) {
+            code = "SingleLeaf";
+        } else if (id == MerkleIssue.Unsorted) {
+            code = "Unsorted";
+        }
+        return string.concat("MerkleIssue: ", code);
+    }
+
+    function toString(
+        ContractOffererIssue id
+    ) internal pure returns (string memory) {
+        string memory code;
+        if (id == ContractOffererIssue.InvalidContractOfferer) {
+            code = "InvalidContractOfferer";
+        }
+        return string.concat("ContractOffererIssue: ", code);
+    }
+
+    function toIssueString(
+        uint16 issueCode
+    ) internal pure returns (string memory issueString) {
+        uint16 issue = (issueCode / 100) * 100;
+        uint8 id = uint8(issueCode % 100);
+        if (issue == 100) {
+            return toString(GenericIssue(id));
+        } else if (issue == 200) {
+            return toString(ERC20Issue(id));
+        } else if (issue == 300) {
+            return toString(ERC721Issue(id));
+        } else if (issue == 400) {
+            return toString(ERC1155Issue(id));
+        } else if (issue == 500) {
+            return toString(ConsiderationIssue(id));
+        } else if (issue == 600) {
+            return toString(OfferIssue(id));
+        } else if (issue == 700) {
+            return toString(PrimaryFeeIssue(id));
+        } else if (issue == 800) {
+            return toString(StatusIssue(id));
+        } else if (issue == 900) {
+            return toString(TimeIssue(id));
+        } else if (issue == 1000) {
+            return toString(ConduitIssue(id));
+        } else if (issue == 1100) {
+            return toString(SignatureIssue(id));
+        } else if (issue == 1200) {
+            return toString(CreatorFeeIssue(id));
+        } else if (issue == 1300) {
+            return toString(NativeIssue(id));
+        } else if (issue == 1400) {
+            return toString(ZoneIssue(id));
+        } else if (issue == 1500) {
+            return toString(MerkleIssue(id));
+        } else if (issue == 1600) {
+            return toString(ContractOffererIssue(id));
+        } else {
+            revert("IssueStringHelpers: Unknown issue code");
+        }
+    }
+
+    function toIssueString(
+        uint16[] memory issueCodes
+    ) internal pure returns (string memory issueString) {
+        for (uint256 i; i < issueCodes.length; i++) {
+            issueString = string.concat(
+                issueString,
+                "\n    ",
+                toIssueString(issueCodes[i])
+            );
+        }
+    }
+}
diff --git a/contracts/seaport/interfaces/ImmutableCreate2FactoryInterface.sol b/contracts/seaport/interfaces/ImmutableCreate2FactoryInterface.sol
new file mode 100644
index 00000000..73fc22d7
--- /dev/null
+++ b/contracts/seaport/interfaces/ImmutableCreate2FactoryInterface.sol
@@ -0,0 +1,96 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+/**
+ * @title ImmutableCreate2FactoryInterface
+ * @author 0age
+ * @notice This contract provides a safeCreate2 function that takes a salt value
+ *         and a block of initialization code as arguments and passes them into
+ *         inline assembly. The contract prevents redeploys by maintaining a
+ *         mapping of all contracts that have already been deployed, and
+ *         prevents frontrunning or other collisions by requiring that the first
+ *         20 bytes of the salt are equal to the address of the caller (this can
+ *         be bypassed by setting the first 20 bytes to the null address). There
+ *         is also a view function that computes the address of the contract
+ *         that will be created when submitting a given salt or nonce along with
+ *         a given block of initialization code.
+ */
+interface ImmutableCreate2FactoryInterface {
+    /**
+     * @dev Create a contract using CREATE2 by submitting a given salt or nonce
+     *      along with the initialization code for the contract. Note that the
+     *      first 20 bytes of the salt must match those of the calling address,
+     *      which prevents contract creation events from being submitted by
+     *      unintended parties.
+     *
+     * @param salt               The nonce that will be passed into the CREATE2
+     *                           call.
+     * @param initializationCode The initialization code that will be passed
+     *                           into the CREATE2 call.
+     *
+     * @return deploymentAddress Address of the contract that will be created.
+     */
+    function safeCreate2(
+        bytes32 salt,
+        bytes calldata initializationCode
+    ) external payable returns (address deploymentAddress);
+
+    /**
+     * @dev Compute the address of the contract that will be created when
+     *      submitting a given salt or nonce to the contract along with the
+     *      contract's initialization code. The CREATE2 address is computed in
+     *      accordance with EIP-1014, and adheres to the formula therein of
+     *      `keccak256( 0xff ++ address ++ salt ++ keccak256(init_code)))[12:]`
+     *      when performing the computation. The computed address is then
+     *      checked for any existing contract code - if so, the null address
+     *      will be returned instead.
+     *
+     * @param salt     The nonce passed into the CREATE2 address calculation.
+     * @param initCode The contract initialization code to be used that will be
+     *                 passed into the CREATE2 address calculation.
+     *
+     * @return deploymentAddress Address of the contract that will be created,
+     *                           or the null address if a contract already
+     *                           exists at that address.
+     */
+    function findCreate2Address(
+        bytes32 salt,
+        bytes calldata initCode
+    ) external view returns (address deploymentAddress);
+
+    /**
+     * @dev Compute the address of the contract that will be created when
+     *      submitting a given salt or nonce to the contract along with the
+     *      keccak256 hash of the contract's initialization code. The CREATE2
+     *      address is computed in accordance with EIP-1014, and adheres to the
+     *      `keccak256( 0xff ++ address ++ salt ++ keccak256(init_code)))[12:]`
+     *      formula when performing the computation. The computed address is
+     *      then checked for any existing contract code - if so, the null
+     *      address will be returned instead.
+     *
+     * @param salt         The nonce passed into the CREATE2 address
+     *                     calculation.
+     * @param initCodeHash The keccak256 hash of the initialization code that
+     *                     will be passed into the CREATE2 address calculation.
+     *
+     * @return deploymentAddress Address of the contract that will be created,
+     *                           or the null address if a contract already
+     *                           exists at that address.
+     */
+    function findCreate2AddressViaHash(
+        bytes32 salt,
+        bytes32 initCodeHash
+    ) external view returns (address deploymentAddress);
+
+    /**
+     * @dev Determine if a contract has already been deployed by the factory to
+     *      a given address.
+     *
+     * @param deploymentAddress The contract address to check.
+     *
+     * @return True if the contract has been deployed, false otherwise.
+     */
+    function hasBeenDeployed(
+        address deploymentAddress
+    ) external view returns (bool);
+}
\ No newline at end of file
diff --git a/contracts/seaport/interfaces/ZoneInteractionErrors.sol b/contracts/seaport/interfaces/ZoneInteractionErrors.sol
new file mode 100644
index 00000000..786a2ca2
--- /dev/null
+++ b/contracts/seaport/interfaces/ZoneInteractionErrors.sol
@@ -0,0 +1,29 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+/**
+ * @title ZoneInteractionErrors
+ * @author 0age
+ * @notice ZoneInteractionErrors contains errors related to zone interaction.
+ */
+interface ZoneInteractionErrors {
+    /**
+     * @dev Revert with an error when attempting to fill an order that specifies
+     *      a restricted submitter as its order type when not submitted by
+     *      either the offerer or the order's zone or approved as valid by the
+     *      zone in question via a call to `isValidOrder`.
+     *
+     * @param orderHash The order hash for the invalid restricted order.
+     */
+    error InvalidRestrictedOrder(bytes32 orderHash);
+
+    /**
+     * @dev Revert with an error when attempting to fill a contract order that
+     *      fails to generate an order successfully, that does not adhere to the
+     *      requirements for minimum spent or maximum received supplied by the
+     *      fulfiller, or that fails the post-execution `ratifyOrder` check..
+     *
+     * @param orderHash The order hash for the invalid contract order.
+     */
+    error InvalidContractOrder(bytes32 orderHash);
+}
diff --git a/contracts/seaport/interfaces/ZoneInterface.sol b/contracts/seaport/interfaces/ZoneInterface.sol
new file mode 100644
index 00000000..048b1e1a
--- /dev/null
+++ b/contracts/seaport/interfaces/ZoneInterface.sol
@@ -0,0 +1,46 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import {
+    ZoneParameters,
+    Schema
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import { IERC165 } from "seaport-types/src/interfaces/IERC165.sol";
+
+/**
+ * @title  ZoneInterface
+ * @notice Contains functions exposed by a zone.
+ */
+interface ZoneInterface is IERC165 {
+    /**
+     * @dev Validates an order.
+     *
+     * @param zoneParameters The context about the order fulfillment and any
+     *                       supplied extraData.
+     *
+     * @return validOrderMagicValue The magic value that indicates a valid
+     *                              order.
+     */
+    function validateOrder(
+        ZoneParameters calldata zoneParameters
+    ) external returns (bytes4 validOrderMagicValue);
+
+    /**
+     * @dev Returns the metadata for this zone.
+     *
+     * @return name The name of the zone.
+     * @return schemas The schemas that the zone implements.
+     */
+    function getSeaportMetadata()
+        external
+        view
+        returns (
+            string memory name,
+            Schema[] memory schemas // map to Seaport Improvement Proposal IDs
+        );
+
+    function supportsInterface(
+        bytes4 interfaceId
+    ) external view override returns (bool);
+}
diff --git a/contracts/seaport/zones/PausableZone.sol b/contracts/seaport/zones/PausableZone.sol
new file mode 100644
index 00000000..673a5c79
--- /dev/null
+++ b/contracts/seaport/zones/PausableZone.sol
@@ -0,0 +1,268 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import { ZoneInterface } from "seaport-types/src/interfaces/ZoneInterface.sol";
+
+import {
+    PausableZoneEventsAndErrors
+} from "./interfaces/PausableZoneEventsAndErrors.sol";
+
+import { ERC165 } from "@openzeppelin/contracts/seaport/utils/introspection/ERC165.sol";
+
+import {
+    SeaportInterface
+} from "seaport-types/src/interfaces/SeaportInterface.sol";
+
+import {
+    AdvancedOrder,
+    CriteriaResolver,
+    Execution,
+    Fulfillment,
+    Order,
+    OrderComponents,
+    Schema,
+    ZoneParameters
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import { PausableZoneInterface } from "./interfaces/PausableZoneInterface.sol";
+
+/**
+ * @title  PausableZone
+ * @author cupOJoseph, BCLeFevre, ryanio
+ * @notice PausableZone is a simple zone implementation that approves every
+ *         order. It can be self-destructed by its controller to pause
+ *         restricted orders that have it set as their zone. Note that this zone
+ *         cannot execute orders that return native tokens to the fulfiller.
+ */
+contract PausableZone is
+    ERC165,
+    PausableZoneEventsAndErrors,
+    ZoneInterface,
+    PausableZoneInterface
+{
+    // Set an immutable controller that can pause the zone & update an operator.
+    address internal immutable _controller;
+
+    // Set an operator that can instruct the zone to cancel or execute orders.
+    address public operator;
+
+    /**
+     * @dev Ensure that the caller is either the operator or controller.
+     */
+    modifier isOperator() {
+        // Ensure that the caller is either the operator or the controller.
+        if (msg.sender != operator && msg.sender != _controller) {
+            revert InvalidOperator();
+        }
+
+        // Continue with function execution.
+        _;
+    }
+
+    /**
+     * @dev Ensure that the caller is the controller.
+     */
+    modifier isController() {
+        // Ensure that the caller is the controller.
+        if (msg.sender != _controller) {
+            revert InvalidController();
+        }
+
+        // Continue with function execution.
+        _;
+    }
+
+    /**
+     * @notice Set the deployer as the controller of the zone.
+     */
+    constructor() {
+        // Set the controller to the deployer.
+        _controller = msg.sender;
+
+        // Emit an event signifying that the zone is unpaused.
+        emit Unpaused();
+    }
+
+    /**
+     * @notice Cancel an arbitrary number of orders that have agreed to use the
+     *         contract as their zone.
+     *
+     * @param seaport  The Seaport address.
+     * @param orders   The orders to cancel.
+     *
+     * @return cancelled A boolean indicating whether the supplied orders have
+     *                   been successfully cancelled.
+     */
+    function cancelOrders(
+        SeaportInterface seaport,
+        OrderComponents[] calldata orders
+    ) external override isOperator returns (bool cancelled) {
+        // Call cancel on Seaport and return its boolean value.
+        cancelled = seaport.cancel(orders);
+    }
+
+    /**
+     * @notice Pause this contract, safely stopping orders from using
+     *         the contract as a zone. Restricted orders with this address as a
+     *         zone will not be fulfillable unless the zone is redeployed to the
+     *         same address.
+     */
+    function pause(address payee) external override isController {
+        // Emit an event signifying that the zone is paused.
+        emit Paused();
+
+        // Destroy the zone, sending any native tokens to the transaction
+        // submitter.
+        selfdestruct(payable(payee));
+    }
+
+    /**
+     * @notice Assign the given address with the ability to operate the zone.
+     *
+     * @param operatorToAssign The address to assign as the operator.
+     */
+    function assignOperator(
+        address operatorToAssign
+    ) external override isController {
+        // Ensure the operator being assigned is not the null address.
+        if (operatorToAssign == address(0)) {
+            revert PauserCanNotBeSetAsZero();
+        }
+
+        // Set the given address as the new operator.
+        operator = operatorToAssign;
+
+        // Emit an event indicating the operator has been updated.
+        emit OperatorUpdated(operatorToAssign);
+    }
+
+    /**
+     * @notice Execute an arbitrary number of matched orders, each with
+     *         an arbitrary number of items for offer and consideration
+     *         along with a set of fulfillments allocating offer components
+     *         to consideration components. Note that this call will revert if
+     *         excess native tokens are returned by Seaport.
+     *
+     * @param seaport      The Seaport address.
+     * @param orders       The orders to match.
+     * @param fulfillments An array of elements allocating offer components
+     *                     to consideration components.
+     *
+     * @return executions An array of elements indicating the sequence of
+     *                    transfers performed as part of matching the given
+     *                    orders.
+     */
+    function executeMatchOrders(
+        SeaportInterface seaport,
+        Order[] calldata orders,
+        Fulfillment[] calldata fulfillments
+    )
+        external
+        payable
+        override
+        isOperator
+        returns (Execution[] memory executions)
+    {
+        // Call matchOrders on Seaport and return the sequence of transfers
+        // performed as part of matching the given orders.
+        executions = seaport.matchOrders{ value: msg.value }(
+            orders,
+            fulfillments
+        );
+    }
+
+    /**
+     * @notice Execute an arbitrary number of matched advanced orders,
+     *         each with an arbitrary number of items for offer and
+     *         consideration along with a set of fulfillments allocating
+     *         offer components to consideration components. Note that this call
+     *         will revert if excess native tokens are returned by Seaport.
+     *
+     * @param seaport           The Seaport address.
+     * @param orders            The orders to match.
+     * @param criteriaResolvers An array where each element contains a reference
+     *                          to a specific order as well as that order's
+     *                          offer or consideration, a token identifier, and
+     *                          a proof that the supplied token identifier is
+     *                          contained in the order's merkle root.
+     * @param fulfillments      An array of elements allocating offer components
+     *                          to consideration components.
+     *
+     * @return executions An array of elements indicating the sequence of
+     *                    transfers performed as part of matching the given
+     *                    orders.
+     */
+    function executeMatchAdvancedOrders(
+        SeaportInterface seaport,
+        AdvancedOrder[] calldata orders,
+        CriteriaResolver[] calldata criteriaResolvers,
+        Fulfillment[] calldata fulfillments
+    )
+        external
+        payable
+        override
+        isOperator
+        returns (Execution[] memory executions)
+    {
+        // Call matchAdvancedOrders on Seaport and return the sequence of
+        // transfers performed as part of matching the given orders.
+        executions = seaport.matchAdvancedOrders{ value: msg.value }(
+            orders,
+            criteriaResolvers,
+            fulfillments,
+            msg.sender
+        );
+    }
+
+    /**
+     * @notice Check if a given order including extraData is currently valid.
+     *
+     * @dev This function is called by Seaport whenever any extraData is
+     *      provided by the caller.
+     *
+     * @custom:param zoneParameters A struct that provides context about the
+     *                              order fulfillment and any supplied
+     *                              extraData, as well as all order hashes
+     *                              fulfilled in a call to a match or
+     *                              fulfillAvailable method.
+     *
+     * @return validOrderMagicValue A magic value indicating if the order is
+     *                              currently valid.
+     */
+    function validateOrder(
+        /**
+         * @custom:name zoneParameters
+         */
+        ZoneParameters calldata
+    ) external pure override returns (bytes4 validOrderMagicValue) {
+        // Return the selector of isValidOrder as the magic value.
+        validOrderMagicValue = ZoneInterface.validateOrder.selector;
+    }
+
+    /**
+     * @dev Returns the metadata for this zone.
+     */
+    function getSeaportMetadata()
+        external
+        pure
+        override
+        returns (
+            string memory name,
+            Schema[] memory schemas // map to Seaport Improvement Proposal IDs
+        )
+    {
+        schemas = new Schema[](1);
+        schemas[0].id = 3003;
+        schemas[0].metadata = new bytes(0);
+
+        return ("PausableZone", schemas);
+    }
+
+    function supportsInterface(
+        bytes4 interfaceId
+    ) public view override(ERC165, ZoneInterface) returns (bool) {
+        return
+            interfaceId == type(ZoneInterface).interfaceId ||
+            super.supportsInterface(interfaceId);
+    }
+}
diff --git a/contracts/seaport/zones/PausableZoneController.sol b/contracts/seaport/zones/PausableZoneController.sol
new file mode 100644
index 00000000..d506404d
--- /dev/null
+++ b/contracts/seaport/zones/PausableZoneController.sol
@@ -0,0 +1,381 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import { PausableZone } from "./PausableZone.sol";
+
+import {
+    PausableZoneControllerInterface
+} from "./interfaces/PausableZoneControllerInterface.sol";
+
+import {
+    PausableZoneEventsAndErrors
+} from "./interfaces/PausableZoneEventsAndErrors.sol";
+
+import {
+    AdvancedOrder,
+    CriteriaResolver,
+    Execution,
+    Fulfillment,
+    Order,
+    OrderComponents
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    SeaportInterface
+} from "seaport-types/src/interfaces/SeaportInterface.sol";
+
+/**
+ * @title  PausableZoneController
+ * @author cupOJoseph, BCLeFevre, stuckinaboot, stephankmin
+ * @notice PausableZoneController enables deploying, pausing and executing
+ *         orders on PausableZones. This deployer is designed to be owned
+ *         by a gnosis safe, DAO, or trusted party.
+ */
+contract PausableZoneController is
+    PausableZoneControllerInterface,
+    PausableZoneEventsAndErrors
+{
+    // Set the owner that can deploy, pause and execute orders on PausableZones.
+    address internal _owner;
+
+    // Set the address of the new potential owner of the zone.
+    address private _potentialOwner;
+
+    // Set the address with the ability to pause the zone.
+    address internal _pauser;
+
+    // Set the immutable zone creation code hash.
+    bytes32 public immutable zoneCreationCode;
+
+    /**
+     * @dev Throws if called by any account other than the owner or pauser.
+     */
+    modifier isPauser() {
+        if (msg.sender != _pauser && msg.sender != _owner) {
+            revert InvalidPauser();
+        }
+        _;
+    }
+
+    /**
+     * @notice Set the owner of the controller and store
+     *         the zone creation code.
+     *
+     * @param ownerAddress The deployer to be set as the owner.
+     */
+    constructor(address ownerAddress) {
+        // Set the owner address as the owner.
+        _owner = ownerAddress;
+
+        // Hash and store the zone creation code.
+        zoneCreationCode = keccak256(type(PausableZone).creationCode);
+    }
+
+    /**
+     * @notice Deploy a PausableZone to a precomputed address.
+     *
+     * @param salt The salt to be used to derive the zone address
+     *
+     * @return derivedAddress The derived address for the zone.
+     */
+    function createZone(
+        bytes32 salt
+    ) external override returns (address derivedAddress) {
+        // Ensure the caller is the owner.
+        if (msg.sender != _owner) {
+            revert CallerIsNotOwner();
+        }
+
+        // Derive the PausableZone address.
+        // This expression demonstrates address computation but is not required.
+        derivedAddress = address(
+            uint160(
+                uint256(
+                    keccak256(
+                        abi.encodePacked(
+                            bytes1(0xff),
+                            address(this),
+                            salt,
+                            zoneCreationCode
+                        )
+                    )
+                )
+            )
+        );
+
+        // Revert if a zone is currently deployed to the derived address.
+        if (derivedAddress.code.length != 0) {
+            revert ZoneAlreadyExists(derivedAddress);
+        }
+
+        // Deploy the zone using the supplied salt.
+        new PausableZone{ salt: salt }();
+
+        // Emit an event signifying that the zone was created.
+        emit ZoneCreated(derivedAddress, salt);
+    }
+
+    /**
+     * @notice Pause orders on a given zone.
+     *
+     * @param zone The address of the zone to be paused.
+     *
+     * @return success A boolean indicating the zone has been paused.
+     */
+    function pause(
+        address zone
+    ) external override isPauser returns (bool success) {
+        // Call pause on the given zone.
+        PausableZone(zone).pause(msg.sender);
+
+        // Return a boolean indicating the pause was successful.
+        success = true;
+    }
+
+    /**
+     * @notice Cancel Seaport orders on a given zone.
+     *
+     * @param pausableZoneAddress The zone that manages the
+     * orders to be cancelled.
+     * @param seaportAddress      The Seaport address.
+     * @param orders              The orders to cancel.
+     */
+    function cancelOrders(
+        address pausableZoneAddress,
+        SeaportInterface seaportAddress,
+        OrderComponents[] calldata orders
+    ) external override {
+        // Ensure the caller is the owner.
+        if (msg.sender != _owner) {
+            revert CallerIsNotOwner();
+        }
+
+        // Create a zone object from the zone address.
+        PausableZone zone = PausableZone(pausableZoneAddress);
+
+        // Call cancelOrders on the given zone.
+        zone.cancelOrders(seaportAddress, orders);
+    }
+
+    /**
+     * @notice Execute an arbitrary number of matched orders on a given zone.
+     *
+     * @param pausableZoneAddress The zone that manages the orders
+     * to be cancelled.
+     * @param seaportAddress      The Seaport address.
+     * @param orders              The orders to match.
+     * @param fulfillments        An array of elements allocating offer
+     *                            components to consideration components.
+     *
+     * @return executions An array of elements indicating the sequence of
+     *                    transfers performed as part of matching the given
+     *                    orders.
+     */
+    function executeMatchOrders(
+        address pausableZoneAddress,
+        SeaportInterface seaportAddress,
+        Order[] calldata orders,
+        Fulfillment[] calldata fulfillments
+    ) external payable override returns (Execution[] memory executions) {
+        // Ensure the caller is the owner.
+        if (msg.sender != _owner) {
+            revert CallerIsNotOwner();
+        }
+
+        // Create a zone object from the zone address.
+        PausableZone zone = PausableZone(pausableZoneAddress);
+
+        // Call executeMatchOrders on the given zone and return the sequence
+        // of transfers performed as part of matching the given orders.
+        executions = zone.executeMatchOrders{ value: msg.value }(
+            seaportAddress,
+            orders,
+            fulfillments
+        );
+    }
+
+    /**
+     * @notice Execute an arbitrary number of matched advanced orders on a given
+     *         zone.
+     *
+     * @param pausableZoneAddress The zone that manages the orders to be
+     *                            cancelled.
+     * @param seaportAddress      The Seaport address.
+     * @param orders              The orders to match.
+     * @param criteriaResolvers   An array where each element contains a
+     *                            reference to a specific order as well as that
+     *                            order's offer or consideration, a token
+     *                            identifier, and a proof that the supplied
+     *                            token identifier is contained in the
+     *                            order's merkle root.
+     * @param fulfillments        An array of elements allocating offer
+     *                            components to consideration components.
+     *
+     * @return executions An array of elements indicating the sequence of
+     *                    transfers performed as part of matching the given
+     *                    orders.
+     */
+    function executeMatchAdvancedOrders(
+        address pausableZoneAddress,
+        SeaportInterface seaportAddress,
+        AdvancedOrder[] calldata orders,
+        CriteriaResolver[] calldata criteriaResolvers,
+        Fulfillment[] calldata fulfillments
+    ) external payable override returns (Execution[] memory executions) {
+        // Ensure the caller is the owner.
+        if (msg.sender != _owner) {
+            revert CallerIsNotOwner();
+        }
+
+        // Create a zone object from the zone address.
+        PausableZone zone = PausableZone(pausableZoneAddress);
+
+        // Call executeMatchOrders on the given zone and return the sequence
+        // of transfers performed as part of matching the given orders.
+        executions = zone.executeMatchAdvancedOrders{ value: msg.value }(
+            seaportAddress,
+            orders,
+            criteriaResolvers,
+            fulfillments
+        );
+    }
+
+    /**
+     * @notice Initiate Zone ownership transfer by assigning a new potential
+     *         owner of this contract. Once set, the new potential owner
+     *         may call `acceptOwnership` to claim ownership.
+     *         Only the owner in question may call this function.
+     *
+     * @param newPotentialOwner The address for which to initiate ownership
+     *                          transfer to.
+     */
+    function transferOwnership(address newPotentialOwner) external override {
+        // Ensure the caller is the owner.
+        if (msg.sender != _owner) {
+            revert CallerIsNotOwner();
+        }
+        // Ensure the new potential owner is not an invalid address.
+        if (newPotentialOwner == address(0)) {
+            revert OwnerCanNotBeSetAsZero();
+        }
+
+        // Emit an event indicating that the potential owner has been updated.
+        emit PotentialOwnerUpdated(newPotentialOwner);
+
+        // Set the new potential owner as the potential owner.
+        _potentialOwner = newPotentialOwner;
+    }
+
+    /**
+     * @notice Clear the currently set potential owner, if any.
+     *         Only the owner of this contract may call this function.
+     */
+    function cancelOwnershipTransfer() external override {
+        // Ensure the caller is the current owner.
+        if (msg.sender != _owner) {
+            revert CallerIsNotOwner();
+        }
+
+        // Emit an event indicating that the potential owner has been cleared.
+        emit PotentialOwnerUpdated(address(0));
+
+        // Clear the current new potential owner.
+        delete _potentialOwner;
+    }
+
+    /**
+     * @notice Accept ownership of this contract. Only the account that the
+     *         current owner has set as the new potential owner may call this
+     *         function.
+     */
+    function acceptOwnership() external override {
+        // Ensure the caller is the potential owner.
+        if (msg.sender != _potentialOwner) {
+            revert CallerIsNotPotentialOwner();
+        }
+
+        // Emit an event indicating that the potential owner has been cleared.
+        emit PotentialOwnerUpdated(address(0));
+
+        // Clear the current new potential owner
+        delete _potentialOwner;
+
+        // Emit an event indicating ownership has been transferred.
+        emit OwnershipTransferred(_owner, msg.sender);
+
+        // Set the caller as the owner of this contract.
+        _owner = msg.sender;
+    }
+
+    /**
+     * @notice Assign the given address with the ability to pause the zone.
+     *
+     * @param pauserToAssign The address to assign the pauser role.
+     */
+    function assignPauser(address pauserToAssign) external override {
+        // Ensure the caller is the owner.
+        if (msg.sender != _owner) {
+            revert CallerIsNotOwner();
+        }
+        // Ensure the pauser to assign is not an invalid address.
+        if (pauserToAssign == address(0)) {
+            revert PauserCanNotBeSetAsZero();
+        }
+
+        // Set the given account as the pauser.
+        _pauser = pauserToAssign;
+
+        // Emit an event indicating the pauser has been assigned.
+        emit PauserUpdated(pauserToAssign);
+    }
+
+    /**
+     * @notice Assign the given address with the ability to operate the
+     *         given zone.
+     *
+     * @param pausableZoneAddress The zone address to assign operator role.
+     * @param operatorToAssign    The address to assign as operator.
+     */
+    function assignOperator(
+        address pausableZoneAddress,
+        address operatorToAssign
+    ) external override {
+        // Ensure the caller is the owner.
+        if (msg.sender != _owner) {
+            revert CallerIsNotOwner();
+        }
+        // Create a zone object from the zone address.
+        PausableZone zone = PausableZone(pausableZoneAddress);
+
+        // Call assignOperator on the zone by passing in the given
+        // operator address.
+        zone.assignOperator(operatorToAssign);
+    }
+
+    /**
+     * @notice An external view function that returns the owner.
+     *
+     * @return The address of the owner.
+     */
+    function owner() external view override returns (address) {
+        return _owner;
+    }
+
+    /**
+     * @notice An external view function that returns the potential owner.
+     *
+     * @return The address of the potential owner.
+     */
+    function potentialOwner() external view override returns (address) {
+        return _potentialOwner;
+    }
+
+    /**
+     * @notice An external view function that returns the pauser.
+     *
+     * @return The address of the pauser.
+     */
+    function pauser() external view override returns (address) {
+        return _pauser;
+    }
+}
diff --git a/contracts/seaport/zones/interfaces/PausableZoneControllerInterface.sol b/contracts/seaport/zones/interfaces/PausableZoneControllerInterface.sol
new file mode 100644
index 00000000..ab48d922
--- /dev/null
+++ b/contracts/seaport/zones/interfaces/PausableZoneControllerInterface.sol
@@ -0,0 +1,170 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import {
+    AdvancedOrder,
+    CriteriaResolver,
+    Execution,
+    Fulfillment,
+    Order,
+    OrderComponents
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+import {
+    SeaportInterface
+} from "seaport-types/src/interfaces/SeaportInterface.sol";
+
+/**
+ * @title  PausableZoneController
+ * @author cupOJoseph, BCLeFevre, stuckinaboot
+ * @notice PausableZoneController enables deploying, pausing and executing
+ *         orders on PausableZones. This deployer is designed to be owned
+ *         by a gnosis safe, DAO, or trusted party.
+ */
+interface PausableZoneControllerInterface {
+    /**
+     * @notice Deploy a PausableZone to a precomputed address.
+     *
+     * @param salt The salt to be used to derive the zone address
+     *
+     * @return derivedAddress The derived address for the zone.
+     */
+    function createZone(bytes32 salt) external returns (address derivedAddress);
+
+    /**
+     * @notice Pause orders on a given zone.
+     *
+     * @param zone The address of the zone to be paused.
+     *
+     * @return success A boolean indicating the zone has been paused.
+     */
+    function pause(address zone) external returns (bool success);
+
+    /**
+     * @notice Cancel Seaport offers on a given zone.
+     *
+     * @param pausableZoneAddress The zone that manages the orders to be
+     *                            cancelled.
+     * @param seaportAddress      The Seaport address.
+     * @param orders              The orders to cancel.
+     */
+    function cancelOrders(
+        address pausableZoneAddress,
+        SeaportInterface seaportAddress,
+        OrderComponents[] calldata orders
+    ) external;
+
+    /**
+     * @notice Execute an arbitrary number of matched orders on a given zone.
+     *
+     * @param pausableZoneAddress The zone that manages the orders to be
+     *                            cancelled.
+     * @param seaportAddress      The Seaport address.
+     * @param orders              The orders to match.
+     * @param fulfillments        An array of elements allocating offer
+     *                            components to consideration components.
+     *
+     * @return executions An array of elements indicating the sequence of
+     *                    transfers performed as part of matching the given
+     *                    orders.
+     */
+    function executeMatchOrders(
+        address pausableZoneAddress,
+        SeaportInterface seaportAddress,
+        Order[] calldata orders,
+        Fulfillment[] calldata fulfillments
+    ) external payable returns (Execution[] memory executions);
+
+    /**
+     * @notice Execute an arbitrary number of matched advanced orders on a
+     *         given zone.
+     *
+     * @param pausableZoneAddress The zone that manages the orders to be
+     *                            cancelled.
+     * @param seaportAddress      The Seaport address.
+     * @param orders              The orders to match.
+     * @param criteriaResolvers   An array where each element contains a
+     *                            reference to a specific order as well as
+     *                            that order's offer or consideration,
+     *                            a token identifier, and a proof that
+     *                            the supplied token identifier is
+     *                            contained in the order's merkle root.
+     * @param fulfillments        An array of elements allocating offer
+     *                            components to consideration components.
+     *
+     * @return executions An array of elements indicating the sequence of
+     *                    transfers performed as part of matching the given
+     *                    orders.
+     */
+    function executeMatchAdvancedOrders(
+        address pausableZoneAddress,
+        SeaportInterface seaportAddress,
+        AdvancedOrder[] calldata orders,
+        CriteriaResolver[] calldata criteriaResolvers,
+        Fulfillment[] calldata fulfillments
+    ) external payable returns (Execution[] memory executions);
+
+    /**
+     * @notice Initiate Zone ownership transfer by assigning a new potential
+     *         owner of this contract. Once set, the new potential owner
+     *         may call `acceptOwnership` to claim ownership.
+     *         Only the owner in question may call this function.
+     *
+     * @param newPotentialOwner The address for which to initiate ownership
+     *                          transfer to.
+     */
+    function transferOwnership(address newPotentialOwner) external;
+
+    /**
+     * @notice Clear the currently set potential owner, if any.
+     *         Only the owner of this contract may call this function.
+     */
+    function cancelOwnershipTransfer() external;
+
+    /**
+     * @notice Accept ownership of this contract. Only the account that the
+     *         current owner has set as the new potential owner may call this
+     *         function.
+     */
+    function acceptOwnership() external;
+
+    /**
+     * @notice Assign the given address with the ability to pause the zone.
+     *
+     * @param pauserToAssign The address to assign the pauser role.
+     */
+    function assignPauser(address pauserToAssign) external;
+
+    /**
+     * @notice Assign the given address with the ability to operate the
+     *         given zone.
+     *
+     * @param pausableZoneAddress The zone address to assign operator role.
+     * @param operatorToAssign    The address to assign as operator.
+     */
+    function assignOperator(
+        address pausableZoneAddress,
+        address operatorToAssign
+    ) external;
+
+    /**
+     * @notice An external view function that returns the owner.
+     *
+     * @return The address of the owner.
+     */
+    function owner() external view returns (address);
+
+    /**
+     * @notice An external view function that returns the potential owner.
+     *
+     * @return The address of the potential owner.
+     */
+    function potentialOwner() external view returns (address);
+
+    /**
+     * @notice An external view function that returns the pauser.
+     *
+     * @return The address of the pauser.
+     */
+    function pauser() external view returns (address);
+}
diff --git a/contracts/seaport/zones/interfaces/PausableZoneEventsAndErrors.sol b/contracts/seaport/zones/interfaces/PausableZoneEventsAndErrors.sol
new file mode 100644
index 00000000..bb80f940
--- /dev/null
+++ b/contracts/seaport/zones/interfaces/PausableZoneEventsAndErrors.sol
@@ -0,0 +1,111 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+/**
+ * @notice PausableZoneEventsAndErrors contains errors and events
+ *         related to zone interaction.
+ */
+interface PausableZoneEventsAndErrors {
+    /**
+     * @dev Emit an event whenever a zone is successfully paused.
+     */
+    event Paused();
+
+    /**
+     * @dev Emit an event whenever a zone is successfully unpaused (created).
+     */
+    event Unpaused();
+
+    /**
+     * @dev Emit an event whenever a zone owner registers a new potential
+     *      owner for that zone.
+     *
+     * @param newPotentialOwner The new potential owner of the zone.
+     */
+    event PotentialOwnerUpdated(address newPotentialOwner);
+
+    /**
+     * @dev Emit an event whenever zone ownership is transferred.
+     *
+     * @param previousOwner The previous owner of the zone.
+     * @param newOwner      The new owner of the zone.
+     */
+    event OwnershipTransferred(address previousOwner, address newOwner);
+
+    /**
+     * @dev Emit an event whenever a new zone is created.
+     *
+     * @param zone The address of the zone.
+     * @param salt The salt used to deploy the zone.
+     */
+    event ZoneCreated(address zone, bytes32 salt);
+
+    /**
+     * @dev Emit an event whenever a zone owner assigns a new pauser
+     *
+     * @param newPauser The new pauser of the zone.
+     */
+    event PauserUpdated(address newPauser);
+
+    /**
+     * @dev Emit an event whenever a zone owner assigns a new operator
+     *
+     * @param newOperator The new operator of the zone.
+     */
+    event OperatorUpdated(address newOperator);
+
+    /**
+     * @dev Revert with an error when attempting to pause the zone
+     *      while the caller is not the owner or pauser of the zone.
+     */
+    error InvalidPauser();
+
+    /**
+     * @dev Revert with an error when attempting to call an operation
+     *      while the caller is not the controller or operator of the zone.
+     */
+    error InvalidOperator();
+
+    /**
+     * @dev Revert with an error when attempting to pause the zone or update the
+     *      operator while the caller is not the controller of the zone.
+     */
+    error InvalidController();
+    /**
+     * @dev Revert with an error when attempting to deploy a zone that is
+     *      currently deployed.
+     */
+    error ZoneAlreadyExists(address zone);
+
+    /**
+     * @dev Revert with an error when the caller does not have the _owner role
+     *
+     */
+    error CallerIsNotOwner();
+
+    /**
+     * @dev Revert with an error when the caller does not have the operator role
+     *
+     */
+    error CallerIsNotOperator();
+
+    /**
+     * @dev Revert with an error when attempting to set the new potential owner
+     *      as the 0 address.
+     *
+     */
+    error OwnerCanNotBeSetAsZero();
+
+    /**
+     * @dev Revert with an error when attempting to set the new potential pauser
+     *      as the 0 address.
+     *
+     */
+    error PauserCanNotBeSetAsZero();
+
+    /**
+     * @dev Revert with an error when the caller does not have
+     *      the potentialOwner role.
+     */
+    error CallerIsNotPotentialOwner();
+}
diff --git a/contracts/seaport/zones/interfaces/PausableZoneInterface.sol b/contracts/seaport/zones/interfaces/PausableZoneInterface.sol
new file mode 100644
index 00000000..19d5d76d
--- /dev/null
+++ b/contracts/seaport/zones/interfaces/PausableZoneInterface.sol
@@ -0,0 +1,102 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import {
+    SeaportInterface
+} from "seaport-types/src/interfaces/SeaportInterface.sol";
+
+import {
+    AdvancedOrder,
+    CriteriaResolver,
+    Execution,
+    Fulfillment,
+    Order,
+    OrderComponents
+} from "seaport-types/src/lib/ConsiderationStructs.sol";
+
+/**
+ * @title  PausableZone
+ * @author cupOJoseph, BCLeFevre, ryanio
+ * @notice PausableZone is a simple zone implementation that approves every
+ *         order. It can be self-destructed by its controller to pause
+ *         restricted orders that have it set as their zone.
+ */
+interface PausableZoneInterface {
+    /**
+     * @notice Cancel an arbitrary number of orders that have agreed to use the
+     *         contract as their zone.
+     *
+     * @param seaport  The Seaport address.
+     * @param orders   The orders to cancel.
+     *
+     * @return cancelled A boolean indicating whether the supplied orders have
+     *                   been successfully cancelled.
+     */
+    function cancelOrders(
+        SeaportInterface seaport,
+        OrderComponents[] calldata orders
+    ) external returns (bool cancelled);
+
+    /**
+     * @notice Execute an arbitrary number of matched orders, each with
+     *         an arbitrary number of items for offer and consideration
+     *         along with a set of fulfillments allocating offer components
+     *         to consideration components.
+     *
+     * @param seaport      The Seaport address.
+     * @param orders       The orders to match.
+     * @param fulfillments An array of elements allocating offer components
+     *                     to consideration components.
+     *
+     * @return executions An array of elements indicating the sequence of
+     *                    transfers performed as part of matching the given
+     *                    orders.
+     */
+    function executeMatchOrders(
+        SeaportInterface seaport,
+        Order[] calldata orders,
+        Fulfillment[] calldata fulfillments
+    ) external payable returns (Execution[] memory executions);
+
+    /**
+     * @notice Execute an arbitrary number of matched advanced orders,
+     *         each with an arbitrary number of items for offer and
+     *         consideration along with a set of fulfillments allocating
+     *         offer components to consideration components.
+     *
+     * @param seaport           The Seaport address.
+     * @param orders            The orders to match.
+     * @param criteriaResolvers An array where each element contains a reference
+     *                          to a specific order as well as that order's
+     *                          offer or consideration, a token identifier, and
+     *                          a proof that the supplied token identifier is
+     *                          contained in the order's merkle root.
+     * @param fulfillments      An array of elements allocating offer components
+     *                          to consideration components.
+     *
+     * @return executions An array of elements indicating the sequence of
+     *                    transfers performed as part of matching the given
+     *                    orders.
+     */
+    function executeMatchAdvancedOrders(
+        SeaportInterface seaport,
+        AdvancedOrder[] calldata orders,
+        CriteriaResolver[] calldata criteriaResolvers,
+        Fulfillment[] calldata fulfillments
+    ) external payable returns (Execution[] memory executions);
+
+    /**
+     * @notice Pause this contract, safely stopping orders from using
+     *         the contract as a zone. Restricted orders with this address as a
+     *         zone will not be fulfillable unless the zone is redeployed to the
+     *         same address.
+     */
+    function pause(address payee) external;
+
+    /**
+     * @notice Assign the given address with the ability to operate the zone.
+     *
+     * @param operatorToAssign The address to assign as the operator.
+     */
+    function assignOperator(address operatorToAssign) external;
+}
diff --git a/forks/README.md b/forks/README.md
new file mode 100644
index 00000000..84b22140
--- /dev/null
+++ b/forks/README.md
@@ -0,0 +1,10 @@
+
+# Forks
+
+This directory contains submodules pointing at repositories maintained by Immutable as forks of other smart contract projects. The `contracts` folders of these projects will be extracted by `yarn build` and merged into the main `contracts` folder.  
+
+To add a new fork, first add it into the `forks` folder as a git submodule. Then, add the configuration for the fork into `forks.json` file. 
+
+## Dependency Management
+
+If the forked repository has dependencies for their smart contracts, add them to the `dependencies` array.  
\ No newline at end of file
diff --git a/forks/forks.json b/forks/forks.json
new file mode 100644
index 00000000..9823ec16
--- /dev/null
+++ b/forks/forks.json
@@ -0,0 +1,25 @@
+[
+    {
+        "name": "passport",
+        "contracts": "passport/src/contracts",
+        "upstream": "https://github.com/immutable/wallet-contracts",
+        "ignore": [
+            "mocks"
+        ]
+    },
+    {
+        "name": "seaport",
+        "upstream": "https://github.com/immutable/seaport",
+        "contracts": "seaport/contracts",
+        "dependencies": [
+            "@openzeppelin/contracts",
+            "seaport-core",
+            "seaport-sol",
+            "seaport-types",
+            "solady"
+        ],
+        "ignore": [
+            "test"
+        ]
+    }
+]
\ No newline at end of file
diff --git a/forks/passport/.eslintignore b/forks/passport/.eslintignore
new file mode 100644
index 00000000..f0a62d69
--- /dev/null
+++ b/forks/passport/.eslintignore
@@ -0,0 +1,3 @@
+.eslintrc.js
+node_modules
+src/gen
diff --git a/forks/passport/.eslintrc.js b/forks/passport/.eslintrc.js
new file mode 100644
index 00000000..284d5a52
--- /dev/null
+++ b/forks/passport/.eslintrc.js
@@ -0,0 +1,38 @@
+const { off } = require("process")
+
+module.exports = {
+  parser: '@typescript-eslint/parser',
+  parserOptions: {
+    ecmaVersion: 2018,
+    sourceType: 'module'
+  },
+
+  extends: [
+    'plugin:@typescript-eslint/recommended',
+    'plugin:import/errors',
+    'plugin:import/warnings',
+    'plugin:import/typescript',
+    'prettier'
+  ],
+
+  rules: {
+    '@typescript-eslint/no-unused-vars': 'off',
+    '@typescript-eslint/no-explicit-any': 'off',
+    '@typescript-eslint/no-non-null-assertion': 'off',
+    '@typescript-eslint/explicit-module-boundary-types': 'off',
+    '@typescript-eslint/ban-types': 'off',
+    '@typescript-eslint/ban-ts-comment': 'off',
+    '@typescript-eslint/no-empty-function': 'off',
+    '@typescript-eslint/no-inferrable-types': 'off',
+    '@typescript-eslint/no-var-requires': 'off',
+    '@typescript-eslint/no-non-null-asserted-optional-chain': 'off',
+
+    'prefer-spread': 'off',
+    'prefer-const': 'off',
+
+    'import/no-unresolved': 'off',
+    // 'import/no-default-export': 2,
+    'import/no-named-as-default-member': 'off',
+
+  }
+}
diff --git a/forks/passport/.gitattributes b/forks/passport/.gitattributes
new file mode 100644
index 00000000..52031de5
--- /dev/null
+++ b/forks/passport/.gitattributes
@@ -0,0 +1 @@
+*.sol linguist-language=Solidity
diff --git a/forks/passport/.github/workflows/ci.yml b/forks/passport/.github/workflows/ci.yml
new file mode 100644
index 00000000..55a6692c
--- /dev/null
+++ b/forks/passport/.github/workflows/ci.yml
@@ -0,0 +1,158 @@
+
+on: [push]
+
+name: ci
+
+jobs:
+  install:
+    name: Install dependencies
+    runs-on: ubuntu-latest
+    steps:
+      - uses: actions/checkout@v1
+      - uses: actions/setup-node@v1
+        with:
+          node-version: 18.15
+      - uses: actions/cache@master
+        id: yarn-cache
+        with:
+          path: |
+            node_modules
+            */*/node_modules
+          key: ${{ runner.os }}-lerna-${{ hashFiles('**/package.json', '**/yarn.lock') }}
+      - run: yarn install --network-concurrency 1
+        if: ${{ steps.yarn-cache.outputs.cache-hit != 'true' }}
+
+  benchmark:
+    name: Benchmark
+    runs-on: ubuntu-latest
+    needs: [install]
+    steps:
+      - uses: actions/checkout@v1
+      - uses: actions/setup-node@v1
+        with:
+          node-version: 18.15
+      - uses: actions/cache@master
+        id: yarn-cache
+        with:
+          path: |
+            node_modules
+            */*/node_modules
+          key: ${{ runner.os }}-lerna-${{ hashFiles('**/package.json', '**/yarn.lock') }}
+      - run: yarn benchmark
+
+  lint-ts:
+    name: Typescript lint
+    runs-on: ubuntu-latest
+    needs: [install]
+    steps:
+      - uses: actions/checkout@v1
+      - uses: actions/setup-node@v1
+        with:
+          node-version: 18.15
+      - uses: actions/cache@master
+        id: yarn-cache
+        with:
+          path: |
+            node_modules
+            */*/node_modules
+          key: ${{ runner.os }}-lerna-${{ hashFiles('**/package.json', '**/yarn.lock') }}
+      - run: yarn lint:ts
+
+  lint-sol:
+    name: Solidity lint
+    runs-on: ubuntu-latest
+    needs: [install]
+    steps:
+      - uses: actions/checkout@v1
+      - uses: actions/setup-node@v1
+        with:
+          node-version: 18.15
+      - uses: actions/cache@master
+        id: yarn-cache
+        with:
+          path: |
+            node_modules
+            */*/node_modules
+          key: ${{ runner.os }}-lerna-${{ hashFiles('**/package.json', '**/yarn.lock') }}
+      - run: yarn lint:sol
+
+  test:
+    name: Test contracts
+    runs-on: ubuntu-latest
+    needs: [install]
+    steps:
+      - uses: actions/checkout@v1
+      - uses: actions/setup-node@v1
+        with:
+          node-version: 18.15
+      - uses: actions/cache@master
+        id: yarn-cache
+        with:
+          path: |
+            node_modules
+            */*/node_modules
+          key: ${{ runner.os }}-lerna-${{ hashFiles('**/package.json', '**/yarn.lock') }}
+      - run: yarn test
+
+  slither:
+    name: Slither test
+    runs-on: ubuntu-latest
+    needs: [install]
+    steps:
+      - uses: actions/checkout@v3
+      - uses: actions/setup-node@v1
+        with:
+          node-version: 18.15
+      - uses: actions/cache@master
+        id: yarn-cache
+        with:
+          path: |
+            node_modules
+            */*/node_modules
+          key: ${{ runner.os }}-lerna-${{ hashFiles('**/package.json', '**/yarn.lock') }}
+      - run: |
+          npx hardhat compile
+          cp -r ./src/artifacts/ ./artifacts/
+          cp -r ./src/contracts/ ./contracts/
+      - uses: crytic/slither-action@v0.3.0
+     
+  # coverage:
+  #   name: Coverage
+  #   runs-on: ubuntu-latest
+  #   needs: [install]
+  #   steps:
+  #     - uses: actions/checkout@v1
+  #     - uses: actions/setup-node@v1
+  #       with:
+  #         node-version: 18
+  #     - uses: actions/cache@master
+  #       id: yarn-cache
+  #       with:
+  #         path: |
+  #           node_modules
+  #           */*/node_modules
+  #         key: ${{ runner.os }}-lerna-${{ hashFiles('**/package.json', '**/yarn.lock') }}
+  #     - run: yarn coverage || true
+  #     - name: Coveralls
+  #       uses: coverallsapp/github-action@master
+  #       with:
+  #         github-token: ${{ secrets.GITHUB_TOKEN }}
+
+  build:
+    name: Is latest build commited?
+    runs-on: ubuntu-latest
+    needs: [install]
+    steps:
+      - uses: actions/checkout@v1
+      - uses: actions/setup-node@v1
+        with:
+          node-version: 18.15
+      - uses: actions/cache@master
+        id: yarn-cache
+        with:
+          path: |
+            node_modules
+            */*/node_modules
+          key: ${{ runner.os }}-lerna-${{ hashFiles('**/package.json', '**/yarn.lock') }}
+      - run: yarn test
+      - run: git diff --exit-code
diff --git a/forks/passport/.gitignore b/forks/passport/.gitignore
new file mode 100644
index 00000000..c4c15775
--- /dev/null
+++ b/forks/passport/.gitignore
@@ -0,0 +1,26 @@
+.vscode
+node_modules/
+
+npm-debug.log*
+yarn-debug.log*
+yarn-error.log*
+lerna-debug.log*
+
+*.js.map
+build
+coverage
+coverage.json
+
+config/*.env
+
+src/gen
+src/artifacts
+
+# MAC OS custom directory attributes file.
+.DS_Store
+
+# IntelliJ IDE directory
+.idea
+*.iml
+
+.env
\ No newline at end of file
diff --git a/forks/passport/.prettierrc b/forks/passport/.prettierrc
new file mode 100644
index 00000000..421afa97
--- /dev/null
+++ b/forks/passport/.prettierrc
@@ -0,0 +1,9 @@
+{
+  "tabWidth": 2,
+  "useTabs": false,
+  "semi": false,
+  "singleQuote": true,
+  "trailingComma": "none",
+  "arrowParens": "avoid",
+  "printWidth": 130
+}
diff --git a/forks/passport/.solcover.js b/forks/passport/.solcover.js
new file mode 100644
index 00000000..fab8bc0e
--- /dev/null
+++ b/forks/passport/.solcover.js
@@ -0,0 +1,3 @@
+module.exports = {
+  skipFiles: ['mocks', 'migrations']
+}
diff --git a/forks/passport/.solhint.json b/forks/passport/.solhint.json
new file mode 100644
index 00000000..efa775ff
--- /dev/null
+++ b/forks/passport/.solhint.json
@@ -0,0 +1,24 @@
+{
+  "extends": "solhint:recommended",
+  "rules": {
+    "quotes": "off",
+    "const-name-snakecase": "off",
+    "contract-name-camelcase": "off",
+    "event-name-camelcase": "off",
+    "func-name-mixedcase": "off",
+    "func-param-name-mixedcase": "off",
+    "modifier-name-mixedcase": "off",
+    "private-vars-leading-underscore": "off",
+    "use-forbidden-name": "off",
+    "var-name-mixedcase": "off",
+    "func-order": "off",
+    "imports-on-top": "off",
+    "ordering": "off",
+    "no-unused-vars": "off",
+    "no-inline-assembly": "off",
+    "visibility-modifier-order": "off",
+    "compiler-version": ["warn", "0.8.17"],
+    "func-visibility": ["warn", {"ignoreConstructors":true}],
+    "reason-string": ["warn", {"maxLength": 96}]
+  }
+}
diff --git a/forks/passport/CONTRIBUTING.md b/forks/passport/CONTRIBUTING.md
new file mode 100644
index 00000000..b7ea09fa
--- /dev/null
+++ b/forks/passport/CONTRIBUTING.md
@@ -0,0 +1,79 @@
+# Contributing to Immutable Wallet Contracts
+
+We greatly appreciate all community feedback and contributions to this repository. This guide will walk you through the different ways you can contribute, including opening GitHub issues, pull requests, requesting features, and providing general feedback. For any security disclosures, please email security@immutable.com.
+
+## Opening an issue
+
+GitHub issues are a great way to discuss and coordinate new features, changes, bugs, etc.
+
+You should create an issue to open discussions when:
+
+- **Before starting development on a significant pull request** - Use GitHub issues as a way to collaboratively propose, design, and discuss any development work you plan to do.
+- **Proposing new features or ideas** - If you just have an idea or feature you'd like to see, drop the details into a GitHub issue for us to keep track of. Be specific with any requests, and include descriptions of what you would like to be added and what problems it would solve.
+- **Reporting a bug or minor security vulnerability** - As above, opening an issue helps us keep track of and prioritize any bugs or minor security vulnerabilities. Make sure to include all the relevant information, e.g. code snippets, error messages, stack traces, environment, reproduction steps, etc, to help us effectively diagnose and address the issue. Please also reach out to security@immutable.com.
+
+There is usually no need to create an issue when:
+
+- **An issue or open PR for the topic already exists** - Please continue or reopen the existing thread to keep all the discussion in one place.
+- **The PR is relatively small and obvious (e.g. small fixes, typos, etc)** - If the changes do not require a discussion, simply filling out the PR description template will suffice.
+- **Providing comments or general feedback** - Please use GitHub issues for actionable feedback or tasks (e.g. feature requests or bugs).
+
+## Opening a Pull Request
+
+If you'd like to contribute to the code and open a pull request (PR), here are some general guidelines to follow:
+
+- Most PRs should have an accompanying GitHub issue (see above on opening issues). When submitting your PR for review, please link the corresponding issue in the PR description.
+- Alongside every PR there should be a brief summary of what changes have been made and why, as per the PR template we provide. Be sure to fill this out to help share context with the reviewers and others.
+- Please follow existing code styles to keep formatting in this repository consistent.
+
+### Contributor workflow
+
+To start contributing to this repository, have a look at this guide to contributing to open source projects.
+
+You should fork the repository and make your changes in your local repository under a new branch, before opening a pull request for your branch back upstream to the original repository. This is a helpful guide that walks through all of these steps in detail.
+
+1. Fork the repository (Fork button in the top right of the repository GitHub UI).
+2. Clone your fork to your local machine.
+
+```
+git clone https://GitHub.com/<your-GitHub-username>/<repository-name>.git
+```
+
+3. Add the original repository as the upstream remote.
+
+```
+cd <repository-name>
+git remote add upstream https://GitHub.com/immutable/<repository-name>.git
+```
+
+4. Make sure your fork is up to date with the original repository (if necessary).
+
+```
+git pull upstream main
+```
+
+5. Create a new branch and make your changes.
+
+```
+git checkout -b <your-branch-name>
+```
+
+6. Be sure to run the tests and set up the relevant linters to ensure all GitHub checks pass (see GitHub issues: https://docs.github.com/en/issues/tracking-your-work-with-issues/about-issues for more information).
+
+```
+npm test
+```
+
+7. Add and commit your changes, including a comprehensive commit message summarising your changes, then push changes to your fork. (e.g. Fixed formatting issue with ImmutableERC721PermissionedMintable.sol)
+
+```
+git add *
+git commit -m "fix typos"
+git push origin <your-branch-name>
+```
+
+8. Open a pull request into the original repository through GitHub in your web browser. Remember to fill out the PR template to the best of your ability to share any context with reviewers.
+
+**IMPORTANT**: Please make sure to read through the [cla](cla.txt) file as part of opening a pull request.
+
+9. We will review the pull requests and request any necessary changes. If all the checks (linting, compilation, tests) pass and everything looks good, your code will be merged into the original repository. Congratulations, and thank you for your contribution!
diff --git a/forks/passport/LICENSE b/forks/passport/LICENSE
new file mode 100644
index 00000000..a4b287cb
--- /dev/null
+++ b/forks/passport/LICENSE
@@ -0,0 +1,220 @@
+   Copyright (c) 2023-present Immutable Pty Ltd.
+   Copyright (c) 2017-present Horizon Blockchain Games Inc.
+
+   Licensed under the Apache License, Version 2.0 (the "License");
+   you may not use this file except in compliance with the License.
+   You may obtain a copy of the License at
+
+       http://www.apache.org/licenses/LICENSE-2.0
+
+   Unless required by applicable law or agreed to in writing, software
+   distributed under the License is distributed on an "AS IS" BASIS,
+   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+   See the License for the specific language governing permissions and
+   limitations under the License.
+
+
+   ------------------------------------------------------------------------
+
+
+                                 Apache License
+                           Version 2.0, January 2004
+                        http://www.apache.org/licenses/
+
+   TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
+
+   1. Definitions.
+
+      "License" shall mean the terms and conditions for use, reproduction,
+      and distribution as defined by Sections 1 through 9 of this document.
+
+      "Licensor" shall mean the copyright owner or entity authorized by
+      the copyright owner that is granting the License.
+
+      "Legal Entity" shall mean the union of the acting entity and all
+      other entities that control, are controlled by, or are under common
+      control with that entity. For the purposes of this definition,
+      "control" means (i) the power, direct or indirect, to cause the
+      direction or management of such entity, whether by contract or
+      otherwise, or (ii) ownership of fifty percent (50%) or more of the
+      outstanding shares, or (iii) beneficial ownership of such entity.
+
+      "You" (or "Your") shall mean an individual or Legal Entity
+      exercising permissions granted by this License.
+
+      "Source" form shall mean the preferred form for making modifications,
+      including but not limited to software source code, documentation
+      source, and configuration files.
+
+      "Object" form shall mean any form resulting from mechanical
+      transformation or translation of a Source form, including but
+      not limited to compiled object code, generated documentation,
+      and conversions to other media types.
+
+      "Work" shall mean the work of authorship, whether in Source or
+      Object form, made available under the License, as indicated by a
+      copyright notice that is included in or attached to the work
+      (an example is provided in the Appendix below).
+
+      "Derivative Works" shall mean any work, whether in Source or Object
+      form, that is based on (or derived from) the Work and for which the
+      editorial revisions, annotations, elaborations, or other modifications
+      represent, as a whole, an original work of authorship. For the purposes
+      of this License, Derivative Works shall not include works that remain
+      separable from, or merely link (or bind by name) to the interfaces of,
+      the Work and Derivative Works thereof.
+
+      "Contribution" shall mean any work of authorship, including
+      the original version of the Work and any modifications or additions
+      to that Work or Derivative Works thereof, that is intentionally
+      submitted to Licensor for inclusion in the Work by the copyright owner
+      or by an individual or Legal Entity authorized to submit on behalf of
+      the copyright owner. For the purposes of this definition, "submitted"
+      means any form of electronic, verbal, or written communication sent
+      to the Licensor or its representatives, including but not limited to
+      communication on electronic mailing lists, source code control systems,
+      and issue tracking systems that are managed by, or on behalf of, the
+      Licensor for the purpose of discussing and improving the Work, but
+      excluding communication that is conspicuously marked or otherwise
+      designated in writing by the copyright owner as "Not a Contribution."
+
+      "Contributor" shall mean Licensor and any individual or Legal Entity
+      on behalf of whom a Contribution has been received by Licensor and
+      subsequently incorporated within the Work.
+
+   2. Grant of Copyright License. Subject to the terms and conditions of
+      this License, each Contributor hereby grants to You a perpetual,
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+      copyright license to reproduce, prepare Derivative Works of,
+      publicly display, publicly perform, sublicense, and distribute the
+      Work and such Derivative Works in Source or Object form.
+
+   3. Grant of Patent License. Subject to the terms and conditions of
+      this License, each Contributor hereby grants to You a perpetual,
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+      (except as stated in this section) patent license to make, have made,
+      use, offer to sell, sell, import, and otherwise transfer the Work,
+      where such license applies only to those patent claims licensable
+      by such Contributor that are necessarily infringed by their
+      Contribution(s) alone or by combination of their Contribution(s)
+      with the Work to which such Contribution(s) was submitted. If You
+      institute patent litigation against any entity (including a
+      cross-claim or counterclaim in a lawsuit) alleging that the Work
+      or a Contribution incorporated within the Work constitutes direct
+      or contributory patent infringement, then any patent licenses
+      granted to You under this License for that Work shall terminate
+      as of the date such litigation is filed.
+
+   4. Redistribution. You may reproduce and distribute copies of the
+      Work or Derivative Works thereof in any medium, with or without
+      modifications, and in Source or Object form, provided that You
+      meet the following conditions:
+
+      (a) You must give any other recipients of the Work or
+          Derivative Works a copy of this License; and
+
+      (b) You must cause any modified files to carry prominent notices
+          stating that You changed the files; and
+
+      (c) You must retain, in the Source form of any Derivative Works
+          that You distribute, all copyright, patent, trademark, and
+          attribution notices from the Source form of the Work,
+          excluding those notices that do not pertain to any part of
+          the Derivative Works; and
+
+      (d) If the Work includes a "NOTICE" text file as part of its
+          distribution, then any Derivative Works that You distribute must
+          include a readable copy of the attribution notices contained
+          within such NOTICE file, excluding those notices that do not
+          pertain to any part of the Derivative Works, in at least one
+          of the following places: within a NOTICE text file distributed
+          as part of the Derivative Works; within the Source form or
+          documentation, if provided along with the Derivative Works; or,
+          within a display generated by the Derivative Works, if and
+          wherever such third-party notices normally appear. The contents
+          of the NOTICE file are for informational purposes only and
+          do not modify the License. You may add Your own attribution
+          notices within Derivative Works that You distribute, alongside
+          or as an addendum to the NOTICE text from the Work, provided
+          that such additional attribution notices cannot be construed
+          as modifying the License.
+
+      You may add Your own copyright statement to Your modifications and
+      may provide additional or different license terms and conditions
+      for use, reproduction, or distribution of Your modifications, or
+      for any such Derivative Works as a whole, provided Your use,
+      reproduction, and distribution of the Work otherwise complies with
+      the conditions stated in this License.
+
+   5. Submission of Contributions. Unless You explicitly state otherwise,
+      any Contribution intentionally submitted for inclusion in the Work
+      by You to the Licensor shall be under the terms and conditions of
+      this License, without any additional terms or conditions.
+      Notwithstanding the above, nothing herein shall supersede or modify
+      the terms of any separate license agreement you may have executed
+      with Licensor regarding such Contributions.
+
+   6. Trademarks. This License does not grant permission to use the trade
+      names, trademarks, service marks, or product names of the Licensor,
+      except as required for reasonable and customary use in describing the
+      origin of the Work and reproducing the content of the NOTICE file.
+
+   7. Disclaimer of Warranty. Unless required by applicable law or
+      agreed to in writing, Licensor provides the Work (and each
+      Contributor provides its Contributions) on an "AS IS" BASIS,
+      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+      implied, including, without limitation, any warranties or conditions
+      of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
+      PARTICULAR PURPOSE. You are solely responsible for determining the
+      appropriateness of using or redistributing the Work and assume any
+      risks associated with Your exercise of permissions under this License.
+
+   8. Limitation of Liability. In no event and under no legal theory,
+      whether in tort (including negligence), contract, or otherwise,
+      unless required by applicable law (such as deliberate and grossly
+      negligent acts) or agreed to in writing, shall any Contributor be
+      liable to You for damages, including any direct, indirect, special,
+      incidental, or consequential damages of any character arising as a
+      result of this License or out of the use or inability to use the
+      Work (including but not limited to damages for loss of goodwill,
+      work stoppage, computer failure or malfunction, or any and all
+      other commercial damages or losses), even if such Contributor
+      has been advised of the possibility of such damages.
+
+   9. Accepting Warranty or Additional Liability. While redistributing
+      the Work or Derivative Works thereof, You may choose to offer,
+      and charge a fee for, acceptance of support, warranty, indemnity,
+      or other liability obligations and/or rights consistent with this
+      License. However, in accepting such obligations, You may act only
+      on Your own behalf and on Your sole responsibility, not on behalf
+      of any other Contributor, and only if You agree to indemnify,
+      defend, and hold each Contributor harmless for any liability
+      incurred by, or claims asserted against, such Contributor by reason
+      of your accepting any such warranty or additional liability.
+
+   END OF TERMS AND CONDITIONS
+
+   APPENDIX: How to apply the Apache License to your work.
+
+      To apply the Apache License to your work, attach the following
+      boilerplate notice, with the fields enclosed by brackets "[]"
+      replaced with your own identifying information. (Don't include
+      the brackets!)  The text should be enclosed in the appropriate
+      comment syntax for the file format. We also recommend that a
+      file or class name and description of purpose be included on the
+      same "printed page" as the copyright notice for easier
+      identification within third-party archives.
+
+   Copyright [yyyy] [name of copyright owner]
+
+   Licensed under the Apache License, Version 2.0 (the "License");
+   you may not use this file except in compliance with the License.
+   You may obtain a copy of the License at
+
+       http://www.apache.org/licenses/LICENSE-2.0
+
+   Unless required by applicable law or agreed to in writing, software
+   distributed under the License is distributed on an "AS IS" BASIS,
+   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+   See the License for the specific language governing permissions and
+   limitations under the License.
diff --git a/forks/passport/README.md b/forks/passport/README.md
new file mode 100644
index 00000000..64b57ee1
--- /dev/null
+++ b/forks/passport/README.md
@@ -0,0 +1,83 @@
+# Immutable Wallet Contracts
+
+This is a fork of of Sequence's [wallet-contracts repo](https://github.com/0xsequence/wallet-contracts).
+
+## Immutable Changes
+
+The following changes have been made:
+
+- The minimal wallet proxy contract (bytecode of which is in
+  src/contracts/Wallet.sol) now supports a function `PROXY_getImplementation` to
+  return the address of the implementation contract pointed to by the proxy
+  contract.
+- The interface for the `PROXY_getImplementation` function is defined in
+  src/contracts/IWalletProxy.sol.
+- The source code for the new bytecode is in `src/contracts/WalletProxy.yul`.
+- All references to the bytecode of the proxy now use the definition in
+  `src/contracts/Wallet.sol` (rather than having their own definition).
+- `StartupWalletImpl.sol` is a minimal initial implementation that updates the
+  implementation address and then delegate calls to the wallet implementation.
+  This contract determines the latest wallet implementation contract to use by
+  contacting the new contract `LatestWalletImplLocator.sol`.
+- `MainModuleDynamicAuth.sol` which uses `ModuleAuth.sol` is a type of wallet
+  implementation that initially does authentication by checking that the
+  contract address matches being generated using the image hash, the factory
+  contract, and the proxy start-up bytes (which include the address of the
+  wallet implementation contract: the instance of `StartupWalletImpl`). The
+  image hash is stored and subsequence transactions verify by comparing the
+  calculated image hash with the stored image hash.
+- Tests for the new wallet factory features have been included in
+  `ImmutableFactory.spec.ts`.
+- Tests for the startup and dynamic authentication have been included in
+  `ImmutableStartup.spec.ts`.
+- Tests for the Immutable deployment have been included in
+  `ImmutableDeployment.spec.ts`.
+
+## Security Review
+
+`@imtbl/wallet-contracts` has been audited by three independant parties
+
+- [Consensys Diligence](./audits/Consensys_Diligence.md) - May 2020
+- [Quantstamp - initial audit](./audits/Quantstamp_Arcadeum_Report_Final.pdf) - July 2020
+- [Quantstamp - audit of new capability, nested Sequence signers](./audits/sequence_quantstamp_audit_feb_2021.pdf) - Feb 2021
+- [Halborm - audit of security improvements and customisations for Immutable environment](./audits/202309_Halborn_Final.pdf) - Sept 2023. 
+See [background information for this audit](./audits/202309_audit_background.md). 
+
+## Dev env & release
+
+This repository is configured as a yarn workspace, and has multiple pacakge.json
+files. Specifically, we have the root ./package.json for the development
+environment, contract compilation and testing. Contract source code and
+distribution files are packaged in "src/package.json".
+
+To release a new version, make sure to bump the version, tag it, and run `yarn
+release`. The `release` command will publish the `@imtbl/wallet-contracts`
+package in the "src/" folder, separate from the root package.
+
+## Contribution
+
+We aim to build robust and feature-rich standards to help all developers onboard
+and build their projects on Immuable zkEVM, and we welcome any and all feedback
+and contributions to this repository! See our [contribution
+guideline](CONTRIBUTING.md) for more details on opening Github issues, pull
+requests requesting features, minor security vulnerabilities and providing
+general feedback.
+
+## Disclaimers
+
+These contracts are in an experimental stage and are subject to change without
+notice. The code must still be formally audited or reviewed and may have
+security vulnerabilities. Do not use it in production. We take no responsibility
+for your implementation decisions and any security problems you might
+experience.
+
+We will audit these contracts before our mainnet launch.
+
+## Security
+
+Please see our [security.md](security.md)  page for information on how to report security issues.
+
+## License
+
+Immutable zkEVM Contracts are released under the Apache-2.0 license. See
+[LICENSE.md](LICENSE.md) for more details.
diff --git a/forks/passport/audits/202309-audit-background-diagrams/address-retrieval.jpg b/forks/passport/audits/202309-audit-background-diagrams/address-retrieval.jpg
new file mode 100644
index 00000000..c6220d9b
Binary files /dev/null and b/forks/passport/audits/202309-audit-background-diagrams/address-retrieval.jpg differ
diff --git a/forks/passport/audits/202309-audit-background-diagrams/data-flow.jpg b/forks/passport/audits/202309-audit-background-diagrams/data-flow.jpg
new file mode 100644
index 00000000..006bf55d
Binary files /dev/null and b/forks/passport/audits/202309-audit-background-diagrams/data-flow.jpg differ
diff --git a/forks/passport/audits/202309-audit-background-diagrams/factory-deployment.jpg b/forks/passport/audits/202309-audit-background-diagrams/factory-deployment.jpg
new file mode 100644
index 00000000..f0a3192e
Binary files /dev/null and b/forks/passport/audits/202309-audit-background-diagrams/factory-deployment.jpg differ
diff --git a/forks/passport/audits/202309-audit-background-diagrams/high-level-diagram.jpg b/forks/passport/audits/202309-audit-background-diagrams/high-level-diagram.jpg
new file mode 100644
index 00000000..b17fa224
Binary files /dev/null and b/forks/passport/audits/202309-audit-background-diagrams/high-level-diagram.jpg differ
diff --git a/forks/passport/audits/202309-audit-background-diagrams/immutable-signer.png b/forks/passport/audits/202309-audit-background-diagrams/immutable-signer.png
new file mode 100644
index 00000000..87bdfd22
Binary files /dev/null and b/forks/passport/audits/202309-audit-background-diagrams/immutable-signer.png differ
diff --git a/forks/passport/audits/202309-audit-background-diagrams/incremental-deployment.gif b/forks/passport/audits/202309-audit-background-diagrams/incremental-deployment.gif
new file mode 100644
index 00000000..df109998
Binary files /dev/null and b/forks/passport/audits/202309-audit-background-diagrams/incremental-deployment.gif differ
diff --git a/forks/passport/audits/202309-audit-background-diagrams/multicall-deploy.png b/forks/passport/audits/202309-audit-background-diagrams/multicall-deploy.png
new file mode 100644
index 00000000..1221aea7
Binary files /dev/null and b/forks/passport/audits/202309-audit-background-diagrams/multicall-deploy.png differ
diff --git a/forks/passport/audits/202309-audit-background-diagrams/startup-wallet.png b/forks/passport/audits/202309-audit-background-diagrams/startup-wallet.png
new file mode 100644
index 00000000..0097b896
Binary files /dev/null and b/forks/passport/audits/202309-audit-background-diagrams/startup-wallet.png differ
diff --git a/forks/passport/audits/202309-audit-background-diagrams/wallet-deployment.jpg b/forks/passport/audits/202309-audit-background-diagrams/wallet-deployment.jpg
new file mode 100644
index 00000000..14c1d527
Binary files /dev/null and b/forks/passport/audits/202309-audit-background-diagrams/wallet-deployment.jpg differ
diff --git a/forks/passport/audits/202309-audit-background-diagrams/wallet-interactions.jpg b/forks/passport/audits/202309-audit-background-diagrams/wallet-interactions.jpg
new file mode 100644
index 00000000..a49f903d
Binary files /dev/null and b/forks/passport/audits/202309-audit-background-diagrams/wallet-interactions.jpg differ
diff --git a/forks/passport/audits/202309_Halborn_Final.pdf b/forks/passport/audits/202309_Halborn_Final.pdf
new file mode 100644
index 00000000..6fabd332
Binary files /dev/null and b/forks/passport/audits/202309_Halborn_Final.pdf differ
diff --git a/forks/passport/audits/202309_audit_background.md b/forks/passport/audits/202309_audit_background.md
new file mode 100644
index 00000000..9615cc4b
--- /dev/null
+++ b/forks/passport/audits/202309_audit_background.md
@@ -0,0 +1,401 @@
+# Immutable Wallet Contracts
+
+As part of building the [Immutable Passport][passport] wallet, we are implementing the generation of counterfactual smart contract wallet (SCW) addresses and the infrastructure that will eventually deploy those wallets to EVM networks.
+
+[passport]: https://www.immutable.com/products/passport
+
+The need for SCWs is driven by wanting to make accounts more secure in the case of externally owned accounts private key exfiltration on mobile platforms through malicious apps embedding Passport wallets through WebViews. The extra security comes from the wallets being multisig and verifying transactions using signatures from two parties: Immutable itself and the user. In this case, even when the user key is compromised, exfiltration of funds is still not possible.
+
+Immutable signs transactions after an OTP 2FA, which translates to 2FA security for the user assets. The wallets are counterfactual because we use a StarkEx L2 and the users might not need to interact with the address on EVM networks unless they are bridging their assets. This is, however, not the only use case, as we also want to support users native to zkEVM networks (or other EVM-compatible networks) not only StarkEx [^1].
+
+[^1]:
+    One caveat we have identified is that different theoretically
+    EVM-compatible networks have slightly different `CREATE2` behavior, which
+    could lead to an user having different addresses in different networks.
+
+To implement this, we have modified the [0xSequence's smart contract wallet][0xSequence] to include the following features:
+
+- Modified [Factory contract][upstream-factory-contract] with better
+  counterfactual addresses support.
+- New [MainModule][main-module] to support wallets which can update their list
+  of signers.
+- New [StartupWallet][startup-wallet] which acts as a temporary main wallet
+  module placeholder that finds the latest implementation at runtime during the
+  execution of the first transaction.
+- A [MultiCall contract][multicall-contract] that enables the SCW deployment to
+  be bundled with its first transaction.
+- [ImmutableSigner][immutable-signer]: a simplified SCW that can be used as a
+  signer for the main 0xSequence wallet, enabling key rotation without changing
+  the main wallet signer.
+- An improved [Proxy contract][proxy-contract] that is gas-efficient and enables
+  onchain implementation address retrieval.
+
+A more exhaustive list of changes can be found in the [README.md](README.md).
+
+[0xSequence]: https://github.com/0xsequence/wallet-contracts
+[upstream-factory-contract]: https://github.com/0xsequence/wallet-contracts/blob/master/src/contracts/Factory.sol
+[main-module]: https://github.com/immutable/wallet-contracts/blob/main/src/contracts/modules/MainModuleDynamicAuth.sol
+[startup-wallet]: https://github.com/immutable/wallet-contracts/blob/main/src/contracts/startup/StartupWalletImpl.sol
+[multicall-contract]: https://github.com/immutable/wallet-contracts/blob/main/src/contracts/MultiCallDeploy.sol
+[immutable-signer]: https://github.com/immutable/wallet-contracts/blob/main/src/contracts/signer/ImmutableSigner.sol
+[proxy-contract]: https://github.com/immutable/wallet-contracts/blob/main/src/contracts/WalletProxy.yul
+
+Since the implementations of all of our contracts were not available at the time of CFA account generation, we also use reserved nonces to decouple deployment from contract implementation[^2].
+
+[^2]:
+    Since `CREATE` depends on nonce and deployer address, implementation does
+    not change the address.
+
+## Previous Audits
+
+The changes we are making to 0xSequence’s contracts are relatively small when compared to the scope of the whole set of contracts, and work within the framework defined by the original contracts. As a result, previous audits to 0xSequence’s contracts might be useful as both context and starting point for auditing our changes. Those are available in 0xSequence’s repository under the [audits][audits] directory.
+
+[audits]: https://github.com/0xsequence/wallet-contracts/tree/master/audits
+
+## Development
+
+Instructions for how to run and test the repo are unchanged from upstream and can be found in our repository. Check the “Dev env & release” section in the README file.
+
+It’s a yarn-managed repository, with the most important commands being:
+
+- `yarn build`
+- `yarn test`
+
+Respectively building contract artifacts and running the test suites.
+
+## Architecture
+
+![High Level Diagram](./202309-audit-background-diagrams/high-level-diagram.jpg 'Interactions between the different contracts in this repo')
+
+The core functionality of the wallet is executing transactions. It also makes sure that:
+
+- The deployment and the first meta transaction of a wallet are executed as a
+  single root transaction.
+- The deployed wallet implementation is the latest at the time of deployment.
+- The wallet can modify its signers, given authorization from the previous
+  signers.
+- Immutable is one of the signers in all wallets, and is able to rotate its key
+  without changing the signer address.
+
+All of those constraints are displayed in the above diagram. The first ever
+transaction for a smart contract wallet would:
+
+1. An EOA with `EXECUTOR_ROLE` calls the `MultiCallDeploy.sol` contract to deploy
+   the user wallet and execute a transaction.
+2. `Factory.sol` deploys an instance of `WalletProxy.yul` to a deterministic
+   address (the user wallet address) based on its own address, the
+   `StartupWalletImpl.sol` address and the signers to the user SCW.
+3. `MultiCallDeploy.sol` calls the `execute()` function on the
+   `WalletProxy.yul`, which will delegate the call to the `StartupWalletImpl.sol`.
+4. `StartupWalletImpl.sol` then finds the most recent wallet implementation
+   address (`MainModuleDynamicAuth.sol`) using `LatestWalletImplLocator.sol`,
+   updates the proxy to point to the implementation in subsequent calls, and
+   delegates the call again to `MainModuleDynamicAuth.sol`.
+5. `MainModuleDynamicAuth.sol` verifies the singnature in the transaction by
+   checking the recovered signer against the user EOA and by calling
+   `ImmutableSigner.sol` (which is the second signer in the multisig).
+6. The transaction gets executed.
+
+In subsequent transactions the flow is much simpler:
+
+1. An EOA with `EXECUTOR_ROLE` calls the `MultiCallDeploy.sol` to execute a
+   transaction.
+2. `MultiCallDeploy.sol` calls the `execute()` function on the
+   `WalletProxy.yul`, which will delegate the call to the
+   `MainModuleDynamicAuth.sol`.
+3. `MainModuleDynamicAuth.sol` verifies the singnature in the transaction by
+   checking the recovered signer against the user EOA and by calling
+   `ImmutableSigner.sol` (which is the second signer in the multisig).
+4. The transaction gets executed.
+
+> 🔀 **Alternate Flow**
+>
+> Alternatively, subsequent transactions could also fully skip the
+> `MultiCallDeploy.sol` contract for improved gas usage, although this would
+> require knowledge if `WalletProxy.yul` for the specific wallet in question is
+> deployed or not:
+>
+> 1. An EOA calls `execute()` on `WalletProxy.yul` which delegates to
+>    `MainModuleDynamicAuth.sol`
+> 2. `MainModuleDynamicAuth.sol` verifies the singnature in the transaction by
+>    checking the recovered signer against the user EOA and by calling
+>    `ImmutableSigner.sol` (which is the second signer in the multisig).
+> 3. The transaction gets executed.
+
+## Incremental Deployment
+
+Since we are building counterfactual infrastructure, it has several levels of materialization, i.e.: it’s incrementally deployed. These levels are:
+
+- **(a)** Nothing is deployed yet, all contract addresses involved are counterfactual[^3].
+- **(b)** Core infrastructure and wallet implementation is deployed.
+- **(c)** User SCW is deployed during the execution of its first transaction.
+
+![Incremental Deployment](./202309-audit-background-diagrams/incremental-deployment.gif 'Contracts are materialized in sequential and incremental steps')
+
+[^3]:
+    One requirement is that we must be able to generate user SCW addresses in
+    this state without executing deployments and they should not change after
+    deployment.
+
+## Wallet Factory
+
+The wallet factory contract interacts with other contracts and externally owned accounts, namely:
+
+- EOAs[^4]
+  - Factory Deployment: deploys the wallet factory contract.
+  - Wallet Deployment: calls deploy() on the factory contract.
+  - Admin: can grant and revoke "Wallet Deployment" permissions to addresses.
+- Contracts
+  - Wallet Proxies
+  - StartupWalletImpl
+
+[^4]:
+    Note that we support both cases where those are the same EOA or different
+    EOAs.
+
+![Data Flow](./202309-audit-background-diagrams/data-flow.jpg)
+
+The wallet factory contract is a modified version of the 0xSequence factory contract, and the StartupWalletImpl contract takes care of updating the proxy to point to the latest wallet implementation address and delegating to that address.
+
+Upgradeability of the wallet implementation is achieved by having the wallet main module contract (i.e.: the wallet implementation) update the address used by the wallet proxy (which is possible because the wallet proxy uses a delegate call).
+
+Upgradeability of signers is achieved by storing a hash of the signers in the wallet storage and allowing the wallet to call itself to change the stored value.
+
+Although not directly relevant to the scope of our changes, these are the planned interactions with the wallet contract itself:
+
+![Wallet Interactions](./202309-audit-background-diagrams/wallet-interactions.jpg)
+
+The wallet will be a 2/2 multisign where Immutable Passport users hold one key and Immutable holds another (Immutable uses a SCW as its signer). The user signs a meta transaction and sends it to passport services, where the second signature will be added and the transaction will be sent to onchain nodes through the “Gas Funds EOA.” The execution runtime will then check the user signature using their address and the Immutable Signature by calling Immutable’s SCW.
+
+## Public Interface
+
+The [wallet factory contract][wallet-factory] has two public methods, both of them taking in the same arguments:
+
+- `function getAddress(address _mainModule, bytes32 _salt)`
+- `function deploy(address _mainModule, bytes32 _salt)`
+
+`getAddress()` is a view function that generates the address for a smart contract wallet before the wallet is deployed, so it can be used as a counterfactual address by Passport. Our functional requirements were:
+
+- Given the same parameters, should return the same address deploy() will actually create the proxy to the wallet on.
+
+`deploy()` deploys a proxy to a smart contract wallet. This method was updated from upstream to support new requirements:
+
+- Should revert if deployment fails.
+- Should only be accessible by the `DEPLOYER_ROLE` role.
+- Should emit `WalletDeployed(_contract, _mainModule, _salt)` if the wallet proxy is successfully deployed.
+
+[wallet-factory]: https://github.com/immutable/wallet-contracts/blob/main/src/contracts/Factory.sol#L11
+
+## Salt and Weights
+
+All our `salt` parameters will be generated using the standard [image hash generation algorithm][image-hash-generation] from 0xSequence, and will always have a `threshold` of 2, and two signers: Immutable with `weight` of 1 and the user EOA with a `weight` of 1. Addresses are sorted to keep deterministic behavior.
+
+```js
+keccak256(abi.encode(keccak256(abi.encode(bytes32(uint256(2)), uint256(1), address1)), uint256(1), address2))
+```
+
+[image-hash-generation]: https://github.com/immutable/wallet-contracts/blob/main/tests/utils/helpers.ts#L370
+
+## External Interactions
+
+At a system level, there’s a single actor interacting with the factory contract: the Passport wallet offchain services. The offchain services are triggered by the end user through interactions with the wallet frontend. As shown in the [architecture diagram](#wallet-factory), Passport services might be implemented using a few different EOAs.
+
+Here we walk through the end to end user journeys to give some extra context.
+
+### Display the SCW address
+
+- The user indicates they want to check their EVM SCW address.
+- The Passport Wallet backend will calculate the salt for the user given the user EOA public key and Immutable’s EOA public key. This will be done offchain using the [image hash generation algorithm](#salt-and-weights).
+- The Passport Wallet backend will call `getAddress()` passing in the startup contract address (stored in the offchain database after that contract is deployed) and the generated salt[^5].
+- The Factory contract will calculate the address and return it.
+- Passport will display the address to the user.
+
+[^5]: Since `getAddress()` is a pure function, this flow can optionally be implemented offchain without calling the contract.
+
+![Address Retrieval](./202309-audit-background-diagrams/address-retrieval.jpg)
+
+### Execute Transaction with the undeployed SCW
+
+1. The user initiates a transaction that requires authorization from the SCW and the smart contract wallet is still not deployed (i.e: counterfactual).
+
+2. The Passport Wallet backend will calculate the `salt` for the user given the user EOA public key and Immutable’s EOA public key. This will be done offchain using the [image hash generation algorithm](#salt-and-weights).
+
+3. The Passport Wallet backend will send a transaction to the relayer, signed by a **Wallet Deployment EOA** and triggering the `deployAndExecute()` method that takes in the parameters to deploy the contract and execute the users first write transaction.
+
+4. The Passport backend will listen for the event emitted to signal the wallet has been deployed
+
+![Wallet Deployment](./202309-audit-background-diagrams/wallet-deployment.jpg)
+
+### Deployment
+
+- The **Factory Deployment EOA** initiates the deployment of the factory contract,
+  passing in the **Admin EOA** address to the contract constructor.
+- The constructor grants admin access to the **Admin EOA**.
+- The **Admin EOA** invokes the factory contract to grant the wallet deployer role
+  to the **Wallet Deployment EOA** that will be used subsequently to invoke the
+  factory contract and deploy wallets.
+
+![Factory Deployment](./202309-audit-background-diagrams/factory-deployment.jpg)
+
+## Testing
+
+We run tests against the full 0xSequence wallet contracts testing suite to
+ensure no regression has been introduced with our changes, in addition to a new
+smaller testing suite written by us for the additional contracts. These
+are all hardhat based tests.
+
+The 0xSequence testing suite was modified in some aspects, however:
+
+- Gas limits were changed in some places.
+- Code was added to give `DEPLOYER_ROLE` to the testing addresses so they could
+  call `Factory.deploy()`.
+- Code that required the `Factory.deploy()` function to not revert when
+  deployment failed was modified to expect a revert.
+- Changes to constructor arguments were reflected in tests.
+
+## New Testing Suite
+
+The [new testing suite][new-testing] has some overlap with 0xSequence’s original
+suite but focuses specifically on the added contracts. The new test files use
+the `Immutable` prefix to differentiate from existing 0xSequence test files.
+
+[new-testing]: https://github.com/immutable/wallet-contracts/blob/main/tests/
+
+### ImmutableDeployment.spec.ts
+
+Focuses on testing the intended end to end deployment, transaction signing and
+transaction execution it's the main testing provided for `ImmutableSigner.sol`.
+It's a good first stop to understanding the architecture.
+
+It tests:
+
+- Deterministic deployment addresses.
+- 2 of 2 transaction execution in the intended configuration.
+- Updating the `ImmutableSigner.sol` EOA signer.
+
+### ImmutableFactory.spec.ts
+
+Has four sections:
+
+- Testing the correctness of the new `getAddress()` function.
+- Testing the correctness of the modified `deploy()` function.
+- Testing negative access to the `deploy()` function.
+- Testing regressions on expected wallet and proxy behavior when using the new
+  factory contract.
+
+### ImmutableMultiCallDeploy.spec.ts
+
+- Tests RBAC on the `deployExecute()` and `deployAndExecute()` functions.
+- Tests wallet deployment.
+- Tests transaction execution.
+
+### ImmutableStartup.spec.ts
+
+Tests the process of pointing the wallet proxies to the startup contract
+initially, and then redirecting to the actual implementation after the first
+transaction.
+
+- Tests implementation upgrades using the startup pattern.
+- Tests initialization to latest implementation.
+- Tests default functionality when using the startup pattern.
+
+### ImmutableSigner.spec.ts
+
+Focuses on unit testing the `ImmutableSigner.sol` contract.
+
+- Allows valid signatures.
+- Disallows invalid signatures.
+- Properly rotates signers.
+- Proper RBAC controls work.
+
+# Added contracts
+
+## MultiCallDeploy
+
+MultiCalllDeploy.sol is a contract that handles the SCW deployment and write transaction in a single transaction. This was an optimisation made for our off-chain relayer service, to avoid the need to submit two separate transactions; the wallet deployment transaction and user transaction.
+
+## Interface
+
+- `function deployExecute(address _mainModule, bytes32 _salt, address factory,  IModuleCalls.Transaction[] calldata _txs, uint256 _nonce, bytes calldata _signature)`
+- `function deployAndExecute(address cfa, address _mainModule, bytes32 _salt, address factory,  IModuleCalls.Transaction[] calldata _txs, uint256 _nonce, bytes calldata _signature)`
+
+## Functionality and usage
+
+MultiCalllDeploy interfaces with the wallet factory contract to invoke deployments, meaning this contract must be granted the deployer role. This contract also implements access control to restrict access to deployments on the factory contract. Additionally, the contract interfaces with the wallet module in order to execute the transactions.
+
+The relayer service will use `deployAndExecute()` as the on-chain entry point. This function will check the existence of the user's SCW. If the SCW is not deployed, it will deploy the proxy contract and immediately execute the user's transaction with this newly deployed proxy state. Else, if the SCW exists, it will execute the users transaction. An example of the first case is if a CFA has assets in its balance and wishes to transfer them (deployment followed by asset transfer)
+
+The alternative is `deployExecute()`, which deploys the wallet and executes the transaction, and does not handle the alternate case (SCW already deployed); this is for gas saving purposes. For the initial release, the relayer service will not use this function.
+
+![Multicall Deploy](./202309-audit-background-diagrams/multicall-deploy.png)
+
+# Immutable Signer
+
+ImmutableSigner.sol is a smart contract used to store the public key of the Immutable signer (the second valid signer defined in the image hash) and validate the 2-of-2 multisig at the time of transaction execution in the wallet module.
+
+## Interface
+
+- `event SignerUpdated(address indexed _previousSigner, address indexed _newSigner)`
+- `function grantSignerRole(address _signerAdmin)`
+- `function updateSigner(address _newSigner)`
+- `isValidSignature(bytes32 _hash, bytes memory _signature)`
+
+## Functionality and usage
+
+ImmutableSigner.sol may only store a single signer address at a time, and uses `updateSigner()` (inclusive of access control) to update the signer to a new signer. This stored signer is what the supplied signatures are validated against.
+
+Delegating the signature validation responsibility to the Immutable signer requires the use of the
+`IERC1271` interface. The wallet module will call the interface on the Immutable Signer, supplying a hash and signature, if the validation succeeds, the `IERC1271_MAGICVALUE` is returned, allowing the transaction flow to continue. This validation flow requires the use of `DYNAMIC_SIGNATURE` flag which is set off-chain in the signing process.
+
+![Immutable Signer](./202309-audit-background-diagrams/immutable-signer.png)
+
+# Startup Wallet
+
+StartupWalletImpl.sol is an implementation contract that stores the address of the latest wallet implementation, ensuring that the wallet proxy deployment data never changes. Previously, an updated module address would needed to be supplied if undeployed proxies referenced a new wallet module.
+
+## Interface
+
+- `walletImplementationLocator()`
+
+## Functionality and usage
+
+By StartupWalletImpl.sol storing the latest wallet module, which is retrieved using an interface to LastWalletImplLocator.sol (manually updated to always point at the most updated wallet module), wallet proxies can ensure they are deployed with the latest implementation.
+
+The above means that the initial implementation contract for all wallet proxies will be StartupWalletImpl.sol, which can be thought of as middle-ware between the proxy and the latest wallet module.
+
+Wallet proxies will forward their first delegate calls to this contract, which will update the implementation address of the proxy to the latest module, and complete the transaction flow, forwarding the delegate call to the latest module. All subsequent calls, will then use the newly updated module as implementation, bypassing the start up wallet contract.
+
+![Startup Wallet](./202309-audit-background-diagrams/startup-wallet.png)
+
+# Yul Proxy
+
+WalletProxy.yul is a yul implementation of a minimal transparent proxy, with an additional function to retrieve the address of the implementation contract of the proxy. The contract was designed to minimize the deployment costs, as this will be a frequent operation.
+
+## Interface
+
+- `PROXY_getImplementation()`
+  - Selector: `0x90611127`
+
+## Functionality and usage
+
+Functionally, WalletProxy.yul is intended to behave identically to a `EIP1167` minimal proxy, with the inclusion of a function to retrieve the implementation address.
+
+With the contract being written in yul, the contract object must explicitly define it’s initcode and runtime code. The initcode copies the runtime code plus an extra 32 bytes, which contains the implementation address. The contract then stores the extra 32 bytes in the storage slot which is equal to the address of the contract.
+
+The runtime code object defines two functions: 1) the `PROXY_getImplementation()` which loads the address stored in the storage slot equal to the address of the proxy contract and 2) a standard fallback function which delegate calls to the implementation address, either returning success or failure.
+
+# Main Module Dynamic Auth
+
+MainModuleDynamicAuth.sol is one of the main wallet implementations, along MainModule.sol and MainModuleUpgradable.sol. It solves some issues with MainModuleUpgradable.sol and extends the behavior of MainModule.sol.
+
+## Interface
+
+- `_isValidImage(bytes32 _imageHash)`
+
+## Functionality and usage
+
+The main goal of this contract is to be a composite implementation of the behavior in both MainModule.sol and MainModuleUpgradable.sol: it initially uses the authorization method of checking the signers in a transaction match the signers used to initialize the SCW address by the wallet factory (i.e.: it’s initially tied to the SCW address), but then it stores that signer hash (called image hash) and uses it in further verifications. This way, the signers of a wallet can be updated without changing the contract address after the initial SCW deployment.
+
+This check is implemented by `_isValidImage(bytes32)` which is in turn called by the core of the wallet authentication implementation in ModuleAuth.sol.
+
+The implementation is imported from ModuleAuthDynamic.sol.
diff --git a/forks/passport/audits/Consensys_Diligence.md b/forks/passport/audits/Consensys_Diligence.md
new file mode 100644
index 00000000..aabcfc35
--- /dev/null
+++ b/forks/passport/audits/Consensys_Diligence.md
@@ -0,0 +1,7 @@
+# Consensys Diligence report May 2020
+
+[View full report](https://diligence.consensys.net/audits/private/cnhjwtpa-horizon-wallet/)
+
+Please note, Sequence Wallet was formerly known as "Arcadeum Wallet". Any references of "Arcadeum"
+are synonymous with "Sequence", and this repository.
+
diff --git a/forks/passport/audits/Quantstamp_Arcadeum_Report_Final.pdf b/forks/passport/audits/Quantstamp_Arcadeum_Report_Final.pdf
new file mode 100644
index 00000000..f78c80a7
Binary files /dev/null and b/forks/passport/audits/Quantstamp_Arcadeum_Report_Final.pdf differ
diff --git a/forks/passport/audits/sequence_quantstamp_audit_feb_2021.pdf b/forks/passport/audits/sequence_quantstamp_audit_feb_2021.pdf
new file mode 100644
index 00000000..27c38975
Binary files /dev/null and b/forks/passport/audits/sequence_quantstamp_audit_feb_2021.pdf differ
diff --git a/forks/passport/cla.txt b/forks/passport/cla.txt
new file mode 100644
index 00000000..ae99e294
--- /dev/null
+++ b/forks/passport/cla.txt
@@ -0,0 +1,97 @@
+Immutable Software Inc. Individual Contributor License Agreement ("Agreement")
+
+
+
+
+Thank you for your interest in Immutable ("Immutable"). In order to clarify the intellectual
+property license granted with Contributions from any person or entity, Immutable must have a
+Contributor License Agreement ("CLA") on file that has been signed by each Contributor,
+indicating agreement to the license terms below. This license is for your protection as a
+Contributor as well as the protection of Immutable and its users; it does not change your
+rights to use your own Contributions for any other purpose.
+
+
+You accept and agree to the following terms and conditions for Your present and future
+Contributions submitted to Immutable. Except for the license granted herein to Immutable
+and recipients of software distributed by Immutable, You reserve all right, title, and
+interest in and to Your Contributions.
+
+
+1. Definitions. "You" (or "Your") shall mean the copyright owner or legal entity authorized
+by the copyright owner that is making this Agreement with Immutable. For legal entities,
+the entity making a Contribution and all other entities that control, are controlled by,
+or are under common control with that entity are considered to be a single Contributor.
+For the purposes of this definition, "control" means (i) the power, direct or indirect, to
+cause the direction or management of such entity, whether by contract or otherwise, or (ii)
+ownership of fifty percent (50%) or more of the outstanding shares, or (iii) beneficial
+ownership of such entity. "Contribution" shall mean any original work of authorship,
+including any modifications or additions to an existing work, that is intentionally
+submitted by You to Immutable for inclusion in, or documentation of, any of the products
+owned or managed by Immutable (the "Work"). For the purposes of this definition,
+"submitted" means any form of electronic, verbal, or written communication sent to
+Immutable or its representatives, including but not limited to communication on
+electronic mailing lists, source code control systems, and issue tracking systems that are
+managed by, or on behalf of, Immutable for the purpose of discussing and improving the
+Work, but excluding communication that is conspicuously marked or otherwise designated
+in writing by You as "Not a Contribution."
+
+
+2. Grant of Copyright License. Subject to the terms and conditions of this Agreement,
+You hereby grant to Immutable and to recipients of software distributed by Immutable a
+perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright
+license to reproduce, prepare derivative works of, publicly display, publicly perform,
+sublicense, and distribute Your Contributions and such derivative works.
+
+
+3. Grant of Patent License. Subject to the terms and conditions of this Agreement, You
+hereby grant to Immutable and to recipients of software distributed by Immutable a perpetual,
+worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this
+section) patent license to make, have made, use, offer to sell, sell, import, and otherwise
+transfer the Work, where such license applies only to those patent claims licensable by You
+that are necessarily infringed by Your Contribution(s) alone or by combination of Your
+Contribution(s) with the Work to which such Contribution(s) was submitted. If any entity
+institutes patent litigation against You or any other entity (including a cross-claim or
+counterclaim in a lawsuit) alleging that your Contribution, or the Work to which you have
+contributed, constitutes direct or contributory patent infringement, then any patent
+licenses granted to that entity under this Agreement for that Contribution or Work shall
+terminate as of the date such litigation is filed.
+
+
+4. Authorization. You represent that you are legally entitled to grant the above license.
+If your employer(s) has rights to intellectual property that you create that includes your
+Contributions, you represent that you have received permission to make Contributions on
+behalf of that employer, that your employer has waived such rights for your Contributions
+to Immutable, or that your employer has executed a separate Corporate CLA with Immutable.
+
+
+5. Original Creation. You represent that each of Your Contributions is Your original
+creation (see section 7 for submissions on behalf of others). You represent that Your
+Contribution submissions include complete details of any third-party license or other
+restriction (including, but not limited to, related patents and trademarks) of which
+you are personally aware and which are associated with any part of Your Contributions.
+
+
+6. No Support Obligation. You are not expected to provide support for Your Contributions,
+except to the extent You desire to provide support. You may provide support for free, for
+a fee, or not at all. Unless required by applicable law or agreed to in writing, You
+provide Your Contributions on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+KIND, either express or implied, including, without limitation, any warranties or
+conditions of TITLE, NON- INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE.
+
+
+7. Licensed Works. Should You wish to submit work that is not Your original creation,
+You may submit it to Immutable separately from any Contribution, identifying the
+complete details of its source and of any license or other restriction (including, but
+not limited to, related patents, trademarks, and license agreements) of which you are
+personally aware, and conspicuously marking the work as "Submitted on behalf of a
+third-party: [named here]".
+
+
+8. Notification. You agree to notify Immutable of any facts or circumstances of which
+you become aware that would make these representations inaccurate in any respect.
+
+
+Please sign:
+
+
+__________________________________ Date: ________________
diff --git a/forks/passport/compileWalletProxyYul.sh b/forks/passport/compileWalletProxyYul.sh
new file mode 100644
index 00000000..1e7f230f
--- /dev/null
+++ b/forks/passport/compileWalletProxyYul.sh
@@ -0,0 +1,3 @@
+# Copyright (c) Immutable Pty Ltd 2018 - 2023
+# Generates WalletProxy bytecode that is stored in Wallet.sol
+solc --strict-assembly src/contracts/WalletProxy.yul --optimize --bin --optimize-runs=1000
diff --git a/forks/passport/config/PROD.env.sample b/forks/passport/config/PROD.env.sample
new file mode 100644
index 00000000..a5f1dfeb
--- /dev/null
+++ b/forks/passport/config/PROD.env.sample
@@ -0,0 +1,3 @@
+ETH_MNEMONIC=""
+INFURA_API_KEY=""
+ETHERSCAN=""
diff --git a/forks/passport/hardhat.config.ts b/forks/passport/hardhat.config.ts
new file mode 100644
index 00000000..7299c8b4
--- /dev/null
+++ b/forks/passport/hardhat.config.ts
@@ -0,0 +1,98 @@
+import { HardhatUserConfig } from 'hardhat/config'
+import { networkConfig } from './utils/config-loader'
+
+import '@nomiclabs/hardhat-truffle5'
+import '@nomiclabs/hardhat-ethers'
+import '@nomiclabs/hardhat-web3'
+import '@nomiclabs/hardhat-etherscan'
+import '@nomicfoundation/hardhat-chai-matchers'
+
+import 'hardhat-gas-reporter'
+import 'solidity-coverage'
+import { HardhatConfig } from 'hardhat/types'
+
+require('dotenv').config()
+
+const ganacheNetwork = {
+  url: 'http://127.0.0.1:8545',
+  blockGasLimit: 6000000000
+}
+
+const config: HardhatUserConfig = {
+  solidity: {
+    compilers: [{ version: '0.8.17' }],
+    settings: {
+      optimizer: {
+        enabled: true,
+        runs: 999999,
+        details: {
+          yul: true
+        }
+      }
+    }
+  },
+  paths: {
+    root: 'src',
+    tests: '../tests'
+  },
+  networks: {
+    // Define here to easily specify private keys
+    devnet: validateEnvironment()
+      ? {
+          url: 'https://rpc.dev.immutable.com',
+          accounts: [process.env.DEPLOYER_PRIV_KEY!, process.env.WALLET_IMPL_CHANGER_PRIV_KEY!]
+        }
+      : {
+          url: 'SET ENVIRONMENT VARIABLES',
+          accounts: []
+        },
+    testnet: validateEnvironment()
+      ? {
+          url: 'https://rpc.testnet.immutable.com',
+          accounts: [process.env.DEPLOYER_PRIV_KEY!, process.env.WALLET_IMPL_CHANGER_PRIV_KEY!]
+        }
+      : {
+          url: 'SET ENVIRONMENT VARIABLES',
+          accounts: []
+        },
+    sepolia: networkConfig('sepolia'),
+    mainnet: networkConfig('mainnet'),
+    ropsten: networkConfig('ropsten'),
+    rinkeby: networkConfig('rinkeby'),
+    kovan: networkConfig('kovan'),
+    goerli: networkConfig('goerli'),
+    matic: networkConfig('matic'),
+    mumbai: networkConfig('mumbai'),
+    arbitrum: networkConfig('arbitrum'),
+    arbitrumTestnet: networkConfig('arbitrum-testnet'),
+    optimism: networkConfig('optimism'),
+    metis: networkConfig('metis'),
+    nova: networkConfig('nova'),
+    avalanche: networkConfig('avalanche'),
+    avalancheTestnet: networkConfig('avalanche-testnet'),
+    ganache: ganacheNetwork,
+    coverage: {
+      url: 'http://localhost:8555'
+    }
+  },
+  etherscan: {
+    // Your API key for Etherscan
+    // Obtain one at https://etherscan.io/
+    apiKey: networkConfig('mainnet').etherscan
+  },
+  mocha: {
+    timeout: process.env.COVERAGE ? 15 * 60 * 1000 : 30 * 1000
+  },
+  gasReporter: {
+    enabled: !!process.env.REPORT_GAS === true,
+    currency: 'USD',
+    gasPrice: 21,
+    showTimeSpent: true
+  }
+}
+
+export default config
+
+function validateEnvironment(): boolean {
+  return !!process.env.DEPLOYER_PRIV_KEY && !!process.env.WALLET_IMPL_CHANGER_PRIV_KEY
+}
diff --git a/forks/passport/package-lock.json b/forks/passport/package-lock.json
new file mode 100644
index 00000000..5b516c42
--- /dev/null
+++ b/forks/passport/package-lock.json
@@ -0,0 +1,26271 @@
+{
+  "name": "root",
+  "lockfileVersion": 2,
+  "requires": true,
+  "packages": {
+    "": {
+      "name": "root",
+      "license": "Apache-2.0",
+      "workspaces": [
+        "src"
+      ],
+      "devDependencies": {
+        "@0xsequence/deployer": "^0.21.5",
+        "@ethersproject/abi": "^5.7.0",
+        "@ethersproject/providers": "^5.7.2",
+        "@nomicfoundation/hardhat-chai-matchers": "^1.0.6",
+        "@nomicfoundation/hardhat-network-helpers": "^1.0.8",
+        "@nomiclabs/hardhat-ethers": "^2.0.2",
+        "@nomiclabs/hardhat-etherscan": "^3.1.0",
+        "@nomiclabs/hardhat-truffle5": "^2.0.0",
+        "@nomiclabs/hardhat-web3": "^2.0.0",
+        "@openzeppelin/contracts": "^4.9.3",
+        "@tenderly/hardhat-tenderly": "^1.0.11",
+        "@typechain/ethers-v5": "^10.1.1",
+        "@types/chai-as-promised": "^7.1.0",
+        "@types/chai-string": "^1.4.1",
+        "@types/mocha": "^10.0.0",
+        "@typescript-eslint/eslint-plugin": "^5.42.1",
+        "@typescript-eslint/parser": "^5.42.1",
+        "bn-chai": "^1.0.1",
+        "chai-as-promised": "^7.1.1",
+        "chai-bignumber": "^3.0.0",
+        "chai-string": "^1.5.0",
+        "child_process": "^1.0.2",
+        "dotenv": "^16.0.3",
+        "eslint": "^8.27.0",
+        "eslint-config-prettier": "^8.1.0",
+        "eslint-plugin-import": "^2.22.0",
+        "eslint-plugin-prettier": "^4.2.1",
+        "ethers": "^5.7.2",
+        "ganache": "^7.5.0",
+        "hardhat": "2.12.2",
+        "hardhat-gas-reporter": "^1.0.9",
+        "husky": "^4.2.3",
+        "rimraf": "^3.0.2",
+        "solhint": "^3.3.4",
+        "solidity-coverage": "^0.8.5",
+        "solmate": "^6.7.0",
+        "ts-node": "^10.9.1",
+        "typechain": "^8.1.1",
+        "typescript": "^4.8.4",
+        "yesno": "^0.4.0"
+      }
+    },
+    "node_modules/@0xsequence/deployer": {
+      "version": "0.21.5",
+      "resolved": "https://registry.npmjs.org/@0xsequence/deployer/-/deployer-0.21.5.tgz",
+      "integrity": "sha512-P+QsiTrbZez//NCRvMc6OV6zWYP/TJzGJ2clFG1852CCPijqm0RJPmdoy5FzHpJd+9T4W7Awo0U2/gseSvFnaA==",
+      "dev": true,
+      "dependencies": {
+        "@0xsequence/utils": "^0.21.5",
+        "@ethersproject/contracts": "^5.0.12",
+        "ethers": "^5.0.32",
+        "ora": "^5.3.0"
+      }
+    },
+    "node_modules/@0xsequence/utils": {
+      "version": "0.21.5",
+      "resolved": "https://registry.npmjs.org/@0xsequence/utils/-/utils-0.21.5.tgz",
+      "integrity": "sha512-6gL0kot2o+E3n8E/1yvQR3wv4vxuJB9w25SNUK/zd9MDJ4oIY8tIy5jc4YrhQwC+FqKPG3k/8skNN9oP3gNXJg==",
+      "dev": true,
+      "dependencies": {
+        "@ethersproject/abstract-signer": "5.0.14",
+        "@ethersproject/properties": "^5.0.9",
+        "ethers": "^5.0.32",
+        "js-base64": "^3.6.0"
+      }
+    },
+    "node_modules/@babel/code-frame": {
+      "version": "7.18.6",
+      "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.18.6.tgz",
+      "integrity": "sha512-TDCmlK5eOvH+eH7cdAFlNXeVJqWIQ7gW9tY1GJIpUtFb6CmjVyq2VM3u71bOyR8CRihcCgMUYoDNyLXao3+70Q==",
+      "dev": true,
+      "dependencies": {
+        "@babel/highlight": "^7.18.6"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/helper-validator-identifier": {
+      "version": "7.19.1",
+      "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.19.1.tgz",
+      "integrity": "sha512-awrNfaMtnHUr653GgGEs++LlAvW6w+DcPrOliSMXWCKo597CwL5Acf/wWdNkf/tfEQE3mjkeD1YOVZOUV/od1w==",
+      "dev": true,
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/highlight": {
+      "version": "7.18.6",
+      "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.18.6.tgz",
+      "integrity": "sha512-u7stbOuYjaPezCuLj29hNW1v64M2Md2qupEKP1fHc7WdOA3DgLh37suiSrZYY7haUB7iBeQZ9P1uiRF359do3g==",
+      "dev": true,
+      "dependencies": {
+        "@babel/helper-validator-identifier": "^7.18.6",
+        "chalk": "^2.0.0",
+        "js-tokens": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/highlight/node_modules/ansi-styles": {
+      "version": "3.2.1",
+      "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+      "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+      "dev": true,
+      "dependencies": {
+        "color-convert": "^1.9.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@babel/highlight/node_modules/chalk": {
+      "version": "2.4.2",
+      "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+      "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+      "dev": true,
+      "dependencies": {
+        "ansi-styles": "^3.2.1",
+        "escape-string-regexp": "^1.0.5",
+        "supports-color": "^5.3.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@babel/highlight/node_modules/color-convert": {
+      "version": "1.9.3",
+      "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+      "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+      "dev": true,
+      "dependencies": {
+        "color-name": "1.1.3"
+      }
+    },
+    "node_modules/@babel/highlight/node_modules/color-name": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+      "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+      "dev": true
+    },
+    "node_modules/@babel/highlight/node_modules/has-flag": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+      "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@babel/highlight/node_modules/supports-color": {
+      "version": "5.5.0",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+      "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+      "dev": true,
+      "dependencies": {
+        "has-flag": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@babel/runtime": {
+      "version": "7.21.0",
+      "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.21.0.tgz",
+      "integrity": "sha512-xwII0//EObnq89Ji5AKYQaRYiW/nZ3llSv29d49IuxPhKbtJoLP+9QUUZ4nVragQVtaVGeZrpB+ZtG/Pdy/POw==",
+      "dev": true,
+      "dependencies": {
+        "regenerator-runtime": "^0.13.11"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@colors/colors": {
+      "version": "1.5.0",
+      "resolved": "https://registry.npmjs.org/@colors/colors/-/colors-1.5.0.tgz",
+      "integrity": "sha512-ooWCrlZP11i8GImSjTHYHLkvFDP48nS4+204nGb1RiX/WXYHmJA2III9/e2DWVabCESdW7hBAEzHRqUn9OUVvQ==",
+      "dev": true,
+      "optional": true,
+      "engines": {
+        "node": ">=0.1.90"
+      }
+    },
+    "node_modules/@cspotcode/source-map-support": {
+      "version": "0.8.1",
+      "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz",
+      "integrity": "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==",
+      "dev": true,
+      "dependencies": {
+        "@jridgewell/trace-mapping": "0.3.9"
+      },
+      "engines": {
+        "node": ">=12"
+      }
+    },
+    "node_modules/@ensdomains/address-encoder": {
+      "version": "0.1.9",
+      "resolved": "https://registry.npmjs.org/@ensdomains/address-encoder/-/address-encoder-0.1.9.tgz",
+      "integrity": "sha512-E2d2gP4uxJQnDu2Kfg1tHNspefzbLT8Tyjrm5sEuim32UkU2sm5xL4VXtgc2X33fmPEw9+jUMpGs4veMbf+PYg==",
+      "dev": true,
+      "dependencies": {
+        "bech32": "^1.1.3",
+        "blakejs": "^1.1.0",
+        "bn.js": "^4.11.8",
+        "bs58": "^4.0.1",
+        "crypto-addr-codec": "^0.1.7",
+        "nano-base32": "^1.0.1",
+        "ripemd160": "^2.0.2"
+      }
+    },
+    "node_modules/@ensdomains/address-encoder/node_modules/bn.js": {
+      "version": "4.12.0",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+      "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+      "dev": true
+    },
+    "node_modules/@ensdomains/ens": {
+      "version": "0.4.5",
+      "resolved": "https://registry.npmjs.org/@ensdomains/ens/-/ens-0.4.5.tgz",
+      "integrity": "sha512-JSvpj1iNMFjK6K+uVl4unqMoa9rf5jopb8cya5UGBWz23Nw8hSNT7efgUx4BTlAPAgpNlEioUfeTyQ6J9ZvTVw==",
+      "deprecated": "Please use @ensdomains/ens-contracts",
+      "dev": true,
+      "dependencies": {
+        "bluebird": "^3.5.2",
+        "eth-ens-namehash": "^2.0.8",
+        "solc": "^0.4.20",
+        "testrpc": "0.0.1",
+        "web3-utils": "^1.0.0-beta.31"
+      }
+    },
+    "node_modules/@ensdomains/ensjs": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/@ensdomains/ensjs/-/ensjs-2.1.0.tgz",
+      "integrity": "sha512-GRbGPT8Z/OJMDuxs75U/jUNEC0tbL0aj7/L/QQznGYKm/tiasp+ndLOaoULy9kKJFC0TBByqfFliEHDgoLhyog==",
+      "dev": true,
+      "dependencies": {
+        "@babel/runtime": "^7.4.4",
+        "@ensdomains/address-encoder": "^0.1.7",
+        "@ensdomains/ens": "0.4.5",
+        "@ensdomains/resolver": "0.2.4",
+        "content-hash": "^2.5.2",
+        "eth-ens-namehash": "^2.0.8",
+        "ethers": "^5.0.13",
+        "js-sha3": "^0.8.0"
+      }
+    },
+    "node_modules/@ensdomains/resolver": {
+      "version": "0.2.4",
+      "resolved": "https://registry.npmjs.org/@ensdomains/resolver/-/resolver-0.2.4.tgz",
+      "integrity": "sha512-bvaTH34PMCbv6anRa9I/0zjLJgY4EuznbEMgbV77JBCQ9KNC46rzi0avuxpOfu+xDjPEtSFGqVEOr5GlUSGudA==",
+      "deprecated": "Please use @ensdomains/ens-contracts",
+      "dev": true
+    },
+    "node_modules/@eslint/eslintrc": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.0.0.tgz",
+      "integrity": "sha512-fluIaaV+GyV24CCu/ggiHdV+j4RNh85yQnAYS/G2mZODZgGmmlrgCydjUcV3YvxCm9x8nMAfThsqTni4KiXT4A==",
+      "dev": true,
+      "dependencies": {
+        "ajv": "^6.12.4",
+        "debug": "^4.3.2",
+        "espree": "^9.4.0",
+        "globals": "^13.19.0",
+        "ignore": "^5.2.0",
+        "import-fresh": "^3.2.1",
+        "js-yaml": "^4.1.0",
+        "minimatch": "^3.1.2",
+        "strip-json-comments": "^3.1.1"
+      },
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "url": "https://opencollective.com/eslint"
+      }
+    },
+    "node_modules/@eslint/eslintrc/node_modules/ajv": {
+      "version": "6.12.6",
+      "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+      "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+      "dev": true,
+      "dependencies": {
+        "fast-deep-equal": "^3.1.1",
+        "fast-json-stable-stringify": "^2.0.0",
+        "json-schema-traverse": "^0.4.1",
+        "uri-js": "^4.2.2"
+      },
+      "funding": {
+        "type": "github",
+        "url": "https://github.com/sponsors/epoberezkin"
+      }
+    },
+    "node_modules/@eslint/eslintrc/node_modules/json-schema-traverse": {
+      "version": "0.4.1",
+      "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+      "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+      "dev": true
+    },
+    "node_modules/@eslint/js": {
+      "version": "8.35.0",
+      "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.35.0.tgz",
+      "integrity": "sha512-JXdzbRiWclLVoD8sNUjR443VVlYqiYmDVT6rGUEIEHU5YJW0gaVZwV2xgM7D4arkvASqD0IlLUVjHiFuxaftRw==",
+      "dev": true,
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      }
+    },
+    "node_modules/@ethereumjs/common": {
+      "version": "2.5.0",
+      "resolved": "https://registry.npmjs.org/@ethereumjs/common/-/common-2.5.0.tgz",
+      "integrity": "sha512-DEHjW6e38o+JmB/NO3GZBpW4lpaiBpkFgXF6jLcJ6gETBYpEyaA5nTimsWBUJR3Vmtm/didUEbNjajskugZORg==",
+      "dev": true,
+      "dependencies": {
+        "crc-32": "^1.2.0",
+        "ethereumjs-util": "^7.1.1"
+      }
+    },
+    "node_modules/@ethereumjs/tx": {
+      "version": "3.3.2",
+      "resolved": "https://registry.npmjs.org/@ethereumjs/tx/-/tx-3.3.2.tgz",
+      "integrity": "sha512-6AaJhwg4ucmwTvw/1qLaZUX5miWrwZ4nLOUsKyb/HtzS3BMw/CasKhdi1ims9mBKeK9sOJCH4qGKOBGyJCeeog==",
+      "dev": true,
+      "dependencies": {
+        "@ethereumjs/common": "^2.5.0",
+        "ethereumjs-util": "^7.1.2"
+      }
+    },
+    "node_modules/@ethersproject/abi": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abi/-/abi-5.7.0.tgz",
+      "integrity": "sha512-351ktp42TiRcYB3H1OP8yajPeAQstMW/yCFokj/AthP9bLHzQFPlOrxOcwYEDkUAICmOHljvN4K39OMTMUa9RA==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/constants": "^5.7.0",
+        "@ethersproject/hash": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/abstract-provider": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abstract-provider/-/abstract-provider-5.7.0.tgz",
+      "integrity": "sha512-R41c9UkchKCpAqStMYUpdunjo3pkEvZC3FAwZn5S5MGbXoMQOHIdHItezTETxAO5bevtMApSyEhn9+CHcDsWBw==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/networks": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/transactions": "^5.7.0",
+        "@ethersproject/web": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/abstract-signer": {
+      "version": "5.0.14",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.0.14.tgz",
+      "integrity": "sha512-JztBwVO7o5OHLh2vyjordlS4/1EjRyaECtc8vPdXTF1i4dXN+J0coeRoPN6ZFbBvi/YbaB6br2fvqhst1VQD/g==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-provider": "^5.0.8",
+        "@ethersproject/bignumber": "^5.0.13",
+        "@ethersproject/bytes": "^5.0.9",
+        "@ethersproject/logger": "^5.0.8",
+        "@ethersproject/properties": "^5.0.7"
+      }
+    },
+    "node_modules/@ethersproject/address": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/address/-/address-5.7.0.tgz",
+      "integrity": "sha512-9wYhYt7aghVGo758POM5nqcOMaE168Q6aRLJZwUmiqSrAungkG74gSSeKEIR7ukixesdRZGPgVqme6vmxs1fkA==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/rlp": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/base64": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/base64/-/base64-5.7.0.tgz",
+      "integrity": "sha512-Dr8tcHt2mEbsZr/mwTPIQAf3Ai0Bks/7gTw9dSqk1mQvhW3XvRlmDJr/4n+wg1JmCl16NZue17CDh8xb/vZ0sQ==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bytes": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/basex": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/basex/-/basex-5.7.0.tgz",
+      "integrity": "sha512-ywlh43GwZLv2Voc2gQVTKBoVQ1mti3d8HK5aMxsfu/nRDnMmNqaSJ3r3n85HBByT8OpoY96SXM1FogC533T4zw==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/bignumber": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/bignumber/-/bignumber-5.7.0.tgz",
+      "integrity": "sha512-n1CAdIHRWjSucQO3MC1zPSVgV/6dy/fjL9pMrPP9peL+QxEg9wOsVqwD4+818B6LUEtaXzVHQiuivzRoxPxUGw==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "bn.js": "^5.2.1"
+      }
+    },
+    "node_modules/@ethersproject/bytes": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/bytes/-/bytes-5.7.0.tgz",
+      "integrity": "sha512-nsbxwgFXWh9NyYWo+U8atvmMsSdKJprTcICAkvbBffT75qDocbuggBU0SJiVK2MuTrp0q+xvLkTnGMPK1+uA9A==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/constants": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/constants/-/constants-5.7.0.tgz",
+      "integrity": "sha512-DHI+y5dBNvkpYUMiRQyxRBYBefZkJfo70VUkUAsRjcPs47muV9evftfZ0PJVCXYbAiCgght0DtcF9srFQmIgWA==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bignumber": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/contracts": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/contracts/-/contracts-5.7.0.tgz",
+      "integrity": "sha512-5GJbzEU3X+d33CdfPhcyS+z8MzsTrBGk/sc+G+59+tPa9yFkl6HQ9D6L0QMgNTA9q8dT0XKxxkyp883XsQvbbg==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abi": "^5.7.0",
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/abstract-signer": "^5.7.0",
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/constants": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/transactions": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/contracts/node_modules/@ethersproject/abstract-signer": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+      "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/hash": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/hash/-/hash-5.7.0.tgz",
+      "integrity": "sha512-qX5WrQfnah1EFnO5zJv1v46a8HW0+E5xuBBDTwMFZLuVTx0tbU2kkx15NqdjxecrLGatQN9FGQKpb1FKdHCt+g==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-signer": "^5.7.0",
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/base64": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/hash/node_modules/@ethersproject/abstract-signer": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+      "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/hdnode": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/hdnode/-/hdnode-5.7.0.tgz",
+      "integrity": "sha512-OmyYo9EENBPPf4ERhR7oj6uAtUAhYGqOnIS+jE5pTXvdKBS99ikzq1E7Iv0ZQZ5V36Lqx1qZLeak0Ra16qpeOg==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-signer": "^5.7.0",
+        "@ethersproject/basex": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/pbkdf2": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/sha2": "^5.7.0",
+        "@ethersproject/signing-key": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0",
+        "@ethersproject/transactions": "^5.7.0",
+        "@ethersproject/wordlists": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/hdnode/node_modules/@ethersproject/abstract-signer": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+      "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/json-wallets": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/json-wallets/-/json-wallets-5.7.0.tgz",
+      "integrity": "sha512-8oee5Xgu6+RKgJTkvEMl2wDgSPSAQ9MB/3JYjFV9jlKvcYHUXZC+cQp0njgmxdHkYWn8s6/IqIZYm0YWCjO/0g==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-signer": "^5.7.0",
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/hdnode": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/pbkdf2": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/random": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0",
+        "@ethersproject/transactions": "^5.7.0",
+        "aes-js": "3.0.0",
+        "scrypt-js": "3.0.1"
+      }
+    },
+    "node_modules/@ethersproject/json-wallets/node_modules/@ethersproject/abstract-signer": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+      "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/keccak256": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/keccak256/-/keccak256-5.7.0.tgz",
+      "integrity": "sha512-2UcPboeL/iW+pSg6vZ6ydF8tCnv3Iu/8tUmLLzWWGzxWKFFqOBQFLo6uLUv6BDrLgCDfN28RJ/wtByx+jZ4KBg==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bytes": "^5.7.0",
+        "js-sha3": "0.8.0"
+      }
+    },
+    "node_modules/@ethersproject/logger": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/logger/-/logger-5.7.0.tgz",
+      "integrity": "sha512-0odtFdXu/XHtjQXJYA3u9G0G8btm0ND5Cu8M7i5vhEcE8/HmF4Lbdqanwyv4uQTr2tx6b7fQRmgLrsnpQlmnig==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ]
+    },
+    "node_modules/@ethersproject/networks": {
+      "version": "5.7.1",
+      "resolved": "https://registry.npmjs.org/@ethersproject/networks/-/networks-5.7.1.tgz",
+      "integrity": "sha512-n/MufjFYv3yFcUyfhnXotyDlNdFb7onmkSy8aQERi2PjNcnWQ66xXxa3XlS8nCcA8aJKJjIIMNJTC7tu80GwpQ==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/pbkdf2": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/pbkdf2/-/pbkdf2-5.7.0.tgz",
+      "integrity": "sha512-oR/dBRZR6GTyaofd86DehG72hY6NpAjhabkhxgr3X2FpJtJuodEl2auADWBZfhDHgVCbu3/H/Ocq2uC6dpNjjw==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/sha2": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/properties": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/properties/-/properties-5.7.0.tgz",
+      "integrity": "sha512-J87jy8suntrAkIZtecpxEPxY//szqr1mlBaYlQ0r4RCaiD2hjheqF9s1LVE8vVuJCXisjIP+JgtK/Do54ej4Sw==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/providers": {
+      "version": "5.7.2",
+      "resolved": "https://registry.npmjs.org/@ethersproject/providers/-/providers-5.7.2.tgz",
+      "integrity": "sha512-g34EWZ1WWAVgr4aptGlVBF8mhl3VWjv+8hoAnzStu8Ah22VHBsuGzP17eb6xDVRzw895G4W7vvx60lFFur/1Rg==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/abstract-signer": "^5.7.0",
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/base64": "^5.7.0",
+        "@ethersproject/basex": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/constants": "^5.7.0",
+        "@ethersproject/hash": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/networks": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/random": "^5.7.0",
+        "@ethersproject/rlp": "^5.7.0",
+        "@ethersproject/sha2": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0",
+        "@ethersproject/transactions": "^5.7.0",
+        "@ethersproject/web": "^5.7.0",
+        "bech32": "1.1.4",
+        "ws": "7.4.6"
+      }
+    },
+    "node_modules/@ethersproject/providers/node_modules/@ethersproject/abstract-signer": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+      "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/random": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/random/-/random-5.7.0.tgz",
+      "integrity": "sha512-19WjScqRA8IIeWclFme75VMXSBvi4e6InrUNuaR4s5pTF2qNhcGdCUwdxUVGtDDqC00sDLCO93jPQoDUH4HVmQ==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/rlp": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/rlp/-/rlp-5.7.0.tgz",
+      "integrity": "sha512-rBxzX2vK8mVF7b0Tol44t5Tb8gomOHkj5guL+HhzQ1yBh/ydjGnpw6at+X6Iw0Kp3OzzzkcKp8N9r0W4kYSs9w==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/sha2": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/sha2/-/sha2-5.7.0.tgz",
+      "integrity": "sha512-gKlH42riwb3KYp0reLsFTokByAKoJdgFCwI+CCiX/k+Jm2mbNs6oOaCjYQSlI1+XBVejwH2KrmCbMAT/GnRDQw==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "hash.js": "1.1.7"
+      }
+    },
+    "node_modules/@ethersproject/signing-key": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/signing-key/-/signing-key-5.7.0.tgz",
+      "integrity": "sha512-MZdy2nL3wO0u7gkB4nA/pEf8lu1TlFswPNmy8AiYkfKTdO6eXBJyUdmHO/ehm/htHw9K/qF8ujnTyUAD+Ry54Q==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "bn.js": "^5.2.1",
+        "elliptic": "6.5.4",
+        "hash.js": "1.1.7"
+      }
+    },
+    "node_modules/@ethersproject/solidity": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/solidity/-/solidity-5.7.0.tgz",
+      "integrity": "sha512-HmabMd2Dt/raavyaGukF4XxizWKhKQ24DoLtdNbBmNKUOPqwjsKQSdV9GQtj9CBEea9DlzETlVER1gYeXXBGaA==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/sha2": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/strings": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/strings/-/strings-5.7.0.tgz",
+      "integrity": "sha512-/9nu+lj0YswRNSH0NXYqrh8775XNyEdUQAuf3f+SmOrnVewcJ5SBNAjF7lpgehKi4abvNNXyf+HX86czCdJ8Mg==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/constants": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/transactions": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/transactions/-/transactions-5.7.0.tgz",
+      "integrity": "sha512-kmcNicCp1lp8qanMTC3RIikGgoJ80ztTyvtsFvCYpSCfkjhD0jZ2LOrnbcuxuToLIUYYf+4XwD1rP+B/erDIhQ==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/constants": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/rlp": "^5.7.0",
+        "@ethersproject/signing-key": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/units": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/units/-/units-5.7.0.tgz",
+      "integrity": "sha512-pD3xLMy3SJu9kG5xDGI7+xhTEmGXlEqXU4OfNapmfnxLVY4EMSSRp7j1k7eezutBPH7RBN/7QPnwR7hzNlEFeg==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/constants": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/wallet": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/wallet/-/wallet-5.7.0.tgz",
+      "integrity": "sha512-MhmXlJXEJFBFVKrDLB4ZdDzxcBxQ3rLyCkhNqVu3CDYvR97E+8r01UgrI+TI99Le+aYm/in/0vp86guJuM7FCA==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/abstract-signer": "^5.7.0",
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/hash": "^5.7.0",
+        "@ethersproject/hdnode": "^5.7.0",
+        "@ethersproject/json-wallets": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/random": "^5.7.0",
+        "@ethersproject/signing-key": "^5.7.0",
+        "@ethersproject/transactions": "^5.7.0",
+        "@ethersproject/wordlists": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/wallet/node_modules/@ethersproject/abstract-signer": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+      "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/web": {
+      "version": "5.7.1",
+      "resolved": "https://registry.npmjs.org/@ethersproject/web/-/web-5.7.1.tgz",
+      "integrity": "sha512-Gueu8lSvyjBWL4cYsWsjh6MtMwM0+H4HvqFPZfB6dV8ctbP9zFAO73VG1cMWae0FLPCtz0peKPpZY8/ugJJX2w==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/base64": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0"
+      }
+    },
+    "node_modules/@ethersproject/wordlists": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/wordlists/-/wordlists-5.7.0.tgz",
+      "integrity": "sha512-S2TFNJNfHWVHNE6cNDjbVlZ6MgE17MIxMbMg2zv3wn+3XSJGosL1m9ZVv3GXCf/2ymSsQ+hRI5IzoMJTG6aoVA==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/hash": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0"
+      }
+    },
+    "node_modules/@fastify/busboy": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/@fastify/busboy/-/busboy-2.1.0.tgz",
+      "integrity": "sha512-+KpH+QxZU7O4675t3mnkQKcZZg56u+K/Ct2K+N2AZYNVK8kyeo/bI18tI8aPm3tvNNRyTWfj6s5tnGNlcbQRsA==",
+      "dev": true,
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@humanwhocodes/config-array": {
+      "version": "0.11.8",
+      "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.8.tgz",
+      "integrity": "sha512-UybHIJzJnR5Qc/MsD9Kr+RpO2h+/P1GhOwdiLPXK5TWk5sgTdu88bTD9UP+CKbPPh5Rni1u0GjAdYQLemG8g+g==",
+      "dev": true,
+      "dependencies": {
+        "@humanwhocodes/object-schema": "^1.2.1",
+        "debug": "^4.1.1",
+        "minimatch": "^3.0.5"
+      },
+      "engines": {
+        "node": ">=10.10.0"
+      }
+    },
+    "node_modules/@humanwhocodes/module-importer": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz",
+      "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==",
+      "dev": true,
+      "engines": {
+        "node": ">=12.22"
+      },
+      "funding": {
+        "type": "github",
+        "url": "https://github.com/sponsors/nzakas"
+      }
+    },
+    "node_modules/@humanwhocodes/object-schema": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-1.2.1.tgz",
+      "integrity": "sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA==",
+      "dev": true
+    },
+    "node_modules/@imtbl/wallet-contracts": {
+      "resolved": "src",
+      "link": true
+    },
+    "node_modules/@jridgewell/resolve-uri": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.0.tgz",
+      "integrity": "sha512-F2msla3tad+Mfht5cJq7LSXcdudKTWCVYUgw6pLFOOHSTtZlj6SWNYAp+AhuqLmWdBO2X5hPrLcu8cVP8fy28w==",
+      "dev": true,
+      "engines": {
+        "node": ">=6.0.0"
+      }
+    },
+    "node_modules/@jridgewell/sourcemap-codec": {
+      "version": "1.4.14",
+      "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.14.tgz",
+      "integrity": "sha512-XPSJHWmi394fuUuzDnGz1wiKqWfo1yXecHQMRf2l6hztTO+nPru658AyDngaBe7isIxEkRsPR3FZh+s7iVa4Uw==",
+      "dev": true
+    },
+    "node_modules/@jridgewell/trace-mapping": {
+      "version": "0.3.9",
+      "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz",
+      "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==",
+      "dev": true,
+      "dependencies": {
+        "@jridgewell/resolve-uri": "^3.0.3",
+        "@jridgewell/sourcemap-codec": "^1.4.10"
+      }
+    },
+    "node_modules/@metamask/eth-sig-util": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/@metamask/eth-sig-util/-/eth-sig-util-4.0.1.tgz",
+      "integrity": "sha512-tghyZKLHZjcdlDqCA3gNZmLeR0XvOE9U1qoQO9ohyAZT6Pya+H9vkBPcsyXytmYLNgVoin7CKCmweo/R43V+tQ==",
+      "dev": true,
+      "dependencies": {
+        "ethereumjs-abi": "^0.6.8",
+        "ethereumjs-util": "^6.2.1",
+        "ethjs-util": "^0.1.6",
+        "tweetnacl": "^1.0.3",
+        "tweetnacl-util": "^0.15.1"
+      },
+      "engines": {
+        "node": ">=12.0.0"
+      }
+    },
+    "node_modules/@metamask/eth-sig-util/node_modules/@types/bn.js": {
+      "version": "4.11.6",
+      "resolved": "https://registry.npmjs.org/@types/bn.js/-/bn.js-4.11.6.tgz",
+      "integrity": "sha512-pqr857jrp2kPuO9uRjZ3PwnJTjoQy+fcdxvBTvHm6dkmEL9q+hDD/2j/0ELOBPtPnS8LjCX0gI9nbl8lVkadpg==",
+      "dev": true,
+      "dependencies": {
+        "@types/node": "*"
+      }
+    },
+    "node_modules/@metamask/eth-sig-util/node_modules/bn.js": {
+      "version": "4.12.0",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+      "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+      "dev": true
+    },
+    "node_modules/@metamask/eth-sig-util/node_modules/ethereumjs-util": {
+      "version": "6.2.1",
+      "resolved": "https://registry.npmjs.org/ethereumjs-util/-/ethereumjs-util-6.2.1.tgz",
+      "integrity": "sha512-W2Ktez4L01Vexijrm5EB6w7dg4n/TgpoYU4avuT5T3Vmnw/eCRtiBrJfQYS/DCSvDIOLn2k57GcHdeBcgVxAqw==",
+      "dev": true,
+      "dependencies": {
+        "@types/bn.js": "^4.11.3",
+        "bn.js": "^4.11.0",
+        "create-hash": "^1.1.2",
+        "elliptic": "^6.5.2",
+        "ethereum-cryptography": "^0.1.3",
+        "ethjs-util": "0.1.6",
+        "rlp": "^2.2.3"
+      }
+    },
+    "node_modules/@metamask/eth-sig-util/node_modules/tweetnacl": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-1.0.3.tgz",
+      "integrity": "sha512-6rt+RN7aOi1nGMyC4Xa5DdYiukl2UWCbcJft7YhxReBGQD7OAM8Pbxw6YMo4r2diNEA8FEmu32YOn9rhaiE5yw==",
+      "dev": true
+    },
+    "node_modules/@morgan-stanley/ts-mocking-bird": {
+      "version": "0.6.4",
+      "resolved": "https://registry.npmjs.org/@morgan-stanley/ts-mocking-bird/-/ts-mocking-bird-0.6.4.tgz",
+      "integrity": "sha512-57VJIflP8eR2xXa9cD1LUawh+Gh+BVQfVu0n6GALyg/AqV/Nz25kDRvws3i9kIe1PTrbsZZOYpsYp6bXPd6nVA==",
+      "dev": true,
+      "dependencies": {
+        "lodash": "^4.17.16",
+        "uuid": "^7.0.3"
+      },
+      "peerDependencies": {
+        "jasmine": "2.x || 3.x || 4.x",
+        "jest": "26.x || 27.x || 28.x",
+        "typescript": ">=4.2"
+      },
+      "peerDependenciesMeta": {
+        "jasmine": {
+          "optional": true
+        },
+        "jest": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/@morgan-stanley/ts-mocking-bird/node_modules/uuid": {
+      "version": "7.0.3",
+      "resolved": "https://registry.npmjs.org/uuid/-/uuid-7.0.3.tgz",
+      "integrity": "sha512-DPSke0pXhTZgoF/d+WSt2QaKMCFSfx7QegxEWT+JOuHF5aWrKEn0G+ztjuJg/gG8/ItK+rbPCD/yNv8yyih6Cg==",
+      "dev": true,
+      "bin": {
+        "uuid": "dist/bin/uuid"
+      }
+    },
+    "node_modules/@noble/hashes": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/@noble/hashes/-/hashes-1.2.0.tgz",
+      "integrity": "sha512-FZfhjEDbT5GRswV3C6uvLPHMiVD6lQBmpoX5+eSiPaMTXte/IKqI5dykDxzZB/WBeK/CDuQRBWarPdi3FNY2zQ==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://paulmillr.com/funding/"
+        }
+      ]
+    },
+    "node_modules/@noble/secp256k1": {
+      "version": "1.7.1",
+      "resolved": "https://registry.npmjs.org/@noble/secp256k1/-/secp256k1-1.7.1.tgz",
+      "integrity": "sha512-hOUk6AyBFmqVrv7k5WAw/LpszxVbj9gGN4JRkIX52fdFAj1UA61KXmZDvqVEm+pOyec3+fIeZB02LYa/pWOArw==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://paulmillr.com/funding/"
+        }
+      ]
+    },
+    "node_modules/@nodelib/fs.scandir": {
+      "version": "2.1.5",
+      "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz",
+      "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==",
+      "dev": true,
+      "dependencies": {
+        "@nodelib/fs.stat": "2.0.5",
+        "run-parallel": "^1.1.9"
+      },
+      "engines": {
+        "node": ">= 8"
+      }
+    },
+    "node_modules/@nodelib/fs.stat": {
+      "version": "2.0.5",
+      "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz",
+      "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==",
+      "dev": true,
+      "engines": {
+        "node": ">= 8"
+      }
+    },
+    "node_modules/@nodelib/fs.walk": {
+      "version": "1.2.8",
+      "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz",
+      "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==",
+      "dev": true,
+      "dependencies": {
+        "@nodelib/fs.scandir": "2.1.5",
+        "fastq": "^1.6.0"
+      },
+      "engines": {
+        "node": ">= 8"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-block": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-block/-/ethereumjs-block-4.0.0.tgz",
+      "integrity": "sha512-bk8uP8VuexLgyIZAHExH1QEovqx0Lzhc9Ntm63nCRKLHXIZkobaFaeCVwTESV7YkPKUk7NiK11s8ryed4CS9yA==",
+      "dev": true,
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-trie": "^5.0.0",
+        "@nomicfoundation/ethereumjs-tx": "^4.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "ethereum-cryptography": "0.1.3"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-block/node_modules/@nomicfoundation/ethereumjs-rlp": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+      "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+      "dev": true,
+      "bin": {
+        "rlp": "bin/rlp"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-blockchain": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-blockchain/-/ethereumjs-blockchain-6.0.0.tgz",
+      "integrity": "sha512-pLFEoea6MWd81QQYSReLlLfH7N9v7lH66JC/NMPN848ySPPQA5renWnE7wPByfQFzNrPBuDDRFFULMDmj1C0xw==",
+      "dev": true,
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-block": "^4.0.0",
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-ethash": "^2.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-trie": "^5.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "abstract-level": "^1.0.3",
+        "debug": "^4.3.3",
+        "ethereum-cryptography": "0.1.3",
+        "level": "^8.0.0",
+        "lru-cache": "^5.1.1",
+        "memory-level": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-blockchain/node_modules/@nomicfoundation/ethereumjs-rlp": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+      "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+      "dev": true,
+      "bin": {
+        "rlp": "bin/rlp"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-blockchain/node_modules/lru-cache": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz",
+      "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==",
+      "dev": true,
+      "dependencies": {
+        "yallist": "^3.0.2"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-blockchain/node_modules/yallist": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
+      "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==",
+      "dev": true
+    },
+    "node_modules/@nomicfoundation/ethereumjs-common": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-common/-/ethereumjs-common-3.0.0.tgz",
+      "integrity": "sha512-WS7qSshQfxoZOpHG/XqlHEGRG1zmyjYrvmATvc4c62+gZXgre1ymYP8ZNgx/3FyZY0TWe9OjFlKOfLqmgOeYwA==",
+      "dev": true,
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "crc-32": "^1.2.0"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-ethash": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-ethash/-/ethereumjs-ethash-2.0.0.tgz",
+      "integrity": "sha512-WpDvnRncfDUuXdsAXlI4lXbqUDOA+adYRQaEezIkxqDkc+LDyYDbd/xairmY98GnQzo1zIqsIL6GB5MoMSJDew==",
+      "dev": true,
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-block": "^4.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "abstract-level": "^1.0.3",
+        "bigint-crypto-utils": "^3.0.23",
+        "ethereum-cryptography": "0.1.3"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-ethash/node_modules/@nomicfoundation/ethereumjs-rlp": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+      "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+      "dev": true,
+      "bin": {
+        "rlp": "bin/rlp"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-evm": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-evm/-/ethereumjs-evm-1.0.0.tgz",
+      "integrity": "sha512-hVS6qRo3V1PLKCO210UfcEQHvlG7GqR8iFzp0yyjTg2TmJQizcChKgWo8KFsdMw6AyoLgLhHGHw4HdlP8a4i+Q==",
+      "dev": true,
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "@types/async-eventemitter": "^0.2.1",
+        "async-eventemitter": "^0.2.4",
+        "debug": "^4.3.3",
+        "ethereum-cryptography": "0.1.3",
+        "mcl-wasm": "^0.7.1",
+        "rustbn.js": "~0.2.0"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-statemanager": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-statemanager/-/ethereumjs-statemanager-1.0.0.tgz",
+      "integrity": "sha512-jCtqFjcd2QejtuAMjQzbil/4NHf5aAWxUc+CvS0JclQpl+7M0bxMofR2AJdtz+P3u0ke2euhYREDiE7iSO31vQ==",
+      "dev": true,
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-trie": "^5.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "debug": "^4.3.3",
+        "ethereum-cryptography": "0.1.3",
+        "functional-red-black-tree": "^1.0.1"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-statemanager/node_modules/@nomicfoundation/ethereumjs-rlp": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+      "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+      "dev": true,
+      "bin": {
+        "rlp": "bin/rlp"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-trie": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-trie/-/ethereumjs-trie-5.0.0.tgz",
+      "integrity": "sha512-LIj5XdE+s+t6WSuq/ttegJzZ1vliwg6wlb+Y9f4RlBpuK35B9K02bO7xU+E6Rgg9RGptkWd6TVLdedTI4eNc2A==",
+      "dev": true,
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "ethereum-cryptography": "0.1.3",
+        "readable-stream": "^3.6.0"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-trie/node_modules/@nomicfoundation/ethereumjs-rlp": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+      "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+      "dev": true,
+      "bin": {
+        "rlp": "bin/rlp"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-tx": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-tx/-/ethereumjs-tx-4.0.0.tgz",
+      "integrity": "sha512-Gg3Lir2lNUck43Kp/3x6TfBNwcWC9Z1wYue9Nz3v4xjdcv6oDW9QSMJxqsKw9QEGoBBZ+gqwpW7+F05/rs/g1w==",
+      "dev": true,
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "ethereum-cryptography": "0.1.3"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-tx/node_modules/@nomicfoundation/ethereumjs-rlp": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+      "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+      "dev": true,
+      "bin": {
+        "rlp": "bin/rlp"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-util": {
+      "version": "8.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-util/-/ethereumjs-util-8.0.0.tgz",
+      "integrity": "sha512-2emi0NJ/HmTG+CGY58fa+DQuAoroFeSH9gKu9O6JnwTtlzJtgfTixuoOqLEgyyzZVvwfIpRueuePb8TonL1y+A==",
+      "dev": true,
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0-beta.2",
+        "ethereum-cryptography": "0.1.3"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-util/node_modules/@nomicfoundation/ethereumjs-rlp": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+      "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+      "dev": true,
+      "bin": {
+        "rlp": "bin/rlp"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-vm": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-vm/-/ethereumjs-vm-6.0.0.tgz",
+      "integrity": "sha512-JMPxvPQ3fzD063Sg3Tp+UdwUkVxMoo1uML6KSzFhMH3hoQi/LMuXBoEHAoW83/vyNS9BxEe6jm6LmT5xdeEJ6w==",
+      "dev": true,
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-block": "^4.0.0",
+        "@nomicfoundation/ethereumjs-blockchain": "^6.0.0",
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-evm": "^1.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-statemanager": "^1.0.0",
+        "@nomicfoundation/ethereumjs-trie": "^5.0.0",
+        "@nomicfoundation/ethereumjs-tx": "^4.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "@types/async-eventemitter": "^0.2.1",
+        "async-eventemitter": "^0.2.4",
+        "debug": "^4.3.3",
+        "ethereum-cryptography": "0.1.3",
+        "functional-red-black-tree": "^1.0.1",
+        "mcl-wasm": "^0.7.1",
+        "rustbn.js": "~0.2.0"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/ethereumjs-vm/node_modules/@nomicfoundation/ethereumjs-rlp": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+      "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+      "dev": true,
+      "bin": {
+        "rlp": "bin/rlp"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/@nomicfoundation/hardhat-chai-matchers": {
+      "version": "1.0.6",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/hardhat-chai-matchers/-/hardhat-chai-matchers-1.0.6.tgz",
+      "integrity": "sha512-f5ZMNmabZeZegEfuxn/0kW+mm7+yV7VNDxLpMOMGXWFJ2l/Ct3QShujzDRF9cOkK9Ui/hbDeOWGZqyQALDXVCQ==",
+      "dev": true,
+      "dependencies": {
+        "@ethersproject/abi": "^5.1.2",
+        "@types/chai-as-promised": "^7.1.3",
+        "chai-as-promised": "^7.1.1",
+        "deep-eql": "^4.0.1",
+        "ordinal": "^1.0.3"
+      },
+      "peerDependencies": {
+        "@nomiclabs/hardhat-ethers": "^2.0.0",
+        "chai": "^4.2.0",
+        "ethers": "^5.0.0",
+        "hardhat": "^2.9.4"
+      }
+    },
+    "node_modules/@nomicfoundation/hardhat-network-helpers": {
+      "version": "1.0.8",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/hardhat-network-helpers/-/hardhat-network-helpers-1.0.8.tgz",
+      "integrity": "sha512-MNqQbzUJZnCMIYvlniC3U+kcavz/PhhQSsY90tbEtUyMj/IQqsLwIRZa4ctjABh3Bz0KCh9OXUZ7Yk/d9hr45Q==",
+      "dev": true,
+      "dependencies": {
+        "ethereumjs-util": "^7.1.4"
+      },
+      "peerDependencies": {
+        "hardhat": "^2.9.5"
+      }
+    },
+    "node_modules/@nomicfoundation/solidity-analyzer": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer/-/solidity-analyzer-0.1.1.tgz",
+      "integrity": "sha512-1LMtXj1puAxyFusBgUIy5pZk3073cNXYnXUpuNKFghHbIit/xZgbk0AokpUADbNm3gyD6bFWl3LRFh3dhVdREg==",
+      "dev": true,
+      "engines": {
+        "node": ">= 12"
+      },
+      "optionalDependencies": {
+        "@nomicfoundation/solidity-analyzer-darwin-arm64": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-darwin-x64": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-freebsd-x64": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-linux-arm64-gnu": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-linux-arm64-musl": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-linux-x64-gnu": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-linux-x64-musl": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-win32-arm64-msvc": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-win32-ia32-msvc": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-win32-x64-msvc": "0.1.1"
+      }
+    },
+    "node_modules/@nomicfoundation/solidity-analyzer-darwin-arm64": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-darwin-arm64/-/solidity-analyzer-darwin-arm64-0.1.1.tgz",
+      "integrity": "sha512-KcTodaQw8ivDZyF+D76FokN/HdpgGpfjc/gFCImdLUyqB6eSWVaZPazMbeAjmfhx3R0zm/NYVzxwAokFKgrc0w==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "optional": true,
+      "os": [
+        "darwin"
+      ],
+      "engines": {
+        "node": ">= 10"
+      }
+    },
+    "node_modules/@nomicfoundation/solidity-analyzer-darwin-x64": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-darwin-x64/-/solidity-analyzer-darwin-x64-0.1.1.tgz",
+      "integrity": "sha512-XhQG4BaJE6cIbjAVtzGOGbK3sn1BO9W29uhk9J8y8fZF1DYz0Doj8QDMfpMu+A6TjPDs61lbsmeYodIDnfveSA==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "optional": true,
+      "os": [
+        "darwin"
+      ],
+      "engines": {
+        "node": ">= 10"
+      }
+    },
+    "node_modules/@nomicfoundation/solidity-analyzer-freebsd-x64": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-freebsd-x64/-/solidity-analyzer-freebsd-x64-0.1.1.tgz",
+      "integrity": "sha512-GHF1VKRdHW3G8CndkwdaeLkVBi5A9u2jwtlS7SLhBc8b5U/GcoL39Q+1CSO3hYqePNP+eV5YI7Zgm0ea6kMHoA==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "optional": true,
+      "os": [
+        "freebsd"
+      ],
+      "engines": {
+        "node": ">= 10"
+      }
+    },
+    "node_modules/@nomicfoundation/solidity-analyzer-linux-arm64-gnu": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-linux-arm64-gnu/-/solidity-analyzer-linux-arm64-gnu-0.1.1.tgz",
+      "integrity": "sha512-g4Cv2fO37ZsUENQ2vwPnZc2zRenHyAxHcyBjKcjaSmmkKrFr64yvzeNO8S3GBFCo90rfochLs99wFVGT/0owpg==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": ">= 10"
+      }
+    },
+    "node_modules/@nomicfoundation/solidity-analyzer-linux-arm64-musl": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-linux-arm64-musl/-/solidity-analyzer-linux-arm64-musl-0.1.1.tgz",
+      "integrity": "sha512-WJ3CE5Oek25OGE3WwzK7oaopY8xMw9Lhb0mlYuJl/maZVo+WtP36XoQTb7bW/i8aAdHW5Z+BqrHMux23pvxG3w==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": ">= 10"
+      }
+    },
+    "node_modules/@nomicfoundation/solidity-analyzer-linux-x64-gnu": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-linux-x64-gnu/-/solidity-analyzer-linux-x64-gnu-0.1.1.tgz",
+      "integrity": "sha512-5WN7leSr5fkUBBjE4f3wKENUy9HQStu7HmWqbtknfXkkil+eNWiBV275IOlpXku7v3uLsXTOKpnnGHJYI2qsdA==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": ">= 10"
+      }
+    },
+    "node_modules/@nomicfoundation/solidity-analyzer-linux-x64-musl": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-linux-x64-musl/-/solidity-analyzer-linux-x64-musl-0.1.1.tgz",
+      "integrity": "sha512-KdYMkJOq0SYPQMmErv/63CwGwMm5XHenEna9X9aB8mQmhDBrYrlAOSsIPgFCUSL0hjxE3xHP65/EPXR/InD2+w==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": ">= 10"
+      }
+    },
+    "node_modules/@nomicfoundation/solidity-analyzer-win32-arm64-msvc": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-win32-arm64-msvc/-/solidity-analyzer-win32-arm64-msvc-0.1.1.tgz",
+      "integrity": "sha512-VFZASBfl4qiBYwW5xeY20exWhmv6ww9sWu/krWSesv3q5hA0o1JuzmPHR4LPN6SUZj5vcqci0O6JOL8BPw+APg==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "optional": true,
+      "os": [
+        "win32"
+      ],
+      "engines": {
+        "node": ">= 10"
+      }
+    },
+    "node_modules/@nomicfoundation/solidity-analyzer-win32-ia32-msvc": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-win32-ia32-msvc/-/solidity-analyzer-win32-ia32-msvc-0.1.1.tgz",
+      "integrity": "sha512-JnFkYuyCSA70j6Si6cS1A9Gh1aHTEb8kOTBApp/c7NRTFGNMH8eaInKlyuuiIbvYFhlXW4LicqyYuWNNq9hkpQ==",
+      "cpu": [
+        "ia32"
+      ],
+      "dev": true,
+      "optional": true,
+      "os": [
+        "win32"
+      ],
+      "engines": {
+        "node": ">= 10"
+      }
+    },
+    "node_modules/@nomicfoundation/solidity-analyzer-win32-x64-msvc": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-win32-x64-msvc/-/solidity-analyzer-win32-x64-msvc-0.1.1.tgz",
+      "integrity": "sha512-HrVJr6+WjIXGnw3Q9u6KQcbZCtk0caVWhCdFADySvRyUxJ8PnzlaP+MhwNE8oyT8OZ6ejHBRrrgjSqDCFXGirw==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "optional": true,
+      "os": [
+        "win32"
+      ],
+      "engines": {
+        "node": ">= 10"
+      }
+    },
+    "node_modules/@nomiclabs/hardhat-ethers": {
+      "version": "2.2.2",
+      "resolved": "https://registry.npmjs.org/@nomiclabs/hardhat-ethers/-/hardhat-ethers-2.2.2.tgz",
+      "integrity": "sha512-NLDlDFL2us07C0jB/9wzvR0kuLivChJWCXTKcj3yqjZqMoYp7g7wwS157F70VHx/+9gHIBGzak5pKDwG8gEefA==",
+      "dev": true,
+      "peerDependencies": {
+        "ethers": "^5.0.0",
+        "hardhat": "^2.0.0"
+      }
+    },
+    "node_modules/@nomiclabs/hardhat-etherscan": {
+      "version": "3.1.7",
+      "resolved": "https://registry.npmjs.org/@nomiclabs/hardhat-etherscan/-/hardhat-etherscan-3.1.7.tgz",
+      "integrity": "sha512-tZ3TvSgpvsQ6B6OGmo1/Au6u8BrAkvs1mIC/eURA3xgIfznUZBhmpne8hv7BXUzw9xNL3fXdpOYgOQlVMTcoHQ==",
+      "dev": true,
+      "dependencies": {
+        "@ethersproject/abi": "^5.1.2",
+        "@ethersproject/address": "^5.0.2",
+        "cbor": "^8.1.0",
+        "chalk": "^2.4.2",
+        "debug": "^4.1.1",
+        "fs-extra": "^7.0.1",
+        "lodash": "^4.17.11",
+        "semver": "^6.3.0",
+        "table": "^6.8.0",
+        "undici": "^5.14.0"
+      },
+      "peerDependencies": {
+        "hardhat": "^2.0.4"
+      }
+    },
+    "node_modules/@nomiclabs/hardhat-etherscan/node_modules/ansi-styles": {
+      "version": "3.2.1",
+      "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+      "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+      "dev": true,
+      "dependencies": {
+        "color-convert": "^1.9.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@nomiclabs/hardhat-etherscan/node_modules/chalk": {
+      "version": "2.4.2",
+      "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+      "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+      "dev": true,
+      "dependencies": {
+        "ansi-styles": "^3.2.1",
+        "escape-string-regexp": "^1.0.5",
+        "supports-color": "^5.3.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@nomiclabs/hardhat-etherscan/node_modules/color-convert": {
+      "version": "1.9.3",
+      "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+      "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+      "dev": true,
+      "dependencies": {
+        "color-name": "1.1.3"
+      }
+    },
+    "node_modules/@nomiclabs/hardhat-etherscan/node_modules/color-name": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+      "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+      "dev": true
+    },
+    "node_modules/@nomiclabs/hardhat-etherscan/node_modules/has-flag": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+      "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@nomiclabs/hardhat-etherscan/node_modules/supports-color": {
+      "version": "5.5.0",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+      "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+      "dev": true,
+      "dependencies": {
+        "has-flag": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@nomiclabs/hardhat-truffle5": {
+      "version": "2.0.7",
+      "resolved": "https://registry.npmjs.org/@nomiclabs/hardhat-truffle5/-/hardhat-truffle5-2.0.7.tgz",
+      "integrity": "sha512-Pw8451IUZp1bTp0QqCHCYfCHs66sCnyxPcaorapu9mfOV9xnZsVaFdtutnhNEiXdiZwbed7LFKpRsde4BjFwig==",
+      "dev": true,
+      "dependencies": {
+        "@nomiclabs/truffle-contract": "^4.2.23",
+        "@types/chai": "^4.2.0",
+        "chai": "^4.2.0",
+        "ethereumjs-util": "^7.1.4",
+        "fs-extra": "^7.0.1"
+      },
+      "peerDependencies": {
+        "@nomiclabs/hardhat-web3": "^2.0.0",
+        "hardhat": "^2.6.4",
+        "web3": "^1.0.0-beta.36"
+      }
+    },
+    "node_modules/@nomiclabs/hardhat-web3": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/@nomiclabs/hardhat-web3/-/hardhat-web3-2.0.0.tgz",
+      "integrity": "sha512-zt4xN+D+fKl3wW2YlTX3k9APR3XZgPkxJYf36AcliJn3oujnKEVRZaHu0PhgLjO+gR+F/kiYayo9fgd2L8970Q==",
+      "dev": true,
+      "dependencies": {
+        "@types/bignumber.js": "^5.0.0"
+      },
+      "peerDependencies": {
+        "hardhat": "^2.0.0",
+        "web3": "^1.0.0-beta.36"
+      }
+    },
+    "node_modules/@nomiclabs/truffle-contract": {
+      "version": "4.5.10",
+      "resolved": "https://registry.npmjs.org/@nomiclabs/truffle-contract/-/truffle-contract-4.5.10.tgz",
+      "integrity": "sha512-nF/6InFV+0hUvutyFgsdOMCoYlr//2fJbRER4itxYtQtc4/O1biTwZIKRu+5l2J5Sq6LU2WX7vZHtDgQdhWxIQ==",
+      "dev": true,
+      "dependencies": {
+        "@ensdomains/ensjs": "^2.0.1",
+        "@truffle/blockchain-utils": "^0.1.3",
+        "@truffle/contract-schema": "^3.4.7",
+        "@truffle/debug-utils": "^6.0.22",
+        "@truffle/error": "^0.1.0",
+        "@truffle/interface-adapter": "^0.5.16",
+        "bignumber.js": "^7.2.1",
+        "ethereum-ens": "^0.8.0",
+        "ethers": "^4.0.0-beta.1",
+        "source-map-support": "^0.5.19"
+      },
+      "peerDependencies": {
+        "web3": "^1.2.1",
+        "web3-core-helpers": "^1.2.1",
+        "web3-core-promievent": "^1.2.1",
+        "web3-eth-abi": "^1.2.1",
+        "web3-utils": "^1.2.1"
+      }
+    },
+    "node_modules/@nomiclabs/truffle-contract/node_modules/bn.js": {
+      "version": "4.12.0",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+      "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+      "dev": true
+    },
+    "node_modules/@nomiclabs/truffle-contract/node_modules/ethers": {
+      "version": "4.0.49",
+      "resolved": "https://registry.npmjs.org/ethers/-/ethers-4.0.49.tgz",
+      "integrity": "sha512-kPltTvWiyu+OktYy1IStSO16i2e7cS9D9OxZ81q2UUaiNPVrm/RTcbxamCXF9VUSKzJIdJV68EAIhTEVBalRWg==",
+      "dev": true,
+      "dependencies": {
+        "aes-js": "3.0.0",
+        "bn.js": "^4.11.9",
+        "elliptic": "6.5.4",
+        "hash.js": "1.1.3",
+        "js-sha3": "0.5.7",
+        "scrypt-js": "2.0.4",
+        "setimmediate": "1.0.4",
+        "uuid": "2.0.1",
+        "xmlhttprequest": "1.8.0"
+      }
+    },
+    "node_modules/@nomiclabs/truffle-contract/node_modules/hash.js": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.3.tgz",
+      "integrity": "sha512-/UETyP0W22QILqS+6HowevwhEFJ3MBJnwTf75Qob9Wz9t0DPuisL8kW8YZMK62dHAKE1c1p+gY1TtOLY+USEHA==",
+      "dev": true,
+      "dependencies": {
+        "inherits": "^2.0.3",
+        "minimalistic-assert": "^1.0.0"
+      }
+    },
+    "node_modules/@nomiclabs/truffle-contract/node_modules/js-sha3": {
+      "version": "0.5.7",
+      "resolved": "https://registry.npmjs.org/js-sha3/-/js-sha3-0.5.7.tgz",
+      "integrity": "sha512-GII20kjaPX0zJ8wzkTbNDYMY7msuZcTWk8S5UOh6806Jq/wz1J8/bnr8uGU0DAUmYDjj2Mr4X1cW8v/GLYnR+g==",
+      "dev": true
+    },
+    "node_modules/@nomiclabs/truffle-contract/node_modules/scrypt-js": {
+      "version": "2.0.4",
+      "resolved": "https://registry.npmjs.org/scrypt-js/-/scrypt-js-2.0.4.tgz",
+      "integrity": "sha512-4KsaGcPnuhtCZQCxFxN3GVYIhKFPTdLd8PLC552XwbMndtD0cjRFAhDuuydXQ0h08ZfPgzqe6EKHozpuH74iDw==",
+      "dev": true
+    },
+    "node_modules/@nomiclabs/truffle-contract/node_modules/setimmediate": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.4.tgz",
+      "integrity": "sha512-/TjEmXQVEzdod/FFskf3o7oOAsGhHf2j1dZqRFbDzq4F3mvvxflIIi4Hd3bLQE9y/CpwqfSQam5JakI/mi3Pog==",
+      "dev": true
+    },
+    "node_modules/@openzeppelin/contracts": {
+      "version": "4.9.3",
+      "resolved": "https://registry.npmjs.org/@openzeppelin/contracts/-/contracts-4.9.3.tgz",
+      "integrity": "sha512-He3LieZ1pP2TNt5JbkPA4PNT9WC3gOTOlDcFGJW4Le4QKqwmiNJCRt44APfxMxvq7OugU/cqYuPcSBzOw38DAg==",
+      "dev": true
+    },
+    "node_modules/@scure/base": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/@scure/base/-/base-1.1.1.tgz",
+      "integrity": "sha512-ZxOhsSyxYwLJj3pLZCefNitxsj093tb2vq90mp2txoYeBqbcjDjqFhyM8eUjq/uFm6zJ+mUuqxlS2FkuSY1MTA==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://paulmillr.com/funding/"
+        }
+      ]
+    },
+    "node_modules/@scure/bip32": {
+      "version": "1.1.5",
+      "resolved": "https://registry.npmjs.org/@scure/bip32/-/bip32-1.1.5.tgz",
+      "integrity": "sha512-XyNh1rB0SkEqd3tXcXMi+Xe1fvg+kUIcoRIEujP1Jgv7DqW2r9lg3Ah0NkFaCs9sTkQAQA8kw7xiRXzENi9Rtw==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://paulmillr.com/funding/"
+        }
+      ],
+      "dependencies": {
+        "@noble/hashes": "~1.2.0",
+        "@noble/secp256k1": "~1.7.0",
+        "@scure/base": "~1.1.0"
+      }
+    },
+    "node_modules/@scure/bip39": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/@scure/bip39/-/bip39-1.1.1.tgz",
+      "integrity": "sha512-t+wDck2rVkh65Hmv280fYdVdY25J9YeEUIgn2LG1WM6gxFkGzcksoDiUkWVpVp3Oex9xGC68JU2dSbUfwZ2jPg==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://paulmillr.com/funding/"
+        }
+      ],
+      "dependencies": {
+        "@noble/hashes": "~1.2.0",
+        "@scure/base": "~1.1.0"
+      }
+    },
+    "node_modules/@sentry/core": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/core/-/core-5.30.0.tgz",
+      "integrity": "sha512-TmfrII8w1PQZSZgPpUESqjB+jC6MvZJZdLtE/0hZ+SrnKhW3x5WlYLvTXZpcWePYBku7rl2wn1RZu6uT0qCTeg==",
+      "dev": true,
+      "dependencies": {
+        "@sentry/hub": "5.30.0",
+        "@sentry/minimal": "5.30.0",
+        "@sentry/types": "5.30.0",
+        "@sentry/utils": "5.30.0",
+        "tslib": "^1.9.3"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/@sentry/hub": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/hub/-/hub-5.30.0.tgz",
+      "integrity": "sha512-2tYrGnzb1gKz2EkMDQcfLrDTvmGcQPuWxLnJKXJvYTQDGLlEvi2tWz1VIHjunmOvJrB5aIQLhm+dcMRwFZDCqQ==",
+      "dev": true,
+      "dependencies": {
+        "@sentry/types": "5.30.0",
+        "@sentry/utils": "5.30.0",
+        "tslib": "^1.9.3"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/@sentry/minimal": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/minimal/-/minimal-5.30.0.tgz",
+      "integrity": "sha512-BwWb/owZKtkDX+Sc4zCSTNcvZUq7YcH3uAVlmh/gtR9rmUvbzAA3ewLuB3myi4wWRAMEtny6+J/FN/x+2wn9Xw==",
+      "dev": true,
+      "dependencies": {
+        "@sentry/hub": "5.30.0",
+        "@sentry/types": "5.30.0",
+        "tslib": "^1.9.3"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/@sentry/node": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/node/-/node-5.30.0.tgz",
+      "integrity": "sha512-Br5oyVBF0fZo6ZS9bxbJZG4ApAjRqAnqFFurMVJJdunNb80brh7a5Qva2kjhm+U6r9NJAB5OmDyPkA1Qnt+QVg==",
+      "dev": true,
+      "dependencies": {
+        "@sentry/core": "5.30.0",
+        "@sentry/hub": "5.30.0",
+        "@sentry/tracing": "5.30.0",
+        "@sentry/types": "5.30.0",
+        "@sentry/utils": "5.30.0",
+        "cookie": "^0.4.1",
+        "https-proxy-agent": "^5.0.0",
+        "lru_map": "^0.3.3",
+        "tslib": "^1.9.3"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/@sentry/node/node_modules/cookie": {
+      "version": "0.4.2",
+      "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.2.tgz",
+      "integrity": "sha512-aSWTXFzaKWkvHO1Ny/s+ePFpvKsPnjc551iI41v3ny/ow6tBG5Vd+FuqGNhh1LxOmVzOlGUriIlOaokOvhaStA==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/@sentry/tracing": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/tracing/-/tracing-5.30.0.tgz",
+      "integrity": "sha512-dUFowCr0AIMwiLD7Fs314Mdzcug+gBVo/+NCMyDw8tFxJkwWAKl7Qa2OZxLQ0ZHjakcj1hNKfCQJ9rhyfOl4Aw==",
+      "dev": true,
+      "dependencies": {
+        "@sentry/hub": "5.30.0",
+        "@sentry/minimal": "5.30.0",
+        "@sentry/types": "5.30.0",
+        "@sentry/utils": "5.30.0",
+        "tslib": "^1.9.3"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/@sentry/types": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/types/-/types-5.30.0.tgz",
+      "integrity": "sha512-R8xOqlSTZ+htqrfteCWU5Nk0CDN5ApUTvrlvBuiH1DyP6czDZ4ktbZB0hAgBlVcK0U+qpD3ag3Tqqpa5Q67rPw==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/@sentry/utils": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/utils/-/utils-5.30.0.tgz",
+      "integrity": "sha512-zaYmoH0NWWtvnJjC9/CBseXMtKHm/tm40sz3YfJRxeQjyzRqNQPgivpd9R/oDJCYj999mzdW382p/qi2ypjLww==",
+      "dev": true,
+      "dependencies": {
+        "@sentry/types": "5.30.0",
+        "tslib": "^1.9.3"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/@sindresorhus/is": {
+      "version": "4.6.0",
+      "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-4.6.0.tgz",
+      "integrity": "sha512-t09vSN3MdfsyCHoFcTRCH/iUtG7OJ0CsjzB8cjAmKc/va/kIgeDI/TxsigdncE/4be734m0cvIYwNaV4i2XqAw==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sindresorhus/is?sponsor=1"
+      }
+    },
+    "node_modules/@solidity-parser/parser": {
+      "version": "0.14.5",
+      "resolved": "https://registry.npmjs.org/@solidity-parser/parser/-/parser-0.14.5.tgz",
+      "integrity": "sha512-6dKnHZn7fg/iQATVEzqyUOyEidbn05q7YA2mQ9hC0MMXhhV3/JrsxmFSYZAcr7j1yUP700LLhTruvJ3MiQmjJg==",
+      "dev": true,
+      "dependencies": {
+        "antlr4ts": "^0.5.0-alpha.4"
+      }
+    },
+    "node_modules/@szmarczak/http-timer": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-5.0.1.tgz",
+      "integrity": "sha512-+PmQX0PiAYPMeVYe237LJAYvOMYW1j2rH5YROyS3b4CTVJum34HfRvKvAzozHAQG0TnHNdUfY9nCeUyRAs//cw==",
+      "dev": true,
+      "dependencies": {
+        "defer-to-connect": "^2.0.1"
+      },
+      "engines": {
+        "node": ">=14.16"
+      }
+    },
+    "node_modules/@tenderly/hardhat-tenderly": {
+      "version": "1.6.1",
+      "resolved": "https://registry.npmjs.org/@tenderly/hardhat-tenderly/-/hardhat-tenderly-1.6.1.tgz",
+      "integrity": "sha512-VhnOcRVB8J1mZk5QUifthsidyEb4p6Qj0nyy07qdnF8GL/kcVDBgqluyapGS1hUhp/SVQqpKHcFTWbVJ/B0l4w==",
+      "dev": true,
+      "dependencies": {
+        "@ethersproject/bignumber": "^5.7.0",
+        "@nomiclabs/hardhat-ethers": "^2.1.1",
+        "axios": "^0.27.2",
+        "ethers": "^5.7.0",
+        "fs-extra": "^10.1.0",
+        "hardhat": "^2.10.2",
+        "hardhat-deploy": "^0.11.14",
+        "js-yaml": "^4.1.0",
+        "tenderly": "^0.4.0",
+        "tslog": "^4.3.1"
+      },
+      "peerDependencies": {
+        "hardhat": "^2.10.2",
+        "tenderly": "^0.4.0"
+      }
+    },
+    "node_modules/@tenderly/hardhat-tenderly/node_modules/fs-extra": {
+      "version": "10.1.0",
+      "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-10.1.0.tgz",
+      "integrity": "sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ==",
+      "dev": true,
+      "dependencies": {
+        "graceful-fs": "^4.2.0",
+        "jsonfile": "^6.0.1",
+        "universalify": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=12"
+      }
+    },
+    "node_modules/@tenderly/hardhat-tenderly/node_modules/jsonfile": {
+      "version": "6.1.0",
+      "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz",
+      "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==",
+      "dev": true,
+      "dependencies": {
+        "universalify": "^2.0.0"
+      },
+      "optionalDependencies": {
+        "graceful-fs": "^4.1.6"
+      }
+    },
+    "node_modules/@tenderly/hardhat-tenderly/node_modules/universalify": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz",
+      "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==",
+      "dev": true,
+      "engines": {
+        "node": ">= 10.0.0"
+      }
+    },
+    "node_modules/@truffle/abi-utils": {
+      "version": "0.3.9",
+      "resolved": "https://registry.npmjs.org/@truffle/abi-utils/-/abi-utils-0.3.9.tgz",
+      "integrity": "sha512-G5dqgwRHx5zwlXjz3QT8OJVfB2cOqWwD6DwKso0KttUt/zejhCjnkKq72rSgyeLMkz7wBB9ERLOsupLBILM8MA==",
+      "dev": true,
+      "dependencies": {
+        "change-case": "3.0.2",
+        "fast-check": "3.1.1",
+        "web3-utils": "1.8.2"
+      }
+    },
+    "node_modules/@truffle/blockchain-utils": {
+      "version": "0.1.6",
+      "resolved": "https://registry.npmjs.org/@truffle/blockchain-utils/-/blockchain-utils-0.1.6.tgz",
+      "integrity": "sha512-SldoNRIFSm3+HMBnSc2jFsu5TWDkCN4X6vL3wrd0t6DIeF7nD6EoPPjxwbFSoqCnkkRxMuZeL6sUx7UMJS/wSA==",
+      "dev": true
+    },
+    "node_modules/@truffle/codec": {
+      "version": "0.14.16",
+      "resolved": "https://registry.npmjs.org/@truffle/codec/-/codec-0.14.16.tgz",
+      "integrity": "sha512-a9UY3n/FnkKN3Q4zOuMFOOcLWb80mdknj+voim4vvXYtJm1aAZQZE5sG9aLnMBTl4TiGLzUtfNDVYY7WgWgDag==",
+      "dev": true,
+      "dependencies": {
+        "@truffle/abi-utils": "^0.3.9",
+        "@truffle/compile-common": "^0.9.4",
+        "big.js": "^6.0.3",
+        "bn.js": "^5.1.3",
+        "cbor": "^5.2.0",
+        "debug": "^4.3.1",
+        "lodash": "^4.17.21",
+        "semver": "7.3.7",
+        "utf8": "^3.0.0",
+        "web3-utils": "1.8.2"
+      }
+    },
+    "node_modules/@truffle/codec/node_modules/bignumber.js": {
+      "version": "9.1.1",
+      "resolved": "https://registry.npmjs.org/bignumber.js/-/bignumber.js-9.1.1.tgz",
+      "integrity": "sha512-pHm4LsMJ6lzgNGVfZHjMoO8sdoRhOzOH4MLmY65Jg70bpxCKu5iOHNJyfF6OyvYw7t8Fpf35RuzUyqnQsj8Vig==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/@truffle/codec/node_modules/cbor": {
+      "version": "5.2.0",
+      "resolved": "https://registry.npmjs.org/cbor/-/cbor-5.2.0.tgz",
+      "integrity": "sha512-5IMhi9e1QU76ppa5/ajP1BmMWZ2FHkhAhjeVKQ/EFCgYSEaeVaoGtL7cxJskf9oCCk+XjzaIdc3IuU/dbA/o2A==",
+      "dev": true,
+      "dependencies": {
+        "bignumber.js": "^9.0.1",
+        "nofilter": "^1.0.4"
+      },
+      "engines": {
+        "node": ">=6.0.0"
+      }
+    },
+    "node_modules/@truffle/codec/node_modules/nofilter": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/nofilter/-/nofilter-1.0.4.tgz",
+      "integrity": "sha512-N8lidFp+fCz+TD51+haYdbDGrcBWwuHX40F5+z0qkUjMJ5Tp+rdSuAkMJ9N9eoolDlEVTf6u5icM+cNKkKW2mA==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/@truffle/codec/node_modules/semver": {
+      "version": "7.3.7",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz",
+      "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==",
+      "dev": true,
+      "dependencies": {
+        "lru-cache": "^6.0.0"
+      },
+      "bin": {
+        "semver": "bin/semver.js"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/@truffle/compile-common": {
+      "version": "0.9.4",
+      "resolved": "https://registry.npmjs.org/@truffle/compile-common/-/compile-common-0.9.4.tgz",
+      "integrity": "sha512-mnqJB/hLiPHNf+WKwt/2MH6lv34xSG/SFCib7+ckAklutUqVLeFo8EwQxinuHNkU7LY0C+YgZXhK1WTCO5YRJQ==",
+      "dev": true,
+      "dependencies": {
+        "@truffle/error": "^0.2.0",
+        "colors": "1.4.0"
+      }
+    },
+    "node_modules/@truffle/compile-common/node_modules/@truffle/error": {
+      "version": "0.2.0",
+      "resolved": "https://registry.npmjs.org/@truffle/error/-/error-0.2.0.tgz",
+      "integrity": "sha512-Fe0/z4WWb7IP2gBnv3l6zqP87Y0kSMs7oiSLakKJq17q3GUunrHSdioKuNspdggxkXIBhEQLhi8C+LJdwmHKWQ==",
+      "dev": true
+    },
+    "node_modules/@truffle/contract-schema": {
+      "version": "3.4.13",
+      "resolved": "https://registry.npmjs.org/@truffle/contract-schema/-/contract-schema-3.4.13.tgz",
+      "integrity": "sha512-emG7upuryYFrsPDbHqeASPWXL824M1tinhQwSPG0phSoa3g+RX9fUNNN/VPmF3tSkXLWUMhRnb7ehxnaCuRbZg==",
+      "dev": true,
+      "dependencies": {
+        "ajv": "^6.10.0",
+        "debug": "^4.3.1"
+      }
+    },
+    "node_modules/@truffle/contract-schema/node_modules/ajv": {
+      "version": "6.12.6",
+      "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+      "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+      "dev": true,
+      "dependencies": {
+        "fast-deep-equal": "^3.1.1",
+        "fast-json-stable-stringify": "^2.0.0",
+        "json-schema-traverse": "^0.4.1",
+        "uri-js": "^4.2.2"
+      },
+      "funding": {
+        "type": "github",
+        "url": "https://github.com/sponsors/epoberezkin"
+      }
+    },
+    "node_modules/@truffle/contract-schema/node_modules/json-schema-traverse": {
+      "version": "0.4.1",
+      "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+      "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+      "dev": true
+    },
+    "node_modules/@truffle/debug-utils": {
+      "version": "6.0.47",
+      "resolved": "https://registry.npmjs.org/@truffle/debug-utils/-/debug-utils-6.0.47.tgz",
+      "integrity": "sha512-bUjdzLPdEKtoUCDzaXkrOoi+PbyAJlMBzGequBK8tirT7xL9bCP2Pd/WxvnmRd7AnfroxGNvXwVXWTItW5SMWQ==",
+      "dev": true,
+      "dependencies": {
+        "@truffle/codec": "^0.14.16",
+        "@trufflesuite/chromafi": "^3.0.0",
+        "bn.js": "^5.1.3",
+        "chalk": "^2.4.2",
+        "debug": "^4.3.1",
+        "highlightjs-solidity": "^2.0.6"
+      }
+    },
+    "node_modules/@truffle/debug-utils/node_modules/ansi-styles": {
+      "version": "3.2.1",
+      "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+      "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+      "dev": true,
+      "dependencies": {
+        "color-convert": "^1.9.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@truffle/debug-utils/node_modules/chalk": {
+      "version": "2.4.2",
+      "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+      "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+      "dev": true,
+      "dependencies": {
+        "ansi-styles": "^3.2.1",
+        "escape-string-regexp": "^1.0.5",
+        "supports-color": "^5.3.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@truffle/debug-utils/node_modules/color-convert": {
+      "version": "1.9.3",
+      "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+      "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+      "dev": true,
+      "dependencies": {
+        "color-name": "1.1.3"
+      }
+    },
+    "node_modules/@truffle/debug-utils/node_modules/color-name": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+      "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+      "dev": true
+    },
+    "node_modules/@truffle/debug-utils/node_modules/has-flag": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+      "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@truffle/debug-utils/node_modules/supports-color": {
+      "version": "5.5.0",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+      "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+      "dev": true,
+      "dependencies": {
+        "has-flag": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@truffle/error": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@truffle/error/-/error-0.1.1.tgz",
+      "integrity": "sha512-sE7c9IHIGdbK4YayH4BC8i8qMjoAOeg6nUXUDZZp8wlU21/EMpaG+CLx+KqcIPyR+GSWIW3Dm0PXkr2nlggFDA==",
+      "dev": true
+    },
+    "node_modules/@truffle/interface-adapter": {
+      "version": "0.5.29",
+      "resolved": "https://registry.npmjs.org/@truffle/interface-adapter/-/interface-adapter-0.5.29.tgz",
+      "integrity": "sha512-6UlJ+f87z7y/dWk9UfbIU+4e80iRsp8h03LEiE5B+PvZbr6cuMjLJUBtBBQZMo3+xrIcS/2u3p5hOxW8OJm8tw==",
+      "dev": true,
+      "dependencies": {
+        "bn.js": "^5.1.3",
+        "ethers": "^4.0.32",
+        "web3": "1.8.2"
+      }
+    },
+    "node_modules/@truffle/interface-adapter/node_modules/ethers": {
+      "version": "4.0.49",
+      "resolved": "https://registry.npmjs.org/ethers/-/ethers-4.0.49.tgz",
+      "integrity": "sha512-kPltTvWiyu+OktYy1IStSO16i2e7cS9D9OxZ81q2UUaiNPVrm/RTcbxamCXF9VUSKzJIdJV68EAIhTEVBalRWg==",
+      "dev": true,
+      "dependencies": {
+        "aes-js": "3.0.0",
+        "bn.js": "^4.11.9",
+        "elliptic": "6.5.4",
+        "hash.js": "1.1.3",
+        "js-sha3": "0.5.7",
+        "scrypt-js": "2.0.4",
+        "setimmediate": "1.0.4",
+        "uuid": "2.0.1",
+        "xmlhttprequest": "1.8.0"
+      }
+    },
+    "node_modules/@truffle/interface-adapter/node_modules/ethers/node_modules/bn.js": {
+      "version": "4.12.0",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+      "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+      "dev": true
+    },
+    "node_modules/@truffle/interface-adapter/node_modules/hash.js": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.3.tgz",
+      "integrity": "sha512-/UETyP0W22QILqS+6HowevwhEFJ3MBJnwTf75Qob9Wz9t0DPuisL8kW8YZMK62dHAKE1c1p+gY1TtOLY+USEHA==",
+      "dev": true,
+      "dependencies": {
+        "inherits": "^2.0.3",
+        "minimalistic-assert": "^1.0.0"
+      }
+    },
+    "node_modules/@truffle/interface-adapter/node_modules/js-sha3": {
+      "version": "0.5.7",
+      "resolved": "https://registry.npmjs.org/js-sha3/-/js-sha3-0.5.7.tgz",
+      "integrity": "sha512-GII20kjaPX0zJ8wzkTbNDYMY7msuZcTWk8S5UOh6806Jq/wz1J8/bnr8uGU0DAUmYDjj2Mr4X1cW8v/GLYnR+g==",
+      "dev": true
+    },
+    "node_modules/@truffle/interface-adapter/node_modules/scrypt-js": {
+      "version": "2.0.4",
+      "resolved": "https://registry.npmjs.org/scrypt-js/-/scrypt-js-2.0.4.tgz",
+      "integrity": "sha512-4KsaGcPnuhtCZQCxFxN3GVYIhKFPTdLd8PLC552XwbMndtD0cjRFAhDuuydXQ0h08ZfPgzqe6EKHozpuH74iDw==",
+      "dev": true
+    },
+    "node_modules/@truffle/interface-adapter/node_modules/setimmediate": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.4.tgz",
+      "integrity": "sha512-/TjEmXQVEzdod/FFskf3o7oOAsGhHf2j1dZqRFbDzq4F3mvvxflIIi4Hd3bLQE9y/CpwqfSQam5JakI/mi3Pog==",
+      "dev": true
+    },
+    "node_modules/@trufflesuite/chromafi": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/@trufflesuite/chromafi/-/chromafi-3.0.0.tgz",
+      "integrity": "sha512-oqWcOqn8nT1bwlPPfidfzS55vqcIDdpfzo3HbU9EnUmcSTX+I8z0UyUFI3tZQjByVJulbzxHxUGS3ZJPwK/GPQ==",
+      "dev": true,
+      "dependencies": {
+        "camelcase": "^4.1.0",
+        "chalk": "^2.3.2",
+        "cheerio": "^1.0.0-rc.2",
+        "detect-indent": "^5.0.0",
+        "highlight.js": "^10.4.1",
+        "lodash.merge": "^4.6.2",
+        "strip-ansi": "^4.0.0",
+        "strip-indent": "^2.0.0"
+      }
+    },
+    "node_modules/@trufflesuite/chromafi/node_modules/ansi-regex": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz",
+      "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@trufflesuite/chromafi/node_modules/ansi-styles": {
+      "version": "3.2.1",
+      "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+      "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+      "dev": true,
+      "dependencies": {
+        "color-convert": "^1.9.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@trufflesuite/chromafi/node_modules/camelcase": {
+      "version": "4.1.0",
+      "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-4.1.0.tgz",
+      "integrity": "sha512-FxAv7HpHrXbh3aPo4o2qxHay2lkLY3x5Mw3KeE4KQE8ysVfziWeRZDwcjauvwBSGEC/nXUPzZy8zeh4HokqOnw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@trufflesuite/chromafi/node_modules/chalk": {
+      "version": "2.4.2",
+      "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+      "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+      "dev": true,
+      "dependencies": {
+        "ansi-styles": "^3.2.1",
+        "escape-string-regexp": "^1.0.5",
+        "supports-color": "^5.3.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@trufflesuite/chromafi/node_modules/color-convert": {
+      "version": "1.9.3",
+      "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+      "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+      "dev": true,
+      "dependencies": {
+        "color-name": "1.1.3"
+      }
+    },
+    "node_modules/@trufflesuite/chromafi/node_modules/color-name": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+      "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+      "dev": true
+    },
+    "node_modules/@trufflesuite/chromafi/node_modules/has-flag": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+      "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@trufflesuite/chromafi/node_modules/strip-ansi": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz",
+      "integrity": "sha512-4XaJ2zQdCzROZDivEVIDPkcQn8LMFSa8kj8Gxb/Lnwzv9A8VctNZ+lfivC/sV3ivW8ElJTERXZoPBRrZKkNKow==",
+      "dev": true,
+      "dependencies": {
+        "ansi-regex": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@trufflesuite/chromafi/node_modules/supports-color": {
+      "version": "5.5.0",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+      "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+      "dev": true,
+      "dependencies": {
+        "has-flag": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/@tsconfig/node10": {
+      "version": "1.0.9",
+      "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.9.tgz",
+      "integrity": "sha512-jNsYVVxU8v5g43Erja32laIDHXeoNvFEpX33OK4d6hljo3jDhCBDhx5dhCCTMWUojscpAagGiRkBKxpdl9fxqA==",
+      "dev": true
+    },
+    "node_modules/@tsconfig/node12": {
+      "version": "1.0.11",
+      "resolved": "https://registry.npmjs.org/@tsconfig/node12/-/node12-1.0.11.tgz",
+      "integrity": "sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==",
+      "dev": true
+    },
+    "node_modules/@tsconfig/node14": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/@tsconfig/node14/-/node14-1.0.3.tgz",
+      "integrity": "sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==",
+      "dev": true
+    },
+    "node_modules/@tsconfig/node16": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/@tsconfig/node16/-/node16-1.0.3.tgz",
+      "integrity": "sha512-yOlFc+7UtL/89t2ZhjPvvB/DeAr3r+Dq58IgzsFkOAvVC6NMJXmCGjbptdXdR9qsX7pKcTL+s87FtYREi2dEEQ==",
+      "dev": true
+    },
+    "node_modules/@typechain/ethers-v5": {
+      "version": "10.2.0",
+      "resolved": "https://registry.npmjs.org/@typechain/ethers-v5/-/ethers-v5-10.2.0.tgz",
+      "integrity": "sha512-ikaq0N/w9fABM+G01OFmU3U3dNnyRwEahkdvi9mqy1a3XwKiPZaF/lu54OcNaEWnpvEYyhhS0N7buCtLQqC92w==",
+      "dev": true,
+      "dependencies": {
+        "lodash": "^4.17.15",
+        "ts-essentials": "^7.0.1"
+      },
+      "peerDependencies": {
+        "@ethersproject/abi": "^5.0.0",
+        "@ethersproject/bytes": "^5.0.0",
+        "@ethersproject/providers": "^5.0.0",
+        "ethers": "^5.1.3",
+        "typechain": "^8.1.1",
+        "typescript": ">=4.3.0"
+      }
+    },
+    "node_modules/@types/async-eventemitter": {
+      "version": "0.2.1",
+      "resolved": "https://registry.npmjs.org/@types/async-eventemitter/-/async-eventemitter-0.2.1.tgz",
+      "integrity": "sha512-M2P4Ng26QbAeITiH7w1d7OxtldgfAe0wobpyJzVK/XOb0cUGKU2R4pfAhqcJBXAe2ife5ZOhSv4wk7p+ffURtg==",
+      "dev": true
+    },
+    "node_modules/@types/bignumber.js": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/@types/bignumber.js/-/bignumber.js-5.0.0.tgz",
+      "integrity": "sha512-0DH7aPGCClywOFaxxjE6UwpN2kQYe9LwuDQMv+zYA97j5GkOMo8e66LYT+a8JYU7jfmUFRZLa9KycxHDsKXJCA==",
+      "deprecated": "This is a stub types definition for bignumber.js (https://github.com/MikeMcl/bignumber.js/). bignumber.js provides its own type definitions, so you don't need @types/bignumber.js installed!",
+      "dev": true,
+      "dependencies": {
+        "bignumber.js": "*"
+      }
+    },
+    "node_modules/@types/bn.js": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/@types/bn.js/-/bn.js-5.1.1.tgz",
+      "integrity": "sha512-qNrYbZqMx0uJAfKnKclPh+dTwK33KfLHYqtyODwd5HnXOjnkhc4qgn3BrK6RWyGZm5+sIFE7Q7Vz6QQtJB7w7g==",
+      "dev": true,
+      "dependencies": {
+        "@types/node": "*"
+      }
+    },
+    "node_modules/@types/cacheable-request": {
+      "version": "6.0.3",
+      "resolved": "https://registry.npmjs.org/@types/cacheable-request/-/cacheable-request-6.0.3.tgz",
+      "integrity": "sha512-IQ3EbTzGxIigb1I3qPZc1rWJnH0BmSKv5QYTalEwweFvyBDLSAe24zP0le/hyi7ecGfZVlIVAg4BZqb8WBwKqw==",
+      "dev": true,
+      "dependencies": {
+        "@types/http-cache-semantics": "*",
+        "@types/keyv": "^3.1.4",
+        "@types/node": "*",
+        "@types/responselike": "^1.0.0"
+      }
+    },
+    "node_modules/@types/chai": {
+      "version": "4.3.4",
+      "resolved": "https://registry.npmjs.org/@types/chai/-/chai-4.3.4.tgz",
+      "integrity": "sha512-KnRanxnpfpjUTqTCXslZSEdLfXExwgNxYPdiO2WGUj8+HDjFi8R3k5RVKPeSCzLjCcshCAtVO2QBbVuAV4kTnw==",
+      "dev": true
+    },
+    "node_modules/@types/chai-as-promised": {
+      "version": "7.1.5",
+      "resolved": "https://registry.npmjs.org/@types/chai-as-promised/-/chai-as-promised-7.1.5.tgz",
+      "integrity": "sha512-jStwss93SITGBwt/niYrkf2C+/1KTeZCZl1LaeezTlqppAKeoQC7jxyqYuP72sxBGKCIbw7oHgbYssIRzT5FCQ==",
+      "dev": true,
+      "dependencies": {
+        "@types/chai": "*"
+      }
+    },
+    "node_modules/@types/chai-string": {
+      "version": "1.4.2",
+      "resolved": "https://registry.npmjs.org/@types/chai-string/-/chai-string-1.4.2.tgz",
+      "integrity": "sha512-ld/1hV5qcPRGuwlPdvRfvM3Ka/iofOk2pH4VkasK4b1JJP1LjNmWWn0LsISf6RRzyhVOvs93rb9tM09e+UuF8Q==",
+      "dev": true,
+      "dependencies": {
+        "@types/chai": "*"
+      }
+    },
+    "node_modules/@types/concat-stream": {
+      "version": "1.6.1",
+      "resolved": "https://registry.npmjs.org/@types/concat-stream/-/concat-stream-1.6.1.tgz",
+      "integrity": "sha512-eHE4cQPoj6ngxBZMvVf6Hw7Mh4jMW4U9lpGmS5GBPB9RYxlFg+CHaVN7ErNY4W9XfLIEn20b4VDYaIrbq0q4uA==",
+      "dev": true,
+      "dependencies": {
+        "@types/node": "*"
+      }
+    },
+    "node_modules/@types/form-data": {
+      "version": "0.0.33",
+      "resolved": "https://registry.npmjs.org/@types/form-data/-/form-data-0.0.33.tgz",
+      "integrity": "sha512-8BSvG1kGm83cyJITQMZSulnl6QV8jqAGreJsc5tPu1Jq0vTSOiY/k24Wx82JRpWwZSqrala6sd5rWi6aNXvqcw==",
+      "dev": true,
+      "dependencies": {
+        "@types/node": "*"
+      }
+    },
+    "node_modules/@types/glob": {
+      "version": "7.2.0",
+      "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.2.0.tgz",
+      "integrity": "sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==",
+      "dev": true,
+      "dependencies": {
+        "@types/minimatch": "*",
+        "@types/node": "*"
+      }
+    },
+    "node_modules/@types/http-cache-semantics": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/@types/http-cache-semantics/-/http-cache-semantics-4.0.1.tgz",
+      "integrity": "sha512-SZs7ekbP8CN0txVG2xVRH6EgKmEm31BOxA07vkFaETzZz1xh+cbt8BcI0slpymvwhx5dlFnQG2rTlPVQn+iRPQ==",
+      "dev": true
+    },
+    "node_modules/@types/json-schema": {
+      "version": "7.0.11",
+      "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.11.tgz",
+      "integrity": "sha512-wOuvG1SN4Us4rez+tylwwwCV1psiNVOkJeM3AUWUNWg/jDQY2+HE/444y5gc+jBmRqASOm2Oeh5c1axHobwRKQ==",
+      "dev": true
+    },
+    "node_modules/@types/json5": {
+      "version": "0.0.29",
+      "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz",
+      "integrity": "sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==",
+      "dev": true
+    },
+    "node_modules/@types/keyv": {
+      "version": "3.1.4",
+      "resolved": "https://registry.npmjs.org/@types/keyv/-/keyv-3.1.4.tgz",
+      "integrity": "sha512-BQ5aZNSCpj7D6K2ksrRCTmKRLEpnPvWDiLPfoGyhZ++8YtiK9d/3DBKPJgry359X/P1PfruyYwvnvwFjuEiEIg==",
+      "dev": true,
+      "dependencies": {
+        "@types/node": "*"
+      }
+    },
+    "node_modules/@types/lru-cache": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/@types/lru-cache/-/lru-cache-5.1.1.tgz",
+      "integrity": "sha512-ssE3Vlrys7sdIzs5LOxCzTVMsU7i9oa/IaW92wF32JFb3CVczqOkru2xspuKczHEbG3nvmPY7IFqVmGGHdNbYw==",
+      "dev": true
+    },
+    "node_modules/@types/minimatch": {
+      "version": "5.1.2",
+      "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-5.1.2.tgz",
+      "integrity": "sha512-K0VQKziLUWkVKiRVrx4a40iPaxTUefQmjtkQofBkYRcoaaL/8rhwDWww9qWbrgicNOgnpIsMxyNIUM4+n6dUIA==",
+      "dev": true
+    },
+    "node_modules/@types/mocha": {
+      "version": "10.0.1",
+      "resolved": "https://registry.npmjs.org/@types/mocha/-/mocha-10.0.1.tgz",
+      "integrity": "sha512-/fvYntiO1GeICvqbQ3doGDIP97vWmvFt83GKguJ6prmQM2iXZfFcq6YE8KteFyRtX2/h5Hf91BYvPodJKFYv5Q==",
+      "dev": true
+    },
+    "node_modules/@types/node": {
+      "version": "18.14.6",
+      "resolved": "https://registry.npmjs.org/@types/node/-/node-18.14.6.tgz",
+      "integrity": "sha512-93+VvleD3mXwlLI/xASjw0FzKcwzl3OdTCzm1LaRfqgS21gfFtK3zDXM5Op9TeeMsJVOaJ2VRDpT9q4Y3d0AvA==",
+      "dev": true
+    },
+    "node_modules/@types/parse-json": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@types/parse-json/-/parse-json-4.0.0.tgz",
+      "integrity": "sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA==",
+      "dev": true
+    },
+    "node_modules/@types/pbkdf2": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/@types/pbkdf2/-/pbkdf2-3.1.0.tgz",
+      "integrity": "sha512-Cf63Rv7jCQ0LaL8tNXmEyqTHuIJxRdlS5vMh1mj5voN4+QFhVZnlZruezqpWYDiJ8UTzhP0VmeLXCmBk66YrMQ==",
+      "dev": true,
+      "dependencies": {
+        "@types/node": "*"
+      }
+    },
+    "node_modules/@types/prettier": {
+      "version": "2.7.2",
+      "resolved": "https://registry.npmjs.org/@types/prettier/-/prettier-2.7.2.tgz",
+      "integrity": "sha512-KufADq8uQqo1pYKVIYzfKbJfBAc0sOeXqGbFaSpv8MRmC/zXgowNZmFcbngndGk922QDmOASEXUZCaY48gs4cg==",
+      "dev": true
+    },
+    "node_modules/@types/qs": {
+      "version": "6.9.7",
+      "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.7.tgz",
+      "integrity": "sha512-FGa1F62FT09qcrueBA6qYTrJPVDzah9a+493+o2PCXsesWHIn27G98TsSMs3WPNbZIEj4+VJf6saSFpvD+3Zsw==",
+      "dev": true
+    },
+    "node_modules/@types/responselike": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/@types/responselike/-/responselike-1.0.0.tgz",
+      "integrity": "sha512-85Y2BjiufFzaMIlvJDvTTB8Fxl2xfLo4HgmHzVBz08w4wDePCTjYw66PdrolO0kzli3yam/YCgRufyo1DdQVTA==",
+      "dev": true,
+      "dependencies": {
+        "@types/node": "*"
+      }
+    },
+    "node_modules/@types/secp256k1": {
+      "version": "4.0.3",
+      "resolved": "https://registry.npmjs.org/@types/secp256k1/-/secp256k1-4.0.3.tgz",
+      "integrity": "sha512-Da66lEIFeIz9ltsdMZcpQvmrmmoqrfju8pm1BH8WbYjZSwUgCwXLb9C+9XYogwBITnbsSaMdVPb2ekf7TV+03w==",
+      "dev": true,
+      "dependencies": {
+        "@types/node": "*"
+      }
+    },
+    "node_modules/@types/semver": {
+      "version": "7.3.13",
+      "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.3.13.tgz",
+      "integrity": "sha512-21cFJr9z3g5dW8B0CVI9g2O9beqaThGQ6ZFBqHfwhzLDKUxaqTIy3vnfah/UPkfOiF2pLq+tGz+W8RyCskuslw==",
+      "dev": true
+    },
+    "node_modules/@typescript-eslint/eslint-plugin": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-5.54.1.tgz",
+      "integrity": "sha512-a2RQAkosH3d3ZIV08s3DcL/mcGc2M/UC528VkPULFxR9VnVPT8pBu0IyBAJJmVsCmhVfwQX1v6q+QGnmSe1bew==",
+      "dev": true,
+      "dependencies": {
+        "@typescript-eslint/scope-manager": "5.54.1",
+        "@typescript-eslint/type-utils": "5.54.1",
+        "@typescript-eslint/utils": "5.54.1",
+        "debug": "^4.3.4",
+        "grapheme-splitter": "^1.0.4",
+        "ignore": "^5.2.0",
+        "natural-compare-lite": "^1.4.0",
+        "regexpp": "^3.2.0",
+        "semver": "^7.3.7",
+        "tsutils": "^3.21.0"
+      },
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/typescript-eslint"
+      },
+      "peerDependencies": {
+        "@typescript-eslint/parser": "^5.0.0",
+        "eslint": "^6.0.0 || ^7.0.0 || ^8.0.0"
+      },
+      "peerDependenciesMeta": {
+        "typescript": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/@typescript-eslint/eslint-plugin/node_modules/semver": {
+      "version": "7.5.4",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz",
+      "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==",
+      "dev": true,
+      "dependencies": {
+        "lru-cache": "^6.0.0"
+      },
+      "bin": {
+        "semver": "bin/semver.js"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/@typescript-eslint/parser": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-5.54.1.tgz",
+      "integrity": "sha512-8zaIXJp/nG9Ff9vQNh7TI+C3nA6q6iIsGJ4B4L6MhZ7mHnTMR4YP5vp2xydmFXIy8rpyIVbNAG44871LMt6ujg==",
+      "dev": true,
+      "dependencies": {
+        "@typescript-eslint/scope-manager": "5.54.1",
+        "@typescript-eslint/types": "5.54.1",
+        "@typescript-eslint/typescript-estree": "5.54.1",
+        "debug": "^4.3.4"
+      },
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/typescript-eslint"
+      },
+      "peerDependencies": {
+        "eslint": "^6.0.0 || ^7.0.0 || ^8.0.0"
+      },
+      "peerDependenciesMeta": {
+        "typescript": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/@typescript-eslint/scope-manager": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-5.54.1.tgz",
+      "integrity": "sha512-zWKuGliXxvuxyM71UA/EcPxaviw39dB2504LqAmFDjmkpO8qNLHcmzlh6pbHs1h/7YQ9bnsO8CCcYCSA8sykUg==",
+      "dev": true,
+      "dependencies": {
+        "@typescript-eslint/types": "5.54.1",
+        "@typescript-eslint/visitor-keys": "5.54.1"
+      },
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/typescript-eslint"
+      }
+    },
+    "node_modules/@typescript-eslint/type-utils": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-5.54.1.tgz",
+      "integrity": "sha512-WREHsTz0GqVYLIbzIZYbmUUr95DKEKIXZNH57W3s+4bVnuF1TKe2jH8ZNH8rO1CeMY3U4j4UQeqPNkHMiGem3g==",
+      "dev": true,
+      "dependencies": {
+        "@typescript-eslint/typescript-estree": "5.54.1",
+        "@typescript-eslint/utils": "5.54.1",
+        "debug": "^4.3.4",
+        "tsutils": "^3.21.0"
+      },
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/typescript-eslint"
+      },
+      "peerDependencies": {
+        "eslint": "*"
+      },
+      "peerDependenciesMeta": {
+        "typescript": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/@typescript-eslint/types": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-5.54.1.tgz",
+      "integrity": "sha512-G9+1vVazrfAfbtmCapJX8jRo2E4MDXxgm/IMOF4oGh3kq7XuK3JRkOg6y2Qu1VsTRmWETyTkWt1wxy7X7/yLkw==",
+      "dev": true,
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/typescript-eslint"
+      }
+    },
+    "node_modules/@typescript-eslint/typescript-estree": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-5.54.1.tgz",
+      "integrity": "sha512-bjK5t+S6ffHnVwA0qRPTZrxKSaFYocwFIkZx5k7pvWfsB1I57pO/0M0Skatzzw1sCkjJ83AfGTL0oFIFiDX3bg==",
+      "dev": true,
+      "dependencies": {
+        "@typescript-eslint/types": "5.54.1",
+        "@typescript-eslint/visitor-keys": "5.54.1",
+        "debug": "^4.3.4",
+        "globby": "^11.1.0",
+        "is-glob": "^4.0.3",
+        "semver": "^7.3.7",
+        "tsutils": "^3.21.0"
+      },
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/typescript-eslint"
+      },
+      "peerDependenciesMeta": {
+        "typescript": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/@typescript-eslint/typescript-estree/node_modules/semver": {
+      "version": "7.5.4",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz",
+      "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==",
+      "dev": true,
+      "dependencies": {
+        "lru-cache": "^6.0.0"
+      },
+      "bin": {
+        "semver": "bin/semver.js"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/@typescript-eslint/utils": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-5.54.1.tgz",
+      "integrity": "sha512-IY5dyQM8XD1zfDe5X8jegX6r2EVU5o/WJnLu/znLPWCBF7KNGC+adacXnt5jEYS9JixDcoccI6CvE4RCjHMzCQ==",
+      "dev": true,
+      "dependencies": {
+        "@types/json-schema": "^7.0.9",
+        "@types/semver": "^7.3.12",
+        "@typescript-eslint/scope-manager": "5.54.1",
+        "@typescript-eslint/types": "5.54.1",
+        "@typescript-eslint/typescript-estree": "5.54.1",
+        "eslint-scope": "^5.1.1",
+        "eslint-utils": "^3.0.0",
+        "semver": "^7.3.7"
+      },
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/typescript-eslint"
+      },
+      "peerDependencies": {
+        "eslint": "^6.0.0 || ^7.0.0 || ^8.0.0"
+      }
+    },
+    "node_modules/@typescript-eslint/utils/node_modules/semver": {
+      "version": "7.5.4",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz",
+      "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==",
+      "dev": true,
+      "dependencies": {
+        "lru-cache": "^6.0.0"
+      },
+      "bin": {
+        "semver": "bin/semver.js"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/@typescript-eslint/visitor-keys": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-5.54.1.tgz",
+      "integrity": "sha512-q8iSoHTgwCfgcRJ2l2x+xCbu8nBlRAlsQ33k24Adj8eoVBE0f8dUeI+bAa8F84Mv05UGbAx57g2zrRsYIooqQg==",
+      "dev": true,
+      "dependencies": {
+        "@typescript-eslint/types": "5.54.1",
+        "eslint-visitor-keys": "^3.3.0"
+      },
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/typescript-eslint"
+      }
+    },
+    "node_modules/abbrev": {
+      "version": "1.0.9",
+      "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.0.9.tgz",
+      "integrity": "sha512-LEyx4aLEC3x6T0UguF6YILf+ntvmOaWsVfENmIW0E9H09vKlLDGelMjjSm0jkDHALj8A8quZ/HapKNigzwge+Q==",
+      "dev": true
+    },
+    "node_modules/abort-controller": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz",
+      "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==",
+      "dev": true,
+      "dependencies": {
+        "event-target-shim": "^5.0.0"
+      },
+      "engines": {
+        "node": ">=6.5"
+      }
+    },
+    "node_modules/abortcontroller-polyfill": {
+      "version": "1.7.5",
+      "resolved": "https://registry.npmjs.org/abortcontroller-polyfill/-/abortcontroller-polyfill-1.7.5.tgz",
+      "integrity": "sha512-JMJ5soJWP18htbbxJjG7bG6yuI6pRhgJ0scHHTfkUjf6wjP912xZWvM+A4sJK3gqd9E8fcPbDnOefbA9Th/FIQ==",
+      "dev": true
+    },
+    "node_modules/abstract-level": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/abstract-level/-/abstract-level-1.0.3.tgz",
+      "integrity": "sha512-t6jv+xHy+VYwc4xqZMn2Pa9DjcdzvzZmQGRjTFc8spIbRGHgBrEKbPq+rYXc7CCo0lxgYvSgKVg9qZAhpVQSjA==",
+      "dev": true,
+      "dependencies": {
+        "buffer": "^6.0.3",
+        "catering": "^2.1.0",
+        "is-buffer": "^2.0.5",
+        "level-supports": "^4.0.0",
+        "level-transcoder": "^1.0.1",
+        "module-error": "^1.0.1",
+        "queue-microtask": "^1.2.3"
+      },
+      "engines": {
+        "node": ">=12"
+      }
+    },
+    "node_modules/abstract-level/node_modules/buffer": {
+      "version": "6.0.3",
+      "resolved": "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz",
+      "integrity": "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "dependencies": {
+        "base64-js": "^1.3.1",
+        "ieee754": "^1.2.1"
+      }
+    },
+    "node_modules/accepts": {
+      "version": "1.3.8",
+      "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz",
+      "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==",
+      "dev": true,
+      "dependencies": {
+        "mime-types": "~2.1.34",
+        "negotiator": "0.6.3"
+      },
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/acorn": {
+      "version": "8.8.2",
+      "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.8.2.tgz",
+      "integrity": "sha512-xjIYgE8HBrkpd/sJqOGNspf8uHG+NOHGOw6a/Urj8taM2EXfdNAH2oFcPeIFfsv3+kz/mJrS5VuMqbNLjCa2vw==",
+      "dev": true,
+      "bin": {
+        "acorn": "bin/acorn"
+      },
+      "engines": {
+        "node": ">=0.4.0"
+      }
+    },
+    "node_modules/acorn-jsx": {
+      "version": "5.3.2",
+      "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz",
+      "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==",
+      "dev": true,
+      "peerDependencies": {
+        "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0"
+      }
+    },
+    "node_modules/acorn-walk": {
+      "version": "8.2.0",
+      "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.2.0.tgz",
+      "integrity": "sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.4.0"
+      }
+    },
+    "node_modules/address": {
+      "version": "1.2.2",
+      "resolved": "https://registry.npmjs.org/address/-/address-1.2.2.tgz",
+      "integrity": "sha512-4B/qKCfeE/ODUaAUpSwfzazo5x29WD4r3vXiWsB7I2mSDAihwEqKO+g8GELZUQSSAo5e1XTYh3ZVfLyxBc12nA==",
+      "dev": true,
+      "engines": {
+        "node": ">= 10.0.0"
+      }
+    },
+    "node_modules/adm-zip": {
+      "version": "0.4.16",
+      "resolved": "https://registry.npmjs.org/adm-zip/-/adm-zip-0.4.16.tgz",
+      "integrity": "sha512-TFi4HBKSGfIKsK5YCkKaaFG2m4PEDyViZmEwof3MTIgzimHLto6muaHVpbrljdIvIrFZzEq/p4nafOeLcYegrg==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.3.0"
+      }
+    },
+    "node_modules/aes-js": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/aes-js/-/aes-js-3.0.0.tgz",
+      "integrity": "sha512-H7wUZRn8WpTq9jocdxQ2c8x2sKo9ZVmzfRE13GiNJXfp7NcKYEdvl3vspKjXox6RIG2VtaRe4JFvxG4rqp2Zuw==",
+      "devOptional": true
+    },
+    "node_modules/agent-base": {
+      "version": "6.0.2",
+      "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz",
+      "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==",
+      "dev": true,
+      "dependencies": {
+        "debug": "4"
+      },
+      "engines": {
+        "node": ">= 6.0.0"
+      }
+    },
+    "node_modules/aggregate-error": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.1.0.tgz",
+      "integrity": "sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==",
+      "dev": true,
+      "dependencies": {
+        "clean-stack": "^2.0.0",
+        "indent-string": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/ajv": {
+      "version": "8.12.0",
+      "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.12.0.tgz",
+      "integrity": "sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==",
+      "dev": true,
+      "dependencies": {
+        "fast-deep-equal": "^3.1.1",
+        "json-schema-traverse": "^1.0.0",
+        "require-from-string": "^2.0.2",
+        "uri-js": "^4.2.2"
+      },
+      "funding": {
+        "type": "github",
+        "url": "https://github.com/sponsors/epoberezkin"
+      }
+    },
+    "node_modules/amdefine": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/amdefine/-/amdefine-1.0.1.tgz",
+      "integrity": "sha512-S2Hw0TtNkMJhIabBwIojKL9YHO5T0n5eNqWJ7Lrlel/zDbftQpxpapi8tZs3X1HWa+u+QeydGmzzNU0m09+Rcg==",
+      "dev": true,
+      "optional": true,
+      "engines": {
+        "node": ">=0.4.2"
+      }
+    },
+    "node_modules/ansi-colors": {
+      "version": "4.1.3",
+      "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.3.tgz",
+      "integrity": "sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/ansi-escapes": {
+      "version": "4.3.2",
+      "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz",
+      "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==",
+      "dev": true,
+      "dependencies": {
+        "type-fest": "^0.21.3"
+      },
+      "engines": {
+        "node": ">=8"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/ansi-regex": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
+      "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/ansi-styles": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
+      "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
+      "dev": true,
+      "dependencies": {
+        "color-convert": "^2.0.1"
+      },
+      "engines": {
+        "node": ">=8"
+      },
+      "funding": {
+        "url": "https://github.com/chalk/ansi-styles?sponsor=1"
+      }
+    },
+    "node_modules/antlr4": {
+      "version": "4.12.0",
+      "resolved": "https://registry.npmjs.org/antlr4/-/antlr4-4.12.0.tgz",
+      "integrity": "sha512-23iB5IzXJZRZeK9TigzUyrNc9pSmNqAerJRBcNq1ETrmttMWRgaYZzC561IgEO3ygKsDJTYDTozABXa4b/fTQQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=16"
+      }
+    },
+    "node_modules/antlr4ts": {
+      "version": "0.5.0-alpha.4",
+      "resolved": "https://registry.npmjs.org/antlr4ts/-/antlr4ts-0.5.0-alpha.4.tgz",
+      "integrity": "sha512-WPQDt1B74OfPv/IMS2ekXAKkTZIHl88uMetg6q3OTqgFxZ/dxDXI0EWLyZid/1Pe6hTftyg5N7gel5wNAGxXyQ==",
+      "dev": true
+    },
+    "node_modules/anymatch": {
+      "version": "3.1.3",
+      "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz",
+      "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==",
+      "dev": true,
+      "dependencies": {
+        "normalize-path": "^3.0.0",
+        "picomatch": "^2.0.4"
+      },
+      "engines": {
+        "node": ">= 8"
+      }
+    },
+    "node_modules/arg": {
+      "version": "4.1.3",
+      "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz",
+      "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==",
+      "dev": true
+    },
+    "node_modules/argparse": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz",
+      "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==",
+      "dev": true
+    },
+    "node_modules/array-back": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/array-back/-/array-back-3.1.0.tgz",
+      "integrity": "sha512-TkuxA4UCOvxuDK6NZYXCalszEzj+TLszyASooky+i742l9TqsOdYCMJJupxRic61hwquNtppB3hgcuq9SVSH1Q==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/array-flatten": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
+      "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==",
+      "dev": true
+    },
+    "node_modules/array-includes": {
+      "version": "3.1.6",
+      "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.6.tgz",
+      "integrity": "sha512-sgTbLvL6cNnw24FnbaDyjmvddQ2ML8arZsgaJhoABMoplz/4QRhtrYS+alr1BUM1Bwp6dhx8vVCBSLG+StwOFw==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "es-abstract": "^1.20.4",
+        "get-intrinsic": "^1.1.3",
+        "is-string": "^1.0.7"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/array-union": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz",
+      "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/array-uniq": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz",
+      "integrity": "sha512-MNha4BWQ6JbwhFhj03YK552f7cb3AzoE8SzeljgChvL1dl3IcvggXVz1DilzySZkCja+CXuZbdW7yATchWn8/Q==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/array.prototype.flat": {
+      "version": "1.3.1",
+      "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.3.1.tgz",
+      "integrity": "sha512-roTU0KWIOmJ4DRLmwKd19Otg0/mT3qPNt0Qb3GWW8iObuZXxrjB/pzn0R3hqpRSWg4HCwqx+0vwOnWnvlOyeIA==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "es-abstract": "^1.20.4",
+        "es-shim-unscopables": "^1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/array.prototype.flatmap": {
+      "version": "1.3.1",
+      "resolved": "https://registry.npmjs.org/array.prototype.flatmap/-/array.prototype.flatmap-1.3.1.tgz",
+      "integrity": "sha512-8UGn9O1FDVvMNB0UlLv4voxRMze7+FpHyF5mSMRjWHUMlpoDViniy05870VlxhfgTnLbpuwTzvD76MTtWxB/mQ==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "es-abstract": "^1.20.4",
+        "es-shim-unscopables": "^1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/asap": {
+      "version": "2.0.6",
+      "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz",
+      "integrity": "sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==",
+      "dev": true
+    },
+    "node_modules/asn1": {
+      "version": "0.2.6",
+      "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.6.tgz",
+      "integrity": "sha512-ix/FxPn0MDjeyJ7i/yoHGFt/EX6LyNbxSEhPPXODPL+KB0VPk86UYfL0lMdy+KCnv+fmvIzySwaK5COwqVbWTQ==",
+      "dev": true,
+      "dependencies": {
+        "safer-buffer": "~2.1.0"
+      }
+    },
+    "node_modules/assert-plus": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz",
+      "integrity": "sha512-NfJ4UzBCcQGLDlQq7nHxH+tv3kyZ0hHQqF5BO6J7tNJeP5do1llPr8dZ8zHonfhAu0PHAdMkSo+8o0wxg9lZWw==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.8"
+      }
+    },
+    "node_modules/assertion-error": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-1.1.0.tgz",
+      "integrity": "sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/ast-parents": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/ast-parents/-/ast-parents-0.0.1.tgz",
+      "integrity": "sha512-XHusKxKz3zoYk1ic8Un640joHbFMhbqneyoZfoKnEGtf2ey9Uh/IdpcQplODdO/kENaMIWsD0nJm4+wX3UNLHA==",
+      "dev": true
+    },
+    "node_modules/astral-regex": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz",
+      "integrity": "sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/async": {
+      "version": "2.6.4",
+      "resolved": "https://registry.npmjs.org/async/-/async-2.6.4.tgz",
+      "integrity": "sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==",
+      "dev": true,
+      "dependencies": {
+        "lodash": "^4.17.14"
+      }
+    },
+    "node_modules/async-eventemitter": {
+      "version": "0.2.4",
+      "resolved": "https://registry.npmjs.org/async-eventemitter/-/async-eventemitter-0.2.4.tgz",
+      "integrity": "sha512-pd20BwL7Yt1zwDFy+8MX8F1+WCT8aQeKj0kQnTrH9WaeRETlRamVhD0JtRPmrV4GfOJ2F9CvdQkZeZhnh2TuHw==",
+      "dev": true,
+      "dependencies": {
+        "async": "^2.4.0"
+      }
+    },
+    "node_modules/async-limiter": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz",
+      "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==",
+      "dev": true
+    },
+    "node_modules/asynckit": {
+      "version": "0.4.0",
+      "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
+      "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==",
+      "dev": true
+    },
+    "node_modules/available-typed-arrays": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.5.tgz",
+      "integrity": "sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/aws-sign2": {
+      "version": "0.7.0",
+      "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz",
+      "integrity": "sha512-08kcGqnYf/YmjoRhfxyu+CLxBjUtHLXLXX/vUfx9l2LYzG3c1m61nrpyFUZI6zeS+Li/wWMMidD9KgrqtGq3mA==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/aws4": {
+      "version": "1.12.0",
+      "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.12.0.tgz",
+      "integrity": "sha512-NmWvPnx0F1SfrQbYwOi7OeaNGokp9XhzNioJ/CSBs8Qa4vxug81mhJEAVZwxXuBmYB5KDRfMq/F3RR0BIU7sWg==",
+      "dev": true
+    },
+    "node_modules/axios": {
+      "version": "0.27.2",
+      "resolved": "https://registry.npmjs.org/axios/-/axios-0.27.2.tgz",
+      "integrity": "sha512-t+yRIyySRTp/wua5xEr+z1q60QmLq8ABsS5O9Me1AsE5dfKqgnCFzwiCZZ/cGNd1lq4/7akDWMxdhVlucjmnOQ==",
+      "dev": true,
+      "dependencies": {
+        "follow-redirects": "^1.14.9",
+        "form-data": "^4.0.0"
+      }
+    },
+    "node_modules/axios/node_modules/form-data": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
+      "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
+      "dev": true,
+      "dependencies": {
+        "asynckit": "^0.4.0",
+        "combined-stream": "^1.0.8",
+        "mime-types": "^2.1.12"
+      },
+      "engines": {
+        "node": ">= 6"
+      }
+    },
+    "node_modules/balanced-match": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
+      "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
+      "dev": true
+    },
+    "node_modules/base-x": {
+      "version": "3.0.9",
+      "resolved": "https://registry.npmjs.org/base-x/-/base-x-3.0.9.tgz",
+      "integrity": "sha512-H7JU6iBHTal1gp56aKoaa//YUxEaAOUiydvrV/pILqIHXTtqxSkATOnDA2u+jZ/61sD+L/412+7kzXRtWukhpQ==",
+      "dev": true,
+      "dependencies": {
+        "safe-buffer": "^5.0.1"
+      }
+    },
+    "node_modules/base64-js": {
+      "version": "1.5.1",
+      "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz",
+      "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ]
+    },
+    "node_modules/bcrypt-pbkdf": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz",
+      "integrity": "sha512-qeFIXtP4MSoi6NLqO12WfqARWWuCKi2Rn/9hJLEmtB5yTNr9DqFWkJRCf2qShWzPeAMRnOgCrq0sg/KLv5ES9w==",
+      "dev": true,
+      "dependencies": {
+        "tweetnacl": "^0.14.3"
+      }
+    },
+    "node_modules/bech32": {
+      "version": "1.1.4",
+      "resolved": "https://registry.npmjs.org/bech32/-/bech32-1.1.4.tgz",
+      "integrity": "sha512-s0IrSOzLlbvX7yp4WBfPITzpAU8sqQcpsmwXDiKwrG4r491vwCO/XpejasRNl0piBMe/DvP4Tz0mIS/X1DPJBQ==",
+      "devOptional": true
+    },
+    "node_modules/big-integer": {
+      "version": "1.6.36",
+      "resolved": "https://registry.npmjs.org/big-integer/-/big-integer-1.6.36.tgz",
+      "integrity": "sha512-t70bfa7HYEA1D9idDbmuv7YbsbVkQ+Hp+8KFSul4aE5e/i1bjCNIRYJZlA8Q8p0r9T8cF/RVvwUgRA//FydEyg==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.6"
+      }
+    },
+    "node_modules/big.js": {
+      "version": "6.2.1",
+      "resolved": "https://registry.npmjs.org/big.js/-/big.js-6.2.1.tgz",
+      "integrity": "sha512-bCtHMwL9LeDIozFn+oNhhFoq+yQ3BNdnsLSASUxLciOb1vgvpHsIO1dsENiGMgbb4SkP5TrzWzRiLddn8ahVOQ==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/bigjs"
+      }
+    },
+    "node_modules/bigint-crypto-utils": {
+      "version": "3.1.8",
+      "resolved": "https://registry.npmjs.org/bigint-crypto-utils/-/bigint-crypto-utils-3.1.8.tgz",
+      "integrity": "sha512-+VMV9Laq8pXLBKKKK49nOoq9bfR3j7NNQAtbA617a4nw9bVLo8rsqkKMBgM2AJWlNX9fEIyYaYX+d0laqYV4tw==",
+      "dev": true,
+      "dependencies": {
+        "bigint-mod-arith": "^3.1.0"
+      },
+      "engines": {
+        "node": ">=10.4.0"
+      }
+    },
+    "node_modules/bigint-mod-arith": {
+      "version": "3.1.2",
+      "resolved": "https://registry.npmjs.org/bigint-mod-arith/-/bigint-mod-arith-3.1.2.tgz",
+      "integrity": "sha512-nx8J8bBeiRR+NlsROFH9jHswW5HO8mgfOSqW0AmjicMMvaONDa8AO+5ViKDUUNytBPWiwfvZP4/Bj4Y3lUfvgQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=10.4.0"
+      }
+    },
+    "node_modules/bignumber.js": {
+      "version": "7.2.1",
+      "resolved": "https://registry.npmjs.org/bignumber.js/-/bignumber.js-7.2.1.tgz",
+      "integrity": "sha512-S4XzBk5sMB+Rcb/LNcpzXr57VRTxgAvaAEDAl1AwRx27j00hT84O6OkteE7u8UB3NuaaygCRrEpqox4uDOrbdQ==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/binary-extensions": {
+      "version": "2.2.0",
+      "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz",
+      "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/bl": {
+      "version": "4.1.0",
+      "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz",
+      "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==",
+      "dev": true,
+      "dependencies": {
+        "buffer": "^5.5.0",
+        "inherits": "^2.0.4",
+        "readable-stream": "^3.4.0"
+      }
+    },
+    "node_modules/blakejs": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/blakejs/-/blakejs-1.2.1.tgz",
+      "integrity": "sha512-QXUSXI3QVc/gJME0dBpXrag1kbzOqCjCX8/b54ntNyW6sjtoqxqRk3LTmXzaJoh71zMsDCjM+47jS7XiwN/+fQ==",
+      "dev": true
+    },
+    "node_modules/bluebird": {
+      "version": "3.7.2",
+      "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz",
+      "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==",
+      "dev": true
+    },
+    "node_modules/bn-chai": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/bn-chai/-/bn-chai-1.0.1.tgz",
+      "integrity": "sha512-7rJXt21DwYiLLpvzLaACixBBoUGkRV1iuFD3wElEhw8Ji9IiY/QsJRtvW+c7ChRgEOyLQkGaSGFUUqBKm21SNA==",
+      "dev": true,
+      "peerDependencies": {
+        "chai": ">= 2.1.2 < 5"
+      }
+    },
+    "node_modules/bn.js": {
+      "version": "5.2.1",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-5.2.1.tgz",
+      "integrity": "sha512-eXRvHzWyYPBuB4NBy0cmYQjGitUrtqwbvlzP3G6VFnNRbsZQIxQ10PbKKHt8gZ/HW/D/747aDl+QkDqg3KQLMQ==",
+      "devOptional": true
+    },
+    "node_modules/body-parser": {
+      "version": "1.20.2",
+      "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.2.tgz",
+      "integrity": "sha512-ml9pReCu3M61kGlqoTm2umSXTlRTuGTx0bfYj+uIUKKYycG5NtSbeetV3faSU6R7ajOPw0g/J1PvK4qNy7s5bA==",
+      "dev": true,
+      "dependencies": {
+        "bytes": "3.1.2",
+        "content-type": "~1.0.5",
+        "debug": "2.6.9",
+        "depd": "2.0.0",
+        "destroy": "1.2.0",
+        "http-errors": "2.0.0",
+        "iconv-lite": "0.4.24",
+        "on-finished": "2.4.1",
+        "qs": "6.11.0",
+        "raw-body": "2.5.2",
+        "type-is": "~1.6.18",
+        "unpipe": "1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.8",
+        "npm": "1.2.8000 || >= 1.4.16"
+      }
+    },
+    "node_modules/body-parser/node_modules/debug": {
+      "version": "2.6.9",
+      "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+      "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+      "dev": true,
+      "dependencies": {
+        "ms": "2.0.0"
+      }
+    },
+    "node_modules/body-parser/node_modules/ms": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+      "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
+      "dev": true
+    },
+    "node_modules/boolbase": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz",
+      "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==",
+      "dev": true
+    },
+    "node_modules/brace-expansion": {
+      "version": "1.1.11",
+      "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
+      "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
+      "dev": true,
+      "dependencies": {
+        "balanced-match": "^1.0.0",
+        "concat-map": "0.0.1"
+      }
+    },
+    "node_modules/braces": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz",
+      "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==",
+      "dev": true,
+      "dependencies": {
+        "fill-range": "^7.0.1"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/brorand": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz",
+      "integrity": "sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==",
+      "devOptional": true
+    },
+    "node_modules/browser-level": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/browser-level/-/browser-level-1.0.1.tgz",
+      "integrity": "sha512-XECYKJ+Dbzw0lbydyQuJzwNXtOpbMSq737qxJN11sIRTErOMShvDpbzTlgju7orJKvx4epULolZAuJGLzCmWRQ==",
+      "dev": true,
+      "dependencies": {
+        "abstract-level": "^1.0.2",
+        "catering": "^2.1.1",
+        "module-error": "^1.0.2",
+        "run-parallel-limit": "^1.1.0"
+      }
+    },
+    "node_modules/browser-stdout": {
+      "version": "1.3.1",
+      "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz",
+      "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==",
+      "dev": true
+    },
+    "node_modules/browserify-aes": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/browserify-aes/-/browserify-aes-1.2.0.tgz",
+      "integrity": "sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA==",
+      "dev": true,
+      "dependencies": {
+        "buffer-xor": "^1.0.3",
+        "cipher-base": "^1.0.0",
+        "create-hash": "^1.1.0",
+        "evp_bytestokey": "^1.0.3",
+        "inherits": "^2.0.1",
+        "safe-buffer": "^5.0.1"
+      }
+    },
+    "node_modules/bs58": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/bs58/-/bs58-4.0.1.tgz",
+      "integrity": "sha512-Ok3Wdf5vOIlBrgCvTq96gBkJw+JUEzdBgyaza5HLtPm7yTHkjRy8+JzNyHF7BHa0bNWOQIp3m5YF0nnFcOIKLw==",
+      "dev": true,
+      "dependencies": {
+        "base-x": "^3.0.2"
+      }
+    },
+    "node_modules/bs58check": {
+      "version": "2.1.2",
+      "resolved": "https://registry.npmjs.org/bs58check/-/bs58check-2.1.2.tgz",
+      "integrity": "sha512-0TS1jicxdU09dwJMNZtVAfzPi6Q6QeN0pM1Fkzrjn+XYHvzMKPU3pHVpva+769iNVSfIYWf7LJ6WR+BuuMf8cA==",
+      "dev": true,
+      "dependencies": {
+        "bs58": "^4.0.0",
+        "create-hash": "^1.1.0",
+        "safe-buffer": "^5.1.2"
+      }
+    },
+    "node_modules/buffer": {
+      "version": "5.7.1",
+      "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz",
+      "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "dependencies": {
+        "base64-js": "^1.3.1",
+        "ieee754": "^1.1.13"
+      }
+    },
+    "node_modules/buffer-from": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz",
+      "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==",
+      "dev": true
+    },
+    "node_modules/buffer-to-arraybuffer": {
+      "version": "0.0.5",
+      "resolved": "https://registry.npmjs.org/buffer-to-arraybuffer/-/buffer-to-arraybuffer-0.0.5.tgz",
+      "integrity": "sha512-3dthu5CYiVB1DEJp61FtApNnNndTckcqe4pFcLdvHtrpG+kcyekCJKg4MRiDcFW7A6AODnXB9U4dwQiCW5kzJQ==",
+      "dev": true
+    },
+    "node_modules/buffer-xor": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/buffer-xor/-/buffer-xor-1.0.3.tgz",
+      "integrity": "sha512-571s0T7nZWK6vB67HI5dyUF7wXiNcfaPPPTl6zYCNApANjIvYJTg7hlud/+cJpdAhS7dVzqMLmfhfHR3rAcOjQ==",
+      "dev": true
+    },
+    "node_modules/bufferutil": {
+      "version": "4.0.7",
+      "resolved": "https://registry.npmjs.org/bufferutil/-/bufferutil-4.0.7.tgz",
+      "integrity": "sha512-kukuqc39WOHtdxtw4UScxF/WVnMFVSQVKhtx3AjZJzhd0RGZZldcrfSEbVsWWe6KNH253574cq5F+wpv0G9pJw==",
+      "dev": true,
+      "hasInstallScript": true,
+      "dependencies": {
+        "node-gyp-build": "^4.3.0"
+      },
+      "engines": {
+        "node": ">=6.14.2"
+      }
+    },
+    "node_modules/bytes": {
+      "version": "3.1.2",
+      "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz",
+      "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/cacheable-lookup": {
+      "version": "6.1.0",
+      "resolved": "https://registry.npmjs.org/cacheable-lookup/-/cacheable-lookup-6.1.0.tgz",
+      "integrity": "sha512-KJ/Dmo1lDDhmW2XDPMo+9oiy/CeqosPguPCrgcVzKyZrL6pM1gU2GmPY/xo6OQPTUaA/c0kwHuywB4E6nmT9ww==",
+      "dev": true,
+      "engines": {
+        "node": ">=10.6.0"
+      }
+    },
+    "node_modules/cacheable-request": {
+      "version": "7.0.2",
+      "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-7.0.2.tgz",
+      "integrity": "sha512-pouW8/FmiPQbuGpkXQ9BAPv/Mo5xDGANgSNXzTzJ8DrKGuXOssM4wIQRjfanNRh3Yu5cfYPvcorqbhg2KIJtew==",
+      "dev": true,
+      "dependencies": {
+        "clone-response": "^1.0.2",
+        "get-stream": "^5.1.0",
+        "http-cache-semantics": "^4.0.0",
+        "keyv": "^4.0.0",
+        "lowercase-keys": "^2.0.0",
+        "normalize-url": "^6.0.1",
+        "responselike": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/cacheable-request/node_modules/get-stream": {
+      "version": "5.2.0",
+      "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz",
+      "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==",
+      "dev": true,
+      "dependencies": {
+        "pump": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=8"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/cacheable-request/node_modules/lowercase-keys": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz",
+      "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/call-bind": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz",
+      "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==",
+      "dev": true,
+      "dependencies": {
+        "function-bind": "^1.1.1",
+        "get-intrinsic": "^1.0.2"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/callsites": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz",
+      "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/camel-case": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/camel-case/-/camel-case-3.0.0.tgz",
+      "integrity": "sha512-+MbKztAYHXPr1jNTSKQF52VpcFjwY5RkR7fxksV8Doo4KAYc5Fl4UJRgthBbTmEx8C54DqahhbLJkDwjI3PI/w==",
+      "dev": true,
+      "dependencies": {
+        "no-case": "^2.2.0",
+        "upper-case": "^1.1.1"
+      }
+    },
+    "node_modules/camelcase": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-3.0.0.tgz",
+      "integrity": "sha512-4nhGqUkc4BqbBBB4Q6zLuD7lzzrHYrjKGeYaEji/3tFR5VdJu9v+LilhGIVe8wxEJPPOeWo7eg8dwY13TZ1BNg==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/caseless": {
+      "version": "0.12.0",
+      "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz",
+      "integrity": "sha512-4tYFyifaFfGacoiObjJegolkwSU4xQNGbVgUiNYVUxbQ2x2lUsFvY4hVgVzGiIe6WLOPqycWXA40l+PWsxthUw==",
+      "dev": true
+    },
+    "node_modules/catering": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/catering/-/catering-2.1.1.tgz",
+      "integrity": "sha512-K7Qy8O9p76sL3/3m7/zLKbRkyOlSZAgzEaLhyj2mXS8PsCud2Eo4hAb8aLtZqHh0QGqLcb9dlJSu6lHRVENm1w==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/cbor": {
+      "version": "8.1.0",
+      "resolved": "https://registry.npmjs.org/cbor/-/cbor-8.1.0.tgz",
+      "integrity": "sha512-DwGjNW9omn6EwP70aXsn7FQJx5kO12tX0bZkaTjzdVFM6/7nhA4t0EENocKGx6D2Bch9PE2KzCUf5SceBdeijg==",
+      "dev": true,
+      "dependencies": {
+        "nofilter": "^3.1.0"
+      },
+      "engines": {
+        "node": ">=12.19"
+      }
+    },
+    "node_modules/chai": {
+      "version": "4.3.7",
+      "resolved": "https://registry.npmjs.org/chai/-/chai-4.3.7.tgz",
+      "integrity": "sha512-HLnAzZ2iupm25PlN0xFreAlBA5zaBSv3og0DdeGA4Ar6h6rJ3A0rolRUKJhSF2V10GZKDgWF/VmAEsNWjCRB+A==",
+      "dev": true,
+      "dependencies": {
+        "assertion-error": "^1.1.0",
+        "check-error": "^1.0.2",
+        "deep-eql": "^4.1.2",
+        "get-func-name": "^2.0.0",
+        "loupe": "^2.3.1",
+        "pathval": "^1.1.1",
+        "type-detect": "^4.0.5"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/chai-as-promised": {
+      "version": "7.1.1",
+      "resolved": "https://registry.npmjs.org/chai-as-promised/-/chai-as-promised-7.1.1.tgz",
+      "integrity": "sha512-azL6xMoi+uxu6z4rhWQ1jbdUhOMhis2PvscD/xjLqNMkv3BPPp2JyyuTHOrf9BOosGpNQ11v6BKv/g57RXbiaA==",
+      "dev": true,
+      "dependencies": {
+        "check-error": "^1.0.2"
+      },
+      "peerDependencies": {
+        "chai": ">= 2.1.2 < 5"
+      }
+    },
+    "node_modules/chai-bignumber": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/chai-bignumber/-/chai-bignumber-3.1.0.tgz",
+      "integrity": "sha512-omxEc80jAU+pZwRmoWr3aEzeLad4JW3iBhLRQlgISvghBdIxrMT7mVAGsDz4WSyCkKowENshH2j9OABAhld7QQ==",
+      "dev": true
+    },
+    "node_modules/chai-string": {
+      "version": "1.5.0",
+      "resolved": "https://registry.npmjs.org/chai-string/-/chai-string-1.5.0.tgz",
+      "integrity": "sha512-sydDC3S3pNAQMYwJrs6dQX0oBQ6KfIPuOZ78n7rocW0eJJlsHPh2t3kwW7xfwYA/1Bf6/arGtSUo16rxR2JFlw==",
+      "dev": true,
+      "peerDependencies": {
+        "chai": "^4.1.2"
+      }
+    },
+    "node_modules/chalk": {
+      "version": "4.1.2",
+      "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz",
+      "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==",
+      "dev": true,
+      "dependencies": {
+        "ansi-styles": "^4.1.0",
+        "supports-color": "^7.1.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/chalk/chalk?sponsor=1"
+      }
+    },
+    "node_modules/change-case": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/change-case/-/change-case-3.0.2.tgz",
+      "integrity": "sha512-Mww+SLF6MZ0U6kdg11algyKd5BARbyM4TbFBepwowYSR5ClfQGCGtxNXgykpN0uF/bstWeaGDT4JWaDh8zWAHA==",
+      "dev": true,
+      "dependencies": {
+        "camel-case": "^3.0.0",
+        "constant-case": "^2.0.0",
+        "dot-case": "^2.1.0",
+        "header-case": "^1.0.0",
+        "is-lower-case": "^1.1.0",
+        "is-upper-case": "^1.1.0",
+        "lower-case": "^1.1.1",
+        "lower-case-first": "^1.0.0",
+        "no-case": "^2.3.2",
+        "param-case": "^2.1.0",
+        "pascal-case": "^2.0.0",
+        "path-case": "^2.1.0",
+        "sentence-case": "^2.1.0",
+        "snake-case": "^2.1.0",
+        "swap-case": "^1.1.0",
+        "title-case": "^2.1.0",
+        "upper-case": "^1.1.1",
+        "upper-case-first": "^1.1.0"
+      }
+    },
+    "node_modules/charenc": {
+      "version": "0.0.2",
+      "resolved": "https://registry.npmjs.org/charenc/-/charenc-0.0.2.tgz",
+      "integrity": "sha512-yrLQ/yVUFXkzg7EDQsPieE/53+0RlaWTs+wBrvW36cyilJ2SaDWfl4Yj7MtLTXleV9uEKefbAGUPv2/iWSooRA==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/check-error": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/check-error/-/check-error-1.0.2.tgz",
+      "integrity": "sha512-BrgHpW9NURQgzoNyjfq0Wu6VFO6D7IZEmJNdtgNqpzGG8RuNFHt2jQxWlAs4HMe119chBnv+34syEZtc6IhLtA==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/cheerio": {
+      "version": "1.0.0-rc.12",
+      "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.0.0-rc.12.tgz",
+      "integrity": "sha512-VqR8m68vM46BNnuZ5NtnGBKIE/DfN0cRIzg9n40EIq9NOv90ayxLBXA8fXC5gquFRGJSTRqBq25Jt2ECLR431Q==",
+      "dev": true,
+      "dependencies": {
+        "cheerio-select": "^2.1.0",
+        "dom-serializer": "^2.0.0",
+        "domhandler": "^5.0.3",
+        "domutils": "^3.0.1",
+        "htmlparser2": "^8.0.1",
+        "parse5": "^7.0.0",
+        "parse5-htmlparser2-tree-adapter": "^7.0.0"
+      },
+      "engines": {
+        "node": ">= 6"
+      },
+      "funding": {
+        "url": "https://github.com/cheeriojs/cheerio?sponsor=1"
+      }
+    },
+    "node_modules/cheerio-select": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/cheerio-select/-/cheerio-select-2.1.0.tgz",
+      "integrity": "sha512-9v9kG0LvzrlcungtnJtpGNxY+fzECQKhK4EGJX2vByejiMX84MFNQw4UxPJl3bFbTMw+Dfs37XaIkCwTZfLh4g==",
+      "dev": true,
+      "dependencies": {
+        "boolbase": "^1.0.0",
+        "css-select": "^5.1.0",
+        "css-what": "^6.1.0",
+        "domelementtype": "^2.3.0",
+        "domhandler": "^5.0.3",
+        "domutils": "^3.0.1"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/fb55"
+      }
+    },
+    "node_modules/child_process": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/child_process/-/child_process-1.0.2.tgz",
+      "integrity": "sha512-Wmza/JzL0SiWz7kl6MhIKT5ceIlnFPJX+lwUGj7Clhy5MMldsSoJR0+uvRzOS5Kv45Mq7t1PoE8TsOA9bzvb6g==",
+      "dev": true
+    },
+    "node_modules/chokidar": {
+      "version": "3.5.3",
+      "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz",
+      "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://paulmillr.com/funding/"
+        }
+      ],
+      "dependencies": {
+        "anymatch": "~3.1.2",
+        "braces": "~3.0.2",
+        "glob-parent": "~5.1.2",
+        "is-binary-path": "~2.1.0",
+        "is-glob": "~4.0.1",
+        "normalize-path": "~3.0.0",
+        "readdirp": "~3.6.0"
+      },
+      "engines": {
+        "node": ">= 8.10.0"
+      },
+      "optionalDependencies": {
+        "fsevents": "~2.3.2"
+      }
+    },
+    "node_modules/chownr": {
+      "version": "1.1.4",
+      "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz",
+      "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==",
+      "dev": true
+    },
+    "node_modules/ci-info": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz",
+      "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==",
+      "dev": true
+    },
+    "node_modules/cids": {
+      "version": "0.7.5",
+      "resolved": "https://registry.npmjs.org/cids/-/cids-0.7.5.tgz",
+      "integrity": "sha512-zT7mPeghoWAu+ppn8+BS1tQ5qGmbMfB4AregnQjA/qHY3GC1m1ptI9GkWNlgeu38r7CuRdXB47uY2XgAYt6QVA==",
+      "deprecated": "This module has been superseded by the multiformats module",
+      "dev": true,
+      "dependencies": {
+        "buffer": "^5.5.0",
+        "class-is": "^1.1.0",
+        "multibase": "~0.6.0",
+        "multicodec": "^1.0.0",
+        "multihashes": "~0.4.15"
+      },
+      "engines": {
+        "node": ">=4.0.0",
+        "npm": ">=3.0.0"
+      }
+    },
+    "node_modules/cids/node_modules/multicodec": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/multicodec/-/multicodec-1.0.4.tgz",
+      "integrity": "sha512-NDd7FeS3QamVtbgfvu5h7fd1IlbaC4EQ0/pgU4zqE2vdHCmBGsUa0TiM8/TdSeG6BMPC92OOCf8F1ocE/Wkrrg==",
+      "deprecated": "This module has been superseded by the multiformats module",
+      "dev": true,
+      "dependencies": {
+        "buffer": "^5.6.0",
+        "varint": "^5.0.0"
+      }
+    },
+    "node_modules/cipher-base": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/cipher-base/-/cipher-base-1.0.4.tgz",
+      "integrity": "sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q==",
+      "dev": true,
+      "dependencies": {
+        "inherits": "^2.0.1",
+        "safe-buffer": "^5.0.1"
+      }
+    },
+    "node_modules/class-is": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/class-is/-/class-is-1.1.0.tgz",
+      "integrity": "sha512-rhjH9AG1fvabIDoGRVH587413LPjTZgmDF9fOFCbFJQV4yuocX1mHxxvXI4g3cGwbVY9wAYIoKlg1N79frJKQw==",
+      "dev": true
+    },
+    "node_modules/classic-level": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/classic-level/-/classic-level-1.2.0.tgz",
+      "integrity": "sha512-qw5B31ANxSluWz9xBzklRWTUAJ1SXIdaVKTVS7HcTGKOAmExx65Wo5BUICW+YGORe2FOUaDghoI9ZDxj82QcFg==",
+      "dev": true,
+      "hasInstallScript": true,
+      "dependencies": {
+        "abstract-level": "^1.0.2",
+        "catering": "^2.1.0",
+        "module-error": "^1.0.1",
+        "napi-macros": "~2.0.0",
+        "node-gyp-build": "^4.3.0"
+      },
+      "engines": {
+        "node": ">=12"
+      }
+    },
+    "node_modules/clean-stack": {
+      "version": "2.2.0",
+      "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz",
+      "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/cli-cursor": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz",
+      "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==",
+      "dev": true,
+      "dependencies": {
+        "restore-cursor": "^3.1.0"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/cli-spinners": {
+      "version": "2.7.0",
+      "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.7.0.tgz",
+      "integrity": "sha512-qu3pN8Y3qHNgE2AFweciB1IfMnmZ/fsNTEE+NOFjmGB2F/7rLhnhzppvpCnN4FovtP26k8lHyy9ptEbNwWFLzw==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/cli-table3": {
+      "version": "0.6.3",
+      "resolved": "https://registry.npmjs.org/cli-table3/-/cli-table3-0.6.3.tgz",
+      "integrity": "sha512-w5Jac5SykAeZJKntOxJCrm63Eg5/4dhMWIcuTbo9rpE+brgaSZo0RuNJZeOyMgsUdhDeojvgyQLmjI+K50ZGyg==",
+      "dev": true,
+      "dependencies": {
+        "string-width": "^4.2.0"
+      },
+      "engines": {
+        "node": "10.* || >= 12.*"
+      },
+      "optionalDependencies": {
+        "@colors/colors": "1.5.0"
+      }
+    },
+    "node_modules/cliui": {
+      "version": "3.2.0",
+      "resolved": "https://registry.npmjs.org/cliui/-/cliui-3.2.0.tgz",
+      "integrity": "sha512-0yayqDxWQbqk3ojkYqUKqaAQ6AfNKeKWRNA8kR0WXzAsdHpP4BIaOmMAG87JGuO6qcobyW4GjxHd9PmhEd+T9w==",
+      "dev": true,
+      "dependencies": {
+        "string-width": "^1.0.1",
+        "strip-ansi": "^3.0.1",
+        "wrap-ansi": "^2.0.0"
+      }
+    },
+    "node_modules/cliui/node_modules/ansi-regex": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
+      "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/cliui/node_modules/is-fullwidth-code-point": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz",
+      "integrity": "sha512-1pqUqRjkhPJ9miNq9SwMfdvi6lBJcd6eFxvfaivQhaH3SgisfiuudvFntdKOmxuee/77l+FPjKrQjWvmPjWrRw==",
+      "dev": true,
+      "dependencies": {
+        "number-is-nan": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/cliui/node_modules/string-width": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz",
+      "integrity": "sha512-0XsVpQLnVCXHJfyEs8tC0zpTVIr5PKKsQtkT29IwupnPTjtPmQ3xT/4yCREF9hYkV/3M3kzcUTSAZT6a6h81tw==",
+      "dev": true,
+      "dependencies": {
+        "code-point-at": "^1.0.0",
+        "is-fullwidth-code-point": "^1.0.0",
+        "strip-ansi": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/cliui/node_modules/strip-ansi": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
+      "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==",
+      "dev": true,
+      "dependencies": {
+        "ansi-regex": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/clone": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz",
+      "integrity": "sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.8"
+      }
+    },
+    "node_modules/clone-response": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.3.tgz",
+      "integrity": "sha512-ROoL94jJH2dUVML2Y/5PEDNaSHgeOdSDicUyS7izcF63G6sTc/FTjLub4b8Il9S8S0beOfYt0TaA5qvFK+w0wA==",
+      "dev": true,
+      "dependencies": {
+        "mimic-response": "^1.0.0"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/code-point-at": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz",
+      "integrity": "sha512-RpAVKQA5T63xEj6/giIbUEtZwJ4UFIc3ZtvEkiaUERylqe8xb5IvqcgOurZLahv93CLKfxcw5YI+DZcUBRyLXA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/color-convert": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
+      "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
+      "dev": true,
+      "dependencies": {
+        "color-name": "~1.1.4"
+      },
+      "engines": {
+        "node": ">=7.0.0"
+      }
+    },
+    "node_modules/color-name": {
+      "version": "1.1.4",
+      "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
+      "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
+      "dev": true
+    },
+    "node_modules/colors": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/colors/-/colors-1.4.0.tgz",
+      "integrity": "sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.1.90"
+      }
+    },
+    "node_modules/combined-stream": {
+      "version": "1.0.8",
+      "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
+      "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
+      "dev": true,
+      "dependencies": {
+        "delayed-stream": "~1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/command-exists": {
+      "version": "1.2.9",
+      "resolved": "https://registry.npmjs.org/command-exists/-/command-exists-1.2.9.tgz",
+      "integrity": "sha512-LTQ/SGc+s0Xc0Fu5WaKnR0YiygZkm9eKFvyS+fRsU7/ZWFF8ykFM6Pc9aCVf1+xasOOZpO3BAVgVrKvsqKHV7w==",
+      "dev": true
+    },
+    "node_modules/command-line-args": {
+      "version": "5.2.1",
+      "resolved": "https://registry.npmjs.org/command-line-args/-/command-line-args-5.2.1.tgz",
+      "integrity": "sha512-H4UfQhZyakIjC74I9d34fGYDwk3XpSr17QhEd0Q3I9Xq1CETHo4Hcuo87WyWHpAF1aSLjLRf5lD9ZGX2qStUvg==",
+      "dev": true,
+      "dependencies": {
+        "array-back": "^3.1.0",
+        "find-replace": "^3.0.0",
+        "lodash.camelcase": "^4.3.0",
+        "typical": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=4.0.0"
+      }
+    },
+    "node_modules/command-line-usage": {
+      "version": "6.1.3",
+      "resolved": "https://registry.npmjs.org/command-line-usage/-/command-line-usage-6.1.3.tgz",
+      "integrity": "sha512-sH5ZSPr+7UStsloltmDh7Ce5fb8XPlHyoPzTpyyMuYCtervL65+ubVZ6Q61cFtFl62UyJlc8/JwERRbAFPUqgw==",
+      "dev": true,
+      "dependencies": {
+        "array-back": "^4.0.2",
+        "chalk": "^2.4.2",
+        "table-layout": "^1.0.2",
+        "typical": "^5.2.0"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/command-line-usage/node_modules/ansi-styles": {
+      "version": "3.2.1",
+      "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+      "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+      "dev": true,
+      "dependencies": {
+        "color-convert": "^1.9.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/command-line-usage/node_modules/array-back": {
+      "version": "4.0.2",
+      "resolved": "https://registry.npmjs.org/array-back/-/array-back-4.0.2.tgz",
+      "integrity": "sha512-NbdMezxqf94cnNfWLL7V/im0Ub+Anbb0IoZhvzie8+4HJ4nMQuzHuy49FkGYCJK2yAloZ3meiB6AVMClbrI1vg==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/command-line-usage/node_modules/chalk": {
+      "version": "2.4.2",
+      "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+      "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+      "dev": true,
+      "dependencies": {
+        "ansi-styles": "^3.2.1",
+        "escape-string-regexp": "^1.0.5",
+        "supports-color": "^5.3.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/command-line-usage/node_modules/color-convert": {
+      "version": "1.9.3",
+      "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+      "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+      "dev": true,
+      "dependencies": {
+        "color-name": "1.1.3"
+      }
+    },
+    "node_modules/command-line-usage/node_modules/color-name": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+      "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+      "dev": true
+    },
+    "node_modules/command-line-usage/node_modules/has-flag": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+      "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/command-line-usage/node_modules/supports-color": {
+      "version": "5.5.0",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+      "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+      "dev": true,
+      "dependencies": {
+        "has-flag": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/command-line-usage/node_modules/typical": {
+      "version": "5.2.0",
+      "resolved": "https://registry.npmjs.org/typical/-/typical-5.2.0.tgz",
+      "integrity": "sha512-dvdQgNDNJo+8B2uBQoqdb11eUCE1JQXhvjC/CZtgvZseVd5TYMXnq0+vuUemXbd/Se29cTaUuPX3YIc2xgbvIg==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/commander": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/commander/-/commander-3.0.2.tgz",
+      "integrity": "sha512-Gar0ASD4BDyKC4hl4DwHqDrmvjoxWKZigVnAbn5H1owvm4CxCPdb0HQDehwNYMJpla5+M2tPmPARzhtYuwpHow==",
+      "dev": true
+    },
+    "node_modules/compare-versions": {
+      "version": "3.6.0",
+      "resolved": "https://registry.npmjs.org/compare-versions/-/compare-versions-3.6.0.tgz",
+      "integrity": "sha512-W6Af2Iw1z4CB7q4uU4hv646dW9GQuBM+YpC0UvUCWSD8w90SJjp+ujJuXaEMtAXBtSqGfMPuFOVn4/+FlaqfBA==",
+      "dev": true
+    },
+    "node_modules/concat-map": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
+      "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==",
+      "dev": true
+    },
+    "node_modules/concat-stream": {
+      "version": "1.6.2",
+      "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz",
+      "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==",
+      "dev": true,
+      "engines": [
+        "node >= 0.8"
+      ],
+      "dependencies": {
+        "buffer-from": "^1.0.0",
+        "inherits": "^2.0.3",
+        "readable-stream": "^2.2.2",
+        "typedarray": "^0.0.6"
+      }
+    },
+    "node_modules/concat-stream/node_modules/readable-stream": {
+      "version": "2.3.8",
+      "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz",
+      "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==",
+      "dev": true,
+      "dependencies": {
+        "core-util-is": "~1.0.0",
+        "inherits": "~2.0.3",
+        "isarray": "~1.0.0",
+        "process-nextick-args": "~2.0.0",
+        "safe-buffer": "~5.1.1",
+        "string_decoder": "~1.1.1",
+        "util-deprecate": "~1.0.1"
+      }
+    },
+    "node_modules/concat-stream/node_modules/safe-buffer": {
+      "version": "5.1.2",
+      "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
+      "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==",
+      "dev": true
+    },
+    "node_modules/concat-stream/node_modules/string_decoder": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz",
+      "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==",
+      "dev": true,
+      "dependencies": {
+        "safe-buffer": "~5.1.0"
+      }
+    },
+    "node_modules/constant-case": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/constant-case/-/constant-case-2.0.0.tgz",
+      "integrity": "sha512-eS0N9WwmjTqrOmR3o83F5vW8Z+9R1HnVz3xmzT2PMFug9ly+Au/fxRWlEBSb6LcZwspSsEn9Xs1uw9YgzAg1EQ==",
+      "dev": true,
+      "dependencies": {
+        "snake-case": "^2.1.0",
+        "upper-case": "^1.1.1"
+      }
+    },
+    "node_modules/content-disposition": {
+      "version": "0.5.4",
+      "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz",
+      "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==",
+      "dev": true,
+      "dependencies": {
+        "safe-buffer": "5.2.1"
+      },
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/content-hash": {
+      "version": "2.5.2",
+      "resolved": "https://registry.npmjs.org/content-hash/-/content-hash-2.5.2.tgz",
+      "integrity": "sha512-FvIQKy0S1JaWV10sMsA7TRx8bpU+pqPkhbsfvOJAdjRXvYxEckAwQWGwtRjiaJfh+E0DvcWUGqcdjwMGFjsSdw==",
+      "dev": true,
+      "dependencies": {
+        "cids": "^0.7.1",
+        "multicodec": "^0.5.5",
+        "multihashes": "^0.4.15"
+      }
+    },
+    "node_modules/content-type": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz",
+      "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/cookie": {
+      "version": "0.5.0",
+      "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.5.0.tgz",
+      "integrity": "sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/cookie-signature": {
+      "version": "1.0.6",
+      "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
+      "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==",
+      "dev": true
+    },
+    "node_modules/core-util-is": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
+      "integrity": "sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==",
+      "dev": true
+    },
+    "node_modules/cors": {
+      "version": "2.8.5",
+      "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz",
+      "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==",
+      "dev": true,
+      "dependencies": {
+        "object-assign": "^4",
+        "vary": "^1"
+      },
+      "engines": {
+        "node": ">= 0.10"
+      }
+    },
+    "node_modules/cosmiconfig": {
+      "version": "7.1.0",
+      "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-7.1.0.tgz",
+      "integrity": "sha512-AdmX6xUzdNASswsFtmwSt7Vj8po9IuqXm0UXz7QKPuEUmPB4XyjGfaAr2PSuELMwkRMVH1EpIkX5bTZGRB3eCA==",
+      "dev": true,
+      "dependencies": {
+        "@types/parse-json": "^4.0.0",
+        "import-fresh": "^3.2.1",
+        "parse-json": "^5.0.0",
+        "path-type": "^4.0.0",
+        "yaml": "^1.10.0"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/cosmiconfig/node_modules/parse-json": {
+      "version": "5.2.0",
+      "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz",
+      "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==",
+      "dev": true,
+      "dependencies": {
+        "@babel/code-frame": "^7.0.0",
+        "error-ex": "^1.3.1",
+        "json-parse-even-better-errors": "^2.3.0",
+        "lines-and-columns": "^1.1.6"
+      },
+      "engines": {
+        "node": ">=8"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/cosmiconfig/node_modules/path-type": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz",
+      "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/crc-32": {
+      "version": "1.2.2",
+      "resolved": "https://registry.npmjs.org/crc-32/-/crc-32-1.2.2.tgz",
+      "integrity": "sha512-ROmzCKrTnOwybPcJApAA6WBWij23HVfGVNKqqrZpuyZOHqK2CwHSvpGuyt/UNNvaIjEd8X5IFGp4Mh+Ie1IHJQ==",
+      "dev": true,
+      "bin": {
+        "crc32": "bin/crc32.njs"
+      },
+      "engines": {
+        "node": ">=0.8"
+      }
+    },
+    "node_modules/create-hash": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz",
+      "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==",
+      "dev": true,
+      "dependencies": {
+        "cipher-base": "^1.0.1",
+        "inherits": "^2.0.1",
+        "md5.js": "^1.3.4",
+        "ripemd160": "^2.0.1",
+        "sha.js": "^2.4.0"
+      }
+    },
+    "node_modules/create-hmac": {
+      "version": "1.1.7",
+      "resolved": "https://registry.npmjs.org/create-hmac/-/create-hmac-1.1.7.tgz",
+      "integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==",
+      "dev": true,
+      "dependencies": {
+        "cipher-base": "^1.0.3",
+        "create-hash": "^1.1.0",
+        "inherits": "^2.0.1",
+        "ripemd160": "^2.0.0",
+        "safe-buffer": "^5.0.1",
+        "sha.js": "^2.4.8"
+      }
+    },
+    "node_modules/create-require": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz",
+      "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==",
+      "dev": true
+    },
+    "node_modules/cross-fetch": {
+      "version": "3.1.5",
+      "resolved": "https://registry.npmjs.org/cross-fetch/-/cross-fetch-3.1.5.tgz",
+      "integrity": "sha512-lvb1SBsI0Z7GDwmuid+mU3kWVBwTVUbe7S0H52yaaAdQOXq2YktTCZdlAcNKFzE6QtRz0snpw9bNiPeOIkkQvw==",
+      "dev": true,
+      "dependencies": {
+        "node-fetch": "2.6.7"
+      }
+    },
+    "node_modules/cross-spawn": {
+      "version": "7.0.3",
+      "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz",
+      "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==",
+      "dev": true,
+      "dependencies": {
+        "path-key": "^3.1.0",
+        "shebang-command": "^2.0.0",
+        "which": "^2.0.1"
+      },
+      "engines": {
+        "node": ">= 8"
+      }
+    },
+    "node_modules/crypt": {
+      "version": "0.0.2",
+      "resolved": "https://registry.npmjs.org/crypt/-/crypt-0.0.2.tgz",
+      "integrity": "sha512-mCxBlsHFYh9C+HVpiEacem8FEBnMXgU9gy4zmNC+SXAZNB/1idgp/aulFJ4FgCi7GPEVbfyng092GqL2k2rmow==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/crypto-addr-codec": {
+      "version": "0.1.7",
+      "resolved": "https://registry.npmjs.org/crypto-addr-codec/-/crypto-addr-codec-0.1.7.tgz",
+      "integrity": "sha512-X4hzfBzNhy4mAc3UpiXEC/L0jo5E8wAa9unsnA8nNXYzXjCcGk83hfC5avJWCSGT8V91xMnAS9AKMHmjw5+XCg==",
+      "dev": true,
+      "dependencies": {
+        "base-x": "^3.0.8",
+        "big-integer": "1.6.36",
+        "blakejs": "^1.1.0",
+        "bs58": "^4.0.1",
+        "ripemd160-min": "0.0.6",
+        "safe-buffer": "^5.2.0",
+        "sha3": "^2.1.1"
+      }
+    },
+    "node_modules/css-select": {
+      "version": "5.1.0",
+      "resolved": "https://registry.npmjs.org/css-select/-/css-select-5.1.0.tgz",
+      "integrity": "sha512-nwoRF1rvRRnnCqqY7updORDsuqKzqYJ28+oSMaJMMgOauh3fvwHqMS7EZpIPqK8GL+g9mKxF1vP/ZjSeNjEVHg==",
+      "dev": true,
+      "dependencies": {
+        "boolbase": "^1.0.0",
+        "css-what": "^6.1.0",
+        "domhandler": "^5.0.2",
+        "domutils": "^3.0.1",
+        "nth-check": "^2.0.1"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/fb55"
+      }
+    },
+    "node_modules/css-what": {
+      "version": "6.1.0",
+      "resolved": "https://registry.npmjs.org/css-what/-/css-what-6.1.0.tgz",
+      "integrity": "sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==",
+      "dev": true,
+      "engines": {
+        "node": ">= 6"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/fb55"
+      }
+    },
+    "node_modules/d": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/d/-/d-1.0.1.tgz",
+      "integrity": "sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==",
+      "dev": true,
+      "dependencies": {
+        "es5-ext": "^0.10.50",
+        "type": "^1.0.1"
+      }
+    },
+    "node_modules/dashdash": {
+      "version": "1.14.1",
+      "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz",
+      "integrity": "sha512-jRFi8UDGo6j+odZiEpjazZaWqEal3w/basFjQHQEwVtZJGDpxbH1MeYluwCS8Xq5wmLJooDlMgvVarmWfGM44g==",
+      "dev": true,
+      "dependencies": {
+        "assert-plus": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=0.10"
+      }
+    },
+    "node_modules/death": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/death/-/death-1.1.0.tgz",
+      "integrity": "sha512-vsV6S4KVHvTGxbEcij7hkWRv0It+sGGWVOM67dQde/o5Xjnr+KmLjxWJii2uEObIrt1CcM9w0Yaovx+iOlIL+w==",
+      "dev": true
+    },
+    "node_modules/debug": {
+      "version": "4.3.4",
+      "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz",
+      "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==",
+      "dev": true,
+      "dependencies": {
+        "ms": "2.1.2"
+      },
+      "engines": {
+        "node": ">=6.0"
+      },
+      "peerDependenciesMeta": {
+        "supports-color": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/decamelize": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz",
+      "integrity": "sha512-z2S+W9X73hAUUki+N+9Za2lBlun89zigOyGrsax+KUQ6wKW4ZoWpEYBkGhQjwAjjDCkWxhY0VKEhk8wzY7F5cA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/decode-uri-component": {
+      "version": "0.2.2",
+      "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.2.tgz",
+      "integrity": "sha512-FqUYQ+8o158GyGTrMFJms9qh3CqTKvAqgqsTnkLI8sKu0028orqBhxNMFkFen0zGyg6epACD32pjVk58ngIErQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10"
+      }
+    },
+    "node_modules/decompress-response": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-6.0.0.tgz",
+      "integrity": "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==",
+      "dev": true,
+      "dependencies": {
+        "mimic-response": "^3.1.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/decompress-response/node_modules/mimic-response": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-3.1.0.tgz",
+      "integrity": "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/deep-eql": {
+      "version": "4.1.3",
+      "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-4.1.3.tgz",
+      "integrity": "sha512-WaEtAOpRA1MQ0eohqZjpGD8zdI0Ovsm8mmFhaDN8dvDZzyoUMcYDnf5Y6iu7HTXxf8JDS23qWa4a+hKCDyOPzw==",
+      "dev": true,
+      "dependencies": {
+        "type-detect": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/deep-extend": {
+      "version": "0.6.0",
+      "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
+      "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
+      "dev": true,
+      "engines": {
+        "node": ">=4.0.0"
+      }
+    },
+    "node_modules/deep-is": {
+      "version": "0.1.4",
+      "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz",
+      "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==",
+      "dev": true
+    },
+    "node_modules/defaults": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.4.tgz",
+      "integrity": "sha512-eFuaLoy/Rxalv2kr+lqMlUnrDWV+3j4pljOIJgLIhI058IQfWJ7vXhyEIHu+HtC738klGALYxOKDO0bQP3tg8A==",
+      "dev": true,
+      "dependencies": {
+        "clone": "^1.0.2"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/defer-to-connect": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-2.0.1.tgz",
+      "integrity": "sha512-4tvttepXG1VaYGrRibk5EwJd1t4udunSOVMdLSAL6mId1ix438oPwPZMALY41FCijukO1L0twNcGsdzS7dHgDg==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/define-lazy-prop": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/define-lazy-prop/-/define-lazy-prop-2.0.0.tgz",
+      "integrity": "sha512-Ds09qNh8yw3khSjiJjiUInaGX9xlqZDY7JVryGxdxV7NPeuqQfplOpQ66yJFZut3jLa5zOwkXw1g9EI2uKh4Og==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/define-properties": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.0.tgz",
+      "integrity": "sha512-xvqAVKGfT1+UAvPwKTVw/njhdQ8ZhXK4lI0bCIuCMrp2up9nPnaDftrLtmpTazqd1o+UY4zgzU+avtMbDP+ldA==",
+      "dev": true,
+      "dependencies": {
+        "has-property-descriptors": "^1.0.0",
+        "object-keys": "^1.1.1"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/delayed-stream": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
+      "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.4.0"
+      }
+    },
+    "node_modules/depd": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
+      "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/destroy": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz",
+      "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.8",
+        "npm": "1.2.8000 || >= 1.4.16"
+      }
+    },
+    "node_modules/detect-indent": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/detect-indent/-/detect-indent-5.0.0.tgz",
+      "integrity": "sha512-rlpvsxUtM0PQvy9iZe640/IWwWYyBsTApREbA1pHOpmOUIl9MkP/U4z7vTtg4Oaojvqhxt7sdufnT0EzGaR31g==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/detect-port": {
+      "version": "1.5.1",
+      "resolved": "https://registry.npmjs.org/detect-port/-/detect-port-1.5.1.tgz",
+      "integrity": "sha512-aBzdj76lueB6uUst5iAs7+0H/oOjqI5D16XUWxlWMIMROhcM0rfsNVk93zTngq1dDNpoXRr++Sus7ETAExppAQ==",
+      "dev": true,
+      "dependencies": {
+        "address": "^1.0.1",
+        "debug": "4"
+      },
+      "bin": {
+        "detect": "bin/detect-port.js",
+        "detect-port": "bin/detect-port.js"
+      }
+    },
+    "node_modules/diff": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/diff/-/diff-5.0.0.tgz",
+      "integrity": "sha512-/VTCrvm5Z0JGty/BWHljh+BAiw3IK+2j87NGMu8Nwc/f48WoDAC395uomO9ZD117ZOBaHmkX1oyLvkVM/aIT3w==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.3.1"
+      }
+    },
+    "node_modules/difflib": {
+      "version": "0.2.4",
+      "resolved": "https://registry.npmjs.org/difflib/-/difflib-0.2.4.tgz",
+      "integrity": "sha512-9YVwmMb0wQHQNr5J9m6BSj6fk4pfGITGQOOs+D9Fl+INODWFOfvhIU1hNv6GgR1RBoC/9NJcwu77zShxV0kT7w==",
+      "dev": true,
+      "dependencies": {
+        "heap": ">= 0.2.0"
+      },
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/dir-glob": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz",
+      "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==",
+      "dev": true,
+      "dependencies": {
+        "path-type": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/dir-glob/node_modules/path-type": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz",
+      "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/doctrine": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz",
+      "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==",
+      "dev": true,
+      "dependencies": {
+        "esutils": "^2.0.2"
+      },
+      "engines": {
+        "node": ">=6.0.0"
+      }
+    },
+    "node_modules/dom-serializer": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-2.0.0.tgz",
+      "integrity": "sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==",
+      "dev": true,
+      "dependencies": {
+        "domelementtype": "^2.3.0",
+        "domhandler": "^5.0.2",
+        "entities": "^4.2.0"
+      },
+      "funding": {
+        "url": "https://github.com/cheeriojs/dom-serializer?sponsor=1"
+      }
+    },
+    "node_modules/dom-walk": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/dom-walk/-/dom-walk-0.1.2.tgz",
+      "integrity": "sha512-6QvTW9mrGeIegrFXdtQi9pk7O/nSK6lSdXW2eqUspN5LWD7UTji2Fqw5V2YLjBpHEoU9Xl/eUWNpDeZvoyOv2w==",
+      "dev": true
+    },
+    "node_modules/domelementtype": {
+      "version": "2.3.0",
+      "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.3.0.tgz",
+      "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/fb55"
+        }
+      ]
+    },
+    "node_modules/domhandler": {
+      "version": "5.0.3",
+      "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-5.0.3.tgz",
+      "integrity": "sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==",
+      "dev": true,
+      "dependencies": {
+        "domelementtype": "^2.3.0"
+      },
+      "engines": {
+        "node": ">= 4"
+      },
+      "funding": {
+        "url": "https://github.com/fb55/domhandler?sponsor=1"
+      }
+    },
+    "node_modules/domutils": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.0.1.tgz",
+      "integrity": "sha512-z08c1l761iKhDFtfXO04C7kTdPBLi41zwOZl00WS8b5eiaebNpY00HKbztwBq+e3vyqWNwWF3mP9YLUeqIrF+Q==",
+      "dev": true,
+      "dependencies": {
+        "dom-serializer": "^2.0.0",
+        "domelementtype": "^2.3.0",
+        "domhandler": "^5.0.1"
+      },
+      "funding": {
+        "url": "https://github.com/fb55/domutils?sponsor=1"
+      }
+    },
+    "node_modules/dot-case": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/dot-case/-/dot-case-2.1.1.tgz",
+      "integrity": "sha512-HnM6ZlFqcajLsyudHq7LeeLDr2rFAVYtDv/hV5qchQEidSck8j9OPUsXY9KwJv/lHMtYlX4DjRQqwFYa+0r8Ug==",
+      "dev": true,
+      "dependencies": {
+        "no-case": "^2.2.0"
+      }
+    },
+    "node_modules/dotenv": {
+      "version": "16.0.3",
+      "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.0.3.tgz",
+      "integrity": "sha512-7GO6HghkA5fYG9TYnNxi14/7K9f5occMlp3zXAuSxn7CKCxt9xbNWG7yF8hTCSUchlfWSe3uLmlPfigevRItzQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=12"
+      }
+    },
+    "node_modules/ecc-jsbn": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz",
+      "integrity": "sha512-eh9O+hwRHNbG4BLTjEl3nw044CkGm5X6LoaCf7LPp7UU8Qrt47JYNi6nPX8xjW97TKGKm1ouctg0QSpZe9qrnw==",
+      "dev": true,
+      "dependencies": {
+        "jsbn": "~0.1.0",
+        "safer-buffer": "^2.1.0"
+      }
+    },
+    "node_modules/ee-first": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
+      "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==",
+      "dev": true
+    },
+    "node_modules/elliptic": {
+      "version": "6.5.4",
+      "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.4.tgz",
+      "integrity": "sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ==",
+      "devOptional": true,
+      "dependencies": {
+        "bn.js": "^4.11.9",
+        "brorand": "^1.1.0",
+        "hash.js": "^1.0.0",
+        "hmac-drbg": "^1.0.1",
+        "inherits": "^2.0.4",
+        "minimalistic-assert": "^1.0.1",
+        "minimalistic-crypto-utils": "^1.0.1"
+      }
+    },
+    "node_modules/elliptic/node_modules/bn.js": {
+      "version": "4.12.0",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+      "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+      "devOptional": true
+    },
+    "node_modules/emoji-regex": {
+      "version": "8.0.0",
+      "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
+      "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
+      "dev": true
+    },
+    "node_modules/encode-utf8": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/encode-utf8/-/encode-utf8-1.0.3.tgz",
+      "integrity": "sha512-ucAnuBEhUK4boH2HjVYG5Q2mQyPorvv0u/ocS+zhdw0S8AlHYY+GOFhP1Gio5z4icpP2ivFSvhtFjQi8+T9ppw==",
+      "dev": true
+    },
+    "node_modules/encodeurl": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
+      "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/end-of-stream": {
+      "version": "1.4.4",
+      "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz",
+      "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==",
+      "dev": true,
+      "dependencies": {
+        "once": "^1.4.0"
+      }
+    },
+    "node_modules/enquirer": {
+      "version": "2.3.6",
+      "resolved": "https://registry.npmjs.org/enquirer/-/enquirer-2.3.6.tgz",
+      "integrity": "sha512-yjNnPr315/FjS4zIsUxYguYUPP2e1NK4d7E7ZOLiyYCcbFBiTMyID+2wvm2w6+pZ/odMA7cRkjhsPbltwBOrLg==",
+      "dev": true,
+      "dependencies": {
+        "ansi-colors": "^4.1.1"
+      },
+      "engines": {
+        "node": ">=8.6"
+      }
+    },
+    "node_modules/entities": {
+      "version": "4.4.0",
+      "resolved": "https://registry.npmjs.org/entities/-/entities-4.4.0.tgz",
+      "integrity": "sha512-oYp7156SP8LkeGD0GF85ad1X9Ai79WtRsZ2gxJqtBuzH+98YUV6jkHEKlZkMbcrjJjIVJNIDP/3WL9wQkoPbWA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.12"
+      },
+      "funding": {
+        "url": "https://github.com/fb55/entities?sponsor=1"
+      }
+    },
+    "node_modules/env-paths": {
+      "version": "2.2.1",
+      "resolved": "https://registry.npmjs.org/env-paths/-/env-paths-2.2.1.tgz",
+      "integrity": "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/error-ex": {
+      "version": "1.3.2",
+      "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz",
+      "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==",
+      "dev": true,
+      "dependencies": {
+        "is-arrayish": "^0.2.1"
+      }
+    },
+    "node_modules/es-abstract": {
+      "version": "1.21.1",
+      "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.21.1.tgz",
+      "integrity": "sha512-QudMsPOz86xYz/1dG1OuGBKOELjCh99IIWHLzy5znUB6j8xG2yMA7bfTV86VSqKF+Y/H08vQPR+9jyXpuC6hfg==",
+      "dev": true,
+      "dependencies": {
+        "available-typed-arrays": "^1.0.5",
+        "call-bind": "^1.0.2",
+        "es-set-tostringtag": "^2.0.1",
+        "es-to-primitive": "^1.2.1",
+        "function-bind": "^1.1.1",
+        "function.prototype.name": "^1.1.5",
+        "get-intrinsic": "^1.1.3",
+        "get-symbol-description": "^1.0.0",
+        "globalthis": "^1.0.3",
+        "gopd": "^1.0.1",
+        "has": "^1.0.3",
+        "has-property-descriptors": "^1.0.0",
+        "has-proto": "^1.0.1",
+        "has-symbols": "^1.0.3",
+        "internal-slot": "^1.0.4",
+        "is-array-buffer": "^3.0.1",
+        "is-callable": "^1.2.7",
+        "is-negative-zero": "^2.0.2",
+        "is-regex": "^1.1.4",
+        "is-shared-array-buffer": "^1.0.2",
+        "is-string": "^1.0.7",
+        "is-typed-array": "^1.1.10",
+        "is-weakref": "^1.0.2",
+        "object-inspect": "^1.12.2",
+        "object-keys": "^1.1.1",
+        "object.assign": "^4.1.4",
+        "regexp.prototype.flags": "^1.4.3",
+        "safe-regex-test": "^1.0.0",
+        "string.prototype.trimend": "^1.0.6",
+        "string.prototype.trimstart": "^1.0.6",
+        "typed-array-length": "^1.0.4",
+        "unbox-primitive": "^1.0.2",
+        "which-typed-array": "^1.1.9"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/es-set-tostringtag": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.1.tgz",
+      "integrity": "sha512-g3OMbtlwY3QewlqAiMLI47KywjWZoEytKr8pf6iTC8uJq5bIAH52Z9pnQ8pVL6whrCto53JZDuUIsifGeLorTg==",
+      "dev": true,
+      "dependencies": {
+        "get-intrinsic": "^1.1.3",
+        "has": "^1.0.3",
+        "has-tostringtag": "^1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      }
+    },
+    "node_modules/es-shim-unscopables": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/es-shim-unscopables/-/es-shim-unscopables-1.0.0.tgz",
+      "integrity": "sha512-Jm6GPcCdC30eMLbZ2x8z2WuRwAws3zTBBKuusffYVUrNj/GVSUAZ+xKMaUpfNDR5IbyNA5LJbaecoUVbmUcB1w==",
+      "dev": true,
+      "dependencies": {
+        "has": "^1.0.3"
+      }
+    },
+    "node_modules/es-to-primitive": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz",
+      "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==",
+      "dev": true,
+      "dependencies": {
+        "is-callable": "^1.1.4",
+        "is-date-object": "^1.0.1",
+        "is-symbol": "^1.0.2"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/es5-ext": {
+      "version": "0.10.62",
+      "resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.62.tgz",
+      "integrity": "sha512-BHLqn0klhEpnOKSrzn/Xsz2UIW8j+cGmo9JLzr8BiUapV8hPL9+FliFqjwr9ngW7jWdnxv6eO+/LqyhJVqgrjA==",
+      "dev": true,
+      "hasInstallScript": true,
+      "dependencies": {
+        "es6-iterator": "^2.0.3",
+        "es6-symbol": "^3.1.3",
+        "next-tick": "^1.1.0"
+      },
+      "engines": {
+        "node": ">=0.10"
+      }
+    },
+    "node_modules/es6-iterator": {
+      "version": "2.0.3",
+      "resolved": "https://registry.npmjs.org/es6-iterator/-/es6-iterator-2.0.3.tgz",
+      "integrity": "sha512-zw4SRzoUkd+cl+ZoE15A9o1oQd920Bb0iOJMQkQhl3jNc03YqVjAhG7scf9C5KWRU/R13Orf588uCC6525o02g==",
+      "dev": true,
+      "dependencies": {
+        "d": "1",
+        "es5-ext": "^0.10.35",
+        "es6-symbol": "^3.1.1"
+      }
+    },
+    "node_modules/es6-promise": {
+      "version": "4.2.8",
+      "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz",
+      "integrity": "sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==",
+      "dev": true
+    },
+    "node_modules/es6-symbol": {
+      "version": "3.1.3",
+      "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.3.tgz",
+      "integrity": "sha512-NJ6Yn3FuDinBaBRWl/q5X/s4koRHBrgKAu+yGI6JCBeiu3qrcbJhwT2GeR/EXVfylRk8dpQVJoLEFhK+Mu31NA==",
+      "dev": true,
+      "dependencies": {
+        "d": "^1.0.1",
+        "ext": "^1.1.2"
+      }
+    },
+    "node_modules/escalade": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz",
+      "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/escape-html": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
+      "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==",
+      "dev": true
+    },
+    "node_modules/escape-string-regexp": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
+      "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.8.0"
+      }
+    },
+    "node_modules/escodegen": {
+      "version": "1.8.1",
+      "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.8.1.tgz",
+      "integrity": "sha512-yhi5S+mNTOuRvyW4gWlg5W1byMaQGWWSYHXsuFZ7GBo7tpyOwi2EdzMP/QWxh9hwkD2m+wDVHJsxhRIj+v/b/A==",
+      "dev": true,
+      "dependencies": {
+        "esprima": "^2.7.1",
+        "estraverse": "^1.9.1",
+        "esutils": "^2.0.2",
+        "optionator": "^0.8.1"
+      },
+      "bin": {
+        "escodegen": "bin/escodegen.js",
+        "esgenerate": "bin/esgenerate.js"
+      },
+      "engines": {
+        "node": ">=0.12.0"
+      },
+      "optionalDependencies": {
+        "source-map": "~0.2.0"
+      }
+    },
+    "node_modules/escodegen/node_modules/esprima": {
+      "version": "2.7.3",
+      "resolved": "https://registry.npmjs.org/esprima/-/esprima-2.7.3.tgz",
+      "integrity": "sha512-OarPfz0lFCiW4/AV2Oy1Rp9qu0iusTKqykwTspGCZtPxmF81JR4MmIebvF1F9+UOKth2ZubLQ4XGGaU+hSn99A==",
+      "dev": true,
+      "bin": {
+        "esparse": "bin/esparse.js",
+        "esvalidate": "bin/esvalidate.js"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/escodegen/node_modules/estraverse": {
+      "version": "1.9.3",
+      "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-1.9.3.tgz",
+      "integrity": "sha512-25w1fMXQrGdoquWnScXZGckOv+Wes+JDnuN/+7ex3SauFRS72r2lFDec0EKPt2YD1wUJ/IrfEex+9yp4hfSOJA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/escodegen/node_modules/levn": {
+      "version": "0.3.0",
+      "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz",
+      "integrity": "sha512-0OO4y2iOHix2W6ujICbKIaEQXvFQHue65vUG3pb5EUomzPI90z9hsA1VsO/dbIIpC53J8gxM9Q4Oho0jrCM/yA==",
+      "dev": true,
+      "dependencies": {
+        "prelude-ls": "~1.1.2",
+        "type-check": "~0.3.2"
+      },
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/escodegen/node_modules/optionator": {
+      "version": "0.8.3",
+      "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz",
+      "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==",
+      "dev": true,
+      "dependencies": {
+        "deep-is": "~0.1.3",
+        "fast-levenshtein": "~2.0.6",
+        "levn": "~0.3.0",
+        "prelude-ls": "~1.1.2",
+        "type-check": "~0.3.2",
+        "word-wrap": "~1.2.3"
+      },
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/escodegen/node_modules/prelude-ls": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz",
+      "integrity": "sha512-ESF23V4SKG6lVSGZgYNpbsiaAkdab6ZgOxe52p7+Kid3W3u3bxR4Vfd/o21dmN7jSt0IwgZ4v5MUd26FEtXE9w==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/escodegen/node_modules/source-map": {
+      "version": "0.2.0",
+      "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.2.0.tgz",
+      "integrity": "sha512-CBdZ2oa/BHhS4xj5DlhjWNHcan57/5YuvfdLf17iVmIpd9KRm+DFLmC6nBNj+6Ua7Kt3TmOjDpQT1aTYOQtoUA==",
+      "dev": true,
+      "optional": true,
+      "dependencies": {
+        "amdefine": ">=0.0.4"
+      },
+      "engines": {
+        "node": ">=0.8.0"
+      }
+    },
+    "node_modules/escodegen/node_modules/type-check": {
+      "version": "0.3.2",
+      "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz",
+      "integrity": "sha512-ZCmOJdvOWDBYJlzAoFkC+Q0+bUyEOS1ltgp1MGU03fqHG+dbi9tBFU2Rd9QKiDZFAYrhPh2JUf7rZRIuHRKtOg==",
+      "dev": true,
+      "dependencies": {
+        "prelude-ls": "~1.1.2"
+      },
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/eslint": {
+      "version": "8.35.0",
+      "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.35.0.tgz",
+      "integrity": "sha512-BxAf1fVL7w+JLRQhWl2pzGeSiGqbWumV4WNvc9Rhp6tiCtm4oHnyPBSEtMGZwrQgudFQ+otqzWoPB7x+hxoWsw==",
+      "dev": true,
+      "dependencies": {
+        "@eslint/eslintrc": "^2.0.0",
+        "@eslint/js": "8.35.0",
+        "@humanwhocodes/config-array": "^0.11.8",
+        "@humanwhocodes/module-importer": "^1.0.1",
+        "@nodelib/fs.walk": "^1.2.8",
+        "ajv": "^6.10.0",
+        "chalk": "^4.0.0",
+        "cross-spawn": "^7.0.2",
+        "debug": "^4.3.2",
+        "doctrine": "^3.0.0",
+        "escape-string-regexp": "^4.0.0",
+        "eslint-scope": "^7.1.1",
+        "eslint-utils": "^3.0.0",
+        "eslint-visitor-keys": "^3.3.0",
+        "espree": "^9.4.0",
+        "esquery": "^1.4.2",
+        "esutils": "^2.0.2",
+        "fast-deep-equal": "^3.1.3",
+        "file-entry-cache": "^6.0.1",
+        "find-up": "^5.0.0",
+        "glob-parent": "^6.0.2",
+        "globals": "^13.19.0",
+        "grapheme-splitter": "^1.0.4",
+        "ignore": "^5.2.0",
+        "import-fresh": "^3.0.0",
+        "imurmurhash": "^0.1.4",
+        "is-glob": "^4.0.0",
+        "is-path-inside": "^3.0.3",
+        "js-sdsl": "^4.1.4",
+        "js-yaml": "^4.1.0",
+        "json-stable-stringify-without-jsonify": "^1.0.1",
+        "levn": "^0.4.1",
+        "lodash.merge": "^4.6.2",
+        "minimatch": "^3.1.2",
+        "natural-compare": "^1.4.0",
+        "optionator": "^0.9.1",
+        "regexpp": "^3.2.0",
+        "strip-ansi": "^6.0.1",
+        "strip-json-comments": "^3.1.0",
+        "text-table": "^0.2.0"
+      },
+      "bin": {
+        "eslint": "bin/eslint.js"
+      },
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "url": "https://opencollective.com/eslint"
+      }
+    },
+    "node_modules/eslint-config-prettier": {
+      "version": "8.7.0",
+      "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-8.7.0.tgz",
+      "integrity": "sha512-HHVXLSlVUhMSmyW4ZzEuvjpwqamgmlfkutD53cYXLikh4pt/modINRcCIApJ84czDxM4GZInwUrromsDdTImTA==",
+      "dev": true,
+      "bin": {
+        "eslint-config-prettier": "bin/cli.js"
+      },
+      "peerDependencies": {
+        "eslint": ">=7.0.0"
+      }
+    },
+    "node_modules/eslint-import-resolver-node": {
+      "version": "0.3.7",
+      "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.7.tgz",
+      "integrity": "sha512-gozW2blMLJCeFpBwugLTGyvVjNoeo1knonXAcatC6bjPBZitotxdWf7Gimr25N4c0AAOo4eOUfaG82IJPDpqCA==",
+      "dev": true,
+      "dependencies": {
+        "debug": "^3.2.7",
+        "is-core-module": "^2.11.0",
+        "resolve": "^1.22.1"
+      }
+    },
+    "node_modules/eslint-import-resolver-node/node_modules/debug": {
+      "version": "3.2.7",
+      "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz",
+      "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==",
+      "dev": true,
+      "dependencies": {
+        "ms": "^2.1.1"
+      }
+    },
+    "node_modules/eslint-module-utils": {
+      "version": "2.7.4",
+      "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.7.4.tgz",
+      "integrity": "sha512-j4GT+rqzCoRKHwURX7pddtIPGySnX9Si/cgMI5ztrcqOPtk5dDEeZ34CQVPphnqkJytlc97Vuk05Um2mJ3gEQA==",
+      "dev": true,
+      "dependencies": {
+        "debug": "^3.2.7"
+      },
+      "engines": {
+        "node": ">=4"
+      },
+      "peerDependenciesMeta": {
+        "eslint": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/eslint-module-utils/node_modules/debug": {
+      "version": "3.2.7",
+      "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz",
+      "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==",
+      "dev": true,
+      "dependencies": {
+        "ms": "^2.1.1"
+      }
+    },
+    "node_modules/eslint-plugin-import": {
+      "version": "2.27.5",
+      "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.27.5.tgz",
+      "integrity": "sha512-LmEt3GVofgiGuiE+ORpnvP+kAm3h6MLZJ4Q5HCyHADofsb4VzXFsRiWj3c0OFiV+3DWFh0qg3v9gcPlfc3zRow==",
+      "dev": true,
+      "dependencies": {
+        "array-includes": "^3.1.6",
+        "array.prototype.flat": "^1.3.1",
+        "array.prototype.flatmap": "^1.3.1",
+        "debug": "^3.2.7",
+        "doctrine": "^2.1.0",
+        "eslint-import-resolver-node": "^0.3.7",
+        "eslint-module-utils": "^2.7.4",
+        "has": "^1.0.3",
+        "is-core-module": "^2.11.0",
+        "is-glob": "^4.0.3",
+        "minimatch": "^3.1.2",
+        "object.values": "^1.1.6",
+        "resolve": "^1.22.1",
+        "semver": "^6.3.0",
+        "tsconfig-paths": "^3.14.1"
+      },
+      "engines": {
+        "node": ">=4"
+      },
+      "peerDependencies": {
+        "eslint": "^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8"
+      }
+    },
+    "node_modules/eslint-plugin-import/node_modules/debug": {
+      "version": "3.2.7",
+      "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz",
+      "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==",
+      "dev": true,
+      "dependencies": {
+        "ms": "^2.1.1"
+      }
+    },
+    "node_modules/eslint-plugin-import/node_modules/doctrine": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz",
+      "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==",
+      "dev": true,
+      "dependencies": {
+        "esutils": "^2.0.2"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/eslint-plugin-prettier": {
+      "version": "4.2.1",
+      "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-4.2.1.tgz",
+      "integrity": "sha512-f/0rXLXUt0oFYs8ra4w49wYZBG5GKZpAYsJSm6rnYL5uVDjd+zowwMwVZHnAjf4edNrKpCDYfXDgmRE/Ak7QyQ==",
+      "dev": true,
+      "dependencies": {
+        "prettier-linter-helpers": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=12.0.0"
+      },
+      "peerDependencies": {
+        "eslint": ">=7.28.0",
+        "prettier": ">=2.0.0"
+      },
+      "peerDependenciesMeta": {
+        "eslint-config-prettier": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/eslint-scope": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz",
+      "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==",
+      "dev": true,
+      "dependencies": {
+        "esrecurse": "^4.3.0",
+        "estraverse": "^4.1.1"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/eslint-utils": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-3.0.0.tgz",
+      "integrity": "sha512-uuQC43IGctw68pJA1RgbQS8/NP7rch6Cwd4j3ZBtgo4/8Flj4eGE7ZYSZRN3iq5pVUv6GPdW5Z1RFleo84uLDA==",
+      "dev": true,
+      "dependencies": {
+        "eslint-visitor-keys": "^2.0.0"
+      },
+      "engines": {
+        "node": "^10.0.0 || ^12.0.0 || >= 14.0.0"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/mysticatea"
+      },
+      "peerDependencies": {
+        "eslint": ">=5"
+      }
+    },
+    "node_modules/eslint-utils/node_modules/eslint-visitor-keys": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.1.0.tgz",
+      "integrity": "sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/eslint-visitor-keys": {
+      "version": "3.3.0",
+      "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.3.0.tgz",
+      "integrity": "sha512-mQ+suqKJVyeuwGYHAdjMFqjCyfl8+Ldnxuyp3ldiMBFKkvytrXUZWaiPCEav8qDHKty44bD+qV1IP4T+w+xXRA==",
+      "dev": true,
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      }
+    },
+    "node_modules/eslint/node_modules/ajv": {
+      "version": "6.12.6",
+      "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+      "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+      "dev": true,
+      "dependencies": {
+        "fast-deep-equal": "^3.1.1",
+        "fast-json-stable-stringify": "^2.0.0",
+        "json-schema-traverse": "^0.4.1",
+        "uri-js": "^4.2.2"
+      },
+      "funding": {
+        "type": "github",
+        "url": "https://github.com/sponsors/epoberezkin"
+      }
+    },
+    "node_modules/eslint/node_modules/escape-string-regexp": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz",
+      "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/eslint/node_modules/eslint-scope": {
+      "version": "7.1.1",
+      "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.1.1.tgz",
+      "integrity": "sha512-QKQM/UXpIiHcLqJ5AOyIW7XZmzjkzQXYE54n1++wb0u9V/abW3l9uQnxX8Z5Xd18xyKIMTUAyQ0k1e8pz6LUrw==",
+      "dev": true,
+      "dependencies": {
+        "esrecurse": "^4.3.0",
+        "estraverse": "^5.2.0"
+      },
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      }
+    },
+    "node_modules/eslint/node_modules/estraverse": {
+      "version": "5.3.0",
+      "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz",
+      "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==",
+      "dev": true,
+      "engines": {
+        "node": ">=4.0"
+      }
+    },
+    "node_modules/eslint/node_modules/find-up": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz",
+      "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==",
+      "dev": true,
+      "dependencies": {
+        "locate-path": "^6.0.0",
+        "path-exists": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/eslint/node_modules/glob-parent": {
+      "version": "6.0.2",
+      "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz",
+      "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==",
+      "dev": true,
+      "dependencies": {
+        "is-glob": "^4.0.3"
+      },
+      "engines": {
+        "node": ">=10.13.0"
+      }
+    },
+    "node_modules/eslint/node_modules/json-schema-traverse": {
+      "version": "0.4.1",
+      "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+      "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+      "dev": true
+    },
+    "node_modules/eslint/node_modules/locate-path": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz",
+      "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==",
+      "dev": true,
+      "dependencies": {
+        "p-locate": "^5.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/eslint/node_modules/p-limit": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
+      "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==",
+      "dev": true,
+      "dependencies": {
+        "yocto-queue": "^0.1.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/eslint/node_modules/p-locate": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz",
+      "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==",
+      "dev": true,
+      "dependencies": {
+        "p-limit": "^3.0.2"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/eslint/node_modules/path-exists": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
+      "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/espree": {
+      "version": "9.4.1",
+      "resolved": "https://registry.npmjs.org/espree/-/espree-9.4.1.tgz",
+      "integrity": "sha512-XwctdmTO6SIvCzd9810yyNzIrOrqNYV9Koizx4C/mRhf9uq0o4yHoCEU/670pOxOL/MSraektvSAji79kX90Vg==",
+      "dev": true,
+      "dependencies": {
+        "acorn": "^8.8.0",
+        "acorn-jsx": "^5.3.2",
+        "eslint-visitor-keys": "^3.3.0"
+      },
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "url": "https://opencollective.com/eslint"
+      }
+    },
+    "node_modules/esquery": {
+      "version": "1.5.0",
+      "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz",
+      "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==",
+      "dev": true,
+      "dependencies": {
+        "estraverse": "^5.1.0"
+      },
+      "engines": {
+        "node": ">=0.10"
+      }
+    },
+    "node_modules/esquery/node_modules/estraverse": {
+      "version": "5.3.0",
+      "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz",
+      "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==",
+      "dev": true,
+      "engines": {
+        "node": ">=4.0"
+      }
+    },
+    "node_modules/esrecurse": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz",
+      "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==",
+      "dev": true,
+      "dependencies": {
+        "estraverse": "^5.2.0"
+      },
+      "engines": {
+        "node": ">=4.0"
+      }
+    },
+    "node_modules/esrecurse/node_modules/estraverse": {
+      "version": "5.3.0",
+      "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz",
+      "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==",
+      "dev": true,
+      "engines": {
+        "node": ">=4.0"
+      }
+    },
+    "node_modules/estraverse": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz",
+      "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4.0"
+      }
+    },
+    "node_modules/esutils": {
+      "version": "2.0.3",
+      "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz",
+      "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/etag": {
+      "version": "1.8.1",
+      "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
+      "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/eth-ens-namehash": {
+      "version": "2.0.8",
+      "resolved": "https://registry.npmjs.org/eth-ens-namehash/-/eth-ens-namehash-2.0.8.tgz",
+      "integrity": "sha512-VWEI1+KJfz4Km//dadyvBBoBeSQ0MHTXPvr8UIXiLW6IanxvAV+DmlZAijZwAyggqGUfwQBeHf7tc9wzc1piSw==",
+      "dev": true,
+      "dependencies": {
+        "idna-uts46-hx": "^2.3.1",
+        "js-sha3": "^0.5.7"
+      }
+    },
+    "node_modules/eth-ens-namehash/node_modules/js-sha3": {
+      "version": "0.5.7",
+      "resolved": "https://registry.npmjs.org/js-sha3/-/js-sha3-0.5.7.tgz",
+      "integrity": "sha512-GII20kjaPX0zJ8wzkTbNDYMY7msuZcTWk8S5UOh6806Jq/wz1J8/bnr8uGU0DAUmYDjj2Mr4X1cW8v/GLYnR+g==",
+      "dev": true
+    },
+    "node_modules/eth-gas-reporter": {
+      "version": "0.2.27",
+      "resolved": "https://registry.npmjs.org/eth-gas-reporter/-/eth-gas-reporter-0.2.27.tgz",
+      "integrity": "sha512-femhvoAM7wL0GcI8ozTdxfuBtBFJ9qsyIAsmKVjlWAHUbdnnXHt+lKzz/kmldM5lA9jLuNHGwuIxorNpLbR1Zw==",
+      "dev": true,
+      "dependencies": {
+        "@solidity-parser/parser": "^0.14.0",
+        "axios": "^1.5.1",
+        "cli-table3": "^0.5.0",
+        "colors": "1.4.0",
+        "ethereum-cryptography": "^1.0.3",
+        "ethers": "^5.7.2",
+        "fs-readdir-recursive": "^1.1.0",
+        "lodash": "^4.17.14",
+        "markdown-table": "^1.1.3",
+        "mocha": "^10.2.0",
+        "req-cwd": "^2.0.0",
+        "sha1": "^1.1.1",
+        "sync-request": "^6.0.0"
+      },
+      "peerDependencies": {
+        "@codechecks/client": "^0.1.0"
+      },
+      "peerDependenciesMeta": {
+        "@codechecks/client": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/eth-gas-reporter/node_modules/ansi-regex": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz",
+      "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/eth-gas-reporter/node_modules/axios": {
+      "version": "1.6.2",
+      "resolved": "https://registry.npmjs.org/axios/-/axios-1.6.2.tgz",
+      "integrity": "sha512-7i24Ri4pmDRfJTR7LDBhsOTtcm+9kjX5WiY1X3wIisx6G9So3pfMkEiU7emUBe46oceVImccTEM3k6C5dbVW8A==",
+      "dev": true,
+      "dependencies": {
+        "follow-redirects": "^1.15.0",
+        "form-data": "^4.0.0",
+        "proxy-from-env": "^1.1.0"
+      }
+    },
+    "node_modules/eth-gas-reporter/node_modules/cli-table3": {
+      "version": "0.5.1",
+      "resolved": "https://registry.npmjs.org/cli-table3/-/cli-table3-0.5.1.tgz",
+      "integrity": "sha512-7Qg2Jrep1S/+Q3EceiZtQcDPWxhAvBw+ERf1162v4sikJrvojMHFqXt8QIVha8UlH9rgU0BeWPytZ9/TzYqlUw==",
+      "dev": true,
+      "dependencies": {
+        "object-assign": "^4.1.0",
+        "string-width": "^2.1.1"
+      },
+      "engines": {
+        "node": ">=6"
+      },
+      "optionalDependencies": {
+        "colors": "^1.1.2"
+      }
+    },
+    "node_modules/eth-gas-reporter/node_modules/ethereum-cryptography": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/ethereum-cryptography/-/ethereum-cryptography-1.2.0.tgz",
+      "integrity": "sha512-6yFQC9b5ug6/17CQpCyE3k9eKBMdhyVjzUy1WkiuY/E4vj/SXDBbCw8QEIaXqf0Mf2SnY6RmpDcwlUmBSS0EJw==",
+      "dev": true,
+      "dependencies": {
+        "@noble/hashes": "1.2.0",
+        "@noble/secp256k1": "1.7.1",
+        "@scure/bip32": "1.1.5",
+        "@scure/bip39": "1.1.1"
+      }
+    },
+    "node_modules/eth-gas-reporter/node_modules/form-data": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
+      "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
+      "dev": true,
+      "dependencies": {
+        "asynckit": "^0.4.0",
+        "combined-stream": "^1.0.8",
+        "mime-types": "^2.1.12"
+      },
+      "engines": {
+        "node": ">= 6"
+      }
+    },
+    "node_modules/eth-gas-reporter/node_modules/is-fullwidth-code-point": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
+      "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/eth-gas-reporter/node_modules/string-width": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz",
+      "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==",
+      "dev": true,
+      "dependencies": {
+        "is-fullwidth-code-point": "^2.0.0",
+        "strip-ansi": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/eth-gas-reporter/node_modules/strip-ansi": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz",
+      "integrity": "sha512-4XaJ2zQdCzROZDivEVIDPkcQn8LMFSa8kj8Gxb/Lnwzv9A8VctNZ+lfivC/sV3ivW8ElJTERXZoPBRrZKkNKow==",
+      "dev": true,
+      "dependencies": {
+        "ansi-regex": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/eth-lib": {
+      "version": "0.1.29",
+      "resolved": "https://registry.npmjs.org/eth-lib/-/eth-lib-0.1.29.tgz",
+      "integrity": "sha512-bfttrr3/7gG4E02HoWTDUcDDslN003OlOoBxk9virpAZQ1ja/jDgwkWB8QfJF7ojuEowrqy+lzp9VcJG7/k5bQ==",
+      "dev": true,
+      "dependencies": {
+        "bn.js": "^4.11.6",
+        "elliptic": "^6.4.0",
+        "nano-json-stream-parser": "^0.1.2",
+        "servify": "^0.1.12",
+        "ws": "^3.0.0",
+        "xhr-request-promise": "^0.1.2"
+      }
+    },
+    "node_modules/eth-lib/node_modules/bn.js": {
+      "version": "4.12.0",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+      "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+      "dev": true
+    },
+    "node_modules/eth-lib/node_modules/safe-buffer": {
+      "version": "5.1.2",
+      "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
+      "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==",
+      "dev": true
+    },
+    "node_modules/eth-lib/node_modules/ws": {
+      "version": "3.3.3",
+      "resolved": "https://registry.npmjs.org/ws/-/ws-3.3.3.tgz",
+      "integrity": "sha512-nnWLa/NwZSt4KQJu51MYlCcSQ5g7INpOrOMt4XV8j4dqTXdmlUmSHQ8/oLC069ckre0fRsgfvsKwbTdtKLCDkA==",
+      "dev": true,
+      "dependencies": {
+        "async-limiter": "~1.0.0",
+        "safe-buffer": "~5.1.0",
+        "ultron": "~1.1.0"
+      }
+    },
+    "node_modules/ethereum-bloom-filters": {
+      "version": "1.0.10",
+      "resolved": "https://registry.npmjs.org/ethereum-bloom-filters/-/ethereum-bloom-filters-1.0.10.tgz",
+      "integrity": "sha512-rxJ5OFN3RwjQxDcFP2Z5+Q9ho4eIdEmSc2ht0fCu8Se9nbXjZ7/031uXoUYJ87KHCOdVeiUuwSnoS7hmYAGVHA==",
+      "dev": true,
+      "dependencies": {
+        "js-sha3": "^0.8.0"
+      }
+    },
+    "node_modules/ethereum-cryptography": {
+      "version": "0.1.3",
+      "resolved": "https://registry.npmjs.org/ethereum-cryptography/-/ethereum-cryptography-0.1.3.tgz",
+      "integrity": "sha512-w8/4x1SGGzc+tO97TASLja6SLd3fRIK2tLVcV2Gx4IB21hE19atll5Cq9o3d0ZmAYC/8aw0ipieTSiekAea4SQ==",
+      "dev": true,
+      "dependencies": {
+        "@types/pbkdf2": "^3.0.0",
+        "@types/secp256k1": "^4.0.1",
+        "blakejs": "^1.1.0",
+        "browserify-aes": "^1.2.0",
+        "bs58check": "^2.1.2",
+        "create-hash": "^1.2.0",
+        "create-hmac": "^1.1.7",
+        "hash.js": "^1.1.7",
+        "keccak": "^3.0.0",
+        "pbkdf2": "^3.0.17",
+        "randombytes": "^2.1.0",
+        "safe-buffer": "^5.1.2",
+        "scrypt-js": "^3.0.0",
+        "secp256k1": "^4.0.1",
+        "setimmediate": "^1.0.5"
+      }
+    },
+    "node_modules/ethereum-ens": {
+      "version": "0.8.0",
+      "resolved": "https://registry.npmjs.org/ethereum-ens/-/ethereum-ens-0.8.0.tgz",
+      "integrity": "sha512-a8cBTF4AWw1Q1Y37V1LSCS9pRY4Mh3f8vCg5cbXCCEJ3eno1hbI/+Ccv9SZLISYpqQhaglP3Bxb/34lS4Qf7Bg==",
+      "dev": true,
+      "dependencies": {
+        "bluebird": "^3.4.7",
+        "eth-ens-namehash": "^2.0.0",
+        "js-sha3": "^0.5.7",
+        "pako": "^1.0.4",
+        "underscore": "^1.8.3",
+        "web3": "^1.0.0-beta.34"
+      }
+    },
+    "node_modules/ethereum-ens/node_modules/js-sha3": {
+      "version": "0.5.7",
+      "resolved": "https://registry.npmjs.org/js-sha3/-/js-sha3-0.5.7.tgz",
+      "integrity": "sha512-GII20kjaPX0zJ8wzkTbNDYMY7msuZcTWk8S5UOh6806Jq/wz1J8/bnr8uGU0DAUmYDjj2Mr4X1cW8v/GLYnR+g==",
+      "dev": true
+    },
+    "node_modules/ethereumjs-abi": {
+      "version": "0.6.8",
+      "resolved": "https://registry.npmjs.org/ethereumjs-abi/-/ethereumjs-abi-0.6.8.tgz",
+      "integrity": "sha512-Tx0r/iXI6r+lRsdvkFDlut0N08jWMnKRZ6Gkq+Nmw75lZe4e6o3EkSnkaBP5NF6+m5PTGAr9JP43N3LyeoglsA==",
+      "dev": true,
+      "dependencies": {
+        "bn.js": "^4.11.8",
+        "ethereumjs-util": "^6.0.0"
+      }
+    },
+    "node_modules/ethereumjs-abi/node_modules/@types/bn.js": {
+      "version": "4.11.6",
+      "resolved": "https://registry.npmjs.org/@types/bn.js/-/bn.js-4.11.6.tgz",
+      "integrity": "sha512-pqr857jrp2kPuO9uRjZ3PwnJTjoQy+fcdxvBTvHm6dkmEL9q+hDD/2j/0ELOBPtPnS8LjCX0gI9nbl8lVkadpg==",
+      "dev": true,
+      "dependencies": {
+        "@types/node": "*"
+      }
+    },
+    "node_modules/ethereumjs-abi/node_modules/bn.js": {
+      "version": "4.12.0",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+      "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+      "dev": true
+    },
+    "node_modules/ethereumjs-abi/node_modules/ethereumjs-util": {
+      "version": "6.2.1",
+      "resolved": "https://registry.npmjs.org/ethereumjs-util/-/ethereumjs-util-6.2.1.tgz",
+      "integrity": "sha512-W2Ktez4L01Vexijrm5EB6w7dg4n/TgpoYU4avuT5T3Vmnw/eCRtiBrJfQYS/DCSvDIOLn2k57GcHdeBcgVxAqw==",
+      "dev": true,
+      "dependencies": {
+        "@types/bn.js": "^4.11.3",
+        "bn.js": "^4.11.0",
+        "create-hash": "^1.1.2",
+        "elliptic": "^6.5.2",
+        "ethereum-cryptography": "^0.1.3",
+        "ethjs-util": "0.1.6",
+        "rlp": "^2.2.3"
+      }
+    },
+    "node_modules/ethereumjs-util": {
+      "version": "7.1.5",
+      "resolved": "https://registry.npmjs.org/ethereumjs-util/-/ethereumjs-util-7.1.5.tgz",
+      "integrity": "sha512-SDl5kKrQAudFBUe5OJM9Ac6WmMyYmXX/6sTmLZ3ffG2eY6ZIGBes3pEDxNN6V72WyOw4CPD5RomKdsa8DAAwLg==",
+      "dev": true,
+      "dependencies": {
+        "@types/bn.js": "^5.1.0",
+        "bn.js": "^5.1.2",
+        "create-hash": "^1.1.2",
+        "ethereum-cryptography": "^0.1.3",
+        "rlp": "^2.2.4"
+      },
+      "engines": {
+        "node": ">=10.0.0"
+      }
+    },
+    "node_modules/ethers": {
+      "version": "5.7.2",
+      "resolved": "https://registry.npmjs.org/ethers/-/ethers-5.7.2.tgz",
+      "integrity": "sha512-wswUsmWo1aOK8rR7DIKiWSw9DbLWe6x98Jrn8wcTflTVvaXhAMaB5zGAXy0GYQEQp9iO1iSHWVyARQm11zUtyg==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abi": "5.7.0",
+        "@ethersproject/abstract-provider": "5.7.0",
+        "@ethersproject/abstract-signer": "5.7.0",
+        "@ethersproject/address": "5.7.0",
+        "@ethersproject/base64": "5.7.0",
+        "@ethersproject/basex": "5.7.0",
+        "@ethersproject/bignumber": "5.7.0",
+        "@ethersproject/bytes": "5.7.0",
+        "@ethersproject/constants": "5.7.0",
+        "@ethersproject/contracts": "5.7.0",
+        "@ethersproject/hash": "5.7.0",
+        "@ethersproject/hdnode": "5.7.0",
+        "@ethersproject/json-wallets": "5.7.0",
+        "@ethersproject/keccak256": "5.7.0",
+        "@ethersproject/logger": "5.7.0",
+        "@ethersproject/networks": "5.7.1",
+        "@ethersproject/pbkdf2": "5.7.0",
+        "@ethersproject/properties": "5.7.0",
+        "@ethersproject/providers": "5.7.2",
+        "@ethersproject/random": "5.7.0",
+        "@ethersproject/rlp": "5.7.0",
+        "@ethersproject/sha2": "5.7.0",
+        "@ethersproject/signing-key": "5.7.0",
+        "@ethersproject/solidity": "5.7.0",
+        "@ethersproject/strings": "5.7.0",
+        "@ethersproject/transactions": "5.7.0",
+        "@ethersproject/units": "5.7.0",
+        "@ethersproject/wallet": "5.7.0",
+        "@ethersproject/web": "5.7.1",
+        "@ethersproject/wordlists": "5.7.0"
+      }
+    },
+    "node_modules/ethers/node_modules/@ethersproject/abstract-signer": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+      "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+      "devOptional": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://gitcoin.co/grants/13/ethersjs-complete-simple-and-tiny-2"
+        },
+        {
+          "type": "individual",
+          "url": "https://www.buymeacoffee.com/ricmoo"
+        }
+      ],
+      "dependencies": {
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0"
+      }
+    },
+    "node_modules/ethjs-unit": {
+      "version": "0.1.6",
+      "resolved": "https://registry.npmjs.org/ethjs-unit/-/ethjs-unit-0.1.6.tgz",
+      "integrity": "sha512-/Sn9Y0oKl0uqQuvgFk/zQgR7aw1g36qX/jzSQ5lSwlO0GigPymk4eGQfeNTD03w1dPOqfz8V77Cy43jH56pagw==",
+      "dev": true,
+      "dependencies": {
+        "bn.js": "4.11.6",
+        "number-to-bn": "1.7.0"
+      },
+      "engines": {
+        "node": ">=6.5.0",
+        "npm": ">=3"
+      }
+    },
+    "node_modules/ethjs-unit/node_modules/bn.js": {
+      "version": "4.11.6",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.6.tgz",
+      "integrity": "sha512-XWwnNNFCuuSQ0m3r3C4LE3EiORltHd9M05pq6FOlVeiophzRbMo50Sbz1ehl8K3Z+jw9+vmgnXefY1hz8X+2wA==",
+      "dev": true
+    },
+    "node_modules/ethjs-util": {
+      "version": "0.1.6",
+      "resolved": "https://registry.npmjs.org/ethjs-util/-/ethjs-util-0.1.6.tgz",
+      "integrity": "sha512-CUnVOQq7gSpDHZVVrQW8ExxUETWrnrvXYvYz55wOU8Uj4VCgw56XC2B/fVqQN+f7gmrnRHSLVnFAwsCuNwji8w==",
+      "dev": true,
+      "dependencies": {
+        "is-hex-prefixed": "1.0.0",
+        "strip-hex-prefix": "1.0.0"
+      },
+      "engines": {
+        "node": ">=6.5.0",
+        "npm": ">=3"
+      }
+    },
+    "node_modules/event-target-shim": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz",
+      "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/eventemitter3": {
+      "version": "4.0.4",
+      "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.4.tgz",
+      "integrity": "sha512-rlaVLnVxtxvoyLsQQFBx53YmXHDxRIzzTLbdfxqi4yocpSjAxXwkU0cScM5JgSKMqEhrZpnvQ2D9gjylR0AimQ==",
+      "dev": true
+    },
+    "node_modules/evp_bytestokey": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz",
+      "integrity": "sha512-/f2Go4TognH/KvCISP7OUsHn85hT9nUkxxA9BEWxFn+Oj9o8ZNLm/40hdlgSLyuOimsrTKLUMEorQexp/aPQeA==",
+      "dev": true,
+      "dependencies": {
+        "md5.js": "^1.3.4",
+        "safe-buffer": "^5.1.1"
+      }
+    },
+    "node_modules/express": {
+      "version": "4.18.2",
+      "resolved": "https://registry.npmjs.org/express/-/express-4.18.2.tgz",
+      "integrity": "sha512-5/PsL6iGPdfQ/lKM1UuielYgv3BUoJfz1aUwU9vHZ+J7gyvwdQXFEBIEIaxeGf0GIcreATNyBExtalisDbuMqQ==",
+      "dev": true,
+      "dependencies": {
+        "accepts": "~1.3.8",
+        "array-flatten": "1.1.1",
+        "body-parser": "1.20.1",
+        "content-disposition": "0.5.4",
+        "content-type": "~1.0.4",
+        "cookie": "0.5.0",
+        "cookie-signature": "1.0.6",
+        "debug": "2.6.9",
+        "depd": "2.0.0",
+        "encodeurl": "~1.0.2",
+        "escape-html": "~1.0.3",
+        "etag": "~1.8.1",
+        "finalhandler": "1.2.0",
+        "fresh": "0.5.2",
+        "http-errors": "2.0.0",
+        "merge-descriptors": "1.0.1",
+        "methods": "~1.1.2",
+        "on-finished": "2.4.1",
+        "parseurl": "~1.3.3",
+        "path-to-regexp": "0.1.7",
+        "proxy-addr": "~2.0.7",
+        "qs": "6.11.0",
+        "range-parser": "~1.2.1",
+        "safe-buffer": "5.2.1",
+        "send": "0.18.0",
+        "serve-static": "1.15.0",
+        "setprototypeof": "1.2.0",
+        "statuses": "2.0.1",
+        "type-is": "~1.6.18",
+        "utils-merge": "1.0.1",
+        "vary": "~1.1.2"
+      },
+      "engines": {
+        "node": ">= 0.10.0"
+      }
+    },
+    "node_modules/express/node_modules/body-parser": {
+      "version": "1.20.1",
+      "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.1.tgz",
+      "integrity": "sha512-jWi7abTbYwajOytWCQc37VulmWiRae5RyTpaCyDcS5/lMdtwSz5lOpDE67srw/HYe35f1z3fDQw+3txg7gNtWw==",
+      "dev": true,
+      "dependencies": {
+        "bytes": "3.1.2",
+        "content-type": "~1.0.4",
+        "debug": "2.6.9",
+        "depd": "2.0.0",
+        "destroy": "1.2.0",
+        "http-errors": "2.0.0",
+        "iconv-lite": "0.4.24",
+        "on-finished": "2.4.1",
+        "qs": "6.11.0",
+        "raw-body": "2.5.1",
+        "type-is": "~1.6.18",
+        "unpipe": "1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.8",
+        "npm": "1.2.8000 || >= 1.4.16"
+      }
+    },
+    "node_modules/express/node_modules/debug": {
+      "version": "2.6.9",
+      "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+      "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+      "dev": true,
+      "dependencies": {
+        "ms": "2.0.0"
+      }
+    },
+    "node_modules/express/node_modules/ms": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+      "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
+      "dev": true
+    },
+    "node_modules/express/node_modules/raw-body": {
+      "version": "2.5.1",
+      "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.1.tgz",
+      "integrity": "sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig==",
+      "dev": true,
+      "dependencies": {
+        "bytes": "3.1.2",
+        "http-errors": "2.0.0",
+        "iconv-lite": "0.4.24",
+        "unpipe": "1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/ext": {
+      "version": "1.7.0",
+      "resolved": "https://registry.npmjs.org/ext/-/ext-1.7.0.tgz",
+      "integrity": "sha512-6hxeJYaL110a9b5TEJSj0gojyHQAmA2ch5Os+ySCiA1QGdS697XWY1pzsrSjqA9LDEEgdB/KypIlR59RcLuHYw==",
+      "dev": true,
+      "dependencies": {
+        "type": "^2.7.2"
+      }
+    },
+    "node_modules/ext/node_modules/type": {
+      "version": "2.7.2",
+      "resolved": "https://registry.npmjs.org/type/-/type-2.7.2.tgz",
+      "integrity": "sha512-dzlvlNlt6AXU7EBSfpAscydQ7gXB+pPGsPnfJnZpiNJBDj7IaJzQlBZYGdEi4R9HmPdBv2XmWJ6YUtoTa7lmCw==",
+      "dev": true
+    },
+    "node_modules/extend": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz",
+      "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==",
+      "dev": true
+    },
+    "node_modules/extsprintf": {
+      "version": "1.3.0",
+      "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz",
+      "integrity": "sha512-11Ndz7Nv+mvAC1j0ktTa7fAb0vLyGGX+rMHNBYQviQDGU0Hw7lhctJANqbPhu9nV9/izT/IntTgZ7Im/9LJs9g==",
+      "dev": true,
+      "engines": [
+        "node >=0.6.0"
+      ]
+    },
+    "node_modules/fast-check": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/fast-check/-/fast-check-3.1.1.tgz",
+      "integrity": "sha512-3vtXinVyuUKCKFKYcwXhGE6NtGWkqF8Yh3rvMZNzmwz8EPrgoc/v4pDdLHyLnCyCI5MZpZZkDEwFyXyEONOxpA==",
+      "dev": true,
+      "dependencies": {
+        "pure-rand": "^5.0.1"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/fast-check"
+      }
+    },
+    "node_modules/fast-deep-equal": {
+      "version": "3.1.3",
+      "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
+      "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==",
+      "dev": true
+    },
+    "node_modules/fast-diff": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/fast-diff/-/fast-diff-1.2.0.tgz",
+      "integrity": "sha512-xJuoT5+L99XlZ8twedaRf6Ax2TgQVxvgZOYoPKqZufmJib0tL2tegPBOZb1pVNgIhlqDlA0eO0c3wBvQcmzx4w==",
+      "dev": true
+    },
+    "node_modules/fast-glob": {
+      "version": "3.2.12",
+      "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.12.tgz",
+      "integrity": "sha512-DVj4CQIYYow0BlaelwK1pHl5n5cRSJfM60UA0zK891sVInoPri2Ekj7+e1CT3/3qxXenpI+nBBmQAcJPJgaj4w==",
+      "dev": true,
+      "dependencies": {
+        "@nodelib/fs.stat": "^2.0.2",
+        "@nodelib/fs.walk": "^1.2.3",
+        "glob-parent": "^5.1.2",
+        "merge2": "^1.3.0",
+        "micromatch": "^4.0.4"
+      },
+      "engines": {
+        "node": ">=8.6.0"
+      }
+    },
+    "node_modules/fast-json-stable-stringify": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz",
+      "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==",
+      "dev": true
+    },
+    "node_modules/fast-levenshtein": {
+      "version": "2.0.6",
+      "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz",
+      "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==",
+      "dev": true
+    },
+    "node_modules/fastq": {
+      "version": "1.15.0",
+      "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.15.0.tgz",
+      "integrity": "sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw==",
+      "dev": true,
+      "dependencies": {
+        "reusify": "^1.0.4"
+      }
+    },
+    "node_modules/file-entry-cache": {
+      "version": "6.0.1",
+      "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz",
+      "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==",
+      "dev": true,
+      "dependencies": {
+        "flat-cache": "^3.0.4"
+      },
+      "engines": {
+        "node": "^10.12.0 || >=12.0.0"
+      }
+    },
+    "node_modules/fill-range": {
+      "version": "7.0.1",
+      "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
+      "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==",
+      "dev": true,
+      "dependencies": {
+        "to-regex-range": "^5.0.1"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/finalhandler": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.2.0.tgz",
+      "integrity": "sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==",
+      "dev": true,
+      "dependencies": {
+        "debug": "2.6.9",
+        "encodeurl": "~1.0.2",
+        "escape-html": "~1.0.3",
+        "on-finished": "2.4.1",
+        "parseurl": "~1.3.3",
+        "statuses": "2.0.1",
+        "unpipe": "~1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/finalhandler/node_modules/debug": {
+      "version": "2.6.9",
+      "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+      "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+      "dev": true,
+      "dependencies": {
+        "ms": "2.0.0"
+      }
+    },
+    "node_modules/finalhandler/node_modules/ms": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+      "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
+      "dev": true
+    },
+    "node_modules/find-replace": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/find-replace/-/find-replace-3.0.0.tgz",
+      "integrity": "sha512-6Tb2myMioCAgv5kfvP5/PkZZ/ntTpVK39fHY7WkWBgvbeE+VHd/tZuZ4mrC+bxh4cfOZeYKVPaJIZtZXV7GNCQ==",
+      "dev": true,
+      "dependencies": {
+        "array-back": "^3.0.1"
+      },
+      "engines": {
+        "node": ">=4.0.0"
+      }
+    },
+    "node_modules/find-up": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/find-up/-/find-up-1.1.2.tgz",
+      "integrity": "sha512-jvElSjyuo4EMQGoTwo1uJU5pQMwTW5lS1x05zzfJuTIyLR3zwO27LYrxNg+dlvKpGOuGy/MzBdXh80g0ve5+HA==",
+      "dev": true,
+      "dependencies": {
+        "path-exists": "^2.0.0",
+        "pinkie-promise": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/find-versions": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/find-versions/-/find-versions-4.0.0.tgz",
+      "integrity": "sha512-wgpWy002tA+wgmO27buH/9KzyEOQnKsG/R0yrcjPT9BOFm0zRBVQbZ95nRGXWMywS8YR5knRbpohio0bcJABxQ==",
+      "dev": true,
+      "dependencies": {
+        "semver-regex": "^3.1.2"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/flat": {
+      "version": "5.0.2",
+      "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz",
+      "integrity": "sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==",
+      "dev": true,
+      "bin": {
+        "flat": "cli.js"
+      }
+    },
+    "node_modules/flat-cache": {
+      "version": "3.0.4",
+      "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz",
+      "integrity": "sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==",
+      "dev": true,
+      "dependencies": {
+        "flatted": "^3.1.0",
+        "rimraf": "^3.0.2"
+      },
+      "engines": {
+        "node": "^10.12.0 || >=12.0.0"
+      }
+    },
+    "node_modules/flatted": {
+      "version": "3.2.7",
+      "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.7.tgz",
+      "integrity": "sha512-5nqDSxl8nn5BSNxyR3n4I6eDmbolI6WT+QqR547RwxQapgjQBmtktdP+HTBb/a/zLsbzERTONyUB5pefh5TtjQ==",
+      "dev": true
+    },
+    "node_modules/fmix": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/fmix/-/fmix-0.1.0.tgz",
+      "integrity": "sha512-Y6hyofImk9JdzU8k5INtTXX1cu8LDlePWDFU5sftm9H+zKCr5SGrVjdhkvsim646cw5zD0nADj8oHyXMZmCZ9w==",
+      "dev": true,
+      "dependencies": {
+        "imul": "^1.0.0"
+      }
+    },
+    "node_modules/follow-redirects": {
+      "version": "1.15.2",
+      "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz",
+      "integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://github.com/sponsors/RubenVerborgh"
+        }
+      ],
+      "engines": {
+        "node": ">=4.0"
+      },
+      "peerDependenciesMeta": {
+        "debug": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/for-each": {
+      "version": "0.3.3",
+      "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz",
+      "integrity": "sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==",
+      "dev": true,
+      "dependencies": {
+        "is-callable": "^1.1.3"
+      }
+    },
+    "node_modules/forever-agent": {
+      "version": "0.6.1",
+      "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz",
+      "integrity": "sha512-j0KLYPhm6zeac4lz3oJ3o65qvgQCcPubiyotZrXqEaG4hNagNYO8qdlUrX5vwqv9ohqeT/Z3j6+yW067yWWdUw==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/form-data": {
+      "version": "2.3.3",
+      "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz",
+      "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==",
+      "dev": true,
+      "dependencies": {
+        "asynckit": "^0.4.0",
+        "combined-stream": "^1.0.6",
+        "mime-types": "^2.1.12"
+      },
+      "engines": {
+        "node": ">= 0.12"
+      }
+    },
+    "node_modules/form-data-encoder": {
+      "version": "1.7.1",
+      "resolved": "https://registry.npmjs.org/form-data-encoder/-/form-data-encoder-1.7.1.tgz",
+      "integrity": "sha512-EFRDrsMm/kyqbTQocNvRXMLjc7Es2Vk+IQFx/YW7hkUH1eBl4J1fqiP34l74Yt0pFLCNpc06fkbVk00008mzjg==",
+      "dev": true
+    },
+    "node_modules/forwarded": {
+      "version": "0.2.0",
+      "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
+      "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/fp-ts": {
+      "version": "1.19.3",
+      "resolved": "https://registry.npmjs.org/fp-ts/-/fp-ts-1.19.3.tgz",
+      "integrity": "sha512-H5KQDspykdHuztLTg+ajGN0Z2qUjcEf3Ybxc6hLt0k7/zPkn29XnKnxlBPyW2XIddWrGaJBzBl4VLYOtk39yZg==",
+      "dev": true
+    },
+    "node_modules/fresh": {
+      "version": "0.5.2",
+      "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
+      "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/fs-extra": {
+      "version": "7.0.1",
+      "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-7.0.1.tgz",
+      "integrity": "sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==",
+      "dev": true,
+      "dependencies": {
+        "graceful-fs": "^4.1.2",
+        "jsonfile": "^4.0.0",
+        "universalify": "^0.1.0"
+      },
+      "engines": {
+        "node": ">=6 <7 || >=8"
+      }
+    },
+    "node_modules/fs-minipass": {
+      "version": "1.2.7",
+      "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-1.2.7.tgz",
+      "integrity": "sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==",
+      "dev": true,
+      "dependencies": {
+        "minipass": "^2.6.0"
+      }
+    },
+    "node_modules/fs-readdir-recursive": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/fs-readdir-recursive/-/fs-readdir-recursive-1.1.0.tgz",
+      "integrity": "sha512-GNanXlVr2pf02+sPN40XN8HG+ePaNcvM0q5mZBd668Obwb0yD5GiUbZOFgwn8kGMY6I3mdyDJzieUy3PTYyTRA==",
+      "dev": true
+    },
+    "node_modules/fs.realpath": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
+      "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==",
+      "dev": true
+    },
+    "node_modules/fsevents": {
+      "version": "2.3.2",
+      "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz",
+      "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==",
+      "dev": true,
+      "hasInstallScript": true,
+      "optional": true,
+      "os": [
+        "darwin"
+      ],
+      "engines": {
+        "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
+      }
+    },
+    "node_modules/function-bind": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz",
+      "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==",
+      "dev": true
+    },
+    "node_modules/function.prototype.name": {
+      "version": "1.1.5",
+      "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.5.tgz",
+      "integrity": "sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.3",
+        "es-abstract": "^1.19.0",
+        "functions-have-names": "^1.2.2"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/functional-red-black-tree": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz",
+      "integrity": "sha512-dsKNQNdj6xA3T+QlADDA7mOSlX0qiMINjn0cgr+eGHGsbSHzTabcIogz2+p/iqP1Xs6EP/sS2SbqH+brGTbq0g==",
+      "dev": true
+    },
+    "node_modules/functions-have-names": {
+      "version": "1.2.3",
+      "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz",
+      "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==",
+      "dev": true,
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/ganache": {
+      "version": "7.7.6",
+      "resolved": "https://registry.npmjs.org/ganache/-/ganache-7.7.6.tgz",
+      "integrity": "sha512-1ba5CERykZijw1kIRGUKKPEUqTDU+sEMElYemAS42w1kunu+/3OS5v+eQsJQ+fCVMEmspploA7S9rEWBcyVsLg==",
+      "bundleDependencies": [
+        "@trufflesuite/bigint-buffer",
+        "keccak",
+        "leveldown",
+        "secp256k1"
+      ],
+      "dev": true,
+      "hasShrinkwrap": true,
+      "dependencies": {
+        "@trufflesuite/bigint-buffer": "1.1.10",
+        "@types/bn.js": "^5.1.0",
+        "@types/lru-cache": "5.1.1",
+        "@types/seedrandom": "3.0.1",
+        "abstract-level": "1.0.3",
+        "abstract-leveldown": "7.2.0",
+        "async-eventemitter": "0.2.4",
+        "emittery": "0.10.0",
+        "keccak": "3.0.2",
+        "leveldown": "6.1.0",
+        "secp256k1": "4.0.3"
+      },
+      "bin": {
+        "ganache": "dist/node/cli.js",
+        "ganache-cli": "dist/node/cli.js"
+      },
+      "optionalDependencies": {
+        "bufferutil": "4.0.5",
+        "utf-8-validate": "5.0.7"
+      }
+    },
+    "node_modules/ganache/node_modules/@trufflesuite/bigint-buffer": {
+      "version": "1.1.10",
+      "resolved": "https://registry.npmjs.org/@trufflesuite/bigint-buffer/-/bigint-buffer-1.1.10.tgz",
+      "integrity": "sha512-pYIQC5EcMmID74t26GCC67946mgTJFiLXOT/BYozgrd4UEY2JHEGLhWi9cMiQCt5BSqFEvKkCHNnoj82SRjiEw==",
+      "dev": true,
+      "hasInstallScript": true,
+      "inBundle": true,
+      "license": "Apache-2.0",
+      "dependencies": {
+        "node-gyp-build": "4.4.0"
+      },
+      "engines": {
+        "node": ">= 14.0.0"
+      }
+    },
+    "node_modules/ganache/node_modules/@trufflesuite/bigint-buffer/node_modules/node-gyp-build": {
+      "version": "4.4.0",
+      "resolved": "https://registry.npmjs.org/node-gyp-build/-/node-gyp-build-4.4.0.tgz",
+      "integrity": "sha512-amJnQCcgtRVw9SvoebO3BKGESClrfXGCUTX9hSn1OuGQTQBOZmVd0Z0OlecpuRksKvbsUqALE8jls/ErClAPuQ==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT",
+      "bin": {
+        "node-gyp-build": "bin.js",
+        "node-gyp-build-optional": "optional.js",
+        "node-gyp-build-test": "build-test.js"
+      }
+    },
+    "node_modules/ganache/node_modules/@types/bn.js": {
+      "version": "5.1.0",
+      "resolved": "https://registry.npmjs.org/@types/bn.js/-/bn.js-5.1.0.tgz",
+      "integrity": "sha512-QSSVYj7pYFN49kW77o2s9xTCwZ8F2xLbjLLSEVh8D2F4JUhZtPAGOFLTD+ffqksBx/u4cE/KImFjyhqCjn/LIA==",
+      "dev": true,
+      "dependencies": {
+        "@types/node": "*"
+      }
+    },
+    "node_modules/ganache/node_modules/@types/lru-cache": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/@types/lru-cache/-/lru-cache-5.1.1.tgz",
+      "integrity": "sha512-ssE3Vlrys7sdIzs5LOxCzTVMsU7i9oa/IaW92wF32JFb3CVczqOkru2xspuKczHEbG3nvmPY7IFqVmGGHdNbYw==",
+      "dev": true
+    },
+    "node_modules/ganache/node_modules/@types/node": {
+      "version": "17.0.0",
+      "resolved": "https://registry.npmjs.org/@types/node/-/node-17.0.0.tgz",
+      "integrity": "sha512-eMhwJXc931Ihh4tkU+Y7GiLzT/y/DBNpNtr4yU9O2w3SYBsr9NaOPhQlLKRmoWtI54uNwuo0IOUFQjVOTZYRvw==",
+      "dev": true
+    },
+    "node_modules/ganache/node_modules/@types/seedrandom": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/@types/seedrandom/-/seedrandom-3.0.1.tgz",
+      "integrity": "sha512-giB9gzDeiCeloIXDgzFBCgjj1k4WxcDrZtGl6h1IqmUPlxF+Nx8Ve+96QCyDZ/HseB/uvDsKbpib9hU5cU53pw==",
+      "dev": true
+    },
+    "node_modules/ganache/node_modules/abstract-level": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/abstract-level/-/abstract-level-1.0.3.tgz",
+      "integrity": "sha512-t6jv+xHy+VYwc4xqZMn2Pa9DjcdzvzZmQGRjTFc8spIbRGHgBrEKbPq+rYXc7CCo0lxgYvSgKVg9qZAhpVQSjA==",
+      "dev": true,
+      "dependencies": {
+        "buffer": "^6.0.3",
+        "catering": "^2.1.0",
+        "is-buffer": "^2.0.5",
+        "level-supports": "^4.0.0",
+        "level-transcoder": "^1.0.1",
+        "module-error": "^1.0.1",
+        "queue-microtask": "^1.2.3"
+      }
+    },
+    "node_modules/ganache/node_modules/abstract-level/node_modules/level-supports": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/level-supports/-/level-supports-4.0.1.tgz",
+      "integrity": "sha512-PbXpve8rKeNcZ9C1mUicC9auIYFyGpkV9/i6g76tLgANwWhtG2v7I4xNBUlkn3lE2/dZF3Pi0ygYGtLc4RXXdA==",
+      "dev": true
+    },
+    "node_modules/ganache/node_modules/abstract-leveldown": {
+      "version": "7.2.0",
+      "resolved": "https://registry.npmjs.org/abstract-leveldown/-/abstract-leveldown-7.2.0.tgz",
+      "integrity": "sha512-DnhQwcFEaYsvYDnACLZhMmCWd3rkOeEvglpa4q5i/5Jlm3UIsWaxVzuXvDLFCSCWRO3yy2/+V/G7FusFgejnfQ==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT",
+      "dependencies": {
+        "buffer": "^6.0.3",
+        "catering": "^2.0.0",
+        "is-buffer": "^2.0.5",
+        "level-concat-iterator": "^3.0.0",
+        "level-supports": "^2.0.1",
+        "queue-microtask": "^1.2.3"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/ganache/node_modules/async": {
+      "version": "2.6.4",
+      "resolved": "https://registry.npmjs.org/async/-/async-2.6.4.tgz",
+      "integrity": "sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==",
+      "dev": true,
+      "dependencies": {
+        "lodash": "^4.17.14"
+      }
+    },
+    "node_modules/ganache/node_modules/async-eventemitter": {
+      "version": "0.2.4",
+      "resolved": "https://registry.npmjs.org/async-eventemitter/-/async-eventemitter-0.2.4.tgz",
+      "integrity": "sha512-pd20BwL7Yt1zwDFy+8MX8F1+WCT8aQeKj0kQnTrH9WaeRETlRamVhD0JtRPmrV4GfOJ2F9CvdQkZeZhnh2TuHw==",
+      "dev": true,
+      "dependencies": {
+        "async": "^2.4.0"
+      }
+    },
+    "node_modules/ganache/node_modules/base64-js": {
+      "version": "1.5.1",
+      "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz",
+      "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "inBundle": true,
+      "license": "MIT"
+    },
+    "node_modules/ganache/node_modules/brorand": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz",
+      "integrity": "sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT"
+    },
+    "node_modules/ganache/node_modules/buffer": {
+      "version": "6.0.3",
+      "resolved": "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz",
+      "integrity": "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "inBundle": true,
+      "license": "MIT",
+      "dependencies": {
+        "base64-js": "^1.3.1",
+        "ieee754": "^1.2.1"
+      }
+    },
+    "node_modules/ganache/node_modules/bufferutil": {
+      "version": "4.0.5",
+      "resolved": "https://registry.npmjs.org/bufferutil/-/bufferutil-4.0.5.tgz",
+      "integrity": "sha512-HTm14iMQKK2FjFLRTM5lAVcyaUzOnqbPtesFIvREgXpJHdQm8bWS+GkQgIkfaBYRHuCnea7w8UVNfwiAQhlr9A==",
+      "dev": true,
+      "hasInstallScript": true,
+      "optional": true,
+      "dependencies": {
+        "node-gyp-build": "^4.3.0"
+      }
+    },
+    "node_modules/ganache/node_modules/catering": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/catering/-/catering-2.1.0.tgz",
+      "integrity": "sha512-M5imwzQn6y+ODBfgi+cfgZv2hIUI6oYU/0f35Mdb1ujGeqeoI5tOnl9Q13DTH7LW+7er+NYq8stNOKZD/Z3U/A==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT",
+      "dependencies": {
+        "queue-tick": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/ganache/node_modules/elliptic": {
+      "version": "6.5.4",
+      "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.4.tgz",
+      "integrity": "sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT",
+      "dependencies": {
+        "bn.js": "^4.11.9",
+        "brorand": "^1.1.0",
+        "hash.js": "^1.0.0",
+        "hmac-drbg": "^1.0.1",
+        "inherits": "^2.0.4",
+        "minimalistic-assert": "^1.0.1",
+        "minimalistic-crypto-utils": "^1.0.1"
+      }
+    },
+    "node_modules/ganache/node_modules/elliptic/node_modules/bn.js": {
+      "version": "4.12.0",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+      "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT"
+    },
+    "node_modules/ganache/node_modules/emittery": {
+      "version": "0.10.0",
+      "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.10.0.tgz",
+      "integrity": "sha512-AGvFfs+d0JKCJQ4o01ASQLGPmSCxgfU9RFXvzPvZdjKK8oscynksuJhWrSTSw7j7Ep/sZct5b5ZhYCi8S/t0HQ==",
+      "dev": true
+    },
+    "node_modules/ganache/node_modules/hash.js": {
+      "version": "1.1.7",
+      "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz",
+      "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT",
+      "dependencies": {
+        "inherits": "^2.0.3",
+        "minimalistic-assert": "^1.0.1"
+      }
+    },
+    "node_modules/ganache/node_modules/hmac-drbg": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz",
+      "integrity": "sha1-0nRXAQJabHdabFRXk+1QL8DGSaE=",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT",
+      "dependencies": {
+        "hash.js": "^1.0.3",
+        "minimalistic-assert": "^1.0.0",
+        "minimalistic-crypto-utils": "^1.0.1"
+      }
+    },
+    "node_modules/ganache/node_modules/ieee754": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz",
+      "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "inBundle": true,
+      "license": "BSD-3-Clause"
+    },
+    "node_modules/ganache/node_modules/inherits": {
+      "version": "2.0.4",
+      "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
+      "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
+      "dev": true,
+      "inBundle": true,
+      "license": "ISC"
+    },
+    "node_modules/ganache/node_modules/is-buffer": {
+      "version": "2.0.5",
+      "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.5.tgz",
+      "integrity": "sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "inBundle": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/ganache/node_modules/keccak": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/keccak/-/keccak-3.0.2.tgz",
+      "integrity": "sha512-PyKKjkH53wDMLGrvmRGSNWgmSxZOUqbnXwKL9tmgbFYA1iAYqW21kfR7mZXV0MlESiefxQQE9X9fTa3X+2MPDQ==",
+      "dev": true,
+      "hasInstallScript": true,
+      "inBundle": true,
+      "license": "MIT",
+      "dependencies": {
+        "node-addon-api": "^2.0.0",
+        "node-gyp-build": "^4.2.0",
+        "readable-stream": "^3.6.0"
+      },
+      "engines": {
+        "node": ">=10.0.0"
+      }
+    },
+    "node_modules/ganache/node_modules/level-concat-iterator": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/level-concat-iterator/-/level-concat-iterator-3.1.0.tgz",
+      "integrity": "sha512-BWRCMHBxbIqPxJ8vHOvKUsaO0v1sLYZtjN3K2iZJsRBYtp+ONsY6Jfi6hy9K3+zolgQRryhIn2NRZjZnWJ9NmQ==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT",
+      "dependencies": {
+        "catering": "^2.1.0"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/ganache/node_modules/level-supports": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/level-supports/-/level-supports-2.1.0.tgz",
+      "integrity": "sha512-E486g1NCjW5cF78KGPrMDRBYzPuueMZ6VBXHT6gC7A8UYWGiM14fGgp+s/L1oFfDWSPV/+SFkYCmZ0SiESkRKA==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/ganache/node_modules/level-transcoder": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/level-transcoder/-/level-transcoder-1.0.1.tgz",
+      "integrity": "sha512-t7bFwFtsQeD8cl8NIoQ2iwxA0CL/9IFw7/9gAjOonH0PWTTiRfY7Hq+Ejbsxh86tXobDQ6IOiddjNYIfOBs06w==",
+      "dev": true,
+      "dependencies": {
+        "buffer": "^6.0.3",
+        "module-error": "^1.0.1"
+      }
+    },
+    "node_modules/ganache/node_modules/leveldown": {
+      "version": "6.1.0",
+      "resolved": "https://registry.npmjs.org/leveldown/-/leveldown-6.1.0.tgz",
+      "integrity": "sha512-8C7oJDT44JXxh04aSSsfcMI8YiaGRhOFI9/pMEL7nWJLVsWajDPTRxsSHTM2WcTVY5nXM+SuRHzPPi0GbnDX+w==",
+      "dev": true,
+      "hasInstallScript": true,
+      "inBundle": true,
+      "license": "MIT",
+      "dependencies": {
+        "abstract-leveldown": "^7.2.0",
+        "napi-macros": "~2.0.0",
+        "node-gyp-build": "^4.3.0"
+      },
+      "engines": {
+        "node": ">=10.12.0"
+      }
+    },
+    "node_modules/ganache/node_modules/lodash": {
+      "version": "4.17.21",
+      "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
+      "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==",
+      "dev": true
+    },
+    "node_modules/ganache/node_modules/minimalistic-assert": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz",
+      "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==",
+      "dev": true,
+      "inBundle": true,
+      "license": "ISC"
+    },
+    "node_modules/ganache/node_modules/minimalistic-crypto-utils": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz",
+      "integrity": "sha1-9sAMHAsIIkblxNmd+4x8CDsrWCo=",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT"
+    },
+    "node_modules/ganache/node_modules/module-error": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/module-error/-/module-error-1.0.2.tgz",
+      "integrity": "sha512-0yuvsqSCv8LbaOKhnsQ/T5JhyFlCYLPXK3U2sgV10zoKQwzs/MyfuQUOZQ1V/6OCOJsK/TRgNVrPuPDqtdMFtA==",
+      "dev": true
+    },
+    "node_modules/ganache/node_modules/napi-macros": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/napi-macros/-/napi-macros-2.0.0.tgz",
+      "integrity": "sha512-A0xLykHtARfueITVDernsAWdtIMbOJgKgcluwENp3AlsKN/PloyO10HtmoqnFAQAcxPkgZN7wdfPfEd0zNGxbg==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT"
+    },
+    "node_modules/ganache/node_modules/node-addon-api": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-2.0.2.tgz",
+      "integrity": "sha512-Ntyt4AIXyaLIuMHF6IOoTakB3K+RWxwtsHNRxllEoA6vPwP9o4866g6YWDLUdnucilZhmkxiHwHr11gAENw+QA==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT"
+    },
+    "node_modules/ganache/node_modules/node-gyp-build": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/node-gyp-build/-/node-gyp-build-4.3.0.tgz",
+      "integrity": "sha512-iWjXZvmboq0ja1pUGULQBexmxq8CV4xBhX7VDOTbL7ZR4FOowwY/VOtRxBN/yKxmdGoIp4j5ysNT4u3S2pDQ3Q==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT",
+      "bin": {
+        "node-gyp-build": "bin.js",
+        "node-gyp-build-optional": "optional.js",
+        "node-gyp-build-test": "build-test.js"
+      }
+    },
+    "node_modules/ganache/node_modules/queue-microtask": {
+      "version": "1.2.3",
+      "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz",
+      "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "inBundle": true,
+      "license": "MIT"
+    },
+    "node_modules/ganache/node_modules/queue-tick": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/queue-tick/-/queue-tick-1.0.0.tgz",
+      "integrity": "sha512-ULWhjjE8BmiICGn3G8+1L9wFpERNxkf8ysxkAer4+TFdRefDaXOCV5m92aMB9FtBVmn/8sETXLXY6BfW7hyaWQ==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT"
+    },
+    "node_modules/ganache/node_modules/readable-stream": {
+      "version": "3.6.0",
+      "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz",
+      "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT",
+      "dependencies": {
+        "inherits": "^2.0.3",
+        "string_decoder": "^1.1.1",
+        "util-deprecate": "^1.0.1"
+      },
+      "engines": {
+        "node": ">= 6"
+      }
+    },
+    "node_modules/ganache/node_modules/safe-buffer": {
+      "version": "5.2.1",
+      "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
+      "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "inBundle": true,
+      "license": "MIT"
+    },
+    "node_modules/ganache/node_modules/secp256k1": {
+      "version": "4.0.3",
+      "resolved": "https://registry.npmjs.org/secp256k1/-/secp256k1-4.0.3.tgz",
+      "integrity": "sha512-NLZVf+ROMxwtEj3Xa562qgv2BK5e2WNmXPiOdVIPLgs6lyTzMvBq0aWTYMI5XCP9jZMVKOcqZLw/Wc4vDkuxhA==",
+      "dev": true,
+      "hasInstallScript": true,
+      "inBundle": true,
+      "license": "MIT",
+      "dependencies": {
+        "elliptic": "^6.5.4",
+        "node-addon-api": "^2.0.0",
+        "node-gyp-build": "^4.2.0"
+      },
+      "engines": {
+        "node": ">=10.0.0"
+      }
+    },
+    "node_modules/ganache/node_modules/string_decoder": {
+      "version": "1.3.0",
+      "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz",
+      "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT",
+      "dependencies": {
+        "safe-buffer": "~5.2.0"
+      }
+    },
+    "node_modules/ganache/node_modules/utf-8-validate": {
+      "version": "5.0.7",
+      "resolved": "https://registry.npmjs.org/utf-8-validate/-/utf-8-validate-5.0.7.tgz",
+      "integrity": "sha512-vLt1O5Pp+flcArHGIyKEQq883nBt8nN8tVBcoL0qUXj2XT1n7p70yGIq2VK98I5FdZ1YHc0wk/koOnHjnXWk1Q==",
+      "dev": true,
+      "hasInstallScript": true,
+      "optional": true,
+      "dependencies": {
+        "node-gyp-build": "^4.3.0"
+      }
+    },
+    "node_modules/ganache/node_modules/util-deprecate": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
+      "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=",
+      "dev": true,
+      "inBundle": true,
+      "license": "MIT"
+    },
+    "node_modules/get-caller-file": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-1.0.3.tgz",
+      "integrity": "sha512-3t6rVToeoZfYSGd8YoLFR2DJkiQrIiUrGcjvFX2mDw3bn6k2OtwHN0TNCLbBO+w8qTvimhDkv+LSscbJY1vE6w==",
+      "dev": true
+    },
+    "node_modules/get-func-name": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/get-func-name/-/get-func-name-2.0.2.tgz",
+      "integrity": "sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/get-intrinsic": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.0.tgz",
+      "integrity": "sha512-L049y6nFOuom5wGyRc3/gdTLO94dySVKRACj1RmJZBQXlbTMhtNIgkWkUHq+jYmZvKf14EW1EoJnnjbmoHij0Q==",
+      "dev": true,
+      "dependencies": {
+        "function-bind": "^1.1.1",
+        "has": "^1.0.3",
+        "has-symbols": "^1.0.3"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/get-port": {
+      "version": "3.2.0",
+      "resolved": "https://registry.npmjs.org/get-port/-/get-port-3.2.0.tgz",
+      "integrity": "sha512-x5UJKlgeUiNT8nyo/AcnwLnZuZNcSjSw0kogRB+Whd1fjjFq4B1hySFxSFWWSn4mIBzg3sRNUDFYc4g5gjPoLg==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/get-stream": {
+      "version": "6.0.1",
+      "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz",
+      "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/get-symbol-description": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.0.tgz",
+      "integrity": "sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "get-intrinsic": "^1.1.1"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/getpass": {
+      "version": "0.1.7",
+      "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz",
+      "integrity": "sha512-0fzj9JxOLfJ+XGLhR8ze3unN0KZCgZwiSSDz168VERjK8Wl8kVSdcu2kspd4s4wtAa1y/qrVRiAA0WclVsu0ng==",
+      "dev": true,
+      "dependencies": {
+        "assert-plus": "^1.0.0"
+      }
+    },
+    "node_modules/ghost-testrpc": {
+      "version": "0.0.2",
+      "resolved": "https://registry.npmjs.org/ghost-testrpc/-/ghost-testrpc-0.0.2.tgz",
+      "integrity": "sha512-i08dAEgJ2g8z5buJIrCTduwPIhih3DP+hOCTyyryikfV8T0bNvHnGXO67i0DD1H4GBDETTclPy9njZbfluQYrQ==",
+      "dev": true,
+      "dependencies": {
+        "chalk": "^2.4.2",
+        "node-emoji": "^1.10.0"
+      },
+      "bin": {
+        "testrpc-sc": "index.js"
+      }
+    },
+    "node_modules/ghost-testrpc/node_modules/ansi-styles": {
+      "version": "3.2.1",
+      "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+      "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+      "dev": true,
+      "dependencies": {
+        "color-convert": "^1.9.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/ghost-testrpc/node_modules/chalk": {
+      "version": "2.4.2",
+      "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+      "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+      "dev": true,
+      "dependencies": {
+        "ansi-styles": "^3.2.1",
+        "escape-string-regexp": "^1.0.5",
+        "supports-color": "^5.3.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/ghost-testrpc/node_modules/color-convert": {
+      "version": "1.9.3",
+      "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+      "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+      "dev": true,
+      "dependencies": {
+        "color-name": "1.1.3"
+      }
+    },
+    "node_modules/ghost-testrpc/node_modules/color-name": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+      "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+      "dev": true
+    },
+    "node_modules/ghost-testrpc/node_modules/has-flag": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+      "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/ghost-testrpc/node_modules/supports-color": {
+      "version": "5.5.0",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+      "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+      "dev": true,
+      "dependencies": {
+        "has-flag": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/glob": {
+      "version": "7.2.3",
+      "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz",
+      "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==",
+      "dev": true,
+      "dependencies": {
+        "fs.realpath": "^1.0.0",
+        "inflight": "^1.0.4",
+        "inherits": "2",
+        "minimatch": "^3.1.1",
+        "once": "^1.3.0",
+        "path-is-absolute": "^1.0.0"
+      },
+      "engines": {
+        "node": "*"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/isaacs"
+      }
+    },
+    "node_modules/glob-parent": {
+      "version": "5.1.2",
+      "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
+      "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
+      "dev": true,
+      "dependencies": {
+        "is-glob": "^4.0.1"
+      },
+      "engines": {
+        "node": ">= 6"
+      }
+    },
+    "node_modules/global": {
+      "version": "4.4.0",
+      "resolved": "https://registry.npmjs.org/global/-/global-4.4.0.tgz",
+      "integrity": "sha512-wv/LAoHdRE3BeTGz53FAamhGlPLhlssK45usmGFThIi4XqnBmjKQ16u+RNbP7WvigRZDxUsM0J3gcQ5yicaL0w==",
+      "dev": true,
+      "dependencies": {
+        "min-document": "^2.19.0",
+        "process": "^0.11.10"
+      }
+    },
+    "node_modules/global-modules": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/global-modules/-/global-modules-2.0.0.tgz",
+      "integrity": "sha512-NGbfmJBp9x8IxyJSd1P+otYK8vonoJactOogrVfFRIAEY1ukil8RSKDz2Yo7wh1oihl51l/r6W4epkeKJHqL8A==",
+      "dev": true,
+      "dependencies": {
+        "global-prefix": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/global-prefix": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/global-prefix/-/global-prefix-3.0.0.tgz",
+      "integrity": "sha512-awConJSVCHVGND6x3tmMaKcQvwXLhjdkmomy2W+Goaui8YPgYgXJZewhg3fWC+DlfqqQuWg8AwqjGTD2nAPVWg==",
+      "dev": true,
+      "dependencies": {
+        "ini": "^1.3.5",
+        "kind-of": "^6.0.2",
+        "which": "^1.3.1"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/global-prefix/node_modules/which": {
+      "version": "1.3.1",
+      "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz",
+      "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==",
+      "dev": true,
+      "dependencies": {
+        "isexe": "^2.0.0"
+      },
+      "bin": {
+        "which": "bin/which"
+      }
+    },
+    "node_modules/globals": {
+      "version": "13.20.0",
+      "resolved": "https://registry.npmjs.org/globals/-/globals-13.20.0.tgz",
+      "integrity": "sha512-Qg5QtVkCy/kv3FUSlu4ukeZDVf9ee0iXLAUYX13gbR17bnejFTzr4iS9bY7kwCf1NztRNm1t91fjOiyx4CSwPQ==",
+      "dev": true,
+      "dependencies": {
+        "type-fest": "^0.20.2"
+      },
+      "engines": {
+        "node": ">=8"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/globals/node_modules/type-fest": {
+      "version": "0.20.2",
+      "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz",
+      "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/globalthis": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.3.tgz",
+      "integrity": "sha512-sFdI5LyBiNTHjRd7cGPWapiHWMOXKyuBNX/cWJ3NfzrZQVa8GI/8cofCl74AOVqq9W5kNmguTIzJ/1s2gyI9wA==",
+      "dev": true,
+      "dependencies": {
+        "define-properties": "^1.1.3"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/globby": {
+      "version": "11.1.0",
+      "resolved": "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz",
+      "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==",
+      "dev": true,
+      "dependencies": {
+        "array-union": "^2.1.0",
+        "dir-glob": "^3.0.1",
+        "fast-glob": "^3.2.9",
+        "ignore": "^5.2.0",
+        "merge2": "^1.4.1",
+        "slash": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/gopd": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz",
+      "integrity": "sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==",
+      "dev": true,
+      "dependencies": {
+        "get-intrinsic": "^1.1.3"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/got": {
+      "version": "12.1.0",
+      "resolved": "https://registry.npmjs.org/got/-/got-12.1.0.tgz",
+      "integrity": "sha512-hBv2ty9QN2RdbJJMK3hesmSkFTjVIHyIDDbssCKnSmq62edGgImJWD10Eb1k77TiV1bxloxqcFAVK8+9pkhOig==",
+      "dev": true,
+      "dependencies": {
+        "@sindresorhus/is": "^4.6.0",
+        "@szmarczak/http-timer": "^5.0.1",
+        "@types/cacheable-request": "^6.0.2",
+        "@types/responselike": "^1.0.0",
+        "cacheable-lookup": "^6.0.4",
+        "cacheable-request": "^7.0.2",
+        "decompress-response": "^6.0.0",
+        "form-data-encoder": "1.7.1",
+        "get-stream": "^6.0.1",
+        "http2-wrapper": "^2.1.10",
+        "lowercase-keys": "^3.0.0",
+        "p-cancelable": "^3.0.0",
+        "responselike": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=14.16"
+      },
+      "funding": {
+        "url": "https://github.com/sindresorhus/got?sponsor=1"
+      }
+    },
+    "node_modules/graceful-fs": {
+      "version": "4.2.10",
+      "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz",
+      "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==",
+      "dev": true
+    },
+    "node_modules/grapheme-splitter": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/grapheme-splitter/-/grapheme-splitter-1.0.4.tgz",
+      "integrity": "sha512-bzh50DW9kTPM00T8y4o8vQg89Di9oLJVLW/KaOGIXJWP/iqCN6WKYkbNOF04vFLJhwcpYUh9ydh/+5vpOqV4YQ==",
+      "dev": true
+    },
+    "node_modules/handlebars": {
+      "version": "4.7.7",
+      "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.7.tgz",
+      "integrity": "sha512-aAcXm5OAfE/8IXkcZvCepKU3VzW1/39Fb5ZuqMtgI/hT8X2YgoMvBY5dLhq/cpOvw7Lk1nK/UF71aLG/ZnVYRA==",
+      "dev": true,
+      "dependencies": {
+        "minimist": "^1.2.5",
+        "neo-async": "^2.6.0",
+        "source-map": "^0.6.1",
+        "wordwrap": "^1.0.0"
+      },
+      "bin": {
+        "handlebars": "bin/handlebars"
+      },
+      "engines": {
+        "node": ">=0.4.7"
+      },
+      "optionalDependencies": {
+        "uglify-js": "^3.1.4"
+      }
+    },
+    "node_modules/har-schema": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz",
+      "integrity": "sha512-Oqluz6zhGX8cyRaTQlFMPw80bSJVG2x/cFb8ZPhUILGgHka9SsokCCOQgpveePerqidZOrT14ipqfJb7ILcW5Q==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/har-validator": {
+      "version": "5.1.5",
+      "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.5.tgz",
+      "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==",
+      "deprecated": "this library is no longer supported",
+      "dev": true,
+      "dependencies": {
+        "ajv": "^6.12.3",
+        "har-schema": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/har-validator/node_modules/ajv": {
+      "version": "6.12.6",
+      "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+      "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+      "dev": true,
+      "dependencies": {
+        "fast-deep-equal": "^3.1.1",
+        "fast-json-stable-stringify": "^2.0.0",
+        "json-schema-traverse": "^0.4.1",
+        "uri-js": "^4.2.2"
+      },
+      "funding": {
+        "type": "github",
+        "url": "https://github.com/sponsors/epoberezkin"
+      }
+    },
+    "node_modules/har-validator/node_modules/json-schema-traverse": {
+      "version": "0.4.1",
+      "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+      "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+      "dev": true
+    },
+    "node_modules/hardhat": {
+      "version": "2.12.2",
+      "resolved": "https://registry.npmjs.org/hardhat/-/hardhat-2.12.2.tgz",
+      "integrity": "sha512-f3ZhzXy1uyQv0UXnAQ8GCBOWjzv++WJNb7bnm10SsyC3dB7vlPpsMWBNhq7aoRxKrNhX9tCev81KFV3i5BTeMQ==",
+      "dev": true,
+      "dependencies": {
+        "@ethersproject/abi": "^5.1.2",
+        "@metamask/eth-sig-util": "^4.0.0",
+        "@nomicfoundation/ethereumjs-block": "^4.0.0",
+        "@nomicfoundation/ethereumjs-blockchain": "^6.0.0",
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-evm": "^1.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-statemanager": "^1.0.0",
+        "@nomicfoundation/ethereumjs-trie": "^5.0.0",
+        "@nomicfoundation/ethereumjs-tx": "^4.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "@nomicfoundation/ethereumjs-vm": "^6.0.0",
+        "@nomicfoundation/solidity-analyzer": "^0.1.0",
+        "@sentry/node": "^5.18.1",
+        "@types/bn.js": "^5.1.0",
+        "@types/lru-cache": "^5.1.0",
+        "abort-controller": "^3.0.0",
+        "adm-zip": "^0.4.16",
+        "aggregate-error": "^3.0.0",
+        "ansi-escapes": "^4.3.0",
+        "chalk": "^2.4.2",
+        "chokidar": "^3.4.0",
+        "ci-info": "^2.0.0",
+        "debug": "^4.1.1",
+        "enquirer": "^2.3.0",
+        "env-paths": "^2.2.0",
+        "ethereum-cryptography": "^1.0.3",
+        "ethereumjs-abi": "^0.6.8",
+        "find-up": "^2.1.0",
+        "fp-ts": "1.19.3",
+        "fs-extra": "^7.0.1",
+        "glob": "7.2.0",
+        "immutable": "^4.0.0-rc.12",
+        "io-ts": "1.10.4",
+        "keccak": "^3.0.2",
+        "lodash": "^4.17.11",
+        "mnemonist": "^0.38.0",
+        "mocha": "^10.0.0",
+        "p-map": "^4.0.0",
+        "qs": "^6.7.0",
+        "raw-body": "^2.4.1",
+        "resolve": "1.17.0",
+        "semver": "^6.3.0",
+        "solc": "0.7.3",
+        "source-map-support": "^0.5.13",
+        "stacktrace-parser": "^0.1.10",
+        "tsort": "0.0.1",
+        "undici": "^5.4.0",
+        "uuid": "^8.3.2",
+        "ws": "^7.4.6"
+      },
+      "bin": {
+        "hardhat": "internal/cli/cli.js"
+      },
+      "engines": {
+        "node": "^14.0.0 || ^16.0.0 || ^18.0.0"
+      },
+      "peerDependencies": {
+        "ts-node": "*",
+        "typescript": "*"
+      },
+      "peerDependenciesMeta": {
+        "ts-node": {
+          "optional": true
+        },
+        "typescript": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/hardhat-deploy": {
+      "version": "0.11.25",
+      "resolved": "https://registry.npmjs.org/hardhat-deploy/-/hardhat-deploy-0.11.25.tgz",
+      "integrity": "sha512-ppSgrVE9A13YgTmf2PQGoyIs9o/jgJOMORrUP/rblU5K8mQ2YHWlPvkzZmP4h+SBW+tNmlnvSrf5K5DmMmExhw==",
+      "dev": true,
+      "dependencies": {
+        "@types/qs": "^6.9.7",
+        "axios": "^0.21.1",
+        "chalk": "^4.1.2",
+        "chokidar": "^3.5.2",
+        "debug": "^4.3.2",
+        "enquirer": "^2.3.6",
+        "ethers": "^5.5.3",
+        "form-data": "^4.0.0",
+        "fs-extra": "^10.0.0",
+        "match-all": "^1.2.6",
+        "murmur-128": "^0.2.1",
+        "qs": "^6.9.4",
+        "zksync-web3": "^0.8.1"
+      }
+    },
+    "node_modules/hardhat-deploy/node_modules/axios": {
+      "version": "0.21.4",
+      "resolved": "https://registry.npmjs.org/axios/-/axios-0.21.4.tgz",
+      "integrity": "sha512-ut5vewkiu8jjGBdqpM44XxjuCjq9LAKeHVmoVfHVzy8eHgxxq8SbAVQNovDA8mVi05kP0Ea/n/UzcSHcTJQfNg==",
+      "dev": true,
+      "dependencies": {
+        "follow-redirects": "^1.14.0"
+      }
+    },
+    "node_modules/hardhat-deploy/node_modules/form-data": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
+      "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
+      "dev": true,
+      "dependencies": {
+        "asynckit": "^0.4.0",
+        "combined-stream": "^1.0.8",
+        "mime-types": "^2.1.12"
+      },
+      "engines": {
+        "node": ">= 6"
+      }
+    },
+    "node_modules/hardhat-deploy/node_modules/fs-extra": {
+      "version": "10.1.0",
+      "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-10.1.0.tgz",
+      "integrity": "sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ==",
+      "dev": true,
+      "dependencies": {
+        "graceful-fs": "^4.2.0",
+        "jsonfile": "^6.0.1",
+        "universalify": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=12"
+      }
+    },
+    "node_modules/hardhat-deploy/node_modules/jsonfile": {
+      "version": "6.1.0",
+      "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz",
+      "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==",
+      "dev": true,
+      "dependencies": {
+        "universalify": "^2.0.0"
+      },
+      "optionalDependencies": {
+        "graceful-fs": "^4.1.6"
+      }
+    },
+    "node_modules/hardhat-deploy/node_modules/universalify": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz",
+      "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==",
+      "dev": true,
+      "engines": {
+        "node": ">= 10.0.0"
+      }
+    },
+    "node_modules/hardhat-gas-reporter": {
+      "version": "1.0.9",
+      "resolved": "https://registry.npmjs.org/hardhat-gas-reporter/-/hardhat-gas-reporter-1.0.9.tgz",
+      "integrity": "sha512-INN26G3EW43adGKBNzYWOlI3+rlLnasXTwW79YNnUhXPDa+yHESgt639dJEs37gCjhkbNKcRRJnomXEuMFBXJg==",
+      "dev": true,
+      "dependencies": {
+        "array-uniq": "1.0.3",
+        "eth-gas-reporter": "^0.2.25",
+        "sha1": "^1.1.1"
+      },
+      "peerDependencies": {
+        "hardhat": "^2.0.2"
+      }
+    },
+    "node_modules/hardhat/node_modules/@nomicfoundation/ethereumjs-rlp": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+      "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+      "dev": true,
+      "bin": {
+        "rlp": "bin/rlp"
+      },
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/hardhat/node_modules/ansi-styles": {
+      "version": "3.2.1",
+      "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+      "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+      "dev": true,
+      "dependencies": {
+        "color-convert": "^1.9.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/hardhat/node_modules/chalk": {
+      "version": "2.4.2",
+      "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+      "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+      "dev": true,
+      "dependencies": {
+        "ansi-styles": "^3.2.1",
+        "escape-string-regexp": "^1.0.5",
+        "supports-color": "^5.3.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/hardhat/node_modules/color-convert": {
+      "version": "1.9.3",
+      "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+      "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+      "dev": true,
+      "dependencies": {
+        "color-name": "1.1.3"
+      }
+    },
+    "node_modules/hardhat/node_modules/color-name": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+      "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+      "dev": true
+    },
+    "node_modules/hardhat/node_modules/ethereum-cryptography": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/ethereum-cryptography/-/ethereum-cryptography-1.2.0.tgz",
+      "integrity": "sha512-6yFQC9b5ug6/17CQpCyE3k9eKBMdhyVjzUy1WkiuY/E4vj/SXDBbCw8QEIaXqf0Mf2SnY6RmpDcwlUmBSS0EJw==",
+      "dev": true,
+      "dependencies": {
+        "@noble/hashes": "1.2.0",
+        "@noble/secp256k1": "1.7.1",
+        "@scure/bip32": "1.1.5",
+        "@scure/bip39": "1.1.1"
+      }
+    },
+    "node_modules/hardhat/node_modules/find-up": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz",
+      "integrity": "sha512-NWzkk0jSJtTt08+FBFMvXoeZnOJD+jTtsRmBYbAIzJdX6l7dLgR7CTubCM5/eDdPUBvLCeVasP1brfVR/9/EZQ==",
+      "dev": true,
+      "dependencies": {
+        "locate-path": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/hardhat/node_modules/glob": {
+      "version": "7.2.0",
+      "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz",
+      "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==",
+      "dev": true,
+      "dependencies": {
+        "fs.realpath": "^1.0.0",
+        "inflight": "^1.0.4",
+        "inherits": "2",
+        "minimatch": "^3.0.4",
+        "once": "^1.3.0",
+        "path-is-absolute": "^1.0.0"
+      },
+      "engines": {
+        "node": "*"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/isaacs"
+      }
+    },
+    "node_modules/hardhat/node_modules/has-flag": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+      "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/hardhat/node_modules/jsonfile": {
+      "version": "2.4.0",
+      "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-2.4.0.tgz",
+      "integrity": "sha512-PKllAqbgLgxHaj8TElYymKCAgrASebJrWpTnEkOaTowt23VKXXN0sUeriJ+eh7y6ufb/CC5ap11pz71/cM0hUw==",
+      "dev": true,
+      "optionalDependencies": {
+        "graceful-fs": "^4.1.6"
+      }
+    },
+    "node_modules/hardhat/node_modules/resolve": {
+      "version": "1.17.0",
+      "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.17.0.tgz",
+      "integrity": "sha512-ic+7JYiV8Vi2yzQGFWOkiZD5Z9z7O2Zhm9XMaTxdJExKasieFCr+yXZ/WmXsckHiKl12ar0y6XiXDx3m4RHn1w==",
+      "dev": true,
+      "dependencies": {
+        "path-parse": "^1.0.6"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/hardhat/node_modules/rimraf": {
+      "version": "2.7.1",
+      "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz",
+      "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==",
+      "dev": true,
+      "dependencies": {
+        "glob": "^7.1.3"
+      },
+      "bin": {
+        "rimraf": "bin.js"
+      }
+    },
+    "node_modules/hardhat/node_modules/solc": {
+      "version": "0.7.3",
+      "resolved": "https://registry.npmjs.org/solc/-/solc-0.7.3.tgz",
+      "integrity": "sha512-GAsWNAjGzIDg7VxzP6mPjdurby3IkGCjQcM8GFYZT6RyaoUZKmMU6Y7YwG+tFGhv7dwZ8rmR4iwFDrrD99JwqA==",
+      "dev": true,
+      "dependencies": {
+        "command-exists": "^1.2.8",
+        "commander": "3.0.2",
+        "follow-redirects": "^1.12.1",
+        "fs-extra": "^0.30.0",
+        "js-sha3": "0.8.0",
+        "memorystream": "^0.3.1",
+        "require-from-string": "^2.0.0",
+        "semver": "^5.5.0",
+        "tmp": "0.0.33"
+      },
+      "bin": {
+        "solcjs": "solcjs"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/hardhat/node_modules/solc/node_modules/fs-extra": {
+      "version": "0.30.0",
+      "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-0.30.0.tgz",
+      "integrity": "sha512-UvSPKyhMn6LEd/WpUaV9C9t3zATuqoqfWc3QdPhPLb58prN9tqYPlPWi8Krxi44loBoUzlobqZ3+8tGpxxSzwA==",
+      "dev": true,
+      "dependencies": {
+        "graceful-fs": "^4.1.2",
+        "jsonfile": "^2.1.0",
+        "klaw": "^1.0.0",
+        "path-is-absolute": "^1.0.0",
+        "rimraf": "^2.2.8"
+      }
+    },
+    "node_modules/hardhat/node_modules/solc/node_modules/semver": {
+      "version": "5.7.2",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz",
+      "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==",
+      "dev": true,
+      "bin": {
+        "semver": "bin/semver"
+      }
+    },
+    "node_modules/hardhat/node_modules/supports-color": {
+      "version": "5.5.0",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+      "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+      "dev": true,
+      "dependencies": {
+        "has-flag": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/hardhat/node_modules/uuid": {
+      "version": "8.3.2",
+      "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz",
+      "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==",
+      "dev": true,
+      "bin": {
+        "uuid": "dist/bin/uuid"
+      }
+    },
+    "node_modules/has": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz",
+      "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==",
+      "dev": true,
+      "dependencies": {
+        "function-bind": "^1.1.1"
+      },
+      "engines": {
+        "node": ">= 0.4.0"
+      }
+    },
+    "node_modules/has-bigints": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz",
+      "integrity": "sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==",
+      "dev": true,
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/has-flag": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
+      "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/has-property-descriptors": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.0.tgz",
+      "integrity": "sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ==",
+      "dev": true,
+      "dependencies": {
+        "get-intrinsic": "^1.1.1"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/has-proto": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.1.tgz",
+      "integrity": "sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/has-symbols": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz",
+      "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/has-tostringtag": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.0.tgz",
+      "integrity": "sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==",
+      "dev": true,
+      "dependencies": {
+        "has-symbols": "^1.0.2"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/hash-base": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/hash-base/-/hash-base-3.1.0.tgz",
+      "integrity": "sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA==",
+      "dev": true,
+      "dependencies": {
+        "inherits": "^2.0.4",
+        "readable-stream": "^3.6.0",
+        "safe-buffer": "^5.2.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/hash.js": {
+      "version": "1.1.7",
+      "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz",
+      "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==",
+      "devOptional": true,
+      "dependencies": {
+        "inherits": "^2.0.3",
+        "minimalistic-assert": "^1.0.1"
+      }
+    },
+    "node_modules/he": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz",
+      "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==",
+      "dev": true,
+      "bin": {
+        "he": "bin/he"
+      }
+    },
+    "node_modules/header-case": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/header-case/-/header-case-1.0.1.tgz",
+      "integrity": "sha512-i0q9mkOeSuhXw6bGgiQCCBgY/jlZuV/7dZXyZ9c6LcBrqwvT8eT719E9uxE5LiZftdl+z81Ugbg/VvXV4OJOeQ==",
+      "dev": true,
+      "dependencies": {
+        "no-case": "^2.2.0",
+        "upper-case": "^1.1.3"
+      }
+    },
+    "node_modules/heap": {
+      "version": "0.2.7",
+      "resolved": "https://registry.npmjs.org/heap/-/heap-0.2.7.tgz",
+      "integrity": "sha512-2bsegYkkHO+h/9MGbn6KWcE45cHZgPANo5LXF7EvWdT0yT2EguSVO1nDgU5c8+ZOPwp2vMNa7YFsJhVcDR9Sdg==",
+      "dev": true
+    },
+    "node_modules/highlight.js": {
+      "version": "10.7.3",
+      "resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-10.7.3.tgz",
+      "integrity": "sha512-tzcUFauisWKNHaRkN4Wjl/ZA07gENAjFl3J/c480dprkGTg5EQstgaNFqBfUqCq54kZRIEcreTsAgF/m2quD7A==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/highlightjs-solidity": {
+      "version": "2.0.6",
+      "resolved": "https://registry.npmjs.org/highlightjs-solidity/-/highlightjs-solidity-2.0.6.tgz",
+      "integrity": "sha512-DySXWfQghjm2l6a/flF+cteroJqD4gI8GSdL4PtvxZSsAHie8m3yVe2JFoRg03ROKT6hp2Lc/BxXkqerNmtQYg==",
+      "dev": true
+    },
+    "node_modules/hmac-drbg": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz",
+      "integrity": "sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==",
+      "devOptional": true,
+      "dependencies": {
+        "hash.js": "^1.0.3",
+        "minimalistic-assert": "^1.0.0",
+        "minimalistic-crypto-utils": "^1.0.1"
+      }
+    },
+    "node_modules/hosted-git-info": {
+      "version": "2.8.9",
+      "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz",
+      "integrity": "sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==",
+      "dev": true
+    },
+    "node_modules/htmlparser2": {
+      "version": "8.0.1",
+      "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-8.0.1.tgz",
+      "integrity": "sha512-4lVbmc1diZC7GUJQtRQ5yBAeUCL1exyMwmForWkRLnwyzWBFxN633SALPMGYaWZvKe9j1pRZJpauvmxENSp/EA==",
+      "dev": true,
+      "funding": [
+        "https://github.com/fb55/htmlparser2?sponsor=1",
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/fb55"
+        }
+      ],
+      "dependencies": {
+        "domelementtype": "^2.3.0",
+        "domhandler": "^5.0.2",
+        "domutils": "^3.0.1",
+        "entities": "^4.3.0"
+      }
+    },
+    "node_modules/http-basic": {
+      "version": "8.1.3",
+      "resolved": "https://registry.npmjs.org/http-basic/-/http-basic-8.1.3.tgz",
+      "integrity": "sha512-/EcDMwJZh3mABI2NhGfHOGOeOZITqfkEO4p/xK+l3NpyncIHUQBoMvCSF/b5GqvKtySC2srL/GGG3+EtlqlmCw==",
+      "dev": true,
+      "dependencies": {
+        "caseless": "^0.12.0",
+        "concat-stream": "^1.6.2",
+        "http-response-object": "^3.0.1",
+        "parse-cache-control": "^1.0.1"
+      },
+      "engines": {
+        "node": ">=6.0.0"
+      }
+    },
+    "node_modules/http-cache-semantics": {
+      "version": "4.1.1",
+      "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.1.tgz",
+      "integrity": "sha512-er295DKPVsV82j5kw1Gjt+ADA/XYHsajl82cGNQG2eyoPkvgUhX+nDIyelzhIWbbsXP39EHcI6l5tYs2FYqYXQ==",
+      "dev": true
+    },
+    "node_modules/http-errors": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz",
+      "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==",
+      "dev": true,
+      "dependencies": {
+        "depd": "2.0.0",
+        "inherits": "2.0.4",
+        "setprototypeof": "1.2.0",
+        "statuses": "2.0.1",
+        "toidentifier": "1.0.1"
+      },
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/http-https": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/http-https/-/http-https-1.0.0.tgz",
+      "integrity": "sha512-o0PWwVCSp3O0wS6FvNr6xfBCHgt0m1tvPLFOCc2iFDKTRAXhB7m8klDf7ErowFH8POa6dVdGatKU5I1YYwzUyg==",
+      "dev": true
+    },
+    "node_modules/http-response-object": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/http-response-object/-/http-response-object-3.0.2.tgz",
+      "integrity": "sha512-bqX0XTF6fnXSQcEJ2Iuyr75yVakyjIDCqroJQ/aHfSdlM743Cwqoi2nDYMzLGWUcuTWGWy8AAvOKXTfiv6q9RA==",
+      "dev": true,
+      "dependencies": {
+        "@types/node": "^10.0.3"
+      }
+    },
+    "node_modules/http-response-object/node_modules/@types/node": {
+      "version": "10.17.60",
+      "resolved": "https://registry.npmjs.org/@types/node/-/node-10.17.60.tgz",
+      "integrity": "sha512-F0KIgDJfy2nA3zMLmWGKxcH2ZVEtCZXHHdOQs2gSaQ27+lNeEfGxzkIw90aXswATX7AZ33tahPbzy6KAfUreVw==",
+      "dev": true
+    },
+    "node_modules/http-signature": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz",
+      "integrity": "sha512-CAbnr6Rz4CYQkLYUtSNXxQPUH2gK8f3iWexVlsnMeD+GjlsQ0Xsy1cOX+mN3dtxYomRy21CiOzU8Uhw6OwncEQ==",
+      "dev": true,
+      "dependencies": {
+        "assert-plus": "^1.0.0",
+        "jsprim": "^1.2.2",
+        "sshpk": "^1.7.0"
+      },
+      "engines": {
+        "node": ">=0.8",
+        "npm": ">=1.3.7"
+      }
+    },
+    "node_modules/http2-wrapper": {
+      "version": "2.2.0",
+      "resolved": "https://registry.npmjs.org/http2-wrapper/-/http2-wrapper-2.2.0.tgz",
+      "integrity": "sha512-kZB0wxMo0sh1PehyjJUWRFEd99KC5TLjZ2cULC4f9iqJBAmKQQXEICjxl5iPJRwP40dpeHFqqhm7tYCvODpqpQ==",
+      "dev": true,
+      "dependencies": {
+        "quick-lru": "^5.1.1",
+        "resolve-alpn": "^1.2.0"
+      },
+      "engines": {
+        "node": ">=10.19.0"
+      }
+    },
+    "node_modules/https-proxy-agent": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz",
+      "integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==",
+      "dev": true,
+      "dependencies": {
+        "agent-base": "6",
+        "debug": "4"
+      },
+      "engines": {
+        "node": ">= 6"
+      }
+    },
+    "node_modules/husky": {
+      "version": "4.3.8",
+      "resolved": "https://registry.npmjs.org/husky/-/husky-4.3.8.tgz",
+      "integrity": "sha512-LCqqsB0PzJQ/AlCgfrfzRe3e3+NvmefAdKQhRYpxS4u6clblBoDdzzvHi8fmxKRzvMxPY/1WZWzomPZww0Anow==",
+      "dev": true,
+      "hasInstallScript": true,
+      "dependencies": {
+        "chalk": "^4.0.0",
+        "ci-info": "^2.0.0",
+        "compare-versions": "^3.6.0",
+        "cosmiconfig": "^7.0.0",
+        "find-versions": "^4.0.0",
+        "opencollective-postinstall": "^2.0.2",
+        "pkg-dir": "^5.0.0",
+        "please-upgrade-node": "^3.2.0",
+        "slash": "^3.0.0",
+        "which-pm-runs": "^1.0.0"
+      },
+      "bin": {
+        "husky-run": "bin/run.js",
+        "husky-upgrade": "lib/upgrader/bin.js"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/husky"
+      }
+    },
+    "node_modules/hyperlinker": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/hyperlinker/-/hyperlinker-1.0.0.tgz",
+      "integrity": "sha512-Ty8UblRWFEcfSuIaajM34LdPXIhbs1ajEX/BBPv24J+enSVaEVY63xQ6lTO9VRYS5LAoghIG0IDJ+p+IPzKUQQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/iconv-lite": {
+      "version": "0.4.24",
+      "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
+      "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
+      "dev": true,
+      "dependencies": {
+        "safer-buffer": ">= 2.1.2 < 3"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/idna-uts46-hx": {
+      "version": "2.3.1",
+      "resolved": "https://registry.npmjs.org/idna-uts46-hx/-/idna-uts46-hx-2.3.1.tgz",
+      "integrity": "sha512-PWoF9Keq6laYdIRwwCdhTPl60xRqAloYNMQLiyUnG42VjT53oW07BXIRM+NK7eQjzXjAk2gUvX9caRxlnF9TAA==",
+      "dev": true,
+      "dependencies": {
+        "punycode": "2.1.0"
+      },
+      "engines": {
+        "node": ">=4.0.0"
+      }
+    },
+    "node_modules/idna-uts46-hx/node_modules/punycode": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.0.tgz",
+      "integrity": "sha512-Yxz2kRwT90aPiWEMHVYnEf4+rhwF1tBmmZ4KepCP+Wkium9JxtWnUm1nqGwpiAHr/tnTSeHqr3wb++jgSkXjhA==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/ieee754": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz",
+      "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ]
+    },
+    "node_modules/ignore": {
+      "version": "5.2.4",
+      "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz",
+      "integrity": "sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==",
+      "dev": true,
+      "engines": {
+        "node": ">= 4"
+      }
+    },
+    "node_modules/immutable": {
+      "version": "4.2.4",
+      "resolved": "https://registry.npmjs.org/immutable/-/immutable-4.2.4.tgz",
+      "integrity": "sha512-WDxL3Hheb1JkRN3sQkyujNlL/xRjAo3rJtaU5xeufUauG66JdMr32bLj4gF+vWl84DIA3Zxw7tiAjneYzRRw+w==",
+      "dev": true
+    },
+    "node_modules/import-fresh": {
+      "version": "3.3.0",
+      "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz",
+      "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==",
+      "dev": true,
+      "dependencies": {
+        "parent-module": "^1.0.0",
+        "resolve-from": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=6"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/imul": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/imul/-/imul-1.0.1.tgz",
+      "integrity": "sha512-WFAgfwPLAjU66EKt6vRdTlKj4nAgIDQzh29JonLa4Bqtl6D8JrIMvWjCnx7xEjVNmP3U0fM5o8ZObk7d0f62bA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/imurmurhash": {
+      "version": "0.1.4",
+      "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
+      "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.8.19"
+      }
+    },
+    "node_modules/indent-string": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz",
+      "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/inflight": {
+      "version": "1.0.6",
+      "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
+      "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==",
+      "dev": true,
+      "dependencies": {
+        "once": "^1.3.0",
+        "wrappy": "1"
+      }
+    },
+    "node_modules/inherits": {
+      "version": "2.0.4",
+      "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
+      "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
+      "devOptional": true
+    },
+    "node_modules/ini": {
+      "version": "1.3.8",
+      "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz",
+      "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==",
+      "dev": true
+    },
+    "node_modules/internal-slot": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.5.tgz",
+      "integrity": "sha512-Y+R5hJrzs52QCG2laLn4udYVnxsfny9CpOhNhUvk/SSSVyF6T27FzRbF0sroPidSu3X8oEAkOn2K804mjpt6UQ==",
+      "dev": true,
+      "dependencies": {
+        "get-intrinsic": "^1.2.0",
+        "has": "^1.0.3",
+        "side-channel": "^1.0.4"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      }
+    },
+    "node_modules/interpret": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/interpret/-/interpret-1.4.0.tgz",
+      "integrity": "sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.10"
+      }
+    },
+    "node_modules/invert-kv": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/invert-kv/-/invert-kv-1.0.0.tgz",
+      "integrity": "sha512-xgs2NH9AE66ucSq4cNG1nhSFghr5l6tdL15Pk+jl46bmmBapgoaY/AacXyaDznAqmGL99TiLSQgO/XazFSKYeQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/io-ts": {
+      "version": "1.10.4",
+      "resolved": "https://registry.npmjs.org/io-ts/-/io-ts-1.10.4.tgz",
+      "integrity": "sha512-b23PteSnYXSONJ6JQXRAlvJhuw8KOtkqa87W4wDtvMrud/DTJd5X+NpOOI+O/zZwVq6v0VLAaJ+1EDViKEuN9g==",
+      "dev": true,
+      "dependencies": {
+        "fp-ts": "^1.0.0"
+      }
+    },
+    "node_modules/ipaddr.js": {
+      "version": "1.9.1",
+      "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
+      "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.10"
+      }
+    },
+    "node_modules/is-arguments": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.1.1.tgz",
+      "integrity": "sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "has-tostringtag": "^1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-array-buffer": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.2.tgz",
+      "integrity": "sha512-y+FyyR/w8vfIRq4eQcM1EYgSTnmHXPqaF+IgzgraytCFq5Xh8lllDVmAZolPJiZttZLeFSINPYMaEJ7/vWUa1w==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "get-intrinsic": "^1.2.0",
+        "is-typed-array": "^1.1.10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-arrayish": {
+      "version": "0.2.1",
+      "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz",
+      "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==",
+      "dev": true
+    },
+    "node_modules/is-bigint": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz",
+      "integrity": "sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==",
+      "dev": true,
+      "dependencies": {
+        "has-bigints": "^1.0.1"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-binary-path": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
+      "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
+      "dev": true,
+      "dependencies": {
+        "binary-extensions": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/is-boolean-object": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.1.2.tgz",
+      "integrity": "sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "has-tostringtag": "^1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-buffer": {
+      "version": "2.0.5",
+      "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.5.tgz",
+      "integrity": "sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/is-callable": {
+      "version": "1.2.7",
+      "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz",
+      "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-core-module": {
+      "version": "2.11.0",
+      "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.11.0.tgz",
+      "integrity": "sha512-RRjxlvLDkD1YJwDbroBHMb+cukurkDWNyHx7D3oNB5x9rb5ogcksMC5wHCadcXoo67gVr/+3GFySh3134zi6rw==",
+      "dev": true,
+      "dependencies": {
+        "has": "^1.0.3"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-date-object": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.5.tgz",
+      "integrity": "sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==",
+      "dev": true,
+      "dependencies": {
+        "has-tostringtag": "^1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-docker": {
+      "version": "2.2.1",
+      "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.2.1.tgz",
+      "integrity": "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==",
+      "dev": true,
+      "bin": {
+        "is-docker": "cli.js"
+      },
+      "engines": {
+        "node": ">=8"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/is-extglob": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
+      "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/is-fullwidth-code-point": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
+      "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/is-function": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/is-function/-/is-function-1.0.2.tgz",
+      "integrity": "sha512-lw7DUp0aWXYg+CBCN+JKkcE0Q2RayZnSvnZBlwgxHBQhqt5pZNVy4Ri7H9GmmXkdu7LUthszM+Tor1u/2iBcpQ==",
+      "dev": true
+    },
+    "node_modules/is-generator-function": {
+      "version": "1.0.10",
+      "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.0.10.tgz",
+      "integrity": "sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==",
+      "dev": true,
+      "dependencies": {
+        "has-tostringtag": "^1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-glob": {
+      "version": "4.0.3",
+      "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz",
+      "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==",
+      "dev": true,
+      "dependencies": {
+        "is-extglob": "^2.1.1"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/is-hex-prefixed": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-hex-prefixed/-/is-hex-prefixed-1.0.0.tgz",
+      "integrity": "sha512-WvtOiug1VFrE9v1Cydwm+FnXd3+w9GaeVUss5W4v/SLy3UW00vP+6iNF2SdnfiBoLy4bTqVdkftNGTUeOFVsbA==",
+      "dev": true,
+      "engines": {
+        "node": ">=6.5.0",
+        "npm": ">=3"
+      }
+    },
+    "node_modules/is-interactive": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-interactive/-/is-interactive-1.0.0.tgz",
+      "integrity": "sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/is-lower-case": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/is-lower-case/-/is-lower-case-1.1.3.tgz",
+      "integrity": "sha512-+5A1e/WJpLLXZEDlgz4G//WYSHyQBD32qa4Jd3Lw06qQlv3fJHnp3YIHjTQSGzHMgzmVKz2ZP3rBxTHkPw/lxA==",
+      "dev": true,
+      "dependencies": {
+        "lower-case": "^1.1.0"
+      }
+    },
+    "node_modules/is-negative-zero": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz",
+      "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-number": {
+      "version": "7.0.0",
+      "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
+      "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.12.0"
+      }
+    },
+    "node_modules/is-number-object": {
+      "version": "1.0.7",
+      "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.0.7.tgz",
+      "integrity": "sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==",
+      "dev": true,
+      "dependencies": {
+        "has-tostringtag": "^1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-path-inside": {
+      "version": "3.0.3",
+      "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz",
+      "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/is-plain-obj": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-2.1.0.tgz",
+      "integrity": "sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/is-regex": {
+      "version": "1.1.4",
+      "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz",
+      "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "has-tostringtag": "^1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-shared-array-buffer": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz",
+      "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-string": {
+      "version": "1.0.7",
+      "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz",
+      "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==",
+      "dev": true,
+      "dependencies": {
+        "has-tostringtag": "^1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-symbol": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz",
+      "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==",
+      "dev": true,
+      "dependencies": {
+        "has-symbols": "^1.0.2"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-typed-array": {
+      "version": "1.1.10",
+      "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.10.tgz",
+      "integrity": "sha512-PJqgEHiWZvMpaFZ3uTc8kHPM4+4ADTlDniuQL7cU/UDA0Ql7F70yGfHph3cLNe+c9toaigv+DFzTJKhc2CtO6A==",
+      "dev": true,
+      "dependencies": {
+        "available-typed-arrays": "^1.0.5",
+        "call-bind": "^1.0.2",
+        "for-each": "^0.3.3",
+        "gopd": "^1.0.1",
+        "has-tostringtag": "^1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-typedarray": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz",
+      "integrity": "sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA==",
+      "dev": true
+    },
+    "node_modules/is-unicode-supported": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz",
+      "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/is-upper-case": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/is-upper-case/-/is-upper-case-1.1.2.tgz",
+      "integrity": "sha512-GQYSJMgfeAmVwh9ixyk888l7OIhNAGKtY6QA+IrWlu9MDTCaXmeozOZ2S9Knj7bQwBO/H6J2kb+pbyTUiMNbsw==",
+      "dev": true,
+      "dependencies": {
+        "upper-case": "^1.1.0"
+      }
+    },
+    "node_modules/is-utf8": {
+      "version": "0.2.1",
+      "resolved": "https://registry.npmjs.org/is-utf8/-/is-utf8-0.2.1.tgz",
+      "integrity": "sha512-rMYPYvCzsXywIsldgLaSoPlw5PfoB/ssr7hY4pLfcodrA5M/eArza1a9VmTiNIBNMjOGr1Ow9mTyU2o69U6U9Q==",
+      "dev": true
+    },
+    "node_modules/is-weakref": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz",
+      "integrity": "sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/is-wsl": {
+      "version": "2.2.0",
+      "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-2.2.0.tgz",
+      "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==",
+      "dev": true,
+      "dependencies": {
+        "is-docker": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/isarray": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
+      "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==",
+      "dev": true
+    },
+    "node_modules/isexe": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
+      "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==",
+      "dev": true
+    },
+    "node_modules/isstream": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz",
+      "integrity": "sha512-Yljz7ffyPbrLpLngrMtZ7NduUgVvi6wG9RJ9IUcyCd59YQ911PBJphODUcbOVbqYfxe1wuYf/LJ8PauMRwsM/g==",
+      "dev": true
+    },
+    "node_modules/js-base64": {
+      "version": "3.7.5",
+      "resolved": "https://registry.npmjs.org/js-base64/-/js-base64-3.7.5.tgz",
+      "integrity": "sha512-3MEt5DTINKqfScXKfJFrRbxkrnk2AxPWGBL/ycjz4dK8iqiSJ06UxD8jh8xuh6p10TX4t2+7FsBYVxxQbMg+qA==",
+      "dev": true
+    },
+    "node_modules/js-sdsl": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/js-sdsl/-/js-sdsl-4.3.0.tgz",
+      "integrity": "sha512-mifzlm2+5nZ+lEcLJMoBK0/IH/bDg8XnJfd/Wq6IP+xoCjLZsTOnV2QpxlVbX9bMnkl5PdEjNtBJ9Cj1NjifhQ==",
+      "dev": true,
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/js-sdsl"
+      }
+    },
+    "node_modules/js-sha3": {
+      "version": "0.8.0",
+      "resolved": "https://registry.npmjs.org/js-sha3/-/js-sha3-0.8.0.tgz",
+      "integrity": "sha512-gF1cRrHhIzNfToc802P800N8PpXS+evLLXfsVpowqmAFR9uwbi89WvXg2QspOmXL8QL86J4T1EpFu+yUkwJY3Q==",
+      "devOptional": true
+    },
+    "node_modules/js-tokens": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
+      "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==",
+      "dev": true
+    },
+    "node_modules/js-yaml": {
+      "version": "4.1.0",
+      "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz",
+      "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==",
+      "dev": true,
+      "dependencies": {
+        "argparse": "^2.0.1"
+      },
+      "bin": {
+        "js-yaml": "bin/js-yaml.js"
+      }
+    },
+    "node_modules/jsbn": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz",
+      "integrity": "sha512-UVU9dibq2JcFWxQPA6KCqj5O42VOmAY3zQUfEKxU0KpTGXwNoCjkX1e13eHNvw/xPynt6pU0rZ1htjWTNTSXsg==",
+      "dev": true
+    },
+    "node_modules/json-buffer": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz",
+      "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==",
+      "dev": true
+    },
+    "node_modules/json-parse-even-better-errors": {
+      "version": "2.3.1",
+      "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz",
+      "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==",
+      "dev": true
+    },
+    "node_modules/json-schema": {
+      "version": "0.4.0",
+      "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.4.0.tgz",
+      "integrity": "sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA==",
+      "dev": true
+    },
+    "node_modules/json-schema-traverse": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz",
+      "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==",
+      "dev": true
+    },
+    "node_modules/json-stable-stringify-without-jsonify": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz",
+      "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==",
+      "dev": true
+    },
+    "node_modules/json-stringify-safe": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz",
+      "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==",
+      "dev": true
+    },
+    "node_modules/json5": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz",
+      "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==",
+      "dev": true,
+      "dependencies": {
+        "minimist": "^1.2.0"
+      },
+      "bin": {
+        "json5": "lib/cli.js"
+      }
+    },
+    "node_modules/jsonfile": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz",
+      "integrity": "sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg==",
+      "dev": true,
+      "optionalDependencies": {
+        "graceful-fs": "^4.1.6"
+      }
+    },
+    "node_modules/jsonschema": {
+      "version": "1.4.1",
+      "resolved": "https://registry.npmjs.org/jsonschema/-/jsonschema-1.4.1.tgz",
+      "integrity": "sha512-S6cATIPVv1z0IlxdN+zUk5EPjkGCdnhN4wVSBlvoUO1tOLJootbo9CquNJmbIh4yikWHiUedhRYrNPn1arpEmQ==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/jsprim": {
+      "version": "1.4.2",
+      "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.2.tgz",
+      "integrity": "sha512-P2bSOMAc/ciLz6DzgjVlGJP9+BrJWu5UDGK70C2iweC5QBIeFf0ZXRvGjEj2uYgrY2MkAAhsSWHDWlFtEroZWw==",
+      "dev": true,
+      "dependencies": {
+        "assert-plus": "1.0.0",
+        "extsprintf": "1.3.0",
+        "json-schema": "0.4.0",
+        "verror": "1.10.0"
+      },
+      "engines": {
+        "node": ">=0.6.0"
+      }
+    },
+    "node_modules/keccak": {
+      "version": "3.0.3",
+      "resolved": "https://registry.npmjs.org/keccak/-/keccak-3.0.3.tgz",
+      "integrity": "sha512-JZrLIAJWuZxKbCilMpNz5Vj7Vtb4scDG3dMXLOsbzBmQGyjwE61BbW7bJkfKKCShXiQZt3T6sBgALRtmd+nZaQ==",
+      "dev": true,
+      "hasInstallScript": true,
+      "dependencies": {
+        "node-addon-api": "^2.0.0",
+        "node-gyp-build": "^4.2.0",
+        "readable-stream": "^3.6.0"
+      },
+      "engines": {
+        "node": ">=10.0.0"
+      }
+    },
+    "node_modules/keyv": {
+      "version": "4.5.2",
+      "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.2.tgz",
+      "integrity": "sha512-5MHbFaKn8cNSmVW7BYnijeAVlE4cYA/SVkifVgrh7yotnfhKmjuXpDKjrABLnT0SfHWV21P8ow07OGfRrNDg8g==",
+      "dev": true,
+      "dependencies": {
+        "json-buffer": "3.0.1"
+      }
+    },
+    "node_modules/kind-of": {
+      "version": "6.0.3",
+      "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz",
+      "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/klaw": {
+      "version": "1.3.1",
+      "resolved": "https://registry.npmjs.org/klaw/-/klaw-1.3.1.tgz",
+      "integrity": "sha512-TED5xi9gGQjGpNnvRWknrwAB1eL5GciPfVFOt3Vk1OJCVDQbzuSfrF3hkUQKlsgKrG1F+0t5W0m+Fje1jIt8rw==",
+      "dev": true,
+      "optionalDependencies": {
+        "graceful-fs": "^4.1.9"
+      }
+    },
+    "node_modules/kleur": {
+      "version": "3.0.3",
+      "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz",
+      "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/lcid": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/lcid/-/lcid-1.0.0.tgz",
+      "integrity": "sha512-YiGkH6EnGrDGqLMITnGjXtGmNtjoXw9SVUzcaos8RBi7Ps0VBylkq+vOcY9QE5poLasPCR849ucFUkl0UzUyOw==",
+      "dev": true,
+      "dependencies": {
+        "invert-kv": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/level": {
+      "version": "8.0.0",
+      "resolved": "https://registry.npmjs.org/level/-/level-8.0.0.tgz",
+      "integrity": "sha512-ypf0jjAk2BWI33yzEaaotpq7fkOPALKAgDBxggO6Q9HGX2MRXn0wbP1Jn/tJv1gtL867+YOjOB49WaUF3UoJNQ==",
+      "dev": true,
+      "dependencies": {
+        "browser-level": "^1.0.1",
+        "classic-level": "^1.2.0"
+      },
+      "engines": {
+        "node": ">=12"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/level"
+      }
+    },
+    "node_modules/level-supports": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/level-supports/-/level-supports-4.0.1.tgz",
+      "integrity": "sha512-PbXpve8rKeNcZ9C1mUicC9auIYFyGpkV9/i6g76tLgANwWhtG2v7I4xNBUlkn3lE2/dZF3Pi0ygYGtLc4RXXdA==",
+      "dev": true,
+      "engines": {
+        "node": ">=12"
+      }
+    },
+    "node_modules/level-transcoder": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/level-transcoder/-/level-transcoder-1.0.1.tgz",
+      "integrity": "sha512-t7bFwFtsQeD8cl8NIoQ2iwxA0CL/9IFw7/9gAjOonH0PWTTiRfY7Hq+Ejbsxh86tXobDQ6IOiddjNYIfOBs06w==",
+      "dev": true,
+      "dependencies": {
+        "buffer": "^6.0.3",
+        "module-error": "^1.0.1"
+      },
+      "engines": {
+        "node": ">=12"
+      }
+    },
+    "node_modules/level-transcoder/node_modules/buffer": {
+      "version": "6.0.3",
+      "resolved": "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz",
+      "integrity": "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "dependencies": {
+        "base64-js": "^1.3.1",
+        "ieee754": "^1.2.1"
+      }
+    },
+    "node_modules/levn": {
+      "version": "0.4.1",
+      "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz",
+      "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==",
+      "dev": true,
+      "dependencies": {
+        "prelude-ls": "^1.2.1",
+        "type-check": "~0.4.0"
+      },
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/lines-and-columns": {
+      "version": "1.2.4",
+      "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz",
+      "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==",
+      "dev": true
+    },
+    "node_modules/load-json-file": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-1.1.0.tgz",
+      "integrity": "sha512-cy7ZdNRXdablkXYNI049pthVeXFurRyb9+hA/dZzerZ0pGTx42z+y+ssxBaVV2l70t1muq5IdKhn4UtcoGUY9A==",
+      "dev": true,
+      "dependencies": {
+        "graceful-fs": "^4.1.2",
+        "parse-json": "^2.2.0",
+        "pify": "^2.0.0",
+        "pinkie-promise": "^2.0.0",
+        "strip-bom": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/locate-path": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz",
+      "integrity": "sha512-NCI2kiDkyR7VeEKm27Kda/iQHyKJe1Bu0FlTbYp3CqJu+9IFe9bLyAjMxf5ZDDbEg+iMPzB5zYyUTSm8wVTKmA==",
+      "dev": true,
+      "dependencies": {
+        "p-locate": "^2.0.0",
+        "path-exists": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/locate-path/node_modules/path-exists": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz",
+      "integrity": "sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/lodash": {
+      "version": "4.17.21",
+      "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
+      "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==",
+      "dev": true
+    },
+    "node_modules/lodash.assign": {
+      "version": "4.2.0",
+      "resolved": "https://registry.npmjs.org/lodash.assign/-/lodash.assign-4.2.0.tgz",
+      "integrity": "sha512-hFuH8TY+Yji7Eja3mGiuAxBqLagejScbG8GbG0j6o9vzn0YL14My+ktnqtZgFTosKymC9/44wP6s7xyuLfnClw==",
+      "dev": true
+    },
+    "node_modules/lodash.camelcase": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz",
+      "integrity": "sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA==",
+      "dev": true
+    },
+    "node_modules/lodash.merge": {
+      "version": "4.6.2",
+      "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz",
+      "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==",
+      "dev": true
+    },
+    "node_modules/lodash.truncate": {
+      "version": "4.4.2",
+      "resolved": "https://registry.npmjs.org/lodash.truncate/-/lodash.truncate-4.4.2.tgz",
+      "integrity": "sha512-jttmRe7bRse52OsWIMDLaXxWqRAmtIUccAQ3garviCqJjafXOfNMO0yMfNpdD6zbGaTU0P5Nz7e7gAT6cKmJRw==",
+      "dev": true
+    },
+    "node_modules/log-symbols": {
+      "version": "4.1.0",
+      "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz",
+      "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==",
+      "dev": true,
+      "dependencies": {
+        "chalk": "^4.1.0",
+        "is-unicode-supported": "^0.1.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/loupe": {
+      "version": "2.3.6",
+      "resolved": "https://registry.npmjs.org/loupe/-/loupe-2.3.6.tgz",
+      "integrity": "sha512-RaPMZKiMy8/JruncMU5Bt6na1eftNoo++R4Y+N2FrxkDVTrGvcyzFTsaGif4QTeKESheMGegbhw6iUAq+5A8zA==",
+      "dev": true,
+      "dependencies": {
+        "get-func-name": "^2.0.0"
+      }
+    },
+    "node_modules/lower-case": {
+      "version": "1.1.4",
+      "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-1.1.4.tgz",
+      "integrity": "sha512-2Fgx1Ycm599x+WGpIYwJOvsjmXFzTSc34IwDWALRA/8AopUKAVPwfJ+h5+f85BCp0PWmmJcWzEpxOpoXycMpdA==",
+      "dev": true
+    },
+    "node_modules/lower-case-first": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/lower-case-first/-/lower-case-first-1.0.2.tgz",
+      "integrity": "sha512-UuxaYakO7XeONbKrZf5FEgkantPf5DUqDayzP5VXZrtRPdH86s4kN47I8B3TW10S4QKiE3ziHNf3kRN//okHjA==",
+      "dev": true,
+      "dependencies": {
+        "lower-case": "^1.1.2"
+      }
+    },
+    "node_modules/lowercase-keys": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-3.0.0.tgz",
+      "integrity": "sha512-ozCC6gdQ+glXOQsveKD0YsDy8DSQFjDTz4zyzEHNV5+JP5D62LmfDZ6o1cycFx9ouG940M5dE8C8CTewdj2YWQ==",
+      "dev": true,
+      "engines": {
+        "node": "^12.20.0 || ^14.13.1 || >=16.0.0"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/lru_map": {
+      "version": "0.3.3",
+      "resolved": "https://registry.npmjs.org/lru_map/-/lru_map-0.3.3.tgz",
+      "integrity": "sha512-Pn9cox5CsMYngeDbmChANltQl+5pi6XmTrraMSzhPmMBbmgcxmqWry0U3PGapCU1yB4/LqCcom7qhHZiF/jGfQ==",
+      "dev": true
+    },
+    "node_modules/lru-cache": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz",
+      "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==",
+      "dev": true,
+      "dependencies": {
+        "yallist": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/make-error": {
+      "version": "1.3.6",
+      "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz",
+      "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==",
+      "dev": true
+    },
+    "node_modules/markdown-table": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/markdown-table/-/markdown-table-1.1.3.tgz",
+      "integrity": "sha512-1RUZVgQlpJSPWYbFSpmudq5nHY1doEIv89gBtF0s4gW1GF2XorxcA/70M5vq7rLv0a6mhOUccRsqkwhwLCIQ2Q==",
+      "dev": true
+    },
+    "node_modules/match-all": {
+      "version": "1.2.6",
+      "resolved": "https://registry.npmjs.org/match-all/-/match-all-1.2.6.tgz",
+      "integrity": "sha512-0EESkXiTkWzrQQntBu2uzKvLu6vVkUGz40nGPbSZuegcfE5UuSzNjLaIu76zJWuaT/2I3Z/8M06OlUOZLGwLlQ==",
+      "dev": true
+    },
+    "node_modules/mcl-wasm": {
+      "version": "0.7.9",
+      "resolved": "https://registry.npmjs.org/mcl-wasm/-/mcl-wasm-0.7.9.tgz",
+      "integrity": "sha512-iJIUcQWA88IJB/5L15GnJVnSQJmf/YaxxV6zRavv83HILHaJQb6y0iFyDMdDO0gN8X37tdxmAOrH/P8B6RB8sQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=8.9.0"
+      }
+    },
+    "node_modules/md5.js": {
+      "version": "1.3.5",
+      "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz",
+      "integrity": "sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==",
+      "dev": true,
+      "dependencies": {
+        "hash-base": "^3.0.0",
+        "inherits": "^2.0.1",
+        "safe-buffer": "^5.1.2"
+      }
+    },
+    "node_modules/media-typer": {
+      "version": "0.3.0",
+      "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
+      "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/memory-level": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/memory-level/-/memory-level-1.0.0.tgz",
+      "integrity": "sha512-UXzwewuWeHBz5krr7EvehKcmLFNoXxGcvuYhC41tRnkrTbJohtS7kVn9akmgirtRygg+f7Yjsfi8Uu5SGSQ4Og==",
+      "dev": true,
+      "dependencies": {
+        "abstract-level": "^1.0.0",
+        "functional-red-black-tree": "^1.0.1",
+        "module-error": "^1.0.1"
+      },
+      "engines": {
+        "node": ">=12"
+      }
+    },
+    "node_modules/memorystream": {
+      "version": "0.3.1",
+      "resolved": "https://registry.npmjs.org/memorystream/-/memorystream-0.3.1.tgz",
+      "integrity": "sha512-S3UwM3yj5mtUSEfP41UZmt/0SCoVYUcU1rkXv+BQ5Ig8ndL4sPoJNBUJERafdPb5jjHJGuMgytgKvKIf58XNBw==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.10.0"
+      }
+    },
+    "node_modules/merge-descriptors": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
+      "integrity": "sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w==",
+      "dev": true
+    },
+    "node_modules/merge2": {
+      "version": "1.4.1",
+      "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz",
+      "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==",
+      "dev": true,
+      "engines": {
+        "node": ">= 8"
+      }
+    },
+    "node_modules/methods": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
+      "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/micromatch": {
+      "version": "4.0.5",
+      "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz",
+      "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==",
+      "dev": true,
+      "dependencies": {
+        "braces": "^3.0.2",
+        "picomatch": "^2.3.1"
+      },
+      "engines": {
+        "node": ">=8.6"
+      }
+    },
+    "node_modules/mime": {
+      "version": "1.6.0",
+      "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
+      "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==",
+      "dev": true,
+      "bin": {
+        "mime": "cli.js"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/mime-db": {
+      "version": "1.52.0",
+      "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
+      "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/mime-types": {
+      "version": "2.1.35",
+      "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
+      "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
+      "dev": true,
+      "dependencies": {
+        "mime-db": "1.52.0"
+      },
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/mimic-fn": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz",
+      "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/mimic-response": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz",
+      "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/min-document": {
+      "version": "2.19.0",
+      "resolved": "https://registry.npmjs.org/min-document/-/min-document-2.19.0.tgz",
+      "integrity": "sha512-9Wy1B3m3f66bPPmU5hdA4DR4PB2OfDU/+GS3yAB7IQozE3tqXaVv2zOjgla7MEGSRv95+ILmOuvhLkOK6wJtCQ==",
+      "dev": true,
+      "dependencies": {
+        "dom-walk": "^0.1.0"
+      }
+    },
+    "node_modules/minimalistic-assert": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz",
+      "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==",
+      "devOptional": true
+    },
+    "node_modules/minimalistic-crypto-utils": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz",
+      "integrity": "sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==",
+      "devOptional": true
+    },
+    "node_modules/minimatch": {
+      "version": "3.1.2",
+      "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
+      "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==",
+      "dev": true,
+      "dependencies": {
+        "brace-expansion": "^1.1.7"
+      },
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/minimist": {
+      "version": "1.2.8",
+      "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz",
+      "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==",
+      "dev": true,
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/minipass": {
+      "version": "2.9.0",
+      "resolved": "https://registry.npmjs.org/minipass/-/minipass-2.9.0.tgz",
+      "integrity": "sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==",
+      "dev": true,
+      "dependencies": {
+        "safe-buffer": "^5.1.2",
+        "yallist": "^3.0.0"
+      }
+    },
+    "node_modules/minipass/node_modules/yallist": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
+      "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==",
+      "dev": true
+    },
+    "node_modules/minizlib": {
+      "version": "1.3.3",
+      "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-1.3.3.tgz",
+      "integrity": "sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==",
+      "dev": true,
+      "dependencies": {
+        "minipass": "^2.9.0"
+      }
+    },
+    "node_modules/mkdirp": {
+      "version": "2.1.5",
+      "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-2.1.5.tgz",
+      "integrity": "sha512-jbjfql+shJtAPrFoKxHOXip4xS+kul9W3OzfzzrqueWK2QMGon2bFH2opl6W9EagBThjEz+iysyi/swOoVfB/w==",
+      "dev": true,
+      "bin": {
+        "mkdirp": "dist/cjs/src/bin.js"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/isaacs"
+      }
+    },
+    "node_modules/mkdirp-promise": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/mkdirp-promise/-/mkdirp-promise-5.0.1.tgz",
+      "integrity": "sha512-Hepn5kb1lJPtVW84RFT40YG1OddBNTOVUZR2bzQUHc+Z03en8/3uX0+060JDhcEzyO08HmipsN9DcnFMxhIL9w==",
+      "deprecated": "This package is broken and no longer maintained. 'mkdirp' itself supports promises now, please switch to that.",
+      "dev": true,
+      "dependencies": {
+        "mkdirp": "*"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/mnemonist": {
+      "version": "0.38.5",
+      "resolved": "https://registry.npmjs.org/mnemonist/-/mnemonist-0.38.5.tgz",
+      "integrity": "sha512-bZTFT5rrPKtPJxj8KSV0WkPyNxl72vQepqqVUAW2ARUpUSF2qXMB6jZj7hW5/k7C1rtpzqbD/IIbJwLXUjCHeg==",
+      "dev": true,
+      "dependencies": {
+        "obliterator": "^2.0.0"
+      }
+    },
+    "node_modules/mocha": {
+      "version": "10.2.0",
+      "resolved": "https://registry.npmjs.org/mocha/-/mocha-10.2.0.tgz",
+      "integrity": "sha512-IDY7fl/BecMwFHzoqF2sg/SHHANeBoMMXFlS9r0OXKDssYE1M5O43wUY/9BVPeIvfH2zmEbBfseqN9gBQZzXkg==",
+      "dev": true,
+      "dependencies": {
+        "ansi-colors": "4.1.1",
+        "browser-stdout": "1.3.1",
+        "chokidar": "3.5.3",
+        "debug": "4.3.4",
+        "diff": "5.0.0",
+        "escape-string-regexp": "4.0.0",
+        "find-up": "5.0.0",
+        "glob": "7.2.0",
+        "he": "1.2.0",
+        "js-yaml": "4.1.0",
+        "log-symbols": "4.1.0",
+        "minimatch": "5.0.1",
+        "ms": "2.1.3",
+        "nanoid": "3.3.3",
+        "serialize-javascript": "6.0.0",
+        "strip-json-comments": "3.1.1",
+        "supports-color": "8.1.1",
+        "workerpool": "6.2.1",
+        "yargs": "16.2.0",
+        "yargs-parser": "20.2.4",
+        "yargs-unparser": "2.0.0"
+      },
+      "bin": {
+        "_mocha": "bin/_mocha",
+        "mocha": "bin/mocha.js"
+      },
+      "engines": {
+        "node": ">= 14.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/mochajs"
+      }
+    },
+    "node_modules/mocha/node_modules/ansi-colors": {
+      "version": "4.1.1",
+      "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz",
+      "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/mocha/node_modules/cliui": {
+      "version": "7.0.4",
+      "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz",
+      "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==",
+      "dev": true,
+      "dependencies": {
+        "string-width": "^4.2.0",
+        "strip-ansi": "^6.0.0",
+        "wrap-ansi": "^7.0.0"
+      }
+    },
+    "node_modules/mocha/node_modules/escape-string-regexp": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz",
+      "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/mocha/node_modules/find-up": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz",
+      "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==",
+      "dev": true,
+      "dependencies": {
+        "locate-path": "^6.0.0",
+        "path-exists": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/mocha/node_modules/get-caller-file": {
+      "version": "2.0.5",
+      "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz",
+      "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==",
+      "dev": true,
+      "engines": {
+        "node": "6.* || 8.* || >= 10.*"
+      }
+    },
+    "node_modules/mocha/node_modules/glob": {
+      "version": "7.2.0",
+      "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz",
+      "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==",
+      "dev": true,
+      "dependencies": {
+        "fs.realpath": "^1.0.0",
+        "inflight": "^1.0.4",
+        "inherits": "2",
+        "minimatch": "^3.0.4",
+        "once": "^1.3.0",
+        "path-is-absolute": "^1.0.0"
+      },
+      "engines": {
+        "node": "*"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/isaacs"
+      }
+    },
+    "node_modules/mocha/node_modules/glob/node_modules/minimatch": {
+      "version": "3.1.2",
+      "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
+      "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==",
+      "dev": true,
+      "dependencies": {
+        "brace-expansion": "^1.1.7"
+      },
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/mocha/node_modules/locate-path": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz",
+      "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==",
+      "dev": true,
+      "dependencies": {
+        "p-locate": "^5.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/mocha/node_modules/minimatch": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.0.1.tgz",
+      "integrity": "sha512-nLDxIFRyhDblz3qMuq+SoRZED4+miJ/G+tdDrjkkkRnjAsBexeGpgjLEQ0blJy7rHhR2b93rhQY4SvyWu9v03g==",
+      "dev": true,
+      "dependencies": {
+        "brace-expansion": "^2.0.1"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/mocha/node_modules/minimatch/node_modules/brace-expansion": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz",
+      "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==",
+      "dev": true,
+      "dependencies": {
+        "balanced-match": "^1.0.0"
+      }
+    },
+    "node_modules/mocha/node_modules/ms": {
+      "version": "2.1.3",
+      "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+      "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
+      "dev": true
+    },
+    "node_modules/mocha/node_modules/p-limit": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
+      "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==",
+      "dev": true,
+      "dependencies": {
+        "yocto-queue": "^0.1.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/mocha/node_modules/p-locate": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz",
+      "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==",
+      "dev": true,
+      "dependencies": {
+        "p-limit": "^3.0.2"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/mocha/node_modules/path-exists": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
+      "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/mocha/node_modules/supports-color": {
+      "version": "8.1.1",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz",
+      "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==",
+      "dev": true,
+      "dependencies": {
+        "has-flag": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/chalk/supports-color?sponsor=1"
+      }
+    },
+    "node_modules/mocha/node_modules/wrap-ansi": {
+      "version": "7.0.0",
+      "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz",
+      "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==",
+      "dev": true,
+      "dependencies": {
+        "ansi-styles": "^4.0.0",
+        "string-width": "^4.1.0",
+        "strip-ansi": "^6.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/chalk/wrap-ansi?sponsor=1"
+      }
+    },
+    "node_modules/mocha/node_modules/y18n": {
+      "version": "5.0.8",
+      "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz",
+      "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/mocha/node_modules/yargs": {
+      "version": "16.2.0",
+      "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz",
+      "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==",
+      "dev": true,
+      "dependencies": {
+        "cliui": "^7.0.2",
+        "escalade": "^3.1.1",
+        "get-caller-file": "^2.0.5",
+        "require-directory": "^2.1.1",
+        "string-width": "^4.2.0",
+        "y18n": "^5.0.5",
+        "yargs-parser": "^20.2.2"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/mocha/node_modules/yargs-parser": {
+      "version": "20.2.4",
+      "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.4.tgz",
+      "integrity": "sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/mock-fs": {
+      "version": "4.14.0",
+      "resolved": "https://registry.npmjs.org/mock-fs/-/mock-fs-4.14.0.tgz",
+      "integrity": "sha512-qYvlv/exQ4+svI3UOvPUpLDF0OMX5euvUH0Ny4N5QyRyhNdgAgUrVH3iUINSzEPLvx0kbo/Bp28GJKIqvE7URw==",
+      "dev": true
+    },
+    "node_modules/module-error": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/module-error/-/module-error-1.0.2.tgz",
+      "integrity": "sha512-0yuvsqSCv8LbaOKhnsQ/T5JhyFlCYLPXK3U2sgV10zoKQwzs/MyfuQUOZQ1V/6OCOJsK/TRgNVrPuPDqtdMFtA==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/ms": {
+      "version": "2.1.2",
+      "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
+      "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==",
+      "dev": true
+    },
+    "node_modules/multibase": {
+      "version": "0.6.1",
+      "resolved": "https://registry.npmjs.org/multibase/-/multibase-0.6.1.tgz",
+      "integrity": "sha512-pFfAwyTjbbQgNc3G7D48JkJxWtoJoBMaR4xQUOuB8RnCgRqaYmWNFeJTTvrJ2w51bjLq2zTby6Rqj9TQ9elSUw==",
+      "deprecated": "This module has been superseded by the multiformats module",
+      "dev": true,
+      "dependencies": {
+        "base-x": "^3.0.8",
+        "buffer": "^5.5.0"
+      }
+    },
+    "node_modules/multicodec": {
+      "version": "0.5.7",
+      "resolved": "https://registry.npmjs.org/multicodec/-/multicodec-0.5.7.tgz",
+      "integrity": "sha512-PscoRxm3f+88fAtELwUnZxGDkduE2HD9Q6GHUOywQLjOGT/HAdhjLDYNZ1e7VR0s0TP0EwZ16LNUTFpoBGivOA==",
+      "deprecated": "This module has been superseded by the multiformats module",
+      "dev": true,
+      "dependencies": {
+        "varint": "^5.0.0"
+      }
+    },
+    "node_modules/multihashes": {
+      "version": "0.4.21",
+      "resolved": "https://registry.npmjs.org/multihashes/-/multihashes-0.4.21.tgz",
+      "integrity": "sha512-uVSvmeCWf36pU2nB4/1kzYZjsXD9vofZKpgudqkceYY5g2aZZXJ5r9lxuzoRLl1OAp28XljXsEJ/X/85ZsKmKw==",
+      "dev": true,
+      "dependencies": {
+        "buffer": "^5.5.0",
+        "multibase": "^0.7.0",
+        "varint": "^5.0.0"
+      }
+    },
+    "node_modules/multihashes/node_modules/multibase": {
+      "version": "0.7.0",
+      "resolved": "https://registry.npmjs.org/multibase/-/multibase-0.7.0.tgz",
+      "integrity": "sha512-TW8q03O0f6PNFTQDvh3xxH03c8CjGaaYrjkl9UQPG6rz53TQzzxJVCIWVjzcbN/Q5Y53Zd0IBQBMVktVgNx4Fg==",
+      "deprecated": "This module has been superseded by the multiformats module",
+      "dev": true,
+      "dependencies": {
+        "base-x": "^3.0.8",
+        "buffer": "^5.5.0"
+      }
+    },
+    "node_modules/murmur-128": {
+      "version": "0.2.1",
+      "resolved": "https://registry.npmjs.org/murmur-128/-/murmur-128-0.2.1.tgz",
+      "integrity": "sha512-WseEgiRkI6aMFBbj8Cg9yBj/y+OdipwVC7zUo3W2W1JAJITwouUOtpqsmGSg67EQmwwSyod7hsVsWY5LsrfQVg==",
+      "dev": true,
+      "dependencies": {
+        "encode-utf8": "^1.0.2",
+        "fmix": "^0.1.0",
+        "imul": "^1.0.0"
+      }
+    },
+    "node_modules/nano-base32": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/nano-base32/-/nano-base32-1.0.1.tgz",
+      "integrity": "sha512-sxEtoTqAPdjWVGv71Q17koMFGsOMSiHsIFEvzOM7cNp8BXB4AnEwmDabm5dorusJf/v1z7QxaZYxUorU9RKaAw==",
+      "dev": true
+    },
+    "node_modules/nano-json-stream-parser": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/nano-json-stream-parser/-/nano-json-stream-parser-0.1.2.tgz",
+      "integrity": "sha512-9MqxMH/BSJC7dnLsEMPyfN5Dvoo49IsPFYMcHw3Bcfc2kN0lpHRBSzlMSVx4HGyJ7s9B31CyBTVehWJoQ8Ctew==",
+      "dev": true
+    },
+    "node_modules/nanoid": {
+      "version": "3.3.3",
+      "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.3.tgz",
+      "integrity": "sha512-p1sjXuopFs0xg+fPASzQ28agW1oHD7xDsd9Xkf3T15H3c/cifrFHVwrh74PdoklAPi+i7MdRsE47vm2r6JoB+w==",
+      "dev": true,
+      "bin": {
+        "nanoid": "bin/nanoid.cjs"
+      },
+      "engines": {
+        "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1"
+      }
+    },
+    "node_modules/napi-macros": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/napi-macros/-/napi-macros-2.0.0.tgz",
+      "integrity": "sha512-A0xLykHtARfueITVDernsAWdtIMbOJgKgcluwENp3AlsKN/PloyO10HtmoqnFAQAcxPkgZN7wdfPfEd0zNGxbg==",
+      "dev": true
+    },
+    "node_modules/natural-compare": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz",
+      "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==",
+      "dev": true
+    },
+    "node_modules/natural-compare-lite": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/natural-compare-lite/-/natural-compare-lite-1.4.0.tgz",
+      "integrity": "sha512-Tj+HTDSJJKaZnfiuw+iaF9skdPpTo2GtEly5JHnWV/hfv2Qj/9RKsGISQtLh2ox3l5EAGw487hnBee0sIJ6v2g==",
+      "dev": true
+    },
+    "node_modules/negotiator": {
+      "version": "0.6.3",
+      "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz",
+      "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/neo-async": {
+      "version": "2.6.2",
+      "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz",
+      "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==",
+      "dev": true
+    },
+    "node_modules/next-tick": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/next-tick/-/next-tick-1.1.0.tgz",
+      "integrity": "sha512-CXdUiJembsNjuToQvxayPZF9Vqht7hewsvy2sOWafLvi2awflj9mOC6bHIg50orX8IJvWKY9wYQ/zB2kogPslQ==",
+      "dev": true
+    },
+    "node_modules/no-case": {
+      "version": "2.3.2",
+      "resolved": "https://registry.npmjs.org/no-case/-/no-case-2.3.2.tgz",
+      "integrity": "sha512-rmTZ9kz+f3rCvK2TD1Ue/oZlns7OGoIWP4fc3llxxRXlOkHKoWPPWJOfFYpITabSow43QJbRIoHQXtt10VldyQ==",
+      "dev": true,
+      "dependencies": {
+        "lower-case": "^1.1.1"
+      }
+    },
+    "node_modules/node-addon-api": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-2.0.2.tgz",
+      "integrity": "sha512-Ntyt4AIXyaLIuMHF6IOoTakB3K+RWxwtsHNRxllEoA6vPwP9o4866g6YWDLUdnucilZhmkxiHwHr11gAENw+QA==",
+      "dev": true
+    },
+    "node_modules/node-emoji": {
+      "version": "1.11.0",
+      "resolved": "https://registry.npmjs.org/node-emoji/-/node-emoji-1.11.0.tgz",
+      "integrity": "sha512-wo2DpQkQp7Sjm2A0cq+sN7EHKO6Sl0ctXeBdFZrL9T9+UywORbufTcTZxom8YqpLQt/FqNMUkOpkZrJVYSKD3A==",
+      "dev": true,
+      "dependencies": {
+        "lodash": "^4.17.21"
+      }
+    },
+    "node_modules/node-fetch": {
+      "version": "2.6.7",
+      "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.7.tgz",
+      "integrity": "sha512-ZjMPFEfVx5j+y2yF35Kzx5sF7kDzxuDj6ziH4FFbOp87zKDZNx8yExJIb05OGF4Nlt9IHFIMBkRl41VdvcNdbQ==",
+      "dev": true,
+      "dependencies": {
+        "whatwg-url": "^5.0.0"
+      },
+      "engines": {
+        "node": "4.x || >=6.0.0"
+      },
+      "peerDependencies": {
+        "encoding": "^0.1.0"
+      },
+      "peerDependenciesMeta": {
+        "encoding": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/node-gyp-build": {
+      "version": "4.6.0",
+      "resolved": "https://registry.npmjs.org/node-gyp-build/-/node-gyp-build-4.6.0.tgz",
+      "integrity": "sha512-NTZVKn9IylLwUzaKjkas1e4u2DLNcV4rdYagA4PWdPwW87Bi7z+BznyKSRwS/761tV/lzCGXplWsiaMjLqP2zQ==",
+      "dev": true,
+      "bin": {
+        "node-gyp-build": "bin.js",
+        "node-gyp-build-optional": "optional.js",
+        "node-gyp-build-test": "build-test.js"
+      }
+    },
+    "node_modules/nofilter": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/nofilter/-/nofilter-3.1.0.tgz",
+      "integrity": "sha512-l2NNj07e9afPnhAhvgVrCD/oy2Ai1yfLpuo3EpiO1jFTsB4sFz6oIfAfSZyQzVpkZQ9xS8ZS5g1jCBgq4Hwo0g==",
+      "dev": true,
+      "engines": {
+        "node": ">=12.19"
+      }
+    },
+    "node_modules/nopt": {
+      "version": "3.0.6",
+      "resolved": "https://registry.npmjs.org/nopt/-/nopt-3.0.6.tgz",
+      "integrity": "sha512-4GUt3kSEYmk4ITxzB/b9vaIDfUVWN/Ml1Fwl11IlnIG2iaJ9O6WXZ9SrYM9NLI8OCBieN2Y8SWC2oJV0RQ7qYg==",
+      "dev": true,
+      "dependencies": {
+        "abbrev": "1"
+      },
+      "bin": {
+        "nopt": "bin/nopt.js"
+      }
+    },
+    "node_modules/normalize-package-data": {
+      "version": "2.5.0",
+      "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz",
+      "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==",
+      "dev": true,
+      "dependencies": {
+        "hosted-git-info": "^2.1.4",
+        "resolve": "^1.10.0",
+        "semver": "2 || 3 || 4 || 5",
+        "validate-npm-package-license": "^3.0.1"
+      }
+    },
+    "node_modules/normalize-package-data/node_modules/semver": {
+      "version": "5.7.2",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz",
+      "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==",
+      "dev": true,
+      "bin": {
+        "semver": "bin/semver"
+      }
+    },
+    "node_modules/normalize-path": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
+      "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/normalize-url": {
+      "version": "6.1.0",
+      "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-6.1.0.tgz",
+      "integrity": "sha512-DlL+XwOy3NxAQ8xuC0okPgK46iuVNAK01YN7RueYBqqFeGsBjV9XmCAzAdgt+667bCl5kPh9EqKKDwnaPG1I7A==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/nth-check": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.1.1.tgz",
+      "integrity": "sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==",
+      "dev": true,
+      "dependencies": {
+        "boolbase": "^1.0.0"
+      },
+      "funding": {
+        "url": "https://github.com/fb55/nth-check?sponsor=1"
+      }
+    },
+    "node_modules/number-is-nan": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz",
+      "integrity": "sha512-4jbtZXNAsfZbAHiiqjLPBiCl16dES1zI4Hpzzxw61Tk+loF+sBDBKx1ICKKKwIqQ7M0mFn1TmkN7euSncWgHiQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/number-to-bn": {
+      "version": "1.7.0",
+      "resolved": "https://registry.npmjs.org/number-to-bn/-/number-to-bn-1.7.0.tgz",
+      "integrity": "sha512-wsJ9gfSz1/s4ZsJN01lyonwuxA1tml6X1yBDnfpMglypcBRFZZkus26EdPSlqS5GJfYddVZa22p3VNb3z5m5Ig==",
+      "dev": true,
+      "dependencies": {
+        "bn.js": "4.11.6",
+        "strip-hex-prefix": "1.0.0"
+      },
+      "engines": {
+        "node": ">=6.5.0",
+        "npm": ">=3"
+      }
+    },
+    "node_modules/number-to-bn/node_modules/bn.js": {
+      "version": "4.11.6",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.6.tgz",
+      "integrity": "sha512-XWwnNNFCuuSQ0m3r3C4LE3EiORltHd9M05pq6FOlVeiophzRbMo50Sbz1ehl8K3Z+jw9+vmgnXefY1hz8X+2wA==",
+      "dev": true
+    },
+    "node_modules/oauth-sign": {
+      "version": "0.9.0",
+      "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz",
+      "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/object-assign": {
+      "version": "4.1.1",
+      "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
+      "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/object-inspect": {
+      "version": "1.12.3",
+      "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.3.tgz",
+      "integrity": "sha512-geUvdk7c+eizMNUDkRpW1wJwgfOiOeHbxBR/hLXK1aT6zmVSO0jsQcs7fj6MGw89jC/cjGfLcNOrtMYtGqm81g==",
+      "dev": true,
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/object-keys": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz",
+      "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.4"
+      }
+    },
+    "node_modules/object.assign": {
+      "version": "4.1.4",
+      "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz",
+      "integrity": "sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "has-symbols": "^1.0.3",
+        "object-keys": "^1.1.1"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/object.values": {
+      "version": "1.1.6",
+      "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.6.tgz",
+      "integrity": "sha512-FVVTkD1vENCsAcwNs9k6jea2uHC/X0+JcjG8YA60FN5CMaJmG95wT9jek/xX9nornqGRrBkKtzuAu2wuHpKqvw==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "es-abstract": "^1.20.4"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/obliterator": {
+      "version": "2.0.4",
+      "resolved": "https://registry.npmjs.org/obliterator/-/obliterator-2.0.4.tgz",
+      "integrity": "sha512-lgHwxlxV1qIg1Eap7LgIeoBWIMFibOjbrYPIPJZcI1mmGAI2m3lNYpK12Y+GBdPQ0U1hRwSord7GIaawz962qQ==",
+      "dev": true
+    },
+    "node_modules/oboe": {
+      "version": "2.1.5",
+      "resolved": "https://registry.npmjs.org/oboe/-/oboe-2.1.5.tgz",
+      "integrity": "sha512-zRFWiF+FoicxEs3jNI/WYUrVEgA7DeET/InK0XQuudGHRg8iIob3cNPrJTKaz4004uaA9Pbe+Dwa8iluhjLZWA==",
+      "dev": true,
+      "dependencies": {
+        "http-https": "^1.0.0"
+      }
+    },
+    "node_modules/on-finished": {
+      "version": "2.4.1",
+      "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz",
+      "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==",
+      "dev": true,
+      "dependencies": {
+        "ee-first": "1.1.1"
+      },
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/once": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
+      "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==",
+      "dev": true,
+      "dependencies": {
+        "wrappy": "1"
+      }
+    },
+    "node_modules/onetime": {
+      "version": "5.1.2",
+      "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz",
+      "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==",
+      "dev": true,
+      "dependencies": {
+        "mimic-fn": "^2.1.0"
+      },
+      "engines": {
+        "node": ">=6"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/open": {
+      "version": "8.4.2",
+      "resolved": "https://registry.npmjs.org/open/-/open-8.4.2.tgz",
+      "integrity": "sha512-7x81NCL719oNbsq/3mh+hVrAWmFuEYUqrq/Iw3kUzH8ReypT9QQ0BLoJS7/G9k6N81XjW4qHWtjWwe/9eLy1EQ==",
+      "dev": true,
+      "dependencies": {
+        "define-lazy-prop": "^2.0.0",
+        "is-docker": "^2.1.1",
+        "is-wsl": "^2.2.0"
+      },
+      "engines": {
+        "node": ">=12"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/opencollective-postinstall": {
+      "version": "2.0.3",
+      "resolved": "https://registry.npmjs.org/opencollective-postinstall/-/opencollective-postinstall-2.0.3.tgz",
+      "integrity": "sha512-8AV/sCtuzUeTo8gQK5qDZzARrulB3egtLzFgteqB2tcT4Mw7B8Kt7JcDHmltjz6FOAHsvTevk70gZEbhM4ZS9Q==",
+      "dev": true,
+      "bin": {
+        "opencollective-postinstall": "index.js"
+      }
+    },
+    "node_modules/optionator": {
+      "version": "0.9.1",
+      "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.1.tgz",
+      "integrity": "sha512-74RlY5FCnhq4jRxVUPKDaRwrVNXMqsGsiW6AJw4XK8hmtm10wC0ypZBLw5IIp85NZMr91+qd1RvvENwg7jjRFw==",
+      "dev": true,
+      "dependencies": {
+        "deep-is": "^0.1.3",
+        "fast-levenshtein": "^2.0.6",
+        "levn": "^0.4.1",
+        "prelude-ls": "^1.2.1",
+        "type-check": "^0.4.0",
+        "word-wrap": "^1.2.3"
+      },
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/ora": {
+      "version": "5.4.1",
+      "resolved": "https://registry.npmjs.org/ora/-/ora-5.4.1.tgz",
+      "integrity": "sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ==",
+      "dev": true,
+      "dependencies": {
+        "bl": "^4.1.0",
+        "chalk": "^4.1.0",
+        "cli-cursor": "^3.1.0",
+        "cli-spinners": "^2.5.0",
+        "is-interactive": "^1.0.0",
+        "is-unicode-supported": "^0.1.0",
+        "log-symbols": "^4.1.0",
+        "strip-ansi": "^6.0.0",
+        "wcwidth": "^1.0.1"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/ordinal": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/ordinal/-/ordinal-1.0.3.tgz",
+      "integrity": "sha512-cMddMgb2QElm8G7vdaa02jhUNbTSrhsgAGUz1OokD83uJTwSUn+nKoNoKVVaRa08yF6sgfO7Maou1+bgLd9rdQ==",
+      "dev": true
+    },
+    "node_modules/os-locale": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/os-locale/-/os-locale-1.4.0.tgz",
+      "integrity": "sha512-PRT7ZORmwu2MEFt4/fv3Q+mEfN4zetKxufQrkShY2oGvUms9r8otu5HfdyIFHkYXjO7laNsoVGmM2MANfuTA8g==",
+      "dev": true,
+      "dependencies": {
+        "lcid": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/os-tmpdir": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz",
+      "integrity": "sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/p-cancelable": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-3.0.0.tgz",
+      "integrity": "sha512-mlVgR3PGuzlo0MmTdk4cXqXWlwQDLnONTAg6sm62XkMJEiRxN3GL3SffkYvqwonbkJBcrI7Uvv5Zh9yjvn2iUw==",
+      "dev": true,
+      "engines": {
+        "node": ">=12.20"
+      }
+    },
+    "node_modules/p-limit": {
+      "version": "1.3.0",
+      "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz",
+      "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==",
+      "dev": true,
+      "dependencies": {
+        "p-try": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/p-locate": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz",
+      "integrity": "sha512-nQja7m7gSKuewoVRen45CtVfODR3crN3goVQ0DDZ9N3yHxgpkuBhZqsaiotSQRrADUrne346peY7kT3TSACykg==",
+      "dev": true,
+      "dependencies": {
+        "p-limit": "^1.1.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/p-map": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/p-map/-/p-map-4.0.0.tgz",
+      "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==",
+      "dev": true,
+      "dependencies": {
+        "aggregate-error": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/p-try": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz",
+      "integrity": "sha512-U1etNYuMJoIz3ZXSrrySFjsXQTWOx2/jdi86L+2pRvph/qMKL6sbcCYdH23fqsbm8TH2Gn0OybpT4eSFlCVHww==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/pako": {
+      "version": "1.0.11",
+      "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz",
+      "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==",
+      "dev": true
+    },
+    "node_modules/param-case": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/param-case/-/param-case-2.1.1.tgz",
+      "integrity": "sha512-eQE845L6ot89sk2N8liD8HAuH4ca6Vvr7VWAWwt7+kvvG5aBcPmmphQ68JsEG2qa9n1TykS2DLeMt363AAH8/w==",
+      "dev": true,
+      "dependencies": {
+        "no-case": "^2.2.0"
+      }
+    },
+    "node_modules/parent-module": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz",
+      "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==",
+      "dev": true,
+      "dependencies": {
+        "callsites": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/parse-cache-control": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/parse-cache-control/-/parse-cache-control-1.0.1.tgz",
+      "integrity": "sha512-60zvsJReQPX5/QP0Kzfd/VrpjScIQ7SHBW6bFCYfEP+fp0Eppr1SHhIO5nd1PjZtvclzSzES9D/p5nFJurwfWg==",
+      "dev": true
+    },
+    "node_modules/parse-headers": {
+      "version": "2.0.5",
+      "resolved": "https://registry.npmjs.org/parse-headers/-/parse-headers-2.0.5.tgz",
+      "integrity": "sha512-ft3iAoLOB/MlwbNXgzy43SWGP6sQki2jQvAyBg/zDFAgr9bfNWZIUj42Kw2eJIl8kEi4PbgE6U1Zau/HwI75HA==",
+      "dev": true
+    },
+    "node_modules/parse-json": {
+      "version": "2.2.0",
+      "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz",
+      "integrity": "sha512-QR/GGaKCkhwk1ePQNYDRKYZ3mwU9ypsKhB0XyFnLQdomyEqk3e8wpW3V5Jp88zbxK4n5ST1nqo+g9juTpownhQ==",
+      "dev": true,
+      "dependencies": {
+        "error-ex": "^1.2.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/parse5": {
+      "version": "7.1.2",
+      "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.1.2.tgz",
+      "integrity": "sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==",
+      "dev": true,
+      "dependencies": {
+        "entities": "^4.4.0"
+      },
+      "funding": {
+        "url": "https://github.com/inikulin/parse5?sponsor=1"
+      }
+    },
+    "node_modules/parse5-htmlparser2-tree-adapter": {
+      "version": "7.0.0",
+      "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.0.0.tgz",
+      "integrity": "sha512-B77tOZrqqfUfnVcOrUvfdLbz4pu4RopLD/4vmu3HUPswwTA8OH0EMW9BlWR2B0RCoiZRAHEUu7IxeP1Pd1UU+g==",
+      "dev": true,
+      "dependencies": {
+        "domhandler": "^5.0.2",
+        "parse5": "^7.0.0"
+      },
+      "funding": {
+        "url": "https://github.com/inikulin/parse5?sponsor=1"
+      }
+    },
+    "node_modules/parseurl": {
+      "version": "1.3.3",
+      "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
+      "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/pascal-case": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/pascal-case/-/pascal-case-2.0.1.tgz",
+      "integrity": "sha512-qjS4s8rBOJa2Xm0jmxXiyh1+OFf6ekCWOvUaRgAQSktzlTbMotS0nmG9gyYAybCWBcuP4fsBeRCKNwGBnMe2OQ==",
+      "dev": true,
+      "dependencies": {
+        "camel-case": "^3.0.0",
+        "upper-case-first": "^1.1.0"
+      }
+    },
+    "node_modules/path-case": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/path-case/-/path-case-2.1.1.tgz",
+      "integrity": "sha512-Ou0N05MioItesaLr9q8TtHVWmJ6fxWdqKB2RohFmNWVyJ+2zeKIeDNWAN6B/Pe7wpzWChhZX6nONYmOnMeJQ/Q==",
+      "dev": true,
+      "dependencies": {
+        "no-case": "^2.2.0"
+      }
+    },
+    "node_modules/path-exists": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-2.1.0.tgz",
+      "integrity": "sha512-yTltuKuhtNeFJKa1PiRzfLAU5182q1y4Eb4XCJ3PBqyzEDkAZRzBrKKBct682ls9reBVHf9udYLN5Nd+K1B9BQ==",
+      "dev": true,
+      "dependencies": {
+        "pinkie-promise": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/path-is-absolute": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
+      "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/path-key": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz",
+      "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/path-parse": {
+      "version": "1.0.7",
+      "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz",
+      "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==",
+      "dev": true
+    },
+    "node_modules/path-to-regexp": {
+      "version": "0.1.7",
+      "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
+      "integrity": "sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ==",
+      "dev": true
+    },
+    "node_modules/path-type": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/path-type/-/path-type-1.1.0.tgz",
+      "integrity": "sha512-S4eENJz1pkiQn9Znv33Q+deTOKmbl+jj1Fl+qiP/vYezj+S8x+J3Uo0ISrx/QoEvIlOaDWJhPaRd1flJ9HXZqg==",
+      "dev": true,
+      "dependencies": {
+        "graceful-fs": "^4.1.2",
+        "pify": "^2.0.0",
+        "pinkie-promise": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/pathval": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/pathval/-/pathval-1.1.1.tgz",
+      "integrity": "sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/pbkdf2": {
+      "version": "3.1.2",
+      "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.1.2.tgz",
+      "integrity": "sha512-iuh7L6jA7JEGu2WxDwtQP1ddOpaJNC4KlDEFfdQajSGgGPNi4OyDc2R7QnbY2bR9QjBVGwgvTdNJZoE7RaxUMA==",
+      "dev": true,
+      "dependencies": {
+        "create-hash": "^1.1.2",
+        "create-hmac": "^1.1.4",
+        "ripemd160": "^2.0.1",
+        "safe-buffer": "^5.0.1",
+        "sha.js": "^2.4.8"
+      },
+      "engines": {
+        "node": ">=0.12"
+      }
+    },
+    "node_modules/performance-now": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz",
+      "integrity": "sha512-7EAHlyLHI56VEIdK57uwHdHKIaAGbnXPiw0yWbarQZOKaKpvUIgW0jWRVLiatnM+XXlSwsanIBH/hzGMJulMow==",
+      "dev": true
+    },
+    "node_modules/picomatch": {
+      "version": "2.3.1",
+      "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz",
+      "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==",
+      "dev": true,
+      "engines": {
+        "node": ">=8.6"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/jonschlinkert"
+      }
+    },
+    "node_modules/pify": {
+      "version": "2.3.0",
+      "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz",
+      "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/pinkie": {
+      "version": "2.0.4",
+      "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz",
+      "integrity": "sha512-MnUuEycAemtSaeFSjXKW/aroV7akBbY+Sv+RkyqFjgAe73F+MR0TBWKBRDkmfWq/HiFmdavfZ1G7h4SPZXaCSg==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/pinkie-promise": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz",
+      "integrity": "sha512-0Gni6D4UcLTbv9c57DfxDGdr41XfgUjqWZu492f0cIGr16zDU06BWP/RAEvOuo7CQ0CNjHaLlM59YJJFm3NWlw==",
+      "dev": true,
+      "dependencies": {
+        "pinkie": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/pkg-dir": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-5.0.0.tgz",
+      "integrity": "sha512-NPE8TDbzl/3YQYY7CSS228s3g2ollTFnc+Qi3tqmqJp9Vg2ovUpixcJEo2HJScN2Ez+kEaal6y70c0ehqJBJeA==",
+      "dev": true,
+      "dependencies": {
+        "find-up": "^5.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/pkg-dir/node_modules/find-up": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz",
+      "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==",
+      "dev": true,
+      "dependencies": {
+        "locate-path": "^6.0.0",
+        "path-exists": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/pkg-dir/node_modules/locate-path": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz",
+      "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==",
+      "dev": true,
+      "dependencies": {
+        "p-locate": "^5.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/pkg-dir/node_modules/p-limit": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
+      "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==",
+      "dev": true,
+      "dependencies": {
+        "yocto-queue": "^0.1.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/pkg-dir/node_modules/p-locate": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz",
+      "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==",
+      "dev": true,
+      "dependencies": {
+        "p-limit": "^3.0.2"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/pkg-dir/node_modules/path-exists": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
+      "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/please-upgrade-node": {
+      "version": "3.2.0",
+      "resolved": "https://registry.npmjs.org/please-upgrade-node/-/please-upgrade-node-3.2.0.tgz",
+      "integrity": "sha512-gQR3WpIgNIKwBMVLkpMUeR3e1/E1y42bqDQZfql+kDeXd8COYfM8PQA4X6y7a8u9Ua9FHmsrrmirW2vHs45hWg==",
+      "dev": true,
+      "dependencies": {
+        "semver-compare": "^1.0.0"
+      }
+    },
+    "node_modules/pluralize": {
+      "version": "8.0.0",
+      "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-8.0.0.tgz",
+      "integrity": "sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/prelude-ls": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz",
+      "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/prettier": {
+      "version": "2.8.8",
+      "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.8.8.tgz",
+      "integrity": "sha512-tdN8qQGvNjw4CHbY+XXk0JgCXn9QiF21a55rBe5LJAU+kDyC4WQn4+awm2Xfk2lQMk5fKup9XgzTZtGkjBdP9Q==",
+      "dev": true,
+      "bin": {
+        "prettier": "bin-prettier.js"
+      },
+      "engines": {
+        "node": ">=10.13.0"
+      },
+      "funding": {
+        "url": "https://github.com/prettier/prettier?sponsor=1"
+      }
+    },
+    "node_modules/prettier-linter-helpers": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.0.tgz",
+      "integrity": "sha512-GbK2cP9nraSSUF9N2XwUwqfzlAFlMNYYl+ShE/V+H8a9uNl/oUqB1w2EL54Jh0OlyRSd8RfWYJ3coVS4TROP2w==",
+      "dev": true,
+      "dependencies": {
+        "fast-diff": "^1.1.2"
+      },
+      "engines": {
+        "node": ">=6.0.0"
+      }
+    },
+    "node_modules/prettier-plugin-solidity": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/prettier-plugin-solidity/-/prettier-plugin-solidity-1.1.3.tgz",
+      "integrity": "sha512-fQ9yucPi2sBbA2U2Xjh6m4isUTJ7S7QLc/XDDsktqqxYfTwdYKJ0EnnywXHwCGAaYbQNK+HIYPL1OemxuMsgeg==",
+      "dev": true,
+      "dependencies": {
+        "@solidity-parser/parser": "^0.16.0",
+        "semver": "^7.3.8",
+        "solidity-comments-extractor": "^0.0.7"
+      },
+      "engines": {
+        "node": ">=12"
+      },
+      "peerDependencies": {
+        "prettier": ">=2.3.0 || >=3.0.0-alpha.0"
+      }
+    },
+    "node_modules/prettier-plugin-solidity/node_modules/@solidity-parser/parser": {
+      "version": "0.16.0",
+      "resolved": "https://registry.npmjs.org/@solidity-parser/parser/-/parser-0.16.0.tgz",
+      "integrity": "sha512-ESipEcHyRHg4Np4SqBCfcXwyxxna1DgFVz69bgpLV8vzl/NP1DtcKsJ4dJZXWQhY/Z4J2LeKBiOkOVZn9ct33Q==",
+      "dev": true,
+      "dependencies": {
+        "antlr4ts": "^0.5.0-alpha.4"
+      }
+    },
+    "node_modules/prettier-plugin-solidity/node_modules/semver": {
+      "version": "7.5.4",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz",
+      "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==",
+      "dev": true,
+      "dependencies": {
+        "lru-cache": "^6.0.0"
+      },
+      "bin": {
+        "semver": "bin/semver.js"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/process": {
+      "version": "0.11.10",
+      "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz",
+      "integrity": "sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.6.0"
+      }
+    },
+    "node_modules/process-nextick-args": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
+      "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==",
+      "dev": true
+    },
+    "node_modules/promise": {
+      "version": "8.3.0",
+      "resolved": "https://registry.npmjs.org/promise/-/promise-8.3.0.tgz",
+      "integrity": "sha512-rZPNPKTOYVNEEKFaq1HqTgOwZD+4/YHS5ukLzQCypkj+OkYx7iv0mA91lJlpPPZ8vMau3IIGj5Qlwrx+8iiSmg==",
+      "dev": true,
+      "dependencies": {
+        "asap": "~2.0.6"
+      }
+    },
+    "node_modules/prompts": {
+      "version": "2.4.2",
+      "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz",
+      "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==",
+      "dev": true,
+      "dependencies": {
+        "kleur": "^3.0.3",
+        "sisteransi": "^1.0.5"
+      },
+      "engines": {
+        "node": ">= 6"
+      }
+    },
+    "node_modules/proxy-addr": {
+      "version": "2.0.7",
+      "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz",
+      "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==",
+      "dev": true,
+      "dependencies": {
+        "forwarded": "0.2.0",
+        "ipaddr.js": "1.9.1"
+      },
+      "engines": {
+        "node": ">= 0.10"
+      }
+    },
+    "node_modules/proxy-from-env": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
+      "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==",
+      "dev": true
+    },
+    "node_modules/psl": {
+      "version": "1.9.0",
+      "resolved": "https://registry.npmjs.org/psl/-/psl-1.9.0.tgz",
+      "integrity": "sha512-E/ZsdU4HLs/68gYzgGTkMicWTLPdAftJLfJFlLUAAKZGkStNU72sZjT66SnMDVOfOWY/YAoiD7Jxa9iHvngcag==",
+      "dev": true
+    },
+    "node_modules/pump": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz",
+      "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==",
+      "dev": true,
+      "dependencies": {
+        "end-of-stream": "^1.1.0",
+        "once": "^1.3.1"
+      }
+    },
+    "node_modules/punycode": {
+      "version": "2.3.0",
+      "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.0.tgz",
+      "integrity": "sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/pure-rand": {
+      "version": "5.0.5",
+      "resolved": "https://registry.npmjs.org/pure-rand/-/pure-rand-5.0.5.tgz",
+      "integrity": "sha512-BwQpbqxSCBJVpamI6ydzcKqyFmnd5msMWUGvzXLm1aXvusbbgkbOto/EUPM00hjveJEaJtdbhUjKSzWRhQVkaw==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "individual",
+          "url": "https://github.com/sponsors/dubzzz"
+        },
+        {
+          "type": "opencollective",
+          "url": "https://opencollective.com/fast-check"
+        }
+      ]
+    },
+    "node_modules/qs": {
+      "version": "6.11.0",
+      "resolved": "https://registry.npmjs.org/qs/-/qs-6.11.0.tgz",
+      "integrity": "sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q==",
+      "dev": true,
+      "dependencies": {
+        "side-channel": "^1.0.4"
+      },
+      "engines": {
+        "node": ">=0.6"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/query-string": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/query-string/-/query-string-5.1.1.tgz",
+      "integrity": "sha512-gjWOsm2SoGlgLEdAGt7a6slVOk9mGiXmPFMqrEhLQ68rhQuBnpfs3+EmlvqKyxnCo9/PPlF+9MtY02S1aFg+Jw==",
+      "dev": true,
+      "dependencies": {
+        "decode-uri-component": "^0.2.0",
+        "object-assign": "^4.1.0",
+        "strict-uri-encode": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/queue-microtask": {
+      "version": "1.2.3",
+      "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz",
+      "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ]
+    },
+    "node_modules/quick-lru": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/quick-lru/-/quick-lru-5.1.1.tgz",
+      "integrity": "sha512-WuyALRjWPDGtt/wzJiadO5AXY+8hZ80hVpe6MyivgraREW751X3SbhRvG3eLKOYN+8VEvqLcf3wdnt44Z4S4SA==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/randombytes": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz",
+      "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==",
+      "dev": true,
+      "dependencies": {
+        "safe-buffer": "^5.1.0"
+      }
+    },
+    "node_modules/range-parser": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
+      "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/raw-body": {
+      "version": "2.5.2",
+      "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.2.tgz",
+      "integrity": "sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==",
+      "dev": true,
+      "dependencies": {
+        "bytes": "3.1.2",
+        "http-errors": "2.0.0",
+        "iconv-lite": "0.4.24",
+        "unpipe": "1.0.0"
+      },
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/read-pkg": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-1.1.0.tgz",
+      "integrity": "sha512-7BGwRHqt4s/uVbuyoeejRn4YmFnYZiFl4AuaeXHlgZf3sONF0SOGlxs2Pw8g6hCKupo08RafIO5YXFNOKTfwsQ==",
+      "dev": true,
+      "dependencies": {
+        "load-json-file": "^1.0.0",
+        "normalize-package-data": "^2.3.2",
+        "path-type": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/read-pkg-up": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-1.0.1.tgz",
+      "integrity": "sha512-WD9MTlNtI55IwYUS27iHh9tK3YoIVhxis8yKhLpTqWtml739uXc9NWTpxoHkfZf3+DkCCsXox94/VWZniuZm6A==",
+      "dev": true,
+      "dependencies": {
+        "find-up": "^1.0.0",
+        "read-pkg": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/readable-stream": {
+      "version": "3.6.1",
+      "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.1.tgz",
+      "integrity": "sha512-+rQmrWMYGA90yenhTYsLWAsLsqVC8osOw6PKE1HDYiO0gdPeKe/xDHNzIAIn4C91YQ6oenEhfYqqc1883qHbjQ==",
+      "dev": true,
+      "dependencies": {
+        "inherits": "^2.0.3",
+        "string_decoder": "^1.1.1",
+        "util-deprecate": "^1.0.1"
+      },
+      "engines": {
+        "node": ">= 6"
+      }
+    },
+    "node_modules/readdirp": {
+      "version": "3.6.0",
+      "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz",
+      "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==",
+      "dev": true,
+      "dependencies": {
+        "picomatch": "^2.2.1"
+      },
+      "engines": {
+        "node": ">=8.10.0"
+      }
+    },
+    "node_modules/rechoir": {
+      "version": "0.6.2",
+      "resolved": "https://registry.npmjs.org/rechoir/-/rechoir-0.6.2.tgz",
+      "integrity": "sha512-HFM8rkZ+i3zrV+4LQjwQ0W+ez98pApMGM3HUrN04j3CqzPOzl9nmP15Y8YXNm8QHGv/eacOVEjqhmWpkRV0NAw==",
+      "dev": true,
+      "dependencies": {
+        "resolve": "^1.1.6"
+      },
+      "engines": {
+        "node": ">= 0.10"
+      }
+    },
+    "node_modules/recursive-readdir": {
+      "version": "2.2.3",
+      "resolved": "https://registry.npmjs.org/recursive-readdir/-/recursive-readdir-2.2.3.tgz",
+      "integrity": "sha512-8HrF5ZsXk5FAH9dgsx3BlUer73nIhuj+9OrQwEbLTPOBzGkL1lsFCR01am+v+0m2Cmbs1nP12hLDl5FA7EszKA==",
+      "dev": true,
+      "dependencies": {
+        "minimatch": "^3.0.5"
+      },
+      "engines": {
+        "node": ">=6.0.0"
+      }
+    },
+    "node_modules/reduce-flatten": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/reduce-flatten/-/reduce-flatten-2.0.0.tgz",
+      "integrity": "sha512-EJ4UNY/U1t2P/2k6oqotuX2Cc3T6nxJwsM0N0asT7dhrtH1ltUxDn4NalSYmPE2rCkVpcf/X6R0wDwcFpzhd4w==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/regenerator-runtime": {
+      "version": "0.13.11",
+      "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.11.tgz",
+      "integrity": "sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg==",
+      "dev": true
+    },
+    "node_modules/regexp.prototype.flags": {
+      "version": "1.4.3",
+      "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.4.3.tgz",
+      "integrity": "sha512-fjggEOO3slI6Wvgjwflkc4NFRCTZAu5CnNfBd5qOMYhWdn67nJBBu34/TkD++eeFmd8C9r9jfXJ27+nSiRkSUA==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.3",
+        "functions-have-names": "^1.2.2"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/regexpp": {
+      "version": "3.2.0",
+      "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.2.0.tgz",
+      "integrity": "sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/mysticatea"
+      }
+    },
+    "node_modules/req-cwd": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/req-cwd/-/req-cwd-2.0.0.tgz",
+      "integrity": "sha512-ueoIoLo1OfB6b05COxAA9UpeoscNpYyM+BqYlA7H6LVF4hKGPXQQSSaD2YmvDVJMkk4UDpAHIeU1zG53IqjvlQ==",
+      "dev": true,
+      "dependencies": {
+        "req-from": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/req-from": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/req-from/-/req-from-2.0.0.tgz",
+      "integrity": "sha512-LzTfEVDVQHBRfjOUMgNBA+V6DWsSnoeKzf42J7l0xa/B4jyPOuuF5MlNSmomLNGemWTnV2TIdjSSLnEn95fOQA==",
+      "dev": true,
+      "dependencies": {
+        "resolve-from": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/req-from/node_modules/resolve-from": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz",
+      "integrity": "sha512-GnlH6vxLymXJNMBo7XP1fJIzBFbdYt49CuTwmB/6N53t+kMPRMFKz783LlQ4tv28XoQfMWinAJX6WCGf2IlaIw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/request": {
+      "version": "2.88.2",
+      "resolved": "https://registry.npmjs.org/request/-/request-2.88.2.tgz",
+      "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==",
+      "deprecated": "request has been deprecated, see https://github.com/request/request/issues/3142",
+      "dev": true,
+      "dependencies": {
+        "aws-sign2": "~0.7.0",
+        "aws4": "^1.8.0",
+        "caseless": "~0.12.0",
+        "combined-stream": "~1.0.6",
+        "extend": "~3.0.2",
+        "forever-agent": "~0.6.1",
+        "form-data": "~2.3.2",
+        "har-validator": "~5.1.3",
+        "http-signature": "~1.2.0",
+        "is-typedarray": "~1.0.0",
+        "isstream": "~0.1.2",
+        "json-stringify-safe": "~5.0.1",
+        "mime-types": "~2.1.19",
+        "oauth-sign": "~0.9.0",
+        "performance-now": "^2.1.0",
+        "qs": "~6.5.2",
+        "safe-buffer": "^5.1.2",
+        "tough-cookie": "~2.5.0",
+        "tunnel-agent": "^0.6.0",
+        "uuid": "^3.3.2"
+      },
+      "engines": {
+        "node": ">= 6"
+      }
+    },
+    "node_modules/request/node_modules/qs": {
+      "version": "6.5.3",
+      "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.3.tgz",
+      "integrity": "sha512-qxXIEh4pCGfHICj1mAJQ2/2XVZkjCDTcEgfoSQxc/fYivUZxTkk7L3bDBJSoNrEzXI17oUO5Dp07ktqE5KzczA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.6"
+      }
+    },
+    "node_modules/request/node_modules/uuid": {
+      "version": "3.4.0",
+      "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz",
+      "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==",
+      "deprecated": "Please upgrade  to version 7 or higher.  Older versions may use Math.random() in certain circumstances, which is known to be problematic.  See https://v8.dev/blog/math-random for details.",
+      "dev": true,
+      "bin": {
+        "uuid": "bin/uuid"
+      }
+    },
+    "node_modules/require-directory": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz",
+      "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/require-from-string": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz",
+      "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/require-main-filename": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-1.0.1.tgz",
+      "integrity": "sha512-IqSUtOVP4ksd1C/ej5zeEh/BIP2ajqpn8c5x+q99gvcIG/Qf0cud5raVnE/Dwd0ua9TXYDoDc0RE5hBSdz22Ug==",
+      "dev": true
+    },
+    "node_modules/resolve": {
+      "version": "1.22.1",
+      "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.1.tgz",
+      "integrity": "sha512-nBpuuYuY5jFsli/JIs1oldw6fOQCBioohqWZg/2hiaOybXOft4lonv85uDOKXdf8rhyK159cxU5cDcK/NKk8zw==",
+      "dev": true,
+      "dependencies": {
+        "is-core-module": "^2.9.0",
+        "path-parse": "^1.0.7",
+        "supports-preserve-symlinks-flag": "^1.0.0"
+      },
+      "bin": {
+        "resolve": "bin/resolve"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/resolve-alpn": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/resolve-alpn/-/resolve-alpn-1.2.1.tgz",
+      "integrity": "sha512-0a1F4l73/ZFZOakJnQ3FvkJ2+gSTQWz/r2KE5OdDY0TxPm5h4GkqkWWfM47T7HsbnOtcJVEF4epCVy6u7Q3K+g==",
+      "dev": true
+    },
+    "node_modules/resolve-from": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz",
+      "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/responselike": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/responselike/-/responselike-2.0.1.tgz",
+      "integrity": "sha512-4gl03wn3hj1HP3yzgdI7d3lCkF95F21Pz4BPGvKHinyQzALR5CapwC8yIi0Rh58DEMQ/SguC03wFj2k0M/mHhw==",
+      "dev": true,
+      "dependencies": {
+        "lowercase-keys": "^2.0.0"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/responselike/node_modules/lowercase-keys": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz",
+      "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/restore-cursor": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz",
+      "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==",
+      "dev": true,
+      "dependencies": {
+        "onetime": "^5.1.0",
+        "signal-exit": "^3.0.2"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/reusify": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz",
+      "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==",
+      "dev": true,
+      "engines": {
+        "iojs": ">=1.0.0",
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/rimraf": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz",
+      "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==",
+      "dev": true,
+      "dependencies": {
+        "glob": "^7.1.3"
+      },
+      "bin": {
+        "rimraf": "bin.js"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/isaacs"
+      }
+    },
+    "node_modules/ripemd160": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/ripemd160/-/ripemd160-2.0.2.tgz",
+      "integrity": "sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==",
+      "dev": true,
+      "dependencies": {
+        "hash-base": "^3.0.0",
+        "inherits": "^2.0.1"
+      }
+    },
+    "node_modules/ripemd160-min": {
+      "version": "0.0.6",
+      "resolved": "https://registry.npmjs.org/ripemd160-min/-/ripemd160-min-0.0.6.tgz",
+      "integrity": "sha512-+GcJgQivhs6S9qvLogusiTcS9kQUfgR75whKuy5jIhuiOfQuJ8fjqxV6EGD5duH1Y/FawFUMtMhyeq3Fbnib8A==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/rlp": {
+      "version": "2.2.7",
+      "resolved": "https://registry.npmjs.org/rlp/-/rlp-2.2.7.tgz",
+      "integrity": "sha512-d5gdPmgQ0Z+AklL2NVXr/IoSjNZFfTVvQWzL/AM2AOcSzYP2xjlb0AC8YyCLc41MSNf6P6QVtjgPdmVtzb+4lQ==",
+      "dev": true,
+      "dependencies": {
+        "bn.js": "^5.2.0"
+      },
+      "bin": {
+        "rlp": "bin/rlp"
+      }
+    },
+    "node_modules/run-parallel": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz",
+      "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "dependencies": {
+        "queue-microtask": "^1.2.2"
+      }
+    },
+    "node_modules/run-parallel-limit": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/run-parallel-limit/-/run-parallel-limit-1.1.0.tgz",
+      "integrity": "sha512-jJA7irRNM91jaKc3Hcl1npHsFLOXOoTkPCUL1JEa1R82O2miplXXRaGdjW/KM/98YQWDhJLiSs793CnXfblJUw==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "dependencies": {
+        "queue-microtask": "^1.2.2"
+      }
+    },
+    "node_modules/rustbn.js": {
+      "version": "0.2.0",
+      "resolved": "https://registry.npmjs.org/rustbn.js/-/rustbn.js-0.2.0.tgz",
+      "integrity": "sha512-4VlvkRUuCJvr2J6Y0ImW7NvTCriMi7ErOAqWk1y69vAdoNIzCF3yPmgeNzx+RQTLEDFq5sHfscn1MwHxP9hNfA==",
+      "dev": true
+    },
+    "node_modules/safe-buffer": {
+      "version": "5.2.1",
+      "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
+      "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ]
+    },
+    "node_modules/safe-regex-test": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.0.tgz",
+      "integrity": "sha512-JBUUzyOgEwXQY1NuPtvcj/qcBDbDmEvWufhlnXZIm75DEHp+afM1r1ujJpJsV/gSM4t59tpDyPi1sd6ZaPFfsA==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "get-intrinsic": "^1.1.3",
+        "is-regex": "^1.1.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/safer-buffer": {
+      "version": "2.1.2",
+      "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
+      "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==",
+      "dev": true
+    },
+    "node_modules/sc-istanbul": {
+      "version": "0.4.6",
+      "resolved": "https://registry.npmjs.org/sc-istanbul/-/sc-istanbul-0.4.6.tgz",
+      "integrity": "sha512-qJFF/8tW/zJsbyfh/iT/ZM5QNHE3CXxtLJbZsL+CzdJLBsPD7SedJZoUA4d8iAcN2IoMp/Dx80shOOd2x96X/g==",
+      "dev": true,
+      "dependencies": {
+        "abbrev": "1.0.x",
+        "async": "1.x",
+        "escodegen": "1.8.x",
+        "esprima": "2.7.x",
+        "glob": "^5.0.15",
+        "handlebars": "^4.0.1",
+        "js-yaml": "3.x",
+        "mkdirp": "0.5.x",
+        "nopt": "3.x",
+        "once": "1.x",
+        "resolve": "1.1.x",
+        "supports-color": "^3.1.0",
+        "which": "^1.1.1",
+        "wordwrap": "^1.0.0"
+      },
+      "bin": {
+        "istanbul": "lib/cli.js"
+      }
+    },
+    "node_modules/sc-istanbul/node_modules/argparse": {
+      "version": "1.0.10",
+      "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz",
+      "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==",
+      "dev": true,
+      "dependencies": {
+        "sprintf-js": "~1.0.2"
+      }
+    },
+    "node_modules/sc-istanbul/node_modules/async": {
+      "version": "1.5.2",
+      "resolved": "https://registry.npmjs.org/async/-/async-1.5.2.tgz",
+      "integrity": "sha512-nSVgobk4rv61R9PUSDtYt7mPVB2olxNR5RWJcAsH676/ef11bUZwvu7+RGYrYauVdDPcO519v68wRhXQtxsV9w==",
+      "dev": true
+    },
+    "node_modules/sc-istanbul/node_modules/esprima": {
+      "version": "2.7.3",
+      "resolved": "https://registry.npmjs.org/esprima/-/esprima-2.7.3.tgz",
+      "integrity": "sha512-OarPfz0lFCiW4/AV2Oy1Rp9qu0iusTKqykwTspGCZtPxmF81JR4MmIebvF1F9+UOKth2ZubLQ4XGGaU+hSn99A==",
+      "dev": true,
+      "bin": {
+        "esparse": "bin/esparse.js",
+        "esvalidate": "bin/esvalidate.js"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/sc-istanbul/node_modules/glob": {
+      "version": "5.0.15",
+      "resolved": "https://registry.npmjs.org/glob/-/glob-5.0.15.tgz",
+      "integrity": "sha512-c9IPMazfRITpmAAKi22dK1VKxGDX9ehhqfABDriL/lzO92xcUKEJPQHrVA/2YHSNFB4iFlykVmWvwo48nr3OxA==",
+      "dev": true,
+      "dependencies": {
+        "inflight": "^1.0.4",
+        "inherits": "2",
+        "minimatch": "2 || 3",
+        "once": "^1.3.0",
+        "path-is-absolute": "^1.0.0"
+      },
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/sc-istanbul/node_modules/has-flag": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-1.0.0.tgz",
+      "integrity": "sha512-DyYHfIYwAJmjAjSSPKANxI8bFY9YtFrgkAfinBojQ8YJTOuOuav64tMUJv584SES4xl74PmuaevIyaLESHdTAA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/sc-istanbul/node_modules/js-yaml": {
+      "version": "3.14.1",
+      "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz",
+      "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==",
+      "dev": true,
+      "dependencies": {
+        "argparse": "^1.0.7",
+        "esprima": "^4.0.0"
+      },
+      "bin": {
+        "js-yaml": "bin/js-yaml.js"
+      }
+    },
+    "node_modules/sc-istanbul/node_modules/js-yaml/node_modules/esprima": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz",
+      "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==",
+      "dev": true,
+      "bin": {
+        "esparse": "bin/esparse.js",
+        "esvalidate": "bin/esvalidate.js"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/sc-istanbul/node_modules/mkdirp": {
+      "version": "0.5.6",
+      "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz",
+      "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==",
+      "dev": true,
+      "dependencies": {
+        "minimist": "^1.2.6"
+      },
+      "bin": {
+        "mkdirp": "bin/cmd.js"
+      }
+    },
+    "node_modules/sc-istanbul/node_modules/resolve": {
+      "version": "1.1.7",
+      "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.1.7.tgz",
+      "integrity": "sha512-9znBF0vBcaSN3W2j7wKvdERPwqTxSpCq+if5C0WoTCyV9n24rua28jeuQ2pL/HOf+yUe/Mef+H/5p60K0Id3bg==",
+      "dev": true
+    },
+    "node_modules/sc-istanbul/node_modules/supports-color": {
+      "version": "3.2.3",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-3.2.3.tgz",
+      "integrity": "sha512-Jds2VIYDrlp5ui7t8abHN2bjAu4LV/q4N2KivFPpGH0lrka0BMq/33AmECUXlKPcHigkNaqfXRENFju+rlcy+A==",
+      "dev": true,
+      "dependencies": {
+        "has-flag": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=0.8.0"
+      }
+    },
+    "node_modules/sc-istanbul/node_modules/which": {
+      "version": "1.3.1",
+      "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz",
+      "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==",
+      "dev": true,
+      "dependencies": {
+        "isexe": "^2.0.0"
+      },
+      "bin": {
+        "which": "bin/which"
+      }
+    },
+    "node_modules/scrypt-js": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/scrypt-js/-/scrypt-js-3.0.1.tgz",
+      "integrity": "sha512-cdwTTnqPu0Hyvf5in5asVdZocVDTNRmR7XEcJuIzMjJeSHybHl7vpB66AzwTaIg6CLSbtjcxc8fqcySfnTkccA==",
+      "devOptional": true
+    },
+    "node_modules/secp256k1": {
+      "version": "4.0.3",
+      "resolved": "https://registry.npmjs.org/secp256k1/-/secp256k1-4.0.3.tgz",
+      "integrity": "sha512-NLZVf+ROMxwtEj3Xa562qgv2BK5e2WNmXPiOdVIPLgs6lyTzMvBq0aWTYMI5XCP9jZMVKOcqZLw/Wc4vDkuxhA==",
+      "dev": true,
+      "hasInstallScript": true,
+      "dependencies": {
+        "elliptic": "^6.5.4",
+        "node-addon-api": "^2.0.0",
+        "node-gyp-build": "^4.2.0"
+      },
+      "engines": {
+        "node": ">=10.0.0"
+      }
+    },
+    "node_modules/semver": {
+      "version": "6.3.1",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz",
+      "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==",
+      "dev": true,
+      "bin": {
+        "semver": "bin/semver.js"
+      }
+    },
+    "node_modules/semver-compare": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/semver-compare/-/semver-compare-1.0.0.tgz",
+      "integrity": "sha512-YM3/ITh2MJ5MtzaM429anh+x2jiLVjqILF4m4oyQB18W7Ggea7BfqdH/wGMK7dDiMghv/6WG7znWMwUDzJiXow==",
+      "dev": true
+    },
+    "node_modules/semver-regex": {
+      "version": "3.1.4",
+      "resolved": "https://registry.npmjs.org/semver-regex/-/semver-regex-3.1.4.tgz",
+      "integrity": "sha512-6IiqeZNgq01qGf0TId0t3NvKzSvUsjcpdEO3AQNeIjR6A2+ckTnQlDpl4qu1bjRv0RzN3FP9hzFmws3lKqRWkA==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/send": {
+      "version": "0.18.0",
+      "resolved": "https://registry.npmjs.org/send/-/send-0.18.0.tgz",
+      "integrity": "sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==",
+      "dev": true,
+      "dependencies": {
+        "debug": "2.6.9",
+        "depd": "2.0.0",
+        "destroy": "1.2.0",
+        "encodeurl": "~1.0.2",
+        "escape-html": "~1.0.3",
+        "etag": "~1.8.1",
+        "fresh": "0.5.2",
+        "http-errors": "2.0.0",
+        "mime": "1.6.0",
+        "ms": "2.1.3",
+        "on-finished": "2.4.1",
+        "range-parser": "~1.2.1",
+        "statuses": "2.0.1"
+      },
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/send/node_modules/debug": {
+      "version": "2.6.9",
+      "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+      "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+      "dev": true,
+      "dependencies": {
+        "ms": "2.0.0"
+      }
+    },
+    "node_modules/send/node_modules/debug/node_modules/ms": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+      "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
+      "dev": true
+    },
+    "node_modules/send/node_modules/ms": {
+      "version": "2.1.3",
+      "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+      "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
+      "dev": true
+    },
+    "node_modules/sentence-case": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/sentence-case/-/sentence-case-2.1.1.tgz",
+      "integrity": "sha512-ENl7cYHaK/Ktwk5OTD+aDbQ3uC8IByu/6Bkg+HDv8Mm+XnBnppVNalcfJTNsp1ibstKh030/JKQQWglDvtKwEQ==",
+      "dev": true,
+      "dependencies": {
+        "no-case": "^2.2.0",
+        "upper-case-first": "^1.1.2"
+      }
+    },
+    "node_modules/serialize-javascript": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.0.tgz",
+      "integrity": "sha512-Qr3TosvguFt8ePWqsvRfrKyQXIiW+nGbYpy8XK24NQHE83caxWt+mIymTT19DGFbNWNLfEwsrkSmN64lVWB9ag==",
+      "dev": true,
+      "dependencies": {
+        "randombytes": "^2.1.0"
+      }
+    },
+    "node_modules/serve-static": {
+      "version": "1.15.0",
+      "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.15.0.tgz",
+      "integrity": "sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==",
+      "dev": true,
+      "dependencies": {
+        "encodeurl": "~1.0.2",
+        "escape-html": "~1.0.3",
+        "parseurl": "~1.3.3",
+        "send": "0.18.0"
+      },
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/servify": {
+      "version": "0.1.12",
+      "resolved": "https://registry.npmjs.org/servify/-/servify-0.1.12.tgz",
+      "integrity": "sha512-/xE6GvsKKqyo1BAY+KxOWXcLpPsUUyji7Qg3bVD7hh1eRze5bR1uYiuDA/k3Gof1s9BTzQZEJK8sNcNGFIzeWw==",
+      "dev": true,
+      "dependencies": {
+        "body-parser": "^1.16.0",
+        "cors": "^2.8.1",
+        "express": "^4.14.0",
+        "request": "^2.79.0",
+        "xhr": "^2.3.3"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/set-blocking": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz",
+      "integrity": "sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw==",
+      "dev": true
+    },
+    "node_modules/setimmediate": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.5.tgz",
+      "integrity": "sha512-MATJdZp8sLqDl/68LfQmbP8zKPLQNV6BIZoIgrscFDQ+RsvK/BxeDQOgyxKKoh0y/8h3BqVFnCqQ/gd+reiIXA==",
+      "dev": true
+    },
+    "node_modules/setprototypeof": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
+      "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==",
+      "dev": true
+    },
+    "node_modules/sha.js": {
+      "version": "2.4.11",
+      "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz",
+      "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==",
+      "dev": true,
+      "dependencies": {
+        "inherits": "^2.0.1",
+        "safe-buffer": "^5.0.1"
+      },
+      "bin": {
+        "sha.js": "bin.js"
+      }
+    },
+    "node_modules/sha1": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/sha1/-/sha1-1.1.1.tgz",
+      "integrity": "sha512-dZBS6OrMjtgVkopB1Gmo4RQCDKiZsqcpAQpkV/aaj+FCrCg8r4I4qMkDPQjBgLIxlmu9k4nUbWq6ohXahOneYA==",
+      "dev": true,
+      "dependencies": {
+        "charenc": ">= 0.0.1",
+        "crypt": ">= 0.0.1"
+      },
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/sha3": {
+      "version": "2.1.4",
+      "resolved": "https://registry.npmjs.org/sha3/-/sha3-2.1.4.tgz",
+      "integrity": "sha512-S8cNxbyb0UGUM2VhRD4Poe5N58gJnJsLJ5vC7FYWGUmGhcsj4++WaIOBFVDxlG0W3To6xBuiRh+i0Qp2oNCOtg==",
+      "dev": true,
+      "dependencies": {
+        "buffer": "6.0.3"
+      }
+    },
+    "node_modules/sha3/node_modules/buffer": {
+      "version": "6.0.3",
+      "resolved": "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz",
+      "integrity": "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ],
+      "dependencies": {
+        "base64-js": "^1.3.1",
+        "ieee754": "^1.2.1"
+      }
+    },
+    "node_modules/shebang-command": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
+      "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==",
+      "dev": true,
+      "dependencies": {
+        "shebang-regex": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/shebang-regex": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz",
+      "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/shelljs": {
+      "version": "0.8.5",
+      "resolved": "https://registry.npmjs.org/shelljs/-/shelljs-0.8.5.tgz",
+      "integrity": "sha512-TiwcRcrkhHvbrZbnRcFYMLl30Dfov3HKqzp5tO5b4pt6G/SezKcYhmDg15zXVBswHmctSAQKznqNW2LO5tTDow==",
+      "dev": true,
+      "dependencies": {
+        "glob": "^7.0.0",
+        "interpret": "^1.0.0",
+        "rechoir": "^0.6.2"
+      },
+      "bin": {
+        "shjs": "bin/shjs"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/side-channel": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz",
+      "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.0",
+        "get-intrinsic": "^1.0.2",
+        "object-inspect": "^1.9.0"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/signal-exit": {
+      "version": "3.0.7",
+      "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz",
+      "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==",
+      "dev": true
+    },
+    "node_modules/simple-concat": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/simple-concat/-/simple-concat-1.0.1.tgz",
+      "integrity": "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/feross"
+        },
+        {
+          "type": "patreon",
+          "url": "https://www.patreon.com/feross"
+        },
+        {
+          "type": "consulting",
+          "url": "https://feross.org/support"
+        }
+      ]
+    },
+    "node_modules/simple-get": {
+      "version": "2.8.2",
+      "resolved": "https://registry.npmjs.org/simple-get/-/simple-get-2.8.2.tgz",
+      "integrity": "sha512-Ijd/rV5o+mSBBs4F/x9oDPtTx9Zb6X9brmnXvMW4J7IR15ngi9q5xxqWBKU744jTZiaXtxaPL7uHG6vtN8kUkw==",
+      "dev": true,
+      "dependencies": {
+        "decompress-response": "^3.3.0",
+        "once": "^1.3.1",
+        "simple-concat": "^1.0.0"
+      }
+    },
+    "node_modules/simple-get/node_modules/decompress-response": {
+      "version": "3.3.0",
+      "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz",
+      "integrity": "sha512-BzRPQuY1ip+qDonAOz42gRm/pg9F768C+npV/4JOsxRC2sq+Rlk+Q4ZCAsOhnIaMrgarILY+RMUIvMmmX1qAEA==",
+      "dev": true,
+      "dependencies": {
+        "mimic-response": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/sisteransi": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz",
+      "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==",
+      "dev": true
+    },
+    "node_modules/slash": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz",
+      "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/slice-ansi": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-4.0.0.tgz",
+      "integrity": "sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ==",
+      "dev": true,
+      "dependencies": {
+        "ansi-styles": "^4.0.0",
+        "astral-regex": "^2.0.0",
+        "is-fullwidth-code-point": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/chalk/slice-ansi?sponsor=1"
+      }
+    },
+    "node_modules/snake-case": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/snake-case/-/snake-case-2.1.0.tgz",
+      "integrity": "sha512-FMR5YoPFwOLuh4rRz92dywJjyKYZNLpMn1R5ujVpIYkbA9p01fq8RMg0FkO4M+Yobt4MjHeLTJVm5xFFBHSV2Q==",
+      "dev": true,
+      "dependencies": {
+        "no-case": "^2.2.0"
+      }
+    },
+    "node_modules/solc": {
+      "version": "0.4.26",
+      "resolved": "https://registry.npmjs.org/solc/-/solc-0.4.26.tgz",
+      "integrity": "sha512-o+c6FpkiHd+HPjmjEVpQgH7fqZ14tJpXhho+/bQXlXbliLIS/xjXb42Vxh+qQY1WCSTMQ0+a5vR9vi0MfhU6mA==",
+      "dev": true,
+      "dependencies": {
+        "fs-extra": "^0.30.0",
+        "memorystream": "^0.3.1",
+        "require-from-string": "^1.1.0",
+        "semver": "^5.3.0",
+        "yargs": "^4.7.1"
+      },
+      "bin": {
+        "solcjs": "solcjs"
+      }
+    },
+    "node_modules/solc/node_modules/fs-extra": {
+      "version": "0.30.0",
+      "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-0.30.0.tgz",
+      "integrity": "sha512-UvSPKyhMn6LEd/WpUaV9C9t3zATuqoqfWc3QdPhPLb58prN9tqYPlPWi8Krxi44loBoUzlobqZ3+8tGpxxSzwA==",
+      "dev": true,
+      "dependencies": {
+        "graceful-fs": "^4.1.2",
+        "jsonfile": "^2.1.0",
+        "klaw": "^1.0.0",
+        "path-is-absolute": "^1.0.0",
+        "rimraf": "^2.2.8"
+      }
+    },
+    "node_modules/solc/node_modules/jsonfile": {
+      "version": "2.4.0",
+      "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-2.4.0.tgz",
+      "integrity": "sha512-PKllAqbgLgxHaj8TElYymKCAgrASebJrWpTnEkOaTowt23VKXXN0sUeriJ+eh7y6ufb/CC5ap11pz71/cM0hUw==",
+      "dev": true,
+      "optionalDependencies": {
+        "graceful-fs": "^4.1.6"
+      }
+    },
+    "node_modules/solc/node_modules/require-from-string": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-1.2.1.tgz",
+      "integrity": "sha512-H7AkJWMobeskkttHyhTVtS0fxpFLjxhbfMa6Bk3wimP7sdPRGL3EyCg3sAQenFfAe+xQ+oAc85Nmtvq0ROM83Q==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/solc/node_modules/rimraf": {
+      "version": "2.7.1",
+      "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz",
+      "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==",
+      "dev": true,
+      "dependencies": {
+        "glob": "^7.1.3"
+      },
+      "bin": {
+        "rimraf": "bin.js"
+      }
+    },
+    "node_modules/solc/node_modules/semver": {
+      "version": "5.7.2",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz",
+      "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==",
+      "dev": true,
+      "bin": {
+        "semver": "bin/semver"
+      }
+    },
+    "node_modules/solhint": {
+      "version": "3.4.1",
+      "resolved": "https://registry.npmjs.org/solhint/-/solhint-3.4.1.tgz",
+      "integrity": "sha512-pzZn2RlZhws1XwvLPVSsxfHrwsteFf5eySOhpAytzXwKQYbTCJV6z8EevYDiSVKMpWrvbKpEtJ055CuEmzp4Xg==",
+      "dev": true,
+      "dependencies": {
+        "@solidity-parser/parser": "^0.16.0",
+        "ajv": "^6.12.6",
+        "antlr4": "^4.11.0",
+        "ast-parents": "^0.0.1",
+        "chalk": "^4.1.2",
+        "commander": "^10.0.0",
+        "cosmiconfig": "^8.0.0",
+        "fast-diff": "^1.2.0",
+        "glob": "^8.0.3",
+        "ignore": "^5.2.4",
+        "js-yaml": "^4.1.0",
+        "lodash": "^4.17.21",
+        "pluralize": "^8.0.0",
+        "semver": "^6.3.0",
+        "strip-ansi": "^6.0.1",
+        "table": "^6.8.1",
+        "text-table": "^0.2.0"
+      },
+      "bin": {
+        "solhint": "solhint.js"
+      },
+      "optionalDependencies": {
+        "prettier": "^2.8.3"
+      }
+    },
+    "node_modules/solhint/node_modules/@solidity-parser/parser": {
+      "version": "0.16.0",
+      "resolved": "https://registry.npmjs.org/@solidity-parser/parser/-/parser-0.16.0.tgz",
+      "integrity": "sha512-ESipEcHyRHg4Np4SqBCfcXwyxxna1DgFVz69bgpLV8vzl/NP1DtcKsJ4dJZXWQhY/Z4J2LeKBiOkOVZn9ct33Q==",
+      "dev": true,
+      "dependencies": {
+        "antlr4ts": "^0.5.0-alpha.4"
+      }
+    },
+    "node_modules/solhint/node_modules/ajv": {
+      "version": "6.12.6",
+      "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+      "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+      "dev": true,
+      "dependencies": {
+        "fast-deep-equal": "^3.1.1",
+        "fast-json-stable-stringify": "^2.0.0",
+        "json-schema-traverse": "^0.4.1",
+        "uri-js": "^4.2.2"
+      },
+      "funding": {
+        "type": "github",
+        "url": "https://github.com/sponsors/epoberezkin"
+      }
+    },
+    "node_modules/solhint/node_modules/brace-expansion": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz",
+      "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==",
+      "dev": true,
+      "dependencies": {
+        "balanced-match": "^1.0.0"
+      }
+    },
+    "node_modules/solhint/node_modules/commander": {
+      "version": "10.0.0",
+      "resolved": "https://registry.npmjs.org/commander/-/commander-10.0.0.tgz",
+      "integrity": "sha512-zS5PnTI22FIRM6ylNW8G4Ap0IEOyk62fhLSD0+uHRT9McRCLGpkVNvao4bjimpK/GShynyQkFFxHhwMcETmduA==",
+      "dev": true,
+      "engines": {
+        "node": ">=14"
+      }
+    },
+    "node_modules/solhint/node_modules/cosmiconfig": {
+      "version": "8.1.0",
+      "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-8.1.0.tgz",
+      "integrity": "sha512-0tLZ9URlPGU7JsKq0DQOQ3FoRsYX8xDZ7xMiATQfaiGMz7EHowNkbU9u1coAOmnh9p/1ySpm0RB3JNWRXM5GCg==",
+      "dev": true,
+      "dependencies": {
+        "import-fresh": "^3.2.1",
+        "js-yaml": "^4.1.0",
+        "parse-json": "^5.0.0",
+        "path-type": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=14"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/d-fischer"
+      }
+    },
+    "node_modules/solhint/node_modules/glob": {
+      "version": "8.1.0",
+      "resolved": "https://registry.npmjs.org/glob/-/glob-8.1.0.tgz",
+      "integrity": "sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==",
+      "dev": true,
+      "dependencies": {
+        "fs.realpath": "^1.0.0",
+        "inflight": "^1.0.4",
+        "inherits": "2",
+        "minimatch": "^5.0.1",
+        "once": "^1.3.0"
+      },
+      "engines": {
+        "node": ">=12"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/isaacs"
+      }
+    },
+    "node_modules/solhint/node_modules/json-schema-traverse": {
+      "version": "0.4.1",
+      "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+      "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+      "dev": true
+    },
+    "node_modules/solhint/node_modules/minimatch": {
+      "version": "5.1.6",
+      "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.6.tgz",
+      "integrity": "sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==",
+      "dev": true,
+      "dependencies": {
+        "brace-expansion": "^2.0.1"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/solhint/node_modules/parse-json": {
+      "version": "5.2.0",
+      "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz",
+      "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==",
+      "dev": true,
+      "dependencies": {
+        "@babel/code-frame": "^7.0.0",
+        "error-ex": "^1.3.1",
+        "json-parse-even-better-errors": "^2.3.0",
+        "lines-and-columns": "^1.1.6"
+      },
+      "engines": {
+        "node": ">=8"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/solhint/node_modules/path-type": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz",
+      "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/solidity-comments-extractor": {
+      "version": "0.0.7",
+      "resolved": "https://registry.npmjs.org/solidity-comments-extractor/-/solidity-comments-extractor-0.0.7.tgz",
+      "integrity": "sha512-wciNMLg/Irp8OKGrh3S2tfvZiZ0NEyILfcRCXCD4mp7SgK/i9gzLfhY2hY7VMCQJ3kH9UB9BzNdibIVMchzyYw==",
+      "dev": true
+    },
+    "node_modules/solidity-coverage": {
+      "version": "0.8.5",
+      "resolved": "https://registry.npmjs.org/solidity-coverage/-/solidity-coverage-0.8.5.tgz",
+      "integrity": "sha512-6C6N6OV2O8FQA0FWA95FdzVH+L16HU94iFgg5wAFZ29UpLFkgNI/DRR2HotG1bC0F4gAc/OMs2BJI44Q/DYlKQ==",
+      "dev": true,
+      "dependencies": {
+        "@ethersproject/abi": "^5.0.9",
+        "@solidity-parser/parser": "^0.16.0",
+        "chalk": "^2.4.2",
+        "death": "^1.1.0",
+        "detect-port": "^1.3.0",
+        "difflib": "^0.2.4",
+        "fs-extra": "^8.1.0",
+        "ghost-testrpc": "^0.0.2",
+        "global-modules": "^2.0.0",
+        "globby": "^10.0.1",
+        "jsonschema": "^1.2.4",
+        "lodash": "^4.17.15",
+        "mocha": "10.2.0",
+        "node-emoji": "^1.10.0",
+        "pify": "^4.0.1",
+        "recursive-readdir": "^2.2.2",
+        "sc-istanbul": "^0.4.5",
+        "semver": "^7.3.4",
+        "shelljs": "^0.8.3",
+        "web3-utils": "^1.3.6"
+      },
+      "bin": {
+        "solidity-coverage": "plugins/bin.js"
+      },
+      "peerDependencies": {
+        "hardhat": "^2.11.0"
+      }
+    },
+    "node_modules/solidity-coverage/node_modules/@solidity-parser/parser": {
+      "version": "0.16.2",
+      "resolved": "https://registry.npmjs.org/@solidity-parser/parser/-/parser-0.16.2.tgz",
+      "integrity": "sha512-PI9NfoA3P8XK2VBkK5oIfRgKDsicwDZfkVq9ZTBCQYGOP1N2owgY2dyLGyU5/J/hQs8KRk55kdmvTLjy3Mu3vg==",
+      "dev": true,
+      "dependencies": {
+        "antlr4ts": "^0.5.0-alpha.4"
+      }
+    },
+    "node_modules/solidity-coverage/node_modules/ansi-styles": {
+      "version": "3.2.1",
+      "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+      "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+      "dev": true,
+      "dependencies": {
+        "color-convert": "^1.9.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/solidity-coverage/node_modules/chalk": {
+      "version": "2.4.2",
+      "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+      "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+      "dev": true,
+      "dependencies": {
+        "ansi-styles": "^3.2.1",
+        "escape-string-regexp": "^1.0.5",
+        "supports-color": "^5.3.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/solidity-coverage/node_modules/color-convert": {
+      "version": "1.9.3",
+      "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+      "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+      "dev": true,
+      "dependencies": {
+        "color-name": "1.1.3"
+      }
+    },
+    "node_modules/solidity-coverage/node_modules/color-name": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+      "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+      "dev": true
+    },
+    "node_modules/solidity-coverage/node_modules/fs-extra": {
+      "version": "8.1.0",
+      "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz",
+      "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==",
+      "dev": true,
+      "dependencies": {
+        "graceful-fs": "^4.2.0",
+        "jsonfile": "^4.0.0",
+        "universalify": "^0.1.0"
+      },
+      "engines": {
+        "node": ">=6 <7 || >=8"
+      }
+    },
+    "node_modules/solidity-coverage/node_modules/globby": {
+      "version": "10.0.2",
+      "resolved": "https://registry.npmjs.org/globby/-/globby-10.0.2.tgz",
+      "integrity": "sha512-7dUi7RvCoT/xast/o/dLN53oqND4yk0nsHkhRgn9w65C4PofCLOoJ39iSOg+qVDdWQPIEj+eszMHQ+aLVwwQSg==",
+      "dev": true,
+      "dependencies": {
+        "@types/glob": "^7.1.1",
+        "array-union": "^2.1.0",
+        "dir-glob": "^3.0.1",
+        "fast-glob": "^3.0.3",
+        "glob": "^7.1.3",
+        "ignore": "^5.1.1",
+        "merge2": "^1.2.3",
+        "slash": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/solidity-coverage/node_modules/has-flag": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+      "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/solidity-coverage/node_modules/pify": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz",
+      "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/solidity-coverage/node_modules/semver": {
+      "version": "7.5.4",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz",
+      "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==",
+      "dev": true,
+      "dependencies": {
+        "lru-cache": "^6.0.0"
+      },
+      "bin": {
+        "semver": "bin/semver.js"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/solidity-coverage/node_modules/supports-color": {
+      "version": "5.5.0",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+      "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+      "dev": true,
+      "dependencies": {
+        "has-flag": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/solmate": {
+      "version": "6.7.0",
+      "resolved": "https://registry.npmjs.org/solmate/-/solmate-6.7.0.tgz",
+      "integrity": "sha512-iMPr+gKbKjXBB12a+Iz5Tua5r7T4yugHaGXDWSJbBZB4Gr3vLeUUvKeLyMxCWWqk1xlLhFDFFuAmOzeyVBuyvQ==",
+      "dev": true
+    },
+    "node_modules/source-map": {
+      "version": "0.6.1",
+      "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz",
+      "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/source-map-support": {
+      "version": "0.5.21",
+      "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz",
+      "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==",
+      "dev": true,
+      "dependencies": {
+        "buffer-from": "^1.0.0",
+        "source-map": "^0.6.0"
+      }
+    },
+    "node_modules/spdx-correct": {
+      "version": "3.2.0",
+      "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.2.0.tgz",
+      "integrity": "sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA==",
+      "dev": true,
+      "dependencies": {
+        "spdx-expression-parse": "^3.0.0",
+        "spdx-license-ids": "^3.0.0"
+      }
+    },
+    "node_modules/spdx-exceptions": {
+      "version": "2.3.0",
+      "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz",
+      "integrity": "sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A==",
+      "dev": true
+    },
+    "node_modules/spdx-expression-parse": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz",
+      "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==",
+      "dev": true,
+      "dependencies": {
+        "spdx-exceptions": "^2.1.0",
+        "spdx-license-ids": "^3.0.0"
+      }
+    },
+    "node_modules/spdx-license-ids": {
+      "version": "3.0.12",
+      "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.12.tgz",
+      "integrity": "sha512-rr+VVSXtRhO4OHbXUiAF7xW3Bo9DuuF6C5jH+q/x15j2jniycgKbxU09Hr0WqlSLUs4i4ltHGXqTe7VHclYWyA==",
+      "dev": true
+    },
+    "node_modules/sprintf-js": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz",
+      "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==",
+      "dev": true
+    },
+    "node_modules/sshpk": {
+      "version": "1.17.0",
+      "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.17.0.tgz",
+      "integrity": "sha512-/9HIEs1ZXGhSPE8X6Ccm7Nam1z8KcoCqPdI7ecm1N33EzAetWahvQWVqLZtaZQ+IDKX4IyA2o0gBzqIMkAagHQ==",
+      "dev": true,
+      "dependencies": {
+        "asn1": "~0.2.3",
+        "assert-plus": "^1.0.0",
+        "bcrypt-pbkdf": "^1.0.0",
+        "dashdash": "^1.12.0",
+        "ecc-jsbn": "~0.1.1",
+        "getpass": "^0.1.1",
+        "jsbn": "~0.1.0",
+        "safer-buffer": "^2.0.2",
+        "tweetnacl": "~0.14.0"
+      },
+      "bin": {
+        "sshpk-conv": "bin/sshpk-conv",
+        "sshpk-sign": "bin/sshpk-sign",
+        "sshpk-verify": "bin/sshpk-verify"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/stacktrace-parser": {
+      "version": "0.1.10",
+      "resolved": "https://registry.npmjs.org/stacktrace-parser/-/stacktrace-parser-0.1.10.tgz",
+      "integrity": "sha512-KJP1OCML99+8fhOHxwwzyWrlUuVX5GQ0ZpJTd1DFXhdkrvg1szxfHhawXUZ3g9TkXORQd4/WG68jMlQZ2p8wlg==",
+      "dev": true,
+      "dependencies": {
+        "type-fest": "^0.7.1"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/stacktrace-parser/node_modules/type-fest": {
+      "version": "0.7.1",
+      "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.7.1.tgz",
+      "integrity": "sha512-Ne2YiiGN8bmrmJJEuTWTLJR32nh/JdL1+PSicowtNb0WFpn59GK8/lfD61bVtzguz7b3PBt74nxpv/Pw5po5Rg==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/statuses": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz",
+      "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/strict-uri-encode": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/strict-uri-encode/-/strict-uri-encode-1.1.0.tgz",
+      "integrity": "sha512-R3f198pcvnB+5IpnBlRkphuE9n46WyVl8I39W/ZUTZLz4nqSP/oLYUrcnJrw462Ds8he4YKMov2efsTIw1BDGQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/string_decoder": {
+      "version": "1.3.0",
+      "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz",
+      "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==",
+      "dev": true,
+      "dependencies": {
+        "safe-buffer": "~5.2.0"
+      }
+    },
+    "node_modules/string-format": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/string-format/-/string-format-2.0.0.tgz",
+      "integrity": "sha512-bbEs3scLeYNXLecRRuk6uJxdXUSj6le/8rNPHChIJTn2V79aXVTR1EH2OH5zLKKoz0V02fOUKZZcw01pLUShZA==",
+      "dev": true
+    },
+    "node_modules/string-width": {
+      "version": "4.2.3",
+      "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
+      "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
+      "dev": true,
+      "dependencies": {
+        "emoji-regex": "^8.0.0",
+        "is-fullwidth-code-point": "^3.0.0",
+        "strip-ansi": "^6.0.1"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/string.prototype.trimend": {
+      "version": "1.0.6",
+      "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.6.tgz",
+      "integrity": "sha512-JySq+4mrPf9EsDBEDYMOb/lM7XQLulwg5R/m1r0PXEFqrV0qHvl58sdTilSXtKOflCsK2E8jxf+GKC0T07RWwQ==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "es-abstract": "^1.20.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/string.prototype.trimstart": {
+      "version": "1.0.6",
+      "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.6.tgz",
+      "integrity": "sha512-omqjMDaY92pbn5HOX7f9IccLA+U1tA9GvtU4JrodiXFfYB7jPzzHpRzpglLAjtUV6bB557zwClJezTqnAiYnQA==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "es-abstract": "^1.20.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/strip-ansi": {
+      "version": "6.0.1",
+      "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
+      "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
+      "dev": true,
+      "dependencies": {
+        "ansi-regex": "^5.0.1"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/strip-bom": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-2.0.0.tgz",
+      "integrity": "sha512-kwrX1y7czp1E69n2ajbG65mIo9dqvJ+8aBQXOGVxqwvNbsXdFM6Lq37dLAY3mknUwru8CfcCbfOLL/gMo+fi3g==",
+      "dev": true,
+      "dependencies": {
+        "is-utf8": "^0.2.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/strip-hex-prefix": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/strip-hex-prefix/-/strip-hex-prefix-1.0.0.tgz",
+      "integrity": "sha512-q8d4ue7JGEiVcypji1bALTos+0pWtyGlivAWyPuTkHzuTCJqrK9sWxYQZUq6Nq3cuyv3bm734IhHvHtGGURU6A==",
+      "dev": true,
+      "dependencies": {
+        "is-hex-prefixed": "1.0.0"
+      },
+      "engines": {
+        "node": ">=6.5.0",
+        "npm": ">=3"
+      }
+    },
+    "node_modules/strip-indent": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/strip-indent/-/strip-indent-2.0.0.tgz",
+      "integrity": "sha512-RsSNPLpq6YUL7QYy44RnPVTn/lcVZtb48Uof3X5JLbF4zD/Gs7ZFDv2HWol+leoQN2mT86LAzSshGfkTlSOpsA==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/strip-json-comments": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz",
+      "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/supports-color": {
+      "version": "7.2.0",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
+      "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
+      "dev": true,
+      "dependencies": {
+        "has-flag": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/supports-preserve-symlinks-flag": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz",
+      "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/swap-case": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/swap-case/-/swap-case-1.1.2.tgz",
+      "integrity": "sha512-BAmWG6/bx8syfc6qXPprof3Mn5vQgf5dwdUNJhsNqU9WdPt5P+ES/wQ5bxfijy8zwZgZZHslC3iAsxsuQMCzJQ==",
+      "dev": true,
+      "dependencies": {
+        "lower-case": "^1.1.1",
+        "upper-case": "^1.1.1"
+      }
+    },
+    "node_modules/swarm-js": {
+      "version": "0.1.42",
+      "resolved": "https://registry.npmjs.org/swarm-js/-/swarm-js-0.1.42.tgz",
+      "integrity": "sha512-BV7c/dVlA3R6ya1lMlSSNPLYrntt0LUq4YMgy3iwpCIc6rZnS5W2wUoctarZ5pXlpKtxDDf9hNziEkcfrxdhqQ==",
+      "dev": true,
+      "dependencies": {
+        "bluebird": "^3.5.0",
+        "buffer": "^5.0.5",
+        "eth-lib": "^0.1.26",
+        "fs-extra": "^4.0.2",
+        "got": "^11.8.5",
+        "mime-types": "^2.1.16",
+        "mkdirp-promise": "^5.0.1",
+        "mock-fs": "^4.1.0",
+        "setimmediate": "^1.0.5",
+        "tar": "^4.0.2",
+        "xhr-request": "^1.0.1"
+      }
+    },
+    "node_modules/swarm-js/node_modules/@szmarczak/http-timer": {
+      "version": "4.0.6",
+      "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-4.0.6.tgz",
+      "integrity": "sha512-4BAffykYOgO+5nzBWYwE3W90sBgLJoUPRWWcL8wlyiM8IB8ipJz3UMJ9KXQd1RKQXpKp8Tutn80HZtWsu2u76w==",
+      "dev": true,
+      "dependencies": {
+        "defer-to-connect": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/swarm-js/node_modules/cacheable-lookup": {
+      "version": "5.0.4",
+      "resolved": "https://registry.npmjs.org/cacheable-lookup/-/cacheable-lookup-5.0.4.tgz",
+      "integrity": "sha512-2/kNscPhpcxrOigMZzbiWF7dz8ilhb/nIHU3EyZiXWXpeq/au8qJ8VhdftMkty3n7Gj6HIGalQG8oiBNB3AJgA==",
+      "dev": true,
+      "engines": {
+        "node": ">=10.6.0"
+      }
+    },
+    "node_modules/swarm-js/node_modules/fs-extra": {
+      "version": "4.0.3",
+      "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-4.0.3.tgz",
+      "integrity": "sha512-q6rbdDd1o2mAnQreO7YADIxf/Whx4AHBiRf6d+/cVT8h44ss+lHgxf1FemcqDnQt9X3ct4McHr+JMGlYSsK7Cg==",
+      "dev": true,
+      "dependencies": {
+        "graceful-fs": "^4.1.2",
+        "jsonfile": "^4.0.0",
+        "universalify": "^0.1.0"
+      }
+    },
+    "node_modules/swarm-js/node_modules/got": {
+      "version": "11.8.6",
+      "resolved": "https://registry.npmjs.org/got/-/got-11.8.6.tgz",
+      "integrity": "sha512-6tfZ91bOr7bOXnK7PRDCGBLa1H4U080YHNaAQ2KsMGlLEzRbk44nsZF2E1IeRc3vtJHPVbKCYgdFbaGO2ljd8g==",
+      "dev": true,
+      "dependencies": {
+        "@sindresorhus/is": "^4.0.0",
+        "@szmarczak/http-timer": "^4.0.5",
+        "@types/cacheable-request": "^6.0.1",
+        "@types/responselike": "^1.0.0",
+        "cacheable-lookup": "^5.0.3",
+        "cacheable-request": "^7.0.2",
+        "decompress-response": "^6.0.0",
+        "http2-wrapper": "^1.0.0-beta.5.2",
+        "lowercase-keys": "^2.0.0",
+        "p-cancelable": "^2.0.0",
+        "responselike": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=10.19.0"
+      },
+      "funding": {
+        "url": "https://github.com/sindresorhus/got?sponsor=1"
+      }
+    },
+    "node_modules/swarm-js/node_modules/http2-wrapper": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/http2-wrapper/-/http2-wrapper-1.0.3.tgz",
+      "integrity": "sha512-V+23sDMr12Wnz7iTcDeJr3O6AIxlnvT/bmaAAAP/Xda35C90p9599p0F1eHR/N1KILWSoWVAiOMFjBBXaXSMxg==",
+      "dev": true,
+      "dependencies": {
+        "quick-lru": "^5.1.1",
+        "resolve-alpn": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=10.19.0"
+      }
+    },
+    "node_modules/swarm-js/node_modules/lowercase-keys": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz",
+      "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/swarm-js/node_modules/p-cancelable": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-2.1.1.tgz",
+      "integrity": "sha512-BZOr3nRQHOntUjTrH8+Lh54smKHoHyur8We1V8DSMVrl5A2malOOwuJRnKRDjSnkoeBh4at6BwEnb5I7Jl31wg==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/sync-request": {
+      "version": "6.1.0",
+      "resolved": "https://registry.npmjs.org/sync-request/-/sync-request-6.1.0.tgz",
+      "integrity": "sha512-8fjNkrNlNCrVc/av+Jn+xxqfCjYaBoHqCsDz6mt030UMxJGr+GSfCV1dQt2gRtlL63+VPidwDVLr7V2OcTSdRw==",
+      "dev": true,
+      "dependencies": {
+        "http-response-object": "^3.0.1",
+        "sync-rpc": "^1.2.1",
+        "then-request": "^6.0.0"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/sync-rpc": {
+      "version": "1.3.6",
+      "resolved": "https://registry.npmjs.org/sync-rpc/-/sync-rpc-1.3.6.tgz",
+      "integrity": "sha512-J8jTXuZzRlvU7HemDgHi3pGnh/rkoqR/OZSjhTyyZrEkkYQbk7Z33AXp37mkPfPpfdOuj7Ex3H/TJM1z48uPQw==",
+      "dev": true,
+      "dependencies": {
+        "get-port": "^3.1.0"
+      }
+    },
+    "node_modules/table": {
+      "version": "6.8.1",
+      "resolved": "https://registry.npmjs.org/table/-/table-6.8.1.tgz",
+      "integrity": "sha512-Y4X9zqrCftUhMeH2EptSSERdVKt/nEdijTOacGD/97EKjhQ/Qs8RTlEGABSJNNN8lac9kheH+af7yAkEWlgneA==",
+      "dev": true,
+      "dependencies": {
+        "ajv": "^8.0.1",
+        "lodash.truncate": "^4.4.2",
+        "slice-ansi": "^4.0.0",
+        "string-width": "^4.2.3",
+        "strip-ansi": "^6.0.1"
+      },
+      "engines": {
+        "node": ">=10.0.0"
+      }
+    },
+    "node_modules/table-layout": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/table-layout/-/table-layout-1.0.2.tgz",
+      "integrity": "sha512-qd/R7n5rQTRFi+Zf2sk5XVVd9UQl6ZkduPFC3S7WEGJAmetDTjY3qPN50eSKzwuzEyQKy5TN2TiZdkIjos2L6A==",
+      "dev": true,
+      "dependencies": {
+        "array-back": "^4.0.1",
+        "deep-extend": "~0.6.0",
+        "typical": "^5.2.0",
+        "wordwrapjs": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/table-layout/node_modules/array-back": {
+      "version": "4.0.2",
+      "resolved": "https://registry.npmjs.org/array-back/-/array-back-4.0.2.tgz",
+      "integrity": "sha512-NbdMezxqf94cnNfWLL7V/im0Ub+Anbb0IoZhvzie8+4HJ4nMQuzHuy49FkGYCJK2yAloZ3meiB6AVMClbrI1vg==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/table-layout/node_modules/typical": {
+      "version": "5.2.0",
+      "resolved": "https://registry.npmjs.org/typical/-/typical-5.2.0.tgz",
+      "integrity": "sha512-dvdQgNDNJo+8B2uBQoqdb11eUCE1JQXhvjC/CZtgvZseVd5TYMXnq0+vuUemXbd/Se29cTaUuPX3YIc2xgbvIg==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/tar": {
+      "version": "4.4.19",
+      "resolved": "https://registry.npmjs.org/tar/-/tar-4.4.19.tgz",
+      "integrity": "sha512-a20gEsvHnWe0ygBY8JbxoM4w3SJdhc7ZAuxkLqh+nvNQN2IOt0B5lLgM490X5Hl8FF0dl0tOf2ewFYAlIFgzVA==",
+      "dev": true,
+      "dependencies": {
+        "chownr": "^1.1.4",
+        "fs-minipass": "^1.2.7",
+        "minipass": "^2.9.0",
+        "minizlib": "^1.3.3",
+        "mkdirp": "^0.5.5",
+        "safe-buffer": "^5.2.1",
+        "yallist": "^3.1.1"
+      },
+      "engines": {
+        "node": ">=4.5"
+      }
+    },
+    "node_modules/tar/node_modules/mkdirp": {
+      "version": "0.5.6",
+      "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz",
+      "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==",
+      "dev": true,
+      "dependencies": {
+        "minimist": "^1.2.6"
+      },
+      "bin": {
+        "mkdirp": "bin/cmd.js"
+      }
+    },
+    "node_modules/tar/node_modules/yallist": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
+      "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==",
+      "dev": true
+    },
+    "node_modules/tenderly": {
+      "version": "0.4.0",
+      "resolved": "https://registry.npmjs.org/tenderly/-/tenderly-0.4.0.tgz",
+      "integrity": "sha512-wZgQ8Z1utc/QoAfVvMHO/ONLXJ3Vw3yjzJzpMmMUR4kvUu861mI7+mL9R1aeUuXI06cv81LZH96Vh+AOseIYoA==",
+      "dev": true,
+      "dependencies": {
+        "axios": "^0.27.2",
+        "cli-table3": "^0.6.2",
+        "commander": "^9.4.0",
+        "express": "^4.18.1",
+        "hyperlinker": "^1.0.0",
+        "js-yaml": "^4.1.0",
+        "open": "^8.4.0",
+        "prompts": "^2.4.2",
+        "tslog": "^4.4.0"
+      },
+      "peerDependencies": {
+        "ts-node": "*",
+        "typescript": "*"
+      },
+      "peerDependenciesMeta": {
+        "ts-node": {
+          "optional": true
+        },
+        "typescript": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/tenderly/node_modules/commander": {
+      "version": "9.5.0",
+      "resolved": "https://registry.npmjs.org/commander/-/commander-9.5.0.tgz",
+      "integrity": "sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==",
+      "dev": true,
+      "engines": {
+        "node": "^12.20.0 || >=14"
+      }
+    },
+    "node_modules/testrpc": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/testrpc/-/testrpc-0.0.1.tgz",
+      "integrity": "sha512-afH1hO+SQ/VPlmaLUFj2636QMeDvPCeQMc/9RBMW0IfjNe9gFD9Ra3ShqYkB7py0do1ZcCna/9acHyzTJ+GcNA==",
+      "deprecated": "testrpc has been renamed to ganache-cli, please use this package from now on.",
+      "dev": true
+    },
+    "node_modules/text-table": {
+      "version": "0.2.0",
+      "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz",
+      "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==",
+      "dev": true
+    },
+    "node_modules/then-request": {
+      "version": "6.0.2",
+      "resolved": "https://registry.npmjs.org/then-request/-/then-request-6.0.2.tgz",
+      "integrity": "sha512-3ZBiG7JvP3wbDzA9iNY5zJQcHL4jn/0BWtXIkagfz7QgOL/LqjCEOBQuJNZfu0XYnv5JhKh+cDxCPM4ILrqruA==",
+      "dev": true,
+      "dependencies": {
+        "@types/concat-stream": "^1.6.0",
+        "@types/form-data": "0.0.33",
+        "@types/node": "^8.0.0",
+        "@types/qs": "^6.2.31",
+        "caseless": "~0.12.0",
+        "concat-stream": "^1.6.0",
+        "form-data": "^2.2.0",
+        "http-basic": "^8.1.1",
+        "http-response-object": "^3.0.1",
+        "promise": "^8.0.0",
+        "qs": "^6.4.0"
+      },
+      "engines": {
+        "node": ">=6.0.0"
+      }
+    },
+    "node_modules/then-request/node_modules/@types/node": {
+      "version": "8.10.66",
+      "resolved": "https://registry.npmjs.org/@types/node/-/node-8.10.66.tgz",
+      "integrity": "sha512-tktOkFUA4kXx2hhhrB8bIFb5TbwzS4uOhKEmwiD+NoiL0qtP2OQ9mFldbgD4dV1djrlBYP6eBuQZiWjuHUpqFw==",
+      "dev": true
+    },
+    "node_modules/timed-out": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/timed-out/-/timed-out-4.0.1.tgz",
+      "integrity": "sha512-G7r3AhovYtr5YKOWQkta8RKAPb+J9IsO4uVmzjl8AZwfhs8UcUwTiD6gcJYSgOtzyjvQKrKYn41syHbUWMkafA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/title-case": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/title-case/-/title-case-2.1.1.tgz",
+      "integrity": "sha512-EkJoZ2O3zdCz3zJsYCsxyq2OC5hrxR9mfdd5I+w8h/tmFfeOxJ+vvkxsKxdmN0WtS9zLdHEgfgVOiMVgv+Po4Q==",
+      "dev": true,
+      "dependencies": {
+        "no-case": "^2.2.0",
+        "upper-case": "^1.0.3"
+      }
+    },
+    "node_modules/tmp": {
+      "version": "0.0.33",
+      "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz",
+      "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==",
+      "dev": true,
+      "dependencies": {
+        "os-tmpdir": "~1.0.2"
+      },
+      "engines": {
+        "node": ">=0.6.0"
+      }
+    },
+    "node_modules/to-regex-range": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
+      "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
+      "dev": true,
+      "dependencies": {
+        "is-number": "^7.0.0"
+      },
+      "engines": {
+        "node": ">=8.0"
+      }
+    },
+    "node_modules/toidentifier": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz",
+      "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.6"
+      }
+    },
+    "node_modules/tough-cookie": {
+      "version": "2.5.0",
+      "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.5.0.tgz",
+      "integrity": "sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==",
+      "dev": true,
+      "dependencies": {
+        "psl": "^1.1.28",
+        "punycode": "^2.1.1"
+      },
+      "engines": {
+        "node": ">=0.8"
+      }
+    },
+    "node_modules/tr46": {
+      "version": "0.0.3",
+      "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz",
+      "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==",
+      "dev": true
+    },
+    "node_modules/ts-command-line-args": {
+      "version": "2.4.2",
+      "resolved": "https://registry.npmjs.org/ts-command-line-args/-/ts-command-line-args-2.4.2.tgz",
+      "integrity": "sha512-mJLQQBOdyD4XI/ZWQY44PIdYde47JhV2xl380O7twPkTQ+Y5vFDHsk8LOeXKuz7dVY5aDCfAzRarNfSqtKOkQQ==",
+      "dev": true,
+      "dependencies": {
+        "@morgan-stanley/ts-mocking-bird": "^0.6.2",
+        "chalk": "^4.1.0",
+        "command-line-args": "^5.1.1",
+        "command-line-usage": "^6.1.0",
+        "string-format": "^2.0.0"
+      },
+      "bin": {
+        "write-markdown": "dist/write-markdown.js"
+      }
+    },
+    "node_modules/ts-essentials": {
+      "version": "7.0.3",
+      "resolved": "https://registry.npmjs.org/ts-essentials/-/ts-essentials-7.0.3.tgz",
+      "integrity": "sha512-8+gr5+lqO3G84KdiTSMRLtuyJ+nTBVRKuCrK4lidMPdVeEp0uqC875uE5NMcaA7YYMN7XsNiFQuMvasF8HT/xQ==",
+      "dev": true,
+      "peerDependencies": {
+        "typescript": ">=3.7.0"
+      }
+    },
+    "node_modules/ts-node": {
+      "version": "10.9.1",
+      "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.9.1.tgz",
+      "integrity": "sha512-NtVysVPkxxrwFGUUxGYhfux8k78pQB3JqYBXlLRZgdGUqTO5wU/UyHop5p70iEbGhB7q5KmiZiU0Y3KlJrScEw==",
+      "dev": true,
+      "dependencies": {
+        "@cspotcode/source-map-support": "^0.8.0",
+        "@tsconfig/node10": "^1.0.7",
+        "@tsconfig/node12": "^1.0.7",
+        "@tsconfig/node14": "^1.0.0",
+        "@tsconfig/node16": "^1.0.2",
+        "acorn": "^8.4.1",
+        "acorn-walk": "^8.1.1",
+        "arg": "^4.1.0",
+        "create-require": "^1.1.0",
+        "diff": "^4.0.1",
+        "make-error": "^1.1.1",
+        "v8-compile-cache-lib": "^3.0.1",
+        "yn": "3.1.1"
+      },
+      "bin": {
+        "ts-node": "dist/bin.js",
+        "ts-node-cwd": "dist/bin-cwd.js",
+        "ts-node-esm": "dist/bin-esm.js",
+        "ts-node-script": "dist/bin-script.js",
+        "ts-node-transpile-only": "dist/bin-transpile.js",
+        "ts-script": "dist/bin-script-deprecated.js"
+      },
+      "peerDependencies": {
+        "@swc/core": ">=1.2.50",
+        "@swc/wasm": ">=1.2.50",
+        "@types/node": "*",
+        "typescript": ">=2.7"
+      },
+      "peerDependenciesMeta": {
+        "@swc/core": {
+          "optional": true
+        },
+        "@swc/wasm": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/ts-node/node_modules/diff": {
+      "version": "4.0.2",
+      "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz",
+      "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.3.1"
+      }
+    },
+    "node_modules/tsconfig-paths": {
+      "version": "3.14.2",
+      "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.14.2.tgz",
+      "integrity": "sha512-o/9iXgCYc5L/JxCHPe3Hvh8Q/2xm5Z+p18PESBU6Ff33695QnCHBEjcytY2q19ua7Mbl/DavtBOLq+oG0RCL+g==",
+      "dev": true,
+      "dependencies": {
+        "@types/json5": "^0.0.29",
+        "json5": "^1.0.2",
+        "minimist": "^1.2.6",
+        "strip-bom": "^3.0.0"
+      }
+    },
+    "node_modules/tsconfig-paths/node_modules/strip-bom": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz",
+      "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/tslib": {
+      "version": "1.14.1",
+      "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz",
+      "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==",
+      "dev": true
+    },
+    "node_modules/tslog": {
+      "version": "4.8.2",
+      "resolved": "https://registry.npmjs.org/tslog/-/tslog-4.8.2.tgz",
+      "integrity": "sha512-eAKIRjxfSKYLs06r1wT7oou6Uv9VN6NW9g0JPidBlqQwPBBl5+84dm7r8zSOPVq1kyfEw1P6B3/FLSpZCorAgA==",
+      "dev": true,
+      "engines": {
+        "node": ">=16"
+      },
+      "funding": {
+        "url": "https://github.com/fullstack-build/tslog?sponsor=1"
+      }
+    },
+    "node_modules/tsort": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/tsort/-/tsort-0.0.1.tgz",
+      "integrity": "sha512-Tyrf5mxF8Ofs1tNoxA13lFeZ2Zrbd6cKbuH3V+MQ5sb6DtBj5FjrXVsRWT8YvNAQTqNoz66dz1WsbigI22aEnw==",
+      "dev": true
+    },
+    "node_modules/tsutils": {
+      "version": "3.21.0",
+      "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.21.0.tgz",
+      "integrity": "sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==",
+      "dev": true,
+      "dependencies": {
+        "tslib": "^1.8.1"
+      },
+      "engines": {
+        "node": ">= 6"
+      },
+      "peerDependencies": {
+        "typescript": ">=2.8.0 || >= 3.2.0-dev || >= 3.3.0-dev || >= 3.4.0-dev || >= 3.5.0-dev || >= 3.6.0-dev || >= 3.6.0-beta || >= 3.7.0-dev || >= 3.7.0-beta"
+      }
+    },
+    "node_modules/tunnel-agent": {
+      "version": "0.6.0",
+      "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz",
+      "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==",
+      "dev": true,
+      "dependencies": {
+        "safe-buffer": "^5.0.1"
+      },
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/tweetnacl": {
+      "version": "0.14.5",
+      "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz",
+      "integrity": "sha512-KXXFFdAbFXY4geFIwoyNK+f5Z1b7swfXABfL7HXCmoIWMKU3dmS26672A4EeQtDzLKy7SXmfBu51JolvEKwtGA==",
+      "dev": true
+    },
+    "node_modules/tweetnacl-util": {
+      "version": "0.15.1",
+      "resolved": "https://registry.npmjs.org/tweetnacl-util/-/tweetnacl-util-0.15.1.tgz",
+      "integrity": "sha512-RKJBIj8lySrShN4w6i/BonWp2Z/uxwC3h4y7xsRrpP59ZboCd0GpEVsOnMDYLMmKBpYhb5TgHzZXy7wTfYFBRw==",
+      "dev": true
+    },
+    "node_modules/type": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/type/-/type-1.2.0.tgz",
+      "integrity": "sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg==",
+      "dev": true
+    },
+    "node_modules/type-check": {
+      "version": "0.4.0",
+      "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz",
+      "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==",
+      "dev": true,
+      "dependencies": {
+        "prelude-ls": "^1.2.1"
+      },
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/type-detect": {
+      "version": "4.0.8",
+      "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz",
+      "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/type-fest": {
+      "version": "0.21.3",
+      "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.21.3.tgz",
+      "integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/type-is": {
+      "version": "1.6.18",
+      "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
+      "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
+      "dev": true,
+      "dependencies": {
+        "media-typer": "0.3.0",
+        "mime-types": "~2.1.24"
+      },
+      "engines": {
+        "node": ">= 0.6"
+      }
+    },
+    "node_modules/typechain": {
+      "version": "8.1.1",
+      "resolved": "https://registry.npmjs.org/typechain/-/typechain-8.1.1.tgz",
+      "integrity": "sha512-uF/sUvnXTOVF2FHKhQYnxHk4su4JjZR8vr4mA2mBaRwHTbwh0jIlqARz9XJr1tA0l7afJGvEa1dTSi4zt039LQ==",
+      "dev": true,
+      "dependencies": {
+        "@types/prettier": "^2.1.1",
+        "debug": "^4.3.1",
+        "fs-extra": "^7.0.0",
+        "glob": "7.1.7",
+        "js-sha3": "^0.8.0",
+        "lodash": "^4.17.15",
+        "mkdirp": "^1.0.4",
+        "prettier": "^2.3.1",
+        "ts-command-line-args": "^2.2.0",
+        "ts-essentials": "^7.0.1"
+      },
+      "bin": {
+        "typechain": "dist/cli/cli.js"
+      },
+      "peerDependencies": {
+        "typescript": ">=4.3.0"
+      }
+    },
+    "node_modules/typechain/node_modules/glob": {
+      "version": "7.1.7",
+      "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.7.tgz",
+      "integrity": "sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==",
+      "dev": true,
+      "dependencies": {
+        "fs.realpath": "^1.0.0",
+        "inflight": "^1.0.4",
+        "inherits": "2",
+        "minimatch": "^3.0.4",
+        "once": "^1.3.0",
+        "path-is-absolute": "^1.0.0"
+      },
+      "engines": {
+        "node": "*"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/isaacs"
+      }
+    },
+    "node_modules/typechain/node_modules/mkdirp": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz",
+      "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==",
+      "dev": true,
+      "bin": {
+        "mkdirp": "bin/cmd.js"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/typed-array-length": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.4.tgz",
+      "integrity": "sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "for-each": "^0.3.3",
+        "is-typed-array": "^1.1.9"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/typedarray": {
+      "version": "0.0.6",
+      "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz",
+      "integrity": "sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==",
+      "dev": true
+    },
+    "node_modules/typedarray-to-buffer": {
+      "version": "3.1.5",
+      "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz",
+      "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==",
+      "dev": true,
+      "dependencies": {
+        "is-typedarray": "^1.0.0"
+      }
+    },
+    "node_modules/typescript": {
+      "version": "4.9.5",
+      "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.9.5.tgz",
+      "integrity": "sha512-1FXk9E2Hm+QzZQ7z+McJiHL4NW1F2EzMu9Nq9i3zAaGqibafqYwCVU6WyWAuyQRRzOlxou8xZSyXLEN8oKj24g==",
+      "dev": true,
+      "bin": {
+        "tsc": "bin/tsc",
+        "tsserver": "bin/tsserver"
+      },
+      "engines": {
+        "node": ">=4.2.0"
+      }
+    },
+    "node_modules/typical": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/typical/-/typical-4.0.0.tgz",
+      "integrity": "sha512-VAH4IvQ7BDFYglMd7BPRDfLgxZZX4O4TFcRDA6EN5X7erNJJq+McIEp8np9aVtxrCJ6qx4GTYVfOWNjcqwZgRw==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/uglify-js": {
+      "version": "3.17.4",
+      "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.17.4.tgz",
+      "integrity": "sha512-T9q82TJI9e/C1TAxYvfb16xO120tMVFZrGA3f9/P4424DNu6ypK103y0GPFVa17yotwSyZW5iYXgjYHkGrJW/g==",
+      "dev": true,
+      "optional": true,
+      "bin": {
+        "uglifyjs": "bin/uglifyjs"
+      },
+      "engines": {
+        "node": ">=0.8.0"
+      }
+    },
+    "node_modules/ultron": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/ultron/-/ultron-1.1.1.tgz",
+      "integrity": "sha512-UIEXBNeYmKptWH6z8ZnqTeS8fV74zG0/eRU9VGkpzz+LIJNs8W/zM/L+7ctCkRrgbNnnR0xxw4bKOr0cW0N0Og==",
+      "dev": true
+    },
+    "node_modules/unbox-primitive": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.2.tgz",
+      "integrity": "sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==",
+      "dev": true,
+      "dependencies": {
+        "call-bind": "^1.0.2",
+        "has-bigints": "^1.0.2",
+        "has-symbols": "^1.0.3",
+        "which-boxed-primitive": "^1.0.2"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/underscore": {
+      "version": "1.13.6",
+      "resolved": "https://registry.npmjs.org/underscore/-/underscore-1.13.6.tgz",
+      "integrity": "sha512-+A5Sja4HP1M08MaXya7p5LvjuM7K6q/2EaC0+iovj/wOcMsTzMvDFbasi/oSapiwOlt252IqsKqPjCl7huKS0A==",
+      "dev": true
+    },
+    "node_modules/undici": {
+      "version": "5.27.2",
+      "resolved": "https://registry.npmjs.org/undici/-/undici-5.27.2.tgz",
+      "integrity": "sha512-iS857PdOEy/y3wlM3yRp+6SNQQ6xU0mmZcwRSriqk+et/cwWAtwmIGf6WkoDN2EK/AMdCO/dfXzIwi+rFMrjjQ==",
+      "dev": true,
+      "dependencies": {
+        "@fastify/busboy": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=14.0"
+      }
+    },
+    "node_modules/universalify": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz",
+      "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==",
+      "dev": true,
+      "engines": {
+        "node": ">= 4.0.0"
+      }
+    },
+    "node_modules/unpipe": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
+      "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/upper-case": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/upper-case/-/upper-case-1.1.3.tgz",
+      "integrity": "sha512-WRbjgmYzgXkCV7zNVpy5YgrHgbBv126rMALQQMrmzOVC4GM2waQ9x7xtm8VU+1yF2kWyPzI9zbZ48n4vSxwfSA==",
+      "dev": true
+    },
+    "node_modules/upper-case-first": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/upper-case-first/-/upper-case-first-1.1.2.tgz",
+      "integrity": "sha512-wINKYvI3Db8dtjikdAqoBbZoP6Q+PZUyfMR7pmwHzjC2quzSkUq5DmPrTtPEqHaz8AGtmsB4TqwapMTM1QAQOQ==",
+      "dev": true,
+      "dependencies": {
+        "upper-case": "^1.1.1"
+      }
+    },
+    "node_modules/uri-js": {
+      "version": "4.4.1",
+      "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz",
+      "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==",
+      "dev": true,
+      "dependencies": {
+        "punycode": "^2.1.0"
+      }
+    },
+    "node_modules/url-set-query": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/url-set-query/-/url-set-query-1.0.0.tgz",
+      "integrity": "sha512-3AChu4NiXquPfeckE5R5cGdiHCMWJx1dwCWOmWIL4KHAziJNOFIYJlpGFeKDvwLPHovZRCxK3cYlwzqI9Vp+Gg==",
+      "dev": true
+    },
+    "node_modules/utf-8-validate": {
+      "version": "5.0.10",
+      "resolved": "https://registry.npmjs.org/utf-8-validate/-/utf-8-validate-5.0.10.tgz",
+      "integrity": "sha512-Z6czzLq4u8fPOyx7TU6X3dvUZVvoJmxSQ+IcrlmagKhilxlhZgxPK6C5Jqbkw1IDUmFTM+cz9QDnnLTwDz/2gQ==",
+      "dev": true,
+      "hasInstallScript": true,
+      "dependencies": {
+        "node-gyp-build": "^4.3.0"
+      },
+      "engines": {
+        "node": ">=6.14.2"
+      }
+    },
+    "node_modules/utf8": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/utf8/-/utf8-3.0.0.tgz",
+      "integrity": "sha512-E8VjFIQ/TyQgp+TZfS6l8yp/xWppSAHzidGiRrqe4bK4XP9pTRyKFgGJpO3SN7zdX4DeomTrwaseCHovfpFcqQ==",
+      "dev": true
+    },
+    "node_modules/util": {
+      "version": "0.12.5",
+      "resolved": "https://registry.npmjs.org/util/-/util-0.12.5.tgz",
+      "integrity": "sha512-kZf/K6hEIrWHI6XqOFUiiMa+79wE/D8Q+NCNAWclkyg3b4d2k7s0QGepNjiABc+aR3N1PAyHL7p6UcLY6LmrnA==",
+      "dev": true,
+      "dependencies": {
+        "inherits": "^2.0.3",
+        "is-arguments": "^1.0.4",
+        "is-generator-function": "^1.0.7",
+        "is-typed-array": "^1.1.3",
+        "which-typed-array": "^1.1.2"
+      }
+    },
+    "node_modules/util-deprecate": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
+      "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==",
+      "dev": true
+    },
+    "node_modules/utils-merge": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
+      "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.4.0"
+      }
+    },
+    "node_modules/uuid": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/uuid/-/uuid-2.0.1.tgz",
+      "integrity": "sha512-nWg9+Oa3qD2CQzHIP4qKUqwNfzKn8P0LtFhotaCTFchsV7ZfDhAybeip/HZVeMIpZi9JgY1E3nUlwaCmZT1sEg==",
+      "deprecated": "Please upgrade  to version 7 or higher.  Older versions may use Math.random() in certain circumstances, which is known to be problematic.  See https://v8.dev/blog/math-random for details.",
+      "dev": true
+    },
+    "node_modules/v8-compile-cache-lib": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz",
+      "integrity": "sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==",
+      "dev": true
+    },
+    "node_modules/validate-npm-package-license": {
+      "version": "3.0.4",
+      "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz",
+      "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==",
+      "dev": true,
+      "dependencies": {
+        "spdx-correct": "^3.0.0",
+        "spdx-expression-parse": "^3.0.0"
+      }
+    },
+    "node_modules/varint": {
+      "version": "5.0.2",
+      "resolved": "https://registry.npmjs.org/varint/-/varint-5.0.2.tgz",
+      "integrity": "sha512-lKxKYG6H03yCZUpAGOPOsMcGxd1RHCu1iKvEHYDPmTyq2HueGhD73ssNBqqQWfvYs04G9iUFRvmAVLW20Jw6ow==",
+      "dev": true
+    },
+    "node_modules/vary": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
+      "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==",
+      "dev": true,
+      "engines": {
+        "node": ">= 0.8"
+      }
+    },
+    "node_modules/verror": {
+      "version": "1.10.0",
+      "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz",
+      "integrity": "sha512-ZZKSmDAEFOijERBLkmYfJ+vmk3w+7hOLYDNkRCuRuMJGEmqYNCNLyBBFwWKVMhfwaEF3WOd0Zlw86U/WC/+nYw==",
+      "dev": true,
+      "engines": [
+        "node >=0.6.0"
+      ],
+      "dependencies": {
+        "assert-plus": "^1.0.0",
+        "core-util-is": "1.0.2",
+        "extsprintf": "^1.2.0"
+      }
+    },
+    "node_modules/wcwidth": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/wcwidth/-/wcwidth-1.0.1.tgz",
+      "integrity": "sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg==",
+      "dev": true,
+      "dependencies": {
+        "defaults": "^1.0.3"
+      }
+    },
+    "node_modules/web3": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3/-/web3-1.8.2.tgz",
+      "integrity": "sha512-92h0GdEHW9wqDICQQKyG4foZBYi0OQkyg4CRml2F7XBl/NG+fu9o6J19kzfFXzSBoA4DnJXbyRgj/RHZv5LRiw==",
+      "dev": true,
+      "hasInstallScript": true,
+      "dependencies": {
+        "web3-bzz": "1.8.2",
+        "web3-core": "1.8.2",
+        "web3-eth": "1.8.2",
+        "web3-eth-personal": "1.8.2",
+        "web3-net": "1.8.2",
+        "web3-shh": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-bzz": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-bzz/-/web3-bzz-1.8.2.tgz",
+      "integrity": "sha512-1EEnxjPnFnvNWw3XeeKuTR8PBxYd0+XWzvaLK7OJC/Go9O8llLGxrxICbKV+8cgIE0sDRBxiYx02X+6OhoAQ9w==",
+      "dev": true,
+      "hasInstallScript": true,
+      "dependencies": {
+        "@types/node": "^12.12.6",
+        "got": "12.1.0",
+        "swarm-js": "^0.1.40"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-bzz/node_modules/@types/node": {
+      "version": "12.20.55",
+      "resolved": "https://registry.npmjs.org/@types/node/-/node-12.20.55.tgz",
+      "integrity": "sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ==",
+      "dev": true
+    },
+    "node_modules/web3-core": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-core/-/web3-core-1.8.2.tgz",
+      "integrity": "sha512-DJTVEAYcNqxkqruJE+Rxp3CIv0y5AZMwPHQmOkz/cz+MM75SIzMTc0AUdXzGyTS8xMF8h3YWMQGgGEy8SBf1PQ==",
+      "dev": true,
+      "dependencies": {
+        "@types/bn.js": "^5.1.0",
+        "@types/node": "^12.12.6",
+        "bignumber.js": "^9.0.0",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-core-requestmanager": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-core-helpers": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-core-helpers/-/web3-core-helpers-1.8.2.tgz",
+      "integrity": "sha512-6B1eLlq9JFrfealZBomd1fmlq1o4A09vrCVQSa51ANoib/jllT3atZrRDr0zt1rfI7TSZTZBXdN/aTdeN99DWw==",
+      "dev": true,
+      "dependencies": {
+        "web3-eth-iban": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-core-method": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-core-method/-/web3-core-method-1.8.2.tgz",
+      "integrity": "sha512-1qnr5mw5wVyULzLOrk4B+ryO3gfGjGd/fx8NR+J2xCGLf1e6OSjxT9vbfuQ3fErk/NjSTWWreieYWLMhaogcRA==",
+      "dev": true,
+      "dependencies": {
+        "@ethersproject/transactions": "^5.6.2",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-promievent": "1.8.2",
+        "web3-core-subscriptions": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-core-promievent": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-core-promievent/-/web3-core-promievent-1.8.2.tgz",
+      "integrity": "sha512-nvkJWDVgoOSsolJldN33tKW6bKKRJX3MCPDYMwP5SUFOA/mCzDEoI88N0JFofDTXkh1k7gOqp1pvwi9heuaxGg==",
+      "dev": true,
+      "dependencies": {
+        "eventemitter3": "4.0.4"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-core-requestmanager": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-core-requestmanager/-/web3-core-requestmanager-1.8.2.tgz",
+      "integrity": "sha512-p1d090RYs5Mu7DK1yyc3GCBVZB/03rBtFhYFoS2EruGzOWs/5Q0grgtpwS/DScdRAm8wB8mYEBhY/RKJWF6B2g==",
+      "dev": true,
+      "dependencies": {
+        "util": "^0.12.5",
+        "web3-core-helpers": "1.8.2",
+        "web3-providers-http": "1.8.2",
+        "web3-providers-ipc": "1.8.2",
+        "web3-providers-ws": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-core-subscriptions": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-core-subscriptions/-/web3-core-subscriptions-1.8.2.tgz",
+      "integrity": "sha512-vXQogHDmAIQcKpXvGiMddBUeP9lnKgYF64+yQJhPNE5PnWr1sAibXuIPV7mIPihpFr/n/DORRj6Wh1pUv9zaTw==",
+      "dev": true,
+      "dependencies": {
+        "eventemitter3": "4.0.4",
+        "web3-core-helpers": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-core/node_modules/@types/node": {
+      "version": "12.20.55",
+      "resolved": "https://registry.npmjs.org/@types/node/-/node-12.20.55.tgz",
+      "integrity": "sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ==",
+      "dev": true
+    },
+    "node_modules/web3-core/node_modules/bignumber.js": {
+      "version": "9.1.1",
+      "resolved": "https://registry.npmjs.org/bignumber.js/-/bignumber.js-9.1.1.tgz",
+      "integrity": "sha512-pHm4LsMJ6lzgNGVfZHjMoO8sdoRhOzOH4MLmY65Jg70bpxCKu5iOHNJyfF6OyvYw7t8Fpf35RuzUyqnQsj8Vig==",
+      "dev": true,
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/web3-eth": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth/-/web3-eth-1.8.2.tgz",
+      "integrity": "sha512-JoTiWWc4F4TInpbvDUGb0WgDYJsFhuIjJlinc5ByjWD88Gvh+GKLsRjjFdbqe5YtwIGT4NymwoC5LQd1K6u/QQ==",
+      "dev": true,
+      "dependencies": {
+        "web3-core": "1.8.2",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-core-subscriptions": "1.8.2",
+        "web3-eth-abi": "1.8.2",
+        "web3-eth-accounts": "1.8.2",
+        "web3-eth-contract": "1.8.2",
+        "web3-eth-ens": "1.8.2",
+        "web3-eth-iban": "1.8.2",
+        "web3-eth-personal": "1.8.2",
+        "web3-net": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-eth-abi": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth-abi/-/web3-eth-abi-1.8.2.tgz",
+      "integrity": "sha512-Om9g3kaRNjqiNPAgKwGT16y+ZwtBzRe4ZJFGjLiSs6v5I7TPNF+rRMWuKnR6jq0azQZDj6rblvKFMA49/k48Og==",
+      "dev": true,
+      "dependencies": {
+        "@ethersproject/abi": "^5.6.3",
+        "web3-utils": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-eth-accounts": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth-accounts/-/web3-eth-accounts-1.8.2.tgz",
+      "integrity": "sha512-c367Ij63VCz9YdyjiHHWLFtN85l6QghgwMQH2B1eM/p9Y5lTlTX7t/Eg/8+f1yoIStXbk2w/PYM2lk+IkbqdLA==",
+      "dev": true,
+      "dependencies": {
+        "@ethereumjs/common": "2.5.0",
+        "@ethereumjs/tx": "3.3.2",
+        "eth-lib": "0.2.8",
+        "ethereumjs-util": "^7.1.5",
+        "scrypt-js": "^3.0.1",
+        "uuid": "^9.0.0",
+        "web3-core": "1.8.2",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-eth-accounts/node_modules/bn.js": {
+      "version": "4.12.0",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+      "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+      "dev": true
+    },
+    "node_modules/web3-eth-accounts/node_modules/eth-lib": {
+      "version": "0.2.8",
+      "resolved": "https://registry.npmjs.org/eth-lib/-/eth-lib-0.2.8.tgz",
+      "integrity": "sha512-ArJ7x1WcWOlSpzdoTBX8vkwlkSQ85CjjifSZtV4co64vWxSV8geWfPI9x4SVYu3DSxnX4yWFVTtGL+j9DUFLNw==",
+      "dev": true,
+      "dependencies": {
+        "bn.js": "^4.11.6",
+        "elliptic": "^6.4.0",
+        "xhr-request-promise": "^0.1.2"
+      }
+    },
+    "node_modules/web3-eth-accounts/node_modules/uuid": {
+      "version": "9.0.0",
+      "resolved": "https://registry.npmjs.org/uuid/-/uuid-9.0.0.tgz",
+      "integrity": "sha512-MXcSTerfPa4uqyzStbRoTgt5XIe3x5+42+q1sDuy3R5MDk66URdLMOZe5aPX/SQd+kuYAh0FdP/pO28IkQyTeg==",
+      "dev": true,
+      "bin": {
+        "uuid": "dist/bin/uuid"
+      }
+    },
+    "node_modules/web3-eth-contract": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth-contract/-/web3-eth-contract-1.8.2.tgz",
+      "integrity": "sha512-ID5A25tHTSBNwOPjiXSVzxruz006ULRIDbzWTYIFTp7NJ7vXu/kynKK2ag/ObuTqBpMbobP8nXcA9b5EDkIdQA==",
+      "dev": true,
+      "dependencies": {
+        "@types/bn.js": "^5.1.0",
+        "web3-core": "1.8.2",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-core-promievent": "1.8.2",
+        "web3-core-subscriptions": "1.8.2",
+        "web3-eth-abi": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-eth-ens": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth-ens/-/web3-eth-ens-1.8.2.tgz",
+      "integrity": "sha512-PWph7C/CnqdWuu1+SH4U4zdrK4t2HNt0I4XzPYFdv9ugE8EuojselioPQXsVGvjql+Nt3jDLvQvggPqlMbvwRw==",
+      "dev": true,
+      "dependencies": {
+        "content-hash": "^2.5.2",
+        "eth-ens-namehash": "2.0.8",
+        "web3-core": "1.8.2",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-promievent": "1.8.2",
+        "web3-eth-abi": "1.8.2",
+        "web3-eth-contract": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-eth-iban": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth-iban/-/web3-eth-iban-1.8.2.tgz",
+      "integrity": "sha512-h3vNblDWkWMuYx93Q27TAJz6lhzpP93EiC3+45D6xoz983p6si773vntoQ+H+5aZhwglBtoiBzdh7PSSOnP/xQ==",
+      "dev": true,
+      "dependencies": {
+        "bn.js": "^5.2.1",
+        "web3-utils": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-eth-personal": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth-personal/-/web3-eth-personal-1.8.2.tgz",
+      "integrity": "sha512-Vg4HfwCr7doiUF/RC+Jz0wT4+cYaXcOWMAW2AHIjHX6Z7Xwa8nrURIeQgeEE62qcEHAzajyAdB1u6bJyTfuCXw==",
+      "dev": true,
+      "dependencies": {
+        "@types/node": "^12.12.6",
+        "web3-core": "1.8.2",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-net": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-eth-personal/node_modules/@types/node": {
+      "version": "12.20.55",
+      "resolved": "https://registry.npmjs.org/@types/node/-/node-12.20.55.tgz",
+      "integrity": "sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ==",
+      "dev": true
+    },
+    "node_modules/web3-net": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-net/-/web3-net-1.8.2.tgz",
+      "integrity": "sha512-1itkDMGmbgb83Dg9nporFes9/fxsU7smJ3oRXlFkg4ZHn8YJyP1MSQFPJWWwSc+GrcCFt4O5IrUTvEkHqE3xag==",
+      "dev": true,
+      "dependencies": {
+        "web3-core": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-providers-http": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-providers-http/-/web3-providers-http-1.8.2.tgz",
+      "integrity": "sha512-2xY94IIEQd16+b+vIBF4IC1p7GVaz9q4EUFscvMUjtEq4ru4Atdzjs9GP+jmcoo49p70II0UV3bqQcz0TQfVyQ==",
+      "dev": true,
+      "dependencies": {
+        "abortcontroller-polyfill": "^1.7.3",
+        "cross-fetch": "^3.1.4",
+        "es6-promise": "^4.2.8",
+        "web3-core-helpers": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-providers-ipc": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-providers-ipc/-/web3-providers-ipc-1.8.2.tgz",
+      "integrity": "sha512-p6fqKVGFg+WiXGHWnB1hu43PbvPkDHTz4RgoEzbXugv5rtv5zfYLqm8Ba6lrJOS5ks9kGKR21a0y3NzE3u7V4w==",
+      "dev": true,
+      "dependencies": {
+        "oboe": "2.1.5",
+        "web3-core-helpers": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-providers-ws": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-providers-ws/-/web3-providers-ws-1.8.2.tgz",
+      "integrity": "sha512-3s/4K+wHgbiN+Zrp9YjMq2eqAF6QGABw7wFftPdx+m5hWImV27/MoIx57c6HffNRqZXmCHnfWWFCNHHsi7wXnA==",
+      "dev": true,
+      "dependencies": {
+        "eventemitter3": "4.0.4",
+        "web3-core-helpers": "1.8.2",
+        "websocket": "^1.0.32"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-shh": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-shh/-/web3-shh-1.8.2.tgz",
+      "integrity": "sha512-uZ+3MAoNcaJsXXNCDnizKJ5viBNeHOFYsCbFhV755Uu52FswzTOw6DtE7yK9nYXMtIhiSgi7nwl1RYzP8pystw==",
+      "dev": true,
+      "hasInstallScript": true,
+      "dependencies": {
+        "web3-core": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-core-subscriptions": "1.8.2",
+        "web3-net": "1.8.2"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/web3-utils": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-utils/-/web3-utils-1.8.2.tgz",
+      "integrity": "sha512-v7j6xhfLQfY7xQDrUP0BKbaNrmZ2/+egbqP9q3KYmOiPpnvAfol+32slgL0WX/5n8VPvKCK5EZ1HGrAVICSToA==",
+      "dev": true,
+      "dependencies": {
+        "bn.js": "^5.2.1",
+        "ethereum-bloom-filters": "^1.0.6",
+        "ethereumjs-util": "^7.1.0",
+        "ethjs-unit": "0.1.6",
+        "number-to-bn": "1.7.0",
+        "randombytes": "^2.1.0",
+        "utf8": "3.0.0"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/webidl-conversions": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz",
+      "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==",
+      "dev": true
+    },
+    "node_modules/websocket": {
+      "version": "1.0.34",
+      "resolved": "https://registry.npmjs.org/websocket/-/websocket-1.0.34.tgz",
+      "integrity": "sha512-PRDso2sGwF6kM75QykIesBijKSVceR6jL2G8NGYyq2XrItNC2P5/qL5XeR056GhA+Ly7JMFvJb9I312mJfmqnQ==",
+      "dev": true,
+      "dependencies": {
+        "bufferutil": "^4.0.1",
+        "debug": "^2.2.0",
+        "es5-ext": "^0.10.50",
+        "typedarray-to-buffer": "^3.1.5",
+        "utf-8-validate": "^5.0.2",
+        "yaeti": "^0.0.6"
+      },
+      "engines": {
+        "node": ">=4.0.0"
+      }
+    },
+    "node_modules/websocket/node_modules/debug": {
+      "version": "2.6.9",
+      "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+      "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+      "dev": true,
+      "dependencies": {
+        "ms": "2.0.0"
+      }
+    },
+    "node_modules/websocket/node_modules/ms": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+      "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
+      "dev": true
+    },
+    "node_modules/whatwg-url": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz",
+      "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==",
+      "dev": true,
+      "dependencies": {
+        "tr46": "~0.0.3",
+        "webidl-conversions": "^3.0.0"
+      }
+    },
+    "node_modules/which": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
+      "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==",
+      "dev": true,
+      "dependencies": {
+        "isexe": "^2.0.0"
+      },
+      "bin": {
+        "node-which": "bin/node-which"
+      },
+      "engines": {
+        "node": ">= 8"
+      }
+    },
+    "node_modules/which-boxed-primitive": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz",
+      "integrity": "sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==",
+      "dev": true,
+      "dependencies": {
+        "is-bigint": "^1.0.1",
+        "is-boolean-object": "^1.1.0",
+        "is-number-object": "^1.0.4",
+        "is-string": "^1.0.5",
+        "is-symbol": "^1.0.3"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/which-module": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/which-module/-/which-module-1.0.0.tgz",
+      "integrity": "sha512-F6+WgncZi/mJDrammbTuHe1q0R5hOXv/mBaiNA2TCNT/LTHusX0V+CJnj9XT8ki5ln2UZyyddDgHfCzyrOH7MQ==",
+      "dev": true
+    },
+    "node_modules/which-pm-runs": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/which-pm-runs/-/which-pm-runs-1.1.0.tgz",
+      "integrity": "sha512-n1brCuqClxfFfq/Rb0ICg9giSZqCS+pLtccdag6C2HyufBrh3fBOiy9nb6ggRMvWOVH5GrdJskj5iGTZNxd7SA==",
+      "dev": true,
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/which-typed-array": {
+      "version": "1.1.9",
+      "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.9.tgz",
+      "integrity": "sha512-w9c4xkx6mPidwp7180ckYWfMmvxpjlZuIudNtDf4N/tTAUB8VJbX25qZoAsrtGuYNnGw3pa0AXgbGKRB8/EceA==",
+      "dev": true,
+      "dependencies": {
+        "available-typed-arrays": "^1.0.5",
+        "call-bind": "^1.0.2",
+        "for-each": "^0.3.3",
+        "gopd": "^1.0.1",
+        "has-tostringtag": "^1.0.0",
+        "is-typed-array": "^1.1.10"
+      },
+      "engines": {
+        "node": ">= 0.4"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/ljharb"
+      }
+    },
+    "node_modules/window-size": {
+      "version": "0.2.0",
+      "resolved": "https://registry.npmjs.org/window-size/-/window-size-0.2.0.tgz",
+      "integrity": "sha512-UD7d8HFA2+PZsbKyaOCEy8gMh1oDtHgJh1LfgjQ4zVXmYjAT/kvz3PueITKuqDiIXQe7yzpPnxX3lNc+AhQMyw==",
+      "dev": true,
+      "bin": {
+        "window-size": "cli.js"
+      },
+      "engines": {
+        "node": ">= 0.10.0"
+      }
+    },
+    "node_modules/word-wrap": {
+      "version": "1.2.5",
+      "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz",
+      "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/wordwrap": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz",
+      "integrity": "sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==",
+      "dev": true
+    },
+    "node_modules/wordwrapjs": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/wordwrapjs/-/wordwrapjs-4.0.1.tgz",
+      "integrity": "sha512-kKlNACbvHrkpIw6oPeYDSmdCTu2hdMHoyXLTcUKala++lx5Y+wjJ/e474Jqv5abnVmwxw08DiTuHmw69lJGksA==",
+      "dev": true,
+      "dependencies": {
+        "reduce-flatten": "^2.0.0",
+        "typical": "^5.2.0"
+      },
+      "engines": {
+        "node": ">=8.0.0"
+      }
+    },
+    "node_modules/wordwrapjs/node_modules/typical": {
+      "version": "5.2.0",
+      "resolved": "https://registry.npmjs.org/typical/-/typical-5.2.0.tgz",
+      "integrity": "sha512-dvdQgNDNJo+8B2uBQoqdb11eUCE1JQXhvjC/CZtgvZseVd5TYMXnq0+vuUemXbd/Se29cTaUuPX3YIc2xgbvIg==",
+      "dev": true,
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/workerpool": {
+      "version": "6.2.1",
+      "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.2.1.tgz",
+      "integrity": "sha512-ILEIE97kDZvF9Wb9f6h5aXK4swSlKGUcOEGiIYb2OOu/IrDU9iwj0fD//SsA6E5ibwJxpEvhullJY4Sl4GcpAw==",
+      "dev": true
+    },
+    "node_modules/wrap-ansi": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz",
+      "integrity": "sha512-vAaEaDM946gbNpH5pLVNR+vX2ht6n0Bt3GXwVB1AuAqZosOvHNF3P7wDnh8KLkSqgUh0uh77le7Owgoz+Z9XBw==",
+      "dev": true,
+      "dependencies": {
+        "string-width": "^1.0.1",
+        "strip-ansi": "^3.0.1"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/wrap-ansi/node_modules/ansi-regex": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
+      "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/wrap-ansi/node_modules/is-fullwidth-code-point": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz",
+      "integrity": "sha512-1pqUqRjkhPJ9miNq9SwMfdvi6lBJcd6eFxvfaivQhaH3SgisfiuudvFntdKOmxuee/77l+FPjKrQjWvmPjWrRw==",
+      "dev": true,
+      "dependencies": {
+        "number-is-nan": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/wrap-ansi/node_modules/string-width": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz",
+      "integrity": "sha512-0XsVpQLnVCXHJfyEs8tC0zpTVIr5PKKsQtkT29IwupnPTjtPmQ3xT/4yCREF9hYkV/3M3kzcUTSAZT6a6h81tw==",
+      "dev": true,
+      "dependencies": {
+        "code-point-at": "^1.0.0",
+        "is-fullwidth-code-point": "^1.0.0",
+        "strip-ansi": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/wrap-ansi/node_modules/strip-ansi": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
+      "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==",
+      "dev": true,
+      "dependencies": {
+        "ansi-regex": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/wrappy": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
+      "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==",
+      "dev": true
+    },
+    "node_modules/ws": {
+      "version": "7.4.6",
+      "resolved": "https://registry.npmjs.org/ws/-/ws-7.4.6.tgz",
+      "integrity": "sha512-YmhHDO4MzaDLB+M9ym/mDA5z0naX8j7SIlT8f8z+I0VtzsRbekxEutHSme7NPS2qE8StCYQNUnfWdXta/Yu85A==",
+      "devOptional": true,
+      "engines": {
+        "node": ">=8.3.0"
+      },
+      "peerDependencies": {
+        "bufferutil": "^4.0.1",
+        "utf-8-validate": "^5.0.2"
+      },
+      "peerDependenciesMeta": {
+        "bufferutil": {
+          "optional": true
+        },
+        "utf-8-validate": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/xhr": {
+      "version": "2.6.0",
+      "resolved": "https://registry.npmjs.org/xhr/-/xhr-2.6.0.tgz",
+      "integrity": "sha512-/eCGLb5rxjx5e3mF1A7s+pLlR6CGyqWN91fv1JgER5mVWg1MZmlhBvy9kjcsOdRk8RrIujotWyJamfyrp+WIcA==",
+      "dev": true,
+      "dependencies": {
+        "global": "~4.4.0",
+        "is-function": "^1.0.1",
+        "parse-headers": "^2.0.0",
+        "xtend": "^4.0.0"
+      }
+    },
+    "node_modules/xhr-request": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/xhr-request/-/xhr-request-1.1.0.tgz",
+      "integrity": "sha512-Y7qzEaR3FDtL3fP30k9wO/e+FBnBByZeybKOhASsGP30NIkRAAkKD/sCnLvgEfAIEC1rcmK7YG8f4oEnIrrWzA==",
+      "dev": true,
+      "dependencies": {
+        "buffer-to-arraybuffer": "^0.0.5",
+        "object-assign": "^4.1.1",
+        "query-string": "^5.0.1",
+        "simple-get": "^2.7.0",
+        "timed-out": "^4.0.1",
+        "url-set-query": "^1.0.0",
+        "xhr": "^2.0.4"
+      }
+    },
+    "node_modules/xhr-request-promise": {
+      "version": "0.1.3",
+      "resolved": "https://registry.npmjs.org/xhr-request-promise/-/xhr-request-promise-0.1.3.tgz",
+      "integrity": "sha512-YUBytBsuwgitWtdRzXDDkWAXzhdGB8bYm0sSzMPZT7Z2MBjMSTHFsyCT1yCRATY+XC69DUrQraRAEgcoCRaIPg==",
+      "dev": true,
+      "dependencies": {
+        "xhr-request": "^1.1.0"
+      }
+    },
+    "node_modules/xmlhttprequest": {
+      "version": "1.8.0",
+      "resolved": "https://registry.npmjs.org/xmlhttprequest/-/xmlhttprequest-1.8.0.tgz",
+      "integrity": "sha512-58Im/U0mlVBLM38NdZjHyhuMtCqa61469k2YP/AaPbvCoV9aQGUpbJBj1QRm2ytRiVQBD/fsw7L2bJGDVQswBA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.4.0"
+      }
+    },
+    "node_modules/xtend": {
+      "version": "4.0.2",
+      "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz",
+      "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.4"
+      }
+    },
+    "node_modules/y18n": {
+      "version": "3.2.2",
+      "resolved": "https://registry.npmjs.org/y18n/-/y18n-3.2.2.tgz",
+      "integrity": "sha512-uGZHXkHnhF0XeeAPgnKfPv1bgKAYyVvmNL1xlKsPYZPaIHxGti2hHqvOCQv71XMsLxu1QjergkqogUnms5D3YQ==",
+      "dev": true
+    },
+    "node_modules/yaeti": {
+      "version": "0.0.6",
+      "resolved": "https://registry.npmjs.org/yaeti/-/yaeti-0.0.6.tgz",
+      "integrity": "sha512-MvQa//+KcZCUkBTIC9blM+CU9J2GzuTytsOUwf2lidtvkx/6gnEp1QvJv34t9vdjhFmha/mUiNDbN0D0mJWdug==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.32"
+      }
+    },
+    "node_modules/yallist": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz",
+      "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==",
+      "dev": true
+    },
+    "node_modules/yaml": {
+      "version": "1.10.2",
+      "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.2.tgz",
+      "integrity": "sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==",
+      "dev": true,
+      "engines": {
+        "node": ">= 6"
+      }
+    },
+    "node_modules/yargs": {
+      "version": "4.8.1",
+      "resolved": "https://registry.npmjs.org/yargs/-/yargs-4.8.1.tgz",
+      "integrity": "sha512-LqodLrnIDM3IFT+Hf/5sxBnEGECrfdC1uIbgZeJmESCSo4HoCAaKEus8MylXHAkdacGc0ye+Qa+dpkuom8uVYA==",
+      "dev": true,
+      "dependencies": {
+        "cliui": "^3.2.0",
+        "decamelize": "^1.1.1",
+        "get-caller-file": "^1.0.1",
+        "lodash.assign": "^4.0.3",
+        "os-locale": "^1.4.0",
+        "read-pkg-up": "^1.0.1",
+        "require-directory": "^2.1.1",
+        "require-main-filename": "^1.0.1",
+        "set-blocking": "^2.0.0",
+        "string-width": "^1.0.1",
+        "which-module": "^1.0.0",
+        "window-size": "^0.2.0",
+        "y18n": "^3.2.1",
+        "yargs-parser": "^2.4.1"
+      }
+    },
+    "node_modules/yargs-parser": {
+      "version": "2.4.1",
+      "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-2.4.1.tgz",
+      "integrity": "sha512-9pIKIJhnI5tonzG6OnCFlz/yln8xHYcGl+pn3xR0Vzff0vzN1PbNRaelgfgRUwZ3s4i3jvxT9WhmUGL4whnasA==",
+      "dev": true,
+      "dependencies": {
+        "camelcase": "^3.0.0",
+        "lodash.assign": "^4.0.6"
+      }
+    },
+    "node_modules/yargs-unparser": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-2.0.0.tgz",
+      "integrity": "sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==",
+      "dev": true,
+      "dependencies": {
+        "camelcase": "^6.0.0",
+        "decamelize": "^4.0.0",
+        "flat": "^5.0.2",
+        "is-plain-obj": "^2.1.0"
+      },
+      "engines": {
+        "node": ">=10"
+      }
+    },
+    "node_modules/yargs-unparser/node_modules/camelcase": {
+      "version": "6.3.0",
+      "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz",
+      "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/yargs-unparser/node_modules/decamelize": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-4.0.0.tgz",
+      "integrity": "sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/yargs/node_modules/ansi-regex": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
+      "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==",
+      "dev": true,
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/yargs/node_modules/is-fullwidth-code-point": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz",
+      "integrity": "sha512-1pqUqRjkhPJ9miNq9SwMfdvi6lBJcd6eFxvfaivQhaH3SgisfiuudvFntdKOmxuee/77l+FPjKrQjWvmPjWrRw==",
+      "dev": true,
+      "dependencies": {
+        "number-is-nan": "^1.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/yargs/node_modules/string-width": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz",
+      "integrity": "sha512-0XsVpQLnVCXHJfyEs8tC0zpTVIr5PKKsQtkT29IwupnPTjtPmQ3xT/4yCREF9hYkV/3M3kzcUTSAZT6a6h81tw==",
+      "dev": true,
+      "dependencies": {
+        "code-point-at": "^1.0.0",
+        "is-fullwidth-code-point": "^1.0.0",
+        "strip-ansi": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/yargs/node_modules/strip-ansi": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
+      "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==",
+      "dev": true,
+      "dependencies": {
+        "ansi-regex": "^2.0.0"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/yesno": {
+      "version": "0.4.0",
+      "resolved": "https://registry.npmjs.org/yesno/-/yesno-0.4.0.tgz",
+      "integrity": "sha512-tdBxmHvbXPBKYIg81bMCB7bVeDmHkRzk5rVJyYYXurwKkHq/MCd8rz4HSJUP7hW0H2NlXiq8IFiWvYKEHhlotA==",
+      "dev": true
+    },
+    "node_modules/yn": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz",
+      "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==",
+      "dev": true,
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/yocto-queue": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz",
+      "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==",
+      "dev": true,
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/zksync-web3": {
+      "version": "0.8.1",
+      "resolved": "https://registry.npmjs.org/zksync-web3/-/zksync-web3-0.8.1.tgz",
+      "integrity": "sha512-1A4aHPQ3MyuGjpv5X/8pVEN+MdZqMjfVmiweQSRjOlklXYu65wT9BGEOtCmMs5d3gIvLp4ssfTeuR5OCKOD2kw==",
+      "dev": true,
+      "peerDependencies": {
+        "ethers": "~5.7.0"
+      }
+    },
+    "src": {
+      "name": "@imtbl/wallet-contracts",
+      "version": "1.0.0",
+      "license": "Apache-2.0",
+      "devDependencies": {
+        "prettier": "^2.8.7",
+        "prettier-plugin-solidity": "^1.1.3"
+      },
+      "optionalDependencies": {
+        "@ethersproject/abi": "^5.7.0",
+        "@ethersproject/providers": "^5.7.2",
+        "ethers": "^5.7.2"
+      }
+    }
+  },
+  "dependencies": {
+    "@0xsequence/deployer": {
+      "version": "0.21.5",
+      "resolved": "https://registry.npmjs.org/@0xsequence/deployer/-/deployer-0.21.5.tgz",
+      "integrity": "sha512-P+QsiTrbZez//NCRvMc6OV6zWYP/TJzGJ2clFG1852CCPijqm0RJPmdoy5FzHpJd+9T4W7Awo0U2/gseSvFnaA==",
+      "dev": true,
+      "requires": {
+        "@0xsequence/utils": "^0.21.5",
+        "@ethersproject/contracts": "^5.0.12",
+        "ethers": "^5.0.32",
+        "ora": "^5.3.0"
+      }
+    },
+    "@0xsequence/utils": {
+      "version": "0.21.5",
+      "resolved": "https://registry.npmjs.org/@0xsequence/utils/-/utils-0.21.5.tgz",
+      "integrity": "sha512-6gL0kot2o+E3n8E/1yvQR3wv4vxuJB9w25SNUK/zd9MDJ4oIY8tIy5jc4YrhQwC+FqKPG3k/8skNN9oP3gNXJg==",
+      "dev": true,
+      "requires": {
+        "@ethersproject/abstract-signer": "5.0.14",
+        "@ethersproject/properties": "^5.0.9",
+        "ethers": "^5.0.32",
+        "js-base64": "^3.6.0"
+      }
+    },
+    "@babel/code-frame": {
+      "version": "7.18.6",
+      "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.18.6.tgz",
+      "integrity": "sha512-TDCmlK5eOvH+eH7cdAFlNXeVJqWIQ7gW9tY1GJIpUtFb6CmjVyq2VM3u71bOyR8CRihcCgMUYoDNyLXao3+70Q==",
+      "dev": true,
+      "requires": {
+        "@babel/highlight": "^7.18.6"
+      }
+    },
+    "@babel/helper-validator-identifier": {
+      "version": "7.19.1",
+      "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.19.1.tgz",
+      "integrity": "sha512-awrNfaMtnHUr653GgGEs++LlAvW6w+DcPrOliSMXWCKo597CwL5Acf/wWdNkf/tfEQE3mjkeD1YOVZOUV/od1w==",
+      "dev": true
+    },
+    "@babel/highlight": {
+      "version": "7.18.6",
+      "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.18.6.tgz",
+      "integrity": "sha512-u7stbOuYjaPezCuLj29hNW1v64M2Md2qupEKP1fHc7WdOA3DgLh37suiSrZYY7haUB7iBeQZ9P1uiRF359do3g==",
+      "dev": true,
+      "requires": {
+        "@babel/helper-validator-identifier": "^7.18.6",
+        "chalk": "^2.0.0",
+        "js-tokens": "^4.0.0"
+      },
+      "dependencies": {
+        "ansi-styles": {
+          "version": "3.2.1",
+          "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+          "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+          "dev": true,
+          "requires": {
+            "color-convert": "^1.9.0"
+          }
+        },
+        "chalk": {
+          "version": "2.4.2",
+          "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+          "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+          "dev": true,
+          "requires": {
+            "ansi-styles": "^3.2.1",
+            "escape-string-regexp": "^1.0.5",
+            "supports-color": "^5.3.0"
+          }
+        },
+        "color-convert": {
+          "version": "1.9.3",
+          "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+          "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+          "dev": true,
+          "requires": {
+            "color-name": "1.1.3"
+          }
+        },
+        "color-name": {
+          "version": "1.1.3",
+          "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+          "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+          "dev": true
+        },
+        "has-flag": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+          "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+          "dev": true
+        },
+        "supports-color": {
+          "version": "5.5.0",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+          "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+          "dev": true,
+          "requires": {
+            "has-flag": "^3.0.0"
+          }
+        }
+      }
+    },
+    "@babel/runtime": {
+      "version": "7.21.0",
+      "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.21.0.tgz",
+      "integrity": "sha512-xwII0//EObnq89Ji5AKYQaRYiW/nZ3llSv29d49IuxPhKbtJoLP+9QUUZ4nVragQVtaVGeZrpB+ZtG/Pdy/POw==",
+      "dev": true,
+      "requires": {
+        "regenerator-runtime": "^0.13.11"
+      }
+    },
+    "@colors/colors": {
+      "version": "1.5.0",
+      "resolved": "https://registry.npmjs.org/@colors/colors/-/colors-1.5.0.tgz",
+      "integrity": "sha512-ooWCrlZP11i8GImSjTHYHLkvFDP48nS4+204nGb1RiX/WXYHmJA2III9/e2DWVabCESdW7hBAEzHRqUn9OUVvQ==",
+      "dev": true,
+      "optional": true
+    },
+    "@cspotcode/source-map-support": {
+      "version": "0.8.1",
+      "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz",
+      "integrity": "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==",
+      "dev": true,
+      "requires": {
+        "@jridgewell/trace-mapping": "0.3.9"
+      }
+    },
+    "@ensdomains/address-encoder": {
+      "version": "0.1.9",
+      "resolved": "https://registry.npmjs.org/@ensdomains/address-encoder/-/address-encoder-0.1.9.tgz",
+      "integrity": "sha512-E2d2gP4uxJQnDu2Kfg1tHNspefzbLT8Tyjrm5sEuim32UkU2sm5xL4VXtgc2X33fmPEw9+jUMpGs4veMbf+PYg==",
+      "dev": true,
+      "requires": {
+        "bech32": "^1.1.3",
+        "blakejs": "^1.1.0",
+        "bn.js": "^4.11.8",
+        "bs58": "^4.0.1",
+        "crypto-addr-codec": "^0.1.7",
+        "nano-base32": "^1.0.1",
+        "ripemd160": "^2.0.2"
+      },
+      "dependencies": {
+        "bn.js": {
+          "version": "4.12.0",
+          "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+          "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+          "dev": true
+        }
+      }
+    },
+    "@ensdomains/ens": {
+      "version": "0.4.5",
+      "resolved": "https://registry.npmjs.org/@ensdomains/ens/-/ens-0.4.5.tgz",
+      "integrity": "sha512-JSvpj1iNMFjK6K+uVl4unqMoa9rf5jopb8cya5UGBWz23Nw8hSNT7efgUx4BTlAPAgpNlEioUfeTyQ6J9ZvTVw==",
+      "dev": true,
+      "requires": {
+        "bluebird": "^3.5.2",
+        "eth-ens-namehash": "^2.0.8",
+        "solc": "^0.4.20",
+        "testrpc": "0.0.1",
+        "web3-utils": "^1.0.0-beta.31"
+      }
+    },
+    "@ensdomains/ensjs": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/@ensdomains/ensjs/-/ensjs-2.1.0.tgz",
+      "integrity": "sha512-GRbGPT8Z/OJMDuxs75U/jUNEC0tbL0aj7/L/QQznGYKm/tiasp+ndLOaoULy9kKJFC0TBByqfFliEHDgoLhyog==",
+      "dev": true,
+      "requires": {
+        "@babel/runtime": "^7.4.4",
+        "@ensdomains/address-encoder": "^0.1.7",
+        "@ensdomains/ens": "0.4.5",
+        "@ensdomains/resolver": "0.2.4",
+        "content-hash": "^2.5.2",
+        "eth-ens-namehash": "^2.0.8",
+        "ethers": "^5.0.13",
+        "js-sha3": "^0.8.0"
+      }
+    },
+    "@ensdomains/resolver": {
+      "version": "0.2.4",
+      "resolved": "https://registry.npmjs.org/@ensdomains/resolver/-/resolver-0.2.4.tgz",
+      "integrity": "sha512-bvaTH34PMCbv6anRa9I/0zjLJgY4EuznbEMgbV77JBCQ9KNC46rzi0avuxpOfu+xDjPEtSFGqVEOr5GlUSGudA==",
+      "dev": true
+    },
+    "@eslint/eslintrc": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.0.0.tgz",
+      "integrity": "sha512-fluIaaV+GyV24CCu/ggiHdV+j4RNh85yQnAYS/G2mZODZgGmmlrgCydjUcV3YvxCm9x8nMAfThsqTni4KiXT4A==",
+      "dev": true,
+      "requires": {
+        "ajv": "^6.12.4",
+        "debug": "^4.3.2",
+        "espree": "^9.4.0",
+        "globals": "^13.19.0",
+        "ignore": "^5.2.0",
+        "import-fresh": "^3.2.1",
+        "js-yaml": "^4.1.0",
+        "minimatch": "^3.1.2",
+        "strip-json-comments": "^3.1.1"
+      },
+      "dependencies": {
+        "ajv": {
+          "version": "6.12.6",
+          "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+          "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+          "dev": true,
+          "requires": {
+            "fast-deep-equal": "^3.1.1",
+            "fast-json-stable-stringify": "^2.0.0",
+            "json-schema-traverse": "^0.4.1",
+            "uri-js": "^4.2.2"
+          }
+        },
+        "json-schema-traverse": {
+          "version": "0.4.1",
+          "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+          "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+          "dev": true
+        }
+      }
+    },
+    "@eslint/js": {
+      "version": "8.35.0",
+      "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.35.0.tgz",
+      "integrity": "sha512-JXdzbRiWclLVoD8sNUjR443VVlYqiYmDVT6rGUEIEHU5YJW0gaVZwV2xgM7D4arkvASqD0IlLUVjHiFuxaftRw==",
+      "dev": true
+    },
+    "@ethereumjs/common": {
+      "version": "2.5.0",
+      "resolved": "https://registry.npmjs.org/@ethereumjs/common/-/common-2.5.0.tgz",
+      "integrity": "sha512-DEHjW6e38o+JmB/NO3GZBpW4lpaiBpkFgXF6jLcJ6gETBYpEyaA5nTimsWBUJR3Vmtm/didUEbNjajskugZORg==",
+      "dev": true,
+      "requires": {
+        "crc-32": "^1.2.0",
+        "ethereumjs-util": "^7.1.1"
+      }
+    },
+    "@ethereumjs/tx": {
+      "version": "3.3.2",
+      "resolved": "https://registry.npmjs.org/@ethereumjs/tx/-/tx-3.3.2.tgz",
+      "integrity": "sha512-6AaJhwg4ucmwTvw/1qLaZUX5miWrwZ4nLOUsKyb/HtzS3BMw/CasKhdi1ims9mBKeK9sOJCH4qGKOBGyJCeeog==",
+      "dev": true,
+      "requires": {
+        "@ethereumjs/common": "^2.5.0",
+        "ethereumjs-util": "^7.1.2"
+      }
+    },
+    "@ethersproject/abi": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abi/-/abi-5.7.0.tgz",
+      "integrity": "sha512-351ktp42TiRcYB3H1OP8yajPeAQstMW/yCFokj/AthP9bLHzQFPlOrxOcwYEDkUAICmOHljvN4K39OMTMUa9RA==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/constants": "^5.7.0",
+        "@ethersproject/hash": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0"
+      }
+    },
+    "@ethersproject/abstract-provider": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abstract-provider/-/abstract-provider-5.7.0.tgz",
+      "integrity": "sha512-R41c9UkchKCpAqStMYUpdunjo3pkEvZC3FAwZn5S5MGbXoMQOHIdHItezTETxAO5bevtMApSyEhn9+CHcDsWBw==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/networks": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/transactions": "^5.7.0",
+        "@ethersproject/web": "^5.7.0"
+      }
+    },
+    "@ethersproject/abstract-signer": {
+      "version": "5.0.14",
+      "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.0.14.tgz",
+      "integrity": "sha512-JztBwVO7o5OHLh2vyjordlS4/1EjRyaECtc8vPdXTF1i4dXN+J0coeRoPN6ZFbBvi/YbaB6br2fvqhst1VQD/g==",
+      "dev": true,
+      "requires": {
+        "@ethersproject/abstract-provider": "^5.0.8",
+        "@ethersproject/bignumber": "^5.0.13",
+        "@ethersproject/bytes": "^5.0.9",
+        "@ethersproject/logger": "^5.0.8",
+        "@ethersproject/properties": "^5.0.7"
+      }
+    },
+    "@ethersproject/address": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/address/-/address-5.7.0.tgz",
+      "integrity": "sha512-9wYhYt7aghVGo758POM5nqcOMaE168Q6aRLJZwUmiqSrAungkG74gSSeKEIR7ukixesdRZGPgVqme6vmxs1fkA==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/rlp": "^5.7.0"
+      }
+    },
+    "@ethersproject/base64": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/base64/-/base64-5.7.0.tgz",
+      "integrity": "sha512-Dr8tcHt2mEbsZr/mwTPIQAf3Ai0Bks/7gTw9dSqk1mQvhW3XvRlmDJr/4n+wg1JmCl16NZue17CDh8xb/vZ0sQ==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bytes": "^5.7.0"
+      }
+    },
+    "@ethersproject/basex": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/basex/-/basex-5.7.0.tgz",
+      "integrity": "sha512-ywlh43GwZLv2Voc2gQVTKBoVQ1mti3d8HK5aMxsfu/nRDnMmNqaSJ3r3n85HBByT8OpoY96SXM1FogC533T4zw==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0"
+      }
+    },
+    "@ethersproject/bignumber": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/bignumber/-/bignumber-5.7.0.tgz",
+      "integrity": "sha512-n1CAdIHRWjSucQO3MC1zPSVgV/6dy/fjL9pMrPP9peL+QxEg9wOsVqwD4+818B6LUEtaXzVHQiuivzRoxPxUGw==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "bn.js": "^5.2.1"
+      }
+    },
+    "@ethersproject/bytes": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/bytes/-/bytes-5.7.0.tgz",
+      "integrity": "sha512-nsbxwgFXWh9NyYWo+U8atvmMsSdKJprTcICAkvbBffT75qDocbuggBU0SJiVK2MuTrp0q+xvLkTnGMPK1+uA9A==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "@ethersproject/constants": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/constants/-/constants-5.7.0.tgz",
+      "integrity": "sha512-DHI+y5dBNvkpYUMiRQyxRBYBefZkJfo70VUkUAsRjcPs47muV9evftfZ0PJVCXYbAiCgght0DtcF9srFQmIgWA==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bignumber": "^5.7.0"
+      }
+    },
+    "@ethersproject/contracts": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/contracts/-/contracts-5.7.0.tgz",
+      "integrity": "sha512-5GJbzEU3X+d33CdfPhcyS+z8MzsTrBGk/sc+G+59+tPa9yFkl6HQ9D6L0QMgNTA9q8dT0XKxxkyp883XsQvbbg==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/abi": "^5.7.0",
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/abstract-signer": "^5.7.0",
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/constants": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/transactions": "^5.7.0"
+      },
+      "dependencies": {
+        "@ethersproject/abstract-signer": {
+          "version": "5.7.0",
+          "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+          "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+          "devOptional": true,
+          "requires": {
+            "@ethersproject/abstract-provider": "^5.7.0",
+            "@ethersproject/bignumber": "^5.7.0",
+            "@ethersproject/bytes": "^5.7.0",
+            "@ethersproject/logger": "^5.7.0",
+            "@ethersproject/properties": "^5.7.0"
+          }
+        }
+      }
+    },
+    "@ethersproject/hash": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/hash/-/hash-5.7.0.tgz",
+      "integrity": "sha512-qX5WrQfnah1EFnO5zJv1v46a8HW0+E5xuBBDTwMFZLuVTx0tbU2kkx15NqdjxecrLGatQN9FGQKpb1FKdHCt+g==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/abstract-signer": "^5.7.0",
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/base64": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0"
+      },
+      "dependencies": {
+        "@ethersproject/abstract-signer": {
+          "version": "5.7.0",
+          "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+          "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+          "devOptional": true,
+          "requires": {
+            "@ethersproject/abstract-provider": "^5.7.0",
+            "@ethersproject/bignumber": "^5.7.0",
+            "@ethersproject/bytes": "^5.7.0",
+            "@ethersproject/logger": "^5.7.0",
+            "@ethersproject/properties": "^5.7.0"
+          }
+        }
+      }
+    },
+    "@ethersproject/hdnode": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/hdnode/-/hdnode-5.7.0.tgz",
+      "integrity": "sha512-OmyYo9EENBPPf4ERhR7oj6uAtUAhYGqOnIS+jE5pTXvdKBS99ikzq1E7Iv0ZQZ5V36Lqx1qZLeak0Ra16qpeOg==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/abstract-signer": "^5.7.0",
+        "@ethersproject/basex": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/pbkdf2": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/sha2": "^5.7.0",
+        "@ethersproject/signing-key": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0",
+        "@ethersproject/transactions": "^5.7.0",
+        "@ethersproject/wordlists": "^5.7.0"
+      },
+      "dependencies": {
+        "@ethersproject/abstract-signer": {
+          "version": "5.7.0",
+          "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+          "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+          "devOptional": true,
+          "requires": {
+            "@ethersproject/abstract-provider": "^5.7.0",
+            "@ethersproject/bignumber": "^5.7.0",
+            "@ethersproject/bytes": "^5.7.0",
+            "@ethersproject/logger": "^5.7.0",
+            "@ethersproject/properties": "^5.7.0"
+          }
+        }
+      }
+    },
+    "@ethersproject/json-wallets": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/json-wallets/-/json-wallets-5.7.0.tgz",
+      "integrity": "sha512-8oee5Xgu6+RKgJTkvEMl2wDgSPSAQ9MB/3JYjFV9jlKvcYHUXZC+cQp0njgmxdHkYWn8s6/IqIZYm0YWCjO/0g==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/abstract-signer": "^5.7.0",
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/hdnode": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/pbkdf2": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/random": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0",
+        "@ethersproject/transactions": "^5.7.0",
+        "aes-js": "3.0.0",
+        "scrypt-js": "3.0.1"
+      },
+      "dependencies": {
+        "@ethersproject/abstract-signer": {
+          "version": "5.7.0",
+          "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+          "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+          "devOptional": true,
+          "requires": {
+            "@ethersproject/abstract-provider": "^5.7.0",
+            "@ethersproject/bignumber": "^5.7.0",
+            "@ethersproject/bytes": "^5.7.0",
+            "@ethersproject/logger": "^5.7.0",
+            "@ethersproject/properties": "^5.7.0"
+          }
+        }
+      }
+    },
+    "@ethersproject/keccak256": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/keccak256/-/keccak256-5.7.0.tgz",
+      "integrity": "sha512-2UcPboeL/iW+pSg6vZ6ydF8tCnv3Iu/8tUmLLzWWGzxWKFFqOBQFLo6uLUv6BDrLgCDfN28RJ/wtByx+jZ4KBg==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bytes": "^5.7.0",
+        "js-sha3": "0.8.0"
+      }
+    },
+    "@ethersproject/logger": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/logger/-/logger-5.7.0.tgz",
+      "integrity": "sha512-0odtFdXu/XHtjQXJYA3u9G0G8btm0ND5Cu8M7i5vhEcE8/HmF4Lbdqanwyv4uQTr2tx6b7fQRmgLrsnpQlmnig==",
+      "devOptional": true
+    },
+    "@ethersproject/networks": {
+      "version": "5.7.1",
+      "resolved": "https://registry.npmjs.org/@ethersproject/networks/-/networks-5.7.1.tgz",
+      "integrity": "sha512-n/MufjFYv3yFcUyfhnXotyDlNdFb7onmkSy8aQERi2PjNcnWQ66xXxa3XlS8nCcA8aJKJjIIMNJTC7tu80GwpQ==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "@ethersproject/pbkdf2": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/pbkdf2/-/pbkdf2-5.7.0.tgz",
+      "integrity": "sha512-oR/dBRZR6GTyaofd86DehG72hY6NpAjhabkhxgr3X2FpJtJuodEl2auADWBZfhDHgVCbu3/H/Ocq2uC6dpNjjw==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/sha2": "^5.7.0"
+      }
+    },
+    "@ethersproject/properties": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/properties/-/properties-5.7.0.tgz",
+      "integrity": "sha512-J87jy8suntrAkIZtecpxEPxY//szqr1mlBaYlQ0r4RCaiD2hjheqF9s1LVE8vVuJCXisjIP+JgtK/Do54ej4Sw==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "@ethersproject/providers": {
+      "version": "5.7.2",
+      "resolved": "https://registry.npmjs.org/@ethersproject/providers/-/providers-5.7.2.tgz",
+      "integrity": "sha512-g34EWZ1WWAVgr4aptGlVBF8mhl3VWjv+8hoAnzStu8Ah22VHBsuGzP17eb6xDVRzw895G4W7vvx60lFFur/1Rg==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/abstract-signer": "^5.7.0",
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/base64": "^5.7.0",
+        "@ethersproject/basex": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/constants": "^5.7.0",
+        "@ethersproject/hash": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/networks": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/random": "^5.7.0",
+        "@ethersproject/rlp": "^5.7.0",
+        "@ethersproject/sha2": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0",
+        "@ethersproject/transactions": "^5.7.0",
+        "@ethersproject/web": "^5.7.0",
+        "bech32": "1.1.4",
+        "ws": "7.4.6"
+      },
+      "dependencies": {
+        "@ethersproject/abstract-signer": {
+          "version": "5.7.0",
+          "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+          "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+          "devOptional": true,
+          "requires": {
+            "@ethersproject/abstract-provider": "^5.7.0",
+            "@ethersproject/bignumber": "^5.7.0",
+            "@ethersproject/bytes": "^5.7.0",
+            "@ethersproject/logger": "^5.7.0",
+            "@ethersproject/properties": "^5.7.0"
+          }
+        }
+      }
+    },
+    "@ethersproject/random": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/random/-/random-5.7.0.tgz",
+      "integrity": "sha512-19WjScqRA8IIeWclFme75VMXSBvi4e6InrUNuaR4s5pTF2qNhcGdCUwdxUVGtDDqC00sDLCO93jPQoDUH4HVmQ==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "@ethersproject/rlp": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/rlp/-/rlp-5.7.0.tgz",
+      "integrity": "sha512-rBxzX2vK8mVF7b0Tol44t5Tb8gomOHkj5guL+HhzQ1yBh/ydjGnpw6at+X6Iw0Kp3OzzzkcKp8N9r0W4kYSs9w==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "@ethersproject/sha2": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/sha2/-/sha2-5.7.0.tgz",
+      "integrity": "sha512-gKlH42riwb3KYp0reLsFTokByAKoJdgFCwI+CCiX/k+Jm2mbNs6oOaCjYQSlI1+XBVejwH2KrmCbMAT/GnRDQw==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "hash.js": "1.1.7"
+      }
+    },
+    "@ethersproject/signing-key": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/signing-key/-/signing-key-5.7.0.tgz",
+      "integrity": "sha512-MZdy2nL3wO0u7gkB4nA/pEf8lu1TlFswPNmy8AiYkfKTdO6eXBJyUdmHO/ehm/htHw9K/qF8ujnTyUAD+Ry54Q==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "bn.js": "^5.2.1",
+        "elliptic": "6.5.4",
+        "hash.js": "1.1.7"
+      }
+    },
+    "@ethersproject/solidity": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/solidity/-/solidity-5.7.0.tgz",
+      "integrity": "sha512-HmabMd2Dt/raavyaGukF4XxizWKhKQ24DoLtdNbBmNKUOPqwjsKQSdV9GQtj9CBEea9DlzETlVER1gYeXXBGaA==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/sha2": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0"
+      }
+    },
+    "@ethersproject/strings": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/strings/-/strings-5.7.0.tgz",
+      "integrity": "sha512-/9nu+lj0YswRNSH0NXYqrh8775XNyEdUQAuf3f+SmOrnVewcJ5SBNAjF7lpgehKi4abvNNXyf+HX86czCdJ8Mg==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/constants": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "@ethersproject/transactions": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/transactions/-/transactions-5.7.0.tgz",
+      "integrity": "sha512-kmcNicCp1lp8qanMTC3RIikGgoJ80ztTyvtsFvCYpSCfkjhD0jZ2LOrnbcuxuToLIUYYf+4XwD1rP+B/erDIhQ==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/constants": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/rlp": "^5.7.0",
+        "@ethersproject/signing-key": "^5.7.0"
+      }
+    },
+    "@ethersproject/units": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/units/-/units-5.7.0.tgz",
+      "integrity": "sha512-pD3xLMy3SJu9kG5xDGI7+xhTEmGXlEqXU4OfNapmfnxLVY4EMSSRp7j1k7eezutBPH7RBN/7QPnwR7hzNlEFeg==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/constants": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0"
+      }
+    },
+    "@ethersproject/wallet": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/wallet/-/wallet-5.7.0.tgz",
+      "integrity": "sha512-MhmXlJXEJFBFVKrDLB4ZdDzxcBxQ3rLyCkhNqVu3CDYvR97E+8r01UgrI+TI99Le+aYm/in/0vp86guJuM7FCA==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/abstract-provider": "^5.7.0",
+        "@ethersproject/abstract-signer": "^5.7.0",
+        "@ethersproject/address": "^5.7.0",
+        "@ethersproject/bignumber": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/hash": "^5.7.0",
+        "@ethersproject/hdnode": "^5.7.0",
+        "@ethersproject/json-wallets": "^5.7.0",
+        "@ethersproject/keccak256": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/random": "^5.7.0",
+        "@ethersproject/signing-key": "^5.7.0",
+        "@ethersproject/transactions": "^5.7.0",
+        "@ethersproject/wordlists": "^5.7.0"
+      },
+      "dependencies": {
+        "@ethersproject/abstract-signer": {
+          "version": "5.7.0",
+          "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+          "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+          "devOptional": true,
+          "requires": {
+            "@ethersproject/abstract-provider": "^5.7.0",
+            "@ethersproject/bignumber": "^5.7.0",
+            "@ethersproject/bytes": "^5.7.0",
+            "@ethersproject/logger": "^5.7.0",
+            "@ethersproject/properties": "^5.7.0"
+          }
+        }
+      }
+    },
+    "@ethersproject/web": {
+      "version": "5.7.1",
+      "resolved": "https://registry.npmjs.org/@ethersproject/web/-/web-5.7.1.tgz",
+      "integrity": "sha512-Gueu8lSvyjBWL4cYsWsjh6MtMwM0+H4HvqFPZfB6dV8ctbP9zFAO73VG1cMWae0FLPCtz0peKPpZY8/ugJJX2w==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/base64": "^5.7.0",
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0"
+      }
+    },
+    "@ethersproject/wordlists": {
+      "version": "5.7.0",
+      "resolved": "https://registry.npmjs.org/@ethersproject/wordlists/-/wordlists-5.7.0.tgz",
+      "integrity": "sha512-S2TFNJNfHWVHNE6cNDjbVlZ6MgE17MIxMbMg2zv3wn+3XSJGosL1m9ZVv3GXCf/2ymSsQ+hRI5IzoMJTG6aoVA==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/bytes": "^5.7.0",
+        "@ethersproject/hash": "^5.7.0",
+        "@ethersproject/logger": "^5.7.0",
+        "@ethersproject/properties": "^5.7.0",
+        "@ethersproject/strings": "^5.7.0"
+      }
+    },
+    "@fastify/busboy": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/@fastify/busboy/-/busboy-2.1.0.tgz",
+      "integrity": "sha512-+KpH+QxZU7O4675t3mnkQKcZZg56u+K/Ct2K+N2AZYNVK8kyeo/bI18tI8aPm3tvNNRyTWfj6s5tnGNlcbQRsA==",
+      "dev": true
+    },
+    "@humanwhocodes/config-array": {
+      "version": "0.11.8",
+      "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.8.tgz",
+      "integrity": "sha512-UybHIJzJnR5Qc/MsD9Kr+RpO2h+/P1GhOwdiLPXK5TWk5sgTdu88bTD9UP+CKbPPh5Rni1u0GjAdYQLemG8g+g==",
+      "dev": true,
+      "requires": {
+        "@humanwhocodes/object-schema": "^1.2.1",
+        "debug": "^4.1.1",
+        "minimatch": "^3.0.5"
+      }
+    },
+    "@humanwhocodes/module-importer": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz",
+      "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==",
+      "dev": true
+    },
+    "@humanwhocodes/object-schema": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-1.2.1.tgz",
+      "integrity": "sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA==",
+      "dev": true
+    },
+    "@imtbl/wallet-contracts": {
+      "version": "file:src",
+      "requires": {
+        "@ethersproject/abi": "^5.7.0",
+        "@ethersproject/providers": "^5.7.2",
+        "ethers": "^5.7.2",
+        "prettier": "^2.8.7",
+        "prettier-plugin-solidity": "^1.1.3"
+      }
+    },
+    "@jridgewell/resolve-uri": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.0.tgz",
+      "integrity": "sha512-F2msla3tad+Mfht5cJq7LSXcdudKTWCVYUgw6pLFOOHSTtZlj6SWNYAp+AhuqLmWdBO2X5hPrLcu8cVP8fy28w==",
+      "dev": true
+    },
+    "@jridgewell/sourcemap-codec": {
+      "version": "1.4.14",
+      "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.14.tgz",
+      "integrity": "sha512-XPSJHWmi394fuUuzDnGz1wiKqWfo1yXecHQMRf2l6hztTO+nPru658AyDngaBe7isIxEkRsPR3FZh+s7iVa4Uw==",
+      "dev": true
+    },
+    "@jridgewell/trace-mapping": {
+      "version": "0.3.9",
+      "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz",
+      "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==",
+      "dev": true,
+      "requires": {
+        "@jridgewell/resolve-uri": "^3.0.3",
+        "@jridgewell/sourcemap-codec": "^1.4.10"
+      }
+    },
+    "@metamask/eth-sig-util": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/@metamask/eth-sig-util/-/eth-sig-util-4.0.1.tgz",
+      "integrity": "sha512-tghyZKLHZjcdlDqCA3gNZmLeR0XvOE9U1qoQO9ohyAZT6Pya+H9vkBPcsyXytmYLNgVoin7CKCmweo/R43V+tQ==",
+      "dev": true,
+      "requires": {
+        "ethereumjs-abi": "^0.6.8",
+        "ethereumjs-util": "^6.2.1",
+        "ethjs-util": "^0.1.6",
+        "tweetnacl": "^1.0.3",
+        "tweetnacl-util": "^0.15.1"
+      },
+      "dependencies": {
+        "@types/bn.js": {
+          "version": "4.11.6",
+          "resolved": "https://registry.npmjs.org/@types/bn.js/-/bn.js-4.11.6.tgz",
+          "integrity": "sha512-pqr857jrp2kPuO9uRjZ3PwnJTjoQy+fcdxvBTvHm6dkmEL9q+hDD/2j/0ELOBPtPnS8LjCX0gI9nbl8lVkadpg==",
+          "dev": true,
+          "requires": {
+            "@types/node": "*"
+          }
+        },
+        "bn.js": {
+          "version": "4.12.0",
+          "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+          "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+          "dev": true
+        },
+        "ethereumjs-util": {
+          "version": "6.2.1",
+          "resolved": "https://registry.npmjs.org/ethereumjs-util/-/ethereumjs-util-6.2.1.tgz",
+          "integrity": "sha512-W2Ktez4L01Vexijrm5EB6w7dg4n/TgpoYU4avuT5T3Vmnw/eCRtiBrJfQYS/DCSvDIOLn2k57GcHdeBcgVxAqw==",
+          "dev": true,
+          "requires": {
+            "@types/bn.js": "^4.11.3",
+            "bn.js": "^4.11.0",
+            "create-hash": "^1.1.2",
+            "elliptic": "^6.5.2",
+            "ethereum-cryptography": "^0.1.3",
+            "ethjs-util": "0.1.6",
+            "rlp": "^2.2.3"
+          }
+        },
+        "tweetnacl": {
+          "version": "1.0.3",
+          "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-1.0.3.tgz",
+          "integrity": "sha512-6rt+RN7aOi1nGMyC4Xa5DdYiukl2UWCbcJft7YhxReBGQD7OAM8Pbxw6YMo4r2diNEA8FEmu32YOn9rhaiE5yw==",
+          "dev": true
+        }
+      }
+    },
+    "@morgan-stanley/ts-mocking-bird": {
+      "version": "0.6.4",
+      "resolved": "https://registry.npmjs.org/@morgan-stanley/ts-mocking-bird/-/ts-mocking-bird-0.6.4.tgz",
+      "integrity": "sha512-57VJIflP8eR2xXa9cD1LUawh+Gh+BVQfVu0n6GALyg/AqV/Nz25kDRvws3i9kIe1PTrbsZZOYpsYp6bXPd6nVA==",
+      "dev": true,
+      "requires": {
+        "lodash": "^4.17.16",
+        "uuid": "^7.0.3"
+      },
+      "dependencies": {
+        "uuid": {
+          "version": "7.0.3",
+          "resolved": "https://registry.npmjs.org/uuid/-/uuid-7.0.3.tgz",
+          "integrity": "sha512-DPSke0pXhTZgoF/d+WSt2QaKMCFSfx7QegxEWT+JOuHF5aWrKEn0G+ztjuJg/gG8/ItK+rbPCD/yNv8yyih6Cg==",
+          "dev": true
+        }
+      }
+    },
+    "@noble/hashes": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/@noble/hashes/-/hashes-1.2.0.tgz",
+      "integrity": "sha512-FZfhjEDbT5GRswV3C6uvLPHMiVD6lQBmpoX5+eSiPaMTXte/IKqI5dykDxzZB/WBeK/CDuQRBWarPdi3FNY2zQ==",
+      "dev": true
+    },
+    "@noble/secp256k1": {
+      "version": "1.7.1",
+      "resolved": "https://registry.npmjs.org/@noble/secp256k1/-/secp256k1-1.7.1.tgz",
+      "integrity": "sha512-hOUk6AyBFmqVrv7k5WAw/LpszxVbj9gGN4JRkIX52fdFAj1UA61KXmZDvqVEm+pOyec3+fIeZB02LYa/pWOArw==",
+      "dev": true
+    },
+    "@nodelib/fs.scandir": {
+      "version": "2.1.5",
+      "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz",
+      "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==",
+      "dev": true,
+      "requires": {
+        "@nodelib/fs.stat": "2.0.5",
+        "run-parallel": "^1.1.9"
+      }
+    },
+    "@nodelib/fs.stat": {
+      "version": "2.0.5",
+      "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz",
+      "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==",
+      "dev": true
+    },
+    "@nodelib/fs.walk": {
+      "version": "1.2.8",
+      "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz",
+      "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==",
+      "dev": true,
+      "requires": {
+        "@nodelib/fs.scandir": "2.1.5",
+        "fastq": "^1.6.0"
+      }
+    },
+    "@nomicfoundation/ethereumjs-block": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-block/-/ethereumjs-block-4.0.0.tgz",
+      "integrity": "sha512-bk8uP8VuexLgyIZAHExH1QEovqx0Lzhc9Ntm63nCRKLHXIZkobaFaeCVwTESV7YkPKUk7NiK11s8ryed4CS9yA==",
+      "dev": true,
+      "requires": {
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-trie": "^5.0.0",
+        "@nomicfoundation/ethereumjs-tx": "^4.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "ethereum-cryptography": "0.1.3"
+      },
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-rlp": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+          "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+          "dev": true
+        }
+      }
+    },
+    "@nomicfoundation/ethereumjs-blockchain": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-blockchain/-/ethereumjs-blockchain-6.0.0.tgz",
+      "integrity": "sha512-pLFEoea6MWd81QQYSReLlLfH7N9v7lH66JC/NMPN848ySPPQA5renWnE7wPByfQFzNrPBuDDRFFULMDmj1C0xw==",
+      "dev": true,
+      "requires": {
+        "@nomicfoundation/ethereumjs-block": "^4.0.0",
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-ethash": "^2.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-trie": "^5.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "abstract-level": "^1.0.3",
+        "debug": "^4.3.3",
+        "ethereum-cryptography": "0.1.3",
+        "level": "^8.0.0",
+        "lru-cache": "^5.1.1",
+        "memory-level": "^1.0.0"
+      },
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-rlp": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+          "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+          "dev": true
+        },
+        "lru-cache": {
+          "version": "5.1.1",
+          "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz",
+          "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==",
+          "dev": true,
+          "requires": {
+            "yallist": "^3.0.2"
+          }
+        },
+        "yallist": {
+          "version": "3.1.1",
+          "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
+          "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==",
+          "dev": true
+        }
+      }
+    },
+    "@nomicfoundation/ethereumjs-common": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-common/-/ethereumjs-common-3.0.0.tgz",
+      "integrity": "sha512-WS7qSshQfxoZOpHG/XqlHEGRG1zmyjYrvmATvc4c62+gZXgre1ymYP8ZNgx/3FyZY0TWe9OjFlKOfLqmgOeYwA==",
+      "dev": true,
+      "requires": {
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "crc-32": "^1.2.0"
+      }
+    },
+    "@nomicfoundation/ethereumjs-ethash": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-ethash/-/ethereumjs-ethash-2.0.0.tgz",
+      "integrity": "sha512-WpDvnRncfDUuXdsAXlI4lXbqUDOA+adYRQaEezIkxqDkc+LDyYDbd/xairmY98GnQzo1zIqsIL6GB5MoMSJDew==",
+      "dev": true,
+      "requires": {
+        "@nomicfoundation/ethereumjs-block": "^4.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "abstract-level": "^1.0.3",
+        "bigint-crypto-utils": "^3.0.23",
+        "ethereum-cryptography": "0.1.3"
+      },
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-rlp": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+          "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+          "dev": true
+        }
+      }
+    },
+    "@nomicfoundation/ethereumjs-evm": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-evm/-/ethereumjs-evm-1.0.0.tgz",
+      "integrity": "sha512-hVS6qRo3V1PLKCO210UfcEQHvlG7GqR8iFzp0yyjTg2TmJQizcChKgWo8KFsdMw6AyoLgLhHGHw4HdlP8a4i+Q==",
+      "dev": true,
+      "requires": {
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "@types/async-eventemitter": "^0.2.1",
+        "async-eventemitter": "^0.2.4",
+        "debug": "^4.3.3",
+        "ethereum-cryptography": "0.1.3",
+        "mcl-wasm": "^0.7.1",
+        "rustbn.js": "~0.2.0"
+      }
+    },
+    "@nomicfoundation/ethereumjs-statemanager": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-statemanager/-/ethereumjs-statemanager-1.0.0.tgz",
+      "integrity": "sha512-jCtqFjcd2QejtuAMjQzbil/4NHf5aAWxUc+CvS0JclQpl+7M0bxMofR2AJdtz+P3u0ke2euhYREDiE7iSO31vQ==",
+      "dev": true,
+      "requires": {
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-trie": "^5.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "debug": "^4.3.3",
+        "ethereum-cryptography": "0.1.3",
+        "functional-red-black-tree": "^1.0.1"
+      },
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-rlp": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+          "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+          "dev": true
+        }
+      }
+    },
+    "@nomicfoundation/ethereumjs-trie": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-trie/-/ethereumjs-trie-5.0.0.tgz",
+      "integrity": "sha512-LIj5XdE+s+t6WSuq/ttegJzZ1vliwg6wlb+Y9f4RlBpuK35B9K02bO7xU+E6Rgg9RGptkWd6TVLdedTI4eNc2A==",
+      "dev": true,
+      "requires": {
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "ethereum-cryptography": "0.1.3",
+        "readable-stream": "^3.6.0"
+      },
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-rlp": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+          "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+          "dev": true
+        }
+      }
+    },
+    "@nomicfoundation/ethereumjs-tx": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-tx/-/ethereumjs-tx-4.0.0.tgz",
+      "integrity": "sha512-Gg3Lir2lNUck43Kp/3x6TfBNwcWC9Z1wYue9Nz3v4xjdcv6oDW9QSMJxqsKw9QEGoBBZ+gqwpW7+F05/rs/g1w==",
+      "dev": true,
+      "requires": {
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "ethereum-cryptography": "0.1.3"
+      },
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-rlp": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+          "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+          "dev": true
+        }
+      }
+    },
+    "@nomicfoundation/ethereumjs-util": {
+      "version": "8.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-util/-/ethereumjs-util-8.0.0.tgz",
+      "integrity": "sha512-2emi0NJ/HmTG+CGY58fa+DQuAoroFeSH9gKu9O6JnwTtlzJtgfTixuoOqLEgyyzZVvwfIpRueuePb8TonL1y+A==",
+      "dev": true,
+      "requires": {
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0-beta.2",
+        "ethereum-cryptography": "0.1.3"
+      },
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-rlp": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+          "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+          "dev": true
+        }
+      }
+    },
+    "@nomicfoundation/ethereumjs-vm": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-vm/-/ethereumjs-vm-6.0.0.tgz",
+      "integrity": "sha512-JMPxvPQ3fzD063Sg3Tp+UdwUkVxMoo1uML6KSzFhMH3hoQi/LMuXBoEHAoW83/vyNS9BxEe6jm6LmT5xdeEJ6w==",
+      "dev": true,
+      "requires": {
+        "@nomicfoundation/ethereumjs-block": "^4.0.0",
+        "@nomicfoundation/ethereumjs-blockchain": "^6.0.0",
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-evm": "^1.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-statemanager": "^1.0.0",
+        "@nomicfoundation/ethereumjs-trie": "^5.0.0",
+        "@nomicfoundation/ethereumjs-tx": "^4.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "@types/async-eventemitter": "^0.2.1",
+        "async-eventemitter": "^0.2.4",
+        "debug": "^4.3.3",
+        "ethereum-cryptography": "0.1.3",
+        "functional-red-black-tree": "^1.0.1",
+        "mcl-wasm": "^0.7.1",
+        "rustbn.js": "~0.2.0"
+      },
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-rlp": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+          "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+          "dev": true
+        }
+      }
+    },
+    "@nomicfoundation/hardhat-chai-matchers": {
+      "version": "1.0.6",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/hardhat-chai-matchers/-/hardhat-chai-matchers-1.0.6.tgz",
+      "integrity": "sha512-f5ZMNmabZeZegEfuxn/0kW+mm7+yV7VNDxLpMOMGXWFJ2l/Ct3QShujzDRF9cOkK9Ui/hbDeOWGZqyQALDXVCQ==",
+      "dev": true,
+      "requires": {
+        "@ethersproject/abi": "^5.1.2",
+        "@types/chai-as-promised": "^7.1.3",
+        "chai-as-promised": "^7.1.1",
+        "deep-eql": "^4.0.1",
+        "ordinal": "^1.0.3"
+      }
+    },
+    "@nomicfoundation/hardhat-network-helpers": {
+      "version": "1.0.8",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/hardhat-network-helpers/-/hardhat-network-helpers-1.0.8.tgz",
+      "integrity": "sha512-MNqQbzUJZnCMIYvlniC3U+kcavz/PhhQSsY90tbEtUyMj/IQqsLwIRZa4ctjABh3Bz0KCh9OXUZ7Yk/d9hr45Q==",
+      "dev": true,
+      "requires": {
+        "ethereumjs-util": "^7.1.4"
+      }
+    },
+    "@nomicfoundation/solidity-analyzer": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer/-/solidity-analyzer-0.1.1.tgz",
+      "integrity": "sha512-1LMtXj1puAxyFusBgUIy5pZk3073cNXYnXUpuNKFghHbIit/xZgbk0AokpUADbNm3gyD6bFWl3LRFh3dhVdREg==",
+      "dev": true,
+      "requires": {
+        "@nomicfoundation/solidity-analyzer-darwin-arm64": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-darwin-x64": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-freebsd-x64": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-linux-arm64-gnu": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-linux-arm64-musl": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-linux-x64-gnu": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-linux-x64-musl": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-win32-arm64-msvc": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-win32-ia32-msvc": "0.1.1",
+        "@nomicfoundation/solidity-analyzer-win32-x64-msvc": "0.1.1"
+      }
+    },
+    "@nomicfoundation/solidity-analyzer-darwin-arm64": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-darwin-arm64/-/solidity-analyzer-darwin-arm64-0.1.1.tgz",
+      "integrity": "sha512-KcTodaQw8ivDZyF+D76FokN/HdpgGpfjc/gFCImdLUyqB6eSWVaZPazMbeAjmfhx3R0zm/NYVzxwAokFKgrc0w==",
+      "dev": true,
+      "optional": true
+    },
+    "@nomicfoundation/solidity-analyzer-darwin-x64": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-darwin-x64/-/solidity-analyzer-darwin-x64-0.1.1.tgz",
+      "integrity": "sha512-XhQG4BaJE6cIbjAVtzGOGbK3sn1BO9W29uhk9J8y8fZF1DYz0Doj8QDMfpMu+A6TjPDs61lbsmeYodIDnfveSA==",
+      "dev": true,
+      "optional": true
+    },
+    "@nomicfoundation/solidity-analyzer-freebsd-x64": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-freebsd-x64/-/solidity-analyzer-freebsd-x64-0.1.1.tgz",
+      "integrity": "sha512-GHF1VKRdHW3G8CndkwdaeLkVBi5A9u2jwtlS7SLhBc8b5U/GcoL39Q+1CSO3hYqePNP+eV5YI7Zgm0ea6kMHoA==",
+      "dev": true,
+      "optional": true
+    },
+    "@nomicfoundation/solidity-analyzer-linux-arm64-gnu": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-linux-arm64-gnu/-/solidity-analyzer-linux-arm64-gnu-0.1.1.tgz",
+      "integrity": "sha512-g4Cv2fO37ZsUENQ2vwPnZc2zRenHyAxHcyBjKcjaSmmkKrFr64yvzeNO8S3GBFCo90rfochLs99wFVGT/0owpg==",
+      "dev": true,
+      "optional": true
+    },
+    "@nomicfoundation/solidity-analyzer-linux-arm64-musl": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-linux-arm64-musl/-/solidity-analyzer-linux-arm64-musl-0.1.1.tgz",
+      "integrity": "sha512-WJ3CE5Oek25OGE3WwzK7oaopY8xMw9Lhb0mlYuJl/maZVo+WtP36XoQTb7bW/i8aAdHW5Z+BqrHMux23pvxG3w==",
+      "dev": true,
+      "optional": true
+    },
+    "@nomicfoundation/solidity-analyzer-linux-x64-gnu": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-linux-x64-gnu/-/solidity-analyzer-linux-x64-gnu-0.1.1.tgz",
+      "integrity": "sha512-5WN7leSr5fkUBBjE4f3wKENUy9HQStu7HmWqbtknfXkkil+eNWiBV275IOlpXku7v3uLsXTOKpnnGHJYI2qsdA==",
+      "dev": true,
+      "optional": true
+    },
+    "@nomicfoundation/solidity-analyzer-linux-x64-musl": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-linux-x64-musl/-/solidity-analyzer-linux-x64-musl-0.1.1.tgz",
+      "integrity": "sha512-KdYMkJOq0SYPQMmErv/63CwGwMm5XHenEna9X9aB8mQmhDBrYrlAOSsIPgFCUSL0hjxE3xHP65/EPXR/InD2+w==",
+      "dev": true,
+      "optional": true
+    },
+    "@nomicfoundation/solidity-analyzer-win32-arm64-msvc": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-win32-arm64-msvc/-/solidity-analyzer-win32-arm64-msvc-0.1.1.tgz",
+      "integrity": "sha512-VFZASBfl4qiBYwW5xeY20exWhmv6ww9sWu/krWSesv3q5hA0o1JuzmPHR4LPN6SUZj5vcqci0O6JOL8BPw+APg==",
+      "dev": true,
+      "optional": true
+    },
+    "@nomicfoundation/solidity-analyzer-win32-ia32-msvc": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-win32-ia32-msvc/-/solidity-analyzer-win32-ia32-msvc-0.1.1.tgz",
+      "integrity": "sha512-JnFkYuyCSA70j6Si6cS1A9Gh1aHTEb8kOTBApp/c7NRTFGNMH8eaInKlyuuiIbvYFhlXW4LicqyYuWNNq9hkpQ==",
+      "dev": true,
+      "optional": true
+    },
+    "@nomicfoundation/solidity-analyzer-win32-x64-msvc": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@nomicfoundation/solidity-analyzer-win32-x64-msvc/-/solidity-analyzer-win32-x64-msvc-0.1.1.tgz",
+      "integrity": "sha512-HrVJr6+WjIXGnw3Q9u6KQcbZCtk0caVWhCdFADySvRyUxJ8PnzlaP+MhwNE8oyT8OZ6ejHBRrrgjSqDCFXGirw==",
+      "dev": true,
+      "optional": true
+    },
+    "@nomiclabs/hardhat-ethers": {
+      "version": "2.2.2",
+      "resolved": "https://registry.npmjs.org/@nomiclabs/hardhat-ethers/-/hardhat-ethers-2.2.2.tgz",
+      "integrity": "sha512-NLDlDFL2us07C0jB/9wzvR0kuLivChJWCXTKcj3yqjZqMoYp7g7wwS157F70VHx/+9gHIBGzak5pKDwG8gEefA==",
+      "dev": true,
+      "requires": {}
+    },
+    "@nomiclabs/hardhat-etherscan": {
+      "version": "3.1.7",
+      "resolved": "https://registry.npmjs.org/@nomiclabs/hardhat-etherscan/-/hardhat-etherscan-3.1.7.tgz",
+      "integrity": "sha512-tZ3TvSgpvsQ6B6OGmo1/Au6u8BrAkvs1mIC/eURA3xgIfznUZBhmpne8hv7BXUzw9xNL3fXdpOYgOQlVMTcoHQ==",
+      "dev": true,
+      "requires": {
+        "@ethersproject/abi": "^5.1.2",
+        "@ethersproject/address": "^5.0.2",
+        "cbor": "^8.1.0",
+        "chalk": "^2.4.2",
+        "debug": "^4.1.1",
+        "fs-extra": "^7.0.1",
+        "lodash": "^4.17.11",
+        "semver": "^6.3.0",
+        "table": "^6.8.0",
+        "undici": "^5.14.0"
+      },
+      "dependencies": {
+        "ansi-styles": {
+          "version": "3.2.1",
+          "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+          "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+          "dev": true,
+          "requires": {
+            "color-convert": "^1.9.0"
+          }
+        },
+        "chalk": {
+          "version": "2.4.2",
+          "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+          "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+          "dev": true,
+          "requires": {
+            "ansi-styles": "^3.2.1",
+            "escape-string-regexp": "^1.0.5",
+            "supports-color": "^5.3.0"
+          }
+        },
+        "color-convert": {
+          "version": "1.9.3",
+          "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+          "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+          "dev": true,
+          "requires": {
+            "color-name": "1.1.3"
+          }
+        },
+        "color-name": {
+          "version": "1.1.3",
+          "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+          "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+          "dev": true
+        },
+        "has-flag": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+          "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+          "dev": true
+        },
+        "supports-color": {
+          "version": "5.5.0",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+          "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+          "dev": true,
+          "requires": {
+            "has-flag": "^3.0.0"
+          }
+        }
+      }
+    },
+    "@nomiclabs/hardhat-truffle5": {
+      "version": "2.0.7",
+      "resolved": "https://registry.npmjs.org/@nomiclabs/hardhat-truffle5/-/hardhat-truffle5-2.0.7.tgz",
+      "integrity": "sha512-Pw8451IUZp1bTp0QqCHCYfCHs66sCnyxPcaorapu9mfOV9xnZsVaFdtutnhNEiXdiZwbed7LFKpRsde4BjFwig==",
+      "dev": true,
+      "requires": {
+        "@nomiclabs/truffle-contract": "^4.2.23",
+        "@types/chai": "^4.2.0",
+        "chai": "^4.2.0",
+        "ethereumjs-util": "^7.1.4",
+        "fs-extra": "^7.0.1"
+      }
+    },
+    "@nomiclabs/hardhat-web3": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/@nomiclabs/hardhat-web3/-/hardhat-web3-2.0.0.tgz",
+      "integrity": "sha512-zt4xN+D+fKl3wW2YlTX3k9APR3XZgPkxJYf36AcliJn3oujnKEVRZaHu0PhgLjO+gR+F/kiYayo9fgd2L8970Q==",
+      "dev": true,
+      "requires": {
+        "@types/bignumber.js": "^5.0.0"
+      }
+    },
+    "@nomiclabs/truffle-contract": {
+      "version": "4.5.10",
+      "resolved": "https://registry.npmjs.org/@nomiclabs/truffle-contract/-/truffle-contract-4.5.10.tgz",
+      "integrity": "sha512-nF/6InFV+0hUvutyFgsdOMCoYlr//2fJbRER4itxYtQtc4/O1biTwZIKRu+5l2J5Sq6LU2WX7vZHtDgQdhWxIQ==",
+      "dev": true,
+      "requires": {
+        "@ensdomains/ensjs": "^2.0.1",
+        "@truffle/blockchain-utils": "^0.1.3",
+        "@truffle/contract-schema": "^3.4.7",
+        "@truffle/debug-utils": "^6.0.22",
+        "@truffle/error": "^0.1.0",
+        "@truffle/interface-adapter": "^0.5.16",
+        "bignumber.js": "^7.2.1",
+        "ethereum-ens": "^0.8.0",
+        "ethers": "^4.0.0-beta.1",
+        "source-map-support": "^0.5.19"
+      },
+      "dependencies": {
+        "bn.js": {
+          "version": "4.12.0",
+          "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+          "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+          "dev": true
+        },
+        "ethers": {
+          "version": "4.0.49",
+          "resolved": "https://registry.npmjs.org/ethers/-/ethers-4.0.49.tgz",
+          "integrity": "sha512-kPltTvWiyu+OktYy1IStSO16i2e7cS9D9OxZ81q2UUaiNPVrm/RTcbxamCXF9VUSKzJIdJV68EAIhTEVBalRWg==",
+          "dev": true,
+          "requires": {
+            "aes-js": "3.0.0",
+            "bn.js": "^4.11.9",
+            "elliptic": "6.5.4",
+            "hash.js": "1.1.3",
+            "js-sha3": "0.5.7",
+            "scrypt-js": "2.0.4",
+            "setimmediate": "1.0.4",
+            "uuid": "2.0.1",
+            "xmlhttprequest": "1.8.0"
+          }
+        },
+        "hash.js": {
+          "version": "1.1.3",
+          "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.3.tgz",
+          "integrity": "sha512-/UETyP0W22QILqS+6HowevwhEFJ3MBJnwTf75Qob9Wz9t0DPuisL8kW8YZMK62dHAKE1c1p+gY1TtOLY+USEHA==",
+          "dev": true,
+          "requires": {
+            "inherits": "^2.0.3",
+            "minimalistic-assert": "^1.0.0"
+          }
+        },
+        "js-sha3": {
+          "version": "0.5.7",
+          "resolved": "https://registry.npmjs.org/js-sha3/-/js-sha3-0.5.7.tgz",
+          "integrity": "sha512-GII20kjaPX0zJ8wzkTbNDYMY7msuZcTWk8S5UOh6806Jq/wz1J8/bnr8uGU0DAUmYDjj2Mr4X1cW8v/GLYnR+g==",
+          "dev": true
+        },
+        "scrypt-js": {
+          "version": "2.0.4",
+          "resolved": "https://registry.npmjs.org/scrypt-js/-/scrypt-js-2.0.4.tgz",
+          "integrity": "sha512-4KsaGcPnuhtCZQCxFxN3GVYIhKFPTdLd8PLC552XwbMndtD0cjRFAhDuuydXQ0h08ZfPgzqe6EKHozpuH74iDw==",
+          "dev": true
+        },
+        "setimmediate": {
+          "version": "1.0.4",
+          "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.4.tgz",
+          "integrity": "sha512-/TjEmXQVEzdod/FFskf3o7oOAsGhHf2j1dZqRFbDzq4F3mvvxflIIi4Hd3bLQE9y/CpwqfSQam5JakI/mi3Pog==",
+          "dev": true
+        }
+      }
+    },
+    "@openzeppelin/contracts": {
+      "version": "4.9.3",
+      "resolved": "https://registry.npmjs.org/@openzeppelin/contracts/-/contracts-4.9.3.tgz",
+      "integrity": "sha512-He3LieZ1pP2TNt5JbkPA4PNT9WC3gOTOlDcFGJW4Le4QKqwmiNJCRt44APfxMxvq7OugU/cqYuPcSBzOw38DAg==",
+      "dev": true
+    },
+    "@scure/base": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/@scure/base/-/base-1.1.1.tgz",
+      "integrity": "sha512-ZxOhsSyxYwLJj3pLZCefNitxsj093tb2vq90mp2txoYeBqbcjDjqFhyM8eUjq/uFm6zJ+mUuqxlS2FkuSY1MTA==",
+      "dev": true
+    },
+    "@scure/bip32": {
+      "version": "1.1.5",
+      "resolved": "https://registry.npmjs.org/@scure/bip32/-/bip32-1.1.5.tgz",
+      "integrity": "sha512-XyNh1rB0SkEqd3tXcXMi+Xe1fvg+kUIcoRIEujP1Jgv7DqW2r9lg3Ah0NkFaCs9sTkQAQA8kw7xiRXzENi9Rtw==",
+      "dev": true,
+      "requires": {
+        "@noble/hashes": "~1.2.0",
+        "@noble/secp256k1": "~1.7.0",
+        "@scure/base": "~1.1.0"
+      }
+    },
+    "@scure/bip39": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/@scure/bip39/-/bip39-1.1.1.tgz",
+      "integrity": "sha512-t+wDck2rVkh65Hmv280fYdVdY25J9YeEUIgn2LG1WM6gxFkGzcksoDiUkWVpVp3Oex9xGC68JU2dSbUfwZ2jPg==",
+      "dev": true,
+      "requires": {
+        "@noble/hashes": "~1.2.0",
+        "@scure/base": "~1.1.0"
+      }
+    },
+    "@sentry/core": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/core/-/core-5.30.0.tgz",
+      "integrity": "sha512-TmfrII8w1PQZSZgPpUESqjB+jC6MvZJZdLtE/0hZ+SrnKhW3x5WlYLvTXZpcWePYBku7rl2wn1RZu6uT0qCTeg==",
+      "dev": true,
+      "requires": {
+        "@sentry/hub": "5.30.0",
+        "@sentry/minimal": "5.30.0",
+        "@sentry/types": "5.30.0",
+        "@sentry/utils": "5.30.0",
+        "tslib": "^1.9.3"
+      }
+    },
+    "@sentry/hub": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/hub/-/hub-5.30.0.tgz",
+      "integrity": "sha512-2tYrGnzb1gKz2EkMDQcfLrDTvmGcQPuWxLnJKXJvYTQDGLlEvi2tWz1VIHjunmOvJrB5aIQLhm+dcMRwFZDCqQ==",
+      "dev": true,
+      "requires": {
+        "@sentry/types": "5.30.0",
+        "@sentry/utils": "5.30.0",
+        "tslib": "^1.9.3"
+      }
+    },
+    "@sentry/minimal": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/minimal/-/minimal-5.30.0.tgz",
+      "integrity": "sha512-BwWb/owZKtkDX+Sc4zCSTNcvZUq7YcH3uAVlmh/gtR9rmUvbzAA3ewLuB3myi4wWRAMEtny6+J/FN/x+2wn9Xw==",
+      "dev": true,
+      "requires": {
+        "@sentry/hub": "5.30.0",
+        "@sentry/types": "5.30.0",
+        "tslib": "^1.9.3"
+      }
+    },
+    "@sentry/node": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/node/-/node-5.30.0.tgz",
+      "integrity": "sha512-Br5oyVBF0fZo6ZS9bxbJZG4ApAjRqAnqFFurMVJJdunNb80brh7a5Qva2kjhm+U6r9NJAB5OmDyPkA1Qnt+QVg==",
+      "dev": true,
+      "requires": {
+        "@sentry/core": "5.30.0",
+        "@sentry/hub": "5.30.0",
+        "@sentry/tracing": "5.30.0",
+        "@sentry/types": "5.30.0",
+        "@sentry/utils": "5.30.0",
+        "cookie": "^0.4.1",
+        "https-proxy-agent": "^5.0.0",
+        "lru_map": "^0.3.3",
+        "tslib": "^1.9.3"
+      },
+      "dependencies": {
+        "cookie": {
+          "version": "0.4.2",
+          "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.2.tgz",
+          "integrity": "sha512-aSWTXFzaKWkvHO1Ny/s+ePFpvKsPnjc551iI41v3ny/ow6tBG5Vd+FuqGNhh1LxOmVzOlGUriIlOaokOvhaStA==",
+          "dev": true
+        }
+      }
+    },
+    "@sentry/tracing": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/tracing/-/tracing-5.30.0.tgz",
+      "integrity": "sha512-dUFowCr0AIMwiLD7Fs314Mdzcug+gBVo/+NCMyDw8tFxJkwWAKl7Qa2OZxLQ0ZHjakcj1hNKfCQJ9rhyfOl4Aw==",
+      "dev": true,
+      "requires": {
+        "@sentry/hub": "5.30.0",
+        "@sentry/minimal": "5.30.0",
+        "@sentry/types": "5.30.0",
+        "@sentry/utils": "5.30.0",
+        "tslib": "^1.9.3"
+      }
+    },
+    "@sentry/types": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/types/-/types-5.30.0.tgz",
+      "integrity": "sha512-R8xOqlSTZ+htqrfteCWU5Nk0CDN5ApUTvrlvBuiH1DyP6czDZ4ktbZB0hAgBlVcK0U+qpD3ag3Tqqpa5Q67rPw==",
+      "dev": true
+    },
+    "@sentry/utils": {
+      "version": "5.30.0",
+      "resolved": "https://registry.npmjs.org/@sentry/utils/-/utils-5.30.0.tgz",
+      "integrity": "sha512-zaYmoH0NWWtvnJjC9/CBseXMtKHm/tm40sz3YfJRxeQjyzRqNQPgivpd9R/oDJCYj999mzdW382p/qi2ypjLww==",
+      "dev": true,
+      "requires": {
+        "@sentry/types": "5.30.0",
+        "tslib": "^1.9.3"
+      }
+    },
+    "@sindresorhus/is": {
+      "version": "4.6.0",
+      "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-4.6.0.tgz",
+      "integrity": "sha512-t09vSN3MdfsyCHoFcTRCH/iUtG7OJ0CsjzB8cjAmKc/va/kIgeDI/TxsigdncE/4be734m0cvIYwNaV4i2XqAw==",
+      "dev": true
+    },
+    "@solidity-parser/parser": {
+      "version": "0.14.5",
+      "resolved": "https://registry.npmjs.org/@solidity-parser/parser/-/parser-0.14.5.tgz",
+      "integrity": "sha512-6dKnHZn7fg/iQATVEzqyUOyEidbn05q7YA2mQ9hC0MMXhhV3/JrsxmFSYZAcr7j1yUP700LLhTruvJ3MiQmjJg==",
+      "dev": true,
+      "requires": {
+        "antlr4ts": "^0.5.0-alpha.4"
+      }
+    },
+    "@szmarczak/http-timer": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-5.0.1.tgz",
+      "integrity": "sha512-+PmQX0PiAYPMeVYe237LJAYvOMYW1j2rH5YROyS3b4CTVJum34HfRvKvAzozHAQG0TnHNdUfY9nCeUyRAs//cw==",
+      "dev": true,
+      "requires": {
+        "defer-to-connect": "^2.0.1"
+      }
+    },
+    "@tenderly/hardhat-tenderly": {
+      "version": "1.6.1",
+      "resolved": "https://registry.npmjs.org/@tenderly/hardhat-tenderly/-/hardhat-tenderly-1.6.1.tgz",
+      "integrity": "sha512-VhnOcRVB8J1mZk5QUifthsidyEb4p6Qj0nyy07qdnF8GL/kcVDBgqluyapGS1hUhp/SVQqpKHcFTWbVJ/B0l4w==",
+      "dev": true,
+      "requires": {
+        "@ethersproject/bignumber": "^5.7.0",
+        "@nomiclabs/hardhat-ethers": "^2.1.1",
+        "axios": "^0.27.2",
+        "ethers": "^5.7.0",
+        "fs-extra": "^10.1.0",
+        "hardhat": "^2.10.2",
+        "hardhat-deploy": "^0.11.14",
+        "js-yaml": "^4.1.0",
+        "tenderly": "^0.4.0",
+        "tslog": "^4.3.1"
+      },
+      "dependencies": {
+        "fs-extra": {
+          "version": "10.1.0",
+          "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-10.1.0.tgz",
+          "integrity": "sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ==",
+          "dev": true,
+          "requires": {
+            "graceful-fs": "^4.2.0",
+            "jsonfile": "^6.0.1",
+            "universalify": "^2.0.0"
+          }
+        },
+        "jsonfile": {
+          "version": "6.1.0",
+          "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz",
+          "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==",
+          "dev": true,
+          "requires": {
+            "graceful-fs": "^4.1.6",
+            "universalify": "^2.0.0"
+          }
+        },
+        "universalify": {
+          "version": "2.0.0",
+          "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz",
+          "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==",
+          "dev": true
+        }
+      }
+    },
+    "@truffle/abi-utils": {
+      "version": "0.3.9",
+      "resolved": "https://registry.npmjs.org/@truffle/abi-utils/-/abi-utils-0.3.9.tgz",
+      "integrity": "sha512-G5dqgwRHx5zwlXjz3QT8OJVfB2cOqWwD6DwKso0KttUt/zejhCjnkKq72rSgyeLMkz7wBB9ERLOsupLBILM8MA==",
+      "dev": true,
+      "requires": {
+        "change-case": "3.0.2",
+        "fast-check": "3.1.1",
+        "web3-utils": "1.8.2"
+      }
+    },
+    "@truffle/blockchain-utils": {
+      "version": "0.1.6",
+      "resolved": "https://registry.npmjs.org/@truffle/blockchain-utils/-/blockchain-utils-0.1.6.tgz",
+      "integrity": "sha512-SldoNRIFSm3+HMBnSc2jFsu5TWDkCN4X6vL3wrd0t6DIeF7nD6EoPPjxwbFSoqCnkkRxMuZeL6sUx7UMJS/wSA==",
+      "dev": true
+    },
+    "@truffle/codec": {
+      "version": "0.14.16",
+      "resolved": "https://registry.npmjs.org/@truffle/codec/-/codec-0.14.16.tgz",
+      "integrity": "sha512-a9UY3n/FnkKN3Q4zOuMFOOcLWb80mdknj+voim4vvXYtJm1aAZQZE5sG9aLnMBTl4TiGLzUtfNDVYY7WgWgDag==",
+      "dev": true,
+      "requires": {
+        "@truffle/abi-utils": "^0.3.9",
+        "@truffle/compile-common": "^0.9.4",
+        "big.js": "^6.0.3",
+        "bn.js": "^5.1.3",
+        "cbor": "^5.2.0",
+        "debug": "^4.3.1",
+        "lodash": "^4.17.21",
+        "semver": "7.3.7",
+        "utf8": "^3.0.0",
+        "web3-utils": "1.8.2"
+      },
+      "dependencies": {
+        "bignumber.js": {
+          "version": "9.1.1",
+          "resolved": "https://registry.npmjs.org/bignumber.js/-/bignumber.js-9.1.1.tgz",
+          "integrity": "sha512-pHm4LsMJ6lzgNGVfZHjMoO8sdoRhOzOH4MLmY65Jg70bpxCKu5iOHNJyfF6OyvYw7t8Fpf35RuzUyqnQsj8Vig==",
+          "dev": true
+        },
+        "cbor": {
+          "version": "5.2.0",
+          "resolved": "https://registry.npmjs.org/cbor/-/cbor-5.2.0.tgz",
+          "integrity": "sha512-5IMhi9e1QU76ppa5/ajP1BmMWZ2FHkhAhjeVKQ/EFCgYSEaeVaoGtL7cxJskf9oCCk+XjzaIdc3IuU/dbA/o2A==",
+          "dev": true,
+          "requires": {
+            "bignumber.js": "^9.0.1",
+            "nofilter": "^1.0.4"
+          }
+        },
+        "nofilter": {
+          "version": "1.0.4",
+          "resolved": "https://registry.npmjs.org/nofilter/-/nofilter-1.0.4.tgz",
+          "integrity": "sha512-N8lidFp+fCz+TD51+haYdbDGrcBWwuHX40F5+z0qkUjMJ5Tp+rdSuAkMJ9N9eoolDlEVTf6u5icM+cNKkKW2mA==",
+          "dev": true
+        },
+        "semver": {
+          "version": "7.3.7",
+          "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz",
+          "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==",
+          "dev": true,
+          "requires": {
+            "lru-cache": "^6.0.0"
+          }
+        }
+      }
+    },
+    "@truffle/compile-common": {
+      "version": "0.9.4",
+      "resolved": "https://registry.npmjs.org/@truffle/compile-common/-/compile-common-0.9.4.tgz",
+      "integrity": "sha512-mnqJB/hLiPHNf+WKwt/2MH6lv34xSG/SFCib7+ckAklutUqVLeFo8EwQxinuHNkU7LY0C+YgZXhK1WTCO5YRJQ==",
+      "dev": true,
+      "requires": {
+        "@truffle/error": "^0.2.0",
+        "colors": "1.4.0"
+      },
+      "dependencies": {
+        "@truffle/error": {
+          "version": "0.2.0",
+          "resolved": "https://registry.npmjs.org/@truffle/error/-/error-0.2.0.tgz",
+          "integrity": "sha512-Fe0/z4WWb7IP2gBnv3l6zqP87Y0kSMs7oiSLakKJq17q3GUunrHSdioKuNspdggxkXIBhEQLhi8C+LJdwmHKWQ==",
+          "dev": true
+        }
+      }
+    },
+    "@truffle/contract-schema": {
+      "version": "3.4.13",
+      "resolved": "https://registry.npmjs.org/@truffle/contract-schema/-/contract-schema-3.4.13.tgz",
+      "integrity": "sha512-emG7upuryYFrsPDbHqeASPWXL824M1tinhQwSPG0phSoa3g+RX9fUNNN/VPmF3tSkXLWUMhRnb7ehxnaCuRbZg==",
+      "dev": true,
+      "requires": {
+        "ajv": "^6.10.0",
+        "debug": "^4.3.1"
+      },
+      "dependencies": {
+        "ajv": {
+          "version": "6.12.6",
+          "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+          "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+          "dev": true,
+          "requires": {
+            "fast-deep-equal": "^3.1.1",
+            "fast-json-stable-stringify": "^2.0.0",
+            "json-schema-traverse": "^0.4.1",
+            "uri-js": "^4.2.2"
+          }
+        },
+        "json-schema-traverse": {
+          "version": "0.4.1",
+          "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+          "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+          "dev": true
+        }
+      }
+    },
+    "@truffle/debug-utils": {
+      "version": "6.0.47",
+      "resolved": "https://registry.npmjs.org/@truffle/debug-utils/-/debug-utils-6.0.47.tgz",
+      "integrity": "sha512-bUjdzLPdEKtoUCDzaXkrOoi+PbyAJlMBzGequBK8tirT7xL9bCP2Pd/WxvnmRd7AnfroxGNvXwVXWTItW5SMWQ==",
+      "dev": true,
+      "requires": {
+        "@truffle/codec": "^0.14.16",
+        "@trufflesuite/chromafi": "^3.0.0",
+        "bn.js": "^5.1.3",
+        "chalk": "^2.4.2",
+        "debug": "^4.3.1",
+        "highlightjs-solidity": "^2.0.6"
+      },
+      "dependencies": {
+        "ansi-styles": {
+          "version": "3.2.1",
+          "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+          "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+          "dev": true,
+          "requires": {
+            "color-convert": "^1.9.0"
+          }
+        },
+        "chalk": {
+          "version": "2.4.2",
+          "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+          "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+          "dev": true,
+          "requires": {
+            "ansi-styles": "^3.2.1",
+            "escape-string-regexp": "^1.0.5",
+            "supports-color": "^5.3.0"
+          }
+        },
+        "color-convert": {
+          "version": "1.9.3",
+          "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+          "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+          "dev": true,
+          "requires": {
+            "color-name": "1.1.3"
+          }
+        },
+        "color-name": {
+          "version": "1.1.3",
+          "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+          "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+          "dev": true
+        },
+        "has-flag": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+          "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+          "dev": true
+        },
+        "supports-color": {
+          "version": "5.5.0",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+          "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+          "dev": true,
+          "requires": {
+            "has-flag": "^3.0.0"
+          }
+        }
+      }
+    },
+    "@truffle/error": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/@truffle/error/-/error-0.1.1.tgz",
+      "integrity": "sha512-sE7c9IHIGdbK4YayH4BC8i8qMjoAOeg6nUXUDZZp8wlU21/EMpaG+CLx+KqcIPyR+GSWIW3Dm0PXkr2nlggFDA==",
+      "dev": true
+    },
+    "@truffle/interface-adapter": {
+      "version": "0.5.29",
+      "resolved": "https://registry.npmjs.org/@truffle/interface-adapter/-/interface-adapter-0.5.29.tgz",
+      "integrity": "sha512-6UlJ+f87z7y/dWk9UfbIU+4e80iRsp8h03LEiE5B+PvZbr6cuMjLJUBtBBQZMo3+xrIcS/2u3p5hOxW8OJm8tw==",
+      "dev": true,
+      "requires": {
+        "bn.js": "^5.1.3",
+        "ethers": "^4.0.32",
+        "web3": "1.8.2"
+      },
+      "dependencies": {
+        "ethers": {
+          "version": "4.0.49",
+          "resolved": "https://registry.npmjs.org/ethers/-/ethers-4.0.49.tgz",
+          "integrity": "sha512-kPltTvWiyu+OktYy1IStSO16i2e7cS9D9OxZ81q2UUaiNPVrm/RTcbxamCXF9VUSKzJIdJV68EAIhTEVBalRWg==",
+          "dev": true,
+          "requires": {
+            "aes-js": "3.0.0",
+            "bn.js": "^4.11.9",
+            "elliptic": "6.5.4",
+            "hash.js": "1.1.3",
+            "js-sha3": "0.5.7",
+            "scrypt-js": "2.0.4",
+            "setimmediate": "1.0.4",
+            "uuid": "2.0.1",
+            "xmlhttprequest": "1.8.0"
+          },
+          "dependencies": {
+            "bn.js": {
+              "version": "4.12.0",
+              "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+              "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+              "dev": true
+            }
+          }
+        },
+        "hash.js": {
+          "version": "1.1.3",
+          "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.3.tgz",
+          "integrity": "sha512-/UETyP0W22QILqS+6HowevwhEFJ3MBJnwTf75Qob9Wz9t0DPuisL8kW8YZMK62dHAKE1c1p+gY1TtOLY+USEHA==",
+          "dev": true,
+          "requires": {
+            "inherits": "^2.0.3",
+            "minimalistic-assert": "^1.0.0"
+          }
+        },
+        "js-sha3": {
+          "version": "0.5.7",
+          "resolved": "https://registry.npmjs.org/js-sha3/-/js-sha3-0.5.7.tgz",
+          "integrity": "sha512-GII20kjaPX0zJ8wzkTbNDYMY7msuZcTWk8S5UOh6806Jq/wz1J8/bnr8uGU0DAUmYDjj2Mr4X1cW8v/GLYnR+g==",
+          "dev": true
+        },
+        "scrypt-js": {
+          "version": "2.0.4",
+          "resolved": "https://registry.npmjs.org/scrypt-js/-/scrypt-js-2.0.4.tgz",
+          "integrity": "sha512-4KsaGcPnuhtCZQCxFxN3GVYIhKFPTdLd8PLC552XwbMndtD0cjRFAhDuuydXQ0h08ZfPgzqe6EKHozpuH74iDw==",
+          "dev": true
+        },
+        "setimmediate": {
+          "version": "1.0.4",
+          "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.4.tgz",
+          "integrity": "sha512-/TjEmXQVEzdod/FFskf3o7oOAsGhHf2j1dZqRFbDzq4F3mvvxflIIi4Hd3bLQE9y/CpwqfSQam5JakI/mi3Pog==",
+          "dev": true
+        }
+      }
+    },
+    "@trufflesuite/chromafi": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/@trufflesuite/chromafi/-/chromafi-3.0.0.tgz",
+      "integrity": "sha512-oqWcOqn8nT1bwlPPfidfzS55vqcIDdpfzo3HbU9EnUmcSTX+I8z0UyUFI3tZQjByVJulbzxHxUGS3ZJPwK/GPQ==",
+      "dev": true,
+      "requires": {
+        "camelcase": "^4.1.0",
+        "chalk": "^2.3.2",
+        "cheerio": "^1.0.0-rc.2",
+        "detect-indent": "^5.0.0",
+        "highlight.js": "^10.4.1",
+        "lodash.merge": "^4.6.2",
+        "strip-ansi": "^4.0.0",
+        "strip-indent": "^2.0.0"
+      },
+      "dependencies": {
+        "ansi-regex": {
+          "version": "3.0.1",
+          "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz",
+          "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==",
+          "dev": true
+        },
+        "ansi-styles": {
+          "version": "3.2.1",
+          "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+          "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+          "dev": true,
+          "requires": {
+            "color-convert": "^1.9.0"
+          }
+        },
+        "camelcase": {
+          "version": "4.1.0",
+          "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-4.1.0.tgz",
+          "integrity": "sha512-FxAv7HpHrXbh3aPo4o2qxHay2lkLY3x5Mw3KeE4KQE8ysVfziWeRZDwcjauvwBSGEC/nXUPzZy8zeh4HokqOnw==",
+          "dev": true
+        },
+        "chalk": {
+          "version": "2.4.2",
+          "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+          "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+          "dev": true,
+          "requires": {
+            "ansi-styles": "^3.2.1",
+            "escape-string-regexp": "^1.0.5",
+            "supports-color": "^5.3.0"
+          }
+        },
+        "color-convert": {
+          "version": "1.9.3",
+          "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+          "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+          "dev": true,
+          "requires": {
+            "color-name": "1.1.3"
+          }
+        },
+        "color-name": {
+          "version": "1.1.3",
+          "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+          "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+          "dev": true
+        },
+        "has-flag": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+          "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+          "dev": true
+        },
+        "strip-ansi": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz",
+          "integrity": "sha512-4XaJ2zQdCzROZDivEVIDPkcQn8LMFSa8kj8Gxb/Lnwzv9A8VctNZ+lfivC/sV3ivW8ElJTERXZoPBRrZKkNKow==",
+          "dev": true,
+          "requires": {
+            "ansi-regex": "^3.0.0"
+          }
+        },
+        "supports-color": {
+          "version": "5.5.0",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+          "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+          "dev": true,
+          "requires": {
+            "has-flag": "^3.0.0"
+          }
+        }
+      }
+    },
+    "@tsconfig/node10": {
+      "version": "1.0.9",
+      "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.9.tgz",
+      "integrity": "sha512-jNsYVVxU8v5g43Erja32laIDHXeoNvFEpX33OK4d6hljo3jDhCBDhx5dhCCTMWUojscpAagGiRkBKxpdl9fxqA==",
+      "dev": true
+    },
+    "@tsconfig/node12": {
+      "version": "1.0.11",
+      "resolved": "https://registry.npmjs.org/@tsconfig/node12/-/node12-1.0.11.tgz",
+      "integrity": "sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==",
+      "dev": true
+    },
+    "@tsconfig/node14": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/@tsconfig/node14/-/node14-1.0.3.tgz",
+      "integrity": "sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==",
+      "dev": true
+    },
+    "@tsconfig/node16": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/@tsconfig/node16/-/node16-1.0.3.tgz",
+      "integrity": "sha512-yOlFc+7UtL/89t2ZhjPvvB/DeAr3r+Dq58IgzsFkOAvVC6NMJXmCGjbptdXdR9qsX7pKcTL+s87FtYREi2dEEQ==",
+      "dev": true
+    },
+    "@typechain/ethers-v5": {
+      "version": "10.2.0",
+      "resolved": "https://registry.npmjs.org/@typechain/ethers-v5/-/ethers-v5-10.2.0.tgz",
+      "integrity": "sha512-ikaq0N/w9fABM+G01OFmU3U3dNnyRwEahkdvi9mqy1a3XwKiPZaF/lu54OcNaEWnpvEYyhhS0N7buCtLQqC92w==",
+      "dev": true,
+      "requires": {
+        "lodash": "^4.17.15",
+        "ts-essentials": "^7.0.1"
+      }
+    },
+    "@types/async-eventemitter": {
+      "version": "0.2.1",
+      "resolved": "https://registry.npmjs.org/@types/async-eventemitter/-/async-eventemitter-0.2.1.tgz",
+      "integrity": "sha512-M2P4Ng26QbAeITiH7w1d7OxtldgfAe0wobpyJzVK/XOb0cUGKU2R4pfAhqcJBXAe2ife5ZOhSv4wk7p+ffURtg==",
+      "dev": true
+    },
+    "@types/bignumber.js": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/@types/bignumber.js/-/bignumber.js-5.0.0.tgz",
+      "integrity": "sha512-0DH7aPGCClywOFaxxjE6UwpN2kQYe9LwuDQMv+zYA97j5GkOMo8e66LYT+a8JYU7jfmUFRZLa9KycxHDsKXJCA==",
+      "dev": true,
+      "requires": {
+        "bignumber.js": "*"
+      }
+    },
+    "@types/bn.js": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/@types/bn.js/-/bn.js-5.1.1.tgz",
+      "integrity": "sha512-qNrYbZqMx0uJAfKnKclPh+dTwK33KfLHYqtyODwd5HnXOjnkhc4qgn3BrK6RWyGZm5+sIFE7Q7Vz6QQtJB7w7g==",
+      "dev": true,
+      "requires": {
+        "@types/node": "*"
+      }
+    },
+    "@types/cacheable-request": {
+      "version": "6.0.3",
+      "resolved": "https://registry.npmjs.org/@types/cacheable-request/-/cacheable-request-6.0.3.tgz",
+      "integrity": "sha512-IQ3EbTzGxIigb1I3qPZc1rWJnH0BmSKv5QYTalEwweFvyBDLSAe24zP0le/hyi7ecGfZVlIVAg4BZqb8WBwKqw==",
+      "dev": true,
+      "requires": {
+        "@types/http-cache-semantics": "*",
+        "@types/keyv": "^3.1.4",
+        "@types/node": "*",
+        "@types/responselike": "^1.0.0"
+      }
+    },
+    "@types/chai": {
+      "version": "4.3.4",
+      "resolved": "https://registry.npmjs.org/@types/chai/-/chai-4.3.4.tgz",
+      "integrity": "sha512-KnRanxnpfpjUTqTCXslZSEdLfXExwgNxYPdiO2WGUj8+HDjFi8R3k5RVKPeSCzLjCcshCAtVO2QBbVuAV4kTnw==",
+      "dev": true
+    },
+    "@types/chai-as-promised": {
+      "version": "7.1.5",
+      "resolved": "https://registry.npmjs.org/@types/chai-as-promised/-/chai-as-promised-7.1.5.tgz",
+      "integrity": "sha512-jStwss93SITGBwt/niYrkf2C+/1KTeZCZl1LaeezTlqppAKeoQC7jxyqYuP72sxBGKCIbw7oHgbYssIRzT5FCQ==",
+      "dev": true,
+      "requires": {
+        "@types/chai": "*"
+      }
+    },
+    "@types/chai-string": {
+      "version": "1.4.2",
+      "resolved": "https://registry.npmjs.org/@types/chai-string/-/chai-string-1.4.2.tgz",
+      "integrity": "sha512-ld/1hV5qcPRGuwlPdvRfvM3Ka/iofOk2pH4VkasK4b1JJP1LjNmWWn0LsISf6RRzyhVOvs93rb9tM09e+UuF8Q==",
+      "dev": true,
+      "requires": {
+        "@types/chai": "*"
+      }
+    },
+    "@types/concat-stream": {
+      "version": "1.6.1",
+      "resolved": "https://registry.npmjs.org/@types/concat-stream/-/concat-stream-1.6.1.tgz",
+      "integrity": "sha512-eHE4cQPoj6ngxBZMvVf6Hw7Mh4jMW4U9lpGmS5GBPB9RYxlFg+CHaVN7ErNY4W9XfLIEn20b4VDYaIrbq0q4uA==",
+      "dev": true,
+      "requires": {
+        "@types/node": "*"
+      }
+    },
+    "@types/form-data": {
+      "version": "0.0.33",
+      "resolved": "https://registry.npmjs.org/@types/form-data/-/form-data-0.0.33.tgz",
+      "integrity": "sha512-8BSvG1kGm83cyJITQMZSulnl6QV8jqAGreJsc5tPu1Jq0vTSOiY/k24Wx82JRpWwZSqrala6sd5rWi6aNXvqcw==",
+      "dev": true,
+      "requires": {
+        "@types/node": "*"
+      }
+    },
+    "@types/glob": {
+      "version": "7.2.0",
+      "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.2.0.tgz",
+      "integrity": "sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==",
+      "dev": true,
+      "requires": {
+        "@types/minimatch": "*",
+        "@types/node": "*"
+      }
+    },
+    "@types/http-cache-semantics": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/@types/http-cache-semantics/-/http-cache-semantics-4.0.1.tgz",
+      "integrity": "sha512-SZs7ekbP8CN0txVG2xVRH6EgKmEm31BOxA07vkFaETzZz1xh+cbt8BcI0slpymvwhx5dlFnQG2rTlPVQn+iRPQ==",
+      "dev": true
+    },
+    "@types/json-schema": {
+      "version": "7.0.11",
+      "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.11.tgz",
+      "integrity": "sha512-wOuvG1SN4Us4rez+tylwwwCV1psiNVOkJeM3AUWUNWg/jDQY2+HE/444y5gc+jBmRqASOm2Oeh5c1axHobwRKQ==",
+      "dev": true
+    },
+    "@types/json5": {
+      "version": "0.0.29",
+      "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz",
+      "integrity": "sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==",
+      "dev": true
+    },
+    "@types/keyv": {
+      "version": "3.1.4",
+      "resolved": "https://registry.npmjs.org/@types/keyv/-/keyv-3.1.4.tgz",
+      "integrity": "sha512-BQ5aZNSCpj7D6K2ksrRCTmKRLEpnPvWDiLPfoGyhZ++8YtiK9d/3DBKPJgry359X/P1PfruyYwvnvwFjuEiEIg==",
+      "dev": true,
+      "requires": {
+        "@types/node": "*"
+      }
+    },
+    "@types/lru-cache": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/@types/lru-cache/-/lru-cache-5.1.1.tgz",
+      "integrity": "sha512-ssE3Vlrys7sdIzs5LOxCzTVMsU7i9oa/IaW92wF32JFb3CVczqOkru2xspuKczHEbG3nvmPY7IFqVmGGHdNbYw==",
+      "dev": true
+    },
+    "@types/minimatch": {
+      "version": "5.1.2",
+      "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-5.1.2.tgz",
+      "integrity": "sha512-K0VQKziLUWkVKiRVrx4a40iPaxTUefQmjtkQofBkYRcoaaL/8rhwDWww9qWbrgicNOgnpIsMxyNIUM4+n6dUIA==",
+      "dev": true
+    },
+    "@types/mocha": {
+      "version": "10.0.1",
+      "resolved": "https://registry.npmjs.org/@types/mocha/-/mocha-10.0.1.tgz",
+      "integrity": "sha512-/fvYntiO1GeICvqbQ3doGDIP97vWmvFt83GKguJ6prmQM2iXZfFcq6YE8KteFyRtX2/h5Hf91BYvPodJKFYv5Q==",
+      "dev": true
+    },
+    "@types/node": {
+      "version": "18.14.6",
+      "resolved": "https://registry.npmjs.org/@types/node/-/node-18.14.6.tgz",
+      "integrity": "sha512-93+VvleD3mXwlLI/xASjw0FzKcwzl3OdTCzm1LaRfqgS21gfFtK3zDXM5Op9TeeMsJVOaJ2VRDpT9q4Y3d0AvA==",
+      "dev": true
+    },
+    "@types/parse-json": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/@types/parse-json/-/parse-json-4.0.0.tgz",
+      "integrity": "sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA==",
+      "dev": true
+    },
+    "@types/pbkdf2": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/@types/pbkdf2/-/pbkdf2-3.1.0.tgz",
+      "integrity": "sha512-Cf63Rv7jCQ0LaL8tNXmEyqTHuIJxRdlS5vMh1mj5voN4+QFhVZnlZruezqpWYDiJ8UTzhP0VmeLXCmBk66YrMQ==",
+      "dev": true,
+      "requires": {
+        "@types/node": "*"
+      }
+    },
+    "@types/prettier": {
+      "version": "2.7.2",
+      "resolved": "https://registry.npmjs.org/@types/prettier/-/prettier-2.7.2.tgz",
+      "integrity": "sha512-KufADq8uQqo1pYKVIYzfKbJfBAc0sOeXqGbFaSpv8MRmC/zXgowNZmFcbngndGk922QDmOASEXUZCaY48gs4cg==",
+      "dev": true
+    },
+    "@types/qs": {
+      "version": "6.9.7",
+      "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.7.tgz",
+      "integrity": "sha512-FGa1F62FT09qcrueBA6qYTrJPVDzah9a+493+o2PCXsesWHIn27G98TsSMs3WPNbZIEj4+VJf6saSFpvD+3Zsw==",
+      "dev": true
+    },
+    "@types/responselike": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/@types/responselike/-/responselike-1.0.0.tgz",
+      "integrity": "sha512-85Y2BjiufFzaMIlvJDvTTB8Fxl2xfLo4HgmHzVBz08w4wDePCTjYw66PdrolO0kzli3yam/YCgRufyo1DdQVTA==",
+      "dev": true,
+      "requires": {
+        "@types/node": "*"
+      }
+    },
+    "@types/secp256k1": {
+      "version": "4.0.3",
+      "resolved": "https://registry.npmjs.org/@types/secp256k1/-/secp256k1-4.0.3.tgz",
+      "integrity": "sha512-Da66lEIFeIz9ltsdMZcpQvmrmmoqrfju8pm1BH8WbYjZSwUgCwXLb9C+9XYogwBITnbsSaMdVPb2ekf7TV+03w==",
+      "dev": true,
+      "requires": {
+        "@types/node": "*"
+      }
+    },
+    "@types/semver": {
+      "version": "7.3.13",
+      "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.3.13.tgz",
+      "integrity": "sha512-21cFJr9z3g5dW8B0CVI9g2O9beqaThGQ6ZFBqHfwhzLDKUxaqTIy3vnfah/UPkfOiF2pLq+tGz+W8RyCskuslw==",
+      "dev": true
+    },
+    "@typescript-eslint/eslint-plugin": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-5.54.1.tgz",
+      "integrity": "sha512-a2RQAkosH3d3ZIV08s3DcL/mcGc2M/UC528VkPULFxR9VnVPT8pBu0IyBAJJmVsCmhVfwQX1v6q+QGnmSe1bew==",
+      "dev": true,
+      "requires": {
+        "@typescript-eslint/scope-manager": "5.54.1",
+        "@typescript-eslint/type-utils": "5.54.1",
+        "@typescript-eslint/utils": "5.54.1",
+        "debug": "^4.3.4",
+        "grapheme-splitter": "^1.0.4",
+        "ignore": "^5.2.0",
+        "natural-compare-lite": "^1.4.0",
+        "regexpp": "^3.2.0",
+        "semver": "^7.3.7",
+        "tsutils": "^3.21.0"
+      },
+      "dependencies": {
+        "semver": {
+          "version": "7.5.4",
+          "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz",
+          "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==",
+          "dev": true,
+          "requires": {
+            "lru-cache": "^6.0.0"
+          }
+        }
+      }
+    },
+    "@typescript-eslint/parser": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-5.54.1.tgz",
+      "integrity": "sha512-8zaIXJp/nG9Ff9vQNh7TI+C3nA6q6iIsGJ4B4L6MhZ7mHnTMR4YP5vp2xydmFXIy8rpyIVbNAG44871LMt6ujg==",
+      "dev": true,
+      "requires": {
+        "@typescript-eslint/scope-manager": "5.54.1",
+        "@typescript-eslint/types": "5.54.1",
+        "@typescript-eslint/typescript-estree": "5.54.1",
+        "debug": "^4.3.4"
+      }
+    },
+    "@typescript-eslint/scope-manager": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-5.54.1.tgz",
+      "integrity": "sha512-zWKuGliXxvuxyM71UA/EcPxaviw39dB2504LqAmFDjmkpO8qNLHcmzlh6pbHs1h/7YQ9bnsO8CCcYCSA8sykUg==",
+      "dev": true,
+      "requires": {
+        "@typescript-eslint/types": "5.54.1",
+        "@typescript-eslint/visitor-keys": "5.54.1"
+      }
+    },
+    "@typescript-eslint/type-utils": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-5.54.1.tgz",
+      "integrity": "sha512-WREHsTz0GqVYLIbzIZYbmUUr95DKEKIXZNH57W3s+4bVnuF1TKe2jH8ZNH8rO1CeMY3U4j4UQeqPNkHMiGem3g==",
+      "dev": true,
+      "requires": {
+        "@typescript-eslint/typescript-estree": "5.54.1",
+        "@typescript-eslint/utils": "5.54.1",
+        "debug": "^4.3.4",
+        "tsutils": "^3.21.0"
+      }
+    },
+    "@typescript-eslint/types": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-5.54.1.tgz",
+      "integrity": "sha512-G9+1vVazrfAfbtmCapJX8jRo2E4MDXxgm/IMOF4oGh3kq7XuK3JRkOg6y2Qu1VsTRmWETyTkWt1wxy7X7/yLkw==",
+      "dev": true
+    },
+    "@typescript-eslint/typescript-estree": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-5.54.1.tgz",
+      "integrity": "sha512-bjK5t+S6ffHnVwA0qRPTZrxKSaFYocwFIkZx5k7pvWfsB1I57pO/0M0Skatzzw1sCkjJ83AfGTL0oFIFiDX3bg==",
+      "dev": true,
+      "requires": {
+        "@typescript-eslint/types": "5.54.1",
+        "@typescript-eslint/visitor-keys": "5.54.1",
+        "debug": "^4.3.4",
+        "globby": "^11.1.0",
+        "is-glob": "^4.0.3",
+        "semver": "^7.3.7",
+        "tsutils": "^3.21.0"
+      },
+      "dependencies": {
+        "semver": {
+          "version": "7.5.4",
+          "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz",
+          "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==",
+          "dev": true,
+          "requires": {
+            "lru-cache": "^6.0.0"
+          }
+        }
+      }
+    },
+    "@typescript-eslint/utils": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-5.54.1.tgz",
+      "integrity": "sha512-IY5dyQM8XD1zfDe5X8jegX6r2EVU5o/WJnLu/znLPWCBF7KNGC+adacXnt5jEYS9JixDcoccI6CvE4RCjHMzCQ==",
+      "dev": true,
+      "requires": {
+        "@types/json-schema": "^7.0.9",
+        "@types/semver": "^7.3.12",
+        "@typescript-eslint/scope-manager": "5.54.1",
+        "@typescript-eslint/types": "5.54.1",
+        "@typescript-eslint/typescript-estree": "5.54.1",
+        "eslint-scope": "^5.1.1",
+        "eslint-utils": "^3.0.0",
+        "semver": "^7.3.7"
+      },
+      "dependencies": {
+        "semver": {
+          "version": "7.5.4",
+          "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz",
+          "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==",
+          "dev": true,
+          "requires": {
+            "lru-cache": "^6.0.0"
+          }
+        }
+      }
+    },
+    "@typescript-eslint/visitor-keys": {
+      "version": "5.54.1",
+      "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-5.54.1.tgz",
+      "integrity": "sha512-q8iSoHTgwCfgcRJ2l2x+xCbu8nBlRAlsQ33k24Adj8eoVBE0f8dUeI+bAa8F84Mv05UGbAx57g2zrRsYIooqQg==",
+      "dev": true,
+      "requires": {
+        "@typescript-eslint/types": "5.54.1",
+        "eslint-visitor-keys": "^3.3.0"
+      }
+    },
+    "abbrev": {
+      "version": "1.0.9",
+      "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.0.9.tgz",
+      "integrity": "sha512-LEyx4aLEC3x6T0UguF6YILf+ntvmOaWsVfENmIW0E9H09vKlLDGelMjjSm0jkDHALj8A8quZ/HapKNigzwge+Q==",
+      "dev": true
+    },
+    "abort-controller": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz",
+      "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==",
+      "dev": true,
+      "requires": {
+        "event-target-shim": "^5.0.0"
+      }
+    },
+    "abortcontroller-polyfill": {
+      "version": "1.7.5",
+      "resolved": "https://registry.npmjs.org/abortcontroller-polyfill/-/abortcontroller-polyfill-1.7.5.tgz",
+      "integrity": "sha512-JMJ5soJWP18htbbxJjG7bG6yuI6pRhgJ0scHHTfkUjf6wjP912xZWvM+A4sJK3gqd9E8fcPbDnOefbA9Th/FIQ==",
+      "dev": true
+    },
+    "abstract-level": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/abstract-level/-/abstract-level-1.0.3.tgz",
+      "integrity": "sha512-t6jv+xHy+VYwc4xqZMn2Pa9DjcdzvzZmQGRjTFc8spIbRGHgBrEKbPq+rYXc7CCo0lxgYvSgKVg9qZAhpVQSjA==",
+      "dev": true,
+      "requires": {
+        "buffer": "^6.0.3",
+        "catering": "^2.1.0",
+        "is-buffer": "^2.0.5",
+        "level-supports": "^4.0.0",
+        "level-transcoder": "^1.0.1",
+        "module-error": "^1.0.1",
+        "queue-microtask": "^1.2.3"
+      },
+      "dependencies": {
+        "buffer": {
+          "version": "6.0.3",
+          "resolved": "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz",
+          "integrity": "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==",
+          "dev": true,
+          "requires": {
+            "base64-js": "^1.3.1",
+            "ieee754": "^1.2.1"
+          }
+        }
+      }
+    },
+    "accepts": {
+      "version": "1.3.8",
+      "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz",
+      "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==",
+      "dev": true,
+      "requires": {
+        "mime-types": "~2.1.34",
+        "negotiator": "0.6.3"
+      }
+    },
+    "acorn": {
+      "version": "8.8.2",
+      "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.8.2.tgz",
+      "integrity": "sha512-xjIYgE8HBrkpd/sJqOGNspf8uHG+NOHGOw6a/Urj8taM2EXfdNAH2oFcPeIFfsv3+kz/mJrS5VuMqbNLjCa2vw==",
+      "dev": true
+    },
+    "acorn-jsx": {
+      "version": "5.3.2",
+      "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz",
+      "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==",
+      "dev": true,
+      "requires": {}
+    },
+    "acorn-walk": {
+      "version": "8.2.0",
+      "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.2.0.tgz",
+      "integrity": "sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA==",
+      "dev": true
+    },
+    "address": {
+      "version": "1.2.2",
+      "resolved": "https://registry.npmjs.org/address/-/address-1.2.2.tgz",
+      "integrity": "sha512-4B/qKCfeE/ODUaAUpSwfzazo5x29WD4r3vXiWsB7I2mSDAihwEqKO+g8GELZUQSSAo5e1XTYh3ZVfLyxBc12nA==",
+      "dev": true
+    },
+    "adm-zip": {
+      "version": "0.4.16",
+      "resolved": "https://registry.npmjs.org/adm-zip/-/adm-zip-0.4.16.tgz",
+      "integrity": "sha512-TFi4HBKSGfIKsK5YCkKaaFG2m4PEDyViZmEwof3MTIgzimHLto6muaHVpbrljdIvIrFZzEq/p4nafOeLcYegrg==",
+      "dev": true
+    },
+    "aes-js": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/aes-js/-/aes-js-3.0.0.tgz",
+      "integrity": "sha512-H7wUZRn8WpTq9jocdxQ2c8x2sKo9ZVmzfRE13GiNJXfp7NcKYEdvl3vspKjXox6RIG2VtaRe4JFvxG4rqp2Zuw==",
+      "devOptional": true
+    },
+    "agent-base": {
+      "version": "6.0.2",
+      "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz",
+      "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==",
+      "dev": true,
+      "requires": {
+        "debug": "4"
+      }
+    },
+    "aggregate-error": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.1.0.tgz",
+      "integrity": "sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==",
+      "dev": true,
+      "requires": {
+        "clean-stack": "^2.0.0",
+        "indent-string": "^4.0.0"
+      }
+    },
+    "ajv": {
+      "version": "8.12.0",
+      "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.12.0.tgz",
+      "integrity": "sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==",
+      "dev": true,
+      "requires": {
+        "fast-deep-equal": "^3.1.1",
+        "json-schema-traverse": "^1.0.0",
+        "require-from-string": "^2.0.2",
+        "uri-js": "^4.2.2"
+      }
+    },
+    "amdefine": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/amdefine/-/amdefine-1.0.1.tgz",
+      "integrity": "sha512-S2Hw0TtNkMJhIabBwIojKL9YHO5T0n5eNqWJ7Lrlel/zDbftQpxpapi8tZs3X1HWa+u+QeydGmzzNU0m09+Rcg==",
+      "dev": true,
+      "optional": true
+    },
+    "ansi-colors": {
+      "version": "4.1.3",
+      "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.3.tgz",
+      "integrity": "sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw==",
+      "dev": true
+    },
+    "ansi-escapes": {
+      "version": "4.3.2",
+      "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz",
+      "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==",
+      "dev": true,
+      "requires": {
+        "type-fest": "^0.21.3"
+      }
+    },
+    "ansi-regex": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
+      "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
+      "dev": true
+    },
+    "ansi-styles": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
+      "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
+      "dev": true,
+      "requires": {
+        "color-convert": "^2.0.1"
+      }
+    },
+    "antlr4": {
+      "version": "4.12.0",
+      "resolved": "https://registry.npmjs.org/antlr4/-/antlr4-4.12.0.tgz",
+      "integrity": "sha512-23iB5IzXJZRZeK9TigzUyrNc9pSmNqAerJRBcNq1ETrmttMWRgaYZzC561IgEO3ygKsDJTYDTozABXa4b/fTQQ==",
+      "dev": true
+    },
+    "antlr4ts": {
+      "version": "0.5.0-alpha.4",
+      "resolved": "https://registry.npmjs.org/antlr4ts/-/antlr4ts-0.5.0-alpha.4.tgz",
+      "integrity": "sha512-WPQDt1B74OfPv/IMS2ekXAKkTZIHl88uMetg6q3OTqgFxZ/dxDXI0EWLyZid/1Pe6hTftyg5N7gel5wNAGxXyQ==",
+      "dev": true
+    },
+    "anymatch": {
+      "version": "3.1.3",
+      "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz",
+      "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==",
+      "dev": true,
+      "requires": {
+        "normalize-path": "^3.0.0",
+        "picomatch": "^2.0.4"
+      }
+    },
+    "arg": {
+      "version": "4.1.3",
+      "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz",
+      "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==",
+      "dev": true
+    },
+    "argparse": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz",
+      "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==",
+      "dev": true
+    },
+    "array-back": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/array-back/-/array-back-3.1.0.tgz",
+      "integrity": "sha512-TkuxA4UCOvxuDK6NZYXCalszEzj+TLszyASooky+i742l9TqsOdYCMJJupxRic61hwquNtppB3hgcuq9SVSH1Q==",
+      "dev": true
+    },
+    "array-flatten": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
+      "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==",
+      "dev": true
+    },
+    "array-includes": {
+      "version": "3.1.6",
+      "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.6.tgz",
+      "integrity": "sha512-sgTbLvL6cNnw24FnbaDyjmvddQ2ML8arZsgaJhoABMoplz/4QRhtrYS+alr1BUM1Bwp6dhx8vVCBSLG+StwOFw==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "es-abstract": "^1.20.4",
+        "get-intrinsic": "^1.1.3",
+        "is-string": "^1.0.7"
+      }
+    },
+    "array-union": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz",
+      "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==",
+      "dev": true
+    },
+    "array-uniq": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz",
+      "integrity": "sha512-MNha4BWQ6JbwhFhj03YK552f7cb3AzoE8SzeljgChvL1dl3IcvggXVz1DilzySZkCja+CXuZbdW7yATchWn8/Q==",
+      "dev": true
+    },
+    "array.prototype.flat": {
+      "version": "1.3.1",
+      "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.3.1.tgz",
+      "integrity": "sha512-roTU0KWIOmJ4DRLmwKd19Otg0/mT3qPNt0Qb3GWW8iObuZXxrjB/pzn0R3hqpRSWg4HCwqx+0vwOnWnvlOyeIA==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "es-abstract": "^1.20.4",
+        "es-shim-unscopables": "^1.0.0"
+      }
+    },
+    "array.prototype.flatmap": {
+      "version": "1.3.1",
+      "resolved": "https://registry.npmjs.org/array.prototype.flatmap/-/array.prototype.flatmap-1.3.1.tgz",
+      "integrity": "sha512-8UGn9O1FDVvMNB0UlLv4voxRMze7+FpHyF5mSMRjWHUMlpoDViniy05870VlxhfgTnLbpuwTzvD76MTtWxB/mQ==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "es-abstract": "^1.20.4",
+        "es-shim-unscopables": "^1.0.0"
+      }
+    },
+    "asap": {
+      "version": "2.0.6",
+      "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz",
+      "integrity": "sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==",
+      "dev": true
+    },
+    "asn1": {
+      "version": "0.2.6",
+      "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.6.tgz",
+      "integrity": "sha512-ix/FxPn0MDjeyJ7i/yoHGFt/EX6LyNbxSEhPPXODPL+KB0VPk86UYfL0lMdy+KCnv+fmvIzySwaK5COwqVbWTQ==",
+      "dev": true,
+      "requires": {
+        "safer-buffer": "~2.1.0"
+      }
+    },
+    "assert-plus": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz",
+      "integrity": "sha512-NfJ4UzBCcQGLDlQq7nHxH+tv3kyZ0hHQqF5BO6J7tNJeP5do1llPr8dZ8zHonfhAu0PHAdMkSo+8o0wxg9lZWw==",
+      "dev": true
+    },
+    "assertion-error": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-1.1.0.tgz",
+      "integrity": "sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==",
+      "dev": true
+    },
+    "ast-parents": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/ast-parents/-/ast-parents-0.0.1.tgz",
+      "integrity": "sha512-XHusKxKz3zoYk1ic8Un640joHbFMhbqneyoZfoKnEGtf2ey9Uh/IdpcQplODdO/kENaMIWsD0nJm4+wX3UNLHA==",
+      "dev": true
+    },
+    "astral-regex": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz",
+      "integrity": "sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==",
+      "dev": true
+    },
+    "async": {
+      "version": "2.6.4",
+      "resolved": "https://registry.npmjs.org/async/-/async-2.6.4.tgz",
+      "integrity": "sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==",
+      "dev": true,
+      "requires": {
+        "lodash": "^4.17.14"
+      }
+    },
+    "async-eventemitter": {
+      "version": "0.2.4",
+      "resolved": "https://registry.npmjs.org/async-eventemitter/-/async-eventemitter-0.2.4.tgz",
+      "integrity": "sha512-pd20BwL7Yt1zwDFy+8MX8F1+WCT8aQeKj0kQnTrH9WaeRETlRamVhD0JtRPmrV4GfOJ2F9CvdQkZeZhnh2TuHw==",
+      "dev": true,
+      "requires": {
+        "async": "^2.4.0"
+      }
+    },
+    "async-limiter": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz",
+      "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==",
+      "dev": true
+    },
+    "asynckit": {
+      "version": "0.4.0",
+      "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
+      "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==",
+      "dev": true
+    },
+    "available-typed-arrays": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.5.tgz",
+      "integrity": "sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw==",
+      "dev": true
+    },
+    "aws-sign2": {
+      "version": "0.7.0",
+      "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz",
+      "integrity": "sha512-08kcGqnYf/YmjoRhfxyu+CLxBjUtHLXLXX/vUfx9l2LYzG3c1m61nrpyFUZI6zeS+Li/wWMMidD9KgrqtGq3mA==",
+      "dev": true
+    },
+    "aws4": {
+      "version": "1.12.0",
+      "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.12.0.tgz",
+      "integrity": "sha512-NmWvPnx0F1SfrQbYwOi7OeaNGokp9XhzNioJ/CSBs8Qa4vxug81mhJEAVZwxXuBmYB5KDRfMq/F3RR0BIU7sWg==",
+      "dev": true
+    },
+    "axios": {
+      "version": "0.27.2",
+      "resolved": "https://registry.npmjs.org/axios/-/axios-0.27.2.tgz",
+      "integrity": "sha512-t+yRIyySRTp/wua5xEr+z1q60QmLq8ABsS5O9Me1AsE5dfKqgnCFzwiCZZ/cGNd1lq4/7akDWMxdhVlucjmnOQ==",
+      "dev": true,
+      "requires": {
+        "follow-redirects": "^1.14.9",
+        "form-data": "^4.0.0"
+      },
+      "dependencies": {
+        "form-data": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
+          "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
+          "dev": true,
+          "requires": {
+            "asynckit": "^0.4.0",
+            "combined-stream": "^1.0.8",
+            "mime-types": "^2.1.12"
+          }
+        }
+      }
+    },
+    "balanced-match": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
+      "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
+      "dev": true
+    },
+    "base-x": {
+      "version": "3.0.9",
+      "resolved": "https://registry.npmjs.org/base-x/-/base-x-3.0.9.tgz",
+      "integrity": "sha512-H7JU6iBHTal1gp56aKoaa//YUxEaAOUiydvrV/pILqIHXTtqxSkATOnDA2u+jZ/61sD+L/412+7kzXRtWukhpQ==",
+      "dev": true,
+      "requires": {
+        "safe-buffer": "^5.0.1"
+      }
+    },
+    "base64-js": {
+      "version": "1.5.1",
+      "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz",
+      "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==",
+      "dev": true
+    },
+    "bcrypt-pbkdf": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz",
+      "integrity": "sha512-qeFIXtP4MSoi6NLqO12WfqARWWuCKi2Rn/9hJLEmtB5yTNr9DqFWkJRCf2qShWzPeAMRnOgCrq0sg/KLv5ES9w==",
+      "dev": true,
+      "requires": {
+        "tweetnacl": "^0.14.3"
+      }
+    },
+    "bech32": {
+      "version": "1.1.4",
+      "resolved": "https://registry.npmjs.org/bech32/-/bech32-1.1.4.tgz",
+      "integrity": "sha512-s0IrSOzLlbvX7yp4WBfPITzpAU8sqQcpsmwXDiKwrG4r491vwCO/XpejasRNl0piBMe/DvP4Tz0mIS/X1DPJBQ==",
+      "devOptional": true
+    },
+    "big-integer": {
+      "version": "1.6.36",
+      "resolved": "https://registry.npmjs.org/big-integer/-/big-integer-1.6.36.tgz",
+      "integrity": "sha512-t70bfa7HYEA1D9idDbmuv7YbsbVkQ+Hp+8KFSul4aE5e/i1bjCNIRYJZlA8Q8p0r9T8cF/RVvwUgRA//FydEyg==",
+      "dev": true
+    },
+    "big.js": {
+      "version": "6.2.1",
+      "resolved": "https://registry.npmjs.org/big.js/-/big.js-6.2.1.tgz",
+      "integrity": "sha512-bCtHMwL9LeDIozFn+oNhhFoq+yQ3BNdnsLSASUxLciOb1vgvpHsIO1dsENiGMgbb4SkP5TrzWzRiLddn8ahVOQ==",
+      "dev": true
+    },
+    "bigint-crypto-utils": {
+      "version": "3.1.8",
+      "resolved": "https://registry.npmjs.org/bigint-crypto-utils/-/bigint-crypto-utils-3.1.8.tgz",
+      "integrity": "sha512-+VMV9Laq8pXLBKKKK49nOoq9bfR3j7NNQAtbA617a4nw9bVLo8rsqkKMBgM2AJWlNX9fEIyYaYX+d0laqYV4tw==",
+      "dev": true,
+      "requires": {
+        "bigint-mod-arith": "^3.1.0"
+      }
+    },
+    "bigint-mod-arith": {
+      "version": "3.1.2",
+      "resolved": "https://registry.npmjs.org/bigint-mod-arith/-/bigint-mod-arith-3.1.2.tgz",
+      "integrity": "sha512-nx8J8bBeiRR+NlsROFH9jHswW5HO8mgfOSqW0AmjicMMvaONDa8AO+5ViKDUUNytBPWiwfvZP4/Bj4Y3lUfvgQ==",
+      "dev": true
+    },
+    "bignumber.js": {
+      "version": "7.2.1",
+      "resolved": "https://registry.npmjs.org/bignumber.js/-/bignumber.js-7.2.1.tgz",
+      "integrity": "sha512-S4XzBk5sMB+Rcb/LNcpzXr57VRTxgAvaAEDAl1AwRx27j00hT84O6OkteE7u8UB3NuaaygCRrEpqox4uDOrbdQ==",
+      "dev": true
+    },
+    "binary-extensions": {
+      "version": "2.2.0",
+      "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz",
+      "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==",
+      "dev": true
+    },
+    "bl": {
+      "version": "4.1.0",
+      "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz",
+      "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==",
+      "dev": true,
+      "requires": {
+        "buffer": "^5.5.0",
+        "inherits": "^2.0.4",
+        "readable-stream": "^3.4.0"
+      }
+    },
+    "blakejs": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/blakejs/-/blakejs-1.2.1.tgz",
+      "integrity": "sha512-QXUSXI3QVc/gJME0dBpXrag1kbzOqCjCX8/b54ntNyW6sjtoqxqRk3LTmXzaJoh71zMsDCjM+47jS7XiwN/+fQ==",
+      "dev": true
+    },
+    "bluebird": {
+      "version": "3.7.2",
+      "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz",
+      "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==",
+      "dev": true
+    },
+    "bn-chai": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/bn-chai/-/bn-chai-1.0.1.tgz",
+      "integrity": "sha512-7rJXt21DwYiLLpvzLaACixBBoUGkRV1iuFD3wElEhw8Ji9IiY/QsJRtvW+c7ChRgEOyLQkGaSGFUUqBKm21SNA==",
+      "dev": true,
+      "requires": {}
+    },
+    "bn.js": {
+      "version": "5.2.1",
+      "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-5.2.1.tgz",
+      "integrity": "sha512-eXRvHzWyYPBuB4NBy0cmYQjGitUrtqwbvlzP3G6VFnNRbsZQIxQ10PbKKHt8gZ/HW/D/747aDl+QkDqg3KQLMQ==",
+      "devOptional": true
+    },
+    "body-parser": {
+      "version": "1.20.2",
+      "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.2.tgz",
+      "integrity": "sha512-ml9pReCu3M61kGlqoTm2umSXTlRTuGTx0bfYj+uIUKKYycG5NtSbeetV3faSU6R7ajOPw0g/J1PvK4qNy7s5bA==",
+      "dev": true,
+      "requires": {
+        "bytes": "3.1.2",
+        "content-type": "~1.0.5",
+        "debug": "2.6.9",
+        "depd": "2.0.0",
+        "destroy": "1.2.0",
+        "http-errors": "2.0.0",
+        "iconv-lite": "0.4.24",
+        "on-finished": "2.4.1",
+        "qs": "6.11.0",
+        "raw-body": "2.5.2",
+        "type-is": "~1.6.18",
+        "unpipe": "1.0.0"
+      },
+      "dependencies": {
+        "debug": {
+          "version": "2.6.9",
+          "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+          "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+          "dev": true,
+          "requires": {
+            "ms": "2.0.0"
+          }
+        },
+        "ms": {
+          "version": "2.0.0",
+          "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+          "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
+          "dev": true
+        }
+      }
+    },
+    "boolbase": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz",
+      "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==",
+      "dev": true
+    },
+    "brace-expansion": {
+      "version": "1.1.11",
+      "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
+      "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
+      "dev": true,
+      "requires": {
+        "balanced-match": "^1.0.0",
+        "concat-map": "0.0.1"
+      }
+    },
+    "braces": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz",
+      "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==",
+      "dev": true,
+      "requires": {
+        "fill-range": "^7.0.1"
+      }
+    },
+    "brorand": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz",
+      "integrity": "sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==",
+      "devOptional": true
+    },
+    "browser-level": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/browser-level/-/browser-level-1.0.1.tgz",
+      "integrity": "sha512-XECYKJ+Dbzw0lbydyQuJzwNXtOpbMSq737qxJN11sIRTErOMShvDpbzTlgju7orJKvx4epULolZAuJGLzCmWRQ==",
+      "dev": true,
+      "requires": {
+        "abstract-level": "^1.0.2",
+        "catering": "^2.1.1",
+        "module-error": "^1.0.2",
+        "run-parallel-limit": "^1.1.0"
+      }
+    },
+    "browser-stdout": {
+      "version": "1.3.1",
+      "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz",
+      "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==",
+      "dev": true
+    },
+    "browserify-aes": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/browserify-aes/-/browserify-aes-1.2.0.tgz",
+      "integrity": "sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA==",
+      "dev": true,
+      "requires": {
+        "buffer-xor": "^1.0.3",
+        "cipher-base": "^1.0.0",
+        "create-hash": "^1.1.0",
+        "evp_bytestokey": "^1.0.3",
+        "inherits": "^2.0.1",
+        "safe-buffer": "^5.0.1"
+      }
+    },
+    "bs58": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/bs58/-/bs58-4.0.1.tgz",
+      "integrity": "sha512-Ok3Wdf5vOIlBrgCvTq96gBkJw+JUEzdBgyaza5HLtPm7yTHkjRy8+JzNyHF7BHa0bNWOQIp3m5YF0nnFcOIKLw==",
+      "dev": true,
+      "requires": {
+        "base-x": "^3.0.2"
+      }
+    },
+    "bs58check": {
+      "version": "2.1.2",
+      "resolved": "https://registry.npmjs.org/bs58check/-/bs58check-2.1.2.tgz",
+      "integrity": "sha512-0TS1jicxdU09dwJMNZtVAfzPi6Q6QeN0pM1Fkzrjn+XYHvzMKPU3pHVpva+769iNVSfIYWf7LJ6WR+BuuMf8cA==",
+      "dev": true,
+      "requires": {
+        "bs58": "^4.0.0",
+        "create-hash": "^1.1.0",
+        "safe-buffer": "^5.1.2"
+      }
+    },
+    "buffer": {
+      "version": "5.7.1",
+      "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz",
+      "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==",
+      "dev": true,
+      "requires": {
+        "base64-js": "^1.3.1",
+        "ieee754": "^1.1.13"
+      }
+    },
+    "buffer-from": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz",
+      "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==",
+      "dev": true
+    },
+    "buffer-to-arraybuffer": {
+      "version": "0.0.5",
+      "resolved": "https://registry.npmjs.org/buffer-to-arraybuffer/-/buffer-to-arraybuffer-0.0.5.tgz",
+      "integrity": "sha512-3dthu5CYiVB1DEJp61FtApNnNndTckcqe4pFcLdvHtrpG+kcyekCJKg4MRiDcFW7A6AODnXB9U4dwQiCW5kzJQ==",
+      "dev": true
+    },
+    "buffer-xor": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/buffer-xor/-/buffer-xor-1.0.3.tgz",
+      "integrity": "sha512-571s0T7nZWK6vB67HI5dyUF7wXiNcfaPPPTl6zYCNApANjIvYJTg7hlud/+cJpdAhS7dVzqMLmfhfHR3rAcOjQ==",
+      "dev": true
+    },
+    "bufferutil": {
+      "version": "4.0.7",
+      "resolved": "https://registry.npmjs.org/bufferutil/-/bufferutil-4.0.7.tgz",
+      "integrity": "sha512-kukuqc39WOHtdxtw4UScxF/WVnMFVSQVKhtx3AjZJzhd0RGZZldcrfSEbVsWWe6KNH253574cq5F+wpv0G9pJw==",
+      "dev": true,
+      "requires": {
+        "node-gyp-build": "^4.3.0"
+      }
+    },
+    "bytes": {
+      "version": "3.1.2",
+      "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz",
+      "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==",
+      "dev": true
+    },
+    "cacheable-lookup": {
+      "version": "6.1.0",
+      "resolved": "https://registry.npmjs.org/cacheable-lookup/-/cacheable-lookup-6.1.0.tgz",
+      "integrity": "sha512-KJ/Dmo1lDDhmW2XDPMo+9oiy/CeqosPguPCrgcVzKyZrL6pM1gU2GmPY/xo6OQPTUaA/c0kwHuywB4E6nmT9ww==",
+      "dev": true
+    },
+    "cacheable-request": {
+      "version": "7.0.2",
+      "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-7.0.2.tgz",
+      "integrity": "sha512-pouW8/FmiPQbuGpkXQ9BAPv/Mo5xDGANgSNXzTzJ8DrKGuXOssM4wIQRjfanNRh3Yu5cfYPvcorqbhg2KIJtew==",
+      "dev": true,
+      "requires": {
+        "clone-response": "^1.0.2",
+        "get-stream": "^5.1.0",
+        "http-cache-semantics": "^4.0.0",
+        "keyv": "^4.0.0",
+        "lowercase-keys": "^2.0.0",
+        "normalize-url": "^6.0.1",
+        "responselike": "^2.0.0"
+      },
+      "dependencies": {
+        "get-stream": {
+          "version": "5.2.0",
+          "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz",
+          "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==",
+          "dev": true,
+          "requires": {
+            "pump": "^3.0.0"
+          }
+        },
+        "lowercase-keys": {
+          "version": "2.0.0",
+          "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz",
+          "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==",
+          "dev": true
+        }
+      }
+    },
+    "call-bind": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz",
+      "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==",
+      "dev": true,
+      "requires": {
+        "function-bind": "^1.1.1",
+        "get-intrinsic": "^1.0.2"
+      }
+    },
+    "callsites": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz",
+      "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==",
+      "dev": true
+    },
+    "camel-case": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/camel-case/-/camel-case-3.0.0.tgz",
+      "integrity": "sha512-+MbKztAYHXPr1jNTSKQF52VpcFjwY5RkR7fxksV8Doo4KAYc5Fl4UJRgthBbTmEx8C54DqahhbLJkDwjI3PI/w==",
+      "dev": true,
+      "requires": {
+        "no-case": "^2.2.0",
+        "upper-case": "^1.1.1"
+      }
+    },
+    "camelcase": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-3.0.0.tgz",
+      "integrity": "sha512-4nhGqUkc4BqbBBB4Q6zLuD7lzzrHYrjKGeYaEji/3tFR5VdJu9v+LilhGIVe8wxEJPPOeWo7eg8dwY13TZ1BNg==",
+      "dev": true
+    },
+    "caseless": {
+      "version": "0.12.0",
+      "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz",
+      "integrity": "sha512-4tYFyifaFfGacoiObjJegolkwSU4xQNGbVgUiNYVUxbQ2x2lUsFvY4hVgVzGiIe6WLOPqycWXA40l+PWsxthUw==",
+      "dev": true
+    },
+    "catering": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/catering/-/catering-2.1.1.tgz",
+      "integrity": "sha512-K7Qy8O9p76sL3/3m7/zLKbRkyOlSZAgzEaLhyj2mXS8PsCud2Eo4hAb8aLtZqHh0QGqLcb9dlJSu6lHRVENm1w==",
+      "dev": true
+    },
+    "cbor": {
+      "version": "8.1.0",
+      "resolved": "https://registry.npmjs.org/cbor/-/cbor-8.1.0.tgz",
+      "integrity": "sha512-DwGjNW9omn6EwP70aXsn7FQJx5kO12tX0bZkaTjzdVFM6/7nhA4t0EENocKGx6D2Bch9PE2KzCUf5SceBdeijg==",
+      "dev": true,
+      "requires": {
+        "nofilter": "^3.1.0"
+      }
+    },
+    "chai": {
+      "version": "4.3.7",
+      "resolved": "https://registry.npmjs.org/chai/-/chai-4.3.7.tgz",
+      "integrity": "sha512-HLnAzZ2iupm25PlN0xFreAlBA5zaBSv3og0DdeGA4Ar6h6rJ3A0rolRUKJhSF2V10GZKDgWF/VmAEsNWjCRB+A==",
+      "dev": true,
+      "requires": {
+        "assertion-error": "^1.1.0",
+        "check-error": "^1.0.2",
+        "deep-eql": "^4.1.2",
+        "get-func-name": "^2.0.0",
+        "loupe": "^2.3.1",
+        "pathval": "^1.1.1",
+        "type-detect": "^4.0.5"
+      }
+    },
+    "chai-as-promised": {
+      "version": "7.1.1",
+      "resolved": "https://registry.npmjs.org/chai-as-promised/-/chai-as-promised-7.1.1.tgz",
+      "integrity": "sha512-azL6xMoi+uxu6z4rhWQ1jbdUhOMhis2PvscD/xjLqNMkv3BPPp2JyyuTHOrf9BOosGpNQ11v6BKv/g57RXbiaA==",
+      "dev": true,
+      "requires": {
+        "check-error": "^1.0.2"
+      }
+    },
+    "chai-bignumber": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/chai-bignumber/-/chai-bignumber-3.1.0.tgz",
+      "integrity": "sha512-omxEc80jAU+pZwRmoWr3aEzeLad4JW3iBhLRQlgISvghBdIxrMT7mVAGsDz4WSyCkKowENshH2j9OABAhld7QQ==",
+      "dev": true
+    },
+    "chai-string": {
+      "version": "1.5.0",
+      "resolved": "https://registry.npmjs.org/chai-string/-/chai-string-1.5.0.tgz",
+      "integrity": "sha512-sydDC3S3pNAQMYwJrs6dQX0oBQ6KfIPuOZ78n7rocW0eJJlsHPh2t3kwW7xfwYA/1Bf6/arGtSUo16rxR2JFlw==",
+      "dev": true,
+      "requires": {}
+    },
+    "chalk": {
+      "version": "4.1.2",
+      "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz",
+      "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==",
+      "dev": true,
+      "requires": {
+        "ansi-styles": "^4.1.0",
+        "supports-color": "^7.1.0"
+      }
+    },
+    "change-case": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/change-case/-/change-case-3.0.2.tgz",
+      "integrity": "sha512-Mww+SLF6MZ0U6kdg11algyKd5BARbyM4TbFBepwowYSR5ClfQGCGtxNXgykpN0uF/bstWeaGDT4JWaDh8zWAHA==",
+      "dev": true,
+      "requires": {
+        "camel-case": "^3.0.0",
+        "constant-case": "^2.0.0",
+        "dot-case": "^2.1.0",
+        "header-case": "^1.0.0",
+        "is-lower-case": "^1.1.0",
+        "is-upper-case": "^1.1.0",
+        "lower-case": "^1.1.1",
+        "lower-case-first": "^1.0.0",
+        "no-case": "^2.3.2",
+        "param-case": "^2.1.0",
+        "pascal-case": "^2.0.0",
+        "path-case": "^2.1.0",
+        "sentence-case": "^2.1.0",
+        "snake-case": "^2.1.0",
+        "swap-case": "^1.1.0",
+        "title-case": "^2.1.0",
+        "upper-case": "^1.1.1",
+        "upper-case-first": "^1.1.0"
+      }
+    },
+    "charenc": {
+      "version": "0.0.2",
+      "resolved": "https://registry.npmjs.org/charenc/-/charenc-0.0.2.tgz",
+      "integrity": "sha512-yrLQ/yVUFXkzg7EDQsPieE/53+0RlaWTs+wBrvW36cyilJ2SaDWfl4Yj7MtLTXleV9uEKefbAGUPv2/iWSooRA==",
+      "dev": true
+    },
+    "check-error": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/check-error/-/check-error-1.0.2.tgz",
+      "integrity": "sha512-BrgHpW9NURQgzoNyjfq0Wu6VFO6D7IZEmJNdtgNqpzGG8RuNFHt2jQxWlAs4HMe119chBnv+34syEZtc6IhLtA==",
+      "dev": true
+    },
+    "cheerio": {
+      "version": "1.0.0-rc.12",
+      "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.0.0-rc.12.tgz",
+      "integrity": "sha512-VqR8m68vM46BNnuZ5NtnGBKIE/DfN0cRIzg9n40EIq9NOv90ayxLBXA8fXC5gquFRGJSTRqBq25Jt2ECLR431Q==",
+      "dev": true,
+      "requires": {
+        "cheerio-select": "^2.1.0",
+        "dom-serializer": "^2.0.0",
+        "domhandler": "^5.0.3",
+        "domutils": "^3.0.1",
+        "htmlparser2": "^8.0.1",
+        "parse5": "^7.0.0",
+        "parse5-htmlparser2-tree-adapter": "^7.0.0"
+      }
+    },
+    "cheerio-select": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/cheerio-select/-/cheerio-select-2.1.0.tgz",
+      "integrity": "sha512-9v9kG0LvzrlcungtnJtpGNxY+fzECQKhK4EGJX2vByejiMX84MFNQw4UxPJl3bFbTMw+Dfs37XaIkCwTZfLh4g==",
+      "dev": true,
+      "requires": {
+        "boolbase": "^1.0.0",
+        "css-select": "^5.1.0",
+        "css-what": "^6.1.0",
+        "domelementtype": "^2.3.0",
+        "domhandler": "^5.0.3",
+        "domutils": "^3.0.1"
+      }
+    },
+    "child_process": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/child_process/-/child_process-1.0.2.tgz",
+      "integrity": "sha512-Wmza/JzL0SiWz7kl6MhIKT5ceIlnFPJX+lwUGj7Clhy5MMldsSoJR0+uvRzOS5Kv45Mq7t1PoE8TsOA9bzvb6g==",
+      "dev": true
+    },
+    "chokidar": {
+      "version": "3.5.3",
+      "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz",
+      "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==",
+      "dev": true,
+      "requires": {
+        "anymatch": "~3.1.2",
+        "braces": "~3.0.2",
+        "fsevents": "~2.3.2",
+        "glob-parent": "~5.1.2",
+        "is-binary-path": "~2.1.0",
+        "is-glob": "~4.0.1",
+        "normalize-path": "~3.0.0",
+        "readdirp": "~3.6.0"
+      }
+    },
+    "chownr": {
+      "version": "1.1.4",
+      "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz",
+      "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==",
+      "dev": true
+    },
+    "ci-info": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz",
+      "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==",
+      "dev": true
+    },
+    "cids": {
+      "version": "0.7.5",
+      "resolved": "https://registry.npmjs.org/cids/-/cids-0.7.5.tgz",
+      "integrity": "sha512-zT7mPeghoWAu+ppn8+BS1tQ5qGmbMfB4AregnQjA/qHY3GC1m1ptI9GkWNlgeu38r7CuRdXB47uY2XgAYt6QVA==",
+      "dev": true,
+      "requires": {
+        "buffer": "^5.5.0",
+        "class-is": "^1.1.0",
+        "multibase": "~0.6.0",
+        "multicodec": "^1.0.0",
+        "multihashes": "~0.4.15"
+      },
+      "dependencies": {
+        "multicodec": {
+          "version": "1.0.4",
+          "resolved": "https://registry.npmjs.org/multicodec/-/multicodec-1.0.4.tgz",
+          "integrity": "sha512-NDd7FeS3QamVtbgfvu5h7fd1IlbaC4EQ0/pgU4zqE2vdHCmBGsUa0TiM8/TdSeG6BMPC92OOCf8F1ocE/Wkrrg==",
+          "dev": true,
+          "requires": {
+            "buffer": "^5.6.0",
+            "varint": "^5.0.0"
+          }
+        }
+      }
+    },
+    "cipher-base": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/cipher-base/-/cipher-base-1.0.4.tgz",
+      "integrity": "sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q==",
+      "dev": true,
+      "requires": {
+        "inherits": "^2.0.1",
+        "safe-buffer": "^5.0.1"
+      }
+    },
+    "class-is": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/class-is/-/class-is-1.1.0.tgz",
+      "integrity": "sha512-rhjH9AG1fvabIDoGRVH587413LPjTZgmDF9fOFCbFJQV4yuocX1mHxxvXI4g3cGwbVY9wAYIoKlg1N79frJKQw==",
+      "dev": true
+    },
+    "classic-level": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/classic-level/-/classic-level-1.2.0.tgz",
+      "integrity": "sha512-qw5B31ANxSluWz9xBzklRWTUAJ1SXIdaVKTVS7HcTGKOAmExx65Wo5BUICW+YGORe2FOUaDghoI9ZDxj82QcFg==",
+      "dev": true,
+      "requires": {
+        "abstract-level": "^1.0.2",
+        "catering": "^2.1.0",
+        "module-error": "^1.0.1",
+        "napi-macros": "~2.0.0",
+        "node-gyp-build": "^4.3.0"
+      }
+    },
+    "clean-stack": {
+      "version": "2.2.0",
+      "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz",
+      "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==",
+      "dev": true
+    },
+    "cli-cursor": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz",
+      "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==",
+      "dev": true,
+      "requires": {
+        "restore-cursor": "^3.1.0"
+      }
+    },
+    "cli-spinners": {
+      "version": "2.7.0",
+      "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.7.0.tgz",
+      "integrity": "sha512-qu3pN8Y3qHNgE2AFweciB1IfMnmZ/fsNTEE+NOFjmGB2F/7rLhnhzppvpCnN4FovtP26k8lHyy9ptEbNwWFLzw==",
+      "dev": true
+    },
+    "cli-table3": {
+      "version": "0.6.3",
+      "resolved": "https://registry.npmjs.org/cli-table3/-/cli-table3-0.6.3.tgz",
+      "integrity": "sha512-w5Jac5SykAeZJKntOxJCrm63Eg5/4dhMWIcuTbo9rpE+brgaSZo0RuNJZeOyMgsUdhDeojvgyQLmjI+K50ZGyg==",
+      "dev": true,
+      "requires": {
+        "@colors/colors": "1.5.0",
+        "string-width": "^4.2.0"
+      }
+    },
+    "cliui": {
+      "version": "3.2.0",
+      "resolved": "https://registry.npmjs.org/cliui/-/cliui-3.2.0.tgz",
+      "integrity": "sha512-0yayqDxWQbqk3ojkYqUKqaAQ6AfNKeKWRNA8kR0WXzAsdHpP4BIaOmMAG87JGuO6qcobyW4GjxHd9PmhEd+T9w==",
+      "dev": true,
+      "requires": {
+        "string-width": "^1.0.1",
+        "strip-ansi": "^3.0.1",
+        "wrap-ansi": "^2.0.0"
+      },
+      "dependencies": {
+        "ansi-regex": {
+          "version": "2.1.1",
+          "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
+          "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==",
+          "dev": true
+        },
+        "is-fullwidth-code-point": {
+          "version": "1.0.0",
+          "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz",
+          "integrity": "sha512-1pqUqRjkhPJ9miNq9SwMfdvi6lBJcd6eFxvfaivQhaH3SgisfiuudvFntdKOmxuee/77l+FPjKrQjWvmPjWrRw==",
+          "dev": true,
+          "requires": {
+            "number-is-nan": "^1.0.0"
+          }
+        },
+        "string-width": {
+          "version": "1.0.2",
+          "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz",
+          "integrity": "sha512-0XsVpQLnVCXHJfyEs8tC0zpTVIr5PKKsQtkT29IwupnPTjtPmQ3xT/4yCREF9hYkV/3M3kzcUTSAZT6a6h81tw==",
+          "dev": true,
+          "requires": {
+            "code-point-at": "^1.0.0",
+            "is-fullwidth-code-point": "^1.0.0",
+            "strip-ansi": "^3.0.0"
+          }
+        },
+        "strip-ansi": {
+          "version": "3.0.1",
+          "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
+          "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==",
+          "dev": true,
+          "requires": {
+            "ansi-regex": "^2.0.0"
+          }
+        }
+      }
+    },
+    "clone": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz",
+      "integrity": "sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg==",
+      "dev": true
+    },
+    "clone-response": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.3.tgz",
+      "integrity": "sha512-ROoL94jJH2dUVML2Y/5PEDNaSHgeOdSDicUyS7izcF63G6sTc/FTjLub4b8Il9S8S0beOfYt0TaA5qvFK+w0wA==",
+      "dev": true,
+      "requires": {
+        "mimic-response": "^1.0.0"
+      }
+    },
+    "code-point-at": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz",
+      "integrity": "sha512-RpAVKQA5T63xEj6/giIbUEtZwJ4UFIc3ZtvEkiaUERylqe8xb5IvqcgOurZLahv93CLKfxcw5YI+DZcUBRyLXA==",
+      "dev": true
+    },
+    "color-convert": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
+      "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
+      "dev": true,
+      "requires": {
+        "color-name": "~1.1.4"
+      }
+    },
+    "color-name": {
+      "version": "1.1.4",
+      "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
+      "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
+      "dev": true
+    },
+    "colors": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/colors/-/colors-1.4.0.tgz",
+      "integrity": "sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==",
+      "dev": true
+    },
+    "combined-stream": {
+      "version": "1.0.8",
+      "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
+      "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
+      "dev": true,
+      "requires": {
+        "delayed-stream": "~1.0.0"
+      }
+    },
+    "command-exists": {
+      "version": "1.2.9",
+      "resolved": "https://registry.npmjs.org/command-exists/-/command-exists-1.2.9.tgz",
+      "integrity": "sha512-LTQ/SGc+s0Xc0Fu5WaKnR0YiygZkm9eKFvyS+fRsU7/ZWFF8ykFM6Pc9aCVf1+xasOOZpO3BAVgVrKvsqKHV7w==",
+      "dev": true
+    },
+    "command-line-args": {
+      "version": "5.2.1",
+      "resolved": "https://registry.npmjs.org/command-line-args/-/command-line-args-5.2.1.tgz",
+      "integrity": "sha512-H4UfQhZyakIjC74I9d34fGYDwk3XpSr17QhEd0Q3I9Xq1CETHo4Hcuo87WyWHpAF1aSLjLRf5lD9ZGX2qStUvg==",
+      "dev": true,
+      "requires": {
+        "array-back": "^3.1.0",
+        "find-replace": "^3.0.0",
+        "lodash.camelcase": "^4.3.0",
+        "typical": "^4.0.0"
+      }
+    },
+    "command-line-usage": {
+      "version": "6.1.3",
+      "resolved": "https://registry.npmjs.org/command-line-usage/-/command-line-usage-6.1.3.tgz",
+      "integrity": "sha512-sH5ZSPr+7UStsloltmDh7Ce5fb8XPlHyoPzTpyyMuYCtervL65+ubVZ6Q61cFtFl62UyJlc8/JwERRbAFPUqgw==",
+      "dev": true,
+      "requires": {
+        "array-back": "^4.0.2",
+        "chalk": "^2.4.2",
+        "table-layout": "^1.0.2",
+        "typical": "^5.2.0"
+      },
+      "dependencies": {
+        "ansi-styles": {
+          "version": "3.2.1",
+          "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+          "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+          "dev": true,
+          "requires": {
+            "color-convert": "^1.9.0"
+          }
+        },
+        "array-back": {
+          "version": "4.0.2",
+          "resolved": "https://registry.npmjs.org/array-back/-/array-back-4.0.2.tgz",
+          "integrity": "sha512-NbdMezxqf94cnNfWLL7V/im0Ub+Anbb0IoZhvzie8+4HJ4nMQuzHuy49FkGYCJK2yAloZ3meiB6AVMClbrI1vg==",
+          "dev": true
+        },
+        "chalk": {
+          "version": "2.4.2",
+          "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+          "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+          "dev": true,
+          "requires": {
+            "ansi-styles": "^3.2.1",
+            "escape-string-regexp": "^1.0.5",
+            "supports-color": "^5.3.0"
+          }
+        },
+        "color-convert": {
+          "version": "1.9.3",
+          "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+          "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+          "dev": true,
+          "requires": {
+            "color-name": "1.1.3"
+          }
+        },
+        "color-name": {
+          "version": "1.1.3",
+          "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+          "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+          "dev": true
+        },
+        "has-flag": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+          "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+          "dev": true
+        },
+        "supports-color": {
+          "version": "5.5.0",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+          "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+          "dev": true,
+          "requires": {
+            "has-flag": "^3.0.0"
+          }
+        },
+        "typical": {
+          "version": "5.2.0",
+          "resolved": "https://registry.npmjs.org/typical/-/typical-5.2.0.tgz",
+          "integrity": "sha512-dvdQgNDNJo+8B2uBQoqdb11eUCE1JQXhvjC/CZtgvZseVd5TYMXnq0+vuUemXbd/Se29cTaUuPX3YIc2xgbvIg==",
+          "dev": true
+        }
+      }
+    },
+    "commander": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/commander/-/commander-3.0.2.tgz",
+      "integrity": "sha512-Gar0ASD4BDyKC4hl4DwHqDrmvjoxWKZigVnAbn5H1owvm4CxCPdb0HQDehwNYMJpla5+M2tPmPARzhtYuwpHow==",
+      "dev": true
+    },
+    "compare-versions": {
+      "version": "3.6.0",
+      "resolved": "https://registry.npmjs.org/compare-versions/-/compare-versions-3.6.0.tgz",
+      "integrity": "sha512-W6Af2Iw1z4CB7q4uU4hv646dW9GQuBM+YpC0UvUCWSD8w90SJjp+ujJuXaEMtAXBtSqGfMPuFOVn4/+FlaqfBA==",
+      "dev": true
+    },
+    "concat-map": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
+      "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==",
+      "dev": true
+    },
+    "concat-stream": {
+      "version": "1.6.2",
+      "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz",
+      "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==",
+      "dev": true,
+      "requires": {
+        "buffer-from": "^1.0.0",
+        "inherits": "^2.0.3",
+        "readable-stream": "^2.2.2",
+        "typedarray": "^0.0.6"
+      },
+      "dependencies": {
+        "readable-stream": {
+          "version": "2.3.8",
+          "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz",
+          "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==",
+          "dev": true,
+          "requires": {
+            "core-util-is": "~1.0.0",
+            "inherits": "~2.0.3",
+            "isarray": "~1.0.0",
+            "process-nextick-args": "~2.0.0",
+            "safe-buffer": "~5.1.1",
+            "string_decoder": "~1.1.1",
+            "util-deprecate": "~1.0.1"
+          }
+        },
+        "safe-buffer": {
+          "version": "5.1.2",
+          "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
+          "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==",
+          "dev": true
+        },
+        "string_decoder": {
+          "version": "1.1.1",
+          "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz",
+          "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==",
+          "dev": true,
+          "requires": {
+            "safe-buffer": "~5.1.0"
+          }
+        }
+      }
+    },
+    "constant-case": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/constant-case/-/constant-case-2.0.0.tgz",
+      "integrity": "sha512-eS0N9WwmjTqrOmR3o83F5vW8Z+9R1HnVz3xmzT2PMFug9ly+Au/fxRWlEBSb6LcZwspSsEn9Xs1uw9YgzAg1EQ==",
+      "dev": true,
+      "requires": {
+        "snake-case": "^2.1.0",
+        "upper-case": "^1.1.1"
+      }
+    },
+    "content-disposition": {
+      "version": "0.5.4",
+      "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz",
+      "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==",
+      "dev": true,
+      "requires": {
+        "safe-buffer": "5.2.1"
+      }
+    },
+    "content-hash": {
+      "version": "2.5.2",
+      "resolved": "https://registry.npmjs.org/content-hash/-/content-hash-2.5.2.tgz",
+      "integrity": "sha512-FvIQKy0S1JaWV10sMsA7TRx8bpU+pqPkhbsfvOJAdjRXvYxEckAwQWGwtRjiaJfh+E0DvcWUGqcdjwMGFjsSdw==",
+      "dev": true,
+      "requires": {
+        "cids": "^0.7.1",
+        "multicodec": "^0.5.5",
+        "multihashes": "^0.4.15"
+      }
+    },
+    "content-type": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz",
+      "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==",
+      "dev": true
+    },
+    "cookie": {
+      "version": "0.5.0",
+      "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.5.0.tgz",
+      "integrity": "sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==",
+      "dev": true
+    },
+    "cookie-signature": {
+      "version": "1.0.6",
+      "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
+      "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==",
+      "dev": true
+    },
+    "core-util-is": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
+      "integrity": "sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==",
+      "dev": true
+    },
+    "cors": {
+      "version": "2.8.5",
+      "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz",
+      "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==",
+      "dev": true,
+      "requires": {
+        "object-assign": "^4",
+        "vary": "^1"
+      }
+    },
+    "cosmiconfig": {
+      "version": "7.1.0",
+      "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-7.1.0.tgz",
+      "integrity": "sha512-AdmX6xUzdNASswsFtmwSt7Vj8po9IuqXm0UXz7QKPuEUmPB4XyjGfaAr2PSuELMwkRMVH1EpIkX5bTZGRB3eCA==",
+      "dev": true,
+      "requires": {
+        "@types/parse-json": "^4.0.0",
+        "import-fresh": "^3.2.1",
+        "parse-json": "^5.0.0",
+        "path-type": "^4.0.0",
+        "yaml": "^1.10.0"
+      },
+      "dependencies": {
+        "parse-json": {
+          "version": "5.2.0",
+          "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz",
+          "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==",
+          "dev": true,
+          "requires": {
+            "@babel/code-frame": "^7.0.0",
+            "error-ex": "^1.3.1",
+            "json-parse-even-better-errors": "^2.3.0",
+            "lines-and-columns": "^1.1.6"
+          }
+        },
+        "path-type": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz",
+          "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==",
+          "dev": true
+        }
+      }
+    },
+    "crc-32": {
+      "version": "1.2.2",
+      "resolved": "https://registry.npmjs.org/crc-32/-/crc-32-1.2.2.tgz",
+      "integrity": "sha512-ROmzCKrTnOwybPcJApAA6WBWij23HVfGVNKqqrZpuyZOHqK2CwHSvpGuyt/UNNvaIjEd8X5IFGp4Mh+Ie1IHJQ==",
+      "dev": true
+    },
+    "create-hash": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz",
+      "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==",
+      "dev": true,
+      "requires": {
+        "cipher-base": "^1.0.1",
+        "inherits": "^2.0.1",
+        "md5.js": "^1.3.4",
+        "ripemd160": "^2.0.1",
+        "sha.js": "^2.4.0"
+      }
+    },
+    "create-hmac": {
+      "version": "1.1.7",
+      "resolved": "https://registry.npmjs.org/create-hmac/-/create-hmac-1.1.7.tgz",
+      "integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==",
+      "dev": true,
+      "requires": {
+        "cipher-base": "^1.0.3",
+        "create-hash": "^1.1.0",
+        "inherits": "^2.0.1",
+        "ripemd160": "^2.0.0",
+        "safe-buffer": "^5.0.1",
+        "sha.js": "^2.4.8"
+      }
+    },
+    "create-require": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz",
+      "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==",
+      "dev": true
+    },
+    "cross-fetch": {
+      "version": "3.1.5",
+      "resolved": "https://registry.npmjs.org/cross-fetch/-/cross-fetch-3.1.5.tgz",
+      "integrity": "sha512-lvb1SBsI0Z7GDwmuid+mU3kWVBwTVUbe7S0H52yaaAdQOXq2YktTCZdlAcNKFzE6QtRz0snpw9bNiPeOIkkQvw==",
+      "dev": true,
+      "requires": {
+        "node-fetch": "2.6.7"
+      }
+    },
+    "cross-spawn": {
+      "version": "7.0.3",
+      "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz",
+      "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==",
+      "dev": true,
+      "requires": {
+        "path-key": "^3.1.0",
+        "shebang-command": "^2.0.0",
+        "which": "^2.0.1"
+      }
+    },
+    "crypt": {
+      "version": "0.0.2",
+      "resolved": "https://registry.npmjs.org/crypt/-/crypt-0.0.2.tgz",
+      "integrity": "sha512-mCxBlsHFYh9C+HVpiEacem8FEBnMXgU9gy4zmNC+SXAZNB/1idgp/aulFJ4FgCi7GPEVbfyng092GqL2k2rmow==",
+      "dev": true
+    },
+    "crypto-addr-codec": {
+      "version": "0.1.7",
+      "resolved": "https://registry.npmjs.org/crypto-addr-codec/-/crypto-addr-codec-0.1.7.tgz",
+      "integrity": "sha512-X4hzfBzNhy4mAc3UpiXEC/L0jo5E8wAa9unsnA8nNXYzXjCcGk83hfC5avJWCSGT8V91xMnAS9AKMHmjw5+XCg==",
+      "dev": true,
+      "requires": {
+        "base-x": "^3.0.8",
+        "big-integer": "1.6.36",
+        "blakejs": "^1.1.0",
+        "bs58": "^4.0.1",
+        "ripemd160-min": "0.0.6",
+        "safe-buffer": "^5.2.0",
+        "sha3": "^2.1.1"
+      }
+    },
+    "css-select": {
+      "version": "5.1.0",
+      "resolved": "https://registry.npmjs.org/css-select/-/css-select-5.1.0.tgz",
+      "integrity": "sha512-nwoRF1rvRRnnCqqY7updORDsuqKzqYJ28+oSMaJMMgOauh3fvwHqMS7EZpIPqK8GL+g9mKxF1vP/ZjSeNjEVHg==",
+      "dev": true,
+      "requires": {
+        "boolbase": "^1.0.0",
+        "css-what": "^6.1.0",
+        "domhandler": "^5.0.2",
+        "domutils": "^3.0.1",
+        "nth-check": "^2.0.1"
+      }
+    },
+    "css-what": {
+      "version": "6.1.0",
+      "resolved": "https://registry.npmjs.org/css-what/-/css-what-6.1.0.tgz",
+      "integrity": "sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==",
+      "dev": true
+    },
+    "d": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/d/-/d-1.0.1.tgz",
+      "integrity": "sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==",
+      "dev": true,
+      "requires": {
+        "es5-ext": "^0.10.50",
+        "type": "^1.0.1"
+      }
+    },
+    "dashdash": {
+      "version": "1.14.1",
+      "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz",
+      "integrity": "sha512-jRFi8UDGo6j+odZiEpjazZaWqEal3w/basFjQHQEwVtZJGDpxbH1MeYluwCS8Xq5wmLJooDlMgvVarmWfGM44g==",
+      "dev": true,
+      "requires": {
+        "assert-plus": "^1.0.0"
+      }
+    },
+    "death": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/death/-/death-1.1.0.tgz",
+      "integrity": "sha512-vsV6S4KVHvTGxbEcij7hkWRv0It+sGGWVOM67dQde/o5Xjnr+KmLjxWJii2uEObIrt1CcM9w0Yaovx+iOlIL+w==",
+      "dev": true
+    },
+    "debug": {
+      "version": "4.3.4",
+      "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz",
+      "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==",
+      "dev": true,
+      "requires": {
+        "ms": "2.1.2"
+      }
+    },
+    "decamelize": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz",
+      "integrity": "sha512-z2S+W9X73hAUUki+N+9Za2lBlun89zigOyGrsax+KUQ6wKW4ZoWpEYBkGhQjwAjjDCkWxhY0VKEhk8wzY7F5cA==",
+      "dev": true
+    },
+    "decode-uri-component": {
+      "version": "0.2.2",
+      "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.2.tgz",
+      "integrity": "sha512-FqUYQ+8o158GyGTrMFJms9qh3CqTKvAqgqsTnkLI8sKu0028orqBhxNMFkFen0zGyg6epACD32pjVk58ngIErQ==",
+      "dev": true
+    },
+    "decompress-response": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-6.0.0.tgz",
+      "integrity": "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==",
+      "dev": true,
+      "requires": {
+        "mimic-response": "^3.1.0"
+      },
+      "dependencies": {
+        "mimic-response": {
+          "version": "3.1.0",
+          "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-3.1.0.tgz",
+          "integrity": "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==",
+          "dev": true
+        }
+      }
+    },
+    "deep-eql": {
+      "version": "4.1.3",
+      "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-4.1.3.tgz",
+      "integrity": "sha512-WaEtAOpRA1MQ0eohqZjpGD8zdI0Ovsm8mmFhaDN8dvDZzyoUMcYDnf5Y6iu7HTXxf8JDS23qWa4a+hKCDyOPzw==",
+      "dev": true,
+      "requires": {
+        "type-detect": "^4.0.0"
+      }
+    },
+    "deep-extend": {
+      "version": "0.6.0",
+      "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
+      "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
+      "dev": true
+    },
+    "deep-is": {
+      "version": "0.1.4",
+      "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz",
+      "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==",
+      "dev": true
+    },
+    "defaults": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.4.tgz",
+      "integrity": "sha512-eFuaLoy/Rxalv2kr+lqMlUnrDWV+3j4pljOIJgLIhI058IQfWJ7vXhyEIHu+HtC738klGALYxOKDO0bQP3tg8A==",
+      "dev": true,
+      "requires": {
+        "clone": "^1.0.2"
+      }
+    },
+    "defer-to-connect": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-2.0.1.tgz",
+      "integrity": "sha512-4tvttepXG1VaYGrRibk5EwJd1t4udunSOVMdLSAL6mId1ix438oPwPZMALY41FCijukO1L0twNcGsdzS7dHgDg==",
+      "dev": true
+    },
+    "define-lazy-prop": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/define-lazy-prop/-/define-lazy-prop-2.0.0.tgz",
+      "integrity": "sha512-Ds09qNh8yw3khSjiJjiUInaGX9xlqZDY7JVryGxdxV7NPeuqQfplOpQ66yJFZut3jLa5zOwkXw1g9EI2uKh4Og==",
+      "dev": true
+    },
+    "define-properties": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.0.tgz",
+      "integrity": "sha512-xvqAVKGfT1+UAvPwKTVw/njhdQ8ZhXK4lI0bCIuCMrp2up9nPnaDftrLtmpTazqd1o+UY4zgzU+avtMbDP+ldA==",
+      "dev": true,
+      "requires": {
+        "has-property-descriptors": "^1.0.0",
+        "object-keys": "^1.1.1"
+      }
+    },
+    "delayed-stream": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
+      "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==",
+      "dev": true
+    },
+    "depd": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
+      "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==",
+      "dev": true
+    },
+    "destroy": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz",
+      "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==",
+      "dev": true
+    },
+    "detect-indent": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/detect-indent/-/detect-indent-5.0.0.tgz",
+      "integrity": "sha512-rlpvsxUtM0PQvy9iZe640/IWwWYyBsTApREbA1pHOpmOUIl9MkP/U4z7vTtg4Oaojvqhxt7sdufnT0EzGaR31g==",
+      "dev": true
+    },
+    "detect-port": {
+      "version": "1.5.1",
+      "resolved": "https://registry.npmjs.org/detect-port/-/detect-port-1.5.1.tgz",
+      "integrity": "sha512-aBzdj76lueB6uUst5iAs7+0H/oOjqI5D16XUWxlWMIMROhcM0rfsNVk93zTngq1dDNpoXRr++Sus7ETAExppAQ==",
+      "dev": true,
+      "requires": {
+        "address": "^1.0.1",
+        "debug": "4"
+      }
+    },
+    "diff": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/diff/-/diff-5.0.0.tgz",
+      "integrity": "sha512-/VTCrvm5Z0JGty/BWHljh+BAiw3IK+2j87NGMu8Nwc/f48WoDAC395uomO9ZD117ZOBaHmkX1oyLvkVM/aIT3w==",
+      "dev": true
+    },
+    "difflib": {
+      "version": "0.2.4",
+      "resolved": "https://registry.npmjs.org/difflib/-/difflib-0.2.4.tgz",
+      "integrity": "sha512-9YVwmMb0wQHQNr5J9m6BSj6fk4pfGITGQOOs+D9Fl+INODWFOfvhIU1hNv6GgR1RBoC/9NJcwu77zShxV0kT7w==",
+      "dev": true,
+      "requires": {
+        "heap": ">= 0.2.0"
+      }
+    },
+    "dir-glob": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz",
+      "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==",
+      "dev": true,
+      "requires": {
+        "path-type": "^4.0.0"
+      },
+      "dependencies": {
+        "path-type": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz",
+          "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==",
+          "dev": true
+        }
+      }
+    },
+    "doctrine": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz",
+      "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==",
+      "dev": true,
+      "requires": {
+        "esutils": "^2.0.2"
+      }
+    },
+    "dom-serializer": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-2.0.0.tgz",
+      "integrity": "sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==",
+      "dev": true,
+      "requires": {
+        "domelementtype": "^2.3.0",
+        "domhandler": "^5.0.2",
+        "entities": "^4.2.0"
+      }
+    },
+    "dom-walk": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/dom-walk/-/dom-walk-0.1.2.tgz",
+      "integrity": "sha512-6QvTW9mrGeIegrFXdtQi9pk7O/nSK6lSdXW2eqUspN5LWD7UTji2Fqw5V2YLjBpHEoU9Xl/eUWNpDeZvoyOv2w==",
+      "dev": true
+    },
+    "domelementtype": {
+      "version": "2.3.0",
+      "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.3.0.tgz",
+      "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==",
+      "dev": true
+    },
+    "domhandler": {
+      "version": "5.0.3",
+      "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-5.0.3.tgz",
+      "integrity": "sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==",
+      "dev": true,
+      "requires": {
+        "domelementtype": "^2.3.0"
+      }
+    },
+    "domutils": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.0.1.tgz",
+      "integrity": "sha512-z08c1l761iKhDFtfXO04C7kTdPBLi41zwOZl00WS8b5eiaebNpY00HKbztwBq+e3vyqWNwWF3mP9YLUeqIrF+Q==",
+      "dev": true,
+      "requires": {
+        "dom-serializer": "^2.0.0",
+        "domelementtype": "^2.3.0",
+        "domhandler": "^5.0.1"
+      }
+    },
+    "dot-case": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/dot-case/-/dot-case-2.1.1.tgz",
+      "integrity": "sha512-HnM6ZlFqcajLsyudHq7LeeLDr2rFAVYtDv/hV5qchQEidSck8j9OPUsXY9KwJv/lHMtYlX4DjRQqwFYa+0r8Ug==",
+      "dev": true,
+      "requires": {
+        "no-case": "^2.2.0"
+      }
+    },
+    "dotenv": {
+      "version": "16.0.3",
+      "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.0.3.tgz",
+      "integrity": "sha512-7GO6HghkA5fYG9TYnNxi14/7K9f5occMlp3zXAuSxn7CKCxt9xbNWG7yF8hTCSUchlfWSe3uLmlPfigevRItzQ==",
+      "dev": true
+    },
+    "ecc-jsbn": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz",
+      "integrity": "sha512-eh9O+hwRHNbG4BLTjEl3nw044CkGm5X6LoaCf7LPp7UU8Qrt47JYNi6nPX8xjW97TKGKm1ouctg0QSpZe9qrnw==",
+      "dev": true,
+      "requires": {
+        "jsbn": "~0.1.0",
+        "safer-buffer": "^2.1.0"
+      }
+    },
+    "ee-first": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
+      "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==",
+      "dev": true
+    },
+    "elliptic": {
+      "version": "6.5.4",
+      "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.4.tgz",
+      "integrity": "sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ==",
+      "devOptional": true,
+      "requires": {
+        "bn.js": "^4.11.9",
+        "brorand": "^1.1.0",
+        "hash.js": "^1.0.0",
+        "hmac-drbg": "^1.0.1",
+        "inherits": "^2.0.4",
+        "minimalistic-assert": "^1.0.1",
+        "minimalistic-crypto-utils": "^1.0.1"
+      },
+      "dependencies": {
+        "bn.js": {
+          "version": "4.12.0",
+          "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+          "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+          "devOptional": true
+        }
+      }
+    },
+    "emoji-regex": {
+      "version": "8.0.0",
+      "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
+      "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
+      "dev": true
+    },
+    "encode-utf8": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/encode-utf8/-/encode-utf8-1.0.3.tgz",
+      "integrity": "sha512-ucAnuBEhUK4boH2HjVYG5Q2mQyPorvv0u/ocS+zhdw0S8AlHYY+GOFhP1Gio5z4icpP2ivFSvhtFjQi8+T9ppw==",
+      "dev": true
+    },
+    "encodeurl": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
+      "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==",
+      "dev": true
+    },
+    "end-of-stream": {
+      "version": "1.4.4",
+      "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz",
+      "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==",
+      "dev": true,
+      "requires": {
+        "once": "^1.4.0"
+      }
+    },
+    "enquirer": {
+      "version": "2.3.6",
+      "resolved": "https://registry.npmjs.org/enquirer/-/enquirer-2.3.6.tgz",
+      "integrity": "sha512-yjNnPr315/FjS4zIsUxYguYUPP2e1NK4d7E7ZOLiyYCcbFBiTMyID+2wvm2w6+pZ/odMA7cRkjhsPbltwBOrLg==",
+      "dev": true,
+      "requires": {
+        "ansi-colors": "^4.1.1"
+      }
+    },
+    "entities": {
+      "version": "4.4.0",
+      "resolved": "https://registry.npmjs.org/entities/-/entities-4.4.0.tgz",
+      "integrity": "sha512-oYp7156SP8LkeGD0GF85ad1X9Ai79WtRsZ2gxJqtBuzH+98YUV6jkHEKlZkMbcrjJjIVJNIDP/3WL9wQkoPbWA==",
+      "dev": true
+    },
+    "env-paths": {
+      "version": "2.2.1",
+      "resolved": "https://registry.npmjs.org/env-paths/-/env-paths-2.2.1.tgz",
+      "integrity": "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==",
+      "dev": true
+    },
+    "error-ex": {
+      "version": "1.3.2",
+      "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz",
+      "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==",
+      "dev": true,
+      "requires": {
+        "is-arrayish": "^0.2.1"
+      }
+    },
+    "es-abstract": {
+      "version": "1.21.1",
+      "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.21.1.tgz",
+      "integrity": "sha512-QudMsPOz86xYz/1dG1OuGBKOELjCh99IIWHLzy5znUB6j8xG2yMA7bfTV86VSqKF+Y/H08vQPR+9jyXpuC6hfg==",
+      "dev": true,
+      "requires": {
+        "available-typed-arrays": "^1.0.5",
+        "call-bind": "^1.0.2",
+        "es-set-tostringtag": "^2.0.1",
+        "es-to-primitive": "^1.2.1",
+        "function-bind": "^1.1.1",
+        "function.prototype.name": "^1.1.5",
+        "get-intrinsic": "^1.1.3",
+        "get-symbol-description": "^1.0.0",
+        "globalthis": "^1.0.3",
+        "gopd": "^1.0.1",
+        "has": "^1.0.3",
+        "has-property-descriptors": "^1.0.0",
+        "has-proto": "^1.0.1",
+        "has-symbols": "^1.0.3",
+        "internal-slot": "^1.0.4",
+        "is-array-buffer": "^3.0.1",
+        "is-callable": "^1.2.7",
+        "is-negative-zero": "^2.0.2",
+        "is-regex": "^1.1.4",
+        "is-shared-array-buffer": "^1.0.2",
+        "is-string": "^1.0.7",
+        "is-typed-array": "^1.1.10",
+        "is-weakref": "^1.0.2",
+        "object-inspect": "^1.12.2",
+        "object-keys": "^1.1.1",
+        "object.assign": "^4.1.4",
+        "regexp.prototype.flags": "^1.4.3",
+        "safe-regex-test": "^1.0.0",
+        "string.prototype.trimend": "^1.0.6",
+        "string.prototype.trimstart": "^1.0.6",
+        "typed-array-length": "^1.0.4",
+        "unbox-primitive": "^1.0.2",
+        "which-typed-array": "^1.1.9"
+      }
+    },
+    "es-set-tostringtag": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.1.tgz",
+      "integrity": "sha512-g3OMbtlwY3QewlqAiMLI47KywjWZoEytKr8pf6iTC8uJq5bIAH52Z9pnQ8pVL6whrCto53JZDuUIsifGeLorTg==",
+      "dev": true,
+      "requires": {
+        "get-intrinsic": "^1.1.3",
+        "has": "^1.0.3",
+        "has-tostringtag": "^1.0.0"
+      }
+    },
+    "es-shim-unscopables": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/es-shim-unscopables/-/es-shim-unscopables-1.0.0.tgz",
+      "integrity": "sha512-Jm6GPcCdC30eMLbZ2x8z2WuRwAws3zTBBKuusffYVUrNj/GVSUAZ+xKMaUpfNDR5IbyNA5LJbaecoUVbmUcB1w==",
+      "dev": true,
+      "requires": {
+        "has": "^1.0.3"
+      }
+    },
+    "es-to-primitive": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz",
+      "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==",
+      "dev": true,
+      "requires": {
+        "is-callable": "^1.1.4",
+        "is-date-object": "^1.0.1",
+        "is-symbol": "^1.0.2"
+      }
+    },
+    "es5-ext": {
+      "version": "0.10.62",
+      "resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.62.tgz",
+      "integrity": "sha512-BHLqn0klhEpnOKSrzn/Xsz2UIW8j+cGmo9JLzr8BiUapV8hPL9+FliFqjwr9ngW7jWdnxv6eO+/LqyhJVqgrjA==",
+      "dev": true,
+      "requires": {
+        "es6-iterator": "^2.0.3",
+        "es6-symbol": "^3.1.3",
+        "next-tick": "^1.1.0"
+      }
+    },
+    "es6-iterator": {
+      "version": "2.0.3",
+      "resolved": "https://registry.npmjs.org/es6-iterator/-/es6-iterator-2.0.3.tgz",
+      "integrity": "sha512-zw4SRzoUkd+cl+ZoE15A9o1oQd920Bb0iOJMQkQhl3jNc03YqVjAhG7scf9C5KWRU/R13Orf588uCC6525o02g==",
+      "dev": true,
+      "requires": {
+        "d": "1",
+        "es5-ext": "^0.10.35",
+        "es6-symbol": "^3.1.1"
+      }
+    },
+    "es6-promise": {
+      "version": "4.2.8",
+      "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz",
+      "integrity": "sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==",
+      "dev": true
+    },
+    "es6-symbol": {
+      "version": "3.1.3",
+      "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.3.tgz",
+      "integrity": "sha512-NJ6Yn3FuDinBaBRWl/q5X/s4koRHBrgKAu+yGI6JCBeiu3qrcbJhwT2GeR/EXVfylRk8dpQVJoLEFhK+Mu31NA==",
+      "dev": true,
+      "requires": {
+        "d": "^1.0.1",
+        "ext": "^1.1.2"
+      }
+    },
+    "escalade": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz",
+      "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==",
+      "dev": true
+    },
+    "escape-html": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
+      "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==",
+      "dev": true
+    },
+    "escape-string-regexp": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
+      "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==",
+      "dev": true
+    },
+    "escodegen": {
+      "version": "1.8.1",
+      "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.8.1.tgz",
+      "integrity": "sha512-yhi5S+mNTOuRvyW4gWlg5W1byMaQGWWSYHXsuFZ7GBo7tpyOwi2EdzMP/QWxh9hwkD2m+wDVHJsxhRIj+v/b/A==",
+      "dev": true,
+      "requires": {
+        "esprima": "^2.7.1",
+        "estraverse": "^1.9.1",
+        "esutils": "^2.0.2",
+        "optionator": "^0.8.1",
+        "source-map": "~0.2.0"
+      },
+      "dependencies": {
+        "esprima": {
+          "version": "2.7.3",
+          "resolved": "https://registry.npmjs.org/esprima/-/esprima-2.7.3.tgz",
+          "integrity": "sha512-OarPfz0lFCiW4/AV2Oy1Rp9qu0iusTKqykwTspGCZtPxmF81JR4MmIebvF1F9+UOKth2ZubLQ4XGGaU+hSn99A==",
+          "dev": true
+        },
+        "estraverse": {
+          "version": "1.9.3",
+          "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-1.9.3.tgz",
+          "integrity": "sha512-25w1fMXQrGdoquWnScXZGckOv+Wes+JDnuN/+7ex3SauFRS72r2lFDec0EKPt2YD1wUJ/IrfEex+9yp4hfSOJA==",
+          "dev": true
+        },
+        "levn": {
+          "version": "0.3.0",
+          "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz",
+          "integrity": "sha512-0OO4y2iOHix2W6ujICbKIaEQXvFQHue65vUG3pb5EUomzPI90z9hsA1VsO/dbIIpC53J8gxM9Q4Oho0jrCM/yA==",
+          "dev": true,
+          "requires": {
+            "prelude-ls": "~1.1.2",
+            "type-check": "~0.3.2"
+          }
+        },
+        "optionator": {
+          "version": "0.8.3",
+          "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz",
+          "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==",
+          "dev": true,
+          "requires": {
+            "deep-is": "~0.1.3",
+            "fast-levenshtein": "~2.0.6",
+            "levn": "~0.3.0",
+            "prelude-ls": "~1.1.2",
+            "type-check": "~0.3.2",
+            "word-wrap": "~1.2.3"
+          }
+        },
+        "prelude-ls": {
+          "version": "1.1.2",
+          "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz",
+          "integrity": "sha512-ESF23V4SKG6lVSGZgYNpbsiaAkdab6ZgOxe52p7+Kid3W3u3bxR4Vfd/o21dmN7jSt0IwgZ4v5MUd26FEtXE9w==",
+          "dev": true
+        },
+        "source-map": {
+          "version": "0.2.0",
+          "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.2.0.tgz",
+          "integrity": "sha512-CBdZ2oa/BHhS4xj5DlhjWNHcan57/5YuvfdLf17iVmIpd9KRm+DFLmC6nBNj+6Ua7Kt3TmOjDpQT1aTYOQtoUA==",
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "amdefine": ">=0.0.4"
+          }
+        },
+        "type-check": {
+          "version": "0.3.2",
+          "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz",
+          "integrity": "sha512-ZCmOJdvOWDBYJlzAoFkC+Q0+bUyEOS1ltgp1MGU03fqHG+dbi9tBFU2Rd9QKiDZFAYrhPh2JUf7rZRIuHRKtOg==",
+          "dev": true,
+          "requires": {
+            "prelude-ls": "~1.1.2"
+          }
+        }
+      }
+    },
+    "eslint": {
+      "version": "8.35.0",
+      "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.35.0.tgz",
+      "integrity": "sha512-BxAf1fVL7w+JLRQhWl2pzGeSiGqbWumV4WNvc9Rhp6tiCtm4oHnyPBSEtMGZwrQgudFQ+otqzWoPB7x+hxoWsw==",
+      "dev": true,
+      "requires": {
+        "@eslint/eslintrc": "^2.0.0",
+        "@eslint/js": "8.35.0",
+        "@humanwhocodes/config-array": "^0.11.8",
+        "@humanwhocodes/module-importer": "^1.0.1",
+        "@nodelib/fs.walk": "^1.2.8",
+        "ajv": "^6.10.0",
+        "chalk": "^4.0.0",
+        "cross-spawn": "^7.0.2",
+        "debug": "^4.3.2",
+        "doctrine": "^3.0.0",
+        "escape-string-regexp": "^4.0.0",
+        "eslint-scope": "^7.1.1",
+        "eslint-utils": "^3.0.0",
+        "eslint-visitor-keys": "^3.3.0",
+        "espree": "^9.4.0",
+        "esquery": "^1.4.2",
+        "esutils": "^2.0.2",
+        "fast-deep-equal": "^3.1.3",
+        "file-entry-cache": "^6.0.1",
+        "find-up": "^5.0.0",
+        "glob-parent": "^6.0.2",
+        "globals": "^13.19.0",
+        "grapheme-splitter": "^1.0.4",
+        "ignore": "^5.2.0",
+        "import-fresh": "^3.0.0",
+        "imurmurhash": "^0.1.4",
+        "is-glob": "^4.0.0",
+        "is-path-inside": "^3.0.3",
+        "js-sdsl": "^4.1.4",
+        "js-yaml": "^4.1.0",
+        "json-stable-stringify-without-jsonify": "^1.0.1",
+        "levn": "^0.4.1",
+        "lodash.merge": "^4.6.2",
+        "minimatch": "^3.1.2",
+        "natural-compare": "^1.4.0",
+        "optionator": "^0.9.1",
+        "regexpp": "^3.2.0",
+        "strip-ansi": "^6.0.1",
+        "strip-json-comments": "^3.1.0",
+        "text-table": "^0.2.0"
+      },
+      "dependencies": {
+        "ajv": {
+          "version": "6.12.6",
+          "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+          "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+          "dev": true,
+          "requires": {
+            "fast-deep-equal": "^3.1.1",
+            "fast-json-stable-stringify": "^2.0.0",
+            "json-schema-traverse": "^0.4.1",
+            "uri-js": "^4.2.2"
+          }
+        },
+        "escape-string-regexp": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz",
+          "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==",
+          "dev": true
+        },
+        "eslint-scope": {
+          "version": "7.1.1",
+          "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.1.1.tgz",
+          "integrity": "sha512-QKQM/UXpIiHcLqJ5AOyIW7XZmzjkzQXYE54n1++wb0u9V/abW3l9uQnxX8Z5Xd18xyKIMTUAyQ0k1e8pz6LUrw==",
+          "dev": true,
+          "requires": {
+            "esrecurse": "^4.3.0",
+            "estraverse": "^5.2.0"
+          }
+        },
+        "estraverse": {
+          "version": "5.3.0",
+          "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz",
+          "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==",
+          "dev": true
+        },
+        "find-up": {
+          "version": "5.0.0",
+          "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz",
+          "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==",
+          "dev": true,
+          "requires": {
+            "locate-path": "^6.0.0",
+            "path-exists": "^4.0.0"
+          }
+        },
+        "glob-parent": {
+          "version": "6.0.2",
+          "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz",
+          "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==",
+          "dev": true,
+          "requires": {
+            "is-glob": "^4.0.3"
+          }
+        },
+        "json-schema-traverse": {
+          "version": "0.4.1",
+          "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+          "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+          "dev": true
+        },
+        "locate-path": {
+          "version": "6.0.0",
+          "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz",
+          "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==",
+          "dev": true,
+          "requires": {
+            "p-locate": "^5.0.0"
+          }
+        },
+        "p-limit": {
+          "version": "3.1.0",
+          "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
+          "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==",
+          "dev": true,
+          "requires": {
+            "yocto-queue": "^0.1.0"
+          }
+        },
+        "p-locate": {
+          "version": "5.0.0",
+          "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz",
+          "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==",
+          "dev": true,
+          "requires": {
+            "p-limit": "^3.0.2"
+          }
+        },
+        "path-exists": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
+          "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
+          "dev": true
+        }
+      }
+    },
+    "eslint-config-prettier": {
+      "version": "8.7.0",
+      "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-8.7.0.tgz",
+      "integrity": "sha512-HHVXLSlVUhMSmyW4ZzEuvjpwqamgmlfkutD53cYXLikh4pt/modINRcCIApJ84czDxM4GZInwUrromsDdTImTA==",
+      "dev": true,
+      "requires": {}
+    },
+    "eslint-import-resolver-node": {
+      "version": "0.3.7",
+      "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.7.tgz",
+      "integrity": "sha512-gozW2blMLJCeFpBwugLTGyvVjNoeo1knonXAcatC6bjPBZitotxdWf7Gimr25N4c0AAOo4eOUfaG82IJPDpqCA==",
+      "dev": true,
+      "requires": {
+        "debug": "^3.2.7",
+        "is-core-module": "^2.11.0",
+        "resolve": "^1.22.1"
+      },
+      "dependencies": {
+        "debug": {
+          "version": "3.2.7",
+          "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz",
+          "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==",
+          "dev": true,
+          "requires": {
+            "ms": "^2.1.1"
+          }
+        }
+      }
+    },
+    "eslint-module-utils": {
+      "version": "2.7.4",
+      "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.7.4.tgz",
+      "integrity": "sha512-j4GT+rqzCoRKHwURX7pddtIPGySnX9Si/cgMI5ztrcqOPtk5dDEeZ34CQVPphnqkJytlc97Vuk05Um2mJ3gEQA==",
+      "dev": true,
+      "requires": {
+        "debug": "^3.2.7"
+      },
+      "dependencies": {
+        "debug": {
+          "version": "3.2.7",
+          "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz",
+          "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==",
+          "dev": true,
+          "requires": {
+            "ms": "^2.1.1"
+          }
+        }
+      }
+    },
+    "eslint-plugin-import": {
+      "version": "2.27.5",
+      "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.27.5.tgz",
+      "integrity": "sha512-LmEt3GVofgiGuiE+ORpnvP+kAm3h6MLZJ4Q5HCyHADofsb4VzXFsRiWj3c0OFiV+3DWFh0qg3v9gcPlfc3zRow==",
+      "dev": true,
+      "requires": {
+        "array-includes": "^3.1.6",
+        "array.prototype.flat": "^1.3.1",
+        "array.prototype.flatmap": "^1.3.1",
+        "debug": "^3.2.7",
+        "doctrine": "^2.1.0",
+        "eslint-import-resolver-node": "^0.3.7",
+        "eslint-module-utils": "^2.7.4",
+        "has": "^1.0.3",
+        "is-core-module": "^2.11.0",
+        "is-glob": "^4.0.3",
+        "minimatch": "^3.1.2",
+        "object.values": "^1.1.6",
+        "resolve": "^1.22.1",
+        "semver": "^6.3.0",
+        "tsconfig-paths": "^3.14.1"
+      },
+      "dependencies": {
+        "debug": {
+          "version": "3.2.7",
+          "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz",
+          "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==",
+          "dev": true,
+          "requires": {
+            "ms": "^2.1.1"
+          }
+        },
+        "doctrine": {
+          "version": "2.1.0",
+          "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz",
+          "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==",
+          "dev": true,
+          "requires": {
+            "esutils": "^2.0.2"
+          }
+        }
+      }
+    },
+    "eslint-plugin-prettier": {
+      "version": "4.2.1",
+      "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-4.2.1.tgz",
+      "integrity": "sha512-f/0rXLXUt0oFYs8ra4w49wYZBG5GKZpAYsJSm6rnYL5uVDjd+zowwMwVZHnAjf4edNrKpCDYfXDgmRE/Ak7QyQ==",
+      "dev": true,
+      "requires": {
+        "prettier-linter-helpers": "^1.0.0"
+      }
+    },
+    "eslint-scope": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz",
+      "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==",
+      "dev": true,
+      "requires": {
+        "esrecurse": "^4.3.0",
+        "estraverse": "^4.1.1"
+      }
+    },
+    "eslint-utils": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-3.0.0.tgz",
+      "integrity": "sha512-uuQC43IGctw68pJA1RgbQS8/NP7rch6Cwd4j3ZBtgo4/8Flj4eGE7ZYSZRN3iq5pVUv6GPdW5Z1RFleo84uLDA==",
+      "dev": true,
+      "requires": {
+        "eslint-visitor-keys": "^2.0.0"
+      },
+      "dependencies": {
+        "eslint-visitor-keys": {
+          "version": "2.1.0",
+          "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.1.0.tgz",
+          "integrity": "sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==",
+          "dev": true
+        }
+      }
+    },
+    "eslint-visitor-keys": {
+      "version": "3.3.0",
+      "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.3.0.tgz",
+      "integrity": "sha512-mQ+suqKJVyeuwGYHAdjMFqjCyfl8+Ldnxuyp3ldiMBFKkvytrXUZWaiPCEav8qDHKty44bD+qV1IP4T+w+xXRA==",
+      "dev": true
+    },
+    "espree": {
+      "version": "9.4.1",
+      "resolved": "https://registry.npmjs.org/espree/-/espree-9.4.1.tgz",
+      "integrity": "sha512-XwctdmTO6SIvCzd9810yyNzIrOrqNYV9Koizx4C/mRhf9uq0o4yHoCEU/670pOxOL/MSraektvSAji79kX90Vg==",
+      "dev": true,
+      "requires": {
+        "acorn": "^8.8.0",
+        "acorn-jsx": "^5.3.2",
+        "eslint-visitor-keys": "^3.3.0"
+      }
+    },
+    "esquery": {
+      "version": "1.5.0",
+      "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz",
+      "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==",
+      "dev": true,
+      "requires": {
+        "estraverse": "^5.1.0"
+      },
+      "dependencies": {
+        "estraverse": {
+          "version": "5.3.0",
+          "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz",
+          "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==",
+          "dev": true
+        }
+      }
+    },
+    "esrecurse": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz",
+      "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==",
+      "dev": true,
+      "requires": {
+        "estraverse": "^5.2.0"
+      },
+      "dependencies": {
+        "estraverse": {
+          "version": "5.3.0",
+          "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz",
+          "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==",
+          "dev": true
+        }
+      }
+    },
+    "estraverse": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz",
+      "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==",
+      "dev": true
+    },
+    "esutils": {
+      "version": "2.0.3",
+      "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz",
+      "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==",
+      "dev": true
+    },
+    "etag": {
+      "version": "1.8.1",
+      "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
+      "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==",
+      "dev": true
+    },
+    "eth-ens-namehash": {
+      "version": "2.0.8",
+      "resolved": "https://registry.npmjs.org/eth-ens-namehash/-/eth-ens-namehash-2.0.8.tgz",
+      "integrity": "sha512-VWEI1+KJfz4Km//dadyvBBoBeSQ0MHTXPvr8UIXiLW6IanxvAV+DmlZAijZwAyggqGUfwQBeHf7tc9wzc1piSw==",
+      "dev": true,
+      "requires": {
+        "idna-uts46-hx": "^2.3.1",
+        "js-sha3": "^0.5.7"
+      },
+      "dependencies": {
+        "js-sha3": {
+          "version": "0.5.7",
+          "resolved": "https://registry.npmjs.org/js-sha3/-/js-sha3-0.5.7.tgz",
+          "integrity": "sha512-GII20kjaPX0zJ8wzkTbNDYMY7msuZcTWk8S5UOh6806Jq/wz1J8/bnr8uGU0DAUmYDjj2Mr4X1cW8v/GLYnR+g==",
+          "dev": true
+        }
+      }
+    },
+    "eth-gas-reporter": {
+      "version": "0.2.27",
+      "resolved": "https://registry.npmjs.org/eth-gas-reporter/-/eth-gas-reporter-0.2.27.tgz",
+      "integrity": "sha512-femhvoAM7wL0GcI8ozTdxfuBtBFJ9qsyIAsmKVjlWAHUbdnnXHt+lKzz/kmldM5lA9jLuNHGwuIxorNpLbR1Zw==",
+      "dev": true,
+      "requires": {
+        "@solidity-parser/parser": "^0.14.0",
+        "axios": "^1.5.1",
+        "cli-table3": "^0.5.0",
+        "colors": "1.4.0",
+        "ethereum-cryptography": "^1.0.3",
+        "ethers": "^5.7.2",
+        "fs-readdir-recursive": "^1.1.0",
+        "lodash": "^4.17.14",
+        "markdown-table": "^1.1.3",
+        "mocha": "^10.2.0",
+        "req-cwd": "^2.0.0",
+        "sha1": "^1.1.1",
+        "sync-request": "^6.0.0"
+      },
+      "dependencies": {
+        "ansi-regex": {
+          "version": "3.0.1",
+          "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz",
+          "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==",
+          "dev": true
+        },
+        "axios": {
+          "version": "1.6.2",
+          "resolved": "https://registry.npmjs.org/axios/-/axios-1.6.2.tgz",
+          "integrity": "sha512-7i24Ri4pmDRfJTR7LDBhsOTtcm+9kjX5WiY1X3wIisx6G9So3pfMkEiU7emUBe46oceVImccTEM3k6C5dbVW8A==",
+          "dev": true,
+          "requires": {
+            "follow-redirects": "^1.15.0",
+            "form-data": "^4.0.0",
+            "proxy-from-env": "^1.1.0"
+          }
+        },
+        "cli-table3": {
+          "version": "0.5.1",
+          "resolved": "https://registry.npmjs.org/cli-table3/-/cli-table3-0.5.1.tgz",
+          "integrity": "sha512-7Qg2Jrep1S/+Q3EceiZtQcDPWxhAvBw+ERf1162v4sikJrvojMHFqXt8QIVha8UlH9rgU0BeWPytZ9/TzYqlUw==",
+          "dev": true,
+          "requires": {
+            "colors": "^1.1.2",
+            "object-assign": "^4.1.0",
+            "string-width": "^2.1.1"
+          }
+        },
+        "ethereum-cryptography": {
+          "version": "1.2.0",
+          "resolved": "https://registry.npmjs.org/ethereum-cryptography/-/ethereum-cryptography-1.2.0.tgz",
+          "integrity": "sha512-6yFQC9b5ug6/17CQpCyE3k9eKBMdhyVjzUy1WkiuY/E4vj/SXDBbCw8QEIaXqf0Mf2SnY6RmpDcwlUmBSS0EJw==",
+          "dev": true,
+          "requires": {
+            "@noble/hashes": "1.2.0",
+            "@noble/secp256k1": "1.7.1",
+            "@scure/bip32": "1.1.5",
+            "@scure/bip39": "1.1.1"
+          }
+        },
+        "form-data": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
+          "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
+          "dev": true,
+          "requires": {
+            "asynckit": "^0.4.0",
+            "combined-stream": "^1.0.8",
+            "mime-types": "^2.1.12"
+          }
+        },
+        "is-fullwidth-code-point": {
+          "version": "2.0.0",
+          "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
+          "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==",
+          "dev": true
+        },
+        "string-width": {
+          "version": "2.1.1",
+          "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz",
+          "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==",
+          "dev": true,
+          "requires": {
+            "is-fullwidth-code-point": "^2.0.0",
+            "strip-ansi": "^4.0.0"
+          }
+        },
+        "strip-ansi": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz",
+          "integrity": "sha512-4XaJ2zQdCzROZDivEVIDPkcQn8LMFSa8kj8Gxb/Lnwzv9A8VctNZ+lfivC/sV3ivW8ElJTERXZoPBRrZKkNKow==",
+          "dev": true,
+          "requires": {
+            "ansi-regex": "^3.0.0"
+          }
+        }
+      }
+    },
+    "eth-lib": {
+      "version": "0.1.29",
+      "resolved": "https://registry.npmjs.org/eth-lib/-/eth-lib-0.1.29.tgz",
+      "integrity": "sha512-bfttrr3/7gG4E02HoWTDUcDDslN003OlOoBxk9virpAZQ1ja/jDgwkWB8QfJF7ojuEowrqy+lzp9VcJG7/k5bQ==",
+      "dev": true,
+      "requires": {
+        "bn.js": "^4.11.6",
+        "elliptic": "^6.4.0",
+        "nano-json-stream-parser": "^0.1.2",
+        "servify": "^0.1.12",
+        "ws": "^3.0.0",
+        "xhr-request-promise": "^0.1.2"
+      },
+      "dependencies": {
+        "bn.js": {
+          "version": "4.12.0",
+          "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+          "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+          "dev": true
+        },
+        "safe-buffer": {
+          "version": "5.1.2",
+          "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
+          "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==",
+          "dev": true
+        },
+        "ws": {
+          "version": "3.3.3",
+          "resolved": "https://registry.npmjs.org/ws/-/ws-3.3.3.tgz",
+          "integrity": "sha512-nnWLa/NwZSt4KQJu51MYlCcSQ5g7INpOrOMt4XV8j4dqTXdmlUmSHQ8/oLC069ckre0fRsgfvsKwbTdtKLCDkA==",
+          "dev": true,
+          "requires": {
+            "async-limiter": "~1.0.0",
+            "safe-buffer": "~5.1.0",
+            "ultron": "~1.1.0"
+          }
+        }
+      }
+    },
+    "ethereum-bloom-filters": {
+      "version": "1.0.10",
+      "resolved": "https://registry.npmjs.org/ethereum-bloom-filters/-/ethereum-bloom-filters-1.0.10.tgz",
+      "integrity": "sha512-rxJ5OFN3RwjQxDcFP2Z5+Q9ho4eIdEmSc2ht0fCu8Se9nbXjZ7/031uXoUYJ87KHCOdVeiUuwSnoS7hmYAGVHA==",
+      "dev": true,
+      "requires": {
+        "js-sha3": "^0.8.0"
+      }
+    },
+    "ethereum-cryptography": {
+      "version": "0.1.3",
+      "resolved": "https://registry.npmjs.org/ethereum-cryptography/-/ethereum-cryptography-0.1.3.tgz",
+      "integrity": "sha512-w8/4x1SGGzc+tO97TASLja6SLd3fRIK2tLVcV2Gx4IB21hE19atll5Cq9o3d0ZmAYC/8aw0ipieTSiekAea4SQ==",
+      "dev": true,
+      "requires": {
+        "@types/pbkdf2": "^3.0.0",
+        "@types/secp256k1": "^4.0.1",
+        "blakejs": "^1.1.0",
+        "browserify-aes": "^1.2.0",
+        "bs58check": "^2.1.2",
+        "create-hash": "^1.2.0",
+        "create-hmac": "^1.1.7",
+        "hash.js": "^1.1.7",
+        "keccak": "^3.0.0",
+        "pbkdf2": "^3.0.17",
+        "randombytes": "^2.1.0",
+        "safe-buffer": "^5.1.2",
+        "scrypt-js": "^3.0.0",
+        "secp256k1": "^4.0.1",
+        "setimmediate": "^1.0.5"
+      }
+    },
+    "ethereum-ens": {
+      "version": "0.8.0",
+      "resolved": "https://registry.npmjs.org/ethereum-ens/-/ethereum-ens-0.8.0.tgz",
+      "integrity": "sha512-a8cBTF4AWw1Q1Y37V1LSCS9pRY4Mh3f8vCg5cbXCCEJ3eno1hbI/+Ccv9SZLISYpqQhaglP3Bxb/34lS4Qf7Bg==",
+      "dev": true,
+      "requires": {
+        "bluebird": "^3.4.7",
+        "eth-ens-namehash": "^2.0.0",
+        "js-sha3": "^0.5.7",
+        "pako": "^1.0.4",
+        "underscore": "^1.8.3",
+        "web3": "^1.0.0-beta.34"
+      },
+      "dependencies": {
+        "js-sha3": {
+          "version": "0.5.7",
+          "resolved": "https://registry.npmjs.org/js-sha3/-/js-sha3-0.5.7.tgz",
+          "integrity": "sha512-GII20kjaPX0zJ8wzkTbNDYMY7msuZcTWk8S5UOh6806Jq/wz1J8/bnr8uGU0DAUmYDjj2Mr4X1cW8v/GLYnR+g==",
+          "dev": true
+        }
+      }
+    },
+    "ethereumjs-abi": {
+      "version": "0.6.8",
+      "resolved": "https://registry.npmjs.org/ethereumjs-abi/-/ethereumjs-abi-0.6.8.tgz",
+      "integrity": "sha512-Tx0r/iXI6r+lRsdvkFDlut0N08jWMnKRZ6Gkq+Nmw75lZe4e6o3EkSnkaBP5NF6+m5PTGAr9JP43N3LyeoglsA==",
+      "dev": true,
+      "requires": {
+        "bn.js": "^4.11.8",
+        "ethereumjs-util": "^6.0.0"
+      },
+      "dependencies": {
+        "@types/bn.js": {
+          "version": "4.11.6",
+          "resolved": "https://registry.npmjs.org/@types/bn.js/-/bn.js-4.11.6.tgz",
+          "integrity": "sha512-pqr857jrp2kPuO9uRjZ3PwnJTjoQy+fcdxvBTvHm6dkmEL9q+hDD/2j/0ELOBPtPnS8LjCX0gI9nbl8lVkadpg==",
+          "dev": true,
+          "requires": {
+            "@types/node": "*"
+          }
+        },
+        "bn.js": {
+          "version": "4.12.0",
+          "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+          "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+          "dev": true
+        },
+        "ethereumjs-util": {
+          "version": "6.2.1",
+          "resolved": "https://registry.npmjs.org/ethereumjs-util/-/ethereumjs-util-6.2.1.tgz",
+          "integrity": "sha512-W2Ktez4L01Vexijrm5EB6w7dg4n/TgpoYU4avuT5T3Vmnw/eCRtiBrJfQYS/DCSvDIOLn2k57GcHdeBcgVxAqw==",
+          "dev": true,
+          "requires": {
+            "@types/bn.js": "^4.11.3",
+            "bn.js": "^4.11.0",
+            "create-hash": "^1.1.2",
+            "elliptic": "^6.5.2",
+            "ethereum-cryptography": "^0.1.3",
+            "ethjs-util": "0.1.6",
+            "rlp": "^2.2.3"
+          }
+        }
+      }
+    },
+    "ethereumjs-util": {
+      "version": "7.1.5",
+      "resolved": "https://registry.npmjs.org/ethereumjs-util/-/ethereumjs-util-7.1.5.tgz",
+      "integrity": "sha512-SDl5kKrQAudFBUe5OJM9Ac6WmMyYmXX/6sTmLZ3ffG2eY6ZIGBes3pEDxNN6V72WyOw4CPD5RomKdsa8DAAwLg==",
+      "dev": true,
+      "requires": {
+        "@types/bn.js": "^5.1.0",
+        "bn.js": "^5.1.2",
+        "create-hash": "^1.1.2",
+        "ethereum-cryptography": "^0.1.3",
+        "rlp": "^2.2.4"
+      }
+    },
+    "ethers": {
+      "version": "5.7.2",
+      "resolved": "https://registry.npmjs.org/ethers/-/ethers-5.7.2.tgz",
+      "integrity": "sha512-wswUsmWo1aOK8rR7DIKiWSw9DbLWe6x98Jrn8wcTflTVvaXhAMaB5zGAXy0GYQEQp9iO1iSHWVyARQm11zUtyg==",
+      "devOptional": true,
+      "requires": {
+        "@ethersproject/abi": "5.7.0",
+        "@ethersproject/abstract-provider": "5.7.0",
+        "@ethersproject/abstract-signer": "5.7.0",
+        "@ethersproject/address": "5.7.0",
+        "@ethersproject/base64": "5.7.0",
+        "@ethersproject/basex": "5.7.0",
+        "@ethersproject/bignumber": "5.7.0",
+        "@ethersproject/bytes": "5.7.0",
+        "@ethersproject/constants": "5.7.0",
+        "@ethersproject/contracts": "5.7.0",
+        "@ethersproject/hash": "5.7.0",
+        "@ethersproject/hdnode": "5.7.0",
+        "@ethersproject/json-wallets": "5.7.0",
+        "@ethersproject/keccak256": "5.7.0",
+        "@ethersproject/logger": "5.7.0",
+        "@ethersproject/networks": "5.7.1",
+        "@ethersproject/pbkdf2": "5.7.0",
+        "@ethersproject/properties": "5.7.0",
+        "@ethersproject/providers": "5.7.2",
+        "@ethersproject/random": "5.7.0",
+        "@ethersproject/rlp": "5.7.0",
+        "@ethersproject/sha2": "5.7.0",
+        "@ethersproject/signing-key": "5.7.0",
+        "@ethersproject/solidity": "5.7.0",
+        "@ethersproject/strings": "5.7.0",
+        "@ethersproject/transactions": "5.7.0",
+        "@ethersproject/units": "5.7.0",
+        "@ethersproject/wallet": "5.7.0",
+        "@ethersproject/web": "5.7.1",
+        "@ethersproject/wordlists": "5.7.0"
+      },
+      "dependencies": {
+        "@ethersproject/abstract-signer": {
+          "version": "5.7.0",
+          "resolved": "https://registry.npmjs.org/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz",
+          "integrity": "sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==",
+          "devOptional": true,
+          "requires": {
+            "@ethersproject/abstract-provider": "^5.7.0",
+            "@ethersproject/bignumber": "^5.7.0",
+            "@ethersproject/bytes": "^5.7.0",
+            "@ethersproject/logger": "^5.7.0",
+            "@ethersproject/properties": "^5.7.0"
+          }
+        }
+      }
+    },
+    "ethjs-unit": {
+      "version": "0.1.6",
+      "resolved": "https://registry.npmjs.org/ethjs-unit/-/ethjs-unit-0.1.6.tgz",
+      "integrity": "sha512-/Sn9Y0oKl0uqQuvgFk/zQgR7aw1g36qX/jzSQ5lSwlO0GigPymk4eGQfeNTD03w1dPOqfz8V77Cy43jH56pagw==",
+      "dev": true,
+      "requires": {
+        "bn.js": "4.11.6",
+        "number-to-bn": "1.7.0"
+      },
+      "dependencies": {
+        "bn.js": {
+          "version": "4.11.6",
+          "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.6.tgz",
+          "integrity": "sha512-XWwnNNFCuuSQ0m3r3C4LE3EiORltHd9M05pq6FOlVeiophzRbMo50Sbz1ehl8K3Z+jw9+vmgnXefY1hz8X+2wA==",
+          "dev": true
+        }
+      }
+    },
+    "ethjs-util": {
+      "version": "0.1.6",
+      "resolved": "https://registry.npmjs.org/ethjs-util/-/ethjs-util-0.1.6.tgz",
+      "integrity": "sha512-CUnVOQq7gSpDHZVVrQW8ExxUETWrnrvXYvYz55wOU8Uj4VCgw56XC2B/fVqQN+f7gmrnRHSLVnFAwsCuNwji8w==",
+      "dev": true,
+      "requires": {
+        "is-hex-prefixed": "1.0.0",
+        "strip-hex-prefix": "1.0.0"
+      }
+    },
+    "event-target-shim": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz",
+      "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==",
+      "dev": true
+    },
+    "eventemitter3": {
+      "version": "4.0.4",
+      "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.4.tgz",
+      "integrity": "sha512-rlaVLnVxtxvoyLsQQFBx53YmXHDxRIzzTLbdfxqi4yocpSjAxXwkU0cScM5JgSKMqEhrZpnvQ2D9gjylR0AimQ==",
+      "dev": true
+    },
+    "evp_bytestokey": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz",
+      "integrity": "sha512-/f2Go4TognH/KvCISP7OUsHn85hT9nUkxxA9BEWxFn+Oj9o8ZNLm/40hdlgSLyuOimsrTKLUMEorQexp/aPQeA==",
+      "dev": true,
+      "requires": {
+        "md5.js": "^1.3.4",
+        "safe-buffer": "^5.1.1"
+      }
+    },
+    "express": {
+      "version": "4.18.2",
+      "resolved": "https://registry.npmjs.org/express/-/express-4.18.2.tgz",
+      "integrity": "sha512-5/PsL6iGPdfQ/lKM1UuielYgv3BUoJfz1aUwU9vHZ+J7gyvwdQXFEBIEIaxeGf0GIcreATNyBExtalisDbuMqQ==",
+      "dev": true,
+      "requires": {
+        "accepts": "~1.3.8",
+        "array-flatten": "1.1.1",
+        "body-parser": "1.20.1",
+        "content-disposition": "0.5.4",
+        "content-type": "~1.0.4",
+        "cookie": "0.5.0",
+        "cookie-signature": "1.0.6",
+        "debug": "2.6.9",
+        "depd": "2.0.0",
+        "encodeurl": "~1.0.2",
+        "escape-html": "~1.0.3",
+        "etag": "~1.8.1",
+        "finalhandler": "1.2.0",
+        "fresh": "0.5.2",
+        "http-errors": "2.0.0",
+        "merge-descriptors": "1.0.1",
+        "methods": "~1.1.2",
+        "on-finished": "2.4.1",
+        "parseurl": "~1.3.3",
+        "path-to-regexp": "0.1.7",
+        "proxy-addr": "~2.0.7",
+        "qs": "6.11.0",
+        "range-parser": "~1.2.1",
+        "safe-buffer": "5.2.1",
+        "send": "0.18.0",
+        "serve-static": "1.15.0",
+        "setprototypeof": "1.2.0",
+        "statuses": "2.0.1",
+        "type-is": "~1.6.18",
+        "utils-merge": "1.0.1",
+        "vary": "~1.1.2"
+      },
+      "dependencies": {
+        "body-parser": {
+          "version": "1.20.1",
+          "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.1.tgz",
+          "integrity": "sha512-jWi7abTbYwajOytWCQc37VulmWiRae5RyTpaCyDcS5/lMdtwSz5lOpDE67srw/HYe35f1z3fDQw+3txg7gNtWw==",
+          "dev": true,
+          "requires": {
+            "bytes": "3.1.2",
+            "content-type": "~1.0.4",
+            "debug": "2.6.9",
+            "depd": "2.0.0",
+            "destroy": "1.2.0",
+            "http-errors": "2.0.0",
+            "iconv-lite": "0.4.24",
+            "on-finished": "2.4.1",
+            "qs": "6.11.0",
+            "raw-body": "2.5.1",
+            "type-is": "~1.6.18",
+            "unpipe": "1.0.0"
+          }
+        },
+        "debug": {
+          "version": "2.6.9",
+          "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+          "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+          "dev": true,
+          "requires": {
+            "ms": "2.0.0"
+          }
+        },
+        "ms": {
+          "version": "2.0.0",
+          "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+          "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
+          "dev": true
+        },
+        "raw-body": {
+          "version": "2.5.1",
+          "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.1.tgz",
+          "integrity": "sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig==",
+          "dev": true,
+          "requires": {
+            "bytes": "3.1.2",
+            "http-errors": "2.0.0",
+            "iconv-lite": "0.4.24",
+            "unpipe": "1.0.0"
+          }
+        }
+      }
+    },
+    "ext": {
+      "version": "1.7.0",
+      "resolved": "https://registry.npmjs.org/ext/-/ext-1.7.0.tgz",
+      "integrity": "sha512-6hxeJYaL110a9b5TEJSj0gojyHQAmA2ch5Os+ySCiA1QGdS697XWY1pzsrSjqA9LDEEgdB/KypIlR59RcLuHYw==",
+      "dev": true,
+      "requires": {
+        "type": "^2.7.2"
+      },
+      "dependencies": {
+        "type": {
+          "version": "2.7.2",
+          "resolved": "https://registry.npmjs.org/type/-/type-2.7.2.tgz",
+          "integrity": "sha512-dzlvlNlt6AXU7EBSfpAscydQ7gXB+pPGsPnfJnZpiNJBDj7IaJzQlBZYGdEi4R9HmPdBv2XmWJ6YUtoTa7lmCw==",
+          "dev": true
+        }
+      }
+    },
+    "extend": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz",
+      "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==",
+      "dev": true
+    },
+    "extsprintf": {
+      "version": "1.3.0",
+      "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz",
+      "integrity": "sha512-11Ndz7Nv+mvAC1j0ktTa7fAb0vLyGGX+rMHNBYQviQDGU0Hw7lhctJANqbPhu9nV9/izT/IntTgZ7Im/9LJs9g==",
+      "dev": true
+    },
+    "fast-check": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/fast-check/-/fast-check-3.1.1.tgz",
+      "integrity": "sha512-3vtXinVyuUKCKFKYcwXhGE6NtGWkqF8Yh3rvMZNzmwz8EPrgoc/v4pDdLHyLnCyCI5MZpZZkDEwFyXyEONOxpA==",
+      "dev": true,
+      "requires": {
+        "pure-rand": "^5.0.1"
+      }
+    },
+    "fast-deep-equal": {
+      "version": "3.1.3",
+      "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
+      "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==",
+      "dev": true
+    },
+    "fast-diff": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/fast-diff/-/fast-diff-1.2.0.tgz",
+      "integrity": "sha512-xJuoT5+L99XlZ8twedaRf6Ax2TgQVxvgZOYoPKqZufmJib0tL2tegPBOZb1pVNgIhlqDlA0eO0c3wBvQcmzx4w==",
+      "dev": true
+    },
+    "fast-glob": {
+      "version": "3.2.12",
+      "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.12.tgz",
+      "integrity": "sha512-DVj4CQIYYow0BlaelwK1pHl5n5cRSJfM60UA0zK891sVInoPri2Ekj7+e1CT3/3qxXenpI+nBBmQAcJPJgaj4w==",
+      "dev": true,
+      "requires": {
+        "@nodelib/fs.stat": "^2.0.2",
+        "@nodelib/fs.walk": "^1.2.3",
+        "glob-parent": "^5.1.2",
+        "merge2": "^1.3.0",
+        "micromatch": "^4.0.4"
+      }
+    },
+    "fast-json-stable-stringify": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz",
+      "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==",
+      "dev": true
+    },
+    "fast-levenshtein": {
+      "version": "2.0.6",
+      "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz",
+      "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==",
+      "dev": true
+    },
+    "fastq": {
+      "version": "1.15.0",
+      "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.15.0.tgz",
+      "integrity": "sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw==",
+      "dev": true,
+      "requires": {
+        "reusify": "^1.0.4"
+      }
+    },
+    "file-entry-cache": {
+      "version": "6.0.1",
+      "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz",
+      "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==",
+      "dev": true,
+      "requires": {
+        "flat-cache": "^3.0.4"
+      }
+    },
+    "fill-range": {
+      "version": "7.0.1",
+      "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
+      "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==",
+      "dev": true,
+      "requires": {
+        "to-regex-range": "^5.0.1"
+      }
+    },
+    "finalhandler": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.2.0.tgz",
+      "integrity": "sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==",
+      "dev": true,
+      "requires": {
+        "debug": "2.6.9",
+        "encodeurl": "~1.0.2",
+        "escape-html": "~1.0.3",
+        "on-finished": "2.4.1",
+        "parseurl": "~1.3.3",
+        "statuses": "2.0.1",
+        "unpipe": "~1.0.0"
+      },
+      "dependencies": {
+        "debug": {
+          "version": "2.6.9",
+          "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+          "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+          "dev": true,
+          "requires": {
+            "ms": "2.0.0"
+          }
+        },
+        "ms": {
+          "version": "2.0.0",
+          "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+          "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
+          "dev": true
+        }
+      }
+    },
+    "find-replace": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/find-replace/-/find-replace-3.0.0.tgz",
+      "integrity": "sha512-6Tb2myMioCAgv5kfvP5/PkZZ/ntTpVK39fHY7WkWBgvbeE+VHd/tZuZ4mrC+bxh4cfOZeYKVPaJIZtZXV7GNCQ==",
+      "dev": true,
+      "requires": {
+        "array-back": "^3.0.1"
+      }
+    },
+    "find-up": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/find-up/-/find-up-1.1.2.tgz",
+      "integrity": "sha512-jvElSjyuo4EMQGoTwo1uJU5pQMwTW5lS1x05zzfJuTIyLR3zwO27LYrxNg+dlvKpGOuGy/MzBdXh80g0ve5+HA==",
+      "dev": true,
+      "requires": {
+        "path-exists": "^2.0.0",
+        "pinkie-promise": "^2.0.0"
+      }
+    },
+    "find-versions": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/find-versions/-/find-versions-4.0.0.tgz",
+      "integrity": "sha512-wgpWy002tA+wgmO27buH/9KzyEOQnKsG/R0yrcjPT9BOFm0zRBVQbZ95nRGXWMywS8YR5knRbpohio0bcJABxQ==",
+      "dev": true,
+      "requires": {
+        "semver-regex": "^3.1.2"
+      }
+    },
+    "flat": {
+      "version": "5.0.2",
+      "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz",
+      "integrity": "sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==",
+      "dev": true
+    },
+    "flat-cache": {
+      "version": "3.0.4",
+      "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz",
+      "integrity": "sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==",
+      "dev": true,
+      "requires": {
+        "flatted": "^3.1.0",
+        "rimraf": "^3.0.2"
+      }
+    },
+    "flatted": {
+      "version": "3.2.7",
+      "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.7.tgz",
+      "integrity": "sha512-5nqDSxl8nn5BSNxyR3n4I6eDmbolI6WT+QqR547RwxQapgjQBmtktdP+HTBb/a/zLsbzERTONyUB5pefh5TtjQ==",
+      "dev": true
+    },
+    "fmix": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/fmix/-/fmix-0.1.0.tgz",
+      "integrity": "sha512-Y6hyofImk9JdzU8k5INtTXX1cu8LDlePWDFU5sftm9H+zKCr5SGrVjdhkvsim646cw5zD0nADj8oHyXMZmCZ9w==",
+      "dev": true,
+      "requires": {
+        "imul": "^1.0.0"
+      }
+    },
+    "follow-redirects": {
+      "version": "1.15.2",
+      "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz",
+      "integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==",
+      "dev": true
+    },
+    "for-each": {
+      "version": "0.3.3",
+      "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz",
+      "integrity": "sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==",
+      "dev": true,
+      "requires": {
+        "is-callable": "^1.1.3"
+      }
+    },
+    "forever-agent": {
+      "version": "0.6.1",
+      "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz",
+      "integrity": "sha512-j0KLYPhm6zeac4lz3oJ3o65qvgQCcPubiyotZrXqEaG4hNagNYO8qdlUrX5vwqv9ohqeT/Z3j6+yW067yWWdUw==",
+      "dev": true
+    },
+    "form-data": {
+      "version": "2.3.3",
+      "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz",
+      "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==",
+      "dev": true,
+      "requires": {
+        "asynckit": "^0.4.0",
+        "combined-stream": "^1.0.6",
+        "mime-types": "^2.1.12"
+      }
+    },
+    "form-data-encoder": {
+      "version": "1.7.1",
+      "resolved": "https://registry.npmjs.org/form-data-encoder/-/form-data-encoder-1.7.1.tgz",
+      "integrity": "sha512-EFRDrsMm/kyqbTQocNvRXMLjc7Es2Vk+IQFx/YW7hkUH1eBl4J1fqiP34l74Yt0pFLCNpc06fkbVk00008mzjg==",
+      "dev": true
+    },
+    "forwarded": {
+      "version": "0.2.0",
+      "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
+      "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==",
+      "dev": true
+    },
+    "fp-ts": {
+      "version": "1.19.3",
+      "resolved": "https://registry.npmjs.org/fp-ts/-/fp-ts-1.19.3.tgz",
+      "integrity": "sha512-H5KQDspykdHuztLTg+ajGN0Z2qUjcEf3Ybxc6hLt0k7/zPkn29XnKnxlBPyW2XIddWrGaJBzBl4VLYOtk39yZg==",
+      "dev": true
+    },
+    "fresh": {
+      "version": "0.5.2",
+      "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
+      "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==",
+      "dev": true
+    },
+    "fs-extra": {
+      "version": "7.0.1",
+      "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-7.0.1.tgz",
+      "integrity": "sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==",
+      "dev": true,
+      "requires": {
+        "graceful-fs": "^4.1.2",
+        "jsonfile": "^4.0.0",
+        "universalify": "^0.1.0"
+      }
+    },
+    "fs-minipass": {
+      "version": "1.2.7",
+      "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-1.2.7.tgz",
+      "integrity": "sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==",
+      "dev": true,
+      "requires": {
+        "minipass": "^2.6.0"
+      }
+    },
+    "fs-readdir-recursive": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/fs-readdir-recursive/-/fs-readdir-recursive-1.1.0.tgz",
+      "integrity": "sha512-GNanXlVr2pf02+sPN40XN8HG+ePaNcvM0q5mZBd668Obwb0yD5GiUbZOFgwn8kGMY6I3mdyDJzieUy3PTYyTRA==",
+      "dev": true
+    },
+    "fs.realpath": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
+      "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==",
+      "dev": true
+    },
+    "fsevents": {
+      "version": "2.3.2",
+      "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz",
+      "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==",
+      "dev": true,
+      "optional": true
+    },
+    "function-bind": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz",
+      "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==",
+      "dev": true
+    },
+    "function.prototype.name": {
+      "version": "1.1.5",
+      "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.5.tgz",
+      "integrity": "sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.3",
+        "es-abstract": "^1.19.0",
+        "functions-have-names": "^1.2.2"
+      }
+    },
+    "functional-red-black-tree": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz",
+      "integrity": "sha512-dsKNQNdj6xA3T+QlADDA7mOSlX0qiMINjn0cgr+eGHGsbSHzTabcIogz2+p/iqP1Xs6EP/sS2SbqH+brGTbq0g==",
+      "dev": true
+    },
+    "functions-have-names": {
+      "version": "1.2.3",
+      "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz",
+      "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==",
+      "dev": true
+    },
+    "ganache": {
+      "version": "7.7.6",
+      "resolved": "https://registry.npmjs.org/ganache/-/ganache-7.7.6.tgz",
+      "integrity": "sha512-1ba5CERykZijw1kIRGUKKPEUqTDU+sEMElYemAS42w1kunu+/3OS5v+eQsJQ+fCVMEmspploA7S9rEWBcyVsLg==",
+      "dev": true,
+      "requires": {
+        "@trufflesuite/bigint-buffer": "1.1.10",
+        "@types/bn.js": "^5.1.0",
+        "@types/lru-cache": "5.1.1",
+        "@types/seedrandom": "3.0.1",
+        "abstract-level": "1.0.3",
+        "abstract-leveldown": "7.2.0",
+        "async-eventemitter": "0.2.4",
+        "bufferutil": "4.0.5",
+        "emittery": "0.10.0",
+        "keccak": "3.0.2",
+        "leveldown": "6.1.0",
+        "secp256k1": "4.0.3",
+        "utf-8-validate": "5.0.7"
+      },
+      "dependencies": {
+        "@trufflesuite/bigint-buffer": {
+          "version": "1.1.10",
+          "resolved": "https://registry.npmjs.org/@trufflesuite/bigint-buffer/-/bigint-buffer-1.1.10.tgz",
+          "integrity": "sha512-pYIQC5EcMmID74t26GCC67946mgTJFiLXOT/BYozgrd4UEY2JHEGLhWi9cMiQCt5BSqFEvKkCHNnoj82SRjiEw==",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "node-gyp-build": "4.4.0"
+          },
+          "dependencies": {
+            "node-gyp-build": {
+              "version": "4.4.0",
+              "resolved": "https://registry.npmjs.org/node-gyp-build/-/node-gyp-build-4.4.0.tgz",
+              "integrity": "sha512-amJnQCcgtRVw9SvoebO3BKGESClrfXGCUTX9hSn1OuGQTQBOZmVd0Z0OlecpuRksKvbsUqALE8jls/ErClAPuQ==",
+              "bundled": true,
+              "dev": true
+            }
+          }
+        },
+        "@types/bn.js": {
+          "version": "5.1.0",
+          "resolved": "https://registry.npmjs.org/@types/bn.js/-/bn.js-5.1.0.tgz",
+          "integrity": "sha512-QSSVYj7pYFN49kW77o2s9xTCwZ8F2xLbjLLSEVh8D2F4JUhZtPAGOFLTD+ffqksBx/u4cE/KImFjyhqCjn/LIA==",
+          "dev": true,
+          "requires": {
+            "@types/node": "*"
+          }
+        },
+        "@types/lru-cache": {
+          "version": "5.1.1",
+          "resolved": "https://registry.npmjs.org/@types/lru-cache/-/lru-cache-5.1.1.tgz",
+          "integrity": "sha512-ssE3Vlrys7sdIzs5LOxCzTVMsU7i9oa/IaW92wF32JFb3CVczqOkru2xspuKczHEbG3nvmPY7IFqVmGGHdNbYw==",
+          "dev": true
+        },
+        "@types/node": {
+          "version": "17.0.0",
+          "resolved": "https://registry.npmjs.org/@types/node/-/node-17.0.0.tgz",
+          "integrity": "sha512-eMhwJXc931Ihh4tkU+Y7GiLzT/y/DBNpNtr4yU9O2w3SYBsr9NaOPhQlLKRmoWtI54uNwuo0IOUFQjVOTZYRvw==",
+          "dev": true
+        },
+        "@types/seedrandom": {
+          "version": "3.0.1",
+          "resolved": "https://registry.npmjs.org/@types/seedrandom/-/seedrandom-3.0.1.tgz",
+          "integrity": "sha512-giB9gzDeiCeloIXDgzFBCgjj1k4WxcDrZtGl6h1IqmUPlxF+Nx8Ve+96QCyDZ/HseB/uvDsKbpib9hU5cU53pw==",
+          "dev": true
+        },
+        "abstract-level": {
+          "version": "1.0.3",
+          "resolved": "https://registry.npmjs.org/abstract-level/-/abstract-level-1.0.3.tgz",
+          "integrity": "sha512-t6jv+xHy+VYwc4xqZMn2Pa9DjcdzvzZmQGRjTFc8spIbRGHgBrEKbPq+rYXc7CCo0lxgYvSgKVg9qZAhpVQSjA==",
+          "dev": true,
+          "requires": {
+            "buffer": "^6.0.3",
+            "catering": "^2.1.0",
+            "is-buffer": "^2.0.5",
+            "level-supports": "^4.0.0",
+            "level-transcoder": "^1.0.1",
+            "module-error": "^1.0.1",
+            "queue-microtask": "^1.2.3"
+          },
+          "dependencies": {
+            "level-supports": {
+              "version": "4.0.1",
+              "resolved": "https://registry.npmjs.org/level-supports/-/level-supports-4.0.1.tgz",
+              "integrity": "sha512-PbXpve8rKeNcZ9C1mUicC9auIYFyGpkV9/i6g76tLgANwWhtG2v7I4xNBUlkn3lE2/dZF3Pi0ygYGtLc4RXXdA==",
+              "dev": true
+            }
+          }
+        },
+        "abstract-leveldown": {
+          "version": "7.2.0",
+          "resolved": "https://registry.npmjs.org/abstract-leveldown/-/abstract-leveldown-7.2.0.tgz",
+          "integrity": "sha512-DnhQwcFEaYsvYDnACLZhMmCWd3rkOeEvglpa4q5i/5Jlm3UIsWaxVzuXvDLFCSCWRO3yy2/+V/G7FusFgejnfQ==",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "buffer": "^6.0.3",
+            "catering": "^2.0.0",
+            "is-buffer": "^2.0.5",
+            "level-concat-iterator": "^3.0.0",
+            "level-supports": "^2.0.1",
+            "queue-microtask": "^1.2.3"
+          }
+        },
+        "async": {
+          "version": "2.6.4",
+          "resolved": "https://registry.npmjs.org/async/-/async-2.6.4.tgz",
+          "integrity": "sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==",
+          "dev": true,
+          "requires": {
+            "lodash": "^4.17.14"
+          }
+        },
+        "async-eventemitter": {
+          "version": "0.2.4",
+          "resolved": "https://registry.npmjs.org/async-eventemitter/-/async-eventemitter-0.2.4.tgz",
+          "integrity": "sha512-pd20BwL7Yt1zwDFy+8MX8F1+WCT8aQeKj0kQnTrH9WaeRETlRamVhD0JtRPmrV4GfOJ2F9CvdQkZeZhnh2TuHw==",
+          "dev": true,
+          "requires": {
+            "async": "^2.4.0"
+          }
+        },
+        "base64-js": {
+          "version": "1.5.1",
+          "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz",
+          "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==",
+          "bundled": true,
+          "dev": true
+        },
+        "brorand": {
+          "version": "1.1.0",
+          "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz",
+          "integrity": "sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=",
+          "bundled": true,
+          "dev": true
+        },
+        "buffer": {
+          "version": "6.0.3",
+          "resolved": "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz",
+          "integrity": "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "base64-js": "^1.3.1",
+            "ieee754": "^1.2.1"
+          }
+        },
+        "bufferutil": {
+          "version": "4.0.5",
+          "resolved": "https://registry.npmjs.org/bufferutil/-/bufferutil-4.0.5.tgz",
+          "integrity": "sha512-HTm14iMQKK2FjFLRTM5lAVcyaUzOnqbPtesFIvREgXpJHdQm8bWS+GkQgIkfaBYRHuCnea7w8UVNfwiAQhlr9A==",
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "node-gyp-build": "^4.3.0"
+          }
+        },
+        "catering": {
+          "version": "2.1.0",
+          "resolved": "https://registry.npmjs.org/catering/-/catering-2.1.0.tgz",
+          "integrity": "sha512-M5imwzQn6y+ODBfgi+cfgZv2hIUI6oYU/0f35Mdb1ujGeqeoI5tOnl9Q13DTH7LW+7er+NYq8stNOKZD/Z3U/A==",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "queue-tick": "^1.0.0"
+          }
+        },
+        "elliptic": {
+          "version": "6.5.4",
+          "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.4.tgz",
+          "integrity": "sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ==",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "bn.js": "^4.11.9",
+            "brorand": "^1.1.0",
+            "hash.js": "^1.0.0",
+            "hmac-drbg": "^1.0.1",
+            "inherits": "^2.0.4",
+            "minimalistic-assert": "^1.0.1",
+            "minimalistic-crypto-utils": "^1.0.1"
+          },
+          "dependencies": {
+            "bn.js": {
+              "version": "4.12.0",
+              "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+              "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+              "bundled": true,
+              "dev": true
+            }
+          }
+        },
+        "emittery": {
+          "version": "0.10.0",
+          "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.10.0.tgz",
+          "integrity": "sha512-AGvFfs+d0JKCJQ4o01ASQLGPmSCxgfU9RFXvzPvZdjKK8oscynksuJhWrSTSw7j7Ep/sZct5b5ZhYCi8S/t0HQ==",
+          "dev": true
+        },
+        "hash.js": {
+          "version": "1.1.7",
+          "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz",
+          "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "inherits": "^2.0.3",
+            "minimalistic-assert": "^1.0.1"
+          }
+        },
+        "hmac-drbg": {
+          "version": "1.0.1",
+          "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz",
+          "integrity": "sha1-0nRXAQJabHdabFRXk+1QL8DGSaE=",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "hash.js": "^1.0.3",
+            "minimalistic-assert": "^1.0.0",
+            "minimalistic-crypto-utils": "^1.0.1"
+          }
+        },
+        "ieee754": {
+          "version": "1.2.1",
+          "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz",
+          "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==",
+          "bundled": true,
+          "dev": true
+        },
+        "inherits": {
+          "version": "2.0.4",
+          "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
+          "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
+          "bundled": true,
+          "dev": true
+        },
+        "is-buffer": {
+          "version": "2.0.5",
+          "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.5.tgz",
+          "integrity": "sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ==",
+          "bundled": true,
+          "dev": true
+        },
+        "keccak": {
+          "version": "3.0.2",
+          "resolved": "https://registry.npmjs.org/keccak/-/keccak-3.0.2.tgz",
+          "integrity": "sha512-PyKKjkH53wDMLGrvmRGSNWgmSxZOUqbnXwKL9tmgbFYA1iAYqW21kfR7mZXV0MlESiefxQQE9X9fTa3X+2MPDQ==",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "node-addon-api": "^2.0.0",
+            "node-gyp-build": "^4.2.0",
+            "readable-stream": "^3.6.0"
+          }
+        },
+        "level-concat-iterator": {
+          "version": "3.1.0",
+          "resolved": "https://registry.npmjs.org/level-concat-iterator/-/level-concat-iterator-3.1.0.tgz",
+          "integrity": "sha512-BWRCMHBxbIqPxJ8vHOvKUsaO0v1sLYZtjN3K2iZJsRBYtp+ONsY6Jfi6hy9K3+zolgQRryhIn2NRZjZnWJ9NmQ==",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "catering": "^2.1.0"
+          }
+        },
+        "level-supports": {
+          "version": "2.1.0",
+          "resolved": "https://registry.npmjs.org/level-supports/-/level-supports-2.1.0.tgz",
+          "integrity": "sha512-E486g1NCjW5cF78KGPrMDRBYzPuueMZ6VBXHT6gC7A8UYWGiM14fGgp+s/L1oFfDWSPV/+SFkYCmZ0SiESkRKA==",
+          "bundled": true,
+          "dev": true
+        },
+        "level-transcoder": {
+          "version": "1.0.1",
+          "resolved": "https://registry.npmjs.org/level-transcoder/-/level-transcoder-1.0.1.tgz",
+          "integrity": "sha512-t7bFwFtsQeD8cl8NIoQ2iwxA0CL/9IFw7/9gAjOonH0PWTTiRfY7Hq+Ejbsxh86tXobDQ6IOiddjNYIfOBs06w==",
+          "dev": true,
+          "requires": {
+            "buffer": "^6.0.3",
+            "module-error": "^1.0.1"
+          }
+        },
+        "leveldown": {
+          "version": "6.1.0",
+          "resolved": "https://registry.npmjs.org/leveldown/-/leveldown-6.1.0.tgz",
+          "integrity": "sha512-8C7oJDT44JXxh04aSSsfcMI8YiaGRhOFI9/pMEL7nWJLVsWajDPTRxsSHTM2WcTVY5nXM+SuRHzPPi0GbnDX+w==",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "abstract-leveldown": "^7.2.0",
+            "napi-macros": "~2.0.0",
+            "node-gyp-build": "^4.3.0"
+          }
+        },
+        "lodash": {
+          "version": "4.17.21",
+          "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
+          "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==",
+          "dev": true
+        },
+        "minimalistic-assert": {
+          "version": "1.0.1",
+          "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz",
+          "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==",
+          "bundled": true,
+          "dev": true
+        },
+        "minimalistic-crypto-utils": {
+          "version": "1.0.1",
+          "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz",
+          "integrity": "sha1-9sAMHAsIIkblxNmd+4x8CDsrWCo=",
+          "bundled": true,
+          "dev": true
+        },
+        "module-error": {
+          "version": "1.0.2",
+          "resolved": "https://registry.npmjs.org/module-error/-/module-error-1.0.2.tgz",
+          "integrity": "sha512-0yuvsqSCv8LbaOKhnsQ/T5JhyFlCYLPXK3U2sgV10zoKQwzs/MyfuQUOZQ1V/6OCOJsK/TRgNVrPuPDqtdMFtA==",
+          "dev": true
+        },
+        "napi-macros": {
+          "version": "2.0.0",
+          "resolved": "https://registry.npmjs.org/napi-macros/-/napi-macros-2.0.0.tgz",
+          "integrity": "sha512-A0xLykHtARfueITVDernsAWdtIMbOJgKgcluwENp3AlsKN/PloyO10HtmoqnFAQAcxPkgZN7wdfPfEd0zNGxbg==",
+          "bundled": true,
+          "dev": true
+        },
+        "node-addon-api": {
+          "version": "2.0.2",
+          "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-2.0.2.tgz",
+          "integrity": "sha512-Ntyt4AIXyaLIuMHF6IOoTakB3K+RWxwtsHNRxllEoA6vPwP9o4866g6YWDLUdnucilZhmkxiHwHr11gAENw+QA==",
+          "bundled": true,
+          "dev": true
+        },
+        "node-gyp-build": {
+          "version": "4.3.0",
+          "resolved": "https://registry.npmjs.org/node-gyp-build/-/node-gyp-build-4.3.0.tgz",
+          "integrity": "sha512-iWjXZvmboq0ja1pUGULQBexmxq8CV4xBhX7VDOTbL7ZR4FOowwY/VOtRxBN/yKxmdGoIp4j5ysNT4u3S2pDQ3Q==",
+          "bundled": true,
+          "dev": true
+        },
+        "queue-microtask": {
+          "version": "1.2.3",
+          "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz",
+          "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==",
+          "bundled": true,
+          "dev": true
+        },
+        "queue-tick": {
+          "version": "1.0.0",
+          "resolved": "https://registry.npmjs.org/queue-tick/-/queue-tick-1.0.0.tgz",
+          "integrity": "sha512-ULWhjjE8BmiICGn3G8+1L9wFpERNxkf8ysxkAer4+TFdRefDaXOCV5m92aMB9FtBVmn/8sETXLXY6BfW7hyaWQ==",
+          "bundled": true,
+          "dev": true
+        },
+        "readable-stream": {
+          "version": "3.6.0",
+          "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz",
+          "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "inherits": "^2.0.3",
+            "string_decoder": "^1.1.1",
+            "util-deprecate": "^1.0.1"
+          }
+        },
+        "safe-buffer": {
+          "version": "5.2.1",
+          "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
+          "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
+          "bundled": true,
+          "dev": true
+        },
+        "secp256k1": {
+          "version": "4.0.3",
+          "resolved": "https://registry.npmjs.org/secp256k1/-/secp256k1-4.0.3.tgz",
+          "integrity": "sha512-NLZVf+ROMxwtEj3Xa562qgv2BK5e2WNmXPiOdVIPLgs6lyTzMvBq0aWTYMI5XCP9jZMVKOcqZLw/Wc4vDkuxhA==",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "elliptic": "^6.5.4",
+            "node-addon-api": "^2.0.0",
+            "node-gyp-build": "^4.2.0"
+          }
+        },
+        "string_decoder": {
+          "version": "1.3.0",
+          "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz",
+          "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==",
+          "bundled": true,
+          "dev": true,
+          "requires": {
+            "safe-buffer": "~5.2.0"
+          }
+        },
+        "utf-8-validate": {
+          "version": "5.0.7",
+          "resolved": "https://registry.npmjs.org/utf-8-validate/-/utf-8-validate-5.0.7.tgz",
+          "integrity": "sha512-vLt1O5Pp+flcArHGIyKEQq883nBt8nN8tVBcoL0qUXj2XT1n7p70yGIq2VK98I5FdZ1YHc0wk/koOnHjnXWk1Q==",
+          "dev": true,
+          "optional": true,
+          "requires": {
+            "node-gyp-build": "^4.3.0"
+          }
+        },
+        "util-deprecate": {
+          "version": "1.0.2",
+          "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
+          "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=",
+          "bundled": true,
+          "dev": true
+        }
+      }
+    },
+    "get-caller-file": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-1.0.3.tgz",
+      "integrity": "sha512-3t6rVToeoZfYSGd8YoLFR2DJkiQrIiUrGcjvFX2mDw3bn6k2OtwHN0TNCLbBO+w8qTvimhDkv+LSscbJY1vE6w==",
+      "dev": true
+    },
+    "get-func-name": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/get-func-name/-/get-func-name-2.0.2.tgz",
+      "integrity": "sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ==",
+      "dev": true
+    },
+    "get-intrinsic": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.0.tgz",
+      "integrity": "sha512-L049y6nFOuom5wGyRc3/gdTLO94dySVKRACj1RmJZBQXlbTMhtNIgkWkUHq+jYmZvKf14EW1EoJnnjbmoHij0Q==",
+      "dev": true,
+      "requires": {
+        "function-bind": "^1.1.1",
+        "has": "^1.0.3",
+        "has-symbols": "^1.0.3"
+      }
+    },
+    "get-port": {
+      "version": "3.2.0",
+      "resolved": "https://registry.npmjs.org/get-port/-/get-port-3.2.0.tgz",
+      "integrity": "sha512-x5UJKlgeUiNT8nyo/AcnwLnZuZNcSjSw0kogRB+Whd1fjjFq4B1hySFxSFWWSn4mIBzg3sRNUDFYc4g5gjPoLg==",
+      "dev": true
+    },
+    "get-stream": {
+      "version": "6.0.1",
+      "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz",
+      "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==",
+      "dev": true
+    },
+    "get-symbol-description": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.0.tgz",
+      "integrity": "sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "get-intrinsic": "^1.1.1"
+      }
+    },
+    "getpass": {
+      "version": "0.1.7",
+      "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz",
+      "integrity": "sha512-0fzj9JxOLfJ+XGLhR8ze3unN0KZCgZwiSSDz168VERjK8Wl8kVSdcu2kspd4s4wtAa1y/qrVRiAA0WclVsu0ng==",
+      "dev": true,
+      "requires": {
+        "assert-plus": "^1.0.0"
+      }
+    },
+    "ghost-testrpc": {
+      "version": "0.0.2",
+      "resolved": "https://registry.npmjs.org/ghost-testrpc/-/ghost-testrpc-0.0.2.tgz",
+      "integrity": "sha512-i08dAEgJ2g8z5buJIrCTduwPIhih3DP+hOCTyyryikfV8T0bNvHnGXO67i0DD1H4GBDETTclPy9njZbfluQYrQ==",
+      "dev": true,
+      "requires": {
+        "chalk": "^2.4.2",
+        "node-emoji": "^1.10.0"
+      },
+      "dependencies": {
+        "ansi-styles": {
+          "version": "3.2.1",
+          "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+          "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+          "dev": true,
+          "requires": {
+            "color-convert": "^1.9.0"
+          }
+        },
+        "chalk": {
+          "version": "2.4.2",
+          "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+          "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+          "dev": true,
+          "requires": {
+            "ansi-styles": "^3.2.1",
+            "escape-string-regexp": "^1.0.5",
+            "supports-color": "^5.3.0"
+          }
+        },
+        "color-convert": {
+          "version": "1.9.3",
+          "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+          "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+          "dev": true,
+          "requires": {
+            "color-name": "1.1.3"
+          }
+        },
+        "color-name": {
+          "version": "1.1.3",
+          "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+          "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+          "dev": true
+        },
+        "has-flag": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+          "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+          "dev": true
+        },
+        "supports-color": {
+          "version": "5.5.0",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+          "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+          "dev": true,
+          "requires": {
+            "has-flag": "^3.0.0"
+          }
+        }
+      }
+    },
+    "glob": {
+      "version": "7.2.3",
+      "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz",
+      "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==",
+      "dev": true,
+      "requires": {
+        "fs.realpath": "^1.0.0",
+        "inflight": "^1.0.4",
+        "inherits": "2",
+        "minimatch": "^3.1.1",
+        "once": "^1.3.0",
+        "path-is-absolute": "^1.0.0"
+      }
+    },
+    "glob-parent": {
+      "version": "5.1.2",
+      "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
+      "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
+      "dev": true,
+      "requires": {
+        "is-glob": "^4.0.1"
+      }
+    },
+    "global": {
+      "version": "4.4.0",
+      "resolved": "https://registry.npmjs.org/global/-/global-4.4.0.tgz",
+      "integrity": "sha512-wv/LAoHdRE3BeTGz53FAamhGlPLhlssK45usmGFThIi4XqnBmjKQ16u+RNbP7WvigRZDxUsM0J3gcQ5yicaL0w==",
+      "dev": true,
+      "requires": {
+        "min-document": "^2.19.0",
+        "process": "^0.11.10"
+      }
+    },
+    "global-modules": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/global-modules/-/global-modules-2.0.0.tgz",
+      "integrity": "sha512-NGbfmJBp9x8IxyJSd1P+otYK8vonoJactOogrVfFRIAEY1ukil8RSKDz2Yo7wh1oihl51l/r6W4epkeKJHqL8A==",
+      "dev": true,
+      "requires": {
+        "global-prefix": "^3.0.0"
+      }
+    },
+    "global-prefix": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/global-prefix/-/global-prefix-3.0.0.tgz",
+      "integrity": "sha512-awConJSVCHVGND6x3tmMaKcQvwXLhjdkmomy2W+Goaui8YPgYgXJZewhg3fWC+DlfqqQuWg8AwqjGTD2nAPVWg==",
+      "dev": true,
+      "requires": {
+        "ini": "^1.3.5",
+        "kind-of": "^6.0.2",
+        "which": "^1.3.1"
+      },
+      "dependencies": {
+        "which": {
+          "version": "1.3.1",
+          "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz",
+          "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==",
+          "dev": true,
+          "requires": {
+            "isexe": "^2.0.0"
+          }
+        }
+      }
+    },
+    "globals": {
+      "version": "13.20.0",
+      "resolved": "https://registry.npmjs.org/globals/-/globals-13.20.0.tgz",
+      "integrity": "sha512-Qg5QtVkCy/kv3FUSlu4ukeZDVf9ee0iXLAUYX13gbR17bnejFTzr4iS9bY7kwCf1NztRNm1t91fjOiyx4CSwPQ==",
+      "dev": true,
+      "requires": {
+        "type-fest": "^0.20.2"
+      },
+      "dependencies": {
+        "type-fest": {
+          "version": "0.20.2",
+          "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz",
+          "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==",
+          "dev": true
+        }
+      }
+    },
+    "globalthis": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.3.tgz",
+      "integrity": "sha512-sFdI5LyBiNTHjRd7cGPWapiHWMOXKyuBNX/cWJ3NfzrZQVa8GI/8cofCl74AOVqq9W5kNmguTIzJ/1s2gyI9wA==",
+      "dev": true,
+      "requires": {
+        "define-properties": "^1.1.3"
+      }
+    },
+    "globby": {
+      "version": "11.1.0",
+      "resolved": "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz",
+      "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==",
+      "dev": true,
+      "requires": {
+        "array-union": "^2.1.0",
+        "dir-glob": "^3.0.1",
+        "fast-glob": "^3.2.9",
+        "ignore": "^5.2.0",
+        "merge2": "^1.4.1",
+        "slash": "^3.0.0"
+      }
+    },
+    "gopd": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz",
+      "integrity": "sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==",
+      "dev": true,
+      "requires": {
+        "get-intrinsic": "^1.1.3"
+      }
+    },
+    "got": {
+      "version": "12.1.0",
+      "resolved": "https://registry.npmjs.org/got/-/got-12.1.0.tgz",
+      "integrity": "sha512-hBv2ty9QN2RdbJJMK3hesmSkFTjVIHyIDDbssCKnSmq62edGgImJWD10Eb1k77TiV1bxloxqcFAVK8+9pkhOig==",
+      "dev": true,
+      "requires": {
+        "@sindresorhus/is": "^4.6.0",
+        "@szmarczak/http-timer": "^5.0.1",
+        "@types/cacheable-request": "^6.0.2",
+        "@types/responselike": "^1.0.0",
+        "cacheable-lookup": "^6.0.4",
+        "cacheable-request": "^7.0.2",
+        "decompress-response": "^6.0.0",
+        "form-data-encoder": "1.7.1",
+        "get-stream": "^6.0.1",
+        "http2-wrapper": "^2.1.10",
+        "lowercase-keys": "^3.0.0",
+        "p-cancelable": "^3.0.0",
+        "responselike": "^2.0.0"
+      }
+    },
+    "graceful-fs": {
+      "version": "4.2.10",
+      "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz",
+      "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==",
+      "dev": true
+    },
+    "grapheme-splitter": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/grapheme-splitter/-/grapheme-splitter-1.0.4.tgz",
+      "integrity": "sha512-bzh50DW9kTPM00T8y4o8vQg89Di9oLJVLW/KaOGIXJWP/iqCN6WKYkbNOF04vFLJhwcpYUh9ydh/+5vpOqV4YQ==",
+      "dev": true
+    },
+    "handlebars": {
+      "version": "4.7.7",
+      "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.7.tgz",
+      "integrity": "sha512-aAcXm5OAfE/8IXkcZvCepKU3VzW1/39Fb5ZuqMtgI/hT8X2YgoMvBY5dLhq/cpOvw7Lk1nK/UF71aLG/ZnVYRA==",
+      "dev": true,
+      "requires": {
+        "minimist": "^1.2.5",
+        "neo-async": "^2.6.0",
+        "source-map": "^0.6.1",
+        "uglify-js": "^3.1.4",
+        "wordwrap": "^1.0.0"
+      }
+    },
+    "har-schema": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz",
+      "integrity": "sha512-Oqluz6zhGX8cyRaTQlFMPw80bSJVG2x/cFb8ZPhUILGgHka9SsokCCOQgpveePerqidZOrT14ipqfJb7ILcW5Q==",
+      "dev": true
+    },
+    "har-validator": {
+      "version": "5.1.5",
+      "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.5.tgz",
+      "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==",
+      "dev": true,
+      "requires": {
+        "ajv": "^6.12.3",
+        "har-schema": "^2.0.0"
+      },
+      "dependencies": {
+        "ajv": {
+          "version": "6.12.6",
+          "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+          "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+          "dev": true,
+          "requires": {
+            "fast-deep-equal": "^3.1.1",
+            "fast-json-stable-stringify": "^2.0.0",
+            "json-schema-traverse": "^0.4.1",
+            "uri-js": "^4.2.2"
+          }
+        },
+        "json-schema-traverse": {
+          "version": "0.4.1",
+          "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+          "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+          "dev": true
+        }
+      }
+    },
+    "hardhat": {
+      "version": "2.12.2",
+      "resolved": "https://registry.npmjs.org/hardhat/-/hardhat-2.12.2.tgz",
+      "integrity": "sha512-f3ZhzXy1uyQv0UXnAQ8GCBOWjzv++WJNb7bnm10SsyC3dB7vlPpsMWBNhq7aoRxKrNhX9tCev81KFV3i5BTeMQ==",
+      "dev": true,
+      "requires": {
+        "@ethersproject/abi": "^5.1.2",
+        "@metamask/eth-sig-util": "^4.0.0",
+        "@nomicfoundation/ethereumjs-block": "^4.0.0",
+        "@nomicfoundation/ethereumjs-blockchain": "^6.0.0",
+        "@nomicfoundation/ethereumjs-common": "^3.0.0",
+        "@nomicfoundation/ethereumjs-evm": "^1.0.0",
+        "@nomicfoundation/ethereumjs-rlp": "^4.0.0",
+        "@nomicfoundation/ethereumjs-statemanager": "^1.0.0",
+        "@nomicfoundation/ethereumjs-trie": "^5.0.0",
+        "@nomicfoundation/ethereumjs-tx": "^4.0.0",
+        "@nomicfoundation/ethereumjs-util": "^8.0.0",
+        "@nomicfoundation/ethereumjs-vm": "^6.0.0",
+        "@nomicfoundation/solidity-analyzer": "^0.1.0",
+        "@sentry/node": "^5.18.1",
+        "@types/bn.js": "^5.1.0",
+        "@types/lru-cache": "^5.1.0",
+        "abort-controller": "^3.0.0",
+        "adm-zip": "^0.4.16",
+        "aggregate-error": "^3.0.0",
+        "ansi-escapes": "^4.3.0",
+        "chalk": "^2.4.2",
+        "chokidar": "^3.4.0",
+        "ci-info": "^2.0.0",
+        "debug": "^4.1.1",
+        "enquirer": "^2.3.0",
+        "env-paths": "^2.2.0",
+        "ethereum-cryptography": "^1.0.3",
+        "ethereumjs-abi": "^0.6.8",
+        "find-up": "^2.1.0",
+        "fp-ts": "1.19.3",
+        "fs-extra": "^7.0.1",
+        "glob": "7.2.0",
+        "immutable": "^4.0.0-rc.12",
+        "io-ts": "1.10.4",
+        "keccak": "^3.0.2",
+        "lodash": "^4.17.11",
+        "mnemonist": "^0.38.0",
+        "mocha": "^10.0.0",
+        "p-map": "^4.0.0",
+        "qs": "^6.7.0",
+        "raw-body": "^2.4.1",
+        "resolve": "1.17.0",
+        "semver": "^6.3.0",
+        "solc": "0.7.3",
+        "source-map-support": "^0.5.13",
+        "stacktrace-parser": "^0.1.10",
+        "tsort": "0.0.1",
+        "undici": "^5.4.0",
+        "uuid": "^8.3.2",
+        "ws": "^7.4.6"
+      },
+      "dependencies": {
+        "@nomicfoundation/ethereumjs-rlp": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.0.tgz",
+          "integrity": "sha512-GaSOGk5QbUk4eBP5qFbpXoZoZUj/NrW7MRa0tKY4Ew4c2HAS0GXArEMAamtFrkazp0BO4K5p2ZCG3b2FmbShmw==",
+          "dev": true
+        },
+        "ansi-styles": {
+          "version": "3.2.1",
+          "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+          "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+          "dev": true,
+          "requires": {
+            "color-convert": "^1.9.0"
+          }
+        },
+        "chalk": {
+          "version": "2.4.2",
+          "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+          "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+          "dev": true,
+          "requires": {
+            "ansi-styles": "^3.2.1",
+            "escape-string-regexp": "^1.0.5",
+            "supports-color": "^5.3.0"
+          }
+        },
+        "color-convert": {
+          "version": "1.9.3",
+          "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+          "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+          "dev": true,
+          "requires": {
+            "color-name": "1.1.3"
+          }
+        },
+        "color-name": {
+          "version": "1.1.3",
+          "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+          "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+          "dev": true
+        },
+        "ethereum-cryptography": {
+          "version": "1.2.0",
+          "resolved": "https://registry.npmjs.org/ethereum-cryptography/-/ethereum-cryptography-1.2.0.tgz",
+          "integrity": "sha512-6yFQC9b5ug6/17CQpCyE3k9eKBMdhyVjzUy1WkiuY/E4vj/SXDBbCw8QEIaXqf0Mf2SnY6RmpDcwlUmBSS0EJw==",
+          "dev": true,
+          "requires": {
+            "@noble/hashes": "1.2.0",
+            "@noble/secp256k1": "1.7.1",
+            "@scure/bip32": "1.1.5",
+            "@scure/bip39": "1.1.1"
+          }
+        },
+        "find-up": {
+          "version": "2.1.0",
+          "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz",
+          "integrity": "sha512-NWzkk0jSJtTt08+FBFMvXoeZnOJD+jTtsRmBYbAIzJdX6l7dLgR7CTubCM5/eDdPUBvLCeVasP1brfVR/9/EZQ==",
+          "dev": true,
+          "requires": {
+            "locate-path": "^2.0.0"
+          }
+        },
+        "glob": {
+          "version": "7.2.0",
+          "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz",
+          "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==",
+          "dev": true,
+          "requires": {
+            "fs.realpath": "^1.0.0",
+            "inflight": "^1.0.4",
+            "inherits": "2",
+            "minimatch": "^3.0.4",
+            "once": "^1.3.0",
+            "path-is-absolute": "^1.0.0"
+          }
+        },
+        "has-flag": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+          "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+          "dev": true
+        },
+        "jsonfile": {
+          "version": "2.4.0",
+          "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-2.4.0.tgz",
+          "integrity": "sha512-PKllAqbgLgxHaj8TElYymKCAgrASebJrWpTnEkOaTowt23VKXXN0sUeriJ+eh7y6ufb/CC5ap11pz71/cM0hUw==",
+          "dev": true,
+          "requires": {
+            "graceful-fs": "^4.1.6"
+          }
+        },
+        "resolve": {
+          "version": "1.17.0",
+          "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.17.0.tgz",
+          "integrity": "sha512-ic+7JYiV8Vi2yzQGFWOkiZD5Z9z7O2Zhm9XMaTxdJExKasieFCr+yXZ/WmXsckHiKl12ar0y6XiXDx3m4RHn1w==",
+          "dev": true,
+          "requires": {
+            "path-parse": "^1.0.6"
+          }
+        },
+        "rimraf": {
+          "version": "2.7.1",
+          "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz",
+          "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==",
+          "dev": true,
+          "requires": {
+            "glob": "^7.1.3"
+          }
+        },
+        "solc": {
+          "version": "0.7.3",
+          "resolved": "https://registry.npmjs.org/solc/-/solc-0.7.3.tgz",
+          "integrity": "sha512-GAsWNAjGzIDg7VxzP6mPjdurby3IkGCjQcM8GFYZT6RyaoUZKmMU6Y7YwG+tFGhv7dwZ8rmR4iwFDrrD99JwqA==",
+          "dev": true,
+          "requires": {
+            "command-exists": "^1.2.8",
+            "commander": "3.0.2",
+            "follow-redirects": "^1.12.1",
+            "fs-extra": "^0.30.0",
+            "js-sha3": "0.8.0",
+            "memorystream": "^0.3.1",
+            "require-from-string": "^2.0.0",
+            "semver": "^5.5.0",
+            "tmp": "0.0.33"
+          },
+          "dependencies": {
+            "fs-extra": {
+              "version": "0.30.0",
+              "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-0.30.0.tgz",
+              "integrity": "sha512-UvSPKyhMn6LEd/WpUaV9C9t3zATuqoqfWc3QdPhPLb58prN9tqYPlPWi8Krxi44loBoUzlobqZ3+8tGpxxSzwA==",
+              "dev": true,
+              "requires": {
+                "graceful-fs": "^4.1.2",
+                "jsonfile": "^2.1.0",
+                "klaw": "^1.0.0",
+                "path-is-absolute": "^1.0.0",
+                "rimraf": "^2.2.8"
+              }
+            },
+            "semver": {
+              "version": "5.7.2",
+              "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz",
+              "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==",
+              "dev": true
+            }
+          }
+        },
+        "supports-color": {
+          "version": "5.5.0",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+          "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+          "dev": true,
+          "requires": {
+            "has-flag": "^3.0.0"
+          }
+        },
+        "uuid": {
+          "version": "8.3.2",
+          "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz",
+          "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==",
+          "dev": true
+        }
+      }
+    },
+    "hardhat-deploy": {
+      "version": "0.11.25",
+      "resolved": "https://registry.npmjs.org/hardhat-deploy/-/hardhat-deploy-0.11.25.tgz",
+      "integrity": "sha512-ppSgrVE9A13YgTmf2PQGoyIs9o/jgJOMORrUP/rblU5K8mQ2YHWlPvkzZmP4h+SBW+tNmlnvSrf5K5DmMmExhw==",
+      "dev": true,
+      "requires": {
+        "@types/qs": "^6.9.7",
+        "axios": "^0.21.1",
+        "chalk": "^4.1.2",
+        "chokidar": "^3.5.2",
+        "debug": "^4.3.2",
+        "enquirer": "^2.3.6",
+        "ethers": "^5.5.3",
+        "form-data": "^4.0.0",
+        "fs-extra": "^10.0.0",
+        "match-all": "^1.2.6",
+        "murmur-128": "^0.2.1",
+        "qs": "^6.9.4",
+        "zksync-web3": "^0.8.1"
+      },
+      "dependencies": {
+        "axios": {
+          "version": "0.21.4",
+          "resolved": "https://registry.npmjs.org/axios/-/axios-0.21.4.tgz",
+          "integrity": "sha512-ut5vewkiu8jjGBdqpM44XxjuCjq9LAKeHVmoVfHVzy8eHgxxq8SbAVQNovDA8mVi05kP0Ea/n/UzcSHcTJQfNg==",
+          "dev": true,
+          "requires": {
+            "follow-redirects": "^1.14.0"
+          }
+        },
+        "form-data": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz",
+          "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
+          "dev": true,
+          "requires": {
+            "asynckit": "^0.4.0",
+            "combined-stream": "^1.0.8",
+            "mime-types": "^2.1.12"
+          }
+        },
+        "fs-extra": {
+          "version": "10.1.0",
+          "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-10.1.0.tgz",
+          "integrity": "sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ==",
+          "dev": true,
+          "requires": {
+            "graceful-fs": "^4.2.0",
+            "jsonfile": "^6.0.1",
+            "universalify": "^2.0.0"
+          }
+        },
+        "jsonfile": {
+          "version": "6.1.0",
+          "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz",
+          "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==",
+          "dev": true,
+          "requires": {
+            "graceful-fs": "^4.1.6",
+            "universalify": "^2.0.0"
+          }
+        },
+        "universalify": {
+          "version": "2.0.0",
+          "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz",
+          "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==",
+          "dev": true
+        }
+      }
+    },
+    "hardhat-gas-reporter": {
+      "version": "1.0.9",
+      "resolved": "https://registry.npmjs.org/hardhat-gas-reporter/-/hardhat-gas-reporter-1.0.9.tgz",
+      "integrity": "sha512-INN26G3EW43adGKBNzYWOlI3+rlLnasXTwW79YNnUhXPDa+yHESgt639dJEs37gCjhkbNKcRRJnomXEuMFBXJg==",
+      "dev": true,
+      "requires": {
+        "array-uniq": "1.0.3",
+        "eth-gas-reporter": "^0.2.25",
+        "sha1": "^1.1.1"
+      }
+    },
+    "has": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz",
+      "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==",
+      "dev": true,
+      "requires": {
+        "function-bind": "^1.1.1"
+      }
+    },
+    "has-bigints": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz",
+      "integrity": "sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==",
+      "dev": true
+    },
+    "has-flag": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
+      "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
+      "dev": true
+    },
+    "has-property-descriptors": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.0.tgz",
+      "integrity": "sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ==",
+      "dev": true,
+      "requires": {
+        "get-intrinsic": "^1.1.1"
+      }
+    },
+    "has-proto": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.1.tgz",
+      "integrity": "sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg==",
+      "dev": true
+    },
+    "has-symbols": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz",
+      "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==",
+      "dev": true
+    },
+    "has-tostringtag": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.0.tgz",
+      "integrity": "sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==",
+      "dev": true,
+      "requires": {
+        "has-symbols": "^1.0.2"
+      }
+    },
+    "hash-base": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/hash-base/-/hash-base-3.1.0.tgz",
+      "integrity": "sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA==",
+      "dev": true,
+      "requires": {
+        "inherits": "^2.0.4",
+        "readable-stream": "^3.6.0",
+        "safe-buffer": "^5.2.0"
+      }
+    },
+    "hash.js": {
+      "version": "1.1.7",
+      "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz",
+      "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==",
+      "devOptional": true,
+      "requires": {
+        "inherits": "^2.0.3",
+        "minimalistic-assert": "^1.0.1"
+      }
+    },
+    "he": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz",
+      "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==",
+      "dev": true
+    },
+    "header-case": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/header-case/-/header-case-1.0.1.tgz",
+      "integrity": "sha512-i0q9mkOeSuhXw6bGgiQCCBgY/jlZuV/7dZXyZ9c6LcBrqwvT8eT719E9uxE5LiZftdl+z81Ugbg/VvXV4OJOeQ==",
+      "dev": true,
+      "requires": {
+        "no-case": "^2.2.0",
+        "upper-case": "^1.1.3"
+      }
+    },
+    "heap": {
+      "version": "0.2.7",
+      "resolved": "https://registry.npmjs.org/heap/-/heap-0.2.7.tgz",
+      "integrity": "sha512-2bsegYkkHO+h/9MGbn6KWcE45cHZgPANo5LXF7EvWdT0yT2EguSVO1nDgU5c8+ZOPwp2vMNa7YFsJhVcDR9Sdg==",
+      "dev": true
+    },
+    "highlight.js": {
+      "version": "10.7.3",
+      "resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-10.7.3.tgz",
+      "integrity": "sha512-tzcUFauisWKNHaRkN4Wjl/ZA07gENAjFl3J/c480dprkGTg5EQstgaNFqBfUqCq54kZRIEcreTsAgF/m2quD7A==",
+      "dev": true
+    },
+    "highlightjs-solidity": {
+      "version": "2.0.6",
+      "resolved": "https://registry.npmjs.org/highlightjs-solidity/-/highlightjs-solidity-2.0.6.tgz",
+      "integrity": "sha512-DySXWfQghjm2l6a/flF+cteroJqD4gI8GSdL4PtvxZSsAHie8m3yVe2JFoRg03ROKT6hp2Lc/BxXkqerNmtQYg==",
+      "dev": true
+    },
+    "hmac-drbg": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz",
+      "integrity": "sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==",
+      "devOptional": true,
+      "requires": {
+        "hash.js": "^1.0.3",
+        "minimalistic-assert": "^1.0.0",
+        "minimalistic-crypto-utils": "^1.0.1"
+      }
+    },
+    "hosted-git-info": {
+      "version": "2.8.9",
+      "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz",
+      "integrity": "sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==",
+      "dev": true
+    },
+    "htmlparser2": {
+      "version": "8.0.1",
+      "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-8.0.1.tgz",
+      "integrity": "sha512-4lVbmc1diZC7GUJQtRQ5yBAeUCL1exyMwmForWkRLnwyzWBFxN633SALPMGYaWZvKe9j1pRZJpauvmxENSp/EA==",
+      "dev": true,
+      "requires": {
+        "domelementtype": "^2.3.0",
+        "domhandler": "^5.0.2",
+        "domutils": "^3.0.1",
+        "entities": "^4.3.0"
+      }
+    },
+    "http-basic": {
+      "version": "8.1.3",
+      "resolved": "https://registry.npmjs.org/http-basic/-/http-basic-8.1.3.tgz",
+      "integrity": "sha512-/EcDMwJZh3mABI2NhGfHOGOeOZITqfkEO4p/xK+l3NpyncIHUQBoMvCSF/b5GqvKtySC2srL/GGG3+EtlqlmCw==",
+      "dev": true,
+      "requires": {
+        "caseless": "^0.12.0",
+        "concat-stream": "^1.6.2",
+        "http-response-object": "^3.0.1",
+        "parse-cache-control": "^1.0.1"
+      }
+    },
+    "http-cache-semantics": {
+      "version": "4.1.1",
+      "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.1.tgz",
+      "integrity": "sha512-er295DKPVsV82j5kw1Gjt+ADA/XYHsajl82cGNQG2eyoPkvgUhX+nDIyelzhIWbbsXP39EHcI6l5tYs2FYqYXQ==",
+      "dev": true
+    },
+    "http-errors": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz",
+      "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==",
+      "dev": true,
+      "requires": {
+        "depd": "2.0.0",
+        "inherits": "2.0.4",
+        "setprototypeof": "1.2.0",
+        "statuses": "2.0.1",
+        "toidentifier": "1.0.1"
+      }
+    },
+    "http-https": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/http-https/-/http-https-1.0.0.tgz",
+      "integrity": "sha512-o0PWwVCSp3O0wS6FvNr6xfBCHgt0m1tvPLFOCc2iFDKTRAXhB7m8klDf7ErowFH8POa6dVdGatKU5I1YYwzUyg==",
+      "dev": true
+    },
+    "http-response-object": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/http-response-object/-/http-response-object-3.0.2.tgz",
+      "integrity": "sha512-bqX0XTF6fnXSQcEJ2Iuyr75yVakyjIDCqroJQ/aHfSdlM743Cwqoi2nDYMzLGWUcuTWGWy8AAvOKXTfiv6q9RA==",
+      "dev": true,
+      "requires": {
+        "@types/node": "^10.0.3"
+      },
+      "dependencies": {
+        "@types/node": {
+          "version": "10.17.60",
+          "resolved": "https://registry.npmjs.org/@types/node/-/node-10.17.60.tgz",
+          "integrity": "sha512-F0KIgDJfy2nA3zMLmWGKxcH2ZVEtCZXHHdOQs2gSaQ27+lNeEfGxzkIw90aXswATX7AZ33tahPbzy6KAfUreVw==",
+          "dev": true
+        }
+      }
+    },
+    "http-signature": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz",
+      "integrity": "sha512-CAbnr6Rz4CYQkLYUtSNXxQPUH2gK8f3iWexVlsnMeD+GjlsQ0Xsy1cOX+mN3dtxYomRy21CiOzU8Uhw6OwncEQ==",
+      "dev": true,
+      "requires": {
+        "assert-plus": "^1.0.0",
+        "jsprim": "^1.2.2",
+        "sshpk": "^1.7.0"
+      }
+    },
+    "http2-wrapper": {
+      "version": "2.2.0",
+      "resolved": "https://registry.npmjs.org/http2-wrapper/-/http2-wrapper-2.2.0.tgz",
+      "integrity": "sha512-kZB0wxMo0sh1PehyjJUWRFEd99KC5TLjZ2cULC4f9iqJBAmKQQXEICjxl5iPJRwP40dpeHFqqhm7tYCvODpqpQ==",
+      "dev": true,
+      "requires": {
+        "quick-lru": "^5.1.1",
+        "resolve-alpn": "^1.2.0"
+      }
+    },
+    "https-proxy-agent": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz",
+      "integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==",
+      "dev": true,
+      "requires": {
+        "agent-base": "6",
+        "debug": "4"
+      }
+    },
+    "husky": {
+      "version": "4.3.8",
+      "resolved": "https://registry.npmjs.org/husky/-/husky-4.3.8.tgz",
+      "integrity": "sha512-LCqqsB0PzJQ/AlCgfrfzRe3e3+NvmefAdKQhRYpxS4u6clblBoDdzzvHi8fmxKRzvMxPY/1WZWzomPZww0Anow==",
+      "dev": true,
+      "requires": {
+        "chalk": "^4.0.0",
+        "ci-info": "^2.0.0",
+        "compare-versions": "^3.6.0",
+        "cosmiconfig": "^7.0.0",
+        "find-versions": "^4.0.0",
+        "opencollective-postinstall": "^2.0.2",
+        "pkg-dir": "^5.0.0",
+        "please-upgrade-node": "^3.2.0",
+        "slash": "^3.0.0",
+        "which-pm-runs": "^1.0.0"
+      }
+    },
+    "hyperlinker": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/hyperlinker/-/hyperlinker-1.0.0.tgz",
+      "integrity": "sha512-Ty8UblRWFEcfSuIaajM34LdPXIhbs1ajEX/BBPv24J+enSVaEVY63xQ6lTO9VRYS5LAoghIG0IDJ+p+IPzKUQQ==",
+      "dev": true
+    },
+    "iconv-lite": {
+      "version": "0.4.24",
+      "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
+      "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
+      "dev": true,
+      "requires": {
+        "safer-buffer": ">= 2.1.2 < 3"
+      }
+    },
+    "idna-uts46-hx": {
+      "version": "2.3.1",
+      "resolved": "https://registry.npmjs.org/idna-uts46-hx/-/idna-uts46-hx-2.3.1.tgz",
+      "integrity": "sha512-PWoF9Keq6laYdIRwwCdhTPl60xRqAloYNMQLiyUnG42VjT53oW07BXIRM+NK7eQjzXjAk2gUvX9caRxlnF9TAA==",
+      "dev": true,
+      "requires": {
+        "punycode": "2.1.0"
+      },
+      "dependencies": {
+        "punycode": {
+          "version": "2.1.0",
+          "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.0.tgz",
+          "integrity": "sha512-Yxz2kRwT90aPiWEMHVYnEf4+rhwF1tBmmZ4KepCP+Wkium9JxtWnUm1nqGwpiAHr/tnTSeHqr3wb++jgSkXjhA==",
+          "dev": true
+        }
+      }
+    },
+    "ieee754": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz",
+      "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==",
+      "dev": true
+    },
+    "ignore": {
+      "version": "5.2.4",
+      "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.4.tgz",
+      "integrity": "sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==",
+      "dev": true
+    },
+    "immutable": {
+      "version": "4.2.4",
+      "resolved": "https://registry.npmjs.org/immutable/-/immutable-4.2.4.tgz",
+      "integrity": "sha512-WDxL3Hheb1JkRN3sQkyujNlL/xRjAo3rJtaU5xeufUauG66JdMr32bLj4gF+vWl84DIA3Zxw7tiAjneYzRRw+w==",
+      "dev": true
+    },
+    "import-fresh": {
+      "version": "3.3.0",
+      "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz",
+      "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==",
+      "dev": true,
+      "requires": {
+        "parent-module": "^1.0.0",
+        "resolve-from": "^4.0.0"
+      }
+    },
+    "imul": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/imul/-/imul-1.0.1.tgz",
+      "integrity": "sha512-WFAgfwPLAjU66EKt6vRdTlKj4nAgIDQzh29JonLa4Bqtl6D8JrIMvWjCnx7xEjVNmP3U0fM5o8ZObk7d0f62bA==",
+      "dev": true
+    },
+    "imurmurhash": {
+      "version": "0.1.4",
+      "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
+      "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==",
+      "dev": true
+    },
+    "indent-string": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz",
+      "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==",
+      "dev": true
+    },
+    "inflight": {
+      "version": "1.0.6",
+      "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
+      "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==",
+      "dev": true,
+      "requires": {
+        "once": "^1.3.0",
+        "wrappy": "1"
+      }
+    },
+    "inherits": {
+      "version": "2.0.4",
+      "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
+      "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
+      "devOptional": true
+    },
+    "ini": {
+      "version": "1.3.8",
+      "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz",
+      "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==",
+      "dev": true
+    },
+    "internal-slot": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.5.tgz",
+      "integrity": "sha512-Y+R5hJrzs52QCG2laLn4udYVnxsfny9CpOhNhUvk/SSSVyF6T27FzRbF0sroPidSu3X8oEAkOn2K804mjpt6UQ==",
+      "dev": true,
+      "requires": {
+        "get-intrinsic": "^1.2.0",
+        "has": "^1.0.3",
+        "side-channel": "^1.0.4"
+      }
+    },
+    "interpret": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/interpret/-/interpret-1.4.0.tgz",
+      "integrity": "sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA==",
+      "dev": true
+    },
+    "invert-kv": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/invert-kv/-/invert-kv-1.0.0.tgz",
+      "integrity": "sha512-xgs2NH9AE66ucSq4cNG1nhSFghr5l6tdL15Pk+jl46bmmBapgoaY/AacXyaDznAqmGL99TiLSQgO/XazFSKYeQ==",
+      "dev": true
+    },
+    "io-ts": {
+      "version": "1.10.4",
+      "resolved": "https://registry.npmjs.org/io-ts/-/io-ts-1.10.4.tgz",
+      "integrity": "sha512-b23PteSnYXSONJ6JQXRAlvJhuw8KOtkqa87W4wDtvMrud/DTJd5X+NpOOI+O/zZwVq6v0VLAaJ+1EDViKEuN9g==",
+      "dev": true,
+      "requires": {
+        "fp-ts": "^1.0.0"
+      }
+    },
+    "ipaddr.js": {
+      "version": "1.9.1",
+      "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
+      "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==",
+      "dev": true
+    },
+    "is-arguments": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.1.1.tgz",
+      "integrity": "sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "has-tostringtag": "^1.0.0"
+      }
+    },
+    "is-array-buffer": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.2.tgz",
+      "integrity": "sha512-y+FyyR/w8vfIRq4eQcM1EYgSTnmHXPqaF+IgzgraytCFq5Xh8lllDVmAZolPJiZttZLeFSINPYMaEJ7/vWUa1w==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "get-intrinsic": "^1.2.0",
+        "is-typed-array": "^1.1.10"
+      }
+    },
+    "is-arrayish": {
+      "version": "0.2.1",
+      "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz",
+      "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==",
+      "dev": true
+    },
+    "is-bigint": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz",
+      "integrity": "sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==",
+      "dev": true,
+      "requires": {
+        "has-bigints": "^1.0.1"
+      }
+    },
+    "is-binary-path": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
+      "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
+      "dev": true,
+      "requires": {
+        "binary-extensions": "^2.0.0"
+      }
+    },
+    "is-boolean-object": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.1.2.tgz",
+      "integrity": "sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "has-tostringtag": "^1.0.0"
+      }
+    },
+    "is-buffer": {
+      "version": "2.0.5",
+      "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.5.tgz",
+      "integrity": "sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ==",
+      "dev": true
+    },
+    "is-callable": {
+      "version": "1.2.7",
+      "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz",
+      "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==",
+      "dev": true
+    },
+    "is-core-module": {
+      "version": "2.11.0",
+      "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.11.0.tgz",
+      "integrity": "sha512-RRjxlvLDkD1YJwDbroBHMb+cukurkDWNyHx7D3oNB5x9rb5ogcksMC5wHCadcXoo67gVr/+3GFySh3134zi6rw==",
+      "dev": true,
+      "requires": {
+        "has": "^1.0.3"
+      }
+    },
+    "is-date-object": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.5.tgz",
+      "integrity": "sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==",
+      "dev": true,
+      "requires": {
+        "has-tostringtag": "^1.0.0"
+      }
+    },
+    "is-docker": {
+      "version": "2.2.1",
+      "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.2.1.tgz",
+      "integrity": "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==",
+      "dev": true
+    },
+    "is-extglob": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
+      "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==",
+      "dev": true
+    },
+    "is-fullwidth-code-point": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
+      "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
+      "dev": true
+    },
+    "is-function": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/is-function/-/is-function-1.0.2.tgz",
+      "integrity": "sha512-lw7DUp0aWXYg+CBCN+JKkcE0Q2RayZnSvnZBlwgxHBQhqt5pZNVy4Ri7H9GmmXkdu7LUthszM+Tor1u/2iBcpQ==",
+      "dev": true
+    },
+    "is-generator-function": {
+      "version": "1.0.10",
+      "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.0.10.tgz",
+      "integrity": "sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==",
+      "dev": true,
+      "requires": {
+        "has-tostringtag": "^1.0.0"
+      }
+    },
+    "is-glob": {
+      "version": "4.0.3",
+      "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz",
+      "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==",
+      "dev": true,
+      "requires": {
+        "is-extglob": "^2.1.1"
+      }
+    },
+    "is-hex-prefixed": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-hex-prefixed/-/is-hex-prefixed-1.0.0.tgz",
+      "integrity": "sha512-WvtOiug1VFrE9v1Cydwm+FnXd3+w9GaeVUss5W4v/SLy3UW00vP+6iNF2SdnfiBoLy4bTqVdkftNGTUeOFVsbA==",
+      "dev": true
+    },
+    "is-interactive": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-interactive/-/is-interactive-1.0.0.tgz",
+      "integrity": "sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==",
+      "dev": true
+    },
+    "is-lower-case": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/is-lower-case/-/is-lower-case-1.1.3.tgz",
+      "integrity": "sha512-+5A1e/WJpLLXZEDlgz4G//WYSHyQBD32qa4Jd3Lw06qQlv3fJHnp3YIHjTQSGzHMgzmVKz2ZP3rBxTHkPw/lxA==",
+      "dev": true,
+      "requires": {
+        "lower-case": "^1.1.0"
+      }
+    },
+    "is-negative-zero": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz",
+      "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==",
+      "dev": true
+    },
+    "is-number": {
+      "version": "7.0.0",
+      "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
+      "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
+      "dev": true
+    },
+    "is-number-object": {
+      "version": "1.0.7",
+      "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.0.7.tgz",
+      "integrity": "sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==",
+      "dev": true,
+      "requires": {
+        "has-tostringtag": "^1.0.0"
+      }
+    },
+    "is-path-inside": {
+      "version": "3.0.3",
+      "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz",
+      "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==",
+      "dev": true
+    },
+    "is-plain-obj": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-2.1.0.tgz",
+      "integrity": "sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==",
+      "dev": true
+    },
+    "is-regex": {
+      "version": "1.1.4",
+      "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz",
+      "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "has-tostringtag": "^1.0.0"
+      }
+    },
+    "is-shared-array-buffer": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz",
+      "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2"
+      }
+    },
+    "is-string": {
+      "version": "1.0.7",
+      "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz",
+      "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==",
+      "dev": true,
+      "requires": {
+        "has-tostringtag": "^1.0.0"
+      }
+    },
+    "is-symbol": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz",
+      "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==",
+      "dev": true,
+      "requires": {
+        "has-symbols": "^1.0.2"
+      }
+    },
+    "is-typed-array": {
+      "version": "1.1.10",
+      "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.10.tgz",
+      "integrity": "sha512-PJqgEHiWZvMpaFZ3uTc8kHPM4+4ADTlDniuQL7cU/UDA0Ql7F70yGfHph3cLNe+c9toaigv+DFzTJKhc2CtO6A==",
+      "dev": true,
+      "requires": {
+        "available-typed-arrays": "^1.0.5",
+        "call-bind": "^1.0.2",
+        "for-each": "^0.3.3",
+        "gopd": "^1.0.1",
+        "has-tostringtag": "^1.0.0"
+      }
+    },
+    "is-typedarray": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz",
+      "integrity": "sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA==",
+      "dev": true
+    },
+    "is-unicode-supported": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz",
+      "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==",
+      "dev": true
+    },
+    "is-upper-case": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/is-upper-case/-/is-upper-case-1.1.2.tgz",
+      "integrity": "sha512-GQYSJMgfeAmVwh9ixyk888l7OIhNAGKtY6QA+IrWlu9MDTCaXmeozOZ2S9Knj7bQwBO/H6J2kb+pbyTUiMNbsw==",
+      "dev": true,
+      "requires": {
+        "upper-case": "^1.1.0"
+      }
+    },
+    "is-utf8": {
+      "version": "0.2.1",
+      "resolved": "https://registry.npmjs.org/is-utf8/-/is-utf8-0.2.1.tgz",
+      "integrity": "sha512-rMYPYvCzsXywIsldgLaSoPlw5PfoB/ssr7hY4pLfcodrA5M/eArza1a9VmTiNIBNMjOGr1Ow9mTyU2o69U6U9Q==",
+      "dev": true
+    },
+    "is-weakref": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz",
+      "integrity": "sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2"
+      }
+    },
+    "is-wsl": {
+      "version": "2.2.0",
+      "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-2.2.0.tgz",
+      "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==",
+      "dev": true,
+      "requires": {
+        "is-docker": "^2.0.0"
+      }
+    },
+    "isarray": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz",
+      "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==",
+      "dev": true
+    },
+    "isexe": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
+      "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==",
+      "dev": true
+    },
+    "isstream": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz",
+      "integrity": "sha512-Yljz7ffyPbrLpLngrMtZ7NduUgVvi6wG9RJ9IUcyCd59YQ911PBJphODUcbOVbqYfxe1wuYf/LJ8PauMRwsM/g==",
+      "dev": true
+    },
+    "js-base64": {
+      "version": "3.7.5",
+      "resolved": "https://registry.npmjs.org/js-base64/-/js-base64-3.7.5.tgz",
+      "integrity": "sha512-3MEt5DTINKqfScXKfJFrRbxkrnk2AxPWGBL/ycjz4dK8iqiSJ06UxD8jh8xuh6p10TX4t2+7FsBYVxxQbMg+qA==",
+      "dev": true
+    },
+    "js-sdsl": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/js-sdsl/-/js-sdsl-4.3.0.tgz",
+      "integrity": "sha512-mifzlm2+5nZ+lEcLJMoBK0/IH/bDg8XnJfd/Wq6IP+xoCjLZsTOnV2QpxlVbX9bMnkl5PdEjNtBJ9Cj1NjifhQ==",
+      "dev": true
+    },
+    "js-sha3": {
+      "version": "0.8.0",
+      "resolved": "https://registry.npmjs.org/js-sha3/-/js-sha3-0.8.0.tgz",
+      "integrity": "sha512-gF1cRrHhIzNfToc802P800N8PpXS+evLLXfsVpowqmAFR9uwbi89WvXg2QspOmXL8QL86J4T1EpFu+yUkwJY3Q==",
+      "devOptional": true
+    },
+    "js-tokens": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
+      "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==",
+      "dev": true
+    },
+    "js-yaml": {
+      "version": "4.1.0",
+      "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz",
+      "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==",
+      "dev": true,
+      "requires": {
+        "argparse": "^2.0.1"
+      }
+    },
+    "jsbn": {
+      "version": "0.1.1",
+      "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz",
+      "integrity": "sha512-UVU9dibq2JcFWxQPA6KCqj5O42VOmAY3zQUfEKxU0KpTGXwNoCjkX1e13eHNvw/xPynt6pU0rZ1htjWTNTSXsg==",
+      "dev": true
+    },
+    "json-buffer": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz",
+      "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==",
+      "dev": true
+    },
+    "json-parse-even-better-errors": {
+      "version": "2.3.1",
+      "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz",
+      "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==",
+      "dev": true
+    },
+    "json-schema": {
+      "version": "0.4.0",
+      "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.4.0.tgz",
+      "integrity": "sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA==",
+      "dev": true
+    },
+    "json-schema-traverse": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz",
+      "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==",
+      "dev": true
+    },
+    "json-stable-stringify-without-jsonify": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz",
+      "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==",
+      "dev": true
+    },
+    "json-stringify-safe": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz",
+      "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==",
+      "dev": true
+    },
+    "json5": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz",
+      "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==",
+      "dev": true,
+      "requires": {
+        "minimist": "^1.2.0"
+      }
+    },
+    "jsonfile": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz",
+      "integrity": "sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg==",
+      "dev": true,
+      "requires": {
+        "graceful-fs": "^4.1.6"
+      }
+    },
+    "jsonschema": {
+      "version": "1.4.1",
+      "resolved": "https://registry.npmjs.org/jsonschema/-/jsonschema-1.4.1.tgz",
+      "integrity": "sha512-S6cATIPVv1z0IlxdN+zUk5EPjkGCdnhN4wVSBlvoUO1tOLJootbo9CquNJmbIh4yikWHiUedhRYrNPn1arpEmQ==",
+      "dev": true
+    },
+    "jsprim": {
+      "version": "1.4.2",
+      "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.2.tgz",
+      "integrity": "sha512-P2bSOMAc/ciLz6DzgjVlGJP9+BrJWu5UDGK70C2iweC5QBIeFf0ZXRvGjEj2uYgrY2MkAAhsSWHDWlFtEroZWw==",
+      "dev": true,
+      "requires": {
+        "assert-plus": "1.0.0",
+        "extsprintf": "1.3.0",
+        "json-schema": "0.4.0",
+        "verror": "1.10.0"
+      }
+    },
+    "keccak": {
+      "version": "3.0.3",
+      "resolved": "https://registry.npmjs.org/keccak/-/keccak-3.0.3.tgz",
+      "integrity": "sha512-JZrLIAJWuZxKbCilMpNz5Vj7Vtb4scDG3dMXLOsbzBmQGyjwE61BbW7bJkfKKCShXiQZt3T6sBgALRtmd+nZaQ==",
+      "dev": true,
+      "requires": {
+        "node-addon-api": "^2.0.0",
+        "node-gyp-build": "^4.2.0",
+        "readable-stream": "^3.6.0"
+      }
+    },
+    "keyv": {
+      "version": "4.5.2",
+      "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.2.tgz",
+      "integrity": "sha512-5MHbFaKn8cNSmVW7BYnijeAVlE4cYA/SVkifVgrh7yotnfhKmjuXpDKjrABLnT0SfHWV21P8ow07OGfRrNDg8g==",
+      "dev": true,
+      "requires": {
+        "json-buffer": "3.0.1"
+      }
+    },
+    "kind-of": {
+      "version": "6.0.3",
+      "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz",
+      "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==",
+      "dev": true
+    },
+    "klaw": {
+      "version": "1.3.1",
+      "resolved": "https://registry.npmjs.org/klaw/-/klaw-1.3.1.tgz",
+      "integrity": "sha512-TED5xi9gGQjGpNnvRWknrwAB1eL5GciPfVFOt3Vk1OJCVDQbzuSfrF3hkUQKlsgKrG1F+0t5W0m+Fje1jIt8rw==",
+      "dev": true,
+      "requires": {
+        "graceful-fs": "^4.1.9"
+      }
+    },
+    "kleur": {
+      "version": "3.0.3",
+      "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz",
+      "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==",
+      "dev": true
+    },
+    "lcid": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/lcid/-/lcid-1.0.0.tgz",
+      "integrity": "sha512-YiGkH6EnGrDGqLMITnGjXtGmNtjoXw9SVUzcaos8RBi7Ps0VBylkq+vOcY9QE5poLasPCR849ucFUkl0UzUyOw==",
+      "dev": true,
+      "requires": {
+        "invert-kv": "^1.0.0"
+      }
+    },
+    "level": {
+      "version": "8.0.0",
+      "resolved": "https://registry.npmjs.org/level/-/level-8.0.0.tgz",
+      "integrity": "sha512-ypf0jjAk2BWI33yzEaaotpq7fkOPALKAgDBxggO6Q9HGX2MRXn0wbP1Jn/tJv1gtL867+YOjOB49WaUF3UoJNQ==",
+      "dev": true,
+      "requires": {
+        "browser-level": "^1.0.1",
+        "classic-level": "^1.2.0"
+      }
+    },
+    "level-supports": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/level-supports/-/level-supports-4.0.1.tgz",
+      "integrity": "sha512-PbXpve8rKeNcZ9C1mUicC9auIYFyGpkV9/i6g76tLgANwWhtG2v7I4xNBUlkn3lE2/dZF3Pi0ygYGtLc4RXXdA==",
+      "dev": true
+    },
+    "level-transcoder": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/level-transcoder/-/level-transcoder-1.0.1.tgz",
+      "integrity": "sha512-t7bFwFtsQeD8cl8NIoQ2iwxA0CL/9IFw7/9gAjOonH0PWTTiRfY7Hq+Ejbsxh86tXobDQ6IOiddjNYIfOBs06w==",
+      "dev": true,
+      "requires": {
+        "buffer": "^6.0.3",
+        "module-error": "^1.0.1"
+      },
+      "dependencies": {
+        "buffer": {
+          "version": "6.0.3",
+          "resolved": "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz",
+          "integrity": "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==",
+          "dev": true,
+          "requires": {
+            "base64-js": "^1.3.1",
+            "ieee754": "^1.2.1"
+          }
+        }
+      }
+    },
+    "levn": {
+      "version": "0.4.1",
+      "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz",
+      "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==",
+      "dev": true,
+      "requires": {
+        "prelude-ls": "^1.2.1",
+        "type-check": "~0.4.0"
+      }
+    },
+    "lines-and-columns": {
+      "version": "1.2.4",
+      "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz",
+      "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==",
+      "dev": true
+    },
+    "load-json-file": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-1.1.0.tgz",
+      "integrity": "sha512-cy7ZdNRXdablkXYNI049pthVeXFurRyb9+hA/dZzerZ0pGTx42z+y+ssxBaVV2l70t1muq5IdKhn4UtcoGUY9A==",
+      "dev": true,
+      "requires": {
+        "graceful-fs": "^4.1.2",
+        "parse-json": "^2.2.0",
+        "pify": "^2.0.0",
+        "pinkie-promise": "^2.0.0",
+        "strip-bom": "^2.0.0"
+      }
+    },
+    "locate-path": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz",
+      "integrity": "sha512-NCI2kiDkyR7VeEKm27Kda/iQHyKJe1Bu0FlTbYp3CqJu+9IFe9bLyAjMxf5ZDDbEg+iMPzB5zYyUTSm8wVTKmA==",
+      "dev": true,
+      "requires": {
+        "p-locate": "^2.0.0",
+        "path-exists": "^3.0.0"
+      },
+      "dependencies": {
+        "path-exists": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz",
+          "integrity": "sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ==",
+          "dev": true
+        }
+      }
+    },
+    "lodash": {
+      "version": "4.17.21",
+      "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
+      "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==",
+      "dev": true
+    },
+    "lodash.assign": {
+      "version": "4.2.0",
+      "resolved": "https://registry.npmjs.org/lodash.assign/-/lodash.assign-4.2.0.tgz",
+      "integrity": "sha512-hFuH8TY+Yji7Eja3mGiuAxBqLagejScbG8GbG0j6o9vzn0YL14My+ktnqtZgFTosKymC9/44wP6s7xyuLfnClw==",
+      "dev": true
+    },
+    "lodash.camelcase": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz",
+      "integrity": "sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA==",
+      "dev": true
+    },
+    "lodash.merge": {
+      "version": "4.6.2",
+      "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz",
+      "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==",
+      "dev": true
+    },
+    "lodash.truncate": {
+      "version": "4.4.2",
+      "resolved": "https://registry.npmjs.org/lodash.truncate/-/lodash.truncate-4.4.2.tgz",
+      "integrity": "sha512-jttmRe7bRse52OsWIMDLaXxWqRAmtIUccAQ3garviCqJjafXOfNMO0yMfNpdD6zbGaTU0P5Nz7e7gAT6cKmJRw==",
+      "dev": true
+    },
+    "log-symbols": {
+      "version": "4.1.0",
+      "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz",
+      "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==",
+      "dev": true,
+      "requires": {
+        "chalk": "^4.1.0",
+        "is-unicode-supported": "^0.1.0"
+      }
+    },
+    "loupe": {
+      "version": "2.3.6",
+      "resolved": "https://registry.npmjs.org/loupe/-/loupe-2.3.6.tgz",
+      "integrity": "sha512-RaPMZKiMy8/JruncMU5Bt6na1eftNoo++R4Y+N2FrxkDVTrGvcyzFTsaGif4QTeKESheMGegbhw6iUAq+5A8zA==",
+      "dev": true,
+      "requires": {
+        "get-func-name": "^2.0.0"
+      }
+    },
+    "lower-case": {
+      "version": "1.1.4",
+      "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-1.1.4.tgz",
+      "integrity": "sha512-2Fgx1Ycm599x+WGpIYwJOvsjmXFzTSc34IwDWALRA/8AopUKAVPwfJ+h5+f85BCp0PWmmJcWzEpxOpoXycMpdA==",
+      "dev": true
+    },
+    "lower-case-first": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/lower-case-first/-/lower-case-first-1.0.2.tgz",
+      "integrity": "sha512-UuxaYakO7XeONbKrZf5FEgkantPf5DUqDayzP5VXZrtRPdH86s4kN47I8B3TW10S4QKiE3ziHNf3kRN//okHjA==",
+      "dev": true,
+      "requires": {
+        "lower-case": "^1.1.2"
+      }
+    },
+    "lowercase-keys": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-3.0.0.tgz",
+      "integrity": "sha512-ozCC6gdQ+glXOQsveKD0YsDy8DSQFjDTz4zyzEHNV5+JP5D62LmfDZ6o1cycFx9ouG940M5dE8C8CTewdj2YWQ==",
+      "dev": true
+    },
+    "lru_map": {
+      "version": "0.3.3",
+      "resolved": "https://registry.npmjs.org/lru_map/-/lru_map-0.3.3.tgz",
+      "integrity": "sha512-Pn9cox5CsMYngeDbmChANltQl+5pi6XmTrraMSzhPmMBbmgcxmqWry0U3PGapCU1yB4/LqCcom7qhHZiF/jGfQ==",
+      "dev": true
+    },
+    "lru-cache": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz",
+      "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==",
+      "dev": true,
+      "requires": {
+        "yallist": "^4.0.0"
+      }
+    },
+    "make-error": {
+      "version": "1.3.6",
+      "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz",
+      "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==",
+      "dev": true
+    },
+    "markdown-table": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/markdown-table/-/markdown-table-1.1.3.tgz",
+      "integrity": "sha512-1RUZVgQlpJSPWYbFSpmudq5nHY1doEIv89gBtF0s4gW1GF2XorxcA/70M5vq7rLv0a6mhOUccRsqkwhwLCIQ2Q==",
+      "dev": true
+    },
+    "match-all": {
+      "version": "1.2.6",
+      "resolved": "https://registry.npmjs.org/match-all/-/match-all-1.2.6.tgz",
+      "integrity": "sha512-0EESkXiTkWzrQQntBu2uzKvLu6vVkUGz40nGPbSZuegcfE5UuSzNjLaIu76zJWuaT/2I3Z/8M06OlUOZLGwLlQ==",
+      "dev": true
+    },
+    "mcl-wasm": {
+      "version": "0.7.9",
+      "resolved": "https://registry.npmjs.org/mcl-wasm/-/mcl-wasm-0.7.9.tgz",
+      "integrity": "sha512-iJIUcQWA88IJB/5L15GnJVnSQJmf/YaxxV6zRavv83HILHaJQb6y0iFyDMdDO0gN8X37tdxmAOrH/P8B6RB8sQ==",
+      "dev": true
+    },
+    "md5.js": {
+      "version": "1.3.5",
+      "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz",
+      "integrity": "sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==",
+      "dev": true,
+      "requires": {
+        "hash-base": "^3.0.0",
+        "inherits": "^2.0.1",
+        "safe-buffer": "^5.1.2"
+      }
+    },
+    "media-typer": {
+      "version": "0.3.0",
+      "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
+      "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==",
+      "dev": true
+    },
+    "memory-level": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/memory-level/-/memory-level-1.0.0.tgz",
+      "integrity": "sha512-UXzwewuWeHBz5krr7EvehKcmLFNoXxGcvuYhC41tRnkrTbJohtS7kVn9akmgirtRygg+f7Yjsfi8Uu5SGSQ4Og==",
+      "dev": true,
+      "requires": {
+        "abstract-level": "^1.0.0",
+        "functional-red-black-tree": "^1.0.1",
+        "module-error": "^1.0.1"
+      }
+    },
+    "memorystream": {
+      "version": "0.3.1",
+      "resolved": "https://registry.npmjs.org/memorystream/-/memorystream-0.3.1.tgz",
+      "integrity": "sha512-S3UwM3yj5mtUSEfP41UZmt/0SCoVYUcU1rkXv+BQ5Ig8ndL4sPoJNBUJERafdPb5jjHJGuMgytgKvKIf58XNBw==",
+      "dev": true
+    },
+    "merge-descriptors": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
+      "integrity": "sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w==",
+      "dev": true
+    },
+    "merge2": {
+      "version": "1.4.1",
+      "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz",
+      "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==",
+      "dev": true
+    },
+    "methods": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
+      "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==",
+      "dev": true
+    },
+    "micromatch": {
+      "version": "4.0.5",
+      "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz",
+      "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==",
+      "dev": true,
+      "requires": {
+        "braces": "^3.0.2",
+        "picomatch": "^2.3.1"
+      }
+    },
+    "mime": {
+      "version": "1.6.0",
+      "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
+      "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==",
+      "dev": true
+    },
+    "mime-db": {
+      "version": "1.52.0",
+      "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
+      "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
+      "dev": true
+    },
+    "mime-types": {
+      "version": "2.1.35",
+      "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
+      "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
+      "dev": true,
+      "requires": {
+        "mime-db": "1.52.0"
+      }
+    },
+    "mimic-fn": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz",
+      "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==",
+      "dev": true
+    },
+    "mimic-response": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz",
+      "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==",
+      "dev": true
+    },
+    "min-document": {
+      "version": "2.19.0",
+      "resolved": "https://registry.npmjs.org/min-document/-/min-document-2.19.0.tgz",
+      "integrity": "sha512-9Wy1B3m3f66bPPmU5hdA4DR4PB2OfDU/+GS3yAB7IQozE3tqXaVv2zOjgla7MEGSRv95+ILmOuvhLkOK6wJtCQ==",
+      "dev": true,
+      "requires": {
+        "dom-walk": "^0.1.0"
+      }
+    },
+    "minimalistic-assert": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz",
+      "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==",
+      "devOptional": true
+    },
+    "minimalistic-crypto-utils": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz",
+      "integrity": "sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==",
+      "devOptional": true
+    },
+    "minimatch": {
+      "version": "3.1.2",
+      "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
+      "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==",
+      "dev": true,
+      "requires": {
+        "brace-expansion": "^1.1.7"
+      }
+    },
+    "minimist": {
+      "version": "1.2.8",
+      "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz",
+      "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==",
+      "dev": true
+    },
+    "minipass": {
+      "version": "2.9.0",
+      "resolved": "https://registry.npmjs.org/minipass/-/minipass-2.9.0.tgz",
+      "integrity": "sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==",
+      "dev": true,
+      "requires": {
+        "safe-buffer": "^5.1.2",
+        "yallist": "^3.0.0"
+      },
+      "dependencies": {
+        "yallist": {
+          "version": "3.1.1",
+          "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
+          "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==",
+          "dev": true
+        }
+      }
+    },
+    "minizlib": {
+      "version": "1.3.3",
+      "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-1.3.3.tgz",
+      "integrity": "sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==",
+      "dev": true,
+      "requires": {
+        "minipass": "^2.9.0"
+      }
+    },
+    "mkdirp": {
+      "version": "2.1.5",
+      "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-2.1.5.tgz",
+      "integrity": "sha512-jbjfql+shJtAPrFoKxHOXip4xS+kul9W3OzfzzrqueWK2QMGon2bFH2opl6W9EagBThjEz+iysyi/swOoVfB/w==",
+      "dev": true
+    },
+    "mkdirp-promise": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/mkdirp-promise/-/mkdirp-promise-5.0.1.tgz",
+      "integrity": "sha512-Hepn5kb1lJPtVW84RFT40YG1OddBNTOVUZR2bzQUHc+Z03en8/3uX0+060JDhcEzyO08HmipsN9DcnFMxhIL9w==",
+      "dev": true,
+      "requires": {
+        "mkdirp": "*"
+      }
+    },
+    "mnemonist": {
+      "version": "0.38.5",
+      "resolved": "https://registry.npmjs.org/mnemonist/-/mnemonist-0.38.5.tgz",
+      "integrity": "sha512-bZTFT5rrPKtPJxj8KSV0WkPyNxl72vQepqqVUAW2ARUpUSF2qXMB6jZj7hW5/k7C1rtpzqbD/IIbJwLXUjCHeg==",
+      "dev": true,
+      "requires": {
+        "obliterator": "^2.0.0"
+      }
+    },
+    "mocha": {
+      "version": "10.2.0",
+      "resolved": "https://registry.npmjs.org/mocha/-/mocha-10.2.0.tgz",
+      "integrity": "sha512-IDY7fl/BecMwFHzoqF2sg/SHHANeBoMMXFlS9r0OXKDssYE1M5O43wUY/9BVPeIvfH2zmEbBfseqN9gBQZzXkg==",
+      "dev": true,
+      "requires": {
+        "ansi-colors": "4.1.1",
+        "browser-stdout": "1.3.1",
+        "chokidar": "3.5.3",
+        "debug": "4.3.4",
+        "diff": "5.0.0",
+        "escape-string-regexp": "4.0.0",
+        "find-up": "5.0.0",
+        "glob": "7.2.0",
+        "he": "1.2.0",
+        "js-yaml": "4.1.0",
+        "log-symbols": "4.1.0",
+        "minimatch": "5.0.1",
+        "ms": "2.1.3",
+        "nanoid": "3.3.3",
+        "serialize-javascript": "6.0.0",
+        "strip-json-comments": "3.1.1",
+        "supports-color": "8.1.1",
+        "workerpool": "6.2.1",
+        "yargs": "16.2.0",
+        "yargs-parser": "20.2.4",
+        "yargs-unparser": "2.0.0"
+      },
+      "dependencies": {
+        "ansi-colors": {
+          "version": "4.1.1",
+          "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz",
+          "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==",
+          "dev": true
+        },
+        "cliui": {
+          "version": "7.0.4",
+          "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz",
+          "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==",
+          "dev": true,
+          "requires": {
+            "string-width": "^4.2.0",
+            "strip-ansi": "^6.0.0",
+            "wrap-ansi": "^7.0.0"
+          }
+        },
+        "escape-string-regexp": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz",
+          "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==",
+          "dev": true
+        },
+        "find-up": {
+          "version": "5.0.0",
+          "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz",
+          "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==",
+          "dev": true,
+          "requires": {
+            "locate-path": "^6.0.0",
+            "path-exists": "^4.0.0"
+          }
+        },
+        "get-caller-file": {
+          "version": "2.0.5",
+          "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz",
+          "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==",
+          "dev": true
+        },
+        "glob": {
+          "version": "7.2.0",
+          "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz",
+          "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==",
+          "dev": true,
+          "requires": {
+            "fs.realpath": "^1.0.0",
+            "inflight": "^1.0.4",
+            "inherits": "2",
+            "minimatch": "^3.0.4",
+            "once": "^1.3.0",
+            "path-is-absolute": "^1.0.0"
+          },
+          "dependencies": {
+            "minimatch": {
+              "version": "3.1.2",
+              "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
+              "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==",
+              "dev": true,
+              "requires": {
+                "brace-expansion": "^1.1.7"
+              }
+            }
+          }
+        },
+        "locate-path": {
+          "version": "6.0.0",
+          "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz",
+          "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==",
+          "dev": true,
+          "requires": {
+            "p-locate": "^5.0.0"
+          }
+        },
+        "minimatch": {
+          "version": "5.0.1",
+          "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.0.1.tgz",
+          "integrity": "sha512-nLDxIFRyhDblz3qMuq+SoRZED4+miJ/G+tdDrjkkkRnjAsBexeGpgjLEQ0blJy7rHhR2b93rhQY4SvyWu9v03g==",
+          "dev": true,
+          "requires": {
+            "brace-expansion": "^2.0.1"
+          },
+          "dependencies": {
+            "brace-expansion": {
+              "version": "2.0.1",
+              "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz",
+              "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==",
+              "dev": true,
+              "requires": {
+                "balanced-match": "^1.0.0"
+              }
+            }
+          }
+        },
+        "ms": {
+          "version": "2.1.3",
+          "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+          "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
+          "dev": true
+        },
+        "p-limit": {
+          "version": "3.1.0",
+          "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
+          "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==",
+          "dev": true,
+          "requires": {
+            "yocto-queue": "^0.1.0"
+          }
+        },
+        "p-locate": {
+          "version": "5.0.0",
+          "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz",
+          "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==",
+          "dev": true,
+          "requires": {
+            "p-limit": "^3.0.2"
+          }
+        },
+        "path-exists": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
+          "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
+          "dev": true
+        },
+        "supports-color": {
+          "version": "8.1.1",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz",
+          "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==",
+          "dev": true,
+          "requires": {
+            "has-flag": "^4.0.0"
+          }
+        },
+        "wrap-ansi": {
+          "version": "7.0.0",
+          "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz",
+          "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==",
+          "dev": true,
+          "requires": {
+            "ansi-styles": "^4.0.0",
+            "string-width": "^4.1.0",
+            "strip-ansi": "^6.0.0"
+          }
+        },
+        "y18n": {
+          "version": "5.0.8",
+          "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz",
+          "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==",
+          "dev": true
+        },
+        "yargs": {
+          "version": "16.2.0",
+          "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz",
+          "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==",
+          "dev": true,
+          "requires": {
+            "cliui": "^7.0.2",
+            "escalade": "^3.1.1",
+            "get-caller-file": "^2.0.5",
+            "require-directory": "^2.1.1",
+            "string-width": "^4.2.0",
+            "y18n": "^5.0.5",
+            "yargs-parser": "^20.2.2"
+          }
+        },
+        "yargs-parser": {
+          "version": "20.2.4",
+          "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.4.tgz",
+          "integrity": "sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==",
+          "dev": true
+        }
+      }
+    },
+    "mock-fs": {
+      "version": "4.14.0",
+      "resolved": "https://registry.npmjs.org/mock-fs/-/mock-fs-4.14.0.tgz",
+      "integrity": "sha512-qYvlv/exQ4+svI3UOvPUpLDF0OMX5euvUH0Ny4N5QyRyhNdgAgUrVH3iUINSzEPLvx0kbo/Bp28GJKIqvE7URw==",
+      "dev": true
+    },
+    "module-error": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/module-error/-/module-error-1.0.2.tgz",
+      "integrity": "sha512-0yuvsqSCv8LbaOKhnsQ/T5JhyFlCYLPXK3U2sgV10zoKQwzs/MyfuQUOZQ1V/6OCOJsK/TRgNVrPuPDqtdMFtA==",
+      "dev": true
+    },
+    "ms": {
+      "version": "2.1.2",
+      "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
+      "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==",
+      "dev": true
+    },
+    "multibase": {
+      "version": "0.6.1",
+      "resolved": "https://registry.npmjs.org/multibase/-/multibase-0.6.1.tgz",
+      "integrity": "sha512-pFfAwyTjbbQgNc3G7D48JkJxWtoJoBMaR4xQUOuB8RnCgRqaYmWNFeJTTvrJ2w51bjLq2zTby6Rqj9TQ9elSUw==",
+      "dev": true,
+      "requires": {
+        "base-x": "^3.0.8",
+        "buffer": "^5.5.0"
+      }
+    },
+    "multicodec": {
+      "version": "0.5.7",
+      "resolved": "https://registry.npmjs.org/multicodec/-/multicodec-0.5.7.tgz",
+      "integrity": "sha512-PscoRxm3f+88fAtELwUnZxGDkduE2HD9Q6GHUOywQLjOGT/HAdhjLDYNZ1e7VR0s0TP0EwZ16LNUTFpoBGivOA==",
+      "dev": true,
+      "requires": {
+        "varint": "^5.0.0"
+      }
+    },
+    "multihashes": {
+      "version": "0.4.21",
+      "resolved": "https://registry.npmjs.org/multihashes/-/multihashes-0.4.21.tgz",
+      "integrity": "sha512-uVSvmeCWf36pU2nB4/1kzYZjsXD9vofZKpgudqkceYY5g2aZZXJ5r9lxuzoRLl1OAp28XljXsEJ/X/85ZsKmKw==",
+      "dev": true,
+      "requires": {
+        "buffer": "^5.5.0",
+        "multibase": "^0.7.0",
+        "varint": "^5.0.0"
+      },
+      "dependencies": {
+        "multibase": {
+          "version": "0.7.0",
+          "resolved": "https://registry.npmjs.org/multibase/-/multibase-0.7.0.tgz",
+          "integrity": "sha512-TW8q03O0f6PNFTQDvh3xxH03c8CjGaaYrjkl9UQPG6rz53TQzzxJVCIWVjzcbN/Q5Y53Zd0IBQBMVktVgNx4Fg==",
+          "dev": true,
+          "requires": {
+            "base-x": "^3.0.8",
+            "buffer": "^5.5.0"
+          }
+        }
+      }
+    },
+    "murmur-128": {
+      "version": "0.2.1",
+      "resolved": "https://registry.npmjs.org/murmur-128/-/murmur-128-0.2.1.tgz",
+      "integrity": "sha512-WseEgiRkI6aMFBbj8Cg9yBj/y+OdipwVC7zUo3W2W1JAJITwouUOtpqsmGSg67EQmwwSyod7hsVsWY5LsrfQVg==",
+      "dev": true,
+      "requires": {
+        "encode-utf8": "^1.0.2",
+        "fmix": "^0.1.0",
+        "imul": "^1.0.0"
+      }
+    },
+    "nano-base32": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/nano-base32/-/nano-base32-1.0.1.tgz",
+      "integrity": "sha512-sxEtoTqAPdjWVGv71Q17koMFGsOMSiHsIFEvzOM7cNp8BXB4AnEwmDabm5dorusJf/v1z7QxaZYxUorU9RKaAw==",
+      "dev": true
+    },
+    "nano-json-stream-parser": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/nano-json-stream-parser/-/nano-json-stream-parser-0.1.2.tgz",
+      "integrity": "sha512-9MqxMH/BSJC7dnLsEMPyfN5Dvoo49IsPFYMcHw3Bcfc2kN0lpHRBSzlMSVx4HGyJ7s9B31CyBTVehWJoQ8Ctew==",
+      "dev": true
+    },
+    "nanoid": {
+      "version": "3.3.3",
+      "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.3.tgz",
+      "integrity": "sha512-p1sjXuopFs0xg+fPASzQ28agW1oHD7xDsd9Xkf3T15H3c/cifrFHVwrh74PdoklAPi+i7MdRsE47vm2r6JoB+w==",
+      "dev": true
+    },
+    "napi-macros": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/napi-macros/-/napi-macros-2.0.0.tgz",
+      "integrity": "sha512-A0xLykHtARfueITVDernsAWdtIMbOJgKgcluwENp3AlsKN/PloyO10HtmoqnFAQAcxPkgZN7wdfPfEd0zNGxbg==",
+      "dev": true
+    },
+    "natural-compare": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz",
+      "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==",
+      "dev": true
+    },
+    "natural-compare-lite": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/natural-compare-lite/-/natural-compare-lite-1.4.0.tgz",
+      "integrity": "sha512-Tj+HTDSJJKaZnfiuw+iaF9skdPpTo2GtEly5JHnWV/hfv2Qj/9RKsGISQtLh2ox3l5EAGw487hnBee0sIJ6v2g==",
+      "dev": true
+    },
+    "negotiator": {
+      "version": "0.6.3",
+      "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz",
+      "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==",
+      "dev": true
+    },
+    "neo-async": {
+      "version": "2.6.2",
+      "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz",
+      "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==",
+      "dev": true
+    },
+    "next-tick": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/next-tick/-/next-tick-1.1.0.tgz",
+      "integrity": "sha512-CXdUiJembsNjuToQvxayPZF9Vqht7hewsvy2sOWafLvi2awflj9mOC6bHIg50orX8IJvWKY9wYQ/zB2kogPslQ==",
+      "dev": true
+    },
+    "no-case": {
+      "version": "2.3.2",
+      "resolved": "https://registry.npmjs.org/no-case/-/no-case-2.3.2.tgz",
+      "integrity": "sha512-rmTZ9kz+f3rCvK2TD1Ue/oZlns7OGoIWP4fc3llxxRXlOkHKoWPPWJOfFYpITabSow43QJbRIoHQXtt10VldyQ==",
+      "dev": true,
+      "requires": {
+        "lower-case": "^1.1.1"
+      }
+    },
+    "node-addon-api": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-2.0.2.tgz",
+      "integrity": "sha512-Ntyt4AIXyaLIuMHF6IOoTakB3K+RWxwtsHNRxllEoA6vPwP9o4866g6YWDLUdnucilZhmkxiHwHr11gAENw+QA==",
+      "dev": true
+    },
+    "node-emoji": {
+      "version": "1.11.0",
+      "resolved": "https://registry.npmjs.org/node-emoji/-/node-emoji-1.11.0.tgz",
+      "integrity": "sha512-wo2DpQkQp7Sjm2A0cq+sN7EHKO6Sl0ctXeBdFZrL9T9+UywORbufTcTZxom8YqpLQt/FqNMUkOpkZrJVYSKD3A==",
+      "dev": true,
+      "requires": {
+        "lodash": "^4.17.21"
+      }
+    },
+    "node-fetch": {
+      "version": "2.6.7",
+      "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.7.tgz",
+      "integrity": "sha512-ZjMPFEfVx5j+y2yF35Kzx5sF7kDzxuDj6ziH4FFbOp87zKDZNx8yExJIb05OGF4Nlt9IHFIMBkRl41VdvcNdbQ==",
+      "dev": true,
+      "requires": {
+        "whatwg-url": "^5.0.0"
+      }
+    },
+    "node-gyp-build": {
+      "version": "4.6.0",
+      "resolved": "https://registry.npmjs.org/node-gyp-build/-/node-gyp-build-4.6.0.tgz",
+      "integrity": "sha512-NTZVKn9IylLwUzaKjkas1e4u2DLNcV4rdYagA4PWdPwW87Bi7z+BznyKSRwS/761tV/lzCGXplWsiaMjLqP2zQ==",
+      "dev": true
+    },
+    "nofilter": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/nofilter/-/nofilter-3.1.0.tgz",
+      "integrity": "sha512-l2NNj07e9afPnhAhvgVrCD/oy2Ai1yfLpuo3EpiO1jFTsB4sFz6oIfAfSZyQzVpkZQ9xS8ZS5g1jCBgq4Hwo0g==",
+      "dev": true
+    },
+    "nopt": {
+      "version": "3.0.6",
+      "resolved": "https://registry.npmjs.org/nopt/-/nopt-3.0.6.tgz",
+      "integrity": "sha512-4GUt3kSEYmk4ITxzB/b9vaIDfUVWN/Ml1Fwl11IlnIG2iaJ9O6WXZ9SrYM9NLI8OCBieN2Y8SWC2oJV0RQ7qYg==",
+      "dev": true,
+      "requires": {
+        "abbrev": "1"
+      }
+    },
+    "normalize-package-data": {
+      "version": "2.5.0",
+      "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz",
+      "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==",
+      "dev": true,
+      "requires": {
+        "hosted-git-info": "^2.1.4",
+        "resolve": "^1.10.0",
+        "semver": "2 || 3 || 4 || 5",
+        "validate-npm-package-license": "^3.0.1"
+      },
+      "dependencies": {
+        "semver": {
+          "version": "5.7.2",
+          "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz",
+          "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==",
+          "dev": true
+        }
+      }
+    },
+    "normalize-path": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
+      "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
+      "dev": true
+    },
+    "normalize-url": {
+      "version": "6.1.0",
+      "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-6.1.0.tgz",
+      "integrity": "sha512-DlL+XwOy3NxAQ8xuC0okPgK46iuVNAK01YN7RueYBqqFeGsBjV9XmCAzAdgt+667bCl5kPh9EqKKDwnaPG1I7A==",
+      "dev": true
+    },
+    "nth-check": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.1.1.tgz",
+      "integrity": "sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==",
+      "dev": true,
+      "requires": {
+        "boolbase": "^1.0.0"
+      }
+    },
+    "number-is-nan": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz",
+      "integrity": "sha512-4jbtZXNAsfZbAHiiqjLPBiCl16dES1zI4Hpzzxw61Tk+loF+sBDBKx1ICKKKwIqQ7M0mFn1TmkN7euSncWgHiQ==",
+      "dev": true
+    },
+    "number-to-bn": {
+      "version": "1.7.0",
+      "resolved": "https://registry.npmjs.org/number-to-bn/-/number-to-bn-1.7.0.tgz",
+      "integrity": "sha512-wsJ9gfSz1/s4ZsJN01lyonwuxA1tml6X1yBDnfpMglypcBRFZZkus26EdPSlqS5GJfYddVZa22p3VNb3z5m5Ig==",
+      "dev": true,
+      "requires": {
+        "bn.js": "4.11.6",
+        "strip-hex-prefix": "1.0.0"
+      },
+      "dependencies": {
+        "bn.js": {
+          "version": "4.11.6",
+          "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.6.tgz",
+          "integrity": "sha512-XWwnNNFCuuSQ0m3r3C4LE3EiORltHd9M05pq6FOlVeiophzRbMo50Sbz1ehl8K3Z+jw9+vmgnXefY1hz8X+2wA==",
+          "dev": true
+        }
+      }
+    },
+    "oauth-sign": {
+      "version": "0.9.0",
+      "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz",
+      "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==",
+      "dev": true
+    },
+    "object-assign": {
+      "version": "4.1.1",
+      "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
+      "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
+      "dev": true
+    },
+    "object-inspect": {
+      "version": "1.12.3",
+      "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.3.tgz",
+      "integrity": "sha512-geUvdk7c+eizMNUDkRpW1wJwgfOiOeHbxBR/hLXK1aT6zmVSO0jsQcs7fj6MGw89jC/cjGfLcNOrtMYtGqm81g==",
+      "dev": true
+    },
+    "object-keys": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz",
+      "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==",
+      "dev": true
+    },
+    "object.assign": {
+      "version": "4.1.4",
+      "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz",
+      "integrity": "sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "has-symbols": "^1.0.3",
+        "object-keys": "^1.1.1"
+      }
+    },
+    "object.values": {
+      "version": "1.1.6",
+      "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.6.tgz",
+      "integrity": "sha512-FVVTkD1vENCsAcwNs9k6jea2uHC/X0+JcjG8YA60FN5CMaJmG95wT9jek/xX9nornqGRrBkKtzuAu2wuHpKqvw==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "es-abstract": "^1.20.4"
+      }
+    },
+    "obliterator": {
+      "version": "2.0.4",
+      "resolved": "https://registry.npmjs.org/obliterator/-/obliterator-2.0.4.tgz",
+      "integrity": "sha512-lgHwxlxV1qIg1Eap7LgIeoBWIMFibOjbrYPIPJZcI1mmGAI2m3lNYpK12Y+GBdPQ0U1hRwSord7GIaawz962qQ==",
+      "dev": true
+    },
+    "oboe": {
+      "version": "2.1.5",
+      "resolved": "https://registry.npmjs.org/oboe/-/oboe-2.1.5.tgz",
+      "integrity": "sha512-zRFWiF+FoicxEs3jNI/WYUrVEgA7DeET/InK0XQuudGHRg8iIob3cNPrJTKaz4004uaA9Pbe+Dwa8iluhjLZWA==",
+      "dev": true,
+      "requires": {
+        "http-https": "^1.0.0"
+      }
+    },
+    "on-finished": {
+      "version": "2.4.1",
+      "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz",
+      "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==",
+      "dev": true,
+      "requires": {
+        "ee-first": "1.1.1"
+      }
+    },
+    "once": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
+      "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==",
+      "dev": true,
+      "requires": {
+        "wrappy": "1"
+      }
+    },
+    "onetime": {
+      "version": "5.1.2",
+      "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz",
+      "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==",
+      "dev": true,
+      "requires": {
+        "mimic-fn": "^2.1.0"
+      }
+    },
+    "open": {
+      "version": "8.4.2",
+      "resolved": "https://registry.npmjs.org/open/-/open-8.4.2.tgz",
+      "integrity": "sha512-7x81NCL719oNbsq/3mh+hVrAWmFuEYUqrq/Iw3kUzH8ReypT9QQ0BLoJS7/G9k6N81XjW4qHWtjWwe/9eLy1EQ==",
+      "dev": true,
+      "requires": {
+        "define-lazy-prop": "^2.0.0",
+        "is-docker": "^2.1.1",
+        "is-wsl": "^2.2.0"
+      }
+    },
+    "opencollective-postinstall": {
+      "version": "2.0.3",
+      "resolved": "https://registry.npmjs.org/opencollective-postinstall/-/opencollective-postinstall-2.0.3.tgz",
+      "integrity": "sha512-8AV/sCtuzUeTo8gQK5qDZzARrulB3egtLzFgteqB2tcT4Mw7B8Kt7JcDHmltjz6FOAHsvTevk70gZEbhM4ZS9Q==",
+      "dev": true
+    },
+    "optionator": {
+      "version": "0.9.1",
+      "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.1.tgz",
+      "integrity": "sha512-74RlY5FCnhq4jRxVUPKDaRwrVNXMqsGsiW6AJw4XK8hmtm10wC0ypZBLw5IIp85NZMr91+qd1RvvENwg7jjRFw==",
+      "dev": true,
+      "requires": {
+        "deep-is": "^0.1.3",
+        "fast-levenshtein": "^2.0.6",
+        "levn": "^0.4.1",
+        "prelude-ls": "^1.2.1",
+        "type-check": "^0.4.0",
+        "word-wrap": "^1.2.3"
+      }
+    },
+    "ora": {
+      "version": "5.4.1",
+      "resolved": "https://registry.npmjs.org/ora/-/ora-5.4.1.tgz",
+      "integrity": "sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ==",
+      "dev": true,
+      "requires": {
+        "bl": "^4.1.0",
+        "chalk": "^4.1.0",
+        "cli-cursor": "^3.1.0",
+        "cli-spinners": "^2.5.0",
+        "is-interactive": "^1.0.0",
+        "is-unicode-supported": "^0.1.0",
+        "log-symbols": "^4.1.0",
+        "strip-ansi": "^6.0.0",
+        "wcwidth": "^1.0.1"
+      }
+    },
+    "ordinal": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/ordinal/-/ordinal-1.0.3.tgz",
+      "integrity": "sha512-cMddMgb2QElm8G7vdaa02jhUNbTSrhsgAGUz1OokD83uJTwSUn+nKoNoKVVaRa08yF6sgfO7Maou1+bgLd9rdQ==",
+      "dev": true
+    },
+    "os-locale": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/os-locale/-/os-locale-1.4.0.tgz",
+      "integrity": "sha512-PRT7ZORmwu2MEFt4/fv3Q+mEfN4zetKxufQrkShY2oGvUms9r8otu5HfdyIFHkYXjO7laNsoVGmM2MANfuTA8g==",
+      "dev": true,
+      "requires": {
+        "lcid": "^1.0.0"
+      }
+    },
+    "os-tmpdir": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz",
+      "integrity": "sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==",
+      "dev": true
+    },
+    "p-cancelable": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-3.0.0.tgz",
+      "integrity": "sha512-mlVgR3PGuzlo0MmTdk4cXqXWlwQDLnONTAg6sm62XkMJEiRxN3GL3SffkYvqwonbkJBcrI7Uvv5Zh9yjvn2iUw==",
+      "dev": true
+    },
+    "p-limit": {
+      "version": "1.3.0",
+      "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz",
+      "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==",
+      "dev": true,
+      "requires": {
+        "p-try": "^1.0.0"
+      }
+    },
+    "p-locate": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz",
+      "integrity": "sha512-nQja7m7gSKuewoVRen45CtVfODR3crN3goVQ0DDZ9N3yHxgpkuBhZqsaiotSQRrADUrne346peY7kT3TSACykg==",
+      "dev": true,
+      "requires": {
+        "p-limit": "^1.1.0"
+      }
+    },
+    "p-map": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/p-map/-/p-map-4.0.0.tgz",
+      "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==",
+      "dev": true,
+      "requires": {
+        "aggregate-error": "^3.0.0"
+      }
+    },
+    "p-try": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz",
+      "integrity": "sha512-U1etNYuMJoIz3ZXSrrySFjsXQTWOx2/jdi86L+2pRvph/qMKL6sbcCYdH23fqsbm8TH2Gn0OybpT4eSFlCVHww==",
+      "dev": true
+    },
+    "pako": {
+      "version": "1.0.11",
+      "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz",
+      "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==",
+      "dev": true
+    },
+    "param-case": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/param-case/-/param-case-2.1.1.tgz",
+      "integrity": "sha512-eQE845L6ot89sk2N8liD8HAuH4ca6Vvr7VWAWwt7+kvvG5aBcPmmphQ68JsEG2qa9n1TykS2DLeMt363AAH8/w==",
+      "dev": true,
+      "requires": {
+        "no-case": "^2.2.0"
+      }
+    },
+    "parent-module": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz",
+      "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==",
+      "dev": true,
+      "requires": {
+        "callsites": "^3.0.0"
+      }
+    },
+    "parse-cache-control": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/parse-cache-control/-/parse-cache-control-1.0.1.tgz",
+      "integrity": "sha512-60zvsJReQPX5/QP0Kzfd/VrpjScIQ7SHBW6bFCYfEP+fp0Eppr1SHhIO5nd1PjZtvclzSzES9D/p5nFJurwfWg==",
+      "dev": true
+    },
+    "parse-headers": {
+      "version": "2.0.5",
+      "resolved": "https://registry.npmjs.org/parse-headers/-/parse-headers-2.0.5.tgz",
+      "integrity": "sha512-ft3iAoLOB/MlwbNXgzy43SWGP6sQki2jQvAyBg/zDFAgr9bfNWZIUj42Kw2eJIl8kEi4PbgE6U1Zau/HwI75HA==",
+      "dev": true
+    },
+    "parse-json": {
+      "version": "2.2.0",
+      "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz",
+      "integrity": "sha512-QR/GGaKCkhwk1ePQNYDRKYZ3mwU9ypsKhB0XyFnLQdomyEqk3e8wpW3V5Jp88zbxK4n5ST1nqo+g9juTpownhQ==",
+      "dev": true,
+      "requires": {
+        "error-ex": "^1.2.0"
+      }
+    },
+    "parse5": {
+      "version": "7.1.2",
+      "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.1.2.tgz",
+      "integrity": "sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==",
+      "dev": true,
+      "requires": {
+        "entities": "^4.4.0"
+      }
+    },
+    "parse5-htmlparser2-tree-adapter": {
+      "version": "7.0.0",
+      "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.0.0.tgz",
+      "integrity": "sha512-B77tOZrqqfUfnVcOrUvfdLbz4pu4RopLD/4vmu3HUPswwTA8OH0EMW9BlWR2B0RCoiZRAHEUu7IxeP1Pd1UU+g==",
+      "dev": true,
+      "requires": {
+        "domhandler": "^5.0.2",
+        "parse5": "^7.0.0"
+      }
+    },
+    "parseurl": {
+      "version": "1.3.3",
+      "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
+      "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==",
+      "dev": true
+    },
+    "pascal-case": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/pascal-case/-/pascal-case-2.0.1.tgz",
+      "integrity": "sha512-qjS4s8rBOJa2Xm0jmxXiyh1+OFf6ekCWOvUaRgAQSktzlTbMotS0nmG9gyYAybCWBcuP4fsBeRCKNwGBnMe2OQ==",
+      "dev": true,
+      "requires": {
+        "camel-case": "^3.0.0",
+        "upper-case-first": "^1.1.0"
+      }
+    },
+    "path-case": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/path-case/-/path-case-2.1.1.tgz",
+      "integrity": "sha512-Ou0N05MioItesaLr9q8TtHVWmJ6fxWdqKB2RohFmNWVyJ+2zeKIeDNWAN6B/Pe7wpzWChhZX6nONYmOnMeJQ/Q==",
+      "dev": true,
+      "requires": {
+        "no-case": "^2.2.0"
+      }
+    },
+    "path-exists": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-2.1.0.tgz",
+      "integrity": "sha512-yTltuKuhtNeFJKa1PiRzfLAU5182q1y4Eb4XCJ3PBqyzEDkAZRzBrKKBct682ls9reBVHf9udYLN5Nd+K1B9BQ==",
+      "dev": true,
+      "requires": {
+        "pinkie-promise": "^2.0.0"
+      }
+    },
+    "path-is-absolute": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
+      "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==",
+      "dev": true
+    },
+    "path-key": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz",
+      "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==",
+      "dev": true
+    },
+    "path-parse": {
+      "version": "1.0.7",
+      "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz",
+      "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==",
+      "dev": true
+    },
+    "path-to-regexp": {
+      "version": "0.1.7",
+      "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
+      "integrity": "sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ==",
+      "dev": true
+    },
+    "path-type": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/path-type/-/path-type-1.1.0.tgz",
+      "integrity": "sha512-S4eENJz1pkiQn9Znv33Q+deTOKmbl+jj1Fl+qiP/vYezj+S8x+J3Uo0ISrx/QoEvIlOaDWJhPaRd1flJ9HXZqg==",
+      "dev": true,
+      "requires": {
+        "graceful-fs": "^4.1.2",
+        "pify": "^2.0.0",
+        "pinkie-promise": "^2.0.0"
+      }
+    },
+    "pathval": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/pathval/-/pathval-1.1.1.tgz",
+      "integrity": "sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ==",
+      "dev": true
+    },
+    "pbkdf2": {
+      "version": "3.1.2",
+      "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.1.2.tgz",
+      "integrity": "sha512-iuh7L6jA7JEGu2WxDwtQP1ddOpaJNC4KlDEFfdQajSGgGPNi4OyDc2R7QnbY2bR9QjBVGwgvTdNJZoE7RaxUMA==",
+      "dev": true,
+      "requires": {
+        "create-hash": "^1.1.2",
+        "create-hmac": "^1.1.4",
+        "ripemd160": "^2.0.1",
+        "safe-buffer": "^5.0.1",
+        "sha.js": "^2.4.8"
+      }
+    },
+    "performance-now": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz",
+      "integrity": "sha512-7EAHlyLHI56VEIdK57uwHdHKIaAGbnXPiw0yWbarQZOKaKpvUIgW0jWRVLiatnM+XXlSwsanIBH/hzGMJulMow==",
+      "dev": true
+    },
+    "picomatch": {
+      "version": "2.3.1",
+      "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz",
+      "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==",
+      "dev": true
+    },
+    "pify": {
+      "version": "2.3.0",
+      "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz",
+      "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==",
+      "dev": true
+    },
+    "pinkie": {
+      "version": "2.0.4",
+      "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz",
+      "integrity": "sha512-MnUuEycAemtSaeFSjXKW/aroV7akBbY+Sv+RkyqFjgAe73F+MR0TBWKBRDkmfWq/HiFmdavfZ1G7h4SPZXaCSg==",
+      "dev": true
+    },
+    "pinkie-promise": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz",
+      "integrity": "sha512-0Gni6D4UcLTbv9c57DfxDGdr41XfgUjqWZu492f0cIGr16zDU06BWP/RAEvOuo7CQ0CNjHaLlM59YJJFm3NWlw==",
+      "dev": true,
+      "requires": {
+        "pinkie": "^2.0.0"
+      }
+    },
+    "pkg-dir": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-5.0.0.tgz",
+      "integrity": "sha512-NPE8TDbzl/3YQYY7CSS228s3g2ollTFnc+Qi3tqmqJp9Vg2ovUpixcJEo2HJScN2Ez+kEaal6y70c0ehqJBJeA==",
+      "dev": true,
+      "requires": {
+        "find-up": "^5.0.0"
+      },
+      "dependencies": {
+        "find-up": {
+          "version": "5.0.0",
+          "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz",
+          "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==",
+          "dev": true,
+          "requires": {
+            "locate-path": "^6.0.0",
+            "path-exists": "^4.0.0"
+          }
+        },
+        "locate-path": {
+          "version": "6.0.0",
+          "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz",
+          "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==",
+          "dev": true,
+          "requires": {
+            "p-locate": "^5.0.0"
+          }
+        },
+        "p-limit": {
+          "version": "3.1.0",
+          "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
+          "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==",
+          "dev": true,
+          "requires": {
+            "yocto-queue": "^0.1.0"
+          }
+        },
+        "p-locate": {
+          "version": "5.0.0",
+          "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz",
+          "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==",
+          "dev": true,
+          "requires": {
+            "p-limit": "^3.0.2"
+          }
+        },
+        "path-exists": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
+          "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
+          "dev": true
+        }
+      }
+    },
+    "please-upgrade-node": {
+      "version": "3.2.0",
+      "resolved": "https://registry.npmjs.org/please-upgrade-node/-/please-upgrade-node-3.2.0.tgz",
+      "integrity": "sha512-gQR3WpIgNIKwBMVLkpMUeR3e1/E1y42bqDQZfql+kDeXd8COYfM8PQA4X6y7a8u9Ua9FHmsrrmirW2vHs45hWg==",
+      "dev": true,
+      "requires": {
+        "semver-compare": "^1.0.0"
+      }
+    },
+    "pluralize": {
+      "version": "8.0.0",
+      "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-8.0.0.tgz",
+      "integrity": "sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA==",
+      "dev": true
+    },
+    "prelude-ls": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz",
+      "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==",
+      "dev": true
+    },
+    "prettier": {
+      "version": "2.8.8",
+      "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.8.8.tgz",
+      "integrity": "sha512-tdN8qQGvNjw4CHbY+XXk0JgCXn9QiF21a55rBe5LJAU+kDyC4WQn4+awm2Xfk2lQMk5fKup9XgzTZtGkjBdP9Q==",
+      "dev": true
+    },
+    "prettier-linter-helpers": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.0.tgz",
+      "integrity": "sha512-GbK2cP9nraSSUF9N2XwUwqfzlAFlMNYYl+ShE/V+H8a9uNl/oUqB1w2EL54Jh0OlyRSd8RfWYJ3coVS4TROP2w==",
+      "dev": true,
+      "requires": {
+        "fast-diff": "^1.1.2"
+      }
+    },
+    "prettier-plugin-solidity": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/prettier-plugin-solidity/-/prettier-plugin-solidity-1.1.3.tgz",
+      "integrity": "sha512-fQ9yucPi2sBbA2U2Xjh6m4isUTJ7S7QLc/XDDsktqqxYfTwdYKJ0EnnywXHwCGAaYbQNK+HIYPL1OemxuMsgeg==",
+      "dev": true,
+      "requires": {
+        "@solidity-parser/parser": "^0.16.0",
+        "semver": "^7.3.8",
+        "solidity-comments-extractor": "^0.0.7"
+      },
+      "dependencies": {
+        "@solidity-parser/parser": {
+          "version": "0.16.0",
+          "resolved": "https://registry.npmjs.org/@solidity-parser/parser/-/parser-0.16.0.tgz",
+          "integrity": "sha512-ESipEcHyRHg4Np4SqBCfcXwyxxna1DgFVz69bgpLV8vzl/NP1DtcKsJ4dJZXWQhY/Z4J2LeKBiOkOVZn9ct33Q==",
+          "dev": true,
+          "requires": {
+            "antlr4ts": "^0.5.0-alpha.4"
+          }
+        },
+        "semver": {
+          "version": "7.5.4",
+          "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz",
+          "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==",
+          "dev": true,
+          "requires": {
+            "lru-cache": "^6.0.0"
+          }
+        }
+      }
+    },
+    "process": {
+      "version": "0.11.10",
+      "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz",
+      "integrity": "sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A==",
+      "dev": true
+    },
+    "process-nextick-args": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
+      "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==",
+      "dev": true
+    },
+    "promise": {
+      "version": "8.3.0",
+      "resolved": "https://registry.npmjs.org/promise/-/promise-8.3.0.tgz",
+      "integrity": "sha512-rZPNPKTOYVNEEKFaq1HqTgOwZD+4/YHS5ukLzQCypkj+OkYx7iv0mA91lJlpPPZ8vMau3IIGj5Qlwrx+8iiSmg==",
+      "dev": true,
+      "requires": {
+        "asap": "~2.0.6"
+      }
+    },
+    "prompts": {
+      "version": "2.4.2",
+      "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz",
+      "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==",
+      "dev": true,
+      "requires": {
+        "kleur": "^3.0.3",
+        "sisteransi": "^1.0.5"
+      }
+    },
+    "proxy-addr": {
+      "version": "2.0.7",
+      "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz",
+      "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==",
+      "dev": true,
+      "requires": {
+        "forwarded": "0.2.0",
+        "ipaddr.js": "1.9.1"
+      }
+    },
+    "proxy-from-env": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
+      "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==",
+      "dev": true
+    },
+    "psl": {
+      "version": "1.9.0",
+      "resolved": "https://registry.npmjs.org/psl/-/psl-1.9.0.tgz",
+      "integrity": "sha512-E/ZsdU4HLs/68gYzgGTkMicWTLPdAftJLfJFlLUAAKZGkStNU72sZjT66SnMDVOfOWY/YAoiD7Jxa9iHvngcag==",
+      "dev": true
+    },
+    "pump": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz",
+      "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==",
+      "dev": true,
+      "requires": {
+        "end-of-stream": "^1.1.0",
+        "once": "^1.3.1"
+      }
+    },
+    "punycode": {
+      "version": "2.3.0",
+      "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.0.tgz",
+      "integrity": "sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA==",
+      "dev": true
+    },
+    "pure-rand": {
+      "version": "5.0.5",
+      "resolved": "https://registry.npmjs.org/pure-rand/-/pure-rand-5.0.5.tgz",
+      "integrity": "sha512-BwQpbqxSCBJVpamI6ydzcKqyFmnd5msMWUGvzXLm1aXvusbbgkbOto/EUPM00hjveJEaJtdbhUjKSzWRhQVkaw==",
+      "dev": true
+    },
+    "qs": {
+      "version": "6.11.0",
+      "resolved": "https://registry.npmjs.org/qs/-/qs-6.11.0.tgz",
+      "integrity": "sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q==",
+      "dev": true,
+      "requires": {
+        "side-channel": "^1.0.4"
+      }
+    },
+    "query-string": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/query-string/-/query-string-5.1.1.tgz",
+      "integrity": "sha512-gjWOsm2SoGlgLEdAGt7a6slVOk9mGiXmPFMqrEhLQ68rhQuBnpfs3+EmlvqKyxnCo9/PPlF+9MtY02S1aFg+Jw==",
+      "dev": true,
+      "requires": {
+        "decode-uri-component": "^0.2.0",
+        "object-assign": "^4.1.0",
+        "strict-uri-encode": "^1.0.0"
+      }
+    },
+    "queue-microtask": {
+      "version": "1.2.3",
+      "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz",
+      "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==",
+      "dev": true
+    },
+    "quick-lru": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/quick-lru/-/quick-lru-5.1.1.tgz",
+      "integrity": "sha512-WuyALRjWPDGtt/wzJiadO5AXY+8hZ80hVpe6MyivgraREW751X3SbhRvG3eLKOYN+8VEvqLcf3wdnt44Z4S4SA==",
+      "dev": true
+    },
+    "randombytes": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz",
+      "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==",
+      "dev": true,
+      "requires": {
+        "safe-buffer": "^5.1.0"
+      }
+    },
+    "range-parser": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
+      "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==",
+      "dev": true
+    },
+    "raw-body": {
+      "version": "2.5.2",
+      "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.2.tgz",
+      "integrity": "sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==",
+      "dev": true,
+      "requires": {
+        "bytes": "3.1.2",
+        "http-errors": "2.0.0",
+        "iconv-lite": "0.4.24",
+        "unpipe": "1.0.0"
+      }
+    },
+    "read-pkg": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-1.1.0.tgz",
+      "integrity": "sha512-7BGwRHqt4s/uVbuyoeejRn4YmFnYZiFl4AuaeXHlgZf3sONF0SOGlxs2Pw8g6hCKupo08RafIO5YXFNOKTfwsQ==",
+      "dev": true,
+      "requires": {
+        "load-json-file": "^1.0.0",
+        "normalize-package-data": "^2.3.2",
+        "path-type": "^1.0.0"
+      }
+    },
+    "read-pkg-up": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-1.0.1.tgz",
+      "integrity": "sha512-WD9MTlNtI55IwYUS27iHh9tK3YoIVhxis8yKhLpTqWtml739uXc9NWTpxoHkfZf3+DkCCsXox94/VWZniuZm6A==",
+      "dev": true,
+      "requires": {
+        "find-up": "^1.0.0",
+        "read-pkg": "^1.0.0"
+      }
+    },
+    "readable-stream": {
+      "version": "3.6.1",
+      "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.1.tgz",
+      "integrity": "sha512-+rQmrWMYGA90yenhTYsLWAsLsqVC8osOw6PKE1HDYiO0gdPeKe/xDHNzIAIn4C91YQ6oenEhfYqqc1883qHbjQ==",
+      "dev": true,
+      "requires": {
+        "inherits": "^2.0.3",
+        "string_decoder": "^1.1.1",
+        "util-deprecate": "^1.0.1"
+      }
+    },
+    "readdirp": {
+      "version": "3.6.0",
+      "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz",
+      "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==",
+      "dev": true,
+      "requires": {
+        "picomatch": "^2.2.1"
+      }
+    },
+    "rechoir": {
+      "version": "0.6.2",
+      "resolved": "https://registry.npmjs.org/rechoir/-/rechoir-0.6.2.tgz",
+      "integrity": "sha512-HFM8rkZ+i3zrV+4LQjwQ0W+ez98pApMGM3HUrN04j3CqzPOzl9nmP15Y8YXNm8QHGv/eacOVEjqhmWpkRV0NAw==",
+      "dev": true,
+      "requires": {
+        "resolve": "^1.1.6"
+      }
+    },
+    "recursive-readdir": {
+      "version": "2.2.3",
+      "resolved": "https://registry.npmjs.org/recursive-readdir/-/recursive-readdir-2.2.3.tgz",
+      "integrity": "sha512-8HrF5ZsXk5FAH9dgsx3BlUer73nIhuj+9OrQwEbLTPOBzGkL1lsFCR01am+v+0m2Cmbs1nP12hLDl5FA7EszKA==",
+      "dev": true,
+      "requires": {
+        "minimatch": "^3.0.5"
+      }
+    },
+    "reduce-flatten": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/reduce-flatten/-/reduce-flatten-2.0.0.tgz",
+      "integrity": "sha512-EJ4UNY/U1t2P/2k6oqotuX2Cc3T6nxJwsM0N0asT7dhrtH1ltUxDn4NalSYmPE2rCkVpcf/X6R0wDwcFpzhd4w==",
+      "dev": true
+    },
+    "regenerator-runtime": {
+      "version": "0.13.11",
+      "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.11.tgz",
+      "integrity": "sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg==",
+      "dev": true
+    },
+    "regexp.prototype.flags": {
+      "version": "1.4.3",
+      "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.4.3.tgz",
+      "integrity": "sha512-fjggEOO3slI6Wvgjwflkc4NFRCTZAu5CnNfBd5qOMYhWdn67nJBBu34/TkD++eeFmd8C9r9jfXJ27+nSiRkSUA==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.3",
+        "functions-have-names": "^1.2.2"
+      }
+    },
+    "regexpp": {
+      "version": "3.2.0",
+      "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.2.0.tgz",
+      "integrity": "sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==",
+      "dev": true
+    },
+    "req-cwd": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/req-cwd/-/req-cwd-2.0.0.tgz",
+      "integrity": "sha512-ueoIoLo1OfB6b05COxAA9UpeoscNpYyM+BqYlA7H6LVF4hKGPXQQSSaD2YmvDVJMkk4UDpAHIeU1zG53IqjvlQ==",
+      "dev": true,
+      "requires": {
+        "req-from": "^2.0.0"
+      }
+    },
+    "req-from": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/req-from/-/req-from-2.0.0.tgz",
+      "integrity": "sha512-LzTfEVDVQHBRfjOUMgNBA+V6DWsSnoeKzf42J7l0xa/B4jyPOuuF5MlNSmomLNGemWTnV2TIdjSSLnEn95fOQA==",
+      "dev": true,
+      "requires": {
+        "resolve-from": "^3.0.0"
+      },
+      "dependencies": {
+        "resolve-from": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz",
+          "integrity": "sha512-GnlH6vxLymXJNMBo7XP1fJIzBFbdYt49CuTwmB/6N53t+kMPRMFKz783LlQ4tv28XoQfMWinAJX6WCGf2IlaIw==",
+          "dev": true
+        }
+      }
+    },
+    "request": {
+      "version": "2.88.2",
+      "resolved": "https://registry.npmjs.org/request/-/request-2.88.2.tgz",
+      "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==",
+      "dev": true,
+      "requires": {
+        "aws-sign2": "~0.7.0",
+        "aws4": "^1.8.0",
+        "caseless": "~0.12.0",
+        "combined-stream": "~1.0.6",
+        "extend": "~3.0.2",
+        "forever-agent": "~0.6.1",
+        "form-data": "~2.3.2",
+        "har-validator": "~5.1.3",
+        "http-signature": "~1.2.0",
+        "is-typedarray": "~1.0.0",
+        "isstream": "~0.1.2",
+        "json-stringify-safe": "~5.0.1",
+        "mime-types": "~2.1.19",
+        "oauth-sign": "~0.9.0",
+        "performance-now": "^2.1.0",
+        "qs": "~6.5.2",
+        "safe-buffer": "^5.1.2",
+        "tough-cookie": "~2.5.0",
+        "tunnel-agent": "^0.6.0",
+        "uuid": "^3.3.2"
+      },
+      "dependencies": {
+        "qs": {
+          "version": "6.5.3",
+          "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.3.tgz",
+          "integrity": "sha512-qxXIEh4pCGfHICj1mAJQ2/2XVZkjCDTcEgfoSQxc/fYivUZxTkk7L3bDBJSoNrEzXI17oUO5Dp07ktqE5KzczA==",
+          "dev": true
+        },
+        "uuid": {
+          "version": "3.4.0",
+          "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz",
+          "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==",
+          "dev": true
+        }
+      }
+    },
+    "require-directory": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz",
+      "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==",
+      "dev": true
+    },
+    "require-from-string": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz",
+      "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==",
+      "dev": true
+    },
+    "require-main-filename": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-1.0.1.tgz",
+      "integrity": "sha512-IqSUtOVP4ksd1C/ej5zeEh/BIP2ajqpn8c5x+q99gvcIG/Qf0cud5raVnE/Dwd0ua9TXYDoDc0RE5hBSdz22Ug==",
+      "dev": true
+    },
+    "resolve": {
+      "version": "1.22.1",
+      "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.1.tgz",
+      "integrity": "sha512-nBpuuYuY5jFsli/JIs1oldw6fOQCBioohqWZg/2hiaOybXOft4lonv85uDOKXdf8rhyK159cxU5cDcK/NKk8zw==",
+      "dev": true,
+      "requires": {
+        "is-core-module": "^2.9.0",
+        "path-parse": "^1.0.7",
+        "supports-preserve-symlinks-flag": "^1.0.0"
+      }
+    },
+    "resolve-alpn": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/resolve-alpn/-/resolve-alpn-1.2.1.tgz",
+      "integrity": "sha512-0a1F4l73/ZFZOakJnQ3FvkJ2+gSTQWz/r2KE5OdDY0TxPm5h4GkqkWWfM47T7HsbnOtcJVEF4epCVy6u7Q3K+g==",
+      "dev": true
+    },
+    "resolve-from": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz",
+      "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==",
+      "dev": true
+    },
+    "responselike": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/responselike/-/responselike-2.0.1.tgz",
+      "integrity": "sha512-4gl03wn3hj1HP3yzgdI7d3lCkF95F21Pz4BPGvKHinyQzALR5CapwC8yIi0Rh58DEMQ/SguC03wFj2k0M/mHhw==",
+      "dev": true,
+      "requires": {
+        "lowercase-keys": "^2.0.0"
+      },
+      "dependencies": {
+        "lowercase-keys": {
+          "version": "2.0.0",
+          "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz",
+          "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==",
+          "dev": true
+        }
+      }
+    },
+    "restore-cursor": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz",
+      "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==",
+      "dev": true,
+      "requires": {
+        "onetime": "^5.1.0",
+        "signal-exit": "^3.0.2"
+      }
+    },
+    "reusify": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz",
+      "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==",
+      "dev": true
+    },
+    "rimraf": {
+      "version": "3.0.2",
+      "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz",
+      "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==",
+      "dev": true,
+      "requires": {
+        "glob": "^7.1.3"
+      }
+    },
+    "ripemd160": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/ripemd160/-/ripemd160-2.0.2.tgz",
+      "integrity": "sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==",
+      "dev": true,
+      "requires": {
+        "hash-base": "^3.0.0",
+        "inherits": "^2.0.1"
+      }
+    },
+    "ripemd160-min": {
+      "version": "0.0.6",
+      "resolved": "https://registry.npmjs.org/ripemd160-min/-/ripemd160-min-0.0.6.tgz",
+      "integrity": "sha512-+GcJgQivhs6S9qvLogusiTcS9kQUfgR75whKuy5jIhuiOfQuJ8fjqxV6EGD5duH1Y/FawFUMtMhyeq3Fbnib8A==",
+      "dev": true
+    },
+    "rlp": {
+      "version": "2.2.7",
+      "resolved": "https://registry.npmjs.org/rlp/-/rlp-2.2.7.tgz",
+      "integrity": "sha512-d5gdPmgQ0Z+AklL2NVXr/IoSjNZFfTVvQWzL/AM2AOcSzYP2xjlb0AC8YyCLc41MSNf6P6QVtjgPdmVtzb+4lQ==",
+      "dev": true,
+      "requires": {
+        "bn.js": "^5.2.0"
+      }
+    },
+    "run-parallel": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz",
+      "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==",
+      "dev": true,
+      "requires": {
+        "queue-microtask": "^1.2.2"
+      }
+    },
+    "run-parallel-limit": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/run-parallel-limit/-/run-parallel-limit-1.1.0.tgz",
+      "integrity": "sha512-jJA7irRNM91jaKc3Hcl1npHsFLOXOoTkPCUL1JEa1R82O2miplXXRaGdjW/KM/98YQWDhJLiSs793CnXfblJUw==",
+      "dev": true,
+      "requires": {
+        "queue-microtask": "^1.2.2"
+      }
+    },
+    "rustbn.js": {
+      "version": "0.2.0",
+      "resolved": "https://registry.npmjs.org/rustbn.js/-/rustbn.js-0.2.0.tgz",
+      "integrity": "sha512-4VlvkRUuCJvr2J6Y0ImW7NvTCriMi7ErOAqWk1y69vAdoNIzCF3yPmgeNzx+RQTLEDFq5sHfscn1MwHxP9hNfA==",
+      "dev": true
+    },
+    "safe-buffer": {
+      "version": "5.2.1",
+      "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
+      "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
+      "dev": true
+    },
+    "safe-regex-test": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.0.tgz",
+      "integrity": "sha512-JBUUzyOgEwXQY1NuPtvcj/qcBDbDmEvWufhlnXZIm75DEHp+afM1r1ujJpJsV/gSM4t59tpDyPi1sd6ZaPFfsA==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "get-intrinsic": "^1.1.3",
+        "is-regex": "^1.1.4"
+      }
+    },
+    "safer-buffer": {
+      "version": "2.1.2",
+      "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
+      "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==",
+      "dev": true
+    },
+    "sc-istanbul": {
+      "version": "0.4.6",
+      "resolved": "https://registry.npmjs.org/sc-istanbul/-/sc-istanbul-0.4.6.tgz",
+      "integrity": "sha512-qJFF/8tW/zJsbyfh/iT/ZM5QNHE3CXxtLJbZsL+CzdJLBsPD7SedJZoUA4d8iAcN2IoMp/Dx80shOOd2x96X/g==",
+      "dev": true,
+      "requires": {
+        "abbrev": "1.0.x",
+        "async": "1.x",
+        "escodegen": "1.8.x",
+        "esprima": "2.7.x",
+        "glob": "^5.0.15",
+        "handlebars": "^4.0.1",
+        "js-yaml": "3.x",
+        "mkdirp": "0.5.x",
+        "nopt": "3.x",
+        "once": "1.x",
+        "resolve": "1.1.x",
+        "supports-color": "^3.1.0",
+        "which": "^1.1.1",
+        "wordwrap": "^1.0.0"
+      },
+      "dependencies": {
+        "argparse": {
+          "version": "1.0.10",
+          "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz",
+          "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==",
+          "dev": true,
+          "requires": {
+            "sprintf-js": "~1.0.2"
+          }
+        },
+        "async": {
+          "version": "1.5.2",
+          "resolved": "https://registry.npmjs.org/async/-/async-1.5.2.tgz",
+          "integrity": "sha512-nSVgobk4rv61R9PUSDtYt7mPVB2olxNR5RWJcAsH676/ef11bUZwvu7+RGYrYauVdDPcO519v68wRhXQtxsV9w==",
+          "dev": true
+        },
+        "esprima": {
+          "version": "2.7.3",
+          "resolved": "https://registry.npmjs.org/esprima/-/esprima-2.7.3.tgz",
+          "integrity": "sha512-OarPfz0lFCiW4/AV2Oy1Rp9qu0iusTKqykwTspGCZtPxmF81JR4MmIebvF1F9+UOKth2ZubLQ4XGGaU+hSn99A==",
+          "dev": true
+        },
+        "glob": {
+          "version": "5.0.15",
+          "resolved": "https://registry.npmjs.org/glob/-/glob-5.0.15.tgz",
+          "integrity": "sha512-c9IPMazfRITpmAAKi22dK1VKxGDX9ehhqfABDriL/lzO92xcUKEJPQHrVA/2YHSNFB4iFlykVmWvwo48nr3OxA==",
+          "dev": true,
+          "requires": {
+            "inflight": "^1.0.4",
+            "inherits": "2",
+            "minimatch": "2 || 3",
+            "once": "^1.3.0",
+            "path-is-absolute": "^1.0.0"
+          }
+        },
+        "has-flag": {
+          "version": "1.0.0",
+          "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-1.0.0.tgz",
+          "integrity": "sha512-DyYHfIYwAJmjAjSSPKANxI8bFY9YtFrgkAfinBojQ8YJTOuOuav64tMUJv584SES4xl74PmuaevIyaLESHdTAA==",
+          "dev": true
+        },
+        "js-yaml": {
+          "version": "3.14.1",
+          "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz",
+          "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==",
+          "dev": true,
+          "requires": {
+            "argparse": "^1.0.7",
+            "esprima": "^4.0.0"
+          },
+          "dependencies": {
+            "esprima": {
+              "version": "4.0.1",
+              "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz",
+              "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==",
+              "dev": true
+            }
+          }
+        },
+        "mkdirp": {
+          "version": "0.5.6",
+          "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz",
+          "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==",
+          "dev": true,
+          "requires": {
+            "minimist": "^1.2.6"
+          }
+        },
+        "resolve": {
+          "version": "1.1.7",
+          "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.1.7.tgz",
+          "integrity": "sha512-9znBF0vBcaSN3W2j7wKvdERPwqTxSpCq+if5C0WoTCyV9n24rua28jeuQ2pL/HOf+yUe/Mef+H/5p60K0Id3bg==",
+          "dev": true
+        },
+        "supports-color": {
+          "version": "3.2.3",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-3.2.3.tgz",
+          "integrity": "sha512-Jds2VIYDrlp5ui7t8abHN2bjAu4LV/q4N2KivFPpGH0lrka0BMq/33AmECUXlKPcHigkNaqfXRENFju+rlcy+A==",
+          "dev": true,
+          "requires": {
+            "has-flag": "^1.0.0"
+          }
+        },
+        "which": {
+          "version": "1.3.1",
+          "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz",
+          "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==",
+          "dev": true,
+          "requires": {
+            "isexe": "^2.0.0"
+          }
+        }
+      }
+    },
+    "scrypt-js": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/scrypt-js/-/scrypt-js-3.0.1.tgz",
+      "integrity": "sha512-cdwTTnqPu0Hyvf5in5asVdZocVDTNRmR7XEcJuIzMjJeSHybHl7vpB66AzwTaIg6CLSbtjcxc8fqcySfnTkccA==",
+      "devOptional": true
+    },
+    "secp256k1": {
+      "version": "4.0.3",
+      "resolved": "https://registry.npmjs.org/secp256k1/-/secp256k1-4.0.3.tgz",
+      "integrity": "sha512-NLZVf+ROMxwtEj3Xa562qgv2BK5e2WNmXPiOdVIPLgs6lyTzMvBq0aWTYMI5XCP9jZMVKOcqZLw/Wc4vDkuxhA==",
+      "dev": true,
+      "requires": {
+        "elliptic": "^6.5.4",
+        "node-addon-api": "^2.0.0",
+        "node-gyp-build": "^4.2.0"
+      }
+    },
+    "semver": {
+      "version": "6.3.1",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz",
+      "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==",
+      "dev": true
+    },
+    "semver-compare": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/semver-compare/-/semver-compare-1.0.0.tgz",
+      "integrity": "sha512-YM3/ITh2MJ5MtzaM429anh+x2jiLVjqILF4m4oyQB18W7Ggea7BfqdH/wGMK7dDiMghv/6WG7znWMwUDzJiXow==",
+      "dev": true
+    },
+    "semver-regex": {
+      "version": "3.1.4",
+      "resolved": "https://registry.npmjs.org/semver-regex/-/semver-regex-3.1.4.tgz",
+      "integrity": "sha512-6IiqeZNgq01qGf0TId0t3NvKzSvUsjcpdEO3AQNeIjR6A2+ckTnQlDpl4qu1bjRv0RzN3FP9hzFmws3lKqRWkA==",
+      "dev": true
+    },
+    "send": {
+      "version": "0.18.0",
+      "resolved": "https://registry.npmjs.org/send/-/send-0.18.0.tgz",
+      "integrity": "sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==",
+      "dev": true,
+      "requires": {
+        "debug": "2.6.9",
+        "depd": "2.0.0",
+        "destroy": "1.2.0",
+        "encodeurl": "~1.0.2",
+        "escape-html": "~1.0.3",
+        "etag": "~1.8.1",
+        "fresh": "0.5.2",
+        "http-errors": "2.0.0",
+        "mime": "1.6.0",
+        "ms": "2.1.3",
+        "on-finished": "2.4.1",
+        "range-parser": "~1.2.1",
+        "statuses": "2.0.1"
+      },
+      "dependencies": {
+        "debug": {
+          "version": "2.6.9",
+          "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+          "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+          "dev": true,
+          "requires": {
+            "ms": "2.0.0"
+          },
+          "dependencies": {
+            "ms": {
+              "version": "2.0.0",
+              "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+              "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
+              "dev": true
+            }
+          }
+        },
+        "ms": {
+          "version": "2.1.3",
+          "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+          "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
+          "dev": true
+        }
+      }
+    },
+    "sentence-case": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/sentence-case/-/sentence-case-2.1.1.tgz",
+      "integrity": "sha512-ENl7cYHaK/Ktwk5OTD+aDbQ3uC8IByu/6Bkg+HDv8Mm+XnBnppVNalcfJTNsp1ibstKh030/JKQQWglDvtKwEQ==",
+      "dev": true,
+      "requires": {
+        "no-case": "^2.2.0",
+        "upper-case-first": "^1.1.2"
+      }
+    },
+    "serialize-javascript": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.0.tgz",
+      "integrity": "sha512-Qr3TosvguFt8ePWqsvRfrKyQXIiW+nGbYpy8XK24NQHE83caxWt+mIymTT19DGFbNWNLfEwsrkSmN64lVWB9ag==",
+      "dev": true,
+      "requires": {
+        "randombytes": "^2.1.0"
+      }
+    },
+    "serve-static": {
+      "version": "1.15.0",
+      "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.15.0.tgz",
+      "integrity": "sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==",
+      "dev": true,
+      "requires": {
+        "encodeurl": "~1.0.2",
+        "escape-html": "~1.0.3",
+        "parseurl": "~1.3.3",
+        "send": "0.18.0"
+      }
+    },
+    "servify": {
+      "version": "0.1.12",
+      "resolved": "https://registry.npmjs.org/servify/-/servify-0.1.12.tgz",
+      "integrity": "sha512-/xE6GvsKKqyo1BAY+KxOWXcLpPsUUyji7Qg3bVD7hh1eRze5bR1uYiuDA/k3Gof1s9BTzQZEJK8sNcNGFIzeWw==",
+      "dev": true,
+      "requires": {
+        "body-parser": "^1.16.0",
+        "cors": "^2.8.1",
+        "express": "^4.14.0",
+        "request": "^2.79.0",
+        "xhr": "^2.3.3"
+      }
+    },
+    "set-blocking": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz",
+      "integrity": "sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw==",
+      "dev": true
+    },
+    "setimmediate": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.5.tgz",
+      "integrity": "sha512-MATJdZp8sLqDl/68LfQmbP8zKPLQNV6BIZoIgrscFDQ+RsvK/BxeDQOgyxKKoh0y/8h3BqVFnCqQ/gd+reiIXA==",
+      "dev": true
+    },
+    "setprototypeof": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
+      "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==",
+      "dev": true
+    },
+    "sha.js": {
+      "version": "2.4.11",
+      "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz",
+      "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==",
+      "dev": true,
+      "requires": {
+        "inherits": "^2.0.1",
+        "safe-buffer": "^5.0.1"
+      }
+    },
+    "sha1": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/sha1/-/sha1-1.1.1.tgz",
+      "integrity": "sha512-dZBS6OrMjtgVkopB1Gmo4RQCDKiZsqcpAQpkV/aaj+FCrCg8r4I4qMkDPQjBgLIxlmu9k4nUbWq6ohXahOneYA==",
+      "dev": true,
+      "requires": {
+        "charenc": ">= 0.0.1",
+        "crypt": ">= 0.0.1"
+      }
+    },
+    "sha3": {
+      "version": "2.1.4",
+      "resolved": "https://registry.npmjs.org/sha3/-/sha3-2.1.4.tgz",
+      "integrity": "sha512-S8cNxbyb0UGUM2VhRD4Poe5N58gJnJsLJ5vC7FYWGUmGhcsj4++WaIOBFVDxlG0W3To6xBuiRh+i0Qp2oNCOtg==",
+      "dev": true,
+      "requires": {
+        "buffer": "6.0.3"
+      },
+      "dependencies": {
+        "buffer": {
+          "version": "6.0.3",
+          "resolved": "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz",
+          "integrity": "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==",
+          "dev": true,
+          "requires": {
+            "base64-js": "^1.3.1",
+            "ieee754": "^1.2.1"
+          }
+        }
+      }
+    },
+    "shebang-command": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
+      "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==",
+      "dev": true,
+      "requires": {
+        "shebang-regex": "^3.0.0"
+      }
+    },
+    "shebang-regex": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz",
+      "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==",
+      "dev": true
+    },
+    "shelljs": {
+      "version": "0.8.5",
+      "resolved": "https://registry.npmjs.org/shelljs/-/shelljs-0.8.5.tgz",
+      "integrity": "sha512-TiwcRcrkhHvbrZbnRcFYMLl30Dfov3HKqzp5tO5b4pt6G/SezKcYhmDg15zXVBswHmctSAQKznqNW2LO5tTDow==",
+      "dev": true,
+      "requires": {
+        "glob": "^7.0.0",
+        "interpret": "^1.0.0",
+        "rechoir": "^0.6.2"
+      }
+    },
+    "side-channel": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz",
+      "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.0",
+        "get-intrinsic": "^1.0.2",
+        "object-inspect": "^1.9.0"
+      }
+    },
+    "signal-exit": {
+      "version": "3.0.7",
+      "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz",
+      "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==",
+      "dev": true
+    },
+    "simple-concat": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/simple-concat/-/simple-concat-1.0.1.tgz",
+      "integrity": "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==",
+      "dev": true
+    },
+    "simple-get": {
+      "version": "2.8.2",
+      "resolved": "https://registry.npmjs.org/simple-get/-/simple-get-2.8.2.tgz",
+      "integrity": "sha512-Ijd/rV5o+mSBBs4F/x9oDPtTx9Zb6X9brmnXvMW4J7IR15ngi9q5xxqWBKU744jTZiaXtxaPL7uHG6vtN8kUkw==",
+      "dev": true,
+      "requires": {
+        "decompress-response": "^3.3.0",
+        "once": "^1.3.1",
+        "simple-concat": "^1.0.0"
+      },
+      "dependencies": {
+        "decompress-response": {
+          "version": "3.3.0",
+          "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz",
+          "integrity": "sha512-BzRPQuY1ip+qDonAOz42gRm/pg9F768C+npV/4JOsxRC2sq+Rlk+Q4ZCAsOhnIaMrgarILY+RMUIvMmmX1qAEA==",
+          "dev": true,
+          "requires": {
+            "mimic-response": "^1.0.0"
+          }
+        }
+      }
+    },
+    "sisteransi": {
+      "version": "1.0.5",
+      "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz",
+      "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==",
+      "dev": true
+    },
+    "slash": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz",
+      "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==",
+      "dev": true
+    },
+    "slice-ansi": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-4.0.0.tgz",
+      "integrity": "sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ==",
+      "dev": true,
+      "requires": {
+        "ansi-styles": "^4.0.0",
+        "astral-regex": "^2.0.0",
+        "is-fullwidth-code-point": "^3.0.0"
+      }
+    },
+    "snake-case": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/snake-case/-/snake-case-2.1.0.tgz",
+      "integrity": "sha512-FMR5YoPFwOLuh4rRz92dywJjyKYZNLpMn1R5ujVpIYkbA9p01fq8RMg0FkO4M+Yobt4MjHeLTJVm5xFFBHSV2Q==",
+      "dev": true,
+      "requires": {
+        "no-case": "^2.2.0"
+      }
+    },
+    "solc": {
+      "version": "0.4.26",
+      "resolved": "https://registry.npmjs.org/solc/-/solc-0.4.26.tgz",
+      "integrity": "sha512-o+c6FpkiHd+HPjmjEVpQgH7fqZ14tJpXhho+/bQXlXbliLIS/xjXb42Vxh+qQY1WCSTMQ0+a5vR9vi0MfhU6mA==",
+      "dev": true,
+      "requires": {
+        "fs-extra": "^0.30.0",
+        "memorystream": "^0.3.1",
+        "require-from-string": "^1.1.0",
+        "semver": "^5.3.0",
+        "yargs": "^4.7.1"
+      },
+      "dependencies": {
+        "fs-extra": {
+          "version": "0.30.0",
+          "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-0.30.0.tgz",
+          "integrity": "sha512-UvSPKyhMn6LEd/WpUaV9C9t3zATuqoqfWc3QdPhPLb58prN9tqYPlPWi8Krxi44loBoUzlobqZ3+8tGpxxSzwA==",
+          "dev": true,
+          "requires": {
+            "graceful-fs": "^4.1.2",
+            "jsonfile": "^2.1.0",
+            "klaw": "^1.0.0",
+            "path-is-absolute": "^1.0.0",
+            "rimraf": "^2.2.8"
+          }
+        },
+        "jsonfile": {
+          "version": "2.4.0",
+          "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-2.4.0.tgz",
+          "integrity": "sha512-PKllAqbgLgxHaj8TElYymKCAgrASebJrWpTnEkOaTowt23VKXXN0sUeriJ+eh7y6ufb/CC5ap11pz71/cM0hUw==",
+          "dev": true,
+          "requires": {
+            "graceful-fs": "^4.1.6"
+          }
+        },
+        "require-from-string": {
+          "version": "1.2.1",
+          "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-1.2.1.tgz",
+          "integrity": "sha512-H7AkJWMobeskkttHyhTVtS0fxpFLjxhbfMa6Bk3wimP7sdPRGL3EyCg3sAQenFfAe+xQ+oAc85Nmtvq0ROM83Q==",
+          "dev": true
+        },
+        "rimraf": {
+          "version": "2.7.1",
+          "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz",
+          "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==",
+          "dev": true,
+          "requires": {
+            "glob": "^7.1.3"
+          }
+        },
+        "semver": {
+          "version": "5.7.2",
+          "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz",
+          "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==",
+          "dev": true
+        }
+      }
+    },
+    "solhint": {
+      "version": "3.4.1",
+      "resolved": "https://registry.npmjs.org/solhint/-/solhint-3.4.1.tgz",
+      "integrity": "sha512-pzZn2RlZhws1XwvLPVSsxfHrwsteFf5eySOhpAytzXwKQYbTCJV6z8EevYDiSVKMpWrvbKpEtJ055CuEmzp4Xg==",
+      "dev": true,
+      "requires": {
+        "@solidity-parser/parser": "^0.16.0",
+        "ajv": "^6.12.6",
+        "antlr4": "^4.11.0",
+        "ast-parents": "^0.0.1",
+        "chalk": "^4.1.2",
+        "commander": "^10.0.0",
+        "cosmiconfig": "^8.0.0",
+        "fast-diff": "^1.2.0",
+        "glob": "^8.0.3",
+        "ignore": "^5.2.4",
+        "js-yaml": "^4.1.0",
+        "lodash": "^4.17.21",
+        "pluralize": "^8.0.0",
+        "prettier": "^2.8.3",
+        "semver": "^6.3.0",
+        "strip-ansi": "^6.0.1",
+        "table": "^6.8.1",
+        "text-table": "^0.2.0"
+      },
+      "dependencies": {
+        "@solidity-parser/parser": {
+          "version": "0.16.0",
+          "resolved": "https://registry.npmjs.org/@solidity-parser/parser/-/parser-0.16.0.tgz",
+          "integrity": "sha512-ESipEcHyRHg4Np4SqBCfcXwyxxna1DgFVz69bgpLV8vzl/NP1DtcKsJ4dJZXWQhY/Z4J2LeKBiOkOVZn9ct33Q==",
+          "dev": true,
+          "requires": {
+            "antlr4ts": "^0.5.0-alpha.4"
+          }
+        },
+        "ajv": {
+          "version": "6.12.6",
+          "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz",
+          "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==",
+          "dev": true,
+          "requires": {
+            "fast-deep-equal": "^3.1.1",
+            "fast-json-stable-stringify": "^2.0.0",
+            "json-schema-traverse": "^0.4.1",
+            "uri-js": "^4.2.2"
+          }
+        },
+        "brace-expansion": {
+          "version": "2.0.1",
+          "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz",
+          "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==",
+          "dev": true,
+          "requires": {
+            "balanced-match": "^1.0.0"
+          }
+        },
+        "commander": {
+          "version": "10.0.0",
+          "resolved": "https://registry.npmjs.org/commander/-/commander-10.0.0.tgz",
+          "integrity": "sha512-zS5PnTI22FIRM6ylNW8G4Ap0IEOyk62fhLSD0+uHRT9McRCLGpkVNvao4bjimpK/GShynyQkFFxHhwMcETmduA==",
+          "dev": true
+        },
+        "cosmiconfig": {
+          "version": "8.1.0",
+          "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-8.1.0.tgz",
+          "integrity": "sha512-0tLZ9URlPGU7JsKq0DQOQ3FoRsYX8xDZ7xMiATQfaiGMz7EHowNkbU9u1coAOmnh9p/1ySpm0RB3JNWRXM5GCg==",
+          "dev": true,
+          "requires": {
+            "import-fresh": "^3.2.1",
+            "js-yaml": "^4.1.0",
+            "parse-json": "^5.0.0",
+            "path-type": "^4.0.0"
+          }
+        },
+        "glob": {
+          "version": "8.1.0",
+          "resolved": "https://registry.npmjs.org/glob/-/glob-8.1.0.tgz",
+          "integrity": "sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==",
+          "dev": true,
+          "requires": {
+            "fs.realpath": "^1.0.0",
+            "inflight": "^1.0.4",
+            "inherits": "2",
+            "minimatch": "^5.0.1",
+            "once": "^1.3.0"
+          }
+        },
+        "json-schema-traverse": {
+          "version": "0.4.1",
+          "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+          "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+          "dev": true
+        },
+        "minimatch": {
+          "version": "5.1.6",
+          "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.6.tgz",
+          "integrity": "sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==",
+          "dev": true,
+          "requires": {
+            "brace-expansion": "^2.0.1"
+          }
+        },
+        "parse-json": {
+          "version": "5.2.0",
+          "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz",
+          "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==",
+          "dev": true,
+          "requires": {
+            "@babel/code-frame": "^7.0.0",
+            "error-ex": "^1.3.1",
+            "json-parse-even-better-errors": "^2.3.0",
+            "lines-and-columns": "^1.1.6"
+          }
+        },
+        "path-type": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz",
+          "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==",
+          "dev": true
+        }
+      }
+    },
+    "solidity-comments-extractor": {
+      "version": "0.0.7",
+      "resolved": "https://registry.npmjs.org/solidity-comments-extractor/-/solidity-comments-extractor-0.0.7.tgz",
+      "integrity": "sha512-wciNMLg/Irp8OKGrh3S2tfvZiZ0NEyILfcRCXCD4mp7SgK/i9gzLfhY2hY7VMCQJ3kH9UB9BzNdibIVMchzyYw==",
+      "dev": true
+    },
+    "solidity-coverage": {
+      "version": "0.8.5",
+      "resolved": "https://registry.npmjs.org/solidity-coverage/-/solidity-coverage-0.8.5.tgz",
+      "integrity": "sha512-6C6N6OV2O8FQA0FWA95FdzVH+L16HU94iFgg5wAFZ29UpLFkgNI/DRR2HotG1bC0F4gAc/OMs2BJI44Q/DYlKQ==",
+      "dev": true,
+      "requires": {
+        "@ethersproject/abi": "^5.0.9",
+        "@solidity-parser/parser": "^0.16.0",
+        "chalk": "^2.4.2",
+        "death": "^1.1.0",
+        "detect-port": "^1.3.0",
+        "difflib": "^0.2.4",
+        "fs-extra": "^8.1.0",
+        "ghost-testrpc": "^0.0.2",
+        "global-modules": "^2.0.0",
+        "globby": "^10.0.1",
+        "jsonschema": "^1.2.4",
+        "lodash": "^4.17.15",
+        "mocha": "10.2.0",
+        "node-emoji": "^1.10.0",
+        "pify": "^4.0.1",
+        "recursive-readdir": "^2.2.2",
+        "sc-istanbul": "^0.4.5",
+        "semver": "^7.3.4",
+        "shelljs": "^0.8.3",
+        "web3-utils": "^1.3.6"
+      },
+      "dependencies": {
+        "@solidity-parser/parser": {
+          "version": "0.16.2",
+          "resolved": "https://registry.npmjs.org/@solidity-parser/parser/-/parser-0.16.2.tgz",
+          "integrity": "sha512-PI9NfoA3P8XK2VBkK5oIfRgKDsicwDZfkVq9ZTBCQYGOP1N2owgY2dyLGyU5/J/hQs8KRk55kdmvTLjy3Mu3vg==",
+          "dev": true,
+          "requires": {
+            "antlr4ts": "^0.5.0-alpha.4"
+          }
+        },
+        "ansi-styles": {
+          "version": "3.2.1",
+          "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
+          "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
+          "dev": true,
+          "requires": {
+            "color-convert": "^1.9.0"
+          }
+        },
+        "chalk": {
+          "version": "2.4.2",
+          "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
+          "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
+          "dev": true,
+          "requires": {
+            "ansi-styles": "^3.2.1",
+            "escape-string-regexp": "^1.0.5",
+            "supports-color": "^5.3.0"
+          }
+        },
+        "color-convert": {
+          "version": "1.9.3",
+          "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
+          "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
+          "dev": true,
+          "requires": {
+            "color-name": "1.1.3"
+          }
+        },
+        "color-name": {
+          "version": "1.1.3",
+          "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
+          "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==",
+          "dev": true
+        },
+        "fs-extra": {
+          "version": "8.1.0",
+          "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz",
+          "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==",
+          "dev": true,
+          "requires": {
+            "graceful-fs": "^4.2.0",
+            "jsonfile": "^4.0.0",
+            "universalify": "^0.1.0"
+          }
+        },
+        "globby": {
+          "version": "10.0.2",
+          "resolved": "https://registry.npmjs.org/globby/-/globby-10.0.2.tgz",
+          "integrity": "sha512-7dUi7RvCoT/xast/o/dLN53oqND4yk0nsHkhRgn9w65C4PofCLOoJ39iSOg+qVDdWQPIEj+eszMHQ+aLVwwQSg==",
+          "dev": true,
+          "requires": {
+            "@types/glob": "^7.1.1",
+            "array-union": "^2.1.0",
+            "dir-glob": "^3.0.1",
+            "fast-glob": "^3.0.3",
+            "glob": "^7.1.3",
+            "ignore": "^5.1.1",
+            "merge2": "^1.2.3",
+            "slash": "^3.0.0"
+          }
+        },
+        "has-flag": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
+          "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
+          "dev": true
+        },
+        "pify": {
+          "version": "4.0.1",
+          "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz",
+          "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==",
+          "dev": true
+        },
+        "semver": {
+          "version": "7.5.4",
+          "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz",
+          "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==",
+          "dev": true,
+          "requires": {
+            "lru-cache": "^6.0.0"
+          }
+        },
+        "supports-color": {
+          "version": "5.5.0",
+          "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
+          "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
+          "dev": true,
+          "requires": {
+            "has-flag": "^3.0.0"
+          }
+        }
+      }
+    },
+    "solmate": {
+      "version": "6.7.0",
+      "resolved": "https://registry.npmjs.org/solmate/-/solmate-6.7.0.tgz",
+      "integrity": "sha512-iMPr+gKbKjXBB12a+Iz5Tua5r7T4yugHaGXDWSJbBZB4Gr3vLeUUvKeLyMxCWWqk1xlLhFDFFuAmOzeyVBuyvQ==",
+      "dev": true
+    },
+    "source-map": {
+      "version": "0.6.1",
+      "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz",
+      "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==",
+      "dev": true
+    },
+    "source-map-support": {
+      "version": "0.5.21",
+      "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz",
+      "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==",
+      "dev": true,
+      "requires": {
+        "buffer-from": "^1.0.0",
+        "source-map": "^0.6.0"
+      }
+    },
+    "spdx-correct": {
+      "version": "3.2.0",
+      "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.2.0.tgz",
+      "integrity": "sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA==",
+      "dev": true,
+      "requires": {
+        "spdx-expression-parse": "^3.0.0",
+        "spdx-license-ids": "^3.0.0"
+      }
+    },
+    "spdx-exceptions": {
+      "version": "2.3.0",
+      "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz",
+      "integrity": "sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A==",
+      "dev": true
+    },
+    "spdx-expression-parse": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz",
+      "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==",
+      "dev": true,
+      "requires": {
+        "spdx-exceptions": "^2.1.0",
+        "spdx-license-ids": "^3.0.0"
+      }
+    },
+    "spdx-license-ids": {
+      "version": "3.0.12",
+      "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.12.tgz",
+      "integrity": "sha512-rr+VVSXtRhO4OHbXUiAF7xW3Bo9DuuF6C5jH+q/x15j2jniycgKbxU09Hr0WqlSLUs4i4ltHGXqTe7VHclYWyA==",
+      "dev": true
+    },
+    "sprintf-js": {
+      "version": "1.0.3",
+      "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz",
+      "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==",
+      "dev": true
+    },
+    "sshpk": {
+      "version": "1.17.0",
+      "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.17.0.tgz",
+      "integrity": "sha512-/9HIEs1ZXGhSPE8X6Ccm7Nam1z8KcoCqPdI7ecm1N33EzAetWahvQWVqLZtaZQ+IDKX4IyA2o0gBzqIMkAagHQ==",
+      "dev": true,
+      "requires": {
+        "asn1": "~0.2.3",
+        "assert-plus": "^1.0.0",
+        "bcrypt-pbkdf": "^1.0.0",
+        "dashdash": "^1.12.0",
+        "ecc-jsbn": "~0.1.1",
+        "getpass": "^0.1.1",
+        "jsbn": "~0.1.0",
+        "safer-buffer": "^2.0.2",
+        "tweetnacl": "~0.14.0"
+      }
+    },
+    "stacktrace-parser": {
+      "version": "0.1.10",
+      "resolved": "https://registry.npmjs.org/stacktrace-parser/-/stacktrace-parser-0.1.10.tgz",
+      "integrity": "sha512-KJP1OCML99+8fhOHxwwzyWrlUuVX5GQ0ZpJTd1DFXhdkrvg1szxfHhawXUZ3g9TkXORQd4/WG68jMlQZ2p8wlg==",
+      "dev": true,
+      "requires": {
+        "type-fest": "^0.7.1"
+      },
+      "dependencies": {
+        "type-fest": {
+          "version": "0.7.1",
+          "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.7.1.tgz",
+          "integrity": "sha512-Ne2YiiGN8bmrmJJEuTWTLJR32nh/JdL1+PSicowtNb0WFpn59GK8/lfD61bVtzguz7b3PBt74nxpv/Pw5po5Rg==",
+          "dev": true
+        }
+      }
+    },
+    "statuses": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz",
+      "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==",
+      "dev": true
+    },
+    "strict-uri-encode": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/strict-uri-encode/-/strict-uri-encode-1.1.0.tgz",
+      "integrity": "sha512-R3f198pcvnB+5IpnBlRkphuE9n46WyVl8I39W/ZUTZLz4nqSP/oLYUrcnJrw462Ds8he4YKMov2efsTIw1BDGQ==",
+      "dev": true
+    },
+    "string_decoder": {
+      "version": "1.3.0",
+      "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz",
+      "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==",
+      "dev": true,
+      "requires": {
+        "safe-buffer": "~5.2.0"
+      }
+    },
+    "string-format": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/string-format/-/string-format-2.0.0.tgz",
+      "integrity": "sha512-bbEs3scLeYNXLecRRuk6uJxdXUSj6le/8rNPHChIJTn2V79aXVTR1EH2OH5zLKKoz0V02fOUKZZcw01pLUShZA==",
+      "dev": true
+    },
+    "string-width": {
+      "version": "4.2.3",
+      "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
+      "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
+      "dev": true,
+      "requires": {
+        "emoji-regex": "^8.0.0",
+        "is-fullwidth-code-point": "^3.0.0",
+        "strip-ansi": "^6.0.1"
+      }
+    },
+    "string.prototype.trimend": {
+      "version": "1.0.6",
+      "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.6.tgz",
+      "integrity": "sha512-JySq+4mrPf9EsDBEDYMOb/lM7XQLulwg5R/m1r0PXEFqrV0qHvl58sdTilSXtKOflCsK2E8jxf+GKC0T07RWwQ==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "es-abstract": "^1.20.4"
+      }
+    },
+    "string.prototype.trimstart": {
+      "version": "1.0.6",
+      "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.6.tgz",
+      "integrity": "sha512-omqjMDaY92pbn5HOX7f9IccLA+U1tA9GvtU4JrodiXFfYB7jPzzHpRzpglLAjtUV6bB557zwClJezTqnAiYnQA==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "define-properties": "^1.1.4",
+        "es-abstract": "^1.20.4"
+      }
+    },
+    "strip-ansi": {
+      "version": "6.0.1",
+      "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
+      "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
+      "dev": true,
+      "requires": {
+        "ansi-regex": "^5.0.1"
+      }
+    },
+    "strip-bom": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-2.0.0.tgz",
+      "integrity": "sha512-kwrX1y7czp1E69n2ajbG65mIo9dqvJ+8aBQXOGVxqwvNbsXdFM6Lq37dLAY3mknUwru8CfcCbfOLL/gMo+fi3g==",
+      "dev": true,
+      "requires": {
+        "is-utf8": "^0.2.0"
+      }
+    },
+    "strip-hex-prefix": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/strip-hex-prefix/-/strip-hex-prefix-1.0.0.tgz",
+      "integrity": "sha512-q8d4ue7JGEiVcypji1bALTos+0pWtyGlivAWyPuTkHzuTCJqrK9sWxYQZUq6Nq3cuyv3bm734IhHvHtGGURU6A==",
+      "dev": true,
+      "requires": {
+        "is-hex-prefixed": "1.0.0"
+      }
+    },
+    "strip-indent": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/strip-indent/-/strip-indent-2.0.0.tgz",
+      "integrity": "sha512-RsSNPLpq6YUL7QYy44RnPVTn/lcVZtb48Uof3X5JLbF4zD/Gs7ZFDv2HWol+leoQN2mT86LAzSshGfkTlSOpsA==",
+      "dev": true
+    },
+    "strip-json-comments": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz",
+      "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==",
+      "dev": true
+    },
+    "supports-color": {
+      "version": "7.2.0",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
+      "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
+      "dev": true,
+      "requires": {
+        "has-flag": "^4.0.0"
+      }
+    },
+    "supports-preserve-symlinks-flag": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz",
+      "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==",
+      "dev": true
+    },
+    "swap-case": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/swap-case/-/swap-case-1.1.2.tgz",
+      "integrity": "sha512-BAmWG6/bx8syfc6qXPprof3Mn5vQgf5dwdUNJhsNqU9WdPt5P+ES/wQ5bxfijy8zwZgZZHslC3iAsxsuQMCzJQ==",
+      "dev": true,
+      "requires": {
+        "lower-case": "^1.1.1",
+        "upper-case": "^1.1.1"
+      }
+    },
+    "swarm-js": {
+      "version": "0.1.42",
+      "resolved": "https://registry.npmjs.org/swarm-js/-/swarm-js-0.1.42.tgz",
+      "integrity": "sha512-BV7c/dVlA3R6ya1lMlSSNPLYrntt0LUq4YMgy3iwpCIc6rZnS5W2wUoctarZ5pXlpKtxDDf9hNziEkcfrxdhqQ==",
+      "dev": true,
+      "requires": {
+        "bluebird": "^3.5.0",
+        "buffer": "^5.0.5",
+        "eth-lib": "^0.1.26",
+        "fs-extra": "^4.0.2",
+        "got": "^11.8.5",
+        "mime-types": "^2.1.16",
+        "mkdirp-promise": "^5.0.1",
+        "mock-fs": "^4.1.0",
+        "setimmediate": "^1.0.5",
+        "tar": "^4.0.2",
+        "xhr-request": "^1.0.1"
+      },
+      "dependencies": {
+        "@szmarczak/http-timer": {
+          "version": "4.0.6",
+          "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-4.0.6.tgz",
+          "integrity": "sha512-4BAffykYOgO+5nzBWYwE3W90sBgLJoUPRWWcL8wlyiM8IB8ipJz3UMJ9KXQd1RKQXpKp8Tutn80HZtWsu2u76w==",
+          "dev": true,
+          "requires": {
+            "defer-to-connect": "^2.0.0"
+          }
+        },
+        "cacheable-lookup": {
+          "version": "5.0.4",
+          "resolved": "https://registry.npmjs.org/cacheable-lookup/-/cacheable-lookup-5.0.4.tgz",
+          "integrity": "sha512-2/kNscPhpcxrOigMZzbiWF7dz8ilhb/nIHU3EyZiXWXpeq/au8qJ8VhdftMkty3n7Gj6HIGalQG8oiBNB3AJgA==",
+          "dev": true
+        },
+        "fs-extra": {
+          "version": "4.0.3",
+          "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-4.0.3.tgz",
+          "integrity": "sha512-q6rbdDd1o2mAnQreO7YADIxf/Whx4AHBiRf6d+/cVT8h44ss+lHgxf1FemcqDnQt9X3ct4McHr+JMGlYSsK7Cg==",
+          "dev": true,
+          "requires": {
+            "graceful-fs": "^4.1.2",
+            "jsonfile": "^4.0.0",
+            "universalify": "^0.1.0"
+          }
+        },
+        "got": {
+          "version": "11.8.6",
+          "resolved": "https://registry.npmjs.org/got/-/got-11.8.6.tgz",
+          "integrity": "sha512-6tfZ91bOr7bOXnK7PRDCGBLa1H4U080YHNaAQ2KsMGlLEzRbk44nsZF2E1IeRc3vtJHPVbKCYgdFbaGO2ljd8g==",
+          "dev": true,
+          "requires": {
+            "@sindresorhus/is": "^4.0.0",
+            "@szmarczak/http-timer": "^4.0.5",
+            "@types/cacheable-request": "^6.0.1",
+            "@types/responselike": "^1.0.0",
+            "cacheable-lookup": "^5.0.3",
+            "cacheable-request": "^7.0.2",
+            "decompress-response": "^6.0.0",
+            "http2-wrapper": "^1.0.0-beta.5.2",
+            "lowercase-keys": "^2.0.0",
+            "p-cancelable": "^2.0.0",
+            "responselike": "^2.0.0"
+          }
+        },
+        "http2-wrapper": {
+          "version": "1.0.3",
+          "resolved": "https://registry.npmjs.org/http2-wrapper/-/http2-wrapper-1.0.3.tgz",
+          "integrity": "sha512-V+23sDMr12Wnz7iTcDeJr3O6AIxlnvT/bmaAAAP/Xda35C90p9599p0F1eHR/N1KILWSoWVAiOMFjBBXaXSMxg==",
+          "dev": true,
+          "requires": {
+            "quick-lru": "^5.1.1",
+            "resolve-alpn": "^1.0.0"
+          }
+        },
+        "lowercase-keys": {
+          "version": "2.0.0",
+          "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz",
+          "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==",
+          "dev": true
+        },
+        "p-cancelable": {
+          "version": "2.1.1",
+          "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-2.1.1.tgz",
+          "integrity": "sha512-BZOr3nRQHOntUjTrH8+Lh54smKHoHyur8We1V8DSMVrl5A2malOOwuJRnKRDjSnkoeBh4at6BwEnb5I7Jl31wg==",
+          "dev": true
+        }
+      }
+    },
+    "sync-request": {
+      "version": "6.1.0",
+      "resolved": "https://registry.npmjs.org/sync-request/-/sync-request-6.1.0.tgz",
+      "integrity": "sha512-8fjNkrNlNCrVc/av+Jn+xxqfCjYaBoHqCsDz6mt030UMxJGr+GSfCV1dQt2gRtlL63+VPidwDVLr7V2OcTSdRw==",
+      "dev": true,
+      "requires": {
+        "http-response-object": "^3.0.1",
+        "sync-rpc": "^1.2.1",
+        "then-request": "^6.0.0"
+      }
+    },
+    "sync-rpc": {
+      "version": "1.3.6",
+      "resolved": "https://registry.npmjs.org/sync-rpc/-/sync-rpc-1.3.6.tgz",
+      "integrity": "sha512-J8jTXuZzRlvU7HemDgHi3pGnh/rkoqR/OZSjhTyyZrEkkYQbk7Z33AXp37mkPfPpfdOuj7Ex3H/TJM1z48uPQw==",
+      "dev": true,
+      "requires": {
+        "get-port": "^3.1.0"
+      }
+    },
+    "table": {
+      "version": "6.8.1",
+      "resolved": "https://registry.npmjs.org/table/-/table-6.8.1.tgz",
+      "integrity": "sha512-Y4X9zqrCftUhMeH2EptSSERdVKt/nEdijTOacGD/97EKjhQ/Qs8RTlEGABSJNNN8lac9kheH+af7yAkEWlgneA==",
+      "dev": true,
+      "requires": {
+        "ajv": "^8.0.1",
+        "lodash.truncate": "^4.4.2",
+        "slice-ansi": "^4.0.0",
+        "string-width": "^4.2.3",
+        "strip-ansi": "^6.0.1"
+      }
+    },
+    "table-layout": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/table-layout/-/table-layout-1.0.2.tgz",
+      "integrity": "sha512-qd/R7n5rQTRFi+Zf2sk5XVVd9UQl6ZkduPFC3S7WEGJAmetDTjY3qPN50eSKzwuzEyQKy5TN2TiZdkIjos2L6A==",
+      "dev": true,
+      "requires": {
+        "array-back": "^4.0.1",
+        "deep-extend": "~0.6.0",
+        "typical": "^5.2.0",
+        "wordwrapjs": "^4.0.0"
+      },
+      "dependencies": {
+        "array-back": {
+          "version": "4.0.2",
+          "resolved": "https://registry.npmjs.org/array-back/-/array-back-4.0.2.tgz",
+          "integrity": "sha512-NbdMezxqf94cnNfWLL7V/im0Ub+Anbb0IoZhvzie8+4HJ4nMQuzHuy49FkGYCJK2yAloZ3meiB6AVMClbrI1vg==",
+          "dev": true
+        },
+        "typical": {
+          "version": "5.2.0",
+          "resolved": "https://registry.npmjs.org/typical/-/typical-5.2.0.tgz",
+          "integrity": "sha512-dvdQgNDNJo+8B2uBQoqdb11eUCE1JQXhvjC/CZtgvZseVd5TYMXnq0+vuUemXbd/Se29cTaUuPX3YIc2xgbvIg==",
+          "dev": true
+        }
+      }
+    },
+    "tar": {
+      "version": "4.4.19",
+      "resolved": "https://registry.npmjs.org/tar/-/tar-4.4.19.tgz",
+      "integrity": "sha512-a20gEsvHnWe0ygBY8JbxoM4w3SJdhc7ZAuxkLqh+nvNQN2IOt0B5lLgM490X5Hl8FF0dl0tOf2ewFYAlIFgzVA==",
+      "dev": true,
+      "requires": {
+        "chownr": "^1.1.4",
+        "fs-minipass": "^1.2.7",
+        "minipass": "^2.9.0",
+        "minizlib": "^1.3.3",
+        "mkdirp": "^0.5.5",
+        "safe-buffer": "^5.2.1",
+        "yallist": "^3.1.1"
+      },
+      "dependencies": {
+        "mkdirp": {
+          "version": "0.5.6",
+          "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz",
+          "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==",
+          "dev": true,
+          "requires": {
+            "minimist": "^1.2.6"
+          }
+        },
+        "yallist": {
+          "version": "3.1.1",
+          "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
+          "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==",
+          "dev": true
+        }
+      }
+    },
+    "tenderly": {
+      "version": "0.4.0",
+      "resolved": "https://registry.npmjs.org/tenderly/-/tenderly-0.4.0.tgz",
+      "integrity": "sha512-wZgQ8Z1utc/QoAfVvMHO/ONLXJ3Vw3yjzJzpMmMUR4kvUu861mI7+mL9R1aeUuXI06cv81LZH96Vh+AOseIYoA==",
+      "dev": true,
+      "requires": {
+        "axios": "^0.27.2",
+        "cli-table3": "^0.6.2",
+        "commander": "^9.4.0",
+        "express": "^4.18.1",
+        "hyperlinker": "^1.0.0",
+        "js-yaml": "^4.1.0",
+        "open": "^8.4.0",
+        "prompts": "^2.4.2",
+        "tslog": "^4.4.0"
+      },
+      "dependencies": {
+        "commander": {
+          "version": "9.5.0",
+          "resolved": "https://registry.npmjs.org/commander/-/commander-9.5.0.tgz",
+          "integrity": "sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==",
+          "dev": true
+        }
+      }
+    },
+    "testrpc": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/testrpc/-/testrpc-0.0.1.tgz",
+      "integrity": "sha512-afH1hO+SQ/VPlmaLUFj2636QMeDvPCeQMc/9RBMW0IfjNe9gFD9Ra3ShqYkB7py0do1ZcCna/9acHyzTJ+GcNA==",
+      "dev": true
+    },
+    "text-table": {
+      "version": "0.2.0",
+      "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz",
+      "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==",
+      "dev": true
+    },
+    "then-request": {
+      "version": "6.0.2",
+      "resolved": "https://registry.npmjs.org/then-request/-/then-request-6.0.2.tgz",
+      "integrity": "sha512-3ZBiG7JvP3wbDzA9iNY5zJQcHL4jn/0BWtXIkagfz7QgOL/LqjCEOBQuJNZfu0XYnv5JhKh+cDxCPM4ILrqruA==",
+      "dev": true,
+      "requires": {
+        "@types/concat-stream": "^1.6.0",
+        "@types/form-data": "0.0.33",
+        "@types/node": "^8.0.0",
+        "@types/qs": "^6.2.31",
+        "caseless": "~0.12.0",
+        "concat-stream": "^1.6.0",
+        "form-data": "^2.2.0",
+        "http-basic": "^8.1.1",
+        "http-response-object": "^3.0.1",
+        "promise": "^8.0.0",
+        "qs": "^6.4.0"
+      },
+      "dependencies": {
+        "@types/node": {
+          "version": "8.10.66",
+          "resolved": "https://registry.npmjs.org/@types/node/-/node-8.10.66.tgz",
+          "integrity": "sha512-tktOkFUA4kXx2hhhrB8bIFb5TbwzS4uOhKEmwiD+NoiL0qtP2OQ9mFldbgD4dV1djrlBYP6eBuQZiWjuHUpqFw==",
+          "dev": true
+        }
+      }
+    },
+    "timed-out": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/timed-out/-/timed-out-4.0.1.tgz",
+      "integrity": "sha512-G7r3AhovYtr5YKOWQkta8RKAPb+J9IsO4uVmzjl8AZwfhs8UcUwTiD6gcJYSgOtzyjvQKrKYn41syHbUWMkafA==",
+      "dev": true
+    },
+    "title-case": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/title-case/-/title-case-2.1.1.tgz",
+      "integrity": "sha512-EkJoZ2O3zdCz3zJsYCsxyq2OC5hrxR9mfdd5I+w8h/tmFfeOxJ+vvkxsKxdmN0WtS9zLdHEgfgVOiMVgv+Po4Q==",
+      "dev": true,
+      "requires": {
+        "no-case": "^2.2.0",
+        "upper-case": "^1.0.3"
+      }
+    },
+    "tmp": {
+      "version": "0.0.33",
+      "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz",
+      "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==",
+      "dev": true,
+      "requires": {
+        "os-tmpdir": "~1.0.2"
+      }
+    },
+    "to-regex-range": {
+      "version": "5.0.1",
+      "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
+      "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
+      "dev": true,
+      "requires": {
+        "is-number": "^7.0.0"
+      }
+    },
+    "toidentifier": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz",
+      "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==",
+      "dev": true
+    },
+    "tough-cookie": {
+      "version": "2.5.0",
+      "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.5.0.tgz",
+      "integrity": "sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==",
+      "dev": true,
+      "requires": {
+        "psl": "^1.1.28",
+        "punycode": "^2.1.1"
+      }
+    },
+    "tr46": {
+      "version": "0.0.3",
+      "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz",
+      "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==",
+      "dev": true
+    },
+    "ts-command-line-args": {
+      "version": "2.4.2",
+      "resolved": "https://registry.npmjs.org/ts-command-line-args/-/ts-command-line-args-2.4.2.tgz",
+      "integrity": "sha512-mJLQQBOdyD4XI/ZWQY44PIdYde47JhV2xl380O7twPkTQ+Y5vFDHsk8LOeXKuz7dVY5aDCfAzRarNfSqtKOkQQ==",
+      "dev": true,
+      "requires": {
+        "@morgan-stanley/ts-mocking-bird": "^0.6.2",
+        "chalk": "^4.1.0",
+        "command-line-args": "^5.1.1",
+        "command-line-usage": "^6.1.0",
+        "string-format": "^2.0.0"
+      }
+    },
+    "ts-essentials": {
+      "version": "7.0.3",
+      "resolved": "https://registry.npmjs.org/ts-essentials/-/ts-essentials-7.0.3.tgz",
+      "integrity": "sha512-8+gr5+lqO3G84KdiTSMRLtuyJ+nTBVRKuCrK4lidMPdVeEp0uqC875uE5NMcaA7YYMN7XsNiFQuMvasF8HT/xQ==",
+      "dev": true,
+      "requires": {}
+    },
+    "ts-node": {
+      "version": "10.9.1",
+      "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.9.1.tgz",
+      "integrity": "sha512-NtVysVPkxxrwFGUUxGYhfux8k78pQB3JqYBXlLRZgdGUqTO5wU/UyHop5p70iEbGhB7q5KmiZiU0Y3KlJrScEw==",
+      "dev": true,
+      "requires": {
+        "@cspotcode/source-map-support": "^0.8.0",
+        "@tsconfig/node10": "^1.0.7",
+        "@tsconfig/node12": "^1.0.7",
+        "@tsconfig/node14": "^1.0.0",
+        "@tsconfig/node16": "^1.0.2",
+        "acorn": "^8.4.1",
+        "acorn-walk": "^8.1.1",
+        "arg": "^4.1.0",
+        "create-require": "^1.1.0",
+        "diff": "^4.0.1",
+        "make-error": "^1.1.1",
+        "v8-compile-cache-lib": "^3.0.1",
+        "yn": "3.1.1"
+      },
+      "dependencies": {
+        "diff": {
+          "version": "4.0.2",
+          "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz",
+          "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==",
+          "dev": true
+        }
+      }
+    },
+    "tsconfig-paths": {
+      "version": "3.14.2",
+      "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.14.2.tgz",
+      "integrity": "sha512-o/9iXgCYc5L/JxCHPe3Hvh8Q/2xm5Z+p18PESBU6Ff33695QnCHBEjcytY2q19ua7Mbl/DavtBOLq+oG0RCL+g==",
+      "dev": true,
+      "requires": {
+        "@types/json5": "^0.0.29",
+        "json5": "^1.0.2",
+        "minimist": "^1.2.6",
+        "strip-bom": "^3.0.0"
+      },
+      "dependencies": {
+        "strip-bom": {
+          "version": "3.0.0",
+          "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz",
+          "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==",
+          "dev": true
+        }
+      }
+    },
+    "tslib": {
+      "version": "1.14.1",
+      "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz",
+      "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==",
+      "dev": true
+    },
+    "tslog": {
+      "version": "4.8.2",
+      "resolved": "https://registry.npmjs.org/tslog/-/tslog-4.8.2.tgz",
+      "integrity": "sha512-eAKIRjxfSKYLs06r1wT7oou6Uv9VN6NW9g0JPidBlqQwPBBl5+84dm7r8zSOPVq1kyfEw1P6B3/FLSpZCorAgA==",
+      "dev": true
+    },
+    "tsort": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/tsort/-/tsort-0.0.1.tgz",
+      "integrity": "sha512-Tyrf5mxF8Ofs1tNoxA13lFeZ2Zrbd6cKbuH3V+MQ5sb6DtBj5FjrXVsRWT8YvNAQTqNoz66dz1WsbigI22aEnw==",
+      "dev": true
+    },
+    "tsutils": {
+      "version": "3.21.0",
+      "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.21.0.tgz",
+      "integrity": "sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==",
+      "dev": true,
+      "requires": {
+        "tslib": "^1.8.1"
+      }
+    },
+    "tunnel-agent": {
+      "version": "0.6.0",
+      "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz",
+      "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==",
+      "dev": true,
+      "requires": {
+        "safe-buffer": "^5.0.1"
+      }
+    },
+    "tweetnacl": {
+      "version": "0.14.5",
+      "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz",
+      "integrity": "sha512-KXXFFdAbFXY4geFIwoyNK+f5Z1b7swfXABfL7HXCmoIWMKU3dmS26672A4EeQtDzLKy7SXmfBu51JolvEKwtGA==",
+      "dev": true
+    },
+    "tweetnacl-util": {
+      "version": "0.15.1",
+      "resolved": "https://registry.npmjs.org/tweetnacl-util/-/tweetnacl-util-0.15.1.tgz",
+      "integrity": "sha512-RKJBIj8lySrShN4w6i/BonWp2Z/uxwC3h4y7xsRrpP59ZboCd0GpEVsOnMDYLMmKBpYhb5TgHzZXy7wTfYFBRw==",
+      "dev": true
+    },
+    "type": {
+      "version": "1.2.0",
+      "resolved": "https://registry.npmjs.org/type/-/type-1.2.0.tgz",
+      "integrity": "sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg==",
+      "dev": true
+    },
+    "type-check": {
+      "version": "0.4.0",
+      "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz",
+      "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==",
+      "dev": true,
+      "requires": {
+        "prelude-ls": "^1.2.1"
+      }
+    },
+    "type-detect": {
+      "version": "4.0.8",
+      "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz",
+      "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==",
+      "dev": true
+    },
+    "type-fest": {
+      "version": "0.21.3",
+      "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.21.3.tgz",
+      "integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==",
+      "dev": true
+    },
+    "type-is": {
+      "version": "1.6.18",
+      "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
+      "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
+      "dev": true,
+      "requires": {
+        "media-typer": "0.3.0",
+        "mime-types": "~2.1.24"
+      }
+    },
+    "typechain": {
+      "version": "8.1.1",
+      "resolved": "https://registry.npmjs.org/typechain/-/typechain-8.1.1.tgz",
+      "integrity": "sha512-uF/sUvnXTOVF2FHKhQYnxHk4su4JjZR8vr4mA2mBaRwHTbwh0jIlqARz9XJr1tA0l7afJGvEa1dTSi4zt039LQ==",
+      "dev": true,
+      "requires": {
+        "@types/prettier": "^2.1.1",
+        "debug": "^4.3.1",
+        "fs-extra": "^7.0.0",
+        "glob": "7.1.7",
+        "js-sha3": "^0.8.0",
+        "lodash": "^4.17.15",
+        "mkdirp": "^1.0.4",
+        "prettier": "^2.3.1",
+        "ts-command-line-args": "^2.2.0",
+        "ts-essentials": "^7.0.1"
+      },
+      "dependencies": {
+        "glob": {
+          "version": "7.1.7",
+          "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.7.tgz",
+          "integrity": "sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==",
+          "dev": true,
+          "requires": {
+            "fs.realpath": "^1.0.0",
+            "inflight": "^1.0.4",
+            "inherits": "2",
+            "minimatch": "^3.0.4",
+            "once": "^1.3.0",
+            "path-is-absolute": "^1.0.0"
+          }
+        },
+        "mkdirp": {
+          "version": "1.0.4",
+          "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz",
+          "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==",
+          "dev": true
+        }
+      }
+    },
+    "typed-array-length": {
+      "version": "1.0.4",
+      "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.4.tgz",
+      "integrity": "sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "for-each": "^0.3.3",
+        "is-typed-array": "^1.1.9"
+      }
+    },
+    "typedarray": {
+      "version": "0.0.6",
+      "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz",
+      "integrity": "sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==",
+      "dev": true
+    },
+    "typedarray-to-buffer": {
+      "version": "3.1.5",
+      "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz",
+      "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==",
+      "dev": true,
+      "requires": {
+        "is-typedarray": "^1.0.0"
+      }
+    },
+    "typescript": {
+      "version": "4.9.5",
+      "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.9.5.tgz",
+      "integrity": "sha512-1FXk9E2Hm+QzZQ7z+McJiHL4NW1F2EzMu9Nq9i3zAaGqibafqYwCVU6WyWAuyQRRzOlxou8xZSyXLEN8oKj24g==",
+      "dev": true
+    },
+    "typical": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/typical/-/typical-4.0.0.tgz",
+      "integrity": "sha512-VAH4IvQ7BDFYglMd7BPRDfLgxZZX4O4TFcRDA6EN5X7erNJJq+McIEp8np9aVtxrCJ6qx4GTYVfOWNjcqwZgRw==",
+      "dev": true
+    },
+    "uglify-js": {
+      "version": "3.17.4",
+      "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.17.4.tgz",
+      "integrity": "sha512-T9q82TJI9e/C1TAxYvfb16xO120tMVFZrGA3f9/P4424DNu6ypK103y0GPFVa17yotwSyZW5iYXgjYHkGrJW/g==",
+      "dev": true,
+      "optional": true
+    },
+    "ultron": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/ultron/-/ultron-1.1.1.tgz",
+      "integrity": "sha512-UIEXBNeYmKptWH6z8ZnqTeS8fV74zG0/eRU9VGkpzz+LIJNs8W/zM/L+7ctCkRrgbNnnR0xxw4bKOr0cW0N0Og==",
+      "dev": true
+    },
+    "unbox-primitive": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.2.tgz",
+      "integrity": "sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==",
+      "dev": true,
+      "requires": {
+        "call-bind": "^1.0.2",
+        "has-bigints": "^1.0.2",
+        "has-symbols": "^1.0.3",
+        "which-boxed-primitive": "^1.0.2"
+      }
+    },
+    "underscore": {
+      "version": "1.13.6",
+      "resolved": "https://registry.npmjs.org/underscore/-/underscore-1.13.6.tgz",
+      "integrity": "sha512-+A5Sja4HP1M08MaXya7p5LvjuM7K6q/2EaC0+iovj/wOcMsTzMvDFbasi/oSapiwOlt252IqsKqPjCl7huKS0A==",
+      "dev": true
+    },
+    "undici": {
+      "version": "5.27.2",
+      "resolved": "https://registry.npmjs.org/undici/-/undici-5.27.2.tgz",
+      "integrity": "sha512-iS857PdOEy/y3wlM3yRp+6SNQQ6xU0mmZcwRSriqk+et/cwWAtwmIGf6WkoDN2EK/AMdCO/dfXzIwi+rFMrjjQ==",
+      "dev": true,
+      "requires": {
+        "@fastify/busboy": "^2.0.0"
+      }
+    },
+    "universalify": {
+      "version": "0.1.2",
+      "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz",
+      "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==",
+      "dev": true
+    },
+    "unpipe": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
+      "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==",
+      "dev": true
+    },
+    "upper-case": {
+      "version": "1.1.3",
+      "resolved": "https://registry.npmjs.org/upper-case/-/upper-case-1.1.3.tgz",
+      "integrity": "sha512-WRbjgmYzgXkCV7zNVpy5YgrHgbBv126rMALQQMrmzOVC4GM2waQ9x7xtm8VU+1yF2kWyPzI9zbZ48n4vSxwfSA==",
+      "dev": true
+    },
+    "upper-case-first": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/upper-case-first/-/upper-case-first-1.1.2.tgz",
+      "integrity": "sha512-wINKYvI3Db8dtjikdAqoBbZoP6Q+PZUyfMR7pmwHzjC2quzSkUq5DmPrTtPEqHaz8AGtmsB4TqwapMTM1QAQOQ==",
+      "dev": true,
+      "requires": {
+        "upper-case": "^1.1.1"
+      }
+    },
+    "uri-js": {
+      "version": "4.4.1",
+      "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz",
+      "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==",
+      "dev": true,
+      "requires": {
+        "punycode": "^2.1.0"
+      }
+    },
+    "url-set-query": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/url-set-query/-/url-set-query-1.0.0.tgz",
+      "integrity": "sha512-3AChu4NiXquPfeckE5R5cGdiHCMWJx1dwCWOmWIL4KHAziJNOFIYJlpGFeKDvwLPHovZRCxK3cYlwzqI9Vp+Gg==",
+      "dev": true
+    },
+    "utf-8-validate": {
+      "version": "5.0.10",
+      "resolved": "https://registry.npmjs.org/utf-8-validate/-/utf-8-validate-5.0.10.tgz",
+      "integrity": "sha512-Z6czzLq4u8fPOyx7TU6X3dvUZVvoJmxSQ+IcrlmagKhilxlhZgxPK6C5Jqbkw1IDUmFTM+cz9QDnnLTwDz/2gQ==",
+      "dev": true,
+      "requires": {
+        "node-gyp-build": "^4.3.0"
+      }
+    },
+    "utf8": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/utf8/-/utf8-3.0.0.tgz",
+      "integrity": "sha512-E8VjFIQ/TyQgp+TZfS6l8yp/xWppSAHzidGiRrqe4bK4XP9pTRyKFgGJpO3SN7zdX4DeomTrwaseCHovfpFcqQ==",
+      "dev": true
+    },
+    "util": {
+      "version": "0.12.5",
+      "resolved": "https://registry.npmjs.org/util/-/util-0.12.5.tgz",
+      "integrity": "sha512-kZf/K6hEIrWHI6XqOFUiiMa+79wE/D8Q+NCNAWclkyg3b4d2k7s0QGepNjiABc+aR3N1PAyHL7p6UcLY6LmrnA==",
+      "dev": true,
+      "requires": {
+        "inherits": "^2.0.3",
+        "is-arguments": "^1.0.4",
+        "is-generator-function": "^1.0.7",
+        "is-typed-array": "^1.1.3",
+        "which-typed-array": "^1.1.2"
+      }
+    },
+    "util-deprecate": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
+      "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==",
+      "dev": true
+    },
+    "utils-merge": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
+      "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==",
+      "dev": true
+    },
+    "uuid": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/uuid/-/uuid-2.0.1.tgz",
+      "integrity": "sha512-nWg9+Oa3qD2CQzHIP4qKUqwNfzKn8P0LtFhotaCTFchsV7ZfDhAybeip/HZVeMIpZi9JgY1E3nUlwaCmZT1sEg==",
+      "dev": true
+    },
+    "v8-compile-cache-lib": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz",
+      "integrity": "sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==",
+      "dev": true
+    },
+    "validate-npm-package-license": {
+      "version": "3.0.4",
+      "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz",
+      "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==",
+      "dev": true,
+      "requires": {
+        "spdx-correct": "^3.0.0",
+        "spdx-expression-parse": "^3.0.0"
+      }
+    },
+    "varint": {
+      "version": "5.0.2",
+      "resolved": "https://registry.npmjs.org/varint/-/varint-5.0.2.tgz",
+      "integrity": "sha512-lKxKYG6H03yCZUpAGOPOsMcGxd1RHCu1iKvEHYDPmTyq2HueGhD73ssNBqqQWfvYs04G9iUFRvmAVLW20Jw6ow==",
+      "dev": true
+    },
+    "vary": {
+      "version": "1.1.2",
+      "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
+      "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==",
+      "dev": true
+    },
+    "verror": {
+      "version": "1.10.0",
+      "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz",
+      "integrity": "sha512-ZZKSmDAEFOijERBLkmYfJ+vmk3w+7hOLYDNkRCuRuMJGEmqYNCNLyBBFwWKVMhfwaEF3WOd0Zlw86U/WC/+nYw==",
+      "dev": true,
+      "requires": {
+        "assert-plus": "^1.0.0",
+        "core-util-is": "1.0.2",
+        "extsprintf": "^1.2.0"
+      }
+    },
+    "wcwidth": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/wcwidth/-/wcwidth-1.0.1.tgz",
+      "integrity": "sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg==",
+      "dev": true,
+      "requires": {
+        "defaults": "^1.0.3"
+      }
+    },
+    "web3": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3/-/web3-1.8.2.tgz",
+      "integrity": "sha512-92h0GdEHW9wqDICQQKyG4foZBYi0OQkyg4CRml2F7XBl/NG+fu9o6J19kzfFXzSBoA4DnJXbyRgj/RHZv5LRiw==",
+      "dev": true,
+      "requires": {
+        "web3-bzz": "1.8.2",
+        "web3-core": "1.8.2",
+        "web3-eth": "1.8.2",
+        "web3-eth-personal": "1.8.2",
+        "web3-net": "1.8.2",
+        "web3-shh": "1.8.2",
+        "web3-utils": "1.8.2"
+      }
+    },
+    "web3-bzz": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-bzz/-/web3-bzz-1.8.2.tgz",
+      "integrity": "sha512-1EEnxjPnFnvNWw3XeeKuTR8PBxYd0+XWzvaLK7OJC/Go9O8llLGxrxICbKV+8cgIE0sDRBxiYx02X+6OhoAQ9w==",
+      "dev": true,
+      "requires": {
+        "@types/node": "^12.12.6",
+        "got": "12.1.0",
+        "swarm-js": "^0.1.40"
+      },
+      "dependencies": {
+        "@types/node": {
+          "version": "12.20.55",
+          "resolved": "https://registry.npmjs.org/@types/node/-/node-12.20.55.tgz",
+          "integrity": "sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ==",
+          "dev": true
+        }
+      }
+    },
+    "web3-core": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-core/-/web3-core-1.8.2.tgz",
+      "integrity": "sha512-DJTVEAYcNqxkqruJE+Rxp3CIv0y5AZMwPHQmOkz/cz+MM75SIzMTc0AUdXzGyTS8xMF8h3YWMQGgGEy8SBf1PQ==",
+      "dev": true,
+      "requires": {
+        "@types/bn.js": "^5.1.0",
+        "@types/node": "^12.12.6",
+        "bignumber.js": "^9.0.0",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-core-requestmanager": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "dependencies": {
+        "@types/node": {
+          "version": "12.20.55",
+          "resolved": "https://registry.npmjs.org/@types/node/-/node-12.20.55.tgz",
+          "integrity": "sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ==",
+          "dev": true
+        },
+        "bignumber.js": {
+          "version": "9.1.1",
+          "resolved": "https://registry.npmjs.org/bignumber.js/-/bignumber.js-9.1.1.tgz",
+          "integrity": "sha512-pHm4LsMJ6lzgNGVfZHjMoO8sdoRhOzOH4MLmY65Jg70bpxCKu5iOHNJyfF6OyvYw7t8Fpf35RuzUyqnQsj8Vig==",
+          "dev": true
+        }
+      }
+    },
+    "web3-core-helpers": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-core-helpers/-/web3-core-helpers-1.8.2.tgz",
+      "integrity": "sha512-6B1eLlq9JFrfealZBomd1fmlq1o4A09vrCVQSa51ANoib/jllT3atZrRDr0zt1rfI7TSZTZBXdN/aTdeN99DWw==",
+      "dev": true,
+      "requires": {
+        "web3-eth-iban": "1.8.2",
+        "web3-utils": "1.8.2"
+      }
+    },
+    "web3-core-method": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-core-method/-/web3-core-method-1.8.2.tgz",
+      "integrity": "sha512-1qnr5mw5wVyULzLOrk4B+ryO3gfGjGd/fx8NR+J2xCGLf1e6OSjxT9vbfuQ3fErk/NjSTWWreieYWLMhaogcRA==",
+      "dev": true,
+      "requires": {
+        "@ethersproject/transactions": "^5.6.2",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-promievent": "1.8.2",
+        "web3-core-subscriptions": "1.8.2",
+        "web3-utils": "1.8.2"
+      }
+    },
+    "web3-core-promievent": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-core-promievent/-/web3-core-promievent-1.8.2.tgz",
+      "integrity": "sha512-nvkJWDVgoOSsolJldN33tKW6bKKRJX3MCPDYMwP5SUFOA/mCzDEoI88N0JFofDTXkh1k7gOqp1pvwi9heuaxGg==",
+      "dev": true,
+      "requires": {
+        "eventemitter3": "4.0.4"
+      }
+    },
+    "web3-core-requestmanager": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-core-requestmanager/-/web3-core-requestmanager-1.8.2.tgz",
+      "integrity": "sha512-p1d090RYs5Mu7DK1yyc3GCBVZB/03rBtFhYFoS2EruGzOWs/5Q0grgtpwS/DScdRAm8wB8mYEBhY/RKJWF6B2g==",
+      "dev": true,
+      "requires": {
+        "util": "^0.12.5",
+        "web3-core-helpers": "1.8.2",
+        "web3-providers-http": "1.8.2",
+        "web3-providers-ipc": "1.8.2",
+        "web3-providers-ws": "1.8.2"
+      }
+    },
+    "web3-core-subscriptions": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-core-subscriptions/-/web3-core-subscriptions-1.8.2.tgz",
+      "integrity": "sha512-vXQogHDmAIQcKpXvGiMddBUeP9lnKgYF64+yQJhPNE5PnWr1sAibXuIPV7mIPihpFr/n/DORRj6Wh1pUv9zaTw==",
+      "dev": true,
+      "requires": {
+        "eventemitter3": "4.0.4",
+        "web3-core-helpers": "1.8.2"
+      }
+    },
+    "web3-eth": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth/-/web3-eth-1.8.2.tgz",
+      "integrity": "sha512-JoTiWWc4F4TInpbvDUGb0WgDYJsFhuIjJlinc5ByjWD88Gvh+GKLsRjjFdbqe5YtwIGT4NymwoC5LQd1K6u/QQ==",
+      "dev": true,
+      "requires": {
+        "web3-core": "1.8.2",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-core-subscriptions": "1.8.2",
+        "web3-eth-abi": "1.8.2",
+        "web3-eth-accounts": "1.8.2",
+        "web3-eth-contract": "1.8.2",
+        "web3-eth-ens": "1.8.2",
+        "web3-eth-iban": "1.8.2",
+        "web3-eth-personal": "1.8.2",
+        "web3-net": "1.8.2",
+        "web3-utils": "1.8.2"
+      }
+    },
+    "web3-eth-abi": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth-abi/-/web3-eth-abi-1.8.2.tgz",
+      "integrity": "sha512-Om9g3kaRNjqiNPAgKwGT16y+ZwtBzRe4ZJFGjLiSs6v5I7TPNF+rRMWuKnR6jq0azQZDj6rblvKFMA49/k48Og==",
+      "dev": true,
+      "requires": {
+        "@ethersproject/abi": "^5.6.3",
+        "web3-utils": "1.8.2"
+      }
+    },
+    "web3-eth-accounts": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth-accounts/-/web3-eth-accounts-1.8.2.tgz",
+      "integrity": "sha512-c367Ij63VCz9YdyjiHHWLFtN85l6QghgwMQH2B1eM/p9Y5lTlTX7t/Eg/8+f1yoIStXbk2w/PYM2lk+IkbqdLA==",
+      "dev": true,
+      "requires": {
+        "@ethereumjs/common": "2.5.0",
+        "@ethereumjs/tx": "3.3.2",
+        "eth-lib": "0.2.8",
+        "ethereumjs-util": "^7.1.5",
+        "scrypt-js": "^3.0.1",
+        "uuid": "^9.0.0",
+        "web3-core": "1.8.2",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "dependencies": {
+        "bn.js": {
+          "version": "4.12.0",
+          "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz",
+          "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==",
+          "dev": true
+        },
+        "eth-lib": {
+          "version": "0.2.8",
+          "resolved": "https://registry.npmjs.org/eth-lib/-/eth-lib-0.2.8.tgz",
+          "integrity": "sha512-ArJ7x1WcWOlSpzdoTBX8vkwlkSQ85CjjifSZtV4co64vWxSV8geWfPI9x4SVYu3DSxnX4yWFVTtGL+j9DUFLNw==",
+          "dev": true,
+          "requires": {
+            "bn.js": "^4.11.6",
+            "elliptic": "^6.4.0",
+            "xhr-request-promise": "^0.1.2"
+          }
+        },
+        "uuid": {
+          "version": "9.0.0",
+          "resolved": "https://registry.npmjs.org/uuid/-/uuid-9.0.0.tgz",
+          "integrity": "sha512-MXcSTerfPa4uqyzStbRoTgt5XIe3x5+42+q1sDuy3R5MDk66URdLMOZe5aPX/SQd+kuYAh0FdP/pO28IkQyTeg==",
+          "dev": true
+        }
+      }
+    },
+    "web3-eth-contract": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth-contract/-/web3-eth-contract-1.8.2.tgz",
+      "integrity": "sha512-ID5A25tHTSBNwOPjiXSVzxruz006ULRIDbzWTYIFTp7NJ7vXu/kynKK2ag/ObuTqBpMbobP8nXcA9b5EDkIdQA==",
+      "dev": true,
+      "requires": {
+        "@types/bn.js": "^5.1.0",
+        "web3-core": "1.8.2",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-core-promievent": "1.8.2",
+        "web3-core-subscriptions": "1.8.2",
+        "web3-eth-abi": "1.8.2",
+        "web3-utils": "1.8.2"
+      }
+    },
+    "web3-eth-ens": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth-ens/-/web3-eth-ens-1.8.2.tgz",
+      "integrity": "sha512-PWph7C/CnqdWuu1+SH4U4zdrK4t2HNt0I4XzPYFdv9ugE8EuojselioPQXsVGvjql+Nt3jDLvQvggPqlMbvwRw==",
+      "dev": true,
+      "requires": {
+        "content-hash": "^2.5.2",
+        "eth-ens-namehash": "2.0.8",
+        "web3-core": "1.8.2",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-promievent": "1.8.2",
+        "web3-eth-abi": "1.8.2",
+        "web3-eth-contract": "1.8.2",
+        "web3-utils": "1.8.2"
+      }
+    },
+    "web3-eth-iban": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth-iban/-/web3-eth-iban-1.8.2.tgz",
+      "integrity": "sha512-h3vNblDWkWMuYx93Q27TAJz6lhzpP93EiC3+45D6xoz983p6si773vntoQ+H+5aZhwglBtoiBzdh7PSSOnP/xQ==",
+      "dev": true,
+      "requires": {
+        "bn.js": "^5.2.1",
+        "web3-utils": "1.8.2"
+      }
+    },
+    "web3-eth-personal": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-eth-personal/-/web3-eth-personal-1.8.2.tgz",
+      "integrity": "sha512-Vg4HfwCr7doiUF/RC+Jz0wT4+cYaXcOWMAW2AHIjHX6Z7Xwa8nrURIeQgeEE62qcEHAzajyAdB1u6bJyTfuCXw==",
+      "dev": true,
+      "requires": {
+        "@types/node": "^12.12.6",
+        "web3-core": "1.8.2",
+        "web3-core-helpers": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-net": "1.8.2",
+        "web3-utils": "1.8.2"
+      },
+      "dependencies": {
+        "@types/node": {
+          "version": "12.20.55",
+          "resolved": "https://registry.npmjs.org/@types/node/-/node-12.20.55.tgz",
+          "integrity": "sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ==",
+          "dev": true
+        }
+      }
+    },
+    "web3-net": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-net/-/web3-net-1.8.2.tgz",
+      "integrity": "sha512-1itkDMGmbgb83Dg9nporFes9/fxsU7smJ3oRXlFkg4ZHn8YJyP1MSQFPJWWwSc+GrcCFt4O5IrUTvEkHqE3xag==",
+      "dev": true,
+      "requires": {
+        "web3-core": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-utils": "1.8.2"
+      }
+    },
+    "web3-providers-http": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-providers-http/-/web3-providers-http-1.8.2.tgz",
+      "integrity": "sha512-2xY94IIEQd16+b+vIBF4IC1p7GVaz9q4EUFscvMUjtEq4ru4Atdzjs9GP+jmcoo49p70II0UV3bqQcz0TQfVyQ==",
+      "dev": true,
+      "requires": {
+        "abortcontroller-polyfill": "^1.7.3",
+        "cross-fetch": "^3.1.4",
+        "es6-promise": "^4.2.8",
+        "web3-core-helpers": "1.8.2"
+      }
+    },
+    "web3-providers-ipc": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-providers-ipc/-/web3-providers-ipc-1.8.2.tgz",
+      "integrity": "sha512-p6fqKVGFg+WiXGHWnB1hu43PbvPkDHTz4RgoEzbXugv5rtv5zfYLqm8Ba6lrJOS5ks9kGKR21a0y3NzE3u7V4w==",
+      "dev": true,
+      "requires": {
+        "oboe": "2.1.5",
+        "web3-core-helpers": "1.8.2"
+      }
+    },
+    "web3-providers-ws": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-providers-ws/-/web3-providers-ws-1.8.2.tgz",
+      "integrity": "sha512-3s/4K+wHgbiN+Zrp9YjMq2eqAF6QGABw7wFftPdx+m5hWImV27/MoIx57c6HffNRqZXmCHnfWWFCNHHsi7wXnA==",
+      "dev": true,
+      "requires": {
+        "eventemitter3": "4.0.4",
+        "web3-core-helpers": "1.8.2",
+        "websocket": "^1.0.32"
+      }
+    },
+    "web3-shh": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-shh/-/web3-shh-1.8.2.tgz",
+      "integrity": "sha512-uZ+3MAoNcaJsXXNCDnizKJ5viBNeHOFYsCbFhV755Uu52FswzTOw6DtE7yK9nYXMtIhiSgi7nwl1RYzP8pystw==",
+      "dev": true,
+      "requires": {
+        "web3-core": "1.8.2",
+        "web3-core-method": "1.8.2",
+        "web3-core-subscriptions": "1.8.2",
+        "web3-net": "1.8.2"
+      }
+    },
+    "web3-utils": {
+      "version": "1.8.2",
+      "resolved": "https://registry.npmjs.org/web3-utils/-/web3-utils-1.8.2.tgz",
+      "integrity": "sha512-v7j6xhfLQfY7xQDrUP0BKbaNrmZ2/+egbqP9q3KYmOiPpnvAfol+32slgL0WX/5n8VPvKCK5EZ1HGrAVICSToA==",
+      "dev": true,
+      "requires": {
+        "bn.js": "^5.2.1",
+        "ethereum-bloom-filters": "^1.0.6",
+        "ethereumjs-util": "^7.1.0",
+        "ethjs-unit": "0.1.6",
+        "number-to-bn": "1.7.0",
+        "randombytes": "^2.1.0",
+        "utf8": "3.0.0"
+      }
+    },
+    "webidl-conversions": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz",
+      "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==",
+      "dev": true
+    },
+    "websocket": {
+      "version": "1.0.34",
+      "resolved": "https://registry.npmjs.org/websocket/-/websocket-1.0.34.tgz",
+      "integrity": "sha512-PRDso2sGwF6kM75QykIesBijKSVceR6jL2G8NGYyq2XrItNC2P5/qL5XeR056GhA+Ly7JMFvJb9I312mJfmqnQ==",
+      "dev": true,
+      "requires": {
+        "bufferutil": "^4.0.1",
+        "debug": "^2.2.0",
+        "es5-ext": "^0.10.50",
+        "typedarray-to-buffer": "^3.1.5",
+        "utf-8-validate": "^5.0.2",
+        "yaeti": "^0.0.6"
+      },
+      "dependencies": {
+        "debug": {
+          "version": "2.6.9",
+          "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+          "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+          "dev": true,
+          "requires": {
+            "ms": "2.0.0"
+          }
+        },
+        "ms": {
+          "version": "2.0.0",
+          "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+          "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
+          "dev": true
+        }
+      }
+    },
+    "whatwg-url": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz",
+      "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==",
+      "dev": true,
+      "requires": {
+        "tr46": "~0.0.3",
+        "webidl-conversions": "^3.0.0"
+      }
+    },
+    "which": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
+      "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==",
+      "dev": true,
+      "requires": {
+        "isexe": "^2.0.0"
+      }
+    },
+    "which-boxed-primitive": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz",
+      "integrity": "sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==",
+      "dev": true,
+      "requires": {
+        "is-bigint": "^1.0.1",
+        "is-boolean-object": "^1.1.0",
+        "is-number-object": "^1.0.4",
+        "is-string": "^1.0.5",
+        "is-symbol": "^1.0.3"
+      }
+    },
+    "which-module": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/which-module/-/which-module-1.0.0.tgz",
+      "integrity": "sha512-F6+WgncZi/mJDrammbTuHe1q0R5hOXv/mBaiNA2TCNT/LTHusX0V+CJnj9XT8ki5ln2UZyyddDgHfCzyrOH7MQ==",
+      "dev": true
+    },
+    "which-pm-runs": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/which-pm-runs/-/which-pm-runs-1.1.0.tgz",
+      "integrity": "sha512-n1brCuqClxfFfq/Rb0ICg9giSZqCS+pLtccdag6C2HyufBrh3fBOiy9nb6ggRMvWOVH5GrdJskj5iGTZNxd7SA==",
+      "dev": true
+    },
+    "which-typed-array": {
+      "version": "1.1.9",
+      "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.9.tgz",
+      "integrity": "sha512-w9c4xkx6mPidwp7180ckYWfMmvxpjlZuIudNtDf4N/tTAUB8VJbX25qZoAsrtGuYNnGw3pa0AXgbGKRB8/EceA==",
+      "dev": true,
+      "requires": {
+        "available-typed-arrays": "^1.0.5",
+        "call-bind": "^1.0.2",
+        "for-each": "^0.3.3",
+        "gopd": "^1.0.1",
+        "has-tostringtag": "^1.0.0",
+        "is-typed-array": "^1.1.10"
+      }
+    },
+    "window-size": {
+      "version": "0.2.0",
+      "resolved": "https://registry.npmjs.org/window-size/-/window-size-0.2.0.tgz",
+      "integrity": "sha512-UD7d8HFA2+PZsbKyaOCEy8gMh1oDtHgJh1LfgjQ4zVXmYjAT/kvz3PueITKuqDiIXQe7yzpPnxX3lNc+AhQMyw==",
+      "dev": true
+    },
+    "word-wrap": {
+      "version": "1.2.5",
+      "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz",
+      "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==",
+      "dev": true
+    },
+    "wordwrap": {
+      "version": "1.0.0",
+      "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz",
+      "integrity": "sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==",
+      "dev": true
+    },
+    "wordwrapjs": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/wordwrapjs/-/wordwrapjs-4.0.1.tgz",
+      "integrity": "sha512-kKlNACbvHrkpIw6oPeYDSmdCTu2hdMHoyXLTcUKala++lx5Y+wjJ/e474Jqv5abnVmwxw08DiTuHmw69lJGksA==",
+      "dev": true,
+      "requires": {
+        "reduce-flatten": "^2.0.0",
+        "typical": "^5.2.0"
+      },
+      "dependencies": {
+        "typical": {
+          "version": "5.2.0",
+          "resolved": "https://registry.npmjs.org/typical/-/typical-5.2.0.tgz",
+          "integrity": "sha512-dvdQgNDNJo+8B2uBQoqdb11eUCE1JQXhvjC/CZtgvZseVd5TYMXnq0+vuUemXbd/Se29cTaUuPX3YIc2xgbvIg==",
+          "dev": true
+        }
+      }
+    },
+    "workerpool": {
+      "version": "6.2.1",
+      "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.2.1.tgz",
+      "integrity": "sha512-ILEIE97kDZvF9Wb9f6h5aXK4swSlKGUcOEGiIYb2OOu/IrDU9iwj0fD//SsA6E5ibwJxpEvhullJY4Sl4GcpAw==",
+      "dev": true
+    },
+    "wrap-ansi": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz",
+      "integrity": "sha512-vAaEaDM946gbNpH5pLVNR+vX2ht6n0Bt3GXwVB1AuAqZosOvHNF3P7wDnh8KLkSqgUh0uh77le7Owgoz+Z9XBw==",
+      "dev": true,
+      "requires": {
+        "string-width": "^1.0.1",
+        "strip-ansi": "^3.0.1"
+      },
+      "dependencies": {
+        "ansi-regex": {
+          "version": "2.1.1",
+          "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
+          "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==",
+          "dev": true
+        },
+        "is-fullwidth-code-point": {
+          "version": "1.0.0",
+          "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz",
+          "integrity": "sha512-1pqUqRjkhPJ9miNq9SwMfdvi6lBJcd6eFxvfaivQhaH3SgisfiuudvFntdKOmxuee/77l+FPjKrQjWvmPjWrRw==",
+          "dev": true,
+          "requires": {
+            "number-is-nan": "^1.0.0"
+          }
+        },
+        "string-width": {
+          "version": "1.0.2",
+          "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz",
+          "integrity": "sha512-0XsVpQLnVCXHJfyEs8tC0zpTVIr5PKKsQtkT29IwupnPTjtPmQ3xT/4yCREF9hYkV/3M3kzcUTSAZT6a6h81tw==",
+          "dev": true,
+          "requires": {
+            "code-point-at": "^1.0.0",
+            "is-fullwidth-code-point": "^1.0.0",
+            "strip-ansi": "^3.0.0"
+          }
+        },
+        "strip-ansi": {
+          "version": "3.0.1",
+          "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
+          "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==",
+          "dev": true,
+          "requires": {
+            "ansi-regex": "^2.0.0"
+          }
+        }
+      }
+    },
+    "wrappy": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
+      "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==",
+      "dev": true
+    },
+    "ws": {
+      "version": "7.4.6",
+      "resolved": "https://registry.npmjs.org/ws/-/ws-7.4.6.tgz",
+      "integrity": "sha512-YmhHDO4MzaDLB+M9ym/mDA5z0naX8j7SIlT8f8z+I0VtzsRbekxEutHSme7NPS2qE8StCYQNUnfWdXta/Yu85A==",
+      "devOptional": true,
+      "requires": {}
+    },
+    "xhr": {
+      "version": "2.6.0",
+      "resolved": "https://registry.npmjs.org/xhr/-/xhr-2.6.0.tgz",
+      "integrity": "sha512-/eCGLb5rxjx5e3mF1A7s+pLlR6CGyqWN91fv1JgER5mVWg1MZmlhBvy9kjcsOdRk8RrIujotWyJamfyrp+WIcA==",
+      "dev": true,
+      "requires": {
+        "global": "~4.4.0",
+        "is-function": "^1.0.1",
+        "parse-headers": "^2.0.0",
+        "xtend": "^4.0.0"
+      }
+    },
+    "xhr-request": {
+      "version": "1.1.0",
+      "resolved": "https://registry.npmjs.org/xhr-request/-/xhr-request-1.1.0.tgz",
+      "integrity": "sha512-Y7qzEaR3FDtL3fP30k9wO/e+FBnBByZeybKOhASsGP30NIkRAAkKD/sCnLvgEfAIEC1rcmK7YG8f4oEnIrrWzA==",
+      "dev": true,
+      "requires": {
+        "buffer-to-arraybuffer": "^0.0.5",
+        "object-assign": "^4.1.1",
+        "query-string": "^5.0.1",
+        "simple-get": "^2.7.0",
+        "timed-out": "^4.0.1",
+        "url-set-query": "^1.0.0",
+        "xhr": "^2.0.4"
+      }
+    },
+    "xhr-request-promise": {
+      "version": "0.1.3",
+      "resolved": "https://registry.npmjs.org/xhr-request-promise/-/xhr-request-promise-0.1.3.tgz",
+      "integrity": "sha512-YUBytBsuwgitWtdRzXDDkWAXzhdGB8bYm0sSzMPZT7Z2MBjMSTHFsyCT1yCRATY+XC69DUrQraRAEgcoCRaIPg==",
+      "dev": true,
+      "requires": {
+        "xhr-request": "^1.1.0"
+      }
+    },
+    "xmlhttprequest": {
+      "version": "1.8.0",
+      "resolved": "https://registry.npmjs.org/xmlhttprequest/-/xmlhttprequest-1.8.0.tgz",
+      "integrity": "sha512-58Im/U0mlVBLM38NdZjHyhuMtCqa61469k2YP/AaPbvCoV9aQGUpbJBj1QRm2ytRiVQBD/fsw7L2bJGDVQswBA==",
+      "dev": true
+    },
+    "xtend": {
+      "version": "4.0.2",
+      "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz",
+      "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==",
+      "dev": true
+    },
+    "y18n": {
+      "version": "3.2.2",
+      "resolved": "https://registry.npmjs.org/y18n/-/y18n-3.2.2.tgz",
+      "integrity": "sha512-uGZHXkHnhF0XeeAPgnKfPv1bgKAYyVvmNL1xlKsPYZPaIHxGti2hHqvOCQv71XMsLxu1QjergkqogUnms5D3YQ==",
+      "dev": true
+    },
+    "yaeti": {
+      "version": "0.0.6",
+      "resolved": "https://registry.npmjs.org/yaeti/-/yaeti-0.0.6.tgz",
+      "integrity": "sha512-MvQa//+KcZCUkBTIC9blM+CU9J2GzuTytsOUwf2lidtvkx/6gnEp1QvJv34t9vdjhFmha/mUiNDbN0D0mJWdug==",
+      "dev": true
+    },
+    "yallist": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz",
+      "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==",
+      "dev": true
+    },
+    "yaml": {
+      "version": "1.10.2",
+      "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.2.tgz",
+      "integrity": "sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==",
+      "dev": true
+    },
+    "yargs": {
+      "version": "4.8.1",
+      "resolved": "https://registry.npmjs.org/yargs/-/yargs-4.8.1.tgz",
+      "integrity": "sha512-LqodLrnIDM3IFT+Hf/5sxBnEGECrfdC1uIbgZeJmESCSo4HoCAaKEus8MylXHAkdacGc0ye+Qa+dpkuom8uVYA==",
+      "dev": true,
+      "requires": {
+        "cliui": "^3.2.0",
+        "decamelize": "^1.1.1",
+        "get-caller-file": "^1.0.1",
+        "lodash.assign": "^4.0.3",
+        "os-locale": "^1.4.0",
+        "read-pkg-up": "^1.0.1",
+        "require-directory": "^2.1.1",
+        "require-main-filename": "^1.0.1",
+        "set-blocking": "^2.0.0",
+        "string-width": "^1.0.1",
+        "which-module": "^1.0.0",
+        "window-size": "^0.2.0",
+        "y18n": "^3.2.1",
+        "yargs-parser": "^2.4.1"
+      },
+      "dependencies": {
+        "ansi-regex": {
+          "version": "2.1.1",
+          "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz",
+          "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==",
+          "dev": true
+        },
+        "is-fullwidth-code-point": {
+          "version": "1.0.0",
+          "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz",
+          "integrity": "sha512-1pqUqRjkhPJ9miNq9SwMfdvi6lBJcd6eFxvfaivQhaH3SgisfiuudvFntdKOmxuee/77l+FPjKrQjWvmPjWrRw==",
+          "dev": true,
+          "requires": {
+            "number-is-nan": "^1.0.0"
+          }
+        },
+        "string-width": {
+          "version": "1.0.2",
+          "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz",
+          "integrity": "sha512-0XsVpQLnVCXHJfyEs8tC0zpTVIr5PKKsQtkT29IwupnPTjtPmQ3xT/4yCREF9hYkV/3M3kzcUTSAZT6a6h81tw==",
+          "dev": true,
+          "requires": {
+            "code-point-at": "^1.0.0",
+            "is-fullwidth-code-point": "^1.0.0",
+            "strip-ansi": "^3.0.0"
+          }
+        },
+        "strip-ansi": {
+          "version": "3.0.1",
+          "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz",
+          "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==",
+          "dev": true,
+          "requires": {
+            "ansi-regex": "^2.0.0"
+          }
+        }
+      }
+    },
+    "yargs-parser": {
+      "version": "2.4.1",
+      "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-2.4.1.tgz",
+      "integrity": "sha512-9pIKIJhnI5tonzG6OnCFlz/yln8xHYcGl+pn3xR0Vzff0vzN1PbNRaelgfgRUwZ3s4i3jvxT9WhmUGL4whnasA==",
+      "dev": true,
+      "requires": {
+        "camelcase": "^3.0.0",
+        "lodash.assign": "^4.0.6"
+      }
+    },
+    "yargs-unparser": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-2.0.0.tgz",
+      "integrity": "sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==",
+      "dev": true,
+      "requires": {
+        "camelcase": "^6.0.0",
+        "decamelize": "^4.0.0",
+        "flat": "^5.0.2",
+        "is-plain-obj": "^2.1.0"
+      },
+      "dependencies": {
+        "camelcase": {
+          "version": "6.3.0",
+          "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz",
+          "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==",
+          "dev": true
+        },
+        "decamelize": {
+          "version": "4.0.0",
+          "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-4.0.0.tgz",
+          "integrity": "sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==",
+          "dev": true
+        }
+      }
+    },
+    "yesno": {
+      "version": "0.4.0",
+      "resolved": "https://registry.npmjs.org/yesno/-/yesno-0.4.0.tgz",
+      "integrity": "sha512-tdBxmHvbXPBKYIg81bMCB7bVeDmHkRzk5rVJyYYXurwKkHq/MCd8rz4HSJUP7hW0H2NlXiq8IFiWvYKEHhlotA==",
+      "dev": true
+    },
+    "yn": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz",
+      "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==",
+      "dev": true
+    },
+    "yocto-queue": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz",
+      "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==",
+      "dev": true
+    },
+    "zksync-web3": {
+      "version": "0.8.1",
+      "resolved": "https://registry.npmjs.org/zksync-web3/-/zksync-web3-0.8.1.tgz",
+      "integrity": "sha512-1A4aHPQ3MyuGjpv5X/8pVEN+MdZqMjfVmiweQSRjOlklXYu65wT9BGEOtCmMs5d3gIvLp4ssfTeuR5OCKOD2kw==",
+      "dev": true,
+      "requires": {}
+    }
+  }
+}
diff --git a/forks/passport/package.json b/forks/passport/package.json
new file mode 100644
index 00000000..10b219b0
--- /dev/null
+++ b/forks/passport/package.json
@@ -0,0 +1,94 @@
+{
+  "name": "root",
+  "private": true,
+  "workspaces": [
+    "src"
+  ],
+  "license": "Apache-2.0",
+  "scripts": {
+    "build": "yarn clean && yarn compile && yarn adapter",
+    "clean": "rimraf src/artifacts && rimraf src/gen && rimraf cache",
+    "compile": "hardhat --max-memory 4096 compile",
+    "adapter": "yarn adapter:gen && yarn adapter:build",
+    "adapter:gen": "rimraf src/gen/typechain && typechain --target ethers-v5 --out-dir src/gen/typechain './src/artifacts/contracts/**/*[^dbg].json'",
+    "adapter:build": "rimraf src/gen/adapter && tsc -p ./src/tsconfig.adapter.json",
+    "test": "yarn build && hardhat test",
+    "quicktest": "hardhat test",
+    "precoverage": "yarn build",
+    "coverage": "COVERAGE=true NET_ID=1 hardhat coverage --network coverage",
+    "benchmark": "BENCHMARK=true yarn test",
+    "lint": "yarn lint:ts && yarn lint:sol",
+    "lint:fix": "yarn lint:ts:fix && yarn lint:sol:fix",
+    "lint:sol": "solhint './src/contracts/**/*.sol'",
+    "lint:sol:fix": "solhint './src/contracts/**/*.sol' --fix",
+    "lint:ts": "eslint -c .eslintrc.js './**/*.ts'",
+    "lint:ts:fix": "eslint -c .eslintrc.js --fix './**/*.ts'",
+    "format": "prettier --write ./**/*.ts",
+    "start:ganache": "ganache --chain.chainId ${npm_package_config_ganacheChainID} --chain.networkId ${npm_package_config_ganacheChainID} --server.port ${npm_package_config_ganachePort} --miner.blockGasLimit ${npm_package_config_ganacheGasLimit} --miner.defaultGasPrice ${npm_package_config_ganacheGasPrice} --wallet.defaultBalance ${npm_package_config_etherBalance} --wallet.mnemonic \"${npm_package_config_mnemonic}\" ${npm_package_config_extra}",
+    "start:ganache:verbose": "yarn run start:ganache --verbose",
+    "stop:ganache": "ps aux | grep ganache | grep -v grep | awk '{print $2}' | xargs kill -9",
+    "deploy:ganache": "hardhat run --network ganache utils/deploy-contracts.ts",
+    "deploy": "hardhat run utils/deploy-contracts.ts --network",
+    "deployToDevnet": "npx hardhat run scripts/DeployWalletContracts.ts --network devnet",
+    "deployToTestnet": "npx hardhat run scripts/DeployWalletContracts.ts --network testnet",
+    "verify": "hardhat verify --network",
+    "release": "yarn publish src"
+  },
+  "husky": {
+    "hooks": {
+      "pre-commit": "yarn lint",
+      "pre-push": "yarn lint && yarn test"
+    }
+  },
+  "devDependencies": {
+    "@0xsequence/deployer": "^0.21.5",
+    "@ethersproject/abi": "^5.7.0",
+    "@ethersproject/providers": "^5.7.2",
+    "@nomicfoundation/hardhat-chai-matchers": "^1.0.6",
+    "@nomicfoundation/hardhat-network-helpers": "^1.0.8",
+    "@nomiclabs/hardhat-ethers": "^2.0.2",
+    "@nomiclabs/hardhat-etherscan": "^3.1.0",
+    "@nomiclabs/hardhat-truffle5": "^2.0.0",
+    "@nomiclabs/hardhat-web3": "^2.0.0",
+    "@openzeppelin/contracts": "^4.9.3",
+    "@tenderly/hardhat-tenderly": "^1.0.11",
+    "@typechain/ethers-v5": "^10.1.1",
+    "@types/chai-as-promised": "^7.1.0",
+    "@types/chai-string": "^1.4.1",
+    "@types/mocha": "^10.0.0",
+    "@typescript-eslint/eslint-plugin": "^5.42.1",
+    "@typescript-eslint/parser": "^5.42.1",
+    "bn-chai": "^1.0.1",
+    "chai-as-promised": "^7.1.1",
+    "chai-bignumber": "^3.0.0",
+    "chai-string": "^1.5.0",
+    "child_process": "^1.0.2",
+    "dotenv": "^16.0.3",
+    "eslint": "^8.27.0",
+    "eslint-config-prettier": "^8.1.0",
+    "eslint-plugin-import": "^2.22.0",
+    "eslint-plugin-prettier": "^4.2.1",
+    "ethers": "^5.7.2",
+    "ganache": "^7.5.0",
+    "hardhat": "2.12.2",
+    "hardhat-gas-reporter": "^1.0.9",
+    "husky": "^4.2.3",
+    "rimraf": "^3.0.2",
+    "solhint": "^3.3.4",
+    "solidity-coverage": "^0.8.5",
+    "solmate": "^6.7.0",
+    "ts-node": "^10.9.1",
+    "typechain": "^8.1.1",
+    "typescript": "^4.8.4",
+    "yesno": "^0.4.0"
+  },
+  "config": {
+    "mnemonic": "test test test test test test test test test test test junk",
+    "ganacheChainID": 127001,
+    "ganachePort": 8545,
+    "ganacheGasLimit": "0xfffffffffff",
+    "ganacheGasPrice": "2",
+    "etherBalance": "100000",
+    "extra": ""
+  }
+}
diff --git a/forks/passport/scripts/DeployWalletContracts.ts b/forks/passport/scripts/DeployWalletContracts.ts
new file mode 100644
index 00000000..b44acb44
--- /dev/null
+++ b/forks/passport/scripts/DeployWalletContracts.ts
@@ -0,0 +1,191 @@
+import * as path from 'path'
+import * as fs from 'fs'
+import * as hre from 'hardhat'
+import { SignerWithAddress } from '@nomiclabs/hardhat-ethers/signers'
+import { ethers } from 'ethers'
+import { ethers as hardhat } from 'hardhat'
+import { expect } from 'chai'
+
+require('dotenv').config()
+
+const outputPath = path.join(__dirname, './deploy_output.json')
+let deployer: SignerWithAddress
+
+async function deploy() {
+  // TODO: We should obtain an API key for blockscout and uncomment
+  // the following lines before a mainnet deployment
+  // if (typeof process.env.ETHERSCAN_API_KEY === 'undefined') {
+  //     throw new Error('Etherscan API KEY has not been defined');
+  // }
+
+  // /dev/imx-evm-relayer/EOA_SUBMITTER
+  let relayerSubmitterEOAPubKey = '0xBC52cE84FceFd2D941D1127608D6Cf598f9633d3'
+  // /dev/imx-evm-relayer/IMMUTABLE_SIGNER_CONTRACT
+  let immutableSignerPubKey = '0x1cE50560686b1297B6311F36B47dbe5d6E04D0f8'
+  // Administration accounts
+  let multiCallAdminPubKey = '0x575be326c482a487add43974e0eaf232e3366e13'
+  let factoryAdminPubKey = '0xddb70ddcd14dbd57ae18ec591f47454e4fc818bb'
+  let walletImplLocatorAdmin = '0xb49c99a17776c10350c2be790e13d4d8dfb1c578'
+  let signerRootAdminPubKey = '0x65af83f71a05d7f6d06ef9a57c9294b4128ccc2c'
+  let signerAdminPubKey = '0x69d09644159e7327dbfd0af9a66f8e332c593e79'
+
+  // Required private keys:
+  // 1. Deployer
+  // 2. walletImplLocatorChanger
+  const [contractDeployer, walletImplLocatorImplChanger] = await hardhat.getSigners()
+  deployer = contractDeployer
+
+  // TOTAL deployment cost = 0.009766773 GWEI = 0.000000000009766773 ETHER
+  // Deployments with esimated gas costs (GWEI)
+  console.log('Deploying contracts...')
+  // 1. Deploy multi call deploy
+  // EST gas cost: 0.001561956
+  const multiCallDeploy = await deployMultiCallDeploy(multiCallAdminPubKey, relayerSubmitterEOAPubKey)
+  await multiCallDeploy.deployTransaction.wait()
+  console.log('Multi Call Deploy deployed to: ', multiCallDeploy.address)
+
+  // 2. Deploy factory with multi call deploy address as deployer role EST
+  // EST gas cost: 0.001239658
+  const factory = await deployFactory(factoryAdminPubKey, multiCallDeploy.address)
+  await factory.deployTransaction.wait()
+  console.log(`Factory deployed to: ${factory.address} with hash ${factory.deployTransaction.hash}`)
+
+  // 3. Deploy wallet impl locator
+  // EST gas cost: 0.001021586
+  const walletImplLocator = await deployWalletImplLocator(walletImplLocatorAdmin, walletImplLocatorImplChanger.address)
+  await walletImplLocator.deployTransaction.wait()
+  console.log(
+    `Wallet Implementation Locator deployed to: ${walletImplLocator.address} with hash ${walletImplLocator.deployTransaction.hash}`
+  )
+
+  // 4. Deploy startup wallet impl
+  // EST gas cost: 0.000175659
+  const startupWalletImpl = await deployStartUp(walletImplLocator.address)
+  await startupWalletImpl.deployTransaction.wait()
+  console.log(
+    `Startup Wallet Impl deployed to: ${startupWalletImpl.address} with hash ${startupWalletImpl.deployTransaction.hash}`
+  )
+
+  // 5. Deploy main module dynamic auth
+  // EST gas cost: 0.003911813
+  const mainModule = await deployMainModule(factory.address, startupWalletImpl.address)
+  await mainModule.deployTransaction.wait()
+  console.log(`Main Module Dynamic Auth deployed to: ${mainModule.address} with hash ${mainModule.deployTransaction.hash}`)
+
+  // 6. Deploy immutable signer
+  // EST gas cost: 0.001856101
+  const immutableSigner = await deployImmutableSigner(signerRootAdminPubKey, signerAdminPubKey, immutableSignerPubKey)
+  await immutableSigner.deployTransaction.wait()
+  console.log('Finished deploying contracts')
+
+  // Fund the implementation changer
+  // WARNING: If the deployment fails at this step, DO NOT RERUN without commenting out the code a prior which deploys
+  // the contracts.
+  // TODO: Code below can be improved by calculating the amount that is required to be transferred.
+  const fundingTx = await deployer.sendTransaction({
+    to: await walletImplLocatorImplChanger.getAddress(),
+    value: ethers.utils.parseEther('10')
+  })
+  await fundingTx.wait()
+
+  console.log(`Transfered funds to the wallet locator implementer changer with hash ${fundingTx.hash}`)
+
+  // Set implementation address on impl locator to dyanmic module auth addr
+  const tx = await walletImplLocator.connect(walletImplLocatorImplChanger).changeWalletImplementation(mainModule.address)
+  await tx.wait()
+  console.log('Wallet Implentation Locator implementation changed to: ', mainModule.address)
+
+  // Output JSON file with addresses and role addresses
+  const JSONOutput = {
+    FactoryAddress: factory.address,
+    WalletImplLocatorAddress: walletImplLocator.address,
+    StartupWalletImplAddress: startupWalletImpl.address,
+    MainModuleDynamicAuthAddress: mainModule.address,
+    ImmutableSignerContractAddress: immutableSigner.address,
+    MultiCallDeployAddress: multiCallDeploy.address,
+    DeployerAddress: deployer.address,
+    FactoryAdminAddress: factoryAdminPubKey,
+    FactoryDeployerAddress: relayerSubmitterEOAPubKey,
+    WalletImplLocatorAdminAddress: walletImplLocatorAdmin,
+    WalletImplLocatorImplChangerAddress: walletImplLocatorImplChanger.address,
+    SignerRootAdminAddress: signerRootAdminPubKey,
+    SignerAdminAddress: signerAdminPubKey,
+    ImmutableSignerAddress: immutableSignerPubKey,
+    MultiCallAdminAddress: multiCallAdminPubKey,
+    MultiCallExecutorAddress: relayerSubmitterEOAPubKey
+  }
+  fs.writeFileSync(outputPath, JSON.stringify(JSONOutput, null, 1))
+
+  // Verify contracts on etherscan
+  // console.log("Verifying contracts on etherscan...");
+  // await verifyContract(multiCallDeploy.address, [multiCallAdminPubKey, relayerSubmitterEOAPubKey]);
+  // await verifyContract(factory.address,  [factoryAdminPubKey, multiCallDeploy.address]);
+  // await verifyContract(walletImplLocator.address, [walletImplLocatorAdmin, walletImplLocatorImplChanger.address]);
+  // await verifyContract(startupWalletImpl.address, [walletImplLocator.address]);
+  // await verifyContract(mainModule.address, [factory.address, startupWalletImpl.address], true, "contracts/modules/MainModuleDynamicAuth.sol:MainModuleDynamicAuth");
+  // await verifyContract(immutableSigner.address, [signerRootAdminPubKey, signerAdminPubKey, relayerSubmitterEOAPubKey]);
+  console.log('Skipping contract verification... (Etherscan not available on Immutable zkEVM)')
+}
+
+async function deployFactory(adminAddr: string, deployerAddr: string): Promise<ethers.Contract> {
+  const Factory = await hardhat.getContractFactory('Factory')
+  return await Factory.connect(deployer).deploy(adminAddr, deployerAddr)
+}
+
+async function deployWalletImplLocator(adminAddr: string, implChangerAddr: string): Promise<ethers.Contract> {
+  const LatestWalletImplLocator = await hardhat.getContractFactory('LatestWalletImplLocator')
+  return await LatestWalletImplLocator.connect(deployer).deploy(adminAddr, implChangerAddr)
+}
+
+async function deployStartUp(walletImplLocatorAddr: string): Promise<ethers.Contract> {
+  const StartupWalletImplImpl = await hardhat.getContractFactory('StartupWalletImpl')
+  return await StartupWalletImplImpl.connect(deployer).deploy(walletImplLocatorAddr)
+}
+
+async function deployMainModule(factoryAddr: string, startUpAddr: string): Promise<ethers.Contract> {
+  const MainModuleDynamicAuth = await hardhat.getContractFactory('MainModuleDynamicAuth')
+  return await MainModuleDynamicAuth.connect(deployer).deploy(factoryAddr, startUpAddr)
+}
+
+async function deployImmutableSigner(
+  rootAdminAddr: string,
+  signerAdminAddr: string,
+  signerAddr: string
+): Promise<ethers.Contract> {
+  const ImmutableSigner = await hardhat.getContractFactory('ImmutableSigner')
+  return await ImmutableSigner.connect(deployer).deploy(rootAdminAddr, signerAdminAddr, signerAddr)
+}
+
+async function deployMultiCallDeploy(adminAddr: string, executorAddr: string): Promise<ethers.Contract> {
+  const MultiCallDeploy = await hardhat.getContractFactory('MultiCallDeploy')
+  return await MultiCallDeploy.connect(deployer).deploy(adminAddr, executorAddr, {})
+}
+
+async function verifyContract(
+  contractAddr: string,
+  constructorArgs: any[],
+  requiesContractPath: boolean = false,
+  contractPath: string = ''
+) {
+  try {
+    if (requiesContractPath) {
+      await hre.run('verify:verify', {
+        contract: contractPath,
+        address: contractAddr,
+        constructorArguments: constructorArgs
+      })
+    } else {
+      await hre.run('verify:verify', {
+        address: contractAddr,
+        constructorArguments: constructorArgs
+      })
+    }
+  } catch (error) {
+    expect(error.message.toLowerCase().includes('already verified')).to.be.equal(true)
+  }
+}
+
+deploy().catch(error => {
+  console.error(error)
+  process.exitCode = 1
+})
diff --git a/forks/passport/scripts/deploy_output.json b/forks/passport/scripts/deploy_output.json
new file mode 100644
index 00000000..aa446acb
--- /dev/null
+++ b/forks/passport/scripts/deploy_output.json
@@ -0,0 +1,18 @@
+{
+ "FactoryAddress": "0x55b9d1cd803d5acA8ea23ccd96f6a756DED9f5a9",
+ "WalletImplLocatorAddress": "0x657d339B8616033FEE25F66eA1d00C3f30b14171",
+ "StartupWalletImplAddress": "0x8df826438e652f7124fe07F413fA3556cd57edB5",
+ "MainModuleDynamicAuthAddress": "0x9bF35E2E14878812909c86e67492e397068b57A7",
+ "ImmutableSignerContractAddress": "0x1B1D383526A2815d26550eb314B5d7e055132733",
+ "MultiCallDeployAddress": "0x04c7A8Eff5109a953f1DF1f83195b6eD3a7e04b6",
+ "DeployerAddress": "0x19b2680f8E79B9C21847e0f441DeEa259d124073",
+ "FactoryAdminAddress": "0xddb70ddcd14dbd57ae18ec591f47454e4fc818bb",
+ "FactoryDeployerAddress": "0xBC52cE84FceFd2D941D1127608D6Cf598f9633d3",
+ "WalletImplLocatorAdminAddress": "0xb49c99a17776c10350c2be790e13d4d8dfb1c578",
+ "WalletImplLocatorImplChangerAddress": "0xb49C99A17776C10350C2bE790e13d4d8dFB1C578",
+ "SignerRootAdminAddress": "0x65af83f71a05d7f6d06ef9a57c9294b4128ccc2c",
+ "SignerAdminAddress": "0x69d09644159e7327dbfd0af9a66f8e332c593e79",
+ "ImmutableSignerAddress": "0x1cE50560686b1297B6311F36B47dbe5d6E04D0f8",
+ "MultiCallAdminAddress": "0x575be326c482a487add43974e0eaf232e3366e13",
+ "MultiCallExecutorAddress": "0xBC52cE84FceFd2D941D1127608D6Cf598f9633d3"
+}
\ No newline at end of file
diff --git a/forks/passport/security.md b/forks/passport/security.md
new file mode 100644
index 00000000..7d29dd3b
--- /dev/null
+++ b/forks/passport/security.md
@@ -0,0 +1,29 @@
+# Security Policy
+
+Security vulnerabilities should be disclosed to the project maintainers through our public [immutable bug bounty program](https://bugcrowd.com/immutable-og) or email us at security@immutable.com.
+
+## Security Patches
+
+Security vulnerabilities will be patched as soon as responsibly possible, and published as an advisory on this repository and on the affected npm packages.
+
+Projects that build on Immutable's contracts are encouraged to clearly state, in their source code and websites, how to be contacted about security issues in the event that a direct notification is considered necessary. We recommend including it in the NatSpec for the contract as `/// @custom:security-contact security@example.com`.
+
+Additionally, we recommend installing the library through npm and setting up vulnerability alerts such as [Dependabot].
+
+[Dependabot]: https://docs.github.com/en/code-security/supply-chain-security/understanding-your-software-supply-chain/about-supply-chain-security#what-is-dependabot
+
+### Supported Versions
+
+Security patches will be released for the latest minor of a given major release. For example, if an issue is found in versions >=1.1.0 and the latest is 1.8.0, the patch will be released only in version 1.8.1.
+
+Only critical severity bug fixes will be backported to past major releases.
+
+| Version | Critical security fixes | Other security fixes |
+| ------- | ----------------------- | -------------------- |
+| 0.x     | :white_check_mark:      | :white_check_mark:   |
+
+Note as well that the Solidity language itself only guarantees security updates for the latest release.
+
+## Legal
+
+Smart contracts are a nascent techology and carry a high level of technical risk and uncertainty. Immutable's zkEVM Contracts are made available under the Apache-2.0 License, which disclaims all warranties in relation to the project and which limits the liability of those that contribute and maintain the project, including Immutable. In your use of this project, you are solely responsible for any use of Immutable zkEVM Contracts and you assume all risks associated with any such use. This Security Policy in no way evidences or represents an on-going duty by any contributor, including Immutable, to correct any flaws or alert you to all or any of the potential risks of utilizing the project.
\ No newline at end of file
diff --git a/forks/passport/src/.gitignore b/forks/passport/src/.gitignore
new file mode 100644
index 00000000..7a1266ae
--- /dev/null
+++ b/forks/passport/src/.gitignore
@@ -0,0 +1,4 @@
+cache
+artifacts/build-info
+artifacts/**/*.dbg.json
+gen
\ No newline at end of file
diff --git a/forks/passport/src/.npmignore b/forks/passport/src/.npmignore
new file mode 100644
index 00000000..7e7a9a2d
--- /dev/null
+++ b/forks/passport/src/.npmignore
@@ -0,0 +1,3 @@
+cache
+artifacts/build-info
+artifacts/**/*.dbg.json
diff --git a/forks/passport/src/contracts/Factory.sol b/forks/passport/src/contracts/Factory.sol
new file mode 100644
index 00000000..40f5c6ae
--- /dev/null
+++ b/forks/passport/src/contracts/Factory.sol
@@ -0,0 +1,59 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+import '@openzeppelin/contracts/access/AccessControl.sol';
+import "./Wallet.sol";
+
+/**
+ * @title Factory
+ * @notice Factory contract to retrieve counterfactual wallet addresses and
+ * deploy new Sequence wallet instances to those addresses
+ */
+contract Factory is AccessControl {
+  // Role to deploy new wallets
+  bytes32 public constant DEPLOYER_ROLE = keccak256('DEPLOYER_ROLE');
+
+  event WalletDeployed(address indexed wallet, address indexed mainModule, bytes32 salt);
+
+  constructor(address _admin, address _deployer) {
+    _grantRole(DEFAULT_ADMIN_ROLE, _admin);
+    _grantRole(DEPLOYER_ROLE, _deployer);
+  }
+
+  /**
+   * @notice Returns a deterministic contract address given a salt
+   * @param _mainModule Address of the main module to be used by the wallet
+   * @param _salt Salt used to generate the address
+   * @return _address The deterministic address
+   */
+  function getAddress(address _mainModule, bytes32 _salt) external view returns (address _address) {
+    bytes32 _hash = keccak256(
+      abi.encodePacked(
+        bytes1(0xff),
+        address(this),
+        _salt,
+        keccak256(abi.encodePacked(Wallet.creationCode, uint256(uint160(_mainModule))))
+      )
+    );
+    return address(uint160(uint256(_hash)));
+  }
+
+  /**
+   * @notice Will deploy a new wallet instance using create2
+   * @param _mainModule Address of the main module to be used by the wallet
+   * @param _salt Salt used to generate the wallet, which is the imageHash
+   *       of the wallet's configuration.
+   * @dev It is recommended to not have more than 200 signers as opcode repricing
+   *      could make transactions impossible to execute as all the signers must be
+   *      passed for each transaction.
+   */
+  function deploy(address _mainModule, bytes32 _salt) external payable onlyRole(DEPLOYER_ROLE) returns (address _contract) {
+    bytes memory code = abi.encodePacked(Wallet.creationCode, uint256(uint160(_mainModule)));
+    assembly {
+      _contract := create2(callvalue(), add(code, 32), mload(code), _salt)
+    }
+    // check deployment success
+    require(_contract != address(0), 'WalletFactory: deployment failed');
+    // emit event, increases gas cost by ~2k
+    emit WalletDeployed(_contract, _mainModule, _salt);
+  }
+}
diff --git a/forks/passport/src/contracts/IWalletProxy.sol b/forks/passport/src/contracts/IWalletProxy.sol
new file mode 100644
index 00000000..02d88473
--- /dev/null
+++ b/forks/passport/src/contracts/IWalletProxy.sol
@@ -0,0 +1,11 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity ^0.8.3;
+
+/**
+ * Interface that WalletProxy.yul implements.
+ */
+interface IWalletProxy {
+    /// @dev Retrieve current implementation contract used by proxy
+    function PROXY_getImplementation() external view returns (address implementation);
+}
\ No newline at end of file
diff --git a/forks/passport/src/contracts/MultiCallDeploy.sol b/forks/passport/src/contracts/MultiCallDeploy.sol
new file mode 100644
index 00000000..63a88450
--- /dev/null
+++ b/forks/passport/src/contracts/MultiCallDeploy.sol
@@ -0,0 +1,72 @@
+// Copyright (c) Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./modules/commons/interfaces/IModuleCalls.sol";
+import '@openzeppelin/contracts/access/AccessControl.sol';
+import "./interfaces/IFactory.sol";
+
+/**
+ * @title MultiCallDeploy
+ * @notice This contract is bundles the wallet deployment and the users first write transaction into a single transaction.
+ *         Contract usage is intended for the submitter inside the relayer service, which will call either of the functions.
+ */
+contract MultiCallDeploy is AccessControl {
+    // Role to execute functions
+    bytes32 public constant EXECUTOR_ROLE = keccak256('EXECUTOR_ROLE');
+
+    constructor(address _admin, address _executor) {
+        _grantRole(DEFAULT_ADMIN_ROLE, _admin);
+        _grantRole(EXECUTOR_ROLE, _executor);
+    }
+
+    /*
+    * @dev Grants EXECUTOR_ROLE to an user.
+    * @param _executor Address that will be allowed to execute functions
+    */
+    function grantExecutorRole(address _executor) external onlyRole(DEFAULT_ADMIN_ROLE) {
+        _grantRole(EXECUTOR_ROLE, _executor);
+    }
+
+    /*
+    * @dev Deploy wallet and execute transaction.
+    * @param _mainModule Address of the main module to be used by the wallet
+    * @param _salt Salt used to generate the address
+    * @param factory address of the factory contract
+    * @param _txs transaction to execute
+    * @param _nonce nonce of the wallet
+    * @param _signature transaction signature from wallet
+    */
+    function deployExecute(address _mainModule, bytes32 _salt, address factory,  IModuleCalls.Transaction[] calldata _txs, uint256 _nonce, bytes calldata _signature) external onlyRole(EXECUTOR_ROLE) {
+        address ret = IFactory(factory).deploy(_mainModule, _salt);
+        IModuleCalls(ret).execute(_txs, _nonce, _signature);
+    }
+
+    /*
+    * @dev Handles deployment of wallet and transaction execution for both cases
+    * @param cfa counter factual address of the wallet
+    * @param _mainModule Address of the main module to be used by the wallet
+    * @param _salt Salt used to generate the address
+    * @param factory address of the factory contract
+    * @param _txs transaction to execute
+    * @param _nonce nonce of the wallet
+    * @param _signature transaction signature from wallet
+    */
+    function deployAndExecute(address cfa, address _mainModule, bytes32 _salt, address factory,  IModuleCalls.Transaction[] calldata _txs, uint256 _nonce, bytes calldata _signature) external onlyRole(EXECUTOR_ROLE){
+        // Get code size at CFA
+        uint32 size;
+        assembly {
+            size := extcodesize(cfa)
+        }
+
+        // If size is 0, deploy the proxy and execute write tx
+        // Else, execute the users transaction
+        if (size == 0) {
+            address ret = IFactory(factory).deploy(_mainModule, _salt);
+            require(cfa == ret, "MultiCallDeploy: deployed address does not match CFA");
+            IModuleCalls(ret).execute(_txs, _nonce, _signature);
+        } else {
+            IModuleCalls(cfa).execute(_txs, _nonce, _signature);
+        }
+    }
+}
\ No newline at end of file
diff --git a/forks/passport/src/contracts/Wallet.sol b/forks/passport/src/contracts/Wallet.sol
new file mode 100644
index 00000000..65532d0e
--- /dev/null
+++ b/forks/passport/src/contracts/Wallet.sol
@@ -0,0 +1,10 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+// Holds the creation code of the WalletProxy.yul used by smart contract wallet instances.
+// Generate this bytecode using ./compileWalletProxyYul.sh
+library Wallet {
+    // This bytecode must precisely match that in tests/utils/helpers.ts
+    bytes internal constant creationCode = hex"6054600f3d396034805130553df3fe63906111273d3560e01c14602b57363d3d373d3d3d3d369030545af43d82803e156027573d90f35b3d90fd5b30543d5260203df3";
+}
diff --git a/forks/passport/src/contracts/WalletProxy.yul b/forks/passport/src/contracts/WalletProxy.yul
new file mode 100644
index 00000000..2bf1f8e9
--- /dev/null
+++ b/forks/passport/src/contracts/WalletProxy.yul
@@ -0,0 +1,81 @@
+// Copyright (c) Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+//
+// This Yul code creates a minimalist transparent proxy with a function to fetch
+// the address of the contract being proxied to using the interface described in
+// IWalletProxy.sol .
+//
+object "ProxyGetImplYul" {
+    // This is the initcode of the contract.
+    code {
+        // Copy the runtime code plus the address of the implementation 
+        // parameter (32 bytes) which is appended to the end to memory.
+        // copy s bytes from code at position f to mem at position t
+        // codecopy(t, f, s)
+        // This will turn into a memory->memory copy for Ewasm and
+        // a codecopy for EVM
+        // The constant 0x54 is datasize("runtime") + 32. The solc compiler is
+        // unable to do constant addition as part of the compilation process, hence
+        // the constant.
+        // If the runtime code is to be updated, uncomment the following line, and comment
+        // out the following line, so that datasize("runtime") can be determined. It will
+        // be the byte following the 0x60 push1 opcode.
+//        datacopy(returndatasize(), dataoffset("runtime"), add(datasize("runtime"), 32))
+        datacopy(returndatasize(), dataoffset("runtime"), 0x54)
+
+        // Store the implementation address at the storage slot which is 
+        // equivalent to the deployed address of this contract.
+        let implAddress := mload(datasize("runtime"))
+        sstore(address(), implAddress)
+
+        // now return the runtime object (the currently
+        // executing code is the constructor code)
+        return(returndatasize(), datasize("runtime"))
+    }
+
+
+    // Code for deployed contract
+    object "runtime" {
+        code {
+            // Load the function selector (the first four bytes of calldata) by shifting the 
+            // word to the right. 
+            let selector := shr(224, calldataload(returndatasize()))
+
+            if eq(selector, 0x90611127) /* Function selector for "PROXY_getImplementation()" */ {
+                let impl := sload(address())
+                mstore(returndatasize(), impl)
+                return(returndatasize(), 0x20)
+            }
+
+            // Load calldata to memory location 0.
+            // Copy s bytes from calldata at position f to mem at position t
+            // calldatacopy(t, f, s)
+            calldatacopy(returndatasize(), returndatasize(), calldatasize())
+
+            // Use returndatasize to load zero.
+            let zero := returndatasize()
+
+            // Execute delegate call. Have outsize set to zero, to indicate
+            // don't return any data automatically.
+            // Call contract at address a with input mem[in…(in+insize)) 
+            // providing g gas and v wei and output area 
+            // mem[out…(out+outsize)) returning 0 on error 
+            // (eg. out of gas) and 1 on success
+            // delegatecall(g, a, in, insize, out, outsize)
+            // Use sload(address()) to load the implemntation address.
+            let success := delegatecall(gas(), sload(address()), returndatasize(), calldatasize(), returndatasize(), returndatasize())
+
+            // Copy the return result to memory location 0.
+            // Copy s bytes from returndata at position f to mem at position t
+            // returndatacopy(t, f, s)
+            returndatacopy(zero, zero, returndatasize())
+
+            // Return or revert: memory location 0 contains either the return value
+            // or the revert information.
+            if iszero(success) {
+                revert (zero,returndatasize())
+            }
+            return (zero,returndatasize())
+        }
+    }
+}
\ No newline at end of file
diff --git a/forks/passport/src/contracts/interfaces/IERC1271Wallet.sol b/forks/passport/src/contracts/interfaces/IERC1271Wallet.sol
new file mode 100644
index 00000000..794798d8
--- /dev/null
+++ b/forks/passport/src/contracts/interfaces/IERC1271Wallet.sol
@@ -0,0 +1,38 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IERC1271Wallet {
+
+  /**
+   * @notice Verifies whether the provided signature is valid with respect to the provided data
+   * @dev MUST return the correct magic value if the signature provided is valid for the provided data
+   *   > The bytes4 magic value to return when signature is valid is 0x20c13b0b : bytes4(keccak256("isValidSignature(bytes,bytes)")
+   *   > This function MAY modify Ethereum's state
+   * @param _data       Arbitrary length data signed on the behalf of address(this)
+   * @param _signature  Signature byte array associated with _data
+   * @return magicValue Magic value 0x20c13b0b if the signature is valid and 0x0 otherwise
+   */
+  function isValidSignature(
+    bytes calldata _data,
+    bytes calldata _signature)
+    external
+    view
+    returns (bytes4 magicValue);
+
+  /**
+   * @notice Verifies whether the provided signature is valid with respect to the provided hash
+   * @dev MUST return the correct magic value if the signature provided is valid for the provided hash
+   *   > The bytes4 magic value to return when signature is valid is 0x20c13b0b : bytes4(keccak256("isValidSignature(bytes,bytes)")
+   *   > This function MAY modify Ethereum's state
+   * @param _hash       keccak256 hash that was signed
+   * @param _signature  Signature byte array associated with _data
+   * @return magicValue Magic value 0x20c13b0b if the signature is valid and 0x0 otherwise
+   */
+  function isValidSignature(
+    bytes32 _hash,
+    bytes calldata _signature)
+    external
+    view
+    returns (bytes4 magicValue);
+}
\ No newline at end of file
diff --git a/forks/passport/src/contracts/interfaces/IFactory.sol b/forks/passport/src/contracts/interfaces/IFactory.sol
new file mode 100644
index 00000000..94258ad0
--- /dev/null
+++ b/forks/passport/src/contracts/interfaces/IFactory.sol
@@ -0,0 +1,29 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+/**
+ * @title IFactory
+ * @notice Factory interface to interact with wallet factory
+ */
+interface IFactory {
+  event WalletDeployed(address indexed wallet, address indexed mainModule, bytes32 salt);
+
+  /**
+   * @notice Returns a deterministic contract address given a salt
+   * @param _mainModule Address of the main module to be used by the wallet
+   * @param _salt Salt used to generate the address
+   * @return _address The deterministic address
+   */
+  function getAddress(address _mainModule, bytes32 _salt) external view returns (address);
+
+  /**
+   * @notice Will deploy a new wallet instance using create2
+   * @param _mainModule Address of the main module to be used by the wallet
+   * @param _salt Salt used to generate the wallet, which is the imageHash
+   *       of the wallet's configuration.
+   * @dev It is recommended to not have more than 200 signers as opcode repricing
+   *      could make transactions impossible to execute as all the signers must be
+   *      passed for each transaction.
+   */
+  function deploy(address _mainModule, bytes32 _salt) external payable returns (address);
+}
diff --git a/forks/passport/src/contracts/interfaces/receivers/IERC1155Receiver.sol b/forks/passport/src/contracts/interfaces/receivers/IERC1155Receiver.sol
new file mode 100644
index 00000000..f61d6466
--- /dev/null
+++ b/forks/passport/src/contracts/interfaces/receivers/IERC1155Receiver.sol
@@ -0,0 +1,8 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IERC1155Receiver {
+  function onERC1155Received(address, address, uint256, uint256, bytes calldata) external returns (bytes4);
+  function onERC1155BatchReceived(address, address, uint256[] calldata, uint256[] calldata, bytes calldata) external returns (bytes4);
+}
diff --git a/forks/passport/src/contracts/interfaces/receivers/IERC223Receiver.sol b/forks/passport/src/contracts/interfaces/receivers/IERC223Receiver.sol
new file mode 100644
index 00000000..dc092c69
--- /dev/null
+++ b/forks/passport/src/contracts/interfaces/receivers/IERC223Receiver.sol
@@ -0,0 +1,7 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IERC223Receiver {
+  function tokenFallback(address, uint256, bytes calldata) external;
+}
diff --git a/forks/passport/src/contracts/interfaces/receivers/IERC721Receiver.sol b/forks/passport/src/contracts/interfaces/receivers/IERC721Receiver.sol
new file mode 100644
index 00000000..495be6b1
--- /dev/null
+++ b/forks/passport/src/contracts/interfaces/receivers/IERC721Receiver.sol
@@ -0,0 +1,7 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IERC721Receiver {
+  function onERC721Received(address, address, uint256, bytes calldata) external returns (bytes4);
+}
diff --git a/forks/passport/src/contracts/migrations/Migrations.sol b/forks/passport/src/contracts/migrations/Migrations.sol
new file mode 100644
index 00000000..03215be8
--- /dev/null
+++ b/forks/passport/src/contracts/migrations/Migrations.sol
@@ -0,0 +1,26 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract Migrations {
+  address public owner;
+  uint public last_completed_migration;
+
+  constructor() public {
+    owner = msg.sender;
+  }
+
+  modifier restricted() {
+    if (msg.sender == owner)
+      _;
+  }
+
+  function setCompleted(uint completed) public restricted {
+    last_completed_migration = completed;
+  }
+
+  function upgrade(address new_address) public restricted {
+    Migrations upgraded = Migrations(new_address);
+    upgraded.setCompleted(last_completed_migration);
+  }
+}
diff --git a/forks/passport/src/contracts/mocks/AlwaysRevertMock.sol b/forks/passport/src/contracts/mocks/AlwaysRevertMock.sol
new file mode 100644
index 00000000..77b1aa4b
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/AlwaysRevertMock.sol
@@ -0,0 +1,9 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract AlwaysRevertMock {
+  fallback() external payable {
+    revert("AlwaysRevertMock#fallback: ALWAYS_REVERT");
+  }
+}
diff --git a/forks/passport/src/contracts/mocks/CallReceiverMock.sol b/forks/passport/src/contracts/mocks/CallReceiverMock.sol
new file mode 100644
index 00000000..f0dda9cd
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/CallReceiverMock.sol
@@ -0,0 +1,23 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract CallReceiverMock {
+  uint256 public lastValA;
+  bytes public lastValB;
+
+  bool revertFlag;
+
+  constructor() public payable { }
+
+  function setRevertFlag(bool _revertFlag) external {
+    revertFlag = _revertFlag;
+  }
+
+  function testCall(uint256 _valA, bytes calldata _valB) external payable {
+    require(!revertFlag, "CallReceiverMock#testCall: REVERT_FLAG");
+
+    lastValA = _valA;
+    lastValB = _valB;
+  }
+}
diff --git a/forks/passport/src/contracts/mocks/CustomModule.sol b/forks/passport/src/contracts/mocks/CustomModule.sol
new file mode 100644
index 00000000..12ccc619
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/CustomModule.sol
@@ -0,0 +1,13 @@
+pragma solidity 0.8.17;
+
+contract CustomModule {
+    string public str;
+
+    function getStr() public view returns (string memory) {
+        return str;
+    }
+
+    function setStr(string memory _str) public {
+        str = _str;
+    }
+}
diff --git a/forks/passport/src/contracts/mocks/DelegateCallMock.sol b/forks/passport/src/contracts/mocks/DelegateCallMock.sol
new file mode 100644
index 00000000..cfd500ae
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/DelegateCallMock.sol
@@ -0,0 +1,24 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract DelegateCallMock {
+  event Readed(uint256 _val);
+
+  uint256 private constant REVERT_SLOT = uint256(keccak256("revert-flag"));
+
+  mapping(uint256 => uint256) private store;
+
+  function setRevertFlag(bool _revertFlag) external {
+    store[REVERT_SLOT] = _revertFlag ? 1 : 0;
+  }
+
+  function write(uint256 _key, uint256 _val) external {
+    require(store[REVERT_SLOT] == 0, "DelegateCallMock#write: REVERT_FLAG");
+    store[_key] = _val;
+  }
+
+  function read(uint256 _key) external {
+    emit Readed(store[_key]);
+  }
+}
diff --git a/forks/passport/src/contracts/mocks/ERC165CheckerMock.sol b/forks/passport/src/contracts/mocks/ERC165CheckerMock.sol
new file mode 100644
index 00000000..2f453b92
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/ERC165CheckerMock.sol
@@ -0,0 +1,56 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract ERC165CheckerMock {
+  bytes4 constant InvalidID = 0xffffffff;
+  bytes4 constant ERC165ID = 0x01ffc9a7;
+
+  function doesContractImplementInterface(address _contract, bytes4 _interfaceId) external view returns (bool) {
+    uint256 success;
+    uint256 result;
+
+    (success, result) = noThrowCall(_contract, ERC165ID);
+    if (success == 0 || result == 0) {
+      return false;
+    }
+
+    (success, result) = noThrowCall(_contract, InvalidID);
+    if (success == 0 || result != 0) {
+      return false;
+    }
+
+    (success, result) = noThrowCall(_contract, _interfaceId);
+    if (success == 1 && result == 1) {
+      return true;
+    }
+    return false;
+  }
+
+  function noThrowCall(
+    address _contract,
+    bytes4 _interfaceId
+  ) private view returns (
+    uint256 success,
+    uint256 result
+  ) {
+    bytes4 erc165ID = ERC165ID;
+
+    assembly {
+      let x := mload(0x40)               // Find empty storage location using "free memory pointer"
+      mstore(x, erc165ID)                // Place signature at beginning of empty storage
+      mstore(add(x, 0x04), _interfaceId) // Place first argument directly next to signature
+
+      success := staticcall(
+        30000,     // 30k gas
+        _contract, // To addr
+        x,         // Inputs are stored at location x
+        0x24,      // Inputs are 36 bytes long
+        x,         // Store output over input (saves space)
+        0x20       // Outputs are 32 bytes long
+      )
+
+      result := mload(x)                 // Load the result
+    }
+  }
+}
diff --git a/forks/passport/src/contracts/mocks/ERC20Mock.sol b/forks/passport/src/contracts/mocks/ERC20Mock.sol
new file mode 100644
index 00000000..0ec14633
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/ERC20Mock.sol
@@ -0,0 +1,9 @@
+pragma solidity ^0.8.0;
+
+import '@openzeppelin/contracts/token/ERC20/ERC20.sol';
+
+contract ERC20Mock is ERC20 {
+    constructor()ERC20("Mock", "M"){
+        _mint(msg.sender, 100 ether);
+    }
+}
\ No newline at end of file
diff --git a/forks/passport/src/contracts/mocks/GasBurnerMock.sol b/forks/passport/src/contracts/mocks/GasBurnerMock.sol
new file mode 100644
index 00000000..5deb005b
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/GasBurnerMock.sol
@@ -0,0 +1,18 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract GasBurnerMock {
+  event ProvidedGas(uint256 _val);
+
+  function burnGas(uint256 _burn) external {
+    emit ProvidedGas(gasleft());
+
+    bytes32 stub;
+    uint256 initial = gasleft();
+
+    while (initial - gasleft() < _burn) {
+      stub = keccak256(abi.encode(stub));
+    }
+  }
+}
diff --git a/forks/passport/src/contracts/mocks/HookCallerMock.sol b/forks/passport/src/contracts/mocks/HookCallerMock.sol
new file mode 100644
index 00000000..f5b0a933
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/HookCallerMock.sol
@@ -0,0 +1,78 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../interfaces/receivers/IERC1155Receiver.sol";
+import "../interfaces/receivers/IERC721Receiver.sol";
+import "../interfaces/receivers/IERC223Receiver.sol";
+
+import "../interfaces/IERC1271Wallet.sol";
+
+
+contract HookCallerMock {
+  function callERC1155Received(address _addr) external {
+    bytes4 result = IERC1155Receiver(_addr).onERC1155Received(
+      address(this),
+      msg.sender,
+      1,
+      2,
+      msg.data
+    );
+
+    require(result == 0xf23a6e61, "HookCallerMock#callERC1155Received: INVALID_RETURN");
+  }
+
+  function callERC1155BatchReceived(address _addr) external {
+    uint256[] memory ids = new uint256[](3);
+    ids[0] = 1;
+    ids[1] = 2;
+    ids[2] = 3;
+
+    uint256[] memory values = new uint256[](3);
+    values[0] = 200;
+    values[1] = 300;
+    values[2] = 400;
+
+    bytes4 result = IERC1155Receiver(_addr).onERC1155BatchReceived(
+      address(this),
+      msg.sender,
+      ids,
+      values,
+      msg.data
+    );
+
+    require(result == 0xbc197c81, "HookCallerMock#callERC1155BatchReceived: INVALID_RETURN");
+  }
+
+  function callERC721Received(address _addr) external {
+    bytes4 result = IERC721Receiver(_addr).onERC721Received(
+      address(this),
+      msg.sender,
+      1,
+      msg.data
+    );
+
+    require(result == 0x150b7a02, "HookCallerMock#callERC721Received: INVALID_RETURN");
+  }
+
+  function callERC223Received(address _addr) external {
+    IERC223Receiver(_addr).tokenFallback(msg.sender, 1, msg.data);
+  }
+
+  function callERC1271isValidSignatureData(
+    address _addr,
+    bytes calldata _data,
+    bytes calldata _signature
+  ) external view {
+    bytes4 result = IERC1271Wallet(_addr).isValidSignature(_data, _signature);
+    require(result == 0x20c13b0b, "HookCallerMock#callERC1271isValidSignatureData: INVALID_RETURN");
+  }
+
+  function callERC1271isValidSignatureHash(
+    address _addr,
+    bytes32 _hash,
+    bytes calldata _signature
+  ) external view {
+    bytes4 result = IERC1271Wallet(_addr).isValidSignature(_hash, _signature);
+    require(result == 0x1626ba7e, "HookCallerMock#callERC1271isValidSignatureHash: INVALID_RETURN");
+  }
+}
diff --git a/forks/passport/src/contracts/mocks/HookMock.sol b/forks/passport/src/contracts/mocks/HookMock.sol
new file mode 100644
index 00000000..d2be3583
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/HookMock.sol
@@ -0,0 +1,9 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract HookMock {
+  function onHookMockCall(uint256 _num) external pure returns (uint256) {
+    return _num * 2;
+  }
+}
diff --git a/forks/passport/src/contracts/mocks/LibBytesImpl.sol b/forks/passport/src/contracts/mocks/LibBytesImpl.sol
new file mode 100644
index 00000000..2ebfa07c
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/LibBytesImpl.sol
@@ -0,0 +1,37 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../utils/LibBytes.sol";
+
+
+contract LibBytesImpl {
+  using LibBytes for bytes;
+
+  function readFirstUint16(bytes calldata _data) external pure returns (uint16, uint256) {
+    return _data.readFirstUint16();
+  }
+
+  function readUint8Uint8(bytes calldata _data, uint256 _index) external pure returns (uint8, uint8, uint256) {
+    return _data.readUint8Uint8(_index);
+  }
+
+  function readAddress(bytes calldata _data, uint256 _index) external pure returns (address, uint256) {
+    return _data.readAddress(_index);
+  }
+
+  function readBytes66(bytes calldata _data, uint256 _index) external pure returns (bytes memory, uint256) {
+    return _data.readBytes66(_index);
+  }
+
+  function readBytes32(bytes calldata _data, uint256 _index) external pure returns (bytes32) {
+    return _data.readBytes32(_index);
+  }
+
+  function readUint16(bytes calldata _data, uint256 _index) external pure returns (uint16, uint256) {
+    return _data.readUint16(_index);
+  }
+
+  function readBytes(bytes calldata _data, uint256 _index, uint256 _size) external view returns (bytes memory, uint256) {
+    return _data.readBytes(_index, _size);
+  }
+}
diff --git a/forks/passport/src/contracts/mocks/MainModule.sol b/forks/passport/src/contracts/mocks/MainModule.sol
new file mode 100644
index 00000000..e792c012
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/MainModule.sol
@@ -0,0 +1,7 @@
+pragma solidity 0.8.17;
+// installed under @sequence/wallet-contracts alias instead of @0xsequence/wallet-contracts as the '0x' has problems with typechain
+import "../modules/MainModule.sol";
+
+contract MainModuleMock is MainModule {
+    constructor(address _factory) MainModule(_factory) {}
+}
diff --git a/forks/passport/src/contracts/mocks/MainModuleMockV1.sol b/forks/passport/src/contracts/mocks/MainModuleMockV1.sol
new file mode 100644
index 00000000..47807abc
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/MainModuleMockV1.sol
@@ -0,0 +1,12 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../modules/MainModuleDynamicAuth.sol";
+
+contract MainModuleMockV1 is MainModuleDynamicAuth {
+    // solhint-disable-next-line no-empty-blocks
+    constructor(address _factory, address _startup) MainModuleDynamicAuth(_factory, _startup) {}
+
+
+}
diff --git a/forks/passport/src/contracts/mocks/MainModuleMockV2.sol b/forks/passport/src/contracts/mocks/MainModuleMockV2.sol
new file mode 100644
index 00000000..d0ed7087
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/MainModuleMockV2.sol
@@ -0,0 +1,15 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../modules/MainModuleDynamicAuth.sol";
+
+contract MainModuleMockV2 is MainModuleDynamicAuth {
+    // solhint-disable-next-line no-empty-blocks
+    constructor(address _factory, address _startup) MainModuleDynamicAuth(_factory, _startup) {}
+
+    function version() external pure override returns (uint256) {
+        return 2;
+    }
+
+}
\ No newline at end of file
diff --git a/forks/passport/src/contracts/mocks/MainModuleMockV3.sol b/forks/passport/src/contracts/mocks/MainModuleMockV3.sol
new file mode 100644
index 00000000..1f590a7c
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/MainModuleMockV3.sol
@@ -0,0 +1,15 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../modules/MainModuleDynamicAuth.sol";
+
+contract MainModuleMockV3 is MainModuleDynamicAuth {
+    // solhint-disable-next-line no-empty-blocks
+    constructor(address _factory, address _startup) MainModuleDynamicAuth(_factory, _startup) {}
+
+    function version() external pure override returns (uint256) {
+        return 3;
+    }
+
+}
\ No newline at end of file
diff --git a/forks/passport/src/contracts/mocks/ModuleMock.sol b/forks/passport/src/contracts/mocks/ModuleMock.sol
new file mode 100644
index 00000000..a0f1f840
--- /dev/null
+++ b/forks/passport/src/contracts/mocks/ModuleMock.sol
@@ -0,0 +1,11 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract ModuleMock {
+  event Pong();
+
+  function ping() external {
+    emit Pong();
+  }
+}
diff --git a/forks/passport/src/contracts/modules/GuestModule.sol b/forks/passport/src/contracts/modules/GuestModule.sol
new file mode 100644
index 00000000..972bf598
--- /dev/null
+++ b/forks/passport/src/contracts/modules/GuestModule.sol
@@ -0,0 +1,117 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../utils/SignatureValidator.sol";
+
+import "./commons/Implementation.sol";
+import "./commons/ModuleAuth.sol";
+import "./commons/ModuleHooks.sol";
+import "./commons/ModuleCalls.sol";
+import "./commons/ModuleUpdate.sol";
+import "./commons/ModuleCreator.sol";
+
+import "../interfaces/receivers/IERC1155Receiver.sol";
+import "../interfaces/receivers/IERC721Receiver.sol";
+
+import "../interfaces/IERC1271Wallet.sol";
+
+
+/**
+ * GuestModule implements an Arcadeum wallet without signatures, nonce or replay protection.
+ * executing transactions using this wallet is not an authenticated process, and can be done by any address.
+ *
+ * @notice This contract is completely public with no security, designed to execute pre-signed transactions
+ *   and use Arcadeum tools without using the wallets.
+ */
+contract GuestModule is
+  ModuleAuth,
+  ModuleCalls,
+  ModuleCreator
+{
+  /**
+   * @notice Allow any caller to execute an action
+   * @param _txs Transactions to process
+   */
+  function execute(
+    Transaction[] memory _txs,
+    uint256,
+    bytes memory
+  ) public override {
+    // Hash transaction bundle
+    bytes32 txHash = _subDigest(keccak256(abi.encode('guest:', _txs)));
+
+    // Execute the transactions
+    _executeGuest(txHash, _txs);
+  }
+
+  /**
+   * @notice Allow any caller to execute an action
+   * @param _txs Transactions to process
+   */
+  function selfExecute(
+    Transaction[] memory _txs
+  ) public override {
+    // Hash transaction bundle
+    bytes32 txHash = _subDigest(keccak256(abi.encode('self:', _txs)));
+
+    // Execute the transactions
+    _executeGuest(txHash, _txs);
+  }
+
+  /**
+   * @notice Executes a list of transactions
+   * @param _txHash  Hash of the batch of transactions
+   * @param _txs  Transactions to execute
+   */
+  function _executeGuest(
+    bytes32 _txHash,
+    Transaction[] memory _txs
+  ) private {
+    // Execute transaction
+    for (uint256 i = 0; i < _txs.length; i++) {
+      Transaction memory transaction = _txs[i];
+
+      bool success;
+      bytes memory result;
+
+      require(!transaction.delegateCall, 'GuestModule#_executeGuest: delegateCall not allowed');
+      require(gasleft() >= transaction.gasLimit, "GuestModule#_executeGuest: NOT_ENOUGH_GAS");
+
+      // solhint-disable
+      (success, result) = transaction.target.call{
+        value: transaction.value,
+        gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+      }(transaction.data);
+      // solhint-enable
+
+      if (success) {
+        emit TxExecuted(_txHash);
+      } else {
+        _revertBytes(transaction, _txHash, result);
+      }
+    }
+  }
+
+  /**
+   * @notice Validates any signature image, because the wallet is public and has now owner.
+   * @return true, all signatures are valid, false, no updates required
+   */
+  function _isValidImage(bytes32) internal override view returns (bool, bool) {
+    return (true, false);
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(
+    bytes4 _interfaceID
+  ) public override (
+    ModuleAuth,
+    ModuleCalls,
+    ModuleCreator
+  ) pure returns (bool) {
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/MainModule.sol b/forks/passport/src/contracts/modules/MainModule.sol
new file mode 100644
index 00000000..e1b6f4dd
--- /dev/null
+++ b/forks/passport/src/contracts/modules/MainModule.sol
@@ -0,0 +1,54 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../utils/SignatureValidator.sol";
+
+import "./commons/Implementation.sol";
+import "./commons/ModuleAuthFixed.sol";
+import "./commons/ModuleHooks.sol";
+import "./commons/ModuleCalls.sol";
+import "./commons/ModuleUpdate.sol";
+import "./commons/ModuleCreator.sol";
+
+import "../interfaces/receivers/IERC1155Receiver.sol";
+import "../interfaces/receivers/IERC721Receiver.sol";
+
+import "../interfaces/IERC1271Wallet.sol";
+
+
+/**
+ * @notice Contains the core functionality arcadeum wallets will inherit.
+ * @dev If using a new main module, developpers must ensure that all inherited
+ *      contracts by the mainmodule don't conflict and are accounted for to be
+ *      supported by the supportsInterface method.
+ */
+contract MainModule is
+  ModuleAuthFixed,
+  ModuleCalls,
+  ModuleUpdate,
+  ModuleHooks,
+  ModuleCreator
+{
+  constructor(
+    address _factory
+  ) public ModuleAuthFixed(
+    _factory
+  ) { }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(
+    bytes4 _interfaceID
+  ) public override(
+    ModuleAuth,
+    ModuleCalls,
+    ModuleUpdate,
+    ModuleHooks,
+    ModuleCreator
+  ) pure returns (bool) {
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/MainModuleDynamicAuth.sol b/forks/passport/src/contracts/modules/MainModuleDynamicAuth.sol
new file mode 100644
index 00000000..149ce806
--- /dev/null
+++ b/forks/passport/src/contracts/modules/MainModuleDynamicAuth.sol
@@ -0,0 +1,52 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./commons/ModuleAuthDynamic.sol";
+import "./commons/ModuleReceivers.sol";
+import "./commons/ModuleCalls.sol";
+import "./commons/ModuleUpdate.sol";
+
+
+/**
+ * TODO Peter update docs
+ * @notice Contains the core functionality arcadeum wallets will inherit with
+ *         the added functionality that the main-module can be changed.
+ * @dev If using a new main module, developpers must ensure that all inherited
+ *      contracts by the mainmodule don't conflict and are accounted for to be
+ *      supported by the supportsInterface method.
+ */
+contract MainModuleDynamicAuth is
+  ModuleAuthDynamic,
+  ModuleCalls,
+  ModuleReceivers,
+  ModuleUpdate
+{
+
+  // solhint-disable-next-line no-empty-blocks
+  constructor(address _factory, address _startup) ModuleAuthDynamic (_factory, _startup) { }
+
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @dev If using a new main module, developpers must ensure that all inherited
+   *      contracts by the mainmodule don't conflict and are accounted for to be
+   *      supported by the supportsInterface method.
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(
+    bytes4 _interfaceID
+  ) public override(
+    ModuleAuthUpgradable,
+    ModuleCalls,
+    ModuleReceivers,
+    ModuleUpdate
+  ) pure returns (bool) {
+    return super.supportsInterface(_interfaceID);
+  }
+
+  function version() external pure virtual returns (uint256) {
+    return 1;
+  }
+}
diff --git a/forks/passport/src/contracts/modules/MainModuleGasEstimation.sol b/forks/passport/src/contracts/modules/MainModuleGasEstimation.sol
new file mode 100644
index 00000000..f1fddbab
--- /dev/null
+++ b/forks/passport/src/contracts/modules/MainModuleGasEstimation.sol
@@ -0,0 +1,93 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./commons/ModuleIgnoreAuthUpgradable.sol";
+import "./commons/ModuleIgnoreNonceCalls.sol";
+import "./commons/ModuleHooks.sol";
+import "./commons/ModuleUpdate.sol";
+import "./commons/ModuleCreator.sol";
+
+
+/**
+ * @notice Contains an alternative implementation of the MainModules that skips validation of
+ *   signatures, this implementation SHOULD NOT be used directly on a wallet.
+ *
+ *   Intended to be used only for gas estimation, using eth_call and overrides.
+ */
+contract MainModuleGasEstimation is
+  ModuleIgnoreAuthUpgradable,
+  ModuleIgnoreNonceCalls,
+  ModuleUpdate,
+  ModuleHooks,
+  ModuleCreator
+{
+  struct SimulateResult {
+    bool executed;
+    bool succeeded;
+    bytes result;
+    uint256 gasUsed;
+  }
+
+  /**
+   * @notice Simulate each transaction in a bundle for gas usage and execution result
+   * @param _txs Transactions to process
+   * @return The gas used and execution result for each transaction in the bundle
+   */
+  function simulateExecute(Transaction[] calldata _txs) public virtual returns (SimulateResult[] memory) {
+    SimulateResult[] memory results = new SimulateResult[](_txs.length);
+
+    // Execute transaction
+    for (uint256 i = 0; i < _txs.length; i++) {
+      Transaction memory transaction = _txs[i];
+
+      require(gasleft() >= transaction.gasLimit, "MainModuleGasEstimation#simulateExecute: NOT_ENOUGH_GAS");
+
+      results[i].executed = true;
+
+      if (transaction.delegateCall) {
+        uint256 initialGas = gasleft();
+
+        (results[i].succeeded, results[i].result) = transaction.target.delegatecall{
+          gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+        }(transaction.data);
+
+        results[i].gasUsed = initialGas - gasleft();
+      } else {
+        uint256 initialGas = gasleft();
+
+        (results[i].succeeded, results[i].result) = transaction.target.call{
+          value: transaction.value,
+          gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+        }(transaction.data);
+
+        results[i].gasUsed = initialGas - gasleft();
+      }
+
+      if (!results[i].succeeded && _txs[i].revertOnError) {
+        break;
+      }
+    }
+
+    return results;
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @dev If using a new main module, developpers must ensure that all inherited
+   *      contracts by the mainmodule don't conflict and are accounted for to be
+   *      supported by the supportsInterface method.
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(
+    bytes4 _interfaceID
+  ) public override(
+    ModuleIgnoreAuthUpgradable,
+    ModuleIgnoreNonceCalls,
+    ModuleUpdate,
+    ModuleHooks,
+    ModuleCreator
+  ) pure returns (bool) {
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/MainModuleUpgradable.sol b/forks/passport/src/contracts/modules/MainModuleUpgradable.sol
new file mode 100644
index 00000000..588682da
--- /dev/null
+++ b/forks/passport/src/contracts/modules/MainModuleUpgradable.sol
@@ -0,0 +1,44 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./commons/ModuleAuthUpgradable.sol";
+import "./commons/ModuleHooks.sol";
+import "./commons/ModuleCalls.sol";
+import "./commons/ModuleUpdate.sol";
+import "./commons/ModuleCreator.sol";
+
+
+/**
+ * @notice Contains the core functionality arcadeum wallets will inherit with
+ *         the added functionality that the main-module can be changed.
+ * @dev If using a new main module, developpers must ensure that all inherited
+ *      contracts by the mainmodule don't conflict and are accounted for to be
+ *      supported by the supportsInterface method.
+ */
+contract MainModuleUpgradable is
+  ModuleAuthUpgradable,
+  ModuleCalls,
+  ModuleUpdate,
+  ModuleHooks,
+  ModuleCreator
+{
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @dev If using a new main module, developpers must ensure that all inherited
+   *      contracts by the mainmodule don't conflict and are accounted for to be
+   *      supported by the supportsInterface method.
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(
+    bytes4 _interfaceID
+  ) public override(
+    ModuleAuthUpgradable,
+    ModuleCalls,
+    ModuleUpdate,
+    ModuleHooks,
+    ModuleCreator
+  ) pure returns (bool) {
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ImageHashKey.sol b/forks/passport/src/contracts/modules/commons/ImageHashKey.sol
new file mode 100644
index 00000000..e1d9ec44
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ImageHashKey.sol
@@ -0,0 +1,8 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+library ImageHashKey {
+  // Randomly generated to avoid collisions, with:
+  // xxd -len 32 -plain -cols 32 /dev/urandom
+  bytes32 internal constant IMAGE_HASH_KEY = bytes32(0xad348b32c79cd46ad46d61aede26d38affaee58f9a122f91eb271e08720464bf);
+}
diff --git a/forks/passport/src/contracts/modules/commons/Implementation.sol b/forks/passport/src/contracts/modules/commons/Implementation.sol
new file mode 100644
index 00000000..a1c65df9
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/Implementation.sol
@@ -0,0 +1,31 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+/**
+ * @dev Allows modules to access the implementation slot
+ */
+contract Implementation {
+  /**
+   * @notice Updates the Wallet implementation
+   * @param _imp New implementation address
+   * @dev The wallet implementation is stored on the storage slot
+   *   defined by the address of the wallet itself
+   *   WARNING updating this value may break the wallet and users
+   *   must be confident that the new implementation is safe.
+   */
+  function _setImplementation(address _imp) internal {
+    assembly {
+      sstore(address(), _imp)
+    }
+  }
+
+  /**
+   * @notice Returns the Wallet implementation
+   * @return _imp The address of the current Wallet implementation
+   */
+  function _getImplementation() internal view returns (address _imp) {
+    assembly {
+      _imp := sload(address())
+    }
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleAuth.sol b/forks/passport/src/contracts/modules/commons/ModuleAuth.sol
new file mode 100644
index 00000000..25cdc1d9
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleAuth.sol
@@ -0,0 +1,228 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../../utils/LibBytes.sol";
+import "../../utils/SignatureValidator.sol";
+import "../../interfaces/IERC1271Wallet.sol";
+
+import "./interfaces/IModuleAuth.sol";
+
+import "./ModuleERC165.sol";
+
+
+abstract contract ModuleAuth is IModuleAuth, ModuleERC165, SignatureValidator, IERC1271Wallet {
+  using LibBytes for bytes;
+
+  uint256 private constant FLAG_SIGNATURE = 0;
+  uint256 private constant FLAG_ADDRESS = 1;
+  uint256 private constant FLAG_DYNAMIC_SIGNATURE = 2;
+
+  bytes4 private constant SELECTOR_ERC1271_BYTES_BYTES = 0x20c13b0b;
+  bytes4 private constant SELECTOR_ERC1271_BYTES32_BYTES = 0x1626ba7e;
+
+  /**
+   * @notice Verify if signer is default wallet owner
+   * @param _hash       Hashed signed message
+   * @param _signature  Array of signatures with signers ordered
+   *                    like the the keys in the multisig configs
+   *
+   * @dev The signature must be solidity packed and contain the total number of owners,
+   *      the threshold, the weight and either the address or a signature for each owner.
+   *
+   *      Each weight & (address or signature) pair is prefixed by a flag that signals if such pair
+   *      contains an address or a signature. The aggregated weight of the signatures must surpass the threshold.
+   *
+   *      Flag types:
+   *        0x00 - Signature
+   *        0x01 - Address
+   *
+   *      E.g:
+   *      abi.encodePacked(
+   *        uint16 threshold,
+   *        uint8 01,  uint8 weight_1, address signer_1,
+   *        uint8 00, uint8 weight_2, bytes signature_2,
+   *        ...
+   *        uint8 01,  uint8 weight_5, address signer_5
+   *      )
+   */
+  function _signatureValidation(
+    bytes32 _hash,
+    bytes memory _signature
+  )
+    internal override returns (bool)
+  {
+    (bool verified, bool needsUpdate, bytes32 imageHash) = _signatureValidationWithUpdateCheck(_hash, _signature);
+    if (needsUpdate) {
+      updateImageHashInternal(imageHash);
+    }
+    return verified;
+  }
+
+  function _signatureValidationInternal(
+    bytes32 _hash,
+    bytes memory _signature
+  )
+    internal view returns (bool)
+  {
+    (bool verified, , ) = _signatureValidationWithUpdateCheck(_hash, _signature);
+    return verified;
+  }
+
+
+
+  function _signatureValidationWithUpdateCheck(
+    bytes32 _hash,
+    bytes memory _signature
+  )
+    internal view returns (bool, bool, bytes32)
+  {
+    (
+      uint16 threshold,  // required threshold signature
+      uint256 rindex     // read index
+    ) = _signature.readFirstUint16();
+
+    // Start image hash generation
+    bytes32 imageHash = bytes32(uint256(threshold));
+
+    // Acumulated weight of signatures
+    uint256 totalWeight;
+
+    // Iterate until the image is completed
+    while (rindex < _signature.length) {
+      // Read next item type and addrWeight
+      uint256 flag; uint256 addrWeight; address addr;
+      (flag, addrWeight, rindex) = _signature.readUint8Uint8(rindex);
+
+      if (flag == FLAG_ADDRESS) {
+        // Read plain address
+        (addr, rindex) = _signature.readAddress(rindex);
+      } else if (flag == FLAG_SIGNATURE) {
+        // Read single signature and recover signer
+        bytes memory signature;
+        (signature, rindex) = _signature.readBytes66(rindex);
+        addr = recoverSigner(_hash, signature);
+
+        // Acumulate total weight of the signature
+        totalWeight += addrWeight;
+      } else if (flag == FLAG_DYNAMIC_SIGNATURE) {
+        // Read signer
+        (addr, rindex) = _signature.readAddress(rindex);
+
+        // Read signature size
+        uint256 size;
+        (size, rindex) = _signature.readUint16(rindex);
+
+        // Read dynamic size signature
+        bytes memory signature;
+        (signature, rindex) = _signature.readBytes(rindex, size);
+        require(isValidSignature(_hash, addr, signature), "ModuleAuth#_signatureValidation: INVALID_SIGNATURE");
+
+        // Acumulate total weight of the signature
+        totalWeight += addrWeight;
+      } else {
+        revert("ModuleAuth#_signatureValidation INVALID_FLAG");
+      }
+
+      // Write weight and address to image
+      imageHash = keccak256(abi.encode(imageHash, addrWeight, addr));
+    }
+
+    (bool verified, bool needsUpdate) = _isValidImage(imageHash);
+    return ((totalWeight >= threshold && verified), needsUpdate, imageHash);
+  }
+
+
+  /**
+   * @notice Validates the signature image
+   * @param _imageHash Hashed image of signature
+   * @return true if the signature image is valid, and true if the image hash should be updated.
+   */
+  function _isValidImage(bytes32 _imageHash) internal view virtual returns (bool, bool);
+
+  /**
+   * @notice Will hash _data to be signed (similar to EIP-712)
+   * @param _digest Pre-final digest
+   * @return hashed data for this wallet
+   */
+  function _subDigest(bytes32 _digest) internal override view returns (bytes32) {
+    uint256 chainId; assembly { chainId := chainid() }
+    return keccak256(
+      abi.encodePacked(
+        "\x19\x01",
+        chainId,
+        address(this),
+        _digest
+      )
+    );
+  }
+
+  /**
+   * @notice Updates the signers configuration of the wallet
+   * @param _imageHash New required image hash of the signature
+   * @dev It is recommended to not have more than 200 signers as opcode repricing
+   *      could make transactions impossible to execute as all the signers must be
+   *      passed for each transaction.
+   */
+  // solhint-disable-next-line no-empty-blocks
+  function updateImageHashInternal(bytes32 _imageHash) internal virtual {
+    // Default implementation does nothing
+  }
+  
+
+
+  /**
+   * @notice Verifies whether the provided signature is valid with respect to the provided data
+   * @dev MUST return the correct magic value if the signature provided is valid for the provided data
+   *   > The bytes4 magic value to return when signature is valid is 0x20c13b0b : bytes4(keccak256("isValidSignature(bytes,bytes)"))
+   * @param _data       Arbitrary length data signed on the behalf of address(this)
+   * @param _signatures Signature byte array associated with _data.
+   *                    Encoded as abi.encode(Signature[], Configs)
+   * @return magicValue Magic value 0x20c13b0b if the signature is valid and 0x0 otherwise
+   */
+  function isValidSignature(
+    bytes calldata _data,
+    bytes calldata _signatures
+  ) external override view returns (bytes4) {
+    // Validate signatures
+    if (_signatureValidationInternal(_subDigest(keccak256(_data)), _signatures)) {
+      return SELECTOR_ERC1271_BYTES_BYTES;
+    }
+    return 0;
+  }
+
+  /**
+   * @notice Verifies whether the provided signature is valid with respect to the provided hash
+   * @dev MUST return the correct magic value if the signature provided is valid for the provided hash
+   *   > The bytes4 magic value to return when signature is valid is 0x1626ba7e : bytes4(keccak256("isValidSignature(bytes32,bytes)"))
+   * @param _hash       keccak256 hash that was signed
+   * @param _signatures Signature byte array associated with _data.
+   *                    Encoded as abi.encode(Signature[], Configs)
+   * @return magicValue Magic value 0x1626ba7e if the signature is valid and 0x0 otherwise
+   */
+  function isValidSignature(
+    bytes32 _hash,
+    bytes calldata _signatures
+  ) external override view returns (bytes4) {
+    // Validate signatures
+    if (_signatureValidationInternal(_subDigest(_hash), _signatures)) {
+      return SELECTOR_ERC1271_BYTES32_BYTES;
+    }
+    return 0;
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (
+      _interfaceID == type(IModuleAuth).interfaceId ||
+      _interfaceID == type(IERC1271Wallet).interfaceId
+    ) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleAuthDynamic.sol b/forks/passport/src/contracts/modules/commons/ModuleAuthDynamic.sol
new file mode 100644
index 00000000..505c4ed8
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleAuthDynamic.sol
@@ -0,0 +1,57 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./ModuleAuthUpgradable.sol";
+import "./ImageHashKey.sol";
+import "../../Wallet.sol";
+
+
+abstract contract ModuleAuthDynamic is ModuleAuthUpgradable {
+  bytes32 public immutable INIT_CODE_HASH;
+  address public immutable FACTORY;
+
+  constructor(address _factory, address _startupWalletImpl) {
+    // Build init code hash of the deployed wallets using that module
+    bytes32 initCodeHash = keccak256(abi.encodePacked(Wallet.creationCode, uint256(uint160(_startupWalletImpl))));
+
+    INIT_CODE_HASH = initCodeHash;
+    FACTORY = _factory;
+  }
+
+  /**
+   * @notice Validates the signature image with the salt used to deploy the contract
+   *         if there is no stored image hash. This will happen prior to the first meta 
+   *         transaction. Subsequently, validate the 
+   *         signature image with a valid image hash defined in the contract storage
+   * @param _imageHash Hash image of signature
+   * @return true if the signature image is valid, and true if the image hash needs to be updated
+   */
+  function _isValidImage(bytes32 _imageHash) internal view override returns (bool, bool) {
+    bytes32 storedImageHash = ModuleStorage.readBytes32(ImageHashKey.IMAGE_HASH_KEY);
+    if (storedImageHash == 0) {
+      // No image hash stored. Check that the image hash was used as the salt when 
+      // deploying the wallet proxy contract.
+      bool authenticated = address(
+        uint160(uint256(
+          keccak256(
+            abi.encodePacked(
+              bytes1(0xff),
+              FACTORY,
+              _imageHash,
+              INIT_CODE_HASH
+            )
+          )
+        ))
+      ) == address(this);
+      // Indicate need to update = true. This will trigger a call to store the image hash
+      return (authenticated, true);
+    }
+
+    // Image hash has been stored. 
+    return ((_imageHash != bytes32(0) && _imageHash == storedImageHash), false);
+  }
+}
+
+
+
diff --git a/forks/passport/src/contracts/modules/commons/ModuleAuthFixed.sol b/forks/passport/src/contracts/modules/commons/ModuleAuthFixed.sol
new file mode 100644
index 00000000..22cf41a4
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleAuthFixed.sol
@@ -0,0 +1,45 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./ModuleAuth.sol";
+import "../../Wallet.sol";
+
+/**
+ *  Implements ModuleAuth by validating the signature image against
+ *  the salt used to deploy the contract
+ *
+ *  This module allows wallets to be deployed with a default configuration
+ *  without using any aditional contract storage
+ */
+abstract contract ModuleAuthFixed is ModuleAuth {
+  bytes32 public immutable INIT_CODE_HASH;
+  address public immutable FACTORY;
+
+  constructor(address _factory) {
+    // Build init code hash of the deployed wallets using that module
+    bytes32 initCodeHash = keccak256(abi.encodePacked(Wallet.creationCode, uint256(uint160(address(this)))));
+
+    INIT_CODE_HASH = initCodeHash;
+    FACTORY = _factory;
+  }
+
+  /**
+   * @notice Validates the signature image with the salt used to deploy the contract
+   * @param _imageHash Hash image of signature
+   * @return true if the signature image is valid, and always false, indicating no updates required
+   */
+  function _isValidImage(bytes32 _imageHash) internal view override returns (bool, bool) {
+    return ((address(
+      uint160(uint256(
+        keccak256(
+          abi.encodePacked(
+            bytes1(0xff),
+            FACTORY,
+            _imageHash,
+            INIT_CODE_HASH
+          )
+        )
+      ))
+    ) == address(this)), false);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleAuthUpgradable.sol b/forks/passport/src/contracts/modules/commons/ModuleAuthUpgradable.sol
new file mode 100644
index 00000000..c3a80e95
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleAuthUpgradable.sol
@@ -0,0 +1,68 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./interfaces/IModuleAuthUpgradable.sol";
+
+import "./ModuleSelfAuth.sol";
+import "./ModuleAuth.sol";
+import "./ModuleStorage.sol";
+import "./ImageHashKey.sol";
+
+
+abstract contract ModuleAuthUpgradable is IModuleAuthUpgradable, ModuleAuth, ModuleSelfAuth {
+  event ImageHashUpdated(bytes32 newImageHash);
+
+  /**
+   * @notice Updates the signers configuration of the wallet
+   * @param _imageHash New required image hash of the signature
+   * @dev It is recommended to not have more than 200 signers as opcode repricing
+   *      could make transactions impossible to execute as all the signers must be
+   *      passed for each transaction.
+   */
+  function updateImageHash(bytes32 _imageHash) external override onlySelf {
+    updateImageHashInternal(_imageHash);
+  }
+
+  /**
+   * @notice Returns the current image hash of the wallet
+   */
+  function imageHash() external override view returns (bytes32) {
+    return ModuleStorage.readBytes32(ImageHashKey.IMAGE_HASH_KEY);
+  }
+
+  /**
+   * @notice Validates the signature image with a valid image hash defined
+   *   in the contract storage
+   * @param _imageHash Hash image of signature
+   * @return true if the signature image is valid, and always false indicating no updates required
+   */
+  function _isValidImage(bytes32 _imageHash) internal virtual view override returns (bool, bool) {
+    return ((_imageHash != bytes32(0) && _imageHash == ModuleStorage.readBytes32(ImageHashKey.IMAGE_HASH_KEY)), false);
+  }
+
+  /**
+   * @notice Updates the signers configuration of the wallet
+   * @param _imageHash New required image hash of the signature
+   * @dev It is recommended to not have more than 200 signers as opcode repricing
+   *      could make transactions impossible to execute as all the signers must be
+   *      passed for each transaction.
+   */
+  function updateImageHashInternal(bytes32 _imageHash) internal override {
+    require(_imageHash != bytes32(0), "ModuleAuthUpgradable#updateImageHash INVALID_IMAGE_HASH");
+    ModuleStorage.writeBytes32(ImageHashKey.IMAGE_HASH_KEY, _imageHash);
+    emit ImageHashUpdated(_imageHash);
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (_interfaceID == type(IModuleAuthUpgradable).interfaceId) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleCalls.sol b/forks/passport/src/contracts/modules/commons/ModuleCalls.sol
new file mode 100644
index 00000000..3574c99e
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleCalls.sol
@@ -0,0 +1,192 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./ModuleSelfAuth.sol";
+import "./ModuleStorage.sol";
+import "./ModuleERC165.sol";
+import "./NonceKey.sol";
+
+import "./interfaces/IModuleCalls.sol";
+import "./interfaces/IModuleAuth.sol";
+
+
+abstract contract ModuleCalls is IModuleCalls, IModuleAuth, ModuleERC165, ModuleSelfAuth {
+  uint256 private constant NONCE_BITS = 96;
+  bytes32 private constant NONCE_MASK = bytes32((1 << NONCE_BITS) - 1);
+
+  /**
+   * @notice Returns the next nonce of the default nonce space
+   * @dev The default nonce space is 0x00
+   * @return The next nonce
+   */
+  function nonce() external override virtual view returns (uint256) {
+    return readNonce(0);
+  }
+
+  /**
+   * @notice Returns the next nonce of the given nonce space
+   * @param _space Nonce space, each space keeps an independent nonce count
+   * @return The next nonce
+   */
+  function readNonce(uint256 _space) public override virtual view returns (uint256) {
+    return uint256(ModuleStorage.readBytes32Map(NonceKey.NONCE_KEY, bytes32(_space)));
+  }
+
+  /**
+   * @notice Changes the next nonce of the given nonce space
+   * @param _space Nonce space, each space keeps an independent nonce count
+   * @param _nonce Nonce to write on the space
+   */
+  function _writeNonce(uint256 _space, uint256 _nonce) private {
+    ModuleStorage.writeBytes32Map(NonceKey.NONCE_KEY, bytes32(_space), bytes32(_nonce));
+  }
+
+  /**
+   * @notice Allow wallet owner to execute an action
+   * @dev Relayers must ensure that the gasLimit specified for each transaction
+   *      is acceptable to them. A user could specify large enough that it could
+   *      consume all the gas available.
+   * @param _txs        Transactions to process
+   * @param _nonce      Signature nonce (may contain an encoded space)
+   * @param _signature  Encoded signature
+   */
+  function execute(
+    Transaction[] memory _txs,
+    uint256 _nonce,
+    bytes memory _signature
+  ) public override virtual {
+    // Validate and update nonce
+    _validateNonce(_nonce);
+
+    // Hash transaction bundle
+    bytes32 txHash = _subDigest(keccak256(abi.encode(_nonce, _txs)));
+
+    // Verify that signatures are valid
+    require(
+      _signatureValidation(txHash, _signature),
+      "ModuleCalls#execute: INVALID_SIGNATURE"
+    );
+
+    // Execute the transactions
+    _execute(txHash, _txs);
+  }
+
+  /**
+   * @notice Allow wallet to execute an action
+   *   without signing the message
+   * @param _txs  Transactions to execute
+   */
+  function selfExecute(
+    Transaction[] memory _txs
+  ) public override virtual onlySelf {
+    // Hash transaction bundle
+    bytes32 txHash = _subDigest(keccak256(abi.encode('self:', _txs)));
+
+    // Execute the transactions
+    _execute(txHash, _txs);
+  }
+
+  /**
+   * @notice Executes a list of transactions
+   * @param _txHash  Hash of the batch of transactions
+   * @param _txs  Transactions to execute
+   */
+  function _execute(
+    bytes32 _txHash,
+    Transaction[] memory _txs
+  ) private {
+    // Execute transaction
+    for (uint256 i = 0; i < _txs.length; i++) {
+      Transaction memory transaction = _txs[i];
+
+      bool success;
+      bytes memory result;
+
+      require(gasleft() >= transaction.gasLimit, "ModuleCalls#_execute: NOT_ENOUGH_GAS");
+
+      if (transaction.delegateCall) {
+        (success, result) = transaction.target.delegatecall{
+          gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+        }(transaction.data);
+      } else {
+        (success, result) = transaction.target.call{
+          value: transaction.value,
+          gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+        }(transaction.data);
+      }
+
+      if (success) {
+        emit TxExecuted(_txHash);
+      } else {
+        _revertBytes(transaction, _txHash, result);
+      }
+    }
+  }
+
+  /**
+   * @notice Verify if a nonce is valid
+   * @param _rawNonce Nonce to validate (may contain an encoded space)
+   * @dev A valid nonce must be above the last one used
+   *   with a maximum delta of 100
+   */
+  function _validateNonce(uint256 _rawNonce) private {
+    // Retrieve current nonce for this wallet
+    (uint256 space, uint256 providedNonce) = _decodeNonce(_rawNonce);
+    uint256 currentNonce = readNonce(space);
+
+    // Verify if nonce is valid
+    require(
+      providedNonce == currentNonce,
+      "MainModule#_auth: INVALID_NONCE"
+    );
+
+    // Update signature nonce
+    uint256 newNonce = providedNonce + 1;
+    _writeNonce(space, newNonce);
+    emit NonceChange(space, newNonce);
+  }
+
+  /**
+   * @notice Logs a failed transaction, reverts if the transaction is not optional
+   * @param _tx      Transaction that is reverting
+   * @param _txHash  Hash of the transaction
+   * @param _reason  Encoded revert message
+   */
+  function _revertBytes(
+    Transaction memory _tx,
+    bytes32 _txHash,
+    bytes memory _reason
+  ) internal {
+    if (_tx.revertOnError) {
+      assembly { revert(add(_reason, 0x20), mload(_reason)) }
+    } else {
+      emit TxFailed(_txHash, _reason);
+    }
+  }
+
+  /**
+   * @notice Decodes a raw nonce
+   * @dev A raw nonce is encoded using the first 160 bits for the space
+   *  and the last 96 bits for the nonce
+   * @param _rawNonce Nonce to be decoded
+   * @return _space The nonce space of the raw nonce
+   * @return _nonce The nonce of the raw nonce
+   */
+  function _decodeNonce(uint256 _rawNonce) private pure returns (uint256 _space, uint256 _nonce) {
+    _nonce = uint256(bytes32(_rawNonce) & NONCE_MASK);
+    _space = _rawNonce >> NONCE_BITS;
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (_interfaceID == type(IModuleCalls).interfaceId) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleCreator.sol b/forks/passport/src/contracts/modules/commons/ModuleCreator.sol
new file mode 100644
index 00000000..28b21784
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleCreator.sol
@@ -0,0 +1,36 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./interfaces/IModuleCreator.sol";
+
+import "./ModuleSelfAuth.sol";
+import "./ModuleERC165.sol";
+
+
+contract ModuleCreator is IModuleCreator, ModuleERC165, ModuleSelfAuth {
+  event CreatedContract(address _contract);
+
+  /**
+   * @notice Creates a contract forwarding eth value
+   * @param _code Creation code of the contract
+   * @return addr The address of the created contract
+   */
+  function createContract(bytes memory _code) public override payable onlySelf returns (address addr) {
+    assembly { addr := create(callvalue(), add(_code, 32), mload(_code)) }
+    require(addr != address(0), 'ModuleCreator: creation failed');
+    emit CreatedContract(addr);
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (_interfaceID == type(IModuleCreator).interfaceId) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleERC165.sol b/forks/passport/src/contracts/modules/commons/ModuleERC165.sol
new file mode 100644
index 00000000..af033f1c
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleERC165.sol
@@ -0,0 +1,18 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+abstract contract ModuleERC165 {
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @dev Adding new hooks will not lead to them being reported by this function
+   *      without upgrading the wallet. In addition, developpers must ensure that 
+   *      all inherited contracts by the mainmodule don't conflict and are accounted
+   *      to be supported by the supportsInterface method.
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) virtual public pure returns (bool) {
+    return _interfaceID == this.supportsInterface.selector;
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleHooks.sol b/forks/passport/src/contracts/modules/commons/ModuleHooks.sol
new file mode 100644
index 00000000..d3cb9859
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleHooks.sol
@@ -0,0 +1,146 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./interfaces/IModuleHooks.sol";
+
+import "./ModuleSelfAuth.sol";
+import "./ModuleStorage.sol";
+import "./ModuleERC165.sol";
+
+import "../../interfaces/receivers/IERC1155Receiver.sol";
+import "../../interfaces/receivers/IERC721Receiver.sol";
+import "../../interfaces/receivers/IERC223Receiver.sol";
+
+
+contract ModuleHooks is IERC1155Receiver, IERC721Receiver, IModuleHooks, ModuleERC165, ModuleSelfAuth {
+  // Randomly generated to avoid collisions, with:
+  // xxd -len 32 -plain -cols 32 /dev/urandom
+  bytes32 private constant HOOKS_KEY = bytes32(0x5f198cb61cfcc209f357b4ede4ad2218c53e3b4cb7e8fa1e8b0ec5e3951acbaa);
+
+  /**
+   * @notice Reads the implementation hook of a signature
+   * @param _signature Signature function
+   * @return The address of the implementation hook, address(0) if none
+  */
+  function readHook(bytes4 _signature) external override view returns (address) {
+    return _readHook(_signature);
+  }
+
+  /**
+   * @notice Adds a new hook to handle a given function selector
+   * @param _signature Signature function linked to the hook
+   * @param _implementation Hook implementation contract
+   * @dev Can't overwrite hooks that are part of the mainmodule (those defined below)
+   */
+  function addHook(bytes4 _signature, address _implementation) external override onlySelf {
+    require(_readHook(_signature) == address(0), "ModuleHooks#addHook: HOOK_ALREADY_REGISTERED");
+    _writeHook(_signature, _implementation);
+  }
+
+  /**
+   * @notice Removes a registered hook
+   * @param _signature Signature function linked to the hook
+   * @dev Can't remove hooks that are part of the mainmodule (those defined below) 
+   *      without upgrading the wallet
+   */
+  function removeHook(bytes4 _signature) external override onlySelf {
+    require(_readHook(_signature) != address(0), "ModuleHooks#removeHook: HOOK_NOT_REGISTERED");
+    _writeHook(_signature, address(0));
+  }
+
+  /**
+   * @notice Reads the implementation hook of a signature
+   * @param _signature Signature function
+   * @return The address of the implementation hook, address(0) if none
+  */
+  function _readHook(bytes4 _signature) private view returns (address) {
+    return address(uint160(uint256(ModuleStorage.readBytes32Map(HOOKS_KEY, _signature))));
+  }
+
+  /**
+   * @notice Writes the implementation hook of a signature
+   * @param _signature Signature function
+   * @param _implementation Hook implementation contract
+  */
+  function _writeHook(bytes4 _signature, address _implementation) private {
+    ModuleStorage.writeBytes32Map(HOOKS_KEY, _signature, bytes32(uint256(uint160(_implementation))));
+  }
+
+  /**
+   * @notice Handle the receipt of a single ERC1155 token type.
+   * @return `bytes4(keccak256("onERC1155Received(address,address,uint256,uint256,bytes)"))`
+   */
+  function onERC1155Received(
+    address,
+    address,
+    uint256,
+    uint256,
+    bytes calldata
+  ) external override returns (bytes4) {
+    return ModuleHooks.onERC1155Received.selector;
+  }
+
+  /**
+   * @notice Handle the receipt of multiple ERC1155 token types.
+   * @return `bytes4(keccak256("onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)"))`
+   */
+  function onERC1155BatchReceived(
+    address,
+    address,
+    uint256[] calldata,
+    uint256[] calldata,
+    bytes calldata
+  ) external override returns (bytes4) {
+    return ModuleHooks.onERC1155BatchReceived.selector;
+  }
+
+  /**
+   * @notice Handle the receipt of a single ERC721 token.
+   * @return `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))`
+   */
+  function onERC721Received(address, address, uint256, bytes calldata) external override returns (bytes4) {
+    return ModuleHooks.onERC721Received.selector;
+  }
+
+  /**
+   * @notice Routes fallback calls through hooks
+   */
+  fallback() external payable {
+      if (msg.data.length >= 4) {
+      address target = _readHook(msg.sig);
+      if (target != address(0)) {
+        (bool success, bytes memory result) = target.delegatecall(msg.data);
+        assembly {
+          if iszero(success)  {
+            revert(add(result, 0x20), mload(result))
+          }
+
+          return(add(result, 0x20), mload(result))
+        }
+      }
+    }
+  }
+
+  /**
+   * @notice Allows the wallet to receive ETH
+   */
+  receive() external payable { }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (
+      _interfaceID == type(IModuleHooks).interfaceId ||
+      _interfaceID == type(IERC1155Receiver).interfaceId ||
+      _interfaceID == type(IERC721Receiver).interfaceId ||
+      _interfaceID == type(IERC223Receiver).interfaceId
+    ) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleIgnoreAuthUpgradable.sol b/forks/passport/src/contracts/modules/commons/ModuleIgnoreAuthUpgradable.sol
new file mode 100644
index 00000000..7198e42e
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleIgnoreAuthUpgradable.sol
@@ -0,0 +1,61 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./interfaces/IModuleAuthUpgradable.sol";
+
+import "./ModuleSelfAuth.sol";
+import "./ModuleAuth.sol";
+import "./ImageHashKey.sol";
+import "./ModuleStorage.sol";
+
+
+/**
+  @notice Implements ModuleAuthUpgradable but ignores the validity of the signature
+    should only be used during gas estimation.
+*/
+abstract contract ModuleIgnoreAuthUpgradable is IModuleAuthUpgradable, ModuleAuth, ModuleSelfAuth {
+  event ImageHashUpdated(bytes32 newImageHash);
+
+  /**
+   * @notice Updates the signers configuration of the wallet
+   * @param _imageHash New required image hash of the signature
+   * @dev It is recommended to not have more than 200 signers as opcode repricing
+   *      could make transactions impossible to execute as all the signers must be
+   *      passed for each transaction.
+   */
+  function updateImageHash(bytes32 _imageHash) external override virtual onlySelf {
+    require(_imageHash != bytes32(0), "ModuleAuthUpgradable#updateImageHash INVALID_IMAGE_HASH");
+    ModuleStorage.writeBytes32(ImageHashKey.IMAGE_HASH_KEY, _imageHash);
+    emit ImageHashUpdated(_imageHash);
+  }
+
+  /**
+   * @notice Returns the current image hash of the wallet
+   */
+  function imageHash() external override virtual view returns (bytes32) {
+    return ModuleStorage.readBytes32(ImageHashKey.IMAGE_HASH_KEY);
+  }
+
+  /**
+   * @notice Removes the signature validation from the module, by returning true for any _imageHash
+   * @param _imageHash Hash image of signature
+   * @return true always, hence always valid, and false always, indicating no update required
+   */
+  function _isValidImage(bytes32 _imageHash) internal override view returns (bool, bool) {
+    // Still validates the imageHash using the original mechanism for a more acurate estimation
+    return (true, false);
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (_interfaceID == type(IModuleAuthUpgradable).interfaceId) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleIgnoreNonceCalls.sol b/forks/passport/src/contracts/modules/commons/ModuleIgnoreNonceCalls.sol
new file mode 100644
index 00000000..e3cbdb07
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleIgnoreNonceCalls.sol
@@ -0,0 +1,199 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./ModuleSelfAuth.sol";
+import "./ModuleStorage.sol";
+import "./ModuleERC165.sol";
+import "./NonceKey.sol";
+
+import "./interfaces/IModuleCalls.sol";
+import "./interfaces/IModuleAuth.sol";
+
+/**
+  @notice Implements ModuleCalls but ignores the validity of the nonce
+    should only be used during gas estimation.
+*/
+abstract contract ModuleIgnoreNonceCalls is IModuleCalls, IModuleAuth, ModuleERC165, ModuleSelfAuth {
+  uint256 private constant NONCE_BITS = 96;
+  bytes32 private constant NONCE_MASK = bytes32((1 << NONCE_BITS) - 1);
+
+  /**
+   * @notice Returns the next nonce of the default nonce space
+   * @dev The default nonce space is 0x00
+   * @return The next nonce
+   */
+  function nonce() external override virtual view returns (uint256) {
+    return readNonce(0);
+  }
+
+  /**
+   * @notice Returns the next nonce of the given nonce space
+   * @param _space Nonce space, each space keeps an independent nonce count
+   * @return The next nonce
+   */
+  function readNonce(uint256 _space) public override virtual view returns (uint256) {
+    return uint256(ModuleStorage.readBytes32Map(NonceKey.NONCE_KEY, bytes32(_space)));
+  }
+
+  /**
+   * @notice Changes the next nonce of the given nonce space
+   * @param _space Nonce space, each space keeps an independent nonce count
+   * @param _nonce Nonce to write on the space
+   */
+  function _writeNonce(uint256 _space, uint256 _nonce) private {
+    ModuleStorage.writeBytes32Map(NonceKey.NONCE_KEY, bytes32(_space), bytes32(_nonce));
+  }
+
+  /**
+   * @notice Allow wallet owner to execute an action
+   * @dev Relayers must ensure that the gasLimit specified for each transaction
+   *      is acceptable to them. A user could specify large enough that it could
+   *      consume all the gas available.
+   * @param _txs        Transactions to process
+   * @param _nonce      Signature nonce (may contain an encoded space)
+   * @param _signature  Encoded signature
+   */
+  function execute(
+    Transaction[] memory _txs,
+    uint256 _nonce,
+    bytes memory _signature
+  ) public override virtual {
+    // Validate and update nonce
+    _validateNonce(_nonce);
+
+    // Hash transaction bundle
+    bytes32 txHash = _subDigest(keccak256(abi.encode(_nonce, _txs)));
+
+    // Verify that signatures are valid
+    require(
+      _signatureValidation(txHash, _signature),
+      "ModuleCalls#execute: INVALID_SIGNATURE"
+    );
+
+    // Execute the transactions
+    _execute(txHash, _txs);
+  }
+
+  /**
+   * @notice Allow wallet to execute an action
+   *   without signing the message
+   * @param _txs  Transactions to execute
+   */
+  function selfExecute(
+    Transaction[] memory _txs
+  ) public override virtual onlySelf {
+    // Hash transaction bundle
+    bytes32 txHash = _subDigest(keccak256(abi.encode('self:', _txs)));
+
+    // Execute the transactions
+    _execute(txHash, _txs);
+  }
+
+  /**
+   * @notice Executes a list of transactions
+   * @param _txHash  Hash of the batch of transactions
+   * @param _txs  Transactions to execute
+   */
+  function _execute(
+    bytes32 _txHash,
+    Transaction[] memory _txs
+  ) private {
+    // Execute transaction
+    for (uint256 i = 0; i < _txs.length; i++) {
+      Transaction memory transaction = _txs[i];
+
+      bool success;
+      bytes memory result;
+
+      require(gasleft() >= transaction.gasLimit, "ModuleCalls#_execute: NOT_ENOUGH_GAS");
+
+      if (transaction.delegateCall) {
+        (success, result) = transaction.target.delegatecall{
+          gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+        }(transaction.data);
+      } else {
+        (success, result) = transaction.target.call{
+          value: transaction.value,
+          gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+        }(transaction.data);
+      }
+
+      if (success) {
+        emit TxExecuted(_txHash);
+      } else {
+        _revertBytes(transaction, _txHash, result);
+      }
+    }
+  }
+
+  /**
+   * @notice Verify if a nonce is valid
+   * @param _rawNonce Nonce to validate (may contain an encoded space)
+   * @dev A valid nonce must be above the last one used
+   *   with a maximum delta of 100
+   */
+  function _validateNonce(uint256 _rawNonce) private {
+    // Retrieve current nonce for this wallet
+    (uint256 space, uint256 providedNonce) = _decodeNonce(_rawNonce);
+    uint256 currentNonce = readNonce(space);
+
+    // Verify if nonce is valid
+    // 
+    // Skip nonce validation for gas estimation, but keep the statement to
+    // arrive at a closer gas expenditure when compared with the version in
+    // ModuleCalls.sol
+    require(
+      (providedNonce == currentNonce) || true,
+      "MainModule#_auth: INVALID_NONCE"
+    );
+
+    // Update signature nonce
+    uint256 newNonce = providedNonce + 1;
+    _writeNonce(space, newNonce);
+    emit NonceChange(space, newNonce);
+  }
+
+  /**
+   * @notice Logs a failed transaction, reverts if the transaction is not optional
+   * @param _tx      Transaction that is reverting
+   * @param _txHash  Hash of the transaction
+   * @param _reason  Encoded revert message
+   */
+  function _revertBytes(
+    Transaction memory _tx,
+    bytes32 _txHash,
+    bytes memory _reason
+  ) internal {
+    if (_tx.revertOnError) {
+      assembly { revert(add(_reason, 0x20), mload(_reason)) }
+    } else {
+      emit TxFailed(_txHash, _reason);
+    }
+  }
+
+  /**
+   * @notice Decodes a raw nonce
+   * @dev A raw nonce is encoded using the first 160 bits for the space
+   *  and the last 96 bits for the nonce
+   * @param _rawNonce Nonce to be decoded
+   * @return _space The nonce space of the raw nonce
+   * @return _nonce The nonce of the raw nonce
+   */
+  function _decodeNonce(uint256 _rawNonce) private pure returns (uint256 _space, uint256 _nonce) {
+    _nonce = uint256(bytes32(_rawNonce) & NONCE_MASK);
+    _space = _rawNonce >> NONCE_BITS;
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (_interfaceID == type(IModuleCalls).interfaceId) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleReceivers.sol b/forks/passport/src/contracts/modules/commons/ModuleReceivers.sol
new file mode 100644
index 00000000..27788146
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleReceivers.sol
@@ -0,0 +1,68 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./ModuleERC165.sol";
+
+import "../../interfaces/receivers/IERC1155Receiver.sol";
+import "../../interfaces/receivers/IERC721Receiver.sol";
+import "../../interfaces/receivers/IERC223Receiver.sol";
+
+contract ModuleReceivers is IERC1155Receiver, IERC721Receiver, ModuleERC165 {
+  /**
+   * @notice Handle the receipt of a single ERC1155 token type.
+   * @return `bytes4(keccak256("onERC1155Received(address,address,uint256,uint256,bytes)"))`
+   */
+  function onERC1155Received(
+    address,
+    address,
+    uint256,
+    uint256,
+    bytes calldata
+  ) external override returns (bytes4) {
+    return ModuleReceivers.onERC1155Received.selector;
+  }
+
+  /**
+   * @notice Handle the receipt of multiple ERC1155 token types.
+   * @return `bytes4(keccak256("onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)"))`
+   */
+  function onERC1155BatchReceived(
+    address,
+    address,
+    uint256[] calldata,
+    uint256[] calldata,
+    bytes calldata
+  ) external override returns (bytes4) {
+    return ModuleReceivers.onERC1155BatchReceived.selector;
+  }
+
+  /**
+   * @notice Handle the receipt of a single ERC721 token.
+   * @return `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))`
+   */
+  function onERC721Received(address, address, uint256, bytes calldata) external override returns (bytes4) {
+    return ModuleReceivers.onERC721Received.selector;
+  }
+
+  /**
+   * @notice Allows the wallet to receive ETH
+   */
+  receive() external payable { }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (
+      _interfaceID == type(IERC1155Receiver).interfaceId ||
+      _interfaceID == type(IERC721Receiver).interfaceId ||
+      _interfaceID == type(IERC223Receiver).interfaceId
+    ) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleSelfAuth.sol b/forks/passport/src/contracts/modules/commons/ModuleSelfAuth.sol
new file mode 100644
index 00000000..9a59d7b2
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleSelfAuth.sol
@@ -0,0 +1,10 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract ModuleSelfAuth {
+  modifier onlySelf() {
+    require(msg.sender == address(this), "ModuleSelfAuth#onlySelf: NOT_AUTHORIZED");
+    _;
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleStorage.sol b/forks/passport/src/contracts/modules/commons/ModuleStorage.sol
new file mode 100644
index 00000000..ad98b9b4
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleStorage.sol
@@ -0,0 +1,23 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+library ModuleStorage {
+  function writeBytes32(bytes32 _key, bytes32 _val) internal {
+    assembly { sstore(_key, _val) }
+  }
+
+  function readBytes32(bytes32 _key) internal view returns (bytes32 val) {
+    assembly { val := sload(_key) }
+  }
+
+  function writeBytes32Map(bytes32 _key, bytes32 _subKey, bytes32 _val) internal {
+    bytes32 key = keccak256(abi.encode(_key, _subKey));
+    assembly { sstore(key, _val) }
+  }
+
+  function readBytes32Map(bytes32 _key, bytes32 _subKey) internal view returns (bytes32 val) {
+    bytes32 key = keccak256(abi.encode(_key, _subKey));
+    assembly { val := sload(key) }
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/ModuleUpdate.sol b/forks/passport/src/contracts/modules/commons/ModuleUpdate.sol
new file mode 100644
index 00000000..044e3ef4
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/ModuleUpdate.sol
@@ -0,0 +1,41 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./interfaces/IModuleUpdate.sol";
+
+import "./Implementation.sol";
+import "./ModuleSelfAuth.sol";
+import "./ModuleERC165.sol";
+
+import "../../utils/LibAddress.sol";
+
+
+contract ModuleUpdate is IModuleUpdate, ModuleERC165, ModuleSelfAuth, Implementation {
+  using LibAddress for address;
+
+  event ImplementationUpdated(address newImplementation);
+
+  /**
+   * @notice Updates the implementation of the base wallet
+   * @param _implementation New main module implementation
+   * @dev WARNING Updating the implementation can brick the wallet
+   */
+  function updateImplementation(address _implementation) external override onlySelf {
+    require(_implementation.isContract(), "ModuleUpdate#updateImplementation: INVALID_IMPLEMENTATION");
+    _setImplementation(_implementation);
+    emit ImplementationUpdated(_implementation);
+  }
+
+  /**
+   * @notice Query if a contract implements an interface
+   * @param _interfaceID The interface identifier, as specified in ERC-165
+   * @return `true` if the contract implements `_interfaceID`
+   */
+  function supportsInterface(bytes4 _interfaceID) public override virtual pure returns (bool) {
+    if (_interfaceID == type(IModuleUpdate).interfaceId) {
+      return true;
+    }
+
+    return super.supportsInterface(_interfaceID);
+  }
+}
diff --git a/forks/passport/src/contracts/modules/commons/NonceKey.sol b/forks/passport/src/contracts/modules/commons/NonceKey.sol
new file mode 100644
index 00000000..64539afe
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/NonceKey.sol
@@ -0,0 +1,9 @@
+
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+library NonceKey {
+  // Randomly generated to avoid collisions, with:
+  // xxd -len 32 -plain -cols 32 /dev/urandom
+  bytes32 internal constant NONCE_KEY = bytes32(0xc40e2218089ef03fc40794d84d38778f688da53b98c9236b084936bfafc9a601);
+}
diff --git a/forks/passport/src/contracts/modules/commons/interfaces/IModuleAuth.sol b/forks/passport/src/contracts/modules/commons/interfaces/IModuleAuth.sol
new file mode 100644
index 00000000..233ac2c7
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/interfaces/IModuleAuth.sol
@@ -0,0 +1,25 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+abstract contract IModuleAuth {
+  /**
+   * @notice Hashed _data to be signed
+   * @param _digest Pre-final digest
+   * @return hashed data for this wallet
+   */
+  function _subDigest(
+    bytes32 _digest
+  ) internal virtual view returns (bytes32);
+
+  /**
+   * @notice Verify if signer is default wallet owner
+   * @param _hash Hashed signed message
+   * @param _signature Encoded signature
+   * @return True is the signature is valid
+   */
+  function _signatureValidation(
+    bytes32 _hash,
+    bytes memory _signature
+  ) internal virtual returns (bool);
+}
diff --git a/forks/passport/src/contracts/modules/commons/interfaces/IModuleAuthUpgradable.sol b/forks/passport/src/contracts/modules/commons/interfaces/IModuleAuthUpgradable.sol
new file mode 100644
index 00000000..f26ce97f
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/interfaces/IModuleAuthUpgradable.sol
@@ -0,0 +1,16 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IModuleAuthUpgradable {
+  /**
+   * @notice Updates the signers configuration of the wallet
+   * @param _imageHash New required image hash of the signature
+   */
+  function updateImageHash(bytes32 _imageHash) external;
+
+  /**
+   * @notice Returns the current image hash of the wallet
+   */
+  function imageHash() external view returns (bytes32);
+}
diff --git a/forks/passport/src/contracts/modules/commons/interfaces/IModuleCalls.sol b/forks/passport/src/contracts/modules/commons/interfaces/IModuleCalls.sol
new file mode 100644
index 00000000..ad0e78f4
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/interfaces/IModuleCalls.sol
@@ -0,0 +1,55 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IModuleCalls {
+  // Events
+  event NonceChange(uint256 _space, uint256 _newNonce);
+  event TxFailed(bytes32 _tx, bytes _reason);
+  event TxExecuted(bytes32 _tx) anonymous;
+
+  // Transaction structure
+  struct Transaction {
+    bool delegateCall;   // Performs delegatecall
+    bool revertOnError;  // Reverts transaction bundle if tx fails
+    uint256 gasLimit;    // Maximum gas to be forwarded
+    address target;      // Address of the contract to call
+    uint256 value;       // Amount of ETH to pass with the call
+    bytes data;          // calldata to pass
+  }
+
+  /**
+   * @notice Returns the next nonce of the default nonce space
+   * @dev The default nonce space is 0x00
+   * @return The next nonce
+   */
+  function nonce() external view returns (uint256);
+
+  /**
+   * @notice Returns the next nonce of the given nonce space
+   * @param _space Nonce space, each space keeps an independent nonce count
+   * @return The next nonce
+   */
+  function readNonce(uint256 _space) external view returns (uint256);
+
+  /**
+   * @notice Allow wallet owner to execute an action
+   * @param _txs        Transactions to process
+   * @param _nonce      Signature nonce (may contain an encoded space)
+   * @param _signature  Encoded signature
+   */
+  function execute(
+    Transaction[] calldata _txs,
+    uint256 _nonce,
+    bytes calldata _signature
+  ) external;
+
+  /**
+   * @notice Allow wallet to execute an action
+   *   without signing the message
+   * @param _txs  Transactions to execute
+   */
+  function selfExecute(
+    Transaction[] calldata _txs
+  ) external;
+}
diff --git a/forks/passport/src/contracts/modules/commons/interfaces/IModuleCreator.sol b/forks/passport/src/contracts/modules/commons/interfaces/IModuleCreator.sol
new file mode 100644
index 00000000..63c490ec
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/interfaces/IModuleCreator.sol
@@ -0,0 +1,12 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IModuleCreator {
+  /**
+   * @notice Creates a contract forwarding eth value
+   * @param _code Creation code of the contract
+   * @return addr The address of the created contract
+   */
+  function createContract(bytes calldata _code) external payable returns (address addr);
+}
diff --git a/forks/passport/src/contracts/modules/commons/interfaces/IModuleHooks.sol b/forks/passport/src/contracts/modules/commons/interfaces/IModuleHooks.sol
new file mode 100644
index 00000000..5ab2ae56
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/interfaces/IModuleHooks.sol
@@ -0,0 +1,25 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IModuleHooks {
+  /**
+   * @notice Reads the implementation hook of a signature
+   * @param _signature Signature function
+   * @return The address of the implementation hook, address(0) if none
+  */
+  function readHook(bytes4 _signature) external view returns (address);
+
+  /**
+   * @notice Adds a new hook to handle a given function selector
+   * @param _signature Signature function linked to the hook
+   * @param _implementation Hook implementation contract
+   */
+  function addHook(bytes4 _signature, address _implementation) external;
+
+  /**
+   * @notice Removes a registered hook
+   * @param _signature Signature function linked to the hook
+   */
+  function removeHook(bytes4 _signature) external;
+}
diff --git a/forks/passport/src/contracts/modules/commons/interfaces/IModuleUpdate.sol b/forks/passport/src/contracts/modules/commons/interfaces/IModuleUpdate.sol
new file mode 100644
index 00000000..ed06243a
--- /dev/null
+++ b/forks/passport/src/contracts/modules/commons/interfaces/IModuleUpdate.sol
@@ -0,0 +1,12 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+interface IModuleUpdate {
+  /**
+   * @notice Updates the implementation of the base wallet
+   * @param _implementation New main module implementation
+   * @dev WARNING Updating the implementation can brick the wallet
+   */
+  function updateImplementation(address _implementation) external;
+}
diff --git a/forks/passport/src/contracts/modules/utils/GasEstimator.sol b/forks/passport/src/contracts/modules/utils/GasEstimator.sol
new file mode 100644
index 00000000..4447868e
--- /dev/null
+++ b/forks/passport/src/contracts/modules/utils/GasEstimator.sol
@@ -0,0 +1,16 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+contract GasEstimator {
+  function estimate(
+    address _to,
+    bytes calldata _data
+  ) external returns (bool success, bytes memory result, uint256 gas) {
+    // solhint-disable
+    uint256 initialGas = gasleft();
+    (success, result) = _to.call(_data);
+    gas = initialGas - gasleft();
+    // solhint-enable
+  }
+}
diff --git a/forks/passport/src/contracts/modules/utils/MultiCallUtils.sol b/forks/passport/src/contracts/modules/utils/MultiCallUtils.sol
new file mode 100644
index 00000000..bb7d3174
--- /dev/null
+++ b/forks/passport/src/contracts/modules/utils/MultiCallUtils.sol
@@ -0,0 +1,99 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../commons/interfaces/IModuleCalls.sol";
+
+
+contract MultiCallUtils {
+  function multiCall(
+    IModuleCalls.Transaction[] memory _txs
+  ) public payable returns (
+    bool[] memory _successes,
+    bytes[] memory _results
+  ) {
+    _successes = new bool[](_txs.length);
+    _results = new bytes[](_txs.length);
+
+    for (uint256 i = 0; i < _txs.length; i++) {
+      IModuleCalls.Transaction memory transaction = _txs[i];
+
+      require(!transaction.delegateCall, 'MultiCallUtils#multiCall: delegateCall not allowed');
+      require(gasleft() >= transaction.gasLimit, "MultiCallUtils#multiCall: NOT_ENOUGH_GAS");
+
+      // solhint-disable
+      (_successes[i], _results[i]) = transaction.target.call{
+        value: transaction.value,
+        gas: transaction.gasLimit == 0 ? gasleft() : transaction.gasLimit
+      }(transaction.data);
+      // solhint-enable
+
+      require(_successes[i] || !_txs[i].revertOnError, 'MultiCallUtils#multiCall: CALL_REVERTED');
+    }
+  }
+
+  // ///
+  // Globals
+  // ///
+
+  function callBlockhash(uint256 _i) external view returns (bytes32) {
+    return blockhash(_i);
+  }
+
+  function callCoinbase() external view returns (address) {
+    return block.coinbase;
+  }
+
+  function callDifficulty() external view returns (uint256) {
+    return block.difficulty;
+  }
+
+  function callGasLimit() external view returns (uint256) {
+    return block.gaslimit;
+  }
+
+  function callBlockNumber() external view returns (uint256) {
+    return block.number;
+  }
+
+  function callTimestamp() external view returns (uint256) {
+    return block.timestamp;
+  }
+
+  function callGasLeft() external view returns (uint256) {
+    return gasleft();
+  }
+
+  function callGasPrice() external view returns (uint256) {
+    return tx.gasprice;
+  }
+
+  function callOrigin() external view returns (address) {
+    return tx.origin;
+  }
+
+  function callBalanceOf(address _addr) external view returns (uint256) {
+    return _addr.balance;
+  }
+
+  function callCodeSize(address _addr) external view returns (uint256 size) {
+    assembly { size := extcodesize(_addr) }
+  }
+
+  function callCode(address _addr) external view returns (bytes memory code) {
+    assembly {
+      let size := extcodesize(_addr)
+      code := mload(0x40)
+      mstore(0x40, add(code, and(add(add(size, 0x20), 0x1f), not(0x1f))))
+      mstore(code, size)
+      extcodecopy(_addr, add(code, 0x20), 0, size)
+    }
+  }
+
+  function callCodeHash(address _addr) external view returns (bytes32 codeHash) {
+    assembly { codeHash := extcodehash(_addr) }
+  }
+
+  function callChainId() external view returns (uint256 id) {
+    assembly { id := chainid() }
+  }
+}
diff --git a/forks/passport/src/contracts/modules/utils/RequireUtils.sol b/forks/passport/src/contracts/modules/utils/RequireUtils.sol
new file mode 100644
index 00000000..fa285d19
--- /dev/null
+++ b/forks/passport/src/contracts/modules/utils/RequireUtils.sol
@@ -0,0 +1,281 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../commons/interfaces/IModuleCalls.sol";
+import "../commons/interfaces/IModuleAuthUpgradable.sol";
+import "../../interfaces/IERC1271Wallet.sol";
+import "../../utils/SignatureValidator.sol";
+import "../../utils/LibBytes.sol";
+import "../../Wallet.sol";
+
+contract RequireUtils is SignatureValidator {
+  using LibBytes for bytes;
+
+  uint256 private constant NONCE_BITS = 96;
+  bytes32 private constant NONCE_MASK = bytes32((1 << NONCE_BITS) - 1);
+
+  uint256 private constant FLAG_SIGNATURE = 0;
+  uint256 private constant FLAG_ADDRESS = 1;
+  uint256 private constant FLAG_DYNAMIC_SIGNATURE = 2;
+
+  bytes32 private immutable INIT_CODE_HASH;
+  address private immutable FACTORY;
+
+  struct Member {
+    uint256 weight;
+    address signer;
+  }
+
+  event RequiredConfig(
+    address indexed _wallet,
+    bytes32 indexed _imageHash,
+    uint256 _threshold,
+    bytes _signers
+  );
+
+  event RequiredSigner(
+    address indexed _wallet,
+    address indexed _signer
+  );
+
+  mapping(address => uint256) public lastSignerUpdate;
+  mapping(address => uint256) public lastWalletUpdate;
+  mapping(address => bytes32) public knownImageHashes;
+  mapping(bytes32 => uint256) public lastImageHashUpdate;
+
+  constructor(address _factory, address _mainModule) public {
+    FACTORY = _factory;
+    INIT_CODE_HASH = keccak256(abi.encodePacked(Wallet.creationCode, uint256(uint160(_mainModule))));
+  }
+
+  /**
+   * @notice Publishes the current configuration of a Sequence wallets using logs
+   * @dev Used for fast lookup of a wallet configuration based on its image-hash, compatible with updated and counter-factual wallets.
+   *
+   * @param _wallet      Sequence wallet
+   * @param _threshold   Thershold of the current configuration
+   * @param _members     Members of the current configuration
+   * @param _index       True if an index in contract-storage is desired 
+   */
+  function publishConfig(
+    address _wallet,
+    uint256 _threshold,
+    Member[] calldata _members,
+    bool _index
+  ) external {
+    // Compute expected imageHash
+    bytes32 imageHash = bytes32(uint256(_threshold));
+    for (uint256 i = 0; i < _members.length; i++) {
+      imageHash = keccak256(abi.encode(imageHash, _members[i].weight, _members[i].signer));
+    }
+
+    // Check against wallet imageHash
+    (bool succeed, bytes memory data) = _wallet.call(abi.encodePacked(IModuleAuthUpgradable(_wallet).imageHash.selector));
+    if (succeed && data.length == 32) {
+      // Check contract defined
+      bytes32 currentImageHash = abi.decode(data, (bytes32));
+      require(currentImageHash == imageHash, "RequireUtils#publishConfig: UNEXPECTED_IMAGE_HASH");
+    } else {
+      // Check counter-factual
+      require(address(
+        uint160(uint256(
+          keccak256(
+            abi.encodePacked(
+              bytes1(0xff),
+              FACTORY,
+              imageHash,
+              INIT_CODE_HASH
+            )
+          )
+        ))
+      ) == _wallet, "RequireUtils#publishConfig: UNEXPECTED_COUNTERFACTUAL_IMAGE_HASH");
+
+      // Register known image-hash for counter-factual wallet
+      if (_index) knownImageHashes[_wallet] = imageHash;
+    }
+
+    // Emit event for easy config retrieval
+    emit RequiredConfig(_wallet, imageHash, _threshold, abi.encode(_members));
+
+    if (_index) {
+      // Register last event for given wallet
+      lastWalletUpdate[_wallet] = block.number;
+
+      // Register last event for image-hash
+      lastImageHashUpdate[imageHash] = block.number;
+    }
+  }
+
+  /**
+   * @notice Publishes the configuration and set of signers for a counter-factual Sequence wallets using logs
+   * @dev Used for fast lookup of a wallet based on its signer members, only signing members are included in the logs
+   *   as a mechanism to avoid poisoning of the directory of wallets.
+   *
+   *   Only the initial counter-factual configuration can be published, to publish updated configurations see `publishConfig`.
+   *
+   * @param _wallet      Sequence wallet
+   * @param _hash        Any hash signed by the wallet
+   * @param _sizeMembers Number of members on the counter-factual configuration
+   * @param _signature   Signature for the given hash
+   * @param _index       True if an index in contract-storage is desired 
+   */
+  function publishInitialSigners(
+    address _wallet,
+    bytes32 _hash,
+    uint256 _sizeMembers,
+    bytes memory _signature,
+    bool _index
+  ) external {
+    // Decode and index signature
+    (
+      uint16 threshold,  // required threshold signature
+      uint256 rindex     // read index
+    ) = _signature.readFirstUint16();
+
+    // Generate sub-digest
+    bytes32 subDigest; {
+      uint256 chainId; assembly { chainId := chainid() }
+      subDigest = keccak256(
+        abi.encodePacked(
+          "\x19\x01",
+          chainId,
+          _wallet,
+          _hash
+        )
+      );
+    }
+
+    // Recover signature
+    bytes32 imageHash = bytes32(uint256(threshold));
+
+    Member[] memory members = new Member[](_sizeMembers);
+    uint256 membersIndex = 0;
+
+    while (rindex < _signature.length) {
+      // Read next item type and addrWeight
+      uint256 flag; uint256 addrWeight; address addr;
+      (flag, addrWeight, rindex) = _signature.readUint8Uint8(rindex);
+
+      if (flag == FLAG_ADDRESS) {
+        // Read plain address
+        (addr, rindex) = _signature.readAddress(rindex);
+      } else if (flag == FLAG_SIGNATURE) {
+        // Read single signature and recover signer
+        bytes memory signature;
+        (signature, rindex) = _signature.readBytes66(rindex);
+        addr = recoverSigner(subDigest, signature);
+
+        // Publish signer
+        _publishSigner(_wallet, addr, _index);
+      } else if (flag == FLAG_DYNAMIC_SIGNATURE) {
+        // Read signer
+        (addr, rindex) = _signature.readAddress(rindex);
+
+        {
+          // Read signature size
+          uint256 size;
+          (size, rindex) = _signature.readUint16(rindex);
+
+          // Read dynamic size signature
+          bytes memory signature;
+          (signature, rindex) = _signature.readBytes(rindex, size);
+          require(isValidSignature(subDigest, addr, signature), "ModuleAuth#_signatureValidation: INVALID_SIGNATURE");
+        }
+
+        // Publish signer
+        _publishSigner(_wallet, addr, _index);
+      } else {
+        revert("RequireUtils#publishInitialSigners: INVALID_SIGNATURE_FLAG");
+      }
+
+      // Store member on array
+      members[membersIndex] = Member(addrWeight, addr);
+      membersIndex++;
+
+      // Write weight and address to image
+      imageHash = keccak256(abi.encode(imageHash, addrWeight, addr));
+    }
+
+    require(membersIndex == _sizeMembers, "RequireUtils#publishInitialSigners: INVALID_MEMBERS_COUNT");
+
+    // Check against counter-factual imageHash
+    require(address(
+      uint160(uint256(
+        keccak256(
+          abi.encodePacked(
+            bytes1(0xff),
+            FACTORY,
+            imageHash,
+            INIT_CODE_HASH
+          )
+        )
+      ))
+    ) == _wallet, "RequireUtils#publishInitialSigners: UNEXPECTED_COUNTERFACTUAL_IMAGE_HASH");
+
+    // Emit event for easy config retrieval
+    emit RequiredConfig(_wallet, imageHash, threshold, abi.encode(members));
+
+    if (_index) {
+      // Register last event for given wallet
+      lastWalletUpdate[_wallet] = block.number;
+
+      // Register last event for image-hash
+      lastImageHashUpdate[imageHash] = block.number;
+
+      // Register known image-hash for counter-factual wallet
+      knownImageHashes[_wallet] = imageHash;
+    }
+  }
+
+  /**
+   * @notice Validates that a given expiration hasn't expired
+   * @dev Used as an optional transaction on a Sequence batch, to create expirable transactions.
+   *
+   * @param _expiration  Expiration to check
+   */
+  function requireNonExpired(uint256 _expiration) external view {
+    require(block.timestamp <= _expiration, "RequireUtils#requireNonExpired: EXPIRED");
+  }
+
+  /**
+   * @notice Validates that a given wallet has reached a given nonce
+   * @dev Used as an optional transaction on a Sequence batch, to define transaction execution order
+   *
+   * @param _wallet Sequence wallet
+   * @param _nonce  Required nonce
+   */
+  function requireMinNonce(address _wallet, uint256 _nonce) external view {
+    (uint256 space, uint256 nonce) = _decodeNonce(_nonce);
+    uint256 currentNonce = IModuleCalls(_wallet).readNonce(space);
+    require(currentNonce >= nonce, "RequireUtils#requireMinNonce: NONCE_BELOW_REQUIRED");
+  }
+
+  /**
+   * @notice Decodes a raw nonce
+   * @dev A raw nonce is encoded using the first 160 bits for the space
+   *  and the last 96 bits for the nonce
+   * @param _rawNonce Nonce to be decoded
+   * @return _space The nonce space of the raw nonce
+   * @return _nonce The nonce of the raw nonce
+   */
+  function _decodeNonce(uint256 _rawNonce) private pure returns (uint256 _space, uint256 _nonce) {
+    _nonce = uint256(bytes32(_rawNonce) & NONCE_MASK);
+    _space = _rawNonce >> NONCE_BITS;
+  }
+
+  /**
+   * @notice Publishes a signer that was validated to sign for a particular wallet
+   * @param _wallet Address of the wallet
+   * @param _signer Address of the signer
+   * @param _index True if an index on contract storage is desired
+   */
+  function _publishSigner(address _wallet, address _signer, bool _index) private {
+    // Required signer event
+    emit RequiredSigner(_wallet, _signer);
+
+    if (_index) {
+      // Register last event for given signer
+      lastSignerUpdate[_signer] = block.number;
+    }
+  }
+}
diff --git a/forks/passport/src/contracts/modules/utils/SequenceUtils.sol b/forks/passport/src/contracts/modules/utils/SequenceUtils.sol
new file mode 100644
index 00000000..5e2ec872
--- /dev/null
+++ b/forks/passport/src/contracts/modules/utils/SequenceUtils.sol
@@ -0,0 +1,19 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./MultiCallUtils.sol";
+import "./RequireUtils.sol";
+
+
+contract SequenceUtils is 
+  MultiCallUtils,
+  RequireUtils
+{
+  constructor(
+    address _factory,
+    address _mainModule
+  ) RequireUtils(
+    _factory,
+    _mainModule
+  ) {}
+}
diff --git a/forks/passport/src/contracts/modules/utils/libs/RequireFreshSigner.sol b/forks/passport/src/contracts/modules/utils/libs/RequireFreshSigner.sol
new file mode 100644
index 00000000..e0df82a6
--- /dev/null
+++ b/forks/passport/src/contracts/modules/utils/libs/RequireFreshSigner.sol
@@ -0,0 +1,17 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../RequireUtils.sol";
+
+
+contract RequireFreshSigner {
+  RequireUtils public immutable REQUIRE_UTILS;
+
+  constructor (RequireUtils _requireUtils) {
+    REQUIRE_UTILS = _requireUtils;
+  }
+
+  function requireFreshSigner(address _signer) external {
+    require(REQUIRE_UTILS.lastSignerUpdate(_signer) == 0, "RequireFreshSigner#requireFreshSigner: DUPLICATED_SIGNER");
+  }
+}
diff --git a/forks/passport/src/contracts/signer/ImmutableSigner.sol b/forks/passport/src/contracts/signer/ImmutableSigner.sol
new file mode 100644
index 00000000..b0e43700
--- /dev/null
+++ b/forks/passport/src/contracts/signer/ImmutableSigner.sol
@@ -0,0 +1,114 @@
+// Copyright Immutable Pty Ltd 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import {SignatureValidator} from '../utils/SignatureValidator.sol';
+import {AccessControl} from '@openzeppelin/contracts/access/AccessControl.sol';
+
+/**
+ * @title ImmutableSigner
+ *
+ * @notice ERC-1271 Wallet implementation. Verifies signatures using a public
+ * key in storage matching an offchain private key. The key can be upgraded by
+ * the role SIGNER_ADMIN_ROLE, set during initialization.
+ */
+contract ImmutableSigner is SignatureValidator, AccessControl {
+  // @dev A signer only valid until the validUntil unix timestmap
+  struct ExpirableSigner {
+    address signer;
+    uint256 validUntil;
+  }
+
+  address public primarySigner;
+  ExpirableSigner public rolloverSigner;
+
+  bytes32 public constant SIGNER_ADMIN_ROLE = keccak256('SIGNER_ADMIN_ROLE');
+
+  /*
+   * @notice Emitted whenever the main authorized signer is updated.
+   */
+  event PrimarySignerUpdated(address indexed _previousSigner, address indexed _newSigner);
+  /*
+   * @notice Emitted whenever a temporary signer is enabled.
+   */
+  event RolloverSignerEnabled(address indexed _signer, uint256 _validUntil);
+
+
+  constructor(address _rootAdmin, address _signerAdmin, address _signer) {
+    _grantRole(DEFAULT_ADMIN_ROLE, _rootAdmin);
+    _grantRole(SIGNER_ADMIN_ROLE, _signerAdmin);
+
+    primarySigner = _signer;
+
+    // We do not need a rollover signer initially as there's no rollover
+    // happening, so validUntil must be in the past. Luckily that aligns with
+    // its default initial value.
+    //
+    // rolloverSigner.validUntil = 0;
+  }
+
+  /*
+   * @dev Grants SIGNER_ADMIN_ROLE to an user.
+   * @param _signerAdmin Address that will be allowed to update the wallet signer.
+   */
+  function grantSignerRole(address _signerAdmin) external onlyRole(DEFAULT_ADMIN_ROLE) {
+    _grantRole(SIGNER_ADMIN_ROLE, _signerAdmin);
+  }
+
+  /*
+   * @dev Updates the authorized public key address. Any old keys are expired effective immediately.
+   * @param _newSigner The address of the new authorized signer.
+   */
+  function updateSigner(address _newSigner) public onlyRole(SIGNER_ADMIN_ROLE) {
+    address previousSigner = primarySigner;
+    primarySigner = _newSigner;
+
+    // Make sure any rollover is immediately finished.
+    rolloverSigner.validUntil = 0;
+
+    emit PrimarySignerUpdated(previousSigner, _newSigner);
+  }
+
+  /*
+   * @dev Updates the authorized public key address. Allows the previous public
+   * key to remain valid for a specified rollover period.
+   *
+   * @param _newSigner The address of the new authorized signer.
+   * @param _rolloverPeriod Period for which the previous key will still be valid.
+   */
+  function updateSignerWithRolloverPeriod(
+    address _newSigner,
+    uint256 _rolloverPeriod
+  ) public onlyRole(SIGNER_ADMIN_ROLE) {
+    address previousSigner = primarySigner;
+    primarySigner = _newSigner;
+
+    rolloverSigner.signer = previousSigner;
+    rolloverSigner.validUntil = block.timestamp + _rolloverPeriod;
+
+    emit PrimarySignerUpdated(previousSigner, _newSigner);
+    emit RolloverSignerEnabled(rolloverSigner.signer, rolloverSigner.validUntil);
+  }
+
+  /**
+   * @notice Verifies whether the provided signature is valid with respect to the provided hash
+   * @dev MUST return the correct magic value if the signature provided is valid for the provided hash
+   *   > The bytes4 magic value to return when signature is valid is 0x1626ba7e
+   * @param _hash       keccak256 hash that was signed
+   * @param _signature  Signature byte array associated with _data
+   * @return magicValue Magic value 0x1626ba7e if the signature is valid and 0x0 otherwise
+   */
+  function isValidSignature(bytes32 _hash, bytes memory _signature) external view returns (bytes4) {
+    if (recoverSigner(_hash, _signature) == primarySigner) {
+      return ERC1271_MAGICVALUE_BYTES32;
+    }
+
+    if (block.timestamp <= rolloverSigner.validUntil) {
+      if (recoverSigner(_hash, _signature) == rolloverSigner.signer) {
+        return ERC1271_MAGICVALUE_BYTES32;
+      }
+    }
+
+    return 0;
+  }
+}
diff --git a/forks/passport/src/contracts/startup/ILatestWalletImplLocator.sol b/forks/passport/src/contracts/startup/ILatestWalletImplLocator.sol
new file mode 100644
index 00000000..6cb4985b
--- /dev/null
+++ b/forks/passport/src/contracts/startup/ILatestWalletImplLocator.sol
@@ -0,0 +1,15 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+/**
+ * @title ILatestWalletImplLocator
+ * @notice Interface for using the latest wallet implementation locator contract.
+ */
+interface ILatestWalletImplLocator {
+    /**
+     * Return the address of the latest wallet implementation contract.
+     */
+    function latestWalletImplementation() external returns (address);
+}
\ No newline at end of file
diff --git a/forks/passport/src/contracts/startup/LatestWalletImplLocator.sol b/forks/passport/src/contracts/startup/LatestWalletImplLocator.sol
new file mode 100644
index 00000000..54644cca
--- /dev/null
+++ b/forks/passport/src/contracts/startup/LatestWalletImplLocator.sol
@@ -0,0 +1,38 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import '@openzeppelin/contracts/access/AccessControl.sol';
+import "./ILatestWalletImplLocator.sol";
+
+/**
+ * @title LatestWalletImplLocator
+ * @notice Contract to return the address of the latest wallet implementation contract.
+ */
+contract LatestWalletImplLocator is ILatestWalletImplLocator, AccessControl {
+    // Role to change the implementation contract address.
+    bytes32 public constant IMPLCHANGER_ROLE = keccak256('IMPLCHANGER_ROLE');
+
+    address public latestWalletImplementation;
+
+    event ImplChanged(address indexed _whoBy, address indexed _newImpl);
+
+    /**
+     * @param _admin Role that can grant / revoke roles: DEFAULT_ADMIN and IMPLCHANGER.
+     * @param _implChanger Initial address that can change the latest wallet implementation address.
+     */
+    constructor(address _admin, address _implChanger) {
+        _grantRole(DEFAULT_ADMIN_ROLE, _admin);
+        _grantRole(IMPLCHANGER_ROLE, _implChanger);
+    }
+
+
+    /**
+     * Change the address of the latest wallet implementation contract.
+     * @param _newImpl Address of the main module to be used by the wallet.
+     */
+    function changeWalletImplementation(address _newImpl) external onlyRole(IMPLCHANGER_ROLE) {
+        latestWalletImplementation = _newImpl;
+        emit ImplChanged(msg.sender, _newImpl);
+    }
+}
\ No newline at end of file
diff --git a/forks/passport/src/contracts/startup/StartupWalletImpl.sol b/forks/passport/src/contracts/startup/StartupWalletImpl.sol
new file mode 100644
index 00000000..33b8b37e
--- /dev/null
+++ b/forks/passport/src/contracts/startup/StartupWalletImpl.sol
@@ -0,0 +1,46 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "./ILatestWalletImplLocator.sol";
+
+/**
+ * @title StartupWalletImpl
+ * @notice Initial wallet implementation contract used to setup the proxy with the latest
+ *        implementation contract and ensure that the minimal proxy deployment data never
+ *        needs to change.
+ *        Note that this code executed in the context of the WalletProxy.yul. As such,
+ *        the address of the Latest Wallet Implementation Locator contract needs to be
+ *        inserted directly into the code of the contract. This is achieved by using an 
+ *        immutable variable.
+ */
+contract StartupWalletImpl {
+    address public immutable walletImplementationLocator;
+
+    constructor (address _walletImplementationLocator) {
+        walletImplementationLocator = _walletImplementationLocator;
+    }
+
+
+    fallback() external payable {
+        // Get the address of the latest version of the wallet implementation.
+        ILatestWalletImplLocator locator = ILatestWalletImplLocator(walletImplementationLocator);
+        address latestImplAddr = locator.latestWalletImplementation();
+
+        // solhint-disable-next-line no-inline-assembly
+        assembly{
+            // Store the address of the implementation to use in the storage slot defined by the
+            // address of this contract.
+            sstore(address(), latestImplAddr)
+
+            // Now do a standard delegate call to the wallet implementation and return the
+            // results / errors.
+            calldatacopy(0, 0, calldatasize())
+            let result := delegatecall(gas(), latestImplAddr, 0, calldatasize(), 0, 0)
+            returndatacopy(0, 0, returndatasize())
+            switch result
+            case 0 {revert(0, returndatasize())}
+            default {return (0, returndatasize())}
+        }
+    }
+}
diff --git a/forks/passport/src/contracts/utils/LibAddress.sol b/forks/passport/src/contracts/utils/LibAddress.sol
new file mode 100644
index 00000000..7508365f
--- /dev/null
+++ b/forks/passport/src/contracts/utils/LibAddress.sol
@@ -0,0 +1,18 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+
+library LibAddress {
+  /**
+   * @notice Will return true if provided address is a contract
+   * @param account Address to verify if contract or not
+   * @dev This contract will return false if called within the constructor of
+   *      a contract's deployment, as the code is not yet stored on-chain.
+   */
+  function isContract(address account) internal view returns (bool) {
+    uint256 csize;
+    // solhint-disable-next-line no-inline-assembly
+    assembly { csize := extcodesize(account) }
+    return csize != 0;
+  }
+}
diff --git a/forks/passport/src/contracts/utils/LibBytes.sol b/forks/passport/src/contracts/utils/LibBytes.sol
new file mode 100644
index 00000000..b8ffc64c
--- /dev/null
+++ b/forks/passport/src/contracts/utils/LibBytes.sol
@@ -0,0 +1,195 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+library LibBytes {
+  using LibBytes for bytes;
+
+  /***********************************|
+  |        Read Bytes Functions       |
+  |__________________________________*/
+
+  /**
+   * @dev Read firsts uint16 value.
+   * @param data Byte array to be read.
+   * @return a uint16 value of data at index zero.
+   * @return newIndex Updated index after reading the values.
+   */
+  function readFirstUint16(
+    bytes memory data
+  ) internal pure returns (
+    uint16 a,
+    uint256 newIndex
+  ) {
+    assembly {
+      let word := mload(add(32, data))
+      a := shr(240, word)
+      newIndex := 2
+    }
+    require(2 <= data.length, "LibBytes#readFirstUint16: OUT_OF_BOUNDS");
+  }
+
+  /**
+   * @dev Reads consecutive bool (8 bits) and uint8 values.
+   * @param data Byte array to be read.
+   * @param index Index in byte array of uint8 and uint8 values.
+   * @return a uint8 value of data at given index.
+   * @return b uint8 value of data at given index + 8.
+   * @return newIndex Updated index after reading the values.
+   */
+  function readUint8Uint8(
+    bytes memory data,
+    uint256 index
+  ) internal pure returns (
+    uint8 a,
+    uint8 b,
+    uint256 newIndex
+  ) {
+    assembly {
+      let word := mload(add(index, add(32, data)))
+      a := shr(248, word)
+      b := and(shr(240, word), 0xff)
+      newIndex := add(index, 2)
+    }
+    assert(newIndex > index);
+    require(newIndex <= data.length, "LibBytes#readUint8Uint8: OUT_OF_BOUNDS");
+  }
+
+  /**
+   * @dev Reads an address value from a position in a byte array.
+   * @param data Byte array to be read.
+   * @param index Index in byte array of address value.
+   * @return a address value of data at given index.
+   * @return newIndex Updated index after reading the value.
+   */
+  function readAddress(
+    bytes memory data,
+    uint256 index
+  ) internal pure returns (
+    address a,
+    uint256 newIndex
+  ) {
+    assembly {
+      let word := mload(add(index, add(32, data)))
+      a := and(shr(96, word), 0xffffffffffffffffffffffffffffffffffffffff)
+      newIndex := add(index, 20)
+    }
+    assert(newIndex > index);
+    require(newIndex <= data.length, "LibBytes#readAddress: OUT_OF_BOUNDS");
+  }
+
+  /**
+   * @dev Reads 66 bytes from a position in a byte array.
+   * @param data Byte array to be read.
+   * @param index Index in byte array of 66 bytes value.
+   * @return a 66 bytes bytes array value of data at given index.
+   * @return newIndex Updated index after reading the value.
+   */
+  function readBytes66(
+    bytes memory data,
+    uint256 index
+  ) internal pure returns (
+    bytes memory a,
+    uint256 newIndex
+  ) {
+    a = new bytes(66);
+    assembly {
+      let offset := add(32, add(data, index))
+      mstore(add(a, 32), mload(offset))
+      mstore(add(a, 64), mload(add(offset, 32)))
+      mstore(add(a, 66), mload(add(offset, 34)))
+      newIndex := add(index, 66)
+    }
+    assert(newIndex > index);
+    require(newIndex <= data.length, "LibBytes#readBytes66: OUT_OF_BOUNDS");
+  }
+
+  /**
+   * @dev Reads a bytes32 value from a position in a byte array.
+   * @param b Byte array containing a bytes32 value.
+   * @param index Index in byte array of bytes32 value.
+   * @return result bytes32 value from byte array.
+   */
+  function readBytes32(
+    bytes memory b,
+    uint256 index
+  )
+    internal
+    pure
+    returns (bytes32 result)
+  {
+    require(
+      b.length >= index + 32,
+      "LibBytes#readBytes32: GREATER_OR_EQUAL_TO_32_LENGTH_REQUIRED"
+    );
+
+    // Arrays are prefixed by a 256 bit length parameter
+    uint256 pos = index + 32;
+
+    // Read the bytes32 from array memory
+    assembly {
+      result := mload(add(b, pos))
+    }
+    return result;
+  }
+
+  /**
+   * @dev Reads an uint16 value from a position in a byte array.
+   * @param data Byte array to be read.
+   * @param index Index in byte array of uint16 value.
+   * @return a uint16 value of data at given index.
+   * @return newIndex Updated index after reading the value.
+   */
+  function readUint16(
+    bytes memory data,
+    uint256 index
+  ) internal pure returns (uint16 a, uint256 newIndex) {
+    assembly {
+      let word := mload(add(index, add(32, data)))
+      a := and(shr(240, word), 0xffff)
+      newIndex := add(index, 2)
+    }
+    assert(newIndex > index);
+    require(newIndex <= data.length, "LibBytes#readUint16: OUT_OF_BOUNDS");
+  }
+
+  /**
+   * @dev Reads bytes from a position in a byte array.
+   * @param data Byte array to be read.
+   * @param index Index in byte array of bytes value.
+   * @param size Number of bytes to read.
+   * @return a bytes bytes array value of data at given index.
+   * @return newIndex Updated index after reading the value.
+   */
+  function readBytes(
+    bytes memory data,
+    uint256 index,
+    uint256 size
+  ) internal pure returns (bytes memory a, uint256 newIndex) {
+    a = new bytes(size);
+
+    assembly {
+      let offset := add(32, add(data, index))
+
+      let i := 0 let n := 32
+      // Copy each word, except last one
+      for { } lt(n, size) { i := n n := add(n, 32) } {
+        mstore(add(a, n), mload(add(offset, i)))
+      }
+
+      // Load word after new array
+      let suffix := add(a, add(32, size))
+      let suffixWord := mload(suffix)
+
+      // Copy last word, overwrites after array 
+      mstore(add(a, n), mload(add(offset, i)))
+
+      // Restore after array
+      mstore(suffix, suffixWord)
+
+      newIndex := add(index, size)
+    }
+
+    assert(newIndex >= index);
+    require(newIndex <= data.length, "LibBytes#readBytes: OUT_OF_BOUNDS");
+  }
+}
diff --git a/forks/passport/src/contracts/utils/SignatureValidator.sol b/forks/passport/src/contracts/utils/SignatureValidator.sol
new file mode 100644
index 00000000..255b820a
--- /dev/null
+++ b/forks/passport/src/contracts/utils/SignatureValidator.sol
@@ -0,0 +1,143 @@
+// SPDX-License-Identifier: Apache-2.0
+pragma solidity 0.8.17;
+
+import "../interfaces/IERC1271Wallet.sol";
+
+import "./LibBytes.sol";
+
+/**
+ * @dev Contains logic for signature validation.
+ * Signatures from wallet contracts assume ERC-1271 support (https://github.com/ethereum/EIPs/blob/master/EIPS/eip-1271.md)
+ * Notes: Methods are strongly inspired by contracts in https://github.com/0xProject/0x-monorepo/blob/development/
+ */
+contract SignatureValidator {
+  using LibBytes for bytes;
+
+  /***********************************|
+  |             Variables             |
+  |__________________________________*/
+
+  // bytes4(keccak256("isValidSignature(bytes,bytes)"))
+  bytes4 constant internal ERC1271_MAGICVALUE = 0x20c13b0b;
+
+  // bytes4(keccak256("isValidSignature(bytes32,bytes)"))
+  bytes4 constant internal ERC1271_MAGICVALUE_BYTES32 = 0x1626ba7e;
+
+  // Allowed signature types.
+  uint256 private constant SIG_TYPE_EIP712 = 1;
+  uint256 private constant SIG_TYPE_ETH_SIGN = 2;
+  uint256 private constant SIG_TYPE_WALLET_BYTES32 = 3;
+
+  /***********************************|
+  |        Signature Functions        |
+  |__________________________________*/
+
+ /**
+   * @notice Recover the signer of hash, assuming it's an EOA account
+   * @dev Only for SignatureType.EIP712 and SignatureType.EthSign signatures
+   * @param _hash      Hash that was signed
+   *   encoded as (bytes32 r, bytes32 s, uint8 v, ... , SignatureType sigType)
+   */
+  function recoverSigner(
+    bytes32 _hash,
+    bytes memory _signature
+  ) internal pure returns (address signer) {
+    require(_signature.length == 66, "SignatureValidator#recoverSigner: invalid signature length");
+    uint256 signatureType = uint8(_signature[_signature.length - 1]);
+
+    // Variables are not scoped in Solidity.
+    uint8 v = uint8(_signature[64]);
+    bytes32 r = _signature.readBytes32(0);
+    bytes32 s = _signature.readBytes32(32);
+
+    // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature
+    // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines
+    // the valid range for s in (281): 0 < s < secp256k1n ÷ 2 + 1, and for v in (282): v ∈ {27, 28}. Most
+    // signatures from current libraries generate a unique signature with an s-value in the lower half order.
+    //
+    // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value
+    // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or
+    // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept
+    // these malleable signatures as well.
+    //
+    // Source OpenZeppelin
+    // https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/cryptography/ECDSA.sol
+
+    if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) {
+      revert("SignatureValidator#recoverSigner: invalid signature 's' value");
+    }
+
+    if (v != 27 && v != 28) {
+      revert("SignatureValidator#recoverSigner: invalid signature 'v' value");
+    }
+
+    // Signature using EIP712
+    if (signatureType == SIG_TYPE_EIP712) {
+      signer = ecrecover(_hash, v, r, s);
+
+    // Signed using web3.eth_sign() or Ethers wallet.signMessage()
+    } else if (signatureType == SIG_TYPE_ETH_SIGN) {
+      signer = ecrecover(
+        keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", _hash)),
+        v,
+        r,
+        s
+      );
+
+    } else {
+      // Anything other signature types are illegal (We do not return false because
+      // the signature may actually be valid, just not in a format
+      // that we currently support. In this case returning false
+      // may lead the caller to incorrectly believe that the
+      // signature was invalid.)
+      revert("SignatureValidator#recoverSigner: UNSUPPORTED_SIGNATURE_TYPE");
+    }
+
+    // Prevent signer from being 0x0
+    require(
+      signer != address(0x0),
+      "SignatureValidator#recoverSigner: INVALID_SIGNER"
+    );
+
+    return signer;
+  }
+
+ /**
+   * @notice Returns true if the provided signature is valid for the given signer.
+   * @dev Supports SignatureType.EIP712, SignatureType.EthSign, and ERC1271 signatures
+   * @param _hash      Hash that was signed
+   * @param _signer    Address of the signer candidate
+   * @param _signature Signature byte array
+   */
+  function isValidSignature(
+    bytes32 _hash,
+    address _signer,
+    bytes memory _signature
+  ) internal view returns (bool valid) {
+    // Check for valid signature length
+    if (_signature.length == 0) {
+      revert("SignatureValidator#isValidSignature: signature is empty");
+    }
+    
+    uint256 signatureType = uint8(_signature[_signature.length - 1]);
+
+    if (signatureType == SIG_TYPE_EIP712 || signatureType == SIG_TYPE_ETH_SIGN) {
+      // Recover signer and compare with provided
+      valid = recoverSigner(_hash, _signature) == _signer;
+
+    } else if (signatureType == SIG_TYPE_WALLET_BYTES32) {
+      // Remove signature type before calling ERC1271, restore after call
+      uint256 prevSize; assembly { prevSize := mload(_signature) mstore(_signature, sub(prevSize, 1)) }
+      valid = ERC1271_MAGICVALUE_BYTES32 == IERC1271Wallet(_signer).isValidSignature(_hash, _signature);
+      assembly { mstore(_signature, prevSize) }
+
+    } else {
+      // Anything other signature types are illegal (We do not return false because
+      // the signature may actually be valid, just not in a format
+      // that we currently support. In this case returning false
+      // may lead the caller to incorrectly believe that the
+      // signature was invalid.)
+      revert("SignatureValidator#isValidSignature: UNSUPPORTED_SIGNATURE_TYPE");
+    }
+  }
+}
diff --git a/forks/passport/src/networks/arbitrum-testnet.json b/forks/passport/src/networks/arbitrum-testnet.json
new file mode 100644
index 00000000..e9114b37
--- /dev/null
+++ b/forks/passport/src/networks/arbitrum-testnet.json
@@ -0,0 +1,22 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/arbitrum.json b/forks/passport/src/networks/arbitrum.json
new file mode 100644
index 00000000..e9114b37
--- /dev/null
+++ b/forks/passport/src/networks/arbitrum.json
@@ -0,0 +1,22 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/avalanche.json b/forks/passport/src/networks/avalanche.json
new file mode 100644
index 00000000..a9b24933
--- /dev/null
+++ b/forks/passport/src/networks/avalanche.json
@@ -0,0 +1,26 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  },
+  {
+    "contractName": "RequireFreshSignerLib",
+    "address": "0xE6B9B21C077F382333220a072e4c44280b873907"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/avalancheTestnet.json b/forks/passport/src/networks/avalancheTestnet.json
new file mode 100644
index 00000000..a9b24933
--- /dev/null
+++ b/forks/passport/src/networks/avalancheTestnet.json
@@ -0,0 +1,26 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  },
+  {
+    "contractName": "RequireFreshSignerLib",
+    "address": "0xE6B9B21C077F382333220a072e4c44280b873907"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/ganache.json b/forks/passport/src/networks/ganache.json
new file mode 100644
index 00000000..f5dc436d
--- /dev/null
+++ b/forks/passport/src/networks/ganache.json
@@ -0,0 +1,22 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xFec500b5070bCD8Ddb3cC16C00c4DeB7d1e716ad"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xB1597358509F1857B4566Fd9BefeA64D920FFFE2"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x2e8c375f11f065fdc9b8B919Fac7B7F359bE10ea"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x7EC4B9C7d437a815dbB23D068bD10b2c1FA469E4"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0x3fc78b29faDafd9D3be6e76cDA39c0c9a706863E"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/goerli.json b/forks/passport/src/networks/goerli.json
new file mode 100644
index 00000000..a9b24933
--- /dev/null
+++ b/forks/passport/src/networks/goerli.json
@@ -0,0 +1,26 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  },
+  {
+    "contractName": "RequireFreshSignerLib",
+    "address": "0xE6B9B21C077F382333220a072e4c44280b873907"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/kovan.json b/forks/passport/src/networks/kovan.json
new file mode 100644
index 00000000..a9b24933
--- /dev/null
+++ b/forks/passport/src/networks/kovan.json
@@ -0,0 +1,26 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  },
+  {
+    "contractName": "RequireFreshSignerLib",
+    "address": "0xE6B9B21C077F382333220a072e4c44280b873907"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/mainnet.json b/forks/passport/src/networks/mainnet.json
new file mode 100644
index 00000000..e9114b37
--- /dev/null
+++ b/forks/passport/src/networks/mainnet.json
@@ -0,0 +1,22 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/matic.json b/forks/passport/src/networks/matic.json
new file mode 100644
index 00000000..a9b24933
--- /dev/null
+++ b/forks/passport/src/networks/matic.json
@@ -0,0 +1,26 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  },
+  {
+    "contractName": "RequireFreshSignerLib",
+    "address": "0xE6B9B21C077F382333220a072e4c44280b873907"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/mumbai.json b/forks/passport/src/networks/mumbai.json
new file mode 100644
index 00000000..a9b24933
--- /dev/null
+++ b/forks/passport/src/networks/mumbai.json
@@ -0,0 +1,26 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  },
+  {
+    "contractName": "RequireFreshSignerLib",
+    "address": "0xE6B9B21C077F382333220a072e4c44280b873907"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/nova.json b/forks/passport/src/networks/nova.json
new file mode 100644
index 00000000..a9b24933
--- /dev/null
+++ b/forks/passport/src/networks/nova.json
@@ -0,0 +1,26 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  },
+  {
+    "contractName": "RequireFreshSignerLib",
+    "address": "0xE6B9B21C077F382333220a072e4c44280b873907"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/optimism.json b/forks/passport/src/networks/optimism.json
new file mode 100644
index 00000000..a9b24933
--- /dev/null
+++ b/forks/passport/src/networks/optimism.json
@@ -0,0 +1,26 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  },
+  {
+    "contractName": "RequireFreshSignerLib",
+    "address": "0xE6B9B21C077F382333220a072e4c44280b873907"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/rinkeby.json b/forks/passport/src/networks/rinkeby.json
new file mode 100644
index 00000000..a9b24933
--- /dev/null
+++ b/forks/passport/src/networks/rinkeby.json
@@ -0,0 +1,26 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  },
+  {
+    "contractName": "RequireFreshSignerLib",
+    "address": "0xE6B9B21C077F382333220a072e4c44280b873907"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/networks/ropsten.json b/forks/passport/src/networks/ropsten.json
new file mode 100644
index 00000000..a9b24933
--- /dev/null
+++ b/forks/passport/src/networks/ropsten.json
@@ -0,0 +1,26 @@
+[
+  {
+    "contractName": "WalletFactory",
+    "address": "0xf9D09D634Fb818b05149329C1dcCFAeA53639d96"
+  },
+  {
+    "contractName": "MainModule",
+    "address": "0xd01F11855bCcb95f88D7A48492F66410d4637313"
+  },
+  {
+    "contractName": "MainModuleUpgradable",
+    "address": "0x7EFE6cE415956c5f80C6530cC6cc81b4808F6118"
+  },
+  {
+    "contractName": "GuestModule",
+    "address": "0x02390F3E6E5FD1C6786CB78FD3027C117a9955A7"
+  },
+  {
+    "contractName": "SequenceUtils",
+    "address": "0xd130B43062D875a4B7aF3f8fc036Bc6e9D3E1B3E"
+  },
+  {
+    "contractName": "RequireFreshSignerLib",
+    "address": "0xE6B9B21C077F382333220a072e4c44280b873907"
+  }
+]
\ No newline at end of file
diff --git a/forks/passport/src/package.json b/forks/passport/src/package.json
new file mode 100644
index 00000000..8a236b35
--- /dev/null
+++ b/forks/passport/src/package.json
@@ -0,0 +1,29 @@
+{
+  "name": "@imtbl/wallet-contracts",
+  "version": "1.0.0",
+  "description": "Immutable Smart Wallet contracts",
+  "repository": "https://github.com/immutable/wallet-contracts",
+  "homepage": "https://www.immutable.com",
+  "license": "Apache-2.0",
+  "main": "gen/adapter/index.js",
+  "types": "gen/adapter/index.d.ts",
+  "scripts": {},
+  "optionalDependencies": {
+    "@ethersproject/abi": "^5.7.0",
+    "@ethersproject/providers": "^5.7.2",
+    "ethers": "^5.7.2"
+  },
+  "dependencies": {},
+  "devDependencies": {
+    "prettier": "^2.8.7",
+    "prettier-plugin-solidity": "^1.1.3"
+  },
+  "keywords": [
+    "Ethereum",
+    "Immutable",
+    "Sequence",
+    "Wallet",
+    "Smart Wallet",
+    "Meta Transactions"
+  ]
+}
diff --git a/forks/passport/src/tsconfig.adapter.json b/forks/passport/src/tsconfig.adapter.json
new file mode 100644
index 00000000..7e9435e7
--- /dev/null
+++ b/forks/passport/src/tsconfig.adapter.json
@@ -0,0 +1,36 @@
+{
+  "compilerOptions": {
+    "target": "es2020",
+    "module": "commonjs",
+    "moduleResolution": "node",
+    "declaration": true,
+    "sourceMap": false,
+    "allowSyntheticDefaultImports": true,
+    "resolveJsonModule": true,
+    "downlevelIteration": true,
+    "removeComments": false,
+    "skipLibCheck": true,
+
+    "strictNullChecks": true,
+    "noImplicitUseStrict": true,
+    "noImplicitAny": false,
+    "noImplicitReturns": true,
+    "noImplicitThis": true,
+    "noUnusedParameters": false,
+    "noErrorTruncation": true,
+    "esModuleInterop": true,
+
+    "experimentalDecorators": true,
+    "forceConsistentCasingInFileNames": true,
+    "allowJs": false,
+    "checkJs": false,
+
+    "baseUrl": ".",
+    "outDir": "./gen/adapter",
+    "lib": ["es2020"],
+  },
+
+  "include": [
+    "./gen/typechain"
+  ]
+}
diff --git a/forks/passport/tests/ERC165.spec.ts b/forks/passport/tests/ERC165.spec.ts
new file mode 100644
index 00000000..3575c1f3
--- /dev/null
+++ b/forks/passport/tests/ERC165.spec.ts
@@ -0,0 +1,187 @@
+import { ethers as hardhat, web3 } from 'hardhat'
+import { ethers } from 'ethers'
+import { expect, encodeImageHash, signAndExecuteMetaTx, interfaceIdOf, addressOf } from './utils'
+
+import {
+  MainModule,
+  MainModuleUpgradable,
+  MainModuleDynamicAuth,
+  StartupWalletImpl,
+  LatestWalletImplLocator,
+  Factory,
+  Factory__factory,
+  MainModule__factory,
+  MainModuleUpgradable__factory,
+  MainModuleDynamicAuth__factory,
+  ERC165CheckerMock__factory,
+  StartupWalletImpl__factory,
+  LatestWalletImplLocator__factory,
+} from '../src'
+
+ethers.utils.Logger.setLogLevel(ethers.utils.Logger.levels.ERROR)
+
+const interfaceIds = [
+  'IModuleHooks',
+  'IERC223Receiver',
+  'IERC721Receiver',
+  'IERC1155Receiver',
+  'IERC1271Wallet',
+  'IModuleCalls',
+  'IModuleCreator',
+  'IModuleUpdate'
+]
+
+const dynamicModuleInterfaceIds = [
+  'IERC223Receiver',
+  'IERC721Receiver',
+  'IERC1155Receiver',
+  'IERC1271Wallet',
+  'IModuleCalls',
+  'IModuleUpdate'
+]
+
+contract('ERC165', () => {
+  // let provider: ethers.providers.Provider
+  let signer: ethers.Signer
+  let networkId: number
+
+  let factory: Factory
+  let mainModule: MainModule
+  let moduleUpgradable: MainModuleUpgradable
+  let moduleDynamicAuth: MainModuleDynamicAuth
+  let startupWalletImpl: StartupWalletImpl
+  let moduleLocator: LatestWalletImplLocator
+
+  let owner: ethers.Wallet
+  let wallet: MainModule
+
+  let erc165checker
+
+  before(async () => {
+    // get signer and provider from hardhat
+    signer = (await hardhat.getSigners())[0]
+    // provider = hardhat.provider
+
+    // Get network ID
+    networkId = process.env.NET_ID ? Number(process.env.NET_ID) : await web3.eth.net.getId()
+
+    // Deploy wallet factory
+    factory = await new Factory__factory().connect(signer).deploy(await signer.getAddress(), await signer.getAddress())
+    // Startup and Locator
+    moduleLocator = await new LatestWalletImplLocator__factory().connect(signer).deploy(await signer.getAddress(), await signer.getAddress())
+    startupWalletImpl = await new StartupWalletImpl__factory().connect(signer).deploy(moduleLocator.address)
+    // Deploy MainModule
+    mainModule = await new MainModule__factory().connect(signer).deploy(factory.address)
+    moduleUpgradable = await new MainModuleUpgradable__factory().connect(signer).deploy()
+    moduleDynamicAuth = await new MainModuleDynamicAuth__factory().connect(signer).deploy(factory.address, startupWalletImpl.address)
+    // Deploy ERC165 Checker
+    erc165checker = await new ERC165CheckerMock__factory().connect(signer).deploy()
+
+    // Update the implementation address for the startup wallet
+    await moduleLocator.connect(signer).changeWalletImplementation(moduleDynamicAuth.address)
+  })
+
+  beforeEach(async () => {
+    owner = new ethers.Wallet(ethers.utils.randomBytes(32))
+    const salt = encodeImageHash(1, [{ weight: 1, address: owner.address }])
+    await factory.deploy(mainModule.address, salt)
+    wallet = MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+  })
+
+  describe('Implement all interfaces for ERC165 on MainModule', () => {
+    interfaceIds.forEach(element => {
+      it(`Should return implements ${element} interfaceId`, async () => {
+        const interfaceId = interfaceIdOf(new ethers.utils.Interface(artifacts.require(element).abi))
+        expect(web3.utils.toBN(interfaceId)).to.not.eq.BN(0)
+
+        const erc165result = await erc165checker.doesContractImplementInterface(wallet.address, interfaceId)
+        expect(erc165result).to.be.true
+      })
+    })
+  })
+  describe('Implement all interfaces for ERC165 on MainModuleUpgradable', () => {
+    beforeEach(async () => {
+      const newOwner = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const newImageHash = encodeImageHash(1, [{ weight: 1, address: newOwner.address }])
+      const newWallet = MainModuleUpgradable__factory.connect(wallet.address, signer)
+
+      const migrateTransactions = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: ethers.constants.Two.pow(21),
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('updateImplementation', [moduleUpgradable.address])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: ethers.constants.Two.pow(21),
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: newWallet.interface.encodeFunctionData('updateImageHash', [newImageHash])
+        }
+      ]
+
+      await signAndExecuteMetaTx(wallet, owner, migrateTransactions, networkId)
+      wallet = newWallet as unknown as MainModule
+    })
+    interfaceIds.concat('IModuleAuthUpgradable').forEach(element => {
+      it(`Should return implements ${element} interfaceId`, async () => {
+        const interfaceId = interfaceIdOf(new ethers.utils.Interface(artifacts.require(element).abi))
+        expect(web3.utils.toBN(interfaceId)).to.not.eq.BN(0)
+
+        const erc165result = await erc165checker.doesContractImplementInterface(wallet.address, interfaceId)
+        expect(erc165result).to.be.true
+      })
+    })
+  })
+  describe('Implement all interfaces for ERC165 on MainModuleDynamicAuth', () => {
+    beforeEach(async () => {
+      owner = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt = encodeImageHash(1, [{ weight: 1, address: owner.address }])
+
+      // Bypass the startup contract initialization to make sure the gas costs
+      // are low, by pointing directly to the implementation contract.
+      await factory.deploy(moduleDynamicAuth.address, salt)
+      wallet = MainModule__factory.connect(addressOf(factory.address, moduleDynamicAuth.address, salt), signer)
+    })
+
+    // Should implement a fixed set of interfaces
+    dynamicModuleInterfaceIds.forEach(id => {
+      it(`Should implement the ${id} interface`, async () => {
+        const interfaceId = interfaceIdOf(new ethers.utils.Interface(artifacts.require(id).abi))
+        expect(web3.utils.toBN(interfaceId)).to.not.eq.BN(0)
+
+        const erc165result = await erc165checker.doesContractImplementInterface(wallet.address, interfaceId)
+        expect(erc165result).to.be.true
+      })
+    })
+
+    // And should not implement other interfaces
+    interfaceIds.filter(id => !dynamicModuleInterfaceIds.includes(id)).forEach(id => {
+      it(`Should not implement the ${id} interface`, async () => {
+        const interfaceId = interfaceIdOf(new ethers.utils.Interface(artifacts.require(id).abi))
+        expect(web3.utils.toBN(interfaceId)).to.not.eq.BN(0)
+
+        const erc165result = await erc165checker.doesContractImplementInterface(wallet.address, interfaceId)
+        expect(erc165result).to.be.false
+      })
+    })
+  })
+  describe('Manually defined interfaces', () => {
+    const interfaces = [
+      ['ERC165', '0x01ffc9a7'],
+      ['ERC721', '0x150b7a02'],
+      ['ERC1155', '0x4e2312e0']
+    ]
+
+    interfaces.forEach(i => {
+      it(`Should implement ${i[0]} interface`, async () => {
+        const erc165result = await erc165checker.doesContractImplementInterface(wallet.address, i[1])
+        expect(erc165result).to.be.true
+      })
+    })
+  })
+})
diff --git a/forks/passport/tests/Factory.spec.ts b/forks/passport/tests/Factory.spec.ts
new file mode 100644
index 00000000..2d543efe
--- /dev/null
+++ b/forks/passport/tests/Factory.spec.ts
@@ -0,0 +1,41 @@
+import { ethers as hardhat, web3 } from 'hardhat'
+import { ethers } from 'ethers'
+import { expect, addressOf } from './utils'
+
+import { ModuleMock, Factory, ModuleMock__factory, Factory__factory } from '../src'
+
+ethers.utils.Logger.setLogLevel(ethers.utils.Logger.levels.ERROR)
+
+contract('Factory', (accounts: string[]) => {
+  let signer: ethers.Signer
+  let moduleMock: ModuleMock
+  let factory: Factory
+
+  beforeEach(async () => {
+    signer = (await hardhat.getSigners())[0]
+    moduleMock = await new ModuleMock__factory().connect(signer).deploy()
+    factory = await new Factory__factory().connect(signer).deploy(await signer.getAddress(), await signer.getAddress())
+  })
+
+  describe('Deploy wallets', () => {
+    it('Should deploy wallet', async () => {
+      // const saltHash = encodeImageHash(1, [{ weight: 1, address: accounts[0] }])
+      const hash = ethers.utils.hexlify(ethers.utils.randomBytes(32))
+      await factory.deploy(moduleMock.address, hash)
+    })
+    it('Should predict wallet address', async () => {
+      const hash = ethers.utils.hexlify(ethers.utils.randomBytes(32))
+      const predict = addressOf(factory.address, moduleMock.address, hash)
+      await factory.deploy(moduleMock.address, hash)
+      expect(await web3.eth.getCode(predict)).to.not.equal('0x')
+    })
+    it('Should initialize with main module', async () => {
+      const hash = ethers.utils.hexlify(ethers.utils.randomBytes(32))
+      await factory.deploy(moduleMock.address, hash)
+      const address = addressOf(factory.address, moduleMock.address, hash)
+      const wallet = await ModuleMock__factory.connect(address, signer)
+      const receipt = await (await wallet.ping()).wait()
+      expect(receipt.events![0].event).to.equal('Pong')
+    })
+  })
+})
diff --git a/forks/passport/tests/GasEstimation.spec.ts b/forks/passport/tests/GasEstimation.spec.ts
new file mode 100644
index 00000000..7b9f3a8b
--- /dev/null
+++ b/forks/passport/tests/GasEstimation.spec.ts
@@ -0,0 +1,522 @@
+import { ethers } from 'ethers'
+import { ethers as hardhat, web3 } from 'hardhat'
+import {
+  expect,
+  encodeImageHash,
+  signAndEncodeMetaTxn,
+  addressOf,
+  multiSignAndEncodeMetaTxn,
+  multiSignAndExecuteMetaTx
+} from './utils'
+
+import {
+  Factory__factory,
+  GasEstimator__factory,
+  CallReceiverMock__factory,
+  GuestModule__factory,
+  MainModuleGasEstimation__factory,
+  ModuleMock__factory
+} from '../src'
+import { GasEstimator, CallReceiverMock, MainModuleGasEstimation, Factory, GuestModule, MainModule } from 'src/gen/typechain'
+
+ethers.utils.Logger.setLogLevel(ethers.utils.Logger.levels.ERROR)
+
+function txBaseCost(data: ethers.BytesLike): number {
+  const bytes = ethers.utils.arrayify(data)
+  return bytes
+    .reduce((p, c) => (c == 0 ? p.add(4) : p.add(16)), ethers.constants.Zero)
+    .add(21000)
+    .toNumber()
+}
+
+// NOTE/TODO/REVIEW: all of the 'approximately' gasUsed numbers are way too high, previously we could compute
+// gas limit to a variance of 2000, and now have to use 40_000+. The contracts are the same, the only difference
+// is a newer version of ethers, and newer version of hardhat.
+//
+// Another theory, it looks like the baseCost is already factored into the estimation somehow? .. or, hardhat
+// doesn't include the baseCost for its transactions, so maybe this is off by the configuration.
+// You can see this if you compare 'estimated' below to the 'gasUsed'
+
+contract('Estimate gas usage', (accounts: string[]) => {
+  let signer: ethers.Signer
+  let networkId: number
+
+  let gasEstimator: GasEstimator
+  let callReceiver: CallReceiverMock
+  let mainModule: MainModuleGasEstimation
+  let guestModule: GuestModule
+  let factory: Factory
+
+  let estimate: typeof gasEstimator.functions.estimate
+
+  let owner: ethers.Wallet
+  let salt: string
+  let address: string
+
+  before(async () => {
+    // get signer from hardhat
+    signer = (await hardhat.getSigners())[0]
+
+    // Get network ID
+    networkId = process.env.NET_ID ? Number(process.env.NET_ID) : await web3.eth.net.getId()
+
+    // Deploy
+    gasEstimator = await new GasEstimator__factory().connect(signer).deploy()
+    callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+    // Deploy wallet factory
+    factory = await new Factory__factory().connect(signer).deploy(await signer.getAddress(), await signer.getAddress())
+
+    // Deploy MainModuleGasEstimation (hardhat doesn't support overwrites, so we use this as the real module)
+    mainModule = await new MainModuleGasEstimation__factory().connect(signer).deploy()
+
+    guestModule = await new GuestModule__factory().connect(signer).deploy()
+
+    estimate = gasEstimator.functions.estimate
+  })
+
+  beforeEach(async () => {
+    await callReceiver.setRevertFlag(false)
+    await callReceiver.testCall(0, [])
+
+    owner = ethers.Wallet.createRandom()
+
+    salt = encodeImageHash(1, [{ weight: 1, address: owner.address }])
+    address = addressOf(factory.address, mainModule.address, salt)
+  })
+
+  describe('Estimate gas of transactions', () => {
+    context('without wallet deployed', () => {
+      // NOTE: upstream (0xSequence) simulates deploy + transaction, here we
+      // simulate only the transaction itself.
+      const bundle = (txData: string) => {
+        return [
+          {
+            delegateCall: false,
+            revertOnError: true,
+            target: address,
+            gasLimit: 0,
+            data: txData,
+            value: 0
+          }
+        ]
+      }
+
+      it('Should estimate wallet deployment', async () => {
+        const factoryData = factory.interface.encodeFunctionData('deploy', [mainModule.address, salt])
+        const estimated = ethers.BigNumber.from((await estimate(factory.address, factoryData)).gasLimit).toNumber()
+        const realTx = await factory.deploy(mainModule.address, salt)
+        const realTxReceipt = await realTx.wait()
+
+        // NOTE: we set a high variance here as gas estimation on factory is a bit tricky from hardhat.
+        // or perhaps the txBaseCost() is wrong somehow..? could be hardhat's opcode pricing is off too.
+        expect(estimated + txBaseCost(factoryData)).to.approximately(realTxReceipt.gasUsed.toNumber(), 100_000)
+      })
+
+      it('Should estimate wallet deployment + upgrade', async () => {
+        // const newImplementation = (await ModuleMockArtifact.new()) as ModuleMock
+        const newImplementation = await new ModuleMock__factory().connect(signer).deploy()
+
+        const transaction = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: 0,
+          target: owner.address,
+          value: ethers.constants.Zero,
+          data: mainModule.interface.encodeFunctionData('updateImplementation', [newImplementation.address])
+        }
+
+        const txData = await signAndEncodeMetaTxn(mainModule as unknown as MainModule, owner, [transaction], networkId)
+        const txDataNoSignature = await signAndEncodeMetaTxn(
+          mainModule as unknown as MainModule,
+          ethers.Wallet.createRandom(),
+          [transaction],
+          networkId
+        )
+
+        const bundleDataNoSignature = guestModule.interface.encodeFunctionData('execute', [bundle(txDataNoSignature), 0, []])
+
+        const estimated = ethers.BigNumber.from((await estimate(guestModule.address, bundleDataNoSignature)).gasLimit).toNumber()
+        const realTx = await guestModule.execute(bundle(txData), 0, [])
+        const realTxReceipt = await realTx.wait()
+
+        expect(estimated + txBaseCost(bundleDataNoSignature)).to.approximately(realTxReceipt.gasUsed.toNumber(), 40_000)
+      })
+      it('Should estimate wallet deployment + upgrade + transaction', async () => {
+        const newImplementation = await new ModuleMock__factory().connect(signer).deploy()
+
+        const transaction = [
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: 0,
+            target: owner.address,
+            value: ethers.constants.Zero,
+            data: mainModule.interface.encodeFunctionData('updateImplementation', [newImplementation.address])
+          },
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: 0,
+            target: callReceiver.address,
+            value: ethers.constants.Zero,
+            data: callReceiver.interface.encodeFunctionData('testCall', [1, ethers.utils.hexlify(ethers.utils.randomBytes(299))])
+          }
+        ]
+
+        const txData = await signAndEncodeMetaTxn(mainModule as any, owner, transaction, networkId)
+        const txDataNoSignature = await signAndEncodeMetaTxn(
+          mainModule as any,
+          ethers.Wallet.createRandom(),
+          transaction,
+          networkId
+        )
+
+        const bundleDataNoSignature = guestModule.interface.encodeFunctionData('execute', [bundle(txDataNoSignature), 0, []])
+
+        const estimated = ethers.BigNumber.from((await estimate(guestModule.address, bundleDataNoSignature)).gasLimit).toNumber()
+        const realTx = await guestModule.execute(bundle(txData), 0, [])
+        const realTxReceipt = await realTx.wait()
+
+        expect(estimated + txBaseCost(bundleDataNoSignature)).to.approximately(realTxReceipt.gasUsed.toNumber(), 50_000)
+      })
+      it('Should estimate wallet deployment + upgrade + failed transaction', async () => {
+        const newImplementation = await new ModuleMock__factory().connect(signer).deploy()
+
+        await callReceiver.setRevertFlag(true)
+
+        const transaction = [
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: 0,
+            target: owner.address,
+            value: ethers.constants.Zero,
+            data: mainModule.interface.encodeFunctionData('updateImplementation', [newImplementation.address])
+          },
+          {
+            delegateCall: false,
+            revertOnError: false,
+            gasLimit: 0,
+            target: callReceiver.address,
+            value: ethers.constants.Zero,
+            data: callReceiver.interface.encodeFunctionData('testCall', [1, ethers.utils.hexlify(ethers.utils.randomBytes(299))])
+          }
+        ]
+
+        const txData = await signAndEncodeMetaTxn(mainModule as any, owner, transaction, networkId)
+        const txDataNoSignature = await signAndEncodeMetaTxn(
+          mainModule as any,
+          ethers.Wallet.createRandom(),
+          transaction,
+          networkId
+        )
+
+        const bundleDataNoSignature = guestModule.interface.encodeFunctionData('execute', [bundle(txDataNoSignature), 0, []])
+
+        const estimated = ethers.BigNumber.from((await estimate(guestModule.address, bundleDataNoSignature)).gasLimit).toNumber()
+        const realTx = await guestModule.execute(bundle(txData), 0, [])
+        const realTxReceipt = await realTx.wait()
+
+        expect(estimated + txBaseCost(bundleDataNoSignature)).to.approximately(realTxReceipt.gasUsed.toNumber(), 50_000)
+      })
+      it('Should estimate wallet deployment + upgrade + fixed gas transaction', async () => {
+        const newImplementation = await new ModuleMock__factory().connect(signer).deploy()
+
+        const transaction = [
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: 0,
+            target: owner.address,
+            value: ethers.constants.Zero,
+            data: mainModule.interface.encodeFunctionData('updateImplementation', [newImplementation.address])
+          },
+          {
+            delegateCall: false,
+            revertOnError: false,
+            gasLimit: 900000,
+            target: callReceiver.address,
+            value: ethers.constants.Zero,
+            data: callReceiver.interface.encodeFunctionData('testCall', [1, ethers.utils.hexlify(ethers.utils.randomBytes(299))])
+          }
+        ]
+
+        const txData = await signAndEncodeMetaTxn(mainModule as any, owner, transaction, networkId)
+        const txDataNoSignature = await signAndEncodeMetaTxn(
+          mainModule as any,
+          ethers.Wallet.createRandom(),
+          transaction,
+          networkId
+        )
+
+        const bundleDataNoSignature = guestModule.interface.encodeFunctionData('execute', [bundle(txDataNoSignature), 0, []])
+
+        const estimated = ethers.BigNumber.from((await estimate(guestModule.address, bundleDataNoSignature)).gasLimit).toNumber()
+        const realTx = await guestModule.execute(bundle(txData), 0, [])
+        const realTxReceipt = await realTx.wait()
+
+        expect(estimated + txBaseCost(bundleDataNoSignature)).to.approximately(realTxReceipt.gasUsed.toNumber(), 50_000)
+      })
+    })
+
+    const options = [
+      {
+        name: 'single signer',
+        signers: 1
+      },
+      {
+        name: 'many signers',
+        signers: 32
+      }
+    ]
+
+    options.map(o => {
+      context(`with wallet deployed and ${o.name}`, () => {
+        let wallet: MainModule
+
+        let owners: ethers.Wallet[]
+        let config: { weight: number; address: string }[]
+        let threshold: number
+
+        let accounts: { weight: number; owner: ethers.Wallet }[]
+        let fakeAccounts: { weight: number; owner: ethers.Wallet }[]
+
+        beforeEach(async () => {
+          threshold = o.signers
+
+          owners = new Array(o.signers).fill(0).map(() => ethers.Wallet.createRandom())
+          config = owners.map(o => ({ weight: 1, address: o.address }))
+          salt = encodeImageHash(threshold, config)
+          address = addressOf(factory.address, mainModule.address, salt)
+
+          await factory.deploy(mainModule.address, salt)
+
+          wallet = (await MainModuleGasEstimation__factory.connect(address, signer)) as unknown as MainModule
+
+          accounts = owners.map(c => ({ weight: 1, owner: c }))
+          fakeAccounts = owners.map(c => ({ weight: 1, owner: ethers.Wallet.createRandom() }))
+        })
+
+        it('Should estimate single transaction', async () => {
+          const transaction = [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: 0,
+              target: callReceiver.address,
+              value: ethers.constants.Zero,
+              data: callReceiver.interface.encodeFunctionData('testCall', [
+                1,
+                ethers.utils.hexlify(ethers.utils.randomBytes(299))
+              ])
+            }
+          ]
+
+          const txDataNoSignature = await multiSignAndEncodeMetaTxn(
+            mainModule as any,
+            fakeAccounts,
+            threshold,
+            transaction,
+            networkId
+          )
+
+          const estimated = ethers.BigNumber.from((await estimate(address, txDataNoSignature)).gasLimit).toNumber()
+          const tx = await multiSignAndExecuteMetaTx(wallet, accounts, threshold, transaction, networkId, 0)
+          const gasUsed = (await tx.wait()).gasUsed.toNumber()
+
+          expect(estimated + txBaseCost(txDataNoSignature)).to.approximately(gasUsed, 100_000)
+        })
+        it('Should estimate multiple transactions', async () => {
+          const transaction = [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: 0,
+              target: callReceiver.address,
+              value: ethers.constants.Zero,
+              data: callReceiver.interface.encodeFunctionData('testCall', [
+                1,
+                ethers.utils.hexlify(ethers.utils.randomBytes(299))
+              ])
+            },
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: 0,
+              target: callReceiver.address,
+              value: ethers.constants.Zero,
+              data: callReceiver.interface.encodeFunctionData('testCall', [
+                1,
+                ethers.utils.hexlify(ethers.utils.randomBytes(2299))
+              ])
+            }
+          ]
+
+          const txDataNoSignature = await multiSignAndEncodeMetaTxn(
+            mainModule as any,
+            fakeAccounts,
+            threshold,
+            transaction,
+            networkId
+          )
+
+          const estimated = ethers.BigNumber.from((await estimate(address, txDataNoSignature)).gasLimit).toNumber()
+          const tx = await multiSignAndExecuteMetaTx(wallet, accounts, threshold, transaction, networkId, 0)
+          const gasUsed = (await tx.wait()).gasUsed.toNumber()
+
+          expect(estimated + txBaseCost(txDataNoSignature)).to.approximately(gasUsed, 150_000)
+        })
+        it('Should estimate multiple transactions with bad nonce', async () => {
+          const transaction = [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: 0,
+              target: callReceiver.address,
+              value: ethers.constants.Zero,
+              data: callReceiver.interface.encodeFunctionData('testCall', [
+                1,
+                ethers.utils.hexlify(ethers.utils.randomBytes(299))
+              ])
+            },
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: 0,
+              target: callReceiver.address,
+              value: ethers.constants.Zero,
+              data: callReceiver.interface.encodeFunctionData('testCall', [
+                1,
+                ethers.utils.hexlify(ethers.utils.randomBytes(2299))
+              ])
+            }
+          ]
+
+          const txDataNoSignature = await multiSignAndEncodeMetaTxn(
+            mainModule as any,
+            fakeAccounts,
+            threshold,
+            transaction,
+            networkId,
+            999999999
+          )
+
+          const estimated = ethers.BigNumber.from((await estimate(address, txDataNoSignature)).gasLimit).toNumber()
+          const tx = await multiSignAndExecuteMetaTx(wallet, accounts, threshold, transaction, networkId, 0)
+          const gasUsed = (await tx.wait()).gasUsed.toNumber()
+
+          expect(estimated + txBaseCost(txDataNoSignature)).to.approximately(gasUsed, 150_000)
+        })
+        it('Should estimate multiple transactions with failing transactions', async () => {
+          const altCallReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+          await altCallReceiver.setRevertFlag(true)
+
+          const transaction = [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: 0,
+              target: callReceiver.address,
+              value: ethers.constants.Zero,
+              data: callReceiver.interface.encodeFunctionData('testCall', [
+                1,
+                ethers.utils.hexlify(ethers.utils.randomBytes(299))
+              ])
+            },
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: 0,
+              target: callReceiver.address,
+              value: ethers.constants.Zero,
+              data: callReceiver.interface.encodeFunctionData('testCall', [
+                1,
+                ethers.utils.hexlify(ethers.utils.randomBytes(2299))
+              ])
+            },
+            {
+              delegateCall: false,
+              revertOnError: false,
+              gasLimit: 0,
+              target: altCallReceiver.address,
+              value: ethers.constants.Zero,
+              data: callReceiver.interface.encodeFunctionData('testCall', [
+                1,
+                ethers.utils.hexlify(ethers.utils.randomBytes(229))
+              ])
+            }
+          ]
+
+          const txDataNoSignature = await multiSignAndEncodeMetaTxn(
+            mainModule as any,
+            fakeAccounts,
+            threshold,
+            transaction,
+            networkId
+          )
+
+          const estimated = ethers.BigNumber.from((await estimate(address, txDataNoSignature)).gasLimit).toNumber()
+          const tx = await multiSignAndExecuteMetaTx(wallet, accounts, threshold, transaction, networkId, 0)
+          const gasUsed = (await tx.wait()).gasUsed.toNumber()
+
+          expect(estimated + txBaseCost(txDataNoSignature)).to.approximately(gasUsed, 120_000)
+        })
+        it('Should estimate multiple transactions with failing transactions and fixed gas limits', async () => {
+          const altCallReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+          await altCallReceiver.setRevertFlag(true)
+
+          const transaction = [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: 0,
+              target: callReceiver.address,
+              value: ethers.constants.Zero,
+              data: callReceiver.interface.encodeFunctionData('testCall', [
+                1,
+                ethers.utils.hexlify(ethers.utils.randomBytes(299))
+              ])
+            },
+            {
+              delegateCall: false,
+              revertOnError: false,
+              gasLimit: 90000,
+              target: callReceiver.address,
+              value: ethers.constants.Zero,
+              data: callReceiver.interface.encodeFunctionData('testCall', [
+                1,
+                ethers.utils.hexlify(ethers.utils.randomBytes(2299))
+              ])
+            },
+            {
+              delegateCall: false,
+              revertOnError: false,
+              gasLimit: 0,
+              target: altCallReceiver.address,
+              value: ethers.constants.Zero,
+              data: callReceiver.interface.encodeFunctionData('testCall', [
+                1,
+                ethers.utils.hexlify(ethers.utils.randomBytes(229))
+              ])
+            }
+          ]
+
+          const txDataNoSignature = await multiSignAndEncodeMetaTxn(
+            mainModule as any,
+            fakeAccounts,
+            threshold,
+            transaction,
+            networkId
+          )
+
+          const estimated = ethers.BigNumber.from((await estimate(address, txDataNoSignature)).gasLimit).toNumber()
+          const tx = await multiSignAndExecuteMetaTx(wallet, accounts, threshold, transaction, networkId, 0)
+          const gasUsed = (await tx.wait()).gasUsed.toNumber()
+
+          expect(estimated + txBaseCost(txDataNoSignature)).to.approximately(gasUsed, 150_000)
+        })
+      })
+    })
+  })
+})
diff --git a/forks/passport/tests/GuestModule.spec.ts b/forks/passport/tests/GuestModule.spec.ts
new file mode 100644
index 00000000..358cc20e
--- /dev/null
+++ b/forks/passport/tests/GuestModule.spec.ts
@@ -0,0 +1,140 @@
+import { ethers } from 'ethers'
+import { expect, RevertError } from './utils'
+
+import { GuestModule, CallReceiverMock, HookCallerMock, MainModuleUpgradable } from 'src/gen/typechain'
+
+import { CallReceiverMock__factory, GuestModule__factory, HookCallerMock__factory, MainModuleUpgradable__factory } from '../src'
+
+import { ethers as hardhat, web3 } from 'hardhat'
+
+ethers.utils.Logger.setLogLevel(ethers.utils.Logger.levels.ERROR)
+
+contract('GuestModule', (accounts: string[]) => {
+  let signer: ethers.Signer
+  let guestModule: GuestModule
+  let callReceiver: CallReceiverMock
+  let hookMock: HookCallerMock
+
+  describe('GuestModule wallet', () => {
+    before(async () => {
+      signer = (await hardhat.getSigners())[0]
+
+      // Deploy
+      guestModule = await (new GuestModule__factory()).connect(signer).deploy()
+      callReceiver = await (new CallReceiverMock__factory()).connect(signer).deploy()
+      hookMock = await (new HookCallerMock__factory()).connect(signer).deploy()
+    })
+
+    let valA
+    let valB
+    let transactions
+
+    beforeEach(async () => {
+      valA = web3.utils.toBN(web3.utils.randomHex(3)).toNumber()
+      valB = web3.utils.randomHex(120)
+
+      transactions = [
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: ethers.constants.Two.pow(20),
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+      ]
+    })
+
+    it('Should accept transactions without signature', async () => {
+      await guestModule.execute(transactions, 0, [])
+
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+    it('Should accept transactions on selfExecute', async () => {
+      await guestModule.selfExecute(transactions)
+
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+    it('Should accept transactions with random signature', async () => {
+      const signature = web3.utils.randomHex(69)
+
+      await guestModule.execute(transactions, 0, signature)
+
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+    it('Should accept transactions with random nonce', async () => {
+      const nonce = 9123891
+
+      await guestModule.execute(transactions, nonce, [])
+
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+    it('Should revert on delegateCall transactions', async () => {
+      const transactions = [
+        {
+          delegateCall: true,
+          revertOnError: false,
+          gasLimit: 1000000,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+      ]
+
+      const tx = guestModule.selfExecute(transactions)
+      await expect(tx).to.be.rejectedWith(RevertError('GuestModule#_executeGuest: delegateCall not allowed'))
+    })
+    it('Should not accept ETH', async () => {
+      const tx = signer.sendTransaction({
+        to: guestModule.address,
+        value: 1
+      })
+      await expect(tx).to.be.rejected
+    })
+    it('Should not implement hooks', async () => {
+      const tx = hookMock.callERC1155Received(guestModule.address)
+      await expect(tx).to.be.rejected
+    })
+    it('Should not be upgradeable', async () => {
+      const mainModule = await (new MainModuleUpgradable__factory()).connect(signer).deploy()
+      const newImageHash = web3.utils.randomHex(32)
+
+      const migrateBundle = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: ethers.constants.Two.pow(18),
+          target: guestModule.address,
+          value: ethers.constants.Zero,
+          data: mainModule.interface.encodeFunctionData('updateImplementation', [mainModule.address])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: ethers.constants.Two.pow(18),
+          target: guestModule.address,
+          value: ethers.constants.Zero,
+          data: mainModule.interface.encodeFunctionData('updateImageHash', [newImageHash])
+        }
+      ]
+
+      const migrateTransaction = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: ethers.constants.Two.pow(18),
+          target: guestModule.address,
+          value: ethers.constants.Zero,
+          data: guestModule.interface.encodeFunctionData('selfExecute', [migrateBundle])
+        }
+      ]
+
+      const tx = guestModule.selfExecute(migrateTransaction)
+      await expect(tx).to.be.rejected
+    })
+  })
+})
diff --git a/forks/passport/tests/ImmutableDeployment.spec.ts b/forks/passport/tests/ImmutableDeployment.spec.ts
new file mode 100644
index 00000000..41b19f14
--- /dev/null
+++ b/forks/passport/tests/ImmutableDeployment.spec.ts
@@ -0,0 +1,399 @@
+import { ethers as hardhat } from 'hardhat'
+import { ethers } from 'ethers'
+import { getContractAddress } from '@ethersproject/address'
+import {
+  Factory,
+  Factory__factory,
+  MainModule__factory,
+  ImmutableSigner,
+  ImmutableSigner__factory,
+  LatestWalletImplLocator__factory,
+  StartupWalletImpl__factory,
+  MainModuleDynamicAuth,
+  MainModuleDynamicAuth__factory,
+} from '../src'
+import {
+  encodeImageHash,
+  expect,
+  addressOf,
+  encodeMetaTransactionsData,
+  walletMultiSign,
+  ethSign,
+} from './utils'
+import { LatestWalletImplLocator, StartupWalletImpl } from 'src/gen/typechain'
+
+describe('E2E Immutable Wallet Deployment', () => {
+  let contractDeployerEOA: ethers.Signer
+  let relayerEOA: ethers.Signer
+
+  let userEOA: ethers.Signer
+  let immutableEOA: ethers.Signer
+  let randomEOA: ethers.Signer
+  let adminEOA: ethers.Signer
+  let walletDeployerEOA: ethers.Signer
+
+  // All contracts involved in the wallet ecosystem
+  let factory: Factory
+  let mainModuleDynamicAuth: MainModuleDynamicAuth
+  let immutableSigner: ImmutableSigner
+  let moduleLocator: LatestWalletImplLocator
+  let startupWallet: StartupWalletImpl
+
+  const WALLET_FACTORY_NONCE = 1
+  const STARTUP_WALLET_NONCE = 2
+  const IMMUTABLE_SIGNER_NONCE = 3
+  const LOCATOR_NONCE = 4
+  const MAIN_MODULE_DYNAMIC_AUTH_NONCE = 5
+
+  beforeEach(async () => {
+    [
+      userEOA,
+      immutableEOA,
+      randomEOA,
+      adminEOA,
+      walletDeployerEOA,
+      relayerEOA,
+      contractDeployerEOA
+    ] = await hardhat.getSigners()
+
+    await hardhat.provider.send("hardhat_reset", [])
+
+    // Matches the production environment where the first transaction (nonce 0)
+    // is used for testing.
+    contractDeployerEOA.sendTransaction({ to: ethers.constants.AddressZero, value: 0 })
+
+    // Nonce 1
+    factory = await new Factory__factory()
+      .connect(contractDeployerEOA)
+      .deploy(await adminEOA.getAddress(), await walletDeployerEOA.getAddress())
+
+    // Calculate the locator address ahead of time
+    const moduleLocatorAddress = getContractAddress({
+      from: await contractDeployerEOA.getAddress(),
+      nonce: LOCATOR_NONCE
+    })
+
+    // Nonce 2
+    startupWallet = await new StartupWalletImpl__factory()
+      .connect(contractDeployerEOA)
+      .deploy(moduleLocatorAddress)
+
+    // Nonce 3
+    immutableSigner = await new ImmutableSigner__factory()
+      .connect(contractDeployerEOA)
+      .deploy(await adminEOA.getAddress(), await adminEOA.getAddress(), await immutableEOA.getAddress())
+
+    // NOTE: Those could possibly be deployed by a separate account instead.
+
+    // Nonce 4
+    moduleLocator = await new LatestWalletImplLocator__factory()
+      .connect(contractDeployerEOA)
+      .deploy(await adminEOA.getAddress(), await walletDeployerEOA.getAddress())
+
+    // Nonce 5
+    mainModuleDynamicAuth = await new MainModuleDynamicAuth__factory()
+      .connect(contractDeployerEOA)
+      .deploy(factory.address, startupWallet.address)
+
+    // Setup the latest implementation address
+    await moduleLocator
+      .connect(walletDeployerEOA)
+      .changeWalletImplementation(mainModuleDynamicAuth.address)
+  })
+
+  it('Should create deterministic contract addresses', async () => {
+    // Generate deployed contract addresses offchain from the deployer address
+    // and fixed nonces.
+    const factoryAddress = getContractAddress({
+      from: await contractDeployerEOA.getAddress(),
+      nonce: WALLET_FACTORY_NONCE
+    })
+
+    const startupWalletAddress = getContractAddress({
+      from: await contractDeployerEOA.getAddress(),
+      nonce: STARTUP_WALLET_NONCE
+    })
+
+    const immutableSignerAddress = getContractAddress({
+      from: await contractDeployerEOA.getAddress(),
+      nonce: IMMUTABLE_SIGNER_NONCE
+    })
+
+    const locatorAddress = getContractAddress({
+      from: await contractDeployerEOA.getAddress(),
+      nonce: LOCATOR_NONCE
+    })
+
+    const mainModuleAddress = getContractAddress({
+      from: await contractDeployerEOA.getAddress(),
+      nonce: MAIN_MODULE_DYNAMIC_AUTH_NONCE
+    })
+
+    // Check they match against the actual deployed addresses
+    expect(factory.address).to.equal(factoryAddress)
+    expect(mainModuleDynamicAuth.address).to.equal(mainModuleAddress)
+    expect(immutableSigner.address).to.equal(immutableSignerAddress)
+    expect(startupWallet.address).to.equal(startupWalletAddress)
+    expect(moduleLocator.address).to.equal(locatorAddress)
+  })
+
+  it('Should execute a transaction signed by the ImmutableSigner', async () => {
+    // Deploy wallet
+    const walletSalt = encodeImageHash(2, [
+      { weight: 1, address: await userEOA.getAddress() },
+      { weight: 1, address: immutableSigner.address }
+    ])
+    const walletAddress = addressOf(factory.address, startupWallet.address, walletSalt)
+    const walletDeploymentTx = await factory.connect(walletDeployerEOA).deploy(startupWallet.address, walletSalt)
+    await walletDeploymentTx.wait()
+
+    // Connect to the generated user address
+    const wallet = MainModule__factory.connect(walletAddress, relayerEOA)
+
+    // Transfer funds to the SCW
+    const transferTx = await relayerEOA.sendTransaction({ to: walletAddress, value: 1 })
+    await transferTx.wait()
+
+    // Return funds
+    const transaction = {
+      delegateCall: false,
+      revertOnError: true,
+      gasLimit: 1000000,
+      target: await randomEOA.getAddress(),
+      value: 1,
+      data: []
+    }
+
+    // Build meta-transaction
+    const networkId = (await hardhat.provider.getNetwork()).chainId
+    const nonce = 0
+    const data = encodeMetaTransactionsData(wallet.address, [transaction], networkId, nonce)
+
+    const signature = await walletMultiSign(
+      [
+        { weight: 1, owner: userEOA as ethers.Wallet },
+        // 03 -> Call the address' isValidSignature()
+        { weight: 1, owner: immutableSigner.address, signature: (await ethSign(immutableEOA as ethers.Wallet, data)) + '03' }
+      ],
+      2,
+      data,
+      false
+    )
+
+    const originalBalance = await randomEOA.getBalance();
+
+    const executionTx = await wallet.execute([transaction], nonce, signature)
+    await executionTx.wait()
+
+    expect(await randomEOA.getBalance()).to.equal(originalBalance.add(1))
+  })
+
+  it('Should execute multiple transactions signed by the ImmutableSigner', async () => {
+    // Deploy wallet
+    const walletSalt = encodeImageHash(2, [
+      { weight: 1, address: await userEOA.getAddress() },
+      { weight: 1, address: immutableSigner.address }
+    ])
+    const walletAddress = addressOf(factory.address, startupWallet.address, walletSalt)
+    const walletDeploymentTx = await factory.connect(walletDeployerEOA).deploy(startupWallet.address, walletSalt)
+    await walletDeploymentTx.wait()
+
+    // Connect to the generated user address
+    const wallet = MainModule__factory.connect(walletAddress, relayerEOA)
+
+    // Transfer funds to the SCW
+    const transferTx = await relayerEOA.sendTransaction({ to: walletAddress, value: 5 })
+    await transferTx.wait()
+
+    for (const nonce of [0, 1, 2, 3, 4]) {
+      // Return funds
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: 1000000,
+        target: await randomEOA.getAddress(),
+        value: 1,
+        data: []
+      }
+
+      // Build meta-transaction
+      const networkId = (await hardhat.provider.getNetwork()).chainId
+      const data = encodeMetaTransactionsData(wallet.address, [transaction], networkId, nonce)
+
+      const signature = await walletMultiSign(
+        [
+          { weight: 1, owner: userEOA as ethers.Wallet },
+          // 03 -> Call the address' isValidSignature()
+          { weight: 1, owner: immutableSigner.address, signature: (await ethSign(immutableEOA as ethers.Wallet, data)) + '03' }
+        ],
+        2,
+        data,
+        false
+      )
+
+      const originalBalance = await randomEOA.getBalance();
+
+      const executionTx = await wallet.execute([transaction], nonce, signature)
+      await executionTx.wait()
+
+      expect(await randomEOA.getBalance()).to.equal(originalBalance.add(1))
+    }
+  })
+
+  it('Should not execute a transaction not signed by the ImmutableSigner', async () => {
+    // Deploy wallet
+    const walletSalt = encodeImageHash(2, [
+      { weight: 1, address: await userEOA.getAddress() },
+      { weight: 1, address: immutableSigner.address }
+    ])
+    const walletAddress = addressOf(factory.address, startupWallet.address, walletSalt)
+    const walletDeploymentTx = await factory.connect(walletDeployerEOA).deploy(startupWallet.address, walletSalt)
+    await walletDeploymentTx.wait()
+
+    // Connect to the generated user address
+    const wallet = MainModule__factory.connect(walletAddress, relayerEOA)
+
+    // Transfer funds to the SCW
+    const transferTx = await relayerEOA.sendTransaction({ to: walletAddress, value: 1 })
+    await transferTx.wait()
+
+    // Return funds
+    const transaction = {
+      delegateCall: false,
+      revertOnError: true,
+      gasLimit: 1000000,
+      target: await randomEOA.getAddress(),
+      value: 1,
+      data: []
+    }
+
+    // Build meta-transaction
+    const networkId = (await hardhat.provider.getNetwork()).chainId
+    const nonce = 0
+    const data = encodeMetaTransactionsData(wallet.address, [transaction], networkId, nonce)
+
+    const signature = await walletMultiSign(
+      [
+        { weight: 1, owner: userEOA as ethers.Wallet },
+        { weight: 1, owner: immutableSigner.address, signature: (await ethSign(randomEOA as ethers.Wallet, data)) + '03' }
+      ],
+      2,
+      data,
+      false
+    )
+
+    await expect(wallet.execute([transaction], nonce, signature)).to.be.revertedWith(
+      'ModuleAuth#_signatureValidation: INVALID_SIGNATURE'
+    )
+  })
+
+  it('Should not execute a transaction signed by the ImmutableSigner with the incorrect data', async () => {
+    // Deploy wallet
+    const walletSalt = encodeImageHash(2, [
+      { weight: 1, address: await userEOA.getAddress() },
+      { weight: 1, address: immutableSigner.address }
+    ])
+    const walletAddress = addressOf(factory.address, startupWallet.address, walletSalt)
+    const walletDeploymentTx = await factory.connect(walletDeployerEOA).deploy(startupWallet.address, walletSalt)
+    await walletDeploymentTx.wait()
+
+    // Connect to the generated user address
+    const wallet = MainModule__factory.connect(walletAddress, relayerEOA)
+
+    // Transfer funds to the SCW
+    const transferTx = await relayerEOA.sendTransaction({ to: walletAddress, value: 1 })
+    await transferTx.wait()
+
+    // Return funds
+    const transaction = {
+      delegateCall: false,
+      revertOnError: true,
+      gasLimit: 1000000,
+      target: await relayerEOA.getAddress(),
+      value: 1,
+      data: []
+    }
+
+    // Build meta-transaction
+    const networkId = (await hardhat.provider.getNetwork()).chainId
+    const nonce = 0
+    const data = encodeMetaTransactionsData(wallet.address, [transaction], networkId, nonce)
+
+    const meaninglessDataDisturbance = '05'
+    const signature = await walletMultiSign(
+      [
+        { weight: 1, owner: userEOA as ethers.Wallet },
+        // 03 -> Call the address' isValidSignature()
+        {
+          weight: 1,
+          owner: immutableSigner.address,
+          signature: (await ethSign(immutableEOA as ethers.Wallet, data + meaninglessDataDisturbance)) + '03'
+        }
+      ],
+      2,
+      data,
+      false
+    )
+
+    await expect(wallet.execute([transaction], nonce, signature)).to.be.revertedWith(
+      'ModuleAuth#_signatureValidation: INVALID_SIGNATURE'
+    )
+  })
+
+  it('Should not execute a transaction signed by an outdated signer', async () => {
+    // Deploy wallet
+    const walletSalt = encodeImageHash(2, [
+      { weight: 1, address: await userEOA.getAddress() },
+      { weight: 1, address: immutableSigner.address }
+    ])
+    const walletAddress = addressOf(factory.address, startupWallet.address, walletSalt)
+    const walletDeploymentTx = await factory.connect(walletDeployerEOA).deploy(startupWallet.address, walletSalt)
+    await walletDeploymentTx.wait()
+
+    // Connect to the generated user address
+    const wallet = MainModule__factory.connect(walletAddress, relayerEOA)
+
+    // Transfer funds to the SCW
+    const transferTx = await relayerEOA.sendTransaction({ to: walletAddress, value: 1 })
+    await transferTx.wait()
+
+    // Change the signer
+    expect(await immutableSigner.connect(adminEOA).updateSigner(await randomEOA.getAddress()))
+      .to.emit(immutableSigner, 'SignerUpdated')
+      .withArgs(await immutableEOA.getAddress(), await randomEOA.getAddress())
+
+    // Return funds
+    const transaction = {
+      delegateCall: false,
+      revertOnError: true,
+      gasLimit: 1000000,
+      target: await relayerEOA.getAddress(),
+      value: 1,
+      data: []
+    }
+
+    // Build meta-transaction
+    const networkId = (await hardhat.provider.getNetwork()).chainId
+    const nonce = 0
+    const data = encodeMetaTransactionsData(wallet.address, [transaction], networkId, nonce)
+
+    const signature = await walletMultiSign(
+      [
+        { weight: 1, owner: userEOA as ethers.Wallet },
+        // 03 -> Call the address' isValidSignature()
+        {
+          weight: 1,
+          owner: immutableSigner.address,
+          signature: (await ethSign(immutableEOA as ethers.Wallet, data)) + '03'
+        }
+      ],
+      2,
+      data,
+      false
+    )
+
+    await expect(wallet.execute([transaction], nonce, signature)).to.be.revertedWith(
+      'ModuleAuth#_signatureValidation: INVALID_SIGNATURE'
+    )
+  })
+})
diff --git a/forks/passport/tests/ImmutableFactory.spec.ts b/forks/passport/tests/ImmutableFactory.spec.ts
new file mode 100644
index 00000000..40a65ae3
--- /dev/null
+++ b/forks/passport/tests/ImmutableFactory.spec.ts
@@ -0,0 +1,353 @@
+import { loadFixture } from '@nomicfoundation/hardhat-network-helpers'
+import { expect } from 'chai'
+import { ethers } from 'hardhat'
+import { addressOf, encodeImageHash, multiSignAndExecuteMetaTx, signAndExecuteMetaTx } from './utils/helpers'
+import { CustomModule__factory, MainModule } from '../src/gen/typechain'
+
+describe('Wallet Factory', function () {
+  let salts: string[] = []
+
+  async function setupFactoryFixture() {
+    // Contracts are deployed using the first signer/account by default
+    const [owner, acc1] = await ethers.getSigners()
+
+    const WalletFactory = await ethers.getContractFactory('Factory')
+    const factory = await WalletFactory.deploy(owner.address, await owner.getAddress())
+
+    const MainModule = await ethers.getContractFactory('MainModuleMock')
+    const mainModule = await MainModule.deploy(factory.address)
+
+    const CustomModule = await ethers.getContractFactory('CustomModule')
+    const customModule = await CustomModule.deploy()
+
+    salts = [
+      encodeImageHash(1, [{ weight: 1, address: owner.address }]),
+      encodeImageHash(1, [{ weight: 1, address: acc1.address }]),
+      encodeImageHash(2, [
+        { weight: 1, address: owner.address },
+        { weight: 1, address: acc1.address }
+      ]),
+      encodeImageHash(3, [
+        { weight: 2, address: owner.address },
+        { weight: 1, address: acc1.address }
+      ])
+    ]
+
+    return {
+      owner,
+      acc1,
+      factory,
+      mainModule,
+      customModule
+    }
+  }
+
+  describe('getAddress', function () {
+    it('Should return deterministic contract address', async function () {
+      const { factory, mainModule } = await loadFixture(setupFactoryFixture)
+      for (const salt of salts) {
+        expect(await factory.getAddress(mainModule.address, salt)).to.equal(addressOf(factory.address, mainModule.address, salt))
+      }
+    })
+
+    it('Should depend on all parameters and generate no collisions', async function () {
+      const { factory, mainModule, customModule } = await loadFixture(setupFactoryFixture)
+
+      // Call getAddress() with different addresses and different salts
+      const addresses = await Promise.all(
+        [mainModule.address, customModule.address].flatMap(address => salts.map(async salt => factory.getAddress(address, salt)))
+      )
+
+      const addressSet = new Set(addresses)
+
+      // Results should have no duplicates
+      expect(addressSet.size).to.equal(addresses.length)
+    })
+  })
+
+  describe('deploy', function () {
+    it('Should not be able to interact with an undeployed contract', async function () {
+      const { owner, factory, mainModule } = await loadFixture(setupFactoryFixture)
+      // revert on attempt to call nonce() on undeployed contract
+      await expect(
+        (
+          await ethers.getContractAt(
+            'MainModuleMock',
+            addressOf(factory.address, mainModule.address, encodeImageHash(1, [{ weight: 1, address: owner.address }]))
+          )
+        ).nonce()
+      ).to.be.revertedWithoutReason()
+    })
+
+    it('Should deploy wallets to the expected address', async function () {
+      const { factory, mainModule } = await loadFixture(setupFactoryFixture)
+
+      for (const salt of salts) {
+        expect(await factory.deploy(mainModule.address, salt))
+          .to.emit(factory, 'WalletDeployed')
+          .withArgs(addressOf(factory.address, mainModule.address, salt), mainModule.address, salt)
+
+        const deployedContract = await ethers.getContractAt(
+          'MainModuleMock',
+          addressOf(factory.address, mainModule.address, salt)
+        )
+        // initial wallet nonce = 0
+        expect(await deployedContract.nonce()).to.equal(0)
+      }
+    })
+
+    it('Should be able to deploy with a custom module and retrieve the address of the custom module as an implementation', async function () {
+      const { owner, factory, customModule } = await loadFixture(setupFactoryFixture)
+
+      const salt = encodeImageHash(1, [{ weight: 1, address: owner.address }])
+      expect(await factory.deploy(customModule.address, salt))
+        .to.emit(factory, 'WalletDeployed')
+        .withArgs(addressOf(factory.address, customModule.address, salt), customModule.address, salt)
+      const contract = CustomModule__factory.connect(addressOf(factory.address, customModule.address, salt), owner)
+
+      // check functionality
+      await contract.setStr('new_string')
+      expect(await contract.getStr()).to.equal('new_string')
+
+      // Retrieve implementation from Proxy's storage
+      const proxy = await ethers.getContractAt('IWalletProxy', await addressOf(factory.address, customModule.address, salt))
+      expect(await proxy.PROXY_getImplementation()).to.be.equal(customModule.address)
+    })
+
+    it('Should not be able to deploy multiple wallets with the same salt', async function () {
+      const { acc1, factory, mainModule } = await loadFixture(setupFactoryFixture)
+      const salt = encodeImageHash(1, [{ weight: 1, address: acc1.address }])
+
+      expect(await factory.deploy(mainModule.address, salt))
+        .to.emit(factory, 'WalletDeployed')
+        .withArgs(addressOf(factory.address, mainModule.address, salt), mainModule.address, salt)
+
+      await expect(factory.deploy(mainModule.address, salt)).to.be.revertedWith('WalletFactory: deployment failed')
+    })
+
+    it('Should generate contracts with independent storage (i.e.: delegatecall)', async function () {
+      const { owner, acc1, factory, customModule } = await loadFixture(setupFactoryFixture)
+
+      const salts = [
+        encodeImageHash(1, [{ weight: 1, address: owner.address }]),
+        encodeImageHash(1, [{ weight: 1, address: acc1.address }])
+      ]
+
+      // Deploy two contracts pointing to the same implementation
+      await Promise.all(salts.map(async salt => factory.deploy(customModule.address, salt)))
+
+      const contracts = salts.map(salt =>
+        CustomModule__factory.connect(addressOf(factory.address, customModule.address, salt), owner)
+      )
+
+      // Set the string stored in each contract to their address
+      await Promise.all(contracts.map(async contract => contract.setStr(contract.address)))
+
+      for (const contract of contracts) {
+        expect(await contract.getStr()).to.equal(contract.address)
+      }
+    })
+  })
+
+  describe('access control', function () {
+    it('Should not be able to deploy if not owner', async function () {
+      const { acc1, factory, mainModule } = await loadFixture(setupFactoryFixture)
+      await expect(
+        factory.connect(acc1).deploy(mainModule.address, encodeImageHash(1, [{ weight: 1, address: acc1.address }]))
+      ).to.be.revertedWith(
+        'AccessControl: account 0x70997970c51812dc3a010c7d01b50e0d17dc79c8 is missing role 0xfc425f2263d0df187444b70e47283d622c70181c5baebb1306a01edba1ce184c'
+      )
+    })
+
+    it('Should not change generated addresses when deploying from a different account', async function () {
+      const { owner, acc1, factory, mainModule } = await loadFixture(setupFactoryFixture)
+      const salt = encodeImageHash(1, [{ weight: 1, address: acc1.address }])
+      await factory.connect(owner).grantRole(await factory.DEPLOYER_ROLE(), acc1.address)
+      expect(await factory.connect(acc1).deploy(mainModule.address, salt))
+        .to.emit(factory, 'WalletDeployed')
+        .withArgs(addressOf(factory.address, mainModule.address, salt), mainModule.address, salt)
+    })
+  })
+
+  describe('wallet', function () {
+    const initNonce = ethers.constants.Zero
+    const transaction = {
+      delegateCall: false,
+      revertOnError: true,
+      gasLimit: ethers.constants.Two.pow(21),
+      target: ethers.constants.AddressZero,
+      value: ethers.constants.Zero,
+      data: []
+    }
+
+    it('Should be able to send a transaction from a deployed wallet', async function () {
+      const { factory, mainModule } = await loadFixture(setupFactoryFixture)
+
+      const acc = ethers.Wallet.createRandom()
+      const salt = encodeImageHash(1, [{ weight: 1, address: acc.address }])
+      await factory.deploy(mainModule.address, salt)
+      const wallet = (await ethers.getContractAt(
+        'MainModuleMock',
+        addressOf(factory.address, mainModule.address, salt)
+      )) as MainModule
+
+      const networkId = (await ethers.provider.getNetwork()).chainId
+      // check wallet nonce before and after transaction
+      expect((await wallet.nonce()).toNumber()).to.equal(0)
+      await signAndExecuteMetaTx(wallet, acc, [transaction], networkId, initNonce)
+      expect((await wallet.nonce()).toNumber()).to.equal(1)
+    })
+
+    it('Should not be able to send a transaction with invalid signer', async function () {
+      const { factory, mainModule } = await loadFixture(setupFactoryFixture)
+
+      const acc = ethers.Wallet.createRandom()
+      const invalidSigner = ethers.Wallet.createRandom()
+      const salt = encodeImageHash(1, [{ weight: 1, address: acc.address }])
+      await factory.deploy(mainModule.address, salt)
+      const wallet = (await ethers.getContractAt(
+        'MainModuleMock',
+        addressOf(factory.address, mainModule.address, salt)
+      )) as MainModule
+
+      const networkId = (await ethers.provider.getNetwork()).chainId
+      await expect(signAndExecuteMetaTx(wallet, invalidSigner, [transaction], networkId, initNonce)).to.be.revertedWith(
+        'ModuleCalls#execute: INVALID_SIGNATURE'
+      )
+    })
+
+    it('Should be able to send a 2 of 2 multisig transaction', async function () {
+      const { factory, mainModule } = await loadFixture(setupFactoryFixture)
+
+      const signer1 = ethers.Wallet.createRandom()
+      const signer2 = ethers.Wallet.createRandom()
+      const salt = encodeImageHash(2, [
+        { weight: 1, address: signer1.address },
+        { weight: 1, address: signer2.address }
+      ])
+      await factory.deploy(mainModule.address, salt)
+      const wallet = (await ethers.getContractAt(
+        'MainModuleMock',
+        addressOf(factory.address, mainModule.address, salt)
+      )) as MainModule
+
+      const networkId = (await ethers.provider.getNetwork()).chainId
+      // check wallet nonce before and after transaction
+      expect((await wallet.nonce()).toNumber()).to.equal(0)
+      await multiSignAndExecuteMetaTx(
+        wallet,
+        [
+          {
+            weight: 1,
+            owner: signer1
+          },
+          {
+            weight: 1,
+            owner: signer2
+          }
+        ],
+        2,
+        [transaction],
+        networkId,
+        initNonce
+      )
+      expect((await wallet.nonce()).toNumber()).to.equal(1)
+    })
+
+    it('Should not be able to send a 2 of 2 multisig transaction with one invalid signer', async function () {
+      const { factory, mainModule } = await loadFixture(setupFactoryFixture)
+
+      const signer1 = ethers.Wallet.createRandom()
+      const signer2 = ethers.Wallet.createRandom()
+      const salt = encodeImageHash(2, [
+        { weight: 1, address: signer1.address },
+        { weight: 1, address: signer2.address }
+      ])
+      await factory.deploy(mainModule.address, salt)
+      const wallet = (await ethers.getContractAt(
+        'MainModuleMock',
+        addressOf(factory.address, mainModule.address, salt)
+      )) as MainModule
+
+      const networkId = (await ethers.provider.getNetwork()).chainId
+      await expect(
+        multiSignAndExecuteMetaTx(
+          wallet,
+          [
+            {
+              weight: 1,
+              owner: signer1
+            },
+            {
+              weight: 1,
+              owner: signer1 // invalid signer
+            }
+          ],
+          2,
+          [transaction],
+          networkId,
+          initNonce
+        )
+      ).to.be.revertedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+    })
+
+    it('Should be able to upgrade a 2 of 2 multisig wallet to a custom module', async function () {
+      const { factory, mainModule, customModule } = await loadFixture(setupFactoryFixture)
+
+      const signer1 = ethers.Wallet.createRandom()
+      const signer2 = ethers.Wallet.createRandom()
+      const salt = encodeImageHash(2, [
+        { weight: 1, address: signer1.address },
+        { weight: 1, address: signer2.address }
+      ])
+      await factory.deploy(mainModule.address, salt)
+      const originalWallet = (await ethers.getContractAt(
+        'MainModuleMock',
+        addressOf(factory.address, mainModule.address, salt)
+      )) as MainModule
+
+      const networkId = (await ethers.provider.getNetwork()).chainId
+    
+      // Retrieve implementation from Proxy's storage
+      const proxy = await ethers.getContractAt('IWalletProxy', await addressOf(factory.address, mainModule.address, salt))
+      expect(await proxy.PROXY_getImplementation()).to.be.equal(mainModule.address)
+ 
+      // upgrade implementation tx
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: ethers.constants.Two.pow(21),
+        target: originalWallet.address,
+        value: ethers.constants.Zero,
+        data: originalWallet.interface.encodeFunctionData('updateImplementation', [customModule.address])
+      }
+      await multiSignAndExecuteMetaTx(
+        originalWallet,
+        [
+          {
+            weight: 1,
+            owner: signer1
+          },
+          {
+            weight: 1,
+            owner: signer2
+          }
+        ],
+        2,
+        [transaction],
+        networkId,
+        initNonce
+      )
+      // check functionality
+      const updatedWallet = await ethers.getContractAt(
+        'CustomModule',
+        addressOf(factory.address, mainModule.address, salt) // still generate using the original module address
+      )
+      expect(updatedWallet.address).to.equal(originalWallet.address) // wallet address remains unchanged
+      expect(await proxy.PROXY_getImplementation()).to.be.equal(customModule.address) // proxy now set to custom module address
+      await updatedWallet.setStr('new_string')
+      expect(await updatedWallet.getStr()).to.equal('new_string')
+      await expect(originalWallet.nonce()).to.be.revertedWithoutReason() // original wallet functions no longer exposed
+    })
+  })
+})
diff --git a/forks/passport/tests/ImmutableMultiCallDeploy.spec.ts b/forks/passport/tests/ImmutableMultiCallDeploy.spec.ts
new file mode 100644
index 00000000..619e37b1
--- /dev/null
+++ b/forks/passport/tests/ImmutableMultiCallDeploy.spec.ts
@@ -0,0 +1,234 @@
+import { loadFixture } from '@nomicfoundation/hardhat-network-helpers'
+import { expect } from 'chai'
+import { ethers } from 'hardhat'
+import { addressOf, encodeImageHash, encodeMetaTransactionsData, walletMultiSign } from './utils/helpers'
+
+describe('Wallet Factory', function () {
+  async function setupFactoryFixture() {
+    // Network ID
+    const networkId = (await ethers.provider.getNetwork()).chainId
+
+    // Wallet TX
+    const optimalGasLimit = ethers.constants.Two.pow(21)
+
+    // Contracts are deployed using the first signer/account by default
+    const [owner, executor, acc1] = await ethers.getSigners()
+
+    const WalletFactory = await ethers.getContractFactory('Factory')
+    const factory = await WalletFactory.deploy(owner.address, await owner.getAddress())
+
+    const MainModule = await ethers.getContractFactory('MainModuleMock')
+    const mainModule = await MainModule.deploy(factory.address)
+
+    const MultiCall = await ethers.getContractFactory('MultiCallDeploy')
+    const multiCall = await MultiCall.deploy(owner.address, executor.address)
+
+    const deployerRole = await factory.DEPLOYER_ROLE()
+    await factory.connect(owner).grantRole(deployerRole, multiCall.address)
+
+    const Token = await ethers.getContractFactory('ERC20Mock')
+
+    const token = await Token.connect(owner).deploy()
+
+    return {
+      owner,
+      executor,
+      acc1,
+      factory,
+      mainModule,
+      multiCall,
+      networkId,
+      optimalGasLimit,
+      token
+    }
+  }
+
+  describe('deployAndExecute', function () {
+    it('Should deploy and execute transfers', async function () {
+      const owner_a = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt = encodeImageHash(1, [{ weight: 1, address: owner_a.address }])
+      const { factory, mainModule, multiCall, executor, acc1, owner, networkId, optimalGasLimit, token } = await loadFixture(
+        setupFactoryFixture
+      )
+
+      // CFA
+      const cfa = addressOf(factory.address, mainModule.address, salt)
+
+      // Transfer tokens to CFA
+      await token.connect(owner).transfer(cfa, ethers.utils.parseEther('5'))
+      expect(await token.balanceOf(cfa)).to.equal(ethers.utils.parseEther('5'))
+
+      // We don't want delegate call here as the state is contained to the ERC20 contract
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: token.address,
+        value: ethers.constants.Zero,
+        data: token.interface.encodeFunctionData('transfer', [acc1.address, ethers.utils.parseEther('2.5')])
+      }
+
+      // Signing
+      const data = encodeMetaTransactionsData(cfa, [transaction], networkId, ethers.constants.Zero)
+      const sig = walletMultiSign([{ weight: 1, owner: owner_a }], 1, data, false)
+
+      // Execution
+      await multiCall.connect(executor).deployAndExecute(cfa, mainModule.address, salt, factory.address, [transaction], 0, sig)
+      expect(await token.balanceOf(cfa)).to.equal(ethers.utils.parseEther('2.5'))
+
+      // Transfer remaining, resign Tx with incremented nonce
+      // Here the deployment will be skipped and the transaction will be executed
+      const dataTwo = encodeMetaTransactionsData(cfa, [transaction], networkId, 1)
+      const sigTwo = walletMultiSign([{ weight: 1, owner: owner_a }], 1, dataTwo, false)
+      await multiCall.connect(executor).deployAndExecute(cfa, mainModule.address, salt, factory.address, [transaction], 1, sigTwo)
+      expect(await token.balanceOf(cfa)).to.equal(0)
+    })
+
+    it('Should fail with a CFA mismatch', async function () {
+      const owner_b = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt = encodeImageHash(1, [{ weight: 1, address: owner_b.address }])
+      const { factory, mainModule, multiCall, acc1, executor, owner, networkId, optimalGasLimit } = await loadFixture(setupFactoryFixture)
+
+      // CFA
+      const cfa = addressOf(factory.address, mainModule.address, salt)
+
+      // Mint tokens
+      const Token = await ethers.getContractFactory('ERC20Mock')
+      const token = await Token.connect(owner).deploy()
+
+      // Transfer tokens to CFA
+      await token.connect(owner).transfer(cfa, ethers.utils.parseEther('5'))
+      expect(await token.balanceOf(cfa)).to.equal(ethers.utils.parseEther('5'))
+
+      // We don't want delegate call here as the state is contained to the ERC20 contract
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: token.address,
+        value: ethers.constants.Zero,
+        data: token.interface.encodeFunctionData('transfer', [acc1.address, ethers.utils.parseEther('2.5')])
+      }
+
+      // Signing
+      const data = encodeMetaTransactionsData(cfa, [transaction], networkId, ethers.constants.Zero)
+      const sig = walletMultiSign([{ weight: 1, owner: owner_b }], 1, data, false)
+
+      // Execution
+      await expect(
+        multiCall
+          .connect(executor)
+          .deployAndExecute(owner_b.address, mainModule.address, salt, factory.address, [transaction], 0, sig)
+      ).to.be.revertedWith('MultiCallDeploy: deployed address does not match CFA')
+    })
+  })
+
+  describe('deployExecute', function () {
+    it('Should deploy and and execute transfer', async function () {
+      const owner_c = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt = encodeImageHash(1, [{ weight: 1, address: owner_c.address }])
+      const { factory, mainModule, multiCall, acc1, executor, owner, networkId, optimalGasLimit, token } = await loadFixture(
+        setupFactoryFixture
+      )
+
+      // CFA
+      const cfa = addressOf(factory.address, mainModule.address, salt)
+
+      // Transfer tokens to CFA
+      await token.connect(owner).transfer(cfa, ethers.utils.parseEther('5'))
+      expect(await token.balanceOf(cfa)).to.equal(ethers.utils.parseEther('5'))
+
+      // We don't want delegate call here as the state is contained to the ERC20 contract
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: token.address,
+        value: ethers.constants.Zero,
+        data: token.interface.encodeFunctionData('transfer', [acc1.address, ethers.utils.parseEther('2.5')])
+      }
+
+      // Signing
+      const data = encodeMetaTransactionsData(cfa, [transaction], networkId, ethers.constants.Zero)
+      const sig = walletMultiSign([{ weight: 1, owner: owner_c }], 1, data, false)
+
+      // Execution
+      await multiCall.connect(executor).deployExecute(mainModule.address, salt, factory.address, [transaction], 0, sig)
+      expect(await token.balanceOf(cfa)).to.equal(ethers.utils.parseEther('2.5'))
+    })
+
+    it('Should fail if the wallet is already deployed', async function () {
+      const owner_d = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt = encodeImageHash(1, [{ weight: 1, address: owner_d.address }])
+      const { factory, mainModule, multiCall, acc1, executor, owner, networkId, optimalGasLimit, token } = await loadFixture(
+        setupFactoryFixture
+      )
+
+      // CFA
+      const cfa = addressOf(factory.address, mainModule.address, salt)
+
+      // Transfer tokens to CFA
+      await token.connect(owner).transfer(cfa, ethers.utils.parseEther('5'))
+      expect(await token.balanceOf(cfa)).to.equal(ethers.utils.parseEther('5'))
+
+      // We don't want delegate call here as the state is contained to the ERC20 contract
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: token.address,
+        value: ethers.constants.Zero,
+        data: token.interface.encodeFunctionData('transfer', [acc1.address, ethers.utils.parseEther('2.5')])
+      }
+
+      // Signing
+      const data = encodeMetaTransactionsData(cfa, [transaction], networkId, ethers.constants.Zero)
+      const sig = walletMultiSign([{ weight: 1, owner: owner_d }], 1, data, false)
+
+      // Execution
+      await multiCall.connect(executor).deployExecute(mainModule.address, salt, factory.address, [transaction], 0, sig)
+
+      // Repeat above process, re-triggering deployment
+      const dataTwo = encodeMetaTransactionsData(cfa, [transaction], networkId, 1)
+      const sigTwo = walletMultiSign([{ weight: 1, owner: owner_d }], 1, dataTwo, false)
+
+      // Attempt to re-deploy
+      await expect(
+        multiCall.connect(executor).deployExecute(mainModule.address, salt, factory.address, [transaction], 0, sigTwo)
+      ).to.be.revertedWith('WalletFactory: deployment failed')
+    })
+
+    it('Should fail if the submitter does not have the executor role', async function () {
+      const owner_d = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt = encodeImageHash(1, [{ weight: 1, address: owner_d.address }])
+      const { factory, mainModule, multiCall, acc1, executor, owner, networkId, optimalGasLimit, token } = await loadFixture(
+        setupFactoryFixture
+      )
+
+      // CFA
+      const cfa = addressOf(factory.address, mainModule.address, salt)
+
+      // Transfer tokens to CFA
+      await token.connect(owner).transfer(cfa, ethers.utils.parseEther('5'))
+      expect(await token.balanceOf(cfa)).to.equal(ethers.utils.parseEther('5'))
+
+      // We don't want delegate call here as the state is contained to the ERC20 contract
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: token.address,
+        value: ethers.constants.Zero,
+        data: token.interface.encodeFunctionData('transfer', [acc1.address, ethers.utils.parseEther('2.5')])
+      }
+
+      // Signing
+      const data = encodeMetaTransactionsData(cfa, [transaction], networkId, ethers.constants.Zero)
+      const sig = walletMultiSign([{ weight: 1, owner: owner_d }], 1, data, false)
+
+      // Execution
+      await expect(multiCall.connect(acc1).deployExecute(mainModule.address, salt, factory.address, [transaction], 0, sig)).to.be.revertedWith("AccessControl: account 0x3c44cdddb6a900fa2b585dd299e03d12fa4293bc is missing role 0xd8aa0f3194971a2a116679f7c2090f6939c8d4e01a2a8d7e41d55e5351469e63")
+      await expect(multiCall.connect(acc1).deployAndExecute(cfa, mainModule.address, salt, factory.address, [transaction], 0, sig)).to.be.revertedWith("AccessControl: account 0x3c44cdddb6a900fa2b585dd299e03d12fa4293bc is missing role 0xd8aa0f3194971a2a116679f7c2090f6939c8d4e01a2a8d7e41d55e5351469e63")
+    })
+  })
+})
\ No newline at end of file
diff --git a/forks/passport/tests/ImmutableSigner.spec.ts b/forks/passport/tests/ImmutableSigner.spec.ts
new file mode 100644
index 00000000..369c5f6b
--- /dev/null
+++ b/forks/passport/tests/ImmutableSigner.spec.ts
@@ -0,0 +1,183 @@
+import { ethers as hardhat, expect } from 'hardhat'
+import { ethers } from 'ethers'
+import { keccak256, randomBytes, toUtf8Bytes } from 'ethers/lib/utils'
+import { ImmutableSigner, ImmutableSigner__factory } from '../src'
+import { ethSign } from './utils'
+
+describe('ImmutableSigner', () => {
+  let deployerEOA: ethers.Signer
+  let rootAdminEOA: ethers.Signer
+  let signerAdminEOA: ethers.Signer
+  let signerEOA: ethers.Signer
+  let randomEOA: ethers.Signer
+
+  let immutableSigner: ImmutableSigner
+
+  const ERC1271_OK = '0x1626ba7e'
+  const ERC1271_INVALID = '0x00000000'
+
+  beforeEach(async () => {
+    ;[deployerEOA, rootAdminEOA, signerAdminEOA, signerEOA, randomEOA] = await hardhat.getSigners()
+
+    immutableSigner = await new ImmutableSigner__factory()
+      .connect(deployerEOA)
+      .deploy(await rootAdminEOA.getAddress(), await signerAdminEOA.getAddress(), await signerEOA.getAddress())
+  })
+
+  describe('isValidSignature', () => {
+    it('Should return ERC1271_MAGICVALUE_BYTES32 for a hash signed by the primary signer', async () => {
+      const data = '0x00'
+      const hash = keccak256(data)
+      const signature = ethSign(signerEOA as ethers.Wallet, hash, true)
+
+      expect(await immutableSigner.isValidSignature(hash, signature)).to.equal(ERC1271_OK)
+    })
+
+    it('Should return 0 for a hash signed by a random signer', async () => {
+      const data = '0x00'
+      const hash = keccak256(data)
+      const signature = ethSign(randomEOA as ethers.Wallet, hash, true)
+
+      expect(await immutableSigner.isValidSignature(hash, signature)).to.equal(ERC1271_INVALID)
+    })
+
+    it('Should return 0 for a random signature', async () => {
+      const data = '0x00'
+      const hash = keccak256(data)
+
+      // Random input hence could fail with a couple different messages
+      await expect(immutableSigner.isValidSignature(hash, randomBytes(66))).to.be.reverted
+    })
+
+    it('Should return 0 for a random 16 byte signature', async () => {
+      const data = '0x00'
+      const hash = keccak256(data)
+
+      await expect(immutableSigner.isValidSignature(hash, randomBytes(16))).to.be.revertedWith(
+        'SignatureValidator#recoverSigner: invalid signature length'
+      )
+    })
+
+    it('Should return 0 for a 0 byte signature', async () => {
+      const data = '0x00'
+      const hash = keccak256(data)
+
+      await expect(immutableSigner.isValidSignature(hash, [])).to.be.revertedWith(
+        'SignatureValidator#recoverSigner: invalid signature length'
+      )
+    })
+  })
+
+  describe('updateSigner', () => {
+    it('Should immediately update the primary signer', async () => {
+      const data = '0x00'
+      const hash = keccak256(data)
+
+      const randomEOASignature = ethSign(randomEOA as ethers.Wallet, hash, true)
+
+      await immutableSigner.connect(signerAdminEOA).updateSigner(await randomEOA.getAddress())
+
+      expect(await immutableSigner.isValidSignature(hash, randomEOASignature)).to.equal(ERC1271_OK)
+    })
+
+    it('Should immediately expire the previous primary signer', async () => {
+      const data = '0x00'
+      const hash = keccak256(data)
+
+      const signerEOASignature = ethSign(signerEOA as ethers.Wallet, hash, true)
+
+      await immutableSigner.connect(signerAdminEOA).updateSigner(await randomEOA.getAddress())
+
+      expect(await immutableSigner.isValidSignature(hash, signerEOASignature)).to.equal(ERC1271_INVALID)
+    })
+
+    it('Should immediately expire the previous rollover signer', async () => {
+      // Set a rollover signer and make sure it's valid
+      await immutableSigner.connect(signerAdminEOA).updateSignerWithRolloverPeriod(await randomEOA.getAddress(), 2)
+
+      const data = '0x00'
+      const hash = keccak256(data)
+      const signerEOASignature = ethSign(signerEOA as ethers.Wallet, hash, true)
+      expect(await immutableSigner.isValidSignature(hash, signerEOASignature)).to.equal(ERC1271_OK)
+
+      // Still within the valid time range
+      await hardhat.provider.send('evm_increaseTime', [2])
+      await hardhat.provider.send('evm_mine', [])
+      await immutableSigner.connect(signerAdminEOA).updateSigner(await randomEOA.getAddress())
+      expect(await immutableSigner.isValidSignature(hash, signerEOASignature)).to.equal(ERC1271_INVALID)
+    })
+
+    it('Should not allow changes from EOAs without SIGNER_ADMIN_ROLE', async () => {
+      const randomAddress = (await randomEOA.getAddress()).toLowerCase()
+
+      await expect(
+        immutableSigner.connect(randomEOA).updateSigner(await randomEOA.getAddress(), { gasLimit: 300_000 })
+      ).to.be.revertedWith(
+        `AccessControl: account ${randomAddress} is missing role ${keccak256(toUtf8Bytes('SIGNER_ADMIN_ROLE'))}`
+      )
+    })
+  })
+
+  describe('updateSignerWithRolloverPeriod', () => {
+    it('Should allow both the new signer and the rollover signer to be used', async () => {
+      await immutableSigner.connect(signerAdminEOA).updateSignerWithRolloverPeriod(await randomEOA.getAddress(), 2)
+
+      const data = '0x00'
+      const hash = keccak256(data)
+      const randomEOASignature = ethSign(randomEOA as ethers.Wallet, hash, true)
+      const signerEOASignature = ethSign(signerEOA as ethers.Wallet, hash, true)
+
+      // Within the valid time range
+      await hardhat.provider.send('evm_increaseTime', [2])
+      await hardhat.provider.send('evm_mine', [])
+      expect(await immutableSigner.isValidSignature(hash, randomEOASignature)).to.equal(ERC1271_OK)
+      expect(await immutableSigner.isValidSignature(hash, signerEOASignature)).to.equal(ERC1271_OK)
+    })
+
+    it('Should expire the rollover signer after the specified timestamp', async () => {
+      await immutableSigner.connect(signerAdminEOA).updateSignerWithRolloverPeriod(await randomEOA.getAddress(), 1)
+
+      const data = '0x00'
+      const hash = keccak256(data)
+      const signerEOASignature = ethSign(signerEOA as ethers.Wallet, hash, true)
+
+      // After the valid time range
+      await hardhat.provider.send('evm_increaseTime', [2])
+      await hardhat.provider.send('evm_mine', [])
+      expect(await immutableSigner.isValidSignature(hash, signerEOASignature)).to.equal(ERC1271_INVALID)
+    })
+
+    it('Should allow the primary signer after the specified rollover timestamp', async () => {
+      await immutableSigner.connect(signerAdminEOA).updateSignerWithRolloverPeriod(await randomEOA.getAddress(), 1)
+
+      const data = '0x00'
+      const hash = keccak256(data)
+      const randomEOASignature = ethSign(randomEOA as ethers.Wallet, hash, true)
+
+      // After the valid time range
+      await hardhat.provider.send('evm_increaseTime', [2])
+      await hardhat.provider.send('evm_mine', [])
+      expect(await immutableSigner.isValidSignature(hash, randomEOASignature)).to.equal(ERC1271_OK)
+    })
+
+    it('Should not allow signers other than the primary and rollover to be used during rollover', async () => {
+      await immutableSigner.connect(signerAdminEOA).updateSignerWithRolloverPeriod(await randomEOA.getAddress(), 2)
+
+      const data = '0x00'
+      const hash = keccak256(data)
+
+      const signatures = [
+        ethSign(deployerEOA as ethers.Wallet, hash, true),
+        ethSign(rootAdminEOA as ethers.Wallet, hash, true),
+        ethSign(signerAdminEOA as ethers.Wallet, hash, true)
+      ]
+
+      // Within rollover
+      await hardhat.provider.send('evm_increaseTime', [2])
+      await hardhat.provider.send('evm_mine', [])
+      for (const signature of signatures) {
+        expect(await immutableSigner.isValidSignature(hash, signature)).to.equal(ERC1271_INVALID)
+      }
+    })
+  })
+})
diff --git a/forks/passport/tests/ImmutableStartup.spec.ts b/forks/passport/tests/ImmutableStartup.spec.ts
new file mode 100644
index 00000000..54a1b927
--- /dev/null
+++ b/forks/passport/tests/ImmutableStartup.spec.ts
@@ -0,0 +1,359 @@
+// Copyright Immutable Pty Ltd 2018 - 2023
+// SPDX-License-Identifier: Apache-2.0
+import { loadFixture } from '@nomicfoundation/hardhat-network-helpers'
+import { expect } from 'chai'
+import { ethers } from 'hardhat'
+import { addressOf, encodeImageHash, walletSign, signAndExecuteMetaTx } from './utils/helpers'
+import { CustomModule__factory, MainModule } from '../src/gen/typechain'
+import { startup } from 'src/gen/adapter'
+
+describe('Wallet Factory', function () {
+//  let salts: string[] = []
+
+  async function setupStartupFixture() {
+    // Contracts are deployed using the first signer/account by default
+    const [owner, acc1] = await ethers.getSigners()
+
+    const WalletFactory = await ethers.getContractFactory('Factory')
+    const factory = await WalletFactory.deploy(owner.address, owner.address)
+
+    const LatestWalletImplLocator = await ethers.getContractFactory('LatestWalletImplLocator')
+    const latestWalletImplLocator = await LatestWalletImplLocator.deploy(owner.address, owner.address)
+
+    const StartupWalletImpl= await ethers.getContractFactory('StartupWalletImpl')
+    const startupWalletImpl = await StartupWalletImpl.deploy(latestWalletImplLocator.address)
+
+    const MainModule = await ethers.getContractFactory('MainModuleMockV1')
+    const mainModuleV1 = await MainModule.deploy(factory.address, startupWalletImpl.address)
+
+    await latestWalletImplLocator.changeWalletImplementation(mainModuleV1.address)
+
+
+
+
+    const salt = encodeImageHash(1, [{ weight: 1, address: owner.address }])
+
+
+    return {
+      owner,
+      acc1,
+      factory,
+      mainModuleV1,
+      startupWalletImpl,
+      salt,
+      latestWalletImplLocator
+
+    }
+  }
+
+  describe('Startup', function () {
+
+    it('Should deploy wallet proxy to the expected address', async function () {
+      const { factory, startupWalletImpl, salt } = await loadFixture(setupStartupFixture)
+
+      expect(await factory.deploy(startupWalletImpl.address, salt))
+          .to.emit(factory, 'WalletDeployed')
+          .withArgs(addressOf(factory.address, startupWalletImpl.address, salt), startupWalletImpl.address, salt)
+
+      const deployedContract = await ethers.getContractAt(
+          'MainModuleMockV1',
+          addressOf(factory.address, startupWalletImpl.address, salt)
+      )
+
+      // Attempt a function call to check that the contract has been deployed.
+      // This will proxy through to the uninitialised MainModule
+      // initial wallet nonce = 0
+      expect(await deployedContract.nonce()).to.equal(0)
+
+      // Check the wallet implementation version
+      expect(await deployedContract.version()).to.equal(1)
+    })
+
+    it('Get implementation before first transaction should indicate startup', async function () {
+      const { factory, startupWalletImpl, salt } = await loadFixture(setupStartupFixture)
+
+      await factory.deploy(startupWalletImpl.address, salt)
+      const deployedContract = await ethers.getContractAt(
+          'IWalletProxy',
+          addressOf(factory.address, startupWalletImpl.address, salt)
+        )
+
+        expect(await deployedContract.PROXY_getImplementation()).to.equal(startupWalletImpl.address)
+    })
+
+
+    it('Get implementation after first transaction should indicate implementation', async function () {
+      const { factory, mainModuleV1, startupWalletImpl} = await loadFixture(setupStartupFixture)
+
+      const acc = ethers.Wallet.createRandom()
+      const salt = encodeImageHash(1, [{ weight: 1, address: acc.address }])
+
+      // Deploy the proxy
+      await factory.deploy(startupWalletImpl.address, salt)
+
+      const deployedAddress = addressOf(factory.address, startupWalletImpl.address, salt)
+
+      const wallet = (await ethers.getContractAt('MainModuleMockV1', deployedAddress)) as MainModule
+      const networkId = (await ethers.provider.getNetwork()).chainId
+
+      const initNonce = ethers.constants.Zero
+
+      const valA = 7
+      const valB = '0x0d'
+      const CallReceiver = await ethers.getContractFactory('CallReceiverMock')
+      const callReceiver = await CallReceiver.deploy()
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: ethers.constants.Two.pow(21),
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+      }
+
+      // check wallet nonce before and after transaction
+      expect((await wallet.nonce()).toNumber()).to.equal(0)
+      await signAndExecuteMetaTx(wallet, acc, [transaction], networkId, initNonce)
+      expect((await wallet.nonce()).toNumber()).to.equal(1)
+
+      // Check that the values in the target contract were altered
+      expect(await callReceiver.lastValA()).to.equal(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+
+      // Check that the get implementation now points to the main module.
+      const deployedProxy = await ethers.getContractAt('IWalletProxy', deployedAddress)
+
+      // Check that get implementation returns the address of main
+      expect(await deployedProxy.PROXY_getImplementation()).to.equal(mainModuleV1.address)
+    })
+
+
+
+
+    it('Should be able to upgrade implementation contracts', async function () {
+      const { factory, startupWalletImpl } = await loadFixture(setupStartupFixture)
+
+      const acc = ethers.Wallet.createRandom()
+      const salt = encodeImageHash(1, [{ weight: 1, address: acc.address }])
+
+
+      //console.log("Deploy wallet proxy using MainModuleMockV1")
+      await factory.deploy(startupWalletImpl.address, salt)
+      const deployedAddress = addressOf(factory.address, startupWalletImpl.address, salt)
+
+      const wallet = await ethers.getContractAt('MainModuleMockV1', deployedAddress)
+      // Check the wallet implementation version
+      expect(await wallet.version()).to.equal(1)
+
+      const walletMainModule = await ethers.getContractAt('MainModuleMockV1', deployedAddress) as MainModule
+
+
+      //console.log("Deploy MainModuleMockV2")
+      const MainModuleV2 = await ethers.getContractFactory('MainModuleMockV2')
+      const mainModuleV2 = await MainModuleV2.deploy(factory.address, startupWalletImpl.address)
+
+      // console.log("Upgrade wallet proxy to using MainModuleMockV2")
+      const networkId = (await ethers.provider.getNetwork()).chainId
+      // upgrade implementation tx
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: ethers.constants.Two.pow(21),
+        target: wallet.address,
+        value: ethers.constants.Zero,
+        data: wallet.interface.encodeFunctionData('updateImplementation', [mainModuleV2.address])
+      }
+
+      let nonce = await wallet.nonce()
+      await signAndExecuteMetaTx(walletMainModule, acc, [transaction], networkId, nonce)
+
+      // Check that get implementation returns the address of main
+      const walletAsProxy = await ethers.getContractAt('IWalletProxy', deployedAddress)
+      expect(await walletAsProxy.PROXY_getImplementation()).to.equal(mainModuleV2.address)
+
+      // Check the wallet implementation version
+      expect(await wallet.version()).to.equal(2)
+
+
+      // Now check that a transaction can be executed
+      //console.log("Execute a function call to Call Receiver Mock")
+      const valA = 7
+      const valB = '0x0d'
+      const CallReceiver = await ethers.getContractFactory('CallReceiverMock')
+      const callReceiver = await CallReceiver.deploy()
+      nonce = await wallet.nonce()
+
+      const transaction2 = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: ethers.constants.Two.pow(21),
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+      }
+      await signAndExecuteMetaTx(walletMainModule, acc, [transaction2], networkId, nonce)
+
+      // Check that the values in the target contract were altered
+      expect(await callReceiver.lastValA()).to.equal(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+
+
+    it('Deploying using upgrade should work', async function () {
+      const { factory, startupWalletImpl, latestWalletImplLocator } = await loadFixture(setupStartupFixture)
+
+      const MainModuleV2 = await ethers.getContractFactory('MainModuleMockV2')
+      const mainModuleV2 = await MainModuleV2.deploy(factory.address, startupWalletImpl.address)
+      
+      await latestWalletImplLocator.changeWalletImplementation(mainModuleV2.address)
+
+
+      const acc = ethers.Wallet.createRandom()
+      const salt = encodeImageHash(1, [{ weight: 1, address: acc.address }])
+
+
+      //console.log("Deploy wallet proxy using MainModuleMockV2")
+      await factory.deploy(startupWalletImpl.address, salt)
+      const deployedAddress = addressOf(factory.address, startupWalletImpl.address, salt)
+
+      const wallet = await ethers.getContractAt('MainModuleMockV2', deployedAddress)
+      // Check the wallet implementation version
+      expect(await wallet.version()).to.equal(2)
+
+      // Check that get implementation returns the address of main
+      const walletAsProxy = await ethers.getContractAt('IWalletProxy', deployedAddress)
+      expect(await walletAsProxy.PROXY_getImplementation()).to.equal(startupWalletImpl.address)
+
+      // Now check that a transaction can be executed
+      //console.log("Execute a function call to Call Receiver Mock")
+      const walletMainModule = await ethers.getContractAt('MainModuleMockV1', deployedAddress) as MainModule
+      const valA = 7
+      const valB = '0x0d'
+      const CallReceiver = await ethers.getContractFactory('CallReceiverMock')
+      const callReceiver = await CallReceiver.deploy()
+      let nonce = await wallet.nonce()
+      const networkId = (await ethers.provider.getNetwork()).chainId
+
+      const transaction2 = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: ethers.constants.Two.pow(21),
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+      }
+      await signAndExecuteMetaTx(walletMainModule, acc, [transaction2], networkId, nonce)
+
+      // Check that the values in the target contract were altered
+      expect(await callReceiver.lastValA()).to.equal(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+
+      // Check that get implementation returns the address of main
+      expect(await walletAsProxy.PROXY_getImplementation()).to.equal(mainModuleV2.address)
+    })
+
+
+
+
+    it('Should be able to execute multiple meta transactions', async function () {
+      const { factory, startupWalletImpl } = await loadFixture(setupStartupFixture)
+
+      const acc = ethers.Wallet.createRandom()
+      const salt = encodeImageHash(1, [{ weight: 1, address: acc.address }])
+
+
+      //console.log("Deploy wallet proxy using MainModuleMockV1")
+      await factory.deploy(startupWalletImpl.address, salt)
+      const deployedAddress = addressOf(factory.address, startupWalletImpl.address, salt)
+
+      const walletMainModule = await ethers.getContractAt('MainModuleMockV1', deployedAddress) as MainModule
+
+      // Now check that a transaction can be executed
+      // This will be authenticated based on the deployed address
+      const valA = 7
+      const valB = '0x0d'
+      const CallReceiver = await ethers.getContractFactory('CallReceiverMock')
+      const callReceiver = await CallReceiver.deploy()
+
+      const networkId = (await ethers.provider.getNetwork()).chainId
+      let nonce = await walletMainModule.nonce()
+      let transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: ethers.constants.Two.pow(21),
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+      }
+      await signAndExecuteMetaTx(walletMainModule, acc, [transaction], networkId, nonce)
+
+      // Check that the values in the target contract were altered
+      expect(await callReceiver.lastValA()).to.equal(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+
+
+      // Now check that a second transaction can be executed
+      // This will be authenticated solely based on the image Hash
+      const valC = 6
+      const valD = '0x053456'
+
+      nonce = await walletMainModule.nonce()
+      transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: ethers.constants.Two.pow(21),
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valC, valD])
+      }
+      await signAndExecuteMetaTx(walletMainModule, acc, [transaction], networkId, nonce)
+
+      // Check that the values in the target contract were altered
+      expect(await callReceiver.lastValA()).to.equal(valC)
+      expect(await callReceiver.lastValB()).to.equal(valD)
+    })
+
+
+    it('Check isSignatureValid before the first transaction', async function () {
+      const { factory, startupWalletImpl } = await loadFixture(setupStartupFixture)
+
+      const acc = ethers.Wallet.createRandom()
+      const salt = encodeImageHash(1, [{ weight: 1, address: acc.address }])
+
+
+      //console.log("Deploy wallet proxy using MainModuleMockV1")
+      await factory.deploy(startupWalletImpl.address, salt)
+      const deployedAddress = addressOf(factory.address, startupWalletImpl.address, salt)
+
+      const wallet = await ethers.getContractAt('MainModuleMockV1', deployedAddress)
+      const walletMainModule = await ethers.getContractAt('MainModuleMockV1', deployedAddress) as MainModule
+
+      expect(await wallet.version()).to.equal(1)
+
+      const networkId = (await ethers.provider.getNetwork()).chainId
+
+      const data = await ethers.utils.randomBytes(95)
+      let messageSubDigest = ethers.utils.solidityPack(
+        ['string', 'uint256', 'address', 'bytes'],
+        ['\x19\x01', networkId, wallet.address, ethers.utils.keccak256(data)]
+      )
+
+      const hash = ethers.utils.keccak256(data)
+      const hashSubDigest = ethers.utils.solidityPack(
+        ['string', 'uint256', 'address', 'bytes'],
+        ['\x19\x01', networkId, wallet.address, ethers.utils.solidityPack(['bytes32'], [hash])]
+      )
+
+      let signature = await walletSign(acc, messageSubDigest)
+      let isValidSigFuncSelector = await wallet['isValidSignature(bytes,bytes)'](data, signature)
+      expect(parseInt(isValidSigFuncSelector, 16)).to.equal(0x20c13b0b)
+
+      signature = await walletSign(acc, hashSubDigest)
+      isValidSigFuncSelector = await wallet['isValidSignature(bytes32,bytes)'](hash, signature)
+      expect(parseInt(isValidSigFuncSelector, 16)).to.equal(0x1626ba7e)
+    })
+
+  })
+})
+
diff --git a/forks/passport/tests/LibBytes.spec.ts b/forks/passport/tests/LibBytes.spec.ts
new file mode 100644
index 00000000..31fa0237
--- /dev/null
+++ b/forks/passport/tests/LibBytes.spec.ts
@@ -0,0 +1,268 @@
+import { ethers } from 'ethers'
+import { expect } from './utils'
+
+import { LibBytesImpl } from 'src/gen/typechain'
+
+ethers.utils.Logger.setLogLevel(ethers.utils.Logger.levels.ERROR)
+
+const LibBytesImplArtifact = artifacts.require('LibBytesImpl')
+
+import { web3 } from 'hardhat'
+
+contract('LibBytes', (accounts: string[]) => {
+  let libBytes
+
+  before(async () => {
+    libBytes = (await LibBytesImplArtifact.new()) as LibBytesImpl
+  })
+
+  describe('readFirstUint16', () => {
+    it('Should read first uint16', async () => {
+      const res = await libBytes.readFirstUint16('0x03021e4453120a')
+      expect(res[0]).to.eq.BN(770)
+      expect(res[1]).to.eq.BN(2)
+    })
+    it('Should read first uint16 of 2 byte array', async () => {
+      const res = await libBytes.readFirstUint16('0xff0a')
+      expect(res[0]).to.eq.BN(65290)
+      expect(res[1]).to.eq.BN(2)
+    })
+    it('Should fail first uint16 out of bounds', async () => {
+      const tx = libBytes.readFirstUint16('0x5a')
+      await expect(tx).to.be.rejectedWith('LibBytes#readFirstUint16: OUT_OF_BOUNDS')
+    })
+  })
+
+  describe('readUint8Uint8', () => {
+    it('Should read bool and uint8 at index zero', async () => {
+      const res = await libBytes.readUint8Uint8('0x011e4453120a', 0)
+      expect(res[0]).to.eq.BN(1)
+      expect(res[1]).to.eq.BN(30)
+      expect(res[2]).to.eq.BN(2)
+    })
+    it('Should read bool and uint8 at given index', async () => {
+      const res = await libBytes.readUint8Uint8('0x5a9c2a0019d401d3', 3)
+      expect(res[0]).to.eq.BN(0)
+      expect(res[1]).to.eq.BN(25)
+      expect(res[2]).to.eq.BN(5)
+    })
+    it('Should read bool and uint8 at last index', async () => {
+      const res = await libBytes.readUint8Uint8('0x020414', 1)
+      expect(res[0]).to.eq.BN(4)
+      expect(res[1]).to.eq.BN(20)
+      expect(res[2]).to.eq.BN(3)
+    })
+    it('Should fail read bool and uint8 out of bounds', async () => {
+      const tx = libBytes.readUint8Uint8('0x5a', 0)
+      await expect(tx).to.be.rejectedWith('LibBytes#readUint8Uint8: OUT_OF_BOUNDS')
+    })
+    it('Should fail read bool and uint16 fully out of bounds', async () => {
+      const tx = libBytes.readUint8Uint8('0x5a9ca2', 12)
+      await expect(tx).to.be.rejectedWith('LibBytes#readUint8Uint8: OUT_OF_BOUNDS')
+    })
+  })
+
+  describe('readAddress', () => {
+    let addr
+    beforeEach(async () => {
+      addr = web3.utils.toChecksumAddress(web3.utils.randomHex(20))
+    })
+    it('Should read address at index zero', async () => {
+      const data = addr.concat(web3.utils.randomHex(9).slice(2))
+
+      const res = await libBytes.readAddress(data, 0)
+      expect(res[0]).to.equal(addr)
+      expect(res[1]).to.eq.BN(20)
+    })
+    it('Should read address at given index', async () => {
+      const data = web3.utils
+        .randomHex(13)
+        .concat(addr.slice(2))
+        .concat(web3.utils.randomHex(6).slice(2))
+
+      const res = await libBytes.readAddress(data, 13)
+      expect(res[0]).to.equal(addr)
+      expect(res[1]).to.eq.BN(33)
+    })
+    it('Should read address at last index', async () => {
+      const data = web3.utils.randomHex(44).concat(addr.slice(2))
+
+      const res = await libBytes.readAddress(data, 44)
+      expect(res[0]).to.equal(addr)
+      expect(res[1]).to.eq.BN(64)
+    })
+    it('Should fail read address out of bounds', async () => {
+      const data = web3.utils.randomHex(44).concat(addr.slice(2))
+      const tx = libBytes.readAddress(data, 45)
+      await expect(tx).to.be.rejectedWith('LibBytes#readAddress: OUT_OF_BOUNDS')
+    })
+    it('Should fail read address totally out of bounds', async () => {
+      const tx = libBytes.readAddress('0x010203', 345)
+      await expect(tx).to.be.rejectedWith('LibBytes#readAddress: OUT_OF_BOUNDS')
+    })
+  })
+
+  describe('readBytes66', () => {
+    let bytes66
+    beforeEach(async () => {
+      bytes66 = web3.utils.randomHex(66)
+    })
+    it('Should read bytes66 at index zero', async () => {
+      const data = bytes66.concat(web3.utils.randomHex(12).slice(2))
+
+      const res = await libBytes.readBytes66(data, 0)
+      expect(res[0]).to.equal(bytes66)
+      expect(res[1]).to.eq.BN(66)
+    })
+    it('Should read bytes66 at given index', async () => {
+      const data = web3.utils
+        .randomHex(18)
+        .concat(bytes66.slice(2))
+        .concat(web3.utils.randomHex(62).slice(2))
+
+      const res = await libBytes.readBytes66(data, 18)
+      expect(res[0]).to.equal(bytes66)
+      expect(res[1]).to.eq.BN(84)
+    })
+    it('Should read bytes66 at last index', async () => {
+      const data = web3.utils.randomHex(33).concat(bytes66.slice(2))
+
+      const res = await libBytes.readBytes66(data, 33)
+      expect(res[0]).to.equal(bytes66)
+      expect(res[1]).to.eq.BN(99)
+    })
+    it('Should fail read bytes66 out of bounds', async () => {
+      const data = web3.utils.randomHex(33).concat(bytes66.slice(2))
+      const tx = libBytes.readBytes66(data, 34)
+      await expect(tx).to.be.rejectedWith('LibBytes#readBytes66: OUT_OF_BOUNDS')
+    })
+    it('Should fail read bytes66 totally out of bounds', async () => {
+      const tx = libBytes.readBytes66('0x010203', 345)
+      await expect(tx).to.be.rejectedWith('LibBytes#readBytes66: OUT_OF_BOUNDS')
+    })
+  })
+
+  describe('readBytes32', () => {
+    let bytes32
+    beforeEach(async () => {
+      bytes32 = web3.utils.randomHex(32)
+    })
+    it('Should read bytes32 at index zero', async () => {
+      const data = bytes32.concat(web3.utils.randomHex(16).slice(2))
+
+      const res = await libBytes.readBytes32(data, 0)
+      expect(res).to.equal(bytes32)
+    })
+    it('Should read bytes32 at given index', async () => {
+      const data = web3.utils
+        .randomHex(12)
+        .concat(bytes32.slice(2))
+        .concat(web3.utils.randomHex(44).slice(2))
+
+      const res = await libBytes.readBytes32(data, 12)
+      expect(res).to.equal(bytes32)
+    })
+    it('Should read bytes32 at last index', async () => {
+      const data = web3.utils.randomHex(11).concat(bytes32.slice(2))
+
+      const res = await libBytes.readBytes32(data, 11)
+      expect(res).to.equal(bytes32)
+    })
+    it('Should fail read bytes32 out of bounds', async () => {
+      const data = web3.utils.randomHex(11).concat(bytes32.slice(2))
+      const tx = libBytes.readBytes32(data, 12)
+      await expect(tx).to.be.rejectedWith('LibBytes#readBytes32: GREATER_OR_EQUAL_TO_32_LENGTH_REQUIRED')
+    })
+    it('Should fail read bytes32 totally out of bounds', async () => {
+      const tx = libBytes.readBytes32('0x010203', 3145)
+      await expect(tx).to.be.rejectedWith('LibBytes#readBytes32: GREATER_OR_EQUAL_TO_32_LENGTH_REQUIRED')
+    })
+  })
+
+  describe('readUint16', () => {
+    it('Should read uint16 at index zero', async () => {
+      const res = await libBytes.readUint16('0x5202', 0)
+      expect(res[0]).to.eq.BN(20994)
+      expect(res[1]).to.eq.BN(2)
+    })
+    it('Should read uint16 at given index', async () => {
+      const res = await libBytes.readUint16('0x5a9c2a1019d401d3', 3)
+      expect(res[0]).to.eq.BN(4121)
+      expect(res[1]).to.eq.BN(5)
+    })
+    it('Should read uint16 at last index', async () => {
+      const res = await libBytes.readUint16('0x020414', 1)
+      expect(res[0]).to.eq.BN(1044)
+      expect(res[1]).to.eq.BN(3)
+    })
+    it('Should fail read uint16 out of bounds', async () => {
+      const tx = libBytes.readUint16('0x5a', 0)
+      await expect(tx).to.be.rejectedWith('LibBytes#readUint16: OUT_OF_BOUNDS')
+    })
+    it('Should fail read uint16 fully out of bounds', async () => {
+      const tx = libBytes.readUint16('0x5a9ca2', 12)
+      await expect(tx).to.be.rejectedWith('LibBytes#readUint16: OUT_OF_BOUNDS')
+    })
+  })
+
+  describe('readBytes', () => {
+    let size
+    let bytes
+
+    const modes = [{
+      name: "Big bytes",
+      size: () => ethers.BigNumber.from(web3.utils.randomHex(2)).toNumber()
+    }, {
+      name: "Max bytes",
+      size: () => 65535
+    }, {
+      name: "Small bytes",
+      size: () => ethers.BigNumber.from(web3.utils.randomHex(1)).toNumber()
+    }].concat([...new Array(130)].map((_, i) => ({
+      name: `${i} bytes`,
+      size: () => i
+    })))
+
+    modes.forEach((mode) => {
+      context(mode.name, () => {
+        beforeEach(async () => {
+          size = mode.size()
+          bytes = web3.utils.randomHex(size)
+        })
+        it('Should read bytes at index zero', async () => {
+          const data = bytes.concat(web3.utils.randomHex(16).slice(2))
+
+          const res = await libBytes.readBytes(data, 0, size)
+          expect(res[0]).to.equal(size === 0 ? null : bytes)
+          expect(res[1]).to.eq.BN(size)
+        })
+        it('Should read bytes at given index', async () => {
+          const data = web3.utils
+            .randomHex(12)
+            .concat(bytes.slice(2))
+            .concat(web3.utils.randomHex(44).slice(2))
+
+          const res = await libBytes.readBytes(data, 12, size)
+          expect(res[0]).to.equal(size === 0 ? null : bytes)
+          expect(res[1]).to.eq.BN(size + 12)
+        })
+        it('Should read bytes at last index', async () => {
+          const data = web3.utils.randomHex(11).concat(bytes.slice(2))
+
+          const res = await libBytes.readBytes(data, 11, size)
+          expect(res[0]).to.equal(size === 0 ? null : bytes)
+          expect(res[1]).to.eq.BN(size + 11)
+        })
+        it('Should fail read bytes out of bounds', async () => {
+          const data = web3.utils.randomHex(11).concat(bytes.slice(2))
+          const tx = libBytes.readBytes(data, 12, size)
+          await expect(tx).to.be.rejectedWith('LibBytes#readBytes: OUT_OF_BOUNDS')
+        })
+        it('Should fail read bytes totally out of bounds', async () => {
+          const tx = libBytes.readBytes('0x010203', 3145, size)
+          await expect(tx).to.be.rejectedWith('LibBytes#readBytes: OUT_OF_BOUNDS')
+        })
+      })
+    })
+  })
+})
diff --git a/forks/passport/tests/MainModule.bench.ts b/forks/passport/tests/MainModule.bench.ts
new file mode 100644
index 00000000..5173026f
--- /dev/null
+++ b/forks/passport/tests/MainModule.bench.ts
@@ -0,0 +1,241 @@
+import { ethers } from 'ethers'
+import { signAndExecuteMetaTx, encodeImageHash, multiSignAndExecuteMetaTx, addressOf } from './utils'
+
+import { MainModule, Factory } from 'src/gen/typechain'
+
+ethers.utils.Logger.setLogLevel(ethers.utils.Logger.levels.ERROR)
+
+const FactoryArtifact = artifacts.require('Factory')
+const MainModuleArtifact = artifacts.require('MainModule')
+
+const runs = 256
+import { web3, ethers as hh } from 'hardhat'
+
+const optimalGasLimit = ethers.constants.Two.pow(22)
+
+function report(test: string, values: number[]) {
+  const min = Math.min(...values)
+  const max = Math.max(...values)
+  const avg = values
+    .map(n => ethers.BigNumber.from(n))
+    .reduce((p, n) => p.add(n))
+    .div(values.length)
+    .toNumber()
+
+  console.info(` -> ${test} runs: ${values.length} cost min: ${min} max: ${max} avg: ${avg}`)
+}
+
+contract('MainModule', () => {
+  let factory
+  let module
+
+  let networkId
+
+  before(async () => {
+    // Get signer
+    const [signer] = await hh.getSigners()
+    // Deploy wallet factory
+    factory = (await FactoryArtifact.new(signer.address, await signer.getAddress())) as Factory
+    // Deploy MainModule
+    module = (await MainModuleArtifact.new(factory.address)) as MainModule
+    // Get network ID
+    networkId = await web3.eth.net.getId()
+  })
+
+  if (process.env.BENCHMARK) {
+    describe.only('Benchmark', function () {
+      ;(this as any).timeout(0)
+
+      it('Deploy a wallet', async () => {
+        const results: number[] = []
+
+        for (let i = 0; i < runs; i++) {
+          const owner = new ethers.Wallet(ethers.utils.randomBytes(32))
+          const salt = encodeImageHash(1, [{ weight: 1, address: owner.address }])
+          const tx = await factory.deploy(module.address, salt)
+          results.push(tx.receipt.gasUsed)
+        }
+
+        report('deploy wallets', results)
+      })
+
+      it('Relay 1/1 transaction', async () => {
+        const results: number[] = []
+
+        const transaction = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: ethers.constants.AddressZero,
+          value: ethers.constants.Zero,
+          data: []
+        }
+
+        for (let i = 0; i < runs; i++) {
+          const owner = new ethers.Wallet(ethers.utils.randomBytes(32))
+          const salt = encodeImageHash(1, [{ weight: 1, address: owner.address }])
+          await factory.deploy(module.address, salt)
+          const wallet = (await MainModuleArtifact.at(addressOf(factory.address, module.address, salt))) as MainModule
+
+          const tx = (await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)) as any
+          results.push(tx.receipt.gasUsed)
+        }
+
+        report('relay 1/1 transaction', results)
+      })
+
+      const batches = [2, 3, 5, 10, 50, 100]
+      batches.forEach(n => {
+        it(`Relay 1/1 ${n} transactions`, async () => {
+          const results: number[] = []
+
+          const transactions = new Array(n).fill(0).map(() => ({
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: optimalGasLimit,
+            target: ethers.constants.AddressZero,
+            value: ethers.constants.Zero,
+            data: []
+          }))
+
+          for (let i = 0; i < runs; i++) {
+            const owner = new ethers.Wallet(ethers.utils.randomBytes(32))
+            const salt = encodeImageHash(1, [{ weight: 1, address: owner.address }])
+            await factory.deploy(module.address, salt)
+            const wallet = (await MainModuleArtifact.at(addressOf(factory.address, module.address, salt))) as MainModule
+
+            const tx = (await signAndExecuteMetaTx(wallet, owner, transactions, networkId)) as any
+            results.push(tx.receipt.gasUsed)
+          }
+
+          report(`relay 1/1 ${n} transactions`, results)
+        })
+      })
+      batches.forEach(n => {
+        const ntxs = Math.floor(n / 2)
+        const nfailing = n - ntxs
+        it(`Relay 1/1 ${ntxs} transactions and ${nfailing} failing transactions`, async () => {
+          const results: number[] = []
+
+          const transactions = new Array(ntxs)
+            .fill(0)
+            .map(() => ({
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: optimalGasLimit,
+              target: ethers.constants.AddressZero,
+              value: ethers.constants.Zero,
+              data: []
+            }))
+            .concat(
+              new Array(nfailing).fill(0).map(() => ({
+                delegateCall: false,
+                revertOnError: false,
+                gasLimit: optimalGasLimit,
+                target: factory.address,
+                value: ethers.constants.Zero,
+                data: []
+              }))
+            )
+
+          for (let i = 0; i < runs; i++) {
+            const owner = new ethers.Wallet(ethers.utils.randomBytes(32))
+            const salt = encodeImageHash(1, [{ weight: 1, address: owner.address }])
+            await factory.deploy(module.address, salt)
+            const wallet = (await MainModuleArtifact.at(addressOf(factory.address, module.address, salt))) as MainModule
+
+            const tx = (await signAndExecuteMetaTx(wallet, owner, transactions, networkId)) as any
+            results.push(tx.receipt.gasUsed)
+          }
+
+          report(`relay 1/1 ${ntxs} transactions and ${nfailing} failing transactions`, results)
+        })
+      })
+
+      it('Relay 2/5 transaction', async () => {
+        const results: number[] = []
+
+        const threshold = 4
+        const transaction = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: ethers.constants.AddressZero,
+          value: ethers.constants.Zero,
+          data: []
+        }
+
+        for (let i = 0; i < runs; i++) {
+          const owners = Array(5)
+            .fill(0)
+            .map(() => new ethers.Wallet(ethers.utils.randomBytes(32)))
+          const weights = [3, 3, 1, 1, 1]
+
+          const salt = encodeImageHash(
+            threshold,
+            owners.map((owner, i) => ({
+              weight: weights[i],
+              address: owner.address
+            }))
+          )
+
+          await factory.deploy(module.address, salt)
+          const wallet = (await MainModuleArtifact.at(addressOf(factory.address, module.address, salt))) as MainModule
+
+          const signers = [0, 3]
+
+          const accounts = owners.map((owner, i) => ({
+            weight: weights[i],
+            owner: signers.includes(i) ? owner : owner.address
+          }))
+
+          const tx = (await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId)) as any
+          results.push(tx.receipt.gasUsed)
+        }
+
+        report('relay 2/5 transaction', results)
+      })
+      it('Relay 255/255 transaction', async () => {
+        const results: number[] = []
+
+        const threshold = 255
+        const weight = 1
+        const transaction = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: ethers.constants.AddressZero,
+          value: ethers.constants.Zero,
+          data: []
+        }
+
+        for (let i = 0; i < runs; i++) {
+          const owners = Array(255)
+            .fill(0)
+            .map(() => new ethers.Wallet(ethers.utils.randomBytes(32)))
+
+          const salt = encodeImageHash(
+            threshold,
+            owners.map(owner => ({
+              weight: weight,
+              address: owner.address
+            }))
+          )
+
+          await factory.deploy(module.address, salt)
+          const wallet = (await MainModuleArtifact.at(addressOf(factory.address, module.address, salt))) as MainModule
+
+          const accounts = owners.map(owner => ({
+            weight: weight,
+            owner: owner
+          }))
+
+          const tx = (await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId)) as any
+          results.push(tx.receipt.gasUsed)
+        }
+
+        report('relay 255/255 transaction', results)
+      })
+    })
+  }
+})
diff --git a/forks/passport/tests/MainModule.spec.ts b/forks/passport/tests/MainModule.spec.ts
new file mode 100644
index 00000000..75e7eb07
--- /dev/null
+++ b/forks/passport/tests/MainModule.spec.ts
@@ -0,0 +1,4007 @@
+import { ethers as hardhat, web3 } from 'hardhat'
+import { ethers, BigNumberish } from 'ethers'
+import {
+  expect,
+  signAndExecuteMetaTx,
+  RevertError,
+  RevertOutOfGasError,
+  RevertCallException,
+  ethSign,
+  encodeImageHash,
+  walletSign,
+  walletMultiSign,
+  multiSignAndExecuteMetaTx,
+  encodeNonce,
+  moduleStorageKey,
+  encodeMetaTransactionsData,
+  addressOf,
+  multiSignMetaTransactions,
+  compareAddr,
+  nextNonce,
+  encodeMessageData,
+  MetaTransactionsType,
+  encodeMessageSubDigest
+} from './utils'
+
+import {
+  Factory__factory,
+  MainModule__factory,
+  MainModuleUpgradable__factory,
+  RequireUtils__factory,
+  HookCallerMock__factory,
+  CallReceiverMock__factory,
+  DelegateCallMock__factory,
+  HookMock__factory,
+  ModuleMock__factory,
+  AlwaysRevertMock__factory,
+  MainModule,
+  MainModuleUpgradable,
+  Factory,
+  CallReceiverMock,
+  ModuleMock,
+  AlwaysRevertMock,
+  HookCallerMock,
+  HookMock,
+  DelegateCallMock,
+  GasBurnerMock,
+  RequireUtils,
+  GasBurnerMock__factory
+} from '../src'
+
+const CallReceiverMockArtifact = artifacts.require('CallReceiverMock')
+
+const optimalGasLimit = ethers.constants.Two.pow(21)
+
+ethers.utils.Logger.setLogLevel(ethers.utils.Logger.levels.ERROR)
+
+contract('MainModule', (accounts: string[]) => {
+  let signer: ethers.Signer
+  let networkId: number
+
+  let factory: Factory
+  let mainModule: MainModule
+  let moduleUpgradable: MainModuleUpgradable
+  let requireUtils: RequireUtils
+
+  let owner: ethers.Wallet
+  let wallet: MainModule
+
+  before(async () => {
+    // get signer and provider from hardhat
+    signer = (await hardhat.getSigners())[0]
+    // provider = hardhat.provider
+
+    // Get network ID
+    networkId = process.env.NET_ID ? Number(process.env.NET_ID) : await web3.eth.net.getId()
+
+    // Deploy wallet factory
+    factory = await new Factory__factory().connect(signer).deploy(await signer.getAddress(), await signer.getAddress())
+    // Deploy MainModule
+    mainModule = await new MainModule__factory().connect(signer).deploy(factory.address)
+    moduleUpgradable = await new MainModuleUpgradable__factory().connect(signer).deploy()
+
+    // Deploy RequireUtils
+    requireUtils = await new RequireUtils__factory().connect(signer).deploy(factory.address, mainModule.address)
+  })
+
+  beforeEach(async () => {
+    owner = new ethers.Wallet(ethers.utils.randomBytes(32))
+    const salt = encodeImageHash(1, [{ weight: 1, address: owner.address }])
+    await factory.deploy(mainModule.address, salt, { gasLimit: 200000 })
+    wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+  })
+
+  describe('Nested signatures', () => {
+    it('Should accept simple nested signed ERC1271 message', async () => {
+      const hookMock = await new HookCallerMock__factory().connect(signer).deploy()
+
+      // WalletA
+      const owner_a = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt_a = encodeImageHash(1, [{ weight: 1, address: owner_a.address }])
+      await factory.deploy(mainModule.address, salt_a)
+      const wallet_a = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt_a), signer)
+
+      // Top level wallet
+      const salt = encodeImageHash(1, [{ weight: 1, address: wallet_a.address }])
+      await factory.deploy(mainModule.address, salt)
+      const wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+
+      const message = ethers.utils.hexlify(ethers.utils.randomBytes(95))
+
+      const topLevelDigest = ethers.utils.keccak256(encodeMessageData(wallet.address, message, networkId))
+
+      const walletADigest = ethers.utils.keccak256(encodeMessageSubDigest(wallet_a.address, topLevelDigest, networkId))
+
+      const signedWalletA = (await walletMultiSign([{ weight: 1, owner: owner_a }], 1, walletADigest, false, true)) + '03'
+
+      const topLevelSigned = await walletMultiSign(
+        [{ weight: 1, owner: wallet_a.address, signature: signedWalletA }],
+        1,
+        topLevelDigest,
+        false,
+        true
+      )
+
+      await hookMock.callERC1271isValidSignatureData(wallet.address, message, topLevelSigned)
+    })
+
+    it('Should accept simple nested signer', async () => {
+      // WalletA
+      const owner_a = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt_a = encodeImageHash(1, [{ weight: 1, address: owner_a.address }])
+      await factory.deploy(mainModule.address, salt_a)
+      const wallet_a = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt_a), signer)
+
+      // Top level wallet
+      const salt = encodeImageHash(1, [{ weight: 1, address: wallet_a.address }])
+      await factory.deploy(mainModule.address, salt)
+      const wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+      }
+
+      const topLevelDigest = ethers.utils.keccak256(
+        encodeMetaTransactionsData(wallet.address, [transaction], networkId, await nextNonce(wallet))
+      )
+
+      const walletADigest = ethers.utils.keccak256(encodeMessageSubDigest(wallet_a.address, topLevelDigest, networkId))
+
+      const signedWalletA = (await walletMultiSign([{ weight: 1, owner: owner_a }], 1, walletADigest, false, true)) + '03'
+
+      const topLevelSigned = await walletMultiSign(
+        [{ weight: 1, owner: wallet_a.address, signature: signedWalletA }],
+        1,
+        topLevelDigest,
+        false,
+        true
+      )
+
+      await wallet.execute([transaction], await nextNonce(wallet), topLevelSigned)
+
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+
+    it('Should accept two nested signers', async () => {
+      // WalletA
+      const owner_a = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt_a = encodeImageHash(1, [{ weight: 1, address: owner_a.address }])
+      await factory.deploy(mainModule.address, salt_a)
+      const wallet_a = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt_a), signer)
+
+      // WalletB
+      const owner_b = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt_b = encodeImageHash(1, [{ weight: 1, address: owner_b.address }])
+      await factory.deploy(mainModule.address, salt_b)
+      const wallet_b = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt_b), signer)
+
+      // Top level wallet
+      const salt = encodeImageHash(2, [
+        { weight: 1, address: wallet_a.address },
+        { weight: 1, address: wallet_b.address }
+      ])
+      await factory.deploy(mainModule.address, salt)
+      const wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+      }
+
+      const topLevelDigest = ethers.utils.keccak256(
+        encodeMetaTransactionsData(wallet.address, [transaction], networkId, await nextNonce(wallet))
+      )
+
+      // Sign using wallet A
+      const signedWalletA =
+        (await walletMultiSign(
+          [{ weight: 1, owner: owner_a }],
+          1,
+          encodeMessageSubDigest(wallet_a.address, topLevelDigest, networkId)
+        )) + '03'
+
+      // Sign using wallet B
+      const signedWalletB =
+        (await walletMultiSign(
+          [{ weight: 1, owner: owner_b }],
+          1,
+          encodeMessageSubDigest(wallet_b.address, topLevelDigest, networkId)
+        )) + '03'
+
+      const topLevelSigned = await walletMultiSign(
+        [
+          { weight: 1, owner: wallet_a.address, signature: signedWalletA },
+          { weight: 1, owner: wallet_b.address, signature: signedWalletB }
+        ],
+        2,
+        topLevelDigest,
+        false,
+        true
+      )
+
+      await wallet.execute([transaction], await nextNonce(wallet), topLevelSigned)
+
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+
+    it('Should accept mixed nested and eoa signers', async () => {
+      // Wallet A
+      const owner_a = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt_a = encodeImageHash(1, [{ weight: 1, address: owner_a.address }])
+      await factory.deploy(mainModule.address, salt_a)
+      const wallet_a = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt_a), signer)
+
+      // Owner B
+      const owner_b = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+      // Top level wallet
+      const salt = encodeImageHash(2, [
+        { weight: 1, address: wallet_a.address },
+        { weight: 1, address: owner_b.address }
+      ])
+      await factory.deploy(mainModule.address, salt)
+      const wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+      }
+
+      const topLevelDigest = ethers.utils.keccak256(
+        encodeMetaTransactionsData(wallet.address, [transaction], networkId, await nextNonce(wallet))
+      )
+
+      // Sign using wallet A
+      const signedWalletA =
+        (await walletMultiSign(
+          [{ weight: 1, owner: owner_a }],
+          1,
+          encodeMessageSubDigest(wallet_a.address, topLevelDigest, networkId)
+        )) + '03'
+
+      const topLevelSigned = await walletMultiSign(
+        [
+          { weight: 1, owner: wallet_a.address, signature: signedWalletA },
+          { weight: 1, owner: owner_b }
+        ],
+        2,
+        topLevelDigest,
+        false,
+        true
+      )
+
+      await wallet.execute([transaction], await nextNonce(wallet), topLevelSigned)
+
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+
+    const cases = [
+      {
+        name: '2 nested sequence wallets',
+        childs: 1,
+        depth: 2
+      },
+      {
+        name: '64 nested sequence wallets',
+        childs: 1,
+        depth: 64
+      },
+      {
+        name: '97 nested sequence wallets',
+        childs: 1,
+        depth: 97
+      },
+      {
+        name: 'binary tree of sequence wallets',
+        childs: 2,
+        depth: 5
+      },
+      {
+        name: 'ternary tree of sequence wallets',
+        childs: 3,
+        depth: 4
+      },
+      {
+        name: 'hexary tree of sequence wallets',
+        childs: 16,
+        depth: 2
+      },
+      {
+        name: 'random tree of sequence wallets (depth 1)',
+        depth: 1
+      },
+      {
+        name: 'random tree of sequence wallets (depth 2)',
+        depth: 2
+      },
+      {
+        name: 'random tree of sequence wallets (depth 3)',
+        depth: 3
+      },
+      {
+        name: 'random tree of sequence wallets (depth 4)',
+        depth: 4
+      }
+    ]
+
+    cases.map(c => {
+      it(`Should handle ${c.name}`, async () => {
+        type Node = {
+          owner: MainModule | ethers.Wallet
+          childs?: Node[]
+        }
+
+        const gen = async (depth: number, numChilds: number | undefined, max: number): Promise<Node> => {
+          let nchilds = numChilds ? numChilds : Math.floor(Math.random() * 5 + 1)
+          nchilds = depth === 0 && nchilds === 0 ? 1 : nchilds
+          if (depth === max || nchilds === 0) {
+            return {
+              owner: new ethers.Wallet(ethers.utils.randomBytes(32))
+            }
+          }
+
+          const childs = await Promise.all([...new Array(nchilds)].map(() => gen(depth + 1, numChilds, max)))
+          const salt = encodeImageHash(
+            childs.length,
+            childs.map(c => ({ weight: 1, address: c.owner.address }))
+          )
+
+          await factory.deploy(mainModule.address, salt)
+          const wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+
+          return {
+            owner: wallet,
+            childs: childs
+          }
+        }
+
+        const sign = async (node: Node, digest: string): Promise<string> => {
+          if (node.childs) {
+            const subDigest = ethers.utils.keccak256(encodeMessageSubDigest(node.owner.address, digest, networkId))
+
+            const signatures = await Promise.all(node.childs.map(async c => (await sign(c, subDigest)) + (c.childs ? '03' : '')))
+            return await walletMultiSign(
+              node.childs.map((c, i) => ({ weight: 1, owner: c.owner.address, signature: signatures[i] })),
+              node.childs.length,
+              digest,
+              false,
+              true
+            )
+          }
+
+          const eoas = await ethSign(node.owner as ethers.Wallet, digest, true)
+          return eoas
+        }
+
+        const tree = await gen(0, c.childs, c.depth)
+
+        const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+        const valA = 5423
+        const valB = web3.utils.randomHex(120)
+
+        const transaction = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+
+        const topLevelDigest = ethers.utils.defaultAbiCoder.encode(
+          ['uint256', MetaTransactionsType],
+          [await nextNonce(tree.owner as MainModule), [transaction]]
+        )
+
+        const signature = await sign(tree, ethers.utils.keccak256(topLevelDigest))
+        await (tree.owner as MainModule).execute([transaction], await nextNonce(wallet), signature)
+
+        expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+        expect(await callReceiver.lastValB()).to.equal(valB)
+      }).timeout(50000)
+    })
+
+    it('Should reject invalid nested signature', async () => {
+      // WalletA
+      const owner_a = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt_a = encodeImageHash(1, [{ weight: 1, address: owner_a.address }])
+      await factory.deploy(mainModule.address, salt_a)
+      const wallet_a = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt_a), signer)
+
+      // WalletB
+      const owner_b = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt_b = encodeImageHash(1, [{ weight: 1, address: owner_b.address }])
+      await factory.deploy(mainModule.address, salt_b)
+      const wallet_b = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt_b), signer)
+
+      // Top level wallet
+      const salt = encodeImageHash(2, [
+        { weight: 1, address: wallet_a.address },
+        { weight: 1, address: wallet_b.address }
+      ])
+      await factory.deploy(mainModule.address, salt)
+      const wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+      }
+
+      const topLevelDigest = ethers.utils.keccak256(
+        encodeMetaTransactionsData(wallet.address, [transaction], networkId, await nextNonce(wallet))
+      )
+
+      // Sign using wallet A
+      const signedWalletA =
+        (await walletMultiSign(
+          [{ weight: 1, owner: owner_a }],
+          1,
+          encodeMessageSubDigest(wallet_a.address, topLevelDigest, networkId)
+        )) + '03'
+
+      // Sign using wallet A again (invalid)
+      const signedWalletB =
+        (await walletMultiSign(
+          [{ weight: 1, owner: owner_a }],
+          1,
+          encodeMessageSubDigest(wallet_b.address, topLevelDigest, networkId)
+        )) + '03'
+
+      const topLevelSigned = await walletMultiSign(
+        [
+          { weight: 1, owner: wallet_a.address, signature: signedWalletA },
+          { weight: 1, owner: wallet_b.address, signature: signedWalletB }
+        ],
+        2,
+        topLevelDigest,
+        false,
+        true
+      )
+
+      const tx = wallet.execute([transaction], await nextNonce(wallet), topLevelSigned)
+      await expect(tx).to.be.rejectedWith(RevertError('ModuleAuth#_signatureValidation: INVALID_SIGNATURE'))
+    })
+
+    it('Should enforce threshold on nested sigantures', async () => {
+      // WalletA
+      const owner_a = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt_a = encodeImageHash(1, [{ weight: 1, address: owner_a.address }])
+      await factory.deploy(mainModule.address, salt_a)
+      const wallet_a = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt_a), signer)
+
+      // WalletB
+      const owner_b = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt_b = encodeImageHash(1, [{ weight: 1, address: owner_b.address }])
+      await factory.deploy(mainModule.address, salt_b)
+      const wallet_b = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt_b), signer)
+
+      // Top level wallet
+      const salt = encodeImageHash(3, [
+        { weight: 1, address: wallet_a.address },
+        { weight: 1, address: wallet_b.address }
+      ])
+      await factory.deploy(mainModule.address, salt)
+      const wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+      }
+
+      const topLevelDigest = ethers.utils.keccak256(
+        encodeMetaTransactionsData(wallet.address, [transaction], networkId, await nextNonce(wallet))
+      )
+
+      // Sign using wallet A
+      const signedWalletA =
+        (await walletMultiSign(
+          [{ weight: 1, owner: owner_a }],
+          1,
+          encodeMessageSubDigest(wallet_a.address, topLevelDigest, networkId)
+        )) + '03'
+
+      // Sign using wallet B
+      const signedWalletB =
+        (await walletMultiSign(
+          [{ weight: 1, owner: owner_b }],
+          1,
+          encodeMessageSubDigest(wallet_b.address, topLevelDigest, networkId)
+        )) + '03'
+
+      const topLevelSigned = await walletMultiSign(
+        [
+          { weight: 1, owner: wallet_a.address, signature: signedWalletA },
+          { weight: 1, owner: wallet_b.address, signature: signedWalletB }
+        ],
+        2,
+        topLevelDigest,
+        false,
+        true
+      )
+
+      const tx = wallet.execute([transaction], await nextNonce(wallet), topLevelSigned)
+      await expect(tx).to.be.rejectedWith(RevertError('ModuleCalls#execute: INVALID_SIGNATURE'))
+    })
+
+    it('Should read weight of nested wallets', async () => {
+      // WalletA
+      const owner_a = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt_a = encodeImageHash(1, [{ weight: 1, address: owner_a.address }])
+      await factory.deploy(mainModule.address, salt_a)
+      const wallet_a = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt_a), signer)
+
+      // WalletB
+      const owner_b = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt_b = encodeImageHash(1, [{ weight: 1, address: owner_b.address }])
+      await factory.deploy(mainModule.address, salt_b)
+      const wallet_b = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt_b), signer)
+
+      // WalletC
+      const owner_c = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt_c = encodeImageHash(1, [{ weight: 1, address: owner_c.address }])
+      await factory.deploy(mainModule.address, salt_c)
+      const wallet_c = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt_c), signer)
+
+      // Top level wallet
+      const salt = encodeImageHash(2, [
+        { weight: 1, address: wallet_a.address },
+        { weight: 1, address: wallet_b.address },
+        { weight: 2, address: wallet_c.address }
+      ])
+      await factory.deploy(mainModule.address, salt)
+      const wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+      }
+
+      const topLevelDigest = ethers.utils.keccak256(
+        encodeMetaTransactionsData(wallet.address, [transaction], networkId, await nextNonce(wallet))
+      )
+
+      // Sign using wallet A
+      const signedWalletA =
+        (await walletMultiSign(
+          [{ weight: 1, owner: owner_a }],
+          1,
+          encodeMessageSubDigest(wallet_a.address, topLevelDigest, networkId)
+        )) + '03'
+
+      // Sign using wallet B
+      const signedWalletB =
+        (await walletMultiSign(
+          [{ weight: 1, owner: owner_b }],
+          1,
+          encodeMessageSubDigest(wallet_b.address, topLevelDigest, networkId)
+        )) + '03'
+
+      // Sign using wallet C
+      const signedWalletC =
+        (await walletMultiSign(
+          [{ weight: 1, owner: owner_c }],
+          1,
+          encodeMessageSubDigest(wallet_c.address, topLevelDigest, networkId)
+        )) + '03'
+
+      const topLevelSignedOnlyA = await walletMultiSign(
+        [
+          {
+            weight: 1,
+            owner: wallet_a.address,
+            signature: signedWalletA
+          },
+          {
+            weight: 1,
+            owner: wallet_b.address
+          },
+          {
+            weight: 2,
+            owner: wallet_c.address
+          }
+        ],
+        2,
+        topLevelDigest,
+        false,
+        true
+      )
+
+      let tx = wallet.execute([transaction], await nextNonce(wallet), topLevelSignedOnlyA)
+      await expect(tx).to.be.rejectedWith(RevertError('ModuleCalls#execute: INVALID_SIGNATURE'))
+
+      const topLevelSignedOnlyB = await walletMultiSign(
+        [
+          {
+            weight: 1,
+            owner: wallet_a.address
+          },
+          {
+            weight: 1,
+            owner: wallet_b.address,
+            signature: signedWalletB
+          },
+          {
+            weight: 2,
+            owner: wallet_c.address
+          }
+        ],
+        2,
+        topLevelDigest,
+        false,
+        true
+      )
+
+      tx = wallet.execute([transaction], await nextNonce(wallet), topLevelSignedOnlyB)
+      await expect(tx).to.be.rejectedWith(RevertError('ModuleCalls#execute: INVALID_SIGNATURE'))
+
+      const topLevelSignedOnlyC = await walletMultiSign(
+        [
+          {
+            weight: 1,
+            owner: wallet_a.address
+          },
+          {
+            weight: 1,
+            owner: wallet_b.address
+          },
+          {
+            weight: 2,
+            owner: wallet_c.address,
+            signature: signedWalletC
+          }
+        ],
+        2,
+        topLevelDigest,
+        false,
+        true
+      )
+
+      await wallet.execute([transaction], await nextNonce(wallet), topLevelSignedOnlyC)
+
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+  })
+
+  describe('Authentication', () => {
+    it('Should accept initial owner signature', async () => {
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: ethers.constants.AddressZero,
+        value: ethers.constants.Zero,
+        data: []
+      }
+
+      await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+    })
+    it('Should reject non-owner signature', async () => {
+      const impostor = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: ethers.constants.AddressZero,
+        value: ethers.constants.Zero,
+        data: []
+      }
+
+      const tx = signAndExecuteMetaTx(wallet, impostor, [transaction], networkId)
+      await expect(tx).to.be.rejectedWith(RevertError('ModuleCalls#execute: INVALID_SIGNATURE'))
+    })
+    describe('Network ID', () => {
+      it('Should reject a transaction of another network id', async () => {
+        const transaction = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: ethers.constants.AddressZero,
+          value: ethers.constants.Zero,
+          data: []
+        }
+
+        const tx = signAndExecuteMetaTx(wallet, owner, [transaction], networkId - 1)
+        await expect(tx).to.be.rejectedWith(RevertError('ModuleCalls#execute: INVALID_SIGNATURE'))
+      })
+    })
+    describe('Nonce', () => {
+      const spaces = [
+        ethers.BigNumber.from(0),
+        ethers.BigNumber.from(1),
+        ethers.BigNumber.from(7342),
+        ethers.BigNumber.from(ethers.utils.randomBytes(20)),
+        ethers.constants.Two.pow(160).sub(ethers.constants.One)
+      ]
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: ethers.constants.AddressZero,
+        value: ethers.constants.Zero,
+        data: []
+      }
+      context('Using non-encoded nonce', () => {
+        it('Should default to space zero', async () => {
+          const nonce = ethers.constants.Zero
+
+          await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, nonce)
+          expect((await wallet.nonce()).toNumber()).to.eq.BN(1)
+        })
+        it('Should work with zero as initial nonce', async () => {
+          const nonce = ethers.constants.Zero
+
+          await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, nonce)
+          expect((await wallet.readNonce(0)).toNumber()).to.eq.BN(1)
+        })
+        it('Should emit NonceChange event', async () => {
+          const receipt1 = await (await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, 0)).wait()
+          const receipt2 = await (await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, 1)).wait()
+
+          const ev1 = receipt1.events?.find(l => l.event === 'NonceChange')!
+          expect(ev1.event).to.be.equal('NonceChange')
+
+          expect(ev1.args?._space.toNumber()).to.eq.BN(0)
+          expect(ev1.args?._newNonce.toNumber()).to.eq.BN(1)
+
+          const ev2 = receipt2.events?.find(l => l.event === 'NonceChange')!
+          expect(ev2.event).to.be.equal('NonceChange')
+          expect(ev1.args?._space.toNumber()).to.eq.BN(0)
+          expect(ev2.args?._newNonce.toNumber()).to.eq.BN(2)
+        })
+        it('Should fail if nonce did not change', async () => {
+          await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, ethers.constants.Zero)
+          const tx = signAndExecuteMetaTx(wallet, owner, [transaction], networkId, ethers.constants.Zero)
+
+          await expect(tx).to.be.rejectedWith(RevertError('MainModule#_auth: INVALID_NONCE'))
+        })
+        it('Should fail if nonce increased by two', async () => {
+          await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, 0)
+          const tx = signAndExecuteMetaTx(wallet, owner, [transaction], networkId, 2)
+
+          await expect(tx).to.be.rejectedWith(RevertError('MainModule#_auth: INVALID_NONCE'))
+        })
+      })
+      spaces.forEach(space => {
+        context(`using ${space.toHexString()} space`, () => {
+          it('Should work with zero as initial nonce', async () => {
+            const nonce = ethers.constants.Zero
+
+            const encodedNonce = encodeNonce(space, nonce)
+            await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodedNonce)
+            expect((await wallet.readNonce(space)).toNumber()).to.eq.BN(1)
+          })
+          it('Should emit NonceChange event', async () => {
+            const encodedFirstNonce = encodeNonce(space, ethers.constants.Zero)
+            const encodedSecondNonce = encodeNonce(space, ethers.constants.One)
+
+            const receipt1 = await (await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodedFirstNonce)).wait()
+            const receipt2 = await (
+              await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodedSecondNonce)
+            ).wait()
+
+            const ev1 = receipt1.events?.find(l => l.event === 'NonceChange')!
+            expect(ev1.event).to.be.equal('NonceChange')
+
+            expect(ev1.args?._space.toString()).to.eq.BN(space.toString())
+            expect(ev1.args?._newNonce.toString()).to.eq.BN(1)
+
+            const ev2 = receipt2.events?.find(l => l.event === 'NonceChange')!
+            expect(ev2.event).to.be.equal('NonceChange')
+            expect(ev2.args?._space.toString()).to.eq.BN(space.toString())
+            expect(ev2.args?._newNonce.toString()).to.eq.BN(2)
+          })
+          it('Should accept next nonce', async () => {
+            const encodedFirstNonce = encodeNonce(space, ethers.constants.Zero)
+            const encodedSecondNonce = encodeNonce(space, ethers.constants.One)
+
+            await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodedFirstNonce)
+            await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodedSecondNonce)
+
+            expect((await wallet.readNonce(space)).toNumber()).to.eq.BN(2)
+          })
+          it('Should fail if nonce did not change', async () => {
+            const encodedNonce = encodeNonce(space, ethers.constants.Zero)
+
+            await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodedNonce)
+            const tx = signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodedNonce)
+
+            await expect(tx).to.be.rejectedWith(RevertError('MainModule#_auth: INVALID_NONCE'))
+          })
+          it('Should fail if nonce increased by two', async () => {
+            const encodedFirstNonce = encodeNonce(space, ethers.constants.Zero)
+            const encodedSecondNonce = encodeNonce(space, ethers.constants.Two)
+
+            await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodedFirstNonce)
+            const tx = signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodedSecondNonce)
+
+            await expect(tx).to.be.rejectedWith(RevertError('MainModule#_auth: INVALID_NONCE'))
+          })
+          it('Should use nonces storage keys', async () => {
+            const subkey = ethers.utils.defaultAbiCoder.encode(['uint256'], [space])
+            const storageKey = moduleStorageKey('0xc40e2218089ef03fc40794d84d38778f688da53b98c9236b084936bfafc9a601', subkey)
+
+            const nonce = ethers.constants.Zero
+
+            const encodedNonce = encodeNonce(space, nonce)
+            await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodedNonce)
+
+            const storageValue = await web3.eth.getStorageAt(wallet.address, storageKey)
+            expect(web3.utils.toBN(storageValue)).to.eq.BN(1)
+          })
+        })
+      })
+      context('using two spaces simultaneously', () => {
+        it('Should keep separated nonce counts', async () => {
+          await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodeNonce(1, 0))
+
+          expect((await wallet.readNonce(1)).toNumber()).to.eq.BN(1)
+          expect((await wallet.readNonce(2)).toNumber()).to.eq.BN(0)
+
+          await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodeNonce(2, 0))
+
+          expect((await wallet.readNonce(1)).toNumber()).to.eq.BN(1)
+          expect((await wallet.readNonce(2)).toNumber()).to.eq.BN(1)
+
+          await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodeNonce(2, 1))
+          await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodeNonce(2, 2))
+
+          expect((await wallet.readNonce(1)).toNumber()).to.eq.BN(1)
+          expect((await wallet.readNonce(2)).toNumber()).to.eq.BN(3)
+        })
+        it('Should emit different events', async () => {
+          await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodeNonce(1, 0))
+          await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodeNonce(1, 1))
+
+          const receipt1 = await (await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodeNonce(1, 2))).wait()
+          const receipt2 = await (await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodeNonce(2, 0))).wait()
+
+          const ev1 = receipt1.events?.find(l => l.event === 'NonceChange')!
+          expect(ev1.event).to.be.equal('NonceChange')
+          expect(ev1.args?._space.toNumber()).to.eq.BN(1)
+          expect(ev1.args?._newNonce.toNumber()).to.eq.BN(3)
+
+          const ev2 = receipt2.events?.find(l => l.event === 'NonceChange')!
+          expect(ev2.event).to.be.equal('NonceChange')
+          expect(ev2.args?._space.toNumber()).to.eq.BN(2)
+          expect(ev2.args?._newNonce.toNumber()).to.eq.BN(1)
+        })
+        it('Should not accept nonce of different space', async () => {
+          await signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodeNonce(1, 0))
+          const tx = signAndExecuteMetaTx(wallet, owner, [transaction], networkId, encodeNonce(2, 1))
+          await expect(tx).to.be.rejectedWith(RevertError('MainModule#_auth: INVALID_NONCE'))
+        })
+      })
+    })
+    it('Should reject signature with invalid flag', async () => {
+      const signature = '0x00010301'
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: ethers.constants.AddressZero,
+        value: ethers.constants.Zero,
+        data: []
+      }
+
+      const tx = wallet.execute([transaction], 0, signature)
+      await expect(tx).to.be.rejectedWith(RevertError('ModuleAuth#_signatureValidation INVALID_FLAG'))
+    })
+  })
+
+  describe('Upgradeability', () => {
+    it('Should update implementation', async () => {
+      const newImplementation = await new ModuleMock__factory().connect(signer).deploy()
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: wallet.address,
+        value: ethers.constants.Zero,
+        data: wallet.interface.encodeFunctionData('updateImplementation', [newImplementation.address])
+      }
+
+      await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+
+      const mock_wallet = await ModuleMock__factory.connect(wallet.address, signer)
+      expect((await (await mock_wallet.ping()).wait()).events![0].event).to.equal('Pong')
+    })
+    it('Should fail to set implementation to address 0', async () => {
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: wallet.address,
+        value: ethers.constants.Zero,
+        data: wallet.interface.encodeFunctionData('updateImplementation', [ethers.constants.AddressZero])
+      }
+
+      const tx = signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+      await expect(tx).to.be.rejectedWith(RevertError('ModuleUpdate#updateImplementation: INVALID_IMPLEMENTATION'))
+    })
+    it('Should fail to set implementation to non-contract', async () => {
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: wallet.address,
+        value: ethers.constants.Zero,
+        data: wallet.interface.encodeFunctionData('updateImplementation', [accounts[1]])
+      }
+
+      const tx = signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+      await expect(tx).to.be.rejectedWith(RevertError('ModuleUpdate#updateImplementation: INVALID_IMPLEMENTATION'))
+    })
+    it('Should use implementation storage key', async () => {
+      const newImplementation = await new ModuleMock__factory().connect(signer).deploy()
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: wallet.address,
+        value: ethers.constants.Zero,
+        data: wallet.interface.encodeFunctionData('updateImplementation', [newImplementation.address])
+      }
+
+      await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+
+      const storageValue = await web3.eth.getStorageAt(wallet.address, wallet.address)
+
+      expect(ethers.utils.getAddress(ethers.utils.defaultAbiCoder.decode(['address'], storageValue)[0])).to.equal(
+        newImplementation.address
+      )
+    })
+  })
+
+  describe('External calls', () => {
+    it('Should perform call to contract', async () => {
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+      }
+
+      await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+
+    it('Should return error message', async () => {
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+      await callReceiver.setRevertFlag(true)
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: callReceiver.interface.encodeFunctionData('testCall', [0, []])
+      }
+
+      const tx = signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+      await expect(tx).to.be.rejectedWith(RevertError('CallReceiverMock#testCall: REVERT_FLAG'))
+    })
+
+    describe('Batch transactions', () => {
+      it('Should perform multiple calls to contracts in one tx', async () => {
+        const callReceiver1 = await new CallReceiverMock__factory().connect(signer).deploy()
+        const callReceiver2 = await new CallReceiverMock__factory().connect(signer).deploy()
+
+        const val1A = 5423
+        const val1B = web3.utils.randomHex(120)
+
+        const val2A = 695412
+        const val2B = web3.utils.randomHex(35)
+
+        const transactions = [
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: optimalGasLimit,
+            target: callReceiver1.address,
+            value: ethers.constants.Zero,
+            data: callReceiver1.interface.encodeFunctionData('testCall', [val1A, val1B])
+          },
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: optimalGasLimit,
+            target: callReceiver2.address,
+            value: ethers.constants.Zero,
+            data: callReceiver2.interface.encodeFunctionData('testCall', [val2A, val2B])
+          }
+        ]
+
+        await signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+        expect((await callReceiver1.lastValA()).toNumber()).to.eq.BN(val1A)
+        expect(await callReceiver1.lastValB()).to.equal(val1B)
+        expect((await callReceiver2.lastValA()).toNumber()).to.eq.BN(val2A)
+        expect(await callReceiver2.lastValB()).to.equal(val2B)
+      })
+
+      it('Should perform call a contract and transfer eth in one tx', async () => {
+        const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+        const receiver = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+        await signer.sendTransaction({
+          to: wallet.address,
+          value: 100
+        })
+
+        const valA = 5423
+        const valB = web3.utils.randomHex(120)
+
+        const transactions = [
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: optimalGasLimit,
+            target: callReceiver.address,
+            value: ethers.constants.Zero,
+            data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+          },
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: optimalGasLimit,
+            target: receiver.address,
+            value: 26,
+            data: []
+          }
+        ]
+
+        await signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+        expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+        expect(await callReceiver.lastValB()).to.equal(valB)
+        expect(await web3.eth.getBalance(receiver.address)).to.eq.BN(26)
+      })
+      it('Should fail if one transaction fails', async () => {
+        const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+        const receiver = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+        await callReceiver.setRevertFlag(true)
+        await signer.sendTransaction({
+          to: wallet.address,
+          value: 100
+        })
+
+        const transactions = [
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: optimalGasLimit,
+            target: receiver.address,
+            value: 26,
+            data: []
+          },
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: optimalGasLimit,
+            target: callReceiver.address,
+            value: ethers.constants.Zero,
+            data: callReceiver.interface.encodeFunctionData('testCall', [0, []])
+          }
+        ]
+
+        const tx = signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+        await expect(tx).to.be.rejectedWith('CallReceiverMock#testCall: REVERT_FLAG')
+      })
+    })
+  })
+
+  describe('Delegate calls', () => {
+    let mockModule: DelegateCallMock
+
+    beforeEach(async () => {
+      mockModule = await new DelegateCallMock__factory().connect(signer).deploy()
+    })
+
+    it('Should delegate call to module', async () => {
+      const transaction1 = {
+        delegateCall: true,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: mockModule.address,
+        value: 0,
+        data: mockModule.interface.encodeFunctionData('write', [11, 45])
+      }
+
+      await signAndExecuteMetaTx(wallet, owner, [transaction1], networkId)
+
+      const transaction2 = {
+        delegateCall: true,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: mockModule.address,
+        value: 0,
+        data: mockModule.interface.encodeFunctionData('read', [11])
+      }
+
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, [transaction2], networkId)).wait()
+      const val = web3.utils.toBN(tx.logs.slice(-2)[0].data)
+      expect(val).to.eq.BN(45)
+    })
+    context('on delegate call revert', () => {
+      beforeEach(async () => {
+        const transaction = {
+          delegateCall: true,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: mockModule.address,
+          value: 0,
+          data: mockModule.interface.encodeFunctionData('setRevertFlag', [true])
+        }
+
+        await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+      })
+      it('Should pass if delegate call is optional', async () => {
+        const transaction = {
+          delegateCall: true,
+          revertOnError: false,
+          gasLimit: optimalGasLimit,
+          target: mockModule.address,
+          value: 0,
+          data: mockModule.interface.encodeFunctionData('write', [11, 45])
+        }
+
+        await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+      })
+      it('Should fail if delegate call fails', async () => {
+        const transaction = {
+          delegateCall: true,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: mockModule.address,
+          value: 0,
+          data: mockModule.interface.encodeFunctionData('write', [11, 45])
+        }
+
+        const tx = signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+        await expect(tx).to.be.rejectedWith('DelegateCallMock#write: REVERT_FLAG')
+      })
+    })
+  })
+
+  describe('Handle ETH', () => {
+    it('Should receive ETH', async () => {
+      await signer.sendTransaction({
+        to: wallet.address,
+        value: 1
+      })
+    })
+    it('Should transfer ETH', async () => {
+      await signer.sendTransaction({
+        to: wallet.address,
+        value: 100
+      })
+
+      const receiver = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: receiver.address,
+        value: 25,
+        data: []
+      }
+
+      await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+      expect(await web3.eth.getBalance(receiver.address)).to.eq.BN(25)
+    })
+    it('Should call payable function', async () => {
+      await signer.sendTransaction({
+        to: wallet.address,
+        value: 100
+      })
+
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const valA = 63129
+      const valB = web3.utils.randomHex(120)
+      const value = 33
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: callReceiver.address,
+        value: value,
+        data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+      }
+
+      await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+      expect(await web3.eth.getBalance(callReceiver.address)).to.eq.BN(value)
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+  })
+
+  describe('Optional transactions', () => {
+    it('Should skip a skipOnError transaction', async () => {
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+      await callReceiver.setRevertFlag(true)
+
+      const data = callReceiver.interface.encodeFunctionData('testCall', [0, []])
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: false,
+        gasLimit: optimalGasLimit,
+        target: callReceiver.address,
+        value: ethers.constants.Zero,
+        data: data
+      }
+
+      const tx = await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+      const receipt = await tx.wait()
+      const event = receipt.events?.pop()!
+
+      const reason = web3.eth.abi.decodeParameter('string', event.args?._reason.slice(10))
+
+      expect(reason).to.equal('CallReceiverMock#testCall: REVERT_FLAG')
+
+      expect(event.args?._index).to.eq.BN(0)
+    })
+    it('Should skip failing transaction within batch', async () => {
+      const callReceiver1 = await new CallReceiverMock__factory().connect(signer).deploy()
+      const callReceiver2 = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      await callReceiver1.setRevertFlag(true)
+
+      const valA = 912341
+      const valB = web3.utils.randomHex(30)
+
+      const data1 = callReceiver1.interface.encodeFunctionData('testCall', [0, []])
+      const data2 = callReceiver2.interface.encodeFunctionData('testCall', [valA, valB])
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit,
+          target: callReceiver1.address,
+          value: ethers.constants.Zero,
+          data: data1
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver2.address,
+          value: ethers.constants.Zero,
+          data: data2
+        }
+      ]
+
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, transactions, networkId)).wait()
+      const event = tx.events?.find(l => l.event === 'TxFailed')!
+
+      const reason = web3.eth.abi.decodeParameter('string', event.args?._reason.slice(10))
+
+      expect(reason).to.equal('CallReceiverMock#testCall: REVERT_FLAG')
+
+      expect((await callReceiver2.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver2.lastValB()).to.equal(valB)
+    })
+    it('Should skip multiple failing transactions within batch', async () => {
+      const callReceiver1 = await new CallReceiverMock__factory().connect(signer).deploy()
+      const callReceiver2 = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      await callReceiver1.setRevertFlag(true)
+
+      const valA = 912341
+      const valB = web3.utils.randomHex(30)
+
+      const data1 = callReceiver1.interface.encodeFunctionData('testCall', [0, []])
+      const data2 = callReceiver2.interface.encodeFunctionData('testCall', [valA, valB])
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit,
+          target: callReceiver1.address,
+          value: ethers.constants.Zero,
+          data: data1
+        },
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit,
+          target: callReceiver1.address,
+          value: ethers.constants.Zero,
+          data: data1
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver2.address,
+          value: ethers.constants.Zero,
+          data: data2
+        }
+      ]
+
+      const txHash = ethers.utils.keccak256(encodeMetaTransactionsData(wallet.address, transactions, networkId, 0))
+
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, transactions, networkId)).wait()
+
+      const event1 = tx.events![1]
+      const event2 = tx.events![2]
+
+      const reason1 = web3.eth.abi.decodeParameter('string', event1.args?._reason.slice(10))
+      const reason2 = web3.eth.abi.decodeParameter('string', event2.args?._reason.slice(10))
+
+      expect(reason1).to.equal('CallReceiverMock#testCall: REVERT_FLAG')
+      expect(reason2).to.equal('CallReceiverMock#testCall: REVERT_FLAG')
+
+      expect(event1.args?._tx).to.equal(txHash)
+      expect(event2.args?._tx).to.equal(txHash)
+
+      expect((await callReceiver2.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver2.lastValB()).to.equal(valB)
+    })
+    it('Should skip all failing transactions within batch', async () => {
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      await callReceiver.setRevertFlag(true)
+
+      const data = callReceiver.interface.encodeFunctionData('testCall', [0, []])
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: data
+        },
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: data
+        }
+      ]
+
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, transactions, networkId)).wait()
+      const event1 = tx.events?.pop()!
+      const event2 = tx.events?.pop()!
+
+      const reason1 = web3.eth.abi.decodeParameter('string', event1.args?._reason.slice(10))
+      const reason2 = web3.eth.abi.decodeParameter('string', event2.args?._reason.slice(10))
+
+      expect(reason1).to.equal('CallReceiverMock#testCall: REVERT_FLAG')
+      expect(reason2).to.equal('CallReceiverMock#testCall: REVERT_FLAG')
+    })
+
+    it('Should skip skipOnError update implementation action', async () => {
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      await callReceiver.setRevertFlag(true)
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('updateImplementation', [ethers.constants.AddressZero])
+        }
+      ]
+
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, transactions, networkId)).wait()
+      const event = tx.events?.pop()!
+
+      const reason = web3.eth.abi.decodeParameter('string', event.args?._reason.slice(10))
+
+      expect(reason).to.equal('ModuleUpdate#updateImplementation: INVALID_IMPLEMENTATION')
+      expect((await wallet.nonce()).toNumber()).to.eq.BN(1)
+    })
+  })
+
+  describe('Hooks', () => {
+    let hookMock: HookCallerMock
+    before(async () => {
+      hookMock = await new HookCallerMock__factory().connect(signer).deploy()
+    })
+    describe('receive tokens', () => {
+      it('Should implement ERC1155 single transfer hook', async () => {
+        await hookMock.callERC1155Received(wallet.address)
+      })
+      it('Should implement ERC1155 batch transfer hook', async () => {
+        await hookMock.callERC1155BatchReceived(wallet.address)
+      })
+      it('Should implement ERC721 transfer hook', async () => {
+        await hookMock.callERC721Received(wallet.address)
+      })
+      it('Should implement ERC223 transfer hook', async () => {
+        await hookMock.callERC223Received(wallet.address)
+      })
+    })
+    describe('ERC1271 Wallet', () => {
+      let data: string
+      let messageSubDigest: string
+      let hash: string
+      let hashSubDigest: string
+
+      beforeEach(async () => {
+        data = await web3.utils.randomHex(250)
+        messageSubDigest = ethers.utils.solidityPack(
+          ['string', 'uint256', 'address', 'bytes'],
+          ['\x19\x01', networkId, wallet.address, ethers.utils.keccak256(data)]
+        )
+
+        hash = ethers.utils.keccak256(ethers.utils.randomBytes(1024))
+        hashSubDigest = ethers.utils.solidityPack(
+          ['string', 'uint256', 'address', 'bytes'],
+          ['\x19\x01', networkId, wallet.address, ethers.utils.solidityPack(['bytes32'], [hash])]
+        )
+      })
+      it('Should validate arbitrary signed data', async () => {
+        const signature = await walletSign(owner, messageSubDigest)
+        await hookMock.callERC1271isValidSignatureData(wallet.address, data, signature)
+      })
+      it('Should validate arbitrary signed hash', async () => {
+        const signature = await walletSign(owner, hashSubDigest)
+        await hookMock.callERC1271isValidSignatureHash(wallet.address, hash, signature)
+      })
+      it('Should reject data signed by non-owner', async () => {
+        const impostor = new ethers.Wallet(ethers.utils.randomBytes(32))
+        const signature = await walletSign(impostor, messageSubDigest)
+        const tx = hookMock.callERC1271isValidSignatureData(wallet.address, data, signature)
+        await expect(tx).to.be.rejectedWith('HookCallerMock#callERC1271isValidSignatureData: INVALID_RETURN')
+      })
+      it('Should reject hash signed by non-owner', async () => {
+        const impostor = new ethers.Wallet(ethers.utils.randomBytes(32))
+        const signature = await walletSign(impostor, hashSubDigest)
+        const tx = hookMock.callERC1271isValidSignatureHash(wallet.address, hash, signature)
+        await expect(tx).to.be.rejectedWith('HookCallerMock#callERC1271isValidSignatureHash: INVALID_RETURN')
+      })
+    })
+
+    describe('External hooks', () => {
+      let hookMock: HookMock
+      before(async () => {
+        hookMock = await new HookMock__factory().connect(signer).deploy()
+      })
+      it('Should read added hook', async () => {
+        const selector = ethers.utils.id('onHookMockCall(uint256)').substring(0, 10)
+
+        const transaction = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('addHook', [selector, hookMock.address])
+        }
+
+        await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+
+        expect(await wallet.readHook(selector)).to.be.equal(hookMock.address)
+      })
+      it('Should return zero if hook is not registered', async () => {
+        const selector = ethers.utils.id('onHookMockCall(uint256)').substring(0, 10)
+        expect(await wallet.readHook(selector)).to.be.equal(ethers.constants.AddressZero)
+      })
+      it('Should forward call to external hook', async () => {
+        const selector = ethers.utils.id('onHookMockCall(uint256)').substring(0, 10)
+        const transaction = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('addHook', [selector, hookMock.address])
+        }
+
+        await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+
+        const walletHook = await new HookMock__factory().connect(signer).deploy()
+        expect((await walletHook.onHookMockCall(21)).toNumber()).to.eq.BN(42)
+      })
+      it('Should not forward call to deregistered hook', async () => {
+        const selector = ethers.utils.id('onHookMockCall(uint256)').substring(0, 10)
+        const transaction1 = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('addHook', [selector, hookMock.address])
+        }
+
+        await signAndExecuteMetaTx(wallet, owner, [transaction1], networkId)
+
+        const transaction2 = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('removeHook', [selector])
+        }
+
+        await signAndExecuteMetaTx(wallet, owner, [transaction2], networkId)
+
+        const walletHook = await HookMock__factory.connect(wallet.address, signer)
+        const tx = walletHook.onHookMockCall(21)
+        await expect(tx).to.be.rejectedWith(RevertCallException())
+      })
+      it('Should pass calling a non registered hook', async () => {
+        const selector = ethers.utils.id('onHookMockCall(uint256)').substring(0, 10)
+        const data = ethers.utils.defaultAbiCoder.encode(['bytes4'], [selector])
+        await web3.eth.sendTransaction({ from: accounts[0], to: wallet.address, data: data })
+      })
+      it('Should not forward msg.data with less than 4 bytes', async () => {
+        const alwaysRevertMock = await new AlwaysRevertMock__factory().connect(signer).deploy()
+        const paddedSelector = '0x11223300'
+
+        const transaction = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('addHook', [paddedSelector, alwaysRevertMock.address])
+        }
+
+        await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+
+        // Calling the wallet with '0x112233' should not forward the call to the hook
+        // Here the msg.data.length is < 4, thus the call should not be forwarded to the always reverting mock
+        const tx = hardhat.provider
+          .getSigner()
+          .sendTransaction({ to: wallet.address, data: '0x112233' })
+          .then(t => t.wait())
+        await expect(tx).to.be.fulfilled
+
+        // Calling the wallet with '0x11223300' should forward the call to the hook (and thus revert)
+        const tx2 = hardhat.provider
+          .getSigner()
+          .sendTransaction({ to: wallet.address, data: '0x11223300' })
+          .then(t => t.wait())
+        await expect(tx2).to.be.rejected
+      })
+
+      it('Should use hooks storage key', async () => {
+        const selector = ethers.utils.id('onHookMockCall(uint256)').substring(0, 10)
+        const subkey = ethers.utils.defaultAbiCoder.encode(['bytes4'], [selector])
+        const storageKey = moduleStorageKey('0x5f198cb61cfcc209f357b4ede4ad2218c53e3b4cb7e8fa1e8b0ec5e3951acbaa', subkey)
+
+        const transaction = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('addHook', [selector, hookMock.address])
+        }
+
+        await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+        const storageValue = await web3.eth.getStorageAt(wallet.address, storageKey)
+
+        const addr = (() => {
+          try {
+            return ethers.utils.getAddress(ethers.utils.defaultAbiCoder.decode(['address'], storageValue)[0])
+          } catch {
+            return ethers.utils.getAddress(storageValue)
+          }
+        })()
+
+        expect(addr).to.equal(hookMock.address)
+      })
+    })
+  })
+
+  describe('Publish configuration', () => {
+    context('publishConfig', () => {
+      let wallet2addr: string
+      let salt2: string
+      let owner2: ethers.Wallet
+      let threshold = 1
+      beforeEach(async () => {
+        owner2 = new ethers.Wallet(ethers.utils.randomBytes(32))
+        salt2 = encodeImageHash(threshold, [{ weight: 1, address: owner2.address }])
+        wallet2addr = addressOf(factory.address, mainModule.address, salt2)
+      })
+      it('Should publish configuration of a non-deployed wallet', async () => {
+        await signAndExecuteMetaTx(
+          wallet,
+          owner,
+          [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: ethers.constants.Zero,
+              target: requireUtils.address,
+              value: ethers.constants.Zero,
+              data: requireUtils.interface.encodeFunctionData('publishConfig', [
+                wallet2addr,
+                threshold,
+                [
+                  {
+                    signer: owner2.address,
+                    weight: 1
+                  }
+                ],
+                false
+              ])
+            }
+          ],
+          networkId
+        )
+
+        const blockHeight1 = await requireUtils.lastWalletUpdate(wallet2addr)
+        const blockHeight2 = await requireUtils.lastImageHashUpdate(salt2)
+        expect(blockHeight1.toNumber()).to.equal(0)
+        expect(blockHeight2.toNumber()).to.equal(0)
+
+        expect(await requireUtils.knownImageHashes(wallet2addr)).to.equal(ethers.constants.HashZero)
+      })
+      it('Should publish configuration of deployed wallet', async () => {
+        await factory.deploy(mainModule.address, salt2)
+        const wallet2 = await MainModule__factory.connect(wallet2addr, signer)
+        await signAndExecuteMetaTx(
+          wallet2,
+          owner2,
+          [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: ethers.constants.Zero,
+              target: requireUtils.address,
+              value: ethers.constants.Zero,
+              data: requireUtils.interface.encodeFunctionData('publishConfig', [
+                wallet2.address,
+                threshold,
+                [
+                  {
+                    signer: owner2.address,
+                    weight: 1
+                  }
+                ],
+                false
+              ])
+            }
+          ],
+          networkId
+        )
+
+        const blockHeight1 = await requireUtils.lastWalletUpdate(wallet2addr)
+        const blockHeight2 = await requireUtils.lastImageHashUpdate(salt2)
+        expect(blockHeight1.toNumber()).to.equal(0)
+        expect(blockHeight2.toNumber()).to.equal(0)
+
+        expect(await requireUtils.knownImageHashes(wallet2addr)).to.equal(ethers.constants.HashZero)
+      })
+      it('Should fail to publish wrong configuraiton of a non-deployed wallet', async () => {
+        const tx = signAndExecuteMetaTx(
+          wallet,
+          owner,
+          [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: ethers.constants.Zero,
+              target: requireUtils.address,
+              value: ethers.constants.Zero,
+              data: requireUtils.interface.encodeFunctionData('publishConfig', [
+                wallet2addr,
+                threshold,
+                [
+                  {
+                    signer: owner2.address,
+                    weight: 2
+                  }
+                ],
+                false
+              ])
+            }
+          ],
+          networkId
+        )
+        await expect(tx).to.be.rejectedWith(RevertError('RequireUtils#publishConfig: UNEXPECTED_COUNTERFACTUAL_IMAGE_HASH'))
+      })
+      it('Should fail to publish wrong configuration of a non-updated wallet', async () => {
+        await factory.deploy(mainModule.address, salt2)
+        const wallet2 = await MainModule__factory.connect(wallet2addr, signer)
+        const tx = signAndExecuteMetaTx(
+          wallet2,
+          owner2,
+          [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: ethers.constants.Zero,
+              target: requireUtils.address,
+              value: ethers.constants.Zero,
+              data: requireUtils.interface.encodeFunctionData('publishConfig', [
+                wallet2.address,
+                threshold,
+                [
+                  {
+                    signer: owner2.address,
+                    weight: 2
+                  }
+                ],
+                false
+              ])
+            }
+          ],
+          networkId
+        )
+        await expect(tx).to.be.rejectedWith(RevertError('RequireUtils#publishConfig: UNEXPECTED_COUNTERFACTUAL_IMAGE_HASH'))
+      })
+    })
+    context('publishConfig indexed', () => {
+      let wallet2addr: string
+      let salt2: string
+      let owner2: ethers.Wallet
+      let threshold = 1
+      beforeEach(async () => {
+        owner2 = new ethers.Wallet(ethers.utils.randomBytes(32))
+        salt2 = encodeImageHash(threshold, [{ weight: 1, address: owner2.address }])
+        wallet2addr = addressOf(factory.address, mainModule.address, salt2)
+      })
+      it('Should publish configuration of a non-deployed wallet', async () => {
+        const tx = await signAndExecuteMetaTx(
+          wallet,
+          owner,
+          [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: 300_000,
+              target: requireUtils.address,
+              value: ethers.constants.Zero,
+              data: requireUtils.interface.encodeFunctionData('publishConfig', [
+                wallet2addr,
+                threshold,
+                [
+                  {
+                    signer: owner2.address,
+                    weight: 1
+                  }
+                ],
+                true
+              ])
+            }
+          ],
+          networkId
+        )
+
+        const receipt = await tx.wait()
+
+        const blockHeight1 = await requireUtils.lastWalletUpdate(wallet2addr)
+        const blockHeight2 = await requireUtils.lastImageHashUpdate(salt2)
+        expect(receipt.blockNumber).to.equal(blockHeight1.toNumber())
+        expect(blockHeight1.toNumber()).to.equal(blockHeight2.toNumber())
+
+        expect(await requireUtils.knownImageHashes(wallet2addr)).to.equal(salt2)
+      })
+      it('Should publish configuration of a deployed wallet', async () => {
+        await factory.deploy(mainModule.address, salt2)
+        const wallet2 = await MainModule__factory.connect(wallet2addr, signer)
+        const tx = await signAndExecuteMetaTx(
+          wallet2,
+          owner2,
+          [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: ethers.constants.Zero,
+              target: requireUtils.address,
+              value: ethers.constants.Zero,
+              data: requireUtils.interface.encodeFunctionData('publishConfig', [
+                wallet2.address,
+                threshold,
+                [
+                  {
+                    signer: owner2.address,
+                    weight: 1
+                  }
+                ],
+                true
+              ])
+            }
+          ],
+          networkId
+        )
+
+        const receipt = await tx.wait()
+
+        const blockHeight1 = await requireUtils.lastWalletUpdate(wallet2.address)
+        const blockHeight2 = await requireUtils.lastImageHashUpdate(salt2)
+        expect(receipt.blockNumber).to.equal(blockHeight1.toNumber())
+        expect(blockHeight1.toNumber()).to.equal(blockHeight2.toNumber())
+
+        expect(await requireUtils.knownImageHashes(wallet2addr)).to.equal(salt2)
+      })
+      it('Should publish configuration of an updated wallet', async () => {
+        await factory.deploy(mainModule.address, salt2)
+        const wallet2 = await MainModule__factory.connect(wallet2addr, signer)
+
+        const newOwnerA = ethers.Wallet.createRandom()
+        const newImageHash = encodeImageHash(1, [{ weight: 1, address: newOwnerA.address }])
+
+        const newWallet = await MainModuleUpgradable__factory.connect(wallet2.address, signer)
+
+        const migrateBundle = [
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: ethers.constants.Two.pow(18),
+            target: newWallet.address,
+            value: ethers.constants.Zero,
+            data: newWallet.interface.encodeFunctionData('updateImplementation', [moduleUpgradable.address])
+          },
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: ethers.constants.Two.pow(18),
+            target: newWallet.address,
+            value: ethers.constants.Zero,
+            data: newWallet.interface.encodeFunctionData('updateImageHash', [newImageHash])
+          }
+        ]
+
+        const migrateTransaction = [
+          {
+            delegateCall: false,
+            revertOnError: false,
+            gasLimit: optimalGasLimit,
+            target: newWallet.address,
+            value: ethers.constants.Zero,
+            data: newWallet.interface.encodeFunctionData('selfExecute', [migrateBundle])
+          }
+        ]
+
+        await signAndExecuteMetaTx(wallet2, owner2, migrateTransaction, networkId)
+
+        const tx = await signAndExecuteMetaTx(
+          wallet2,
+          newOwnerA,
+          [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: ethers.constants.Zero,
+              target: requireUtils.address,
+              value: ethers.constants.Zero,
+              data: requireUtils.interface.encodeFunctionData('publishConfig', [
+                wallet2.address,
+                threshold,
+                [
+                  {
+                    signer: newOwnerA.address,
+                    weight: 1
+                  }
+                ],
+                true
+              ])
+            }
+          ],
+          networkId
+        )
+
+        const receipt = await tx.wait()
+
+        const blockHeight1 = await requireUtils.lastWalletUpdate(wallet2addr)
+        const blockHeight2 = await requireUtils.lastImageHashUpdate(newImageHash)
+        expect(receipt.blockNumber).to.equal(blockHeight1.toNumber())
+        expect(blockHeight1.toNumber()).to.equal(blockHeight2.toNumber())
+
+        expect((await requireUtils.lastImageHashUpdate(salt2)).toNumber()).to.equal(0)
+
+        expect(await requireUtils.knownImageHashes(wallet2addr)).to.equal(ethers.constants.HashZero)
+        expect(await requireUtils.knownImageHashes(addressOf(factory.address, mainModule.address, newImageHash))).to.equal(
+          ethers.constants.HashZero
+        )
+      })
+      it('Should fail to publish wrong configuration of a non-deployed wallet', async () => {
+        const tx = signAndExecuteMetaTx(
+          wallet,
+          owner,
+          [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: ethers.constants.Zero,
+              target: requireUtils.address,
+              value: ethers.constants.Zero,
+              data: requireUtils.interface.encodeFunctionData('publishConfig', [
+                wallet2addr,
+                threshold,
+                [
+                  {
+                    signer: owner2.address,
+                    weight: 2
+                  }
+                ],
+                true
+              ])
+            }
+          ],
+          networkId
+        )
+        await expect(tx).to.be.rejectedWith(RevertError('RequireUtils#publishConfig: UNEXPECTED_COUNTERFACTUAL_IMAGE_HASH'))
+      })
+      it('Should fail to publish wrong configuration of a deployed wallet', async () => {
+        await factory.deploy(mainModule.address, salt2)
+        const wallet2 = await MainModule__factory.connect(wallet2addr, signer)
+        const tx = signAndExecuteMetaTx(
+          wallet2,
+          owner2,
+          [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: ethers.constants.Zero,
+              target: requireUtils.address,
+              value: ethers.constants.Zero,
+              data: requireUtils.interface.encodeFunctionData('publishConfig', [
+                wallet2.address,
+                threshold,
+                [
+                  {
+                    signer: owner2.address,
+                    weight: 2
+                  }
+                ],
+                true
+              ])
+            }
+          ],
+          networkId
+        )
+        await expect(tx).to.be.rejectedWith(RevertError('RequireUtils#publishConfig: UNEXPECTED_COUNTERFACTUAL_IMAGE_HASH'))
+      })
+    })
+    context('publishInitialSigners', () => {
+      let wallet2addr: string
+      let salt2: string
+      let owner2a: ethers.Wallet
+      let owner2b: ethers.Wallet
+      let owner2c: ethers.Wallet
+      let threshold = 2
+
+      let message: string
+      let digest: string
+      let preSubDigest: string
+      let signature: string
+      let badSignature: string
+      let config: { weight: number; address: string; signer?: ethers.Wallet }[]
+
+      enum SignatureType {
+        EOA,
+        EOADynamic,
+        ERC1271
+      }
+
+      const options = [
+        {
+          name: 'EOA signature indexed',
+          signatureType: SignatureType.EOA,
+          indexed: true
+        },
+        {
+          name: 'EOA signature not indexed',
+          signatureType: SignatureType.EOA,
+          indexed: false
+        },
+        {
+          name: 'dynamic EOA signature indexed',
+          signatureType: SignatureType.EOADynamic,
+          indexed: true
+        },
+        {
+          name: 'dynamic EOA signature not indexed',
+          signatureType: SignatureType.EOADynamic,
+          indexed: false
+        },
+        {
+          name: 'ERC1271 signature indexed',
+          signatureType: SignatureType.ERC1271,
+          indexed: true
+        },
+        {
+          name: 'ERC1271 signature not indexed',
+          signatureType: SignatureType.ERC1271,
+          indexed: false
+        }
+      ]
+
+      options.map(o => {
+        context(o.name, () => {
+          beforeEach(async () => {
+            owner2a = new ethers.Wallet(ethers.utils.randomBytes(32))
+            owner2b = new ethers.Wallet(ethers.utils.randomBytes(32))
+            owner2c = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+            config = [
+              { weight: 1, address: owner2a.address, signer: owner2a },
+              { weight: 1, address: owner2b.address, signer: owner2b },
+              { weight: 1, address: owner2c.address }
+            ]
+            salt2 = encodeImageHash(threshold, config)
+            wallet2addr = addressOf(factory.address, mainModule.address, salt2)
+
+            message = ethers.utils.hexlify(ethers.utils.randomBytes(96))
+            digest = ethers.utils.keccak256(message)
+            preSubDigest = ethers.utils.solidityPack(
+              ['string', 'uint256', 'address', 'bytes'],
+              ['\x19\x01', networkId, wallet2addr, digest]
+            )
+
+            switch (o.signatureType) {
+              case SignatureType.EOA:
+                signature = await walletMultiSign(
+                  [
+                    { weight: 1, owner: owner2a },
+                    { weight: 1, owner: owner2b },
+                    { weight: 1, owner: owner2c.address }
+                  ],
+                  threshold,
+                  preSubDigest
+                )
+                break
+              case SignatureType.EOADynamic:
+                signature = await walletMultiSign(
+                  [
+                    { weight: 1, owner: owner2a },
+                    { weight: 1, owner: owner2b },
+                    { weight: 1, owner: owner2c.address }
+                  ],
+                  threshold,
+                  preSubDigest,
+                  true
+                )
+                break
+              case SignatureType.ERC1271:
+                // Deploy nested sequence wallet
+                const ownern2a = new ethers.Wallet(ethers.utils.randomBytes(32))
+                const ownern2b = new ethers.Wallet(ethers.utils.randomBytes(32))
+                const nconfig = [
+                  { weight: 2, address: ownern2a.address, signer: ownern2a },
+                  { weight: 1, address: ownern2b.address }
+                ]
+                const nsalt = encodeImageHash(2, nconfig)
+                await factory.deploy(mainModule.address, nsalt)
+                const nwalletaddr = addressOf(factory.address, mainModule.address, nsalt)
+                owner2b = {
+                  address: nwalletaddr,
+                  signMessage: async msg => {
+                    const nsubdigest = ethers.utils.solidityPack(
+                      ['string', 'uint256', 'address', 'bytes'],
+                      ['\x19\x01', networkId, nwalletaddr, msg]
+                    )
+
+                    return `${await walletMultiSign(
+                      [
+                        { weight: 2, owner: ownern2a },
+                        { weight: 1, owner: ownern2b.address }
+                      ],
+                      threshold,
+                      nsubdigest
+                    )}03`
+                  }
+                } as any
+
+                // Re-create wallet
+                config = [
+                  { weight: 1, address: owner2a.address, signer: owner2a },
+                  { weight: 1, address: owner2b.address, signer: owner2b },
+                  { weight: 1, address: owner2c.address }
+                ]
+                salt2 = encodeImageHash(threshold, config)
+                wallet2addr = addressOf(factory.address, mainModule.address, salt2)
+                preSubDigest = ethers.utils.solidityPack(
+                  ['string', 'uint256', 'address', 'bytes'],
+                  ['\x19\x01', networkId, wallet2addr, digest]
+                )
+
+                signature = await walletMultiSign(
+                  [
+                    { weight: 1, owner: owner2a },
+                    { weight: 1, owner: owner2b },
+                    { weight: 1, owner: owner2c.address }
+                  ],
+                  threshold,
+                  preSubDigest
+                )
+                break
+            }
+          })
+
+          it('Should publish signers of a non-deployed wallet', async () => {
+            const tx = await signAndExecuteMetaTx(
+              wallet,
+              owner,
+              [
+                {
+                  delegateCall: false,
+                  revertOnError: true,
+                  gasLimit: ethers.constants.Zero,
+                  target: requireUtils.address,
+                  value: ethers.constants.Zero,
+                  data: requireUtils.interface.encodeFunctionData('publishInitialSigners', [
+                    wallet2addr,
+                    digest,
+                    3,
+                    signature,
+                    o.indexed
+                  ])
+                }
+              ],
+              networkId
+            )
+
+            const receipt = await tx.wait()
+
+            const logs = receipt.events!
+
+            const owner2aLog = logs.find(
+              l =>
+                l.topics.length === 3 &&
+                l.topics[0] === '0x600ba597427f042bcd559a0d06fa1732cc104d6dd43cbe8845b5a0e804b2b39f' &&
+                l.topics[2] === ethers.utils.defaultAbiCoder.encode(['address'], [owner2a.address])
+            )!
+
+            expect(owner2aLog).to.not.be.undefined
+            expect(owner2aLog.topics[1]).to.equal(ethers.utils.defaultAbiCoder.encode(['address'], [wallet2addr]))
+
+            const owner2bLog = logs.find(
+              l =>
+                l.topics.length === 3 &&
+                l.topics[0] === '0x600ba597427f042bcd559a0d06fa1732cc104d6dd43cbe8845b5a0e804b2b39f' &&
+                l.topics[2] === ethers.utils.defaultAbiCoder.encode(['address'], [owner2b.address])
+            )!
+
+            expect(owner2bLog).to.not.be.undefined
+            expect(owner2bLog.topics[1]).to.equal(ethers.utils.defaultAbiCoder.encode(['address'], [wallet2addr]))
+
+            const owner2cLog = logs.find(
+              l =>
+                l.topics.length === 3 &&
+                l.topics[0] === '0x600ba597427f042bcd559a0d06fa1732cc104d6dd43cbe8845b5a0e804b2b39f' &&
+                l.topics[2] === ethers.utils.defaultAbiCoder.encode(['address'], [owner2c.address])
+            )
+
+            expect(owner2cLog).to.be.undefined
+
+            const MembersType = `tuple(
+              uint256 weight,
+              address signer
+            )[]`
+
+            const walletLog = logs[logs.length - 2]
+            expect(walletLog.topics[1]).to.equal(ethers.utils.defaultAbiCoder.encode(['address'], [wallet2addr]))
+            expect(walletLog.topics[2]).to.equal(salt2)
+            expect(walletLog.data).to.equal(
+              ethers.utils.defaultAbiCoder.encode(
+                ['uint256', 'bytes'],
+                [
+                  threshold,
+                  ethers.utils.defaultAbiCoder.encode(
+                    [MembersType],
+                    [config.sort((a, b) => compareAddr(a.address, b.address)).map(s => ({ weight: s.weight, signer: s.address }))]
+                  )
+                ]
+              )
+            )
+
+            expect((await requireUtils.lastSignerUpdate(owner2a.address)).toNumber()).to.equal(
+              o.indexed ? receipt.blockNumber : 0
+            )
+            expect((await requireUtils.lastSignerUpdate(owner2b.address)).toNumber()).to.equal(
+              o.indexed ? receipt.blockNumber : 0
+            )
+            expect((await requireUtils.lastSignerUpdate(owner2c.address)).toNumber()).to.equal(0)
+
+            expect((await requireUtils.lastWalletUpdate(wallet2addr)).toNumber()).to.equal(o.indexed ? receipt.blockNumber : 0)
+            expect((await requireUtils.lastImageHashUpdate(salt2)).toNumber()).to.equal(o.indexed ? receipt.blockNumber : 0)
+            expect(await requireUtils.knownImageHashes(wallet2addr)).to.equal(o.indexed ? salt2 : ethers.constants.HashZero)
+          })
+          it('Should publish signers of a deployed wallet', async () => {
+            await factory.deploy(mainModule.address, salt2)
+            const tx = await signAndExecuteMetaTx(
+              wallet,
+              owner,
+              [
+                {
+                  delegateCall: false,
+                  revertOnError: true,
+                  gasLimit: ethers.constants.Zero,
+                  target: requireUtils.address,
+                  value: ethers.constants.Zero,
+                  data: requireUtils.interface.encodeFunctionData('publishInitialSigners', [
+                    wallet2addr,
+                    digest,
+                    3,
+                    signature,
+                    o.indexed
+                  ])
+                }
+              ],
+              networkId
+            )
+
+            const receipt = await tx.wait()
+
+            const logs = receipt.events!
+
+            const owner2aLog = logs.find(
+              l =>
+                l.topics.length === 3 &&
+                l.topics[0] === '0x600ba597427f042bcd559a0d06fa1732cc104d6dd43cbe8845b5a0e804b2b39f' &&
+                l.topics[2] === ethers.utils.defaultAbiCoder.encode(['address'], [owner2a.address])
+            )!
+
+            expect(owner2aLog).to.not.be.undefined
+            expect(owner2aLog.topics[1]).to.equal(ethers.utils.defaultAbiCoder.encode(['address'], [wallet2addr]))
+
+            const owner2bLog = logs.find(
+              l =>
+                l.topics.length === 3 &&
+                l.topics[0] === '0x600ba597427f042bcd559a0d06fa1732cc104d6dd43cbe8845b5a0e804b2b39f' &&
+                l.topics[2] === ethers.utils.defaultAbiCoder.encode(['address'], [owner2b.address])
+            )!
+
+            expect(owner2bLog).to.not.be.undefined
+            expect(owner2bLog.topics[1]).to.equal(ethers.utils.defaultAbiCoder.encode(['address'], [wallet2addr]))
+
+            const owner2cLog = logs.find(
+              l =>
+                l.topics.length === 3 &&
+                l.topics[0] === '0x600ba597427f042bcd559a0d06fa1732cc104d6dd43cbe8845b5a0e804b2b39f' &&
+                l.topics[2] === ethers.utils.defaultAbiCoder.encode(['address'], [owner2c.address])
+            )
+
+            expect(owner2cLog).to.be.undefined
+
+            const MembersType = `tuple(
+              uint256 weight,
+              address signer
+            )[]`
+
+            const walletLog = logs[logs.length - 2]
+            expect(walletLog.topics[1]).to.equal(ethers.utils.defaultAbiCoder.encode(['address'], [wallet2addr]))
+            expect(walletLog.topics[2]).to.equal(salt2)
+            expect(walletLog.data).to.equal(
+              ethers.utils.defaultAbiCoder.encode(
+                ['uint256', 'bytes'],
+                [
+                  threshold,
+                  ethers.utils.defaultAbiCoder.encode(
+                    [MembersType],
+                    [config.sort((a, b) => compareAddr(a.address, b.address)).map(s => ({ weight: s.weight, signer: s.address }))]
+                  )
+                ]
+              )
+            )
+
+            expect((await requireUtils.lastSignerUpdate(owner2a.address)).toNumber()).to.equal(
+              o.indexed ? receipt.blockNumber : 0
+            )
+            expect((await requireUtils.lastSignerUpdate(owner2b.address)).toNumber()).to.equal(
+              o.indexed ? receipt.blockNumber : 0
+            )
+            expect((await requireUtils.lastSignerUpdate(owner2c.address)).toNumber()).to.equal(0)
+
+            expect((await requireUtils.lastWalletUpdate(wallet2addr)).toNumber()).to.equal(o.indexed ? receipt.blockNumber : 0)
+            expect((await requireUtils.lastImageHashUpdate(salt2)).toNumber()).to.equal(o.indexed ? receipt.blockNumber : 0)
+            expect(await requireUtils.knownImageHashes(wallet2addr)).to.equal(o.indexed ? salt2 : ethers.constants.HashZero)
+          })
+          it('Should fail to publish signers with invalid part', async () => {
+            const tx = signAndExecuteMetaTx(
+              wallet,
+              owner,
+              [
+                {
+                  delegateCall: false,
+                  revertOnError: true,
+                  gasLimit: ethers.constants.Zero,
+                  target: requireUtils.address,
+                  value: ethers.constants.Zero,
+                  data: requireUtils.interface.encodeFunctionData('publishInitialSigners', [
+                    wallet2addr,
+                    digest,
+                    1,
+                    '0x0001ff01ab5801a7d398351b8be11c439e05c5b3259aec9b',
+                    o.indexed
+                  ])
+                }
+              ],
+              networkId
+            )
+
+            await expect(tx).to.be.rejectedWith(RevertError('RequireUtils#publishInitialSigners: INVALID_SIGNATURE_FLAG'))
+          })
+          it('Should fail to publish signers with invalid signers count', async () => {
+            const tx = signAndExecuteMetaTx(
+              wallet,
+              owner,
+              [
+                {
+                  delegateCall: false,
+                  revertOnError: true,
+                  gasLimit: ethers.constants.Zero,
+                  target: requireUtils.address,
+                  value: ethers.constants.Zero,
+                  data: requireUtils.interface.encodeFunctionData('publishInitialSigners', [
+                    wallet2addr,
+                    digest,
+                    4,
+                    signature,
+                    o.indexed
+                  ])
+                }
+              ],
+              networkId
+            )
+
+            await expect(tx).to.be.rejectedWith(RevertError('RequireUtils#publishInitialSigners: INVALID_MEMBERS_COUNT'))
+          })
+          it('Should fail to publish signers of non-deployed wallet with invalid signature', async () => {
+            const invalidSignature = await walletMultiSign(
+              [
+                { weight: 1, owner: ethers.Wallet.createRandom() },
+                { weight: 1, owner: owner2c.address }
+              ],
+              threshold,
+              message
+            )
+
+            const tx = signAndExecuteMetaTx(
+              wallet,
+              owner,
+              [
+                {
+                  delegateCall: false,
+                  revertOnError: true,
+                  gasLimit: ethers.constants.Zero,
+                  target: requireUtils.address,
+                  value: ethers.constants.Zero,
+                  data: requireUtils.interface.encodeFunctionData('publishInitialSigners', [
+                    wallet2addr,
+                    digest,
+                    2,
+                    invalidSignature,
+                    o.indexed
+                  ])
+                }
+              ],
+              networkId
+            )
+
+            await expect(tx).to.be.rejectedWith(
+              RevertError('RequireUtils#publishInitialSigners: UNEXPECTED_COUNTERFACTUAL_IMAGE_HASH')
+            )
+          })
+          it('Should fail to publish signers of deployed wallet with invalid signature', async () => {
+            await factory.deploy(mainModule.address, salt2)
+
+            const invalidSignature = await walletMultiSign(
+              [
+                { weight: 1, owner: ethers.Wallet.createRandom() },
+                { weight: 1, owner: owner2c.address }
+              ],
+              threshold,
+              message
+            )
+
+            const tx = signAndExecuteMetaTx(
+              wallet,
+              owner,
+              [
+                {
+                  delegateCall: false,
+                  revertOnError: true,
+                  gasLimit: ethers.constants.Zero,
+                  target: requireUtils.address,
+                  value: ethers.constants.Zero,
+                  data: requireUtils.interface.encodeFunctionData('publishInitialSigners', [
+                    wallet2addr,
+                    digest,
+                    2,
+                    invalidSignature,
+                    o.indexed
+                  ])
+                }
+              ],
+              networkId
+            )
+
+            await expect(tx).to.be.rejectedWith(
+              RevertError('RequireUtils#publishInitialSigners: UNEXPECTED_COUNTERFACTUAL_IMAGE_HASH')
+            )
+          })
+          it('Should fail to publish signers of updated wallet with invalid signature', async () => {
+            await factory.deploy(mainModule.address, salt2)
+
+            const wallet2 = await MainModule__factory.connect(wallet2addr, signer)
+
+            const newOwnerA = ethers.Wallet.createRandom()
+            const newImageHash = encodeImageHash(1, [{ weight: 1, address: newOwnerA.address }])
+
+            const newWallet = await MainModuleUpgradable__factory.connect(wallet2.address, signer)
+
+            const migrateBundle = [
+              {
+                delegateCall: false,
+                revertOnError: true,
+                gasLimit: ethers.constants.Two.pow(18),
+                target: newWallet.address,
+                value: ethers.constants.Zero,
+                data: newWallet.interface.encodeFunctionData('updateImplementation', [moduleUpgradable.address])
+              },
+              {
+                delegateCall: false,
+                revertOnError: true,
+                gasLimit: ethers.constants.Two.pow(18),
+                target: newWallet.address,
+                value: ethers.constants.Zero,
+                data: newWallet.interface.encodeFunctionData('updateImageHash', [newImageHash])
+              }
+            ]
+
+            const migrateTransaction = [
+              {
+                delegateCall: false,
+                revertOnError: false,
+                gasLimit: optimalGasLimit,
+                target: newWallet.address,
+                value: ethers.constants.Zero,
+                data: newWallet.interface.encodeFunctionData('selfExecute', [migrateBundle])
+              }
+            ]
+
+            await multiSignAndExecuteMetaTx(
+              wallet2,
+              config.map(c => ({ weight: c.weight, owner: c.signer ? c.signer : c.address })),
+              2,
+              migrateTransaction,
+              networkId
+            )
+            signature = await walletMultiSign([{ weight: 1, owner: ethers.Wallet.createRandom() }], 1, message)
+
+            const tx = signAndExecuteMetaTx(
+              wallet,
+              owner,
+              [
+                {
+                  delegateCall: false,
+                  revertOnError: true,
+                  gasLimit: ethers.constants.Zero,
+                  target: requireUtils.address,
+                  value: ethers.constants.Zero,
+                  data: requireUtils.interface.encodeFunctionData('publishInitialSigners', [
+                    wallet2addr,
+                    digest,
+                    1,
+                    signature,
+                    o.indexed
+                  ])
+                }
+              ],
+              networkId
+            )
+
+            await expect(tx).to.be.rejectedWith(
+              RevertError('RequireUtils#publishInitialSigners: UNEXPECTED_COUNTERFACTUAL_IMAGE_HASH')
+            )
+          })
+        })
+      })
+    })
+  })
+
+  describe('Update owners', async () => {
+    const transaction = {
+      delegateCall: false,
+      revertOnError: true,
+      gasLimit: optimalGasLimit,
+      target: ethers.constants.AddressZero,
+      value: 0,
+      data: []
+    }
+
+    let newOwnerA: ethers.Wallet
+    let newImageHash
+
+    context('After a migration', async () => {
+      beforeEach(async () => {
+        newOwnerA = new ethers.Wallet(ethers.utils.randomBytes(32))
+        newImageHash = encodeImageHash(1, [{ weight: 1, address: newOwnerA.address }])
+
+        const newWallet = await MainModuleUpgradable__factory.connect(wallet.address, signer)
+
+        const migrateBundle = [
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: ethers.constants.Two.pow(18),
+            target: wallet.address,
+            value: ethers.constants.Zero,
+            data: wallet.interface.encodeFunctionData('updateImplementation', [moduleUpgradable.address])
+          },
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: ethers.constants.Two.pow(18),
+            target: wallet.address,
+            value: ethers.constants.Zero,
+            data: newWallet.interface.encodeFunctionData('updateImageHash', [newImageHash])
+          }
+        ]
+
+        const migrateTransaction = [
+          {
+            delegateCall: false,
+            revertOnError: false,
+            gasLimit: optimalGasLimit,
+            target: wallet.address,
+            value: ethers.constants.Zero,
+            data: wallet.interface.encodeFunctionData('selfExecute', [migrateBundle])
+          }
+        ]
+
+        await signAndExecuteMetaTx(wallet, owner, migrateTransaction, networkId)
+        wallet = newWallet as unknown as MainModule
+      })
+      it('Should implement new upgradable module', async () => {
+        const walletUpgradable = wallet as unknown as MainModuleUpgradable
+        expect(await walletUpgradable.imageHash()).to.equal(newImageHash)
+      })
+      it('Should accept new owner signature', async () => {
+        await signAndExecuteMetaTx(wallet, newOwnerA, [transaction], networkId)
+      })
+      it('Should reject old owner signature', async () => {
+        const tx = signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+        await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+      })
+      it('Should fail to update to invalid image hash', async () => {
+        const walletUpgradable = wallet as unknown as MainModuleUpgradable
+        const transaction = {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: walletUpgradable.interface.encodeFunctionData('updateImageHash', [ethers.constants.HashZero])
+        }
+        const tx = signAndExecuteMetaTx(wallet, newOwnerA, [transaction], networkId)
+        await expect(tx).to.be.rejectedWith('ModuleAuthUpgradable#updateImageHash INVALID_IMAGE_HASH')
+      })
+      it('Should fail to change image hash from non-self address', async () => {
+        const walletUpgradable = wallet as unknown as MainModuleUpgradable
+        const tx = walletUpgradable.updateImageHash(ethers.utils.hexlify(ethers.utils.randomBytes(32)), { from: accounts[0] })
+        await expect(tx).to.be.rejectedWith('ModuleSelfAuth#onlySelf: NOT_AUTHORIZED')
+      })
+      it('Should use image hash storage key', async () => {
+        const storageKey = moduleStorageKey('0xad348b32c79cd46ad46d61aede26d38affaee58f9a122f91eb271e08720464bf')
+        const storageValue = await web3.eth.getStorageAt(wallet.address, storageKey)
+        expect(ethers.utils.defaultAbiCoder.encode(['bytes32'], [storageValue])).to.equal(newImageHash)
+      })
+      it('Should fail to execute transactions on moduleUpgradable implementation', async () => {
+        const tx = moduleUpgradable.execute([transaction], 0, '0x0000')
+        await expect(tx).to.be.rejectedWith(RevertError('ModuleCalls#execute: INVALID_SIGNATURE'))
+      })
+      it('Should update wallet and publish configuration', async () => {
+        const threshold = 2
+        const newOwnerB = new ethers.Wallet(ethers.utils.randomBytes(32))
+        const newOwnerC = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+        const members = [
+          { weight: 1, signer: newOwnerB.address },
+          { weight: 1, signer: newOwnerC.address }
+        ]
+
+        const newImageHash = encodeImageHash(threshold, [
+          { weight: 1, address: newOwnerB.address },
+          { weight: 1, address: newOwnerC.address }
+        ])
+
+        const walletUpgradable = wallet as unknown as MainModuleUpgradable
+
+        const migrateTransactions = [
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: optimalGasLimit,
+            target: wallet.address,
+            value: ethers.constants.Zero,
+            data: walletUpgradable.interface.encodeFunctionData('updateImageHash', [newImageHash])
+          },
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: ethers.constants.Zero,
+            target: requireUtils.address,
+            value: ethers.constants.Zero,
+            data: requireUtils.interface.encodeFunctionData('publishConfig', [
+              wallet.address,
+              threshold,
+              members.sort((a, b) => compareAddr(a.signer, b.signer)),
+              false
+            ])
+          }
+        ]
+
+        await signAndExecuteMetaTx(wallet, newOwnerA, migrateTransactions, networkId)
+      })
+      it('Should fail to update wallet and publish wrong configuration', async () => {
+        const threshold = 2
+        const newOwnerB = new ethers.Wallet(ethers.utils.randomBytes(32))
+        const newOwnerC = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+        const members = [
+          { weight: 1, signer: newOwnerB.address },
+          { weight: 1, signer: newOwnerC.address }
+        ]
+
+        const newImageHash = encodeImageHash(threshold, [
+          { weight: 1, address: newOwnerB.address },
+          { weight: 2, address: newOwnerC.address }
+        ])
+
+        const walletUpgradable = wallet as unknown as MainModuleUpgradable
+
+        const migrateTransactions = [
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: optimalGasLimit,
+            target: wallet.address,
+            value: ethers.constants.Zero,
+            data: walletUpgradable.interface.encodeFunctionData('updateImageHash', [newImageHash])
+          },
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: ethers.constants.Zero,
+            target: requireUtils.address,
+            value: ethers.constants.Zero,
+            data: requireUtils.interface.encodeFunctionData('publishConfig', [
+              wallet.address,
+              threshold,
+              members.sort((a, b) => compareAddr(a.signer, b.signer)),
+              false
+            ])
+          }
+        ]
+
+        const tx = signAndExecuteMetaTx(wallet, newOwnerA, migrateTransactions, networkId)
+        await expect(tx).to.be.rejectedWith(RevertError('RequireUtils#publishConfig: UNEXPECTED_IMAGE_HASH'))
+      })
+      context('After updating the image hash', () => {
+        let threshold = 2
+        let newOwnerB
+        let newOwnerC
+
+        beforeEach(async () => {
+          newOwnerB = new ethers.Wallet(ethers.utils.randomBytes(32))
+          newOwnerC = new ethers.Wallet(ethers.utils.randomBytes(32))
+          newImageHash = encodeImageHash(threshold, [
+            { weight: 1, address: newOwnerB.address },
+            { weight: 1, address: newOwnerC.address }
+          ])
+          const migrateTransactions = [
+            {
+              delegateCall: false,
+              revertOnError: true,
+              gasLimit: optimalGasLimit,
+              target: wallet.address,
+              value: ethers.constants.Zero,
+              data: (wallet as unknown as MainModuleUpgradable).interface.encodeFunctionData('updateImageHash', [newImageHash])
+            }
+          ]
+
+          await signAndExecuteMetaTx(wallet, newOwnerA, migrateTransactions, networkId)
+        })
+        it('Should have updated the image hash', async () => {
+          expect(await (wallet as unknown as MainModuleUpgradable).imageHash()).to.equal(newImageHash)
+        })
+        it('Should accept new owners signatures', async () => {
+          const accounts = [
+            {
+              weight: 1,
+              owner: newOwnerB
+            },
+            {
+              weight: 1,
+              owner: newOwnerC
+            }
+          ]
+          await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId)
+        })
+        it('Should reject old owner signatures', async () => {
+          const tx = signAndExecuteMetaTx(wallet, newOwnerA, [transaction], networkId)
+          await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+        })
+        it('Should use image hash storage key', async () => {
+          const storageKey = moduleStorageKey('0xad348b32c79cd46ad46d61aede26d38affaee58f9a122f91eb271e08720464bf')
+          const storageValue = await web3.eth.getStorageAt(wallet.address, storageKey)
+          expect(ethers.utils.defaultAbiCoder.encode(['bytes32'], [storageValue])).to.equal(newImageHash)
+        })
+      })
+    })
+  })
+
+  describe('Multisignature', async () => {
+    const transaction = {
+      delegateCall: false,
+      revertOnError: true,
+      gasLimit: optimalGasLimit,
+      target: ethers.constants.AddressZero,
+      value: 0,
+      data: []
+    }
+
+    const modes = [
+      {
+        name: 'Forced dynamic signature encoding',
+        force: true
+      },
+      {
+        name: 'Default signature encoding',
+        force: false
+      }
+    ]
+
+    modes.map(mode => {
+      context(mode.name, () => {
+        context('With 1/2 wallet', () => {
+          let owner1: ethers.Wallet
+          let owner2: ethers.Wallet
+          let ownerweight = 1
+          let threshold = 1
+
+          beforeEach(async () => {
+            owner1 = new ethers.Wallet(ethers.utils.randomBytes(32))
+            owner2 = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+            const salt = encodeImageHash(threshold, [
+              {
+                weight: ownerweight,
+                address: owner1.address
+              },
+              {
+                weight: ownerweight,
+                address: owner2.address
+              }
+            ])
+
+            await factory.deploy(mainModule.address, salt)
+            wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+          })
+          it('Should accept signed message by first owner', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1
+              },
+              {
+                weight: ownerweight,
+                owner: owner2.address
+              }
+            ]
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should accept signed message by second owner', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1.address
+              },
+              {
+                weight: ownerweight,
+                owner: owner2
+              }
+            ]
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should accept signed message by both owners', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1
+              },
+              {
+                weight: ownerweight,
+                owner: owner2
+              }
+            ]
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should reject message without signatures', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1.address
+              },
+              {
+                weight: ownerweight,
+                owner: owner2.address
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject message signed by non-owner', async () => {
+            const impostor = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: impostor
+              },
+              {
+                weight: ownerweight,
+                owner: owner2.address
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject signature of invalid length', async () => {
+            const data = encodeMetaTransactionsData(wallet.address, [transaction], networkId, await nextNonce(wallet))
+            const eoasignature = ethers.utils.arrayify(await ethSign(owner2, data, false))
+
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1.address
+              },
+              {
+                weight: ownerweight,
+                owner: owner2.address,
+                signature: [...eoasignature.slice(0, -1), 75, 2]
+              }
+            ]
+
+            const signature = await multiSignMetaTransactions(
+              wallet,
+              accounts,
+              threshold,
+              [transaction],
+              networkId,
+              await nextNonce(wallet),
+              mode.force
+            )
+            const tx = wallet.execute([transaction], await nextNonce(wallet), signature)
+            await expect(tx).to.be.rejectedWith(RevertError('SignatureValidator#recoverSigner: invalid signature length'))
+          })
+        })
+        context('With 2/2 wallet', () => {
+          let owner1
+          let owner2
+          let ownerweight = 1
+          let threshold = 2
+
+          beforeEach(async () => {
+            owner1 = new ethers.Wallet(ethers.utils.randomBytes(32))
+            owner2 = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+            const salt = encodeImageHash(threshold, [
+              {
+                weight: ownerweight,
+                address: owner1.address
+              },
+              {
+                weight: ownerweight,
+                address: owner2.address
+              }
+            ])
+
+            await factory.deploy(mainModule.address, salt)
+            wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+          })
+          it('Should accept signed message by both owners', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1
+              },
+              {
+                weight: ownerweight,
+                owner: owner2
+              }
+            ]
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should reject message without signatures', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1.address
+              },
+              {
+                weight: ownerweight,
+                owner: owner2.address
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject message signed only by first owner', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1
+              },
+              {
+                weight: ownerweight,
+                owner: owner2.address
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject message signed only by second owner', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1
+              },
+              {
+                weight: ownerweight,
+                owner: owner2.address
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject message signed by non-owner', async () => {
+            const impostor = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1
+              },
+              {
+                weight: ownerweight,
+                owner: impostor
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+        })
+        context('With 2/3 wallet', () => {
+          let owner1
+          let owner2
+          let owner3
+
+          let ownerweight = 1
+          let threshold = 2
+
+          beforeEach(async () => {
+            owner1 = new ethers.Wallet(ethers.utils.randomBytes(32))
+            owner2 = new ethers.Wallet(ethers.utils.randomBytes(32))
+            owner3 = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+            const salt = encodeImageHash(threshold, [
+              {
+                weight: ownerweight,
+                address: owner1.address
+              },
+              {
+                weight: ownerweight,
+                address: owner2.address
+              },
+              {
+                weight: ownerweight,
+                address: owner3.address
+              }
+            ])
+
+            await factory.deploy(mainModule.address, salt)
+            wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+          })
+
+          it('Should accept signed message by first and second owner', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1
+              },
+              {
+                weight: ownerweight,
+                owner: owner2
+              },
+              {
+                weight: ownerweight,
+                owner: owner3.address
+              }
+            ]
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should accept signed message by first and last owner', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1
+              },
+              {
+                weight: ownerweight,
+                owner: owner2.address
+              },
+              {
+                weight: ownerweight,
+                owner: owner3
+              }
+            ]
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should accept signed message by second and last owner', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1.address
+              },
+              {
+                weight: ownerweight,
+                owner: owner2
+              },
+              {
+                weight: ownerweight,
+                owner: owner3
+              }
+            ]
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should accept signed message by all owners', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1
+              },
+              {
+                weight: ownerweight,
+                owner: owner2
+              },
+              {
+                weight: ownerweight,
+                owner: owner3
+              }
+            ]
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should reject message signed only by first owner', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1
+              },
+              {
+                weight: ownerweight,
+                owner: owner2.address
+              },
+              {
+                weight: ownerweight,
+                owner: owner3.address
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject message signed only by second owner', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1.address
+              },
+              {
+                weight: ownerweight,
+                owner: owner2
+              },
+              {
+                weight: ownerweight,
+                owner: owner3.address
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject message signed only by last owner', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1.address
+              },
+              {
+                weight: ownerweight,
+                owner: owner2.address
+              },
+              {
+                weight: ownerweight,
+                owner: owner3
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject message not signed', async () => {
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: owner1.address
+              },
+              {
+                weight: ownerweight,
+                owner: owner2.address
+              },
+              {
+                weight: ownerweight,
+                owner: owner3.address
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject message signed by non-owner', async () => {
+            const impostor = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: impostor
+              },
+              {
+                weight: ownerweight,
+                owner: owner2
+              },
+              {
+                weight: ownerweight,
+                owner: owner3.address
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject message if the image lacks an owner', async () => {
+            const impostor = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+            const accounts = [
+              {
+                weight: ownerweight,
+                owner: impostor
+              },
+              {
+                weight: ownerweight,
+                owner: owner2
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+        })
+
+        context('With 255/255 wallet', () => {
+          let owners: ethers.Wallet[]
+          let weight = 1
+          let threshold = 255
+
+          beforeEach(async () => {
+            owners = Array(255)
+              .fill(0)
+              .map(() => new ethers.Wallet(ethers.utils.randomBytes(32)))
+
+            const salt = encodeImageHash(
+              threshold,
+              owners.map(owner => ({
+                weight: weight,
+                address: owner.address
+              }))
+            )
+
+            await factory.deploy(mainModule.address, salt)
+            wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+          })
+
+          it('Should accept message signed by all owners', async () => {
+            const accounts = owners.map(owner => ({
+              weight: weight,
+              owner: owner
+            }))
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should reject message signed by non-owner', async () => {
+            const impostor = new ethers.Wallet(ethers.utils.randomBytes(32))
+            const accounts = [
+              ...owners.slice(1).map(owner => ({
+                weight: weight,
+                owner: owner
+              })),
+              {
+                weight: weight,
+                owner: impostor
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject message missing a signature', async () => {
+            const accounts = owners.slice(1).map(owner => ({
+              weight: weight,
+              owner: owner
+            }))
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+        })
+
+        context('With weighted owners', () => {
+          let owners: ethers.Wallet[]
+          let weights: BigNumberish[]
+          let threshold = 4
+
+          beforeEach(async () => {
+            owners = Array(5)
+              .fill(0)
+              .map(() => new ethers.Wallet(ethers.utils.randomBytes(32)))
+            weights = [3, 3, 1, 1, 1]
+
+            const salt = encodeImageHash(
+              threshold,
+              owners.map((owner, i) => ({
+                weight: weights[i],
+                address: owner.address
+              }))
+            )
+
+            await factory.deploy(mainModule.address, salt)
+            wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+          })
+
+          it('Should accept signed message with (3+1)/4 weight', async () => {
+            const signers = [0, 3]
+
+            const accounts = owners.map((owner, i) => ({
+              weight: weights[i],
+              owner: signers.includes(i) ? owner : owner.address
+            }))
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should accept signed message with (3+3)/4 weight', async () => {
+            const signers = [0, 1]
+
+            const accounts = owners.map((owner, i) => ({
+              weight: weights[i],
+              owner: signers.includes(i) ? owner : owner.address
+            }))
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should accept signed message with (3+3+1+1)/4 weight', async () => {
+            const signers = [0, 1, 2, 3]
+
+            const accounts = owners.map((owner, i) => ({
+              weight: weights[i],
+              owner: signers.includes(i) ? owner : owner.address
+            }))
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should accept signed message with (3+3+1+1+1)/4 weight', async () => {
+            const signers = [0, 1, 2, 3, 4]
+
+            const accounts = owners.map((owner, i) => ({
+              weight: weights[i],
+              owner: signers.includes(i) ? owner : owner.address
+            }))
+
+            await multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+          })
+          it('Should reject signed message with (1)/4 weight', async () => {
+            const signers = [3]
+
+            const accounts = owners.map((owner, i) => ({
+              weight: weights[i],
+              owner: signers.includes(i) ? owner : owner.address
+            }))
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject signed message with (1+1)/4 weight', async () => {
+            const signers = [2, 3]
+
+            const accounts = owners.map((owner, i) => ({
+              weight: weights[i],
+              owner: signers.includes(i) ? owner : owner.address
+            }))
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject signed message with (1+1+1)/4 weight', async () => {
+            const signers = [2, 3, 4]
+
+            const accounts = owners.map((owner, i) => ({
+              weight: weights[i],
+              owner: signers.includes(i) ? owner : owner.address
+            }))
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject signed message with (3)/4 weight', async () => {
+            const signers = [0]
+
+            const accounts = owners.map((owner, i) => ({
+              weight: weights[i],
+              owner: signers.includes(i) ? owner : owner.address
+            }))
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject signed message with (0)/4 weight', async () => {
+            const accounts = owners.map((owner, i) => ({
+              weight: weights[i],
+              owner: owner.address
+            }))
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+          it('Should reject message signed by non-owner', async () => {
+            const impostor = new ethers.Wallet(ethers.utils.randomBytes(32))
+
+            const signers = [0, 1]
+
+            const accounts = [
+              ...owners.map((owner, i) => ({
+                weight: weights[i],
+                owner: signers.includes(i) ? owner : owner.address
+              })),
+              {
+                weight: 4,
+                owner: impostor
+              }
+            ]
+
+            const tx = multiSignAndExecuteMetaTx(wallet, accounts, threshold, [transaction], networkId, undefined, mode.force)
+            await expect(tx).to.be.rejectedWith('ModuleCalls#execute: INVALID_SIGNATURE')
+          })
+        })
+
+        context('Reject invalid signatures', () => {
+          it('Should reject invalid signature type', async () => {
+            const signature = await multiSignMetaTransactions(
+              wallet,
+              [{ weight: 1, owner: owner }],
+              1,
+              [transaction],
+              networkId,
+              0,
+              mode.force
+            )
+            const invalidSignature = signature.slice(0, -2) + 'ff'
+            const tx = wallet.execute([transaction], 0, invalidSignature)
+            const revertError = mode.force
+              ? 'SignatureValidator#isValidSignature: UNSUPPORTED_SIGNATURE_TYPE'
+              : 'SignatureValidator#recoverSigner: UNSUPPORTED_SIGNATURE_TYPE'
+            await expect(tx).to.be.rejectedWith(RevertError(revertError))
+          })
+          it('Should reject invalid s value', async () => {
+            const sig = await multiSignMetaTransactions(
+              wallet,
+              [{ weight: 1, owner: owner }],
+              1,
+              [transaction],
+              networkId,
+              0,
+              mode.force
+            )
+            const prefix = mode.force ? 118 : 74
+            const invalidSignature =
+              sig.slice(0, prefix) + '7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a1' + sig.slice(prefix + 64)
+            const tx = wallet.execute([transaction], 0, invalidSignature)
+            await expect(tx).to.be.rejectedWith(RevertError("SignatureValidator#recoverSigner: invalid signature 's' value"))
+          })
+          it('Should reject invalid v value', async () => {
+            const sig = await multiSignMetaTransactions(
+              wallet,
+              [{ weight: 1, owner: owner }],
+              1,
+              [transaction],
+              networkId,
+              0,
+              mode.force
+            )
+            const prefix = mode.force ? 182 : 138
+            const invalidSignature = sig.slice(0, prefix) + '1a' + sig.slice(prefix + 2)
+            const tx = wallet.execute([transaction], 0, invalidSignature)
+            await expect(tx).to.be.rejectedWith(RevertError("SignatureValidator#recoverSigner: invalid signature 'v' value"))
+          })
+          it('Should reject empty dynamic signature', async () => {
+            const invalidSignature = '0x00010200ABFf4013541fd79ee5b6847C9dF3C9B34183C2830000'
+            const tx = wallet.execute([transaction], 0, invalidSignature)
+            await expect(tx).to.be.rejectedWith(RevertError('SignatureValidator#isValidSignature: signature is empty'))
+          })
+        })
+      })
+    })
+  })
+
+  describe('Gas limit', () => {
+    let gasBurner: GasBurnerMock
+
+    before(async () => {
+      gasBurner = await new GasBurnerMock__factory().connect(signer).deploy()
+    })
+
+    it('Should forward the defined amount of gas', async () => {
+      const gas = 10000
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: gas,
+        target: gasBurner.address,
+        value: ethers.constants.Zero,
+        data: gasBurner.interface.encodeFunctionData('burnGas', [0])
+      }
+
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)).wait()
+      const reported = web3.utils.toBN(tx.logs.slice(-2)[0].data)
+      expect(reported).to.be.lt.BN(gas)
+    })
+    it('Should forward different amounts of gas', async () => {
+      const gasA = 10100
+      const gasB = 350000
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: gasA,
+          target: gasBurner.address,
+          value: ethers.constants.Zero,
+          data: gasBurner.interface.encodeFunctionData('burnGas', [8000])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: gasB,
+          target: gasBurner.address,
+          value: ethers.constants.Zero,
+          data: gasBurner.interface.encodeFunctionData('burnGas', [340000])
+        }
+      ]
+
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, transactions, networkId)).wait()
+
+      const reportedB = web3.utils.toBN(tx.logs.slice(-2)[0].data)
+      const reportedA = web3.utils.toBN(tx.logs.slice(-4)[0].data)
+
+      expect(reportedA).to.be.lt.BN(gasA)
+      expect(reportedB).to.be.lt.BN(gasB)
+      expect(reportedB).to.be.gt.BN(gasA)
+    })
+    it('Should fail if forwarded call runs out of gas', async () => {
+      const gas = 10000
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: gas,
+        target: gasBurner.address,
+        value: ethers.constants.Zero,
+        data: gasBurner.interface.encodeFunctionData('burnGas', [11000])
+      }
+
+      const tx = signAndExecuteMetaTx(wallet, owner, [transaction], networkId)
+      expect(tx).to.be.rejected
+    })
+    it('Should fail without reverting if optional call runs out of gas', async () => {
+      const gas = 10000
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: false,
+        gasLimit: gas,
+        target: gasBurner.address,
+        value: ethers.constants.Zero,
+        data: gasBurner.interface.encodeFunctionData('burnGas', [200_000])
+      }
+
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)).wait()
+      const log = tx.events?.pop()!
+      expect(log.event).to.be.equal('TxFailed')
+    })
+    it('Should continue execution if optional call runs out of gas', async () => {
+      const gas = 10000
+
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const valA = 9512358833
+      const valB = web3.utils.randomHex(1600)
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: gas,
+          target: gasBurner.address,
+          value: ethers.constants.Zero,
+          data: gasBurner.interface.encodeFunctionData('burnGas', [200000])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+      ]
+
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, transactions, networkId)).wait()
+      const log = tx.events?.slice(-2)[0]!
+      expect(log.event).to.be.equal('TxFailed')
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+    it('Should fail if transaction is executed with not enough gas', async () => {
+      const gas = 1000000
+
+      const transaction = {
+        delegateCall: false,
+        revertOnError: false,
+        gasLimit: gas,
+        target: gasBurner.address,
+        value: ethers.constants.Zero,
+        data: gasBurner.interface.encodeFunctionData('burnGas', [0])
+      }
+
+      const signed = await multiSignMetaTransactions(wallet, [{ weight: 1, owner: owner }], 1, [transaction], networkId, 0)
+
+      const tx = wallet.execute([transaction], 0, signed, { gasLimit: 250_000 })
+      await expect(tx).to.be.rejectedWith(RevertError('ModuleCalls#_execute: NOT_ENOUGH_GAS'))
+    })
+  })
+
+  describe('Create contracts', () => {
+    it('Should create a contract', async () => {
+      const deployCode = CallReceiverMockArtifact.bytecode
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('createContract', [deployCode])
+        }
+      ]
+
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, transactions, networkId)).wait()
+      const log = tx.events?.find(l => l.event === 'CreatedContract')!
+
+      expect(log.event).to.equal('CreatedContract')
+
+      const deployed = await CallReceiverMock__factory.connect(log.args?._contract, signer)
+      await deployed.testCall(12345, '0x552299')
+
+      expect((await deployed.lastValA()).toNumber()).to.eq.BN(12345)
+      expect(await deployed.lastValB()).to.equal('0x552299')
+    })
+    it('Should create a contract with value', async () => {
+      const deployCode = CallReceiverMockArtifact.bytecode
+
+      await signer.sendTransaction({
+        to: wallet.address,
+        value: 100
+      })
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: 99,
+          data: wallet.interface.encodeFunctionData('createContract', [deployCode])
+        }
+      ]
+
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, transactions, networkId)).wait()
+      const log = tx.events?.find(l => l.event === 'CreatedContract')!
+
+      expect(await web3.eth.getBalance(log.args?._contract)).to.eq.BN(99)
+    })
+    it('Should revert if contract creation fails', async () => {
+      const deployCode = '0xfb' // Invalid opcode
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('createContract', [deployCode])
+        }
+      ]
+
+      await expect(signAndExecuteMetaTx(wallet, owner, transactions, networkId)).to.be.revertedWith(
+        'ModuleCreator: creation failed'
+      )
+    })
+    it('Should fail to create a contract from non-self', async () => {
+      const deployCode = CallReceiverMockArtifact.bytecode
+
+      const tx = wallet.createContract(deployCode)
+      await expect(tx).to.be.rejectedWith(RevertError('ModuleSelfAuth#onlySelf: NOT_AUTHORIZED'))
+    })
+  })
+
+  describe('Transaction events', () => {
+    const transaction = {
+      delegateCall: false,
+      revertOnError: true,
+      gasLimit: optimalGasLimit,
+      target: ethers.constants.AddressZero,
+      value: ethers.constants.Zero,
+      data: []
+    }
+
+    it('Should emit TxExecuted event', async () => {
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, [transaction], networkId)).wait()
+      const log = tx.logs[1]
+      const txHash = ethers.utils.keccak256(encodeMetaTransactionsData(wallet.address, [transaction], networkId, 0))
+
+      expect(log.topics.length).to.equal(0)
+      expect(log.data).to.be.equal(txHash)
+    })
+
+    it('Should emit multiple TxExecuted events', async () => {
+      const tx = await (await signAndExecuteMetaTx(wallet, owner, [transaction, transaction], networkId)).wait()
+      const log1 = tx.events![1]
+      const log2 = tx.events![2]
+
+      const txHash = ethers.utils.keccak256(encodeMetaTransactionsData(wallet.address, [transaction, transaction], networkId, 0))
+
+      expect(log1.topics.length).to.equal(0)
+      expect(log1.data).to.be.equal(txHash)
+
+      expect(log2.topics.length).to.equal(0)
+      expect(log2.data).to.be.equal(txHash)
+    })
+  })
+
+  describe('Internal bundles', () => {
+    it('Should execute internal bundle', async () => {
+      const callReceiver1 = await new CallReceiverMock__factory().connect(signer).deploy()
+      const callReceiver2 = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const expected1 = await web3.utils.randomHex(552)
+      const expected2 = await web3.utils.randomHex(24)
+
+      const bundle = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit.div(ethers.constants.Two),
+          target: callReceiver1.address,
+          value: ethers.constants.Zero,
+          data: callReceiver1.interface.encodeFunctionData('testCall', [11, expected1])
+        },
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit.div(ethers.constants.Two),
+          target: callReceiver2.address,
+          value: ethers.constants.Zero,
+          data: callReceiver1.interface.encodeFunctionData('testCall', [12, expected2])
+        }
+      ]
+
+      const transaction = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('selfExecute', [bundle])
+        }
+      ]
+
+      await signAndExecuteMetaTx(wallet, owner, transaction, networkId)
+
+      expect((await callReceiver1.lastValA()).toNumber()).to.eq.BN(11)
+      expect((await callReceiver2.lastValA()).toNumber()).to.eq.BN(12)
+
+      expect((await callReceiver1.lastValB()).toString()).to.equal(expected1)
+      expect((await callReceiver2.lastValB()).toString()).to.equal(expected2)
+    })
+    it('Should execute multiple internal bundles', async () => {
+      const data = [
+        [
+          { i: 0, a: 142, b: 412 },
+          { i: 1, a: 123, b: 2 }
+        ],
+        [
+          { i: 2, a: 142, b: 2 },
+          { i: 3, a: 642, b: 33 },
+          { i: 4, a: 122, b: 12 },
+          { i: 5, a: 611, b: 53 }
+        ],
+        [{ i: 6, a: 2, b: 1 }],
+        []
+      ]
+
+      const contracts = await Promise.all(
+        data.flat().map(async () => await new CallReceiverMock__factory().connect(signer).deploy())
+      )
+      const expectedb = await Promise.all(data.flat().map(d => web3.utils.randomHex(d.b)))
+
+      const bundles = data.map(bundle => {
+        return bundle.map(obj => ({
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit.div(ethers.constants.Two),
+          value: ethers.constants.Zero,
+          target: contracts[obj.i].address,
+          data: contracts[obj.i].interface.encodeFunctionData('testCall', [obj.a, expectedb[obj.i]])
+        }))
+      })
+
+      const transactions = bundles.map(bundle => ({
+        delegateCall: false,
+        revertOnError: true,
+        gasLimit: optimalGasLimit,
+        target: wallet.address,
+        value: ethers.constants.Zero,
+        data: wallet.interface.encodeFunctionData('selfExecute', [bundle])
+      }))
+
+      await signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+
+      const lastValsA = await Promise.all(contracts.map((c: CallReceiverMock) => c.lastValA()))
+      const lastValsB = await Promise.all(contracts.map((c: CallReceiverMock) => c.lastValB()))
+
+      lastValsA.forEach((val, i) => expect(val.toNumber()).to.eq.BN(data.flat()[i].a))
+      lastValsB.forEach((val, i) => expect(val).to.equal(expectedb[i]))
+    })
+    it('Should execute nested internal bundles', async () => {
+      const callReceiver1 = await new CallReceiverMock__factory().connect(signer).deploy()
+      const callReceiver2 = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const expected1 = await web3.utils.randomHex(552)
+      const expected2 = await web3.utils.randomHex(24)
+
+      const bundle = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit.div(4),
+          target: callReceiver1.address,
+          value: ethers.constants.Zero,
+          data: callReceiver1.interface.encodeFunctionData('testCall', [11, expected1])
+        },
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit.div(4),
+          target: callReceiver2.address,
+          value: ethers.constants.Zero,
+          data: callReceiver1.interface.encodeFunctionData('testCall', [12, expected2])
+        }
+      ]
+
+      const nestedBundle = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit.div(2),
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('selfExecute', [bundle])
+        }
+      ]
+
+      const transaction = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('selfExecute', [nestedBundle])
+        }
+      ]
+
+      await signAndExecuteMetaTx(wallet, owner, transaction, networkId)
+
+      expect((await callReceiver1.lastValA()).toNumber()).to.eq.BN(11)
+      expect((await callReceiver2.lastValA()).toNumber()).to.eq.BN(12)
+
+      expect(await callReceiver1.lastValB()).to.equal(expected1)
+      expect(await callReceiver2.lastValB()).to.equal(expected2)
+    })
+    it('Should revert bundle without reverting transaction', async () => {
+      const callReceiver1 = await new CallReceiverMock__factory().connect(signer).deploy()
+      const callReceiver2 = await new CallReceiverMock__factory().connect(signer).deploy()
+      const callReceiver3 = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const expected1 = await web3.utils.randomHex(552)
+      const expected2 = await web3.utils.randomHex(24)
+      const expected3 = await web3.utils.randomHex(11)
+
+      const bundle = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: ethers.BigNumber.from(100000),
+          target: callReceiver1.address,
+          value: ethers.constants.Zero,
+          data: callReceiver1.interface.encodeFunctionData('testCall', [11, expected1])
+        },
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: ethers.BigNumber.from(100000),
+          target: callReceiver2.address,
+          value: ethers.constants.Zero,
+          data: callReceiver2.interface.encodeFunctionData('testCall', [12, expected2])
+        },
+        {
+          // This transaction will revert
+          // because Factory has no fallback
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: ethers.BigNumber.from(100000),
+          target: factory.address,
+          value: ethers.constants.Zero,
+          data: callReceiver1.interface.encodeFunctionData('testCall', [12, expected2])
+        }
+      ]
+
+      const transaction = [
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit,
+          target: wallet.address,
+          value: ethers.constants.Zero,
+          data: wallet.interface.encodeFunctionData('selfExecute', [bundle])
+        },
+        {
+          delegateCall: false,
+          revertOnError: false,
+          gasLimit: optimalGasLimit,
+          target: callReceiver3.address,
+          value: ethers.constants.Zero,
+          data: callReceiver3.interface.encodeFunctionData('testCall', [51, expected3])
+        }
+      ]
+
+      const tx = await signAndExecuteMetaTx(wallet, owner, transaction, networkId)
+
+      expect((await callReceiver1.lastValA()).toNumber()).to.eq.BN(0)
+      expect((await callReceiver2.lastValA()).toNumber()).to.eq.BN(0)
+      expect((await callReceiver3.lastValA()).toNumber()).to.eq.BN(51)
+
+      expect(await callReceiver1.lastValB()).to.equal('0x')
+      expect(await callReceiver2.lastValB()).to.equal('0x')
+      expect(await callReceiver3.lastValB()).to.equal(expected3)
+    })
+  })
+})
+
+// 0x000102010b1f05b9dd385e2683500bdfec03b53b0d9acb8f004700010001d4c175bfe46abb1138ad97ec9560aae4b745f0b2957986b3367dbaa3486e40497096518c964bad007597f625e2e978d1d6f5096ed2783cb8ceeb828088e81a811b020303
+// 0x000102010b1f05b9dd385e2683500bdfec03b53b0d9acb8f004700010001d4c175bfe46abb1138ad97ec9560aae4b745f0b2957986b3367dbaa3486e40497096518c964bad007597f625e2e978d1d6f5096ed2783cb8ceeb828088e81a811b0203
+//                                                       00010001d4c175bfe46abb1138ad97ec9560aae4b745f0b2957986b3367dbaa3486e40497096518c964bad007597f625e2e978d1d6f5096ed2783cb8ceeb828088e81a811b02
+//                                                               d4c175bfe46abb1138ad97ec9560aae4b745f0b2957986b3367dbaa3486e40497096518c964bad007597f625e2e978d1d6f5096ed2783cb8ceeb828088e81a811b02
diff --git a/forks/passport/tests/MultiCallUtils.spec.ts b/forks/passport/tests/MultiCallUtils.spec.ts
new file mode 100644
index 00000000..dafeed89
--- /dev/null
+++ b/forks/passport/tests/MultiCallUtils.spec.ts
@@ -0,0 +1,314 @@
+import { ethers as hardhat, web3 } from 'hardhat'
+import { ethers } from 'ethers'
+import { expect, b, RevertError } from './utils'
+
+import { MultiCallUtils, CallReceiverMock, MultiCallUtils__factory, CallReceiverMock__factory } from '../src'
+
+ethers.utils.Logger.setLogLevel(ethers.utils.Logger.levels.ERROR)
+
+const CallReceiverMockArtifact = artifacts.require('CallReceiverMock')
+
+contract('Multi call utils', (accounts: string[]) => {
+  let signer: ethers.Signer
+  let multiCall: MultiCallUtils
+  let callReceiver: CallReceiverMock
+
+  before(async () => {
+    signer = (await hardhat.getSigners())[0]
+    multiCall = await (new MultiCallUtils__factory()).connect(signer).deploy()
+    callReceiver = await (new CallReceiverMock__factory()).connect(signer).deploy()
+  })
+
+  beforeEach(async () => {
+    await callReceiver.setRevertFlag(false)
+  })
+
+  describe('Call multiple contracts', () => {
+    it('Should execute empty call', async () => {
+      const res = await multiCall.callStatic.multiCall([])
+
+      expect(res[0].length).to.equal(0)
+      expect(res[1].length).to.equal(0)
+    })
+    it('Should execute single call', async () => {
+      await callReceiver.testCall(5123, [])
+      const res = await multiCall.callStatic.multiCall([
+        {
+          delegateCall: false,
+          revertOnError: false,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('lastValA'),
+          gasLimit: 0,
+          value: 0
+        }
+      ])
+
+      expect(res[0].length).to.equal(1)
+      expect(res[1].length).to.equal(1)
+      expect(res[0][0]).to.be.true
+      expect(res[1][0]).to.be.equal(ethers.utils.defaultAbiCoder.encode(['uint256'], [5123]))
+    })
+    it('Should execute two calls', async () => {
+      const bytes = ethers.utils.randomBytes(422)
+
+      await callReceiver.testCall(55522, bytes)
+      const res = await multiCall.callStatic.multiCall([
+        {
+          delegateCall: false,
+          revertOnError: false,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('lastValA'),
+          gasLimit: 0,
+          value: 0
+        },
+        {
+          delegateCall: false,
+          revertOnError: false,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('lastValB'),
+          gasLimit: 0,
+          value: 0
+        }
+      ])
+
+      expect(res[0].length).to.equal(2)
+      expect(res[1].length).to.equal(2)
+      expect(res[0][0]).to.be.true
+      expect(res[1][0]).to.be.equal(ethers.utils.defaultAbiCoder.encode(['uint256'], [55522]))
+      expect(res[0][1]).to.be.true
+      expect(ethers.utils.defaultAbiCoder.decode(['bytes'], res[1][1])[0]).to.be.equal(ethers.utils.hexlify(bytes))
+    })
+    it('Should execute calls to multiple contracts', async () => {
+      const callReceiver2 = (await CallReceiverMockArtifact.new()) as CallReceiverMock
+      const bytes = ethers.utils.hexlify(ethers.utils.randomBytes(21))
+
+      await callReceiver.testCall(55522, bytes)
+      await callReceiver2.testCall(66623, [])
+
+      const res = await multiCall.callStatic.multiCall([
+        {
+          delegateCall: false,
+          revertOnError: false,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('lastValA'),
+          gasLimit: 0,
+          value: 0
+        },
+        {
+          delegateCall: false,
+          revertOnError: false,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('lastValB'),
+          gasLimit: 0,
+          value: 0
+        },
+        {
+          delegateCall: false,
+          revertOnError: false,
+          target: callReceiver2.address,
+          data: callReceiver.interface.encodeFunctionData('lastValA'),
+          gasLimit: 0,
+          value: 0
+        }
+      ])
+
+      expect(res[0].length).to.equal(3)
+      expect(res[1].length).to.equal(3)
+      expect(res[0][0]).to.be.true
+      expect(res[1][0]).to.be.equal(ethers.utils.defaultAbiCoder.encode(['uint256'], [55522]))
+      expect(res[0][1]).to.be.true
+      expect(ethers.utils.defaultAbiCoder.decode(['bytes'], res[1][1])[0]).to.be.equal(bytes)
+      expect(res[0][2]).to.be.true
+      expect(res[1][2]).to.be.equal(ethers.utils.defaultAbiCoder.encode(['uint256'], [66623]))
+    })
+    it('Return other calls even if single call fails', async () => {
+      const bytes = ethers.utils.randomBytes(422)
+
+      await callReceiver.testCall(55522, bytes)
+      await callReceiver.setRevertFlag(true)
+
+      const res = await multiCall.callStatic.multiCall([
+        {
+          delegateCall: false,
+          revertOnError: false,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('lastValA'),
+          gasLimit: 0,
+          value: 0
+        },
+        {
+          delegateCall: false,
+          revertOnError: false,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('testCall', [111, bytes]),
+          gasLimit: 0,
+          value: 0
+        }
+      ])
+
+      expect(res[0].length).to.equal(2)
+      expect(res[1].length).to.equal(2)
+      expect(res[0][0]).to.be.true
+      expect(res[1][0]).to.be.equal(ethers.utils.defaultAbiCoder.encode(['uint256'], [55522]))
+      expect(res[0][1]).to.be.false
+    })
+    it('Fail if call with revert on error fails', async () => {
+      const bytes = ethers.utils.randomBytes(422)
+
+      await callReceiver.testCall(55522, bytes)
+      await callReceiver.setRevertFlag(true)
+
+      const tx = multiCall.callStatic.multiCall([
+        {
+          delegateCall: false,
+          revertOnError: false,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('lastValA'),
+          gasLimit: 0,
+          value: 0
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('testCall', [111, bytes]),
+          gasLimit: 0,
+          value: 0
+        }
+      ])
+
+      await expect(tx).to.be.rejectedWith(RevertError('MultiCallUtils#multiCall: CALL_REVERTED'))
+    })
+    it('Fail if batch includes delegate call', async () => {
+      const bytes = ethers.utils.randomBytes(422)
+
+      await callReceiver.testCall(55522, bytes)
+
+      const tx = multiCall.callStatic.multiCall([
+        {
+          delegateCall: true,
+          revertOnError: false,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('lastValA'),
+          gasLimit: 0,
+          value: 0
+        },
+        {
+          delegateCall: false,
+          revertOnError: false,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('testCall', [111, bytes]),
+          gasLimit: 0,
+          value: 0
+        }
+      ])
+
+      await expect(tx).to.be.rejectedWith(RevertError('MultiCallUtils#multiCall: delegateCall not allowed'))
+    })
+    it('Fail if not enough gas for call', async () => {
+      const bytes = ethers.utils.randomBytes(422)
+
+      await callReceiver.testCall(55522, bytes)
+
+      const tx = multiCall.callStatic.multiCall([
+        {
+          delegateCall: false,
+          revertOnError: false,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('lastValA'),
+          gasLimit: 0,
+          value: 0
+        },
+        {
+          delegateCall: false,
+          revertOnError: false,
+          target: callReceiver.address,
+          data: callReceiver.interface.encodeFunctionData('testCall', [111, bytes]),
+          gasLimit: b(2)
+            .pow(256)
+            .sub(b(1)),
+          value: 0
+        }
+      ])
+
+      await expect(tx).to.be.rejectedWith(RevertError('MultiCallUtils#multiCall: NOT_ENOUGH_GAS'))
+    })
+    it('Should call globals', async () => {
+      const emptyAccount = ethers.Wallet.createRandom().address
+
+      await multiCall.callStatic.multiCall([])
+
+      const lastBlock = await web3.eth.getBlock('latest')
+
+      const txs = [
+        ['callBlockhash', [lastBlock.number - 1]],
+        ['callCoinbase', []],
+        ['callDifficulty', []],
+        ['callGasLimit', []],
+        ['callBlockNumber', []],
+        ['callTimestamp', []],
+        ['callGasLeft', []],
+        ['callGasPrice', []],
+        ['callOrigin', []],
+        ['callBalanceOf', [accounts[0]]],
+        ['callBalanceOf', [emptyAccount]],
+        ['callCodeSize', [accounts[0]]],
+        ['callCodeSize', [callReceiver.address]],
+        ['callCode', [accounts[0]]],
+        ['callCode', [callReceiver.address]],
+        ['callCodeHash', [accounts[0]]],
+        ['callCodeHash', [callReceiver.address]],
+        ['callChainId', []]
+      ].map(method => ({
+        delegateCall: false,
+        revertOnError: false,
+        target: multiCall.address,
+        value: 0,
+        gasLimit: 0,
+        data: multiCall.interface.encodeFunctionData(method[0] as any, method[1] as any)
+      }))
+
+      const res = await multiCall.callStatic.multiCall(txs)
+
+      const emptyBytes32 = ethers.utils.defaultAbiCoder.encode(['uint256'], ['0'])
+
+      expect(res[0].length).to.equal(txs.length)
+      expect(res[1].length).to.equal(txs.length)
+
+      // All calls must success
+      expect(res[0].reduce((a: boolean, c: boolean) => a && c)).to.be.true
+
+      expect(res[1][0]).to.not.equal(emptyBytes32, 'return block hash')
+
+      if (!process.env.COVERAGE) {
+        expect(res[1][1]).to.not.equal(emptyBytes32, 'return coinbase')
+        expect(res[1][2]).to.not.equal(emptyBytes32, 'return difficulty')
+      }
+
+      expect(res[1][3]).to.not.equal(emptyBytes32)
+      expect(res[1][4]).to.not.equal(emptyBytes32, 'return block number')
+      expect(res[1][5]).to.not.equal(emptyBytes32, 'return timestamp')
+      expect(res[1][6]).to.not.equal(emptyBytes32, 'return gas left')
+      // expect(res[1][7]).to.not.equal(emptyBytes32, 'return gas price')
+      expect(res[1][8]).to.not.equal(emptyBytes32, 'return origin')
+      expect(res[1][9]).to.not.equal(emptyBytes32, 'return balance of 0x')
+      expect(res[1][10]).to.equal(emptyBytes32, 'return balance of empty account')
+      expect(res[1][11]).to.equal(emptyBytes32, 'return code size of empty account')
+      expect(res[1][12]).to.not.equal(emptyBytes32, 'return code size of contract')
+
+      expect(ethers.utils.defaultAbiCoder.decode(['bytes'], res[1][13])[0]).to.equal('0x')
+
+      const codeSize = ethers.utils.defaultAbiCoder.decode(['uint256'], res[1][12])[0]
+      expect(ethers.utils.defaultAbiCoder.decode(['bytes'], res[1][14])[0].length).to.equal(
+        2 + codeSize * 2,
+        'return code of correct size'
+      )
+
+      expect(res[1][15]).to.not.equal(emptyBytes32)
+      expect(res[1][16]).to.not.equal(emptyBytes32)
+
+      const chainId = process.env.NET_ID ? parseInt(process.env.NET_ID) : await web3.eth.net.getId()
+      expect(ethers.utils.defaultAbiCoder.decode(['uint256'], res[1][17])[0].toNumber()).to.equal(chainId, 'return chain id')
+    })
+  })
+})
diff --git a/forks/passport/tests/RequireUtils.spec.ts b/forks/passport/tests/RequireUtils.spec.ts
new file mode 100644
index 00000000..921f5a52
--- /dev/null
+++ b/forks/passport/tests/RequireUtils.spec.ts
@@ -0,0 +1,395 @@
+import { ethers as hardhat, web3 } from 'hardhat'
+import { ethers } from 'ethers'
+import { expect, signAndExecuteMetaTx, RevertError, encodeImageHash, addressOf, encodeNonce, walletSign } from './utils'
+
+import {
+  MainModule,
+  Factory,
+  RequireUtils,
+  CallReceiverMock,
+  RequireFreshSigner,
+  Factory__factory,
+  MainModule__factory,
+  RequireUtils__factory,
+  RequireFreshSigner__factory,
+  CallReceiverMock__factory
+} from '../src'
+
+ethers.utils.Logger.setLogLevel(ethers.utils.Logger.levels.ERROR)
+
+function now(): number {
+  return Math.floor(Date.now() / 1000)
+}
+
+const optimalGasLimit = ethers.constants.Two.pow(21)
+
+contract('Require utils', (accounts: string[]) => {
+  let signer: ethers.Signer
+  let networkId: number
+
+  let factory: Factory
+  let mainModule: MainModule
+  let requireUtils: RequireUtils
+  let requireFreshSigner: RequireFreshSigner
+
+  let owner: ethers.Wallet
+  let wallet: MainModule
+  let salt: string
+
+  before(async () => {
+    signer = (await hardhat.getSigners())[0]
+    networkId = process.env.NET_ID ? parseInt(process.env.NET_ID) : await web3.eth.net.getId()
+
+    // Deploy wallet factory
+    factory = await new Factory__factory().connect(signer).deploy(await signer.getAddress(), await signer.getAddress())
+    // Deploy MainModule
+    mainModule = await new MainModule__factory().connect(signer).deploy(factory.address)
+    // Deploy expirable util
+    requireUtils = await new RequireUtils__factory().connect(signer).deploy(factory.address, mainModule.address)
+    // Deploy require fresh signer lib
+    requireFreshSigner = await new RequireFreshSigner__factory().connect(signer).deploy(requireUtils.address)
+  })
+
+  beforeEach(async () => {
+    owner = new ethers.Wallet(ethers.utils.randomBytes(32))
+    salt = encodeImageHash(1, [{ weight: 1, address: owner.address }])
+    await factory.deploy(mainModule.address, salt)
+    wallet = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+  })
+
+  const stubTxns = [
+    {
+      delegateCall: false,
+      revertOnError: true,
+      gasLimit: optimalGasLimit,
+      target: ethers.constants.AddressZero,
+      value: ethers.constants.Zero,
+      data: []
+    }
+  ]
+
+  describe('Require fresh signer', () => {
+    it('Should pass if signer is new', async () => {
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+      await signAndExecuteMetaTx(wallet, owner, stubTxns, networkId)
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: requireFreshSigner.address,
+          value: ethers.constants.Zero,
+          data: requireFreshSigner.interface.encodeFunctionData('requireFreshSigner', [owner.address])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+      ]
+
+      await signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+// TODO: investigate why this is failing with gas limits. 
+// This functionality is for  fast lookup of a wallet of a wallet based on its signer members, 
+// and indendent of required functionality (deployment, upgrading, transaction submission, 2-of-2 multi-sig ect), thus leaving it OOS for now is fine
+
+    it('Should fail if signer is not new', async () => {
+      const message = ethers.utils.hexlify(ethers.utils.randomBytes(96))
+      const digest = ethers.utils.keccak256(message)
+      const preSubDigest = ethers.utils.solidityPack(
+        ['string', 'uint256', 'address', 'bytes'],
+        ['\x19\x01', networkId, wallet.address, digest]
+      )
+
+      const signature = await walletSign(owner, preSubDigest)
+
+      await signAndExecuteMetaTx(
+        wallet,
+        owner,
+        [
+          {
+            delegateCall: false,
+            revertOnError: true,
+            gasLimit: ethers.constants.Zero,
+            target: requireUtils.address,
+            value: ethers.constants.Zero,
+            data: requireUtils.interface.encodeFunctionData('publishInitialSigners', [wallet.address, digest, 1, signature, true])
+          }
+        ],
+        networkId
+      )
+
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+      await signAndExecuteMetaTx(wallet, owner, stubTxns, networkId)
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: requireFreshSigner.address,
+          value: ethers.constants.Zero,
+          data: requireFreshSigner.interface.encodeFunctionData('requireFreshSigner', [owner.address])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+      ]
+
+      const tx = signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+      await expect(tx).to.be.rejectedWith(RevertError('RequireFreshSigner#requireFreshSigner: DUPLICATED_SIGNER'))
+    })
+  })
+  describe('Require min-nonce', () => {
+    it('Should pass nonce increased from self-wallet', async () => {
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+      await signAndExecuteMetaTx(wallet, owner, stubTxns, networkId)
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: requireUtils.address,
+          value: ethers.constants.Zero,
+          data: requireUtils.interface.encodeFunctionData('requireMinNonce', [wallet.address, ethers.constants.Two])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+      ]
+
+      await signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+    it('Should pass nonce increased from different wallet', async () => {
+      const owner2 = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt2 = encodeImageHash(1, [{ weight: 1, address: owner2.address }])
+      await factory.deploy(mainModule.address, salt2)
+      const wallet2 = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt2), signer)
+
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+      await signAndExecuteMetaTx(wallet2, owner2, stubTxns, networkId)
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: requireUtils.address,
+          value: ethers.constants.Zero,
+          data: requireUtils.interface.encodeFunctionData('requireMinNonce', [wallet2.address, ethers.constants.One])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+      ]
+
+      await signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+    it('Should fail if nonce is below required on different wallet', async () => {
+      const owner2 = new ethers.Wallet(ethers.utils.randomBytes(32))
+      const salt2 = encodeImageHash(1, [{ weight: 1, address: owner2.address }])
+      await factory.deploy(mainModule.address, salt2)
+      const wallet2 = await MainModule__factory.connect(addressOf(factory.address, mainModule.address, salt), signer)
+
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      try {
+        await signAndExecuteMetaTx(wallet2, owner2, stubTxns, networkId)
+      } catch (err) {
+        // silence exception from txn failure
+      }
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: requireUtils.address,
+          value: ethers.constants.Zero,
+          data: requireUtils.interface.encodeFunctionData('requireMinNonce', [wallet2.address, ethers.constants.Two])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+      ]
+
+      const tx = signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+      await expect(tx).to.be.rejectedWith(RevertError('RequireUtils#requireMinNonce: NONCE_BELOW_REQUIRED'))
+    })
+    it('Should fail if nonce is below required on self-wallet on a different space', async () => {
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+      await signAndExecuteMetaTx(wallet, owner, stubTxns, networkId)
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: requireUtils.address,
+          value: ethers.constants.Zero,
+          data: requireUtils.interface.encodeFunctionData('requireMinNonce', [
+            wallet.address,
+            encodeNonce(ethers.constants.Two, ethers.constants.One)
+          ])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+      ]
+
+      const tx = signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+      await expect(tx).to.be.rejectedWith(RevertError('RequireUtils#requireMinNonce: NONCE_BELOW_REQUIRED'))
+    })
+    it('Should fail if nonce is below required on self-wallet', async () => {
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: requireUtils.address,
+          value: ethers.constants.Zero,
+          data: requireUtils.interface.encodeFunctionData('requireMinNonce', [wallet.address, ethers.constants.Two])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+      ]
+      const tx = signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+      await expect(tx).to.be.rejectedWith(RevertError('RequireUtils#requireMinNonce: NONCE_BELOW_REQUIRED'))
+    })
+  })
+  describe('Expirable transactions', () => {
+    it('Should pass if non expired', async () => {
+      await requireUtils.requireNonExpired(now() + 1480)
+    })
+    it('Should fail if expired', async () => {
+      const tx = requireUtils.requireNonExpired(now() - 1)
+      await expect(tx).to.be.rejectedWith(RevertError('RequireUtils#requireNonExpired: EXPIRED'))
+    })
+    it('Should pass bundle if non expired', async () => {
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: requireUtils.address,
+          value: ethers.constants.Zero,
+          data: requireUtils.interface.encodeFunctionData('requireNonExpired', [now() + 1480])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+      ]
+
+      await signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+      expect((await callReceiver.lastValA()).toNumber()).to.eq.BN(valA)
+      expect(await callReceiver.lastValB()).to.equal(valB)
+    })
+    it('Should fail bundle if expired', async () => {
+      const callReceiver = await new CallReceiverMock__factory().connect(signer).deploy()
+
+      const valA = 5423
+      const valB = web3.utils.randomHex(120)
+
+      const transactions = [
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: requireUtils.address,
+          value: ethers.constants.Zero,
+          data: requireUtils.interface.encodeFunctionData('requireNonExpired', [now() - 1])
+        },
+        {
+          delegateCall: false,
+          revertOnError: true,
+          gasLimit: optimalGasLimit,
+          target: callReceiver.address,
+          value: ethers.constants.Zero,
+          data: callReceiver.interface.encodeFunctionData('testCall', [valA, valB])
+        }
+      ]
+
+      const tx = signAndExecuteMetaTx(wallet, owner, transactions, networkId)
+      await expect(tx).to.be.rejectedWith(RevertError('RequireUtils#requireNonExpired: EXPIRED'))
+    })
+  })
+})
diff --git a/forks/passport/tests/utils/contract.ts b/forks/passport/tests/utils/contract.ts
new file mode 100644
index 00000000..d7b3921e
--- /dev/null
+++ b/forks/passport/tests/utils/contract.ts
@@ -0,0 +1,155 @@
+// Build ethers.Contract instances from ABI JSON files generated by truffle.
+//
+// adapted this utility from the handy work by the counterfactual team at:
+// https://github.com/counterfactual/monorepo/blob/d9be8524a691c45b6aac1b5e1cf2ff81059203df/packages/contracts/utils/contract.ts
+
+import { ethers } from 'ethers'
+
+interface NetworkMapping {
+  [networkId: number]: { address: string }
+}
+
+interface BuildArtifact {
+  readonly contractName: string
+  readonly abi: any[]
+  readonly bytecode: string
+  readonly networks: NetworkMapping
+}
+
+/**
+ * Convenience class for an undeployed contract i.e. only the ABI and bytecode.
+ */
+export class AbstractContract {
+  /**
+   * Load build artifact by name into an abstract contract
+   * @example
+   *  const CountingApp = AbstractContract.fromArtifactName("CountingApp", { StaticCall })
+   * @param artifactName The name of the artifact to load
+   * @param links Optional AbstractContract libraries to link.
+   * @returns Truffle artifact wrapped in an AbstractContract.
+   */
+  public static async fromArtifactName(
+    artifactName: string,
+    links?: { [name: string]: Promise<AbstractContract> }
+  ): Promise<AbstractContract> {
+    // these ABI JSON files are generated by truffle
+    const contract: BuildArtifact = await import(`../../build/contracts/${artifactName}.json`)
+    return AbstractContract.fromBuildArtifact(contract, links, artifactName)
+  }
+
+  /**
+   * Wrap build artifact in abstract contract
+   * @param buildArtifact Truffle contract to wrap
+   * @param links Optional AbstractContract libraries to link.
+   * @returns Truffle artifact wrapped in an AbstractContract.
+   */
+  public static async fromBuildArtifact(
+    buildArtifact: BuildArtifact,
+    links?: { [name: string]: Promise<AbstractContract> },
+    artifactName: string = 'UntitledContract'
+  ): Promise<AbstractContract> {
+    return new AbstractContract(
+      buildArtifact.abi,
+      buildArtifact.bytecode,
+      buildArtifact.networks,
+      links,
+      artifactName
+    )
+  }
+
+  public static async getNetworkID(wallet: ethers.Wallet): Promise<number> {
+    return (await wallet.provider.getNetwork()).chainId
+  }
+
+  /**
+   * @param abi ABI of the abstract contract
+   * @param bytecode Binary of the abstract contract
+   * @param networks Network mapping of deployed addresses
+   * @param links
+   * @param contractName
+   */
+  constructor(
+    readonly abi: string[] | string,
+    readonly bytecode: string,
+    readonly networks: NetworkMapping,
+    readonly links?: { [contractName: string]: Promise<AbstractContract> },
+    readonly contractName?: string
+  ) {}
+
+  /**
+   * Get the deployed singleton instance of this abstract contract, if it exists
+   * @param Signer (with provider) to use for contract calls
+   * @throws Error if AbstractContract has no deployed address
+   */
+  public async getDeployed(wallet: ethers.Wallet): Promise<ethers.Contract> {
+    if (!wallet.provider) {
+      throw new Error('Signer requires provider')
+    }
+    const networkId = (await wallet.provider.getNetwork()).chainId
+    const address = this.getDeployedAddress(networkId)
+    return new ethers.Contract(address, this.abi, wallet)
+  }
+
+  /**
+   * Deploy new instance of contract
+   * @param wallet Wallet (with provider) to use for contract calls
+   * @param args Optional arguments to pass to contract constructor
+   * @returns New contract instance
+   */
+  public async deploy(wallet: ethers.Wallet, args?: any[]): Promise<ethers.Contract> {
+    if (!wallet.provider) {
+      throw new Error('Signer requires provider')
+    }
+
+    const networkId = (await wallet.provider.getNetwork()).chainId
+    const bytecode = (await this.links)
+      ? await this.generateLinkedBytecode(networkId)
+      : this.bytecode
+    const contractFactory = new ethers.ContractFactory(
+      this.abi,
+      bytecode,
+      wallet
+    )
+    return contractFactory.deploy(...(args || []))
+  }
+
+  /**
+   * Connect to a deployed instance of this abstract contract
+   * @param signer Signer (with provider) to use for contract calls
+   * @param address Address of deployed instance to connect to
+   * @returns Contract instance
+   */
+  public async connect(
+    signer: ethers.Signer,
+    address: string
+  ): Promise<ethers.Contract> {
+    return new ethers.Contract(address, this.abi, signer)
+  }
+
+  public getDeployedAddress(networkId: number): string {
+    const info = this.networks[networkId]
+    if (!info) {
+      throw new Error(
+        `Abstract contract ${
+          this.contractName
+        } not deployed on network ${networkId}`
+      )
+    }
+    return info.address
+  }
+
+  public async generateLinkedBytecode(networkId: number): Promise<string> {
+    if (this.links === undefined) {
+      throw new Error('Nothing to link')
+    }
+    let bytecode = this.bytecode
+    for (const name of Object.keys(this.links)) {
+      const library = this.links[name]
+      const regex = new RegExp(`__${name}_+`, 'g')
+      const address = (await library).getDeployedAddress(networkId)
+      const addressHex = address.replace('0x', '')
+      bytecode = bytecode.replace(regex, addressHex)
+    }
+    return bytecode
+  }
+}
diff --git a/forks/passport/tests/utils/helpers.ts b/forks/passport/tests/utils/helpers.ts
new file mode 100644
index 00000000..27007558
--- /dev/null
+++ b/forks/passport/tests/utils/helpers.ts
@@ -0,0 +1,453 @@
+import { ethers , BytesLike, BigNumberish } from 'ethers'
+import { MainModule } from 'src/gen/typechain'
+export const ZERO_ADDRESS = '0x0000000000000000000000000000000000000000'
+
+// createTestWallet creates a new wallet
+export const createTestWallet = (web3: any, addressIndex: number = 0) => {
+  const provider = new Web3DebugProvider(web3.currentProvider)
+
+  const wallet = ethers.Wallet
+    .fromMnemonic(process.env.npm_package_config_mnemonic!, `m/44'/60'/0'/0/${addressIndex}`)
+    .connect(provider)
+
+  const signer = provider.getSigner(addressIndex)
+
+  return { wallet, provider, signer }
+}
+
+// // Check if tx was Reverted with specified message
+// export function RevertError(errorMessage?: string) {
+//   // let prefix = 'VM Exception while processing transaction: revert'
+//   // return errorMessage ? `${prefix + ' ' + errorMessage}` : prefix
+//   return `VM Exception while processing transaction: reverted with reason string '${errorMessage}`
+// }
+
+// Check if tx was Reverted with specified message
+export function RevertError(errorMessage?: string) {
+  if (!errorMessage) {
+    return /Transaction reverted and Hardhat couldn't infer the reason/
+  } else {
+    // return new RegExp(`${errorMessage}`)
+    return new RegExp(`VM Exception while processing transaction: reverted with reason string ["']${errorMessage}["']`)
+  }
+}
+
+export function RevertOutOfGasError() {
+  return /out of gas/
+}
+
+export function RevertCallException() {
+  return /call revert exception/
+}
+
+export interface JSONRPCRequest {
+  jsonrpc: string
+  id: number
+  method: any
+  params: any
+}
+
+export class Web3DebugProvider extends ethers.providers.JsonRpcProvider {
+
+  public reqCounter = 0
+  public reqLog: JSONRPCRequest[] = []
+
+  readonly _web3Provider: ethers.providers.ExternalProvider
+  private _sendAsync: (request: any, callback: (error: any, response: any) => void) => void
+
+  constructor(web3Provider: ethers.providers.ExternalProvider, network?: ethers.providers.Networkish) {
+      // HTTP has a host; IPC has a path.
+      super(web3Provider.host || web3Provider.path || '', network)
+
+      if (web3Provider) {
+        if (web3Provider.sendAsync) {
+          this._sendAsync = web3Provider.sendAsync.bind(web3Provider)
+        } else if (web3Provider.send) {
+          this._sendAsync = web3Provider.send.bind(web3Provider)
+        }
+      }
+
+      if (!web3Provider || !this._sendAsync) {
+        ethers.logger.throwError(
+          'invalid web3Provider',
+          ethers.errors.INVALID_ARGUMENT,
+          { arg: 'web3Provider', value: web3Provider }
+        )
+      }
+
+      ethers.utils.defineReadOnly(this, '_web3Provider', web3Provider)
+  }
+
+  send(method: string, params: any): Promise<any> {
+
+    this.reqCounter++
+
+    return new Promise((resolve, reject) => {
+      let request = {
+        method: method,
+        params: params,
+        id: this.reqCounter,
+        jsonrpc: '2.0'
+      } as JSONRPCRequest
+      this.reqLog.push(request)
+
+      this._sendAsync(request, function(error, result) {
+        if (error) {
+          reject(error)
+          return
+        }
+
+        if (result.error) {
+          // @TODO: not any
+          let error: any = new Error(result.error.message)
+          error.code = result.error.code
+          error.data = result.error.data
+          reject(error)
+          return
+        }
+
+        resolve(result.result)
+      })
+    })
+  }
+
+  getPastRequest(reverseIndex: number = 0): JSONRPCRequest {
+    if (this.reqLog.length === 0) {
+      return { jsonrpc: '2.0', id: 0, method: null, params: null }
+    }
+    return this.reqLog[this.reqLog.length-reverseIndex-1]
+  }
+
+}
+
+// Take a message, hash it and sign it with ETH_SIGN SignatureType
+export async function ethSign(wallet: ethers.Wallet, message: string | Uint8Array, hashed = false) {
+  let hash = hashed ? message : ethers.utils.keccak256(message)
+  let hashArray = ethers.utils.arrayify(hash)
+  let ethsigNoType = await wallet.signMessage(hashArray)
+  return ethsigNoType.endsWith('03') || ethsigNoType.endsWith('02') ? ethsigNoType : ethsigNoType + '02'
+}
+
+export const MetaTransactionsType = `tuple(
+  bool delegateCall,
+  bool revertOnError,
+  uint256 gasLimit,
+  address target,
+  uint256 value,
+  bytes data
+)[]`
+
+export function encodeMessageData(
+  owner: string,
+  message: string,
+  networkId: BigNumberish
+): string {
+  return encodeMessageSubDigest(owner, ethers.utils.keccak256(message), networkId)
+}
+
+export function encodeMessageSubDigest(
+  owner: string,
+  digest: string,
+  networkId: BigNumberish
+): string {
+  return ethers.utils.solidityPack(
+    ['string', 'uint256', 'address', 'bytes32'],
+    ['\x19\x01', networkId, owner, digest]
+  )
+}
+
+export function encodeMetaTransactionsData(
+  owner: string,
+  txs: {
+    delegateCall: boolean;
+    revertOnError: boolean;
+    gasLimit: BigNumberish;
+    target: string;
+    value: BigNumberish;
+    data: BytesLike;
+  }[],
+  networkId: BigNumberish,
+  nonce: BigNumberish
+): string {
+  const transactions = ethers.utils.defaultAbiCoder.encode(['uint256', MetaTransactionsType], [nonce, txs])
+  return encodeMessageData(owner, transactions, networkId)
+}
+
+export async function walletSign(
+  owner: ethers.Wallet,
+  message: string,
+  forceDynamicSize: boolean = false
+) {
+  return walletMultiSign([{ weight: 1, owner: owner }], 1, message, forceDynamicSize)
+}
+
+export function compareAddr(a: string | ethers.Wallet, b: string | ethers.Wallet) {
+  const addrA = typeof a === 'string' ? a : a.address
+  const addrB = typeof b === 'string' ? b : b.address
+
+  const bigA = ethers.BigNumber.from(addrA)
+  const bigB = ethers.BigNumber.from(addrB)
+
+  if (bigA.lt(bigB)) {
+    return -1
+  } else if (bigA.eq(bigB)) {
+    return 0
+  } else {
+    return 1
+  }
+}
+
+export async function walletMultiSign(
+  accounts: {
+    weight: BigNumberish,
+    owner: string | ethers.Wallet,
+    signature?: string
+  }[],
+  threshold: BigNumberish,
+  message: string,
+  forceDynamicSize: boolean = false,
+  hashed = false
+) {
+  const sorted = accounts.sort((a, b) => compareAddr(a.owner, b.owner))
+  const accountBytes = await Promise.all(
+    sorted.map(async (a) => {
+      if (typeof a.owner === 'string' && !a.signature) {
+        return ethers.utils.solidityPack(
+          ['uint8', 'uint8', 'address'],
+          [1, a.weight, a.owner]
+        )
+      } else {
+        const signature = ethers.utils.arrayify(a.signature ? a.signature as string : await ethSign(a.owner as ethers.Wallet, message, hashed))
+        if (forceDynamicSize || signature.length !== 66) {
+          const address = typeof a.owner === 'string' ? a.owner : a.owner.address
+          return ethers.utils.solidityPack(
+            ['uint8', 'uint8', 'address', 'uint16', 'bytes'],
+            [2, a.weight, address, signature.length, signature]
+          )
+        } else {
+          return ethers.utils.solidityPack(
+            ['uint8', 'uint8', 'bytes'],
+            [0, a.weight, signature]
+          )
+        }
+      }
+    })
+  )
+
+  return ethers.utils.solidityPack(
+    ['uint16', ...Array(accounts.length).fill('bytes')],
+    [threshold, ...accountBytes]
+  )
+}
+
+export async function signAndEncodeMetaTxn(
+  wallet: MainModule,
+  owner: ethers.Wallet,
+  txs: {
+    delegateCall: boolean;
+    revertOnError: boolean;
+    gasLimit: BigNumberish;
+    target: string;
+    value: BigNumberish;
+    data: BytesLike;
+  }[],
+  networkId: BigNumberish,
+  nonce: BigNumberish | undefined = undefined,
+  forceDynamicSize: boolean = false
+) {
+  return multiSignAndEncodeMetaTxn(
+    wallet,
+    [{weight: 1, owner }],
+    1,
+    txs,
+    networkId,
+    nonce,
+    forceDynamicSize
+  )
+}
+
+export async function multiSignAndEncodeMetaTxn(
+  wallet: MainModule,
+  accounts: {
+    weight: BigNumberish,
+    owner: string | ethers.Wallet
+  }[],
+  threshold: BigNumberish,
+  txs: {
+    delegateCall: boolean;
+    revertOnError: boolean;
+    gasLimit: BigNumberish;
+    target: string;
+    value: BigNumberish;
+    data: BytesLike;
+  }[],
+  networkId: BigNumberish,
+  nonce: BigNumberish | undefined = undefined,
+  forceDynamicSize: boolean = false
+): Promise<string> {
+  if (!nonce) nonce = await nextNonce(wallet)
+  const signature = await multiSignMetaTransactions(wallet, accounts, threshold, txs, networkId, nonce, forceDynamicSize)
+  return wallet.interface.encodeFunctionData('execute', [txs, nonce, signature])
+}
+
+export async function multiSignMetaTransactions(
+  wallet: MainModule,
+  accounts: {
+    weight: BigNumberish,
+    owner: string | ethers.Wallet
+  }[],
+  threshold: BigNumberish,
+  txs: {
+    delegateCall: boolean;
+    revertOnError: boolean;
+    gasLimit: BigNumberish;
+    target: string;
+    value: BigNumberish;
+    data: BytesLike;
+  }[],
+  networkId: BigNumberish,
+  nonce: BigNumberish,
+  forceDynamicSize: boolean = false
+) {
+  const data = encodeMetaTransactionsData(wallet.address, txs, networkId, nonce)
+  return walletMultiSign(accounts, threshold, data, forceDynamicSize)
+}
+
+export async function nextNonce(wallet: MainModule) {
+  return (await wallet.nonce()).toNumber()
+}
+
+export async function signAndExecuteMetaTx(
+  wallet: MainModule,
+  owner: ethers.Wallet,
+  txs: {
+    delegateCall: boolean;
+    revertOnError: boolean;
+    gasLimit: BigNumberish;
+    target: string;
+    value: BigNumberish;
+    data: BytesLike;
+  }[],
+  networkId: BigNumberish,
+  nonce: BigNumberish | undefined = undefined,
+  forceDynamicSize: boolean = false
+) {
+  return multiSignAndExecuteMetaTx(
+    wallet,
+    [{ weight: 1, owner: owner }],
+    1,
+    txs,
+    networkId,
+    nonce,
+    forceDynamicSize
+  )
+}
+
+export async function multiSignAndExecuteMetaTx(
+  wallet: MainModule,
+  accounts: {
+    weight: BigNumberish,
+    owner: string | ethers.Wallet
+  }[],
+  threshold: BigNumberish,
+  txs: {
+    delegateCall: boolean;
+    revertOnError: boolean;
+    gasLimit: BigNumberish;
+    target: string;
+    value: BigNumberish;
+    data: BytesLike;
+  }[],
+  networkId: BigNumberish,
+  nonce: BigNumberish | undefined = undefined,
+  forceDynamicSize: boolean = false
+) {
+  if (!nonce) nonce = await nextNonce(wallet)
+  const signature = await multiSignMetaTransactions(wallet, accounts, threshold, txs, networkId, nonce, forceDynamicSize)
+  return wallet.execute(txs, nonce, signature)
+}
+
+export function encodeImageHash(
+  threshold: BigNumberish,
+  accounts: {
+    weight: BigNumberish
+    address: string
+  }[]
+) {
+  const sorted = accounts.sort((a, b) => compareAddr(a.address, b.address))
+  let imageHash = ethers.utils.solidityPack(['uint256'], [threshold])
+
+  sorted.forEach((a) => 
+    imageHash = ethers.utils.keccak256(
+      ethers.utils.defaultAbiCoder.encode(
+        ['bytes32', 'uint8', 'address'],
+        [imageHash, a.weight, a.address]
+      )
+    )
+  )
+
+  return imageHash
+}
+
+export function encodeNonce(space: BigNumberish, nonce: BigNumberish) {
+  const shiftedSpace = ethers.BigNumber.from(space).mul(ethers.constants.Two.pow(96))
+  return ethers.BigNumber.from(nonce).add(shiftedSpace)
+}
+
+export function moduleStorageKey(key: string, subkey?: string): string {
+  if (!subkey) {
+    return key;
+  }
+
+  return ethers.utils.keccak256(
+    ethers.utils.defaultAbiCoder.encode(
+      ['bytes32', 'bytes32'],
+      [moduleStorageKey(key), subkey]
+    )
+  )
+}
+
+function xor(a, b) {
+  if (!Buffer.isBuffer(a)) a = Buffer.from(ethers.utils.arrayify(a))
+  if (!Buffer.isBuffer(b)) b = Buffer.from(ethers.utils.arrayify(b))
+  return ethers.utils.hexlify(a.map((v: number, i: number) => v ^ b[i]))
+}
+
+export function interfaceIdOf(int: ethers.utils.Interface): string {
+  const signatures = Object.keys(int.functions)
+    .filter((k) => k.indexOf('(') !== -1)
+    .map((k) => int.getSighash(int.functions[k]))
+
+  return signatures.reduce((p, c) => xor(p, c))
+}
+
+
+// This bytecode must precisely match that in src/contracts/Wallet.sol
+// Original Sequence wallet proxy
+// export const WALLET_CODE = '0x603a600e3d39601a805130553df3363d3d373d3d3d363d30545af43d82803e903d91601857fd5bf3'
+// Solidity wallet proxy with PROXY_getImplementation
+//export const WALLET_CODE = '0x608060405234801561001057600080fd5b5060405161029f38038061029f8339818101604052810190610032919061009e565b803055506100cb565b600080fd5b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b600061006b82610040565b9050919050565b61007b81610060565b811461008657600080fd5b50565b60008151905061009881610072565b92915050565b6000602082840312156100b4576100b361003b565b5b60006100c284828501610089565b91505092915050565b6101c5806100da6000396000f3fe6080604052600436106100225760003560e01c806390611127146100a857610076565b36610076573373ffffffffffffffffffffffffffffffffffffffff16347f606834f57405380c4fb88d1f4850326ad3885f014bab3b568dfbf7a041eef73860405161006c90610113565b60405180910390a3005b60006100806100d3565b90503660008037600080366000845af43d6000803e80600081146100a3573d6000f35b3d6000fd5b3480156100b457600080fd5b506100bd6100d3565b6040516100ca9190610174565b60405180910390f35b60003054905090565b600082825260208201905092915050565b50565b60006100fd6000836100dc565b9150610108826100ed565b600082019050919050565b6000602082019050818103600083015261012c816100f0565b9050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b600061015e82610133565b9050919050565b61016e81610153565b82525050565b60006020820190506101896000830184610165565b9291505056fea2646970667358221220d43fa02972046db2bc81804ebf600d5b46b97e55c738ea899a28224e111b588564736f6c63430008110033'
+// Yul wallet proxy with PROXY_getImplementation
+export const WALLET_CODE = '0x6054600f3d396034805130553df3fe63906111273d3560e01c14602b57363d3d373d3d3d3d369030545af43d82803e156027573d90f35b3d90fd5b30543d5260203df3'
+
+export function addressOf(
+  factory: string,
+  mainModule: string,
+  imageHash: string
+): string {
+  const codeHash = ethers.utils.keccak256(
+    ethers.utils.solidityPack(
+      ['bytes', 'bytes32'],
+      [WALLET_CODE, ethers.utils.hexZeroPad(mainModule, 32)]
+    )
+  )
+
+  const hash = ethers.utils.keccak256(
+    ethers.utils.solidityPack(
+      ['bytes1', 'address', 'bytes32', 'bytes32'],
+      ['0xff', factory, imageHash, codeHash]
+    )
+  )
+
+  return ethers.utils.getAddress(ethers.utils.hexDataSlice(hash, 12))
+}
diff --git a/forks/passport/tests/utils/index.ts b/forks/passport/tests/utils/index.ts
new file mode 100644
index 00000000..68613fd9
--- /dev/null
+++ b/forks/passport/tests/utils/index.ts
@@ -0,0 +1,22 @@
+import * as chai from 'chai'
+import chaiAsPromised from 'chai-as-promised'
+import chaiString from 'chai-string'
+import { ethers } from 'ethers'
+
+import BN = require('bn.js') // used by web3
+
+export * from './contract'
+export * from './helpers'
+
+const BigNumber = ethers.BigNumber
+export { BigNumber }
+
+export const { assert, expect } = chai
+  .use(chaiString)
+  .use(chaiAsPromised)
+  .use(require('chai-bignumber')(BigNumber)) // Used by ethers.js & waffle
+  .use(require('bn-chai')(BN))       // Used by Web3 & truffle
+
+export function b(raw: ethers.BigNumberish): ethers.BigNumber {
+  return ethers.BigNumber.from(raw)
+}
diff --git a/forks/passport/tsconfig.json b/forks/passport/tsconfig.json
new file mode 100644
index 00000000..3915d06c
--- /dev/null
+++ b/forks/passport/tsconfig.json
@@ -0,0 +1,44 @@
+{
+  "compilerOptions": {
+    "target": "es2020",
+    "module": "commonjs",
+    "moduleResolution": "node",
+    "sourceMap": true,
+    "allowSyntheticDefaultImports": true,
+    "resolveJsonModule": true,
+    "downlevelIteration": true,
+    "removeComments": false,
+    "skipLibCheck": true,
+
+    "strictNullChecks": true,
+    "noImplicitUseStrict": true,
+    "noImplicitAny": false,
+    "noImplicitReturns": true,
+    "noImplicitThis": true,
+    "noUnusedParameters": false,
+    "noErrorTruncation": true,
+    "esModuleInterop": true,
+
+    "experimentalDecorators": true,
+    "forceConsistentCasingInFileNames": true,
+    "allowJs": false,
+    "checkJs": false,
+
+    "baseUrl": ".",
+    "outDir": "build",
+    "lib": ["es2020", "dom"],
+
+    "typeRoots": [
+      "./node_modules/@types"
+    ]
+  },
+
+  "include": [
+    "./hardhat.config.ts", "typings", "tests", "utils"
+  ],
+
+  "exclude": [
+    "node_modules",
+    "dist"
+  ]
+}
diff --git a/forks/passport/typings/chai-bignumber.d.ts b/forks/passport/typings/chai-bignumber.d.ts
new file mode 100644
index 00000000..8a0080c8
--- /dev/null
+++ b/forks/passport/typings/chai-bignumber.d.ts
@@ -0,0 +1,16 @@
+/// <reference types="chai" />
+
+declare module 'chai-bignumber' {
+  function chaiBignumber(bignumber: any): (chai: any, utils: any) => void
+
+  namespace chaiBignumber {}
+
+  export = chaiBignumber
+}
+
+declare namespace Chai {
+  // For BDD API
+  interface Assertion extends LanguageChains, NumericComparison, TypeComparison {
+    bignumber: Assertion
+  }
+}
diff --git a/forks/passport/typings/chai-bn.ts b/forks/passport/typings/chai-bn.ts
new file mode 100644
index 00000000..fdf499cd
--- /dev/null
+++ b/forks/passport/typings/chai-bn.ts
@@ -0,0 +1,19 @@
+/* eslint-disable */
+/// <reference types="chai" />
+
+declare module 'chai-bn' {
+  function chaiBN(bignumber: any): (chai: any, utils: any) => void
+
+  namespace chaiBN {}
+
+  export = chaiBN
+}
+
+declare namespace Chai {
+  interface Equal {
+    BN: any
+  }
+  interface NumberComparer {
+    BN: any
+  }
+}
diff --git a/forks/passport/typings/truffle.d.ts b/forks/passport/typings/truffle.d.ts
new file mode 100644
index 00000000..360a1a7c
--- /dev/null
+++ b/forks/passport/typings/truffle.d.ts
@@ -0,0 +1,12 @@
+declare module 'truffle' {
+  import * as truffle from 'truffle-contract'
+
+  interface ArtifactsGlobal {
+    require<A>(name: string): truffle.TruffleContract<A>
+  }
+
+  global {
+    function contract(name: string, callback: (accounts: Array<string>) => void): void
+    const artifacts: ArtifactsGlobal
+  }
+}
diff --git a/forks/passport/utils/config-loader.ts b/forks/passport/utils/config-loader.ts
new file mode 100644
index 00000000..d59135ea
--- /dev/null
+++ b/forks/passport/utils/config-loader.ts
@@ -0,0 +1,151 @@
+import * as dotenv from 'dotenv'
+import * as path from 'path'
+import { HttpNetworkConfig } from 'hardhat/types'
+import { ethers } from 'ethers'
+
+type EthereumNetworksTypes = 'rinkeby' | 'sepolia' |'ropsten' | 'kovan' | 'goerli' | 'mainnet' | 'mumbai' | 'matic' | 'arbitrum' | 'arbitrum-testnet' | 'optimism' | 'metis' | 'nova' | 'avalanche' | 'avalanche-testnet'
+
+export const getEnvConfig = (env: string) => {
+  const envFile = path.resolve(__dirname, `../config/${env}.env`)
+  const envLoad = dotenv.config({ path: envFile })
+
+  if (envLoad.error) {
+    console.warn('No config found, using default')
+    return { ETH_MNEMONIC: ethers.Wallet.createRandom().mnemonic.phrase }
+  }
+
+  return envLoad.parsed || {}
+}
+
+export const networkGasMultiplier = (network: EthereumNetworksTypes): number => {
+  switch (network) {
+    case 'arbitrum-testnet':
+    case 'arbitrum':
+      return 5
+
+    default:
+      return 1
+  }
+}
+
+export const networkRpcUrl = (network: EthereumNetworksTypes): string => {
+  const config = getEnvConfig('PROD')
+
+  switch (network) {
+    case 'mumbai':
+      return 'https://rpc-mumbai.matic.today/'
+
+    case 'matic':
+      return 'https://nodes.sequence.app/matic'
+
+    case 'arbitrum-testnet':
+      return 'https://rinkeby.arbitrum.io/rpc'
+    
+    case 'arbitrum':
+      return 'https://arb1.arbitrum.io/rpc'
+
+    case 'optimism':
+      return 'https://mainnet.optimism.io'
+
+    case 'metis':
+      return 'https://andromeda.metis.io/?owner=1088'
+
+    case 'nova':
+      return 'https://nova.arbitrum.io/rpc'
+
+    case 'avalanche':
+      return 'https://nodes.sequence.app/avalanche'
+
+    case 'avalanche-testnet':
+      return 'https://nodes.sequence.app/avalanche-testnet'
+    
+    case 'sepolia':
+        return 'https://rpc.sepolia.org'
+
+    default:
+      return `https://${network}.infura.io/v3/${config['INFURA_API_KEY']}`
+  }
+}
+
+export const networkChainId = (network: EthereumNetworksTypes): number => {
+  const config = getEnvConfig('PROD')
+
+  switch (network) {
+    case 'mumbai':
+      return 80001
+    
+    case 'ropsten':
+      return 3
+
+    case 'matic':
+      return 137
+
+    case 'arbitrum-testnet':
+      return 421611
+    
+    case 'arbitrum':
+      return 42161
+
+    case 'rinkeby':
+      return 4
+
+    case 'goerli':
+      return 5
+
+    case 'mainnet':
+      return 1
+
+    case 'kovan':
+      return 42
+
+    case 'optimism':
+      return 10
+
+    case 'metis':
+      return 1088
+
+    case 'nova':
+      return 42170
+
+    case 'avalanche':
+      return 43114
+
+    case 'avalanche-testnet':
+      return 43113
+
+    case `sepolia`:
+      return 11155111
+  }
+}
+
+export const etherscanKey = (network: EthereumNetworksTypes): string => {
+  // return process.env.ETHERSCAN_API_KEY!;
+  const config = getEnvConfig('PROD')
+
+  if (network === 'mainnet') {
+    return config['ETHERSCAN']
+  } else {
+    return config[`ETHERSCAN_${network.toUpperCase()}`]
+  }
+}
+
+export const networkConfig = (network: EthereumNetworksTypes): HttpNetworkConfig & { etherscan?: string } => {
+  const config = getEnvConfig('PROD')
+  return {
+    url: networkRpcUrl(network),
+    chainId: networkChainId(network),
+    accounts: {
+      mnemonic: config['ETH_MNEMONIC'],
+      initialIndex: 0,
+      count: 10,
+      path: `m/44'/60'/0'/0`
+    },
+    // accounts: [process.env.DEPLOYER_PRIV_KEY!,process.env.WALLET_IMPL_CHANGER_PRIV_KEY!],
+    gas: 'auto',
+    gasPrice: 'auto',
+    gasMultiplier: networkGasMultiplier(network),
+    timeout: 20000,
+    httpHeaders: {},
+    etherscan: etherscanKey(network)
+  }
+}
diff --git a/forks/passport/utils/deploy-contracts.ts b/forks/passport/utils/deploy-contracts.ts
new file mode 100644
index 00000000..9d7891c8
--- /dev/null
+++ b/forks/passport/utils/deploy-contracts.ts
@@ -0,0 +1,106 @@
+import { network, run, tenderly } from 'hardhat'
+import * as _ from 'lodash'
+import ora from 'ora'
+
+import {
+  MainModule__factory,
+  SequenceUtils__factory,
+  MainModuleUpgradable__factory,
+  GuestModule__factory,
+  Factory__factory,
+  RequireFreshSigner__factory
+} from '../src/gen/typechain'
+
+import { UniversalDeployer } from '@0xsequence/deployer'
+import { ContractFactory, BigNumber, providers } from 'ethers'
+import fs from 'fs'
+
+const prompt = ora()
+
+/**
+ * @notice Deploy core wallet contracts via universal deployer
+ *
+ *   1. Deploy Wallet Factory via UD
+ *   2. Deploy Main Module via UD
+ *   3. Deploy Upgradable Main Module via UD
+ *   4. Deploy Guest Module via UD
+ */
+
+const provider = new providers.Web3Provider(network.provider.send)
+const signer = provider.getSigner()
+const universalDeployer = new UniversalDeployer(network.name, signer.provider)
+const txParams = {
+  gasLimit: 6000000,
+  gasPrice: BigNumber.from(10)
+    .pow(9)
+    .mul(16)
+}
+
+const attempVerify = async <T extends ContractFactory>(name: string, _: new () => T, address: string, ...args: Parameters<T["deploy"]>) => {
+  try {
+    await run("verify:verify", {
+      address: address,
+      constructorArguments: args,
+    })
+  } catch {}
+
+  try {
+    await tenderly.verify({
+      name: name,
+      address: address
+    })
+  } catch {}
+}
+
+const buildNetworkJson = (...contracts: { name: string, address: string }[]) => {
+  return contracts.map((c) => ({
+    contractName: c.name,
+    address: c.address
+  }))
+}
+
+const main = async () => {
+  prompt.info(`Network Name:           ${network.name}`)
+  prompt.info(`Local Deployer Address: ${await signer.getAddress()}`)
+  prompt.info(`Local Deployer Balance: ${await signer.getBalance()}`)
+
+  const walletFactory = await universalDeployer.deploy('WalletFactory', Factory__factory, txParams)
+  const mainModule = await universalDeployer.deploy('MainModule', MainModule__factory, txParams, 0, walletFactory.address)
+  const mainModuleUpgradeable = await universalDeployer.deploy('MainModuleUpgradable', MainModuleUpgradable__factory, txParams)
+  const guestModule = await universalDeployer.deploy('GuestModule', GuestModule__factory, txParams)
+  const sequenceUtils = await universalDeployer.deploy('SequenceUtils', SequenceUtils__factory, txParams, 0, walletFactory.address, mainModule.address)
+  const requireFreshSignerLib = await universalDeployer.deploy('RequireFreshSignerLib', RequireFreshSigner__factory, txParams, 0, sequenceUtils.address)
+
+  prompt.start(`writing deployment information to ${network.name}.json`)
+  fs.writeFileSync(`./src/networks/${network.name}.json`, JSON.stringify(buildNetworkJson(
+    { name: "WalletFactory", address: walletFactory.address },
+    { name: "MainModule", address: mainModule.address },
+    { name: "MainModuleUpgradable", address: mainModuleUpgradeable.address },
+    { name: "GuestModule", address: guestModule.address },
+    { name: "SequenceUtils", address: sequenceUtils.address },
+    { name: "RequireFreshSignerLib", address: requireFreshSignerLib.address }
+  ), null, 2))
+  prompt.succeed()
+
+  prompt.start(`verifying contracts`)
+
+  await attempVerify("Factory", Factory__factory, walletFactory.address)
+  await attempVerify("MainModule", MainModule__factory, mainModule.address, walletFactory.address)
+  await attempVerify("MainModuleUpgradable", MainModuleUpgradable__factory, mainModuleUpgradeable.address)
+  await attempVerify("GuestModule", GuestModule__factory, guestModule.address)
+  await attempVerify("SequenceUtils", SequenceUtils__factory, sequenceUtils.address, walletFactory.address, mainModule.address)
+  await attempVerify("RequireFreshSignerLib", RequireFreshSigner__factory, requireFreshSignerLib.address, sequenceUtils.address)
+
+  prompt.succeed()
+}
+
+// We recommend this pattern to be able to use async/await everywhere
+// and properly handle errors.
+main()
+  .then(() => {
+    process.exit(0)
+  })
+  .catch(error => {
+    console.error(error)
+    process.exit(1)
+  })
diff --git a/forks/passport/yarn.lock b/forks/passport/yarn.lock
new file mode 100644
index 00000000..d3abce39
--- /dev/null
+++ b/forks/passport/yarn.lock
@@ -0,0 +1,8426 @@
+# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
+# yarn lockfile v1
+
+
+"@0xsequence/deployer@^0.21.5":
+  version "0.21.5"
+  resolved "https://registry.yarnpkg.com/@0xsequence/deployer/-/deployer-0.21.5.tgz#787333cb68b3b54a9696ab863614ce4ea4eb8895"
+  integrity sha512-P+QsiTrbZez//NCRvMc6OV6zWYP/TJzGJ2clFG1852CCPijqm0RJPmdoy5FzHpJd+9T4W7Awo0U2/gseSvFnaA==
+  dependencies:
+    "@0xsequence/utils" "^0.21.5"
+    "@ethersproject/contracts" "^5.0.12"
+    ethers "^5.0.32"
+    ora "^5.3.0"
+
+"@0xsequence/utils@^0.21.5":
+  version "0.21.5"
+  resolved "https://registry.yarnpkg.com/@0xsequence/utils/-/utils-0.21.5.tgz#9a0343b970c4cdda1d3bc575308aa919e3b7ecda"
+  integrity sha512-6gL0kot2o+E3n8E/1yvQR3wv4vxuJB9w25SNUK/zd9MDJ4oIY8tIy5jc4YrhQwC+FqKPG3k/8skNN9oP3gNXJg==
+  dependencies:
+    "@ethersproject/abstract-signer" "5.0.14"
+    "@ethersproject/properties" "^5.0.9"
+    ethers "^5.0.32"
+    js-base64 "^3.6.0"
+
+"@aashutoshrathi/word-wrap@^1.2.3":
+  version "1.2.6"
+  resolved "https://registry.yarnpkg.com/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz#bd9154aec9983f77b3a034ecaa015c2e4201f6cf"
+  integrity sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA==
+
+"@babel/code-frame@^7.0.0":
+  version "7.22.10"
+  resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.22.10.tgz#1c20e612b768fefa75f6e90d6ecb86329247f0a3"
+  integrity sha512-/KKIMG4UEL35WmI9OlvMhurwtytjvXoFcGNrOvyG9zIzA8YmPjVtIZUf7b05+TPO7G7/GEmLHDaoCgACHl9hhA==
+  dependencies:
+    "@babel/highlight" "^7.22.10"
+    chalk "^2.4.2"
+
+"@babel/helper-validator-identifier@^7.22.5":
+  version "7.22.5"
+  resolved "https://registry.yarnpkg.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.22.5.tgz#9544ef6a33999343c8740fa51350f30eeaaaf193"
+  integrity sha512-aJXu+6lErq8ltp+JhkJUfk1MTGyuA4v7f3pA+BJ5HLfNC6nAQ0Cpi9uOquUj8Hehg0aUiHzWQbOVJGao6ztBAQ==
+
+"@babel/highlight@^7.22.10":
+  version "7.22.10"
+  resolved "https://registry.yarnpkg.com/@babel/highlight/-/highlight-7.22.10.tgz#02a3f6d8c1cb4521b2fd0ab0da8f4739936137d7"
+  integrity sha512-78aUtVcT7MUscr0K5mIEnkwxPE0MaxkR5RxRwuHaQ+JuU5AmTPhY+do2mdzVTnIJJpyBglql2pehuBIWHug+WQ==
+  dependencies:
+    "@babel/helper-validator-identifier" "^7.22.5"
+    chalk "^2.4.2"
+    js-tokens "^4.0.0"
+
+"@babel/runtime@^7.4.4":
+  version "7.22.10"
+  resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.22.10.tgz#ae3e9631fd947cb7e3610d3e9d8fef5f76696682"
+  integrity sha512-21t/fkKLMZI4pqP2wlmsQAWnYW1PDyKyyUV4vCi+B25ydmdaYTKXPwCj0BzSUnZf4seIiYvSA3jcZ3gdsMFkLQ==
+  dependencies:
+    regenerator-runtime "^0.14.0"
+
+"@colors/colors@1.5.0":
+  version "1.5.0"
+  resolved "https://registry.yarnpkg.com/@colors/colors/-/colors-1.5.0.tgz#bb504579c1cae923e6576a4f5da43d25f97bdbd9"
+  integrity sha512-ooWCrlZP11i8GImSjTHYHLkvFDP48nS4+204nGb1RiX/WXYHmJA2III9/e2DWVabCESdW7hBAEzHRqUn9OUVvQ==
+
+"@cspotcode/source-map-support@^0.8.0":
+  version "0.8.1"
+  resolved "https://registry.yarnpkg.com/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz#00629c35a688e05a88b1cda684fb9d5e73f000a1"
+  integrity sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==
+  dependencies:
+    "@jridgewell/trace-mapping" "0.3.9"
+
+"@ensdomains/address-encoder@^0.1.7":
+  version "0.1.9"
+  resolved "https://registry.yarnpkg.com/@ensdomains/address-encoder/-/address-encoder-0.1.9.tgz#f948c485443d9ef7ed2c0c4790e931c33334d02d"
+  integrity sha512-E2d2gP4uxJQnDu2Kfg1tHNspefzbLT8Tyjrm5sEuim32UkU2sm5xL4VXtgc2X33fmPEw9+jUMpGs4veMbf+PYg==
+  dependencies:
+    bech32 "^1.1.3"
+    blakejs "^1.1.0"
+    bn.js "^4.11.8"
+    bs58 "^4.0.1"
+    crypto-addr-codec "^0.1.7"
+    nano-base32 "^1.0.1"
+    ripemd160 "^2.0.2"
+
+"@ensdomains/ens@0.4.5":
+  version "0.4.5"
+  resolved "https://registry.yarnpkg.com/@ensdomains/ens/-/ens-0.4.5.tgz#e0aebc005afdc066447c6e22feb4eda89a5edbfc"
+  integrity sha512-JSvpj1iNMFjK6K+uVl4unqMoa9rf5jopb8cya5UGBWz23Nw8hSNT7efgUx4BTlAPAgpNlEioUfeTyQ6J9ZvTVw==
+  dependencies:
+    bluebird "^3.5.2"
+    eth-ens-namehash "^2.0.8"
+    solc "^0.4.20"
+    testrpc "0.0.1"
+    web3-utils "^1.0.0-beta.31"
+
+"@ensdomains/ensjs@^2.0.1":
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/@ensdomains/ensjs/-/ensjs-2.1.0.tgz#0a7296c1f3d735ef019320d863a7846a0760c460"
+  integrity sha512-GRbGPT8Z/OJMDuxs75U/jUNEC0tbL0aj7/L/QQznGYKm/tiasp+ndLOaoULy9kKJFC0TBByqfFliEHDgoLhyog==
+  dependencies:
+    "@babel/runtime" "^7.4.4"
+    "@ensdomains/address-encoder" "^0.1.7"
+    "@ensdomains/ens" "0.4.5"
+    "@ensdomains/resolver" "0.2.4"
+    content-hash "^2.5.2"
+    eth-ens-namehash "^2.0.8"
+    ethers "^5.0.13"
+    js-sha3 "^0.8.0"
+
+"@ensdomains/resolver@0.2.4":
+  version "0.2.4"
+  resolved "https://registry.yarnpkg.com/@ensdomains/resolver/-/resolver-0.2.4.tgz#c10fe28bf5efbf49bff4666d909aed0265efbc89"
+  integrity sha512-bvaTH34PMCbv6anRa9I/0zjLJgY4EuznbEMgbV77JBCQ9KNC46rzi0avuxpOfu+xDjPEtSFGqVEOr5GlUSGudA==
+
+"@eslint-community/eslint-utils@^4.2.0":
+  version "4.4.0"
+  resolved "https://registry.yarnpkg.com/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz#a23514e8fb9af1269d5f7788aa556798d61c6b59"
+  integrity sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==
+  dependencies:
+    eslint-visitor-keys "^3.3.0"
+
+"@eslint-community/regexpp@^4.4.0", "@eslint-community/regexpp@^4.6.1":
+  version "4.6.2"
+  resolved "https://registry.yarnpkg.com/@eslint-community/regexpp/-/regexpp-4.6.2.tgz#1816b5f6948029c5eaacb0703b850ee0cb37d8f8"
+  integrity sha512-pPTNuaAG3QMH+buKyBIGJs3g/S5y0caxw0ygM3YyE6yJFySwiGGSzA+mM3KJ8QQvzeLh3blwgSonkFjgQdxzMw==
+
+"@eslint/eslintrc@^2.1.1":
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/@eslint/eslintrc/-/eslintrc-2.1.1.tgz#18d635e24ad35f7276e8a49d135c7d3ca6a46f93"
+  integrity sha512-9t7ZA7NGGK8ckelF0PQCfcxIUzs1Md5rrO6U/c+FIQNanea5UZC0wqKXH4vHBccmu4ZJgZ2idtPeW7+Q2npOEA==
+  dependencies:
+    ajv "^6.12.4"
+    debug "^4.3.2"
+    espree "^9.6.0"
+    globals "^13.19.0"
+    ignore "^5.2.0"
+    import-fresh "^3.2.1"
+    js-yaml "^4.1.0"
+    minimatch "^3.1.2"
+    strip-json-comments "^3.1.1"
+
+"@eslint/js@^8.46.0":
+  version "8.46.0"
+  resolved "https://registry.yarnpkg.com/@eslint/js/-/js-8.46.0.tgz#3f7802972e8b6fe3f88ed1aabc74ec596c456db6"
+  integrity sha512-a8TLtmPi8xzPkCbp/OGFUo5yhRkHM2Ko9kOWP4znJr0WAhWyThaw3PnwX4vOTWOAMsV2uRt32PPDcEz63esSaA==
+
+"@ethereumjs/common@2.5.0":
+  version "2.5.0"
+  resolved "https://registry.yarnpkg.com/@ethereumjs/common/-/common-2.5.0.tgz#ec61551b31bef7a69d1dc634d8932468866a4268"
+  integrity sha512-DEHjW6e38o+JmB/NO3GZBpW4lpaiBpkFgXF6jLcJ6gETBYpEyaA5nTimsWBUJR3Vmtm/didUEbNjajskugZORg==
+  dependencies:
+    crc-32 "^1.2.0"
+    ethereumjs-util "^7.1.1"
+
+"@ethereumjs/common@^2.5.0":
+  version "2.6.5"
+  resolved "https://registry.yarnpkg.com/@ethereumjs/common/-/common-2.6.5.tgz#0a75a22a046272579d91919cb12d84f2756e8d30"
+  integrity sha512-lRyVQOeCDaIVtgfbowla32pzeDv2Obr8oR8Put5RdUBNRGr1VGPGQNGP6elWIpgK3YdpzqTOh4GyUGOureVeeA==
+  dependencies:
+    crc-32 "^1.2.0"
+    ethereumjs-util "^7.1.5"
+
+"@ethereumjs/tx@3.3.2":
+  version "3.3.2"
+  resolved "https://registry.yarnpkg.com/@ethereumjs/tx/-/tx-3.3.2.tgz#348d4624bf248aaab6c44fec2ae67265efe3db00"
+  integrity sha512-6AaJhwg4ucmwTvw/1qLaZUX5miWrwZ4nLOUsKyb/HtzS3BMw/CasKhdi1ims9mBKeK9sOJCH4qGKOBGyJCeeog==
+  dependencies:
+    "@ethereumjs/common" "^2.5.0"
+    ethereumjs-util "^7.1.2"
+
+"@ethersproject/abi@5.7.0", "@ethersproject/abi@^5.0.0-beta.146", "@ethersproject/abi@^5.0.9", "@ethersproject/abi@^5.1.2", "@ethersproject/abi@^5.6.3", "@ethersproject/abi@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/abi/-/abi-5.7.0.tgz#b3f3e045bbbeed1af3947335c247ad625a44e449"
+  integrity sha512-351ktp42TiRcYB3H1OP8yajPeAQstMW/yCFokj/AthP9bLHzQFPlOrxOcwYEDkUAICmOHljvN4K39OMTMUa9RA==
+  dependencies:
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/hash" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+
+"@ethersproject/abstract-provider@5.7.0", "@ethersproject/abstract-provider@^5.0.8", "@ethersproject/abstract-provider@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/abstract-provider/-/abstract-provider-5.7.0.tgz#b0a8550f88b6bf9d51f90e4795d48294630cb9ef"
+  integrity sha512-R41c9UkchKCpAqStMYUpdunjo3pkEvZC3FAwZn5S5MGbXoMQOHIdHItezTETxAO5bevtMApSyEhn9+CHcDsWBw==
+  dependencies:
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/networks" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+    "@ethersproject/web" "^5.7.0"
+
+"@ethersproject/abstract-signer@5.0.14":
+  version "5.0.14"
+  resolved "https://registry.yarnpkg.com/@ethersproject/abstract-signer/-/abstract-signer-5.0.14.tgz#30ef912b0f86599d90fdffc65c110452e7b55cf1"
+  integrity sha512-JztBwVO7o5OHLh2vyjordlS4/1EjRyaECtc8vPdXTF1i4dXN+J0coeRoPN6ZFbBvi/YbaB6br2fvqhst1VQD/g==
+  dependencies:
+    "@ethersproject/abstract-provider" "^5.0.8"
+    "@ethersproject/bignumber" "^5.0.13"
+    "@ethersproject/bytes" "^5.0.9"
+    "@ethersproject/logger" "^5.0.8"
+    "@ethersproject/properties" "^5.0.7"
+
+"@ethersproject/abstract-signer@5.7.0", "@ethersproject/abstract-signer@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz#13f4f32117868452191a4649723cb086d2b596b2"
+  integrity sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==
+  dependencies:
+    "@ethersproject/abstract-provider" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+
+"@ethersproject/address@5.7.0", "@ethersproject/address@^5.0.2", "@ethersproject/address@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/address/-/address-5.7.0.tgz#19b56c4d74a3b0a46bfdbb6cfcc0a153fc697f37"
+  integrity sha512-9wYhYt7aghVGo758POM5nqcOMaE168Q6aRLJZwUmiqSrAungkG74gSSeKEIR7ukixesdRZGPgVqme6vmxs1fkA==
+  dependencies:
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/rlp" "^5.7.0"
+
+"@ethersproject/base64@5.7.0", "@ethersproject/base64@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/base64/-/base64-5.7.0.tgz#ac4ee92aa36c1628173e221d0d01f53692059e1c"
+  integrity sha512-Dr8tcHt2mEbsZr/mwTPIQAf3Ai0Bks/7gTw9dSqk1mQvhW3XvRlmDJr/4n+wg1JmCl16NZue17CDh8xb/vZ0sQ==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+
+"@ethersproject/basex@5.7.0", "@ethersproject/basex@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/basex/-/basex-5.7.0.tgz#97034dc7e8938a8ca943ab20f8a5e492ece4020b"
+  integrity sha512-ywlh43GwZLv2Voc2gQVTKBoVQ1mti3d8HK5aMxsfu/nRDnMmNqaSJ3r3n85HBByT8OpoY96SXM1FogC533T4zw==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+
+"@ethersproject/bignumber@5.7.0", "@ethersproject/bignumber@^5.0.13", "@ethersproject/bignumber@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/bignumber/-/bignumber-5.7.0.tgz#e2f03837f268ba655ffba03a57853e18a18dc9c2"
+  integrity sha512-n1CAdIHRWjSucQO3MC1zPSVgV/6dy/fjL9pMrPP9peL+QxEg9wOsVqwD4+818B6LUEtaXzVHQiuivzRoxPxUGw==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    bn.js "^5.2.1"
+
+"@ethersproject/bytes@5.7.0", "@ethersproject/bytes@^5.0.9", "@ethersproject/bytes@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/bytes/-/bytes-5.7.0.tgz#a00f6ea8d7e7534d6d87f47188af1148d71f155d"
+  integrity sha512-nsbxwgFXWh9NyYWo+U8atvmMsSdKJprTcICAkvbBffT75qDocbuggBU0SJiVK2MuTrp0q+xvLkTnGMPK1+uA9A==
+  dependencies:
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/constants@5.7.0", "@ethersproject/constants@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/constants/-/constants-5.7.0.tgz#df80a9705a7e08984161f09014ea012d1c75295e"
+  integrity sha512-DHI+y5dBNvkpYUMiRQyxRBYBefZkJfo70VUkUAsRjcPs47muV9evftfZ0PJVCXYbAiCgght0DtcF9srFQmIgWA==
+  dependencies:
+    "@ethersproject/bignumber" "^5.7.0"
+
+"@ethersproject/contracts@5.7.0", "@ethersproject/contracts@^5.0.12", "@ethersproject/contracts@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/contracts/-/contracts-5.7.0.tgz#c305e775abd07e48aa590e1a877ed5c316f8bd1e"
+  integrity sha512-5GJbzEU3X+d33CdfPhcyS+z8MzsTrBGk/sc+G+59+tPa9yFkl6HQ9D6L0QMgNTA9q8dT0XKxxkyp883XsQvbbg==
+  dependencies:
+    "@ethersproject/abi" "^5.7.0"
+    "@ethersproject/abstract-provider" "^5.7.0"
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+
+"@ethersproject/hash@5.7.0", "@ethersproject/hash@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/hash/-/hash-5.7.0.tgz#eb7aca84a588508369562e16e514b539ba5240a7"
+  integrity sha512-qX5WrQfnah1EFnO5zJv1v46a8HW0+E5xuBBDTwMFZLuVTx0tbU2kkx15NqdjxecrLGatQN9FGQKpb1FKdHCt+g==
+  dependencies:
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/base64" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+
+"@ethersproject/hdnode@5.7.0", "@ethersproject/hdnode@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/hdnode/-/hdnode-5.7.0.tgz#e627ddc6b466bc77aebf1a6b9e47405ca5aef9cf"
+  integrity sha512-OmyYo9EENBPPf4ERhR7oj6uAtUAhYGqOnIS+jE5pTXvdKBS99ikzq1E7Iv0ZQZ5V36Lqx1qZLeak0Ra16qpeOg==
+  dependencies:
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/basex" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/pbkdf2" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/sha2" "^5.7.0"
+    "@ethersproject/signing-key" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+    "@ethersproject/wordlists" "^5.7.0"
+
+"@ethersproject/json-wallets@5.7.0", "@ethersproject/json-wallets@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/json-wallets/-/json-wallets-5.7.0.tgz#5e3355287b548c32b368d91014919ebebddd5360"
+  integrity sha512-8oee5Xgu6+RKgJTkvEMl2wDgSPSAQ9MB/3JYjFV9jlKvcYHUXZC+cQp0njgmxdHkYWn8s6/IqIZYm0YWCjO/0g==
+  dependencies:
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/hdnode" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/pbkdf2" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/random" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+    aes-js "3.0.0"
+    scrypt-js "3.0.1"
+
+"@ethersproject/keccak256@5.7.0", "@ethersproject/keccak256@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/keccak256/-/keccak256-5.7.0.tgz#3186350c6e1cd6aba7940384ec7d6d9db01f335a"
+  integrity sha512-2UcPboeL/iW+pSg6vZ6ydF8tCnv3Iu/8tUmLLzWWGzxWKFFqOBQFLo6uLUv6BDrLgCDfN28RJ/wtByx+jZ4KBg==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    js-sha3 "0.8.0"
+
+"@ethersproject/logger@5.7.0", "@ethersproject/logger@^5.0.8", "@ethersproject/logger@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/logger/-/logger-5.7.0.tgz#6ce9ae168e74fecf287be17062b590852c311892"
+  integrity sha512-0odtFdXu/XHtjQXJYA3u9G0G8btm0ND5Cu8M7i5vhEcE8/HmF4Lbdqanwyv4uQTr2tx6b7fQRmgLrsnpQlmnig==
+
+"@ethersproject/networks@5.7.1", "@ethersproject/networks@^5.7.0":
+  version "5.7.1"
+  resolved "https://registry.yarnpkg.com/@ethersproject/networks/-/networks-5.7.1.tgz#118e1a981d757d45ccea6bb58d9fd3d9db14ead6"
+  integrity sha512-n/MufjFYv3yFcUyfhnXotyDlNdFb7onmkSy8aQERi2PjNcnWQ66xXxa3XlS8nCcA8aJKJjIIMNJTC7tu80GwpQ==
+  dependencies:
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/pbkdf2@5.7.0", "@ethersproject/pbkdf2@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/pbkdf2/-/pbkdf2-5.7.0.tgz#d2267d0a1f6e123f3771007338c47cccd83d3102"
+  integrity sha512-oR/dBRZR6GTyaofd86DehG72hY6NpAjhabkhxgr3X2FpJtJuodEl2auADWBZfhDHgVCbu3/H/Ocq2uC6dpNjjw==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/sha2" "^5.7.0"
+
+"@ethersproject/properties@5.7.0", "@ethersproject/properties@^5.0.7", "@ethersproject/properties@^5.0.9", "@ethersproject/properties@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/properties/-/properties-5.7.0.tgz#a6e12cb0439b878aaf470f1902a176033067ed30"
+  integrity sha512-J87jy8suntrAkIZtecpxEPxY//szqr1mlBaYlQ0r4RCaiD2hjheqF9s1LVE8vVuJCXisjIP+JgtK/Do54ej4Sw==
+  dependencies:
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/providers@5.7.2", "@ethersproject/providers@^5.7.2":
+  version "5.7.2"
+  resolved "https://registry.yarnpkg.com/@ethersproject/providers/-/providers-5.7.2.tgz#f8b1a4f275d7ce58cf0a2eec222269a08beb18cb"
+  integrity sha512-g34EWZ1WWAVgr4aptGlVBF8mhl3VWjv+8hoAnzStu8Ah22VHBsuGzP17eb6xDVRzw895G4W7vvx60lFFur/1Rg==
+  dependencies:
+    "@ethersproject/abstract-provider" "^5.7.0"
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/base64" "^5.7.0"
+    "@ethersproject/basex" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/hash" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/networks" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/random" "^5.7.0"
+    "@ethersproject/rlp" "^5.7.0"
+    "@ethersproject/sha2" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+    "@ethersproject/web" "^5.7.0"
+    bech32 "1.1.4"
+    ws "7.4.6"
+
+"@ethersproject/random@5.7.0", "@ethersproject/random@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/random/-/random-5.7.0.tgz#af19dcbc2484aae078bb03656ec05df66253280c"
+  integrity sha512-19WjScqRA8IIeWclFme75VMXSBvi4e6InrUNuaR4s5pTF2qNhcGdCUwdxUVGtDDqC00sDLCO93jPQoDUH4HVmQ==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/rlp@5.7.0", "@ethersproject/rlp@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/rlp/-/rlp-5.7.0.tgz#de39e4d5918b9d74d46de93af80b7685a9c21304"
+  integrity sha512-rBxzX2vK8mVF7b0Tol44t5Tb8gomOHkj5guL+HhzQ1yBh/ydjGnpw6at+X6Iw0Kp3OzzzkcKp8N9r0W4kYSs9w==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/sha2@5.7.0", "@ethersproject/sha2@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/sha2/-/sha2-5.7.0.tgz#9a5f7a7824ef784f7f7680984e593a800480c9fb"
+  integrity sha512-gKlH42riwb3KYp0reLsFTokByAKoJdgFCwI+CCiX/k+Jm2mbNs6oOaCjYQSlI1+XBVejwH2KrmCbMAT/GnRDQw==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    hash.js "1.1.7"
+
+"@ethersproject/signing-key@5.7.0", "@ethersproject/signing-key@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/signing-key/-/signing-key-5.7.0.tgz#06b2df39411b00bc57c7c09b01d1e41cf1b16ab3"
+  integrity sha512-MZdy2nL3wO0u7gkB4nA/pEf8lu1TlFswPNmy8AiYkfKTdO6eXBJyUdmHO/ehm/htHw9K/qF8ujnTyUAD+Ry54Q==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    bn.js "^5.2.1"
+    elliptic "6.5.4"
+    hash.js "1.1.7"
+
+"@ethersproject/solidity@5.7.0", "@ethersproject/solidity@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/solidity/-/solidity-5.7.0.tgz#5e9c911d8a2acce2a5ebb48a5e2e0af20b631cb8"
+  integrity sha512-HmabMd2Dt/raavyaGukF4XxizWKhKQ24DoLtdNbBmNKUOPqwjsKQSdV9GQtj9CBEea9DlzETlVER1gYeXXBGaA==
+  dependencies:
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/sha2" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+
+"@ethersproject/strings@5.7.0", "@ethersproject/strings@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/strings/-/strings-5.7.0.tgz#54c9d2a7c57ae8f1205c88a9d3a56471e14d5ed2"
+  integrity sha512-/9nu+lj0YswRNSH0NXYqrh8775XNyEdUQAuf3f+SmOrnVewcJ5SBNAjF7lpgehKi4abvNNXyf+HX86czCdJ8Mg==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/transactions@5.7.0", "@ethersproject/transactions@^5.6.2", "@ethersproject/transactions@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/transactions/-/transactions-5.7.0.tgz#91318fc24063e057885a6af13fdb703e1f993d3b"
+  integrity sha512-kmcNicCp1lp8qanMTC3RIikGgoJ80ztTyvtsFvCYpSCfkjhD0jZ2LOrnbcuxuToLIUYYf+4XwD1rP+B/erDIhQ==
+  dependencies:
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/rlp" "^5.7.0"
+    "@ethersproject/signing-key" "^5.7.0"
+
+"@ethersproject/units@5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/units/-/units-5.7.0.tgz#637b563d7e14f42deeee39245275d477aae1d8b1"
+  integrity sha512-pD3xLMy3SJu9kG5xDGI7+xhTEmGXlEqXU4OfNapmfnxLVY4EMSSRp7j1k7eezutBPH7RBN/7QPnwR7hzNlEFeg==
+  dependencies:
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/wallet@5.7.0", "@ethersproject/wallet@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/wallet/-/wallet-5.7.0.tgz#4e5d0790d96fe21d61d38fb40324e6c7ef350b2d"
+  integrity sha512-MhmXlJXEJFBFVKrDLB4ZdDzxcBxQ3rLyCkhNqVu3CDYvR97E+8r01UgrI+TI99Le+aYm/in/0vp86guJuM7FCA==
+  dependencies:
+    "@ethersproject/abstract-provider" "^5.7.0"
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/hash" "^5.7.0"
+    "@ethersproject/hdnode" "^5.7.0"
+    "@ethersproject/json-wallets" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/random" "^5.7.0"
+    "@ethersproject/signing-key" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+    "@ethersproject/wordlists" "^5.7.0"
+
+"@ethersproject/web@5.7.1", "@ethersproject/web@^5.7.0":
+  version "5.7.1"
+  resolved "https://registry.yarnpkg.com/@ethersproject/web/-/web-5.7.1.tgz#de1f285b373149bee5928f4eb7bcb87ee5fbb4ae"
+  integrity sha512-Gueu8lSvyjBWL4cYsWsjh6MtMwM0+H4HvqFPZfB6dV8ctbP9zFAO73VG1cMWae0FLPCtz0peKPpZY8/ugJJX2w==
+  dependencies:
+    "@ethersproject/base64" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+
+"@ethersproject/wordlists@5.7.0", "@ethersproject/wordlists@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/wordlists/-/wordlists-5.7.0.tgz#8fb2c07185d68c3e09eb3bfd6e779ba2774627f5"
+  integrity sha512-S2TFNJNfHWVHNE6cNDjbVlZ6MgE17MIxMbMg2zv3wn+3XSJGosL1m9ZVv3GXCf/2ymSsQ+hRI5IzoMJTG6aoVA==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/hash" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+
+"@fastify/busboy@^2.0.0":
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/@fastify/busboy/-/busboy-2.1.0.tgz#0709e9f4cb252351c609c6e6d8d6779a8d25edff"
+  integrity sha512-+KpH+QxZU7O4675t3mnkQKcZZg56u+K/Ct2K+N2AZYNVK8kyeo/bI18tI8aPm3tvNNRyTWfj6s5tnGNlcbQRsA==
+
+"@humanwhocodes/config-array@^0.11.10":
+  version "0.11.10"
+  resolved "https://registry.yarnpkg.com/@humanwhocodes/config-array/-/config-array-0.11.10.tgz#5a3ffe32cc9306365fb3fd572596cd602d5e12d2"
+  integrity sha512-KVVjQmNUepDVGXNuoRRdmmEjruj0KfiGSbS8LVc12LMsWDQzRXJ0qdhN8L8uUigKpfEHRhlaQFY0ib1tnUbNeQ==
+  dependencies:
+    "@humanwhocodes/object-schema" "^1.2.1"
+    debug "^4.1.1"
+    minimatch "^3.0.5"
+
+"@humanwhocodes/module-importer@^1.0.1":
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz#af5b2691a22b44be847b0ca81641c5fb6ad0172c"
+  integrity sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==
+
+"@humanwhocodes/object-schema@^1.2.1":
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/@humanwhocodes/object-schema/-/object-schema-1.2.1.tgz#b520529ec21d8e5945a1851dfd1c32e94e39ff45"
+  integrity sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA==
+
+"@jridgewell/resolve-uri@^3.0.3":
+  version "3.1.1"
+  resolved "https://registry.yarnpkg.com/@jridgewell/resolve-uri/-/resolve-uri-3.1.1.tgz#c08679063f279615a3326583ba3a90d1d82cc721"
+  integrity sha512-dSYZh7HhCDtCKm4QakX0xFpsRDqjjtZf/kjI/v3T3Nwt5r8/qz/M19F9ySyOqU94SXBmeG9ttTul+YnR4LOxFA==
+
+"@jridgewell/sourcemap-codec@^1.4.10":
+  version "1.4.15"
+  resolved "https://registry.yarnpkg.com/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz#d7c6e6755c78567a951e04ab52ef0fd26de59f32"
+  integrity sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==
+
+"@jridgewell/trace-mapping@0.3.9":
+  version "0.3.9"
+  resolved "https://registry.yarnpkg.com/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz#6534fd5933a53ba7cbf3a17615e273a0d1273ff9"
+  integrity sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==
+  dependencies:
+    "@jridgewell/resolve-uri" "^3.0.3"
+    "@jridgewell/sourcemap-codec" "^1.4.10"
+
+"@metamask/eth-sig-util@^4.0.0":
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/@metamask/eth-sig-util/-/eth-sig-util-4.0.1.tgz#3ad61f6ea9ad73ba5b19db780d40d9aae5157088"
+  integrity sha512-tghyZKLHZjcdlDqCA3gNZmLeR0XvOE9U1qoQO9ohyAZT6Pya+H9vkBPcsyXytmYLNgVoin7CKCmweo/R43V+tQ==
+  dependencies:
+    ethereumjs-abi "^0.6.8"
+    ethereumjs-util "^6.2.1"
+    ethjs-util "^0.1.6"
+    tweetnacl "^1.0.3"
+    tweetnacl-util "^0.15.1"
+
+"@noble/hashes@1.2.0", "@noble/hashes@~1.2.0":
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/@noble/hashes/-/hashes-1.2.0.tgz#a3150eeb09cc7ab207ebf6d7b9ad311a9bdbed12"
+  integrity sha512-FZfhjEDbT5GRswV3C6uvLPHMiVD6lQBmpoX5+eSiPaMTXte/IKqI5dykDxzZB/WBeK/CDuQRBWarPdi3FNY2zQ==
+
+"@noble/secp256k1@1.7.1", "@noble/secp256k1@~1.7.0":
+  version "1.7.1"
+  resolved "https://registry.yarnpkg.com/@noble/secp256k1/-/secp256k1-1.7.1.tgz#b251c70f824ce3ca7f8dc3df08d58f005cc0507c"
+  integrity sha512-hOUk6AyBFmqVrv7k5WAw/LpszxVbj9gGN4JRkIX52fdFAj1UA61KXmZDvqVEm+pOyec3+fIeZB02LYa/pWOArw==
+
+"@nodelib/fs.scandir@2.1.5":
+  version "2.1.5"
+  resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5"
+  integrity sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==
+  dependencies:
+    "@nodelib/fs.stat" "2.0.5"
+    run-parallel "^1.1.9"
+
+"@nodelib/fs.stat@2.0.5", "@nodelib/fs.stat@^2.0.2":
+  version "2.0.5"
+  resolved "https://registry.yarnpkg.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz#5bd262af94e9d25bd1e71b05deed44876a222e8b"
+  integrity sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==
+
+"@nodelib/fs.walk@^1.2.3", "@nodelib/fs.walk@^1.2.8":
+  version "1.2.8"
+  resolved "https://registry.yarnpkg.com/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz#e95737e8bb6746ddedf69c556953494f196fe69a"
+  integrity sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==
+  dependencies:
+    "@nodelib/fs.scandir" "2.1.5"
+    fastq "^1.6.0"
+
+"@nomicfoundation/ethereumjs-block@4.2.2", "@nomicfoundation/ethereumjs-block@^4.0.0":
+  version "4.2.2"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/ethereumjs-block/-/ethereumjs-block-4.2.2.tgz#f317078c810a54381c682d0c12e1e81acfc11599"
+  integrity sha512-atjpt4gc6ZGZUPHBAQaUJsm1l/VCo7FmyQ780tMGO8QStjLdhz09dXynmhwVTy5YbRr0FOh/uX3QaEM0yIB2Zg==
+  dependencies:
+    "@nomicfoundation/ethereumjs-common" "3.1.2"
+    "@nomicfoundation/ethereumjs-rlp" "4.0.3"
+    "@nomicfoundation/ethereumjs-trie" "5.0.5"
+    "@nomicfoundation/ethereumjs-tx" "4.1.2"
+    "@nomicfoundation/ethereumjs-util" "8.0.6"
+    ethereum-cryptography "0.1.3"
+
+"@nomicfoundation/ethereumjs-blockchain@6.2.2", "@nomicfoundation/ethereumjs-blockchain@^6.0.0":
+  version "6.2.2"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/ethereumjs-blockchain/-/ethereumjs-blockchain-6.2.2.tgz#9f79dd2b3dc73f5d5a220f7d8a734330c4c26320"
+  integrity sha512-6AIB2MoTEPZJLl6IRKcbd8mUmaBAQ/NMe3O7OsAOIiDjMNPPH5KaUQiLfbVlegT4wKIg/GOsFH7XlH2KDVoJNg==
+  dependencies:
+    "@nomicfoundation/ethereumjs-block" "4.2.2"
+    "@nomicfoundation/ethereumjs-common" "3.1.2"
+    "@nomicfoundation/ethereumjs-ethash" "2.0.5"
+    "@nomicfoundation/ethereumjs-rlp" "4.0.3"
+    "@nomicfoundation/ethereumjs-trie" "5.0.5"
+    "@nomicfoundation/ethereumjs-util" "8.0.6"
+    abstract-level "^1.0.3"
+    debug "^4.3.3"
+    ethereum-cryptography "0.1.3"
+    level "^8.0.0"
+    lru-cache "^5.1.1"
+    memory-level "^1.0.0"
+
+"@nomicfoundation/ethereumjs-common@3.1.2", "@nomicfoundation/ethereumjs-common@^3.0.0":
+  version "3.1.2"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/ethereumjs-common/-/ethereumjs-common-3.1.2.tgz#041086da66ed40f2bf2a2116a1f2f0fcf33fb80d"
+  integrity sha512-JAEBpIua62dyObHM9KI2b4wHZcRQYYge9gxiygTWa3lNCr2zo+K0TbypDpgiNij5MCGNWP1eboNfNfx1a3vkvA==
+  dependencies:
+    "@nomicfoundation/ethereumjs-util" "8.0.6"
+    crc-32 "^1.2.0"
+
+"@nomicfoundation/ethereumjs-ethash@2.0.5":
+  version "2.0.5"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/ethereumjs-ethash/-/ethereumjs-ethash-2.0.5.tgz#0c605812f6f4589a9f6d597db537bbf3b86469db"
+  integrity sha512-xlLdcICGgAYyYmnI3r1t0R5fKGBJNDQSOQxXNjVO99JmxJIdXR5MgPo5CSJO1RpyzKOgzi3uIFn8agv564dZEQ==
+  dependencies:
+    "@nomicfoundation/ethereumjs-block" "4.2.2"
+    "@nomicfoundation/ethereumjs-rlp" "4.0.3"
+    "@nomicfoundation/ethereumjs-util" "8.0.6"
+    abstract-level "^1.0.3"
+    bigint-crypto-utils "^3.0.23"
+    ethereum-cryptography "0.1.3"
+
+"@nomicfoundation/ethereumjs-evm@1.3.2", "@nomicfoundation/ethereumjs-evm@^1.0.0":
+  version "1.3.2"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/ethereumjs-evm/-/ethereumjs-evm-1.3.2.tgz#f9d6bafd5c23d07ab75b8649d589af1a43b60bfc"
+  integrity sha512-I00d4MwXuobyoqdPe/12dxUQxTYzX8OckSaWsMcWAfQhgVDvBx6ffPyP/w1aL0NW7MjyerySPcSVfDJAMHjilw==
+  dependencies:
+    "@nomicfoundation/ethereumjs-common" "3.1.2"
+    "@nomicfoundation/ethereumjs-util" "8.0.6"
+    "@types/async-eventemitter" "^0.2.1"
+    async-eventemitter "^0.2.4"
+    debug "^4.3.3"
+    ethereum-cryptography "0.1.3"
+    mcl-wasm "^0.7.1"
+    rustbn.js "~0.2.0"
+
+"@nomicfoundation/ethereumjs-rlp@4.0.3", "@nomicfoundation/ethereumjs-rlp@^4.0.0":
+  version "4.0.3"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/ethereumjs-rlp/-/ethereumjs-rlp-4.0.3.tgz#8d9147fbd0d49e8f4c5ce729d226694a8fe03eb8"
+  integrity sha512-DZMzB/lqPK78T6MluyXqtlRmOMcsZbTTbbEyAjo0ncaff2mqu/k8a79PBcyvpgAhWD/R59Fjq/x3ro5Lof0AtA==
+
+"@nomicfoundation/ethereumjs-statemanager@1.0.5", "@nomicfoundation/ethereumjs-statemanager@^1.0.0":
+  version "1.0.5"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/ethereumjs-statemanager/-/ethereumjs-statemanager-1.0.5.tgz#951cc9ff2c421d40233d2e9d0fe033db2391ee44"
+  integrity sha512-CAhzpzTR5toh/qOJIZUUOnWekUXuRqkkzaGAQrVcF457VhtCmr+ddZjjK50KNZ524c1XP8cISguEVNqJ6ij1sA==
+  dependencies:
+    "@nomicfoundation/ethereumjs-common" "3.1.2"
+    "@nomicfoundation/ethereumjs-rlp" "4.0.3"
+    "@nomicfoundation/ethereumjs-trie" "5.0.5"
+    "@nomicfoundation/ethereumjs-util" "8.0.6"
+    debug "^4.3.3"
+    ethereum-cryptography "0.1.3"
+    functional-red-black-tree "^1.0.1"
+
+"@nomicfoundation/ethereumjs-trie@5.0.5", "@nomicfoundation/ethereumjs-trie@^5.0.0":
+  version "5.0.5"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/ethereumjs-trie/-/ethereumjs-trie-5.0.5.tgz#bf31c9306dcbba2007fad668e96109ddb147040c"
+  integrity sha512-+8sNZrXkzvA1NH5F4kz5RSYl1I6iaRz7mAZRsyxOm0IVY4UaP43Ofvfp/TwOalFunurQrYB5pRO40+8FBcxFMA==
+  dependencies:
+    "@nomicfoundation/ethereumjs-rlp" "4.0.3"
+    "@nomicfoundation/ethereumjs-util" "8.0.6"
+    ethereum-cryptography "0.1.3"
+    readable-stream "^3.6.0"
+
+"@nomicfoundation/ethereumjs-tx@4.1.2", "@nomicfoundation/ethereumjs-tx@^4.0.0":
+  version "4.1.2"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/ethereumjs-tx/-/ethereumjs-tx-4.1.2.tgz#8659fad7f9094b7eb82aa6cc3c8097cb1c42ff31"
+  integrity sha512-emJBJZpmTdUa09cqxQqHaysbBI9Od353ZazeH7WgPb35miMgNY6mb7/3vBA98N5lUW/rgkiItjX0KZfIzihSoQ==
+  dependencies:
+    "@nomicfoundation/ethereumjs-common" "3.1.2"
+    "@nomicfoundation/ethereumjs-rlp" "4.0.3"
+    "@nomicfoundation/ethereumjs-util" "8.0.6"
+    ethereum-cryptography "0.1.3"
+
+"@nomicfoundation/ethereumjs-util@8.0.6", "@nomicfoundation/ethereumjs-util@^8.0.0":
+  version "8.0.6"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/ethereumjs-util/-/ethereumjs-util-8.0.6.tgz#dbce5d258b017b37aa58b3a7c330ad59d10ccf0b"
+  integrity sha512-jOQfF44laa7xRfbfLXojdlcpkvxeHrE2Xu7tSeITsWFgoII163MzjOwFEzSNozHYieFysyoEMhCdP+NY5ikstw==
+  dependencies:
+    "@nomicfoundation/ethereumjs-rlp" "4.0.3"
+    ethereum-cryptography "0.1.3"
+
+"@nomicfoundation/ethereumjs-vm@^6.0.0":
+  version "6.4.2"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/ethereumjs-vm/-/ethereumjs-vm-6.4.2.tgz#af1cf62e6c0054bc2b7febc8556d032433d1b18c"
+  integrity sha512-PRTyxZMP6kx+OdAzBhuH1LD2Yw+hrSpaytftvaK//thDy2OI07S0nrTdbrdk7b8ZVPAc9H9oTwFBl3/wJ3w15g==
+  dependencies:
+    "@nomicfoundation/ethereumjs-block" "4.2.2"
+    "@nomicfoundation/ethereumjs-blockchain" "6.2.2"
+    "@nomicfoundation/ethereumjs-common" "3.1.2"
+    "@nomicfoundation/ethereumjs-evm" "1.3.2"
+    "@nomicfoundation/ethereumjs-rlp" "4.0.3"
+    "@nomicfoundation/ethereumjs-statemanager" "1.0.5"
+    "@nomicfoundation/ethereumjs-trie" "5.0.5"
+    "@nomicfoundation/ethereumjs-tx" "4.1.2"
+    "@nomicfoundation/ethereumjs-util" "8.0.6"
+    "@types/async-eventemitter" "^0.2.1"
+    async-eventemitter "^0.2.4"
+    debug "^4.3.3"
+    ethereum-cryptography "0.1.3"
+    functional-red-black-tree "^1.0.1"
+    mcl-wasm "^0.7.1"
+    rustbn.js "~0.2.0"
+
+"@nomicfoundation/hardhat-chai-matchers@^1.0.6":
+  version "1.0.6"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/hardhat-chai-matchers/-/hardhat-chai-matchers-1.0.6.tgz#72a2e312e1504ee5dd73fe302932736432ba96bc"
+  integrity sha512-f5ZMNmabZeZegEfuxn/0kW+mm7+yV7VNDxLpMOMGXWFJ2l/Ct3QShujzDRF9cOkK9Ui/hbDeOWGZqyQALDXVCQ==
+  dependencies:
+    "@ethersproject/abi" "^5.1.2"
+    "@types/chai-as-promised" "^7.1.3"
+    chai-as-promised "^7.1.1"
+    deep-eql "^4.0.1"
+    ordinal "^1.0.3"
+
+"@nomicfoundation/hardhat-network-helpers@^1.0.8":
+  version "1.0.8"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/hardhat-network-helpers/-/hardhat-network-helpers-1.0.8.tgz#e4fe1be93e8a65508c46d73c41fa26c7e9f84931"
+  integrity sha512-MNqQbzUJZnCMIYvlniC3U+kcavz/PhhQSsY90tbEtUyMj/IQqsLwIRZa4ctjABh3Bz0KCh9OXUZ7Yk/d9hr45Q==
+  dependencies:
+    ethereumjs-util "^7.1.4"
+
+"@nomicfoundation/solidity-analyzer-darwin-arm64@0.1.1":
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/solidity-analyzer-darwin-arm64/-/solidity-analyzer-darwin-arm64-0.1.1.tgz#4c858096b1c17fe58a474fe81b46815f93645c15"
+  integrity sha512-KcTodaQw8ivDZyF+D76FokN/HdpgGpfjc/gFCImdLUyqB6eSWVaZPazMbeAjmfhx3R0zm/NYVzxwAokFKgrc0w==
+
+"@nomicfoundation/solidity-analyzer-darwin-x64@0.1.1":
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/solidity-analyzer-darwin-x64/-/solidity-analyzer-darwin-x64-0.1.1.tgz#6e25ccdf6e2d22389c35553b64fe6f3fdaec432c"
+  integrity sha512-XhQG4BaJE6cIbjAVtzGOGbK3sn1BO9W29uhk9J8y8fZF1DYz0Doj8QDMfpMu+A6TjPDs61lbsmeYodIDnfveSA==
+
+"@nomicfoundation/solidity-analyzer-freebsd-x64@0.1.1":
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/solidity-analyzer-freebsd-x64/-/solidity-analyzer-freebsd-x64-0.1.1.tgz#0a224ea50317139caeebcdedd435c28a039d169c"
+  integrity sha512-GHF1VKRdHW3G8CndkwdaeLkVBi5A9u2jwtlS7SLhBc8b5U/GcoL39Q+1CSO3hYqePNP+eV5YI7Zgm0ea6kMHoA==
+
+"@nomicfoundation/solidity-analyzer-linux-arm64-gnu@0.1.1":
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/solidity-analyzer-linux-arm64-gnu/-/solidity-analyzer-linux-arm64-gnu-0.1.1.tgz#dfa085d9ffab9efb2e7b383aed3f557f7687ac2b"
+  integrity sha512-g4Cv2fO37ZsUENQ2vwPnZc2zRenHyAxHcyBjKcjaSmmkKrFr64yvzeNO8S3GBFCo90rfochLs99wFVGT/0owpg==
+
+"@nomicfoundation/solidity-analyzer-linux-arm64-musl@0.1.1":
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/solidity-analyzer-linux-arm64-musl/-/solidity-analyzer-linux-arm64-musl-0.1.1.tgz#c9e06b5d513dd3ab02a7ac069c160051675889a4"
+  integrity sha512-WJ3CE5Oek25OGE3WwzK7oaopY8xMw9Lhb0mlYuJl/maZVo+WtP36XoQTb7bW/i8aAdHW5Z+BqrHMux23pvxG3w==
+
+"@nomicfoundation/solidity-analyzer-linux-x64-gnu@0.1.1":
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/solidity-analyzer-linux-x64-gnu/-/solidity-analyzer-linux-x64-gnu-0.1.1.tgz#8d328d16839e52571f72f2998c81e46bf320f893"
+  integrity sha512-5WN7leSr5fkUBBjE4f3wKENUy9HQStu7HmWqbtknfXkkil+eNWiBV275IOlpXku7v3uLsXTOKpnnGHJYI2qsdA==
+
+"@nomicfoundation/solidity-analyzer-linux-x64-musl@0.1.1":
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/solidity-analyzer-linux-x64-musl/-/solidity-analyzer-linux-x64-musl-0.1.1.tgz#9b49d0634b5976bb5ed1604a1e1b736f390959bb"
+  integrity sha512-KdYMkJOq0SYPQMmErv/63CwGwMm5XHenEna9X9aB8mQmhDBrYrlAOSsIPgFCUSL0hjxE3xHP65/EPXR/InD2+w==
+
+"@nomicfoundation/solidity-analyzer-win32-arm64-msvc@0.1.1":
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/solidity-analyzer-win32-arm64-msvc/-/solidity-analyzer-win32-arm64-msvc-0.1.1.tgz#e2867af7264ebbcc3131ef837878955dd6a3676f"
+  integrity sha512-VFZASBfl4qiBYwW5xeY20exWhmv6ww9sWu/krWSesv3q5hA0o1JuzmPHR4LPN6SUZj5vcqci0O6JOL8BPw+APg==
+
+"@nomicfoundation/solidity-analyzer-win32-ia32-msvc@0.1.1":
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/solidity-analyzer-win32-ia32-msvc/-/solidity-analyzer-win32-ia32-msvc-0.1.1.tgz#0685f78608dd516c8cdfb4896ed451317e559585"
+  integrity sha512-JnFkYuyCSA70j6Si6cS1A9Gh1aHTEb8kOTBApp/c7NRTFGNMH8eaInKlyuuiIbvYFhlXW4LicqyYuWNNq9hkpQ==
+
+"@nomicfoundation/solidity-analyzer-win32-x64-msvc@0.1.1":
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/solidity-analyzer-win32-x64-msvc/-/solidity-analyzer-win32-x64-msvc-0.1.1.tgz#c9a44f7108646f083b82e851486e0f6aeb785836"
+  integrity sha512-HrVJr6+WjIXGnw3Q9u6KQcbZCtk0caVWhCdFADySvRyUxJ8PnzlaP+MhwNE8oyT8OZ6ejHBRrrgjSqDCFXGirw==
+
+"@nomicfoundation/solidity-analyzer@^0.1.0":
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/@nomicfoundation/solidity-analyzer/-/solidity-analyzer-0.1.1.tgz#f5f4d36d3f66752f59a57e7208cd856f3ddf6f2d"
+  integrity sha512-1LMtXj1puAxyFusBgUIy5pZk3073cNXYnXUpuNKFghHbIit/xZgbk0AokpUADbNm3gyD6bFWl3LRFh3dhVdREg==
+  optionalDependencies:
+    "@nomicfoundation/solidity-analyzer-darwin-arm64" "0.1.1"
+    "@nomicfoundation/solidity-analyzer-darwin-x64" "0.1.1"
+    "@nomicfoundation/solidity-analyzer-freebsd-x64" "0.1.1"
+    "@nomicfoundation/solidity-analyzer-linux-arm64-gnu" "0.1.1"
+    "@nomicfoundation/solidity-analyzer-linux-arm64-musl" "0.1.1"
+    "@nomicfoundation/solidity-analyzer-linux-x64-gnu" "0.1.1"
+    "@nomicfoundation/solidity-analyzer-linux-x64-musl" "0.1.1"
+    "@nomicfoundation/solidity-analyzer-win32-arm64-msvc" "0.1.1"
+    "@nomicfoundation/solidity-analyzer-win32-ia32-msvc" "0.1.1"
+    "@nomicfoundation/solidity-analyzer-win32-x64-msvc" "0.1.1"
+
+"@nomiclabs/hardhat-ethers@^2.0.2", "@nomiclabs/hardhat-ethers@^2.1.1":
+  version "2.2.3"
+  resolved "https://registry.yarnpkg.com/@nomiclabs/hardhat-ethers/-/hardhat-ethers-2.2.3.tgz#b41053e360c31a32c2640c9a45ee981a7e603fe0"
+  integrity sha512-YhzPdzb612X591FOe68q+qXVXGG2ANZRvDo0RRUtimev85rCrAlv/TLMEZw5c+kq9AbzocLTVX/h2jVIFPL9Xg==
+
+"@nomiclabs/hardhat-etherscan@^3.1.0":
+  version "3.1.7"
+  resolved "https://registry.yarnpkg.com/@nomiclabs/hardhat-etherscan/-/hardhat-etherscan-3.1.7.tgz#72e3d5bd5d0ceb695e097a7f6f5ff6fcbf062b9a"
+  integrity sha512-tZ3TvSgpvsQ6B6OGmo1/Au6u8BrAkvs1mIC/eURA3xgIfznUZBhmpne8hv7BXUzw9xNL3fXdpOYgOQlVMTcoHQ==
+  dependencies:
+    "@ethersproject/abi" "^5.1.2"
+    "@ethersproject/address" "^5.0.2"
+    cbor "^8.1.0"
+    chalk "^2.4.2"
+    debug "^4.1.1"
+    fs-extra "^7.0.1"
+    lodash "^4.17.11"
+    semver "^6.3.0"
+    table "^6.8.0"
+    undici "^5.14.0"
+
+"@nomiclabs/hardhat-truffle5@^2.0.0":
+  version "2.0.7"
+  resolved "https://registry.yarnpkg.com/@nomiclabs/hardhat-truffle5/-/hardhat-truffle5-2.0.7.tgz#7519eadd2c6c460c2addc3d4d6efda7a8883361e"
+  integrity sha512-Pw8451IUZp1bTp0QqCHCYfCHs66sCnyxPcaorapu9mfOV9xnZsVaFdtutnhNEiXdiZwbed7LFKpRsde4BjFwig==
+  dependencies:
+    "@nomiclabs/truffle-contract" "^4.2.23"
+    "@types/chai" "^4.2.0"
+    chai "^4.2.0"
+    ethereumjs-util "^7.1.4"
+    fs-extra "^7.0.1"
+
+"@nomiclabs/hardhat-web3@^2.0.0":
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/@nomiclabs/hardhat-web3/-/hardhat-web3-2.0.0.tgz#2d9850cb285a2cebe1bd718ef26a9523542e52a9"
+  integrity sha512-zt4xN+D+fKl3wW2YlTX3k9APR3XZgPkxJYf36AcliJn3oujnKEVRZaHu0PhgLjO+gR+F/kiYayo9fgd2L8970Q==
+  dependencies:
+    "@types/bignumber.js" "^5.0.0"
+
+"@nomiclabs/truffle-contract@^4.2.23":
+  version "4.5.10"
+  resolved "https://registry.yarnpkg.com/@nomiclabs/truffle-contract/-/truffle-contract-4.5.10.tgz#52adcca1068647e1c2b44bf0e6a89fc4ad7f9213"
+  integrity sha512-nF/6InFV+0hUvutyFgsdOMCoYlr//2fJbRER4itxYtQtc4/O1biTwZIKRu+5l2J5Sq6LU2WX7vZHtDgQdhWxIQ==
+  dependencies:
+    "@ensdomains/ensjs" "^2.0.1"
+    "@truffle/blockchain-utils" "^0.1.3"
+    "@truffle/contract-schema" "^3.4.7"
+    "@truffle/debug-utils" "^6.0.22"
+    "@truffle/error" "^0.1.0"
+    "@truffle/interface-adapter" "^0.5.16"
+    bignumber.js "^7.2.1"
+    ethereum-ens "^0.8.0"
+    ethers "^4.0.0-beta.1"
+    source-map-support "^0.5.19"
+
+"@openzeppelin/contracts@^4.9.3":
+  version "4.9.3"
+  resolved "https://registry.yarnpkg.com/@openzeppelin/contracts/-/contracts-4.9.3.tgz#00d7a8cf35a475b160b3f0293a6403c511099364"
+  integrity sha512-He3LieZ1pP2TNt5JbkPA4PNT9WC3gOTOlDcFGJW4Le4QKqwmiNJCRt44APfxMxvq7OugU/cqYuPcSBzOw38DAg==
+
+"@scure/base@~1.1.0":
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/@scure/base/-/base-1.1.1.tgz#ebb651ee52ff84f420097055f4bf46cfba403938"
+  integrity sha512-ZxOhsSyxYwLJj3pLZCefNitxsj093tb2vq90mp2txoYeBqbcjDjqFhyM8eUjq/uFm6zJ+mUuqxlS2FkuSY1MTA==
+
+"@scure/bip32@1.1.5":
+  version "1.1.5"
+  resolved "https://registry.yarnpkg.com/@scure/bip32/-/bip32-1.1.5.tgz#d2ccae16dcc2e75bc1d75f5ef3c66a338d1ba300"
+  integrity sha512-XyNh1rB0SkEqd3tXcXMi+Xe1fvg+kUIcoRIEujP1Jgv7DqW2r9lg3Ah0NkFaCs9sTkQAQA8kw7xiRXzENi9Rtw==
+  dependencies:
+    "@noble/hashes" "~1.2.0"
+    "@noble/secp256k1" "~1.7.0"
+    "@scure/base" "~1.1.0"
+
+"@scure/bip39@1.1.1":
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/@scure/bip39/-/bip39-1.1.1.tgz#b54557b2e86214319405db819c4b6a370cf340c5"
+  integrity sha512-t+wDck2rVkh65Hmv280fYdVdY25J9YeEUIgn2LG1WM6gxFkGzcksoDiUkWVpVp3Oex9xGC68JU2dSbUfwZ2jPg==
+  dependencies:
+    "@noble/hashes" "~1.2.0"
+    "@scure/base" "~1.1.0"
+
+"@sentry/core@5.30.0":
+  version "5.30.0"
+  resolved "https://registry.yarnpkg.com/@sentry/core/-/core-5.30.0.tgz#6b203664f69e75106ee8b5a2fe1d717379b331f3"
+  integrity sha512-TmfrII8w1PQZSZgPpUESqjB+jC6MvZJZdLtE/0hZ+SrnKhW3x5WlYLvTXZpcWePYBku7rl2wn1RZu6uT0qCTeg==
+  dependencies:
+    "@sentry/hub" "5.30.0"
+    "@sentry/minimal" "5.30.0"
+    "@sentry/types" "5.30.0"
+    "@sentry/utils" "5.30.0"
+    tslib "^1.9.3"
+
+"@sentry/hub@5.30.0":
+  version "5.30.0"
+  resolved "https://registry.yarnpkg.com/@sentry/hub/-/hub-5.30.0.tgz#2453be9b9cb903404366e198bd30c7ca74cdc100"
+  integrity sha512-2tYrGnzb1gKz2EkMDQcfLrDTvmGcQPuWxLnJKXJvYTQDGLlEvi2tWz1VIHjunmOvJrB5aIQLhm+dcMRwFZDCqQ==
+  dependencies:
+    "@sentry/types" "5.30.0"
+    "@sentry/utils" "5.30.0"
+    tslib "^1.9.3"
+
+"@sentry/minimal@5.30.0":
+  version "5.30.0"
+  resolved "https://registry.yarnpkg.com/@sentry/minimal/-/minimal-5.30.0.tgz#ce3d3a6a273428e0084adcb800bc12e72d34637b"
+  integrity sha512-BwWb/owZKtkDX+Sc4zCSTNcvZUq7YcH3uAVlmh/gtR9rmUvbzAA3ewLuB3myi4wWRAMEtny6+J/FN/x+2wn9Xw==
+  dependencies:
+    "@sentry/hub" "5.30.0"
+    "@sentry/types" "5.30.0"
+    tslib "^1.9.3"
+
+"@sentry/node@^5.18.1":
+  version "5.30.0"
+  resolved "https://registry.yarnpkg.com/@sentry/node/-/node-5.30.0.tgz#4ca479e799b1021285d7fe12ac0858951c11cd48"
+  integrity sha512-Br5oyVBF0fZo6ZS9bxbJZG4ApAjRqAnqFFurMVJJdunNb80brh7a5Qva2kjhm+U6r9NJAB5OmDyPkA1Qnt+QVg==
+  dependencies:
+    "@sentry/core" "5.30.0"
+    "@sentry/hub" "5.30.0"
+    "@sentry/tracing" "5.30.0"
+    "@sentry/types" "5.30.0"
+    "@sentry/utils" "5.30.0"
+    cookie "^0.4.1"
+    https-proxy-agent "^5.0.0"
+    lru_map "^0.3.3"
+    tslib "^1.9.3"
+
+"@sentry/tracing@5.30.0":
+  version "5.30.0"
+  resolved "https://registry.yarnpkg.com/@sentry/tracing/-/tracing-5.30.0.tgz#501d21f00c3f3be7f7635d8710da70d9419d4e1f"
+  integrity sha512-dUFowCr0AIMwiLD7Fs314Mdzcug+gBVo/+NCMyDw8tFxJkwWAKl7Qa2OZxLQ0ZHjakcj1hNKfCQJ9rhyfOl4Aw==
+  dependencies:
+    "@sentry/hub" "5.30.0"
+    "@sentry/minimal" "5.30.0"
+    "@sentry/types" "5.30.0"
+    "@sentry/utils" "5.30.0"
+    tslib "^1.9.3"
+
+"@sentry/types@5.30.0":
+  version "5.30.0"
+  resolved "https://registry.yarnpkg.com/@sentry/types/-/types-5.30.0.tgz#19709bbe12a1a0115bc790b8942917da5636f402"
+  integrity sha512-R8xOqlSTZ+htqrfteCWU5Nk0CDN5ApUTvrlvBuiH1DyP6czDZ4ktbZB0hAgBlVcK0U+qpD3ag3Tqqpa5Q67rPw==
+
+"@sentry/utils@5.30.0":
+  version "5.30.0"
+  resolved "https://registry.yarnpkg.com/@sentry/utils/-/utils-5.30.0.tgz#9a5bd7ccff85ccfe7856d493bffa64cabc41e980"
+  integrity sha512-zaYmoH0NWWtvnJjC9/CBseXMtKHm/tm40sz3YfJRxeQjyzRqNQPgivpd9R/oDJCYj999mzdW382p/qi2ypjLww==
+  dependencies:
+    "@sentry/types" "5.30.0"
+    tslib "^1.9.3"
+
+"@sindresorhus/is@^4.0.0", "@sindresorhus/is@^4.6.0":
+  version "4.6.0"
+  resolved "https://registry.yarnpkg.com/@sindresorhus/is/-/is-4.6.0.tgz#3c7c9c46e678feefe7a2e5bb609d3dbd665ffb3f"
+  integrity sha512-t09vSN3MdfsyCHoFcTRCH/iUtG7OJ0CsjzB8cjAmKc/va/kIgeDI/TxsigdncE/4be734m0cvIYwNaV4i2XqAw==
+
+"@solidity-parser/parser@^0.14.0":
+  version "0.14.5"
+  resolved "https://registry.yarnpkg.com/@solidity-parser/parser/-/parser-0.14.5.tgz#87bc3cc7b068e08195c219c91cd8ddff5ef1a804"
+  integrity sha512-6dKnHZn7fg/iQATVEzqyUOyEidbn05q7YA2mQ9hC0MMXhhV3/JrsxmFSYZAcr7j1yUP700LLhTruvJ3MiQmjJg==
+  dependencies:
+    antlr4ts "^0.5.0-alpha.4"
+
+"@solidity-parser/parser@^0.16.0":
+  version "0.16.1"
+  resolved "https://registry.yarnpkg.com/@solidity-parser/parser/-/parser-0.16.1.tgz#f7c8a686974e1536da0105466c4db6727311253c"
+  integrity sha512-PdhRFNhbTtu3x8Axm0uYpqOy/lODYQK+MlYSgqIsq2L8SFYEHJPHNUiOTAJbDGzNjjr1/n9AcIayxafR/fWmYw==
+  dependencies:
+    antlr4ts "^0.5.0-alpha.4"
+
+"@szmarczak/http-timer@^4.0.5":
+  version "4.0.6"
+  resolved "https://registry.yarnpkg.com/@szmarczak/http-timer/-/http-timer-4.0.6.tgz#b4a914bb62e7c272d4e5989fe4440f812ab1d807"
+  integrity sha512-4BAffykYOgO+5nzBWYwE3W90sBgLJoUPRWWcL8wlyiM8IB8ipJz3UMJ9KXQd1RKQXpKp8Tutn80HZtWsu2u76w==
+  dependencies:
+    defer-to-connect "^2.0.0"
+
+"@szmarczak/http-timer@^5.0.1":
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/@szmarczak/http-timer/-/http-timer-5.0.1.tgz#c7c1bf1141cdd4751b0399c8fc7b8b664cd5be3a"
+  integrity sha512-+PmQX0PiAYPMeVYe237LJAYvOMYW1j2rH5YROyS3b4CTVJum34HfRvKvAzozHAQG0TnHNdUfY9nCeUyRAs//cw==
+  dependencies:
+    defer-to-connect "^2.0.1"
+
+"@tenderly/hardhat-tenderly@^1.0.11":
+  version "1.7.7"
+  resolved "https://registry.yarnpkg.com/@tenderly/hardhat-tenderly/-/hardhat-tenderly-1.7.7.tgz#b320ee6bea3779e4781eb0299a3a816cbdc83def"
+  integrity sha512-p/jLzRPpoD7J0LGvUFEQjgniDzmP5AzfTgy41qqzyjhOsW0voe7wZI8lXjadl5MEr7rAXN1iH3VncT13qG6+Zw==
+  dependencies:
+    "@ethersproject/bignumber" "^5.7.0"
+    "@nomiclabs/hardhat-ethers" "^2.1.1"
+    axios "^0.27.2"
+    ethers "^5.7.0"
+    fs-extra "^10.1.0"
+    hardhat-deploy "^0.11.14"
+    js-yaml "^4.1.0"
+    tenderly "^0.5.3"
+    tslog "^4.3.1"
+
+"@truffle/abi-utils@^1.0.2":
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/@truffle/abi-utils/-/abi-utils-1.0.2.tgz#41210b234912051433960382af009f339f8a9642"
+  integrity sha512-MefEcxsBlprKIpfW7eh2I5zJqlWM18xk3duL7SW4VhIs6kNEec//hCpEDoE6P0m7GjqY3vk8X4vnf4aLlZkRcA==
+  dependencies:
+    change-case "3.0.2"
+    fast-check "3.1.1"
+    web3-utils "1.10.0"
+
+"@truffle/blockchain-utils@^0.1.3":
+  version "0.1.8"
+  resolved "https://registry.yarnpkg.com/@truffle/blockchain-utils/-/blockchain-utils-0.1.8.tgz#0c1a369aa72f51df5af095678803242ea0a0d6ae"
+  integrity sha512-ZskpYDNHkXD3ota4iU3pZz6kLth87RC+wDn66Rp2Or+DqqJCKdnmS9GDctBi1EcMPDEi0BqpkdrfBuzA9uIkGg==
+
+"@truffle/codec@^0.17.2":
+  version "0.17.2"
+  resolved "https://registry.yarnpkg.com/@truffle/codec/-/codec-0.17.2.tgz#52a3604f73b89964373eec945f20d5cd0f4244d4"
+  integrity sha512-n9HX8R5a5+/j6Y0+lqSzIyU1cUxTRYn/xEWp0Qc1b0Vtltad7wvVh+KLGvbm/KQEX3o1RK1xRIUN2E0QlDeQnA==
+  dependencies:
+    "@truffle/abi-utils" "^1.0.2"
+    "@truffle/compile-common" "^0.9.7"
+    big.js "^6.0.3"
+    bn.js "^5.1.3"
+    cbor "^5.2.0"
+    debug "^4.3.1"
+    lodash "^4.17.21"
+    semver "7.5.2"
+    utf8 "^3.0.0"
+    web3-utils "1.10.0"
+
+"@truffle/compile-common@^0.9.7":
+  version "0.9.7"
+  resolved "https://registry.yarnpkg.com/@truffle/compile-common/-/compile-common-0.9.7.tgz#e8ba6cd49c4d4e7ae4684ba453fb9b2dcd09b347"
+  integrity sha512-TXuVLc5yJ/A0bSWw5OWIdXmcyaPpj3TJQ60ki7w9cIuW65Bazw7P4FRPaVNjR9YGe1FLYJ36GSdd9V3egPbzCg==
+  dependencies:
+    "@truffle/error" "^0.2.1"
+    colors "1.4.0"
+
+"@truffle/contract-schema@^3.4.7":
+  version "3.4.15"
+  resolved "https://registry.yarnpkg.com/@truffle/contract-schema/-/contract-schema-3.4.15.tgz#199789b3f0a61b0e564ee8d62d7a7e5a8e6b749f"
+  integrity sha512-m13e1VlXEdxiXiqv/SmPlqbdtcuhjwIGTICm+JCEO8nt0NYBbdMC2paNkpUvGz9lK139JxIupMHctEV4vgkldw==
+  dependencies:
+    ajv "^6.10.0"
+    debug "^4.3.1"
+
+"@truffle/debug-utils@^6.0.22":
+  version "6.0.56"
+  resolved "https://registry.yarnpkg.com/@truffle/debug-utils/-/debug-utils-6.0.56.tgz#7727891580c52cc9acfbe46d43ca4ef9aa338713"
+  integrity sha512-tCB0nKZirWlQp+0JqqaBxZk7CIwDBIHl8Q9CyGJZA4pdAwKHHMI3FoPAXzubP7X0YuICeQsAmpOAXzOHGYGkjg==
+  dependencies:
+    "@truffle/codec" "^0.17.2"
+    "@trufflesuite/chromafi" "^3.0.0"
+    bn.js "^5.1.3"
+    chalk "^2.4.2"
+    debug "^4.3.1"
+    highlightjs-solidity "^2.0.6"
+
+"@truffle/error@^0.1.0":
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/@truffle/error/-/error-0.1.1.tgz#e52026ac8ca7180d83443dca73c03e07ace2a301"
+  integrity sha512-sE7c9IHIGdbK4YayH4BC8i8qMjoAOeg6nUXUDZZp8wlU21/EMpaG+CLx+KqcIPyR+GSWIW3Dm0PXkr2nlggFDA==
+
+"@truffle/error@^0.2.1":
+  version "0.2.1"
+  resolved "https://registry.yarnpkg.com/@truffle/error/-/error-0.2.1.tgz#71bb8e777a832e0cfe09a8638a70a5177aad8628"
+  integrity sha512-5Qy+z9dg9hP37WNdLnXH4b9MzemWrjTufRq7/DTKqimjyxCP/1zlL8gQEMdiSx1BBtAZz0xypkID/jb7AF/Osg==
+
+"@truffle/interface-adapter@^0.5.16":
+  version "0.5.35"
+  resolved "https://registry.yarnpkg.com/@truffle/interface-adapter/-/interface-adapter-0.5.35.tgz#f0eb1c4a2803190ca249143f545029a8b641fe96"
+  integrity sha512-B5gtJnvsum5j2do393n0UfCT8MklrlAZxuqvEFBeMM9UKnreYct0/D368FVMlZwWo1N50HgGeZ0hlpSJqR/nvg==
+  dependencies:
+    bn.js "^5.1.3"
+    ethers "^4.0.32"
+    web3 "1.10.0"
+
+"@trufflesuite/bigint-buffer@1.1.10":
+  version "1.1.10"
+  resolved "https://registry.yarnpkg.com/@trufflesuite/bigint-buffer/-/bigint-buffer-1.1.10.tgz#a1d9ca22d3cad1a138b78baaf15543637a3e1692"
+  integrity sha512-pYIQC5EcMmID74t26GCC67946mgTJFiLXOT/BYozgrd4UEY2JHEGLhWi9cMiQCt5BSqFEvKkCHNnoj82SRjiEw==
+  dependencies:
+    node-gyp-build "4.4.0"
+
+"@trufflesuite/chromafi@^3.0.0":
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/@trufflesuite/chromafi/-/chromafi-3.0.0.tgz#f6956408c1af6a38a6ed1657783ce59504a1eb8b"
+  integrity sha512-oqWcOqn8nT1bwlPPfidfzS55vqcIDdpfzo3HbU9EnUmcSTX+I8z0UyUFI3tZQjByVJulbzxHxUGS3ZJPwK/GPQ==
+  dependencies:
+    camelcase "^4.1.0"
+    chalk "^2.3.2"
+    cheerio "^1.0.0-rc.2"
+    detect-indent "^5.0.0"
+    highlight.js "^10.4.1"
+    lodash.merge "^4.6.2"
+    strip-ansi "^4.0.0"
+    strip-indent "^2.0.0"
+
+"@trufflesuite/uws-js-unofficial@20.10.0-unofficial.2":
+  version "20.10.0-unofficial.2"
+  resolved "https://registry.yarnpkg.com/@trufflesuite/uws-js-unofficial/-/uws-js-unofficial-20.10.0-unofficial.2.tgz#7ed613ce3260cd5d1773a4d5787a2a106acd1a91"
+  integrity sha512-oQQlnS3oNeGsgS4K3KCSSavJgSb0W9D5ktZs4FacX9VbM7b+NlhjH96d6/G4fMrz+bc5MXRyco419on0X0dvRA==
+  dependencies:
+    ws "8.2.3"
+  optionalDependencies:
+    bufferutil "4.0.5"
+    utf-8-validate "5.0.7"
+
+"@tsconfig/node10@^1.0.7":
+  version "1.0.9"
+  resolved "https://registry.yarnpkg.com/@tsconfig/node10/-/node10-1.0.9.tgz#df4907fc07a886922637b15e02d4cebc4c0021b2"
+  integrity sha512-jNsYVVxU8v5g43Erja32laIDHXeoNvFEpX33OK4d6hljo3jDhCBDhx5dhCCTMWUojscpAagGiRkBKxpdl9fxqA==
+
+"@tsconfig/node12@^1.0.7":
+  version "1.0.11"
+  resolved "https://registry.yarnpkg.com/@tsconfig/node12/-/node12-1.0.11.tgz#ee3def1f27d9ed66dac6e46a295cffb0152e058d"
+  integrity sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==
+
+"@tsconfig/node14@^1.0.0":
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/@tsconfig/node14/-/node14-1.0.3.tgz#e4386316284f00b98435bf40f72f75a09dabf6c1"
+  integrity sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==
+
+"@tsconfig/node16@^1.0.2":
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/@tsconfig/node16/-/node16-1.0.4.tgz#0b92dcc0cc1c81f6f306a381f28e31b1a56536e9"
+  integrity sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==
+
+"@typechain/ethers-v5@^10.1.1":
+  version "10.2.1"
+  resolved "https://registry.yarnpkg.com/@typechain/ethers-v5/-/ethers-v5-10.2.1.tgz#50241e6957683281ecfa03fb5a6724d8a3ce2391"
+  integrity sha512-n3tQmCZjRE6IU4h6lqUGiQ1j866n5MTCBJreNEHHVWXa2u9GJTaeYyU1/k+1qLutkyw+sS6VAN+AbeiTqsxd/A==
+  dependencies:
+    lodash "^4.17.15"
+    ts-essentials "^7.0.1"
+
+"@types/async-eventemitter@^0.2.1":
+  version "0.2.1"
+  resolved "https://registry.yarnpkg.com/@types/async-eventemitter/-/async-eventemitter-0.2.1.tgz#f8e6280e87e8c60b2b938624b0a3530fb3e24712"
+  integrity sha512-M2P4Ng26QbAeITiH7w1d7OxtldgfAe0wobpyJzVK/XOb0cUGKU2R4pfAhqcJBXAe2ife5ZOhSv4wk7p+ffURtg==
+
+"@types/bignumber.js@^5.0.0":
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/@types/bignumber.js/-/bignumber.js-5.0.0.tgz#d9f1a378509f3010a3255e9cc822ad0eeb4ab969"
+  integrity sha512-0DH7aPGCClywOFaxxjE6UwpN2kQYe9LwuDQMv+zYA97j5GkOMo8e66LYT+a8JYU7jfmUFRZLa9KycxHDsKXJCA==
+  dependencies:
+    bignumber.js "*"
+
+"@types/bn.js@^4.11.3":
+  version "4.11.6"
+  resolved "https://registry.yarnpkg.com/@types/bn.js/-/bn.js-4.11.6.tgz#c306c70d9358aaea33cd4eda092a742b9505967c"
+  integrity sha512-pqr857jrp2kPuO9uRjZ3PwnJTjoQy+fcdxvBTvHm6dkmEL9q+hDD/2j/0ELOBPtPnS8LjCX0gI9nbl8lVkadpg==
+  dependencies:
+    "@types/node" "*"
+
+"@types/bn.js@^5.1.0", "@types/bn.js@^5.1.1":
+  version "5.1.1"
+  resolved "https://registry.yarnpkg.com/@types/bn.js/-/bn.js-5.1.1.tgz#b51e1b55920a4ca26e9285ff79936bbdec910682"
+  integrity sha512-qNrYbZqMx0uJAfKnKclPh+dTwK33KfLHYqtyODwd5HnXOjnkhc4qgn3BrK6RWyGZm5+sIFE7Q7Vz6QQtJB7w7g==
+  dependencies:
+    "@types/node" "*"
+
+"@types/cacheable-request@^6.0.1", "@types/cacheable-request@^6.0.2":
+  version "6.0.3"
+  resolved "https://registry.yarnpkg.com/@types/cacheable-request/-/cacheable-request-6.0.3.tgz#a430b3260466ca7b5ca5bfd735693b36e7a9d183"
+  integrity sha512-IQ3EbTzGxIigb1I3qPZc1rWJnH0BmSKv5QYTalEwweFvyBDLSAe24zP0le/hyi7ecGfZVlIVAg4BZqb8WBwKqw==
+  dependencies:
+    "@types/http-cache-semantics" "*"
+    "@types/keyv" "^3.1.4"
+    "@types/node" "*"
+    "@types/responselike" "^1.0.0"
+
+"@types/chai-as-promised@^7.1.0", "@types/chai-as-promised@^7.1.3":
+  version "7.1.5"
+  resolved "https://registry.yarnpkg.com/@types/chai-as-promised/-/chai-as-promised-7.1.5.tgz#6e016811f6c7a64f2eed823191c3a6955094e255"
+  integrity sha512-jStwss93SITGBwt/niYrkf2C+/1KTeZCZl1LaeezTlqppAKeoQC7jxyqYuP72sxBGKCIbw7oHgbYssIRzT5FCQ==
+  dependencies:
+    "@types/chai" "*"
+
+"@types/chai-string@^1.4.1":
+  version "1.4.2"
+  resolved "https://registry.yarnpkg.com/@types/chai-string/-/chai-string-1.4.2.tgz#0f116504a666b6c6a3c42becf86634316c9a19ac"
+  integrity sha512-ld/1hV5qcPRGuwlPdvRfvM3Ka/iofOk2pH4VkasK4b1JJP1LjNmWWn0LsISf6RRzyhVOvs93rb9tM09e+UuF8Q==
+  dependencies:
+    "@types/chai" "*"
+
+"@types/chai@*", "@types/chai@^4.2.0":
+  version "4.3.5"
+  resolved "https://registry.yarnpkg.com/@types/chai/-/chai-4.3.5.tgz#ae69bcbb1bebb68c4ac0b11e9d8ed04526b3562b"
+  integrity sha512-mEo1sAde+UCE6b2hxn332f1g1E8WfYRu6p5SvTKr2ZKC1f7gFJXk4h5PyGP9Dt6gCaG8y8XhwnXWC6Iy2cmBng==
+
+"@types/concat-stream@^1.6.0":
+  version "1.6.1"
+  resolved "https://registry.yarnpkg.com/@types/concat-stream/-/concat-stream-1.6.1.tgz#24bcfc101ecf68e886aaedce60dfd74b632a1b74"
+  integrity sha512-eHE4cQPoj6ngxBZMvVf6Hw7Mh4jMW4U9lpGmS5GBPB9RYxlFg+CHaVN7ErNY4W9XfLIEn20b4VDYaIrbq0q4uA==
+  dependencies:
+    "@types/node" "*"
+
+"@types/form-data@0.0.33":
+  version "0.0.33"
+  resolved "https://registry.yarnpkg.com/@types/form-data/-/form-data-0.0.33.tgz#c9ac85b2a5fd18435b8c85d9ecb50e6d6c893ff8"
+  integrity sha512-8BSvG1kGm83cyJITQMZSulnl6QV8jqAGreJsc5tPu1Jq0vTSOiY/k24Wx82JRpWwZSqrala6sd5rWi6aNXvqcw==
+  dependencies:
+    "@types/node" "*"
+
+"@types/glob@^7.1.1":
+  version "7.2.0"
+  resolved "https://registry.yarnpkg.com/@types/glob/-/glob-7.2.0.tgz#bc1b5bf3aa92f25bd5dd39f35c57361bdce5b2eb"
+  integrity sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==
+  dependencies:
+    "@types/minimatch" "*"
+    "@types/node" "*"
+
+"@types/http-cache-semantics@*":
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/@types/http-cache-semantics/-/http-cache-semantics-4.0.1.tgz#0ea7b61496902b95890dc4c3a116b60cb8dae812"
+  integrity sha512-SZs7ekbP8CN0txVG2xVRH6EgKmEm31BOxA07vkFaETzZz1xh+cbt8BcI0slpymvwhx5dlFnQG2rTlPVQn+iRPQ==
+
+"@types/json-schema@^7.0.9":
+  version "7.0.12"
+  resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.12.tgz#d70faba7039d5fca54c83c7dbab41051d2b6f6cb"
+  integrity sha512-Hr5Jfhc9eYOQNPYO5WLDq/n4jqijdHNlDXjuAQkkt+mWdQR+XJToOHrsD4cPaMXpn6KO7y2+wM8AZEs8VpBLVA==
+
+"@types/json5@^0.0.29":
+  version "0.0.29"
+  resolved "https://registry.yarnpkg.com/@types/json5/-/json5-0.0.29.tgz#ee28707ae94e11d2b827bcbe5270bcea7f3e71ee"
+  integrity sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==
+
+"@types/keyv@^3.1.4":
+  version "3.1.4"
+  resolved "https://registry.yarnpkg.com/@types/keyv/-/keyv-3.1.4.tgz#3ccdb1c6751b0c7e52300bcdacd5bcbf8faa75b6"
+  integrity sha512-BQ5aZNSCpj7D6K2ksrRCTmKRLEpnPvWDiLPfoGyhZ++8YtiK9d/3DBKPJgry359X/P1PfruyYwvnvwFjuEiEIg==
+  dependencies:
+    "@types/node" "*"
+
+"@types/lru-cache@5.1.1", "@types/lru-cache@^5.1.0":
+  version "5.1.1"
+  resolved "https://registry.yarnpkg.com/@types/lru-cache/-/lru-cache-5.1.1.tgz#c48c2e27b65d2a153b19bfc1a317e30872e01eef"
+  integrity sha512-ssE3Vlrys7sdIzs5LOxCzTVMsU7i9oa/IaW92wF32JFb3CVczqOkru2xspuKczHEbG3nvmPY7IFqVmGGHdNbYw==
+
+"@types/minimatch@*":
+  version "5.1.2"
+  resolved "https://registry.yarnpkg.com/@types/minimatch/-/minimatch-5.1.2.tgz#07508b45797cb81ec3f273011b054cd0755eddca"
+  integrity sha512-K0VQKziLUWkVKiRVrx4a40iPaxTUefQmjtkQofBkYRcoaaL/8rhwDWww9qWbrgicNOgnpIsMxyNIUM4+n6dUIA==
+
+"@types/mocha@^10.0.0":
+  version "10.0.1"
+  resolved "https://registry.yarnpkg.com/@types/mocha/-/mocha-10.0.1.tgz#2f4f65bb08bc368ac39c96da7b2f09140b26851b"
+  integrity sha512-/fvYntiO1GeICvqbQ3doGDIP97vWmvFt83GKguJ6prmQM2iXZfFcq6YE8KteFyRtX2/h5Hf91BYvPodJKFYv5Q==
+
+"@types/node@*":
+  version "20.4.9"
+  resolved "https://registry.yarnpkg.com/@types/node/-/node-20.4.9.tgz#c7164e0f8d3f12dfae336af0b1f7fdec8c6b204f"
+  integrity sha512-8e2HYcg7ohnTUbHk8focoklEQYvemQmu9M/f43DZVx43kHn0tE3BY/6gSDxS7k0SprtS0NHvj+L80cGLnoOUcQ==
+
+"@types/node@^10.0.3":
+  version "10.17.60"
+  resolved "https://registry.yarnpkg.com/@types/node/-/node-10.17.60.tgz#35f3d6213daed95da7f0f73e75bcc6980e90597b"
+  integrity sha512-F0KIgDJfy2nA3zMLmWGKxcH2ZVEtCZXHHdOQs2gSaQ27+lNeEfGxzkIw90aXswATX7AZ33tahPbzy6KAfUreVw==
+
+"@types/node@^12.12.6":
+  version "12.20.55"
+  resolved "https://registry.yarnpkg.com/@types/node/-/node-12.20.55.tgz#c329cbd434c42164f846b909bd6f85b5537f6240"
+  integrity sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ==
+
+"@types/node@^8.0.0":
+  version "8.10.66"
+  resolved "https://registry.yarnpkg.com/@types/node/-/node-8.10.66.tgz#dd035d409df322acc83dff62a602f12a5783bbb3"
+  integrity sha512-tktOkFUA4kXx2hhhrB8bIFb5TbwzS4uOhKEmwiD+NoiL0qtP2OQ9mFldbgD4dV1djrlBYP6eBuQZiWjuHUpqFw==
+
+"@types/parse-json@^4.0.0":
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/@types/parse-json/-/parse-json-4.0.0.tgz#2f8bb441434d163b35fb8ffdccd7138927ffb8c0"
+  integrity sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA==
+
+"@types/pbkdf2@^3.0.0":
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/@types/pbkdf2/-/pbkdf2-3.1.0.tgz#039a0e9b67da0cdc4ee5dab865caa6b267bb66b1"
+  integrity sha512-Cf63Rv7jCQ0LaL8tNXmEyqTHuIJxRdlS5vMh1mj5voN4+QFhVZnlZruezqpWYDiJ8UTzhP0VmeLXCmBk66YrMQ==
+  dependencies:
+    "@types/node" "*"
+
+"@types/prettier@^2.1.1":
+  version "2.7.3"
+  resolved "https://registry.yarnpkg.com/@types/prettier/-/prettier-2.7.3.tgz#3e51a17e291d01d17d3fc61422015a933af7a08f"
+  integrity sha512-+68kP9yzs4LMp7VNh8gdzMSPZFL44MLGqiHWvttYJe+6qnuVr4Ek9wSBQoveqY/r+LwjCcU29kNVkidwim+kYA==
+
+"@types/qs@^6.2.31", "@types/qs@^6.9.7":
+  version "6.9.7"
+  resolved "https://registry.yarnpkg.com/@types/qs/-/qs-6.9.7.tgz#63bb7d067db107cc1e457c303bc25d511febf6cb"
+  integrity sha512-FGa1F62FT09qcrueBA6qYTrJPVDzah9a+493+o2PCXsesWHIn27G98TsSMs3WPNbZIEj4+VJf6saSFpvD+3Zsw==
+
+"@types/responselike@^1.0.0":
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/@types/responselike/-/responselike-1.0.0.tgz#251f4fe7d154d2bad125abe1b429b23afd262e29"
+  integrity sha512-85Y2BjiufFzaMIlvJDvTTB8Fxl2xfLo4HgmHzVBz08w4wDePCTjYw66PdrolO0kzli3yam/YCgRufyo1DdQVTA==
+  dependencies:
+    "@types/node" "*"
+
+"@types/secp256k1@^4.0.1":
+  version "4.0.3"
+  resolved "https://registry.yarnpkg.com/@types/secp256k1/-/secp256k1-4.0.3.tgz#1b8e55d8e00f08ee7220b4d59a6abe89c37a901c"
+  integrity sha512-Da66lEIFeIz9ltsdMZcpQvmrmmoqrfju8pm1BH8WbYjZSwUgCwXLb9C+9XYogwBITnbsSaMdVPb2ekf7TV+03w==
+  dependencies:
+    "@types/node" "*"
+
+"@types/seedrandom@3.0.1":
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/@types/seedrandom/-/seedrandom-3.0.1.tgz#1254750a4fec4aff2ebec088ccd0bb02e91fedb4"
+  integrity sha512-giB9gzDeiCeloIXDgzFBCgjj1k4WxcDrZtGl6h1IqmUPlxF+Nx8Ve+96QCyDZ/HseB/uvDsKbpib9hU5cU53pw==
+
+"@types/semver@^7.3.12":
+  version "7.5.0"
+  resolved "https://registry.yarnpkg.com/@types/semver/-/semver-7.5.0.tgz#591c1ce3a702c45ee15f47a42ade72c2fd78978a"
+  integrity sha512-G8hZ6XJiHnuhQKR7ZmysCeJWE08o8T0AXtk5darsCaTVsYZhhgUrq53jizaR2FvsoeCwJhlmwTjkXBY5Pn/ZHw==
+
+"@typescript-eslint/eslint-plugin@^5.42.1":
+  version "5.62.0"
+  resolved "https://registry.yarnpkg.com/@typescript-eslint/eslint-plugin/-/eslint-plugin-5.62.0.tgz#aeef0328d172b9e37d9bab6dbc13b87ed88977db"
+  integrity sha512-TiZzBSJja/LbhNPvk6yc0JrX9XqhQ0hdh6M2svYfsHGejaKFIAGd9MQ+ERIMzLGlN/kZoYIgdxFV0PuljTKXag==
+  dependencies:
+    "@eslint-community/regexpp" "^4.4.0"
+    "@typescript-eslint/scope-manager" "5.62.0"
+    "@typescript-eslint/type-utils" "5.62.0"
+    "@typescript-eslint/utils" "5.62.0"
+    debug "^4.3.4"
+    graphemer "^1.4.0"
+    ignore "^5.2.0"
+    natural-compare-lite "^1.4.0"
+    semver "^7.3.7"
+    tsutils "^3.21.0"
+
+"@typescript-eslint/parser@^5.42.1":
+  version "5.62.0"
+  resolved "https://registry.yarnpkg.com/@typescript-eslint/parser/-/parser-5.62.0.tgz#1b63d082d849a2fcae8a569248fbe2ee1b8a56c7"
+  integrity sha512-VlJEV0fOQ7BExOsHYAGrgbEiZoi8D+Bl2+f6V2RrXerRSylnp+ZBHmPvaIa8cz0Ajx7WO7Z5RqfgYg7ED1nRhA==
+  dependencies:
+    "@typescript-eslint/scope-manager" "5.62.0"
+    "@typescript-eslint/types" "5.62.0"
+    "@typescript-eslint/typescript-estree" "5.62.0"
+    debug "^4.3.4"
+
+"@typescript-eslint/scope-manager@5.62.0":
+  version "5.62.0"
+  resolved "https://registry.yarnpkg.com/@typescript-eslint/scope-manager/-/scope-manager-5.62.0.tgz#d9457ccc6a0b8d6b37d0eb252a23022478c5460c"
+  integrity sha512-VXuvVvZeQCQb5Zgf4HAxc04q5j+WrNAtNh9OwCsCgpKqESMTu3tF/jhZ3xG6T4NZwWl65Bg8KuS2uEvhSfLl0w==
+  dependencies:
+    "@typescript-eslint/types" "5.62.0"
+    "@typescript-eslint/visitor-keys" "5.62.0"
+
+"@typescript-eslint/type-utils@5.62.0":
+  version "5.62.0"
+  resolved "https://registry.yarnpkg.com/@typescript-eslint/type-utils/-/type-utils-5.62.0.tgz#286f0389c41681376cdad96b309cedd17d70346a"
+  integrity sha512-xsSQreu+VnfbqQpW5vnCJdq1Z3Q0U31qiWmRhr98ONQmcp/yhiPJFPq8MXiJVLiksmOKSjIldZzkebzHuCGzew==
+  dependencies:
+    "@typescript-eslint/typescript-estree" "5.62.0"
+    "@typescript-eslint/utils" "5.62.0"
+    debug "^4.3.4"
+    tsutils "^3.21.0"
+
+"@typescript-eslint/types@5.62.0":
+  version "5.62.0"
+  resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-5.62.0.tgz#258607e60effa309f067608931c3df6fed41fd2f"
+  integrity sha512-87NVngcbVXUahrRTqIK27gD2t5Cu1yuCXxbLcFtCzZGlfyVWWh8mLHkoxzjsB6DDNnvdL+fW8MiwPEJyGJQDgQ==
+
+"@typescript-eslint/typescript-estree@5.62.0":
+  version "5.62.0"
+  resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-5.62.0.tgz#7d17794b77fabcac615d6a48fb143330d962eb9b"
+  integrity sha512-CmcQ6uY7b9y694lKdRB8FEel7JbU/40iSAPomu++SjLMntB+2Leay2LO6i8VnJk58MtE9/nQSFIH6jpyRWyYzA==
+  dependencies:
+    "@typescript-eslint/types" "5.62.0"
+    "@typescript-eslint/visitor-keys" "5.62.0"
+    debug "^4.3.4"
+    globby "^11.1.0"
+    is-glob "^4.0.3"
+    semver "^7.3.7"
+    tsutils "^3.21.0"
+
+"@typescript-eslint/utils@5.62.0":
+  version "5.62.0"
+  resolved "https://registry.yarnpkg.com/@typescript-eslint/utils/-/utils-5.62.0.tgz#141e809c71636e4a75daa39faed2fb5f4b10df86"
+  integrity sha512-n8oxjeb5aIbPFEtmQxQYOLI0i9n5ySBEY/ZEHHZqKQSFnxio1rv6dthascc9dLuwrL0RC5mPCxB7vnAVGAYWAQ==
+  dependencies:
+    "@eslint-community/eslint-utils" "^4.2.0"
+    "@types/json-schema" "^7.0.9"
+    "@types/semver" "^7.3.12"
+    "@typescript-eslint/scope-manager" "5.62.0"
+    "@typescript-eslint/types" "5.62.0"
+    "@typescript-eslint/typescript-estree" "5.62.0"
+    eslint-scope "^5.1.1"
+    semver "^7.3.7"
+
+"@typescript-eslint/visitor-keys@5.62.0":
+  version "5.62.0"
+  resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-5.62.0.tgz#2174011917ce582875954ffe2f6912d5931e353e"
+  integrity sha512-07ny+LHRzQXepkGg6w0mFY41fVUNBrL2Roj/++7V1txKugfjm/Ci/qSND03r2RhlJhJYMcTn9AhhSSqQp0Ysyw==
+  dependencies:
+    "@typescript-eslint/types" "5.62.0"
+    eslint-visitor-keys "^3.3.0"
+
+abbrev@1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/abbrev/-/abbrev-1.1.1.tgz#f8f2c887ad10bf67f634f005b6987fed3179aac8"
+  integrity sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==
+
+abbrev@1.0.x:
+  version "1.0.9"
+  resolved "https://registry.yarnpkg.com/abbrev/-/abbrev-1.0.9.tgz#91b4792588a7738c25f35dd6f63752a2f8776135"
+  integrity sha512-LEyx4aLEC3x6T0UguF6YILf+ntvmOaWsVfENmIW0E9H09vKlLDGelMjjSm0jkDHALj8A8quZ/HapKNigzwge+Q==
+
+abort-controller@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/abort-controller/-/abort-controller-3.0.0.tgz#eaf54d53b62bae4138e809ca225c8439a6efb392"
+  integrity sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==
+  dependencies:
+    event-target-shim "^5.0.0"
+
+abortcontroller-polyfill@^1.7.3:
+  version "1.7.5"
+  resolved "https://registry.yarnpkg.com/abortcontroller-polyfill/-/abortcontroller-polyfill-1.7.5.tgz#6738495f4e901fbb57b6c0611d0c75f76c485bed"
+  integrity sha512-JMJ5soJWP18htbbxJjG7bG6yuI6pRhgJ0scHHTfkUjf6wjP912xZWvM+A4sJK3gqd9E8fcPbDnOefbA9Th/FIQ==
+
+abstract-level@1.0.3, abstract-level@^1.0.0, abstract-level@^1.0.2, abstract-level@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/abstract-level/-/abstract-level-1.0.3.tgz#78a67d3d84da55ee15201486ab44c09560070741"
+  integrity sha512-t6jv+xHy+VYwc4xqZMn2Pa9DjcdzvzZmQGRjTFc8spIbRGHgBrEKbPq+rYXc7CCo0lxgYvSgKVg9qZAhpVQSjA==
+  dependencies:
+    buffer "^6.0.3"
+    catering "^2.1.0"
+    is-buffer "^2.0.5"
+    level-supports "^4.0.0"
+    level-transcoder "^1.0.1"
+    module-error "^1.0.1"
+    queue-microtask "^1.2.3"
+
+abstract-leveldown@7.2.0, abstract-leveldown@^7.2.0:
+  version "7.2.0"
+  resolved "https://registry.yarnpkg.com/abstract-leveldown/-/abstract-leveldown-7.2.0.tgz#08d19d4e26fb5be426f7a57004851b39e1795a2e"
+  integrity sha512-DnhQwcFEaYsvYDnACLZhMmCWd3rkOeEvglpa4q5i/5Jlm3UIsWaxVzuXvDLFCSCWRO3yy2/+V/G7FusFgejnfQ==
+  dependencies:
+    buffer "^6.0.3"
+    catering "^2.0.0"
+    is-buffer "^2.0.5"
+    level-concat-iterator "^3.0.0"
+    level-supports "^2.0.1"
+    queue-microtask "^1.2.3"
+
+accepts@~1.3.8:
+  version "1.3.8"
+  resolved "https://registry.yarnpkg.com/accepts/-/accepts-1.3.8.tgz#0bf0be125b67014adcb0b0921e62db7bffe16b2e"
+  integrity sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==
+  dependencies:
+    mime-types "~2.1.34"
+    negotiator "0.6.3"
+
+acorn-jsx@^5.3.2:
+  version "5.3.2"
+  resolved "https://registry.yarnpkg.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz#7ed5bb55908b3b2f1bc55c6af1653bada7f07937"
+  integrity sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==
+
+acorn-walk@^8.1.1:
+  version "8.2.0"
+  resolved "https://registry.yarnpkg.com/acorn-walk/-/acorn-walk-8.2.0.tgz#741210f2e2426454508853a2f44d0ab83b7f69c1"
+  integrity sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA==
+
+acorn@^8.4.1, acorn@^8.9.0:
+  version "8.10.0"
+  resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.10.0.tgz#8be5b3907a67221a81ab23c7889c4c5526b62ec5"
+  integrity sha512-F0SAmZ8iUtS//m8DmCTA0jlh6TDKkHQyK6xc6V4KDTyZKA9dnvX9/3sRTVQrWm79glUAZbnmmNcdYwUIHWVybw==
+
+address@^1.0.1:
+  version "1.2.2"
+  resolved "https://registry.yarnpkg.com/address/-/address-1.2.2.tgz#2b5248dac5485a6390532c6a517fda2e3faac89e"
+  integrity sha512-4B/qKCfeE/ODUaAUpSwfzazo5x29WD4r3vXiWsB7I2mSDAihwEqKO+g8GELZUQSSAo5e1XTYh3ZVfLyxBc12nA==
+
+adm-zip@^0.4.16:
+  version "0.4.16"
+  resolved "https://registry.yarnpkg.com/adm-zip/-/adm-zip-0.4.16.tgz#cf4c508fdffab02c269cbc7f471a875f05570365"
+  integrity sha512-TFi4HBKSGfIKsK5YCkKaaFG2m4PEDyViZmEwof3MTIgzimHLto6muaHVpbrljdIvIrFZzEq/p4nafOeLcYegrg==
+
+aes-js@3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/aes-js/-/aes-js-3.0.0.tgz#e21df10ad6c2053295bcbb8dab40b09dbea87e4d"
+  integrity sha512-H7wUZRn8WpTq9jocdxQ2c8x2sKo9ZVmzfRE13GiNJXfp7NcKYEdvl3vspKjXox6RIG2VtaRe4JFvxG4rqp2Zuw==
+
+agent-base@6:
+  version "6.0.2"
+  resolved "https://registry.yarnpkg.com/agent-base/-/agent-base-6.0.2.tgz#49fff58577cfee3f37176feab4c22e00f86d7f77"
+  integrity sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==
+  dependencies:
+    debug "4"
+
+aggregate-error@^3.0.0:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/aggregate-error/-/aggregate-error-3.1.0.tgz#92670ff50f5359bdb7a3e0d40d0ec30c5737687a"
+  integrity sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==
+  dependencies:
+    clean-stack "^2.0.0"
+    indent-string "^4.0.0"
+
+ajv@^6.10.0, ajv@^6.12.3, ajv@^6.12.4, ajv@^6.12.6:
+  version "6.12.6"
+  resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4"
+  integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==
+  dependencies:
+    fast-deep-equal "^3.1.1"
+    fast-json-stable-stringify "^2.0.0"
+    json-schema-traverse "^0.4.1"
+    uri-js "^4.2.2"
+
+ajv@^8.0.1:
+  version "8.12.0"
+  resolved "https://registry.yarnpkg.com/ajv/-/ajv-8.12.0.tgz#d1a0527323e22f53562c567c00991577dfbe19d1"
+  integrity sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==
+  dependencies:
+    fast-deep-equal "^3.1.1"
+    json-schema-traverse "^1.0.0"
+    require-from-string "^2.0.2"
+    uri-js "^4.2.2"
+
+amdefine@>=0.0.4:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/amdefine/-/amdefine-1.0.1.tgz#4a5282ac164729e93619bcfd3ad151f817ce91f5"
+  integrity sha512-S2Hw0TtNkMJhIabBwIojKL9YHO5T0n5eNqWJ7Lrlel/zDbftQpxpapi8tZs3X1HWa+u+QeydGmzzNU0m09+Rcg==
+
+ansi-colors@3.2.3:
+  version "3.2.3"
+  resolved "https://registry.yarnpkg.com/ansi-colors/-/ansi-colors-3.2.3.tgz#57d35b8686e851e2cc04c403f1c00203976a1813"
+  integrity sha512-LEHHyuhlPY3TmuUYMh2oz89lTShfvgbmzaBcxve9t/9Wuy7Dwf4yoAKcND7KFT1HAQfqZ12qtc+DUrBMeKF9nw==
+
+ansi-colors@4.1.1:
+  version "4.1.1"
+  resolved "https://registry.yarnpkg.com/ansi-colors/-/ansi-colors-4.1.1.tgz#cbb9ae256bf750af1eab344f229aa27fe94ba348"
+  integrity sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==
+
+ansi-colors@^4.1.1:
+  version "4.1.3"
+  resolved "https://registry.yarnpkg.com/ansi-colors/-/ansi-colors-4.1.3.tgz#37611340eb2243e70cc604cad35d63270d48781b"
+  integrity sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw==
+
+ansi-escapes@^4.3.0:
+  version "4.3.2"
+  resolved "https://registry.yarnpkg.com/ansi-escapes/-/ansi-escapes-4.3.2.tgz#6b2291d1db7d98b6521d5f1efa42d0f3a9feb65e"
+  integrity sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==
+  dependencies:
+    type-fest "^0.21.3"
+
+ansi-regex@^2.0.0:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-2.1.1.tgz#c3b33ab5ee360d86e0e628f0468ae7ef27d654df"
+  integrity sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==
+
+ansi-regex@^3.0.0:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-3.0.1.tgz#123d6479e92ad45ad897d4054e3c7ca7db4944e1"
+  integrity sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==
+
+ansi-regex@^4.1.0:
+  version "4.1.1"
+  resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-4.1.1.tgz#164daac87ab2d6f6db3a29875e2d1766582dabed"
+  integrity sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==
+
+ansi-regex@^5.0.1:
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.1.tgz#082cb2c89c9fe8659a311a53bd6a4dc5301db304"
+  integrity sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==
+
+ansi-styles@^3.2.0, ansi-styles@^3.2.1:
+  version "3.2.1"
+  resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.1.tgz#41fbb20243e50b12be0f04b8dedbf07520ce841d"
+  integrity sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==
+  dependencies:
+    color-convert "^1.9.0"
+
+ansi-styles@^4.0.0, ansi-styles@^4.1.0:
+  version "4.3.0"
+  resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937"
+  integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==
+  dependencies:
+    color-convert "^2.0.1"
+
+antlr4@^4.11.0:
+  version "4.13.0"
+  resolved "https://registry.yarnpkg.com/antlr4/-/antlr4-4.13.0.tgz#25c0b17f0d9216de114303d38bafd6f181d5447f"
+  integrity sha512-zooUbt+UscjnWyOrsuY/tVFL4rwrAGwOivpQmvmUDE22hy/lUA467Rc1rcixyRwcRUIXFYBwv7+dClDSHdmmew==
+
+antlr4ts@^0.5.0-alpha.4:
+  version "0.5.0-alpha.4"
+  resolved "https://registry.yarnpkg.com/antlr4ts/-/antlr4ts-0.5.0-alpha.4.tgz#71702865a87478ed0b40c0709f422cf14d51652a"
+  integrity sha512-WPQDt1B74OfPv/IMS2ekXAKkTZIHl88uMetg6q3OTqgFxZ/dxDXI0EWLyZid/1Pe6hTftyg5N7gel5wNAGxXyQ==
+
+anymatch@~3.1.1, anymatch@~3.1.2:
+  version "3.1.3"
+  resolved "https://registry.yarnpkg.com/anymatch/-/anymatch-3.1.3.tgz#790c58b19ba1720a84205b57c618d5ad8524973e"
+  integrity sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==
+  dependencies:
+    normalize-path "^3.0.0"
+    picomatch "^2.0.4"
+
+arg@^4.1.0:
+  version "4.1.3"
+  resolved "https://registry.yarnpkg.com/arg/-/arg-4.1.3.tgz#269fc7ad5b8e42cb63c896d5666017261c144089"
+  integrity sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==
+
+argparse@^1.0.7:
+  version "1.0.10"
+  resolved "https://registry.yarnpkg.com/argparse/-/argparse-1.0.10.tgz#bcd6791ea5ae09725e17e5ad988134cd40b3d911"
+  integrity sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==
+  dependencies:
+    sprintf-js "~1.0.2"
+
+argparse@^2.0.1:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/argparse/-/argparse-2.0.1.tgz#246f50f3ca78a3240f6c997e8a9bd1eac49e4b38"
+  integrity sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==
+
+array-back@^3.0.1, array-back@^3.1.0:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/array-back/-/array-back-3.1.0.tgz#b8859d7a508871c9a7b2cf42f99428f65e96bfb0"
+  integrity sha512-TkuxA4UCOvxuDK6NZYXCalszEzj+TLszyASooky+i742l9TqsOdYCMJJupxRic61hwquNtppB3hgcuq9SVSH1Q==
+
+array-back@^4.0.1, array-back@^4.0.2:
+  version "4.0.2"
+  resolved "https://registry.yarnpkg.com/array-back/-/array-back-4.0.2.tgz#8004e999a6274586beeb27342168652fdb89fa1e"
+  integrity sha512-NbdMezxqf94cnNfWLL7V/im0Ub+Anbb0IoZhvzie8+4HJ4nMQuzHuy49FkGYCJK2yAloZ3meiB6AVMClbrI1vg==
+
+array-buffer-byte-length@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/array-buffer-byte-length/-/array-buffer-byte-length-1.0.0.tgz#fabe8bc193fea865f317fe7807085ee0dee5aead"
+  integrity sha512-LPuwb2P+NrQw3XhxGc36+XSvuBPopovXYTR9Ew++Du9Yb/bx5AzBfrIsBoj0EZUifjQU+sHL21sseZ3jerWO/A==
+  dependencies:
+    call-bind "^1.0.2"
+    is-array-buffer "^3.0.1"
+
+array-flatten@1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-1.1.1.tgz#9a5f699051b1e7073328f2a008968b64ea2955d2"
+  integrity sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==
+
+array-includes@^3.1.6:
+  version "3.1.6"
+  resolved "https://registry.yarnpkg.com/array-includes/-/array-includes-3.1.6.tgz#9e9e720e194f198266ba9e18c29e6a9b0e4b225f"
+  integrity sha512-sgTbLvL6cNnw24FnbaDyjmvddQ2ML8arZsgaJhoABMoplz/4QRhtrYS+alr1BUM1Bwp6dhx8vVCBSLG+StwOFw==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.1.4"
+    es-abstract "^1.20.4"
+    get-intrinsic "^1.1.3"
+    is-string "^1.0.7"
+
+array-union@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/array-union/-/array-union-2.1.0.tgz#b798420adbeb1de828d84acd8a2e23d3efe85e8d"
+  integrity sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==
+
+array-uniq@1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/array-uniq/-/array-uniq-1.0.3.tgz#af6ac877a25cc7f74e058894753858dfdb24fdb6"
+  integrity sha512-MNha4BWQ6JbwhFhj03YK552f7cb3AzoE8SzeljgChvL1dl3IcvggXVz1DilzySZkCja+CXuZbdW7yATchWn8/Q==
+
+array.prototype.findlastindex@^1.2.2:
+  version "1.2.2"
+  resolved "https://registry.yarnpkg.com/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.2.tgz#bc229aef98f6bd0533a2bc61ff95209875526c9b"
+  integrity sha512-tb5thFFlUcp7NdNF6/MpDk/1r/4awWG1FIz3YqDf+/zJSTezBb+/5WViH41obXULHVpDzoiCLpJ/ZO9YbJMsdw==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.1.4"
+    es-abstract "^1.20.4"
+    es-shim-unscopables "^1.0.0"
+    get-intrinsic "^1.1.3"
+
+array.prototype.flat@^1.3.1:
+  version "1.3.1"
+  resolved "https://registry.yarnpkg.com/array.prototype.flat/-/array.prototype.flat-1.3.1.tgz#ffc6576a7ca3efc2f46a143b9d1dda9b4b3cf5e2"
+  integrity sha512-roTU0KWIOmJ4DRLmwKd19Otg0/mT3qPNt0Qb3GWW8iObuZXxrjB/pzn0R3hqpRSWg4HCwqx+0vwOnWnvlOyeIA==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.1.4"
+    es-abstract "^1.20.4"
+    es-shim-unscopables "^1.0.0"
+
+array.prototype.flatmap@^1.3.1:
+  version "1.3.1"
+  resolved "https://registry.yarnpkg.com/array.prototype.flatmap/-/array.prototype.flatmap-1.3.1.tgz#1aae7903c2100433cb8261cd4ed310aab5c4a183"
+  integrity sha512-8UGn9O1FDVvMNB0UlLv4voxRMze7+FpHyF5mSMRjWHUMlpoDViniy05870VlxhfgTnLbpuwTzvD76MTtWxB/mQ==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.1.4"
+    es-abstract "^1.20.4"
+    es-shim-unscopables "^1.0.0"
+
+array.prototype.reduce@^1.0.5:
+  version "1.0.5"
+  resolved "https://registry.yarnpkg.com/array.prototype.reduce/-/array.prototype.reduce-1.0.5.tgz#6b20b0daa9d9734dd6bc7ea66b5bbce395471eac"
+  integrity sha512-kDdugMl7id9COE8R7MHF5jWk7Dqt/fs4Pv+JXoICnYwqpjjjbUurz6w5fT5IG6brLdJhv6/VoHB0H7oyIBXd+Q==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.1.4"
+    es-abstract "^1.20.4"
+    es-array-method-boxes-properly "^1.0.0"
+    is-string "^1.0.7"
+
+arraybuffer.prototype.slice@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.1.tgz#9b5ea3868a6eebc30273da577eb888381c0044bb"
+  integrity sha512-09x0ZWFEjj4WD8PDbykUwo3t9arLn8NIzmmYEJFpYekOAQjpkGSyrQhNoRTcwwcFRu+ycWF78QZ63oWTqSjBcw==
+  dependencies:
+    array-buffer-byte-length "^1.0.0"
+    call-bind "^1.0.2"
+    define-properties "^1.2.0"
+    get-intrinsic "^1.2.1"
+    is-array-buffer "^3.0.2"
+    is-shared-array-buffer "^1.0.2"
+
+asap@~2.0.6:
+  version "2.0.6"
+  resolved "https://registry.yarnpkg.com/asap/-/asap-2.0.6.tgz#e50347611d7e690943208bbdafebcbc2fb866d46"
+  integrity sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==
+
+asn1@~0.2.3:
+  version "0.2.6"
+  resolved "https://registry.yarnpkg.com/asn1/-/asn1-0.2.6.tgz#0d3a7bb6e64e02a90c0303b31f292868ea09a08d"
+  integrity sha512-ix/FxPn0MDjeyJ7i/yoHGFt/EX6LyNbxSEhPPXODPL+KB0VPk86UYfL0lMdy+KCnv+fmvIzySwaK5COwqVbWTQ==
+  dependencies:
+    safer-buffer "~2.1.0"
+
+assert-plus@1.0.0, assert-plus@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/assert-plus/-/assert-plus-1.0.0.tgz#f12e0f3c5d77b0b1cdd9146942e4e96c1e4dd525"
+  integrity sha512-NfJ4UzBCcQGLDlQq7nHxH+tv3kyZ0hHQqF5BO6J7tNJeP5do1llPr8dZ8zHonfhAu0PHAdMkSo+8o0wxg9lZWw==
+
+assertion-error@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/assertion-error/-/assertion-error-1.1.0.tgz#e60b6b0e8f301bd97e5375215bda406c85118c0b"
+  integrity sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==
+
+ast-parents@^0.0.1:
+  version "0.0.1"
+  resolved "https://registry.yarnpkg.com/ast-parents/-/ast-parents-0.0.1.tgz#508fd0f05d0c48775d9eccda2e174423261e8dd3"
+  integrity sha512-XHusKxKz3zoYk1ic8Un640joHbFMhbqneyoZfoKnEGtf2ey9Uh/IdpcQplODdO/kENaMIWsD0nJm4+wX3UNLHA==
+
+astral-regex@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/astral-regex/-/astral-regex-2.0.0.tgz#483143c567aeed4785759c0865786dc77d7d2e31"
+  integrity sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==
+
+async-eventemitter@0.2.4, async-eventemitter@^0.2.4:
+  version "0.2.4"
+  resolved "https://registry.yarnpkg.com/async-eventemitter/-/async-eventemitter-0.2.4.tgz#f5e7c8ca7d3e46aab9ec40a292baf686a0bafaca"
+  integrity sha512-pd20BwL7Yt1zwDFy+8MX8F1+WCT8aQeKj0kQnTrH9WaeRETlRamVhD0JtRPmrV4GfOJ2F9CvdQkZeZhnh2TuHw==
+  dependencies:
+    async "^2.4.0"
+
+async-limiter@~1.0.0:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/async-limiter/-/async-limiter-1.0.1.tgz#dd379e94f0db8310b08291f9d64c3209766617fd"
+  integrity sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==
+
+async@1.x:
+  version "1.5.2"
+  resolved "https://registry.yarnpkg.com/async/-/async-1.5.2.tgz#ec6a61ae56480c0c3cb241c95618e20892f9672a"
+  integrity sha512-nSVgobk4rv61R9PUSDtYt7mPVB2olxNR5RWJcAsH676/ef11bUZwvu7+RGYrYauVdDPcO519v68wRhXQtxsV9w==
+
+async@^2.4.0:
+  version "2.6.4"
+  resolved "https://registry.yarnpkg.com/async/-/async-2.6.4.tgz#706b7ff6084664cd7eae713f6f965433b5504221"
+  integrity sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==
+  dependencies:
+    lodash "^4.17.14"
+
+asynckit@^0.4.0:
+  version "0.4.0"
+  resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79"
+  integrity sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==
+
+available-typed-arrays@^1.0.5:
+  version "1.0.5"
+  resolved "https://registry.yarnpkg.com/available-typed-arrays/-/available-typed-arrays-1.0.5.tgz#92f95616501069d07d10edb2fc37d3e1c65123b7"
+  integrity sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw==
+
+aws-sign2@~0.7.0:
+  version "0.7.0"
+  resolved "https://registry.yarnpkg.com/aws-sign2/-/aws-sign2-0.7.0.tgz#b46e890934a9591f2d2f6f86d7e6a9f1b3fe76a8"
+  integrity sha512-08kcGqnYf/YmjoRhfxyu+CLxBjUtHLXLXX/vUfx9l2LYzG3c1m61nrpyFUZI6zeS+Li/wWMMidD9KgrqtGq3mA==
+
+aws4@^1.8.0:
+  version "1.12.0"
+  resolved "https://registry.yarnpkg.com/aws4/-/aws4-1.12.0.tgz#ce1c9d143389679e253b314241ea9aa5cec980d3"
+  integrity sha512-NmWvPnx0F1SfrQbYwOi7OeaNGokp9XhzNioJ/CSBs8Qa4vxug81mhJEAVZwxXuBmYB5KDRfMq/F3RR0BIU7sWg==
+
+axios@^0.21.1:
+  version "0.21.4"
+  resolved "https://registry.yarnpkg.com/axios/-/axios-0.21.4.tgz#c67b90dc0568e5c1cf2b0b858c43ba28e2eda575"
+  integrity sha512-ut5vewkiu8jjGBdqpM44XxjuCjq9LAKeHVmoVfHVzy8eHgxxq8SbAVQNovDA8mVi05kP0Ea/n/UzcSHcTJQfNg==
+  dependencies:
+    follow-redirects "^1.14.0"
+
+axios@^0.27.2:
+  version "0.27.2"
+  resolved "https://registry.yarnpkg.com/axios/-/axios-0.27.2.tgz#207658cc8621606e586c85db4b41a750e756d972"
+  integrity sha512-t+yRIyySRTp/wua5xEr+z1q60QmLq8ABsS5O9Me1AsE5dfKqgnCFzwiCZZ/cGNd1lq4/7akDWMxdhVlucjmnOQ==
+  dependencies:
+    follow-redirects "^1.14.9"
+    form-data "^4.0.0"
+
+balanced-match@^1.0.0:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee"
+  integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==
+
+base-x@^3.0.2, base-x@^3.0.8:
+  version "3.0.9"
+  resolved "https://registry.yarnpkg.com/base-x/-/base-x-3.0.9.tgz#6349aaabb58526332de9f60995e548a53fe21320"
+  integrity sha512-H7JU6iBHTal1gp56aKoaa//YUxEaAOUiydvrV/pILqIHXTtqxSkATOnDA2u+jZ/61sD+L/412+7kzXRtWukhpQ==
+  dependencies:
+    safe-buffer "^5.0.1"
+
+base64-js@^1.3.1:
+  version "1.5.1"
+  resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a"
+  integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==
+
+bcrypt-pbkdf@^1.0.0:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz#a4301d389b6a43f9b67ff3ca11a3f6637e360e9e"
+  integrity sha512-qeFIXtP4MSoi6NLqO12WfqARWWuCKi2Rn/9hJLEmtB5yTNr9DqFWkJRCf2qShWzPeAMRnOgCrq0sg/KLv5ES9w==
+  dependencies:
+    tweetnacl "^0.14.3"
+
+bech32@1.1.4, bech32@^1.1.3:
+  version "1.1.4"
+  resolved "https://registry.yarnpkg.com/bech32/-/bech32-1.1.4.tgz#e38c9f37bf179b8eb16ae3a772b40c356d4832e9"
+  integrity sha512-s0IrSOzLlbvX7yp4WBfPITzpAU8sqQcpsmwXDiKwrG4r491vwCO/XpejasRNl0piBMe/DvP4Tz0mIS/X1DPJBQ==
+
+big-integer@1.6.36:
+  version "1.6.36"
+  resolved "https://registry.yarnpkg.com/big-integer/-/big-integer-1.6.36.tgz#78631076265d4ae3555c04f85e7d9d2f3a071a36"
+  integrity sha512-t70bfa7HYEA1D9idDbmuv7YbsbVkQ+Hp+8KFSul4aE5e/i1bjCNIRYJZlA8Q8p0r9T8cF/RVvwUgRA//FydEyg==
+
+big.js@^6.0.3:
+  version "6.2.1"
+  resolved "https://registry.yarnpkg.com/big.js/-/big.js-6.2.1.tgz#7205ce763efb17c2e41f26f121c420c6a7c2744f"
+  integrity sha512-bCtHMwL9LeDIozFn+oNhhFoq+yQ3BNdnsLSASUxLciOb1vgvpHsIO1dsENiGMgbb4SkP5TrzWzRiLddn8ahVOQ==
+
+bigint-crypto-utils@^3.0.23:
+  version "3.3.0"
+  resolved "https://registry.yarnpkg.com/bigint-crypto-utils/-/bigint-crypto-utils-3.3.0.tgz#72ad00ae91062cf07f2b1def9594006c279c1d77"
+  integrity sha512-jOTSb+drvEDxEq6OuUybOAv/xxoh3cuYRUIPyu8sSHQNKM303UQ2R1DAo45o1AkcIXw6fzbaFI1+xGGdaXs2lg==
+
+bignumber.js@*, bignumber.js@^9.0.0, bignumber.js@^9.0.1:
+  version "9.1.1"
+  resolved "https://registry.yarnpkg.com/bignumber.js/-/bignumber.js-9.1.1.tgz#c4df7dc496bd849d4c9464344c1aa74228b4dac6"
+  integrity sha512-pHm4LsMJ6lzgNGVfZHjMoO8sdoRhOzOH4MLmY65Jg70bpxCKu5iOHNJyfF6OyvYw7t8Fpf35RuzUyqnQsj8Vig==
+
+bignumber.js@^7.2.1:
+  version "7.2.1"
+  resolved "https://registry.yarnpkg.com/bignumber.js/-/bignumber.js-7.2.1.tgz#80c048759d826800807c4bfd521e50edbba57a5f"
+  integrity sha512-S4XzBk5sMB+Rcb/LNcpzXr57VRTxgAvaAEDAl1AwRx27j00hT84O6OkteE7u8UB3NuaaygCRrEpqox4uDOrbdQ==
+
+binary-extensions@^2.0.0:
+  version "2.2.0"
+  resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-2.2.0.tgz#75f502eeaf9ffde42fc98829645be4ea76bd9e2d"
+  integrity sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==
+
+bl@^4.1.0:
+  version "4.1.0"
+  resolved "https://registry.yarnpkg.com/bl/-/bl-4.1.0.tgz#451535264182bec2fbbc83a62ab98cf11d9f7b3a"
+  integrity sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==
+  dependencies:
+    buffer "^5.5.0"
+    inherits "^2.0.4"
+    readable-stream "^3.4.0"
+
+blakejs@^1.1.0:
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/blakejs/-/blakejs-1.2.1.tgz#5057e4206eadb4a97f7c0b6e197a505042fc3814"
+  integrity sha512-QXUSXI3QVc/gJME0dBpXrag1kbzOqCjCX8/b54ntNyW6sjtoqxqRk3LTmXzaJoh71zMsDCjM+47jS7XiwN/+fQ==
+
+bluebird@^3.4.7, bluebird@^3.5.0, bluebird@^3.5.2:
+  version "3.7.2"
+  resolved "https://registry.yarnpkg.com/bluebird/-/bluebird-3.7.2.tgz#9f229c15be272454ffa973ace0dbee79a1b0c36f"
+  integrity sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==
+
+bn-chai@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/bn-chai/-/bn-chai-1.0.1.tgz#5d6e9654162602a527b08a1546e60cfb44213725"
+  integrity sha512-7rJXt21DwYiLLpvzLaACixBBoUGkRV1iuFD3wElEhw8Ji9IiY/QsJRtvW+c7ChRgEOyLQkGaSGFUUqBKm21SNA==
+
+bn.js@4.11.6:
+  version "4.11.6"
+  resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-4.11.6.tgz#53344adb14617a13f6e8dd2ce28905d1c0ba3215"
+  integrity sha512-XWwnNNFCuuSQ0m3r3C4LE3EiORltHd9M05pq6FOlVeiophzRbMo50Sbz1ehl8K3Z+jw9+vmgnXefY1hz8X+2wA==
+
+bn.js@^4.11.0, bn.js@^4.11.6, bn.js@^4.11.8, bn.js@^4.11.9:
+  version "4.12.0"
+  resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-4.12.0.tgz#775b3f278efbb9718eec7361f483fb36fbbfea88"
+  integrity sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==
+
+bn.js@^5.1.2, bn.js@^5.1.3, bn.js@^5.2.0, bn.js@^5.2.1:
+  version "5.2.1"
+  resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-5.2.1.tgz#0bc527a6a0d18d0aa8d5b0538ce4a77dccfa7b70"
+  integrity sha512-eXRvHzWyYPBuB4NBy0cmYQjGitUrtqwbvlzP3G6VFnNRbsZQIxQ10PbKKHt8gZ/HW/D/747aDl+QkDqg3KQLMQ==
+
+body-parser@1.20.1:
+  version "1.20.1"
+  resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.20.1.tgz#b1812a8912c195cd371a3ee5e66faa2338a5c668"
+  integrity sha512-jWi7abTbYwajOytWCQc37VulmWiRae5RyTpaCyDcS5/lMdtwSz5lOpDE67srw/HYe35f1z3fDQw+3txg7gNtWw==
+  dependencies:
+    bytes "3.1.2"
+    content-type "~1.0.4"
+    debug "2.6.9"
+    depd "2.0.0"
+    destroy "1.2.0"
+    http-errors "2.0.0"
+    iconv-lite "0.4.24"
+    on-finished "2.4.1"
+    qs "6.11.0"
+    raw-body "2.5.1"
+    type-is "~1.6.18"
+    unpipe "1.0.0"
+
+body-parser@^1.16.0:
+  version "1.20.2"
+  resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.20.2.tgz#6feb0e21c4724d06de7ff38da36dad4f57a747fd"
+  integrity sha512-ml9pReCu3M61kGlqoTm2umSXTlRTuGTx0bfYj+uIUKKYycG5NtSbeetV3faSU6R7ajOPw0g/J1PvK4qNy7s5bA==
+  dependencies:
+    bytes "3.1.2"
+    content-type "~1.0.5"
+    debug "2.6.9"
+    depd "2.0.0"
+    destroy "1.2.0"
+    http-errors "2.0.0"
+    iconv-lite "0.4.24"
+    on-finished "2.4.1"
+    qs "6.11.0"
+    raw-body "2.5.2"
+    type-is "~1.6.18"
+    unpipe "1.0.0"
+
+boolbase@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/boolbase/-/boolbase-1.0.0.tgz#68dff5fbe60c51eb37725ea9e3ed310dcc1e776e"
+  integrity sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==
+
+brace-expansion@^1.1.7:
+  version "1.1.11"
+  resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd"
+  integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==
+  dependencies:
+    balanced-match "^1.0.0"
+    concat-map "0.0.1"
+
+brace-expansion@^2.0.1:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-2.0.1.tgz#1edc459e0f0c548486ecf9fc99f2221364b9a0ae"
+  integrity sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==
+  dependencies:
+    balanced-match "^1.0.0"
+
+braces@^3.0.2, braces@~3.0.2:
+  version "3.0.2"
+  resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107"
+  integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==
+  dependencies:
+    fill-range "^7.0.1"
+
+brorand@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/brorand/-/brorand-1.1.0.tgz#12c25efe40a45e3c323eb8675a0a0ce57b22371f"
+  integrity sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==
+
+browser-level@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/browser-level/-/browser-level-1.0.1.tgz#36e8c3183d0fe1c405239792faaab5f315871011"
+  integrity sha512-XECYKJ+Dbzw0lbydyQuJzwNXtOpbMSq737qxJN11sIRTErOMShvDpbzTlgju7orJKvx4epULolZAuJGLzCmWRQ==
+  dependencies:
+    abstract-level "^1.0.2"
+    catering "^2.1.1"
+    module-error "^1.0.2"
+    run-parallel-limit "^1.1.0"
+
+browser-stdout@1.3.1:
+  version "1.3.1"
+  resolved "https://registry.yarnpkg.com/browser-stdout/-/browser-stdout-1.3.1.tgz#baa559ee14ced73452229bad7326467c61fabd60"
+  integrity sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==
+
+browserify-aes@^1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/browserify-aes/-/browserify-aes-1.2.0.tgz#326734642f403dabc3003209853bb70ad428ef48"
+  integrity sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA==
+  dependencies:
+    buffer-xor "^1.0.3"
+    cipher-base "^1.0.0"
+    create-hash "^1.1.0"
+    evp_bytestokey "^1.0.3"
+    inherits "^2.0.1"
+    safe-buffer "^5.0.1"
+
+bs58@^4.0.0, bs58@^4.0.1:
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/bs58/-/bs58-4.0.1.tgz#be161e76c354f6f788ae4071f63f34e8c4f0a42a"
+  integrity sha512-Ok3Wdf5vOIlBrgCvTq96gBkJw+JUEzdBgyaza5HLtPm7yTHkjRy8+JzNyHF7BHa0bNWOQIp3m5YF0nnFcOIKLw==
+  dependencies:
+    base-x "^3.0.2"
+
+bs58check@^2.1.2:
+  version "2.1.2"
+  resolved "https://registry.yarnpkg.com/bs58check/-/bs58check-2.1.2.tgz#53b018291228d82a5aa08e7d796fdafda54aebfc"
+  integrity sha512-0TS1jicxdU09dwJMNZtVAfzPi6Q6QeN0pM1Fkzrjn+XYHvzMKPU3pHVpva+769iNVSfIYWf7LJ6WR+BuuMf8cA==
+  dependencies:
+    bs58 "^4.0.0"
+    create-hash "^1.1.0"
+    safe-buffer "^5.1.2"
+
+buffer-from@^1.0.0:
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.2.tgz#2b146a6fd72e80b4f55d255f35ed59a3a9a41bd5"
+  integrity sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==
+
+buffer-to-arraybuffer@^0.0.5:
+  version "0.0.5"
+  resolved "https://registry.yarnpkg.com/buffer-to-arraybuffer/-/buffer-to-arraybuffer-0.0.5.tgz#6064a40fa76eb43c723aba9ef8f6e1216d10511a"
+  integrity sha512-3dthu5CYiVB1DEJp61FtApNnNndTckcqe4pFcLdvHtrpG+kcyekCJKg4MRiDcFW7A6AODnXB9U4dwQiCW5kzJQ==
+
+buffer-xor@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/buffer-xor/-/buffer-xor-1.0.3.tgz#26e61ed1422fb70dd42e6e36729ed51d855fe8d9"
+  integrity sha512-571s0T7nZWK6vB67HI5dyUF7wXiNcfaPPPTl6zYCNApANjIvYJTg7hlud/+cJpdAhS7dVzqMLmfhfHR3rAcOjQ==
+
+buffer@6.0.3, buffer@^6.0.3:
+  version "6.0.3"
+  resolved "https://registry.yarnpkg.com/buffer/-/buffer-6.0.3.tgz#2ace578459cc8fbe2a70aaa8f52ee63b6a74c6c6"
+  integrity sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==
+  dependencies:
+    base64-js "^1.3.1"
+    ieee754 "^1.2.1"
+
+buffer@^5.0.5, buffer@^5.5.0, buffer@^5.6.0:
+  version "5.7.1"
+  resolved "https://registry.yarnpkg.com/buffer/-/buffer-5.7.1.tgz#ba62e7c13133053582197160851a8f648e99eed0"
+  integrity sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==
+  dependencies:
+    base64-js "^1.3.1"
+    ieee754 "^1.1.13"
+
+bufferutil@4.0.5:
+  version "4.0.5"
+  resolved "https://registry.yarnpkg.com/bufferutil/-/bufferutil-4.0.5.tgz#da9ea8166911cc276bf677b8aed2d02d31f59028"
+  integrity sha512-HTm14iMQKK2FjFLRTM5lAVcyaUzOnqbPtesFIvREgXpJHdQm8bWS+GkQgIkfaBYRHuCnea7w8UVNfwiAQhlr9A==
+  dependencies:
+    node-gyp-build "^4.3.0"
+
+bufferutil@^4.0.1:
+  version "4.0.7"
+  resolved "https://registry.yarnpkg.com/bufferutil/-/bufferutil-4.0.7.tgz#60c0d19ba2c992dd8273d3f73772ffc894c153ad"
+  integrity sha512-kukuqc39WOHtdxtw4UScxF/WVnMFVSQVKhtx3AjZJzhd0RGZZldcrfSEbVsWWe6KNH253574cq5F+wpv0G9pJw==
+  dependencies:
+    node-gyp-build "^4.3.0"
+
+bytes@3.1.2:
+  version "3.1.2"
+  resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.1.2.tgz#8b0beeb98605adf1b128fa4386403c009e0221a5"
+  integrity sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==
+
+cacheable-lookup@^5.0.3:
+  version "5.0.4"
+  resolved "https://registry.yarnpkg.com/cacheable-lookup/-/cacheable-lookup-5.0.4.tgz#5a6b865b2c44357be3d5ebc2a467b032719a7005"
+  integrity sha512-2/kNscPhpcxrOigMZzbiWF7dz8ilhb/nIHU3EyZiXWXpeq/au8qJ8VhdftMkty3n7Gj6HIGalQG8oiBNB3AJgA==
+
+cacheable-lookup@^6.0.4:
+  version "6.1.0"
+  resolved "https://registry.yarnpkg.com/cacheable-lookup/-/cacheable-lookup-6.1.0.tgz#0330a543471c61faa4e9035db583aad753b36385"
+  integrity sha512-KJ/Dmo1lDDhmW2XDPMo+9oiy/CeqosPguPCrgcVzKyZrL6pM1gU2GmPY/xo6OQPTUaA/c0kwHuywB4E6nmT9ww==
+
+cacheable-request@^7.0.2:
+  version "7.0.4"
+  resolved "https://registry.yarnpkg.com/cacheable-request/-/cacheable-request-7.0.4.tgz#7a33ebf08613178b403635be7b899d3e69bbe817"
+  integrity sha512-v+p6ongsrp0yTGbJXjgxPow2+DL93DASP4kXCDKb8/bwRtt9OEF3whggkkDkGNzgcWy2XaF4a8nZglC7uElscg==
+  dependencies:
+    clone-response "^1.0.2"
+    get-stream "^5.1.0"
+    http-cache-semantics "^4.0.0"
+    keyv "^4.0.0"
+    lowercase-keys "^2.0.0"
+    normalize-url "^6.0.1"
+    responselike "^2.0.0"
+
+call-bind@^1.0.0, call-bind@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/call-bind/-/call-bind-1.0.2.tgz#b1d4e89e688119c3c9a903ad30abb2f6a919be3c"
+  integrity sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==
+  dependencies:
+    function-bind "^1.1.1"
+    get-intrinsic "^1.0.2"
+
+callsites@^3.0.0:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73"
+  integrity sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==
+
+camel-case@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/camel-case/-/camel-case-3.0.0.tgz#ca3c3688a4e9cf3a4cda777dc4dcbc713249cf73"
+  integrity sha512-+MbKztAYHXPr1jNTSKQF52VpcFjwY5RkR7fxksV8Doo4KAYc5Fl4UJRgthBbTmEx8C54DqahhbLJkDwjI3PI/w==
+  dependencies:
+    no-case "^2.2.0"
+    upper-case "^1.1.1"
+
+camelcase@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-3.0.0.tgz#32fc4b9fcdaf845fcdf7e73bb97cac2261f0ab0a"
+  integrity sha512-4nhGqUkc4BqbBBB4Q6zLuD7lzzrHYrjKGeYaEji/3tFR5VdJu9v+LilhGIVe8wxEJPPOeWo7eg8dwY13TZ1BNg==
+
+camelcase@^4.1.0:
+  version "4.1.0"
+  resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-4.1.0.tgz#d545635be1e33c542649c69173e5de6acfae34dd"
+  integrity sha512-FxAv7HpHrXbh3aPo4o2qxHay2lkLY3x5Mw3KeE4KQE8ysVfziWeRZDwcjauvwBSGEC/nXUPzZy8zeh4HokqOnw==
+
+camelcase@^5.0.0:
+  version "5.3.1"
+  resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-5.3.1.tgz#e3c9b31569e106811df242f715725a1f4c494320"
+  integrity sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==
+
+camelcase@^6.0.0:
+  version "6.3.0"
+  resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-6.3.0.tgz#5685b95eb209ac9c0c177467778c9c84df58ba9a"
+  integrity sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==
+
+caseless@^0.12.0, caseless@~0.12.0:
+  version "0.12.0"
+  resolved "https://registry.yarnpkg.com/caseless/-/caseless-0.12.0.tgz#1b681c21ff84033c826543090689420d187151dc"
+  integrity sha512-4tYFyifaFfGacoiObjJegolkwSU4xQNGbVgUiNYVUxbQ2x2lUsFvY4hVgVzGiIe6WLOPqycWXA40l+PWsxthUw==
+
+catering@^2.0.0, catering@^2.1.0, catering@^2.1.1:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/catering/-/catering-2.1.1.tgz#66acba06ed5ee28d5286133982a927de9a04b510"
+  integrity sha512-K7Qy8O9p76sL3/3m7/zLKbRkyOlSZAgzEaLhyj2mXS8PsCud2Eo4hAb8aLtZqHh0QGqLcb9dlJSu6lHRVENm1w==
+
+cbor@^5.2.0:
+  version "5.2.0"
+  resolved "https://registry.yarnpkg.com/cbor/-/cbor-5.2.0.tgz#4cca67783ccd6de7b50ab4ed62636712f287a67c"
+  integrity sha512-5IMhi9e1QU76ppa5/ajP1BmMWZ2FHkhAhjeVKQ/EFCgYSEaeVaoGtL7cxJskf9oCCk+XjzaIdc3IuU/dbA/o2A==
+  dependencies:
+    bignumber.js "^9.0.1"
+    nofilter "^1.0.4"
+
+cbor@^8.1.0:
+  version "8.1.0"
+  resolved "https://registry.yarnpkg.com/cbor/-/cbor-8.1.0.tgz#cfc56437e770b73417a2ecbfc9caf6b771af60d5"
+  integrity sha512-DwGjNW9omn6EwP70aXsn7FQJx5kO12tX0bZkaTjzdVFM6/7nhA4t0EENocKGx6D2Bch9PE2KzCUf5SceBdeijg==
+  dependencies:
+    nofilter "^3.1.0"
+
+chai-as-promised@^7.1.1:
+  version "7.1.1"
+  resolved "https://registry.yarnpkg.com/chai-as-promised/-/chai-as-promised-7.1.1.tgz#08645d825deb8696ee61725dbf590c012eb00ca0"
+  integrity sha512-azL6xMoi+uxu6z4rhWQ1jbdUhOMhis2PvscD/xjLqNMkv3BPPp2JyyuTHOrf9BOosGpNQ11v6BKv/g57RXbiaA==
+  dependencies:
+    check-error "^1.0.2"
+
+chai-bignumber@^3.0.0:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/chai-bignumber/-/chai-bignumber-3.1.0.tgz#e196456c760df21f0e124f6df922289ea15a7e4c"
+  integrity sha512-omxEc80jAU+pZwRmoWr3aEzeLad4JW3iBhLRQlgISvghBdIxrMT7mVAGsDz4WSyCkKowENshH2j9OABAhld7QQ==
+
+chai-string@^1.5.0:
+  version "1.5.0"
+  resolved "https://registry.yarnpkg.com/chai-string/-/chai-string-1.5.0.tgz#0bdb2d8a5f1dbe90bc78ec493c1c1c180dd4d3d2"
+  integrity sha512-sydDC3S3pNAQMYwJrs6dQX0oBQ6KfIPuOZ78n7rocW0eJJlsHPh2t3kwW7xfwYA/1Bf6/arGtSUo16rxR2JFlw==
+
+chai@^4.2.0:
+  version "4.3.7"
+  resolved "https://registry.yarnpkg.com/chai/-/chai-4.3.7.tgz#ec63f6df01829088e8bf55fca839bcd464a8ec51"
+  integrity sha512-HLnAzZ2iupm25PlN0xFreAlBA5zaBSv3og0DdeGA4Ar6h6rJ3A0rolRUKJhSF2V10GZKDgWF/VmAEsNWjCRB+A==
+  dependencies:
+    assertion-error "^1.1.0"
+    check-error "^1.0.2"
+    deep-eql "^4.1.2"
+    get-func-name "^2.0.0"
+    loupe "^2.3.1"
+    pathval "^1.1.1"
+    type-detect "^4.0.5"
+
+chalk@^2.3.2, chalk@^2.4.2:
+  version "2.4.2"
+  resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.2.tgz#cd42541677a54333cf541a49108c1432b44c9424"
+  integrity sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==
+  dependencies:
+    ansi-styles "^3.2.1"
+    escape-string-regexp "^1.0.5"
+    supports-color "^5.3.0"
+
+chalk@^4.0.0, chalk@^4.1.0, chalk@^4.1.2:
+  version "4.1.2"
+  resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01"
+  integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==
+  dependencies:
+    ansi-styles "^4.1.0"
+    supports-color "^7.1.0"
+
+change-case@3.0.2:
+  version "3.0.2"
+  resolved "https://registry.yarnpkg.com/change-case/-/change-case-3.0.2.tgz#fd48746cce02f03f0a672577d1d3a8dc2eceb037"
+  integrity sha512-Mww+SLF6MZ0U6kdg11algyKd5BARbyM4TbFBepwowYSR5ClfQGCGtxNXgykpN0uF/bstWeaGDT4JWaDh8zWAHA==
+  dependencies:
+    camel-case "^3.0.0"
+    constant-case "^2.0.0"
+    dot-case "^2.1.0"
+    header-case "^1.0.0"
+    is-lower-case "^1.1.0"
+    is-upper-case "^1.1.0"
+    lower-case "^1.1.1"
+    lower-case-first "^1.0.0"
+    no-case "^2.3.2"
+    param-case "^2.1.0"
+    pascal-case "^2.0.0"
+    path-case "^2.1.0"
+    sentence-case "^2.1.0"
+    snake-case "^2.1.0"
+    swap-case "^1.1.0"
+    title-case "^2.1.0"
+    upper-case "^1.1.1"
+    upper-case-first "^1.1.0"
+
+"charenc@>= 0.0.1":
+  version "0.0.2"
+  resolved "https://registry.yarnpkg.com/charenc/-/charenc-0.0.2.tgz#c0a1d2f3a7092e03774bfa83f14c0fc5790a8667"
+  integrity sha512-yrLQ/yVUFXkzg7EDQsPieE/53+0RlaWTs+wBrvW36cyilJ2SaDWfl4Yj7MtLTXleV9uEKefbAGUPv2/iWSooRA==
+
+check-error@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/check-error/-/check-error-1.0.2.tgz#574d312edd88bb5dd8912e9286dd6c0aed4aac82"
+  integrity sha512-BrgHpW9NURQgzoNyjfq0Wu6VFO6D7IZEmJNdtgNqpzGG8RuNFHt2jQxWlAs4HMe119chBnv+34syEZtc6IhLtA==
+
+cheerio-select@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/cheerio-select/-/cheerio-select-2.1.0.tgz#4d8673286b8126ca2a8e42740d5e3c4884ae21b4"
+  integrity sha512-9v9kG0LvzrlcungtnJtpGNxY+fzECQKhK4EGJX2vByejiMX84MFNQw4UxPJl3bFbTMw+Dfs37XaIkCwTZfLh4g==
+  dependencies:
+    boolbase "^1.0.0"
+    css-select "^5.1.0"
+    css-what "^6.1.0"
+    domelementtype "^2.3.0"
+    domhandler "^5.0.3"
+    domutils "^3.0.1"
+
+cheerio@^1.0.0-rc.2:
+  version "1.0.0-rc.12"
+  resolved "https://registry.yarnpkg.com/cheerio/-/cheerio-1.0.0-rc.12.tgz#788bf7466506b1c6bf5fae51d24a2c4d62e47683"
+  integrity sha512-VqR8m68vM46BNnuZ5NtnGBKIE/DfN0cRIzg9n40EIq9NOv90ayxLBXA8fXC5gquFRGJSTRqBq25Jt2ECLR431Q==
+  dependencies:
+    cheerio-select "^2.1.0"
+    dom-serializer "^2.0.0"
+    domhandler "^5.0.3"
+    domutils "^3.0.1"
+    htmlparser2 "^8.0.1"
+    parse5 "^7.0.0"
+    parse5-htmlparser2-tree-adapter "^7.0.0"
+
+child_process@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/child_process/-/child_process-1.0.2.tgz#b1f7e7fc73d25e7fd1d455adc94e143830182b5a"
+  integrity sha512-Wmza/JzL0SiWz7kl6MhIKT5ceIlnFPJX+lwUGj7Clhy5MMldsSoJR0+uvRzOS5Kv45Mq7t1PoE8TsOA9bzvb6g==
+
+chokidar@3.3.0:
+  version "3.3.0"
+  resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-3.3.0.tgz#12c0714668c55800f659e262d4962a97faf554a6"
+  integrity sha512-dGmKLDdT3Gdl7fBUe8XK+gAtGmzy5Fn0XkkWQuYxGIgWVPPse2CxFA5mtrlD0TOHaHjEUqkWNyP1XdHoJES/4A==
+  dependencies:
+    anymatch "~3.1.1"
+    braces "~3.0.2"
+    glob-parent "~5.1.0"
+    is-binary-path "~2.1.0"
+    is-glob "~4.0.1"
+    normalize-path "~3.0.0"
+    readdirp "~3.2.0"
+  optionalDependencies:
+    fsevents "~2.1.1"
+
+chokidar@3.5.3, chokidar@^3.4.0, chokidar@^3.5.2:
+  version "3.5.3"
+  resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-3.5.3.tgz#1cf37c8707b932bd1af1ae22c0432e2acd1903bd"
+  integrity sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==
+  dependencies:
+    anymatch "~3.1.2"
+    braces "~3.0.2"
+    glob-parent "~5.1.2"
+    is-binary-path "~2.1.0"
+    is-glob "~4.0.1"
+    normalize-path "~3.0.0"
+    readdirp "~3.6.0"
+  optionalDependencies:
+    fsevents "~2.3.2"
+
+chownr@^1.1.4:
+  version "1.1.4"
+  resolved "https://registry.yarnpkg.com/chownr/-/chownr-1.1.4.tgz#6fc9d7b42d32a583596337666e7d08084da2cc6b"
+  integrity sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==
+
+ci-info@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/ci-info/-/ci-info-2.0.0.tgz#67a9e964be31a51e15e5010d58e6f12834002f46"
+  integrity sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==
+
+cids@^0.7.1:
+  version "0.7.5"
+  resolved "https://registry.yarnpkg.com/cids/-/cids-0.7.5.tgz#60a08138a99bfb69b6be4ceb63bfef7a396b28b2"
+  integrity sha512-zT7mPeghoWAu+ppn8+BS1tQ5qGmbMfB4AregnQjA/qHY3GC1m1ptI9GkWNlgeu38r7CuRdXB47uY2XgAYt6QVA==
+  dependencies:
+    buffer "^5.5.0"
+    class-is "^1.1.0"
+    multibase "~0.6.0"
+    multicodec "^1.0.0"
+    multihashes "~0.4.15"
+
+cipher-base@^1.0.0, cipher-base@^1.0.1, cipher-base@^1.0.3:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/cipher-base/-/cipher-base-1.0.4.tgz#8760e4ecc272f4c363532f926d874aae2c1397de"
+  integrity sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q==
+  dependencies:
+    inherits "^2.0.1"
+    safe-buffer "^5.0.1"
+
+class-is@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/class-is/-/class-is-1.1.0.tgz#9d3c0fba0440d211d843cec3dedfa48055005825"
+  integrity sha512-rhjH9AG1fvabIDoGRVH587413LPjTZgmDF9fOFCbFJQV4yuocX1mHxxvXI4g3cGwbVY9wAYIoKlg1N79frJKQw==
+
+classic-level@^1.2.0:
+  version "1.3.0"
+  resolved "https://registry.yarnpkg.com/classic-level/-/classic-level-1.3.0.tgz#5e36680e01dc6b271775c093f2150844c5edd5c8"
+  integrity sha512-iwFAJQYtqRTRM0F6L8h4JCt00ZSGdOyqh7yVrhhjrOpFhmBjNlRUey64MCiyo6UmQHMJ+No3c81nujPv+n9yrg==
+  dependencies:
+    abstract-level "^1.0.2"
+    catering "^2.1.0"
+    module-error "^1.0.1"
+    napi-macros "^2.2.2"
+    node-gyp-build "^4.3.0"
+
+clean-stack@^2.0.0:
+  version "2.2.0"
+  resolved "https://registry.yarnpkg.com/clean-stack/-/clean-stack-2.2.0.tgz#ee8472dbb129e727b31e8a10a427dee9dfe4008b"
+  integrity sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==
+
+cli-cursor@^3.1.0:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/cli-cursor/-/cli-cursor-3.1.0.tgz#264305a7ae490d1d03bf0c9ba7c925d1753af307"
+  integrity sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==
+  dependencies:
+    restore-cursor "^3.1.0"
+
+cli-spinners@^2.5.0:
+  version "2.9.0"
+  resolved "https://registry.yarnpkg.com/cli-spinners/-/cli-spinners-2.9.0.tgz#5881d0ad96381e117bbe07ad91f2008fe6ffd8db"
+  integrity sha512-4/aL9X3Wh0yiMQlE+eeRhWP6vclO3QRtw1JHKIT0FFUs5FjpFmESqtMvYZ0+lbzBw900b95mS0hohy+qn2VK/g==
+
+cli-table3@^0.5.0:
+  version "0.5.1"
+  resolved "https://registry.yarnpkg.com/cli-table3/-/cli-table3-0.5.1.tgz#0252372d94dfc40dbd8df06005f48f31f656f202"
+  integrity sha512-7Qg2Jrep1S/+Q3EceiZtQcDPWxhAvBw+ERf1162v4sikJrvojMHFqXt8QIVha8UlH9rgU0BeWPytZ9/TzYqlUw==
+  dependencies:
+    object-assign "^4.1.0"
+    string-width "^2.1.1"
+  optionalDependencies:
+    colors "^1.1.2"
+
+cli-table3@^0.6.2:
+  version "0.6.3"
+  resolved "https://registry.yarnpkg.com/cli-table3/-/cli-table3-0.6.3.tgz#61ab765aac156b52f222954ffc607a6f01dbeeb2"
+  integrity sha512-w5Jac5SykAeZJKntOxJCrm63Eg5/4dhMWIcuTbo9rpE+brgaSZo0RuNJZeOyMgsUdhDeojvgyQLmjI+K50ZGyg==
+  dependencies:
+    string-width "^4.2.0"
+  optionalDependencies:
+    "@colors/colors" "1.5.0"
+
+cliui@^3.2.0:
+  version "3.2.0"
+  resolved "https://registry.yarnpkg.com/cliui/-/cliui-3.2.0.tgz#120601537a916d29940f934da3b48d585a39213d"
+  integrity sha512-0yayqDxWQbqk3ojkYqUKqaAQ6AfNKeKWRNA8kR0WXzAsdHpP4BIaOmMAG87JGuO6qcobyW4GjxHd9PmhEd+T9w==
+  dependencies:
+    string-width "^1.0.1"
+    strip-ansi "^3.0.1"
+    wrap-ansi "^2.0.0"
+
+cliui@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/cliui/-/cliui-5.0.0.tgz#deefcfdb2e800784aa34f46fa08e06851c7bbbc5"
+  integrity sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==
+  dependencies:
+    string-width "^3.1.0"
+    strip-ansi "^5.2.0"
+    wrap-ansi "^5.1.0"
+
+cliui@^7.0.2:
+  version "7.0.4"
+  resolved "https://registry.yarnpkg.com/cliui/-/cliui-7.0.4.tgz#a0265ee655476fc807aea9df3df8df7783808b4f"
+  integrity sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==
+  dependencies:
+    string-width "^4.2.0"
+    strip-ansi "^6.0.0"
+    wrap-ansi "^7.0.0"
+
+clone-response@^1.0.2:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/clone-response/-/clone-response-1.0.3.tgz#af2032aa47816399cf5f0a1d0db902f517abb8c3"
+  integrity sha512-ROoL94jJH2dUVML2Y/5PEDNaSHgeOdSDicUyS7izcF63G6sTc/FTjLub4b8Il9S8S0beOfYt0TaA5qvFK+w0wA==
+  dependencies:
+    mimic-response "^1.0.0"
+
+clone@^1.0.2:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/clone/-/clone-1.0.4.tgz#da309cc263df15994c688ca902179ca3c7cd7c7e"
+  integrity sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg==
+
+code-point-at@^1.0.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/code-point-at/-/code-point-at-1.1.0.tgz#0d070b4d043a5bea33a2f1a40e2edb3d9a4ccf77"
+  integrity sha512-RpAVKQA5T63xEj6/giIbUEtZwJ4UFIc3ZtvEkiaUERylqe8xb5IvqcgOurZLahv93CLKfxcw5YI+DZcUBRyLXA==
+
+color-convert@^1.9.0:
+  version "1.9.3"
+  resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.3.tgz#bb71850690e1f136567de629d2d5471deda4c1e8"
+  integrity sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==
+  dependencies:
+    color-name "1.1.3"
+
+color-convert@^2.0.1:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3"
+  integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==
+  dependencies:
+    color-name "~1.1.4"
+
+color-name@1.1.3:
+  version "1.1.3"
+  resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25"
+  integrity sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==
+
+color-name@~1.1.4:
+  version "1.1.4"
+  resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2"
+  integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==
+
+colors@1.4.0, colors@^1.1.2:
+  version "1.4.0"
+  resolved "https://registry.yarnpkg.com/colors/-/colors-1.4.0.tgz#c50491479d4c1bdaed2c9ced32cf7c7dc2360f78"
+  integrity sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==
+
+combined-stream@^1.0.6, combined-stream@^1.0.8, combined-stream@~1.0.6:
+  version "1.0.8"
+  resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f"
+  integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==
+  dependencies:
+    delayed-stream "~1.0.0"
+
+command-exists@^1.2.8:
+  version "1.2.9"
+  resolved "https://registry.yarnpkg.com/command-exists/-/command-exists-1.2.9.tgz#c50725af3808c8ab0260fd60b01fbfa25b954f69"
+  integrity sha512-LTQ/SGc+s0Xc0Fu5WaKnR0YiygZkm9eKFvyS+fRsU7/ZWFF8ykFM6Pc9aCVf1+xasOOZpO3BAVgVrKvsqKHV7w==
+
+command-line-args@^5.1.1:
+  version "5.2.1"
+  resolved "https://registry.yarnpkg.com/command-line-args/-/command-line-args-5.2.1.tgz#c44c32e437a57d7c51157696893c5909e9cec42e"
+  integrity sha512-H4UfQhZyakIjC74I9d34fGYDwk3XpSr17QhEd0Q3I9Xq1CETHo4Hcuo87WyWHpAF1aSLjLRf5lD9ZGX2qStUvg==
+  dependencies:
+    array-back "^3.1.0"
+    find-replace "^3.0.0"
+    lodash.camelcase "^4.3.0"
+    typical "^4.0.0"
+
+command-line-usage@^6.1.0:
+  version "6.1.3"
+  resolved "https://registry.yarnpkg.com/command-line-usage/-/command-line-usage-6.1.3.tgz#428fa5acde6a838779dfa30e44686f4b6761d957"
+  integrity sha512-sH5ZSPr+7UStsloltmDh7Ce5fb8XPlHyoPzTpyyMuYCtervL65+ubVZ6Q61cFtFl62UyJlc8/JwERRbAFPUqgw==
+  dependencies:
+    array-back "^4.0.2"
+    chalk "^2.4.2"
+    table-layout "^1.0.2"
+    typical "^5.2.0"
+
+commander@3.0.2:
+  version "3.0.2"
+  resolved "https://registry.yarnpkg.com/commander/-/commander-3.0.2.tgz#6837c3fb677ad9933d1cfba42dd14d5117d6b39e"
+  integrity sha512-Gar0ASD4BDyKC4hl4DwHqDrmvjoxWKZigVnAbn5H1owvm4CxCPdb0HQDehwNYMJpla5+M2tPmPARzhtYuwpHow==
+
+commander@^10.0.0:
+  version "10.0.1"
+  resolved "https://registry.yarnpkg.com/commander/-/commander-10.0.1.tgz#881ee46b4f77d1c1dccc5823433aa39b022cbe06"
+  integrity sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug==
+
+commander@^9.4.0:
+  version "9.5.0"
+  resolved "https://registry.yarnpkg.com/commander/-/commander-9.5.0.tgz#bc08d1eb5cedf7ccb797a96199d41c7bc3e60d30"
+  integrity sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==
+
+compare-versions@^3.6.0:
+  version "3.6.0"
+  resolved "https://registry.yarnpkg.com/compare-versions/-/compare-versions-3.6.0.tgz#1a5689913685e5a87637b8d3ffca75514ec41d62"
+  integrity sha512-W6Af2Iw1z4CB7q4uU4hv646dW9GQuBM+YpC0UvUCWSD8w90SJjp+ujJuXaEMtAXBtSqGfMPuFOVn4/+FlaqfBA==
+
+concat-map@0.0.1:
+  version "0.0.1"
+  resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b"
+  integrity sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==
+
+concat-stream@^1.6.0, concat-stream@^1.6.2:
+  version "1.6.2"
+  resolved "https://registry.yarnpkg.com/concat-stream/-/concat-stream-1.6.2.tgz#904bdf194cd3122fc675c77fc4ac3d4ff0fd1a34"
+  integrity sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==
+  dependencies:
+    buffer-from "^1.0.0"
+    inherits "^2.0.3"
+    readable-stream "^2.2.2"
+    typedarray "^0.0.6"
+
+constant-case@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/constant-case/-/constant-case-2.0.0.tgz#4175764d389d3fa9c8ecd29186ed6005243b6a46"
+  integrity sha512-eS0N9WwmjTqrOmR3o83F5vW8Z+9R1HnVz3xmzT2PMFug9ly+Au/fxRWlEBSb6LcZwspSsEn9Xs1uw9YgzAg1EQ==
+  dependencies:
+    snake-case "^2.1.0"
+    upper-case "^1.1.1"
+
+content-disposition@0.5.4:
+  version "0.5.4"
+  resolved "https://registry.yarnpkg.com/content-disposition/-/content-disposition-0.5.4.tgz#8b82b4efac82512a02bb0b1dcec9d2c5e8eb5bfe"
+  integrity sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==
+  dependencies:
+    safe-buffer "5.2.1"
+
+content-hash@^2.5.2:
+  version "2.5.2"
+  resolved "https://registry.yarnpkg.com/content-hash/-/content-hash-2.5.2.tgz#bbc2655e7c21f14fd3bfc7b7d4bfe6e454c9e211"
+  integrity sha512-FvIQKy0S1JaWV10sMsA7TRx8bpU+pqPkhbsfvOJAdjRXvYxEckAwQWGwtRjiaJfh+E0DvcWUGqcdjwMGFjsSdw==
+  dependencies:
+    cids "^0.7.1"
+    multicodec "^0.5.5"
+    multihashes "^0.4.15"
+
+content-type@~1.0.4, content-type@~1.0.5:
+  version "1.0.5"
+  resolved "https://registry.yarnpkg.com/content-type/-/content-type-1.0.5.tgz#8b773162656d1d1086784c8f23a54ce6d73d7918"
+  integrity sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==
+
+cookie-signature@1.0.6:
+  version "1.0.6"
+  resolved "https://registry.yarnpkg.com/cookie-signature/-/cookie-signature-1.0.6.tgz#e303a882b342cc3ee8ca513a79999734dab3ae2c"
+  integrity sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==
+
+cookie@0.5.0:
+  version "0.5.0"
+  resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.5.0.tgz#d1f5d71adec6558c58f389987c366aa47e994f8b"
+  integrity sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==
+
+cookie@^0.4.1:
+  version "0.4.2"
+  resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.4.2.tgz#0e41f24de5ecf317947c82fc789e06a884824432"
+  integrity sha512-aSWTXFzaKWkvHO1Ny/s+ePFpvKsPnjc551iI41v3ny/ow6tBG5Vd+FuqGNhh1LxOmVzOlGUriIlOaokOvhaStA==
+
+core-util-is@1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7"
+  integrity sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==
+
+core-util-is@~1.0.0:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.3.tgz#a6042d3634c2b27e9328f837b965fac83808db85"
+  integrity sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==
+
+cors@^2.8.1:
+  version "2.8.5"
+  resolved "https://registry.yarnpkg.com/cors/-/cors-2.8.5.tgz#eac11da51592dd86b9f06f6e7ac293b3df875d29"
+  integrity sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==
+  dependencies:
+    object-assign "^4"
+    vary "^1"
+
+cosmiconfig@^7.0.0:
+  version "7.1.0"
+  resolved "https://registry.yarnpkg.com/cosmiconfig/-/cosmiconfig-7.1.0.tgz#1443b9afa596b670082ea46cbd8f6a62b84635f6"
+  integrity sha512-AdmX6xUzdNASswsFtmwSt7Vj8po9IuqXm0UXz7QKPuEUmPB4XyjGfaAr2PSuELMwkRMVH1EpIkX5bTZGRB3eCA==
+  dependencies:
+    "@types/parse-json" "^4.0.0"
+    import-fresh "^3.2.1"
+    parse-json "^5.0.0"
+    path-type "^4.0.0"
+    yaml "^1.10.0"
+
+cosmiconfig@^8.0.0:
+  version "8.2.0"
+  resolved "https://registry.yarnpkg.com/cosmiconfig/-/cosmiconfig-8.2.0.tgz#f7d17c56a590856cd1e7cee98734dca272b0d8fd"
+  integrity sha512-3rTMnFJA1tCOPwRxtgF4wd7Ab2qvDbL8jX+3smjIbS4HlZBagTlpERbdN7iAbWlrfxE3M8c27kTwTawQ7st+OQ==
+  dependencies:
+    import-fresh "^3.2.1"
+    js-yaml "^4.1.0"
+    parse-json "^5.0.0"
+    path-type "^4.0.0"
+
+crc-32@^1.2.0:
+  version "1.2.2"
+  resolved "https://registry.yarnpkg.com/crc-32/-/crc-32-1.2.2.tgz#3cad35a934b8bf71f25ca524b6da51fb7eace2ff"
+  integrity sha512-ROmzCKrTnOwybPcJApAA6WBWij23HVfGVNKqqrZpuyZOHqK2CwHSvpGuyt/UNNvaIjEd8X5IFGp4Mh+Ie1IHJQ==
+
+create-hash@^1.1.0, create-hash@^1.1.2, create-hash@^1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/create-hash/-/create-hash-1.2.0.tgz#889078af11a63756bcfb59bd221996be3a9ef196"
+  integrity sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==
+  dependencies:
+    cipher-base "^1.0.1"
+    inherits "^2.0.1"
+    md5.js "^1.3.4"
+    ripemd160 "^2.0.1"
+    sha.js "^2.4.0"
+
+create-hmac@^1.1.4, create-hmac@^1.1.7:
+  version "1.1.7"
+  resolved "https://registry.yarnpkg.com/create-hmac/-/create-hmac-1.1.7.tgz#69170c78b3ab957147b2b8b04572e47ead2243ff"
+  integrity sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==
+  dependencies:
+    cipher-base "^1.0.3"
+    create-hash "^1.1.0"
+    inherits "^2.0.1"
+    ripemd160 "^2.0.0"
+    safe-buffer "^5.0.1"
+    sha.js "^2.4.8"
+
+create-require@^1.1.0:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/create-require/-/create-require-1.1.1.tgz#c1d7e8f1e5f6cfc9ff65f9cd352d37348756c333"
+  integrity sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==
+
+cross-fetch@^3.1.4:
+  version "3.1.8"
+  resolved "https://registry.yarnpkg.com/cross-fetch/-/cross-fetch-3.1.8.tgz#0327eba65fd68a7d119f8fb2bf9334a1a7956f82"
+  integrity sha512-cvA+JwZoU0Xq+h6WkMvAUqPEYy92Obet6UdKLfW60qn99ftItKjB5T+BkyWOFWe2pUyfQ+IJHmpOTznqk1M6Kg==
+  dependencies:
+    node-fetch "^2.6.12"
+
+cross-spawn@^7.0.2:
+  version "7.0.3"
+  resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.3.tgz#f73a85b9d5d41d045551c177e2882d4ac85728a6"
+  integrity sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==
+  dependencies:
+    path-key "^3.1.0"
+    shebang-command "^2.0.0"
+    which "^2.0.1"
+
+"crypt@>= 0.0.1":
+  version "0.0.2"
+  resolved "https://registry.yarnpkg.com/crypt/-/crypt-0.0.2.tgz#88d7ff7ec0dfb86f713dc87bbb42d044d3e6c41b"
+  integrity sha512-mCxBlsHFYh9C+HVpiEacem8FEBnMXgU9gy4zmNC+SXAZNB/1idgp/aulFJ4FgCi7GPEVbfyng092GqL2k2rmow==
+
+crypto-addr-codec@^0.1.7:
+  version "0.1.7"
+  resolved "https://registry.yarnpkg.com/crypto-addr-codec/-/crypto-addr-codec-0.1.7.tgz#e16cea892730178fe25a38f6d15b680cab3124ae"
+  integrity sha512-X4hzfBzNhy4mAc3UpiXEC/L0jo5E8wAa9unsnA8nNXYzXjCcGk83hfC5avJWCSGT8V91xMnAS9AKMHmjw5+XCg==
+  dependencies:
+    base-x "^3.0.8"
+    big-integer "1.6.36"
+    blakejs "^1.1.0"
+    bs58 "^4.0.1"
+    ripemd160-min "0.0.6"
+    safe-buffer "^5.2.0"
+    sha3 "^2.1.1"
+
+css-select@^5.1.0:
+  version "5.1.0"
+  resolved "https://registry.yarnpkg.com/css-select/-/css-select-5.1.0.tgz#b8ebd6554c3637ccc76688804ad3f6a6fdaea8a6"
+  integrity sha512-nwoRF1rvRRnnCqqY7updORDsuqKzqYJ28+oSMaJMMgOauh3fvwHqMS7EZpIPqK8GL+g9mKxF1vP/ZjSeNjEVHg==
+  dependencies:
+    boolbase "^1.0.0"
+    css-what "^6.1.0"
+    domhandler "^5.0.2"
+    domutils "^3.0.1"
+    nth-check "^2.0.1"
+
+css-what@^6.1.0:
+  version "6.1.0"
+  resolved "https://registry.yarnpkg.com/css-what/-/css-what-6.1.0.tgz#fb5effcf76f1ddea2c81bdfaa4de44e79bac70f4"
+  integrity sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==
+
+d@1, d@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/d/-/d-1.0.1.tgz#8698095372d58dbee346ffd0c7093f99f8f9eb5a"
+  integrity sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==
+  dependencies:
+    es5-ext "^0.10.50"
+    type "^1.0.1"
+
+dashdash@^1.12.0:
+  version "1.14.1"
+  resolved "https://registry.yarnpkg.com/dashdash/-/dashdash-1.14.1.tgz#853cfa0f7cbe2fed5de20326b8dd581035f6e2f0"
+  integrity sha512-jRFi8UDGo6j+odZiEpjazZaWqEal3w/basFjQHQEwVtZJGDpxbH1MeYluwCS8Xq5wmLJooDlMgvVarmWfGM44g==
+  dependencies:
+    assert-plus "^1.0.0"
+
+death@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/death/-/death-1.1.0.tgz#01aa9c401edd92750514470b8266390c66c67318"
+  integrity sha512-vsV6S4KVHvTGxbEcij7hkWRv0It+sGGWVOM67dQde/o5Xjnr+KmLjxWJii2uEObIrt1CcM9w0Yaovx+iOlIL+w==
+
+debug@2.6.9, debug@^2.2.0:
+  version "2.6.9"
+  resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f"
+  integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==
+  dependencies:
+    ms "2.0.0"
+
+debug@3.2.6:
+  version "3.2.6"
+  resolved "https://registry.yarnpkg.com/debug/-/debug-3.2.6.tgz#e83d17de16d8a7efb7717edbe5fb10135eee629b"
+  integrity sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==
+  dependencies:
+    ms "^2.1.1"
+
+debug@4, debug@4.3.4, debug@^4.1.1, debug@^4.3.1, debug@^4.3.2, debug@^4.3.3, debug@^4.3.4:
+  version "4.3.4"
+  resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865"
+  integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==
+  dependencies:
+    ms "2.1.2"
+
+debug@^3.2.7:
+  version "3.2.7"
+  resolved "https://registry.yarnpkg.com/debug/-/debug-3.2.7.tgz#72580b7e9145fb39b6676f9c5e5fb100b934179a"
+  integrity sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==
+  dependencies:
+    ms "^2.1.1"
+
+decamelize@^1.1.1, decamelize@^1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-1.2.0.tgz#f6534d15148269b20352e7bee26f501f9a191290"
+  integrity sha512-z2S+W9X73hAUUki+N+9Za2lBlun89zigOyGrsax+KUQ6wKW4ZoWpEYBkGhQjwAjjDCkWxhY0VKEhk8wzY7F5cA==
+
+decamelize@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-4.0.0.tgz#aa472d7bf660eb15f3494efd531cab7f2a709837"
+  integrity sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==
+
+decode-uri-component@^0.2.0:
+  version "0.2.2"
+  resolved "https://registry.yarnpkg.com/decode-uri-component/-/decode-uri-component-0.2.2.tgz#e69dbe25d37941171dd540e024c444cd5188e1e9"
+  integrity sha512-FqUYQ+8o158GyGTrMFJms9qh3CqTKvAqgqsTnkLI8sKu0028orqBhxNMFkFen0zGyg6epACD32pjVk58ngIErQ==
+
+decompress-response@^3.3.0:
+  version "3.3.0"
+  resolved "https://registry.yarnpkg.com/decompress-response/-/decompress-response-3.3.0.tgz#80a4dd323748384bfa248083622aedec982adff3"
+  integrity sha512-BzRPQuY1ip+qDonAOz42gRm/pg9F768C+npV/4JOsxRC2sq+Rlk+Q4ZCAsOhnIaMrgarILY+RMUIvMmmX1qAEA==
+  dependencies:
+    mimic-response "^1.0.0"
+
+decompress-response@^6.0.0:
+  version "6.0.0"
+  resolved "https://registry.yarnpkg.com/decompress-response/-/decompress-response-6.0.0.tgz#ca387612ddb7e104bd16d85aab00d5ecf09c66fc"
+  integrity sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==
+  dependencies:
+    mimic-response "^3.1.0"
+
+deep-eql@^4.0.1, deep-eql@^4.1.2:
+  version "4.1.3"
+  resolved "https://registry.yarnpkg.com/deep-eql/-/deep-eql-4.1.3.tgz#7c7775513092f7df98d8df9996dd085eb668cc6d"
+  integrity sha512-WaEtAOpRA1MQ0eohqZjpGD8zdI0Ovsm8mmFhaDN8dvDZzyoUMcYDnf5Y6iu7HTXxf8JDS23qWa4a+hKCDyOPzw==
+  dependencies:
+    type-detect "^4.0.0"
+
+deep-extend@~0.6.0:
+  version "0.6.0"
+  resolved "https://registry.yarnpkg.com/deep-extend/-/deep-extend-0.6.0.tgz#c4fa7c95404a17a9c3e8ca7e1537312b736330ac"
+  integrity sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==
+
+deep-is@^0.1.3, deep-is@~0.1.3:
+  version "0.1.4"
+  resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.4.tgz#a6f2dce612fadd2ef1f519b73551f17e85199831"
+  integrity sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==
+
+defaults@^1.0.3:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/defaults/-/defaults-1.0.4.tgz#b0b02062c1e2aa62ff5d9528f0f98baa90978d7a"
+  integrity sha512-eFuaLoy/Rxalv2kr+lqMlUnrDWV+3j4pljOIJgLIhI058IQfWJ7vXhyEIHu+HtC738klGALYxOKDO0bQP3tg8A==
+  dependencies:
+    clone "^1.0.2"
+
+defer-to-connect@^2.0.0, defer-to-connect@^2.0.1:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/defer-to-connect/-/defer-to-connect-2.0.1.tgz#8016bdb4143e4632b77a3449c6236277de520587"
+  integrity sha512-4tvttepXG1VaYGrRibk5EwJd1t4udunSOVMdLSAL6mId1ix438oPwPZMALY41FCijukO1L0twNcGsdzS7dHgDg==
+
+define-lazy-prop@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/define-lazy-prop/-/define-lazy-prop-2.0.0.tgz#3f7ae421129bcaaac9bc74905c98a0009ec9ee7f"
+  integrity sha512-Ds09qNh8yw3khSjiJjiUInaGX9xlqZDY7JVryGxdxV7NPeuqQfplOpQ66yJFZut3jLa5zOwkXw1g9EI2uKh4Og==
+
+define-properties@^1.1.2, define-properties@^1.1.3, define-properties@^1.1.4, define-properties@^1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.2.0.tgz#52988570670c9eacedd8064f4a990f2405849bd5"
+  integrity sha512-xvqAVKGfT1+UAvPwKTVw/njhdQ8ZhXK4lI0bCIuCMrp2up9nPnaDftrLtmpTazqd1o+UY4zgzU+avtMbDP+ldA==
+  dependencies:
+    has-property-descriptors "^1.0.0"
+    object-keys "^1.1.1"
+
+delayed-stream@~1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619"
+  integrity sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==
+
+depd@2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/depd/-/depd-2.0.0.tgz#b696163cc757560d09cf22cc8fad1571b79e76df"
+  integrity sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==
+
+destroy@1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.2.0.tgz#4803735509ad8be552934c67df614f94e66fa015"
+  integrity sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==
+
+detect-indent@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/detect-indent/-/detect-indent-5.0.0.tgz#3871cc0a6a002e8c3e5b3cf7f336264675f06b9d"
+  integrity sha512-rlpvsxUtM0PQvy9iZe640/IWwWYyBsTApREbA1pHOpmOUIl9MkP/U4z7vTtg4Oaojvqhxt7sdufnT0EzGaR31g==
+
+detect-port@^1.3.0:
+  version "1.5.1"
+  resolved "https://registry.yarnpkg.com/detect-port/-/detect-port-1.5.1.tgz#451ca9b6eaf20451acb0799b8ab40dff7718727b"
+  integrity sha512-aBzdj76lueB6uUst5iAs7+0H/oOjqI5D16XUWxlWMIMROhcM0rfsNVk93zTngq1dDNpoXRr++Sus7ETAExppAQ==
+  dependencies:
+    address "^1.0.1"
+    debug "4"
+
+diff@3.5.0:
+  version "3.5.0"
+  resolved "https://registry.yarnpkg.com/diff/-/diff-3.5.0.tgz#800c0dd1e0a8bfbc95835c202ad220fe317e5a12"
+  integrity sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==
+
+diff@5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/diff/-/diff-5.0.0.tgz#7ed6ad76d859d030787ec35855f5b1daf31d852b"
+  integrity sha512-/VTCrvm5Z0JGty/BWHljh+BAiw3IK+2j87NGMu8Nwc/f48WoDAC395uomO9ZD117ZOBaHmkX1oyLvkVM/aIT3w==
+
+diff@^4.0.1:
+  version "4.0.2"
+  resolved "https://registry.yarnpkg.com/diff/-/diff-4.0.2.tgz#60f3aecb89d5fae520c11aa19efc2bb982aade7d"
+  integrity sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==
+
+difflib@^0.2.4:
+  version "0.2.4"
+  resolved "https://registry.yarnpkg.com/difflib/-/difflib-0.2.4.tgz#b5e30361a6db023176d562892db85940a718f47e"
+  integrity sha512-9YVwmMb0wQHQNr5J9m6BSj6fk4pfGITGQOOs+D9Fl+INODWFOfvhIU1hNv6GgR1RBoC/9NJcwu77zShxV0kT7w==
+  dependencies:
+    heap ">= 0.2.0"
+
+dir-glob@^3.0.1:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/dir-glob/-/dir-glob-3.0.1.tgz#56dbf73d992a4a93ba1584f4534063fd2e41717f"
+  integrity sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==
+  dependencies:
+    path-type "^4.0.0"
+
+doctrine@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-2.1.0.tgz#5cd01fc101621b42c4cd7f5d1a66243716d3f39d"
+  integrity sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==
+  dependencies:
+    esutils "^2.0.2"
+
+doctrine@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-3.0.0.tgz#addebead72a6574db783639dc87a121773973961"
+  integrity sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==
+  dependencies:
+    esutils "^2.0.2"
+
+dom-serializer@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/dom-serializer/-/dom-serializer-2.0.0.tgz#e41b802e1eedf9f6cae183ce5e622d789d7d8e53"
+  integrity sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==
+  dependencies:
+    domelementtype "^2.3.0"
+    domhandler "^5.0.2"
+    entities "^4.2.0"
+
+dom-walk@^0.1.0:
+  version "0.1.2"
+  resolved "https://registry.yarnpkg.com/dom-walk/-/dom-walk-0.1.2.tgz#0c548bef048f4d1f2a97249002236060daa3fd84"
+  integrity sha512-6QvTW9mrGeIegrFXdtQi9pk7O/nSK6lSdXW2eqUspN5LWD7UTji2Fqw5V2YLjBpHEoU9Xl/eUWNpDeZvoyOv2w==
+
+domelementtype@^2.3.0:
+  version "2.3.0"
+  resolved "https://registry.yarnpkg.com/domelementtype/-/domelementtype-2.3.0.tgz#5c45e8e869952626331d7aab326d01daf65d589d"
+  integrity sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==
+
+domhandler@^5.0.2, domhandler@^5.0.3:
+  version "5.0.3"
+  resolved "https://registry.yarnpkg.com/domhandler/-/domhandler-5.0.3.tgz#cc385f7f751f1d1fc650c21374804254538c7d31"
+  integrity sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==
+  dependencies:
+    domelementtype "^2.3.0"
+
+domutils@^3.0.1:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/domutils/-/domutils-3.1.0.tgz#c47f551278d3dc4b0b1ab8cbb42d751a6f0d824e"
+  integrity sha512-H78uMmQtI2AhgDJjWeQmHwJJ2bLPD3GMmO7Zja/ZZh84wkm+4ut+IUnUdRa8uCGX88DiVx1j6FRe1XfxEgjEZA==
+  dependencies:
+    dom-serializer "^2.0.0"
+    domelementtype "^2.3.0"
+    domhandler "^5.0.3"
+
+dot-case@^2.1.0:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/dot-case/-/dot-case-2.1.1.tgz#34dcf37f50a8e93c2b3bca8bb7fb9155c7da3bee"
+  integrity sha512-HnM6ZlFqcajLsyudHq7LeeLDr2rFAVYtDv/hV5qchQEidSck8j9OPUsXY9KwJv/lHMtYlX4DjRQqwFYa+0r8Ug==
+  dependencies:
+    no-case "^2.2.0"
+
+dotenv@^16.0.3:
+  version "16.3.1"
+  resolved "https://registry.yarnpkg.com/dotenv/-/dotenv-16.3.1.tgz#369034de7d7e5b120972693352a3bf112172cc3e"
+  integrity sha512-IPzF4w4/Rd94bA9imS68tZBaYyBWSCE47V1RGuMrB94iyTOIEwRmVL2x/4An+6mETpLrKJ5hQkB8W4kFAadeIQ==
+
+ecc-jsbn@~0.1.1:
+  version "0.1.2"
+  resolved "https://registry.yarnpkg.com/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz#3a83a904e54353287874c564b7549386849a98c9"
+  integrity sha512-eh9O+hwRHNbG4BLTjEl3nw044CkGm5X6LoaCf7LPp7UU8Qrt47JYNi6nPX8xjW97TKGKm1ouctg0QSpZe9qrnw==
+  dependencies:
+    jsbn "~0.1.0"
+    safer-buffer "^2.1.0"
+
+ee-first@1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d"
+  integrity sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==
+
+elliptic@6.5.4, elliptic@^6.4.0, elliptic@^6.5.2, elliptic@^6.5.4:
+  version "6.5.4"
+  resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.5.4.tgz#da37cebd31e79a1367e941b592ed1fbebd58abbb"
+  integrity sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ==
+  dependencies:
+    bn.js "^4.11.9"
+    brorand "^1.1.0"
+    hash.js "^1.0.0"
+    hmac-drbg "^1.0.1"
+    inherits "^2.0.4"
+    minimalistic-assert "^1.0.1"
+    minimalistic-crypto-utils "^1.0.1"
+
+emittery@0.10.0:
+  version "0.10.0"
+  resolved "https://registry.yarnpkg.com/emittery/-/emittery-0.10.0.tgz#bb373c660a9d421bb44706ec4967ed50c02a8026"
+  integrity sha512-AGvFfs+d0JKCJQ4o01ASQLGPmSCxgfU9RFXvzPvZdjKK8oscynksuJhWrSTSw7j7Ep/sZct5b5ZhYCi8S/t0HQ==
+
+emoji-regex@^7.0.1:
+  version "7.0.3"
+  resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-7.0.3.tgz#933a04052860c85e83c122479c4748a8e4c72156"
+  integrity sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==
+
+emoji-regex@^8.0.0:
+  version "8.0.0"
+  resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37"
+  integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==
+
+encode-utf8@^1.0.2:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/encode-utf8/-/encode-utf8-1.0.3.tgz#f30fdd31da07fb596f281beb2f6b027851994cda"
+  integrity sha512-ucAnuBEhUK4boH2HjVYG5Q2mQyPorvv0u/ocS+zhdw0S8AlHYY+GOFhP1Gio5z4icpP2ivFSvhtFjQi8+T9ppw==
+
+encodeurl@~1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59"
+  integrity sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==
+
+end-of-stream@^1.1.0:
+  version "1.4.4"
+  resolved "https://registry.yarnpkg.com/end-of-stream/-/end-of-stream-1.4.4.tgz#5ae64a5f45057baf3626ec14da0ca5e4b2431eb0"
+  integrity sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==
+  dependencies:
+    once "^1.4.0"
+
+enquirer@^2.3.0, enquirer@^2.3.6:
+  version "2.4.1"
+  resolved "https://registry.yarnpkg.com/enquirer/-/enquirer-2.4.1.tgz#93334b3fbd74fc7097b224ab4a8fb7e40bf4ae56"
+  integrity sha512-rRqJg/6gd538VHvR3PSrdRBb/1Vy2YfzHqzvbhGIQpDRKIa4FgV/54b5Q1xYSxOOwKvjXweS26E0Q+nAMwp2pQ==
+  dependencies:
+    ansi-colors "^4.1.1"
+    strip-ansi "^6.0.1"
+
+entities@^4.2.0, entities@^4.4.0:
+  version "4.5.0"
+  resolved "https://registry.yarnpkg.com/entities/-/entities-4.5.0.tgz#5d268ea5e7113ec74c4d033b79ea5a35a488fb48"
+  integrity sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==
+
+env-paths@^2.2.0:
+  version "2.2.1"
+  resolved "https://registry.yarnpkg.com/env-paths/-/env-paths-2.2.1.tgz#420399d416ce1fbe9bc0a07c62fa68d67fd0f8f2"
+  integrity sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==
+
+error-ex@^1.2.0, error-ex@^1.3.1:
+  version "1.3.2"
+  resolved "https://registry.yarnpkg.com/error-ex/-/error-ex-1.3.2.tgz#b4ac40648107fdcdcfae242f428bea8a14d4f1bf"
+  integrity sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==
+  dependencies:
+    is-arrayish "^0.2.1"
+
+es-abstract@^1.19.0, es-abstract@^1.20.4, es-abstract@^1.21.2:
+  version "1.22.1"
+  resolved "https://registry.yarnpkg.com/es-abstract/-/es-abstract-1.22.1.tgz#8b4e5fc5cefd7f1660f0f8e1a52900dfbc9d9ccc"
+  integrity sha512-ioRRcXMO6OFyRpyzV3kE1IIBd4WG5/kltnzdxSCqoP8CMGs/Li+M1uF5o7lOkZVFjDs+NLesthnF66Pg/0q0Lw==
+  dependencies:
+    array-buffer-byte-length "^1.0.0"
+    arraybuffer.prototype.slice "^1.0.1"
+    available-typed-arrays "^1.0.5"
+    call-bind "^1.0.2"
+    es-set-tostringtag "^2.0.1"
+    es-to-primitive "^1.2.1"
+    function.prototype.name "^1.1.5"
+    get-intrinsic "^1.2.1"
+    get-symbol-description "^1.0.0"
+    globalthis "^1.0.3"
+    gopd "^1.0.1"
+    has "^1.0.3"
+    has-property-descriptors "^1.0.0"
+    has-proto "^1.0.1"
+    has-symbols "^1.0.3"
+    internal-slot "^1.0.5"
+    is-array-buffer "^3.0.2"
+    is-callable "^1.2.7"
+    is-negative-zero "^2.0.2"
+    is-regex "^1.1.4"
+    is-shared-array-buffer "^1.0.2"
+    is-string "^1.0.7"
+    is-typed-array "^1.1.10"
+    is-weakref "^1.0.2"
+    object-inspect "^1.12.3"
+    object-keys "^1.1.1"
+    object.assign "^4.1.4"
+    regexp.prototype.flags "^1.5.0"
+    safe-array-concat "^1.0.0"
+    safe-regex-test "^1.0.0"
+    string.prototype.trim "^1.2.7"
+    string.prototype.trimend "^1.0.6"
+    string.prototype.trimstart "^1.0.6"
+    typed-array-buffer "^1.0.0"
+    typed-array-byte-length "^1.0.0"
+    typed-array-byte-offset "^1.0.0"
+    typed-array-length "^1.0.4"
+    unbox-primitive "^1.0.2"
+    which-typed-array "^1.1.10"
+
+es-array-method-boxes-properly@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/es-array-method-boxes-properly/-/es-array-method-boxes-properly-1.0.0.tgz#873f3e84418de4ee19c5be752990b2e44718d09e"
+  integrity sha512-wd6JXUmyHmt8T5a2xreUwKcGPq6f1f+WwIJkijUqiGcJz1qqnZgP6XIK+QyIWU5lT7imeNxUll48bziG+TSYcA==
+
+es-set-tostringtag@^2.0.1:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/es-set-tostringtag/-/es-set-tostringtag-2.0.1.tgz#338d502f6f674301d710b80c8592de8a15f09cd8"
+  integrity sha512-g3OMbtlwY3QewlqAiMLI47KywjWZoEytKr8pf6iTC8uJq5bIAH52Z9pnQ8pVL6whrCto53JZDuUIsifGeLorTg==
+  dependencies:
+    get-intrinsic "^1.1.3"
+    has "^1.0.3"
+    has-tostringtag "^1.0.0"
+
+es-shim-unscopables@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/es-shim-unscopables/-/es-shim-unscopables-1.0.0.tgz#702e632193201e3edf8713635d083d378e510241"
+  integrity sha512-Jm6GPcCdC30eMLbZ2x8z2WuRwAws3zTBBKuusffYVUrNj/GVSUAZ+xKMaUpfNDR5IbyNA5LJbaecoUVbmUcB1w==
+  dependencies:
+    has "^1.0.3"
+
+es-to-primitive@^1.2.1:
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/es-to-primitive/-/es-to-primitive-1.2.1.tgz#e55cd4c9cdc188bcefb03b366c736323fc5c898a"
+  integrity sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==
+  dependencies:
+    is-callable "^1.1.4"
+    is-date-object "^1.0.1"
+    is-symbol "^1.0.2"
+
+es5-ext@^0.10.35, es5-ext@^0.10.50:
+  version "0.10.62"
+  resolved "https://registry.yarnpkg.com/es5-ext/-/es5-ext-0.10.62.tgz#5e6adc19a6da524bf3d1e02bbc8960e5eb49a9a5"
+  integrity sha512-BHLqn0klhEpnOKSrzn/Xsz2UIW8j+cGmo9JLzr8BiUapV8hPL9+FliFqjwr9ngW7jWdnxv6eO+/LqyhJVqgrjA==
+  dependencies:
+    es6-iterator "^2.0.3"
+    es6-symbol "^3.1.3"
+    next-tick "^1.1.0"
+
+es6-iterator@^2.0.3:
+  version "2.0.3"
+  resolved "https://registry.yarnpkg.com/es6-iterator/-/es6-iterator-2.0.3.tgz#a7de889141a05a94b0854403b2d0a0fbfa98f3b7"
+  integrity sha512-zw4SRzoUkd+cl+ZoE15A9o1oQd920Bb0iOJMQkQhl3jNc03YqVjAhG7scf9C5KWRU/R13Orf588uCC6525o02g==
+  dependencies:
+    d "1"
+    es5-ext "^0.10.35"
+    es6-symbol "^3.1.1"
+
+es6-promise@^4.2.8:
+  version "4.2.8"
+  resolved "https://registry.yarnpkg.com/es6-promise/-/es6-promise-4.2.8.tgz#4eb21594c972bc40553d276e510539143db53e0a"
+  integrity sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==
+
+es6-symbol@^3.1.1, es6-symbol@^3.1.3:
+  version "3.1.3"
+  resolved "https://registry.yarnpkg.com/es6-symbol/-/es6-symbol-3.1.3.tgz#bad5d3c1bcdac28269f4cb331e431c78ac705d18"
+  integrity sha512-NJ6Yn3FuDinBaBRWl/q5X/s4koRHBrgKAu+yGI6JCBeiu3qrcbJhwT2GeR/EXVfylRk8dpQVJoLEFhK+Mu31NA==
+  dependencies:
+    d "^1.0.1"
+    ext "^1.1.2"
+
+escalade@^3.1.1:
+  version "3.1.1"
+  resolved "https://registry.yarnpkg.com/escalade/-/escalade-3.1.1.tgz#d8cfdc7000965c5a0174b4a82eaa5c0552742e40"
+  integrity sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==
+
+escape-html@~1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/escape-html/-/escape-html-1.0.3.tgz#0258eae4d3d0c0974de1c169188ef0051d1d1988"
+  integrity sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==
+
+escape-string-regexp@1.0.5, escape-string-regexp@^1.0.5:
+  version "1.0.5"
+  resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4"
+  integrity sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==
+
+escape-string-regexp@4.0.0, escape-string-regexp@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz#14ba83a5d373e3d311e5afca29cf5bfad965bf34"
+  integrity sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==
+
+escodegen@1.8.x:
+  version "1.8.1"
+  resolved "https://registry.yarnpkg.com/escodegen/-/escodegen-1.8.1.tgz#5a5b53af4693110bebb0867aa3430dd3b70a1018"
+  integrity sha512-yhi5S+mNTOuRvyW4gWlg5W1byMaQGWWSYHXsuFZ7GBo7tpyOwi2EdzMP/QWxh9hwkD2m+wDVHJsxhRIj+v/b/A==
+  dependencies:
+    esprima "^2.7.1"
+    estraverse "^1.9.1"
+    esutils "^2.0.2"
+    optionator "^0.8.1"
+  optionalDependencies:
+    source-map "~0.2.0"
+
+eslint-config-prettier@^8.1.0:
+  version "8.10.0"
+  resolved "https://registry.yarnpkg.com/eslint-config-prettier/-/eslint-config-prettier-8.10.0.tgz#3a06a662130807e2502fc3ff8b4143d8a0658e11"
+  integrity sha512-SM8AMJdeQqRYT9O9zguiruQZaN7+z+E4eAP9oiLNGKMtomwaB1E9dcgUD6ZAn/eQAb52USbvezbiljfZUhbJcg==
+
+eslint-import-resolver-node@^0.3.7:
+  version "0.3.9"
+  resolved "https://registry.yarnpkg.com/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.9.tgz#d4eaac52b8a2e7c3cd1903eb00f7e053356118ac"
+  integrity sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g==
+  dependencies:
+    debug "^3.2.7"
+    is-core-module "^2.13.0"
+    resolve "^1.22.4"
+
+eslint-module-utils@^2.8.0:
+  version "2.8.0"
+  resolved "https://registry.yarnpkg.com/eslint-module-utils/-/eslint-module-utils-2.8.0.tgz#e439fee65fc33f6bba630ff621efc38ec0375c49"
+  integrity sha512-aWajIYfsqCKRDgUfjEXNN/JlrzauMuSEy5sbd7WXbtW3EH6A6MpwEh42c7qD+MqQo9QMJ6fWLAeIJynx0g6OAw==
+  dependencies:
+    debug "^3.2.7"
+
+eslint-plugin-import@^2.22.0:
+  version "2.28.0"
+  resolved "https://registry.yarnpkg.com/eslint-plugin-import/-/eslint-plugin-import-2.28.0.tgz#8d66d6925117b06c4018d491ae84469eb3cb1005"
+  integrity sha512-B8s/n+ZluN7sxj9eUf7/pRFERX0r5bnFA2dCaLHy2ZeaQEAz0k+ZZkFWRFHJAqxfxQDx6KLv9LeIki7cFdwW+Q==
+  dependencies:
+    array-includes "^3.1.6"
+    array.prototype.findlastindex "^1.2.2"
+    array.prototype.flat "^1.3.1"
+    array.prototype.flatmap "^1.3.1"
+    debug "^3.2.7"
+    doctrine "^2.1.0"
+    eslint-import-resolver-node "^0.3.7"
+    eslint-module-utils "^2.8.0"
+    has "^1.0.3"
+    is-core-module "^2.12.1"
+    is-glob "^4.0.3"
+    minimatch "^3.1.2"
+    object.fromentries "^2.0.6"
+    object.groupby "^1.0.0"
+    object.values "^1.1.6"
+    resolve "^1.22.3"
+    semver "^6.3.1"
+    tsconfig-paths "^3.14.2"
+
+eslint-plugin-prettier@^4.2.1:
+  version "4.2.1"
+  resolved "https://registry.yarnpkg.com/eslint-plugin-prettier/-/eslint-plugin-prettier-4.2.1.tgz#651cbb88b1dab98bfd42f017a12fa6b2d993f94b"
+  integrity sha512-f/0rXLXUt0oFYs8ra4w49wYZBG5GKZpAYsJSm6rnYL5uVDjd+zowwMwVZHnAjf4edNrKpCDYfXDgmRE/Ak7QyQ==
+  dependencies:
+    prettier-linter-helpers "^1.0.0"
+
+eslint-scope@^5.1.1:
+  version "5.1.1"
+  resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-5.1.1.tgz#e786e59a66cb92b3f6c1fb0d508aab174848f48c"
+  integrity sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==
+  dependencies:
+    esrecurse "^4.3.0"
+    estraverse "^4.1.1"
+
+eslint-scope@^7.2.2:
+  version "7.2.2"
+  resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-7.2.2.tgz#deb4f92563390f32006894af62a22dba1c46423f"
+  integrity sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==
+  dependencies:
+    esrecurse "^4.3.0"
+    estraverse "^5.2.0"
+
+eslint-visitor-keys@^3.3.0, eslint-visitor-keys@^3.4.1, eslint-visitor-keys@^3.4.2:
+  version "3.4.2"
+  resolved "https://registry.yarnpkg.com/eslint-visitor-keys/-/eslint-visitor-keys-3.4.2.tgz#8c2095440eca8c933bedcadf16fefa44dbe9ba5f"
+  integrity sha512-8drBzUEyZ2llkpCA67iYrgEssKDUu68V8ChqqOfFupIaG/LCVPUT+CoGJpT77zJprs4T/W7p07LP7zAIMuweVw==
+
+eslint@^8.27.0:
+  version "8.46.0"
+  resolved "https://registry.yarnpkg.com/eslint/-/eslint-8.46.0.tgz#a06a0ff6974e53e643acc42d1dcf2e7f797b3552"
+  integrity sha512-cIO74PvbW0qU8e0mIvk5IV3ToWdCq5FYG6gWPHHkx6gNdjlbAYvtfHmlCMXxjcoVaIdwy/IAt3+mDkZkfvb2Dg==
+  dependencies:
+    "@eslint-community/eslint-utils" "^4.2.0"
+    "@eslint-community/regexpp" "^4.6.1"
+    "@eslint/eslintrc" "^2.1.1"
+    "@eslint/js" "^8.46.0"
+    "@humanwhocodes/config-array" "^0.11.10"
+    "@humanwhocodes/module-importer" "^1.0.1"
+    "@nodelib/fs.walk" "^1.2.8"
+    ajv "^6.12.4"
+    chalk "^4.0.0"
+    cross-spawn "^7.0.2"
+    debug "^4.3.2"
+    doctrine "^3.0.0"
+    escape-string-regexp "^4.0.0"
+    eslint-scope "^7.2.2"
+    eslint-visitor-keys "^3.4.2"
+    espree "^9.6.1"
+    esquery "^1.4.2"
+    esutils "^2.0.2"
+    fast-deep-equal "^3.1.3"
+    file-entry-cache "^6.0.1"
+    find-up "^5.0.0"
+    glob-parent "^6.0.2"
+    globals "^13.19.0"
+    graphemer "^1.4.0"
+    ignore "^5.2.0"
+    imurmurhash "^0.1.4"
+    is-glob "^4.0.0"
+    is-path-inside "^3.0.3"
+    js-yaml "^4.1.0"
+    json-stable-stringify-without-jsonify "^1.0.1"
+    levn "^0.4.1"
+    lodash.merge "^4.6.2"
+    minimatch "^3.1.2"
+    natural-compare "^1.4.0"
+    optionator "^0.9.3"
+    strip-ansi "^6.0.1"
+    text-table "^0.2.0"
+
+espree@^9.6.0, espree@^9.6.1:
+  version "9.6.1"
+  resolved "https://registry.yarnpkg.com/espree/-/espree-9.6.1.tgz#a2a17b8e434690a5432f2f8018ce71d331a48c6f"
+  integrity sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==
+  dependencies:
+    acorn "^8.9.0"
+    acorn-jsx "^5.3.2"
+    eslint-visitor-keys "^3.4.1"
+
+esprima@2.7.x, esprima@^2.7.1:
+  version "2.7.3"
+  resolved "https://registry.yarnpkg.com/esprima/-/esprima-2.7.3.tgz#96e3b70d5779f6ad49cd032673d1c312767ba581"
+  integrity sha512-OarPfz0lFCiW4/AV2Oy1Rp9qu0iusTKqykwTspGCZtPxmF81JR4MmIebvF1F9+UOKth2ZubLQ4XGGaU+hSn99A==
+
+esprima@^4.0.0:
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.1.tgz#13b04cdb3e6c5d19df91ab6987a8695619b0aa71"
+  integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==
+
+esquery@^1.4.2:
+  version "1.5.0"
+  resolved "https://registry.yarnpkg.com/esquery/-/esquery-1.5.0.tgz#6ce17738de8577694edd7361c57182ac8cb0db0b"
+  integrity sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==
+  dependencies:
+    estraverse "^5.1.0"
+
+esrecurse@^4.3.0:
+  version "4.3.0"
+  resolved "https://registry.yarnpkg.com/esrecurse/-/esrecurse-4.3.0.tgz#7ad7964d679abb28bee72cec63758b1c5d2c9921"
+  integrity sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==
+  dependencies:
+    estraverse "^5.2.0"
+
+estraverse@^1.9.1:
+  version "1.9.3"
+  resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-1.9.3.tgz#af67f2dc922582415950926091a4005d29c9bb44"
+  integrity sha512-25w1fMXQrGdoquWnScXZGckOv+Wes+JDnuN/+7ex3SauFRS72r2lFDec0EKPt2YD1wUJ/IrfEex+9yp4hfSOJA==
+
+estraverse@^4.1.1:
+  version "4.3.0"
+  resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-4.3.0.tgz#398ad3f3c5a24948be7725e83d11a7de28cdbd1d"
+  integrity sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==
+
+estraverse@^5.1.0, estraverse@^5.2.0:
+  version "5.3.0"
+  resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-5.3.0.tgz#2eea5290702f26ab8fe5370370ff86c965d21123"
+  integrity sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==
+
+esutils@^2.0.2:
+  version "2.0.3"
+  resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.3.tgz#74d2eb4de0b8da1293711910d50775b9b710ef64"
+  integrity sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==
+
+etag@~1.8.1:
+  version "1.8.1"
+  resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.1.tgz#41ae2eeb65efa62268aebfea83ac7d79299b0887"
+  integrity sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==
+
+eth-ens-namehash@2.0.8, eth-ens-namehash@^2.0.0, eth-ens-namehash@^2.0.8:
+  version "2.0.8"
+  resolved "https://registry.yarnpkg.com/eth-ens-namehash/-/eth-ens-namehash-2.0.8.tgz#229ac46eca86d52e0c991e7cb2aef83ff0f68bcf"
+  integrity sha512-VWEI1+KJfz4Km//dadyvBBoBeSQ0MHTXPvr8UIXiLW6IanxvAV+DmlZAijZwAyggqGUfwQBeHf7tc9wzc1piSw==
+  dependencies:
+    idna-uts46-hx "^2.3.1"
+    js-sha3 "^0.5.7"
+
+eth-gas-reporter@^0.2.25:
+  version "0.2.25"
+  resolved "https://registry.yarnpkg.com/eth-gas-reporter/-/eth-gas-reporter-0.2.25.tgz#546dfa946c1acee93cb1a94c2a1162292d6ff566"
+  integrity sha512-1fRgyE4xUB8SoqLgN3eDfpDfwEfRxh2Sz1b7wzFbyQA+9TekMmvSjjoRu9SKcSVyK+vLkLIsVbJDsTWjw195OQ==
+  dependencies:
+    "@ethersproject/abi" "^5.0.0-beta.146"
+    "@solidity-parser/parser" "^0.14.0"
+    cli-table3 "^0.5.0"
+    colors "1.4.0"
+    ethereum-cryptography "^1.0.3"
+    ethers "^4.0.40"
+    fs-readdir-recursive "^1.1.0"
+    lodash "^4.17.14"
+    markdown-table "^1.1.3"
+    mocha "^7.1.1"
+    req-cwd "^2.0.0"
+    request "^2.88.0"
+    request-promise-native "^1.0.5"
+    sha1 "^1.1.1"
+    sync-request "^6.0.0"
+
+eth-lib@0.2.8:
+  version "0.2.8"
+  resolved "https://registry.yarnpkg.com/eth-lib/-/eth-lib-0.2.8.tgz#b194058bef4b220ad12ea497431d6cb6aa0623c8"
+  integrity sha512-ArJ7x1WcWOlSpzdoTBX8vkwlkSQ85CjjifSZtV4co64vWxSV8geWfPI9x4SVYu3DSxnX4yWFVTtGL+j9DUFLNw==
+  dependencies:
+    bn.js "^4.11.6"
+    elliptic "^6.4.0"
+    xhr-request-promise "^0.1.2"
+
+eth-lib@^0.1.26:
+  version "0.1.29"
+  resolved "https://registry.yarnpkg.com/eth-lib/-/eth-lib-0.1.29.tgz#0c11f5060d42da9f931eab6199084734f4dbd1d9"
+  integrity sha512-bfttrr3/7gG4E02HoWTDUcDDslN003OlOoBxk9virpAZQ1ja/jDgwkWB8QfJF7ojuEowrqy+lzp9VcJG7/k5bQ==
+  dependencies:
+    bn.js "^4.11.6"
+    elliptic "^6.4.0"
+    nano-json-stream-parser "^0.1.2"
+    servify "^0.1.12"
+    ws "^3.0.0"
+    xhr-request-promise "^0.1.2"
+
+ethereum-bloom-filters@^1.0.6:
+  version "1.0.10"
+  resolved "https://registry.yarnpkg.com/ethereum-bloom-filters/-/ethereum-bloom-filters-1.0.10.tgz#3ca07f4aed698e75bd134584850260246a5fed8a"
+  integrity sha512-rxJ5OFN3RwjQxDcFP2Z5+Q9ho4eIdEmSc2ht0fCu8Se9nbXjZ7/031uXoUYJ87KHCOdVeiUuwSnoS7hmYAGVHA==
+  dependencies:
+    js-sha3 "^0.8.0"
+
+ethereum-cryptography@0.1.3, ethereum-cryptography@^0.1.3:
+  version "0.1.3"
+  resolved "https://registry.yarnpkg.com/ethereum-cryptography/-/ethereum-cryptography-0.1.3.tgz#8d6143cfc3d74bf79bbd8edecdf29e4ae20dd191"
+  integrity sha512-w8/4x1SGGzc+tO97TASLja6SLd3fRIK2tLVcV2Gx4IB21hE19atll5Cq9o3d0ZmAYC/8aw0ipieTSiekAea4SQ==
+  dependencies:
+    "@types/pbkdf2" "^3.0.0"
+    "@types/secp256k1" "^4.0.1"
+    blakejs "^1.1.0"
+    browserify-aes "^1.2.0"
+    bs58check "^2.1.2"
+    create-hash "^1.2.0"
+    create-hmac "^1.1.7"
+    hash.js "^1.1.7"
+    keccak "^3.0.0"
+    pbkdf2 "^3.0.17"
+    randombytes "^2.1.0"
+    safe-buffer "^5.1.2"
+    scrypt-js "^3.0.0"
+    secp256k1 "^4.0.1"
+    setimmediate "^1.0.5"
+
+ethereum-cryptography@^1.0.3:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/ethereum-cryptography/-/ethereum-cryptography-1.2.0.tgz#5ccfa183e85fdaf9f9b299a79430c044268c9b3a"
+  integrity sha512-6yFQC9b5ug6/17CQpCyE3k9eKBMdhyVjzUy1WkiuY/E4vj/SXDBbCw8QEIaXqf0Mf2SnY6RmpDcwlUmBSS0EJw==
+  dependencies:
+    "@noble/hashes" "1.2.0"
+    "@noble/secp256k1" "1.7.1"
+    "@scure/bip32" "1.1.5"
+    "@scure/bip39" "1.1.1"
+
+ethereum-ens@^0.8.0:
+  version "0.8.0"
+  resolved "https://registry.yarnpkg.com/ethereum-ens/-/ethereum-ens-0.8.0.tgz#6d0f79acaa61fdbc87d2821779c4e550243d4c57"
+  integrity sha512-a8cBTF4AWw1Q1Y37V1LSCS9pRY4Mh3f8vCg5cbXCCEJ3eno1hbI/+Ccv9SZLISYpqQhaglP3Bxb/34lS4Qf7Bg==
+  dependencies:
+    bluebird "^3.4.7"
+    eth-ens-namehash "^2.0.0"
+    js-sha3 "^0.5.7"
+    pako "^1.0.4"
+    underscore "^1.8.3"
+    web3 "^1.0.0-beta.34"
+
+ethereumjs-abi@^0.6.8:
+  version "0.6.8"
+  resolved "https://registry.yarnpkg.com/ethereumjs-abi/-/ethereumjs-abi-0.6.8.tgz#71bc152db099f70e62f108b7cdfca1b362c6fcae"
+  integrity sha512-Tx0r/iXI6r+lRsdvkFDlut0N08jWMnKRZ6Gkq+Nmw75lZe4e6o3EkSnkaBP5NF6+m5PTGAr9JP43N3LyeoglsA==
+  dependencies:
+    bn.js "^4.11.8"
+    ethereumjs-util "^6.0.0"
+
+ethereumjs-util@^6.0.0, ethereumjs-util@^6.2.1:
+  version "6.2.1"
+  resolved "https://registry.yarnpkg.com/ethereumjs-util/-/ethereumjs-util-6.2.1.tgz#fcb4e4dd5ceacb9d2305426ab1a5cd93e3163b69"
+  integrity sha512-W2Ktez4L01Vexijrm5EB6w7dg4n/TgpoYU4avuT5T3Vmnw/eCRtiBrJfQYS/DCSvDIOLn2k57GcHdeBcgVxAqw==
+  dependencies:
+    "@types/bn.js" "^4.11.3"
+    bn.js "^4.11.0"
+    create-hash "^1.1.2"
+    elliptic "^6.5.2"
+    ethereum-cryptography "^0.1.3"
+    ethjs-util "0.1.6"
+    rlp "^2.2.3"
+
+ethereumjs-util@^7.1.0, ethereumjs-util@^7.1.1, ethereumjs-util@^7.1.2, ethereumjs-util@^7.1.4, ethereumjs-util@^7.1.5:
+  version "7.1.5"
+  resolved "https://registry.yarnpkg.com/ethereumjs-util/-/ethereumjs-util-7.1.5.tgz#9ecf04861e4fbbeed7465ece5f23317ad1129181"
+  integrity sha512-SDl5kKrQAudFBUe5OJM9Ac6WmMyYmXX/6sTmLZ3ffG2eY6ZIGBes3pEDxNN6V72WyOw4CPD5RomKdsa8DAAwLg==
+  dependencies:
+    "@types/bn.js" "^5.1.0"
+    bn.js "^5.1.2"
+    create-hash "^1.1.2"
+    ethereum-cryptography "^0.1.3"
+    rlp "^2.2.4"
+
+ethers@^4.0.0-beta.1, ethers@^4.0.32, ethers@^4.0.40:
+  version "4.0.49"
+  resolved "https://registry.yarnpkg.com/ethers/-/ethers-4.0.49.tgz#0eb0e9161a0c8b4761be547396bbe2fb121a8894"
+  integrity sha512-kPltTvWiyu+OktYy1IStSO16i2e7cS9D9OxZ81q2UUaiNPVrm/RTcbxamCXF9VUSKzJIdJV68EAIhTEVBalRWg==
+  dependencies:
+    aes-js "3.0.0"
+    bn.js "^4.11.9"
+    elliptic "6.5.4"
+    hash.js "1.1.3"
+    js-sha3 "0.5.7"
+    scrypt-js "2.0.4"
+    setimmediate "1.0.4"
+    uuid "2.0.1"
+    xmlhttprequest "1.8.0"
+
+ethers@^5.0.13, ethers@^5.0.32, ethers@^5.5.3, ethers@^5.7.0, ethers@^5.7.2:
+  version "5.7.2"
+  resolved "https://registry.yarnpkg.com/ethers/-/ethers-5.7.2.tgz#3a7deeabbb8c030d4126b24f84e525466145872e"
+  integrity sha512-wswUsmWo1aOK8rR7DIKiWSw9DbLWe6x98Jrn8wcTflTVvaXhAMaB5zGAXy0GYQEQp9iO1iSHWVyARQm11zUtyg==
+  dependencies:
+    "@ethersproject/abi" "5.7.0"
+    "@ethersproject/abstract-provider" "5.7.0"
+    "@ethersproject/abstract-signer" "5.7.0"
+    "@ethersproject/address" "5.7.0"
+    "@ethersproject/base64" "5.7.0"
+    "@ethersproject/basex" "5.7.0"
+    "@ethersproject/bignumber" "5.7.0"
+    "@ethersproject/bytes" "5.7.0"
+    "@ethersproject/constants" "5.7.0"
+    "@ethersproject/contracts" "5.7.0"
+    "@ethersproject/hash" "5.7.0"
+    "@ethersproject/hdnode" "5.7.0"
+    "@ethersproject/json-wallets" "5.7.0"
+    "@ethersproject/keccak256" "5.7.0"
+    "@ethersproject/logger" "5.7.0"
+    "@ethersproject/networks" "5.7.1"
+    "@ethersproject/pbkdf2" "5.7.0"
+    "@ethersproject/properties" "5.7.0"
+    "@ethersproject/providers" "5.7.2"
+    "@ethersproject/random" "5.7.0"
+    "@ethersproject/rlp" "5.7.0"
+    "@ethersproject/sha2" "5.7.0"
+    "@ethersproject/signing-key" "5.7.0"
+    "@ethersproject/solidity" "5.7.0"
+    "@ethersproject/strings" "5.7.0"
+    "@ethersproject/transactions" "5.7.0"
+    "@ethersproject/units" "5.7.0"
+    "@ethersproject/wallet" "5.7.0"
+    "@ethersproject/web" "5.7.1"
+    "@ethersproject/wordlists" "5.7.0"
+
+ethjs-unit@0.1.6:
+  version "0.1.6"
+  resolved "https://registry.yarnpkg.com/ethjs-unit/-/ethjs-unit-0.1.6.tgz#c665921e476e87bce2a9d588a6fe0405b2c41699"
+  integrity sha512-/Sn9Y0oKl0uqQuvgFk/zQgR7aw1g36qX/jzSQ5lSwlO0GigPymk4eGQfeNTD03w1dPOqfz8V77Cy43jH56pagw==
+  dependencies:
+    bn.js "4.11.6"
+    number-to-bn "1.7.0"
+
+ethjs-util@0.1.6, ethjs-util@^0.1.6:
+  version "0.1.6"
+  resolved "https://registry.yarnpkg.com/ethjs-util/-/ethjs-util-0.1.6.tgz#f308b62f185f9fe6237132fb2a9818866a5cd536"
+  integrity sha512-CUnVOQq7gSpDHZVVrQW8ExxUETWrnrvXYvYz55wOU8Uj4VCgw56XC2B/fVqQN+f7gmrnRHSLVnFAwsCuNwji8w==
+  dependencies:
+    is-hex-prefixed "1.0.0"
+    strip-hex-prefix "1.0.0"
+
+event-target-shim@^5.0.0:
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/event-target-shim/-/event-target-shim-5.0.1.tgz#5d4d3ebdf9583d63a5333ce2deb7480ab2b05789"
+  integrity sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==
+
+eventemitter3@4.0.4:
+  version "4.0.4"
+  resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-4.0.4.tgz#b5463ace635a083d018bdc7c917b4c5f10a85384"
+  integrity sha512-rlaVLnVxtxvoyLsQQFBx53YmXHDxRIzzTLbdfxqi4yocpSjAxXwkU0cScM5JgSKMqEhrZpnvQ2D9gjylR0AimQ==
+
+evp_bytestokey@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz#7fcbdb198dc71959432efe13842684e0525acb02"
+  integrity sha512-/f2Go4TognH/KvCISP7OUsHn85hT9nUkxxA9BEWxFn+Oj9o8ZNLm/40hdlgSLyuOimsrTKLUMEorQexp/aPQeA==
+  dependencies:
+    md5.js "^1.3.4"
+    safe-buffer "^5.1.1"
+
+express@^4.14.0, express@^4.18.1:
+  version "4.18.2"
+  resolved "https://registry.yarnpkg.com/express/-/express-4.18.2.tgz#3fabe08296e930c796c19e3c516979386ba9fd59"
+  integrity sha512-5/PsL6iGPdfQ/lKM1UuielYgv3BUoJfz1aUwU9vHZ+J7gyvwdQXFEBIEIaxeGf0GIcreATNyBExtalisDbuMqQ==
+  dependencies:
+    accepts "~1.3.8"
+    array-flatten "1.1.1"
+    body-parser "1.20.1"
+    content-disposition "0.5.4"
+    content-type "~1.0.4"
+    cookie "0.5.0"
+    cookie-signature "1.0.6"
+    debug "2.6.9"
+    depd "2.0.0"
+    encodeurl "~1.0.2"
+    escape-html "~1.0.3"
+    etag "~1.8.1"
+    finalhandler "1.2.0"
+    fresh "0.5.2"
+    http-errors "2.0.0"
+    merge-descriptors "1.0.1"
+    methods "~1.1.2"
+    on-finished "2.4.1"
+    parseurl "~1.3.3"
+    path-to-regexp "0.1.7"
+    proxy-addr "~2.0.7"
+    qs "6.11.0"
+    range-parser "~1.2.1"
+    safe-buffer "5.2.1"
+    send "0.18.0"
+    serve-static "1.15.0"
+    setprototypeof "1.2.0"
+    statuses "2.0.1"
+    type-is "~1.6.18"
+    utils-merge "1.0.1"
+    vary "~1.1.2"
+
+ext@^1.1.2:
+  version "1.7.0"
+  resolved "https://registry.yarnpkg.com/ext/-/ext-1.7.0.tgz#0ea4383c0103d60e70be99e9a7f11027a33c4f5f"
+  integrity sha512-6hxeJYaL110a9b5TEJSj0gojyHQAmA2ch5Os+ySCiA1QGdS697XWY1pzsrSjqA9LDEEgdB/KypIlR59RcLuHYw==
+  dependencies:
+    type "^2.7.2"
+
+extend@~3.0.2:
+  version "3.0.2"
+  resolved "https://registry.yarnpkg.com/extend/-/extend-3.0.2.tgz#f8b1136b4071fbd8eb140aff858b1019ec2915fa"
+  integrity sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==
+
+extsprintf@1.3.0:
+  version "1.3.0"
+  resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.3.0.tgz#96918440e3041a7a414f8c52e3c574eb3c3e1e05"
+  integrity sha512-11Ndz7Nv+mvAC1j0ktTa7fAb0vLyGGX+rMHNBYQviQDGU0Hw7lhctJANqbPhu9nV9/izT/IntTgZ7Im/9LJs9g==
+
+extsprintf@^1.2.0:
+  version "1.4.1"
+  resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.4.1.tgz#8d172c064867f235c0c84a596806d279bf4bcc07"
+  integrity sha512-Wrk35e8ydCKDj/ArClo1VrPVmN8zph5V4AtHwIuHhvMXsKf73UT3BOD+azBIW+3wOJ4FhEH7zyaJCFvChjYvMA==
+
+fast-check@3.1.1:
+  version "3.1.1"
+  resolved "https://registry.yarnpkg.com/fast-check/-/fast-check-3.1.1.tgz#72c5ae7022a4e86504762e773adfb8a5b0b01252"
+  integrity sha512-3vtXinVyuUKCKFKYcwXhGE6NtGWkqF8Yh3rvMZNzmwz8EPrgoc/v4pDdLHyLnCyCI5MZpZZkDEwFyXyEONOxpA==
+  dependencies:
+    pure-rand "^5.0.1"
+
+fast-deep-equal@^3.1.1, fast-deep-equal@^3.1.3:
+  version "3.1.3"
+  resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525"
+  integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==
+
+fast-diff@^1.1.2, fast-diff@^1.2.0:
+  version "1.3.0"
+  resolved "https://registry.yarnpkg.com/fast-diff/-/fast-diff-1.3.0.tgz#ece407fa550a64d638536cd727e129c61616e0f0"
+  integrity sha512-VxPP4NqbUjj6MaAOafWeUn2cXWLcCtljklUtZf0Ind4XQ+QPtmA0b18zZy0jIQx+ExRVCR/ZQpBmik5lXshNsw==
+
+fast-glob@^3.0.3, fast-glob@^3.2.9:
+  version "3.3.1"
+  resolved "https://registry.yarnpkg.com/fast-glob/-/fast-glob-3.3.1.tgz#784b4e897340f3dbbef17413b3f11acf03c874c4"
+  integrity sha512-kNFPyjhh5cKjrUltxs+wFx+ZkbRaxxmZ+X0ZU31SOsxCEtP9VPgtq2teZw1DebupL5GmDaNQ6yKMMVcM41iqDg==
+  dependencies:
+    "@nodelib/fs.stat" "^2.0.2"
+    "@nodelib/fs.walk" "^1.2.3"
+    glob-parent "^5.1.2"
+    merge2 "^1.3.0"
+    micromatch "^4.0.4"
+
+fast-json-stable-stringify@^2.0.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz#874bf69c6f404c2b5d99c481341399fd55892633"
+  integrity sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==
+
+fast-levenshtein@^2.0.6, fast-levenshtein@~2.0.6:
+  version "2.0.6"
+  resolved "https://registry.yarnpkg.com/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz#3d8a5c66883a16a30ca8643e851f19baa7797917"
+  integrity sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==
+
+fastq@^1.6.0:
+  version "1.15.0"
+  resolved "https://registry.yarnpkg.com/fastq/-/fastq-1.15.0.tgz#d04d07c6a2a68fe4599fea8d2e103a937fae6b3a"
+  integrity sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw==
+  dependencies:
+    reusify "^1.0.4"
+
+file-entry-cache@^6.0.1:
+  version "6.0.1"
+  resolved "https://registry.yarnpkg.com/file-entry-cache/-/file-entry-cache-6.0.1.tgz#211b2dd9659cb0394b073e7323ac3c933d522027"
+  integrity sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==
+  dependencies:
+    flat-cache "^3.0.4"
+
+fill-range@^7.0.1:
+  version "7.0.1"
+  resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.0.1.tgz#1919a6a7c75fe38b2c7c77e5198535da9acdda40"
+  integrity sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==
+  dependencies:
+    to-regex-range "^5.0.1"
+
+finalhandler@1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.2.0.tgz#7d23fe5731b207b4640e4fcd00aec1f9207a7b32"
+  integrity sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==
+  dependencies:
+    debug "2.6.9"
+    encodeurl "~1.0.2"
+    escape-html "~1.0.3"
+    on-finished "2.4.1"
+    parseurl "~1.3.3"
+    statuses "2.0.1"
+    unpipe "~1.0.0"
+
+find-replace@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/find-replace/-/find-replace-3.0.0.tgz#3e7e23d3b05167a76f770c9fbd5258b0def68c38"
+  integrity sha512-6Tb2myMioCAgv5kfvP5/PkZZ/ntTpVK39fHY7WkWBgvbeE+VHd/tZuZ4mrC+bxh4cfOZeYKVPaJIZtZXV7GNCQ==
+  dependencies:
+    array-back "^3.0.1"
+
+find-up@3.0.0, find-up@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/find-up/-/find-up-3.0.0.tgz#49169f1d7993430646da61ecc5ae355c21c97b73"
+  integrity sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==
+  dependencies:
+    locate-path "^3.0.0"
+
+find-up@5.0.0, find-up@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/find-up/-/find-up-5.0.0.tgz#4c92819ecb7083561e4f4a240a86be5198f536fc"
+  integrity sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==
+  dependencies:
+    locate-path "^6.0.0"
+    path-exists "^4.0.0"
+
+find-up@^1.0.0:
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/find-up/-/find-up-1.1.2.tgz#6b2e9822b1a2ce0a60ab64d610eccad53cb24d0f"
+  integrity sha512-jvElSjyuo4EMQGoTwo1uJU5pQMwTW5lS1x05zzfJuTIyLR3zwO27LYrxNg+dlvKpGOuGy/MzBdXh80g0ve5+HA==
+  dependencies:
+    path-exists "^2.0.0"
+    pinkie-promise "^2.0.0"
+
+find-up@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/find-up/-/find-up-2.1.0.tgz#45d1b7e506c717ddd482775a2b77920a3c0c57a7"
+  integrity sha512-NWzkk0jSJtTt08+FBFMvXoeZnOJD+jTtsRmBYbAIzJdX6l7dLgR7CTubCM5/eDdPUBvLCeVasP1brfVR/9/EZQ==
+  dependencies:
+    locate-path "^2.0.0"
+
+find-versions@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/find-versions/-/find-versions-4.0.0.tgz#3c57e573bf97769b8cb8df16934b627915da4965"
+  integrity sha512-wgpWy002tA+wgmO27buH/9KzyEOQnKsG/R0yrcjPT9BOFm0zRBVQbZ95nRGXWMywS8YR5knRbpohio0bcJABxQ==
+  dependencies:
+    semver-regex "^3.1.2"
+
+flat-cache@^3.0.4:
+  version "3.0.4"
+  resolved "https://registry.yarnpkg.com/flat-cache/-/flat-cache-3.0.4.tgz#61b0338302b2fe9f957dcc32fc2a87f1c3048b11"
+  integrity sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==
+  dependencies:
+    flatted "^3.1.0"
+    rimraf "^3.0.2"
+
+flat@^4.1.0:
+  version "4.1.1"
+  resolved "https://registry.yarnpkg.com/flat/-/flat-4.1.1.tgz#a392059cc382881ff98642f5da4dde0a959f309b"
+  integrity sha512-FmTtBsHskrU6FJ2VxCnsDb84wu9zhmO3cUX2kGFb5tuwhfXxGciiT0oRY+cck35QmG+NmGh5eLz6lLCpWTqwpA==
+  dependencies:
+    is-buffer "~2.0.3"
+
+flat@^5.0.2:
+  version "5.0.2"
+  resolved "https://registry.yarnpkg.com/flat/-/flat-5.0.2.tgz#8ca6fe332069ffa9d324c327198c598259ceb241"
+  integrity sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==
+
+flatted@^3.1.0:
+  version "3.2.7"
+  resolved "https://registry.yarnpkg.com/flatted/-/flatted-3.2.7.tgz#609f39207cb614b89d0765b477cb2d437fbf9787"
+  integrity sha512-5nqDSxl8nn5BSNxyR3n4I6eDmbolI6WT+QqR547RwxQapgjQBmtktdP+HTBb/a/zLsbzERTONyUB5pefh5TtjQ==
+
+fmix@^0.1.0:
+  version "0.1.0"
+  resolved "https://registry.yarnpkg.com/fmix/-/fmix-0.1.0.tgz#c7bbf124dec42c9d191cfb947d0a9778dd986c0c"
+  integrity sha512-Y6hyofImk9JdzU8k5INtTXX1cu8LDlePWDFU5sftm9H+zKCr5SGrVjdhkvsim646cw5zD0nADj8oHyXMZmCZ9w==
+  dependencies:
+    imul "^1.0.0"
+
+follow-redirects@^1.12.1, follow-redirects@^1.14.0, follow-redirects@^1.14.9:
+  version "1.15.2"
+  resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.2.tgz#b460864144ba63f2681096f274c4e57026da2c13"
+  integrity sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==
+
+for-each@^0.3.3:
+  version "0.3.3"
+  resolved "https://registry.yarnpkg.com/for-each/-/for-each-0.3.3.tgz#69b447e88a0a5d32c3e7084f3f1710034b21376e"
+  integrity sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==
+  dependencies:
+    is-callable "^1.1.3"
+
+forever-agent@~0.6.1:
+  version "0.6.1"
+  resolved "https://registry.yarnpkg.com/forever-agent/-/forever-agent-0.6.1.tgz#fbc71f0c41adeb37f96c577ad1ed42d8fdacca91"
+  integrity sha512-j0KLYPhm6zeac4lz3oJ3o65qvgQCcPubiyotZrXqEaG4hNagNYO8qdlUrX5vwqv9ohqeT/Z3j6+yW067yWWdUw==
+
+form-data-encoder@1.7.1:
+  version "1.7.1"
+  resolved "https://registry.yarnpkg.com/form-data-encoder/-/form-data-encoder-1.7.1.tgz#ac80660e4f87ee0d3d3c3638b7da8278ddb8ec96"
+  integrity sha512-EFRDrsMm/kyqbTQocNvRXMLjc7Es2Vk+IQFx/YW7hkUH1eBl4J1fqiP34l74Yt0pFLCNpc06fkbVk00008mzjg==
+
+form-data@^2.2.0:
+  version "2.5.1"
+  resolved "https://registry.yarnpkg.com/form-data/-/form-data-2.5.1.tgz#f2cbec57b5e59e23716e128fe44d4e5dd23895f4"
+  integrity sha512-m21N3WOmEEURgk6B9GLOE4RuWOFf28Lhh9qGYeNlGq4VDXUlJy2th2slBNU8Gp8EzloYZOibZJ7t5ecIrFSjVA==
+  dependencies:
+    asynckit "^0.4.0"
+    combined-stream "^1.0.6"
+    mime-types "^2.1.12"
+
+form-data@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/form-data/-/form-data-4.0.0.tgz#93919daeaf361ee529584b9b31664dc12c9fa452"
+  integrity sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==
+  dependencies:
+    asynckit "^0.4.0"
+    combined-stream "^1.0.8"
+    mime-types "^2.1.12"
+
+form-data@~2.3.2:
+  version "2.3.3"
+  resolved "https://registry.yarnpkg.com/form-data/-/form-data-2.3.3.tgz#dcce52c05f644f298c6a7ab936bd724ceffbf3a6"
+  integrity sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==
+  dependencies:
+    asynckit "^0.4.0"
+    combined-stream "^1.0.6"
+    mime-types "^2.1.12"
+
+forwarded@0.2.0:
+  version "0.2.0"
+  resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.2.0.tgz#2269936428aad4c15c7ebe9779a84bf0b2a81811"
+  integrity sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==
+
+fp-ts@1.19.3:
+  version "1.19.3"
+  resolved "https://registry.yarnpkg.com/fp-ts/-/fp-ts-1.19.3.tgz#261a60d1088fbff01f91256f91d21d0caaaaa96f"
+  integrity sha512-H5KQDspykdHuztLTg+ajGN0Z2qUjcEf3Ybxc6hLt0k7/zPkn29XnKnxlBPyW2XIddWrGaJBzBl4VLYOtk39yZg==
+
+fp-ts@^1.0.0:
+  version "1.19.5"
+  resolved "https://registry.yarnpkg.com/fp-ts/-/fp-ts-1.19.5.tgz#3da865e585dfa1fdfd51785417357ac50afc520a"
+  integrity sha512-wDNqTimnzs8QqpldiId9OavWK2NptormjXnRJTQecNjzwfyp6P/8s/zG8e4h3ja3oqkKaY72UlTjQYt/1yXf9A==
+
+fresh@0.5.2:
+  version "0.5.2"
+  resolved "https://registry.yarnpkg.com/fresh/-/fresh-0.5.2.tgz#3d8cadd90d976569fa835ab1f8e4b23a105605a7"
+  integrity sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==
+
+fs-extra@^0.30.0:
+  version "0.30.0"
+  resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-0.30.0.tgz#f233ffcc08d4da7d432daa449776989db1df93f0"
+  integrity sha512-UvSPKyhMn6LEd/WpUaV9C9t3zATuqoqfWc3QdPhPLb58prN9tqYPlPWi8Krxi44loBoUzlobqZ3+8tGpxxSzwA==
+  dependencies:
+    graceful-fs "^4.1.2"
+    jsonfile "^2.1.0"
+    klaw "^1.0.0"
+    path-is-absolute "^1.0.0"
+    rimraf "^2.2.8"
+
+fs-extra@^10.0.0, fs-extra@^10.1.0:
+  version "10.1.0"
+  resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-10.1.0.tgz#02873cfbc4084dde127eaa5f9905eef2325d1abf"
+  integrity sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ==
+  dependencies:
+    graceful-fs "^4.2.0"
+    jsonfile "^6.0.1"
+    universalify "^2.0.0"
+
+fs-extra@^4.0.2:
+  version "4.0.3"
+  resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-4.0.3.tgz#0d852122e5bc5beb453fb028e9c0c9bf36340c94"
+  integrity sha512-q6rbdDd1o2mAnQreO7YADIxf/Whx4AHBiRf6d+/cVT8h44ss+lHgxf1FemcqDnQt9X3ct4McHr+JMGlYSsK7Cg==
+  dependencies:
+    graceful-fs "^4.1.2"
+    jsonfile "^4.0.0"
+    universalify "^0.1.0"
+
+fs-extra@^7.0.0, fs-extra@^7.0.1:
+  version "7.0.1"
+  resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-7.0.1.tgz#4f189c44aa123b895f722804f55ea23eadc348e9"
+  integrity sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==
+  dependencies:
+    graceful-fs "^4.1.2"
+    jsonfile "^4.0.0"
+    universalify "^0.1.0"
+
+fs-extra@^8.1.0:
+  version "8.1.0"
+  resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-8.1.0.tgz#49d43c45a88cd9677668cb7be1b46efdb8d2e1c0"
+  integrity sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==
+  dependencies:
+    graceful-fs "^4.2.0"
+    jsonfile "^4.0.0"
+    universalify "^0.1.0"
+
+fs-minipass@^1.2.7:
+  version "1.2.7"
+  resolved "https://registry.yarnpkg.com/fs-minipass/-/fs-minipass-1.2.7.tgz#ccff8570841e7fe4265693da88936c55aed7f7c7"
+  integrity sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==
+  dependencies:
+    minipass "^2.6.0"
+
+fs-readdir-recursive@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/fs-readdir-recursive/-/fs-readdir-recursive-1.1.0.tgz#e32fc030a2ccee44a6b5371308da54be0b397d27"
+  integrity sha512-GNanXlVr2pf02+sPN40XN8HG+ePaNcvM0q5mZBd668Obwb0yD5GiUbZOFgwn8kGMY6I3mdyDJzieUy3PTYyTRA==
+
+fs.realpath@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f"
+  integrity sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==
+
+fsevents@~2.1.1:
+  version "2.1.3"
+  resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.1.3.tgz#fb738703ae8d2f9fe900c33836ddebee8b97f23e"
+  integrity sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==
+
+fsevents@~2.3.2:
+  version "2.3.2"
+  resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.2.tgz#8a526f78b8fdf4623b709e0b975c52c24c02fd1a"
+  integrity sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==
+
+function-bind@^1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.1.tgz#a56899d3ea3c9bab874bb9773b7c5ede92f4895d"
+  integrity sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==
+
+function.prototype.name@^1.1.5:
+  version "1.1.5"
+  resolved "https://registry.yarnpkg.com/function.prototype.name/-/function.prototype.name-1.1.5.tgz#cce0505fe1ffb80503e6f9e46cc64e46a12a9621"
+  integrity sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.1.3"
+    es-abstract "^1.19.0"
+    functions-have-names "^1.2.2"
+
+functional-red-black-tree@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz#1b0ab3bd553b2a0d6399d29c0e3ea0b252078327"
+  integrity sha512-dsKNQNdj6xA3T+QlADDA7mOSlX0qiMINjn0cgr+eGHGsbSHzTabcIogz2+p/iqP1Xs6EP/sS2SbqH+brGTbq0g==
+
+functions-have-names@^1.2.2, functions-have-names@^1.2.3:
+  version "1.2.3"
+  resolved "https://registry.yarnpkg.com/functions-have-names/-/functions-have-names-1.2.3.tgz#0404fe4ee2ba2f607f0e0ec3c80bae994133b834"
+  integrity sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==
+
+ganache@^7.5.0:
+  version "7.9.0"
+  resolved "https://registry.yarnpkg.com/ganache/-/ganache-7.9.0.tgz#561deceb376b1c4e8998ac8e5a842574507d3295"
+  integrity sha512-KdsTZaAKqDXTNDMKnLzg0ngX8wnZKyVGm7HD03GIyUMVRuXI83s0CUEaGIDWRUWTQP7BE8sDh7QtbW+NoX4zrQ==
+  dependencies:
+    "@trufflesuite/bigint-buffer" "1.1.10"
+    "@trufflesuite/uws-js-unofficial" "20.10.0-unofficial.2"
+    "@types/bn.js" "^5.1.0"
+    "@types/lru-cache" "5.1.1"
+    "@types/seedrandom" "3.0.1"
+    abstract-level "1.0.3"
+    abstract-leveldown "7.2.0"
+    async-eventemitter "0.2.4"
+    emittery "0.10.0"
+    keccak "3.0.2"
+    leveldown "6.1.0"
+    secp256k1 "4.0.3"
+  optionalDependencies:
+    bufferutil "4.0.5"
+    utf-8-validate "5.0.7"
+
+get-caller-file@^1.0.1:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-1.0.3.tgz#f978fa4c90d1dfe7ff2d6beda2a515e713bdcf4a"
+  integrity sha512-3t6rVToeoZfYSGd8YoLFR2DJkiQrIiUrGcjvFX2mDw3bn6k2OtwHN0TNCLbBO+w8qTvimhDkv+LSscbJY1vE6w==
+
+get-caller-file@^2.0.1, get-caller-file@^2.0.5:
+  version "2.0.5"
+  resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-2.0.5.tgz#4f94412a82db32f36e3b0b9741f8a97feb031f7e"
+  integrity sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==
+
+get-func-name@^2.0.0:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/get-func-name/-/get-func-name-2.0.2.tgz#0d7cf20cd13fda808669ffa88f4ffc7a3943fc41"
+  integrity sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ==
+
+get-intrinsic@^1.0.2, get-intrinsic@^1.1.1, get-intrinsic@^1.1.3, get-intrinsic@^1.2.0, get-intrinsic@^1.2.1:
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.2.1.tgz#d295644fed4505fc9cde952c37ee12b477a83d82"
+  integrity sha512-2DcsyfABl+gVHEfCOaTrWgyt+tb6MSEGmKq+kI5HwLbIYgjgmMcV8KQ41uaKz1xxUcn9tJtgFbQUEVcEbd0FYw==
+  dependencies:
+    function-bind "^1.1.1"
+    has "^1.0.3"
+    has-proto "^1.0.1"
+    has-symbols "^1.0.3"
+
+get-port@^3.1.0:
+  version "3.2.0"
+  resolved "https://registry.yarnpkg.com/get-port/-/get-port-3.2.0.tgz#dd7ce7de187c06c8bf353796ac71e099f0980ebc"
+  integrity sha512-x5UJKlgeUiNT8nyo/AcnwLnZuZNcSjSw0kogRB+Whd1fjjFq4B1hySFxSFWWSn4mIBzg3sRNUDFYc4g5gjPoLg==
+
+get-stream@^5.1.0:
+  version "5.2.0"
+  resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-5.2.0.tgz#4966a1795ee5ace65e706c4b7beb71257d6e22d3"
+  integrity sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==
+  dependencies:
+    pump "^3.0.0"
+
+get-stream@^6.0.1:
+  version "6.0.1"
+  resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-6.0.1.tgz#a262d8eef67aced57c2852ad6167526a43cbf7b7"
+  integrity sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==
+
+get-symbol-description@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/get-symbol-description/-/get-symbol-description-1.0.0.tgz#7fdb81c900101fbd564dd5f1a30af5aadc1e58d6"
+  integrity sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==
+  dependencies:
+    call-bind "^1.0.2"
+    get-intrinsic "^1.1.1"
+
+getpass@^0.1.1:
+  version "0.1.7"
+  resolved "https://registry.yarnpkg.com/getpass/-/getpass-0.1.7.tgz#5eff8e3e684d569ae4cb2b1282604e8ba62149fa"
+  integrity sha512-0fzj9JxOLfJ+XGLhR8ze3unN0KZCgZwiSSDz168VERjK8Wl8kVSdcu2kspd4s4wtAa1y/qrVRiAA0WclVsu0ng==
+  dependencies:
+    assert-plus "^1.0.0"
+
+ghost-testrpc@^0.0.2:
+  version "0.0.2"
+  resolved "https://registry.yarnpkg.com/ghost-testrpc/-/ghost-testrpc-0.0.2.tgz#c4de9557b1d1ae7b2d20bbe474a91378ca90ce92"
+  integrity sha512-i08dAEgJ2g8z5buJIrCTduwPIhih3DP+hOCTyyryikfV8T0bNvHnGXO67i0DD1H4GBDETTclPy9njZbfluQYrQ==
+  dependencies:
+    chalk "^2.4.2"
+    node-emoji "^1.10.0"
+
+glob-parent@^5.1.2, glob-parent@~5.1.0, glob-parent@~5.1.2:
+  version "5.1.2"
+  resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-5.1.2.tgz#869832c58034fe68a4093c17dc15e8340d8401c4"
+  integrity sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==
+  dependencies:
+    is-glob "^4.0.1"
+
+glob-parent@^6.0.2:
+  version "6.0.2"
+  resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-6.0.2.tgz#6d237d99083950c79290f24c7642a3de9a28f9e3"
+  integrity sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==
+  dependencies:
+    is-glob "^4.0.3"
+
+glob@7.1.3:
+  version "7.1.3"
+  resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.3.tgz#3960832d3f1574108342dafd3a67b332c0969df1"
+  integrity sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==
+  dependencies:
+    fs.realpath "^1.0.0"
+    inflight "^1.0.4"
+    inherits "2"
+    minimatch "^3.0.4"
+    once "^1.3.0"
+    path-is-absolute "^1.0.0"
+
+glob@7.1.7:
+  version "7.1.7"
+  resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.7.tgz#3b193e9233f01d42d0b3f78294bbeeb418f94a90"
+  integrity sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==
+  dependencies:
+    fs.realpath "^1.0.0"
+    inflight "^1.0.4"
+    inherits "2"
+    minimatch "^3.0.4"
+    once "^1.3.0"
+    path-is-absolute "^1.0.0"
+
+glob@7.2.0:
+  version "7.2.0"
+  resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.0.tgz#d15535af7732e02e948f4c41628bd910293f6023"
+  integrity sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==
+  dependencies:
+    fs.realpath "^1.0.0"
+    inflight "^1.0.4"
+    inherits "2"
+    minimatch "^3.0.4"
+    once "^1.3.0"
+    path-is-absolute "^1.0.0"
+
+glob@^5.0.15:
+  version "5.0.15"
+  resolved "https://registry.yarnpkg.com/glob/-/glob-5.0.15.tgz#1bc936b9e02f4a603fcc222ecf7633d30b8b93b1"
+  integrity sha512-c9IPMazfRITpmAAKi22dK1VKxGDX9ehhqfABDriL/lzO92xcUKEJPQHrVA/2YHSNFB4iFlykVmWvwo48nr3OxA==
+  dependencies:
+    inflight "^1.0.4"
+    inherits "2"
+    minimatch "2 || 3"
+    once "^1.3.0"
+    path-is-absolute "^1.0.0"
+
+glob@^7.0.0, glob@^7.1.3:
+  version "7.2.3"
+  resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.3.tgz#b8df0fb802bbfa8e89bd1d938b4e16578ed44f2b"
+  integrity sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==
+  dependencies:
+    fs.realpath "^1.0.0"
+    inflight "^1.0.4"
+    inherits "2"
+    minimatch "^3.1.1"
+    once "^1.3.0"
+    path-is-absolute "^1.0.0"
+
+glob@^8.0.3:
+  version "8.1.0"
+  resolved "https://registry.yarnpkg.com/glob/-/glob-8.1.0.tgz#d388f656593ef708ee3e34640fdfb99a9fd1c33e"
+  integrity sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==
+  dependencies:
+    fs.realpath "^1.0.0"
+    inflight "^1.0.4"
+    inherits "2"
+    minimatch "^5.0.1"
+    once "^1.3.0"
+
+global-modules@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/global-modules/-/global-modules-2.0.0.tgz#997605ad2345f27f51539bea26574421215c7780"
+  integrity sha512-NGbfmJBp9x8IxyJSd1P+otYK8vonoJactOogrVfFRIAEY1ukil8RSKDz2Yo7wh1oihl51l/r6W4epkeKJHqL8A==
+  dependencies:
+    global-prefix "^3.0.0"
+
+global-prefix@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/global-prefix/-/global-prefix-3.0.0.tgz#fc85f73064df69f50421f47f883fe5b913ba9b97"
+  integrity sha512-awConJSVCHVGND6x3tmMaKcQvwXLhjdkmomy2W+Goaui8YPgYgXJZewhg3fWC+DlfqqQuWg8AwqjGTD2nAPVWg==
+  dependencies:
+    ini "^1.3.5"
+    kind-of "^6.0.2"
+    which "^1.3.1"
+
+global@~4.4.0:
+  version "4.4.0"
+  resolved "https://registry.yarnpkg.com/global/-/global-4.4.0.tgz#3e7b105179006a323ed71aafca3e9c57a5cc6406"
+  integrity sha512-wv/LAoHdRE3BeTGz53FAamhGlPLhlssK45usmGFThIi4XqnBmjKQ16u+RNbP7WvigRZDxUsM0J3gcQ5yicaL0w==
+  dependencies:
+    min-document "^2.19.0"
+    process "^0.11.10"
+
+globals@^13.19.0:
+  version "13.20.0"
+  resolved "https://registry.yarnpkg.com/globals/-/globals-13.20.0.tgz#ea276a1e508ffd4f1612888f9d1bad1e2717bf82"
+  integrity sha512-Qg5QtVkCy/kv3FUSlu4ukeZDVf9ee0iXLAUYX13gbR17bnejFTzr4iS9bY7kwCf1NztRNm1t91fjOiyx4CSwPQ==
+  dependencies:
+    type-fest "^0.20.2"
+
+globalthis@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/globalthis/-/globalthis-1.0.3.tgz#5852882a52b80dc301b0660273e1ed082f0b6ccf"
+  integrity sha512-sFdI5LyBiNTHjRd7cGPWapiHWMOXKyuBNX/cWJ3NfzrZQVa8GI/8cofCl74AOVqq9W5kNmguTIzJ/1s2gyI9wA==
+  dependencies:
+    define-properties "^1.1.3"
+
+globby@^10.0.1:
+  version "10.0.2"
+  resolved "https://registry.yarnpkg.com/globby/-/globby-10.0.2.tgz#277593e745acaa4646c3ab411289ec47a0392543"
+  integrity sha512-7dUi7RvCoT/xast/o/dLN53oqND4yk0nsHkhRgn9w65C4PofCLOoJ39iSOg+qVDdWQPIEj+eszMHQ+aLVwwQSg==
+  dependencies:
+    "@types/glob" "^7.1.1"
+    array-union "^2.1.0"
+    dir-glob "^3.0.1"
+    fast-glob "^3.0.3"
+    glob "^7.1.3"
+    ignore "^5.1.1"
+    merge2 "^1.2.3"
+    slash "^3.0.0"
+
+globby@^11.1.0:
+  version "11.1.0"
+  resolved "https://registry.yarnpkg.com/globby/-/globby-11.1.0.tgz#bd4be98bb042f83d796f7e3811991fbe82a0d34b"
+  integrity sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==
+  dependencies:
+    array-union "^2.1.0"
+    dir-glob "^3.0.1"
+    fast-glob "^3.2.9"
+    ignore "^5.2.0"
+    merge2 "^1.4.1"
+    slash "^3.0.0"
+
+gopd@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/gopd/-/gopd-1.0.1.tgz#29ff76de69dac7489b7c0918a5788e56477c332c"
+  integrity sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==
+  dependencies:
+    get-intrinsic "^1.1.3"
+
+got@12.1.0:
+  version "12.1.0"
+  resolved "https://registry.yarnpkg.com/got/-/got-12.1.0.tgz#099f3815305c682be4fd6b0ee0726d8e4c6b0af4"
+  integrity sha512-hBv2ty9QN2RdbJJMK3hesmSkFTjVIHyIDDbssCKnSmq62edGgImJWD10Eb1k77TiV1bxloxqcFAVK8+9pkhOig==
+  dependencies:
+    "@sindresorhus/is" "^4.6.0"
+    "@szmarczak/http-timer" "^5.0.1"
+    "@types/cacheable-request" "^6.0.2"
+    "@types/responselike" "^1.0.0"
+    cacheable-lookup "^6.0.4"
+    cacheable-request "^7.0.2"
+    decompress-response "^6.0.0"
+    form-data-encoder "1.7.1"
+    get-stream "^6.0.1"
+    http2-wrapper "^2.1.10"
+    lowercase-keys "^3.0.0"
+    p-cancelable "^3.0.0"
+    responselike "^2.0.0"
+
+got@^11.8.5:
+  version "11.8.6"
+  resolved "https://registry.yarnpkg.com/got/-/got-11.8.6.tgz#276e827ead8772eddbcfc97170590b841823233a"
+  integrity sha512-6tfZ91bOr7bOXnK7PRDCGBLa1H4U080YHNaAQ2KsMGlLEzRbk44nsZF2E1IeRc3vtJHPVbKCYgdFbaGO2ljd8g==
+  dependencies:
+    "@sindresorhus/is" "^4.0.0"
+    "@szmarczak/http-timer" "^4.0.5"
+    "@types/cacheable-request" "^6.0.1"
+    "@types/responselike" "^1.0.0"
+    cacheable-lookup "^5.0.3"
+    cacheable-request "^7.0.2"
+    decompress-response "^6.0.0"
+    http2-wrapper "^1.0.0-beta.5.2"
+    lowercase-keys "^2.0.0"
+    p-cancelable "^2.0.0"
+    responselike "^2.0.0"
+
+graceful-fs@^4.1.2, graceful-fs@^4.1.6, graceful-fs@^4.1.9, graceful-fs@^4.2.0:
+  version "4.2.11"
+  resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.11.tgz#4183e4e8bf08bb6e05bbb2f7d2e0c8f712ca40e3"
+  integrity sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==
+
+graphemer@^1.4.0:
+  version "1.4.0"
+  resolved "https://registry.yarnpkg.com/graphemer/-/graphemer-1.4.0.tgz#fb2f1d55e0e3a1849aeffc90c4fa0dd53a0e66c6"
+  integrity sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==
+
+growl@1.10.5:
+  version "1.10.5"
+  resolved "https://registry.yarnpkg.com/growl/-/growl-1.10.5.tgz#f2735dc2283674fa67478b10181059355c369e5e"
+  integrity sha512-qBr4OuELkhPenW6goKVXiv47US3clb3/IbuWF9KNKEijAy9oeHxU9IgzjvJhHkUzhaj7rOUD7+YGWqUjLp5oSA==
+
+handlebars@^4.0.1:
+  version "4.7.8"
+  resolved "https://registry.yarnpkg.com/handlebars/-/handlebars-4.7.8.tgz#41c42c18b1be2365439188c77c6afae71c0cd9e9"
+  integrity sha512-vafaFqs8MZkRrSX7sFVUdo3ap/eNiLnb4IakshzvP56X5Nr1iGKAIqdX6tMlm6HcNRIkr6AxO5jFEoJzzpT8aQ==
+  dependencies:
+    minimist "^1.2.5"
+    neo-async "^2.6.2"
+    source-map "^0.6.1"
+    wordwrap "^1.0.0"
+  optionalDependencies:
+    uglify-js "^3.1.4"
+
+har-schema@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/har-schema/-/har-schema-2.0.0.tgz#a94c2224ebcac04782a0d9035521f24735b7ec92"
+  integrity sha512-Oqluz6zhGX8cyRaTQlFMPw80bSJVG2x/cFb8ZPhUILGgHka9SsokCCOQgpveePerqidZOrT14ipqfJb7ILcW5Q==
+
+har-validator@~5.1.3:
+  version "5.1.5"
+  resolved "https://registry.yarnpkg.com/har-validator/-/har-validator-5.1.5.tgz#1f0803b9f8cb20c0fa13822df1ecddb36bde1efd"
+  integrity sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==
+  dependencies:
+    ajv "^6.12.3"
+    har-schema "^2.0.0"
+
+hardhat-deploy@^0.11.14:
+  version "0.11.34"
+  resolved "https://registry.yarnpkg.com/hardhat-deploy/-/hardhat-deploy-0.11.34.tgz#61252ebf5dfdda7b0b31298dd5580b0735c05910"
+  integrity sha512-N6xcwD8LSMV/IyfEr8TfR2YRbOh9Q4QvitR9MKZRTXQmgQiiMGjX+2efMjKgNMxwCVlmpfnE1tyDxOJOOUseLQ==
+  dependencies:
+    "@ethersproject/abi" "^5.7.0"
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/contracts" "^5.7.0"
+    "@ethersproject/providers" "^5.7.2"
+    "@ethersproject/solidity" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+    "@ethersproject/wallet" "^5.7.0"
+    "@types/qs" "^6.9.7"
+    axios "^0.21.1"
+    chalk "^4.1.2"
+    chokidar "^3.5.2"
+    debug "^4.3.2"
+    enquirer "^2.3.6"
+    ethers "^5.5.3"
+    form-data "^4.0.0"
+    fs-extra "^10.0.0"
+    match-all "^1.2.6"
+    murmur-128 "^0.2.1"
+    qs "^6.9.4"
+    zksync-web3 "^0.14.3"
+
+hardhat-gas-reporter@^1.0.9:
+  version "1.0.9"
+  resolved "https://registry.yarnpkg.com/hardhat-gas-reporter/-/hardhat-gas-reporter-1.0.9.tgz#9a2afb354bc3b6346aab55b1c02ca556d0e16450"
+  integrity sha512-INN26G3EW43adGKBNzYWOlI3+rlLnasXTwW79YNnUhXPDa+yHESgt639dJEs37gCjhkbNKcRRJnomXEuMFBXJg==
+  dependencies:
+    array-uniq "1.0.3"
+    eth-gas-reporter "^0.2.25"
+    sha1 "^1.1.1"
+
+hardhat@2.12.2:
+  version "2.12.2"
+  resolved "https://registry.yarnpkg.com/hardhat/-/hardhat-2.12.2.tgz#6ae985007b20c1f381c6573799d66c1438c4c802"
+  integrity sha512-f3ZhzXy1uyQv0UXnAQ8GCBOWjzv++WJNb7bnm10SsyC3dB7vlPpsMWBNhq7aoRxKrNhX9tCev81KFV3i5BTeMQ==
+  dependencies:
+    "@ethersproject/abi" "^5.1.2"
+    "@metamask/eth-sig-util" "^4.0.0"
+    "@nomicfoundation/ethereumjs-block" "^4.0.0"
+    "@nomicfoundation/ethereumjs-blockchain" "^6.0.0"
+    "@nomicfoundation/ethereumjs-common" "^3.0.0"
+    "@nomicfoundation/ethereumjs-evm" "^1.0.0"
+    "@nomicfoundation/ethereumjs-rlp" "^4.0.0"
+    "@nomicfoundation/ethereumjs-statemanager" "^1.0.0"
+    "@nomicfoundation/ethereumjs-trie" "^5.0.0"
+    "@nomicfoundation/ethereumjs-tx" "^4.0.0"
+    "@nomicfoundation/ethereumjs-util" "^8.0.0"
+    "@nomicfoundation/ethereumjs-vm" "^6.0.0"
+    "@nomicfoundation/solidity-analyzer" "^0.1.0"
+    "@sentry/node" "^5.18.1"
+    "@types/bn.js" "^5.1.0"
+    "@types/lru-cache" "^5.1.0"
+    abort-controller "^3.0.0"
+    adm-zip "^0.4.16"
+    aggregate-error "^3.0.0"
+    ansi-escapes "^4.3.0"
+    chalk "^2.4.2"
+    chokidar "^3.4.0"
+    ci-info "^2.0.0"
+    debug "^4.1.1"
+    enquirer "^2.3.0"
+    env-paths "^2.2.0"
+    ethereum-cryptography "^1.0.3"
+    ethereumjs-abi "^0.6.8"
+    find-up "^2.1.0"
+    fp-ts "1.19.3"
+    fs-extra "^7.0.1"
+    glob "7.2.0"
+    immutable "^4.0.0-rc.12"
+    io-ts "1.10.4"
+    keccak "^3.0.2"
+    lodash "^4.17.11"
+    mnemonist "^0.38.0"
+    mocha "^10.0.0"
+    p-map "^4.0.0"
+    qs "^6.7.0"
+    raw-body "^2.4.1"
+    resolve "1.17.0"
+    semver "^6.3.0"
+    solc "0.7.3"
+    source-map-support "^0.5.13"
+    stacktrace-parser "^0.1.10"
+    tsort "0.0.1"
+    undici "^5.4.0"
+    uuid "^8.3.2"
+    ws "^7.4.6"
+
+has-bigints@^1.0.1, has-bigints@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/has-bigints/-/has-bigints-1.0.2.tgz#0871bd3e3d51626f6ca0966668ba35d5602d6eaa"
+  integrity sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==
+
+has-flag@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-1.0.0.tgz#9d9e793165ce017a00f00418c43f942a7b1d11fa"
+  integrity sha512-DyYHfIYwAJmjAjSSPKANxI8bFY9YtFrgkAfinBojQ8YJTOuOuav64tMUJv584SES4xl74PmuaevIyaLESHdTAA==
+
+has-flag@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz#b5d454dc2199ae225699f3467e5a07f3b955bafd"
+  integrity sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==
+
+has-flag@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b"
+  integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==
+
+has-property-descriptors@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/has-property-descriptors/-/has-property-descriptors-1.0.0.tgz#610708600606d36961ed04c196193b6a607fa861"
+  integrity sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ==
+  dependencies:
+    get-intrinsic "^1.1.1"
+
+has-proto@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/has-proto/-/has-proto-1.0.1.tgz#1885c1305538958aff469fef37937c22795408e0"
+  integrity sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg==
+
+has-symbols@^1.0.0, has-symbols@^1.0.2, has-symbols@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.3.tgz#bb7b2c4349251dce87b125f7bdf874aa7c8b39f8"
+  integrity sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==
+
+has-tostringtag@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/has-tostringtag/-/has-tostringtag-1.0.0.tgz#7e133818a7d394734f941e73c3d3f9291e658b25"
+  integrity sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==
+  dependencies:
+    has-symbols "^1.0.2"
+
+has@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/has/-/has-1.0.3.tgz#722d7cbfc1f6aa8241f16dd814e011e1f41e8796"
+  integrity sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==
+  dependencies:
+    function-bind "^1.1.1"
+
+hash-base@^3.0.0:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/hash-base/-/hash-base-3.1.0.tgz#55c381d9e06e1d2997a883b4a3fddfe7f0d3af33"
+  integrity sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA==
+  dependencies:
+    inherits "^2.0.4"
+    readable-stream "^3.6.0"
+    safe-buffer "^5.2.0"
+
+hash.js@1.1.3:
+  version "1.1.3"
+  resolved "https://registry.yarnpkg.com/hash.js/-/hash.js-1.1.3.tgz#340dedbe6290187151c1ea1d777a3448935df846"
+  integrity sha512-/UETyP0W22QILqS+6HowevwhEFJ3MBJnwTf75Qob9Wz9t0DPuisL8kW8YZMK62dHAKE1c1p+gY1TtOLY+USEHA==
+  dependencies:
+    inherits "^2.0.3"
+    minimalistic-assert "^1.0.0"
+
+hash.js@1.1.7, hash.js@^1.0.0, hash.js@^1.0.3, hash.js@^1.1.7:
+  version "1.1.7"
+  resolved "https://registry.yarnpkg.com/hash.js/-/hash.js-1.1.7.tgz#0babca538e8d4ee4a0f8988d68866537a003cf42"
+  integrity sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==
+  dependencies:
+    inherits "^2.0.3"
+    minimalistic-assert "^1.0.1"
+
+he@1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/he/-/he-1.2.0.tgz#84ae65fa7eafb165fddb61566ae14baf05664f0f"
+  integrity sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==
+
+header-case@^1.0.0:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/header-case/-/header-case-1.0.1.tgz#9535973197c144b09613cd65d317ef19963bd02d"
+  integrity sha512-i0q9mkOeSuhXw6bGgiQCCBgY/jlZuV/7dZXyZ9c6LcBrqwvT8eT719E9uxE5LiZftdl+z81Ugbg/VvXV4OJOeQ==
+  dependencies:
+    no-case "^2.2.0"
+    upper-case "^1.1.3"
+
+"heap@>= 0.2.0":
+  version "0.2.7"
+  resolved "https://registry.yarnpkg.com/heap/-/heap-0.2.7.tgz#1e6adf711d3f27ce35a81fe3b7bd576c2260a8fc"
+  integrity sha512-2bsegYkkHO+h/9MGbn6KWcE45cHZgPANo5LXF7EvWdT0yT2EguSVO1nDgU5c8+ZOPwp2vMNa7YFsJhVcDR9Sdg==
+
+highlight.js@^10.4.1:
+  version "10.7.3"
+  resolved "https://registry.yarnpkg.com/highlight.js/-/highlight.js-10.7.3.tgz#697272e3991356e40c3cac566a74eef681756531"
+  integrity sha512-tzcUFauisWKNHaRkN4Wjl/ZA07gENAjFl3J/c480dprkGTg5EQstgaNFqBfUqCq54kZRIEcreTsAgF/m2quD7A==
+
+highlightjs-solidity@^2.0.6:
+  version "2.0.6"
+  resolved "https://registry.yarnpkg.com/highlightjs-solidity/-/highlightjs-solidity-2.0.6.tgz#e7a702a2b05e0a97f185e6ba39fd4846ad23a990"
+  integrity sha512-DySXWfQghjm2l6a/flF+cteroJqD4gI8GSdL4PtvxZSsAHie8m3yVe2JFoRg03ROKT6hp2Lc/BxXkqerNmtQYg==
+
+hmac-drbg@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/hmac-drbg/-/hmac-drbg-1.0.1.tgz#d2745701025a6c775a6c545793ed502fc0c649a1"
+  integrity sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==
+  dependencies:
+    hash.js "^1.0.3"
+    minimalistic-assert "^1.0.0"
+    minimalistic-crypto-utils "^1.0.1"
+
+hosted-git-info@^2.1.4:
+  version "2.8.9"
+  resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-2.8.9.tgz#dffc0bf9a21c02209090f2aa69429e1414daf3f9"
+  integrity sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==
+
+htmlparser2@^8.0.1:
+  version "8.0.2"
+  resolved "https://registry.yarnpkg.com/htmlparser2/-/htmlparser2-8.0.2.tgz#f002151705b383e62433b5cf466f5b716edaec21"
+  integrity sha512-GYdjWKDkbRLkZ5geuHs5NY1puJ+PXwP7+fHPRz06Eirsb9ugf6d8kkXav6ADhcODhFFPMIXyxkxSuMf3D6NCFA==
+  dependencies:
+    domelementtype "^2.3.0"
+    domhandler "^5.0.3"
+    domutils "^3.0.1"
+    entities "^4.4.0"
+
+http-basic@^8.1.1:
+  version "8.1.3"
+  resolved "https://registry.yarnpkg.com/http-basic/-/http-basic-8.1.3.tgz#a7cabee7526869b9b710136970805b1004261bbf"
+  integrity sha512-/EcDMwJZh3mABI2NhGfHOGOeOZITqfkEO4p/xK+l3NpyncIHUQBoMvCSF/b5GqvKtySC2srL/GGG3+EtlqlmCw==
+  dependencies:
+    caseless "^0.12.0"
+    concat-stream "^1.6.2"
+    http-response-object "^3.0.1"
+    parse-cache-control "^1.0.1"
+
+http-cache-semantics@^4.0.0:
+  version "4.1.1"
+  resolved "https://registry.yarnpkg.com/http-cache-semantics/-/http-cache-semantics-4.1.1.tgz#abe02fcb2985460bf0323be664436ec3476a6d5a"
+  integrity sha512-er295DKPVsV82j5kw1Gjt+ADA/XYHsajl82cGNQG2eyoPkvgUhX+nDIyelzhIWbbsXP39EHcI6l5tYs2FYqYXQ==
+
+http-errors@2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-2.0.0.tgz#b7774a1486ef73cf7667ac9ae0858c012c57b9d3"
+  integrity sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==
+  dependencies:
+    depd "2.0.0"
+    inherits "2.0.4"
+    setprototypeof "1.2.0"
+    statuses "2.0.1"
+    toidentifier "1.0.1"
+
+http-https@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/http-https/-/http-https-1.0.0.tgz#2f908dd5f1db4068c058cd6e6d4ce392c913389b"
+  integrity sha512-o0PWwVCSp3O0wS6FvNr6xfBCHgt0m1tvPLFOCc2iFDKTRAXhB7m8klDf7ErowFH8POa6dVdGatKU5I1YYwzUyg==
+
+http-response-object@^3.0.1:
+  version "3.0.2"
+  resolved "https://registry.yarnpkg.com/http-response-object/-/http-response-object-3.0.2.tgz#7f435bb210454e4360d074ef1f989d5ea8aa9810"
+  integrity sha512-bqX0XTF6fnXSQcEJ2Iuyr75yVakyjIDCqroJQ/aHfSdlM743Cwqoi2nDYMzLGWUcuTWGWy8AAvOKXTfiv6q9RA==
+  dependencies:
+    "@types/node" "^10.0.3"
+
+http-signature@~1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/http-signature/-/http-signature-1.2.0.tgz#9aecd925114772f3d95b65a60abb8f7c18fbace1"
+  integrity sha512-CAbnr6Rz4CYQkLYUtSNXxQPUH2gK8f3iWexVlsnMeD+GjlsQ0Xsy1cOX+mN3dtxYomRy21CiOzU8Uhw6OwncEQ==
+  dependencies:
+    assert-plus "^1.0.0"
+    jsprim "^1.2.2"
+    sshpk "^1.7.0"
+
+http2-wrapper@^1.0.0-beta.5.2:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/http2-wrapper/-/http2-wrapper-1.0.3.tgz#b8f55e0c1f25d4ebd08b3b0c2c079f9590800b3d"
+  integrity sha512-V+23sDMr12Wnz7iTcDeJr3O6AIxlnvT/bmaAAAP/Xda35C90p9599p0F1eHR/N1KILWSoWVAiOMFjBBXaXSMxg==
+  dependencies:
+    quick-lru "^5.1.1"
+    resolve-alpn "^1.0.0"
+
+http2-wrapper@^2.1.10:
+  version "2.2.0"
+  resolved "https://registry.yarnpkg.com/http2-wrapper/-/http2-wrapper-2.2.0.tgz#b80ad199d216b7d3680195077bd7b9060fa9d7f3"
+  integrity sha512-kZB0wxMo0sh1PehyjJUWRFEd99KC5TLjZ2cULC4f9iqJBAmKQQXEICjxl5iPJRwP40dpeHFqqhm7tYCvODpqpQ==
+  dependencies:
+    quick-lru "^5.1.1"
+    resolve-alpn "^1.2.0"
+
+https-proxy-agent@^5.0.0:
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz#c59ef224a04fe8b754f3db0063a25ea30d0005d6"
+  integrity sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==
+  dependencies:
+    agent-base "6"
+    debug "4"
+
+husky@^4.2.3:
+  version "4.3.8"
+  resolved "https://registry.yarnpkg.com/husky/-/husky-4.3.8.tgz#31144060be963fd6850e5cc8f019a1dfe194296d"
+  integrity sha512-LCqqsB0PzJQ/AlCgfrfzRe3e3+NvmefAdKQhRYpxS4u6clblBoDdzzvHi8fmxKRzvMxPY/1WZWzomPZww0Anow==
+  dependencies:
+    chalk "^4.0.0"
+    ci-info "^2.0.0"
+    compare-versions "^3.6.0"
+    cosmiconfig "^7.0.0"
+    find-versions "^4.0.0"
+    opencollective-postinstall "^2.0.2"
+    pkg-dir "^5.0.0"
+    please-upgrade-node "^3.2.0"
+    slash "^3.0.0"
+    which-pm-runs "^1.0.0"
+
+hyperlinker@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/hyperlinker/-/hyperlinker-1.0.0.tgz#23dc9e38a206b208ee49bc2d6c8ef47027df0c0e"
+  integrity sha512-Ty8UblRWFEcfSuIaajM34LdPXIhbs1ajEX/BBPv24J+enSVaEVY63xQ6lTO9VRYS5LAoghIG0IDJ+p+IPzKUQQ==
+
+iconv-lite@0.4.24:
+  version "0.4.24"
+  resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b"
+  integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==
+  dependencies:
+    safer-buffer ">= 2.1.2 < 3"
+
+idna-uts46-hx@^2.3.1:
+  version "2.3.1"
+  resolved "https://registry.yarnpkg.com/idna-uts46-hx/-/idna-uts46-hx-2.3.1.tgz#a1dc5c4df37eee522bf66d969cc980e00e8711f9"
+  integrity sha512-PWoF9Keq6laYdIRwwCdhTPl60xRqAloYNMQLiyUnG42VjT53oW07BXIRM+NK7eQjzXjAk2gUvX9caRxlnF9TAA==
+  dependencies:
+    punycode "2.1.0"
+
+ieee754@^1.1.13, ieee754@^1.2.1:
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352"
+  integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==
+
+ignore@^5.1.1, ignore@^5.2.0, ignore@^5.2.4:
+  version "5.2.4"
+  resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.2.4.tgz#a291c0c6178ff1b960befe47fcdec301674a6324"
+  integrity sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ==
+
+immutable@^4.0.0-rc.12:
+  version "4.3.2"
+  resolved "https://registry.yarnpkg.com/immutable/-/immutable-4.3.2.tgz#f89d910f8dfb6e15c03b2cae2faaf8c1f66455fe"
+  integrity sha512-oGXzbEDem9OOpDWZu88jGiYCvIsLHMvGw+8OXlpsvTFvIQplQbjg1B1cvKg8f7Hoch6+NGjpPsH1Fr+Mc2D1aA==
+
+import-fresh@^3.2.1:
+  version "3.3.0"
+  resolved "https://registry.yarnpkg.com/import-fresh/-/import-fresh-3.3.0.tgz#37162c25fcb9ebaa2e6e53d5b4d88ce17d9e0c2b"
+  integrity sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==
+  dependencies:
+    parent-module "^1.0.0"
+    resolve-from "^4.0.0"
+
+imul@^1.0.0:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/imul/-/imul-1.0.1.tgz#9d5867161e8b3de96c2c38d5dc7cb102f35e2ac9"
+  integrity sha512-WFAgfwPLAjU66EKt6vRdTlKj4nAgIDQzh29JonLa4Bqtl6D8JrIMvWjCnx7xEjVNmP3U0fM5o8ZObk7d0f62bA==
+
+imurmurhash@^0.1.4:
+  version "0.1.4"
+  resolved "https://registry.yarnpkg.com/imurmurhash/-/imurmurhash-0.1.4.tgz#9218b9b2b928a238b13dc4fb6b6d576f231453ea"
+  integrity sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==
+
+indent-string@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/indent-string/-/indent-string-4.0.0.tgz#624f8f4497d619b2d9768531d58f4122854d7251"
+  integrity sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==
+
+inflight@^1.0.4:
+  version "1.0.6"
+  resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9"
+  integrity sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==
+  dependencies:
+    once "^1.3.0"
+    wrappy "1"
+
+inherits@2, inherits@2.0.4, inherits@^2.0.1, inherits@^2.0.3, inherits@^2.0.4, inherits@~2.0.3:
+  version "2.0.4"
+  resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c"
+  integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==
+
+ini@^1.3.5:
+  version "1.3.8"
+  resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.8.tgz#a29da425b48806f34767a4efce397269af28432c"
+  integrity sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==
+
+internal-slot@^1.0.5:
+  version "1.0.5"
+  resolved "https://registry.yarnpkg.com/internal-slot/-/internal-slot-1.0.5.tgz#f2a2ee21f668f8627a4667f309dc0f4fb6674986"
+  integrity sha512-Y+R5hJrzs52QCG2laLn4udYVnxsfny9CpOhNhUvk/SSSVyF6T27FzRbF0sroPidSu3X8oEAkOn2K804mjpt6UQ==
+  dependencies:
+    get-intrinsic "^1.2.0"
+    has "^1.0.3"
+    side-channel "^1.0.4"
+
+interpret@^1.0.0:
+  version "1.4.0"
+  resolved "https://registry.yarnpkg.com/interpret/-/interpret-1.4.0.tgz#665ab8bc4da27a774a40584e812e3e0fa45b1a1e"
+  integrity sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA==
+
+invert-kv@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/invert-kv/-/invert-kv-1.0.0.tgz#104a8e4aaca6d3d8cd157a8ef8bfab2d7a3ffdb6"
+  integrity sha512-xgs2NH9AE66ucSq4cNG1nhSFghr5l6tdL15Pk+jl46bmmBapgoaY/AacXyaDznAqmGL99TiLSQgO/XazFSKYeQ==
+
+io-ts@1.10.4:
+  version "1.10.4"
+  resolved "https://registry.yarnpkg.com/io-ts/-/io-ts-1.10.4.tgz#cd5401b138de88e4f920adbcb7026e2d1967e6e2"
+  integrity sha512-b23PteSnYXSONJ6JQXRAlvJhuw8KOtkqa87W4wDtvMrud/DTJd5X+NpOOI+O/zZwVq6v0VLAaJ+1EDViKEuN9g==
+  dependencies:
+    fp-ts "^1.0.0"
+
+ipaddr.js@1.9.1:
+  version "1.9.1"
+  resolved "https://registry.yarnpkg.com/ipaddr.js/-/ipaddr.js-1.9.1.tgz#bff38543eeb8984825079ff3a2a8e6cbd46781b3"
+  integrity sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==
+
+is-arguments@^1.0.4:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/is-arguments/-/is-arguments-1.1.1.tgz#15b3f88fda01f2a97fec84ca761a560f123efa9b"
+  integrity sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==
+  dependencies:
+    call-bind "^1.0.2"
+    has-tostringtag "^1.0.0"
+
+is-array-buffer@^3.0.1, is-array-buffer@^3.0.2:
+  version "3.0.2"
+  resolved "https://registry.yarnpkg.com/is-array-buffer/-/is-array-buffer-3.0.2.tgz#f2653ced8412081638ecb0ebbd0c41c6e0aecbbe"
+  integrity sha512-y+FyyR/w8vfIRq4eQcM1EYgSTnmHXPqaF+IgzgraytCFq5Xh8lllDVmAZolPJiZttZLeFSINPYMaEJ7/vWUa1w==
+  dependencies:
+    call-bind "^1.0.2"
+    get-intrinsic "^1.2.0"
+    is-typed-array "^1.1.10"
+
+is-arrayish@^0.2.1:
+  version "0.2.1"
+  resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.2.1.tgz#77c99840527aa8ecb1a8ba697b80645a7a926a9d"
+  integrity sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==
+
+is-bigint@^1.0.1:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/is-bigint/-/is-bigint-1.0.4.tgz#08147a1875bc2b32005d41ccd8291dffc6691df3"
+  integrity sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==
+  dependencies:
+    has-bigints "^1.0.1"
+
+is-binary-path@~2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/is-binary-path/-/is-binary-path-2.1.0.tgz#ea1f7f3b80f064236e83470f86c09c254fb45b09"
+  integrity sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==
+  dependencies:
+    binary-extensions "^2.0.0"
+
+is-boolean-object@^1.1.0:
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/is-boolean-object/-/is-boolean-object-1.1.2.tgz#5c6dc200246dd9321ae4b885a114bb1f75f63719"
+  integrity sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==
+  dependencies:
+    call-bind "^1.0.2"
+    has-tostringtag "^1.0.0"
+
+is-buffer@^2.0.5, is-buffer@~2.0.3:
+  version "2.0.5"
+  resolved "https://registry.yarnpkg.com/is-buffer/-/is-buffer-2.0.5.tgz#ebc252e400d22ff8d77fa09888821a24a658c191"
+  integrity sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ==
+
+is-callable@^1.1.3, is-callable@^1.1.4, is-callable@^1.2.7:
+  version "1.2.7"
+  resolved "https://registry.yarnpkg.com/is-callable/-/is-callable-1.2.7.tgz#3bc2a85ea742d9e36205dcacdd72ca1fdc51b055"
+  integrity sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==
+
+is-core-module@^2.12.1, is-core-module@^2.13.0:
+  version "2.13.0"
+  resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.13.0.tgz#bb52aa6e2cbd49a30c2ba68c42bf3435ba6072db"
+  integrity sha512-Z7dk6Qo8pOCp3l4tsX2C5ZVas4V+UxwQodwZhLopL91TX8UyyHEXafPcyoeeWuLrwzHcr3igO78wNLwHJHsMCQ==
+  dependencies:
+    has "^1.0.3"
+
+is-date-object@^1.0.1:
+  version "1.0.5"
+  resolved "https://registry.yarnpkg.com/is-date-object/-/is-date-object-1.0.5.tgz#0841d5536e724c25597bf6ea62e1bd38298df31f"
+  integrity sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==
+  dependencies:
+    has-tostringtag "^1.0.0"
+
+is-docker@^2.0.0, is-docker@^2.1.1:
+  version "2.2.1"
+  resolved "https://registry.yarnpkg.com/is-docker/-/is-docker-2.2.1.tgz#33eeabe23cfe86f14bde4408a02c0cfb853acdaa"
+  integrity sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==
+
+is-extglob@^2.1.1:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-2.1.1.tgz#a88c02535791f02ed37c76a1b9ea9773c833f8c2"
+  integrity sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==
+
+is-fullwidth-code-point@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz#ef9e31386f031a7f0d643af82fde50c457ef00cb"
+  integrity sha512-1pqUqRjkhPJ9miNq9SwMfdvi6lBJcd6eFxvfaivQhaH3SgisfiuudvFntdKOmxuee/77l+FPjKrQjWvmPjWrRw==
+  dependencies:
+    number-is-nan "^1.0.0"
+
+is-fullwidth-code-point@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz#a3b30a5c4f199183167aaab93beefae3ddfb654f"
+  integrity sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==
+
+is-fullwidth-code-point@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz#f116f8064fe90b3f7844a38997c0b75051269f1d"
+  integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==
+
+is-function@^1.0.1:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/is-function/-/is-function-1.0.2.tgz#4f097f30abf6efadac9833b17ca5dc03f8144e08"
+  integrity sha512-lw7DUp0aWXYg+CBCN+JKkcE0Q2RayZnSvnZBlwgxHBQhqt5pZNVy4Ri7H9GmmXkdu7LUthszM+Tor1u/2iBcpQ==
+
+is-generator-function@^1.0.7:
+  version "1.0.10"
+  resolved "https://registry.yarnpkg.com/is-generator-function/-/is-generator-function-1.0.10.tgz#f1558baf1ac17e0deea7c0415c438351ff2b3c72"
+  integrity sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==
+  dependencies:
+    has-tostringtag "^1.0.0"
+
+is-glob@^4.0.0, is-glob@^4.0.1, is-glob@^4.0.3, is-glob@~4.0.1:
+  version "4.0.3"
+  resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-4.0.3.tgz#64f61e42cbbb2eec2071a9dac0b28ba1e65d5084"
+  integrity sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==
+  dependencies:
+    is-extglob "^2.1.1"
+
+is-hex-prefixed@1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/is-hex-prefixed/-/is-hex-prefixed-1.0.0.tgz#7d8d37e6ad77e5d127148913c573e082d777f554"
+  integrity sha512-WvtOiug1VFrE9v1Cydwm+FnXd3+w9GaeVUss5W4v/SLy3UW00vP+6iNF2SdnfiBoLy4bTqVdkftNGTUeOFVsbA==
+
+is-interactive@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/is-interactive/-/is-interactive-1.0.0.tgz#cea6e6ae5c870a7b0a0004070b7b587e0252912e"
+  integrity sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==
+
+is-lower-case@^1.1.0:
+  version "1.1.3"
+  resolved "https://registry.yarnpkg.com/is-lower-case/-/is-lower-case-1.1.3.tgz#7e147be4768dc466db3bfb21cc60b31e6ad69393"
+  integrity sha512-+5A1e/WJpLLXZEDlgz4G//WYSHyQBD32qa4Jd3Lw06qQlv3fJHnp3YIHjTQSGzHMgzmVKz2ZP3rBxTHkPw/lxA==
+  dependencies:
+    lower-case "^1.1.0"
+
+is-negative-zero@^2.0.2:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/is-negative-zero/-/is-negative-zero-2.0.2.tgz#7bf6f03a28003b8b3965de3ac26f664d765f3150"
+  integrity sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==
+
+is-number-object@^1.0.4:
+  version "1.0.7"
+  resolved "https://registry.yarnpkg.com/is-number-object/-/is-number-object-1.0.7.tgz#59d50ada4c45251784e9904f5246c742f07a42fc"
+  integrity sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==
+  dependencies:
+    has-tostringtag "^1.0.0"
+
+is-number@^7.0.0:
+  version "7.0.0"
+  resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b"
+  integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==
+
+is-path-inside@^3.0.3:
+  version "3.0.3"
+  resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-3.0.3.tgz#d231362e53a07ff2b0e0ea7fed049161ffd16283"
+  integrity sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==
+
+is-plain-obj@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/is-plain-obj/-/is-plain-obj-2.1.0.tgz#45e42e37fccf1f40da8e5f76ee21515840c09287"
+  integrity sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==
+
+is-regex@^1.1.4:
+  version "1.1.4"
+  resolved "https://registry.yarnpkg.com/is-regex/-/is-regex-1.1.4.tgz#eef5663cd59fa4c0ae339505323df6854bb15958"
+  integrity sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==
+  dependencies:
+    call-bind "^1.0.2"
+    has-tostringtag "^1.0.0"
+
+is-shared-array-buffer@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz#8f259c573b60b6a32d4058a1a07430c0a7344c79"
+  integrity sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==
+  dependencies:
+    call-bind "^1.0.2"
+
+is-string@^1.0.5, is-string@^1.0.7:
+  version "1.0.7"
+  resolved "https://registry.yarnpkg.com/is-string/-/is-string-1.0.7.tgz#0dd12bf2006f255bb58f695110eff7491eebc0fd"
+  integrity sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==
+  dependencies:
+    has-tostringtag "^1.0.0"
+
+is-symbol@^1.0.2, is-symbol@^1.0.3:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/is-symbol/-/is-symbol-1.0.4.tgz#a6dac93b635b063ca6872236de88910a57af139c"
+  integrity sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==
+  dependencies:
+    has-symbols "^1.0.2"
+
+is-typed-array@^1.1.10, is-typed-array@^1.1.3, is-typed-array@^1.1.9:
+  version "1.1.12"
+  resolved "https://registry.yarnpkg.com/is-typed-array/-/is-typed-array-1.1.12.tgz#d0bab5686ef4a76f7a73097b95470ab199c57d4a"
+  integrity sha512-Z14TF2JNG8Lss5/HMqt0//T9JeHXttXy5pH/DBU4vi98ozO2btxzq9MwYDZYnKwU8nRsz/+GVFVRDq3DkVuSPg==
+  dependencies:
+    which-typed-array "^1.1.11"
+
+is-typedarray@^1.0.0, is-typedarray@~1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/is-typedarray/-/is-typedarray-1.0.0.tgz#e479c80858df0c1b11ddda6940f96011fcda4a9a"
+  integrity sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA==
+
+is-unicode-supported@^0.1.0:
+  version "0.1.0"
+  resolved "https://registry.yarnpkg.com/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz#3f26c76a809593b52bfa2ecb5710ed2779b522a7"
+  integrity sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==
+
+is-upper-case@^1.1.0:
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/is-upper-case/-/is-upper-case-1.1.2.tgz#8d0b1fa7e7933a1e58483600ec7d9661cbaf756f"
+  integrity sha512-GQYSJMgfeAmVwh9ixyk888l7OIhNAGKtY6QA+IrWlu9MDTCaXmeozOZ2S9Knj7bQwBO/H6J2kb+pbyTUiMNbsw==
+  dependencies:
+    upper-case "^1.1.0"
+
+is-utf8@^0.2.0:
+  version "0.2.1"
+  resolved "https://registry.yarnpkg.com/is-utf8/-/is-utf8-0.2.1.tgz#4b0da1442104d1b336340e80797e865cf39f7d72"
+  integrity sha512-rMYPYvCzsXywIsldgLaSoPlw5PfoB/ssr7hY4pLfcodrA5M/eArza1a9VmTiNIBNMjOGr1Ow9mTyU2o69U6U9Q==
+
+is-weakref@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/is-weakref/-/is-weakref-1.0.2.tgz#9529f383a9338205e89765e0392efc2f100f06f2"
+  integrity sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==
+  dependencies:
+    call-bind "^1.0.2"
+
+is-wsl@^2.2.0:
+  version "2.2.0"
+  resolved "https://registry.yarnpkg.com/is-wsl/-/is-wsl-2.2.0.tgz#74a4c76e77ca9fd3f932f290c17ea326cd157271"
+  integrity sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==
+  dependencies:
+    is-docker "^2.0.0"
+
+isarray@^2.0.5:
+  version "2.0.5"
+  resolved "https://registry.yarnpkg.com/isarray/-/isarray-2.0.5.tgz#8af1e4c1221244cc62459faf38940d4e644a5723"
+  integrity sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==
+
+isarray@~1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11"
+  integrity sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==
+
+isexe@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10"
+  integrity sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==
+
+isstream@~0.1.2:
+  version "0.1.2"
+  resolved "https://registry.yarnpkg.com/isstream/-/isstream-0.1.2.tgz#47e63f7af55afa6f92e1500e690eb8b8529c099a"
+  integrity sha512-Yljz7ffyPbrLpLngrMtZ7NduUgVvi6wG9RJ9IUcyCd59YQ911PBJphODUcbOVbqYfxe1wuYf/LJ8PauMRwsM/g==
+
+js-base64@^3.6.0:
+  version "3.7.5"
+  resolved "https://registry.yarnpkg.com/js-base64/-/js-base64-3.7.5.tgz#21e24cf6b886f76d6f5f165bfcd69cc55b9e3fca"
+  integrity sha512-3MEt5DTINKqfScXKfJFrRbxkrnk2AxPWGBL/ycjz4dK8iqiSJ06UxD8jh8xuh6p10TX4t2+7FsBYVxxQbMg+qA==
+
+js-sha3@0.5.7, js-sha3@^0.5.7:
+  version "0.5.7"
+  resolved "https://registry.yarnpkg.com/js-sha3/-/js-sha3-0.5.7.tgz#0d4ffd8002d5333aabaf4a23eed2f6374c9f28e7"
+  integrity sha512-GII20kjaPX0zJ8wzkTbNDYMY7msuZcTWk8S5UOh6806Jq/wz1J8/bnr8uGU0DAUmYDjj2Mr4X1cW8v/GLYnR+g==
+
+js-sha3@0.8.0, js-sha3@^0.8.0:
+  version "0.8.0"
+  resolved "https://registry.yarnpkg.com/js-sha3/-/js-sha3-0.8.0.tgz#b9b7a5da73afad7dedd0f8c463954cbde6818840"
+  integrity sha512-gF1cRrHhIzNfToc802P800N8PpXS+evLLXfsVpowqmAFR9uwbi89WvXg2QspOmXL8QL86J4T1EpFu+yUkwJY3Q==
+
+js-tokens@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-4.0.0.tgz#19203fb59991df98e3a287050d4647cdeaf32499"
+  integrity sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==
+
+js-yaml@3.13.1:
+  version "3.13.1"
+  resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.13.1.tgz#aff151b30bfdfa8e49e05da22e7415e9dfa37847"
+  integrity sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==
+  dependencies:
+    argparse "^1.0.7"
+    esprima "^4.0.0"
+
+js-yaml@3.x:
+  version "3.14.1"
+  resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.14.1.tgz#dae812fdb3825fa306609a8717383c50c36a0537"
+  integrity sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==
+  dependencies:
+    argparse "^1.0.7"
+    esprima "^4.0.0"
+
+js-yaml@4.1.0, js-yaml@^4.1.0:
+  version "4.1.0"
+  resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-4.1.0.tgz#c1fb65f8f5017901cdd2c951864ba18458a10602"
+  integrity sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==
+  dependencies:
+    argparse "^2.0.1"
+
+jsbn@~0.1.0:
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/jsbn/-/jsbn-0.1.1.tgz#a5e654c2e5a2deb5f201d96cefbca80c0ef2f513"
+  integrity sha512-UVU9dibq2JcFWxQPA6KCqj5O42VOmAY3zQUfEKxU0KpTGXwNoCjkX1e13eHNvw/xPynt6pU0rZ1htjWTNTSXsg==
+
+json-buffer@3.0.1:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/json-buffer/-/json-buffer-3.0.1.tgz#9338802a30d3b6605fbe0613e094008ca8c05a13"
+  integrity sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==
+
+json-parse-even-better-errors@^2.3.0:
+  version "2.3.1"
+  resolved "https://registry.yarnpkg.com/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz#7c47805a94319928e05777405dc12e1f7a4ee02d"
+  integrity sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==
+
+json-schema-traverse@^0.4.1:
+  version "0.4.1"
+  resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660"
+  integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==
+
+json-schema-traverse@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz#ae7bcb3656ab77a73ba5c49bf654f38e6b6860e2"
+  integrity sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==
+
+json-schema@0.4.0:
+  version "0.4.0"
+  resolved "https://registry.yarnpkg.com/json-schema/-/json-schema-0.4.0.tgz#f7de4cf6efab838ebaeb3236474cbba5a1930ab5"
+  integrity sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA==
+
+json-stable-stringify-without-jsonify@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz#9db7b59496ad3f3cfef30a75142d2d930ad72651"
+  integrity sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==
+
+json-stringify-safe@~5.0.1:
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb"
+  integrity sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==
+
+json5@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/json5/-/json5-1.0.2.tgz#63d98d60f21b313b77c4d6da18bfa69d80e1d593"
+  integrity sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==
+  dependencies:
+    minimist "^1.2.0"
+
+jsonfile@^2.1.0:
+  version "2.4.0"
+  resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-2.4.0.tgz#3736a2b428b87bbda0cc83b53fa3d633a35c2ae8"
+  integrity sha512-PKllAqbgLgxHaj8TElYymKCAgrASebJrWpTnEkOaTowt23VKXXN0sUeriJ+eh7y6ufb/CC5ap11pz71/cM0hUw==
+  optionalDependencies:
+    graceful-fs "^4.1.6"
+
+jsonfile@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-4.0.0.tgz#8771aae0799b64076b76640fca058f9c10e33ecb"
+  integrity sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg==
+  optionalDependencies:
+    graceful-fs "^4.1.6"
+
+jsonfile@^6.0.1:
+  version "6.1.0"
+  resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-6.1.0.tgz#bc55b2634793c679ec6403094eb13698a6ec0aae"
+  integrity sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==
+  dependencies:
+    universalify "^2.0.0"
+  optionalDependencies:
+    graceful-fs "^4.1.6"
+
+jsonschema@^1.2.4:
+  version "1.4.1"
+  resolved "https://registry.yarnpkg.com/jsonschema/-/jsonschema-1.4.1.tgz#cc4c3f0077fb4542982973d8a083b6b34f482dab"
+  integrity sha512-S6cATIPVv1z0IlxdN+zUk5EPjkGCdnhN4wVSBlvoUO1tOLJootbo9CquNJmbIh4yikWHiUedhRYrNPn1arpEmQ==
+
+jsprim@^1.2.2:
+  version "1.4.2"
+  resolved "https://registry.yarnpkg.com/jsprim/-/jsprim-1.4.2.tgz#712c65533a15c878ba59e9ed5f0e26d5b77c5feb"
+  integrity sha512-P2bSOMAc/ciLz6DzgjVlGJP9+BrJWu5UDGK70C2iweC5QBIeFf0ZXRvGjEj2uYgrY2MkAAhsSWHDWlFtEroZWw==
+  dependencies:
+    assert-plus "1.0.0"
+    extsprintf "1.3.0"
+    json-schema "0.4.0"
+    verror "1.10.0"
+
+keccak@3.0.2:
+  version "3.0.2"
+  resolved "https://registry.yarnpkg.com/keccak/-/keccak-3.0.2.tgz#4c2c6e8c54e04f2670ee49fa734eb9da152206e0"
+  integrity sha512-PyKKjkH53wDMLGrvmRGSNWgmSxZOUqbnXwKL9tmgbFYA1iAYqW21kfR7mZXV0MlESiefxQQE9X9fTa3X+2MPDQ==
+  dependencies:
+    node-addon-api "^2.0.0"
+    node-gyp-build "^4.2.0"
+    readable-stream "^3.6.0"
+
+keccak@^3.0.0, keccak@^3.0.2:
+  version "3.0.3"
+  resolved "https://registry.yarnpkg.com/keccak/-/keccak-3.0.3.tgz#4bc35ad917be1ef54ff246f904c2bbbf9ac61276"
+  integrity sha512-JZrLIAJWuZxKbCilMpNz5Vj7Vtb4scDG3dMXLOsbzBmQGyjwE61BbW7bJkfKKCShXiQZt3T6sBgALRtmd+nZaQ==
+  dependencies:
+    node-addon-api "^2.0.0"
+    node-gyp-build "^4.2.0"
+    readable-stream "^3.6.0"
+
+keyv@^4.0.0:
+  version "4.5.3"
+  resolved "https://registry.yarnpkg.com/keyv/-/keyv-4.5.3.tgz#00873d2b046df737963157bd04f294ca818c9c25"
+  integrity sha512-QCiSav9WaX1PgETJ+SpNnx2PRRapJ/oRSXM4VO5OGYGSjrxbKPVFVhB3l2OCbLCk329N8qyAtsJjSjvVBWzEug==
+  dependencies:
+    json-buffer "3.0.1"
+
+kind-of@^6.0.2:
+  version "6.0.3"
+  resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-6.0.3.tgz#07c05034a6c349fa06e24fa35aa76db4580ce4dd"
+  integrity sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==
+
+klaw@^1.0.0:
+  version "1.3.1"
+  resolved "https://registry.yarnpkg.com/klaw/-/klaw-1.3.1.tgz#4088433b46b3b1ba259d78785d8e96f73ba02439"
+  integrity sha512-TED5xi9gGQjGpNnvRWknrwAB1eL5GciPfVFOt3Vk1OJCVDQbzuSfrF3hkUQKlsgKrG1F+0t5W0m+Fje1jIt8rw==
+  optionalDependencies:
+    graceful-fs "^4.1.9"
+
+kleur@^3.0.3:
+  version "3.0.3"
+  resolved "https://registry.yarnpkg.com/kleur/-/kleur-3.0.3.tgz#a79c9ecc86ee1ce3fa6206d1216c501f147fc07e"
+  integrity sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==
+
+lcid@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/lcid/-/lcid-1.0.0.tgz#308accafa0bc483a3867b4b6f2b9506251d1b835"
+  integrity sha512-YiGkH6EnGrDGqLMITnGjXtGmNtjoXw9SVUzcaos8RBi7Ps0VBylkq+vOcY9QE5poLasPCR849ucFUkl0UzUyOw==
+  dependencies:
+    invert-kv "^1.0.0"
+
+level-concat-iterator@^3.0.0:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/level-concat-iterator/-/level-concat-iterator-3.1.0.tgz#5235b1f744bc34847ed65a50548aa88d22e881cf"
+  integrity sha512-BWRCMHBxbIqPxJ8vHOvKUsaO0v1sLYZtjN3K2iZJsRBYtp+ONsY6Jfi6hy9K3+zolgQRryhIn2NRZjZnWJ9NmQ==
+  dependencies:
+    catering "^2.1.0"
+
+level-supports@^2.0.1:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/level-supports/-/level-supports-2.1.0.tgz#9af908d853597ecd592293b2fad124375be79c5f"
+  integrity sha512-E486g1NCjW5cF78KGPrMDRBYzPuueMZ6VBXHT6gC7A8UYWGiM14fGgp+s/L1oFfDWSPV/+SFkYCmZ0SiESkRKA==
+
+level-supports@^4.0.0:
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/level-supports/-/level-supports-4.0.1.tgz#431546f9d81f10ff0fea0e74533a0e875c08c66a"
+  integrity sha512-PbXpve8rKeNcZ9C1mUicC9auIYFyGpkV9/i6g76tLgANwWhtG2v7I4xNBUlkn3lE2/dZF3Pi0ygYGtLc4RXXdA==
+
+level-transcoder@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/level-transcoder/-/level-transcoder-1.0.1.tgz#f8cef5990c4f1283d4c86d949e73631b0bc8ba9c"
+  integrity sha512-t7bFwFtsQeD8cl8NIoQ2iwxA0CL/9IFw7/9gAjOonH0PWTTiRfY7Hq+Ejbsxh86tXobDQ6IOiddjNYIfOBs06w==
+  dependencies:
+    buffer "^6.0.3"
+    module-error "^1.0.1"
+
+level@^8.0.0:
+  version "8.0.0"
+  resolved "https://registry.yarnpkg.com/level/-/level-8.0.0.tgz#41b4c515dabe28212a3e881b61c161ffead14394"
+  integrity sha512-ypf0jjAk2BWI33yzEaaotpq7fkOPALKAgDBxggO6Q9HGX2MRXn0wbP1Jn/tJv1gtL867+YOjOB49WaUF3UoJNQ==
+  dependencies:
+    browser-level "^1.0.1"
+    classic-level "^1.2.0"
+
+leveldown@6.1.0:
+  version "6.1.0"
+  resolved "https://registry.yarnpkg.com/leveldown/-/leveldown-6.1.0.tgz#7ab1297706f70c657d1a72b31b40323aa612b9ee"
+  integrity sha512-8C7oJDT44JXxh04aSSsfcMI8YiaGRhOFI9/pMEL7nWJLVsWajDPTRxsSHTM2WcTVY5nXM+SuRHzPPi0GbnDX+w==
+  dependencies:
+    abstract-leveldown "^7.2.0"
+    napi-macros "~2.0.0"
+    node-gyp-build "^4.3.0"
+
+levn@^0.4.1:
+  version "0.4.1"
+  resolved "https://registry.yarnpkg.com/levn/-/levn-0.4.1.tgz#ae4562c007473b932a6200d403268dd2fffc6ade"
+  integrity sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==
+  dependencies:
+    prelude-ls "^1.2.1"
+    type-check "~0.4.0"
+
+levn@~0.3.0:
+  version "0.3.0"
+  resolved "https://registry.yarnpkg.com/levn/-/levn-0.3.0.tgz#3b09924edf9f083c0490fdd4c0bc4421e04764ee"
+  integrity sha512-0OO4y2iOHix2W6ujICbKIaEQXvFQHue65vUG3pb5EUomzPI90z9hsA1VsO/dbIIpC53J8gxM9Q4Oho0jrCM/yA==
+  dependencies:
+    prelude-ls "~1.1.2"
+    type-check "~0.3.2"
+
+lines-and-columns@^1.1.6:
+  version "1.2.4"
+  resolved "https://registry.yarnpkg.com/lines-and-columns/-/lines-and-columns-1.2.4.tgz#eca284f75d2965079309dc0ad9255abb2ebc1632"
+  integrity sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==
+
+load-json-file@^1.0.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/load-json-file/-/load-json-file-1.1.0.tgz#956905708d58b4bab4c2261b04f59f31c99374c0"
+  integrity sha512-cy7ZdNRXdablkXYNI049pthVeXFurRyb9+hA/dZzerZ0pGTx42z+y+ssxBaVV2l70t1muq5IdKhn4UtcoGUY9A==
+  dependencies:
+    graceful-fs "^4.1.2"
+    parse-json "^2.2.0"
+    pify "^2.0.0"
+    pinkie-promise "^2.0.0"
+    strip-bom "^2.0.0"
+
+locate-path@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-2.0.0.tgz#2b568b265eec944c6d9c0de9c3dbbbca0354cd8e"
+  integrity sha512-NCI2kiDkyR7VeEKm27Kda/iQHyKJe1Bu0FlTbYp3CqJu+9IFe9bLyAjMxf5ZDDbEg+iMPzB5zYyUTSm8wVTKmA==
+  dependencies:
+    p-locate "^2.0.0"
+    path-exists "^3.0.0"
+
+locate-path@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-3.0.0.tgz#dbec3b3ab759758071b58fe59fc41871af21400e"
+  integrity sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==
+  dependencies:
+    p-locate "^3.0.0"
+    path-exists "^3.0.0"
+
+locate-path@^6.0.0:
+  version "6.0.0"
+  resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-6.0.0.tgz#55321eb309febbc59c4801d931a72452a681d286"
+  integrity sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==
+  dependencies:
+    p-locate "^5.0.0"
+
+lodash.assign@^4.0.3, lodash.assign@^4.0.6:
+  version "4.2.0"
+  resolved "https://registry.yarnpkg.com/lodash.assign/-/lodash.assign-4.2.0.tgz#0d99f3ccd7a6d261d19bdaeb9245005d285808e7"
+  integrity sha512-hFuH8TY+Yji7Eja3mGiuAxBqLagejScbG8GbG0j6o9vzn0YL14My+ktnqtZgFTosKymC9/44wP6s7xyuLfnClw==
+
+lodash.camelcase@^4.3.0:
+  version "4.3.0"
+  resolved "https://registry.yarnpkg.com/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz#b28aa6288a2b9fc651035c7711f65ab6190331a6"
+  integrity sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA==
+
+lodash.merge@^4.6.2:
+  version "4.6.2"
+  resolved "https://registry.yarnpkg.com/lodash.merge/-/lodash.merge-4.6.2.tgz#558aa53b43b661e1925a0afdfa36a9a1085fe57a"
+  integrity sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==
+
+lodash.truncate@^4.4.2:
+  version "4.4.2"
+  resolved "https://registry.yarnpkg.com/lodash.truncate/-/lodash.truncate-4.4.2.tgz#5a350da0b1113b837ecfffd5812cbe58d6eae193"
+  integrity sha512-jttmRe7bRse52OsWIMDLaXxWqRAmtIUccAQ3garviCqJjafXOfNMO0yMfNpdD6zbGaTU0P5Nz7e7gAT6cKmJRw==
+
+lodash@^4.17.11, lodash@^4.17.14, lodash@^4.17.15, lodash@^4.17.19, lodash@^4.17.21:
+  version "4.17.21"
+  resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c"
+  integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==
+
+log-symbols@3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/log-symbols/-/log-symbols-3.0.0.tgz#f3a08516a5dea893336a7dee14d18a1cfdab77c4"
+  integrity sha512-dSkNGuI7iG3mfvDzUuYZyvk5dD9ocYCYzNU6CYDE6+Xqd+gwme6Z00NS3dUh8mq/73HaEtT7m6W+yUPtU6BZnQ==
+  dependencies:
+    chalk "^2.4.2"
+
+log-symbols@4.1.0, log-symbols@^4.1.0:
+  version "4.1.0"
+  resolved "https://registry.yarnpkg.com/log-symbols/-/log-symbols-4.1.0.tgz#3fbdbb95b4683ac9fc785111e792e558d4abd503"
+  integrity sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==
+  dependencies:
+    chalk "^4.1.0"
+    is-unicode-supported "^0.1.0"
+
+loupe@^2.3.1:
+  version "2.3.6"
+  resolved "https://registry.yarnpkg.com/loupe/-/loupe-2.3.6.tgz#76e4af498103c532d1ecc9be102036a21f787b53"
+  integrity sha512-RaPMZKiMy8/JruncMU5Bt6na1eftNoo++R4Y+N2FrxkDVTrGvcyzFTsaGif4QTeKESheMGegbhw6iUAq+5A8zA==
+  dependencies:
+    get-func-name "^2.0.0"
+
+lower-case-first@^1.0.0:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/lower-case-first/-/lower-case-first-1.0.2.tgz#e5da7c26f29a7073be02d52bac9980e5922adfa1"
+  integrity sha512-UuxaYakO7XeONbKrZf5FEgkantPf5DUqDayzP5VXZrtRPdH86s4kN47I8B3TW10S4QKiE3ziHNf3kRN//okHjA==
+  dependencies:
+    lower-case "^1.1.2"
+
+lower-case@^1.1.0, lower-case@^1.1.1, lower-case@^1.1.2:
+  version "1.1.4"
+  resolved "https://registry.yarnpkg.com/lower-case/-/lower-case-1.1.4.tgz#9a2cabd1b9e8e0ae993a4bf7d5875c39c42e8eac"
+  integrity sha512-2Fgx1Ycm599x+WGpIYwJOvsjmXFzTSc34IwDWALRA/8AopUKAVPwfJ+h5+f85BCp0PWmmJcWzEpxOpoXycMpdA==
+
+lowercase-keys@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-2.0.0.tgz#2603e78b7b4b0006cbca2fbcc8a3202558ac9479"
+  integrity sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==
+
+lowercase-keys@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-3.0.0.tgz#c5e7d442e37ead247ae9db117a9d0a467c89d4f2"
+  integrity sha512-ozCC6gdQ+glXOQsveKD0YsDy8DSQFjDTz4zyzEHNV5+JP5D62LmfDZ6o1cycFx9ouG940M5dE8C8CTewdj2YWQ==
+
+lru-cache@^5.1.1:
+  version "5.1.1"
+  resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-5.1.1.tgz#1da27e6710271947695daf6848e847f01d84b920"
+  integrity sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==
+  dependencies:
+    yallist "^3.0.2"
+
+lru-cache@^6.0.0:
+  version "6.0.0"
+  resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz#6d6fe6570ebd96aaf90fcad1dafa3b2566db3a94"
+  integrity sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==
+  dependencies:
+    yallist "^4.0.0"
+
+lru_map@^0.3.3:
+  version "0.3.3"
+  resolved "https://registry.yarnpkg.com/lru_map/-/lru_map-0.3.3.tgz#b5c8351b9464cbd750335a79650a0ec0e56118dd"
+  integrity sha512-Pn9cox5CsMYngeDbmChANltQl+5pi6XmTrraMSzhPmMBbmgcxmqWry0U3PGapCU1yB4/LqCcom7qhHZiF/jGfQ==
+
+make-error@^1.1.1:
+  version "1.3.6"
+  resolved "https://registry.yarnpkg.com/make-error/-/make-error-1.3.6.tgz#2eb2e37ea9b67c4891f684a1394799af484cf7a2"
+  integrity sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==
+
+markdown-table@^1.1.3:
+  version "1.1.3"
+  resolved "https://registry.yarnpkg.com/markdown-table/-/markdown-table-1.1.3.tgz#9fcb69bcfdb8717bfd0398c6ec2d93036ef8de60"
+  integrity sha512-1RUZVgQlpJSPWYbFSpmudq5nHY1doEIv89gBtF0s4gW1GF2XorxcA/70M5vq7rLv0a6mhOUccRsqkwhwLCIQ2Q==
+
+match-all@^1.2.6:
+  version "1.2.6"
+  resolved "https://registry.yarnpkg.com/match-all/-/match-all-1.2.6.tgz#66d276ad6b49655551e63d3a6ee53e8be0566f8d"
+  integrity sha512-0EESkXiTkWzrQQntBu2uzKvLu6vVkUGz40nGPbSZuegcfE5UuSzNjLaIu76zJWuaT/2I3Z/8M06OlUOZLGwLlQ==
+
+mcl-wasm@^0.7.1:
+  version "0.7.9"
+  resolved "https://registry.yarnpkg.com/mcl-wasm/-/mcl-wasm-0.7.9.tgz#c1588ce90042a8700c3b60e40efb339fc07ab87f"
+  integrity sha512-iJIUcQWA88IJB/5L15GnJVnSQJmf/YaxxV6zRavv83HILHaJQb6y0iFyDMdDO0gN8X37tdxmAOrH/P8B6RB8sQ==
+
+md5.js@^1.3.4:
+  version "1.3.5"
+  resolved "https://registry.yarnpkg.com/md5.js/-/md5.js-1.3.5.tgz#b5d07b8e3216e3e27cd728d72f70d1e6a342005f"
+  integrity sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==
+  dependencies:
+    hash-base "^3.0.0"
+    inherits "^2.0.1"
+    safe-buffer "^5.1.2"
+
+media-typer@0.3.0:
+  version "0.3.0"
+  resolved "https://registry.yarnpkg.com/media-typer/-/media-typer-0.3.0.tgz#8710d7af0aa626f8fffa1ce00168545263255748"
+  integrity sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==
+
+memory-level@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/memory-level/-/memory-level-1.0.0.tgz#7323c3fd368f9af2f71c3cd76ba403a17ac41692"
+  integrity sha512-UXzwewuWeHBz5krr7EvehKcmLFNoXxGcvuYhC41tRnkrTbJohtS7kVn9akmgirtRygg+f7Yjsfi8Uu5SGSQ4Og==
+  dependencies:
+    abstract-level "^1.0.0"
+    functional-red-black-tree "^1.0.1"
+    module-error "^1.0.1"
+
+memorystream@^0.3.1:
+  version "0.3.1"
+  resolved "https://registry.yarnpkg.com/memorystream/-/memorystream-0.3.1.tgz#86d7090b30ce455d63fbae12dda51a47ddcaf9b2"
+  integrity sha512-S3UwM3yj5mtUSEfP41UZmt/0SCoVYUcU1rkXv+BQ5Ig8ndL4sPoJNBUJERafdPb5jjHJGuMgytgKvKIf58XNBw==
+
+merge-descriptors@1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/merge-descriptors/-/merge-descriptors-1.0.1.tgz#b00aaa556dd8b44568150ec9d1b953f3f90cbb61"
+  integrity sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w==
+
+merge2@^1.2.3, merge2@^1.3.0, merge2@^1.4.1:
+  version "1.4.1"
+  resolved "https://registry.yarnpkg.com/merge2/-/merge2-1.4.1.tgz#4368892f885e907455a6fd7dc55c0c9d404990ae"
+  integrity sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==
+
+methods@~1.1.2:
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/methods/-/methods-1.1.2.tgz#5529a4d67654134edcc5266656835b0f851afcee"
+  integrity sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==
+
+micromatch@^4.0.4:
+  version "4.0.5"
+  resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.5.tgz#bc8999a7cbbf77cdc89f132f6e467051b49090c6"
+  integrity sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==
+  dependencies:
+    braces "^3.0.2"
+    picomatch "^2.3.1"
+
+mime-db@1.52.0:
+  version "1.52.0"
+  resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.52.0.tgz#bbabcdc02859f4987301c856e3387ce5ec43bf70"
+  integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==
+
+mime-types@^2.1.12, mime-types@^2.1.16, mime-types@~2.1.19, mime-types@~2.1.24, mime-types@~2.1.34:
+  version "2.1.35"
+  resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.35.tgz#381a871b62a734450660ae3deee44813f70d959a"
+  integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==
+  dependencies:
+    mime-db "1.52.0"
+
+mime@1.6.0:
+  version "1.6.0"
+  resolved "https://registry.yarnpkg.com/mime/-/mime-1.6.0.tgz#32cd9e5c64553bd58d19a568af452acff04981b1"
+  integrity sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==
+
+mimic-fn@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b"
+  integrity sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==
+
+mimic-response@^1.0.0:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/mimic-response/-/mimic-response-1.0.1.tgz#4923538878eef42063cb8a3e3b0798781487ab1b"
+  integrity sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==
+
+mimic-response@^3.1.0:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/mimic-response/-/mimic-response-3.1.0.tgz#2d1d59af9c1b129815accc2c46a022a5ce1fa3c9"
+  integrity sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==
+
+min-document@^2.19.0:
+  version "2.19.0"
+  resolved "https://registry.yarnpkg.com/min-document/-/min-document-2.19.0.tgz#7bd282e3f5842ed295bb748cdd9f1ffa2c824685"
+  integrity sha512-9Wy1B3m3f66bPPmU5hdA4DR4PB2OfDU/+GS3yAB7IQozE3tqXaVv2zOjgla7MEGSRv95+ILmOuvhLkOK6wJtCQ==
+  dependencies:
+    dom-walk "^0.1.0"
+
+minimalistic-assert@^1.0.0, minimalistic-assert@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz#2e194de044626d4a10e7f7fbc00ce73e83e4d5c7"
+  integrity sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==
+
+minimalistic-crypto-utils@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz#f6c00c1c0b082246e5c4d99dfb8c7c083b2b582a"
+  integrity sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==
+
+"minimatch@2 || 3", minimatch@^3.0.4, minimatch@^3.0.5, minimatch@^3.1.1, minimatch@^3.1.2:
+  version "3.1.2"
+  resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.1.2.tgz#19cd194bfd3e428f049a70817c038d89ab4be35b"
+  integrity sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==
+  dependencies:
+    brace-expansion "^1.1.7"
+
+minimatch@3.0.4:
+  version "3.0.4"
+  resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.0.4.tgz#5166e286457f03306064be5497e8dbb0c3d32083"
+  integrity sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==
+  dependencies:
+    brace-expansion "^1.1.7"
+
+minimatch@5.0.1:
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-5.0.1.tgz#fb9022f7528125187c92bd9e9b6366be1cf3415b"
+  integrity sha512-nLDxIFRyhDblz3qMuq+SoRZED4+miJ/G+tdDrjkkkRnjAsBexeGpgjLEQ0blJy7rHhR2b93rhQY4SvyWu9v03g==
+  dependencies:
+    brace-expansion "^2.0.1"
+
+minimatch@^5.0.1:
+  version "5.1.6"
+  resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-5.1.6.tgz#1cfcb8cf5522ea69952cd2af95ae09477f122a96"
+  integrity sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==
+  dependencies:
+    brace-expansion "^2.0.1"
+
+minimist@^1.2.0, minimist@^1.2.5, minimist@^1.2.6:
+  version "1.2.8"
+  resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.8.tgz#c1a464e7693302e082a075cee0c057741ac4772c"
+  integrity sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==
+
+minipass@^2.6.0, minipass@^2.9.0:
+  version "2.9.0"
+  resolved "https://registry.yarnpkg.com/minipass/-/minipass-2.9.0.tgz#e713762e7d3e32fed803115cf93e04bca9fcc9a6"
+  integrity sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==
+  dependencies:
+    safe-buffer "^5.1.2"
+    yallist "^3.0.0"
+
+minizlib@^1.3.3:
+  version "1.3.3"
+  resolved "https://registry.yarnpkg.com/minizlib/-/minizlib-1.3.3.tgz#2290de96818a34c29551c8a8d301216bd65a861d"
+  integrity sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==
+  dependencies:
+    minipass "^2.9.0"
+
+mkdirp-promise@^5.0.1:
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/mkdirp-promise/-/mkdirp-promise-5.0.1.tgz#e9b8f68e552c68a9c1713b84883f7a1dd039b8a1"
+  integrity sha512-Hepn5kb1lJPtVW84RFT40YG1OddBNTOVUZR2bzQUHc+Z03en8/3uX0+060JDhcEzyO08HmipsN9DcnFMxhIL9w==
+  dependencies:
+    mkdirp "*"
+
+mkdirp@*:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-3.0.1.tgz#e44e4c5607fb279c168241713cc6e0fea9adcb50"
+  integrity sha512-+NsyUUAZDmo6YVHzL/stxSu3t9YS1iljliy3BSDrXJ/dkn1KYdmtZODGGjLcc9XLgVVpH4KshHB8XmZgMhaBXg==
+
+mkdirp@0.5.5:
+  version "0.5.5"
+  resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.5.5.tgz#d91cefd62d1436ca0f41620e251288d420099def"
+  integrity sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==
+  dependencies:
+    minimist "^1.2.5"
+
+mkdirp@0.5.x, mkdirp@^0.5.5:
+  version "0.5.6"
+  resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.5.6.tgz#7def03d2432dcae4ba1d611445c48396062255f6"
+  integrity sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==
+  dependencies:
+    minimist "^1.2.6"
+
+mkdirp@^1.0.4:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-1.0.4.tgz#3eb5ed62622756d79a5f0e2a221dfebad75c2f7e"
+  integrity sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==
+
+mnemonist@^0.38.0:
+  version "0.38.5"
+  resolved "https://registry.yarnpkg.com/mnemonist/-/mnemonist-0.38.5.tgz#4adc7f4200491237fe0fa689ac0b86539685cade"
+  integrity sha512-bZTFT5rrPKtPJxj8KSV0WkPyNxl72vQepqqVUAW2ARUpUSF2qXMB6jZj7hW5/k7C1rtpzqbD/IIbJwLXUjCHeg==
+  dependencies:
+    obliterator "^2.0.0"
+
+mocha@10.2.0, mocha@^10.0.0:
+  version "10.2.0"
+  resolved "https://registry.yarnpkg.com/mocha/-/mocha-10.2.0.tgz#1fd4a7c32ba5ac372e03a17eef435bd00e5c68b8"
+  integrity sha512-IDY7fl/BecMwFHzoqF2sg/SHHANeBoMMXFlS9r0OXKDssYE1M5O43wUY/9BVPeIvfH2zmEbBfseqN9gBQZzXkg==
+  dependencies:
+    ansi-colors "4.1.1"
+    browser-stdout "1.3.1"
+    chokidar "3.5.3"
+    debug "4.3.4"
+    diff "5.0.0"
+    escape-string-regexp "4.0.0"
+    find-up "5.0.0"
+    glob "7.2.0"
+    he "1.2.0"
+    js-yaml "4.1.0"
+    log-symbols "4.1.0"
+    minimatch "5.0.1"
+    ms "2.1.3"
+    nanoid "3.3.3"
+    serialize-javascript "6.0.0"
+    strip-json-comments "3.1.1"
+    supports-color "8.1.1"
+    workerpool "6.2.1"
+    yargs "16.2.0"
+    yargs-parser "20.2.4"
+    yargs-unparser "2.0.0"
+
+mocha@^7.1.1:
+  version "7.2.0"
+  resolved "https://registry.yarnpkg.com/mocha/-/mocha-7.2.0.tgz#01cc227b00d875ab1eed03a75106689cfed5a604"
+  integrity sha512-O9CIypScywTVpNaRrCAgoUnJgozpIofjKUYmJhiCIJMiuYnLI6otcb1/kpW9/n/tJODHGZ7i8aLQoDVsMtOKQQ==
+  dependencies:
+    ansi-colors "3.2.3"
+    browser-stdout "1.3.1"
+    chokidar "3.3.0"
+    debug "3.2.6"
+    diff "3.5.0"
+    escape-string-regexp "1.0.5"
+    find-up "3.0.0"
+    glob "7.1.3"
+    growl "1.10.5"
+    he "1.2.0"
+    js-yaml "3.13.1"
+    log-symbols "3.0.0"
+    minimatch "3.0.4"
+    mkdirp "0.5.5"
+    ms "2.1.1"
+    node-environment-flags "1.0.6"
+    object.assign "4.1.0"
+    strip-json-comments "2.0.1"
+    supports-color "6.0.0"
+    which "1.3.1"
+    wide-align "1.1.3"
+    yargs "13.3.2"
+    yargs-parser "13.1.2"
+    yargs-unparser "1.6.0"
+
+mock-fs@^4.1.0:
+  version "4.14.0"
+  resolved "https://registry.yarnpkg.com/mock-fs/-/mock-fs-4.14.0.tgz#ce5124d2c601421255985e6e94da80a7357b1b18"
+  integrity sha512-qYvlv/exQ4+svI3UOvPUpLDF0OMX5euvUH0Ny4N5QyRyhNdgAgUrVH3iUINSzEPLvx0kbo/Bp28GJKIqvE7URw==
+
+module-error@^1.0.1, module-error@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/module-error/-/module-error-1.0.2.tgz#8d1a48897ca883f47a45816d4fb3e3c6ba404d86"
+  integrity sha512-0yuvsqSCv8LbaOKhnsQ/T5JhyFlCYLPXK3U2sgV10zoKQwzs/MyfuQUOZQ1V/6OCOJsK/TRgNVrPuPDqtdMFtA==
+
+ms@2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8"
+  integrity sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==
+
+ms@2.1.1:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.1.tgz#30a5864eb3ebb0a66f2ebe6d727af06a09d86e0a"
+  integrity sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==
+
+ms@2.1.2:
+  version "2.1.2"
+  resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009"
+  integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==
+
+ms@2.1.3, ms@^2.1.1:
+  version "2.1.3"
+  resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2"
+  integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==
+
+multibase@^0.7.0:
+  version "0.7.0"
+  resolved "https://registry.yarnpkg.com/multibase/-/multibase-0.7.0.tgz#1adfc1c50abe05eefeb5091ac0c2728d6b84581b"
+  integrity sha512-TW8q03O0f6PNFTQDvh3xxH03c8CjGaaYrjkl9UQPG6rz53TQzzxJVCIWVjzcbN/Q5Y53Zd0IBQBMVktVgNx4Fg==
+  dependencies:
+    base-x "^3.0.8"
+    buffer "^5.5.0"
+
+multibase@~0.6.0:
+  version "0.6.1"
+  resolved "https://registry.yarnpkg.com/multibase/-/multibase-0.6.1.tgz#b76df6298536cc17b9f6a6db53ec88f85f8cc12b"
+  integrity sha512-pFfAwyTjbbQgNc3G7D48JkJxWtoJoBMaR4xQUOuB8RnCgRqaYmWNFeJTTvrJ2w51bjLq2zTby6Rqj9TQ9elSUw==
+  dependencies:
+    base-x "^3.0.8"
+    buffer "^5.5.0"
+
+multicodec@^0.5.5:
+  version "0.5.7"
+  resolved "https://registry.yarnpkg.com/multicodec/-/multicodec-0.5.7.tgz#1fb3f9dd866a10a55d226e194abba2dcc1ee9ffd"
+  integrity sha512-PscoRxm3f+88fAtELwUnZxGDkduE2HD9Q6GHUOywQLjOGT/HAdhjLDYNZ1e7VR0s0TP0EwZ16LNUTFpoBGivOA==
+  dependencies:
+    varint "^5.0.0"
+
+multicodec@^1.0.0:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/multicodec/-/multicodec-1.0.4.tgz#46ac064657c40380c28367c90304d8ed175a714f"
+  integrity sha512-NDd7FeS3QamVtbgfvu5h7fd1IlbaC4EQ0/pgU4zqE2vdHCmBGsUa0TiM8/TdSeG6BMPC92OOCf8F1ocE/Wkrrg==
+  dependencies:
+    buffer "^5.6.0"
+    varint "^5.0.0"
+
+multihashes@^0.4.15, multihashes@~0.4.15:
+  version "0.4.21"
+  resolved "https://registry.yarnpkg.com/multihashes/-/multihashes-0.4.21.tgz#dc02d525579f334a7909ade8a122dabb58ccfcb5"
+  integrity sha512-uVSvmeCWf36pU2nB4/1kzYZjsXD9vofZKpgudqkceYY5g2aZZXJ5r9lxuzoRLl1OAp28XljXsEJ/X/85ZsKmKw==
+  dependencies:
+    buffer "^5.5.0"
+    multibase "^0.7.0"
+    varint "^5.0.0"
+
+murmur-128@^0.2.1:
+  version "0.2.1"
+  resolved "https://registry.yarnpkg.com/murmur-128/-/murmur-128-0.2.1.tgz#a9f6568781d2350ecb1bf80c14968cadbeaa4b4d"
+  integrity sha512-WseEgiRkI6aMFBbj8Cg9yBj/y+OdipwVC7zUo3W2W1JAJITwouUOtpqsmGSg67EQmwwSyod7hsVsWY5LsrfQVg==
+  dependencies:
+    encode-utf8 "^1.0.2"
+    fmix "^0.1.0"
+    imul "^1.0.0"
+
+nano-base32@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/nano-base32/-/nano-base32-1.0.1.tgz#ba548c879efcfb90da1c4d9e097db4a46c9255ef"
+  integrity sha512-sxEtoTqAPdjWVGv71Q17koMFGsOMSiHsIFEvzOM7cNp8BXB4AnEwmDabm5dorusJf/v1z7QxaZYxUorU9RKaAw==
+
+nano-json-stream-parser@^0.1.2:
+  version "0.1.2"
+  resolved "https://registry.yarnpkg.com/nano-json-stream-parser/-/nano-json-stream-parser-0.1.2.tgz#0cc8f6d0e2b622b479c40d499c46d64b755c6f5f"
+  integrity sha512-9MqxMH/BSJC7dnLsEMPyfN5Dvoo49IsPFYMcHw3Bcfc2kN0lpHRBSzlMSVx4HGyJ7s9B31CyBTVehWJoQ8Ctew==
+
+nanoid@3.3.3:
+  version "3.3.3"
+  resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.3.tgz#fd8e8b7aa761fe807dba2d1b98fb7241bb724a25"
+  integrity sha512-p1sjXuopFs0xg+fPASzQ28agW1oHD7xDsd9Xkf3T15H3c/cifrFHVwrh74PdoklAPi+i7MdRsE47vm2r6JoB+w==
+
+napi-macros@^2.2.2:
+  version "2.2.2"
+  resolved "https://registry.yarnpkg.com/napi-macros/-/napi-macros-2.2.2.tgz#817fef20c3e0e40a963fbf7b37d1600bd0201044"
+  integrity sha512-hmEVtAGYzVQpCKdbQea4skABsdXW4RUh5t5mJ2zzqowJS2OyXZTU1KhDVFhx+NlWZ4ap9mqR9TcDO3LTTttd+g==
+
+napi-macros@~2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/napi-macros/-/napi-macros-2.0.0.tgz#2b6bae421e7b96eb687aa6c77a7858640670001b"
+  integrity sha512-A0xLykHtARfueITVDernsAWdtIMbOJgKgcluwENp3AlsKN/PloyO10HtmoqnFAQAcxPkgZN7wdfPfEd0zNGxbg==
+
+natural-compare-lite@^1.4.0:
+  version "1.4.0"
+  resolved "https://registry.yarnpkg.com/natural-compare-lite/-/natural-compare-lite-1.4.0.tgz#17b09581988979fddafe0201e931ba933c96cbb4"
+  integrity sha512-Tj+HTDSJJKaZnfiuw+iaF9skdPpTo2GtEly5JHnWV/hfv2Qj/9RKsGISQtLh2ox3l5EAGw487hnBee0sIJ6v2g==
+
+natural-compare@^1.4.0:
+  version "1.4.0"
+  resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7"
+  integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==
+
+negotiator@0.6.3:
+  version "0.6.3"
+  resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.3.tgz#58e323a72fedc0d6f9cd4d31fe49f51479590ccd"
+  integrity sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==
+
+neo-async@^2.6.2:
+  version "2.6.2"
+  resolved "https://registry.yarnpkg.com/neo-async/-/neo-async-2.6.2.tgz#b4aafb93e3aeb2d8174ca53cf163ab7d7308305f"
+  integrity sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==
+
+next-tick@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/next-tick/-/next-tick-1.1.0.tgz#1836ee30ad56d67ef281b22bd199f709449b35eb"
+  integrity sha512-CXdUiJembsNjuToQvxayPZF9Vqht7hewsvy2sOWafLvi2awflj9mOC6bHIg50orX8IJvWKY9wYQ/zB2kogPslQ==
+
+no-case@^2.2.0, no-case@^2.3.2:
+  version "2.3.2"
+  resolved "https://registry.yarnpkg.com/no-case/-/no-case-2.3.2.tgz#60b813396be39b3f1288a4c1ed5d1e7d28b464ac"
+  integrity sha512-rmTZ9kz+f3rCvK2TD1Ue/oZlns7OGoIWP4fc3llxxRXlOkHKoWPPWJOfFYpITabSow43QJbRIoHQXtt10VldyQ==
+  dependencies:
+    lower-case "^1.1.1"
+
+node-addon-api@^2.0.0:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/node-addon-api/-/node-addon-api-2.0.2.tgz#432cfa82962ce494b132e9d72a15b29f71ff5d32"
+  integrity sha512-Ntyt4AIXyaLIuMHF6IOoTakB3K+RWxwtsHNRxllEoA6vPwP9o4866g6YWDLUdnucilZhmkxiHwHr11gAENw+QA==
+
+node-emoji@^1.10.0:
+  version "1.11.0"
+  resolved "https://registry.yarnpkg.com/node-emoji/-/node-emoji-1.11.0.tgz#69a0150e6946e2f115e9d7ea4df7971e2628301c"
+  integrity sha512-wo2DpQkQp7Sjm2A0cq+sN7EHKO6Sl0ctXeBdFZrL9T9+UywORbufTcTZxom8YqpLQt/FqNMUkOpkZrJVYSKD3A==
+  dependencies:
+    lodash "^4.17.21"
+
+node-environment-flags@1.0.6:
+  version "1.0.6"
+  resolved "https://registry.yarnpkg.com/node-environment-flags/-/node-environment-flags-1.0.6.tgz#a30ac13621f6f7d674260a54dede048c3982c088"
+  integrity sha512-5Evy2epuL+6TM0lCQGpFIj6KwiEsGh1SrHUhTbNX+sLbBtjidPZFAnVK9y5yU1+h//RitLbRHTIMyxQPtxMdHw==
+  dependencies:
+    object.getownpropertydescriptors "^2.0.3"
+    semver "^5.7.0"
+
+node-fetch@^2.6.12:
+  version "2.6.12"
+  resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-2.6.12.tgz#02eb8e22074018e3d5a83016649d04df0e348fba"
+  integrity sha512-C/fGU2E8ToujUivIO0H+tpQ6HWo4eEmchoPIoXtxCrVghxdKq+QOHqEZW7tuP3KlV3bC8FRMO5nMCC7Zm1VP6g==
+  dependencies:
+    whatwg-url "^5.0.0"
+
+node-gyp-build@4.4.0:
+  version "4.4.0"
+  resolved "https://registry.yarnpkg.com/node-gyp-build/-/node-gyp-build-4.4.0.tgz#42e99687ce87ddeaf3a10b99dc06abc11021f3f4"
+  integrity sha512-amJnQCcgtRVw9SvoebO3BKGESClrfXGCUTX9hSn1OuGQTQBOZmVd0Z0OlecpuRksKvbsUqALE8jls/ErClAPuQ==
+
+node-gyp-build@^4.2.0, node-gyp-build@^4.3.0:
+  version "4.6.0"
+  resolved "https://registry.yarnpkg.com/node-gyp-build/-/node-gyp-build-4.6.0.tgz#0c52e4cbf54bbd28b709820ef7b6a3c2d6209055"
+  integrity sha512-NTZVKn9IylLwUzaKjkas1e4u2DLNcV4rdYagA4PWdPwW87Bi7z+BznyKSRwS/761tV/lzCGXplWsiaMjLqP2zQ==
+
+nofilter@^1.0.4:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/nofilter/-/nofilter-1.0.4.tgz#78d6f4b6a613e7ced8b015cec534625f7667006e"
+  integrity sha512-N8lidFp+fCz+TD51+haYdbDGrcBWwuHX40F5+z0qkUjMJ5Tp+rdSuAkMJ9N9eoolDlEVTf6u5icM+cNKkKW2mA==
+
+nofilter@^3.1.0:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/nofilter/-/nofilter-3.1.0.tgz#c757ba68801d41ff930ba2ec55bab52ca184aa66"
+  integrity sha512-l2NNj07e9afPnhAhvgVrCD/oy2Ai1yfLpuo3EpiO1jFTsB4sFz6oIfAfSZyQzVpkZQ9xS8ZS5g1jCBgq4Hwo0g==
+
+nopt@3.x:
+  version "3.0.6"
+  resolved "https://registry.yarnpkg.com/nopt/-/nopt-3.0.6.tgz#c6465dbf08abcd4db359317f79ac68a646b28ff9"
+  integrity sha512-4GUt3kSEYmk4ITxzB/b9vaIDfUVWN/Ml1Fwl11IlnIG2iaJ9O6WXZ9SrYM9NLI8OCBieN2Y8SWC2oJV0RQ7qYg==
+  dependencies:
+    abbrev "1"
+
+normalize-package-data@^2.3.2:
+  version "2.5.0"
+  resolved "https://registry.yarnpkg.com/normalize-package-data/-/normalize-package-data-2.5.0.tgz#e66db1838b200c1dfc233225d12cb36520e234a8"
+  integrity sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==
+  dependencies:
+    hosted-git-info "^2.1.4"
+    resolve "^1.10.0"
+    semver "2 || 3 || 4 || 5"
+    validate-npm-package-license "^3.0.1"
+
+normalize-path@^3.0.0, normalize-path@~3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-3.0.0.tgz#0dcd69ff23a1c9b11fd0978316644a0388216a65"
+  integrity sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==
+
+normalize-url@^6.0.1:
+  version "6.1.0"
+  resolved "https://registry.yarnpkg.com/normalize-url/-/normalize-url-6.1.0.tgz#40d0885b535deffe3f3147bec877d05fe4c5668a"
+  integrity sha512-DlL+XwOy3NxAQ8xuC0okPgK46iuVNAK01YN7RueYBqqFeGsBjV9XmCAzAdgt+667bCl5kPh9EqKKDwnaPG1I7A==
+
+nth-check@^2.0.1:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/nth-check/-/nth-check-2.1.1.tgz#c9eab428effce36cd6b92c924bdb000ef1f1ed1d"
+  integrity sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==
+  dependencies:
+    boolbase "^1.0.0"
+
+number-is-nan@^1.0.0:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/number-is-nan/-/number-is-nan-1.0.1.tgz#097b602b53422a522c1afb8790318336941a011d"
+  integrity sha512-4jbtZXNAsfZbAHiiqjLPBiCl16dES1zI4Hpzzxw61Tk+loF+sBDBKx1ICKKKwIqQ7M0mFn1TmkN7euSncWgHiQ==
+
+number-to-bn@1.7.0:
+  version "1.7.0"
+  resolved "https://registry.yarnpkg.com/number-to-bn/-/number-to-bn-1.7.0.tgz#bb3623592f7e5f9e0030b1977bd41a0c53fe1ea0"
+  integrity sha512-wsJ9gfSz1/s4ZsJN01lyonwuxA1tml6X1yBDnfpMglypcBRFZZkus26EdPSlqS5GJfYddVZa22p3VNb3z5m5Ig==
+  dependencies:
+    bn.js "4.11.6"
+    strip-hex-prefix "1.0.0"
+
+oauth-sign@~0.9.0:
+  version "0.9.0"
+  resolved "https://registry.yarnpkg.com/oauth-sign/-/oauth-sign-0.9.0.tgz#47a7b016baa68b5fa0ecf3dee08a85c679ac6455"
+  integrity sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==
+
+object-assign@^4, object-assign@^4.1.0, object-assign@^4.1.1:
+  version "4.1.1"
+  resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863"
+  integrity sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==
+
+object-inspect@^1.12.3, object-inspect@^1.9.0:
+  version "1.12.3"
+  resolved "https://registry.yarnpkg.com/object-inspect/-/object-inspect-1.12.3.tgz#ba62dffd67ee256c8c086dfae69e016cd1f198b9"
+  integrity sha512-geUvdk7c+eizMNUDkRpW1wJwgfOiOeHbxBR/hLXK1aT6zmVSO0jsQcs7fj6MGw89jC/cjGfLcNOrtMYtGqm81g==
+
+object-keys@^1.0.11, object-keys@^1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.1.1.tgz#1c47f272df277f3b1daf061677d9c82e2322c60e"
+  integrity sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==
+
+object.assign@4.1.0:
+  version "4.1.0"
+  resolved "https://registry.yarnpkg.com/object.assign/-/object.assign-4.1.0.tgz#968bf1100d7956bb3ca086f006f846b3bc4008da"
+  integrity sha512-exHJeq6kBKj58mqGyTQ9DFvrZC/eR6OwxzoM9YRoGBqrXYonaFyGiFMuc9VZrXf7DarreEwMpurG3dd+CNyW5w==
+  dependencies:
+    define-properties "^1.1.2"
+    function-bind "^1.1.1"
+    has-symbols "^1.0.0"
+    object-keys "^1.0.11"
+
+object.assign@^4.1.4:
+  version "4.1.4"
+  resolved "https://registry.yarnpkg.com/object.assign/-/object.assign-4.1.4.tgz#9673c7c7c351ab8c4d0b516f4343ebf4dfb7799f"
+  integrity sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.1.4"
+    has-symbols "^1.0.3"
+    object-keys "^1.1.1"
+
+object.fromentries@^2.0.6:
+  version "2.0.6"
+  resolved "https://registry.yarnpkg.com/object.fromentries/-/object.fromentries-2.0.6.tgz#cdb04da08c539cffa912dcd368b886e0904bfa73"
+  integrity sha512-VciD13dswC4j1Xt5394WR4MzmAQmlgN72phd/riNp9vtD7tp4QQWJ0R4wvclXcafgcYK8veHRed2W6XeGBvcfg==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.1.4"
+    es-abstract "^1.20.4"
+
+object.getownpropertydescriptors@^2.0.3:
+  version "2.1.6"
+  resolved "https://registry.yarnpkg.com/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.6.tgz#5e5c384dd209fa4efffead39e3a0512770ccc312"
+  integrity sha512-lq+61g26E/BgHv0ZTFgRvi7NMEPuAxLkFU7rukXjc/AlwH4Am5xXVnIXy3un1bg/JPbXHrixRkK1itUzzPiIjQ==
+  dependencies:
+    array.prototype.reduce "^1.0.5"
+    call-bind "^1.0.2"
+    define-properties "^1.2.0"
+    es-abstract "^1.21.2"
+    safe-array-concat "^1.0.0"
+
+object.groupby@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/object.groupby/-/object.groupby-1.0.0.tgz#cb29259cf90f37e7bac6437686c1ea8c916d12a9"
+  integrity sha512-70MWG6NfRH9GnbZOikuhPPYzpUpof9iW2J9E4dW7FXTqPNb6rllE6u39SKwwiNh8lCwX3DDb5OgcKGiEBrTTyw==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.2.0"
+    es-abstract "^1.21.2"
+    get-intrinsic "^1.2.1"
+
+object.values@^1.1.6:
+  version "1.1.6"
+  resolved "https://registry.yarnpkg.com/object.values/-/object.values-1.1.6.tgz#4abbaa71eba47d63589d402856f908243eea9b1d"
+  integrity sha512-FVVTkD1vENCsAcwNs9k6jea2uHC/X0+JcjG8YA60FN5CMaJmG95wT9jek/xX9nornqGRrBkKtzuAu2wuHpKqvw==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.1.4"
+    es-abstract "^1.20.4"
+
+obliterator@^2.0.0:
+  version "2.0.4"
+  resolved "https://registry.yarnpkg.com/obliterator/-/obliterator-2.0.4.tgz#fa650e019b2d075d745e44f1effeb13a2adbe816"
+  integrity sha512-lgHwxlxV1qIg1Eap7LgIeoBWIMFibOjbrYPIPJZcI1mmGAI2m3lNYpK12Y+GBdPQ0U1hRwSord7GIaawz962qQ==
+
+oboe@2.1.5:
+  version "2.1.5"
+  resolved "https://registry.yarnpkg.com/oboe/-/oboe-2.1.5.tgz#5554284c543a2266d7a38f17e073821fbde393cd"
+  integrity sha512-zRFWiF+FoicxEs3jNI/WYUrVEgA7DeET/InK0XQuudGHRg8iIob3cNPrJTKaz4004uaA9Pbe+Dwa8iluhjLZWA==
+  dependencies:
+    http-https "^1.0.0"
+
+on-finished@2.4.1:
+  version "2.4.1"
+  resolved "https://registry.yarnpkg.com/on-finished/-/on-finished-2.4.1.tgz#58c8c44116e54845ad57f14ab10b03533184ac3f"
+  integrity sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==
+  dependencies:
+    ee-first "1.1.1"
+
+once@1.x, once@^1.3.0, once@^1.3.1, once@^1.4.0:
+  version "1.4.0"
+  resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1"
+  integrity sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==
+  dependencies:
+    wrappy "1"
+
+onetime@^5.1.0:
+  version "5.1.2"
+  resolved "https://registry.yarnpkg.com/onetime/-/onetime-5.1.2.tgz#d0e96ebb56b07476df1dd9c4806e5237985ca45e"
+  integrity sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==
+  dependencies:
+    mimic-fn "^2.1.0"
+
+open@^8.4.0:
+  version "8.4.2"
+  resolved "https://registry.yarnpkg.com/open/-/open-8.4.2.tgz#5b5ffe2a8f793dcd2aad73e550cb87b59cb084f9"
+  integrity sha512-7x81NCL719oNbsq/3mh+hVrAWmFuEYUqrq/Iw3kUzH8ReypT9QQ0BLoJS7/G9k6N81XjW4qHWtjWwe/9eLy1EQ==
+  dependencies:
+    define-lazy-prop "^2.0.0"
+    is-docker "^2.1.1"
+    is-wsl "^2.2.0"
+
+opencollective-postinstall@^2.0.2:
+  version "2.0.3"
+  resolved "https://registry.yarnpkg.com/opencollective-postinstall/-/opencollective-postinstall-2.0.3.tgz#7a0fff978f6dbfa4d006238fbac98ed4198c3259"
+  integrity sha512-8AV/sCtuzUeTo8gQK5qDZzARrulB3egtLzFgteqB2tcT4Mw7B8Kt7JcDHmltjz6FOAHsvTevk70gZEbhM4ZS9Q==
+
+optionator@^0.8.1:
+  version "0.8.3"
+  resolved "https://registry.yarnpkg.com/optionator/-/optionator-0.8.3.tgz#84fa1d036fe9d3c7e21d99884b601167ec8fb495"
+  integrity sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==
+  dependencies:
+    deep-is "~0.1.3"
+    fast-levenshtein "~2.0.6"
+    levn "~0.3.0"
+    prelude-ls "~1.1.2"
+    type-check "~0.3.2"
+    word-wrap "~1.2.3"
+
+optionator@^0.9.3:
+  version "0.9.3"
+  resolved "https://registry.yarnpkg.com/optionator/-/optionator-0.9.3.tgz#007397d44ed1872fdc6ed31360190f81814e2c64"
+  integrity sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg==
+  dependencies:
+    "@aashutoshrathi/word-wrap" "^1.2.3"
+    deep-is "^0.1.3"
+    fast-levenshtein "^2.0.6"
+    levn "^0.4.1"
+    prelude-ls "^1.2.1"
+    type-check "^0.4.0"
+
+ora@^5.3.0:
+  version "5.4.1"
+  resolved "https://registry.yarnpkg.com/ora/-/ora-5.4.1.tgz#1b2678426af4ac4a509008e5e4ac9e9959db9e18"
+  integrity sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ==
+  dependencies:
+    bl "^4.1.0"
+    chalk "^4.1.0"
+    cli-cursor "^3.1.0"
+    cli-spinners "^2.5.0"
+    is-interactive "^1.0.0"
+    is-unicode-supported "^0.1.0"
+    log-symbols "^4.1.0"
+    strip-ansi "^6.0.0"
+    wcwidth "^1.0.1"
+
+ordinal@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/ordinal/-/ordinal-1.0.3.tgz#1a3c7726a61728112f50944ad7c35c06ae3a0d4d"
+  integrity sha512-cMddMgb2QElm8G7vdaa02jhUNbTSrhsgAGUz1OokD83uJTwSUn+nKoNoKVVaRa08yF6sgfO7Maou1+bgLd9rdQ==
+
+os-locale@^1.4.0:
+  version "1.4.0"
+  resolved "https://registry.yarnpkg.com/os-locale/-/os-locale-1.4.0.tgz#20f9f17ae29ed345e8bde583b13d2009803c14d9"
+  integrity sha512-PRT7ZORmwu2MEFt4/fv3Q+mEfN4zetKxufQrkShY2oGvUms9r8otu5HfdyIFHkYXjO7laNsoVGmM2MANfuTA8g==
+  dependencies:
+    lcid "^1.0.0"
+
+os-tmpdir@~1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/os-tmpdir/-/os-tmpdir-1.0.2.tgz#bbe67406c79aa85c5cfec766fe5734555dfa1274"
+  integrity sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==
+
+p-cancelable@^2.0.0:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/p-cancelable/-/p-cancelable-2.1.1.tgz#aab7fbd416582fa32a3db49859c122487c5ed2cf"
+  integrity sha512-BZOr3nRQHOntUjTrH8+Lh54smKHoHyur8We1V8DSMVrl5A2malOOwuJRnKRDjSnkoeBh4at6BwEnb5I7Jl31wg==
+
+p-cancelable@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/p-cancelable/-/p-cancelable-3.0.0.tgz#63826694b54d61ca1c20ebcb6d3ecf5e14cd8050"
+  integrity sha512-mlVgR3PGuzlo0MmTdk4cXqXWlwQDLnONTAg6sm62XkMJEiRxN3GL3SffkYvqwonbkJBcrI7Uvv5Zh9yjvn2iUw==
+
+p-limit@^1.1.0:
+  version "1.3.0"
+  resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-1.3.0.tgz#b86bd5f0c25690911c7590fcbfc2010d54b3ccb8"
+  integrity sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==
+  dependencies:
+    p-try "^1.0.0"
+
+p-limit@^2.0.0:
+  version "2.3.0"
+  resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-2.3.0.tgz#3dd33c647a214fdfffd835933eb086da0dc21db1"
+  integrity sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==
+  dependencies:
+    p-try "^2.0.0"
+
+p-limit@^3.0.2:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-3.1.0.tgz#e1daccbe78d0d1388ca18c64fea38e3e57e3706b"
+  integrity sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==
+  dependencies:
+    yocto-queue "^0.1.0"
+
+p-locate@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-2.0.0.tgz#20a0103b222a70c8fd39cc2e580680f3dde5ec43"
+  integrity sha512-nQja7m7gSKuewoVRen45CtVfODR3crN3goVQ0DDZ9N3yHxgpkuBhZqsaiotSQRrADUrne346peY7kT3TSACykg==
+  dependencies:
+    p-limit "^1.1.0"
+
+p-locate@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-3.0.0.tgz#322d69a05c0264b25997d9f40cd8a891ab0064a4"
+  integrity sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==
+  dependencies:
+    p-limit "^2.0.0"
+
+p-locate@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-5.0.0.tgz#83c8315c6785005e3bd021839411c9e110e6d834"
+  integrity sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==
+  dependencies:
+    p-limit "^3.0.2"
+
+p-map@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/p-map/-/p-map-4.0.0.tgz#bb2f95a5eda2ec168ec9274e06a747c3e2904d2b"
+  integrity sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==
+  dependencies:
+    aggregate-error "^3.0.0"
+
+p-try@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/p-try/-/p-try-1.0.0.tgz#cbc79cdbaf8fd4228e13f621f2b1a237c1b207b3"
+  integrity sha512-U1etNYuMJoIz3ZXSrrySFjsXQTWOx2/jdi86L+2pRvph/qMKL6sbcCYdH23fqsbm8TH2Gn0OybpT4eSFlCVHww==
+
+p-try@^2.0.0:
+  version "2.2.0"
+  resolved "https://registry.yarnpkg.com/p-try/-/p-try-2.2.0.tgz#cb2868540e313d61de58fafbe35ce9004d5540e6"
+  integrity sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==
+
+pako@^1.0.4:
+  version "1.0.11"
+  resolved "https://registry.yarnpkg.com/pako/-/pako-1.0.11.tgz#6c9599d340d54dfd3946380252a35705a6b992bf"
+  integrity sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==
+
+param-case@^2.1.0:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/param-case/-/param-case-2.1.1.tgz#df94fd8cf6531ecf75e6bef9a0858fbc72be2247"
+  integrity sha512-eQE845L6ot89sk2N8liD8HAuH4ca6Vvr7VWAWwt7+kvvG5aBcPmmphQ68JsEG2qa9n1TykS2DLeMt363AAH8/w==
+  dependencies:
+    no-case "^2.2.0"
+
+parent-module@^1.0.0:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/parent-module/-/parent-module-1.0.1.tgz#691d2709e78c79fae3a156622452d00762caaaa2"
+  integrity sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==
+  dependencies:
+    callsites "^3.0.0"
+
+parse-cache-control@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/parse-cache-control/-/parse-cache-control-1.0.1.tgz#8eeab3e54fa56920fe16ba38f77fa21aacc2d74e"
+  integrity sha512-60zvsJReQPX5/QP0Kzfd/VrpjScIQ7SHBW6bFCYfEP+fp0Eppr1SHhIO5nd1PjZtvclzSzES9D/p5nFJurwfWg==
+
+parse-headers@^2.0.0:
+  version "2.0.5"
+  resolved "https://registry.yarnpkg.com/parse-headers/-/parse-headers-2.0.5.tgz#069793f9356a54008571eb7f9761153e6c770da9"
+  integrity sha512-ft3iAoLOB/MlwbNXgzy43SWGP6sQki2jQvAyBg/zDFAgr9bfNWZIUj42Kw2eJIl8kEi4PbgE6U1Zau/HwI75HA==
+
+parse-json@^2.2.0:
+  version "2.2.0"
+  resolved "https://registry.yarnpkg.com/parse-json/-/parse-json-2.2.0.tgz#f480f40434ef80741f8469099f8dea18f55a4dc9"
+  integrity sha512-QR/GGaKCkhwk1ePQNYDRKYZ3mwU9ypsKhB0XyFnLQdomyEqk3e8wpW3V5Jp88zbxK4n5ST1nqo+g9juTpownhQ==
+  dependencies:
+    error-ex "^1.2.0"
+
+parse-json@^5.0.0:
+  version "5.2.0"
+  resolved "https://registry.yarnpkg.com/parse-json/-/parse-json-5.2.0.tgz#c76fc66dee54231c962b22bcc8a72cf2f99753cd"
+  integrity sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==
+  dependencies:
+    "@babel/code-frame" "^7.0.0"
+    error-ex "^1.3.1"
+    json-parse-even-better-errors "^2.3.0"
+    lines-and-columns "^1.1.6"
+
+parse5-htmlparser2-tree-adapter@^7.0.0:
+  version "7.0.0"
+  resolved "https://registry.yarnpkg.com/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.0.0.tgz#23c2cc233bcf09bb7beba8b8a69d46b08c62c2f1"
+  integrity sha512-B77tOZrqqfUfnVcOrUvfdLbz4pu4RopLD/4vmu3HUPswwTA8OH0EMW9BlWR2B0RCoiZRAHEUu7IxeP1Pd1UU+g==
+  dependencies:
+    domhandler "^5.0.2"
+    parse5 "^7.0.0"
+
+parse5@^7.0.0:
+  version "7.1.2"
+  resolved "https://registry.yarnpkg.com/parse5/-/parse5-7.1.2.tgz#0736bebbfd77793823240a23b7fc5e010b7f8e32"
+  integrity sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==
+  dependencies:
+    entities "^4.4.0"
+
+parseurl@~1.3.3:
+  version "1.3.3"
+  resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.3.tgz#9da19e7bee8d12dff0513ed5b76957793bc2e8d4"
+  integrity sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==
+
+pascal-case@^2.0.0:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/pascal-case/-/pascal-case-2.0.1.tgz#2d578d3455f660da65eca18ef95b4e0de912761e"
+  integrity sha512-qjS4s8rBOJa2Xm0jmxXiyh1+OFf6ekCWOvUaRgAQSktzlTbMotS0nmG9gyYAybCWBcuP4fsBeRCKNwGBnMe2OQ==
+  dependencies:
+    camel-case "^3.0.0"
+    upper-case-first "^1.1.0"
+
+path-case@^2.1.0:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/path-case/-/path-case-2.1.1.tgz#94b8037c372d3fe2906e465bb45e25d226e8eea5"
+  integrity sha512-Ou0N05MioItesaLr9q8TtHVWmJ6fxWdqKB2RohFmNWVyJ+2zeKIeDNWAN6B/Pe7wpzWChhZX6nONYmOnMeJQ/Q==
+  dependencies:
+    no-case "^2.2.0"
+
+path-exists@^2.0.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-2.1.0.tgz#0feb6c64f0fc518d9a754dd5efb62c7022761f4b"
+  integrity sha512-yTltuKuhtNeFJKa1PiRzfLAU5182q1y4Eb4XCJ3PBqyzEDkAZRzBrKKBct682ls9reBVHf9udYLN5Nd+K1B9BQ==
+  dependencies:
+    pinkie-promise "^2.0.0"
+
+path-exists@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-3.0.0.tgz#ce0ebeaa5f78cb18925ea7d810d7b59b010fd515"
+  integrity sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ==
+
+path-exists@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-4.0.0.tgz#513bdbe2d3b95d7762e8c1137efa195c6c61b5b3"
+  integrity sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==
+
+path-is-absolute@^1.0.0:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f"
+  integrity sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==
+
+path-key@^3.1.0:
+  version "3.1.1"
+  resolved "https://registry.yarnpkg.com/path-key/-/path-key-3.1.1.tgz#581f6ade658cbba65a0d3380de7753295054f375"
+  integrity sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==
+
+path-parse@^1.0.6, path-parse@^1.0.7:
+  version "1.0.7"
+  resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.7.tgz#fbc114b60ca42b30d9daf5858e4bd68bbedb6735"
+  integrity sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==
+
+path-to-regexp@0.1.7:
+  version "0.1.7"
+  resolved "https://registry.yarnpkg.com/path-to-regexp/-/path-to-regexp-0.1.7.tgz#df604178005f522f15eb4490e7247a1bfaa67f8c"
+  integrity sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ==
+
+path-type@^1.0.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/path-type/-/path-type-1.1.0.tgz#59c44f7ee491da704da415da5a4070ba4f8fe441"
+  integrity sha512-S4eENJz1pkiQn9Znv33Q+deTOKmbl+jj1Fl+qiP/vYezj+S8x+J3Uo0ISrx/QoEvIlOaDWJhPaRd1flJ9HXZqg==
+  dependencies:
+    graceful-fs "^4.1.2"
+    pify "^2.0.0"
+    pinkie-promise "^2.0.0"
+
+path-type@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/path-type/-/path-type-4.0.0.tgz#84ed01c0a7ba380afe09d90a8c180dcd9d03043b"
+  integrity sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==
+
+pathval@^1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/pathval/-/pathval-1.1.1.tgz#8534e77a77ce7ac5a2512ea21e0fdb8fcf6c3d8d"
+  integrity sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ==
+
+pbkdf2@^3.0.17:
+  version "3.1.2"
+  resolved "https://registry.yarnpkg.com/pbkdf2/-/pbkdf2-3.1.2.tgz#dd822aa0887580e52f1a039dc3eda108efae3075"
+  integrity sha512-iuh7L6jA7JEGu2WxDwtQP1ddOpaJNC4KlDEFfdQajSGgGPNi4OyDc2R7QnbY2bR9QjBVGwgvTdNJZoE7RaxUMA==
+  dependencies:
+    create-hash "^1.1.2"
+    create-hmac "^1.1.4"
+    ripemd160 "^2.0.1"
+    safe-buffer "^5.0.1"
+    sha.js "^2.4.8"
+
+performance-now@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/performance-now/-/performance-now-2.1.0.tgz#6309f4e0e5fa913ec1c69307ae364b4b377c9e7b"
+  integrity sha512-7EAHlyLHI56VEIdK57uwHdHKIaAGbnXPiw0yWbarQZOKaKpvUIgW0jWRVLiatnM+XXlSwsanIBH/hzGMJulMow==
+
+picomatch@^2.0.4, picomatch@^2.2.1, picomatch@^2.3.1:
+  version "2.3.1"
+  resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.1.tgz#3ba3833733646d9d3e4995946c1365a67fb07a42"
+  integrity sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==
+
+pify@^2.0.0:
+  version "2.3.0"
+  resolved "https://registry.yarnpkg.com/pify/-/pify-2.3.0.tgz#ed141a6ac043a849ea588498e7dca8b15330e90c"
+  integrity sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==
+
+pify@^4.0.1:
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/pify/-/pify-4.0.1.tgz#4b2cd25c50d598735c50292224fd8c6df41e3231"
+  integrity sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==
+
+pinkie-promise@^2.0.0:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/pinkie-promise/-/pinkie-promise-2.0.1.tgz#2135d6dfa7a358c069ac9b178776288228450ffa"
+  integrity sha512-0Gni6D4UcLTbv9c57DfxDGdr41XfgUjqWZu492f0cIGr16zDU06BWP/RAEvOuo7CQ0CNjHaLlM59YJJFm3NWlw==
+  dependencies:
+    pinkie "^2.0.0"
+
+pinkie@^2.0.0:
+  version "2.0.4"
+  resolved "https://registry.yarnpkg.com/pinkie/-/pinkie-2.0.4.tgz#72556b80cfa0d48a974e80e77248e80ed4f7f870"
+  integrity sha512-MnUuEycAemtSaeFSjXKW/aroV7akBbY+Sv+RkyqFjgAe73F+MR0TBWKBRDkmfWq/HiFmdavfZ1G7h4SPZXaCSg==
+
+pkg-dir@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/pkg-dir/-/pkg-dir-5.0.0.tgz#a02d6aebe6ba133a928f74aec20bafdfe6b8e760"
+  integrity sha512-NPE8TDbzl/3YQYY7CSS228s3g2ollTFnc+Qi3tqmqJp9Vg2ovUpixcJEo2HJScN2Ez+kEaal6y70c0ehqJBJeA==
+  dependencies:
+    find-up "^5.0.0"
+
+please-upgrade-node@^3.2.0:
+  version "3.2.0"
+  resolved "https://registry.yarnpkg.com/please-upgrade-node/-/please-upgrade-node-3.2.0.tgz#aeddd3f994c933e4ad98b99d9a556efa0e2fe942"
+  integrity sha512-gQR3WpIgNIKwBMVLkpMUeR3e1/E1y42bqDQZfql+kDeXd8COYfM8PQA4X6y7a8u9Ua9FHmsrrmirW2vHs45hWg==
+  dependencies:
+    semver-compare "^1.0.0"
+
+pluralize@^8.0.0:
+  version "8.0.0"
+  resolved "https://registry.yarnpkg.com/pluralize/-/pluralize-8.0.0.tgz#1a6fa16a38d12a1901e0320fa017051c539ce3b1"
+  integrity sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA==
+
+prelude-ls@^1.2.1:
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.2.1.tgz#debc6489d7a6e6b0e7611888cec880337d316396"
+  integrity sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==
+
+prelude-ls@~1.1.2:
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.1.2.tgz#21932a549f5e52ffd9a827f570e04be62a97da54"
+  integrity sha512-ESF23V4SKG6lVSGZgYNpbsiaAkdab6ZgOxe52p7+Kid3W3u3bxR4Vfd/o21dmN7jSt0IwgZ4v5MUd26FEtXE9w==
+
+prettier-linter-helpers@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/prettier-linter-helpers/-/prettier-linter-helpers-1.0.0.tgz#d23d41fe1375646de2d0104d3454a3008802cf7b"
+  integrity sha512-GbK2cP9nraSSUF9N2XwUwqfzlAFlMNYYl+ShE/V+H8a9uNl/oUqB1w2EL54Jh0OlyRSd8RfWYJ3coVS4TROP2w==
+  dependencies:
+    fast-diff "^1.1.2"
+
+prettier-plugin-solidity@^1.1.3:
+  version "1.1.3"
+  resolved "https://registry.yarnpkg.com/prettier-plugin-solidity/-/prettier-plugin-solidity-1.1.3.tgz#9a35124f578404caf617634a8cab80862d726cba"
+  integrity sha512-fQ9yucPi2sBbA2U2Xjh6m4isUTJ7S7QLc/XDDsktqqxYfTwdYKJ0EnnywXHwCGAaYbQNK+HIYPL1OemxuMsgeg==
+  dependencies:
+    "@solidity-parser/parser" "^0.16.0"
+    semver "^7.3.8"
+    solidity-comments-extractor "^0.0.7"
+
+prettier@^2.3.1, prettier@^2.8.3, prettier@^2.8.7:
+  version "2.8.8"
+  resolved "https://registry.yarnpkg.com/prettier/-/prettier-2.8.8.tgz#e8c5d7e98a4305ffe3de2e1fc4aca1a71c28b1da"
+  integrity sha512-tdN8qQGvNjw4CHbY+XXk0JgCXn9QiF21a55rBe5LJAU+kDyC4WQn4+awm2Xfk2lQMk5fKup9XgzTZtGkjBdP9Q==
+
+process-nextick-args@~2.0.0:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-2.0.1.tgz#7820d9b16120cc55ca9ae7792680ae7dba6d7fe2"
+  integrity sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==
+
+process@^0.11.10:
+  version "0.11.10"
+  resolved "https://registry.yarnpkg.com/process/-/process-0.11.10.tgz#7332300e840161bda3e69a1d1d91a7d4bc16f182"
+  integrity sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A==
+
+promise@^8.0.0:
+  version "8.3.0"
+  resolved "https://registry.yarnpkg.com/promise/-/promise-8.3.0.tgz#8cb333d1edeb61ef23869fbb8a4ea0279ab60e0a"
+  integrity sha512-rZPNPKTOYVNEEKFaq1HqTgOwZD+4/YHS5ukLzQCypkj+OkYx7iv0mA91lJlpPPZ8vMau3IIGj5Qlwrx+8iiSmg==
+  dependencies:
+    asap "~2.0.6"
+
+prompts@^2.4.2:
+  version "2.4.2"
+  resolved "https://registry.yarnpkg.com/prompts/-/prompts-2.4.2.tgz#7b57e73b3a48029ad10ebd44f74b01722a4cb069"
+  integrity sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==
+  dependencies:
+    kleur "^3.0.3"
+    sisteransi "^1.0.5"
+
+proxy-addr@~2.0.7:
+  version "2.0.7"
+  resolved "https://registry.yarnpkg.com/proxy-addr/-/proxy-addr-2.0.7.tgz#f19fe69ceab311eeb94b42e70e8c2070f9ba1025"
+  integrity sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==
+  dependencies:
+    forwarded "0.2.0"
+    ipaddr.js "1.9.1"
+
+psl@^1.1.28:
+  version "1.9.0"
+  resolved "https://registry.yarnpkg.com/psl/-/psl-1.9.0.tgz#d0df2a137f00794565fcaf3b2c00cd09f8d5a5a7"
+  integrity sha512-E/ZsdU4HLs/68gYzgGTkMicWTLPdAftJLfJFlLUAAKZGkStNU72sZjT66SnMDVOfOWY/YAoiD7Jxa9iHvngcag==
+
+pump@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/pump/-/pump-3.0.0.tgz#b4a2116815bde2f4e1ea602354e8c75565107a64"
+  integrity sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==
+  dependencies:
+    end-of-stream "^1.1.0"
+    once "^1.3.1"
+
+punycode@2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.1.0.tgz#5f863edc89b96db09074bad7947bf09056ca4e7d"
+  integrity sha512-Yxz2kRwT90aPiWEMHVYnEf4+rhwF1tBmmZ4KepCP+Wkium9JxtWnUm1nqGwpiAHr/tnTSeHqr3wb++jgSkXjhA==
+
+punycode@^2.1.0, punycode@^2.1.1:
+  version "2.3.0"
+  resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.3.0.tgz#f67fa67c94da8f4d0cfff981aee4118064199b8f"
+  integrity sha512-rRV+zQD8tVFys26lAGR9WUuS4iUAngJScM+ZRSKtvl5tKeZ2t5bvdNFdNHBW9FWR4guGHlgmsZ1G7BSm2wTbuA==
+
+pure-rand@^5.0.1:
+  version "5.0.5"
+  resolved "https://registry.yarnpkg.com/pure-rand/-/pure-rand-5.0.5.tgz#bda2a7f6a1fc0f284d78d78ca5902f26f2ad35cf"
+  integrity sha512-BwQpbqxSCBJVpamI6ydzcKqyFmnd5msMWUGvzXLm1aXvusbbgkbOto/EUPM00hjveJEaJtdbhUjKSzWRhQVkaw==
+
+qs@6.11.0:
+  version "6.11.0"
+  resolved "https://registry.yarnpkg.com/qs/-/qs-6.11.0.tgz#fd0d963446f7a65e1367e01abd85429453f0c37a"
+  integrity sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q==
+  dependencies:
+    side-channel "^1.0.4"
+
+qs@^6.4.0, qs@^6.7.0, qs@^6.9.4:
+  version "6.11.2"
+  resolved "https://registry.yarnpkg.com/qs/-/qs-6.11.2.tgz#64bea51f12c1f5da1bc01496f48ffcff7c69d7d9"
+  integrity sha512-tDNIz22aBzCDxLtVH++VnTfzxlfeK5CbqohpSqpJgj1Wg/cQbStNAz3NuqCs5vV+pjBsK4x4pN9HlVh7rcYRiA==
+  dependencies:
+    side-channel "^1.0.4"
+
+qs@~6.5.2:
+  version "6.5.3"
+  resolved "https://registry.yarnpkg.com/qs/-/qs-6.5.3.tgz#3aeeffc91967ef6e35c0e488ef46fb296ab76aad"
+  integrity sha512-qxXIEh4pCGfHICj1mAJQ2/2XVZkjCDTcEgfoSQxc/fYivUZxTkk7L3bDBJSoNrEzXI17oUO5Dp07ktqE5KzczA==
+
+query-string@^5.0.1:
+  version "5.1.1"
+  resolved "https://registry.yarnpkg.com/query-string/-/query-string-5.1.1.tgz#a78c012b71c17e05f2e3fa2319dd330682efb3cb"
+  integrity sha512-gjWOsm2SoGlgLEdAGt7a6slVOk9mGiXmPFMqrEhLQ68rhQuBnpfs3+EmlvqKyxnCo9/PPlF+9MtY02S1aFg+Jw==
+  dependencies:
+    decode-uri-component "^0.2.0"
+    object-assign "^4.1.0"
+    strict-uri-encode "^1.0.0"
+
+queue-microtask@^1.2.2, queue-microtask@^1.2.3:
+  version "1.2.3"
+  resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.3.tgz#4929228bbc724dfac43e0efb058caf7b6cfb6243"
+  integrity sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==
+
+quick-lru@^5.1.1:
+  version "5.1.1"
+  resolved "https://registry.yarnpkg.com/quick-lru/-/quick-lru-5.1.1.tgz#366493e6b3e42a3a6885e2e99d18f80fb7a8c932"
+  integrity sha512-WuyALRjWPDGtt/wzJiadO5AXY+8hZ80hVpe6MyivgraREW751X3SbhRvG3eLKOYN+8VEvqLcf3wdnt44Z4S4SA==
+
+randombytes@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/randombytes/-/randombytes-2.1.0.tgz#df6f84372f0270dc65cdf6291349ab7a473d4f2a"
+  integrity sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==
+  dependencies:
+    safe-buffer "^5.1.0"
+
+range-parser@~1.2.1:
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/range-parser/-/range-parser-1.2.1.tgz#3cf37023d199e1c24d1a55b84800c2f3e6468031"
+  integrity sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==
+
+raw-body@2.5.1:
+  version "2.5.1"
+  resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.5.1.tgz#fe1b1628b181b700215e5fd42389f98b71392857"
+  integrity sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig==
+  dependencies:
+    bytes "3.1.2"
+    http-errors "2.0.0"
+    iconv-lite "0.4.24"
+    unpipe "1.0.0"
+
+raw-body@2.5.2, raw-body@^2.4.1:
+  version "2.5.2"
+  resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.5.2.tgz#99febd83b90e08975087e8f1f9419a149366b68a"
+  integrity sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==
+  dependencies:
+    bytes "3.1.2"
+    http-errors "2.0.0"
+    iconv-lite "0.4.24"
+    unpipe "1.0.0"
+
+read-pkg-up@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/read-pkg-up/-/read-pkg-up-1.0.1.tgz#9d63c13276c065918d57f002a57f40a1b643fb02"
+  integrity sha512-WD9MTlNtI55IwYUS27iHh9tK3YoIVhxis8yKhLpTqWtml739uXc9NWTpxoHkfZf3+DkCCsXox94/VWZniuZm6A==
+  dependencies:
+    find-up "^1.0.0"
+    read-pkg "^1.0.0"
+
+read-pkg@^1.0.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/read-pkg/-/read-pkg-1.1.0.tgz#f5ffaa5ecd29cb31c0474bca7d756b6bb29e3f28"
+  integrity sha512-7BGwRHqt4s/uVbuyoeejRn4YmFnYZiFl4AuaeXHlgZf3sONF0SOGlxs2Pw8g6hCKupo08RafIO5YXFNOKTfwsQ==
+  dependencies:
+    load-json-file "^1.0.0"
+    normalize-package-data "^2.3.2"
+    path-type "^1.0.0"
+
+readable-stream@^2.2.2:
+  version "2.3.8"
+  resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.3.8.tgz#91125e8042bba1b9887f49345f6277027ce8be9b"
+  integrity sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==
+  dependencies:
+    core-util-is "~1.0.0"
+    inherits "~2.0.3"
+    isarray "~1.0.0"
+    process-nextick-args "~2.0.0"
+    safe-buffer "~5.1.1"
+    string_decoder "~1.1.1"
+    util-deprecate "~1.0.1"
+
+readable-stream@^3.4.0, readable-stream@^3.6.0:
+  version "3.6.2"
+  resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.2.tgz#56a9b36ea965c00c5a93ef31eb111a0f11056967"
+  integrity sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==
+  dependencies:
+    inherits "^2.0.3"
+    string_decoder "^1.1.1"
+    util-deprecate "^1.0.1"
+
+readdirp@~3.2.0:
+  version "3.2.0"
+  resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-3.2.0.tgz#c30c33352b12c96dfb4b895421a49fd5a9593839"
+  integrity sha512-crk4Qu3pmXwgxdSgGhgA/eXiJAPQiX4GMOZZMXnqKxHX7TaoL+3gQVo/WeuAiogr07DpnfjIMpXXa+PAIvwPGQ==
+  dependencies:
+    picomatch "^2.0.4"
+
+readdirp@~3.6.0:
+  version "3.6.0"
+  resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-3.6.0.tgz#74a370bd857116e245b29cc97340cd431a02a6c7"
+  integrity sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==
+  dependencies:
+    picomatch "^2.2.1"
+
+rechoir@^0.6.2:
+  version "0.6.2"
+  resolved "https://registry.yarnpkg.com/rechoir/-/rechoir-0.6.2.tgz#85204b54dba82d5742e28c96756ef43af50e3384"
+  integrity sha512-HFM8rkZ+i3zrV+4LQjwQ0W+ez98pApMGM3HUrN04j3CqzPOzl9nmP15Y8YXNm8QHGv/eacOVEjqhmWpkRV0NAw==
+  dependencies:
+    resolve "^1.1.6"
+
+recursive-readdir@^2.2.2:
+  version "2.2.3"
+  resolved "https://registry.yarnpkg.com/recursive-readdir/-/recursive-readdir-2.2.3.tgz#e726f328c0d69153bcabd5c322d3195252379372"
+  integrity sha512-8HrF5ZsXk5FAH9dgsx3BlUer73nIhuj+9OrQwEbLTPOBzGkL1lsFCR01am+v+0m2Cmbs1nP12hLDl5FA7EszKA==
+  dependencies:
+    minimatch "^3.0.5"
+
+reduce-flatten@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/reduce-flatten/-/reduce-flatten-2.0.0.tgz#734fd84e65f375d7ca4465c69798c25c9d10ae27"
+  integrity sha512-EJ4UNY/U1t2P/2k6oqotuX2Cc3T6nxJwsM0N0asT7dhrtH1ltUxDn4NalSYmPE2rCkVpcf/X6R0wDwcFpzhd4w==
+
+regenerator-runtime@^0.14.0:
+  version "0.14.0"
+  resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.14.0.tgz#5e19d68eb12d486f797e15a3c6a918f7cec5eb45"
+  integrity sha512-srw17NI0TUWHuGa5CFGGmhfNIeja30WMBfbslPNhf6JrqQlLN5gcrvig1oqPxiVaXb0oW0XRKtH6Nngs5lKCIA==
+
+regexp.prototype.flags@^1.5.0:
+  version "1.5.0"
+  resolved "https://registry.yarnpkg.com/regexp.prototype.flags/-/regexp.prototype.flags-1.5.0.tgz#fe7ce25e7e4cca8db37b6634c8a2c7009199b9cb"
+  integrity sha512-0SutC3pNudRKgquxGoRGIz946MZVHqbNfPjBdxeOhBrdgDKlRoXmYLQN9xRbrR09ZXWeGAdPuif7egofn6v5LA==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.2.0"
+    functions-have-names "^1.2.3"
+
+req-cwd@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/req-cwd/-/req-cwd-2.0.0.tgz#d4082b4d44598036640fb73ddea01ed53db49ebc"
+  integrity sha512-ueoIoLo1OfB6b05COxAA9UpeoscNpYyM+BqYlA7H6LVF4hKGPXQQSSaD2YmvDVJMkk4UDpAHIeU1zG53IqjvlQ==
+  dependencies:
+    req-from "^2.0.0"
+
+req-from@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/req-from/-/req-from-2.0.0.tgz#d74188e47f93796f4aa71df6ee35ae689f3e0e70"
+  integrity sha512-LzTfEVDVQHBRfjOUMgNBA+V6DWsSnoeKzf42J7l0xa/B4jyPOuuF5MlNSmomLNGemWTnV2TIdjSSLnEn95fOQA==
+  dependencies:
+    resolve-from "^3.0.0"
+
+request-promise-core@1.1.4:
+  version "1.1.4"
+  resolved "https://registry.yarnpkg.com/request-promise-core/-/request-promise-core-1.1.4.tgz#3eedd4223208d419867b78ce815167d10593a22f"
+  integrity sha512-TTbAfBBRdWD7aNNOoVOBH4pN/KigV6LyapYNNlAPA8JwbovRti1E88m3sYAwsLi5ryhPKsE9APwnjFTgdUjTpw==
+  dependencies:
+    lodash "^4.17.19"
+
+request-promise-native@^1.0.5:
+  version "1.0.9"
+  resolved "https://registry.yarnpkg.com/request-promise-native/-/request-promise-native-1.0.9.tgz#e407120526a5efdc9a39b28a5679bf47b9d9dc28"
+  integrity sha512-wcW+sIUiWnKgNY0dqCpOZkUbF/I+YPi+f09JZIDa39Ec+q82CpSYniDp+ISgTTbKmnpJWASeJBPZmoxH84wt3g==
+  dependencies:
+    request-promise-core "1.1.4"
+    stealthy-require "^1.1.1"
+    tough-cookie "^2.3.3"
+
+request@^2.79.0, request@^2.88.0:
+  version "2.88.2"
+  resolved "https://registry.yarnpkg.com/request/-/request-2.88.2.tgz#d73c918731cb5a87da047e207234146f664d12b3"
+  integrity sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==
+  dependencies:
+    aws-sign2 "~0.7.0"
+    aws4 "^1.8.0"
+    caseless "~0.12.0"
+    combined-stream "~1.0.6"
+    extend "~3.0.2"
+    forever-agent "~0.6.1"
+    form-data "~2.3.2"
+    har-validator "~5.1.3"
+    http-signature "~1.2.0"
+    is-typedarray "~1.0.0"
+    isstream "~0.1.2"
+    json-stringify-safe "~5.0.1"
+    mime-types "~2.1.19"
+    oauth-sign "~0.9.0"
+    performance-now "^2.1.0"
+    qs "~6.5.2"
+    safe-buffer "^5.1.2"
+    tough-cookie "~2.5.0"
+    tunnel-agent "^0.6.0"
+    uuid "^3.3.2"
+
+require-directory@^2.1.1:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/require-directory/-/require-directory-2.1.1.tgz#8c64ad5fd30dab1c976e2344ffe7f792a6a6df42"
+  integrity sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==
+
+require-from-string@^1.1.0:
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/require-from-string/-/require-from-string-1.2.1.tgz#529c9ccef27380adfec9a2f965b649bbee636418"
+  integrity sha512-H7AkJWMobeskkttHyhTVtS0fxpFLjxhbfMa6Bk3wimP7sdPRGL3EyCg3sAQenFfAe+xQ+oAc85Nmtvq0ROM83Q==
+
+require-from-string@^2.0.0, require-from-string@^2.0.2:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/require-from-string/-/require-from-string-2.0.2.tgz#89a7fdd938261267318eafe14f9c32e598c36909"
+  integrity sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==
+
+require-main-filename@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/require-main-filename/-/require-main-filename-1.0.1.tgz#97f717b69d48784f5f526a6c5aa8ffdda055a4d1"
+  integrity sha512-IqSUtOVP4ksd1C/ej5zeEh/BIP2ajqpn8c5x+q99gvcIG/Qf0cud5raVnE/Dwd0ua9TXYDoDc0RE5hBSdz22Ug==
+
+require-main-filename@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/require-main-filename/-/require-main-filename-2.0.0.tgz#d0b329ecc7cc0f61649f62215be69af54aa8989b"
+  integrity sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==
+
+resolve-alpn@^1.0.0, resolve-alpn@^1.2.0:
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/resolve-alpn/-/resolve-alpn-1.2.1.tgz#b7adbdac3546aaaec20b45e7d8265927072726f9"
+  integrity sha512-0a1F4l73/ZFZOakJnQ3FvkJ2+gSTQWz/r2KE5OdDY0TxPm5h4GkqkWWfM47T7HsbnOtcJVEF4epCVy6u7Q3K+g==
+
+resolve-from@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-3.0.0.tgz#b22c7af7d9d6881bc8b6e653335eebcb0a188748"
+  integrity sha512-GnlH6vxLymXJNMBo7XP1fJIzBFbdYt49CuTwmB/6N53t+kMPRMFKz783LlQ4tv28XoQfMWinAJX6WCGf2IlaIw==
+
+resolve-from@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-4.0.0.tgz#4abcd852ad32dd7baabfe9b40e00a36db5f392e6"
+  integrity sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==
+
+resolve@1.1.x:
+  version "1.1.7"
+  resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.1.7.tgz#203114d82ad2c5ed9e8e0411b3932875e889e97b"
+  integrity sha512-9znBF0vBcaSN3W2j7wKvdERPwqTxSpCq+if5C0WoTCyV9n24rua28jeuQ2pL/HOf+yUe/Mef+H/5p60K0Id3bg==
+
+resolve@1.17.0:
+  version "1.17.0"
+  resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.17.0.tgz#b25941b54968231cc2d1bb76a79cb7f2c0bf8444"
+  integrity sha512-ic+7JYiV8Vi2yzQGFWOkiZD5Z9z7O2Zhm9XMaTxdJExKasieFCr+yXZ/WmXsckHiKl12ar0y6XiXDx3m4RHn1w==
+  dependencies:
+    path-parse "^1.0.6"
+
+resolve@^1.1.6, resolve@^1.10.0, resolve@^1.22.3, resolve@^1.22.4:
+  version "1.22.4"
+  resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.22.4.tgz#1dc40df46554cdaf8948a486a10f6ba1e2026c34"
+  integrity sha512-PXNdCiPqDqeUou+w1C2eTQbNfxKSuMxqTCuvlmmMsk1NWHL5fRrhY6Pl0qEYYc6+QqGClco1Qj8XnjPego4wfg==
+  dependencies:
+    is-core-module "^2.13.0"
+    path-parse "^1.0.7"
+    supports-preserve-symlinks-flag "^1.0.0"
+
+responselike@^2.0.0:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/responselike/-/responselike-2.0.1.tgz#9a0bc8fdc252f3fb1cca68b016591059ba1422bc"
+  integrity sha512-4gl03wn3hj1HP3yzgdI7d3lCkF95F21Pz4BPGvKHinyQzALR5CapwC8yIi0Rh58DEMQ/SguC03wFj2k0M/mHhw==
+  dependencies:
+    lowercase-keys "^2.0.0"
+
+restore-cursor@^3.1.0:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-3.1.0.tgz#39f67c54b3a7a58cea5236d95cf0034239631f7e"
+  integrity sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==
+  dependencies:
+    onetime "^5.1.0"
+    signal-exit "^3.0.2"
+
+reusify@^1.0.4:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/reusify/-/reusify-1.0.4.tgz#90da382b1e126efc02146e90845a88db12925d76"
+  integrity sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==
+
+rimraf@^2.2.8:
+  version "2.7.1"
+  resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-2.7.1.tgz#35797f13a7fdadc566142c29d4f07ccad483e3ec"
+  integrity sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==
+  dependencies:
+    glob "^7.1.3"
+
+rimraf@^3.0.2:
+  version "3.0.2"
+  resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-3.0.2.tgz#f1a5402ba6220ad52cc1282bac1ae3aa49fd061a"
+  integrity sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==
+  dependencies:
+    glob "^7.1.3"
+
+ripemd160-min@0.0.6:
+  version "0.0.6"
+  resolved "https://registry.yarnpkg.com/ripemd160-min/-/ripemd160-min-0.0.6.tgz#a904b77658114474d02503e819dcc55853b67e62"
+  integrity sha512-+GcJgQivhs6S9qvLogusiTcS9kQUfgR75whKuy5jIhuiOfQuJ8fjqxV6EGD5duH1Y/FawFUMtMhyeq3Fbnib8A==
+
+ripemd160@^2.0.0, ripemd160@^2.0.1, ripemd160@^2.0.2:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/ripemd160/-/ripemd160-2.0.2.tgz#a1c1a6f624751577ba5d07914cbc92850585890c"
+  integrity sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==
+  dependencies:
+    hash-base "^3.0.0"
+    inherits "^2.0.1"
+
+rlp@^2.2.3, rlp@^2.2.4:
+  version "2.2.7"
+  resolved "https://registry.yarnpkg.com/rlp/-/rlp-2.2.7.tgz#33f31c4afac81124ac4b283e2bd4d9720b30beaf"
+  integrity sha512-d5gdPmgQ0Z+AklL2NVXr/IoSjNZFfTVvQWzL/AM2AOcSzYP2xjlb0AC8YyCLc41MSNf6P6QVtjgPdmVtzb+4lQ==
+  dependencies:
+    bn.js "^5.2.0"
+
+run-parallel-limit@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/run-parallel-limit/-/run-parallel-limit-1.1.0.tgz#be80e936f5768623a38a963262d6bef8ff11e7ba"
+  integrity sha512-jJA7irRNM91jaKc3Hcl1npHsFLOXOoTkPCUL1JEa1R82O2miplXXRaGdjW/KM/98YQWDhJLiSs793CnXfblJUw==
+  dependencies:
+    queue-microtask "^1.2.2"
+
+run-parallel@^1.1.9:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/run-parallel/-/run-parallel-1.2.0.tgz#66d1368da7bdf921eb9d95bd1a9229e7f21a43ee"
+  integrity sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==
+  dependencies:
+    queue-microtask "^1.2.2"
+
+rustbn.js@~0.2.0:
+  version "0.2.0"
+  resolved "https://registry.yarnpkg.com/rustbn.js/-/rustbn.js-0.2.0.tgz#8082cb886e707155fd1cb6f23bd591ab8d55d0ca"
+  integrity sha512-4VlvkRUuCJvr2J6Y0ImW7NvTCriMi7ErOAqWk1y69vAdoNIzCF3yPmgeNzx+RQTLEDFq5sHfscn1MwHxP9hNfA==
+
+safe-array-concat@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/safe-array-concat/-/safe-array-concat-1.0.0.tgz#2064223cba3c08d2ee05148eedbc563cd6d84060"
+  integrity sha512-9dVEFruWIsnie89yym+xWTAYASdpw3CJV7Li/6zBewGf9z2i1j31rP6jnY0pHEO4QZh6N0K11bFjWmdR8UGdPQ==
+  dependencies:
+    call-bind "^1.0.2"
+    get-intrinsic "^1.2.0"
+    has-symbols "^1.0.3"
+    isarray "^2.0.5"
+
+safe-buffer@5.2.1, safe-buffer@^5.0.1, safe-buffer@^5.1.0, safe-buffer@^5.1.1, safe-buffer@^5.1.2, safe-buffer@^5.2.0, safe-buffer@^5.2.1, safe-buffer@~5.2.0:
+  version "5.2.1"
+  resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6"
+  integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==
+
+safe-buffer@~5.1.0, safe-buffer@~5.1.1:
+  version "5.1.2"
+  resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.2.tgz#991ec69d296e0313747d59bdfd2b745c35f8828d"
+  integrity sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==
+
+safe-regex-test@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/safe-regex-test/-/safe-regex-test-1.0.0.tgz#793b874d524eb3640d1873aad03596db2d4f2295"
+  integrity sha512-JBUUzyOgEwXQY1NuPtvcj/qcBDbDmEvWufhlnXZIm75DEHp+afM1r1ujJpJsV/gSM4t59tpDyPi1sd6ZaPFfsA==
+  dependencies:
+    call-bind "^1.0.2"
+    get-intrinsic "^1.1.3"
+    is-regex "^1.1.4"
+
+"safer-buffer@>= 2.1.2 < 3", safer-buffer@^2.0.2, safer-buffer@^2.1.0, safer-buffer@~2.1.0:
+  version "2.1.2"
+  resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a"
+  integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==
+
+sc-istanbul@^0.4.5:
+  version "0.4.6"
+  resolved "https://registry.yarnpkg.com/sc-istanbul/-/sc-istanbul-0.4.6.tgz#cf6784355ff2076f92d70d59047d71c13703e839"
+  integrity sha512-qJFF/8tW/zJsbyfh/iT/ZM5QNHE3CXxtLJbZsL+CzdJLBsPD7SedJZoUA4d8iAcN2IoMp/Dx80shOOd2x96X/g==
+  dependencies:
+    abbrev "1.0.x"
+    async "1.x"
+    escodegen "1.8.x"
+    esprima "2.7.x"
+    glob "^5.0.15"
+    handlebars "^4.0.1"
+    js-yaml "3.x"
+    mkdirp "0.5.x"
+    nopt "3.x"
+    once "1.x"
+    resolve "1.1.x"
+    supports-color "^3.1.0"
+    which "^1.1.1"
+    wordwrap "^1.0.0"
+
+scrypt-js@2.0.4:
+  version "2.0.4"
+  resolved "https://registry.yarnpkg.com/scrypt-js/-/scrypt-js-2.0.4.tgz#32f8c5149f0797672e551c07e230f834b6af5f16"
+  integrity sha512-4KsaGcPnuhtCZQCxFxN3GVYIhKFPTdLd8PLC552XwbMndtD0cjRFAhDuuydXQ0h08ZfPgzqe6EKHozpuH74iDw==
+
+scrypt-js@3.0.1, scrypt-js@^3.0.0, scrypt-js@^3.0.1:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/scrypt-js/-/scrypt-js-3.0.1.tgz#d314a57c2aef69d1ad98a138a21fe9eafa9ee312"
+  integrity sha512-cdwTTnqPu0Hyvf5in5asVdZocVDTNRmR7XEcJuIzMjJeSHybHl7vpB66AzwTaIg6CLSbtjcxc8fqcySfnTkccA==
+
+secp256k1@4.0.3, secp256k1@^4.0.1:
+  version "4.0.3"
+  resolved "https://registry.yarnpkg.com/secp256k1/-/secp256k1-4.0.3.tgz#c4559ecd1b8d3c1827ed2d1b94190d69ce267303"
+  integrity sha512-NLZVf+ROMxwtEj3Xa562qgv2BK5e2WNmXPiOdVIPLgs6lyTzMvBq0aWTYMI5XCP9jZMVKOcqZLw/Wc4vDkuxhA==
+  dependencies:
+    elliptic "^6.5.4"
+    node-addon-api "^2.0.0"
+    node-gyp-build "^4.2.0"
+
+semver-compare@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/semver-compare/-/semver-compare-1.0.0.tgz#0dee216a1c941ab37e9efb1788f6afc5ff5537fc"
+  integrity sha512-YM3/ITh2MJ5MtzaM429anh+x2jiLVjqILF4m4oyQB18W7Ggea7BfqdH/wGMK7dDiMghv/6WG7znWMwUDzJiXow==
+
+semver-regex@^3.1.2:
+  version "3.1.4"
+  resolved "https://registry.yarnpkg.com/semver-regex/-/semver-regex-3.1.4.tgz#13053c0d4aa11d070a2f2872b6b1e3ae1e1971b4"
+  integrity sha512-6IiqeZNgq01qGf0TId0t3NvKzSvUsjcpdEO3AQNeIjR6A2+ckTnQlDpl4qu1bjRv0RzN3FP9hzFmws3lKqRWkA==
+
+"semver@2 || 3 || 4 || 5", semver@^5.3.0, semver@^5.5.0, semver@^5.7.0:
+  version "5.7.2"
+  resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.2.tgz#48d55db737c3287cd4835e17fa13feace1c41ef8"
+  integrity sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==
+
+semver@7.5.2:
+  version "7.5.2"
+  resolved "https://registry.yarnpkg.com/semver/-/semver-7.5.2.tgz#5b851e66d1be07c1cdaf37dfc856f543325a2beb"
+  integrity sha512-SoftuTROv/cRjCze/scjGyiDtcUyxw1rgYQSZY7XTmtR5hX+dm76iDbTH8TkLPHCQmlbQVSSbNZCPM2hb0knnQ==
+  dependencies:
+    lru-cache "^6.0.0"
+
+semver@^6.3.0, semver@^6.3.1:
+  version "6.3.1"
+  resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4"
+  integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==
+
+semver@^7.3.4, semver@^7.3.7, semver@^7.3.8, semver@^7.5.2:
+  version "7.5.4"
+  resolved "https://registry.yarnpkg.com/semver/-/semver-7.5.4.tgz#483986ec4ed38e1c6c48c34894a9182dbff68a6e"
+  integrity sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==
+  dependencies:
+    lru-cache "^6.0.0"
+
+send@0.18.0:
+  version "0.18.0"
+  resolved "https://registry.yarnpkg.com/send/-/send-0.18.0.tgz#670167cc654b05f5aa4a767f9113bb371bc706be"
+  integrity sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==
+  dependencies:
+    debug "2.6.9"
+    depd "2.0.0"
+    destroy "1.2.0"
+    encodeurl "~1.0.2"
+    escape-html "~1.0.3"
+    etag "~1.8.1"
+    fresh "0.5.2"
+    http-errors "2.0.0"
+    mime "1.6.0"
+    ms "2.1.3"
+    on-finished "2.4.1"
+    range-parser "~1.2.1"
+    statuses "2.0.1"
+
+sentence-case@^2.1.0:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/sentence-case/-/sentence-case-2.1.1.tgz#1f6e2dda39c168bf92d13f86d4a918933f667ed4"
+  integrity sha512-ENl7cYHaK/Ktwk5OTD+aDbQ3uC8IByu/6Bkg+HDv8Mm+XnBnppVNalcfJTNsp1ibstKh030/JKQQWglDvtKwEQ==
+  dependencies:
+    no-case "^2.2.0"
+    upper-case-first "^1.1.2"
+
+serialize-javascript@6.0.0:
+  version "6.0.0"
+  resolved "https://registry.yarnpkg.com/serialize-javascript/-/serialize-javascript-6.0.0.tgz#efae5d88f45d7924141da8b5c3a7a7e663fefeb8"
+  integrity sha512-Qr3TosvguFt8ePWqsvRfrKyQXIiW+nGbYpy8XK24NQHE83caxWt+mIymTT19DGFbNWNLfEwsrkSmN64lVWB9ag==
+  dependencies:
+    randombytes "^2.1.0"
+
+serve-static@1.15.0:
+  version "1.15.0"
+  resolved "https://registry.yarnpkg.com/serve-static/-/serve-static-1.15.0.tgz#faaef08cffe0a1a62f60cad0c4e513cff0ac9540"
+  integrity sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==
+  dependencies:
+    encodeurl "~1.0.2"
+    escape-html "~1.0.3"
+    parseurl "~1.3.3"
+    send "0.18.0"
+
+servify@^0.1.12:
+  version "0.1.12"
+  resolved "https://registry.yarnpkg.com/servify/-/servify-0.1.12.tgz#142ab7bee1f1d033b66d0707086085b17c06db95"
+  integrity sha512-/xE6GvsKKqyo1BAY+KxOWXcLpPsUUyji7Qg3bVD7hh1eRze5bR1uYiuDA/k3Gof1s9BTzQZEJK8sNcNGFIzeWw==
+  dependencies:
+    body-parser "^1.16.0"
+    cors "^2.8.1"
+    express "^4.14.0"
+    request "^2.79.0"
+    xhr "^2.3.3"
+
+set-blocking@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/set-blocking/-/set-blocking-2.0.0.tgz#045f9782d011ae9a6803ddd382b24392b3d890f7"
+  integrity sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw==
+
+setimmediate@1.0.4:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/setimmediate/-/setimmediate-1.0.4.tgz#20e81de622d4a02588ce0c8da8973cbcf1d3138f"
+  integrity sha512-/TjEmXQVEzdod/FFskf3o7oOAsGhHf2j1dZqRFbDzq4F3mvvxflIIi4Hd3bLQE9y/CpwqfSQam5JakI/mi3Pog==
+
+setimmediate@^1.0.5:
+  version "1.0.5"
+  resolved "https://registry.yarnpkg.com/setimmediate/-/setimmediate-1.0.5.tgz#290cbb232e306942d7d7ea9b83732ab7856f8285"
+  integrity sha512-MATJdZp8sLqDl/68LfQmbP8zKPLQNV6BIZoIgrscFDQ+RsvK/BxeDQOgyxKKoh0y/8h3BqVFnCqQ/gd+reiIXA==
+
+setprototypeof@1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.2.0.tgz#66c9a24a73f9fc28cbe66b09fed3d33dcaf1b424"
+  integrity sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==
+
+sha.js@^2.4.0, sha.js@^2.4.8:
+  version "2.4.11"
+  resolved "https://registry.yarnpkg.com/sha.js/-/sha.js-2.4.11.tgz#37a5cf0b81ecbc6943de109ba2960d1b26584ae7"
+  integrity sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==
+  dependencies:
+    inherits "^2.0.1"
+    safe-buffer "^5.0.1"
+
+sha1@^1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/sha1/-/sha1-1.1.1.tgz#addaa7a93168f393f19eb2b15091618e2700f848"
+  integrity sha512-dZBS6OrMjtgVkopB1Gmo4RQCDKiZsqcpAQpkV/aaj+FCrCg8r4I4qMkDPQjBgLIxlmu9k4nUbWq6ohXahOneYA==
+  dependencies:
+    charenc ">= 0.0.1"
+    crypt ">= 0.0.1"
+
+sha3@^2.1.1:
+  version "2.1.4"
+  resolved "https://registry.yarnpkg.com/sha3/-/sha3-2.1.4.tgz#000fac0fe7c2feac1f48a25e7a31b52a6492cc8f"
+  integrity sha512-S8cNxbyb0UGUM2VhRD4Poe5N58gJnJsLJ5vC7FYWGUmGhcsj4++WaIOBFVDxlG0W3To6xBuiRh+i0Qp2oNCOtg==
+  dependencies:
+    buffer "6.0.3"
+
+shebang-command@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-2.0.0.tgz#ccd0af4f8835fbdc265b82461aaf0c36663f34ea"
+  integrity sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==
+  dependencies:
+    shebang-regex "^3.0.0"
+
+shebang-regex@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-3.0.0.tgz#ae16f1644d873ecad843b0307b143362d4c42172"
+  integrity sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==
+
+shelljs@^0.8.3:
+  version "0.8.5"
+  resolved "https://registry.yarnpkg.com/shelljs/-/shelljs-0.8.5.tgz#de055408d8361bed66c669d2f000538ced8ee20c"
+  integrity sha512-TiwcRcrkhHvbrZbnRcFYMLl30Dfov3HKqzp5tO5b4pt6G/SezKcYhmDg15zXVBswHmctSAQKznqNW2LO5tTDow==
+  dependencies:
+    glob "^7.0.0"
+    interpret "^1.0.0"
+    rechoir "^0.6.2"
+
+side-channel@^1.0.4:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/side-channel/-/side-channel-1.0.4.tgz#efce5c8fdc104ee751b25c58d4290011fa5ea2cf"
+  integrity sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==
+  dependencies:
+    call-bind "^1.0.0"
+    get-intrinsic "^1.0.2"
+    object-inspect "^1.9.0"
+
+signal-exit@^3.0.2:
+  version "3.0.7"
+  resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.7.tgz#a9a1767f8af84155114eaabd73f99273c8f59ad9"
+  integrity sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==
+
+simple-concat@^1.0.0:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/simple-concat/-/simple-concat-1.0.1.tgz#f46976082ba35c2263f1c8ab5edfe26c41c9552f"
+  integrity sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==
+
+simple-get@^2.7.0:
+  version "2.8.2"
+  resolved "https://registry.yarnpkg.com/simple-get/-/simple-get-2.8.2.tgz#5708fb0919d440657326cd5fe7d2599d07705019"
+  integrity sha512-Ijd/rV5o+mSBBs4F/x9oDPtTx9Zb6X9brmnXvMW4J7IR15ngi9q5xxqWBKU744jTZiaXtxaPL7uHG6vtN8kUkw==
+  dependencies:
+    decompress-response "^3.3.0"
+    once "^1.3.1"
+    simple-concat "^1.0.0"
+
+sisteransi@^1.0.5:
+  version "1.0.5"
+  resolved "https://registry.yarnpkg.com/sisteransi/-/sisteransi-1.0.5.tgz#134d681297756437cc05ca01370d3a7a571075ed"
+  integrity sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==
+
+slash@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/slash/-/slash-3.0.0.tgz#6539be870c165adbd5240220dbe361f1bc4d4634"
+  integrity sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==
+
+slice-ansi@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-4.0.0.tgz#500e8dd0fd55b05815086255b3195adf2a45fe6b"
+  integrity sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ==
+  dependencies:
+    ansi-styles "^4.0.0"
+    astral-regex "^2.0.0"
+    is-fullwidth-code-point "^3.0.0"
+
+snake-case@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/snake-case/-/snake-case-2.1.0.tgz#41bdb1b73f30ec66a04d4e2cad1b76387d4d6d9f"
+  integrity sha512-FMR5YoPFwOLuh4rRz92dywJjyKYZNLpMn1R5ujVpIYkbA9p01fq8RMg0FkO4M+Yobt4MjHeLTJVm5xFFBHSV2Q==
+  dependencies:
+    no-case "^2.2.0"
+
+solc@0.7.3:
+  version "0.7.3"
+  resolved "https://registry.yarnpkg.com/solc/-/solc-0.7.3.tgz#04646961bd867a744f63d2b4e3c0701ffdc7d78a"
+  integrity sha512-GAsWNAjGzIDg7VxzP6mPjdurby3IkGCjQcM8GFYZT6RyaoUZKmMU6Y7YwG+tFGhv7dwZ8rmR4iwFDrrD99JwqA==
+  dependencies:
+    command-exists "^1.2.8"
+    commander "3.0.2"
+    follow-redirects "^1.12.1"
+    fs-extra "^0.30.0"
+    js-sha3 "0.8.0"
+    memorystream "^0.3.1"
+    require-from-string "^2.0.0"
+    semver "^5.5.0"
+    tmp "0.0.33"
+
+solc@^0.4.20:
+  version "0.4.26"
+  resolved "https://registry.yarnpkg.com/solc/-/solc-0.4.26.tgz#5390a62a99f40806b86258c737c1cf653cc35cb5"
+  integrity sha512-o+c6FpkiHd+HPjmjEVpQgH7fqZ14tJpXhho+/bQXlXbliLIS/xjXb42Vxh+qQY1WCSTMQ0+a5vR9vi0MfhU6mA==
+  dependencies:
+    fs-extra "^0.30.0"
+    memorystream "^0.3.1"
+    require-from-string "^1.1.0"
+    semver "^5.3.0"
+    yargs "^4.7.1"
+
+solhint@^3.3.4:
+  version "3.5.1"
+  resolved "https://registry.yarnpkg.com/solhint/-/solhint-3.5.1.tgz#3cf47c173f1a223770c3bef719122d6d2d157371"
+  integrity sha512-29+vUIwUmsasKuIzCYOqiKlu4We7+BVYNuoKmDMWsmjfPDoJQpM65eBzYuqEr18lwvXpQAbjTGdzvm4iZIO9Uw==
+  dependencies:
+    "@solidity-parser/parser" "^0.16.0"
+    ajv "^6.12.6"
+    antlr4 "^4.11.0"
+    ast-parents "^0.0.1"
+    chalk "^4.1.2"
+    commander "^10.0.0"
+    cosmiconfig "^8.0.0"
+    fast-diff "^1.2.0"
+    glob "^8.0.3"
+    ignore "^5.2.4"
+    js-yaml "^4.1.0"
+    lodash "^4.17.21"
+    pluralize "^8.0.0"
+    semver "^7.5.2"
+    strip-ansi "^6.0.1"
+    table "^6.8.1"
+    text-table "^0.2.0"
+  optionalDependencies:
+    prettier "^2.8.3"
+
+solidity-comments-extractor@^0.0.7:
+  version "0.0.7"
+  resolved "https://registry.yarnpkg.com/solidity-comments-extractor/-/solidity-comments-extractor-0.0.7.tgz#99d8f1361438f84019795d928b931f4e5c39ca19"
+  integrity sha512-wciNMLg/Irp8OKGrh3S2tfvZiZ0NEyILfcRCXCD4mp7SgK/i9gzLfhY2hY7VMCQJ3kH9UB9BzNdibIVMchzyYw==
+
+solidity-coverage@^0.8.5:
+  version "0.8.5"
+  resolved "https://registry.yarnpkg.com/solidity-coverage/-/solidity-coverage-0.8.5.tgz#64071c3a0c06a0cecf9a7776c35f49edc961e875"
+  integrity sha512-6C6N6OV2O8FQA0FWA95FdzVH+L16HU94iFgg5wAFZ29UpLFkgNI/DRR2HotG1bC0F4gAc/OMs2BJI44Q/DYlKQ==
+  dependencies:
+    "@ethersproject/abi" "^5.0.9"
+    "@solidity-parser/parser" "^0.16.0"
+    chalk "^2.4.2"
+    death "^1.1.0"
+    detect-port "^1.3.0"
+    difflib "^0.2.4"
+    fs-extra "^8.1.0"
+    ghost-testrpc "^0.0.2"
+    global-modules "^2.0.0"
+    globby "^10.0.1"
+    jsonschema "^1.2.4"
+    lodash "^4.17.15"
+    mocha "10.2.0"
+    node-emoji "^1.10.0"
+    pify "^4.0.1"
+    recursive-readdir "^2.2.2"
+    sc-istanbul "^0.4.5"
+    semver "^7.3.4"
+    shelljs "^0.8.3"
+    web3-utils "^1.3.6"
+
+solmate@^6.7.0:
+  version "6.7.0"
+  resolved "https://registry.yarnpkg.com/solmate/-/solmate-6.7.0.tgz#5e07b62498babec273aceb322a3e45d0e058e042"
+  integrity sha512-iMPr+gKbKjXBB12a+Iz5Tua5r7T4yugHaGXDWSJbBZB4Gr3vLeUUvKeLyMxCWWqk1xlLhFDFFuAmOzeyVBuyvQ==
+
+source-map-support@^0.5.13, source-map-support@^0.5.19:
+  version "0.5.21"
+  resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.21.tgz#04fe7c7f9e1ed2d662233c28cb2b35b9f63f6e4f"
+  integrity sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==
+  dependencies:
+    buffer-from "^1.0.0"
+    source-map "^0.6.0"
+
+source-map@^0.6.0, source-map@^0.6.1:
+  version "0.6.1"
+  resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263"
+  integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==
+
+source-map@~0.2.0:
+  version "0.2.0"
+  resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.2.0.tgz#dab73fbcfc2ba819b4de03bd6f6eaa48164b3f9d"
+  integrity sha512-CBdZ2oa/BHhS4xj5DlhjWNHcan57/5YuvfdLf17iVmIpd9KRm+DFLmC6nBNj+6Ua7Kt3TmOjDpQT1aTYOQtoUA==
+  dependencies:
+    amdefine ">=0.0.4"
+
+spdx-correct@^3.0.0:
+  version "3.2.0"
+  resolved "https://registry.yarnpkg.com/spdx-correct/-/spdx-correct-3.2.0.tgz#4f5ab0668f0059e34f9c00dce331784a12de4e9c"
+  integrity sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA==
+  dependencies:
+    spdx-expression-parse "^3.0.0"
+    spdx-license-ids "^3.0.0"
+
+spdx-exceptions@^2.1.0:
+  version "2.3.0"
+  resolved "https://registry.yarnpkg.com/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz#3f28ce1a77a00372683eade4a433183527a2163d"
+  integrity sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A==
+
+spdx-expression-parse@^3.0.0:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz#cf70f50482eefdc98e3ce0a6833e4a53ceeba679"
+  integrity sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==
+  dependencies:
+    spdx-exceptions "^2.1.0"
+    spdx-license-ids "^3.0.0"
+
+spdx-license-ids@^3.0.0:
+  version "3.0.13"
+  resolved "https://registry.yarnpkg.com/spdx-license-ids/-/spdx-license-ids-3.0.13.tgz#7189a474c46f8d47c7b0da4b987bb45e908bd2d5"
+  integrity sha512-XkD+zwiqXHikFZm4AX/7JSCXA98U5Db4AFd5XUg/+9UNtnH75+Z9KxtpYiJZx36mUDVOwH83pl7yvCer6ewM3w==
+
+sprintf-js@~1.0.2:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.0.3.tgz#04e6926f662895354f3dd015203633b857297e2c"
+  integrity sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==
+
+sshpk@^1.7.0:
+  version "1.17.0"
+  resolved "https://registry.yarnpkg.com/sshpk/-/sshpk-1.17.0.tgz#578082d92d4fe612b13007496e543fa0fbcbe4c5"
+  integrity sha512-/9HIEs1ZXGhSPE8X6Ccm7Nam1z8KcoCqPdI7ecm1N33EzAetWahvQWVqLZtaZQ+IDKX4IyA2o0gBzqIMkAagHQ==
+  dependencies:
+    asn1 "~0.2.3"
+    assert-plus "^1.0.0"
+    bcrypt-pbkdf "^1.0.0"
+    dashdash "^1.12.0"
+    ecc-jsbn "~0.1.1"
+    getpass "^0.1.1"
+    jsbn "~0.1.0"
+    safer-buffer "^2.0.2"
+    tweetnacl "~0.14.0"
+
+stacktrace-parser@^0.1.10:
+  version "0.1.10"
+  resolved "https://registry.yarnpkg.com/stacktrace-parser/-/stacktrace-parser-0.1.10.tgz#29fb0cae4e0d0b85155879402857a1639eb6051a"
+  integrity sha512-KJP1OCML99+8fhOHxwwzyWrlUuVX5GQ0ZpJTd1DFXhdkrvg1szxfHhawXUZ3g9TkXORQd4/WG68jMlQZ2p8wlg==
+  dependencies:
+    type-fest "^0.7.1"
+
+statuses@2.0.1:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/statuses/-/statuses-2.0.1.tgz#55cb000ccf1d48728bd23c685a063998cf1a1b63"
+  integrity sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==
+
+stealthy-require@^1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/stealthy-require/-/stealthy-require-1.1.1.tgz#35b09875b4ff49f26a777e509b3090a3226bf24b"
+  integrity sha512-ZnWpYnYugiOVEY5GkcuJK1io5V8QmNYChG62gSit9pQVGErXtrKuPC55ITaVSukmMta5qpMU7vqLt2Lnni4f/g==
+
+strict-uri-encode@^1.0.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/strict-uri-encode/-/strict-uri-encode-1.1.0.tgz#279b225df1d582b1f54e65addd4352e18faa0713"
+  integrity sha512-R3f198pcvnB+5IpnBlRkphuE9n46WyVl8I39W/ZUTZLz4nqSP/oLYUrcnJrw462Ds8he4YKMov2efsTIw1BDGQ==
+
+string-format@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/string-format/-/string-format-2.0.0.tgz#f2df2e7097440d3b65de31b6d40d54c96eaffb9b"
+  integrity sha512-bbEs3scLeYNXLecRRuk6uJxdXUSj6le/8rNPHChIJTn2V79aXVTR1EH2OH5zLKKoz0V02fOUKZZcw01pLUShZA==
+
+string-width@^1.0.1:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/string-width/-/string-width-1.0.2.tgz#118bdf5b8cdc51a2a7e70d211e07e2b0b9b107d3"
+  integrity sha512-0XsVpQLnVCXHJfyEs8tC0zpTVIr5PKKsQtkT29IwupnPTjtPmQ3xT/4yCREF9hYkV/3M3kzcUTSAZT6a6h81tw==
+  dependencies:
+    code-point-at "^1.0.0"
+    is-fullwidth-code-point "^1.0.0"
+    strip-ansi "^3.0.0"
+
+"string-width@^1.0.2 || 2", string-width@^2.1.1:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/string-width/-/string-width-2.1.1.tgz#ab93f27a8dc13d28cac815c462143a6d9012ae9e"
+  integrity sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==
+  dependencies:
+    is-fullwidth-code-point "^2.0.0"
+    strip-ansi "^4.0.0"
+
+string-width@^3.0.0, string-width@^3.1.0:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/string-width/-/string-width-3.1.0.tgz#22767be21b62af1081574306f69ac51b62203961"
+  integrity sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==
+  dependencies:
+    emoji-regex "^7.0.1"
+    is-fullwidth-code-point "^2.0.0"
+    strip-ansi "^5.1.0"
+
+string-width@^4.1.0, string-width@^4.2.0, string-width@^4.2.3:
+  version "4.2.3"
+  resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010"
+  integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==
+  dependencies:
+    emoji-regex "^8.0.0"
+    is-fullwidth-code-point "^3.0.0"
+    strip-ansi "^6.0.1"
+
+string.prototype.trim@^1.2.7:
+  version "1.2.7"
+  resolved "https://registry.yarnpkg.com/string.prototype.trim/-/string.prototype.trim-1.2.7.tgz#a68352740859f6893f14ce3ef1bb3037f7a90533"
+  integrity sha512-p6TmeT1T3411M8Cgg9wBTMRtY2q9+PNy9EV1i2lIXUN/btt763oIfxwN3RR8VU6wHX8j/1CFy0L+YuThm6bgOg==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.1.4"
+    es-abstract "^1.20.4"
+
+string.prototype.trimend@^1.0.6:
+  version "1.0.6"
+  resolved "https://registry.yarnpkg.com/string.prototype.trimend/-/string.prototype.trimend-1.0.6.tgz#c4a27fa026d979d79c04f17397f250a462944533"
+  integrity sha512-JySq+4mrPf9EsDBEDYMOb/lM7XQLulwg5R/m1r0PXEFqrV0qHvl58sdTilSXtKOflCsK2E8jxf+GKC0T07RWwQ==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.1.4"
+    es-abstract "^1.20.4"
+
+string.prototype.trimstart@^1.0.6:
+  version "1.0.6"
+  resolved "https://registry.yarnpkg.com/string.prototype.trimstart/-/string.prototype.trimstart-1.0.6.tgz#e90ab66aa8e4007d92ef591bbf3cd422c56bdcf4"
+  integrity sha512-omqjMDaY92pbn5HOX7f9IccLA+U1tA9GvtU4JrodiXFfYB7jPzzHpRzpglLAjtUV6bB557zwClJezTqnAiYnQA==
+  dependencies:
+    call-bind "^1.0.2"
+    define-properties "^1.1.4"
+    es-abstract "^1.20.4"
+
+string_decoder@^1.1.1:
+  version "1.3.0"
+  resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz#42f114594a46cf1a8e30b0a84f56c78c3edac21e"
+  integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==
+  dependencies:
+    safe-buffer "~5.2.0"
+
+string_decoder@~1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.1.1.tgz#9cf1611ba62685d7030ae9e4ba34149c3af03fc8"
+  integrity sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==
+  dependencies:
+    safe-buffer "~5.1.0"
+
+strip-ansi@^3.0.0, strip-ansi@^3.0.1:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-3.0.1.tgz#6a385fb8853d952d5ff05d0e8aaf94278dc63dcf"
+  integrity sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==
+  dependencies:
+    ansi-regex "^2.0.0"
+
+strip-ansi@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-4.0.0.tgz#a8479022eb1ac368a871389b635262c505ee368f"
+  integrity sha512-4XaJ2zQdCzROZDivEVIDPkcQn8LMFSa8kj8Gxb/Lnwzv9A8VctNZ+lfivC/sV3ivW8ElJTERXZoPBRrZKkNKow==
+  dependencies:
+    ansi-regex "^3.0.0"
+
+strip-ansi@^5.0.0, strip-ansi@^5.1.0, strip-ansi@^5.2.0:
+  version "5.2.0"
+  resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-5.2.0.tgz#8c9a536feb6afc962bdfa5b104a5091c1ad9c0ae"
+  integrity sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==
+  dependencies:
+    ansi-regex "^4.1.0"
+
+strip-ansi@^6.0.0, strip-ansi@^6.0.1:
+  version "6.0.1"
+  resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9"
+  integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==
+  dependencies:
+    ansi-regex "^5.0.1"
+
+strip-bom@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-2.0.0.tgz#6219a85616520491f35788bdbf1447a99c7e6b0e"
+  integrity sha512-kwrX1y7czp1E69n2ajbG65mIo9dqvJ+8aBQXOGVxqwvNbsXdFM6Lq37dLAY3mknUwru8CfcCbfOLL/gMo+fi3g==
+  dependencies:
+    is-utf8 "^0.2.0"
+
+strip-bom@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-3.0.0.tgz#2334c18e9c759f7bdd56fdef7e9ae3d588e68ed3"
+  integrity sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==
+
+strip-hex-prefix@1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/strip-hex-prefix/-/strip-hex-prefix-1.0.0.tgz#0c5f155fef1151373377de9dbb588da05500e36f"
+  integrity sha512-q8d4ue7JGEiVcypji1bALTos+0pWtyGlivAWyPuTkHzuTCJqrK9sWxYQZUq6Nq3cuyv3bm734IhHvHtGGURU6A==
+  dependencies:
+    is-hex-prefixed "1.0.0"
+
+strip-indent@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/strip-indent/-/strip-indent-2.0.0.tgz#5ef8db295d01e6ed6cbf7aab96998d7822527b68"
+  integrity sha512-RsSNPLpq6YUL7QYy44RnPVTn/lcVZtb48Uof3X5JLbF4zD/Gs7ZFDv2HWol+leoQN2mT86LAzSshGfkTlSOpsA==
+
+strip-json-comments@2.0.1:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-2.0.1.tgz#3c531942e908c2697c0ec344858c286c7ca0a60a"
+  integrity sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==
+
+strip-json-comments@3.1.1, strip-json-comments@^3.1.1:
+  version "3.1.1"
+  resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-3.1.1.tgz#31f1281b3832630434831c310c01cccda8cbe006"
+  integrity sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==
+
+supports-color@6.0.0:
+  version "6.0.0"
+  resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-6.0.0.tgz#76cfe742cf1f41bb9b1c29ad03068c05b4c0e40a"
+  integrity sha512-on9Kwidc1IUQo+bQdhi8+Tijpo0e1SS6RoGo2guUwn5vdaxw8RXOF9Vb2ws+ihWOmh4JnCJOvaziZWP1VABaLg==
+  dependencies:
+    has-flag "^3.0.0"
+
+supports-color@8.1.1:
+  version "8.1.1"
+  resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-8.1.1.tgz#cd6fc17e28500cff56c1b86c0a7fd4a54a73005c"
+  integrity sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==
+  dependencies:
+    has-flag "^4.0.0"
+
+supports-color@^3.1.0:
+  version "3.2.3"
+  resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-3.2.3.tgz#65ac0504b3954171d8a64946b2ae3cbb8a5f54f6"
+  integrity sha512-Jds2VIYDrlp5ui7t8abHN2bjAu4LV/q4N2KivFPpGH0lrka0BMq/33AmECUXlKPcHigkNaqfXRENFju+rlcy+A==
+  dependencies:
+    has-flag "^1.0.0"
+
+supports-color@^5.3.0:
+  version "5.5.0"
+  resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-5.5.0.tgz#e2e69a44ac8772f78a1ec0b35b689df6530efc8f"
+  integrity sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==
+  dependencies:
+    has-flag "^3.0.0"
+
+supports-color@^7.1.0:
+  version "7.2.0"
+  resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz#1b7dcdcb32b8138801b3e478ba6a51caa89648da"
+  integrity sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==
+  dependencies:
+    has-flag "^4.0.0"
+
+supports-preserve-symlinks-flag@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz#6eda4bd344a3c94aea376d4cc31bc77311039e09"
+  integrity sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==
+
+swap-case@^1.1.0:
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/swap-case/-/swap-case-1.1.2.tgz#c39203a4587385fad3c850a0bd1bcafa081974e3"
+  integrity sha512-BAmWG6/bx8syfc6qXPprof3Mn5vQgf5dwdUNJhsNqU9WdPt5P+ES/wQ5bxfijy8zwZgZZHslC3iAsxsuQMCzJQ==
+  dependencies:
+    lower-case "^1.1.1"
+    upper-case "^1.1.1"
+
+swarm-js@^0.1.40:
+  version "0.1.42"
+  resolved "https://registry.yarnpkg.com/swarm-js/-/swarm-js-0.1.42.tgz#497995c62df6696f6e22372f457120e43e727979"
+  integrity sha512-BV7c/dVlA3R6ya1lMlSSNPLYrntt0LUq4YMgy3iwpCIc6rZnS5W2wUoctarZ5pXlpKtxDDf9hNziEkcfrxdhqQ==
+  dependencies:
+    bluebird "^3.5.0"
+    buffer "^5.0.5"
+    eth-lib "^0.1.26"
+    fs-extra "^4.0.2"
+    got "^11.8.5"
+    mime-types "^2.1.16"
+    mkdirp-promise "^5.0.1"
+    mock-fs "^4.1.0"
+    setimmediate "^1.0.5"
+    tar "^4.0.2"
+    xhr-request "^1.0.1"
+
+sync-request@^6.0.0:
+  version "6.1.0"
+  resolved "https://registry.yarnpkg.com/sync-request/-/sync-request-6.1.0.tgz#e96217565b5e50bbffe179868ba75532fb597e68"
+  integrity sha512-8fjNkrNlNCrVc/av+Jn+xxqfCjYaBoHqCsDz6mt030UMxJGr+GSfCV1dQt2gRtlL63+VPidwDVLr7V2OcTSdRw==
+  dependencies:
+    http-response-object "^3.0.1"
+    sync-rpc "^1.2.1"
+    then-request "^6.0.0"
+
+sync-rpc@^1.2.1:
+  version "1.3.6"
+  resolved "https://registry.yarnpkg.com/sync-rpc/-/sync-rpc-1.3.6.tgz#b2e8b2550a12ccbc71df8644810529deb68665a7"
+  integrity sha512-J8jTXuZzRlvU7HemDgHi3pGnh/rkoqR/OZSjhTyyZrEkkYQbk7Z33AXp37mkPfPpfdOuj7Ex3H/TJM1z48uPQw==
+  dependencies:
+    get-port "^3.1.0"
+
+table-layout@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/table-layout/-/table-layout-1.0.2.tgz#c4038a1853b0136d63365a734b6931cf4fad4a04"
+  integrity sha512-qd/R7n5rQTRFi+Zf2sk5XVVd9UQl6ZkduPFC3S7WEGJAmetDTjY3qPN50eSKzwuzEyQKy5TN2TiZdkIjos2L6A==
+  dependencies:
+    array-back "^4.0.1"
+    deep-extend "~0.6.0"
+    typical "^5.2.0"
+    wordwrapjs "^4.0.0"
+
+table@^6.8.0, table@^6.8.1:
+  version "6.8.1"
+  resolved "https://registry.yarnpkg.com/table/-/table-6.8.1.tgz#ea2b71359fe03b017a5fbc296204471158080bdf"
+  integrity sha512-Y4X9zqrCftUhMeH2EptSSERdVKt/nEdijTOacGD/97EKjhQ/Qs8RTlEGABSJNNN8lac9kheH+af7yAkEWlgneA==
+  dependencies:
+    ajv "^8.0.1"
+    lodash.truncate "^4.4.2"
+    slice-ansi "^4.0.0"
+    string-width "^4.2.3"
+    strip-ansi "^6.0.1"
+
+tar@^4.0.2:
+  version "4.4.19"
+  resolved "https://registry.yarnpkg.com/tar/-/tar-4.4.19.tgz#2e4d7263df26f2b914dee10c825ab132123742f3"
+  integrity sha512-a20gEsvHnWe0ygBY8JbxoM4w3SJdhc7ZAuxkLqh+nvNQN2IOt0B5lLgM490X5Hl8FF0dl0tOf2ewFYAlIFgzVA==
+  dependencies:
+    chownr "^1.1.4"
+    fs-minipass "^1.2.7"
+    minipass "^2.9.0"
+    minizlib "^1.3.3"
+    mkdirp "^0.5.5"
+    safe-buffer "^5.2.1"
+    yallist "^3.1.1"
+
+tenderly@^0.5.3:
+  version "0.5.3"
+  resolved "https://registry.yarnpkg.com/tenderly/-/tenderly-0.5.3.tgz#115653ff33fc33e3be41ab7dd669fdbe0f28a6fb"
+  integrity sha512-sR+sbZKZzt3b2+moXJsrkBvbava1/4mGulIfuZw8bwr2OpCH8N00dME1t89JC8RjVnQjy4VewVFHyWANdn5zYQ==
+  dependencies:
+    axios "^0.27.2"
+    cli-table3 "^0.6.2"
+    commander "^9.4.0"
+    express "^4.18.1"
+    hyperlinker "^1.0.0"
+    js-yaml "^4.1.0"
+    open "^8.4.0"
+    prompts "^2.4.2"
+    tslog "^4.4.0"
+
+testrpc@0.0.1:
+  version "0.0.1"
+  resolved "https://registry.yarnpkg.com/testrpc/-/testrpc-0.0.1.tgz#83e2195b1f5873aec7be1af8cbe6dcf39edb7aed"
+  integrity sha512-afH1hO+SQ/VPlmaLUFj2636QMeDvPCeQMc/9RBMW0IfjNe9gFD9Ra3ShqYkB7py0do1ZcCna/9acHyzTJ+GcNA==
+
+text-table@^0.2.0:
+  version "0.2.0"
+  resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4"
+  integrity sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==
+
+then-request@^6.0.0:
+  version "6.0.2"
+  resolved "https://registry.yarnpkg.com/then-request/-/then-request-6.0.2.tgz#ec18dd8b5ca43aaee5cb92f7e4c1630e950d4f0c"
+  integrity sha512-3ZBiG7JvP3wbDzA9iNY5zJQcHL4jn/0BWtXIkagfz7QgOL/LqjCEOBQuJNZfu0XYnv5JhKh+cDxCPM4ILrqruA==
+  dependencies:
+    "@types/concat-stream" "^1.6.0"
+    "@types/form-data" "0.0.33"
+    "@types/node" "^8.0.0"
+    "@types/qs" "^6.2.31"
+    caseless "~0.12.0"
+    concat-stream "^1.6.0"
+    form-data "^2.2.0"
+    http-basic "^8.1.1"
+    http-response-object "^3.0.1"
+    promise "^8.0.0"
+    qs "^6.4.0"
+
+timed-out@^4.0.1:
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/timed-out/-/timed-out-4.0.1.tgz#f32eacac5a175bea25d7fab565ab3ed8741ef56f"
+  integrity sha512-G7r3AhovYtr5YKOWQkta8RKAPb+J9IsO4uVmzjl8AZwfhs8UcUwTiD6gcJYSgOtzyjvQKrKYn41syHbUWMkafA==
+
+title-case@^2.1.0:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/title-case/-/title-case-2.1.1.tgz#3e127216da58d2bc5becf137ab91dae3a7cd8faa"
+  integrity sha512-EkJoZ2O3zdCz3zJsYCsxyq2OC5hrxR9mfdd5I+w8h/tmFfeOxJ+vvkxsKxdmN0WtS9zLdHEgfgVOiMVgv+Po4Q==
+  dependencies:
+    no-case "^2.2.0"
+    upper-case "^1.0.3"
+
+tmp@0.0.33:
+  version "0.0.33"
+  resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.0.33.tgz#6d34335889768d21b2bcda0aa277ced3b1bfadf9"
+  integrity sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==
+  dependencies:
+    os-tmpdir "~1.0.2"
+
+to-regex-range@^5.0.1:
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4"
+  integrity sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==
+  dependencies:
+    is-number "^7.0.0"
+
+toidentifier@1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/toidentifier/-/toidentifier-1.0.1.tgz#3be34321a88a820ed1bd80dfaa33e479fbb8dd35"
+  integrity sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==
+
+tough-cookie@^2.3.3, tough-cookie@~2.5.0:
+  version "2.5.0"
+  resolved "https://registry.yarnpkg.com/tough-cookie/-/tough-cookie-2.5.0.tgz#cd9fb2a0aa1d5a12b473bd9fb96fa3dcff65ade2"
+  integrity sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==
+  dependencies:
+    psl "^1.1.28"
+    punycode "^2.1.1"
+
+tr46@~0.0.3:
+  version "0.0.3"
+  resolved "https://registry.yarnpkg.com/tr46/-/tr46-0.0.3.tgz#8184fd347dac9cdc185992f3a6622e14b9d9ab6a"
+  integrity sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==
+
+ts-command-line-args@^2.2.0:
+  version "2.5.1"
+  resolved "https://registry.yarnpkg.com/ts-command-line-args/-/ts-command-line-args-2.5.1.tgz#e64456b580d1d4f6d948824c274cf6fa5f45f7f0"
+  integrity sha512-H69ZwTw3rFHb5WYpQya40YAX2/w7Ut75uUECbgBIsLmM+BNuYnxsltfyyLMxy6sEeKxgijLTnQtLd0nKd6+IYw==
+  dependencies:
+    chalk "^4.1.0"
+    command-line-args "^5.1.1"
+    command-line-usage "^6.1.0"
+    string-format "^2.0.0"
+
+ts-essentials@^7.0.1:
+  version "7.0.3"
+  resolved "https://registry.yarnpkg.com/ts-essentials/-/ts-essentials-7.0.3.tgz#686fd155a02133eedcc5362dc8b5056cde3e5a38"
+  integrity sha512-8+gr5+lqO3G84KdiTSMRLtuyJ+nTBVRKuCrK4lidMPdVeEp0uqC875uE5NMcaA7YYMN7XsNiFQuMvasF8HT/xQ==
+
+ts-node@^10.9.1:
+  version "10.9.1"
+  resolved "https://registry.yarnpkg.com/ts-node/-/ts-node-10.9.1.tgz#e73de9102958af9e1f0b168a6ff320e25adcff4b"
+  integrity sha512-NtVysVPkxxrwFGUUxGYhfux8k78pQB3JqYBXlLRZgdGUqTO5wU/UyHop5p70iEbGhB7q5KmiZiU0Y3KlJrScEw==
+  dependencies:
+    "@cspotcode/source-map-support" "^0.8.0"
+    "@tsconfig/node10" "^1.0.7"
+    "@tsconfig/node12" "^1.0.7"
+    "@tsconfig/node14" "^1.0.0"
+    "@tsconfig/node16" "^1.0.2"
+    acorn "^8.4.1"
+    acorn-walk "^8.1.1"
+    arg "^4.1.0"
+    create-require "^1.1.0"
+    diff "^4.0.1"
+    make-error "^1.1.1"
+    v8-compile-cache-lib "^3.0.1"
+    yn "3.1.1"
+
+tsconfig-paths@^3.14.2:
+  version "3.14.2"
+  resolved "https://registry.yarnpkg.com/tsconfig-paths/-/tsconfig-paths-3.14.2.tgz#6e32f1f79412decd261f92d633a9dc1cfa99f088"
+  integrity sha512-o/9iXgCYc5L/JxCHPe3Hvh8Q/2xm5Z+p18PESBU6Ff33695QnCHBEjcytY2q19ua7Mbl/DavtBOLq+oG0RCL+g==
+  dependencies:
+    "@types/json5" "^0.0.29"
+    json5 "^1.0.2"
+    minimist "^1.2.6"
+    strip-bom "^3.0.0"
+
+tslib@^1.8.1, tslib@^1.9.3:
+  version "1.14.1"
+  resolved "https://registry.yarnpkg.com/tslib/-/tslib-1.14.1.tgz#cf2d38bdc34a134bcaf1091c41f6619e2f672d00"
+  integrity sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==
+
+tslog@^4.3.1, tslog@^4.4.0:
+  version "4.9.1"
+  resolved "https://registry.yarnpkg.com/tslog/-/tslog-4.9.1.tgz#b395bf003de35ed1537bbb97d1e180280debae09"
+  integrity sha512-N6SkH+ApNh65tyKV7CfXmcShfw7KTnoTASQkKr+nZbhM73iwi/3a9ZJeYKRRvboLxkx3UNnNViMo1s/hNdSZLg==
+
+tsort@0.0.1:
+  version "0.0.1"
+  resolved "https://registry.yarnpkg.com/tsort/-/tsort-0.0.1.tgz#e2280f5e817f8bf4275657fd0f9aebd44f5a2786"
+  integrity sha512-Tyrf5mxF8Ofs1tNoxA13lFeZ2Zrbd6cKbuH3V+MQ5sb6DtBj5FjrXVsRWT8YvNAQTqNoz66dz1WsbigI22aEnw==
+
+tsutils@^3.21.0:
+  version "3.21.0"
+  resolved "https://registry.yarnpkg.com/tsutils/-/tsutils-3.21.0.tgz#b48717d394cea6c1e096983eed58e9d61715b623"
+  integrity sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==
+  dependencies:
+    tslib "^1.8.1"
+
+tunnel-agent@^0.6.0:
+  version "0.6.0"
+  resolved "https://registry.yarnpkg.com/tunnel-agent/-/tunnel-agent-0.6.0.tgz#27a5dea06b36b04a0a9966774b290868f0fc40fd"
+  integrity sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==
+  dependencies:
+    safe-buffer "^5.0.1"
+
+tweetnacl-util@^0.15.1:
+  version "0.15.1"
+  resolved "https://registry.yarnpkg.com/tweetnacl-util/-/tweetnacl-util-0.15.1.tgz#b80fcdb5c97bcc508be18c44a4be50f022eea00b"
+  integrity sha512-RKJBIj8lySrShN4w6i/BonWp2Z/uxwC3h4y7xsRrpP59ZboCd0GpEVsOnMDYLMmKBpYhb5TgHzZXy7wTfYFBRw==
+
+tweetnacl@^0.14.3, tweetnacl@~0.14.0:
+  version "0.14.5"
+  resolved "https://registry.yarnpkg.com/tweetnacl/-/tweetnacl-0.14.5.tgz#5ae68177f192d4456269d108afa93ff8743f4f64"
+  integrity sha512-KXXFFdAbFXY4geFIwoyNK+f5Z1b7swfXABfL7HXCmoIWMKU3dmS26672A4EeQtDzLKy7SXmfBu51JolvEKwtGA==
+
+tweetnacl@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/tweetnacl/-/tweetnacl-1.0.3.tgz#ac0af71680458d8a6378d0d0d050ab1407d35596"
+  integrity sha512-6rt+RN7aOi1nGMyC4Xa5DdYiukl2UWCbcJft7YhxReBGQD7OAM8Pbxw6YMo4r2diNEA8FEmu32YOn9rhaiE5yw==
+
+type-check@^0.4.0, type-check@~0.4.0:
+  version "0.4.0"
+  resolved "https://registry.yarnpkg.com/type-check/-/type-check-0.4.0.tgz#07b8203bfa7056c0657050e3ccd2c37730bab8f1"
+  integrity sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==
+  dependencies:
+    prelude-ls "^1.2.1"
+
+type-check@~0.3.2:
+  version "0.3.2"
+  resolved "https://registry.yarnpkg.com/type-check/-/type-check-0.3.2.tgz#5884cab512cf1d355e3fb784f30804b2b520db72"
+  integrity sha512-ZCmOJdvOWDBYJlzAoFkC+Q0+bUyEOS1ltgp1MGU03fqHG+dbi9tBFU2Rd9QKiDZFAYrhPh2JUf7rZRIuHRKtOg==
+  dependencies:
+    prelude-ls "~1.1.2"
+
+type-detect@^4.0.0, type-detect@^4.0.5:
+  version "4.0.8"
+  resolved "https://registry.yarnpkg.com/type-detect/-/type-detect-4.0.8.tgz#7646fb5f18871cfbb7749e69bd39a6388eb7450c"
+  integrity sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==
+
+type-fest@^0.20.2:
+  version "0.20.2"
+  resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.20.2.tgz#1bf207f4b28f91583666cb5fbd327887301cd5f4"
+  integrity sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==
+
+type-fest@^0.21.3:
+  version "0.21.3"
+  resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.21.3.tgz#d260a24b0198436e133fa26a524a6d65fa3b2e37"
+  integrity sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==
+
+type-fest@^0.7.1:
+  version "0.7.1"
+  resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.7.1.tgz#8dda65feaf03ed78f0a3f9678f1869147f7c5c48"
+  integrity sha512-Ne2YiiGN8bmrmJJEuTWTLJR32nh/JdL1+PSicowtNb0WFpn59GK8/lfD61bVtzguz7b3PBt74nxpv/Pw5po5Rg==
+
+type-is@~1.6.18:
+  version "1.6.18"
+  resolved "https://registry.yarnpkg.com/type-is/-/type-is-1.6.18.tgz#4e552cd05df09467dcbc4ef739de89f2cf37c131"
+  integrity sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==
+  dependencies:
+    media-typer "0.3.0"
+    mime-types "~2.1.24"
+
+type@^1.0.1:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/type/-/type-1.2.0.tgz#848dd7698dafa3e54a6c479e759c4bc3f18847a0"
+  integrity sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg==
+
+type@^2.7.2:
+  version "2.7.2"
+  resolved "https://registry.yarnpkg.com/type/-/type-2.7.2.tgz#2376a15a3a28b1efa0f5350dcf72d24df6ef98d0"
+  integrity sha512-dzlvlNlt6AXU7EBSfpAscydQ7gXB+pPGsPnfJnZpiNJBDj7IaJzQlBZYGdEi4R9HmPdBv2XmWJ6YUtoTa7lmCw==
+
+typechain@^8.1.1:
+  version "8.3.1"
+  resolved "https://registry.yarnpkg.com/typechain/-/typechain-8.3.1.tgz#dccbc839b94877997536c356380eff7325395cfb"
+  integrity sha512-fA7clol2IP/56yq6vkMTR+4URF1nGjV82Wx6Rf09EsqD4tkzMAvEaqYxVFCavJm/1xaRga/oD55K+4FtuXwQOQ==
+  dependencies:
+    "@types/prettier" "^2.1.1"
+    debug "^4.3.1"
+    fs-extra "^7.0.0"
+    glob "7.1.7"
+    js-sha3 "^0.8.0"
+    lodash "^4.17.15"
+    mkdirp "^1.0.4"
+    prettier "^2.3.1"
+    ts-command-line-args "^2.2.0"
+    ts-essentials "^7.0.1"
+
+typed-array-buffer@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/typed-array-buffer/-/typed-array-buffer-1.0.0.tgz#18de3e7ed7974b0a729d3feecb94338d1472cd60"
+  integrity sha512-Y8KTSIglk9OZEr8zywiIHG/kmQ7KWyjseXs1CbSo8vC42w7hg2HgYTxSWwP0+is7bWDc1H+Fo026CpHFwm8tkw==
+  dependencies:
+    call-bind "^1.0.2"
+    get-intrinsic "^1.2.1"
+    is-typed-array "^1.1.10"
+
+typed-array-byte-length@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/typed-array-byte-length/-/typed-array-byte-length-1.0.0.tgz#d787a24a995711611fb2b87a4052799517b230d0"
+  integrity sha512-Or/+kvLxNpeQ9DtSydonMxCx+9ZXOswtwJn17SNLvhptaXYDJvkFFP5zbfU/uLmvnBJlI4yrnXRxpdWH/M5tNA==
+  dependencies:
+    call-bind "^1.0.2"
+    for-each "^0.3.3"
+    has-proto "^1.0.1"
+    is-typed-array "^1.1.10"
+
+typed-array-byte-offset@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/typed-array-byte-offset/-/typed-array-byte-offset-1.0.0.tgz#cbbe89b51fdef9cd6aaf07ad4707340abbc4ea0b"
+  integrity sha512-RD97prjEt9EL8YgAgpOkf3O4IF9lhJFr9g0htQkm0rchFp/Vx7LW5Q8fSXXub7BXAODyUQohRMyOc3faCPd0hg==
+  dependencies:
+    available-typed-arrays "^1.0.5"
+    call-bind "^1.0.2"
+    for-each "^0.3.3"
+    has-proto "^1.0.1"
+    is-typed-array "^1.1.10"
+
+typed-array-length@^1.0.4:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/typed-array-length/-/typed-array-length-1.0.4.tgz#89d83785e5c4098bec72e08b319651f0eac9c1bb"
+  integrity sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng==
+  dependencies:
+    call-bind "^1.0.2"
+    for-each "^0.3.3"
+    is-typed-array "^1.1.9"
+
+typedarray-to-buffer@^3.1.5:
+  version "3.1.5"
+  resolved "https://registry.yarnpkg.com/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz#a97ee7a9ff42691b9f783ff1bc5112fe3fca9080"
+  integrity sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==
+  dependencies:
+    is-typedarray "^1.0.0"
+
+typedarray@^0.0.6:
+  version "0.0.6"
+  resolved "https://registry.yarnpkg.com/typedarray/-/typedarray-0.0.6.tgz#867ac74e3864187b1d3d47d996a78ec5c8830777"
+  integrity sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==
+
+typescript@^4.8.4:
+  version "4.9.5"
+  resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.9.5.tgz#095979f9bcc0d09da324d58d03ce8f8374cbe65a"
+  integrity sha512-1FXk9E2Hm+QzZQ7z+McJiHL4NW1F2EzMu9Nq9i3zAaGqibafqYwCVU6WyWAuyQRRzOlxou8xZSyXLEN8oKj24g==
+
+typical@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/typical/-/typical-4.0.0.tgz#cbeaff3b9d7ae1e2bbfaf5a4e6f11eccfde94fc4"
+  integrity sha512-VAH4IvQ7BDFYglMd7BPRDfLgxZZX4O4TFcRDA6EN5X7erNJJq+McIEp8np9aVtxrCJ6qx4GTYVfOWNjcqwZgRw==
+
+typical@^5.2.0:
+  version "5.2.0"
+  resolved "https://registry.yarnpkg.com/typical/-/typical-5.2.0.tgz#4daaac4f2b5315460804f0acf6cb69c52bb93066"
+  integrity sha512-dvdQgNDNJo+8B2uBQoqdb11eUCE1JQXhvjC/CZtgvZseVd5TYMXnq0+vuUemXbd/Se29cTaUuPX3YIc2xgbvIg==
+
+uglify-js@^3.1.4:
+  version "3.17.4"
+  resolved "https://registry.yarnpkg.com/uglify-js/-/uglify-js-3.17.4.tgz#61678cf5fa3f5b7eb789bb345df29afb8257c22c"
+  integrity sha512-T9q82TJI9e/C1TAxYvfb16xO120tMVFZrGA3f9/P4424DNu6ypK103y0GPFVa17yotwSyZW5iYXgjYHkGrJW/g==
+
+ultron@~1.1.0:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/ultron/-/ultron-1.1.1.tgz#9fe1536a10a664a65266a1e3ccf85fd36302bc9c"
+  integrity sha512-UIEXBNeYmKptWH6z8ZnqTeS8fV74zG0/eRU9VGkpzz+LIJNs8W/zM/L+7ctCkRrgbNnnR0xxw4bKOr0cW0N0Og==
+
+unbox-primitive@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/unbox-primitive/-/unbox-primitive-1.0.2.tgz#29032021057d5e6cdbd08c5129c226dff8ed6f9e"
+  integrity sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==
+  dependencies:
+    call-bind "^1.0.2"
+    has-bigints "^1.0.2"
+    has-symbols "^1.0.3"
+    which-boxed-primitive "^1.0.2"
+
+underscore@^1.8.3:
+  version "1.13.6"
+  resolved "https://registry.yarnpkg.com/underscore/-/underscore-1.13.6.tgz#04786a1f589dc6c09f761fc5f45b89e935136441"
+  integrity sha512-+A5Sja4HP1M08MaXya7p5LvjuM7K6q/2EaC0+iovj/wOcMsTzMvDFbasi/oSapiwOlt252IqsKqPjCl7huKS0A==
+
+undici@^5.14.0, undici@^5.4.0:
+  version "5.27.2"
+  resolved "https://registry.yarnpkg.com/undici/-/undici-5.27.2.tgz#a270c563aea5b46cc0df2550523638c95c5d4411"
+  integrity sha512-iS857PdOEy/y3wlM3yRp+6SNQQ6xU0mmZcwRSriqk+et/cwWAtwmIGf6WkoDN2EK/AMdCO/dfXzIwi+rFMrjjQ==
+  dependencies:
+    "@fastify/busboy" "^2.0.0"
+
+universalify@^0.1.0:
+  version "0.1.2"
+  resolved "https://registry.yarnpkg.com/universalify/-/universalify-0.1.2.tgz#b646f69be3942dabcecc9d6639c80dc105efaa66"
+  integrity sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==
+
+universalify@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/universalify/-/universalify-2.0.0.tgz#75a4984efedc4b08975c5aeb73f530d02df25717"
+  integrity sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==
+
+unpipe@1.0.0, unpipe@~1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/unpipe/-/unpipe-1.0.0.tgz#b2bf4ee8514aae6165b4817829d21b2ef49904ec"
+  integrity sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==
+
+upper-case-first@^1.1.0, upper-case-first@^1.1.2:
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/upper-case-first/-/upper-case-first-1.1.2.tgz#5d79bedcff14419518fd2edb0a0507c9b6859115"
+  integrity sha512-wINKYvI3Db8dtjikdAqoBbZoP6Q+PZUyfMR7pmwHzjC2quzSkUq5DmPrTtPEqHaz8AGtmsB4TqwapMTM1QAQOQ==
+  dependencies:
+    upper-case "^1.1.1"
+
+upper-case@^1.0.3, upper-case@^1.1.0, upper-case@^1.1.1, upper-case@^1.1.3:
+  version "1.1.3"
+  resolved "https://registry.yarnpkg.com/upper-case/-/upper-case-1.1.3.tgz#f6b4501c2ec4cdd26ba78be7222961de77621598"
+  integrity sha512-WRbjgmYzgXkCV7zNVpy5YgrHgbBv126rMALQQMrmzOVC4GM2waQ9x7xtm8VU+1yF2kWyPzI9zbZ48n4vSxwfSA==
+
+uri-js@^4.2.2:
+  version "4.4.1"
+  resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e"
+  integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==
+  dependencies:
+    punycode "^2.1.0"
+
+url-set-query@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/url-set-query/-/url-set-query-1.0.0.tgz#016e8cfd7c20ee05cafe7795e892bd0702faa339"
+  integrity sha512-3AChu4NiXquPfeckE5R5cGdiHCMWJx1dwCWOmWIL4KHAziJNOFIYJlpGFeKDvwLPHovZRCxK3cYlwzqI9Vp+Gg==
+
+utf-8-validate@5.0.7:
+  version "5.0.7"
+  resolved "https://registry.yarnpkg.com/utf-8-validate/-/utf-8-validate-5.0.7.tgz#c15a19a6af1f7ad9ec7ddc425747ca28c3644922"
+  integrity sha512-vLt1O5Pp+flcArHGIyKEQq883nBt8nN8tVBcoL0qUXj2XT1n7p70yGIq2VK98I5FdZ1YHc0wk/koOnHjnXWk1Q==
+  dependencies:
+    node-gyp-build "^4.3.0"
+
+utf-8-validate@^5.0.2:
+  version "5.0.10"
+  resolved "https://registry.yarnpkg.com/utf-8-validate/-/utf-8-validate-5.0.10.tgz#d7d10ea39318171ca982718b6b96a8d2442571a2"
+  integrity sha512-Z6czzLq4u8fPOyx7TU6X3dvUZVvoJmxSQ+IcrlmagKhilxlhZgxPK6C5Jqbkw1IDUmFTM+cz9QDnnLTwDz/2gQ==
+  dependencies:
+    node-gyp-build "^4.3.0"
+
+utf8@3.0.0, utf8@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/utf8/-/utf8-3.0.0.tgz#f052eed1364d696e769ef058b183df88c87f69d1"
+  integrity sha512-E8VjFIQ/TyQgp+TZfS6l8yp/xWppSAHzidGiRrqe4bK4XP9pTRyKFgGJpO3SN7zdX4DeomTrwaseCHovfpFcqQ==
+
+util-deprecate@^1.0.1, util-deprecate@~1.0.1:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf"
+  integrity sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==
+
+util@^0.12.5:
+  version "0.12.5"
+  resolved "https://registry.yarnpkg.com/util/-/util-0.12.5.tgz#5f17a6059b73db61a875668781a1c2b136bd6fbc"
+  integrity sha512-kZf/K6hEIrWHI6XqOFUiiMa+79wE/D8Q+NCNAWclkyg3b4d2k7s0QGepNjiABc+aR3N1PAyHL7p6UcLY6LmrnA==
+  dependencies:
+    inherits "^2.0.3"
+    is-arguments "^1.0.4"
+    is-generator-function "^1.0.7"
+    is-typed-array "^1.1.3"
+    which-typed-array "^1.1.2"
+
+utils-merge@1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.1.tgz#9f95710f50a267947b2ccc124741c1028427e713"
+  integrity sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==
+
+uuid@2.0.1:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/uuid/-/uuid-2.0.1.tgz#c2a30dedb3e535d72ccf82e343941a50ba8533ac"
+  integrity sha512-nWg9+Oa3qD2CQzHIP4qKUqwNfzKn8P0LtFhotaCTFchsV7ZfDhAybeip/HZVeMIpZi9JgY1E3nUlwaCmZT1sEg==
+
+uuid@^3.3.2:
+  version "3.4.0"
+  resolved "https://registry.yarnpkg.com/uuid/-/uuid-3.4.0.tgz#b23e4358afa8a202fe7a100af1f5f883f02007ee"
+  integrity sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==
+
+uuid@^8.3.2:
+  version "8.3.2"
+  resolved "https://registry.yarnpkg.com/uuid/-/uuid-8.3.2.tgz#80d5b5ced271bb9af6c445f21a1a04c606cefbe2"
+  integrity sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==
+
+uuid@^9.0.0:
+  version "9.0.0"
+  resolved "https://registry.yarnpkg.com/uuid/-/uuid-9.0.0.tgz#592f550650024a38ceb0c562f2f6aa435761efb5"
+  integrity sha512-MXcSTerfPa4uqyzStbRoTgt5XIe3x5+42+q1sDuy3R5MDk66URdLMOZe5aPX/SQd+kuYAh0FdP/pO28IkQyTeg==
+
+v8-compile-cache-lib@^3.0.1:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz#6336e8d71965cb3d35a1bbb7868445a7c05264bf"
+  integrity sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==
+
+validate-npm-package-license@^3.0.1:
+  version "3.0.4"
+  resolved "https://registry.yarnpkg.com/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz#fc91f6b9c7ba15c857f4cb2c5defeec39d4f410a"
+  integrity sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==
+  dependencies:
+    spdx-correct "^3.0.0"
+    spdx-expression-parse "^3.0.0"
+
+varint@^5.0.0:
+  version "5.0.2"
+  resolved "https://registry.yarnpkg.com/varint/-/varint-5.0.2.tgz#5b47f8a947eb668b848e034dcfa87d0ff8a7f7a4"
+  integrity sha512-lKxKYG6H03yCZUpAGOPOsMcGxd1RHCu1iKvEHYDPmTyq2HueGhD73ssNBqqQWfvYs04G9iUFRvmAVLW20Jw6ow==
+
+vary@^1, vary@~1.1.2:
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.2.tgz#2299f02c6ded30d4a5961b0b9f74524a18f634fc"
+  integrity sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==
+
+verror@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/verror/-/verror-1.10.0.tgz#3a105ca17053af55d6e270c1f8288682e18da400"
+  integrity sha512-ZZKSmDAEFOijERBLkmYfJ+vmk3w+7hOLYDNkRCuRuMJGEmqYNCNLyBBFwWKVMhfwaEF3WOd0Zlw86U/WC/+nYw==
+  dependencies:
+    assert-plus "^1.0.0"
+    core-util-is "1.0.2"
+    extsprintf "^1.2.0"
+
+wcwidth@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/wcwidth/-/wcwidth-1.0.1.tgz#f0b0dcf915bc5ff1528afadb2c0e17b532da2fe8"
+  integrity sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg==
+  dependencies:
+    defaults "^1.0.3"
+
+web3-bzz@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-bzz/-/web3-bzz-1.10.0.tgz#ac74bc71cdf294c7080a79091079192f05c5baed"
+  integrity sha512-o9IR59io3pDUsXTsps5pO5hW1D5zBmg46iNc2t4j2DkaYHNdDLwk2IP9ukoM2wg47QILfPEJYzhTfkS/CcX0KA==
+  dependencies:
+    "@types/node" "^12.12.6"
+    got "12.1.0"
+    swarm-js "^0.1.40"
+
+web3-core-helpers@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-core-helpers/-/web3-core-helpers-1.10.0.tgz#1016534c51a5df77ed4f94d1fcce31de4af37fad"
+  integrity sha512-pIxAzFDS5vnbXvfvLSpaA1tfRykAe9adw43YCKsEYQwH0gCLL0kMLkaCX3q+Q8EVmAh+e1jWL/nl9U0de1+++g==
+  dependencies:
+    web3-eth-iban "1.10.0"
+    web3-utils "1.10.0"
+
+web3-core-method@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-core-method/-/web3-core-method-1.10.0.tgz#82668197fa086e8cc8066742e35a9d72535e3412"
+  integrity sha512-4R700jTLAMKDMhQ+nsVfIXvH6IGJlJzGisIfMKWAIswH31h5AZz7uDUW2YctI+HrYd+5uOAlS4OJeeT9bIpvkA==
+  dependencies:
+    "@ethersproject/transactions" "^5.6.2"
+    web3-core-helpers "1.10.0"
+    web3-core-promievent "1.10.0"
+    web3-core-subscriptions "1.10.0"
+    web3-utils "1.10.0"
+
+web3-core-promievent@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-core-promievent/-/web3-core-promievent-1.10.0.tgz#cbb5b3a76b888df45ed3a8d4d8d4f54ccb66a37b"
+  integrity sha512-68N7k5LWL5R38xRaKFrTFT2pm2jBNFaM4GioS00YjAKXRQ3KjmhijOMG3TICz6Aa5+6GDWYelDNx21YAeZ4YTg==
+  dependencies:
+    eventemitter3 "4.0.4"
+
+web3-core-requestmanager@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-core-requestmanager/-/web3-core-requestmanager-1.10.0.tgz#4b34f6e05837e67c70ff6f6993652afc0d54c340"
+  integrity sha512-3z/JKE++Os62APml4dvBM+GAuId4h3L9ckUrj7ebEtS2AR0ixyQPbrBodgL91Sv7j7cQ3Y+hllaluqjguxvSaQ==
+  dependencies:
+    util "^0.12.5"
+    web3-core-helpers "1.10.0"
+    web3-providers-http "1.10.0"
+    web3-providers-ipc "1.10.0"
+    web3-providers-ws "1.10.0"
+
+web3-core-subscriptions@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-core-subscriptions/-/web3-core-subscriptions-1.10.0.tgz#b534592ee1611788fc0cb0b95963b9b9b6eacb7c"
+  integrity sha512-HGm1PbDqsxejI075gxBc5OSkwymilRWZufIy9zEpnWKNmfbuv5FfHgW1/chtJP6aP3Uq2vHkvTDl3smQBb8l+g==
+  dependencies:
+    eventemitter3 "4.0.4"
+    web3-core-helpers "1.10.0"
+
+web3-core@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-core/-/web3-core-1.10.0.tgz#9aa07c5deb478cf356c5d3b5b35afafa5fa8e633"
+  integrity sha512-fWySwqy2hn3TL89w5TM8wXF1Z2Q6frQTKHWmP0ppRQorEK8NcHJRfeMiv/mQlSKoTS1F6n/nv2uyZsixFycjYQ==
+  dependencies:
+    "@types/bn.js" "^5.1.1"
+    "@types/node" "^12.12.6"
+    bignumber.js "^9.0.0"
+    web3-core-helpers "1.10.0"
+    web3-core-method "1.10.0"
+    web3-core-requestmanager "1.10.0"
+    web3-utils "1.10.0"
+
+web3-eth-abi@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-eth-abi/-/web3-eth-abi-1.10.0.tgz#53a7a2c95a571e205e27fd9e664df4919483cce1"
+  integrity sha512-cwS+qRBWpJ43aI9L3JS88QYPfFcSJJ3XapxOQ4j40v6mk7ATpA8CVK1vGTzpihNlOfMVRBkR95oAj7oL6aiDOg==
+  dependencies:
+    "@ethersproject/abi" "^5.6.3"
+    web3-utils "1.10.0"
+
+web3-eth-accounts@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-eth-accounts/-/web3-eth-accounts-1.10.0.tgz#2942beca0a4291455f32cf09de10457a19a48117"
+  integrity sha512-wiq39Uc3mOI8rw24wE2n15hboLE0E9BsQLdlmsL4Zua9diDS6B5abXG0XhFcoNsXIGMWXVZz4TOq3u4EdpXF/Q==
+  dependencies:
+    "@ethereumjs/common" "2.5.0"
+    "@ethereumjs/tx" "3.3.2"
+    eth-lib "0.2.8"
+    ethereumjs-util "^7.1.5"
+    scrypt-js "^3.0.1"
+    uuid "^9.0.0"
+    web3-core "1.10.0"
+    web3-core-helpers "1.10.0"
+    web3-core-method "1.10.0"
+    web3-utils "1.10.0"
+
+web3-eth-contract@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-eth-contract/-/web3-eth-contract-1.10.0.tgz#8e68c7654576773ec3c91903f08e49d0242c503a"
+  integrity sha512-MIC5FOzP/+2evDksQQ/dpcXhSqa/2hFNytdl/x61IeWxhh6vlFeSjq0YVTAyIzdjwnL7nEmZpjfI6y6/Ufhy7w==
+  dependencies:
+    "@types/bn.js" "^5.1.1"
+    web3-core "1.10.0"
+    web3-core-helpers "1.10.0"
+    web3-core-method "1.10.0"
+    web3-core-promievent "1.10.0"
+    web3-core-subscriptions "1.10.0"
+    web3-eth-abi "1.10.0"
+    web3-utils "1.10.0"
+
+web3-eth-ens@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-eth-ens/-/web3-eth-ens-1.10.0.tgz#96a676524e0b580c87913f557a13ed810cf91cd9"
+  integrity sha512-3hpGgzX3qjgxNAmqdrC2YUQMTfnZbs4GeLEmy8aCWziVwogbuqQZ+Gzdfrym45eOZodk+lmXyLuAdqkNlvkc1g==
+  dependencies:
+    content-hash "^2.5.2"
+    eth-ens-namehash "2.0.8"
+    web3-core "1.10.0"
+    web3-core-helpers "1.10.0"
+    web3-core-promievent "1.10.0"
+    web3-eth-abi "1.10.0"
+    web3-eth-contract "1.10.0"
+    web3-utils "1.10.0"
+
+web3-eth-iban@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-eth-iban/-/web3-eth-iban-1.10.0.tgz#5a46646401965b0f09a4f58e7248c8a8cd22538a"
+  integrity sha512-0l+SP3IGhInw7Q20LY3IVafYEuufo4Dn75jAHT7c2aDJsIolvf2Lc6ugHkBajlwUneGfbRQs/ccYPQ9JeMUbrg==
+  dependencies:
+    bn.js "^5.2.1"
+    web3-utils "1.10.0"
+
+web3-eth-personal@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-eth-personal/-/web3-eth-personal-1.10.0.tgz#94d525f7a29050a0c2a12032df150ac5ea633071"
+  integrity sha512-anseKn98w/d703eWq52uNuZi7GhQeVjTC5/svrBWEKob0WZ5kPdo+EZoFN0sp5a5ubbrk/E0xSl1/M5yORMtpg==
+  dependencies:
+    "@types/node" "^12.12.6"
+    web3-core "1.10.0"
+    web3-core-helpers "1.10.0"
+    web3-core-method "1.10.0"
+    web3-net "1.10.0"
+    web3-utils "1.10.0"
+
+web3-eth@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-eth/-/web3-eth-1.10.0.tgz#38b905e2759697c9624ab080cfcf4e6c60b3a6cf"
+  integrity sha512-Z5vT6slNMLPKuwRyKGbqeGYC87OAy8bOblaqRTgg94CXcn/mmqU7iPIlG4506YdcdK3x6cfEDG7B6w+jRxypKA==
+  dependencies:
+    web3-core "1.10.0"
+    web3-core-helpers "1.10.0"
+    web3-core-method "1.10.0"
+    web3-core-subscriptions "1.10.0"
+    web3-eth-abi "1.10.0"
+    web3-eth-accounts "1.10.0"
+    web3-eth-contract "1.10.0"
+    web3-eth-ens "1.10.0"
+    web3-eth-iban "1.10.0"
+    web3-eth-personal "1.10.0"
+    web3-net "1.10.0"
+    web3-utils "1.10.0"
+
+web3-net@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-net/-/web3-net-1.10.0.tgz#be53e7f5dafd55e7c9013d49c505448b92c9c97b"
+  integrity sha512-NLH/N3IshYWASpxk4/18Ge6n60GEvWBVeM8inx2dmZJVmRI6SJIlUxbL8jySgiTn3MMZlhbdvrGo8fpUW7a1GA==
+  dependencies:
+    web3-core "1.10.0"
+    web3-core-method "1.10.0"
+    web3-utils "1.10.0"
+
+web3-providers-http@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-providers-http/-/web3-providers-http-1.10.0.tgz#864fa48675e7918c9a4374e5f664b32c09d0151b"
+  integrity sha512-eNr965YB8a9mLiNrkjAWNAPXgmQWfpBfkkn7tpEFlghfww0u3I0tktMZiaToJVcL2+Xq+81cxbkpeWJ5XQDwOA==
+  dependencies:
+    abortcontroller-polyfill "^1.7.3"
+    cross-fetch "^3.1.4"
+    es6-promise "^4.2.8"
+    web3-core-helpers "1.10.0"
+
+web3-providers-ipc@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-providers-ipc/-/web3-providers-ipc-1.10.0.tgz#9747c7a6aee96a51488e32fa7c636c3460b39889"
+  integrity sha512-OfXG1aWN8L1OUqppshzq8YISkWrYHaATW9H8eh0p89TlWMc1KZOL9vttBuaBEi96D/n0eYDn2trzt22bqHWfXA==
+  dependencies:
+    oboe "2.1.5"
+    web3-core-helpers "1.10.0"
+
+web3-providers-ws@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-providers-ws/-/web3-providers-ws-1.10.0.tgz#cb0b87b94c4df965cdf486af3a8cd26daf3975e5"
+  integrity sha512-sK0fNcglW36yD5xjnjtSGBnEtf59cbw4vZzJ+CmOWIKGIR96mP5l684g0WD0Eo+f4NQc2anWWXG74lRc9OVMCQ==
+  dependencies:
+    eventemitter3 "4.0.4"
+    web3-core-helpers "1.10.0"
+    websocket "^1.0.32"
+
+web3-shh@1.10.0:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-shh/-/web3-shh-1.10.0.tgz#c2979b87e0f67a7fef2ce9ee853bd7bfbe9b79a8"
+  integrity sha512-uNUUuNsO2AjX41GJARV9zJibs11eq6HtOe6Wr0FtRUcj8SN6nHeYIzwstAvJ4fXA53gRqFMTxdntHEt9aXVjpg==
+  dependencies:
+    web3-core "1.10.0"
+    web3-core-method "1.10.0"
+    web3-core-subscriptions "1.10.0"
+    web3-net "1.10.0"
+
+web3-utils@1.10.0, web3-utils@^1.0.0-beta.31, web3-utils@^1.3.6:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3-utils/-/web3-utils-1.10.0.tgz#ca4c1b431a765c14ac7f773e92e0fd9377ccf578"
+  integrity sha512-kSaCM0uMcZTNUSmn5vMEhlo02RObGNRRCkdX0V9UTAU0+lrvn0HSaudyCo6CQzuXUsnuY2ERJGCGPfeWmv19Rg==
+  dependencies:
+    bn.js "^5.2.1"
+    ethereum-bloom-filters "^1.0.6"
+    ethereumjs-util "^7.1.0"
+    ethjs-unit "0.1.6"
+    number-to-bn "1.7.0"
+    randombytes "^2.1.0"
+    utf8 "3.0.0"
+
+web3@1.10.0, web3@^1.0.0-beta.34:
+  version "1.10.0"
+  resolved "https://registry.yarnpkg.com/web3/-/web3-1.10.0.tgz#2fde0009f59aa756c93e07ea2a7f3ab971091274"
+  integrity sha512-YfKY9wSkGcM8seO+daR89oVTcbu18NsVfvOngzqMYGUU0pPSQmE57qQDvQzUeoIOHAnXEBNzrhjQJmm8ER0rng==
+  dependencies:
+    web3-bzz "1.10.0"
+    web3-core "1.10.0"
+    web3-eth "1.10.0"
+    web3-eth-personal "1.10.0"
+    web3-net "1.10.0"
+    web3-shh "1.10.0"
+    web3-utils "1.10.0"
+
+webidl-conversions@^3.0.0:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871"
+  integrity sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==
+
+websocket@^1.0.32:
+  version "1.0.34"
+  resolved "https://registry.yarnpkg.com/websocket/-/websocket-1.0.34.tgz#2bdc2602c08bf2c82253b730655c0ef7dcab3111"
+  integrity sha512-PRDso2sGwF6kM75QykIesBijKSVceR6jL2G8NGYyq2XrItNC2P5/qL5XeR056GhA+Ly7JMFvJb9I312mJfmqnQ==
+  dependencies:
+    bufferutil "^4.0.1"
+    debug "^2.2.0"
+    es5-ext "^0.10.50"
+    typedarray-to-buffer "^3.1.5"
+    utf-8-validate "^5.0.2"
+    yaeti "^0.0.6"
+
+whatwg-url@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/whatwg-url/-/whatwg-url-5.0.0.tgz#966454e8765462e37644d3626f6742ce8b70965d"
+  integrity sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==
+  dependencies:
+    tr46 "~0.0.3"
+    webidl-conversions "^3.0.0"
+
+which-boxed-primitive@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz#13757bc89b209b049fe5d86430e21cf40a89a8e6"
+  integrity sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==
+  dependencies:
+    is-bigint "^1.0.1"
+    is-boolean-object "^1.1.0"
+    is-number-object "^1.0.4"
+    is-string "^1.0.5"
+    is-symbol "^1.0.3"
+
+which-module@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/which-module/-/which-module-1.0.0.tgz#bba63ca861948994ff307736089e3b96026c2a4f"
+  integrity sha512-F6+WgncZi/mJDrammbTuHe1q0R5hOXv/mBaiNA2TCNT/LTHusX0V+CJnj9XT8ki5ln2UZyyddDgHfCzyrOH7MQ==
+
+which-module@^2.0.0:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/which-module/-/which-module-2.0.1.tgz#776b1fe35d90aebe99e8ac15eb24093389a4a409"
+  integrity sha512-iBdZ57RDvnOR9AGBhML2vFZf7h8vmBjhoaZqODJBFWHVtKkDmKuHai3cx5PgVMrX5YDNp27AofYbAwctSS+vhQ==
+
+which-pm-runs@^1.0.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/which-pm-runs/-/which-pm-runs-1.1.0.tgz#35ccf7b1a0fce87bd8b92a478c9d045785d3bf35"
+  integrity sha512-n1brCuqClxfFfq/Rb0ICg9giSZqCS+pLtccdag6C2HyufBrh3fBOiy9nb6ggRMvWOVH5GrdJskj5iGTZNxd7SA==
+
+which-typed-array@^1.1.10, which-typed-array@^1.1.11, which-typed-array@^1.1.2:
+  version "1.1.11"
+  resolved "https://registry.yarnpkg.com/which-typed-array/-/which-typed-array-1.1.11.tgz#99d691f23c72aab6768680805a271b69761ed61a"
+  integrity sha512-qe9UWWpkeG5yzZ0tNYxDmd7vo58HDBc39mZ0xWWpolAGADdFOzkfamWLDxkOWcvHQKVmdTyQdLD4NOfjLWTKew==
+  dependencies:
+    available-typed-arrays "^1.0.5"
+    call-bind "^1.0.2"
+    for-each "^0.3.3"
+    gopd "^1.0.1"
+    has-tostringtag "^1.0.0"
+
+which@1.3.1, which@^1.1.1, which@^1.3.1:
+  version "1.3.1"
+  resolved "https://registry.yarnpkg.com/which/-/which-1.3.1.tgz#a45043d54f5805316da8d62f9f50918d3da70b0a"
+  integrity sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==
+  dependencies:
+    isexe "^2.0.0"
+
+which@^2.0.1:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/which/-/which-2.0.2.tgz#7c6a8dd0a636a0327e10b59c9286eee93f3f51b1"
+  integrity sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==
+  dependencies:
+    isexe "^2.0.0"
+
+wide-align@1.1.3:
+  version "1.1.3"
+  resolved "https://registry.yarnpkg.com/wide-align/-/wide-align-1.1.3.tgz#ae074e6bdc0c14a431e804e624549c633b000457"
+  integrity sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==
+  dependencies:
+    string-width "^1.0.2 || 2"
+
+window-size@^0.2.0:
+  version "0.2.0"
+  resolved "https://registry.yarnpkg.com/window-size/-/window-size-0.2.0.tgz#b4315bb4214a3d7058ebeee892e13fa24d98b075"
+  integrity sha512-UD7d8HFA2+PZsbKyaOCEy8gMh1oDtHgJh1LfgjQ4zVXmYjAT/kvz3PueITKuqDiIXQe7yzpPnxX3lNc+AhQMyw==
+
+word-wrap@~1.2.3:
+  version "1.2.5"
+  resolved "https://registry.yarnpkg.com/word-wrap/-/word-wrap-1.2.5.tgz#d2c45c6dd4fbce621a66f136cbe328afd0410b34"
+  integrity sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==
+
+wordwrap@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/wordwrap/-/wordwrap-1.0.0.tgz#27584810891456a4171c8d0226441ade90cbcaeb"
+  integrity sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==
+
+wordwrapjs@^4.0.0:
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/wordwrapjs/-/wordwrapjs-4.0.1.tgz#d9790bccfb110a0fc7836b5ebce0937b37a8b98f"
+  integrity sha512-kKlNACbvHrkpIw6oPeYDSmdCTu2hdMHoyXLTcUKala++lx5Y+wjJ/e474Jqv5abnVmwxw08DiTuHmw69lJGksA==
+  dependencies:
+    reduce-flatten "^2.0.0"
+    typical "^5.2.0"
+
+workerpool@6.2.1:
+  version "6.2.1"
+  resolved "https://registry.yarnpkg.com/workerpool/-/workerpool-6.2.1.tgz#46fc150c17d826b86a008e5a4508656777e9c343"
+  integrity sha512-ILEIE97kDZvF9Wb9f6h5aXK4swSlKGUcOEGiIYb2OOu/IrDU9iwj0fD//SsA6E5ibwJxpEvhullJY4Sl4GcpAw==
+
+wrap-ansi@^2.0.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-2.1.0.tgz#d8fc3d284dd05794fe84973caecdd1cf824fdd85"
+  integrity sha512-vAaEaDM946gbNpH5pLVNR+vX2ht6n0Bt3GXwVB1AuAqZosOvHNF3P7wDnh8KLkSqgUh0uh77le7Owgoz+Z9XBw==
+  dependencies:
+    string-width "^1.0.1"
+    strip-ansi "^3.0.1"
+
+wrap-ansi@^5.1.0:
+  version "5.1.0"
+  resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-5.1.0.tgz#1fd1f67235d5b6d0fee781056001bfb694c03b09"
+  integrity sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==
+  dependencies:
+    ansi-styles "^3.2.0"
+    string-width "^3.0.0"
+    strip-ansi "^5.0.0"
+
+wrap-ansi@^7.0.0:
+  version "7.0.0"
+  resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43"
+  integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==
+  dependencies:
+    ansi-styles "^4.0.0"
+    string-width "^4.1.0"
+    strip-ansi "^6.0.0"
+
+wrappy@1:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f"
+  integrity sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==
+
+ws@7.4.6:
+  version "7.4.6"
+  resolved "https://registry.yarnpkg.com/ws/-/ws-7.4.6.tgz#5654ca8ecdeee47c33a9a4bf6d28e2be2980377c"
+  integrity sha512-YmhHDO4MzaDLB+M9ym/mDA5z0naX8j7SIlT8f8z+I0VtzsRbekxEutHSme7NPS2qE8StCYQNUnfWdXta/Yu85A==
+
+ws@8.2.3:
+  version "8.2.3"
+  resolved "https://registry.yarnpkg.com/ws/-/ws-8.2.3.tgz#63a56456db1b04367d0b721a0b80cae6d8becbba"
+  integrity sha512-wBuoj1BDpC6ZQ1B7DWQBYVLphPWkm8i9Y0/3YdHjHKHiohOJ1ws+3OccDWtH+PoC9DZD5WOTrJvNbWvjS6JWaA==
+
+ws@^3.0.0:
+  version "3.3.3"
+  resolved "https://registry.yarnpkg.com/ws/-/ws-3.3.3.tgz#f1cf84fe2d5e901ebce94efaece785f187a228f2"
+  integrity sha512-nnWLa/NwZSt4KQJu51MYlCcSQ5g7INpOrOMt4XV8j4dqTXdmlUmSHQ8/oLC069ckre0fRsgfvsKwbTdtKLCDkA==
+  dependencies:
+    async-limiter "~1.0.0"
+    safe-buffer "~5.1.0"
+    ultron "~1.1.0"
+
+ws@^7.4.6:
+  version "7.5.9"
+  resolved "https://registry.yarnpkg.com/ws/-/ws-7.5.9.tgz#54fa7db29f4c7cec68b1ddd3a89de099942bb591"
+  integrity sha512-F+P9Jil7UiSKSkppIiD94dN07AwvFixvLIj1Og1Rl9GGMuNipJnV9JzjD6XuqmAeiswGvUmNLjr5cFuXwNS77Q==
+
+xhr-request-promise@^0.1.2:
+  version "0.1.3"
+  resolved "https://registry.yarnpkg.com/xhr-request-promise/-/xhr-request-promise-0.1.3.tgz#2d5f4b16d8c6c893be97f1a62b0ed4cf3ca5f96c"
+  integrity sha512-YUBytBsuwgitWtdRzXDDkWAXzhdGB8bYm0sSzMPZT7Z2MBjMSTHFsyCT1yCRATY+XC69DUrQraRAEgcoCRaIPg==
+  dependencies:
+    xhr-request "^1.1.0"
+
+xhr-request@^1.0.1, xhr-request@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/xhr-request/-/xhr-request-1.1.0.tgz#f4a7c1868b9f198723444d82dcae317643f2e2ed"
+  integrity sha512-Y7qzEaR3FDtL3fP30k9wO/e+FBnBByZeybKOhASsGP30NIkRAAkKD/sCnLvgEfAIEC1rcmK7YG8f4oEnIrrWzA==
+  dependencies:
+    buffer-to-arraybuffer "^0.0.5"
+    object-assign "^4.1.1"
+    query-string "^5.0.1"
+    simple-get "^2.7.0"
+    timed-out "^4.0.1"
+    url-set-query "^1.0.0"
+    xhr "^2.0.4"
+
+xhr@^2.0.4, xhr@^2.3.3:
+  version "2.6.0"
+  resolved "https://registry.yarnpkg.com/xhr/-/xhr-2.6.0.tgz#b69d4395e792b4173d6b7df077f0fc5e4e2b249d"
+  integrity sha512-/eCGLb5rxjx5e3mF1A7s+pLlR6CGyqWN91fv1JgER5mVWg1MZmlhBvy9kjcsOdRk8RrIujotWyJamfyrp+WIcA==
+  dependencies:
+    global "~4.4.0"
+    is-function "^1.0.1"
+    parse-headers "^2.0.0"
+    xtend "^4.0.0"
+
+xmlhttprequest@1.8.0:
+  version "1.8.0"
+  resolved "https://registry.yarnpkg.com/xmlhttprequest/-/xmlhttprequest-1.8.0.tgz#67fe075c5c24fef39f9d65f5f7b7fe75171968fc"
+  integrity sha512-58Im/U0mlVBLM38NdZjHyhuMtCqa61469k2YP/AaPbvCoV9aQGUpbJBj1QRm2ytRiVQBD/fsw7L2bJGDVQswBA==
+
+xtend@^4.0.0:
+  version "4.0.2"
+  resolved "https://registry.yarnpkg.com/xtend/-/xtend-4.0.2.tgz#bb72779f5fa465186b1f438f674fa347fdb5db54"
+  integrity sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==
+
+y18n@^3.2.1:
+  version "3.2.2"
+  resolved "https://registry.yarnpkg.com/y18n/-/y18n-3.2.2.tgz#85c901bd6470ce71fc4bb723ad209b70f7f28696"
+  integrity sha512-uGZHXkHnhF0XeeAPgnKfPv1bgKAYyVvmNL1xlKsPYZPaIHxGti2hHqvOCQv71XMsLxu1QjergkqogUnms5D3YQ==
+
+y18n@^4.0.0:
+  version "4.0.3"
+  resolved "https://registry.yarnpkg.com/y18n/-/y18n-4.0.3.tgz#b5f259c82cd6e336921efd7bfd8bf560de9eeedf"
+  integrity sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ==
+
+y18n@^5.0.5:
+  version "5.0.8"
+  resolved "https://registry.yarnpkg.com/y18n/-/y18n-5.0.8.tgz#7f4934d0f7ca8c56f95314939ddcd2dd91ce1d55"
+  integrity sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==
+
+yaeti@^0.0.6:
+  version "0.0.6"
+  resolved "https://registry.yarnpkg.com/yaeti/-/yaeti-0.0.6.tgz#f26f484d72684cf42bedfb76970aa1608fbf9577"
+  integrity sha512-MvQa//+KcZCUkBTIC9blM+CU9J2GzuTytsOUwf2lidtvkx/6gnEp1QvJv34t9vdjhFmha/mUiNDbN0D0mJWdug==
+
+yallist@^3.0.0, yallist@^3.0.2, yallist@^3.1.1:
+  version "3.1.1"
+  resolved "https://registry.yarnpkg.com/yallist/-/yallist-3.1.1.tgz#dbb7daf9bfd8bac9ab45ebf602b8cbad0d5d08fd"
+  integrity sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==
+
+yallist@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz#9bb92790d9c0effec63be73519e11a35019a3a72"
+  integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==
+
+yaml@^1.10.0:
+  version "1.10.2"
+  resolved "https://registry.yarnpkg.com/yaml/-/yaml-1.10.2.tgz#2301c5ffbf12b467de8da2333a459e29e7920e4b"
+  integrity sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==
+
+yargs-parser@13.1.2, yargs-parser@^13.1.2:
+  version "13.1.2"
+  resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-13.1.2.tgz#130f09702ebaeef2650d54ce6e3e5706f7a4fb38"
+  integrity sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==
+  dependencies:
+    camelcase "^5.0.0"
+    decamelize "^1.2.0"
+
+yargs-parser@20.2.4:
+  version "20.2.4"
+  resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.4.tgz#b42890f14566796f85ae8e3a25290d205f154a54"
+  integrity sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==
+
+yargs-parser@^2.4.1:
+  version "2.4.1"
+  resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-2.4.1.tgz#85568de3cf150ff49fa51825f03a8c880ddcc5c4"
+  integrity sha512-9pIKIJhnI5tonzG6OnCFlz/yln8xHYcGl+pn3xR0Vzff0vzN1PbNRaelgfgRUwZ3s4i3jvxT9WhmUGL4whnasA==
+  dependencies:
+    camelcase "^3.0.0"
+    lodash.assign "^4.0.6"
+
+yargs-parser@^20.2.2:
+  version "20.2.9"
+  resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.9.tgz#2eb7dc3b0289718fc295f362753845c41a0c94ee"
+  integrity sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==
+
+yargs-unparser@1.6.0:
+  version "1.6.0"
+  resolved "https://registry.yarnpkg.com/yargs-unparser/-/yargs-unparser-1.6.0.tgz#ef25c2c769ff6bd09e4b0f9d7c605fb27846ea9f"
+  integrity sha512-W9tKgmSn0DpSatfri0nx52Joq5hVXgeLiqR/5G0sZNDoLZFOr/xjBUDcShCOGNsBnEMNo1KAMBkTej1Hm62HTw==
+  dependencies:
+    flat "^4.1.0"
+    lodash "^4.17.15"
+    yargs "^13.3.0"
+
+yargs-unparser@2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/yargs-unparser/-/yargs-unparser-2.0.0.tgz#f131f9226911ae5d9ad38c432fe809366c2325eb"
+  integrity sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==
+  dependencies:
+    camelcase "^6.0.0"
+    decamelize "^4.0.0"
+    flat "^5.0.2"
+    is-plain-obj "^2.1.0"
+
+yargs@13.3.2, yargs@^13.3.0:
+  version "13.3.2"
+  resolved "https://registry.yarnpkg.com/yargs/-/yargs-13.3.2.tgz#ad7ffefec1aa59565ac915f82dccb38a9c31a2dd"
+  integrity sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==
+  dependencies:
+    cliui "^5.0.0"
+    find-up "^3.0.0"
+    get-caller-file "^2.0.1"
+    require-directory "^2.1.1"
+    require-main-filename "^2.0.0"
+    set-blocking "^2.0.0"
+    string-width "^3.0.0"
+    which-module "^2.0.0"
+    y18n "^4.0.0"
+    yargs-parser "^13.1.2"
+
+yargs@16.2.0:
+  version "16.2.0"
+  resolved "https://registry.yarnpkg.com/yargs/-/yargs-16.2.0.tgz#1c82bf0f6b6a66eafce7ef30e376f49a12477f66"
+  integrity sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==
+  dependencies:
+    cliui "^7.0.2"
+    escalade "^3.1.1"
+    get-caller-file "^2.0.5"
+    require-directory "^2.1.1"
+    string-width "^4.2.0"
+    y18n "^5.0.5"
+    yargs-parser "^20.2.2"
+
+yargs@^4.7.1:
+  version "4.8.1"
+  resolved "https://registry.yarnpkg.com/yargs/-/yargs-4.8.1.tgz#c0c42924ca4aaa6b0e6da1739dfb216439f9ddc0"
+  integrity sha512-LqodLrnIDM3IFT+Hf/5sxBnEGECrfdC1uIbgZeJmESCSo4HoCAaKEus8MylXHAkdacGc0ye+Qa+dpkuom8uVYA==
+  dependencies:
+    cliui "^3.2.0"
+    decamelize "^1.1.1"
+    get-caller-file "^1.0.1"
+    lodash.assign "^4.0.3"
+    os-locale "^1.4.0"
+    read-pkg-up "^1.0.1"
+    require-directory "^2.1.1"
+    require-main-filename "^1.0.1"
+    set-blocking "^2.0.0"
+    string-width "^1.0.1"
+    which-module "^1.0.0"
+    window-size "^0.2.0"
+    y18n "^3.2.1"
+    yargs-parser "^2.4.1"
+
+yesno@^0.4.0:
+  version "0.4.0"
+  resolved "https://registry.yarnpkg.com/yesno/-/yesno-0.4.0.tgz#5d674f14d339f0bd4b0edc47f899612c74fcd895"
+  integrity sha512-tdBxmHvbXPBKYIg81bMCB7bVeDmHkRzk5rVJyYYXurwKkHq/MCd8rz4HSJUP7hW0H2NlXiq8IFiWvYKEHhlotA==
+
+yn@3.1.1:
+  version "3.1.1"
+  resolved "https://registry.yarnpkg.com/yn/-/yn-3.1.1.tgz#1e87401a09d767c1d5eab26a6e4c185182d2eb50"
+  integrity sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==
+
+yocto-queue@^0.1.0:
+  version "0.1.0"
+  resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b"
+  integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==
+
+zksync-web3@^0.14.3:
+  version "0.14.3"
+  resolved "https://registry.yarnpkg.com/zksync-web3/-/zksync-web3-0.14.3.tgz#64ac2a16d597464c3fc4ae07447a8007631c57c9"
+  integrity sha512-hT72th4AnqyLW1d5Jlv8N2B/qhEnl2NePK2A3org7tAa24niem/UAaHMkEvmWI3SF9waYUPtqAtjpf+yvQ9zvQ==
diff --git a/forks/seaport b/forks/seaport
new file mode 160000
index 00000000..edd1b827
--- /dev/null
+++ b/forks/seaport
@@ -0,0 +1 @@
+Subproject commit edd1b827e3e2ca393dd08b228d27914c2de50193
diff --git a/hardhat.config.ts b/hardhat.config.ts
index 25a399a8..63ce55b2 100644
--- a/hardhat.config.ts
+++ b/hardhat.config.ts
@@ -25,13 +25,26 @@ task("accounts", "Prints the list of accounts", async (taskArgs, hre) => {
 // Go to https://hardhat.org/config/ to learn more
 const config: HardhatUserConfig = {
   solidity: {
-    version: "0.8.19",
-    settings: {
-      optimizer: {
-        enabled: true,
-        runs: 200,
+    compilers: [
+      {
+        version: "0.8.19",
+        settings: {
+          optimizer: {
+            enabled: true,
+            runs: 200,
+          },
+        },
       },
-    },
+      {
+        version: "0.8.17",
+        settings: {
+          optimizer: {
+            enabled: true,
+            runs: 200,
+          },
+        },
+      },
+    ],
   },
   paths: {
     tests: "./test",
diff --git a/package.json b/package.json
index d8f3385a..730ac3b4 100644
--- a/package.json
+++ b/package.json
@@ -10,7 +10,7 @@
   ],
   "scripts": {
     "compile": "hardhat compile",
-    "build": "tsc",
+    "build": "git submodule update --remote && ts-node scripts/build.ts && yarn install && tsc",
     "test": "hardhat test",
     "prettier:solidity": "./node_modules/.bin/prettier --write contracts/**/*.sol",
     "lint": "eslint . --ext .ts,.tsx --fix && solhint --config ./.solhint.json  contracts/**/*.sol",
@@ -54,6 +54,7 @@
     "hardhat-gas-reporter": "^1.0.9",
     "prettier": "^2.8.8",
     "prettier-plugin-solidity": "^1.1.3",
+    "replace-in-file": "^7.0.2",
     "solhint": "^3.3.8",
     "solhint-plugin-prettier": "^0.0.5",
     "solidity-coverage": "^0.8.4",
@@ -62,8 +63,13 @@
     "typescript": "^4.9.5"
   },
   "dependencies": {
-    "@openzeppelin/contracts": "^4.9.3",
+    "@openzeppelin/contracts": "^4.3.2",
     "solidity-bits": "^0.4.0",
-    "solidity-bytes-utils": "^0.8.0"
+    "solidity-bytes-utils": "^0.8.0",
+    "seaport-core": "immutable/seaport-core#1.5.0+im.1",
+    "seaport-sol": "^1.5.0",
+    "seaport-types": "^0.0.1",
+    "@openzeppelin/contracts/seaport": "npm:@openzeppelin/contracts@^4.9.2",
+    "solady": "^0.0.84"
   }
-}
+}
\ No newline at end of file
diff --git a/scripts/build.ts b/scripts/build.ts
new file mode 100644
index 00000000..9b8f1c86
--- /dev/null
+++ b/scripts/build.ts
@@ -0,0 +1,50 @@
+import fs from "fs";
+import replace from "replace-in-file";
+import forks from "../forks/forks.json";
+import main from "../package.json";
+
+forks.forEach((fork) => {
+  const dest = `contracts/${fork.name}`;
+  fs.rmSync(dest, { recursive: true, force: true });
+  fs.cpSync(`forks/${fork.contracts}`, dest, {
+    recursive: true,
+    filter: (src: string, dest: string) => {
+      const matches = fork.ignore.filter((ig) => src.includes(`forks/${fork.contracts}/${ig}`));
+      // only include if there are no matches
+      return matches.length === 0;
+    },
+  });
+
+  const rmTemplate = `DO NOT MODIFY THESE CONTRACTS DIRECTLY. This folder has been automatically extracted from ${fork.upstream} via a submodule in this repository's forks directory. See the upstream repository for full context.`;
+
+  fs.writeFileSync(`${dest}/README.md`, rmTemplate);
+
+  if (fork.dependencies) {
+    const pkg = JSON.parse(fs.readFileSync(`forks/${fork.name}/package.json`));
+    fork.dependencies.forEach((dep) => {
+      const depVersion = pkg.dependencies[dep] ? pkg.dependencies[dep] : pkg.devDependencies[dep];
+
+      const mainDepVersion = (main.dependencies as any)[dep];
+      // What happens if the version clashes!
+      if (mainDepVersion && depVersion !== mainDepVersion) {
+        // Create a custom name e.g. @openzeppelin/contracts/seaport
+        const customDepName = `${dep}/${fork.name}`;
+        (main.dependencies as any)[customDepName] = `npm:${dep}@${depVersion}`;
+        // Now replace all of the references to the dependency
+        // inside the fork's copied contracts folder
+        // e.g. '@openzeppelin/contracts/x/y.sol --> @openzeppelin/contracts/seaport/x/y.sol
+        replace.sync({
+          from: dep,
+          to: customDepName,
+          files: [`${dest}/**/*.sol`],
+        });
+      } else {
+        // No clash in the dependencies, just add it
+        (main.dependencies as any)[dep] = depVersion;
+      }
+    });
+  }
+});
+
+// Update our main package.json
+fs.writeFileSync("package.json", JSON.stringify(main, null, 2), "utf8");
diff --git a/yarn.lock b/yarn.lock
index 8465a95c..de25f9f9 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -1142,10 +1142,15 @@
     find-up "^4.1.0"
     fs-extra "^8.1.0"
 
-"@openzeppelin/contracts@^4.9.3":
-  version "4.9.3"
-  resolved "https://registry.yarnpkg.com/@openzeppelin/contracts/-/contracts-4.9.3.tgz#00d7a8cf35a475b160b3f0293a6403c511099364"
-  integrity sha512-He3LieZ1pP2TNt5JbkPA4PNT9WC3gOTOlDcFGJW4Le4QKqwmiNJCRt44APfxMxvq7OugU/cqYuPcSBzOw38DAg==
+"@openzeppelin/contracts/seaport@npm:@openzeppelin/contracts@^4.9.2":
+  version "4.9.5"
+  resolved "https://registry.yarnpkg.com/@openzeppelin/contracts/-/contracts-4.9.5.tgz#1eed23d4844c861a1835b5d33507c1017fa98de8"
+  integrity sha512-ZK+W5mVhRppff9BE6YdR8CC52C8zAvsVAiWhEtQ5+oNxFE6h1WdeWo+FJSF8KKvtxxVYZ7MTP/5KoVpAU3aSWg==
+
+"@openzeppelin/contracts@^4.3.2":
+  version "4.9.4"
+  resolved "https://registry.yarnpkg.com/@openzeppelin/contracts/-/contracts-4.9.4.tgz#9241960f6aa5b9f73eb59452d6d85b9e7f21f393"
+  integrity sha512-cZ47skkw2iUz7ApIkgin5bNff7GkCJCZev48HKp81+sRBE0So9yi+Nm5O9G2BMysbjSdA9o9dKDUx0J9Yy1LUQ==
 
 "@openzeppelin/test-helpers@^0.5.16":
   version "0.5.16"
@@ -2934,6 +2939,15 @@ cliui@^7.0.2:
     strip-ansi "^6.0.0"
     wrap-ansi "^7.0.0"
 
+cliui@^8.0.1:
+  version "8.0.1"
+  resolved "https://registry.yarnpkg.com/cliui/-/cliui-8.0.1.tgz#0c04b075db02cbfe60dc8e6cf2f5486b1a3608aa"
+  integrity sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==
+  dependencies:
+    string-width "^4.2.0"
+    strip-ansi "^6.0.1"
+    wrap-ansi "^7.0.0"
+
 clone-response@^1.0.2:
   version "1.0.3"
   resolved "https://registry.yarnpkg.com/clone-response/-/clone-response-1.0.3.tgz#af2032aa47816399cf5f0a1d0db902f517abb8c3"
@@ -4937,7 +4951,7 @@ glob@^7.0.0, glob@^7.1.3:
     once "^1.3.0"
     path-is-absolute "^1.0.0"
 
-glob@^8.0.3:
+glob@^8.0.3, glob@^8.1.0:
   version "8.1.0"
   resolved "https://registry.yarnpkg.com/glob/-/glob-8.1.0.tgz#d388f656593ef708ee3e34640fdfb99a9fd1c33e"
   integrity sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==
@@ -7521,6 +7535,15 @@ regexpp@^3.0.0:
   resolved "https://registry.yarnpkg.com/regexpp/-/regexpp-3.2.0.tgz#0425a2768d8f23bad70ca4b90461fa2f1213e1b2"
   integrity sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==
 
+replace-in-file@^7.0.2:
+  version "7.0.2"
+  resolved "https://registry.yarnpkg.com/replace-in-file/-/replace-in-file-7.0.2.tgz#c38e2143134836586f902319e72f47b55e7c5fae"
+  integrity sha512-tPG+Qmqf+x2Rf1WVdb/9B5tFIf6KJ5hs3fgxh1OTzPRUugPPvyAva7NvCJtnSpmyq6r+ABYcuUOqZkm6yzGSUw==
+  dependencies:
+    chalk "^4.1.2"
+    glob "^8.1.0"
+    yargs "^17.7.2"
+
 req-cwd@^2.0.0:
   version "2.0.0"
   resolved "https://registry.yarnpkg.com/req-cwd/-/req-cwd-2.0.0.tgz#d4082b4d44598036640fb73ddea01ed53db49ebc"
@@ -7786,6 +7809,32 @@ scrypt-js@3.0.1, scrypt-js@^3.0.0, scrypt-js@^3.0.1:
   resolved "https://registry.yarnpkg.com/scrypt-js/-/scrypt-js-3.0.1.tgz#d314a57c2aef69d1ad98a138a21fe9eafa9ee312"
   integrity sha512-cdwTTnqPu0Hyvf5in5asVdZocVDTNRmR7XEcJuIzMjJeSHybHl7vpB66AzwTaIg6CLSbtjcxc8fqcySfnTkccA==
 
+seaport-core@^0.0.1:
+  version "0.0.1"
+  resolved "https://registry.yarnpkg.com/seaport-core/-/seaport-core-0.0.1.tgz#99db0b605d0fbbfd43ca7a4724e64374ce47f6d4"
+  integrity sha512-fgdSIC0ru8xK+fdDfF4bgTFH8ssr6EwbPejC2g/JsWzxy+FvG7JfaX57yn/eIv6hoscgZL87Rm+kANncgwLH3A==
+  dependencies:
+    seaport-types "^0.0.1"
+
+seaport-core@immutable/seaport-core#1.5.0+im.1:
+  version "1.5.0"
+  resolved "https://codeload.github.com/immutable/seaport-core/tar.gz/33e9030f308500b422926a1be12d7a1e4d6adc06"
+  dependencies:
+    seaport-types "^0.0.1"
+
+seaport-sol@^1.5.0:
+  version "1.5.3"
+  resolved "https://registry.yarnpkg.com/seaport-sol/-/seaport-sol-1.5.3.tgz#ccb0047bcefb7d29bcd379faddf3a5a9902d0c3a"
+  integrity sha512-6g91hs15v4zBx/ZN9YD0evhQSDhdMKu83c2CgBgtc517D8ipaYaFO71ZD6V0Z6/I0cwNfuN/ZljcA1J+xXr65A==
+  dependencies:
+    seaport-core "^0.0.1"
+    seaport-types "^0.0.1"
+
+seaport-types@^0.0.1:
+  version "0.0.1"
+  resolved "https://registry.yarnpkg.com/seaport-types/-/seaport-types-0.0.1.tgz#e2a32fe8641853d7dadb1b0232d911d88ccc3f1a"
+  integrity sha512-m7MLa7sq3YPwojxXiVvoX1PM9iNVtQIn7AdEtBnKTwgxPfGRWUlbs/oMgetpjT/ZYTmv3X5/BghOcstWYvKqRA==
+
 secp256k1@4.0.3, secp256k1@^4.0.1:
   version "4.0.3"
   resolved "https://registry.yarnpkg.com/secp256k1/-/secp256k1-4.0.3.tgz#c4559ecd1b8d3c1827ed2d1b94190d69ce267303"
@@ -8009,6 +8058,11 @@ snake-case@^2.1.0:
   dependencies:
     no-case "^2.2.0"
 
+solady@^0.0.84:
+  version "0.0.84"
+  resolved "https://registry.yarnpkg.com/solady/-/solady-0.0.84.tgz#95476df1936ef349003e88d8a4853185eb0b7267"
+  integrity sha512-1ccuZWcMR+g8Ont5LUTqMSFqrmEC+rYAbo6DjZFzdL7AJAtLiaOzO25BKZR10h6YBZNaqO5zUOFy09R6/AzAKQ==
+
 solc@0.7.3:
   version "0.7.3"
   resolved "https://registry.yarnpkg.com/solc/-/solc-0.7.3.tgz#04646961bd867a744f63d2b4e3c0701ffdc7d78a"
@@ -9710,6 +9764,11 @@ yargs-parser@^20.2.2:
   resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.9.tgz#2eb7dc3b0289718fc295f362753845c41a0c94ee"
   integrity sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==
 
+yargs-parser@^21.1.1:
+  version "21.1.1"
+  resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-21.1.1.tgz#9096bceebf990d21bb31fa9516e0ede294a77d35"
+  integrity sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==
+
 yargs-unparser@1.6.0:
   version "1.6.0"
   resolved "https://registry.yarnpkg.com/yargs-unparser/-/yargs-unparser-1.6.0.tgz#ef25c2c769ff6bd09e4b0f9d7c605fb27846ea9f"
@@ -9758,6 +9817,19 @@ yargs@16.2.0:
     y18n "^5.0.5"
     yargs-parser "^20.2.2"
 
+yargs@^17.7.2:
+  version "17.7.2"
+  resolved "https://registry.yarnpkg.com/yargs/-/yargs-17.7.2.tgz#991df39aca675a192b816e1e0363f9d75d2aa269"
+  integrity sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==
+  dependencies:
+    cliui "^8.0.1"
+    escalade "^3.1.1"
+    get-caller-file "^2.0.5"
+    require-directory "^2.1.1"
+    string-width "^4.2.3"
+    y18n "^5.0.5"
+    yargs-parser "^21.1.1"
+
 yargs@^4.7.1:
   version "4.8.1"
   resolved "https://registry.yarnpkg.com/yargs/-/yargs-4.8.1.tgz#c0c42924ca4aaa6b0e6da1739dfb216439f9ddc0"