本文将提供有关如何生成ECDSA私钥的指南,然后使用PHP7.0++导出到以太坊钱包地址。
你可以找到以下工作实现:
composer.json
{
"require": {
"sop/asn1": "^3.3",
"sop/crypto-encoding": "^0.2.0",
"sop/crypto-types": "^0.2.1",
"kornrunner/keccak": "^1.0",
"symfony/dotenv": "^4.0",
"sc0vu/web3.php": "dev-master"
}
}
GenerateEthereumWallet.php
require_once "vendor/autoload.php";
use Sop\CryptoTypes\Asymmetric\EC\ECPublicKey;
use Sop\CryptoTypes\Asymmetric\EC\ECPrivateKey;
use Sop\CryptoEncoding\PEM;
use kornrunner\keccak;
$config = [
'private_key_type' => OPENSSL_KEYTYPE_EC,
'curve_name' => 'secp256k1'
];
$res = openssl_pkey_new($config);
if (!$res) {
echo 'ERROR: Fail to generate private key. -> ' . openssl_error_string();
exit;
}
openssl_pkey_export($res, $priv_key);
$key_detail = openssl_pkey_get_details($res);
$pub_key = $key_detail["key"];
$priv_pem = PEM::fromString($priv_key);
$ec_priv_key = ECPrivateKey::fromPEM($priv_pem);
$ec_priv_seq = $ec_priv_key->toASN1();
$priv_key_hex = bin2hex($ec_priv_seq->at(1)->asOctetString()->string());
$priv_key_len = strlen($priv_key_hex) / 2;
$pub_key_hex = bin2hex($ec_priv_seq->at(3)->asTagged()->asExplicit()->asBitString()->string());
$pub_key_len = strlen($pub_key_hex) / 2;
$pub_key_hex_2 = substr($pub_key_hex, 2);
$pub_key_len_2 = strlen($pub_key_hex_2) / 2;
$hash = Keccak::hash(hex2bin($pub_key_hex_2), 256);
$wallet_address = '0x' . substr($hash, -40);
$wallet_private_key = '0x' . $priv_key_hex;
echo "\r\n ETH Wallet Address: " . $wallet_address;
echo "\r\n Private Key: " . $wallet_private_key;
代码条件
该代码需要PHP 7.0++,OpenSSL扩展和PHP Composer。需要使用PHP Composer来安装第三方软件包。
$ composer install
生成私钥
以太坊标准是使用secp256k1曲线生成私钥。在我的教程中,我使用OpenSSL函数生成PEM格式的椭圆曲线私钥,如下所示:
$config = [
'private_key_type' => OPENSSL_KEYTYPE_EC,
'curve_name' => 'secp256k1'
];
$res = openssl_pkey_new($config);
if (!$res) {
echo 'ERROR: Fail to generate private key. -> ' . openssl_error_string();
exit;
}
openssl_pkey_export($res, $priv_key);
$priv_pem = PEM::fromString($priv_key);
之后我需要将私钥转换为ASN1序列,下面是ANS1序列的结构。
ECPrivateKey ::= SEQUENCE (ecPrivkeyVer1),
privateKey OCTET STRING,
parameters [0] ECParameters } OPTIONAL,
publicKey [1] BIT STRING OPTIONAL
}
下面的代码是我如何从ANS1序列结构中查询十六进制字符串中的公钥和私钥。
$ec_priv_seq = $ec_priv_key->toASN1();
$priv_key_hex = bin2hex($ec_priv_seq->at(1)->asOctetString()->string());
$priv_key_len = strlen($priv_key_hex) / 2;
$pub_key_hex = bin2hex($ec_priv_seq->at(3)->asTagged()->asExplicit()->asBitString()->string());
$pub_key_len = strlen($pub_key_hex) / 2;
衍生以太坊钱包地址
以太坊钱包地址来自公钥。每个EC公钥始终以0x04开头。为了获得以太坊钱包地址的正确哈希值,我们需要删除前导0x04。
$pub_key_hex_2 = substr($pub_key_hex, 2);
$pub_key_len_2 = strlen($pub_key_hex_2) / 2;
我们继续使用EC公钥的Keccak256哈希。 以太坊钱包地址长度为20个字节,长度为40个字符,因此我们只需要哈希数据的最后20个字节。
$hash