Buffer.php 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312
  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 ArrayAccess;
  13. class Buffer implements ArrayAccess
  14. {
  15. /**
  16. * data
  17. *
  18. * @var array
  19. */
  20. protected $data = [];
  21. /**
  22. * encoding
  23. *
  24. * @var string
  25. */
  26. protected $encoding = '';
  27. /**
  28. * construct
  29. *
  30. * @param mixed $data
  31. * @param string $encoding the data encoding
  32. * @return void
  33. */
  34. public function __construct($data=[], string $encoding='utf8')
  35. {
  36. $this->encoding = strtolower($encoding);
  37. if ($data) {
  38. $this->data = $this->decodeToData($data);
  39. }
  40. }
  41. /**
  42. * offsetSet
  43. *
  44. * @param mixed $offset
  45. * @param mixed $value
  46. * @return void
  47. */
  48. public function offsetSet($offset, $value)
  49. {
  50. if (is_null($offset)) {
  51. $this->data[] = $value;
  52. } else {
  53. $this->data[$offset] = $value;
  54. }
  55. }
  56. /**
  57. * offsetExists
  58. *
  59. * @param mixed $offset
  60. * @return bool
  61. */
  62. public function offsetExists($offset)
  63. {
  64. return isset($this->data[$offset]);
  65. }
  66. /**
  67. * offsetUnset
  68. *
  69. * @param mixed $offset
  70. * @return void
  71. */
  72. public function offsetUnset($offset)
  73. {
  74. unset($this->data[$offset]);
  75. }
  76. /**
  77. * offsetGet
  78. *
  79. * @param mixed $offset
  80. * @return mixed
  81. */
  82. public function offsetGet($offset)
  83. {
  84. return isset($this->data[$offset]) ? $this->data[$offset] : null;
  85. }
  86. /**
  87. * toString
  88. *
  89. * @param string $encoding
  90. * @return string
  91. */
  92. public function toString(string $encoding='utf8')
  93. {
  94. $output = '';
  95. $input = $this->data;
  96. switch ($encoding) {
  97. case 'hex':
  98. foreach ($input as $data) {
  99. $hex = dechex($data);
  100. // pad zero
  101. if ((strlen($hex) % 2) !== 0) {
  102. $hex = '0' . $hex;
  103. }
  104. $output .= $hex;
  105. }
  106. break;
  107. case 'ascii':
  108. foreach ($input as $data) {
  109. $output .= chr($data);
  110. }
  111. break;
  112. case 'utf8':
  113. // $length = count($input);
  114. // for ($i = array_keys($input)[0]; $i < $length; $i += 3) {
  115. // $output .= chr($input[$i]) . chr($input[$i + 1]) . chr($input[$i + 2]);
  116. // }
  117. // change to bytes string
  118. foreach ($input as $data) {
  119. $output .= chr($data);
  120. }
  121. break;
  122. default:
  123. throw new InvalidArgumentException('ToString encoding must be valid.');
  124. break;
  125. }
  126. return $output;
  127. }
  128. /**
  129. * length
  130. *
  131. * @return int
  132. */
  133. public function length()
  134. {
  135. return count($this->data);
  136. }
  137. /**
  138. * concat
  139. *
  140. * @param mixed $inputs
  141. * @return \RLP\Buffer
  142. */
  143. public function concat()
  144. {
  145. $inputs = func_get_args();
  146. foreach ($inputs as $input) {
  147. if (is_array($input)) {
  148. $input = new Buffer($input);
  149. }
  150. if ($input instanceof Buffer) {
  151. $length = $input->length();
  152. for ($i = 0; $i < $length; $i++) {
  153. $this->data[] = $input[$i];
  154. }
  155. } else {
  156. throw new InvalidArgumentException('Input must be array or Buffer when call concat.');
  157. }
  158. }
  159. return $this;
  160. }
  161. /**
  162. * slice
  163. *
  164. * @param int $start
  165. * @param mixed $end
  166. * @return \RLP\Buffer
  167. */
  168. public function slice(int $start=0, $end=null)
  169. {
  170. if ($end === null) {
  171. $end = $this->length();
  172. }
  173. if ($end > 0) {
  174. $end -= $start;
  175. } elseif ($end === 0) {
  176. return new Buffer([]);
  177. }
  178. $sliced = array_slice($this->data, $start, $end);
  179. return new Buffer($sliced);
  180. }
  181. /**
  182. * decodeToData
  183. *
  184. * @param mixed $input
  185. * @return array
  186. */
  187. protected function decodeToData($input)
  188. {
  189. $output = [];
  190. if (is_array($input)) {
  191. $output = $this->arrayToData($input);
  192. } elseif (is_int($input)) {
  193. $output = $this->intToData($input);
  194. } elseif (is_numeric($input)) {
  195. $output = $this->numericToData($input);
  196. } elseif (is_string($input)) {
  197. $output = $this->stringToData($input, $this->encoding);
  198. }
  199. return $output;
  200. }
  201. /**
  202. * arrayToData
  203. *
  204. * @param array $inputs
  205. * @return array
  206. */
  207. protected function arrayToData(array $inputs)
  208. {
  209. $output = [];
  210. foreach ($inputs as $input) {
  211. if (is_array($input)) {
  212. // throw exception, maybe support future
  213. // $output[] = $this->arrayToData($input);
  214. throw new InvalidArgumentException('Do not use multidimensional array.');
  215. } elseif (is_string($input)) {
  216. $output = array_merge($output, $this->stringToData($input, $this->encoding));
  217. } elseif (is_numeric($input)) {
  218. $output = array_merge($output, $this->numericToData($input));
  219. }
  220. }
  221. return $output;
  222. }
  223. /**
  224. * stringToData
  225. *
  226. * @param string $input
  227. * @param string $encoding
  228. * @return array
  229. */
  230. protected function stringToData(string $input, string $encoding)
  231. {
  232. $output = [];
  233. switch ($encoding) {
  234. case 'hex':
  235. if (strpos($input, '0x') === 0) {
  236. // hex string
  237. $input = str_replace('0x', '', $input);
  238. }
  239. if (strlen($input) % 2 !== 0) {
  240. $input = '0' . $input;
  241. }
  242. // $splited = str_split($input, 2);
  243. // foreach ($splited as $data) {
  244. // $output[] = hexdec($data);
  245. // }
  246. $output = array_map('hexdec', str_split($input, 2));
  247. break;
  248. case 'ascii':
  249. $output = array_map('ord', str_split($input, 1));
  250. break;
  251. case 'utf8':
  252. $output = unpack('C*', $input);
  253. break;
  254. default:
  255. throw new InvalidArgumentException('StringToData encoding must be valid.');
  256. break;
  257. }
  258. return $output;
  259. }
  260. /**
  261. * numericToData
  262. *
  263. * @param mixed $intput
  264. * @return array
  265. */
  266. protected function numericToData($intput)
  267. {
  268. $output = (int) $intput;
  269. return [$output];
  270. }
  271. /**
  272. * intToData
  273. *
  274. * @param mixed $intput
  275. * @return array
  276. */
  277. protected function intToData($input)
  278. {
  279. return array_fill(0, $input, 0);
  280. }
  281. }