Utils.php 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. <?php
  2. namespace App\Services\Web3;
  3. use kornrunner\Keccak;
  4. use stdClass;
  5. class Utils
  6. {
  7. const SHA3_NULL_HASH = 'c5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470';
  8. public static function remove0x($value)
  9. {
  10. if (strtolower(substr($value, 0, 2)) == '0x') {
  11. return substr($value, 2);
  12. }
  13. return $value;
  14. }
  15. public static function add0x($value): string
  16. {
  17. return '0x' . self::remove0x($value);
  18. }
  19. public static function pubKeyToAddress($pubkey)
  20. {
  21. return '0x' . substr(Keccak::hash(substr(hex2bin($pubkey), 1), 256), 24);
  22. }
  23. /**
  24. * RLPencode
  25. */
  26. public static function rawEncode(array $input): string
  27. {
  28. $rlp = new RLP\RLP;
  29. $data = [];
  30. foreach ($input as $item) {
  31. // If the value is invalid: 0, 0x0, list it as an empty string
  32. $data[] = $item && hexdec(Utils::remove0x($item)) != 0 ? Utils::add0x($item) : '';
  33. }
  34. return $rlp->encode($data)->toString('hex');
  35. }
  36. /**
  37. *
  38. * @param string $str
  39. * @param int $bit
  40. * @return string
  41. */
  42. public static function fill0($str, $bit = 64)
  43. {
  44. $str_len = strlen($str);
  45. $zero = '';
  46. for ($i = $str_len; $i < $bit; $i++) {
  47. $zero .= "0";
  48. }
  49. $real_str = $zero . $str;
  50. return $real_str;
  51. }
  52. /**
  53. * ether to wei
  54. */
  55. public static function ethToWei($value, $hex = false)
  56. {
  57. $value = bcmul($value, '1000000000000000000');
  58. if ($hex) {
  59. return self::decToHex($value, $hex);
  60. }
  61. return $value;
  62. }
  63. /**
  64. * wei to ether
  65. */
  66. public static function weiToEth($value, $hex = false)
  67. {
  68. if (strtolower(substr($value, 0, 2)) == '0x') {
  69. $value = self::hexToDec(self::remove0x($value));
  70. }
  71. $value = bcdiv($value, '1000000000000000000', 18);
  72. if ($hex) {
  73. return '0x' . self::decToHex($value);
  74. }
  75. return $value;
  76. }
  77. /**
  78. * change to hex(0x)
  79. * @param string|number $value
  80. * @param boolean $mark
  81. * @return string
  82. */
  83. public static function decToHex($value, $mark = true)
  84. {
  85. $hexvalues = [
  86. '0', '1', '2', '3', '4', '5', '6', '7',
  87. '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'
  88. ];
  89. $hexval = '';
  90. while ($value != '0') {
  91. $hexval = $hexvalues[bcmod($value, '16')] . $hexval;
  92. $value = bcdiv($value, '16', 0);
  93. }
  94. return ($mark ? '0x' . $hexval : $hexval);
  95. }
  96. /**
  97. * change to hex(0x)
  98. * @param string $number hex number
  99. * @return string
  100. */
  101. public static function hexToDec($number)
  102. {
  103. // have 0x,remove it
  104. $number = self::remove0x(strtolower($number));
  105. $decvalues = [
  106. '0' => '0', '1' => '1', '2' => '2',
  107. '3' => '3', '4' => '4', '5' => '5',
  108. '6' => '6', '7' => '7', '8' => '8',
  109. '9' => '9', 'a' => '10', 'b' => '11',
  110. 'c' => '12', 'd' => '13', 'e' => '14',
  111. 'f' => '15'];
  112. $decval = '0';
  113. $number = strrev($number);
  114. for ($i = 0; $i < strlen($number); $i++) {
  115. $decval = bcadd(bcmul(bcpow('16', $i, 0), $decvalues[$number[$i]]), $decval);
  116. }
  117. return $decval;
  118. }
  119. /**
  120. * jsonMethodToString
  121. *
  122. * @param stdClass|array $json
  123. * @return string
  124. */
  125. public static function jsonMethodToString($json)
  126. {
  127. if ($json instanceof stdClass) {
  128. // one way to change whole json stdClass to array type
  129. // $jsonString = json_encode($json);
  130. // if (JSON_ERROR_NONE !== json_last_error()) {
  131. // throw new InvalidArgumentException('json_decode error: ' . json_last_error_msg());
  132. // }
  133. // $json = json_decode($jsonString, true);
  134. // another way to change whole json to array type but need the depth
  135. // $json = self::jsonToArray($json, $depth)
  136. // another way to change json to array type but not whole json stdClass
  137. $json = (array)$json;
  138. $typeName = [];
  139. foreach ($json['inputs'] as $param) {
  140. if (isset($param->type)) {
  141. $typeName[] = $param->type;
  142. }
  143. }
  144. return $json['name'] . '(' . implode(',', $typeName) . ')';
  145. } elseif (!is_array($json)) {
  146. throw new \Exception('jsonMethodToString json must be array or stdClass.');
  147. }
  148. if (isset($json['name']) && strpos($json['name'], '(') > 0) {
  149. return $json['name'];
  150. }
  151. $typeName = [];
  152. foreach ($json['inputs'] as $param) {
  153. if (isset($param['type'])) {
  154. $typeName[] = $param['type'];
  155. }
  156. }
  157. return $json['name'] . '(' . implode(',', $typeName) . ')';
  158. }
  159. /**
  160. * sha3
  161. * keccak256
  162. *
  163. * @param string $value
  164. * @return string
  165. */
  166. public static function sha3($value)
  167. {
  168. if (!is_string($value)) {
  169. throw new InvalidArgumentException('The value to sha3 function must be string.');
  170. }
  171. if (strpos($value, '0x') === 0) {
  172. $value = self::hexToBin($value);
  173. }
  174. $hash = Keccak::hash($value, 256);
  175. if ($hash === self::SHA3_NULL_HASH) {
  176. return null;
  177. }
  178. return '0x' . $hash;
  179. }
  180. /**
  181. * hexToBin
  182. *
  183. * @param string
  184. * @return string
  185. */
  186. public static function hexToBin($value)
  187. {
  188. if (!is_string($value)) {
  189. throw new \Exception('The value to hexToBin function must be string.');
  190. }
  191. if (self::isZeroPrefixed($value)) {
  192. $count = 1;
  193. $value = str_replace('0x', '', $value, $count);
  194. }
  195. return pack('H*', $value);
  196. }
  197. /**
  198. * isZeroPrefixed
  199. *
  200. * @param string
  201. * @return bool
  202. */
  203. public static function isZeroPrefixed($value)
  204. {
  205. if (!is_string($value)) {
  206. throw new \Exception('The value to isZeroPrefixed function must be string.');
  207. }
  208. return (strpos($value, '0x') === 0);
  209. }
  210. public static function hexToString($value){
  211. return pack("H*",$value);
  212. }
  213. }