-
Notifications
You must be signed in to change notification settings - Fork 118
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
ARC-65: AVM Run Time Errors based on program bytecode #315
Conversation
I think it would make sense to include a special prefix to the log to signify the fact that it is an error message. This would remove the requirement of parsing the error message for the last three TEAL lines to determine if the last log was indeed an error message, which seems a bit fragile to me. On a related note, I think we should include a reference implementation for the showing the error message client-side.
|
Scratch that last part. I could imagine this being useful where you have an app ID for an ARC implementation where the implementation varies so you might not necessarily have the source map. Ie. ARC200 tokens. Still, I think it would be useful to add some more context to the ARC describing these sort of scenarios. |
From memory you only get logs back from a failed transaction if you do a simulate call (been a while since I checked so I could be wrong)? AlgoKit utils does have some logic to perform simulate on error (if debug config enabled). Also, the nice thing about this is the current pattern which is to use a TEAL comment and then if source maps are present so the 3 lines surrounding the error line would work nicely still with this model (and in fact would also work if you did a disassembly too?) |
Long story short: I like this technique |
In the example we suggest the usage of fixed-length error codes ( I think we could formalize this better in the ARC, defining a precise regex for the error code and messages. @nullun WDYT? @joe-p @nullun do you have suggestion for the error code/message regex?
It's hard to come up with a general reference implementation, since the program could be anything and contain any error. It was easier to demonstrate a specific example. What do you suggest as reference implementation on top of the existing example?
This approach to AVM run time errors simplifies a lot the overhead of generating, synching, shipping and parsing (client side) the Source Map jointly with the Algod API response. Also, this would work also for non-ARC4 Smart Contracts. It's a general approach, closer to the pure AVM (due to the lack of a native error raising opcode). |
This was actually changed a few months back in algorand/go-algorand#5875 and released in v3.22.0, allowing failed transactions to return PC, appID, stack, scratch, and logs. Although |
Definitely agree with Joe, this should likely be defined as part of the standard so there's a sane way to parse them. I do like what is already there.
Something like |
One other thing to think about that isn't currently in the ARC is encoding. It's implied that error messages are UTF-8 strings but it should be explicit about that. Or we could think of a way to do UTF-8 strings now but also support event signatures, so this could be combined with something like ARC56 for dynamic error messages that don't take up too many bytes on-chain. |
@nullun @joe-p @robdmoore I've updated the ARC based on discussion feedback! If it's ok I'll try to push it for a final call. |
ARCs/arc-0065.md
Outdated
|
||
> Errors consist of a _code_ and an optional _short message_. | ||
|
||
Errors **MUST** statisfy the regexp: `s/^ERR:(\w*)\s-\s(.*)$/`. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I realise this regex wouldn't satisfy the below example of ERR:042
where there isn't a short message.
Maybe the following would be better? Welcome others input here.
s/^ERR:(\w*)(\s?-\s?(.*))?$/
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Just realised my example above created a third capturing group. This may be better: ERR:(\w*)(?:\s?-\s?(.*))?
Which then gives you 2 capturing groups, the first for the error code, the second for the error message.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good point! Also: both numeric (ERR:0042
) and alphanumeric error codes should be supported (either "labels" or things like "hex" codes, like ERR:BADREQUEST
or ERR:0xAA
). I'll update the spec!
ARCs/arc-0065.md
Outdated
} | ||
``` | ||
|
||
The `message` field contains the error `ERR:001 - Invalid Method`. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As discussed, I think changing it to rely on the logs
array would be better. Just decode each value and compare against the regex.
Here's an example I did via the command line with an application that attempts to use ABI methods first, then falls back to non-ABI methods. It resulted in two logged errors before failing.
curl -s -X POST http://127.0.0.1:4001/v2/transactions --data-binary @appl_call.stxn | jq -r '."data"."eval-states"[0]."logs"[] | @base64d' | grep -E "ERR:(\w*)(?:\s?-\s?(.*))?"
ERR:01 - Invalid Method
ERR:02 - No arguments supplied
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
IIRC the SDKs (e.g. JS) currently only throw the error message, e.g. when you do sendRawTransaction
in JS/TS.
To me, including the error code in the surfaced message makes it significantly more useful
> Compilers **MAY** optimize for program bytecode size by storing the error prefixes | ||
in the `bytecblock` and concatenating the error message at the cost of some extra | ||
opcodes. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Relying on .logs
is great when you have them, but I believe a lot of SDKs would hide .logs
e.g. v2 SDK throwing from a sendRawTransaction
call. The error message is mostly guaranteed to be propagated through the SDK, backend->frontend in case of a backend executor, etc. Hiding the error code from the algod error message would significantly detract value imo.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Merg all node stacks in one holding
Please stop spamming. |
@SudoWeezy since ARC-56 is now finalized, are we good to finalize this? |
Sure, will do it right now |
This ARC formalizes the conventions to rise AVM run time errors based on program bytecode.