以太坊交易哈希:如何提前获得?

时间:2018-01-12 20:40:23

标签: hash transactions ethereum

请告诉我如何获得交易哈希?

// I have these tx opts:
var txOpts = {
  "to":"0x345cA3e014Aaf5dcA488057592ee47305D9B3e10",
  "nonce":"0x8",
  "gasPrice":1,
  "gas":250000,
  "value":"0xde0b6b3a7640000",
  "data":"0xd0e30db0"
}

// I create and sign tx:
var tx = new ethereumjs.Tx(txOpts);
tx.sign(new ethereumjs.Buffer.Buffer("c87509a1c067bbde78beb793e6fa76530b6382a4c0241e5e4a9ec0a0f44dc0d3", "hex"));

// I've got raw tx:
var rawTx = tx.serialize().toString('hex');
"f86c08018303d09094345ca3e014aaf5dca488057592ee47305d9b3e10880de0b6b3a764000084d0e30db01ca0625e358100f4aacb9a65e6e054d963138565e3ceafb20eae4c9c8aaa583a29eea01d8f74faba33ab577ec36ac383dd5bd5298216bcf69fe2c09bba2d3003ecd008"

当我将此tx发送到ganache-cli时,我会收到此日志:

eth_sendRawTransaction
   > {
   >   "jsonrpc": "2.0",
   >   "id": 7,
   >   "method": "eth_sendRawTransaction",
   >   "params": [
   >     "0xf86c08018303d09094345ca3e014aaf5dca488057592ee47305d9b3e10880de0b6b3a764000084d0e30db01ca0625e358100f4aacb9a65e6e054d963138565e3ceafb20eae4c9c8aaa583a29eea01d8f74faba33ab577ec36ac383dd5bd5298216bcf69fe2c09bba2d3003ecd008"
   >   ],
   >   "external": true
   > }

  Transaction: 0x73d419e3a9a63aab7ee4f3be43c0df0175f4395615945d395c827bffb3bbfecc
  Gas usage: 29634
  Block Number: 9
  Block Time: Fri Jan 12 2018 23:21:22 GMT-0700

事务哈希是0x73d419e3a9a63aab7ee4f3be43c0df0175f4395615945d395c827bffb3bbfecc 如何提前获得?

// use ethereumjs.hash function (is that rlp-hash?):
var hash = tx.hash().toString('hex')
"**71ef26c4c1c1b01a5f87525e8e9b3ca7ffe5c9ae30ee1e70b353bf9b14db96be**"

这不等于73d419e3a9a63aab7ee4f3be43c0df0175f4395615945d395c827bffb3bbfecc!

在挖掘之前是否有可能提前计算CORRECT事务哈希?请告诉我应该使用哪些功能。

非常感谢!

4 个答案:

答案 0 :(得分:1)

这只是ganache的一个问题。 queueRawTransaction使用FakeTransaction,它忽略了您的交易中的实际签名,而是伪造它。我不完全确定它为什么会这样工作,但我相信你的代码会为提交给实际以太坊网络的交易做正确的事。

如果你想获得相同的事务哈希ganache是计算,这可以解决问题,但它不是你从以太坊本身得到的事务哈希:

const EthereumTx = require('ethereumjs-tx');
const FakeTx = require('ethereumjs-tx/fake.js');

var txOpts = {
  "to":"0x345cA3e014Aaf5dcA488057592ee47305D9B3e10",
  "nonce":"0x8",
  "gasPrice":1,
  "gas":250000,
  "value":"0xde0b6b3a7640000",
  "data":"0xd0e30db0"
}

var tx = new EthereumTx(txOpts);
tx.sign(Buffer.from("c87509a1c067bbde78beb793e6fa76530b6382a4c0241e5e4a9ec0a0f44dc0d3", "hex"));

const raw = tx.serialize().toString('hex');
const fake = new FakeTx(raw);
fake.from = tx.getSenderAddress();

console.log(fake.hash(true).toString('hex'));

// Output:
// 71ef26c4c1c1b01a5f87525e8e9b3ca7ffe5c9ae30ee1e70b353bf9b14db96be

答案 1 :(得分:1)

我们从ganache得到的实际tx哈希是:

  

0x73d419e3a9a63aab7ee4f3be43c0df0175f4395615945d395c827bffb3bbfecc

FakeTx和ethereumjs-tx都为签名的事务返回相同的哈希值:

  

71ef26c4c1c1b01a5f87525e8e9b3ca7ffe5c9ae30ee1e70b353bf9b14db96be

但是如果thransaction没有签名“tx.hash()。toString('hex')”返回与ganache cli完全相同的值。即使没有将'from'添加到txOpts。

答案 2 :(得分:1)

不可能预先获得交易哈希,只能在执行交易后获得交易哈希。

但是,每当任何矿工不进行交易时,此交易哈希值都是临时的,那么很有可能,如果您发送的交易费用非常少,那么任何矿工都可能不会选择您的交易。

答案 3 :(得分:0)

我挖了一点点来找到答案。

您不仅需要调用tx.hash().toString('hex'),还需要像这样tx.hash(true).toString('hex')进行调用,其中true代表includeSignature,用于计算哈希值。

如果您好奇的话,这就是方法(https://github.com/ethereumjs/ethereumjs-tx/blob/9a6324f64f4da1cb550a3eec4eaef95da4ab441b/src/transaction.ts#L177)。

您也可以像这样从rawTx获得相同的结果:

import * as ethUtil from 'ethereumjs-util';
const rawTx = `0x${signedTx.serialize().toString('hex')}`;    // 0x is important
const res1 = `0x${ethUtil.keccak(rawTx).toString('hex')}`;
const res2 = `0x${signedTx.hash(true).toString('hex')}`;
res1 === res2; // true

我希望这会有所帮助。