# This file contains the code to convert syntax trees to Lua code use "nomnom/code_obj.nom" use "nomnom/parser.nom" use "nomnom/pretty_errors.nom" # TODO: use pretty_errors local action [report compile error at %pos %err]: barf "Compile error at \%pos: \%err" local action [report compile error at %pos %err hint %hint]: barf "Compile error at \%pos: \%err\n\%hint" action [compile %tree using %compile_actions]: assume (%tree is a "Syntax Tree") if all of [..] %tree.version, action (Nomsu version) %tree.version != (Nomsu version) action (1 upgraded from 2 to 3) ..then: %tree = (upgrade %tree from %tree.version to (Nomsu version)) if %tree.type is: "Action": %stub = %tree.stub %compile_action = %compile_actions.%stub # Don't apply compiler actions to methods if (%compile_action and (not %tree.target)): # TODO: restore this: #%args = [%tree, %compile_actions] %args = [%nomsu, %tree] for % in (%tree::get args): %args::add % %result = (call %compile_action with %args) if (%result == (nil)): report compile error at %tree.source "\ ..The compile-time action here (\(%tree.stub)) failed to return any value." ..hint "\ ..Look at the implementation of (\(%tree.stub)) and make sure it's returning something." if (%result is a "Syntax Tree"): if (%result == %tree): report compile error at %tree.source "\ ..The compile-time action here (\(%tree.stub)) is producing an endless loop." ..hint "\ ..Look at the implementation of (\(%tree.stub)) and make sure it's not just returning the original tree." return (compile %result using %compile_actions) return %result %lua = (Lua Value from %tree) if %tree.target: # Method call %target_lua = (compile %tree.target using %compile_actions) if (((%target_lua::as smext)::matches "^%(.*%)$") or ((%target_lua::as smext)::matches "^[_a-zA-Z][_a-zA-Z0-9]*$")): %lua::add [%target_lua, ":"] ..else: %lua::add ["(", %target_lua, "):"] %lua::add [%stub as lua id, "("] %args = [] for %tok in %tree at %i: if (%tok is text): do next %tok # TODO: maybe don't translate Lua comments #if (%tok.type == "Comment"): do next %tok if (%tok.type == "Block"): %values = [] for %line in %tok: #unless (%line.type == "Comment"): %values::add (compile %line using %compile_actions) if all of (%.is_value for % in %values): if ((size of %values) == 1): %arg_lua = %values.1 ..else: %arg_lua = (Lua Value from %tok ["("]) %arg_lua::add %values joined with " and nil or " %arg_lua::add ")" ..else: %arg_lua = (Lua Value from %tok ["((function()"]) for %v in %values at %i: if %v.is_value: %v = (%v::as statements ("return " if (%i == (size of %values) else ""))) %arg_lua::add ["\n ", %v] %arg_lua::add "\nend)())" ..else: %arg_lua = (compile %tok using %compile_actions) unless %arg_lua.is_value: if (%tok.type == "Action"): report compile error at %tok "\ ..Can't use this as an argument to (\%stub), since it's not \ ..an expression, it produces: \%arg_lua" ..hint "\ ..Check the implementation of (\(%tok.stub)) to see if it \ ..is actually meant to produce an expression." ..else: report compile error at %tok "\ ..Can't use this as an argument to (\%stub), since it's \ ..not an expression, it produces: \%arg_lua" assume (%arg_lua != %lua) or barf "Huh? \%tree .\%i = \%tok -> \%arg_lua" %args::add %arg_lua %lua::add %args joined with ", " %lua::add ")" return %lua "EscapedNomsu": return (Lua Value from %tree ((%tree.1)::as nomsu)) "Block": %lua = (Lua Code from %tree) %lua::add (..) ((compile %line using %compile_actions)::as statements) ..for %line in %tree ..joined with "\n" return %lua "Text": %lua = (Lua Value from %tree) %lua_bits = [] %string_buffer = "" for % in %tree: if (% is text): %string_buffer = "\%string_buffer\%" do next % if (%string_buffer != ""): %lua_bits::add (%string_buffer::as lua) %string_buffer = "" %bit_lua = (compile % using %compile_actions) unless %bit_lua.is_value: report compile error at % "\ ..Can't use this as a string interpolation value, since it doesn't have a value." if (%.type != "Text"): %bit_lua = (Lua Value from % ["tostring(",%bit_lua,")"]) %lua_bits::add %bit_lua if ((%string_buffer != "") or ((size of %lua_bits) == 0)): %lua_bits::add (%string_buffer::as lua) %lua::add %lua_bits joined with ("..") if ((size of %lua_bits) > 1): %lua::parenthesize return %lua "List": %lua = (Lua Value from %tree ["List{"]) %lua::add ((compile % using %compile_actions) for % in %tree) joined with ", " or ",\n " %lua::add "}" return %lua "Dict": %lua = (Lua Value from %tree ["Dict{"]) %lua::add ((compile % using %compile_actions) for % in %tree) joined with ", " or ",\n " %lua::add "}" return %lua "DictEntry": set {%key:%tree.1, %value:%tree.2} %key_lua = (compile %key using %compile_actions) unless %key_lua.is_value: report compile error at %tree.1 "\ ..Can't use this as a dict key, since it's not an expression." %value_lua = (..) (compile %value using %compile_actions) if %value ..else (Lua Value from %key ["true"]) unless %value_lua.is_value: report compile error at %tree.2 "\ ..Can't use this as a dict value, since it's not an expression." %key_str = ((%key_lua::as smext)::matching "^[\"']([a-zA-Z_][a-zA-Z0-9_]*)['\"]$") if: %key_str: return (Lua Code from %tree [%key_str, "=", %value_lua]) ((%key_lua::as smext).1 == "["): # NOTE: this *must* use a space after the [ to avoid freaking out Lua's parser if the inner expression is a long string. Lua parses x[[[y]]] as x("[y]"), not as x["y"] return (Lua Code from %tree ["[ ", %key_lua, "]=", %value_lua]) else: return (Lua Code from %tree ["[", %key_lua, "]=", %value_lua]) "IndexChain": %lua = (compile %tree.1 using %compile_actions) unless %lua.is_value: report compile error at %tree.1 "\ ..Can't index into this, since it's not an expression." %first_char = (%lua::as smext).1 if (any of [%first_char == "{", %first_char == "\"", %first_char == "["]): %lua::parenthesize for %i in 2 to (size of %tree): %key = %tree.%i %key_lua = (compile %key using %compile_actions) unless %key_lua.is_value: report compile error at %key "\ ..Can't use this as an index, since it's not an expression." %key_lua_str = (%key_lua::as smext) %lua_id = (%key_lua_str::matching "^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$") if: %lua_id: %lua::add [".", %lua_id] (%key_lua_str.1 == "["): # NOTE: this *must* use a space after the [ to avoid freaking out Lua's parser if the inner expression is a long string. Lua parses x[[[y]]] as x("[y]"), not as x["y"] %lua::add ["[ ", %key_lua, " ]"] else: %lua::add ["[", %key_lua, "]"] return %lua "Number": return (Lua Value from %tree ["\(%tree.1)"]) "Var": return (Lua Value from %tree [%tree.1::as lua id]) "FileChunks": barf "\ ..Can't convert FileChunks to a single block of lua, since each chunk's \ ..compilation depends on the earlier chunks" "Comment": # TODO: de-implement? return (Lua Code from %tree "-- \(%tree.1::with "\n" -> "\n-- ")") "Error": barf "Can't compile errors" else: barf "Unknown type: \(%tree.type)"