Error handling
Sometimes contracts can produce errors. There are two kinds of GenVM errors:
- unrecoverable errors
exit(x)
where (if then return is considered to beNone
)- unhandled
Exception
, which will result inexit(1)
- rollbacks
raise Rollback("message")
--- raises regular Python exception that can be caught, if it escapes your contract, then it will call the latergl.rollback_immediate("message")
--- immediately terminates current sub-vm without unwinding the stack, which is more cost-efficient, but can't be caught in the current sub-vm
Built-in gl.eq_principle_*
functions family will raise Rollback(sub_vm_message)
in case non-deterministic block agreed on a rollback.
Which means that deterministic block can catch and inspect such error from a non-deterministic block and act on it.
Note that rollback message is compared for strict equality, which means that you should not include stacktrace there, but rather use some descriptive error codes.
Difference between rollbacks and unrecoverable errors is that unrecoverable errors can't be handled by user code and their "message" contains only short description provided by the GenVM. However, there are two exceptions of when a contract can observe unrecoverable errors:
- If leaders non-deterministic block produced an error. It is discouraged to do that and indicates that some unexpected (by contract) corner case was reached
- You can get one out of
gl.advanced.sandbox
method
# { "Depends": "py-genlayer:test" }
from genlayer import *
class Contract(gl.Contract):
@gl.public.write
def main(self):
def run():
raise gl.Rollback('my message')
try:
gl.eq_principle_strict_eq(run)
except Rollback as r:
print(r.msg) # will print `my message`