-- This file contains the datastructures used to represent parsed Nomsu syntax trees, -- as well as the logic for converting them to Lua code. utils = require 'utils' {:repr, :stringify, :min, :max, :equivalent, :set, :is_list, :sum} = utils immutable = require 'immutable' {:insert, :remove, :concat} = table {:Lua, :Nomsu, :Location} = require "code_obj" MAX_LINE = 80 -- For beautification purposes, try not to make lines much longer than this value Types = {} Types.DictEntry = immutable({"key","value"}, {name:"DictEntry"}) Types.is_node = (n)-> type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n) -- Helper method: Tree = (name, methods)-> with methods .__tostring = => "#{@name}(#{repr(@value)}, #{repr @source})" .with_value = (value)=> getmetatable(self)(value, @source) .type = name .name = name .original_nomsu = => leading_space = 0 src_file = FILE_CACHE[@source.filename] while src_file\sub(@source.start-leading_space-1, @source.start-leading_space-1) == " " leading_space += 1 if src_file\sub(@source.start-leading_space-1, @source.start-leading_space-1) != "\n" leading_space = 0 ret = tostring(@source\get_text!)\gsub("\n"..((" ")\rep(leading_space)), "\n") return ret Types[name] = immutable {"value","source"}, methods Tree "Nomsu", as_lua: (nomsu)=> Lua.Value(@source, "nomsu:parse(Nomsu(",repr(@value.source),", ",repr(tostring(@value\as_nomsu!)),"))") as_nomsu: (inline=false)=> nomsu = @value\as_nomsu(true) if nomsu == nil and not inline nomsu = @value\as_nomsu! return nomsu and Nomsu(@source, "\\:\n ", nomsu) return nomsu and Nomsu(@source, "\\(", nomsu, ")") map: (fn)=> fn(self) or @with_value(@value\map(fn)) Tree "Block", as_lua: (nomsu)=> lua = Lua(@source) for i,line in ipairs @value line_lua = line\as_lua(nomsu) if i > 1 lua\append "\n" line_lua\convert_to_statements! lua\append line_lua return lua as_nomsu: (inline=false)=> if inline nomsu = Nomsu(@source) for i,line in ipairs @value if i > 1 nomsu\append "; " line_nomsu = line\as_nomsu(true) return nil unless line_nomsu nomsu\append line_nomsu return nomsu nomsu = Nomsu(@source) for i, line in ipairs @value line = assert(line\as_nomsu(nil, true), "Could not convert line to nomsu") nomsu\append line if i < #@value nomsu\append "\n" if tostring(line)\match("\n") nomsu\append "\n" return nomsu map: (fn)=> fn(self) or @with_value(Tuple(unpack([v\map(fn) for v in *@value]))) math_expression = re.compile [[ ([+-] " ")* "%" (" " [*/^+-] (" " [+-])* " %")+ !. ]] Tree "Action", as_lua: (nomsu)=> stub = @get_stub! compile_action = nomsu.environment.COMPILE_ACTIONS[stub] if compile_action args = [arg for arg in *@value when arg.type != "Word"] -- Force all compile-time actions to take a tree location args = [args[p-1] for p in *nomsu.environment.ARG_ORDERS[compile_action][stub]] -- Force Lua to avoid tail call optimization for debugging purposes ret = compile_action(self, unpack(args)) if not ret then error("Failed to produce any Lua") return ret action = rawget(nomsu.environment.ACTIONS, stub) lua = Lua.Value(@source) if not action and math_expression\match(stub) -- This is a bit of a hack, but this code handles arbitrarily complex -- math expressions like 2*x + 3^2 without having to define a single -- action for every possibility. for i,tok in ipairs @value if tok.type == "Word" lua\append tok.value else tok_lua = tok\as_lua(nomsu) unless tok_lua.is_value src = tok.source\get_text! error("non-expression value inside math expression: #{colored.yellow src}") if tok.type == "Action" tok_lua\parenthesize! lua\append tok_lua if i < #@value lua\append " " return lua args = {} for i, tok in ipairs @value if tok.type == "Word" then continue arg_lua = tok\as_lua(nomsu) unless arg_lua.is_value line, src = tok.source\get_line!, tok.source\get_text! error "#{line}: Cannot use:\n#{colored.yellow src}\nas an argument to #{stub}, since it's not an expression, it produces: #{repr arg_lua}", 0 insert args, arg_lua if action args = [args[p] for p in *nomsu.environment.ARG_ORDERS[action][stub]] -- Not really worth bothering with ACTIONS.foo(...) style since almost every action -- has arguments, so it won't work lua\append "ACTIONS[",repr(stub),"](" for i, arg in ipairs args lua\append arg if i < #args then lua\append ", " lua\append ")" return lua get_stub: (include_names=false)=> bits = if include_names [(t.type == "Word" and t.value or "%#{t.value}") for t in *@value] else [(t.type == "Word" and t.value or "%") for t in *@value] return concat(bits, " ") as_nomsu: (inline=false, can_use_colon=false)=> if inline nomsu = Nomsu(@source) for i,bit in ipairs @value if bit.type == "Word" if i > 1 nomsu\append " " nomsu\append bit.value else arg_nomsu = bit\as_nomsu(true) return nil unless arg_nomsu unless i == 1 nomsu\append " " if bit.type == "Action" or bit.type == "Block" arg_nomsu\parenthesize! nomsu\append arg_nomsu return nomsu else nomsu = Nomsu(@source) next_space = "" -- TODO: track line length as we go and use 80-that instead of 80 for wrapping last_colon = nil for i,bit in ipairs @value if bit.type == "Word" nomsu\append next_space, bit.value next_space = " " else arg_nomsu = if last_colon == i-1 and bit.type == "Action" then nil elseif bit.type == "Block" then nil else bit\as_nomsu(true) if arg_nomsu and #arg_nomsu < MAX_LINE if bit.type == "Action" if can_use_colon and i > 1 nomsu\append next_space\match("[^ ]*"), ": ", arg_nomsu next_space = "\n.." last_colon = i else nomsu\append next_space, "(", arg_nomsu, ")" next_space = " " else nomsu\append next_space, arg_nomsu next_space = " " else arg_nomsu = bit\as_nomsu(nil, true) return nil unless nomsu -- These types carry their own indentation if bit.type != "List" and bit.type != "Dict" and bit.type != "Text" if i == 1 arg_nomsu = Nomsu(bit.source, "(..)\n ", arg_nomsu) else arg_nomsu = Nomsu(bit.source, "\n ", arg_nomsu) if last_colon == i-1 and (bit.type == "Action" or bit.type == "Block") next_space = "" nomsu\append next_space, arg_nomsu next_space = "\n.." if next_space == " " and #(tostring(nomsu)\match("[^\n]*$")) > MAX_LINE next_space = "\n.." return nomsu map: (fn)=> fn(self) or @with_value(Tuple(unpack([v\map(fn) for v in *@value]))) Tree "Text", as_lua: (nomsu)=> lua = Lua.Value(@source) string_buffer = "" for bit in *@value if type(bit) == "string" string_buffer ..= bit continue if string_buffer ~= "" if #lua.bits > 0 then lua\append ".." lua\append repr(string_buffer) string_buffer = "" bit_lua = bit\as_lua(nomsu) unless bit_lua.is_value line, src = bit.source\get_line!, bit.source\get_text! error "#{line}: Cannot use #{colored.yellow bit} as a string interpolation value, since it's not an expression.", 0 if #lua.bits > 0 then lua\append ".." if bit.type != "Text" bit_lua = Lua.Value(bit.source, "stringify(",bit_lua,")") lua\append bit_lua if string_buffer ~= "" or #lua.bits == 0 if #lua.bits > 0 then lua\append ".." lua\append repr(string_buffer) if #lua.bits > 1 lua\parenthesize! return lua as_nomsu: (inline=false)=> if inline nomsu = Nomsu(@source, '"') for bit in *@value if type(bit) == 'string' -- TODO: unescape better? nomsu\append (bit\gsub("\\","\\\\")\gsub("\n","\\n")) else interp_nomsu = bit\as_nomsu(true) if interp_nomsu if bit.type != "Word" and bit.type != "List" and bit.type != "Dict" and bit.type != "Text" interp_nomsu\parenthesize! nomsu\append "\\", interp_nomsu else return nil nomsu\append '"' return nomsu else inline_version = @as_nomsu(true) if inline_version and #inline_version <= MAX_LINE return inline_version nomsu = Nomsu(@source, '".."\n ') for i, bit in ipairs @value if type(bit) == 'string' nomsu\append (bit\gsub("\\","\\\\")\gsub("\n","\n ")) else interp_nomsu = bit\as_nomsu(true) if interp_nomsu if bit.type != "Word" and bit.type != "List" and bit.type != "Dict" and bit.type != "Text" interp_nomsu\parenthesize! nomsu\append "\\", interp_nomsu else interp_nomsu = bit\as_nomsu! return nil unless interp_nomsu nomsu\append "\\\n ", interp_nomsu if i < #@value nomsu\append "\n .." return nomsu map: (fn)=> fn(self) or @with_value(Tuple(unpack([type(v) == 'string' and v or v\map(fn) for v in *@value]))) Tree "List", as_lua: (nomsu)=> lua = Lua.Value @source, "{" line_length = 0 for i, item in ipairs @value item_lua = item\as_lua(nomsu) unless item_lua.is_value line, src = item.source\get_line!, item.source\get_text! error "#{line}: Cannot use #{colored.yellow src} as a list item, since it's not an expression.", 0 lua\append item_lua item_string = tostring(item_lua) last_line = item_string\match("[^\n]*$") if item_string\match("\n") line_length = #last_line else line_length += #last_line if i < #@value if line_length >= MAX_LINE lua\append ",\n " line_length = 0 else lua\append ", " line_length += 2 lua\append "}" return lua as_nomsu: (inline=false)=> if inline nomsu = Nomsu(@source, "[") for i, item in ipairs @value item_nomsu = item\as_nomsu(true) return nil unless item_nomsu if i > 1 nomsu\append ", " nomsu\append item_nomsu nomsu\append "]" return nomsu else inline_version = @as_nomsu(true) if inline_version and #inline_version <= MAX_LINE return inline_version nomsu = Nomsu(@source, "[..]") line = Nomsu(@source, "\n ") for item in *@value item_nomsu = item\as_nomsu(true) if item_nomsu and #line + #", " + #item_nomsu <= MAX_LINE if #line.bits > 1 line\append ", " line\append item_nomsu else unless item_nomsu item_nomsu = item\as_nomsu! return nil unless item_nomsu if #line.bits > 1 nomsu\append line line = Nomsu(item.source, "\n ") line\append item_nomsu if #line.bits > 1 nomsu\append line return nomsu map: (fn)=> fn(self) or @with_value(Tuple(unpack([v\map(fn) for v in *@value]))) Tree "Dict", as_lua: (nomsu)=> lua = Lua.Value @source, "{" line_length = 0 for i, entry in ipairs @value key_lua = entry.key\as_lua(nomsu) unless key_lua.is_value line, src = key.source\get_line!, key.source\get_text! error "#{line}: Cannot use #{colored.yellow src} as a dict key, since it's not an expression.", 0 value_lua = entry.value and entry.value\as_lua(nomsu) or Lua.Value(entry.key.source, "true") unless value_lua.is_value line, src = value.source\get_line!, value.source\get_text! error "#{line}: Cannot use #{colored.yellow src} as a dict value, since it's not an expression.", 0 key_str = tostring(key_lua)\match([=[["']([a-zA-Z_][a-zA-Z0-9_]*)['"]]=]) if key_str lua\append key_str,"=",value_lua elseif tostring(key_lua)\sub(1,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\append "[ ",key_lua,"]=",value_lua else lua\append "[",key_lua,"]=",value_lua -- TODO: maybe make this more accurate? It's only a heuristic, so eh... newlines, last_line = ("[#{key_lua}=#{value_lua}")\match("^(.-)([^\n]*)$") if #newlines > 0 line_length = #last_line else line_length += #last_line if i < #@value if line_length >= MAX_LINE lua\append ",\n " line_length = 0 else lua\append ", " line_length += 2 lua\append "}" return lua as_nomsu: (inline=false)=> if inline nomsu = Nomsu(@source, "{") for i, entry in ipairs @value key_nomsu = entry.key\as_nomsu(true) return nil unless key_nomsu if entry.key.type == "Action" or entry.key.type == "Block" key_nomsu\parenthesize! value_nomsu = entry.value and entry.value\as_nomsu(true) or Nomsu(entry.key.source, "") return nil unless value_nomsu if i > 1 nomsu\append ", " nomsu\append key_nomsu,":",value_nomsu nomsu\append "}" return nomsu else inline_version = @as_nomsu(true) if inline_version then return inline_version nomsu = Nomsu(@source, "{..}") line = Nomsu(@source, "\n ") for entry in *@value key_nomsu = entry.key\as_nomsu(true) return nil unless key_nomsu if entry.key.type == "Action" or entry.key.type == "Block" key_nomsu\parenthesize! value_nomsu = entry.value and entry.value\as_nomsu(true) or Nomsu(entry.key.source, "") if value_nomsu and #line + #", " + #key_nomsu + #":" + #value_nomsu <= MAX_LINE if #line.bits > 1 line\append ", " line\append key_nomsu if entry.value then line\append ":",value_nomsu else unless value_nomsu value_nomsu = entry.value\as_nomsu! return nil unless value_nomsu if #line.bits > 1 nomsu\append line line = Nomsu(bit.source, "\n ") line\append key_nomsu if entry.value then line\append ":",value_nomsu if #line.bits > 1 nomsu\append line return nomsu map: (fn)=> DictEntry = Types.DictEntry fn(self) or @with_value(Tuple(unpack([DictEntry(e.key\map(fn), e.value\map(fn)) for e in *@value]))) Tree "IndexChain", as_lua: (nomsu)=> lua = @value[1]\as_lua(nomsu) unless lua.is_value line, src = @value[1].source\get_line!, @value[1].source\get_text! error "#{line}: Cannot index #{colored.yellow src}, since it's not an expression.", 0 first_char = tostring(lua)\sub(1,1) if first_char == "{" or first_char == '"' or first_char == "[" lua\parenthesize! for i=2,#@value key = @value[i] if key.type == 'Text' and #key.value == 1 and type(key.value[1]) == 'string' and key.value[1]\match("^[a-zA-Z_][a-zA-Z0-9_]*$") lua\append ".#{key.value[1]}" continue key_lua = key\as_lua(nomsu) unless key_lua.is_value line, src = key.source\get_line!, key.source\get_text! error "#{line}: Cannot use #{colored.yellow src} as an index, since it's not an expression.", 0 -- 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"] if tostring(key_lua)\sub(1,1) == '[' lua\append "[ ",key_lua,"]" else lua\append "[",key_lua,"]" return lua as_nomsu: (inline=false)=> nomsu = Nomsu(@source) for i, bit in ipairs @value if i > 1 nomsu\append "." bit_nomsu = bit\as_nomsu(true) return nil unless bit_nomsu if bit.type == "Action" or bit.type == "Block" bit_nomsu\parenthesize! nomsu\append bit_nomsu return nomsu map: (fn)=> fn(self) or @with_value(Tuple(unpack([v\map(fn) for v in *@value]))) Tree "Number", as_lua: (nomsu)=> Lua.Value(@source, tostring(@value)) as_nomsu: (inline=false)=> return Nomsu(@source, tostring(@value)) map: (fn)=> fn(self) or self Tree "Var", as_lua: (nomsu)=> lua_id = "_"..(@value\gsub "%W", (verboten)-> if verboten == "_" then "__" else ("_%x")\format(verboten\byte!)) Lua.Value(@source, lua_id) as_nomsu: (inline=false)=> return Nomsu(@source, "%", @value) map: (fn)=> fn(self) or self Tree "Word", as_lua: (nomsu)=> error("Attempt to convert Word to lua") as_nomsu: (inline=false)=> return Nomsu(@source, @value) map: (fn)=> fn(self) or self Tree "Comment", as_lua: (nomsu)=> Lua(@source, "--"..@value\gsub("\n","\n--").."\n") as_nomsu: (inline=false)=> return nil if inline if @value\match("\n") return Nomsu(@source, "#..", @value\gsub("\n", "\n ")) else return Nomsu(@source, "#", @value) map: (fn)=> fn(self) or self return Types