RLP.php 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. <?php
  2. /**
  3. * This file is part of rlp package.
  4. *
  5. * (c) Kuan-Cheng,Lai <alk03073135@gmail.com>
  6. *
  7. * @author Peter Lai <alk03073135@gmail.com>
  8. * @license MIT
  9. */
  10. namespace App\Services\Web3\RLP;
  11. use InvalidArgumentException;
  12. use RuntimeException;
  13. class RLP
  14. {
  15. /**
  16. * encode
  17. *
  18. * @param mixed $inputs array of string
  19. * @return Buffer
  20. */
  21. public function encode($inputs): Buffer
  22. {
  23. $output = new Buffer;
  24. if (is_array($inputs)) {
  25. $result = new Buffer;
  26. foreach ($inputs as $input) {
  27. $output->concat($this->encode($input));
  28. }
  29. return $result->concat($this->encodeLength($output->length(), 192), $output);
  30. }
  31. $input = $this->toBuffer($inputs);
  32. $length = $input->length();
  33. if ($length === 1 && $input[0] < 128) {
  34. return $input;
  35. } else {
  36. return $output->concat($this->encodeLength($length, 128), $input);
  37. }
  38. }
  39. /**
  40. * decode
  41. * Maybe use bignumber future.
  42. *
  43. * @param string $input
  44. * @return array
  45. */
  46. public function decode(string $input): array
  47. {
  48. // if (!is_string($input)) {
  49. // throw new InvalidArgumentException('Input must be string when call decode.');
  50. // }
  51. $input = $this->toBuffer($input);
  52. $decoded = $this->decodeData($input);
  53. return $decoded['data'];
  54. }
  55. /**
  56. * decodeData
  57. * Maybe use bignumber future.
  58. *
  59. * @param Buffer $input
  60. * @return array
  61. */
  62. protected function decodeData(Buffer $input): array
  63. {
  64. $firstByte = $input[0];
  65. $output = new Buffer;
  66. if ($firstByte <= 0x7f) {
  67. return [
  68. 'data' => $input->slice(0, 1),
  69. 'remainder' => $input->slice(1)
  70. ];
  71. } elseif ($firstByte <= 0xb7) {
  72. $length = $firstByte - 0x7f;
  73. $data = new Buffer([]);
  74. if ($firstByte !== 0x80) {
  75. // for ($i = 1; $i < $length; $i++) {
  76. // $data[] = $input[$i];
  77. // }
  78. $data = $input->slice(1, $length);
  79. }
  80. if ($length === 2 && $data[0] < 0x80) {
  81. throw new RuntimeException('Byte must be less than 0x80.');
  82. }
  83. return [
  84. 'data' => $data,
  85. 'remainder' => $input->slice($length)
  86. ];
  87. } elseif ($firstByte <= 0xbf) {
  88. $llength = $firstByte - 0xb6;
  89. $hexLength = $input->slice(1, $llength)->toString('hex');
  90. if ($hexLength === '00') {
  91. throw new RuntimeException('Invalid RLP.');
  92. }
  93. $length = hexdec($hexLength);
  94. $data = $input->slice($llength, $length + $llength);
  95. if ($data->length() < $length) {
  96. throw new RuntimeException('Invalid RLP.');
  97. }
  98. return [
  99. 'data' => $data,
  100. 'remainder' => $input->slice($length + $llength)
  101. ];
  102. } elseif ($firstByte <= 0xf7) {
  103. $length = $firstByte - 0xbf;
  104. $innerRemainder = $input->slice(1, $length);
  105. $decoded = [];
  106. while ($innerRemainder->length()) {
  107. $data = $this->decodeData($innerRemainder);
  108. $decoded[] = $data['data'];
  109. $innerRemainder = $data['remainder'];
  110. }
  111. return [
  112. 'data' => $decoded,
  113. 'remainder' => $input->slice($length)
  114. ];
  115. } else {
  116. $llength = $firstByte - 0xf6;
  117. $hexLength = $input->slice(1, $llength)->toString('hex');
  118. $decoded = [];
  119. if ($hexLength === '00') {
  120. throw new RuntimeException('Invalid RLP.');
  121. }
  122. $length = hexdec($hexLength);
  123. $totalLength = $llength + $length;
  124. if ($totalLength > $input->length()) {
  125. throw new RuntimeException('Invalid RLP: total length is bigger than data length.');
  126. }
  127. $innerRemainder = $input->slice($llength, $totalLength);
  128. if ($innerRemainder->length() === 0) {
  129. throw new RuntimeException('Invalid RLP: list has invalid length.');
  130. }
  131. while ($innerRemainder->length()) {
  132. $data = $this->decodeData($innerRemainder);
  133. $decoded[] = $data['data'];
  134. $innerRemainder = $data['remainder'];
  135. }
  136. return [
  137. 'data' => $decoded,
  138. 'remainder' => $input->slice($length)
  139. ];
  140. }
  141. }
  142. /**
  143. * encodeLength
  144. *
  145. * @param int $length
  146. * @param int $offset
  147. * @return Buffer
  148. */
  149. protected function encodeLength(int $length, int $offset): Buffer
  150. {
  151. // if (!is_int($length) || !is_int($offset)) {
  152. // throw new InvalidArgumentException('Length and offset must be int when call encodeLength.');
  153. // }
  154. if ($length < 56) {
  155. // var_dump($length, $offset);
  156. return new Buffer(strval($length + $offset));
  157. }
  158. $hexLength = $this->intToHex($length);
  159. $firstByte = $this->intToHex($offset + 55 + (strlen($hexLength) / 2));
  160. return new Buffer(strval($firstByte . $hexLength), 'hex');
  161. }
  162. /**
  163. * intToHex
  164. *
  165. * @param int $value
  166. * @return string
  167. */
  168. protected function intToHex(int $value): string
  169. {
  170. // if (!is_int($value)) {
  171. // throw new InvalidArgumentException('Value must be int when call intToHex.');
  172. // }
  173. $hex = dechex($value);
  174. return $this->padToEven($hex);
  175. }
  176. /**
  177. * padToEven
  178. *
  179. * @param string $value
  180. * @return string
  181. */
  182. protected function padToEven(string $value): string
  183. {
  184. // if (!is_string($value)) {
  185. // throw new InvalidArgumentException('Value must be string when call padToEven.');
  186. // }
  187. if ((strlen($value) % 2) !== 0 ) {
  188. $value = '0' . $value;
  189. }
  190. return $value;
  191. }
  192. /**
  193. * toArray
  194. * Format input to value, deprecated when we have toBuffer.
  195. *
  196. * @param mixed $input
  197. * @return array
  198. */
  199. protected function toArray($input)
  200. {
  201. if (is_string($input)) {
  202. if (strpos($input, '0x') === 0) {
  203. // hex string
  204. $value = str_replace('0x', '', $input);
  205. return $input;
  206. } else {
  207. return str_split($input, 1);
  208. }
  209. }
  210. throw new InvalidArgumentException('The input type didn\'t support.');
  211. }
  212. /**
  213. * toBuffer
  214. * Format input to buffer.
  215. *
  216. * @param mixed $input
  217. * @return Buffer
  218. */
  219. protected function toBuffer($input): Buffer
  220. {
  221. if (is_string($input)) {
  222. if (strpos($input, '0x') === 0) {
  223. // hex string
  224. // $input = str_replace('0x', '', $input);
  225. return new Buffer($input, 'hex');
  226. }
  227. return new Buffer(str_split($input, 1));
  228. } elseif (is_numeric($input)) {
  229. if (!$input || $input < 0) {
  230. return new Buffer([]);
  231. }
  232. if (is_float($input)) {
  233. $input = number_format($input, 0, '', '');
  234. var_dump($input);
  235. }
  236. $gmpInput = gmp_init($input, 10);
  237. return new Buffer('0x' . gmp_strval($gmpInput, 16), 'hex');
  238. } elseif ($input === null) {
  239. return new Buffer([]);
  240. } elseif (is_array($input)) {
  241. return new Buffer($input);
  242. } elseif ($input instanceof Buffer) {
  243. return $input;
  244. }
  245. throw new InvalidArgumentException('The input type didn\'t support.');
  246. }
  247. }