Contract.php 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. <?php
  2. namespace App\Services\Web3;
  3. use kornrunner\Keccak;
  4. class Contract
  5. {
  6. private $address;
  7. private $abi;
  8. private $web3;
  9. private $event;
  10. private $eventHash;
  11. private $function;
  12. private function __construct(Web3 $web3, $abi, $address)
  13. {
  14. $this->abi = $abi;
  15. $this->web3 = $web3;
  16. $this->address = $address;
  17. $json = json_decode($this->abi, true);
  18. foreach ($json as $item) {
  19. switch ($item['type']) {
  20. case 'event':
  21. $this->event[$item['name']] = $item;
  22. $hash = $this->encodeEventSignature($this->getEventByName($item['name']));
  23. $this->eventHash[$hash] = $item['name'];
  24. break;
  25. case 'function':
  26. $this->function[$item['name']] = $item;
  27. break;
  28. }
  29. }
  30. }
  31. public static function at(Web3 $web3, $abi, $address): Contract
  32. {
  33. return new Contract($web3, $abi, $address);
  34. }
  35. public function getFunctionArrayByName($name)
  36. {
  37. return $this->function[$name];
  38. }
  39. public function getEventArrayByName($name)
  40. {
  41. return $this->event[$name];
  42. }
  43. public function getWeb3(): Web3
  44. {
  45. return $this->web3;
  46. }
  47. public function getAddress()
  48. {
  49. return $this->address;
  50. }
  51. public function getFunctionByName($name): string
  52. {
  53. $inputs = $this->function[$name]['inputs'];
  54. return $this->extracted($name, $inputs);
  55. }
  56. public function getEventByName($name): string
  57. {
  58. $inputs = $this->event[$name]['inputs'];
  59. return $this->extracted($name, $inputs);
  60. }
  61. public function getFunctionName(): array
  62. {
  63. return array_keys($this->function);
  64. }
  65. public function getEventName(): array
  66. {
  67. return array_keys($this->event);
  68. }
  69. public function decodeEvent($hash)
  70. {
  71. if (!array_key_exists($hash, $this->eventHash)) {
  72. throw new \Exception("this hash not in the constract");
  73. }
  74. return $this->eventHash[$hash];
  75. }
  76. public function getAbi()
  77. {
  78. return $this->abi;
  79. }
  80. /**
  81. * @throws \Exception
  82. */
  83. public function send(Wallet $wallet, $function, array $param, $config = [])
  84. {
  85. $data = $this->getData($function, $param, $wallet->getAddress());
  86. $neirong = $this->web3->estimateGas($data['to'], $data['data'], $data['from'], null, $data['value']);
  87. // if(isset($neirong['code'])){
  88. // return $neirong;
  89. // }
  90. $neirong = '0x5480';
  91. $data['gas'] = dechex(hexdec($neirong) * 1.5);
  92. $data['gasPrice'] = $this->web3->gasPrice();
  93. $data['nonce'] = $this->web3->getTransactionCount($wallet->getAddress(), 'pending');
  94. unset($data['from']);
  95. // $chainId = $this->web3->chainId();
  96. $data = array_merge([
  97. 'nonce' => '01',
  98. 'gasPrice' => '',
  99. 'gas' => '',
  100. 'to' => '',
  101. 'value' => '',
  102. 'data' => '',
  103. ], $data);
  104. $signature = $wallet->sign(Utils::rawEncode($data));
  105. $chainId = 0;
  106. $data['v'] = dechex($signature->recoveryParam + 27 + ($chainId ? $chainId * 2 + 8 : 0));
  107. $data['r'] = $signature->r->toString('hex');
  108. $data['s'] = $signature->s->toString('hex');
  109. $signRaw = Utils::add0x(Utils::rawEncode($data));
  110. return $this->web3->sendRawTransaction($signRaw);
  111. }
  112. /**
  113. * @param $function
  114. * @param $param
  115. * @param null $from
  116. * @return array
  117. * @throws \Exception
  118. */
  119. private function getData($function, $param, $from = null): array
  120. {
  121. if (!array_key_exists($function, $this->function)) {
  122. throw new \Exception(" function not in contract ");
  123. }
  124. $function = $this->getFunctionArrayByName($function);
  125. if (count($param) != count($function['inputs'])) {
  126. throw new \Exception("please send full param");
  127. }
  128. $data = [
  129. 'to' => $this->address,
  130. 'value' => '0x0'
  131. ];
  132. if (!empty($from)) {
  133. $data['from'] = $from;
  134. }
  135. $hash = Keccak::hash($this->getFunctionByName($function['name']), 256);
  136. $hashSub = mb_substr($hash, 0, 8, 'utf-8');
  137. $data['data'] = '0x' . $hashSub;
  138. $input = $function['inputs'];
  139. for ($i = 0; $i < count($param); $i++) {
  140. $value = '';
  141. switch ($input[$i]['type']) {
  142. case 'address':
  143. $value = Utils::remove0x($param[$i]);
  144. break;
  145. case 'uint8':
  146. case 'uint16':
  147. case 'uint24':
  148. case 'uint32':
  149. case 'uint40':
  150. case 'uint48':
  151. case 'uint56':
  152. case 'uint64':
  153. case 'uint72':
  154. case 'uint80':
  155. case 'uint88':
  156. case 'uint96':
  157. case 'uint104':
  158. case 'uint112':
  159. case 'uint120':
  160. case 'uint128':
  161. case 'uint136':
  162. case 'uint144':
  163. case 'uint152':
  164. case 'uint160':
  165. case 'uint168':
  166. case 'uint176':
  167. case 'uint184':
  168. case 'uint192':
  169. case 'uint200':
  170. case 'uint208':
  171. case 'uint216':
  172. case 'uint224':
  173. case 'uint232':
  174. case 'uint240':
  175. case 'uint248':
  176. case 'uint256':
  177. $value = Utils::decToHex($param[$i], false);
  178. break;
  179. }
  180. $data['data'] = $data['data'] . Utils::fill0($value);
  181. }
  182. return $data;
  183. }
  184. /**
  185. * encodeEventSignature
  186. * TODO: Fix same event name with different params
  187. *
  188. * @param string|stdClass|array $functionName
  189. * @return string
  190. */
  191. public function encodeEventSignature($functionName)
  192. {
  193. if (!is_string($functionName)) {
  194. $functionName = Utils::jsonMethodToString($functionName);
  195. }
  196. return Utils::sha3($functionName);
  197. }
  198. public function call($function, $param = [], $quantity = Quantity::latest)
  199. {
  200. $data = $this->getData($function, $param);
  201. return $this->web3->call($data['to'], $data['data'], null, null, null, null, $quantity);
  202. }
  203. /**
  204. * @param $name
  205. * @param $inputs
  206. * @return string
  207. */
  208. public function extracted($name, $inputs): string
  209. {
  210. $res = $name . '(';
  211. foreach ($inputs as $input) {
  212. $res .= $input['type'] . ',';
  213. }
  214. if (count($inputs) > 0) {
  215. $res = substr($res, 0, strlen($res) - 1);
  216. }
  217. $res .= ')';
  218. return $res;
  219. }
  220. }