#!/usr/bin/env nomsu -V6.13.12.8 # This file contains basic error reporting code use "core/metaprogramming.nom" use "core/operators.nom" ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ (fail $msg) compiles to "error(\(($msg as lua expr) if $msg else "nil"), 0);" (assume $condition) compiles to: lua> (" local \$assumption = 'Assumption failed: '..tostring((\$condition):get_source_code()) ") return Lua (" if not \($condition as lua expr) then error(\(quote "\$assumption"), 0) end ") (assume $a == $b) compiles to: lua> "local \$assumption = 'Assumption failed: '..tostring(\(\($a == $b) as nomsu))" define mangler return Lua (" do local \(mangle "a"), \(mangle "b") = \($a as lua expr), \($b as lua expr) if \(mangle "a") ~= \(mangle "b") then error(\(quote "\$assumption").."\\n"..tostring(\(mangle "a")).." != "..tostring(\ ..\(mangle "b")), 0) end end ") test: try: fail $worked = (no) try: fail ..if it fails: $worked = (yes) ..if it succeeds: fail "'try' incorrectly ran success case." unless $worked: fail "'try' failed to recover from failure" # Try/except [ try $action if it succeeds $success if it fails $fallback try $action if it fails $fallback if it succeeds $success ] all compile to: $success_lua = ($success as lua) if ((#"\$success_lua") > 0): $success_lua, add "\n" $success_lua, prepend "-- Success:\n" $success_lua, add "if not _fell_through then return table.unpack(_result, 2) end" $fallback_lua = ($fallback as lua) if ((#"\$fallback_lua") > 0): $fallback_lua, prepend "\nlocal function failure() return _result[2] end\n" $fallback_lua, prepend "-- Failure:" return Lua (" do local _fell_through = false local _result = {pcall(function() \($action as lua) _fell_through = true end)} if _result[1] then \$success_lua else \$fallback_lua end end ") ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ (try $action) parses as try $action if it succeeds (do nothing) if it fails (do nothing) (try $action if it fails $msg $fallback) parses as try $action if it succeeds (do nothing) if it fails $msg $fallback (try $action if it succeeds $success) parses as try $action if it succeeds $success if it fails (do nothing) (try $action if it fails $fallback if it succeeds $success) parses as try $action if it succeeds $success if it fails $fallback test: $success = (no) try: do: fail ..then always: $success = (yes) ..if it succeeds: fail "'try ... then always ...' didn't propagate failure" unless $success: fail "'try ... then always ...' didn't execute the 'always' code" (do $action then always $final_action) compiles to (" do -- do/then always local _fell_through = false local _results = {pcall(function() \($action as lua) _fell_through = true end)} \($final_action as lua) if not _results[1] then error(_results[2], 0) end if not _fell_through then return table.unpack(_results, 2) end end ") ~~~ (barf $) parses as (fail $) (assume $1 or barf $2) parses as (unless $1: fail $2)