之前我们说过NEO智能合约的调试问题,过去了一段时间,有很多东西都发生了比较大的变化。让我们重新再来探讨一下这个话题。

先说日期,2018年4月27日,马上迎来劳动节。

 

以后可能还会再次谈论这个话题。

 

然后明确几个问题,把一些经常纠结的地方先说清楚

  1. 如果你连neo同步区块还没搞明白,先不要凑智能合约开发这个热闹。
  2. 我们只讨论C#开发智能合约的调试问题,其它的暂无工具开发计划。
  3. NEO官方没有智能合约调试工具,所以我们的工具是有很多定制的成分。
    1. 为什么要调试

一次编码,按下f5,一切正常,挥一挥衣袖就潇洒的离开,那是传说。

当代码执行不符合预期时,我们需要深入分析问题,需要更精细的观测手段。

 

而NEO官方并没有提供这样的工具,于是我们开发了一揽子工具用于解决调试问题

他们是:

  1. NeonDebug 替换的c#智能合约编译器
  2. NeoGUI-NEL 定制版NeoGUI 或者 NeoCli-NEl 定制版的NeoCli
  3. NeonDebugGUI,智能合约执行情况观测工具

 

为了3的正常运行,1 2 都是必须的,如果你想知道为什么,源码可以告诉你答案

 

https://github.com/NewEconoLab/neondebug

https://github.com/NewEconoLab/neo-gui-nel

 

  1. 合约的编译

你可曾听过NeoVM,AVM文件,就是智能合约编译生成的那东西,就是NeoVM的字节码,叫AVM是因为以前叫小蚁,ant的A,这个扩展名没一起改。

 

为了方便不那么了解虚拟机的开发者理解,我和C#的类库机制做一个对比。

 

这个AVM相当于是DLL,要调试一个c# DLL,你需要DLL PDB cs源码三部分内容。DLL存储字节码,PDB存储字节码地址和源码位置的映射,这样才能在断点时根据字节码执行位置定位到源码位置,而cs源码,就是为了让你调试时对照到代码

 

那么:

DLL-》AVM

PDB-》?

cs-》cs

 

标准的c#编译器neon.exe(如果不知道neon.exe,再好好补一下C#智能合约编译的相关资料),只生成了一个AVM和一个ABI文件,这个ABI是接口说明。

PDB对应的角色是空缺的

 

所以你需要 NEL 的 Neon.exe ,编译neondebug仓库的neondebug项目可以得到这个替换的neon.exe

 

设置环境变量,保证在任何路径可以执行neon

注意红框部分,有Debug字样

 

这个替换的编译器,就专门解决那个缺失的PDB问题,这个编译器和官方编译器代码随时同步更新,你直接使用这个替换的编译器就行,任何更新都会同步,不需要切换回官方编译器。

 

重启VS2017之后,来编译一个智能合约试试

注意红框部分,已经使用了我们的替换版本编译 有(debug)字样

下一行告诉你,在c:\neo\smartcontracts输出了些什么,让我们来看看。

首先原来的avm照样输出,这些是另外多输出的

 

文件名全部替换为了合约hash,avm还是那个avm

Abi.json还是那个abi

Cs是把智能合约源码copy 过来的,只支持一个智能合约项目有一个代码文件,所以不要分文件。

Map就是替代PDB的那个了,他记录字节码地址和源代码地址的映射关系

 

生成在这个路径也是有意义的,调试查看工具会直接来这个路径找合约,以前需要用调试查看工具再打开一次合约,现在不用了,直接用同一个路径。

 

  1. 合约的调用

A,用neogui 测试执行合约

    用neogui-nel,测试执行一个合约,只有nel定制版支持。

    

试运行固定产生一个0x00 的log文件,看这个log就可以调试了,这个下一节再说,我们这里之说产生log文件的办法。

 

产生这个log是调试的唯一办法。

 

B 、用NEOCLI 测试执行合约

调用NEOCLI -NEL的 rpc invokescript 接口,会在返回值多返回一个fulllog字段,将其保存为文件,就是这个llvmhex文件,可以用来调试。

只有nel定制版支持

C、用NEOGUI发交易

用NEL定制版NEOGUI发起交易,只有nel定制版支持

 

 

发起交易,

会得到交易id同名的llvm文件

Id[0]

Id[1]

这种为鉴权合约的log

只有通过neogui nel定制版发起交易 或者 neocli nel定制版的 rpc接口 sendraw发起交易

才会产生鉴权合约的log,才可以调试鉴权合约,其他只调试应用合约

D、通过NEOCLI的sendraw RPC 发起交易

只有nel定制版支持

会得到交易id同名的llvm文件

Id[0]

Id[1]

这种为鉴权合约的log

E、通过其他方式产生的交易

由于交易会同步给每一个节点,定制的neo-gui-nel,和neo-gui-cli

可以得到所有的交易,无论是谁产生的。

但是做智能合约的调试的LOG是很耗时的,所以他并不总是打开的,有一个开关去控制他

这个开关在config.json里面

NEOGUI-nel的设置为true,表示只生成这个GUI自己发起的交易的LOG

打开这个同步就会变得非常慢,调试的时候再用。

 

NEOGUI-nel不修改这个config 也可以打开生成任何人发起的交易的开关

这里按一下让他显示为=false,然后之后同步的每一笔交易都会生成调试LOG了

这个只在内存修改,每次打开neo-gui都要看一眼。

 

  1. 合约的调试

编译,启动NeonDebugGUI

略,他也在NeonDebug仓库中

 

使用Load 打开对应的llvmhex.txt文件即可调试

这两个load,一个是打开一个对话框,一个是输入txid自己去找,结果都一样。

 

功能介绍

打开调试LOG以后

图中红黄蓝部分先注意一下

红色部分最重要,首先观察合约执行状态 halt break 为vm正常停机,程序正常走到结束。

Fault break为vm异常停机,你可以直接反着从最下边看看到底怎么了。

 

红色部分为整个VM内Dump下来的执行流程,也就是整个智能合约的过程到底干了什么,从红色部分选择最上面一项,然后用键盘的方向键下,往下挪,先完整的看几个合约,可以建立一些基本的感觉。

 

黄色部分为智能合约源码,当红色部分选中条目变化时,如果能找到对应的源码,黄色部分就会跳转高亮。

 

通常最主要的调试观测方法就是选红色、看黄色。

 

然后蓝色部分收集了所有syscall,因为neoVM的虚拟机自身是纯计算,所有对链产生影响或者和链互访的功能都是通过syscall进行的,看一个合约有哪些syscall就能快速了解这个合约的行为,当然这需要建立在熟悉的基础上,只要多看,就能熟悉。

调试小提示

  1. 红色部分最重要
  2. 选中条目时,执行栈显示的这条指令执行之后的状态。

比如

Equal指令是==表达式,他显示的false只能告诉你是不想等,要看是谁和谁比较,往前倒退一条指令

倒退后观测计算栈,valuetool 会帮助你显示这个数据可能是什么

  1. 是多条虚拟机指令对应一条cs源码,并且有些是对应不到的。

记住这一点,而不能只盯着源码,左边最重要,左边要选到关键的代码。

你要搞清楚那些最关键的指令,比如syscall appcall equal

这并不会有多复杂,知道找关键指令就会快很多。

  1. 一定要熟悉syscall

最重要的storage.put 就是一个syscall

五一劳动节,讲讲NEO智能合约的调试的更多相关文章

  1. NEO智能合约开发(二)再续不可能的任务

      NEO智能合约开发中,应用合约比较简单,是的你没看错,应用合约比较简单. 应用合约三部曲,发布.调用.看结果.除了看结果工具比较缺乏,发布调用neogui最起码可以支撑你测试.   鉴权合约比较麻 ...

  2. NEO智能合约开发(一)不可能完成的任务

    悬赏任务 兹有如下合约 public static object Main(string method, object[] args) { if (Runtime.Trigger == Trigger ...

  3. 用 C# 编写 NEO 智能合约

    工具 -> 扩展和更新安装 NeoContractPlugin 插件 打开 Visual Studio 2017,打开 工具, 扩展和更新 ,在左侧点击 联机 ,搜索 Neo,安装 NeoCon ...

  4. neo智能合约的生命周期

  5. 通过METAMASK调试和发布智能合约指南(转载)

    2017-12-07 芯链团队 小明微思考 原文地址:https://mp.weixin.qq.com/s?__biz=MzA4Mzk2MzUzNg==&mid=2651223347& ...

  6. NEO从入门到开窗(2) - 智能合约的面相

    一.啰嗦两句 昨天讲了智能合约的一生,那丫长啥样啊?今儿我就跟各位唠叨唠叨. 二.一个简单的智能合约 下面这段就是NEO实例源码里的一个,干撒用的?聪明的你一眼儿就看出来了吧,就是一个所谓域名合约的增 ...

  7. NEO从入门到开窗(1) - 一个智能合约的诞生

    一.啰嗦两句 最近一直都在研究区块链,BitCoin,Etherenum, Hyper Ledger Fabric还有今天的主角小蚂蚁,当然出名以后改了一个艺名叫NEO.区块链大部分都是用Golang ...

  8. EOS智能合约开发(四):智能合约部署及调试(附编程示例)

    EOS智能合约开发(一):EOS环境搭建和创建节点 EOS智能合约开发(二):EOS创建和管理钱包 EOS智能合约开发(三):EOS创建和管理账号 部署智能合约的示例代码如下: $ cleos set ...

  9. 【精解】EOS智能合约演练

    EOS,智能合约,abi,wasm,cleos,eosiocpp,开发调试,钱包,账户,签名权限 热身 本文旨在针对EOS智能合约进行一个完整的实操演练,过程中深入熟悉掌握整个EOS智能合约的流程,过 ...

随机推荐

  1. tomcat8 源码分析 | 组件及启动过程

    tomcat 8 源码分析 ,本文主要讲解tomcat拥有哪些组件,容器,又是如何启动的 推荐访问我的个人网站,排版更好看呦: https://chenmingyu.top/tomcat-source ...

  2. python6-深浅拷贝 元组类型 字典类型 集合类型

    一,深浅拷贝 (一) 值拷贝:应用场景最多 案例:1.ls = [1, 'abc', [10]]    ls1 = ls     # :ls1直接将ls中存放的地址拿过来#  :    ls内部的值发 ...

  3. JN_0005:PS改变图片指定内容颜色

    1,打开图片. 2,选择选区,抽取出独立图存 选中选区,按ctrl + alt + j ,抽取图层. 3,选中图层,再按住 ctrl,点击图层图标 的白色选区处,即可选中图层中的内容. 4,选中图层内 ...

  4. 第十九节: 结合【表达式目录树】来封装EF的BaseDal层的方法

    一. 简介 该章节,可以说是一个简单轻松的章节,只要你对Expression表达式树.EF的基本使用.泛型有所了解,那么本章节实质上就是一个非常简单的封装章节,便于我们快捷开发. PS:在该章节对于E ...

  5. [Luogu P4180][BJWC 2010]严格次小生成树

    严格次小生成树,关键是“严格”,如果是不严格的其实只需要枚举每条不在最小生成树的边,如果得到边权和大于等于最小生成树的结束就行.原理就是因为Kruskal非常贪心,只要随便改一条边就能得到一个非严格的 ...

  6. 玩转ubuntu之初体验

    安装的是ubuntu16.04.1,分区和具体细节就不多说了,非常简单 安装完之后要做的几件事: 1.设置root密码 #设置root密码 sudo passwd root 2.检查并更新系统 #ub ...

  7. PYthon3:简单几步实现冒泡排序

    1.概念理解: 冒泡排序:可以简单的理解为是列表中相近的元素,两两比较,小的在前面.最多需要len()-1次排序. 2.例子:a=[11,7,4,56,35,0] 3.代码实现: 4.输出结果: 第1 ...

  8. laravel5.4 打印输出 sql 语句

    直接打印 sql 语句 DB::connection('test_link')->enableQueryLog(); //执行代码 $log = DB::connection('test_lin ...

  9. 3D Slicer中文教程(二)—软件功能界面介绍

    1.界面介绍 2.菜单及工具栏介绍 (1)菜单 File-文件菜单 文件菜单包含用于加载MRML场景的选项,用于从互联网下载样本数据集或各种类型的各个数据集.此处还提供了保存场景和数据的选项. Edi ...

  10. Prisma GraphQL 服务器 生产者 "https://www.prisma.io"

    Prisma   一个 GraphQL  服务器 生产者     "https://www.prisma.io" , 关注一下