使用 bitcoinjs 发送比特币交易

Posted

技术标签:

【中文标题】使用 bitcoinjs 发送比特币交易【英文标题】:Sending a bitcoin transaction using bitcoinjs 【发布时间】:2017-05-31 21:05:11 【问题描述】:

谁能解释我如何???我已经使用 bitcoinjs 设置了两个钱包。

我想从这里发送 100000 satoshis: 1G4iprWu7Q8tNbQLA8UBM2GearcnzwFrxM

到这里: 1HsrKvboax8J3X1sgsRdWybEwnUNWsDw4Y

如果这里需要它是1G4iprWu7Q8tNbQLA8UBM2GearcnzwFrxM 的最后一笔交易

我使用的代码来自 bitcoinjs.org 网站:

var tx = new bitcoin.TransactionBuilder()

// Add the input (who is paying):
// [previous transaction hash, index of the output to use]
var txId = 'aa94ab02c182214f090e99a0d57021caffd0f195a81c24602b1028b130b63e31'
tx.addInput(txId, 0)

// Add the output (who to pay to):
// [payee's address, amount in satoshis]
tx.addOutput("1Gokm82v6DmtwKEB8AiVhm82hyFSsEvBDK", 15000)

// Initialize a private key using WIF
var privateKeyWIF = 'L1uyy5qTuGrVXrmrsvHWHgVzW9kKdrp27wBC7Vs6nZDTF2BRUVwy'
var keyPair = bitcoin.ECPair.fromWIF(privateKeyWIF)

// Sign the first input with the new key
tx.sign(0, keyPair)

// Print transaction serialized as hex
console.log(tx.build().toHex())
// => 0100000001313eb630b128102b60241ca895f1d0ffca21 ...

// You could now push the transaction onto the Bitcoin network manually
// (see https://blockchain.info/pushtx)

现在我假设 var txId 是上次交易 here 的交易 ID

`tx.addInput`` 是我放置费用的地方吗?如果是的话,100个就够了吗?

tx.addOutput 是 obvs,所以我可以接受!

var privateKeyWIF* 是我把来自发送地址的私钥放在哪里了吗?

不知道var keyPairtx.sign 做了什么!

任何可以帮助告诉我细节应该去哪里的人将不胜感激!对于这个例子,假设我的发件人地址的私钥是 5Kb8kLf9zgWQnogidDA76MzPL6TsZZY36hWXMssSzNydYXYB9KF

干杯

【问题讨论】:

你读过documentation - 我假设那是你正在使用的库 我记得有一次这让我很困惑,所以我放弃了。祝你好运 是的,我已阅读文档并进行了多次尝试。我正在使用的代码生成十六进制格式的交易,但不是所需的交易,我只需要知道它的去向 【参考方案1】:

比特币交易通常将先前的交易输出作为新的交易输入。

首先,您需要查看您之前的交易。如您所见here 或在 txId 上方的 sn-p 中是: 4303caa91ca5a2236396af3bfae26b70a223a2bcee2fe8d23a7b24626c3c4bd2


   "hash":"4303caa91ca5a2236396af3bfae26b70a223a2bcee2fe8d23a7b24626c3c4bd2",
   "ver":1,
   "vin_sz":1,
   "vout_sz":2,
   "size":225,
   "weight":900,
   "fee":18713,
   "relayed_by":"0.0.0.0",
   "lock_time":448506,
   "tx_index":7399058499716239,
   "double_spend":false,
   "time":1484590465,
   "block_index":448507,
   "block_height":448507,
   "inputs":[
      
         "sequence":4294967294,
         "witness":"",
         "script":"47304402204fdab6f26efa32f49c79d8c91b5d42ea39760cff5afc34351f8595453bd3c9d102201ff8e25e4ababe3c3515617e90f97516cda50061ad0c02f215e4a008d580ede4012102e801e52ff6e7d1ad0bb46ef6732d8fcfae1e122a47ea5690e084b9fa4a73d106",
         "index":0,
         "prev_out":
            "spent":true,
            "script":"76a914d7819c081b247a146922b8f90f89181b02a3c66f88ac",
            "spending_outpoints":[
               
                  "tx_index":7399058499716239,
                  "n":0
               
            ],
            "tx_index":356328836300976,
            "value":469871307,
            "addr":"1LeVYfpp1LQQUBqsmruVW6cMbYrZtmunPr",
            "n":1,
            "type":0
         
      
   ],
   "out":[
      
         "type":0,
         "spent":true,
         "value":469352594,
         "spending_outpoints":[
            
               "tx_index":5608049158171792,
               "n":0
            
         ],
         "n":0,
         "tx_index":7399058499716239,
         "script":"76a9147adfb2779ee7710b0dffeff8fb77c82867ed43e588ac",
         "addr":"1CChR9sShAc61MNLhMoLzy87w5DwP6jRvv"
      ,
      
         "type":0,
         "spent":true,
         "value":500000,
         "spending_outpoints":[
            
               "tx_index":8253272257190039,
               "n":0
            
         ],
         "n":1,
         "tx_index":7399058499716239,
         "script":"76a914a53e0c670a2836baef1ea93de125f1fdd77370dd88ac",
         "addr":"1G4iprWu7Q8tNbQLA8UBM2GearcnzwFrxM"
      
   ]

您的地址 1G4iprWu7Q8tNbQLA8UBM2GearcnzwFrxM 显示为第二个输出 (index 1)。

这个地址的总金额是500000 satoshis


那么,让我们创建我们的交易

var txb = new bitcoin.TransactionBuilder()
txb.addInput('4303caa91ca5a2236396af3bfae26b70a223a2bcee2fe8d23a7b24626c3c4bd2', 1)

现在我们将发送 100000 聪

txb.addOutput('1cMh228HTCiwS8ZsaakH8A8wze1JR5ZsP', 100000)

我们需要将更改发送到您拥有的某个地址

txb.addOutput('1cMh228HTCiwS8ZsaakH8A8wze1JR5ZsP', 398130)

剩下的是矿工费。您可以使用online services 估算理想费用。

(in)500000 Satoshis - (out)100000 - (out)398130 = (fee)1870

现在我们需要签署交易以确认您拥有输入地址

var yourAddressPrivateKeyWIF = 'L1uyy5qTuGrVXrmrsvHWHgVzW9kKdrp27wBC7Vs6nZDTF2BRUVwy'
var yourAddresskeyPair = bitcoin.ECPair.fromWIF(yourAddressPrivateKeyWIF)
txb.sign(0, yourAddresskeyPair)

如果你运行yourAddresskeyPair.getAdress(),输出应该是你的地址1G4iprWu7Q8tNbQLA8UBM2GearcnzwFrxM

现在您可以将交易发送到网络。获取十六进制格式的原始交易:

txb.build().toHex()

在此处粘贴结果https://blockchain.info/pushtx 或在此处查看其他解决方案:https://github.com/bitcoinjs/bitcoinjs-lib/issues/839

【讨论】:

【参考方案2】:

我发现很多人对这部分感到困惑 - “如何将交易发布到网络”。简而言之,您需要访问一个单独的比特币核心完整节点。完整节点包括您将原始交易十六进制发布到的 JSON API,然后完整节点基本上在幕后为您完成其余工作(通过八卦网络广播到其他节点,然后将交易添加到自己的mempools,然后等待矿工对其进行处理,以便将其添加到区块链中。

将交易发送到网络有两种主要方式(通过 Bitcoin Core 中的 JSON RPC API 或通过 bitcoin-cli 发送):

    您可以添加一个允许您通过套接字访问网络服务器的 nodeJS 库,然后像从您的 nodeJS 代码中调用任何其他 REST 服务一样调用它。 为了测试,使用完整节点附带的“bitcoin-cli”更容易。确保您的完整节点正在运行,通过 SSH 连接到它,然后从 bitcoin-cli 中调用 sendrawtransaction 命令。具体来说,在 UNIX 提示符下,只有一行:
% bitcoin-cli sendrawtransaction 0000000001186f9f....00000000

那个大长十六进制数字是您的 bitcoinjs 代码将生成的交易十六进制数。未经确认的交易应该会在几秒钟内出现在您的钱包中。即已验证,但尚未确认。

【讨论】:

【参考方案3】:
   let bitcoin = require('bitcoinjs-lib');
   const TESTNET = bitcoin.networks.testnet;
   const keyPair = bitcoin.ECPair.makeRandom( network: TESTNET );
   const  address  = bitcoin.payments.p2pkh(
      pubkey: keyPair.publicKey,
      network: TESTNET,
   );;
   let wif = keyPair.toWIF()
// console.log(address, keyPair.publicKey.toString('hex') , keyPair.privateKey.toString('hex') , wif);

   let fkeyPair = bitcoin.ECPair.fromWIF(fwif , TESTNET);
   const result = await axios.get(`https://testnet.blockchain.info/rawaddr/$fAddress`);
   let balance = result.data.final_balance;
   let latestTx = result.data.txs[0].hash;
   console.log('testAddress balance:' , balance);
   console.log('latest tx: ', latestTx);
   var txb = new bitcoin.TransactionBuilder(bitcoin.networks.testnet);
   let sendAmount = 15000;
   let fee = 26456;
   let whatIsLeft = balance - fee - sendAmount;
   txb.addInput(latestTx, 1);
   txb.addOutput(f2Address, sendAmount);
   txb.addOutput(f2Address, whatIsLeft);
   txb.sign(0, fkeyPair);
   let body = txb.build().toHex();
   console.log(body);

【讨论】:

【参考方案4】:

如果

首先了解wif format是什么。

输入

From github 我们有:

Transaction.prototype.addInput = function (hash, index, sequence, scriptSig)

所以你必须传递交易的哈希和索引(哪个输出将是你的输入)你应该检查what a bitcoin tx is

密钥对

未验证,但可能keyPair 是您的 Pk 和 pk。

签名

还请验证因为我没有,但逻辑上tx.sign(index, keyPair) 应该使用keyPair 中的私钥对输入index 的tx (SIGHASH_ALL) 进行签名。如果您有多个输入,当然您必须为每个输入提供一个签名。看看mastering bitcoin

【讨论】:

感谢您的信息,虽然这并没有解释如何将交易推送到网络上

以上是关于使用 bitcoinjs 发送比特币交易的主要内容,如果未能解决你的问题,请参考以下文章

MyEclipse上运行比特币全节点bitcoinj

如何用分类帐签署比特币 psbt?

sh 使用BlockCypher Transaction API创建并发送比特币交易

区块链比特币学习 - 5 -创币交易

比特币的账户结构

区块链比特币学习 - 4 - 交易池