Web3.php 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823
  1. <?php
  2. namespace App\Services\Web3;
  3. use GuzzleHttp\Client as GuzzleHttp;
  4. class Web3
  5. {
  6. private $url;
  7. protected $requestId = 0;
  8. protected $client;
  9. protected $chainId;
  10. public function getProvider()
  11. {
  12. return $this->provider;
  13. }
  14. public function __construct($url)
  15. {
  16. $this->url = $url;
  17. $this->client = new GuzzleHttp(array_merge(['timeout' => 60, 'verify' => false], ['base_uri' => $url]));
  18. }
  19. /**
  20. * Returns the current client version
  21. * @throws \Exception
  22. * Returns: String - the current client version
  23. */
  24. public function clientVersion()
  25. {
  26. return $this->request('web3_clientVersion', []);
  27. }
  28. /**
  29. * Returns Keccak-256 (not the standardized SHA3-256) of the given data.
  30. * @param string data
  31. * @throws \Exception
  32. * Returns: DATA - The SHA3 result of the given string.
  33. */
  34. public function sha3(string $data)
  35. {
  36. return $this->request('web3_sha3', [$data]);
  37. }
  38. /**
  39. * Returns the current network id.
  40. * @throws \Exception
  41. * Returns: String - The current network id.
  42. * "1": Ethereum Mainnet
  43. * "2": Morden Testnet (deprecated)
  44. * "3": Ropsten Testnet
  45. * "4": Rinkeby Testnet
  46. * "42": Kovan Testnet
  47. */
  48. public function netVersion()
  49. {
  50. if (empty($this->chainId)) {
  51. $this->chainId = $this->request('net_version', []);
  52. }
  53. return $this->chainId;
  54. }
  55. /**
  56. * Returns true if client is actively listening for network connections.
  57. * @throws \Exception
  58. * Returns: Boolean - true when listening, otherwise false.
  59. */
  60. public function netListening()
  61. {
  62. return $this->request('net_listening', []);
  63. }
  64. /**
  65. * Returns number of peers currently connected to the client.
  66. * @throws \Exception
  67. * Returns: QUANTITY - integer of the number of connected peers.
  68. */
  69. public function netPeerCount()
  70. {
  71. return $this->request('net_peerCount', []);
  72. }
  73. /**
  74. * Returns the current ethereum protocol version.
  75. * @throws \Exception
  76. * Returns: String - The current ethereum protocol version
  77. */
  78. public function protocolVersion()
  79. {
  80. return $this->request('eth_protocolVersion', []);
  81. }
  82. /**
  83. * Returns an object with data about the sync status or false.
  84. * @throws \Exception
  85. * Returns: Object|Boolean, An object with sync status data or FALSE, when not syncing:
  86. * startingBlock: QUANTITY - The block at which the import started (will only be reset, after the sync reached his head)
  87. * currentBlock: QUANTITY - The current block, same as eth_blockNumber
  88. * highestBlock: QUANTITY - The estimated highest block
  89. */
  90. public function syncing()
  91. {
  92. return $this->request('eth_syncing', []);
  93. }
  94. /**
  95. * Returns the client coinbase address.
  96. * @throws \Exception
  97. * Returns: DATA, 20 bytes - the current coinbase address.
  98. */
  99. public function coinbase()
  100. {
  101. return $this->request('eth_coinbase', []);
  102. }
  103. /**
  104. * Returns true if client is actively mining new blocks.
  105. * @throws \Exception
  106. * Returns: Boolean - returns true of the client is mining, otherwise false.
  107. */
  108. public function mining()
  109. {
  110. return $this->request('eth_mining', []);
  111. }
  112. /**
  113. * Returns the number of hashes per second that the node is mining with.
  114. * @throws \Exception
  115. * Returns: QUANTITY - number of hashes per second.
  116. */
  117. public function hashrate()
  118. {
  119. return $this->request('eth_hashrate', []);
  120. }
  121. /**
  122. * Returns the current price per gas in wei.
  123. * @throws \Exception
  124. * Returns: QUANTITY - integer of the current gas price in wei.
  125. */
  126. public function gasPrice()
  127. {
  128. return $this->request('eth_gasPrice', []);
  129. }
  130. /**
  131. * Returns a list of addresses owned by client.
  132. * @throws \Exception
  133. * Returns: Array of DATA, 20 Bytes - addresses owned by the client.
  134. */
  135. public function accounts()
  136. {
  137. return $this->request('eth_accounts', []);
  138. }
  139. /**
  140. * Returns the balance of the account of given address.
  141. * @param $address , 20 Bytes - address to check for balance.
  142. * @param string $block - integer block number, or the string "latest", "earliest" or "pending", see the default block parameter
  143. * @throws \Exception
  144. * Returns: QUANTITY - integer of the current balance in wei.
  145. */
  146. public function getBalance($address, string $block = 'latest')
  147. {
  148. return $this->request('eth_getBalance', [$address, $block]);
  149. }
  150. /**
  151. * Returns the value from a storage position at a given address.
  152. * @param $data
  153. * @param $quantity1
  154. * @param string $quantity2
  155. * @return mixed
  156. * @throws \Exception
  157. * Returns: DATA - the value at this storage position.
  158. */
  159. public function getStorageAt($data, $quantity1, string $quantity2 = Quantity::latest)
  160. {
  161. return $this->request('eth_getStorageAt', [$data, $quantity1, $quantity2]);
  162. }
  163. /**
  164. * Returns the number of transactions sent from an address.
  165. * @param $address
  166. * @param $quantity
  167. * @throws \Exception
  168. * Returns: QUANTITY - integer of the number of transactions send from this address.
  169. */
  170. public function getTransactionCount($address, $quantity = Quantity::latest)
  171. {
  172. return $this->request('eth_getTransactionCount', [$address, $quantity]);
  173. }
  174. /**
  175. * Returns the number of transactions in a block from a block matching the given block hash.
  176. * @param $hash
  177. * @throws \Exception
  178. * Returns: QUANTITY - integer of the number of transactions in this block.
  179. */
  180. public function getBlockTransactionCountByHash($hash)
  181. {
  182. return $this->request('eth_getBlockTransactionCountByHash', [$hash]);
  183. }
  184. /**
  185. * Returns the number of transactions in a block matching the given block number.
  186. * @param string $quantity
  187. * @throws \Exception
  188. * Returns: QUANTITY - integer of the number of transactions in this block.
  189. */
  190. public function getBlockTransactionCountByNumber(string $quantity = Quantity::latest)
  191. {
  192. return $this->request('eth_getBlockTransactionCountByNumber', [$quantity]);
  193. }
  194. /**
  195. * Returns the number of uncles in a block from a block matching the given block hash.
  196. * @param $hash
  197. * @throws \Exception
  198. * Returns: QUANTITY - integer of the number of uncles in this block.
  199. */
  200. public function getUncleCountByBlockHash($hash)
  201. {
  202. return $this->request('eth_getUncleCountByBlockHash', [$hash]);
  203. }
  204. /**
  205. * Returns the number of uncles in a block from a block matching the given block number.
  206. * @param $quantity
  207. * @throws \Exception
  208. * Returns: QUANTITY - integer of the number of uncles in this block.
  209. */
  210. public function getUncleCountByBlockNumber($quantity = Quantity::latest)
  211. {
  212. return $this->request('eth_getUncleCountByBlockNumber', [$quantity]);
  213. }
  214. /**
  215. * Returns code at a given address.
  216. * @param $data
  217. * @param string $quantity
  218. * @throws \Exception
  219. * Returns: DATA - the code from the given address.
  220. */
  221. public function getCode($data, string $quantity = Quantity::latest)
  222. {
  223. return $this->request('eth_getCode', [$data, $quantity]);
  224. }
  225. /**
  226. * The sign method calculates an Ethereum specific signature with:
  227. * sign(keccak256("\x19Ethereum Signed Message:\n" + len(message) + message))).
  228. *
  229. * By adding a prefix to the message makes the calculated signature recognisable as an Ethereum specific signature.
  230. * This prevents misuse where a malicious DApp can sign arbitrary data (e.g. transaction) and use the signature to impersonate the victim.
  231. *
  232. * Note the address to sign with must be unlocked.
  233. * @param $address
  234. * @param $data
  235. * @throws \Exception
  236. * Returns: DATA: Signature
  237. */
  238. public function sign($address, $data)
  239. {
  240. return $this->request('eth_sign', [$address, $data]);
  241. }
  242. /**
  243. * Creates new message call transaction or a contract creation, if the data field contains code.
  244. * @param $from
  245. * @param $to
  246. * @param $gas
  247. * @param $value
  248. * @param $data
  249. * @param $gasPrice
  250. * @param $nonce
  251. * @throws \Exception
  252. * Returns: DATA, 32 Bytes - the transaction hash, or the zero hash if the transaction is not yet available.
  253. * Use eth_getTransactionReceipt to get the contract address, after the transaction was mined, when you created a contract.
  254. */
  255. public function sendTransaction($from, $to, $gas = 90000, $value = '0x0', $data = '0x0', $gasPrice = null, $nonce = null)
  256. {
  257. $data = [
  258. 'from' => $from,
  259. 'to' => $to,
  260. 'gas' => $gas,
  261. 'value' => $value,
  262. 'data' => $data,
  263. ];
  264. if (!empty($nonce)) {
  265. $data['nonce'] = $nonce;
  266. }
  267. if (!empty($gasPrice)) {
  268. $data['gasPrice'] = $gasPrice;
  269. }
  270. return $this->request('eth_sendTransaction', $data);
  271. }
  272. /**
  273. * Creates new message call transaction or a contract creation for signed transactions.
  274. * @param $data
  275. * @throws \Exception
  276. * Returns: DATA, 32 Bytes - the transaction hash, or the zero hash if the transaction is not yet available.
  277. * Use eth_getTransactionReceipt to get the contract address, after the transaction was mined, when you created a contract.
  278. */
  279. public function sendRawTransaction($data)
  280. {
  281. return $this->request('eth_sendRawTransaction', [$data]);
  282. }
  283. /**
  284. * Executes a new message call immediately without creating a transaction on the block chain.
  285. * @param $to
  286. * @param string $data
  287. * @param $from
  288. * @param $gas
  289. * @param $value
  290. * @param $gasPrice
  291. * @return mixed
  292. * @throws \Exception Returns: DATA - the return value of executed contract.
  293. */
  294. public function call( $to, string $data = '0x0', $from = null, $gas = null, $value = null, $gasPrice = null,$quantity=Quantity::latest)
  295. {
  296. $data0 = [
  297. 'to' => $to,
  298. ];
  299. if (!empty($gas)) {
  300. $data0['gas'] = $gas;
  301. }
  302. if (!empty($from)) {
  303. $data0['from'] = $from;
  304. }
  305. if (!empty($value)) {
  306. $data0['value'] = $value;
  307. }
  308. if (!empty($data)) {
  309. $data0['data'] = $data;
  310. }
  311. if (!empty($gasPrice)) {
  312. $data0['gasPrice'] = $gasPrice;
  313. }
  314. return $this->request('eth_call', [$data0,$quantity]);
  315. }
  316. /**
  317. * Generates and returns an estimate of how much gas is necessary to allow the transaction to complete.The transaction will not be added to the blockchain.
  318. * Note that the estimate may be significantly more than the amount of gas actually used by the transaction, for a variety of reasons including EVM mechanics and node performance.
  319. * @param $to
  320. * @param string $data
  321. * @param $from
  322. * @param $gas
  323. * @param $value
  324. * @param $gasPrice
  325. * @param $nonce
  326. * @throws \Exception
  327. * Returns: QUANTITY - the amount of gas used.
  328. */
  329. public function estimateGas($to, string $data = '0x0', $from = null, $gas = null, $value = null, $gasPrice = null, $nonce = null)
  330. {
  331. $data0 = [];
  332. if (!empty($from)) {
  333. $data0['from'] = $from;
  334. }
  335. $data0['to'] = $to;
  336. if (!empty($gas)) {
  337. $data0['gas'] = $gas;
  338. }
  339. if (!empty($value)) {
  340. $data0['value'] = $value;
  341. }
  342. if (!empty($data)) {
  343. $data0['data'] = $data;
  344. }
  345. if (!empty($gasPrice)) {
  346. $data0['gasPrice'] = $gasPrice;
  347. }
  348. return $this->request('eth_estimateGas', [$data0]);
  349. }
  350. /**
  351. * Returns information about a block by hash.
  352. * @param $hash
  353. * @throws \Exception
  354. * Returns: Object - A block object, or null when no block was found:
  355. * number: QUANTITY - the block number. null when its pending block.
  356. * hash: DATA, 32 Bytes - hash of the block. null when its pending block.
  357. * parentHash: DATA, 32 Bytes - hash of the parent block.
  358. * nonce: DATA, 8 Bytes - hash of the generated proof-of-work. null when its pending block.
  359. * sha3Uncles: DATA, 32 Bytes - SHA3 of the uncles data in the block.
  360. * logsBloom: DATA, 256 Bytes - the bloom filter for the logs of the block. null when its pending block.
  361. * transactionsRoot: DATA, 32 Bytes - the root of the transaction trie of the block.
  362. * stateRoot: DATA, 32 Bytes - the root of the final state trie of the block.
  363. * receiptsRoot: DATA, 32 Bytes - the root of the receipts trie of the block.
  364. * miner: DATA, 20 Bytes - the address of the beneficiary to whom the mining rewards were given.
  365. * difficulty: QUANTITY - integer of the difficulty for this block.
  366. * totalDifficulty: QUANTITY - integer of the total difficulty of the chain until this block.
  367. * extraData: DATA - the “extra data” field of this block.
  368. * size: QUANTITY - integer the size of this block in bytes.
  369. * gasLimit: QUANTITY - the maximum gas allowed in this block.
  370. * gasUsed: QUANTITY - the total used gas by all transactions in this block.
  371. * timestamp: QUANTITY - the unix timestamp for when the block was collated.
  372. * transactions: Array - Array of transaction objects, or 32 Bytes transaction hashes depending on the last given parameter.
  373. * uncles: Array - Array of uncle hashes.
  374. */
  375. public function getBlockByHash($hash)
  376. {
  377. return $this->request('eth_getBlockByHash', [$hash]);
  378. }
  379. /**
  380. * Returns information about a block by block number.
  381. * @param string $quantity
  382. * @param bool $allObject
  383. * @throws \Exception
  384. * Returns: See eth_getBlockByHash
  385. */
  386. public function getBlockByNumber(string $quantity = Quantity::latest, bool $allObject = false)
  387. {
  388. return $this->request('eth_getBlockByNumber', [$quantity, $allObject]);
  389. }
  390. /**
  391. * Returns the information about a transaction requested by transaction hash.
  392. * @param $hash
  393. * @throws \Exception
  394. * Returns: Object - A transaction object, or null when no transaction was found:
  395. * blockHash: DATA, 32 Bytes - hash of the block where this transaction was in. null when its pending.
  396. * blockNumber: QUANTITY - block number where this transaction was in. null when its pending.
  397. * from: DATA, 20 Bytes - address of the sender.
  398. * gas: QUANTITY - gas provided by the sender.
  399. * gasPrice: QUANTITY - gas price provided by the sender in Wei.
  400. * hash: DATA, 32 Bytes - hash of the transaction.
  401. * input: DATA - the data send along with the transaction.
  402. * nonce: QUANTITY - the number of transactions made by the sender prior to this one.
  403. * to: DATA, 20 Bytes - address of the receiver. null when its a contract creation transaction.
  404. * transactionIndex: QUANTITY - integer of the transactions index position in the block. null when its pending.
  405. * value: QUANTITY - value transferred in Wei.
  406. * v: QUANTITY - ECDSA recovery id
  407. * r: DATA, 32 Bytes - ECDSA signature r
  408. * s: DATA, 32 Bytes - ECDSA signature s
  409. */
  410. public function getTransactionByHash($hash)
  411. {
  412. return $this->request('eth_getTransactionByHash', [$hash]);
  413. }
  414. /**
  415. * Returns information about a transaction by block hash and transaction index position.
  416. * @param $hash
  417. * @param int $index
  418. * @throws \Exception
  419. * Returns: See eth_getTransactionByHash
  420. */
  421. public function getTransactionByBlockHashAndIndex($hash, int $index = 0)
  422. {
  423. return $this->request('eth_getTransactionByBlockHashAndIndex', [$hash, $index]);
  424. }
  425. /**
  426. * Returns information about a transaction by block number and transaction index position.
  427. * @param string $quantity
  428. * @param int $index
  429. * @throws \Exception
  430. * Returns: See eth_getTransactionByHash
  431. */
  432. public function getTransactionByBlockNumberAndIndex(string $quantity = Quantity::latest, int $index = 0)
  433. {
  434. return $this->request('eth_getTransactionByBlockNumberAndIndex', [$quantity, $index]);
  435. }
  436. /**
  437. * Returns the receipt of a transaction by transaction hash.
  438. * Note That the receipt is not available for pending transactions.
  439. * @param $hash
  440. * @throws \Exception
  441. * Returns: Object - A transaction receipt object, or null when no receipt was found:
  442. * transactionHash : DATA, 32 Bytes - hash of the transaction.
  443. * transactionIndex: QUANTITY - integer of the transactions index position in the block.
  444. * blockHash: DATA, 32 Bytes - hash of the block where this transaction was in.
  445. * blockNumber: QUANTITY - block number where this transaction was in.
  446. * from: DATA, 20 Bytes - address of the sender.
  447. * to: DATA, 20 Bytes - address of the receiver. null when its a contract creation transaction.
  448. * cumulativeGasUsed : QUANTITY - The total amount of gas used when this transaction was executed in the block.
  449. * gasUsed : QUANTITY - The amount of gas used by this specific transaction alone.
  450. * contractAddress : DATA, 20 Bytes - The contract address created, if the transaction was a contract creation, otherwise null.
  451. * logs: Array - Array of log objects, which this transaction generated.
  452. * logsBloom: DATA, 256 Bytes - Bloom filter for light clients to quickly retrieve related logs.
  453. * It also returns either :
  454. * root : DATA 32 bytes of post-transaction stateroot (pre Byzantium)
  455. * status: QUANTITY either 1 (success) or 0 (failure)
  456. */
  457. public function getTransactionReceipt($hash)
  458. {
  459. return $this->request('eth_getTransactionReceipt', [$hash]);
  460. }
  461. /**
  462. * Returns information about a uncle of a block by hash and uncle index position.
  463. * @param $hash
  464. * @param int $index
  465. * @throws \Exception
  466. * Returns: See eth_getBlockByHash
  467. */
  468. public function getUncleByBlockHashAndIndex($hash, int $index = 0)
  469. {
  470. return $this->request('eth_getUncleByBlockHashAndIndex', [$hash, $index]);
  471. }
  472. /**
  473. * Returns information about a uncle of a block by number and uncle index position.
  474. * @param string $quantity
  475. * @param int $index
  476. * @throws \Exception
  477. * Returns: See eth_getBlockByHash
  478. * Note: An uncle doesn’t contain individual transactions.
  479. */
  480. public function getUncleByBlockNumberAndIndex(string $quantity = Quantity::latest, int $index = 0)
  481. {
  482. return $this->request('eth_getUncleByBlockNumberAndIndex', [$quantity, $index]);
  483. }
  484. /**
  485. * Returns a list of available compilers in the client.
  486. * @throws \Exception
  487. * Returns: Array - Array of available compilers.
  488. */
  489. public function getCompilers()
  490. {
  491. return $this->request('eth_getCompilers', []);
  492. }
  493. /**
  494. * Returns compiled solidity code.
  495. * @param $source
  496. * @throws \Exception
  497. * Returns: DATA - The compiled source code.
  498. */
  499. public function compileSolidity($source)
  500. {
  501. return $this->request('eth_compileSolidity', [$source]);
  502. }
  503. /**
  504. * Returns compiled LLL code.
  505. * @param $source
  506. * @throws \Exception
  507. * Returns: DATA - The compiled source code.
  508. */
  509. public function compileLLL($source)
  510. {
  511. return $this->request('eth_compileLLL', [$source]);
  512. }
  513. /**
  514. * Returns compiled serpent code.
  515. * @param $source
  516. * @throws \Exception
  517. * Returns: DATA - The compiled source code.
  518. */
  519. public function compileSerpent($source)
  520. {
  521. return $this->request('eth_compileSerpent', [$source]);
  522. }
  523. /**
  524. * Creates a filter object, based on filter options, to notify when the state changes (logs).
  525. * To check if the state has changed, call eth_getFilterChanges.
  526. *
  527. * A note on specifying topic filters:
  528. * Topics are order-dependent. A transaction with a log with topics [A, B] will be matched by the following topic filters:
  529. * [] “anything”
  530. * [A] “A in first position (and anything after)”
  531. * [null, B] “anything in first position AND B in second position (and anything after)”
  532. * [A, B] “A in first position AND B in second position (and anything after)”
  533. * [[A, B], [A, B]] “(A OR B) in first position AND (A OR B) in second position (and anything after)”
  534. *
  535. * @param $fromBlock
  536. * @param $toBlock
  537. * @param $address
  538. * @param $topic
  539. * @throws \Exception
  540. * Returns: QUANTITY - A filter id.
  541. */
  542. public function newFilter($fromBlock = null, $toBlock = null, $address = null, $topic = [])
  543. {
  544. $data = $this->getData($fromBlock, $toBlock, $address, $topic);
  545. return $this->request('eth_newFilter', [$data]);
  546. }
  547. /**
  548. * Creates a filter in the node, to notify when a new block arrives.To check if the state has changed, call eth_getFilterChanges.
  549. * @throws \Exception
  550. * Returns: QUANTITY - A filter id.
  551. */
  552. public function newBlockFilter()
  553. {
  554. return $this->request('eth_newBlockFilter', []);
  555. }
  556. /**
  557. * Creates a filter in the node, to notify when new pending transactions arrive.To check if the state has changed, call eth_getFilterChanges.
  558. * @throws \Exception
  559. * Returns: QUANTITY - A filter id.
  560. */
  561. public function newPendingTransactionFilter()
  562. {
  563. return $this->request('eth_newPendingTransactionFilter', []);
  564. }
  565. /**
  566. * Uninstalls a filter with given id. Should always be called when watch is no longer needed.
  567. * Additonally Filters timeout when they aren’t requested with eth_getFilterChanges for a period of time.
  568. * @param $filterNumber
  569. * @throws \Exception
  570. * Returns: Boolean - true if the filter was successfully uninstalled, otherwise false.
  571. */
  572. public function uninstallFilter($filterNumber)
  573. {
  574. return $this->request('eth_uninstallFilter', [$filterNumber]);
  575. }
  576. /**
  577. * Polling method for a filter, which returns an array of logs which occurred since last poll.
  578. * @param $filterNumber
  579. * @throws \Exception
  580. * Returns: Array - Array of log objects, or an empty array if nothing has changed since last poll.
  581. * For filters created with eth_newBlockFilter the return are block hashes (DATA, 32 Bytes), e.g. ["0x3454645634534..."].
  582. * For filters created with eth_newPendingTransactionFilter the return are transaction hashes (DATA, 32 Bytes), e.g. ["0x6345343454645..."].
  583. * For filters created with eth_newFilter logs are objects with following params:
  584. * removed: TAG - true when the log was removed, due to a chain reorganization. false if its a valid log.
  585. * logIndex: QUANTITY - integer of the log index position in the block. null when its pending log.
  586. * transactionIndex: QUANTITY - integer of the transactions index position log was created from. null when its pending log.
  587. * transactionHash: DATA, 32 Bytes - hash of the transactions this log was created from. null when its pending log.
  588. * blockHash: DATA, 32 Bytes - hash of the block where this log was in. null when its pending. null when its pending log.
  589. * blockNumber: QUANTITY - the block number where this log was in. null when its pending. null when its pending log.
  590. * address: DATA, 20 Bytes - address from which this log originated.
  591. * data: DATA - contains one or more 32 Bytes non-indexed arguments of the log.
  592. * topics: Array of DATA - Array of 0 to 4 32 Bytes DATA of indexed log arguments.
  593. * (In solidity: The first topic is the hash of the signature of the event (e.g. Deposit(address,bytes32,uint256)), except you declared the event with the anonymous specifier.)
  594. */
  595. public function getFilterChanges($filterNumber)
  596. {
  597. return $this->request('eth_getFilterChanges', [$filterNumber]);
  598. }
  599. /**
  600. * Returns an array of all logs matching filter with given id.
  601. * @param $filterNumber
  602. * @throws \Exception
  603. * Returns: See eth_getFilterChanges
  604. */
  605. public function getFilterLogs($filterNumber)
  606. {
  607. return $this->request('eth_getFilterLogs', [$filterNumber]);
  608. }
  609. /**
  610. * Returns an array of all logs matching a given filter object.
  611. * @param $fromBlock
  612. * @param $toBlock
  613. * @param $address
  614. * @param $topic
  615. * @throws \Exception
  616. * Returns: See eth_getFilterChanges
  617. */
  618. public function getLogs(string $fromBlock = null, string $toBlock = null, $address = null, $topic = [])
  619. {
  620. $data = $this->getData($fromBlock, $toBlock, $address, $topic);
  621. return $this->request('eth_getLogs', [$data]);
  622. }
  623. /**
  624. * Returns the hash of the current block, the seedHash, and the boundary condition to be met (“target”).
  625. * @throws \Exception
  626. * Returns: Array - Array with the following properties:
  627. * DATA, 32 Bytes - current block header pow-hash
  628. * DATA, 32 Bytes - the seed hash used for the DAG.
  629. * DATA, 32 Bytes - the boundary condition (“target”), 2^256 / difficulty.
  630. */
  631. public function getWork()
  632. {
  633. return $this->request('eth_getWork', []);
  634. }
  635. /**
  636. * Used for submitting a proof-of-work solution.
  637. * @param $data1
  638. * @param $data2
  639. * @param $data3
  640. * @throws \Exception
  641. * Returns: Boolean - returns true if the provided solution is valid, otherwise false.
  642. */
  643. public function submitWork($data1, $data2, $data3)
  644. {
  645. return $this->request('eth_submitWork', [$data1, $data2 . $data3]);
  646. }
  647. /**
  648. * Used for submitting mining hashrate.
  649. * @param $hashRate
  650. * @param $id
  651. * @throws \Exception
  652. * Returns: Boolean - returns true if submitting went through succesfully and false otherwise.
  653. */
  654. public function submitHashrate($hashRate, $id)
  655. {
  656. return $this->request('eth_submitHashrate', [$hashRate, $id]);
  657. }
  658. /**
  659. * Returns the number of most recent block.
  660. * @throws \Exception
  661. * Returns: QUANTITY - integer of the current block number the client is on.
  662. */
  663. public function blockNumber()
  664. {
  665. return $this->request('eth_blockNumber', []);
  666. }
  667. /**
  668. * Returns the current whisper protocol version.
  669. * @throws \Exception
  670. * Returns: String - The current whisper protocol version
  671. */
  672. public function shhVersion()
  673. {
  674. return $this->request('shh_version', []);
  675. }
  676. /**
  677. * Returns the current network id.
  678. * @throws \Exception
  679. * Returns: String - The current network id.
  680. * "1": Ethereum Mainnet
  681. * "2": Morden Testnet (deprecated)
  682. * "3": Ropsten Testnet
  683. * "4": Rinkeby Testnet
  684. * "42": Kovan Testnet
  685. */
  686. public function chainId(): int
  687. {
  688. return $this->netVersion();
  689. }
  690. public function request($method, $params = []){
  691. $data = [
  692. 'json' => [
  693. 'jsonrpc' => '2.0',
  694. 'method' => $method,
  695. 'params' => $params,
  696. 'id' => $this->requestId++,
  697. ]
  698. ];
  699. $res = $this->client->post('', $data);
  700. $body = json_decode($res->getBody());
  701. if (isset($body->error) && !empty($body->error)) {
  702. return ['code'=>400,'msg'=>$body->error->message . " [Method] {$method}"];
  703. throw new \Exception($body->error->message . " [Method] {$method}", $body->error->code);
  704. }
  705. return $body->result;
  706. }
  707. /**
  708. * @param $fromBlock
  709. * @param $toBlock
  710. * @param $address
  711. * @param $topic
  712. * @return array
  713. */
  714. private function getData($fromBlock, $toBlock, $address, $topic): array
  715. {
  716. $data = [];
  717. if (!empty($fromBlock)) {
  718. $data['fromBlock'] = $fromBlock;
  719. }else{
  720. $data['fromBlock'] ="0x0";
  721. }
  722. if (!empty($toBlock)) {
  723. $data['toBlock'] = $toBlock;
  724. }
  725. if (!empty($address)) {
  726. $data['address'] = $address;
  727. }
  728. if (!empty($topic)) {
  729. $data['topic'] = $topic;
  730. }
  731. return $data;
  732. }
  733. }