Wallet.php 1.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869
  1. <?php
  2. namespace App\Services\Web3;
  3. use Elliptic\EC;
  4. use kornrunner\Keccak;
  5. use Web3\RLP\RLP;
  6. class Wallet
  7. {
  8. private $address;
  9. private $privateKey;
  10. private function __construct($address, $privateKey)
  11. {
  12. $this->address = $address;
  13. $this->privateKey = $privateKey;
  14. }
  15. static public function create(): Wallet
  16. {
  17. $ec = new EC('secp256k1');
  18. $kp = $ec->genKeyPair();
  19. $privateKey = $kp->getPrivate('hex');
  20. $pub = $kp->getPublic('hex');
  21. $address = Utils::pubKeyToAddress($pub);
  22. return new Wallet($address, $privateKey);
  23. }
  24. static public function createByPrivate($privateKey): Wallet
  25. {
  26. $ec = new EC('secp256k1');
  27. // Generate keys
  28. $key = $ec->keyFromPrivate($privateKey);
  29. $pub = $key->getPublic('hex');
  30. // get address based on public key
  31. return new Wallet(strtolower(Utils::pubKeyToAddress($pub)), $privateKey);
  32. }
  33. public function getAddress()
  34. {
  35. return $this->address;
  36. }
  37. public function getPrivateKey()
  38. {
  39. return Utils::add0x($this->privateKey);
  40. }
  41. /**
  42. * @throws \Exception
  43. */
  44. public function sign($data)
  45. {
  46. if (empty($this->privateKey)) {
  47. throw new \Exception("please unlock this address");
  48. }
  49. $hash = Keccak::hash(hex2bin($data), 256);
  50. $ec = new EC('secp256k1');
  51. // Generate keys
  52. $key = $ec->keyFromPrivate($this->privateKey);
  53. // Sign message (can be hex sequence or array)
  54. return $key->sign($hash, ['canonical' => true]);
  55. }
  56. }