2

我正在尝试在 Python 中生成与 Solidity 相同的整数值 sha3.keccak_256。

以下是 Solidity 的作用:

pragma solidity ^0.4.18;

contract GenerateHash{
    function generateHashVal(int id, int date) pure public returns (bytes32){
        //Using values - (123,1522228250);
        return keccak256(id,date);
    }
}

这样生成的哈希是 0xdf4ccab87521641ffc0a552aea55be3a0c583544dc761541784ec656668f4c5a

但是在 Python3 中,我无法为整数值生成相同的值。如果我将其类型转换为字符串,那么我可以获得一些值,但这与 Solidity 的值不匹配:

>>> s=sha3.keccak_256(repr(data).encode('utf-8')).hexdigest()
>>> print(s)
37aafdecdf8b7e9da212361dfbb20d96826ae5cc912ac972f315228c0cdc51c5
>>> print(data)
1231522228250

任何帮助表示赞赏。

4

2 回答 2

1

您需要构建正确的二进制表示,它应该是两个连接在一起的 32 字节整数。可能还有其他方法可以做到这一点,但这里有一种方法可以将所有内容转换为具有正确填充的十六进制,然后用于binascii.unhexlify转换为字节序列:

sha3.keccak_256(binascii.unhexlify('{:064x}{:064x}'.format(123, 1522228250))).hexdigest()

# output: 'df4ccab87521641ffc0a552aea55be3a0c583544dc761541784ec656668f4c5a'
于 2018-03-28T18:03:27.873 回答
0

解决方案

查看web3python 库,它有许多有用的功能*。在这种情况下,我们可以使用Web3.soliditySha3(). 您传入类型和数据,它会为您生成哈希,如下所示:

from web3 import Web3

result = Web3.soliditySha3(['uint256', 'uint256'], [123, 1522228250])

assert result == "0xdf4ccab87521641ffc0a552aea55be3a0c583544dc761541784ec656668f4c5a"

Web3 v4 注意

v4 即将推出。它已经从十六进制字符串切换到bytes许多地方,包括这个函数。因此,如果您使用安装pip install --pre web3(我建议您这样做,直到 v4 稳定),则对上述内容进行轻微修改:

assert Web3.toHex(result) == "0xdf4ccab87521641ffc0a552aea55be3a0c583544dc761541784ec656668f4c5a"

其他资源

以太坊 StackExchange 是一个针对以太坊特定问题的非常活跃的站点。您可能会得到更快的响应,并找到更多已经提出的问题。例如,这里提出了一个类似的问题:Python and Solidity keccak256 function gives different results

* 免责声明:我是web3.py repo的贡献者。

于 2018-03-29T16:38:32.070 回答