专栏名称: 区块链技术学习
致力于区块链技术的学习和普及,对区块链技术和相关企业事件进行深度分析和研判,探索去中心化账本技术应用领域。
目录
相关文章推荐
51好读  ›  专栏  ›  区块链技术学习

使用PHP生成以太坊钱包和密钥对

区块链技术学习  · 公众号  ·  · 2018-11-05 21:44

正文

来源:汇智网

链接:http://blog.hubwiz.com/2018/10/27/ethereum-php-wallet-keypairs/

本文将提供有关如何生成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);
// 然后将其转换为ASN1结构
$ec_priv_seq = $ec_priv_key->toASN1();
// HEX中的私钥和公钥
$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才能正确hash它
$pub_key_hex_2 = substr($pub_key_hex, 2);
$pub_key_len_2 = strlen($pub_key_hex_2) / 2;
// Hash
$hash = Keccak::hash(hex2bin($pub_key_hex_2), 256);
// 以太坊地址长度为20个字节。 (40个十六进制字符长)
// 我们只需要最后20个字节作为以太坊地址
$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;
}
// Generate Private Key
openssl_pkey_export($res, $priv_key);
// PEM Format
$priv_pem = PEM::fromString($priv_key);

之后我需要将私钥转换为ASN1序列,下面是ANS1序列的结构。

ECPrivateKey ::= SEQUENCE {
    version        INTEGER { ecPrivkeyVer1(1) }
 (ecPrivkeyVer1),
    privateKey     OCTET STRING,
    parameters [0] ECParameters {{ NamedCurve }} OPTIONAL,
    publicKey  [1] BIT STRING OPTIONAL
}

下面的代码是我如何从ANS1序列结构中查询十六进制字符串中的公钥和私钥。

// Then convert it to ASN1 Structure
$ec_priv_seq = $ec_priv_key->toASN1();
// Private Key & Public Key in HEX
$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






请到「今天看啥」查看全文