|
|
|
Tutorial on debugging transactions with Remix
|
|
|
|
===============================================
|
|
|
|
|
|
|
|
The goal of this tutorial is to explain how to debug transaction using
|
|
|
|
Remix.
|
|
|
|
|
|
|
|
Start debugging
|
|
|
|
---------------
|
|
|
|
|
|
|
|
There are two different ways to start debugging, each way correspond to
|
|
|
|
a different use case.
|
|
|
|
|
|
|
|
### From the Transaction GUI
|
|
|
|
|
|
|
|
We will not explain in detail here how to write or deploy contract. Let
|
|
|
|
us start with a basic contract (replace this one by your's):
|
|
|
|
|
|
|
|
``` {.sourceCode .none}
|
|
|
|
contract Donation {
|
|
|
|
address owner;
|
|
|
|
event fundMoved(address _to, uint _amount);
|
|
|
|
modifier onlyowner { if (msg.sender == owner) _; }
|
|
|
|
address[] _giver;
|
|
|
|
uint[] _values;
|
|
|
|
|
|
|
|
function Donation() {
|
|
|
|
owner = msg.sender;
|
|
|
|
}
|
|
|
|
|
|
|
|
function donate() payable {
|
|
|
|
addGiver(msg.value);
|
|
|
|
}
|
|
|
|
|
|
|
|
function moveFund(address _to, uint _amount) onlyowner {
|
|
|
|
uint balance = this.balance;
|
|
|
|
uint amount = _amount;
|
|
|
|
if (_amount <= this.balance) {
|
|
|
|
if (_to.send(this.balance)) {
|
|
|
|
fundMoved(_to, _amount);
|
|
|
|
} else {
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function addGiver(uint _amount) internal {
|
|
|
|
_giver.push(msg.sender);
|
|
|
|
_values.push(_amount);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
For the purpose of this tutorial, we will run the `JavaScript VM`
|
|
|
|
(that's the default mode when you don't use Remix with Mist or
|
|
|
|
Metamask). This simulates a custom blockchain. You could do the same
|
|
|
|
using a proper backend node.
|
|
|
|
|
|
|
|
Now, let's deploy the contract:
|
|
|
|
|
|
|
|
Right panel / Red button `Create`
|
|
|
|
|
|
|
|
data:image/s3,"s3://crabby-images/39072/39072ac872f23906e8e0c73a2935c0b99c0edfaf" alt="image"
|
|
|
|
|
|
|
|
Then we should call the `Donate` function (that will send Ether to the
|
|
|
|
contract).
|
|
|
|
|
|
|
|
Let's set the amount of Ether:
|
|
|
|
|
|
|
|
Right panel / second tab from the left - fill in the ´´value´´ input (´1
|
|
|
|
ether´ for instance)
|
|
|
|
|
|
|
|
data:image/s3,"s3://crabby-images/34bbd/34bbdc7ca226b0f78a1c96b13dfc2a4d2c1ba2f5" alt="image"
|
|
|
|
|
|
|
|
Then click on `Donate`. As we are using the `JavaScript VM`, everything
|
|
|
|
goes almost instantly.
|
|
|
|
|
|
|
|
Remix displays also some information related to each transaction result.
|
|
|
|
In the terminal, the transaction is logged and you can start debugging
|
|
|
|
it.
|
|
|
|
|
|
|
|
data:image/s3,"s3://crabby-images/5ad9a/5ad9aa629bedb914013286eed3690bca366b21f3" alt="image"
|
|
|
|
|
|
|
|
### From the Debugger
|
|
|
|
|
|
|
|
The debugger can be found in the right panel / 5th tab from the left.
|
|
|
|
|
|
|
|
You can start a debug session by providing either a `transaction hash`
|
|
|
|
or a `block number` and `transaction index`.
|
|
|
|
|
|
|
|
data:image/s3,"s3://crabby-images/81b8d/81b8d2624de81f88a5cb1f682b7b585403707dbe" alt="image"
|
|
|
|
|
|
|
|
Click the `play` button to start debugging.
|
|
|
|
|
|
|
|
Using the debugger
|
|
|
|
------------------
|
|
|
|
|
|
|
|
The debugger allows one to see detailed informations about the
|
|
|
|
transaction's execution. It uses the editor (left panel) to display the
|
|
|
|
location in the source code where the current execution is.
|
|
|
|
|
|
|
|
The transaction panel displays basic information about the current
|
|
|
|
transaction.
|
|
|
|
|
|
|
|
data:image/s3,"s3://crabby-images/c20dc/c20dc015a0d3b49cfb888b1d50c5e22c6af7923e" alt="image"
|
|
|
|
|
|
|
|
The navigation part contains a slider and buttons that can be used to
|
|
|
|
step through the transaction execution.
|
|
|
|
|
|
|
|
From the left to the right:
|
|
|
|
|
|
|
|
step over back, step into back, step into forward, step over forward,
|
|
|
|
jump out (jump out of the current call), jump to the previous
|
|
|
|
breakpoint, jump to the next breakpoint.
|
|
|
|
|
|
|
|
data:image/s3,"s3://crabby-images/98ef1/98ef1f8add9083daab64b8a1f76a2189b9cf628d" alt="image"
|
|
|
|
|
|
|
|
11 panels give detailed information about the execution:
|
|
|
|
|
|
|
|
### Instructions
|
|
|
|
|
|
|
|
data:image/s3,"s3://crabby-images/b0f81/b0f81437077017433e9b5e80f734af0c8892768f" alt="image"
|
|
|
|
|
|
|
|
The Instructions panel displays the bytecode of the current executing
|
|
|
|
contract- with the current step highlighted.
|
|
|
|
|
|
|
|
Important note: When this panel is hidden, the slider will have a
|
|
|
|
courser granularity and only stop at expression boundaries, even if they
|
|
|
|
are compiled into multiple EVM instructions. When the panel is
|
|
|
|
displayed, it will be possible to step over every instruction, even
|
|
|
|
those that refers to the same expression.
|
|
|
|
|
|
|
|
### Solidity Locals
|
|
|
|
|
|
|
|
data:image/s3,"s3://crabby-images/8b4f1/8b4f17571d7401668cd0154e0adfc79a16fdf65f" alt="image"
|
|
|
|
|
|
|
|
The Solidity Locals panel displays local variables associated with the
|
|
|
|
current context.
|
|
|
|
|
|
|
|
### Solidity State
|
|
|
|
|
|
|
|
data:image/s3,"s3://crabby-images/506a2/506a20aef47de284f3db9e97415fea6699be37cd" alt="image"
|
|
|
|
|
|
|
|
The Solidity State panel displays state variables of the current
|
|
|
|
executing contract.
|
|
|
|
|
|
|
|
### Low level panels
|
|
|
|
|
|
|
|
These panels display low level informations about the execution:
|
|
|
|
|
|
|
|
> - Stack
|
|
|
|
> - Storages Changes
|
|
|
|
> - Memory
|
|
|
|
> - Call Data
|
|
|
|
> - Call Stack
|
|
|
|
> - Return Value (only if the current step is a RETURN opcode)
|
|
|
|
> - Full Storages Changes (only at the end of the execution - display
|
|
|
|
> every storage change of every modified contract)
|
|
|
|
|
|
|
|
### Reverted Transaction
|
|
|
|
|
|
|
|
A transaction could be reverted (either because of out of gas exception,
|
|
|
|
Solidity `throw` or low level exception).
|
|
|
|
|
|
|
|
In that case it is important to be aware of the exception and to locate
|
|
|
|
where the exception is in the source code.
|
|
|
|
|
|
|
|
Remix will warn you when the execution throws an exception. The
|
|
|
|
`warning` button will jump to the last opcode before the exception
|
|
|
|
happened.
|
|
|
|
|
|
|
|
data:image/s3,"s3://crabby-images/c2d55/c2d55117ed6d24f0c2529c547e7d076fa66b080a" alt="image"
|
|
|
|
|
|
|
|
### Breakpoints
|
|
|
|
|
|
|
|
The two last buttons from the navigation area are used to jump either
|
|
|
|
back to the previous breakpoint or forward to the next breakpoint.
|
|
|
|
|
|
|
|
Breakpoints can be added and removed by clicking on the line number.
|
|
|
|
|
|
|
|
data:image/s3,"s3://crabby-images/c2163/c2163c1d6e7c502107c6deeade29a80ff376ee83" alt="image"
|
|
|
|
|
|
|
|
When a debug session is started, the execution will jump to the first
|
|
|
|
encountered breakpoint.
|
|
|
|
|
|
|
|
Important note: If you add a breakpoint to a line that declares a
|
|
|
|
variable, it might be triggered twice: Once for initializing the
|
|
|
|
variable to zero and second time for assigning the actual value. As an
|
|
|
|
example, assume you are debugging the following contract:
|
|
|
|
|
|
|
|
``` {.sourceCode .none}
|
|
|
|
contract ctr {
|
|
|
|
function hid () {
|
|
|
|
uint p = 45;
|
|
|
|
uint m;
|
|
|
|
m = 89;
|
|
|
|
uint l = 34;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
And let's says that breakpoints are set for the lines
|
|
|
|
|
|
|
|
`uint p = 45;`
|
|
|
|
|
|
|
|
`m = 89;`
|
|
|
|
|
|
|
|
`uint l = 34;`
|
|
|
|
|
|
|
|
then clicking on `Jump to next breakpoint` will stop at the following
|
|
|
|
lines in the given order:
|
|
|
|
|
|
|
|
> `uint p = 45;` (declaration of p)
|
|
|
|
>
|
|
|
|
> `uint l = 34;` (declaration of l)
|
|
|
|
>
|
|
|
|
> `uint p = 45;` (45 assigned to p)
|
|
|
|
>
|
|
|
|
> `m = 89;` (89 assigned to m)
|
|
|
|
>
|
|
|
|
> `uint l = 34;` (34 assigned to l)
|