美文网首页
nodejs调用bitcoin转账功能

nodejs调用bitcoin转账功能

作者: 那个大螺丝 | 来源:发表于2018-12-14 01:01 被阅读0次

    nodejs与btc网络交互使用bitcoinjs-lib模组
    使用BTC测试网进行测试

    安装模组

    npm i bitcoinjs-lib
    

    根据随机字符构造两个函数

    function rng () { return Buffer.from('tmd123zzzzzzzzzzzzzzzzzzzzzzzzzz') }
    function rng2 () { return Buffer.from('tmd321333zzzzzzzzzzzzzzzzzzzzzzz') }
    

    生成钱包1, 指定测试链

    const alice = bitcoin.ECPair.makeRandom({network: testnet, rng});
    const { address } = bitcoin.payments.p2pkh({ pubkey: alice.publicKey, network: testnet });
    // console.log(address);
    // console.log(alice.toWIF())
    // 地址: mmH6e8tfLyvrrnFF3o1scaNsPXShGY89rb
    // 私钥: L182iNrxy9rSPQfEg5X1P1sR9qAbS1pbxPosguD8Sx5tYauV3bYm
    

    生成钱包2

    const bob = bitcoin.ECPair.makeRandom({network: testnet, rng: rng2 });
    const { address:address2 } = bitcoin.payments.p2pkh({ pubkey: bob.publicKey, network: testnet });
    // console.log(address2);
    // console.log(bob.toWIF())
    // 地址: miAMpCdoM3SuRMRoEVHp8smFdDAz29WA9g
    // 私钥: L182iNub4Z8Ly6H13jcx7i82jKq3czc7WzFso3z249LuMcMjzQXo
    
    • 比特币转账需要提取或合并所有未花费的交易中的比特币,才能实现交易。
    • 描述起来比较复杂,下图是个例子,我们从区块浏览器查到我最近一笔未花费支出,某个地址转了0.03个BTC给我,在这一笔交易中,我有0.03个BTC。
    • 如果我要转出0.5个BTC给他人,我需要合并其他的交易,否则将会余额不足,虽然在我这个地址中有足够的BTC。


      1.png
      2.png
    单个哈希交易。
    const transfer1 = async () => {
    
      // 注意要指定交易对象是测试链的
      const txb = new bitcoin.TransactionBuilder(testnet);
      txb.setVersion(1);
      // 在这个交易中, bob在第0个位置,上图所示
      txb.addInput('5799a647d6b89a9f73122d75faee6f5a0210bd3cb22c48a70d35eac33ce5d426', 0);
      
      // 这里把btc转给 alice 的地址,金额是0.02 但是要*100000000, 也就是2000000,
      // 剩余的金额没有设置招零地址接收,则被视为手续费,被区块网络收取
      // 每一笔交易只有已花费和未花费两种状态,不存在消费一部分的状态,
      // 所以合并多笔交易的话,只要未花费,都可以合并。
      txb.addOutput('mmH6e8tfLyvrrnFF3o1scaNsPXShGY89rb', 2000000);
    
      // 签名交易,0代表索引,输入排序,这里只有一个输入,所以是第0位。
      txb.sign(0, bob);
    
      // 序列化成一串字符
      const tx = txb.build().toHex();
      console.log(tx);
    
      // 在一个测试链的节点把交易广布出去
      const result = await fetch('https://api.blockcypher.com/v1/btc/test3/txs/push',{
        method:'post',
        headers:{'Content-Type':'application/json'},
        body:JSON.stringify({tx})
      });
    
      // 打印结果
      console.log(result);
    };
    
    transfer1();
    
    多笔交易,自动合并。
    const transfer2 = async () => {
      const url = 'https://api.blockcypher.com/v1/btc/test3/addrs/';
      const res = await fetch(url+address2);
      const json = await res.json();
      const balance = json.balance;
      // console.log(balance/100000000);
    
      const txrefs = json.txrefs;
    
      // 过滤掉已经被花费了的交易,以及自己不在接收列表的交易
      const unspentList = txrefs.filter(item=> !item.spent_by && item.tx_output_n !== -1);
      // 这个地址还可以查询余额
      // console.log(unspentList);
    
      // 构建交易对象
      const txb = new bitcoin.TransactionBuilder(testnet);
      txb.setVersion(1);
    
      // 批量插入未花费交易
      unspentList.forEach(item=>txb.addInput(item.tx_hash, item.tx_output_n));
    
      // 转出账户
      txb.addOutput('mmH6e8tfLyvrrnFF3o1scaNsPXShGY89rb', 2000000);
    
      // 设置找零地址,如果忘记了,就会丢失所有BTC !!!!!!!!
      // 如果不预留手续费,交易可能会一直不被打包!
     // 3000 为手续费
      txb.addOutput('miAMpCdoM3SuRMRoEVHp8smFdDAz29WA9g',balance - 2000000 - 3000);
      
      // 批量签名,根据索引即可
      unspentList.forEach((item,index)=>{txb.sign(index, bob)});
    
      // 序列化交易
      const tx = txb.build().toHex();
      // console.log(tx);
    
      // 在一个测试链的节点把交易广布出去
      const result = await fetch('https://api.blockcypher.com/v1/btc/test3/txs/push',{
        method:'post',
        headers:{'Content-Type':'application/json'},
        body:JSON.stringify({tx})
      });
    
      // 打印结果
      console.log(result);
    
    
    };
    
    transfer2();
    

    全部代码

    相关文章

      网友评论

          本文标题:nodejs调用bitcoin转账功能

          本文链接:https://www.haomeiwen.com/subject/izoihqtx.html