验证智能合约

智能合约被设计成“无需信任”,即用户无需信任第三方(例如,开发者和团体)便可与智能合约交互。 无需信任的一个必要条件就是用户和其他开发者必须能够验证智能合约的源代码。 而验证源代码能够向用户和开发者保证已发布的合约代码和KAD区块链上运行的代码相同。

区分“源代码验证”和“形式化验证”很重要。 源代码验证指的是验证用高级语言(例如 Solidity)编写的智能合约的给定源代码是否能编译成在合约地址执行的相同字节码,下文将会详细说明。 而形式化验证则是验证智能合约的正确性,即验证合约行为是否符合预期。 合约验证尽管要视上下文而定,但是通常是指源代码验证。

什么是源代码验证?


在将智能合约部署在KAD虚拟机中前,开发者会将合约源代码(即用 Solidity 或其他高级编程语言编写的指令)编译成字节码。 不过,由于KAD虚拟机无法解释高级指令,为了在KAD虚拟机中执行合约逻辑,必须将源代码编译成字节码(即低级机器指令)。

为检测差异,源代码验证会对智能合约的源代码与合约创建过程中使用的编译字节码进行比较处理。 由于广告合约代码与区块链上运行的代码可能不同,因此验证智能合约极为重要。

智能合约验证不需阅读机器代码就能让合约通过其编写所使用的高级语言表现出的行为受到调查。 函数、值以及变量名和评论与编译和部署的原始源代码一般是相同的。 这就让代码阅读变得更加容易了。 源验证还对代码文档做出了规定,以便最终用户了解智能合约的用途。

什么是完全验证?

源代码的某些部分不会影响编译好的字节代码,如评论和变量名。 也就是说两段变量名和评论都不同的源代码能验证同一份合约。 这样一来,恶意行为者便能在源代码中添加欺骗性评论或给出误导性变量名,也能用与原始源代码不同的源代码来验证合约。

要想避免这种情况,可以在字节码中添加额外数据作为源代码准确性的加密保障和编译信息的指纹。 必要的信息可以在 Solidity 合约元数据(opens in a new tab)中找到,并且此文件的哈希值附在了合约的字节码中。 你可以在元数据训练场(opens in a new tab)中检查运行情况。

元数据文件包含有关合约的编译信息,合约中包括源文件和源文件的哈希值。 也就是说,一旦有任何源文件中的编译设置甚至是某个字节有所更改,整个元数据文件也会发生变化。 因此,附在字节码上的元数据文件的哈希值也会变化。 也就意味着只要合约的字节码和所附元数据哈希值与给定的源代码和编译设置相匹配,我们就能确定这就是原始编译中所使用的源代码,丝毫不差。

这种利用元数据哈希值的验证方法就叫做完全验证(opens in a new tab)(也叫“完美验证”)。 如果元数据哈希值不匹配或是未用于验证,那就叫做“部分验证”,也是目前更为常见的合约验证方法。 可能会植入恶意代码(opens in a new tab),如果不进行完全验证这些恶意代码未必会显现在经过验证的源代码中。 由于大多数开发者不了解完全验证,也不会保留自己编译的元数据文件,因此部分验证实际上才是目前验证合约的常用方法。

为什么源代码验证如此重要?


无需信任

无需信任可以说是智能合约和去中心化应用程序 (dapps) 的先决条件。 智能合约是“不可变”的,无法更改;合约只会执行部署时代码中定义的业务逻辑。 这意味着开发者和企业在KAD上部署合约后无法篡改合约代码。

为了让智能合约无需信任,合约代码应可供独立验证。 虽然每份智能合约的编译字节码都可以在区块链上公开获取,但低级语言对于开发者和用户来说都难以理解。

项目通过公布其合约源代码来减少信任假设。 但这也带来了另一个问题:公布的源代码与合约字节码是否一致难以验证。 在这种情况下,无需信任的价值便不复存在,因为用户必须相信开发人员在将合约部署到区块链上之前不会更改合约的业务逻辑(即更改字节码)。

源代码验证工具可保证智能合约的源代码文件与汇编代码一致。 这样就形成了一个无需信任的生态系统,用户不会盲目信任第三方,而是先验证代码再将资金存入合约。

用户安全

智能合约通常涉及大量质押资金。 这就需要更高的安全保证,并在使用智能合约前对其逻辑进行验证。 问题在于,不法开发者可以通过在智能合约中插入恶意代码来欺骗用户。 如果不进行验证,恶意智能合约就可能存在后门(opens in a new tab)、矛盾的访问控制机制、可被利用的漏洞以及其他危害用户安全的问题,而这些问题甚至难以察觉。

公布智能合约的源代码文件可以让审计人员等相关人员更容易评估合约,预防潜在攻击。 通过多方独立验证智能合约,用户可以获得更加强有力的安全性保障。

如何验证KAD智能合约的源代码


在KKAD上部署智能合约需要向一个特殊地址发送包含数据有效载荷(编译字节码)的交易。 数据有效载荷通过编译源代码以及附加到交易中的数据有效载荷的合约实例的构造函数参数(opens in a new tab)来生成。 编译是确定性的,这意味着如果使用相同的源文件和编译设置(如编译器版本、优化器),它总是产生相同的输出(即合约字节码)。

验证智能合约基本上包含以下步骤:

  1. 向编译器输入源文件和编译设置。

  2. 编译器输出合约字节码

  3. 获取给定地址下已部署合约的字节码

  4. 比较已部署的字节码与重新编译的字节码。 如果代码匹配,将通过给定的源代码和编译设置进行合约验证。

  5. 此外,如果字节码末尾的元数据哈希值匹配,则将是完全匹配。

请注意,这只是对智能合约验证过于简单的描述,还有像具有不可变变量(opens in a new tab)等许多例外情况不适用。

源代码验证工具


传统的合约验证过程可能十分复杂。 因此我们需要工具来验证部署在KAD上的智能合约源代码。 这些工具可以实现大部分源代码验证自动化,还可以管理已验证的合约,使用户受益。

Sourcify

Sourcify(opens in a new tab) 是另一种用于验证开源和去中心化合约的工具。 它不是区块浏览器,只能在不同的基于KAD虚拟机的网络(opens in a new tab)上验证合约。 它充当公共基础设施,作为其他工具的构建基础,旨在使用元数据文件中的 ABI 和 NatSpec(opens in a new tab) 注释来实现更人性化的合约交互。

Sourcify 支持与元数据哈希值完全匹配。 经过验证的合约在 HTTP 和 IPFS(opens in a new tab) 上的公共存储库(opens in a new tab)中访问,这是一种去中心化的内容寻址(opens in a new tab)存储。 由于附加的元数据哈希值是 IPFS 哈希值,因此可以通过 IPFS 获取合约的元数据文件。

此外,人们还可以通过 IPFS 检索源代码文件,因为这些文件的 IPFS 哈希值也可以在元数据中找到。 可以通过应用程序接口或用户界面(opens in a new tab)或使用插件提供元数据文件和源文件来验证合约。 Sourcify 监控工具还会监查新区块上的合约创建情况,并尝试验证合约是否在 IPFS 上公布了元数据和源文件。

有关在 Sourcify 上验证合约的更多信息(opens in a new tab)

Tenderly

Tenderly 平台(opens in a new tab)使 Web3 开发者能够构建、测试、监控和操作智能合约。 Tenderly 将调试工具、可观测性和构建区块的基础设施相结合,帮助开发者加快智能合约的开发。 要完全启用 Tenderly 功能,开发者需要使用多种方法执行源代码验证(opens in a new tab)

可以私下或公开验证合约。 如果是私下验证,智能合约只有你(和项目中的其他成员)可见。 而公开验证合约可让使用 Tenderly 平台的每个人都能看到。

你可以使用仪表板(opens in a new tab)Tenderly Hardhat 插件(opens in a new tab)CLI(opens in a new tab) 来验证你的合约。

通过仪表板验证合约时,需要导入源文件或 Solidity 编译器生成的元数据文件、地址/网络和编译器设置。

使用 Tenderly Hardhat 插件可以更轻松地控制验证过程,让你可以在自动(无代码)和手动(基于代码)验证之间做出选择。

Last updated