"use strict"; var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); }; import assert from "assert"; //import Web3HttpProvider from "web3-providers-http"; import { ethers } from "ethers"; const bnify = ethers.BigNumber.from; const blockchainData = { homestead: { addresses: [ { address: "0xAC1639CF97a3A46D431e6d1216f576622894cBB5", balance: bnify("4813414100000000"), code: "0x" }, // Splitter contract { address: "0x3474627D4F63A678266BC17171D87f8570936622", code: "0x606060405260e060020a60003504630b3ed5368114602e57806337b0574a14605257806356fa47f0146062575b005b602c6004356000546101009004600160a060020a03908116339091161460bb575b50565b60005460ff166060908152602090f35b602c60043560005460ff1615609657600160a060020a038116600034606082818181858883f193505050501515604f576002565b33600160a060020a0316600034606082818181858883f193505050501515604f576002565b600080546101009004600160a060020a03169082606082818181858883f193505050501515604f57600256", storage: { "0": "0x0000000000000000000000b2682160c482eb985ec9f3e364eec0a904c44c2300" } }, { address: "0x8ba1f109551bD432803012645Ac136ddd64DBA72", name: "ricmoo.firefly.eth" }, ], blocks: [ { hash: "0x3d6122660cc824376f11ee842f83addc3525e2dd6756b9bcf0affa6aa88cf741", parentHash: "0xb495a1d7e6663152ae92708da4843337b958146015a2802f4193a410044698c9", number: 3, timestamp: 1438270048, nonce: "0x2e9344e0cbde83ce", difficulty: 17154715646, gasLimit: bnify("0x1388"), gasUsed: bnify("0"), miner: "0x5088D623ba0fcf0131E0897a91734A4D83596AA0", extraData: "0x476574682f76312e302e302d66633739643332642f6c696e75782f676f312e34", transactions: [] } ], transactions: [ { hash: "0xccc90ab97a74c952fb3376c4a3efb566a58a10df62eb4d44a61e106fcf10ec61", blockHash: "0x9653f180a5720f3634816eb945a6d722adee52cc47526f6357ac10adaf368135", blockNumber: 4097745, transactionIndex: 18, type: 0, from: "0x32DEF047DeFd076DB21A2D759aff2A591c972248", gasPrice: bnify("0x4a817c800"), gasLimit: bnify("0x3d090"), to: "0x6fC21092DA55B392b045eD78F4732bff3C580e2c", value: bnify("0x186cc6acd4b0000"), nonce: 0, data: "0xf2c298be000000000000000000000000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000000067269636d6f6f0000000000000000000000000000000000000000000000000000", r: "0x1e5605197a03e3f0a168f14749168dfeefc44c9228312dacbffdcbbb13263265", s: "0x269c3e5b3558267ad91b0a887d51f9f10098771c67b82ea6cb74f29638754f54", v: 38, creates: null, //raw: "0xf8d2808504a817c8008303d090946fc21092da55b392b045ed78f4732bff3c580e2c880186cc6acd4b0000b864f2c298be000000000000000000000000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000000067269636d6f6f000000000000000000000000000000000000000000000000000026a01e5605197a03e3f0a168f14749168dfeefc44c9228312dacbffdcbbb13263265a0269c3e5b3558267ad91b0a887d51f9f10098771c67b82ea6cb74f29638754f54", chainId: 1 } ], transactionReceipts: [ { blockHash: "0x36b4af7f0538559e581c8588f16477df0f676439ea67fe8d7a2ae4abb20e2566", blockNumber: 0x3c92b5, type: 0, contractAddress: null, cumulativeGasUsed: 0x1cca2e, from: "0x18C6045651826824FEBBD39d8560584078d1b247", gasUsed: 0x14bb7, logs: [ { address: "0x314159265dD8dbb310642f98f50C066173C1259b", blockHash: "0x36b4af7f0538559e581c8588f16477df0f676439ea67fe8d7a2ae4abb20e2566", blockNumber: 0x3c92b5, data: "0x00000000000000000000000018c6045651826824febbd39d8560584078d1b247", logIndex: 0x1a, topics: [ "0xce0457fe73731f824cc272376169235128c118b49d344817417c6d108d155e82", "0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae", "0xf0106919d12469348e14ad6a051d0656227e1aba2fefed41737fdf78421b20e1" ], transactionHash: "0xc6fcb7d00d536e659a4559d2de29afa9e364094438fef3e72ba80728ce1cb616", transactionIndex: 0x39, }, { address: "0x6090A6e47849629b7245Dfa1Ca21D94cd15878Ef", blockHash: "0x36b4af7f0538559e581c8588f16477df0f676439ea67fe8d7a2ae4abb20e2566", blockNumber: 0x3c92b5, data: "0x000000000000000000000000000000000000000000000000002386f26fc1000000000000000000000000000000000000000000000000000000000000595a32ce", logIndex: 0x1b, topics: [ "0x0f0c27adfd84b60b6f456b0e87cdccb1e5fb9603991588d87fa99f5b6b61e670", "0xf0106919d12469348e14ad6a051d0656227e1aba2fefed41737fdf78421b20e1", "0x00000000000000000000000018c6045651826824febbd39d8560584078d1b247" ], transactionHash: "0xc6fcb7d00d536e659a4559d2de29afa9e364094438fef3e72ba80728ce1cb616", transactionIndex: 0x39, } ], logsBloom: "0x00000000000000040000000000100000010000000000000040000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000200000010000000004000000000000000000000000000000000002000000000000000000000000400000000020000000000000000000000000000000000000004000000000000000000000000000000000000000000000000801000000000000000000000020000000000040000000040000000000000000002000000004000000000000000000000000000000000000000000000010000000000000000000000000000000000200000000000000000", root: "0x9b550a9a640ce50331b64504ef87aaa7e2aaf97344acb6ff111f879b319d2590", status: null, to: "0x6090A6e47849629b7245Dfa1Ca21D94cd15878Ef", transactionHash: "0xc6fcb7d00d536e659a4559d2de29afa9e364094438fef3e72ba80728ce1cb616", transactionIndex: 0x39 }, // Byzantium block { byzantium: true, blockHash: "0x34e5a6cfbdbb84f7625df1de69d218ade4da72f4a2558064a156674e72e976c9", blockNumber: 0x444f76, type: 0, contractAddress: null, cumulativeGasUsed: 0x15bfe7, from: "0x18C6045651826824FEBBD39d8560584078d1b247", gasUsed: 0x1b968, logs: [ { address: "0xb90E64082D00437e65A76d4c8187596BC213480a", blockHash: "0x34e5a6cfbdbb84f7625df1de69d218ade4da72f4a2558064a156674e72e976c9", blockNumber: 0x444f76, data: "0x", logIndex: 0x10, topics: [ "0x748d071d1992ee1bfe7a39058114d0a50d5798fe8eb3a9bfb4687f024629a2ce", "0x5574aa58f7191ccab6de6cf75fe2ea0484f010b852fdd8c6b7ae151d6c2f4b83" ], transactionHash: "0x7f1c6a58dc880438236d0b0a4ae166e9e9a038dbea8ec074149bd8b176332cac", transactionIndex: 0x1e, } ], logsBloom: "0x00000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001000000000000000200000000000000008000000000000000000000000000000000000000000000000000000000000000010000000000000000000800000000000000000000000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000800000000000000000800000000000000000000000000000000000000", status: 1, to: "0xb90E64082D00437e65A76d4c8187596BC213480a", transactionHash: "0x7f1c6a58dc880438236d0b0a4ae166e9e9a038dbea8ec074149bd8b176332cac", transactionIndex: 0x1e } ] }, kovan: { addresses: [ { address: "0x09c967A0385eE3B3717779738cA0B9D116e0EcE7", balance: bnify("997787946734641021"), code: "0x" }, ], blocks: [ { hash: "0xf0ec9bf41b99a6bd1f6cd29f91302f71a1a82d14634d2e207edea4b7962f3676", parentHash: "0xf110ecd84454f116e2222378e7bca81ac3e59be0dac96d7ec56d5ef1c3bc1d64", number: 3, timestamp: 1488459452, difficulty: 131072, gasLimit: bnify("0x5b48ec"), gasUsed: bnify("0"), miner: "0x00A0A24b9f0E5EC7Aa4c7389b8302fd0123194dE", extraData: "0xd5830105048650617269747986312e31352e31826c69", transactions: [] }, // Kovan Test Case with difficulty > 53-bits; See #711 { hash: "0xd92891a6eeaed4892289edf9bd5ebff261da5c6a51f7131cc1a481c6f4d1aa75", parentHash: "0xcc769a02513be1df80eee7d3a5cb87f14f37baee03c13f3e3ad1e7bdcaf7dac3", number: 16265864, timestamp: 1579621004, difficulty: null, gasLimit: bnify("0x989680"), gasUsed: bnify("0x0705bf"), miner: "0x596e8221A30bFe6e7eFF67Fee664A01C73BA3C56", extraData: "0xde830206088f5061726974792d457468657265756d86312e34302e30826c69", transactions: [ "0x20e6760fa1297fb06c8c20e6ed99581e0ba964d51167ea3c8ff580bfcb10bfc3", "0x0ce7eba48b1bbdee05823b79ae24e741f3f290d0abfef8ae9adf32db108b7dd6", "0x1fa2baafa844bf4853e4abbbf49532bf570210d589dc626dbf7ebc4832bdfa5d", "0xdb5d1fa54d30a4b6aee0b242a2c68ea52d3dd28703f69e6e30871827850aa2fa", "0xcc898db85d7d2493d4778faf640be32a4a3b7f5f987257bdc0009ce75a18eeaa" ] }, ], transactions: [], transactionReceipts: [] }, rinkeby: { addresses: [ { address: "0xd09a624630a656a7dbb122cb05e41c12c7cd8c0e", balance: bnify("3000000000000000000"), code: "0x" }, ], blocks: [ { hash: "0x9eb9db9c3ec72918c7db73ae44e520139e95319c421ed6f9fc11fa8dd0cddc56", parentHash: "0x9b095b36c15eaf13044373aef8ee0bd3a382a5abb92e402afa44b8249c3a90e9", number: 3, timestamp: 1492010489, nonce: "0x0000000000000000", difficulty: 2, gasLimit: bnify("0x47e7c4"), gasUsed: bnify(0), // miner: "0x42EB768f2244C8811C63729A21A3569731535f06", extraData: "0xd783010600846765746887676f312e372e33856c696e757800000000000000004e10f96536e45ceca7e34cc1bdda71db3f3bb029eb69afd28b57eb0202c0ec0859d383a99f63503c4df9ab6c1dc63bf6b9db77be952f47d86d2d7b208e77397301", transactions: [] }, ], transactions: [], transactionReceipts: [] }, ropsten: { addresses: [ { address: "0x03a6F7a5ce5866d9A0CCC1D4C980b8d523f80480", balance: bnify("15861113897828552666"), code: "0x" }, ], blocks: [ { hash: "0xaf2f2d55e6514389bcc388ccaf40c6ebf7b3814a199a214f1203fb674076e6df", parentHash: "0x88e8bc1dd383672e96d77ee247e7524622ff3b15c337bd33ef602f15ba82d920", number: 3, timestamp: 1479642588, nonce: "0x04668f72247a130c", difficulty: 996427, gasLimit: bnify("0xff4033"), gasUsed: bnify("0"), miner: "0xD1aEb42885A43b72B518182Ef893125814811048", extraData: "0xd883010503846765746887676f312e372e318664617277696e", transactions: [] }, ], transactions: [ // Berlin tests (EIP-2930) { hash: "0x48bff7b0e603200118a672f7c622ab7d555a28f98938edb8318803eed7ea7395", type: 1, accessList: [ { address: "0x0000000000000000000000000000000000000000", storageKeys: [] } ], blockHash: "0x378e24bcd568bd24cf1f54d38f13f038ee28d89e82af4f2a0d79c1f88dcd8aac", blockNumber: 9812343, from: "0x32162F3581E88a5f62e8A61892B42C46E2c18f7b", gasPrice: bnify("0x65cf89a0"), gasLimit: bnify("0x5b68"), to: "0x32162F3581E88a5f62e8A61892B42C46E2c18f7b", value: bnify("0"), nonce: 13, data: "0x", r: "0x9659cba42376dbea1433cd6afc9c8ffa38dbeff5408ffdca0ebde6207281a3ec", s: "0x27efbab3e6ed30b088ce0a50533364778e101c9e52acf318daec131da64e7758", v: 0, creates: null, chainId: 3 }, { hash: "0x1675a417e728fd3562d628d06955ef35b913573d9e417eb4e6a209998499c9d3", type: 1, accessList: [ { address: "0x0000000000000000000000000000000000000000", storageKeys: [ "0xdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef", "0x0000000000111111111122222222223333333333444444444455555555556666", "0xdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef" ] } ], blockHash: "0x7565688256f5801768237993b47ca0608796b3ace0c4b8b6e623c6092bef14b8", blockNumber: 9812365, from: "0x32162F3581E88a5f62e8A61892B42C46E2c18f7b", gasPrice: bnify("0x65cf89a0"), gasLimit: bnify("0x71ac"), to: "0x32162F3581E88a5f62e8A61892B42C46E2c18f7b", value: bnify("0"), nonce: 14, data: "0x", r: "0xb0646756f89817d70cdb40aa2ae8b5f43ef65d0926dcf71a7dca5280c93763df", s: "0x4d32dbd9a44a2c5639b8434b823938202f75b0a8459f3fcd9f37b2495b7a66a6", v: 0, creates: null, chainId: 3 }, // London Tests (EIP-1559) { hash: '0xb8c7871d9d8597ee8a50395d8b39dafa280c90337dc501d0db1321806c6ea98c', blockHash: '0xfd824501af65b1d0f21ea9eb7ec83f45108fcd6fd1bca5d6414ba5923ad87b49', blockNumber: 10512507, transactionIndex: 5, type: 2, creates: null, from: '0xad252DD6C011E613610A36368f04aC84D5185b7c', //gasPrice: bnify("0x0268ab0ed6"), maxPriorityFeePerGas: bnify("0x0268ab0ed6"), maxFeePerGas: bnify("0x0268ab0ed6"), gasLimit: bnify("0x5208"), to: '0x8210357f377E901f18E45294e86a2A32215Cc3C9', value: bnify("0x7b"), nonce: 0, data: '0x', r: '0x7426c348119eed4e9e0525b52aa77edbbf1107610702b4642fa9d2688dce6fa7', s: '0x03f606ad1f12af5876280a34601a4eb3919b797cf3878161e2d24b61d2609846', v: 1, accessList: [], chainId: 3, }, ], transactionReceipts: [ { blockHash: "0xc9235b8253fce455942147aa8b450d23081b867ffbb2a1e4dec934827cd80f8f", blockNumber: 0x1564d8, type: 0, contractAddress: null, cumulativeGasUsed: bnify("0x80b9"), from: "0xb346D5019EeafC028CfC01A5f789399C2314ae8D", gasUsed: bnify("0x80b9"), logs: [ { address: "0x6fC21092DA55B392b045eD78F4732bff3C580e2c", blockHash: "0xc9235b8253fce455942147aa8b450d23081b867ffbb2a1e4dec934827cd80f8f", blockNumber: 0x1564d8, data: "0x00000000000000000000000006b5955a67d827cdf91823e3bb8f069e6c89c1d6000000000000000000000000000000000000000000000000016345785d8a0000", logIndex: 0x0, topics: [ "0xac375770417e1cb46c89436efcf586a74d0298fee9838f66a38d40c65959ffda" ], transactionHash: "0x55c477790b105e69e98afadf0505cbda606414b0187356137132bf24945016ce", transactionIndex: 0x0, } ], logsBloom: "0x00000000000000800000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000400000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000000000000000000000001000000000000000000000010000000000000100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", root: "0xf1c3506ab619ac1b5e8f1ca355b16d6b9a1b7436b2960b0e9ec9a91f4238b5cc", to: "0x6fC21092DA55B392b045eD78F4732bff3C580e2c", transactionHash: "0x55c477790b105e69e98afadf0505cbda606414b0187356137132bf24945016ce", transactionIndex: 0x0 }, // Byzantium Receipt { byzantium: true, blockHash: "0x61d343e0e081b60ac53bab381e07bdd5d0815b204091a576fd05106b814e7e1e", blockNumber: 0x1e1e3b, contractAddress: null, cumulativeGasUsed: bnify("0x4142f"), from: "0xdc8F20170C0946ACCF9627b3EB1513CFD1c0499f", gasUsed: bnify("0x1eb6d"), logs: [ { address: "0xCBf1735Aad8C4B337903cD44b419eFE6538aaB40", blockHash: "0x61d343e0e081b60ac53bab381e07bdd5d0815b204091a576fd05106b814e7e1e", blockNumber: 0x1e1e3b, data: "0x000000000000000000000000b70560a43a9abf6ea2016f40a3e84b8821e134c5f6c95607c490f4f379c0160ef5c8898770f8a52959abf0e9de914647b377fa290000000000000000000000000000000000000000000000000000000000001c20000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000001400000000000000000000000000000000000000000000000000000000000030d4000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000355524c0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000004c6a736f6e2868747470733a2f2f6170692e6b72616b656e2e636f6d2f302f7075626c69632f5469636b65723f706169723d455448555344292e726573756c742e584554485a5553442e632e300000000000000000000000000000000000000000", logIndex: 0x1, topics: ["0xb76d0edd90c6a07aa3ff7a222d7f5933e29c6acc660c059c97837f05c4ca1a84"], transactionHash: "0xf724f1d6813f13fb523c5f6af6261d06d41138dd094fff723e09fb0f893f03e6", transactionIndex: 0x2, } ], logsBloom: "0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000800000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000000008000000000000000000000080000000202000000", status: 1, to: "0xB70560a43A9aBf6ea2016F40a3e84B8821E134c5", transactionHash: "0xf724f1d6813f13fb523c5f6af6261d06d41138dd094fff723e09fb0f893f03e6", transactionIndex: 0x2 }, // London Tests (EIP-1559) { blockNumber: 10512507, blockHash: '0xfd824501af65b1d0f21ea9eb7ec83f45108fcd6fd1bca5d6414ba5923ad87b49', transactionHash: '0xb8c7871d9d8597ee8a50395d8b39dafa280c90337dc501d0db1321806c6ea98c', transactionIndex: 5, byzantium: true, type: 2, to: '0x8210357f377E901f18E45294e86a2A32215Cc3C9', from: '0xad252DD6C011E613610A36368f04aC84D5185b7c', contractAddress: null, gasUsed: bnify("0x5208"), logsBloom: '0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000', logs: [], cumulativeGasUsed: bnify("0x038f3e"), effectiveGasPrice: bnify("0x268ab0ed6"), status: 1, } ], }, goerli: { addresses: [ { address: "0x06B5955A67D827CDF91823E3bB8F069e6c89c1D6", balance: bnify("314159000000000000"), code: "0x" }, ], blocks: [ { hash: "0xd5daa825732729bb0d2fd187a1b888e6bfc890f1fc5333984740d9052afb2920", parentHash: "0xe675f1362d82cdd1ec260b16fb046c17f61d8a84808150f5d715ccce775f575e", number: 3, timestamp: 1548947483, difficulty: 2, gasLimit: bnify("10455073"), gasUsed: bnify("0"), // miner: "0xe0a2Bd4258D2768837BAa26A28fE71Dc079f84c7", extraData: "0x506172697479205465636820417574686f7269747900000000000000000000002822e1b202411c38084d96c84302b8361ec4840a51cd2fad9cb4bd9921cad7e64bc2e5dc7b41f3f75b33358be3aec718cf4d4317ace940e01b3581a95c9259ac01", transactions: [] }, // Blockhash with leading zero; see #629 { hash: "0x0f305466552efa183a0de26b6fda26d55a872dbc02aca8b5852cc2a361ce9ee4", parentHash: "0x6723e880e01c15c5ac894abcae0f5b55ea809a31eaf5618998928f7d9cbc5118", number: 1479831, timestamp: 1571216171, difficulty: 2, gasLimit: bnify(0x7a1200), gasUsed: bnify("0x0d0ef5"), // miner: "0x22eA9f6b28DB76A7162054c05ed812dEb2f519Cd", extraData: "0x0000000000000000000000000000000000000000000000000000000000000000f4e6fc1fbd88adf57a272d98f725487f872ef0495a54c2b873a58d14e010bf517cc5650417f18cfd4ad2396272c564a7da1265ae27c397609293f488ec57d68e01", transactions: [ "0xea29f0764f03c5c67ac53a866a28ce23a4a032c2de4327e452b39f482920761a", "0x0eef23ffb59ac41762fdfa55d9e47e82fa7f0b70b1e8ec486d72fe1fee15f6de", "0xba1eeb67ac6e8d1aa900ff6fbd84ac46869c9e100b33f787acfb234cd9c93f9f", "0x4f412ab735b29ddc8b1ff7abe4bfece7ad4684aa20e260fbc42aed75a0d387ea", "0x2f1fddcc7a2c4b2b7d83c5cadec4e7b71c34cec65da99b1114bd2b044ae0636c" ] } ], transactions: [], transactionReceipts: [ { blockHash: "0x2384e8e8bdcf6eb87ec7c138fa503ac34adb32cac817e4b35f14d4339eaa1993", blockNumber: 47464, byzantium: true, type: 0, contractAddress: null, cumulativeGasUsed: bnify(21000), from: "0x8c1e1e5b47980D214965f3bd8ea34C413E120ae4", gasUsed: bnify(21000), logsBloom: "0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", to: "0x58Bb4221245461E1d4cf886f18a01E3Df40Bd359", transactionHash: "0xec8b1ac5d787f36c738cc7793fec606283b41f1efa69df4ae6b2a014dcd12797", transactionIndex: 0, logs: [], status: 1 } ], } }; blockchainData["default"] = blockchainData.homestead; function equals(name, actual, expected) { if (expected && expected.eq) { if (actual == null) { assert.ok(false, name + " - actual big number null"); } expected = ethers.BigNumber.from(expected); actual = ethers.BigNumber.from(actual); assert.ok(expected.eq(actual), name + " matches"); } else if (Array.isArray(expected)) { if (actual == null) { assert.ok(false, name + " - actual array null"); } assert.equal(actual.length, expected.length, name + " array lengths match"); for (let i = 0; i < expected.length; i++) { equals("(" + name + " - item " + i + ")", actual[i], expected[i]); } } else if (typeof (expected) === "object") { if (actual == null) { if (expected === actual) { return; } assert.ok(false, name + " - actual object null"); } let keys = {}; Object.keys(expected).forEach((key) => { keys[key] = true; }); Object.keys(actual).forEach((key) => { keys[key] = true; }); Object.keys(keys).forEach((key) => { equals("(" + name + " - key + " + key + ")", actual[key], expected[key]); }); } else { if (actual == null) { assert.ok(false, name + " - actual null"); } assert.equal(actual, expected, name + " matches"); } } function waiter(duration) { return new Promise((resolve) => { const timer = setTimeout(resolve, duration); if (timer.unref) { timer.unref(); } }); } const allNetworks = ["default", "homestead", "ropsten", "rinkeby", "kovan", "goerli"]; // We use separate API keys because otherwise the testcases sometimes // fail during CI because our default keys are pretty heavily used const _ApiKeys = { alchemy: "YrPw6SWb20vJDRFkhWq8aKnTQ8JRNRHM", etherscan: "FPFGK6JSW2UHJJ2666FG93KP7WC999MNW7", infura: "49a0efa3aaee4fd99797bfa94d8ce2f1", }; const _ApiKeysPocket = { homestead: "6004bcd10040261633ade990", ropsten: "6004bd4d0040261633ade991", rinkeby: "6004bda20040261633ade994", goerli: "6004bd860040261633ade992", }; function getApiKeys(network) { if (network === "default" || network == null) { network = "homestead"; } const apiKeys = ethers.utils.shallowCopy(_ApiKeys); apiKeys.pocket = _ApiKeysPocket[network]; return apiKeys; } const providerFunctions = [ { name: "getDefaultProvider", networks: allNetworks, create: (network) => { if (network == "default") { return ethers.getDefaultProvider(null, getApiKeys(network)); } return ethers.getDefaultProvider(network, getApiKeys(network)); } }, { name: "AlchemyProvider", networks: allNetworks, create: (network) => { if (network == "default") { return new ethers.providers.AlchemyProvider(null, getApiKeys(network).alchemy); } return new ethers.providers.AlchemyProvider(network, getApiKeys(network).alchemy); } }, /* { name: "CloudflareProvider", networks: [ "default", "homestead" ], create: (network: string) => { return new ethers.providers.CloudflareProvider(network); } }, */ { name: "InfuraProvider", networks: allNetworks, create: (network) => { if (network == "default") { return new ethers.providers.InfuraProvider(null, getApiKeys(network).infura); } return new ethers.providers.InfuraProvider(network, getApiKeys(network).infura); } }, { name: "EtherscanProvider", networks: allNetworks, create: (network) => { if (network == "default") { return new ethers.providers.EtherscanProvider(null, getApiKeys(network).etherscan); } return new ethers.providers.EtherscanProvider(network, getApiKeys(network).etherscan); } }, { name: "NodesmithProvider", networks: [], create: (network) => { throw new Error("not tested"); } }, { name: "PocketProvider", // note: sans-kovan // @TODO: Pocket is being incredibly unreliable right now; removing it so // we can pass the CI //networks: [ "default", "homestead", "ropsten", "rinkeby", "goerli" ], networks: ["default", "homestead"], create: (network) => { if (network == "default") { return new ethers.providers.PocketProvider(null, { applicationId: getApiKeys(network).pocket, loadBalancer: true }); } return new ethers.providers.PocketProvider(network, { applicationId: getApiKeys(network).pocket, loadBalancer: true }); } }, { name: "Web3Provider", networks: [], create: (network) => { throw new Error("not tested"); } } ]; // This wallet can be funded and used for various test cases const fundWallet = ethers.Wallet.createRandom(); const testFunctions = []; Object.keys(blockchainData).forEach((network) => { function addSimpleTest(name, func, expected) { testFunctions.push({ name: name, networks: [network], execute: (provider) => __awaiter(this, void 0, void 0, function* () { const value = yield func(provider); equals(name, expected, value); }) }); } function addObjectTest(name, func, expected, checkSkip) { testFunctions.push({ name, networks: [network], checkSkip, execute: (provider) => __awaiter(this, void 0, void 0, function* () { const value = yield func(provider); Object.keys(expected).forEach((key) => { equals(`${name}.${key}`, value[key], expected[key]); }); }) }); } const tests = blockchainData[network]; // And address test case can have any of the following: // - balance // - code // - storage // - ENS name tests.addresses.forEach((test) => { if (test.balance) { addSimpleTest(`fetches account balance: ${test.address}`, (provider) => { return provider.getBalance(test.address); }, test.balance); } if (test.code) { addSimpleTest(`fetches account code: ${test.address}`, (provider) => { return provider.getCode(test.address); }, test.code); } if (test.storage) { Object.keys(test.storage).forEach((position) => { addSimpleTest(`fetches storage: ${test.address}:${position}`, (provider) => { return provider.getStorageAt(test.address, bnify(position)); }, test.storage[position]); }); } if (test.name) { addSimpleTest(`fetches ENS name: ${test.address}`, (provider) => { return provider.resolveName(test.name); }, test.address); } }); tests.blocks.forEach((test) => { addObjectTest(`fetches block (by number) #${test.number}`, (provider) => { return provider.getBlock(test.number); }, test); }); tests.blocks.forEach((test) => { addObjectTest(`fetches block (by hash) ${test.hash}`, (provider) => { return provider.getBlock(test.hash); }, test, (provider, network, test) => { return (provider === "EtherscanProvider"); }); }); tests.transactions.forEach((test) => { const hash = test.hash; addObjectTest(`fetches transaction ${hash}`, (provider) => __awaiter(void 0, void 0, void 0, function* () { const tx = yield provider.getTransaction(hash); // This changes with every block assert.equal(typeof (tx.confirmations), "number", "confirmations is a number"); delete tx.confirmations; assert.equal(typeof (tx.wait), "function", "wait is a function"); delete tx.wait; return tx; }), test, (provider, network, test) => { // Temporary; pocket is being broken again for old transactions return provider === "PocketProvider"; //return false; }); }); tests.transactionReceipts.forEach((test) => { const hash = test.transactionHash; addObjectTest(`fetches transaction receipt ${hash}`, (provider) => __awaiter(void 0, void 0, void 0, function* () { const receipt = yield provider.getTransactionReceipt(hash); if (test.status === null) { assert.ok(receipt.status === undefined, "no status"); receipt.status = null; } // This changes with every block; so just make sure it is a number assert.equal(typeof (receipt.confirmations), "number", "confirmations is a number"); delete receipt.confirmations; return receipt; }), test, (provider, network, test) => { // Temporary; pocket is being broken again for old transactions return provider === "PocketProvider"; //return false; }); }); }); (function () { function addErrorTest(code, func) { testFunctions.push({ name: `throws correct ${code} error`, networks: ["ropsten"], checkSkip: (provider, network, test) => { return false; }, execute: (provider) => __awaiter(this, void 0, void 0, function* () { try { const value = yield func(provider); console.log(value); assert.ok(false, "did not throw"); } catch (error) { assert.equal(error.code, code, "incorrect error thrown"); } }) }); } /* @TODO: Use this for testing pre-EIP-155 transactions on specific networks addErrorTest(ethers.utils.Logger.errors.NONCE_EXPIRED, async (provider: ethers.providers.Provider) => { return provider.sendTransaction("0xf86480850218711a0082520894000000000000000000000000000000000000000002801ba038aaddcaaae7d3fa066dfd6f196c8348e1bb210f2c121d36cb2c24ef20cea1fba008ae378075d3cd75aae99ab75a70da82161dffb2c8263dabc5d8adecfa9447fa"); }); */ // Wallet(id("foobar1234")) addErrorTest(ethers.utils.Logger.errors.NONCE_EXPIRED, (provider) => __awaiter(this, void 0, void 0, function* () { return provider.sendTransaction("0xf86480850218711a00825208940000000000000000000000000000000000000000038029a04320fd28c8e6c95da9229d960d14ffa3de81f83abe3ad9c189642c83d7d951f3a009aac89e04a8bafdcf618e21fed5e7b1144ca1083a301fd5fde28b0419eb63ce"); })); addErrorTest(ethers.utils.Logger.errors.INSUFFICIENT_FUNDS, (provider) => __awaiter(this, void 0, void 0, function* () { const txProps = { to: "0x8ba1f109551bD432803012645Ac136ddd64DBA72", gasPrice: 9000000000, gasLimit: 21000, chainId: 3, value: 1, }; const wallet = ethers.Wallet.createRandom(); const tx = yield wallet.signTransaction(txProps); return provider.sendTransaction(tx); })); addErrorTest(ethers.utils.Logger.errors.INSUFFICIENT_FUNDS, (provider) => __awaiter(this, void 0, void 0, function* () { const txProps = { to: "0x8ba1f109551bD432803012645Ac136ddd64DBA72", gasPrice: 9000000000, gasLimit: 21000, value: 1, // @TODO: Remove this once all providers are eip-1559 savvy type: 0, }; const wallet = ethers.Wallet.createRandom().connect(provider); return wallet.sendTransaction(txProps); })); addErrorTest(ethers.utils.Logger.errors.UNPREDICTABLE_GAS_LIMIT, (provider) => __awaiter(this, void 0, void 0, function* () { return provider.estimateGas({ to: "0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e" // ENS contract }); })); })(); testFunctions.push({ name: "sends a legacy transaction", extras: ["funding"], timeout: 900, networks: ["ropsten"], checkSkip: (provider, network, test) => { return false; }, execute: (provider) => __awaiter(void 0, void 0, void 0, function* () { const gasPrice = (yield provider.getGasPrice()).mul(10); const wallet = fundWallet.connect(provider); const addr = "0x8210357f377E901f18E45294e86a2A32215Cc3C9"; yield waiter(3000); const b0 = yield provider.getBalance(wallet.address); assert.ok(b0.gt(ethers.constants.Zero), "balance is non-zero"); const tx = yield wallet.sendTransaction({ type: 0, to: addr, value: 123, gasPrice: gasPrice }); yield tx.wait(); yield waiter(3000); const b1 = yield provider.getBalance(wallet.address); assert.ok(b0.gt(b1), "balance is decreased"); }) }); testFunctions.push({ name: "sends an EIP-2930 transaction", extras: ["funding"], timeout: 900, networks: ["ropsten"], checkSkip: (provider, network, test) => { return false; }, execute: (provider) => __awaiter(void 0, void 0, void 0, function* () { const gasPrice = (yield provider.getGasPrice()).mul(10); const wallet = fundWallet.connect(provider); const addr = "0x8210357f377E901f18E45294e86a2A32215Cc3C9"; yield waiter(3000); const b0 = yield provider.getBalance(wallet.address); assert.ok(b0.gt(ethers.constants.Zero), "balance is non-zero"); const tx = yield wallet.sendTransaction({ type: 1, accessList: { "0x8ba1f109551bD432803012645Ac136ddd64DBA72": [ "0x0000000000000000000000000000000000000000000000000000000000000000", "0x0000000000000000000000000000000000000000000000000000000000000042", ] }, to: addr, value: 123, gasPrice: gasPrice }); yield tx.wait(); yield waiter(3000); const b1 = yield provider.getBalance(wallet.address); assert.ok(b0.gt(b1), "balance is decreased"); }) }); testFunctions.push({ name: "sends an EIP-1559 transaction", extras: ["funding"], timeout: 900, networks: ["ropsten"], checkSkip: (provider, network, test) => { // These don't support EIP-1559 yet for sending return (provider === "AlchemyProvider"); }, execute: (provider) => __awaiter(void 0, void 0, void 0, function* () { const wallet = fundWallet.connect(provider); const addr = "0x8210357f377E901f18E45294e86a2A32215Cc3C9"; yield waiter(3000); const b0 = yield provider.getBalance(wallet.address); assert.ok(b0.gt(ethers.constants.Zero), "balance is non-zero"); const tx = yield wallet.sendTransaction({ type: 2, accessList: { "0x8ba1f109551bD432803012645Ac136ddd64DBA72": [ "0x0000000000000000000000000000000000000000000000000000000000000000", "0x0000000000000000000000000000000000000000000000000000000000000042", ] }, to: addr, value: 123, }); yield tx.wait(); yield waiter(3000); const b1 = yield provider.getBalance(wallet.address); assert.ok(b0.gt(b1), "balance is decreased"); }) }); describe("Test Provider Methods", function () { let fundReceipt = null; const faucet = "0x8210357f377E901f18E45294e86a2A32215Cc3C9"; before(function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(300000); // Get some ether from the faucet const provider = new ethers.providers.InfuraProvider("ropsten", getApiKeys("ropsten").infura); const funder = yield ethers.utils.fetchJson(`https:/\/api.ethers.io/api/v1/?action=fundAccount&address=${fundWallet.address.toLowerCase()}`); fundReceipt = provider.waitForTransaction(funder.hash); fundReceipt.then((receipt) => { console.log(`*** Funded: ${fundWallet.address}`); }); }); }); after(function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(300000); // Wait until the funding is complete yield fundReceipt; // Refund all unused ether to the faucet const provider = new ethers.providers.InfuraProvider("ropsten", getApiKeys("ropsten").infura); const gasPrice = yield provider.getGasPrice(); const balance = yield provider.getBalance(fundWallet.address); const tx = yield fundWallet.connect(provider).sendTransaction({ to: faucet, gasLimit: 21000, gasPrice: gasPrice, value: balance.sub(gasPrice.mul(21000)) }); console.log(`*** Sweep Transaction:`, tx.hash); }); }); providerFunctions.forEach(({ name, networks, create }) => { networks.forEach((network) => { const provider = create(network); testFunctions.forEach((test) => { // Skip tests not supported on this network if (test.networks.indexOf(network) === -1) { return; } if (test.checkSkip && test.checkSkip(name, network, test)) { return; } // How many attempts to try? const attempts = (test.attempts != null) ? test.attempts : 3; const timeout = (test.timeout != null) ? test.timeout : 60; const extras = (test.extras || []).reduce((accum, key) => { accum[key] = true; return accum; }, {}); it(`${name}.${network ? network : "default"} ${test.name}`, function () { return __awaiter(this, void 0, void 0, function* () { // Multiply by 2 to make sure this never happens; we want our // timeout logic to success, not allow a done() called multiple // times because our logic returns after the timeout has occurred. this.timeout(2 * (1000 + timeout * 1000 * attempts)); // Wait for the funding transaction to be mined if (extras.funding) { yield fundReceipt; } // We wait at least 1 seconds between tests if (!extras.nowait) { yield waiter(1000); } let error = null; for (let attempt = 0; attempt < attempts; attempt++) { try { const result = yield Promise.race([ test.execute(provider), waiter(timeout * 1000).then((result) => { throw new Error("timeout"); }) ]); return result; } catch (attemptError) { console.log(`*** Failed attempt ${attempt + 1}: ${attemptError.message}`); error = attemptError; // On failure, wait 5s yield waiter(5000); } } throw error; }); }); }); }); }); }); describe("Extra tests", function () { it("etherscan long-request #1093", function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(60000); yield waiter(2000); const provider = new ethers.providers.EtherscanProvider(null, getApiKeys(null).etherscan); const value = yield provider.call({ to: "0xbf320b8336b131e0270295c15478d91741f9fc11", data: "0x3ad206cc000000000000000000000000f6e914d07d12636759868a61e52973d17ed7111b0000000000000000000000000000000000000000000000000000000000000040000000000000000000000000000000000000000000000000000000000000006400000000000000000000000022b3faaa8df978f6bafe18aade18dc2e3dfa0e0c000000000000000000000000998b3b82bc9dba173990be7afb772788b5acb8bd000000000000000000000000ba11d00c5f74255f56a5e366f4f77f5a186d7f55000000000000000000000000c7579bb99af590ec71c316e1ac4436c5350395940000000000000000000000002a05d22db079bc40c2f77a1d1ff703a56e631cc10000000000000000000000000d8775f648430679a709e98d2b0cb6250d2887ef0000000000000000000000009a0242b7a33dacbe40edb927834f96eb39f8fbcb000000000000000000000000c78593c17482ea5de44fdd84896ffd903972878e000000000000000000000000e7d3e4413e29ae35b0893140f4500965c74365e500000000000000000000000037d40510a2f5bc98aa7a0f7bf4b3453bcfb90ac10000000000000000000000004a6058666cf1057eac3cd3a5a614620547559fc900000000000000000000000035a69642857083ba2f30bfab735dacc7f0bac96900000000000000000000000084f7c44b6fed1080f647e354d552595be2cc602f0000000000000000000000001500205f50bf3fd976466d0662905c9ff254fc9c000000000000000000000000660b612ec57754d949ac1a09d0c2937a010dee05000000000000000000000000acfa209fb73bf3dd5bbfb1101b9bc999c49062a5000000000000000000000000865d176351f287fe1b0010805b110d08699c200a000000000000000000000000633a8f8e557702039463f9f2eb20b7936fff8c050000000000000000000000001961b3331969ed52770751fc718ef530838b6dee0000000000000000000000002fb12bccf6f5dd338b76be784a93ade0724256900000000000000000000000004d8fc1453a0f359e99c9675954e656d80d996fbf0000000000000000000000006aeb95f06cda84ca345c2de0f3b7f96923a44f4c0000000000000000000000008aa33a7899fcc8ea5fbe6a608a109c3893a1b8b200000000000000000000000014c926f2290044b647e1bf2072e67b495eff1905000000000000000000000000763186eb8d4856d536ed4478302971214febc6a90000000000000000000000008a1e3930fde1f151471c368fdbb39f3f63a65b55000000000000000000000000a8daa52ded91f7c82b4bb02b4b87c6a841db1fd500000000000000000000000033803edf44a71b9579f54cd429b53b06c0eeab83000000000000000000000000026e62dded1a6ad07d93d39f96b9eabd59665e0d00000000000000000000000047da42696a866cdc61a4c809a515500a242909c100000000000000000000000008b4c866ae9d1be56a06e0c302054b4ffe067b43000000000000000000000000420335d3deef2d5b87524ff9d0fb441f71ea621f000000000000000000000000983f7cc12d0b5d512b0f91f51a4aa478ac4def46000000000000000000000000b2bfeb70b903f1baac7f2ba2c62934c7e5b974c40000000000000000000000009b11b1b271a224a271619f3419b1b080fdec5b4a0000000000000000000000007b1309c1522afd4e66c31e1e6d0ec1319e1eba5e000000000000000000000000959529102cfde07b1196bd27adedc196d75f84f6000000000000000000000000107c4504cd79c5d2696ea0030a8dd4e92601b82e000000000000000000000000539efe69bcdd21a83efd9122571a64cc25e0282b000000000000000000000000e5a7c12972f3bbfe70ed29521c8949b8af6a0970000000000000000000000000f8ad7dfe656188a23e89da09506adf7ad9290d5d0000000000000000000000005732046a883704404f284ce41ffadd5b007fd668000000000000000000000000df6ef343350780bf8c3410bf062e0c015b1dd671000000000000000000000000f028adee51533b1b47beaa890feb54a457f51e89000000000000000000000000dd6bf56ca2ada24c683fac50e37783e55b57af9f000000000000000000000000ef51c9377feb29856e61625caf9390bd0b67ea18000000000000000000000000c80c5e40220172b36adee2c951f26f2a577810c50000000000000000000000001f573d6fb3f13d689ff844b4ce37794d79a7ff1c000000000000000000000000d2d6158683aee4cc838067727209a0aaf4359de30000000000000000000000007cdec53fe4770729dac314756c10e2f37b8d2b2f000000000000000000000000cc34366e3842ca1bd36c1f324d15257960fcc8010000000000000000000000006b01c3170ae1efebee1a3159172cb3f7a5ecf9e5000000000000000000000000139d9397274bb9e2c29a9aa8aa0b5874d30d62e300000000000000000000000063f584fa56e60e4d0fe8802b27c7e6e3b33e007f000000000000000000000000780116d91e5592e58a3b3c76a351571b39abcec60000000000000000000000000e511aa1a137aad267dfe3a6bfca0b856c1a3682000000000000000000000000327682779bab2bf4d1337e8974ab9de8275a7ca80000000000000000000000001b80eeeadcc590f305945bcc258cfa770bbe18900000000000000000000000005af2be193a6abca9c8817001f45744777db307560000000000000000000000009e77d5a1251b6f7d456722a6eac6d2d5980bd891000000000000000000000000e25f0974fea47682f6a7386e4217da70512ec997000000000000000000000000558ec3152e2eb2174905cd19aea4e34a23de9ad6000000000000000000000000b736ba66aad83adb2322d1f199bfa32b3962f13c000000000000000000000000509a38b7a1cc0dcd83aa9d06214663d9ec7c7f4a0000000000000000000000000327112423f3a68efdf1fcf402f6c5cb9f7c33fd0000000000000000000000005acd19b9c91e596b1f062f18e3d02da7ed8d1e5000000000000000000000000003df4c372a29376d2c8df33a1b5f001cd8d68b0e0000000000000000000000006aac8cb9861e42bf8259f5abdc6ae3ae89909e11000000000000000000000000d96b9fd7586d9ea24c950d24399be4fb65372fdd00000000000000000000000073dd069c299a5d691e9836243bcaec9c8c1d87340000000000000000000000005ecd84482176db90bb741ddc8c2f9ccc290e29ce000000000000000000000000fa456cf55250a839088b27ee32a424d7dacb54ff000000000000000000000000b683d83a532e2cb7dfa5275eed3698436371cc9f000000000000000000000000ccbf21ba6ef00802ab06637896b799f7101f54a20000000000000000000000007b123f53421b1bf8533339bfbdc7c98aa94163db0000000000000000000000006ecccf7ebc3497a9334f4fe957a7d5fa933c5bcc0000000000000000000000004fabb145d64652a948d72533023f6e7a623c7c53000000000000000000000000e1aee98495365fc179699c1bb3e761fa716bee6200000000000000000000000056d811088235f11c8920698a204a5010a788f4b300000000000000000000000026e75307fc0c021472feb8f727839531f112f3170000000000000000000000007d4b8cce0591c9044a22ee543533b72e976e36c30000000000000000000000003c6a7ab47b5f058be0e7c7fe1a4b7925b8aca40e0000000000000000000000001d462414fe14cf489c7a21cac78509f4bf8cd7c000000000000000000000000043044f861ec040db59a7e324c40507addb67314200000000000000000000000004f2e7221fdb1b52a68169b25793e51478ff0329000000000000000000000000954b890704693af242613edef1b603825afcd708000000000000000000000000a8f93faee440644f89059a2c88bdc9bf3be5e2ea0000000000000000000000001234567461d3f8db7496581774bd869c83d51c9300000000000000000000000056ba2ee7890461f463f7be02aac3099f6d5811a80000000000000000000000006c8c6b02e7b2be14d4fa6022dfd6d75921d90e4e000000000000000000000000f444cd92e09cc8b2a23cd2eecb3c1e4cc8da6958000000000000000000000000cf8f9555d55ce45a3a33a81d6ef99a2a2e71dee2000000000000000000000000076c97e1c869072ee22f8c91978c99b4bcb0259100000000000000000000000017b26400621695c2d8c2d8869f6259e82d7544c4000000000000000000000000679badc551626e01b23ceecefbc9b877ea18fc46000000000000000000000000336f646f87d9f6bc6ed42dd46e8b3fd9dbd15c220000000000000000000000005d3a536e4d6dbd6114cc1ead35777bab948e3643000000000000000000000000f5dce57282a584d2746faf1593d3121fcac444dc0000000000000000000000001d9e20e581a5468644fe74ccb6a46278ef377f9e000000000000000000000000177d39ac676ed1c67a2b268ad7f1e58826e5b0af" }); assert.ok(!!value); }); }); }); /* describe("Test extra Etherscan operations", function() { let provider = new providers.EtherscanProvider(); it("fethces the current price of ether", function() { this.timeout(20000); return provider.getEtherPrice().then(function(price) { assert.ok(typeof(price) === "number", "Etherscan price returns a number"); assert.ok(price > 0.0, "Etherscan price returns non-zero"); }); }); it("fetches the history", function() { this.timeout(100000); return provider.getHistory("ricmoo.firefly.eth").then(function(history) { assert.ok(history.length > 40, "Etherscan history returns results"); assert.equal(history[0].hash, "0xd25f550cfdff90c086a6496a84dbb2c4577df15b1416e5b3319a3e4ebb5b25d8", "Etherscan history returns correct transaction"); }); }); }); */ describe("Test Basic Authentication", function () { //this.retries(3); function test(name, url) { it("tests " + name, function () { this.timeout(60000); return ethers.utils.fetchJson(url).then((data) => { assert.equal(data.authenticated, true, "authenticates user"); }); }); } let secure = { url: "https://httpbin.org/basic-auth/user/passwd", user: "user", password: "passwd" }; let insecure = { url: "http://httpbin.org/basic-auth/user/passwd", user: "user", password: "passwd" }; let insecureForced = { url: "http://httpbin.org/basic-auth/user/passwd", user: "user", password: "passwd", allowInsecureAuthentication: true }; test("secure url", secure); test("insecure url", insecureForced); it("tests insecure connections fail", function () { this.timeout(60000); assert.throws(() => { return ethers.utils.fetchJson(insecure); }, (error) => { return (error.reason === "basic authentication requires a secure https url"); }, "throws an exception for insecure connections"); }); }); describe("Test API Key Formatting", function () { it("Infura API Key", function () { const projectId = "someProjectId"; const projectSecret = "someSecretKey"; // Test simple projectId const apiKeyString = ethers.providers.InfuraProvider.getApiKey(projectId); assert.equal(apiKeyString.apiKey, projectId); assert.equal(apiKeyString.projectId, projectId); assert.ok(apiKeyString.secretKey == null); // Test complex API key with projectId const apiKeyObject = ethers.providers.InfuraProvider.getApiKey({ projectId }); assert.equal(apiKeyObject.apiKey, projectId); assert.equal(apiKeyObject.projectId, projectId); assert.ok(apiKeyObject.projectSecret == null); // Test complex API key with projectId and projectSecret const apiKeyObject2 = ethers.providers.InfuraProvider.getApiKey({ projectId: projectId, projectSecret: projectSecret }); assert.equal(apiKeyObject2.apiKey, projectId); assert.equal(apiKeyObject2.projectId, projectId); assert.equal(apiKeyObject2.projectSecret, projectSecret); // Fails on invalid projectId type assert.throws(() => { const apiKey = ethers.providers.InfuraProvider.getApiKey({ projectId: 1234, projectSecret: projectSecret }); console.log(apiKey); }, (error) => { return (error.argument === "projectId" && error.reason === "projectSecret requires a projectId"); }); // Fails on invalid projectSecret type assert.throws(() => { const apiKey = ethers.providers.InfuraProvider.getApiKey({ projectId: projectId, projectSecret: 1234 }); console.log(apiKey); }, (error) => { return (error.argument === "projectSecret" && error.reason === "invalid projectSecret"); }); { const provider = new ethers.providers.InfuraProvider("homestead", { projectId: projectId, projectSecret: projectSecret }); assert.equal(provider.network.name, "homestead"); assert.equal(provider.apiKey, projectId); assert.equal(provider.projectId, projectId); assert.equal(provider.projectSecret, projectSecret); } // Attempt an unsupported network assert.throws(() => { const provider = new ethers.providers.InfuraProvider("imaginary"); console.log(provider); }, (error) => { return (error.argument === "network" && error.reason === "unsupported network"); }); }); it("Pocket API key", function () { const applicationId = "someApplicationId"; const applicationSecretKey = "someApplicationSecret"; // Test simple applicationId const apiKeyString = ethers.providers.PocketProvider.getApiKey(applicationId); assert.equal(apiKeyString.applicationId, applicationId); assert.ok(apiKeyString.applicationSecretKey == null); // Test complex API key with applicationId const apiKeyObject = ethers.providers.PocketProvider.getApiKey({ applicationId }); assert.equal(apiKeyObject.applicationId, applicationId); assert.ok(apiKeyObject.applicationSecretKey == null); // Test complex API key with applicationId and applicationSecretKey const apiKeyObject2 = ethers.providers.PocketProvider.getApiKey({ applicationId: applicationId, applicationSecretKey: applicationSecretKey }); assert.equal(apiKeyObject2.applicationId, applicationId); assert.equal(apiKeyObject2.applicationSecretKey, applicationSecretKey); // Test complex API key with loadBalancer [true, false].forEach((loadBalancer) => { const apiKeyObject = ethers.providers.PocketProvider.getApiKey({ applicationId, loadBalancer }); assert.equal(apiKeyObject.applicationId, applicationId); assert.equal(apiKeyObject.loadBalancer, loadBalancer); assert.ok(apiKeyObject.applicationSecretKey == null); const apiKeyObject2 = ethers.providers.PocketProvider.getApiKey({ applicationId, applicationSecretKey, loadBalancer }); assert.equal(apiKeyObject2.applicationId, applicationId); assert.equal(apiKeyObject2.applicationSecretKey, applicationSecretKey); assert.equal(apiKeyObject2.loadBalancer, loadBalancer); }); // Fails on invalid applicationId type assert.throws(() => { const apiKey = ethers.providers.PocketProvider.getApiKey({ applicationId: 1234, applicationSecretKey: applicationSecretKey }); console.log(apiKey); }, (error) => { return (error.argument === "applicationId" && error.reason === "applicationSecretKey requires an applicationId"); }); // Fails on invalid projectSecret type assert.throws(() => { const apiKey = ethers.providers.PocketProvider.getApiKey({ applicationId: applicationId, applicationSecretKey: 1234 }); console.log(apiKey); }, (error) => { return (error.argument === "applicationSecretKey" && error.reason === "invalid applicationSecretKey"); }); { const provider = new ethers.providers.PocketProvider("homestead", { applicationId: applicationId, applicationSecretKey: applicationSecretKey }); assert.equal(provider.network.name, "homestead"); assert.equal(provider.applicationId, applicationId); assert.equal(provider.applicationSecretKey, applicationSecretKey); } // Attempt an unsupported network assert.throws(() => { const provider = new ethers.providers.PocketProvider("imaginary"); console.log(provider); }, (error) => { return (error.argument === "network" && error.reason === "unsupported network"); }); }); }); describe("Test WebSocketProvider", function () { this.retries(3); function testWebSocketProvider(provider) { return __awaiter(this, void 0, void 0, function* () { yield provider.destroy(); }); } it("InfuraProvider.getWebSocketProvider", function () { return __awaiter(this, void 0, void 0, function* () { const provider = ethers.providers.InfuraProvider.getWebSocketProvider(); yield testWebSocketProvider(provider); }); }); }); describe("Test Events", function () { this.retries(3); function testBlockEvent(provider) { return __awaiter(this, void 0, void 0, function* () { return new Promise((resolve, reject) => { let firstBlockNumber = null; const handler = (blockNumber) => { if (firstBlockNumber == null) { firstBlockNumber = blockNumber; return; } provider.removeListener("block", handler); if (firstBlockNumber + 1 === blockNumber) { resolve(true); } else { reject(new Error("blockNumber fail")); } }; provider.on("block", handler); }); }); } it("InfuraProvider", function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(60000); const provider = new ethers.providers.InfuraProvider("rinkeby"); yield testBlockEvent(provider); }); }); }); describe("Bad ENS resolution", function () { const provider = providerFunctions[0].create("ropsten"); it("signer has a bad ENS name", function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(300000); const wallet = new ethers.Wallet(ethers.utils.id("random-wallet"), provider); // If "to" is specified as an ENS name, it cannot resolve to null try { const tx = yield wallet.sendTransaction({ to: "junk", value: 1 }); console.log("TX", tx); assert.ok(false, "failed to throw an exception"); } catch (error) { assert.ok(error.argument === "tx.to" && error.value === "junk"); } // But promises that resolve to null are ok const tos = [null, Promise.resolve(null)]; for (let i = 0; i < tos.length; i++) { const to = tos[i]; try { const tx = yield wallet.sendTransaction({ to, value: 1 }); console.log("TX", tx); } catch (error) { assert.ok(error.code === "INSUFFICIENT_FUNDS"); } } }); }); }); describe("Resolve ENS avatar", function () { [ { title: "data", name: "data-avatar.tests.eth", value: "" }, { title: "ipfs", name: "ipfs-avatar.tests.eth", value: "https:/\/gateway.ipfs.io/ipfs/QmQsQgpda6JAYkFoeVcj5iPbwV3xRcvaiXv3bhp1VuYUqw" }, { title: "url", name: "url-avatar.tests.eth", value: "https:/\/ethers.org/static/logo.png" }, ].forEach((test) => { it(`Resolves avatar for ${test.title}`, function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(60000); const provider = ethers.getDefaultProvider("ropsten", getApiKeys("ropsten")); const avatar = yield provider.getAvatar(test.name); assert.equal(test.value, avatar, "avatar url"); }); }); }); [ { title: "ERC-1155", name: "nick.eth", value: "https:/\/lh3.googleusercontent.com/hKHZTZSTmcznonu8I6xcVZio1IF76fq0XmcxnvUykC-FGuVJ75UPdLDlKJsfgVXH9wOSmkyHw0C39VAYtsGyxT7WNybjQ6s3fM3macE" }, { title: "ERC-721", name: "brantly.eth", value: "https:/\/api.wrappedpunks.com/images/punks/2430.png" } ].forEach((test) => { it(`Resolves avatar for ${test.title}`, function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(60000); const provider = ethers.getDefaultProvider("homestead", getApiKeys("homestead")); const avatar = yield provider.getAvatar(test.name); assert.equal(avatar, test.value, "avatar url"); }); }); }); }); describe("Test EIP-2544 ENS wildcards", function () { const provider = (providerFunctions[0].create("ropsten")); it("Resolves recursively", function () { return __awaiter(this, void 0, void 0, function* () { const resolver = yield provider.getResolver("ricmoose.hatch.eth"); assert.equal(resolver.address, "0x8fc4C380c5d539aE631daF3Ca9182b40FB21D1ae", "found the correct resolver"); assert.equal(yield resolver.supportsWildcard(), true, "supportsWildcard"); assert.equal((yield resolver.getAvatar()).url, "https://static.ricmoo.com/uploads/profile-06cb9c3031c9.jpg", "gets passed-through avatar"); assert.equal(yield resolver.getAddress(), "0x4FaBE0A3a4DDd9968A7b4565184Ad0eFA7BE5411", "gets resolved address"); }); }); }); describe("Test CCIP execution", function () { const address = "0xAe375B05A08204C809b3cA67C680765661998886"; const ABI = [ //'error OffchainLookup(address sender, string[] urls, bytes callData, bytes4 callbackFunction, bytes extraData)', 'function testGet(bytes callData) view returns (bytes32)', 'function testGetFail(bytes callData) view returns (bytes32)', 'function testGetSenderFail(bytes callData) view returns (bytes32)', 'function testGetFallback(bytes callData) view returns (bytes32)', 'function testGetMissing(bytes callData) view returns (bytes32)', 'function testPost(bytes callData) view returns (bytes32)', 'function verifyTest(bytes result, bytes extraData) pure returns (bytes32)' ]; const provider = providerFunctions[0].create("ropsten"); const contract = new ethers.Contract(address, ABI, provider); // This matches the verify method in the Solidity contract against the // processed data from the endpoint const verify = function (sender, data, result) { const check = ethers.utils.concat([ ethers.utils.arrayify(ethers.utils.arrayify(sender).length), sender, ethers.utils.arrayify(ethers.utils.arrayify(data).length), data ]); assert.equal(result, ethers.utils.keccak256(check), "response is equal"); }; it("testGet passes under normal operation", function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(60000); const data = "0x1234"; const result = yield contract.testGet(data, { ccipReadEnabled: true }); verify(ethers.constants.AddressZero, data, result); }); }); it("testGet should fail with CCIP not explicitly enabled by overrides", function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(60000); try { const data = "0x1234"; const result = yield contract.testGet(data); console.log(result); assert.fail("throw-failed"); } catch (error) { if (error.message === "throw-failed") { throw error; } if (error.code !== "CALL_EXCEPTION") { console.log(error); assert.fail("failed"); } } }); }); it("testGet should fail with CCIP explicitly disabled on provider", function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(60000); const provider = providerFunctions[0].create("ropsten"); provider.disableCcipRead = true; const contract = new ethers.Contract(address, ABI, provider); try { const data = "0x1234"; const result = yield contract.testGet(data, { ccipReadEnabled: true }); console.log(result); assert.fail("throw-failed"); } catch (error) { if (error.message === "throw-failed") { throw error; } if (error.code !== "CALL_EXCEPTION") { console.log(error); assert.fail("failed"); } } }); }); it("testGetFail should fail if all URLs 5xx", function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(60000); try { const data = "0x1234"; const result = yield contract.testGetFail(data, { ccipReadEnabled: true }); console.log(result); assert.fail("throw-failed"); } catch (error) { if (error.message === "throw-failed") { throw error; } if (error.code !== "SERVER_ERROR" || (error.errorMessages || []).pop() !== "hello world") { console.log(error); assert.fail("failed"); } } }); }); it("testGetSenderFail should fail if sender does not match", function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(60000); try { const data = "0x1234"; const result = yield contract.testGetSenderFail(data, { ccipReadEnabled: true }); console.log(result); assert.fail("throw-failed"); } catch (error) { if (error.message === "throw-failed") { throw error; } if (error.code !== "CALL_EXCEPTION") { console.log(error); assert.fail("failed"); } } }); }); it("testGetMissing should fail if early URL 4xx", function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(60000); try { const data = "0x1234"; const result = yield contract.testGetMissing(data, { ccipReadEnabled: true }); console.log(result); assert.fail("throw-failed"); } catch (error) { if (error.message === "throw-failed") { throw error; } if (error.code !== "SERVER_ERROR" || error.errorMessage !== "hello world") { console.log(error); assert.fail("failed"); } } }); }); it("testGetFallback passes if any URL returns correctly", function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(60000); const data = "0x123456"; const result = yield contract.testGetFallback(data, { ccipReadEnabled: true }); verify(ethers.constants.AddressZero, data, result); }); }); it("testPost passes under normal operation", function () { return __awaiter(this, void 0, void 0, function* () { this.timeout(60000); const data = "0x1234"; const result = yield contract.testPost(data, { ccipReadEnabled: true }); verify(ethers.constants.AddressZero, data, result); }); }); }); //# sourceMappingURL=test-providers.js.map