diff --git a/code_obj.lua b/code_obj.lua index edbd446..a861bf2 100644 --- a/code_obj.lua +++ b/code_obj.lua @@ -105,11 +105,7 @@ do self.trailing_line_len = #trailing_text end else - if #b.indents > 1 then - self.trailing_line_len = b.trailing_line_len - else - self.trailing_line_len = self.trailing_line_len + #tostring(b) - end + self.trailing_line_len = math.min(self.trailing_line_len + #tostring(b), b.trailing_line_len) if self.current_indent ~= 0 then indents[#bits] = self.current_indent end diff --git a/code_obj.moon b/code_obj.moon index caca4cf..0981d5a 100644 --- a/code_obj.moon +++ b/code_obj.moon @@ -73,10 +73,7 @@ class Code @current_indent = #spaces @trailing_line_len = #trailing_text else - if #b.indents > 1 - @trailing_line_len = b.trailing_line_len - else - @trailing_line_len += #tostring(b) + @trailing_line_len = math.min(@trailing_line_len + #tostring(b), b.trailing_line_len) if @current_indent != 0 indents[#bits] = @current_indent @__str = nil diff --git a/core/metaprogramming.nom b/core/metaprogramming.nom index 2be7132..17b3be4 100644 --- a/core/metaprogramming.nom +++ b/core/metaprogramming.nom @@ -164,12 +164,11 @@ compile [%tree with %t -> %replacement] to \(%replacement as lua return) end) -compile [%tree with vars %v] to - Lua value ".." - \(%tree as lua expr):map(function(t) - local replacements = \(%v as lua expr) - if t.type == "Var" then - return replacements[t[1]] +action [%tree with vars %replacements] + =lua ".." + \%tree:map(function(\%t) + if \%t.type == "Var" then + return \%replacements[\%t[1]] end end) diff --git a/nomsu.1.peg b/nomsu.1.peg index 1551b6d..39c0a9a 100644 --- a/nomsu.1.peg +++ b/nomsu.1.peg @@ -1,6 +1,7 @@ -- Nomsu version 2 file (FileChunks): {:curr_indent: ' '* :} + ("#!" (!"nomsu" [^%nl])* "nomsu" %ws+ "-V" %ws* {:version: ([0-9.]+ -> tonumber) :} [^%nl]*)? comment? blank_lines? (chunk (nl_nodent chunk_delimeter nl_nodent chunk)*)? blank_lines? @@ -100,12 +101,12 @@ inline_text_interpolation: ) indented_text (Text): - '".."' eol %nl {:curr_indent: indent :} - (indented_plain_text / text_interpolation / {~ blank_lines (=curr_indent -> "") ~})* + '".."' eol %nl {%nl*} {:curr_indent: indent :} + (indented_plain_text / text_interpolation / {~ %nl+ (=curr_indent -> "") ~})* (!! [^%nl]+ -> "Unexpected character while parsing Text" !!)? indented_plain_text (Text): {~ (("\\" -> "\") / (("\" blank_lines =curr_indent "..") -> "") / (!text_interpolation "\") / [^%nl\]+)+ - (blank_lines (=curr_indent -> ""))* ~} + (%nl+ (=curr_indent -> ""))* ~} text_interpolation: inline_text_interpolation / ("\" indented_expression blank_lines =curr_indent "..") diff --git a/nomsu.2.peg b/nomsu.2.peg index f6a5d23..58ae33d 100644 --- a/nomsu.2.peg +++ b/nomsu.2.peg @@ -1,6 +1,7 @@ -- Nomsu version 2 file (FileChunks): {:curr_indent: ' '* :} + ("#!" (!"nomsu" [^%nl])* "nomsu" %ws+ "-V" %ws* {:version: ([0-9.]+ -> tonumber) :} [^%nl]*)? comment? blank_lines? (chunk (nl_nodent section_division nl_nodent chunk)*)? blank_lines? @@ -61,13 +62,16 @@ index_chain (IndexChain): -- Actions need either at least 1 word, or at least 2 tokens inline_action (Action): !section_division - ( (inline_expression (%ws* (inline_expression / word))+) - / (word (%ws* (inline_expression / word))*)) + ( ((smushed_action / inline_expression) (%ws* (smushed_action / inline_expression / word))+) + / (word (%ws* (smushed_action / inline_expression / word))*)) (%ws* inline_block)? action (Action): !section_division - ( (expression ((nl_nodent "..")? %ws* (expression / word))+) - / (word ((nl_nodent "..")? %ws* (expression / word))*)) + ( ((smushed_action / expression) ((nl_nodent "..")? %ws* (smushed_action / expression / word))+) + / (word ((nl_nodent "..")? %ws* (smushed_action / expression / word))*)) +smushed_action (Action): + !section_division + (index_chain / noindex_inline_expression / word+) (index_chain / noindex_inline_expression / word+ / "(" %ws* ")")+ word: !number { %operator_char+ / %ident_char+ } @@ -93,20 +97,20 @@ inline_text_interpolation: ) indented_text (Text): - '".."' eol %nl {:curr_indent: indent :} - (indented_plain_text / text_interpolation / {~ blank_lines (=curr_indent -> "") ~})* + '".."' eol %nl {%nl*} {:curr_indent: indent :} + (indented_plain_text / text_interpolation / {~ %nl+ (=curr_indent -> "") ~})* (!! [^%nl]+ -> "Unexpected character while parsing Text" !!)? indented_plain_text (Text): {~ (("\\" -> "\") / (("\" blank_lines =curr_indent "..") -> "") / (!text_interpolation "\") / [^%nl\]+)+ - (blank_lines (=curr_indent -> ""))* ~} + (%nl+ (=curr_indent -> ""))* ~} text_interpolation: inline_text_interpolation / ("\" indented_expression blank_lines =curr_indent "..") number (Number): (("-"? (([0-9]+ "." [0-9]+) / ("." [0-9]+) / ([0-9]+)))-> tonumber) --- Variables can be nameless (i.e. just %) and can't contain operators like apostrophe --- which is a hack to allow %'s to parse as "%" and "' s" separately -variable (Var): "%" {(%ident_char+ ((!"'" %operator_char+) / %ident_char+)*)?} +-- Variables can be nameless (i.e. just %) and can only contain identifier chars. +-- This ensures you don't get weird parsings of `%x+%y` or `%'s thing`. +variable (Var): "%" {%ident_char*} inline_list (List): !('[..]') diff --git a/nomsu.lua b/nomsu.lua index 6080637..0f22eff 100644 --- a/nomsu.lua +++ b/nomsu.lua @@ -271,10 +271,11 @@ run = function() end end if #file_queue == 0 then - nomsu:run([[use "core" + nomsu:run([[#!/usr/bin/env nomsu -V2 +use "core" use "lib/consolecolor.nom" action [quit, exit]: lua> "os.exit(0)" -action [help] +action [help]: say ".." This is the Nomsu v\(Nomsu version) interactive console. You can type in Nomsu code here and hit 'enter' twice to run it. diff --git a/nomsu.moon b/nomsu.moon index cba9eee..c2e5e05 100755 --- a/nomsu.moon +++ b/nomsu.moon @@ -175,10 +175,11 @@ run = -> if #file_queue == 0 -- Run in interactive mode (REPL) nomsu\run [[ +#!/usr/bin/env nomsu -V2 use "core" use "lib/consolecolor.nom" action [quit, exit]: lua> "os.exit(0)" -action [help] +action [help]: say ".." This is the Nomsu v\(Nomsu version) interactive console. You can type in Nomsu code here and hit 'enter' twice to run it. diff --git a/nomsu_compiler.lua b/nomsu_compiler.lua index 57ed5c1..54bdc16 100644 --- a/nomsu_compiler.lua +++ b/nomsu_compiler.lua @@ -722,60 +722,67 @@ do local MIN_COLON_LEN = 20 NomsuCompiler.tree_to_nomsu = function(self, tree, options) options = options or { } - if not (options.pop_comments) then - local comments - do - local _accum_0 = { } - local _len_0 = 1 - for p, c in pairs(tree.comments or { }) do - if tree.source.start <= p and p <= tree.source.stop then - _accum_0[_len_0] = { - comment = c, - pos = p - } - _len_0 = _len_0 + 1 + options.consumed_comments = options.consumed_comments or { } + local pop_comments + pop_comments = function(pos, prefix, suffix) + if prefix == nil then + prefix = '' + end + if suffix == nil then + suffix = '' + end + local find_comments + find_comments = function(t) + if t.comments and t.source.filename == tree.source.filename then + local _list_0 = t.comments + for _index_0 = 1, #_list_0 do + local c = _list_0[_index_0] + if not (options.consumed_comments[c]) then + coroutine.yield(c) + end end end - comments = _accum_0 - end - table.sort(comments, function(a, b) - return a.pos < b.pos - end) - local comment_i = 1 - options.pop_comments = function(pos, prefix) - if prefix == nil then - prefix = '' - end - local nomsu = NomsuCode(tree.source) - while comments[comment_i] and comments[comment_i].pos <= pos do - local comment = comments[comment_i].comment - nomsu:append("#" .. (gsub(comment, "\n", "\n ")) .. "\n") - if comment:match("^\n.") then - nomsu:append("\n") + for _index_0 = 1, #t do + local x = t[_index_0] + if AST.is_syntax_tree(x) then + find_comments(x) end - comment_i = comment_i + 1 end - if #nomsu.bits == 0 then - return '' - end - nomsu:prepend(prefix) - return nomsu end + local nomsu = NomsuCode(tree.source) + for comment in coroutine.wrap(function() + return find_comments(tree) + end) do + if comment.pos > pos then + break + end + options.consumed_comments[comment] = true + nomsu:append("#" .. (gsub(comment.comment, "\n", "\n ")) .. "\n") + if comment.comment:match("^\n.") then + nomsu:append("\n") + end + end + if #nomsu.bits == 0 then + return '' + end + nomsu:prepend(prefix) + nomsu:append(suffix) + return nomsu end local recurse recurse = function(t, opts) opts = opts or { } - opts.pop_comments = options.pop_comments + opts.consumed_comments = options.consumed_comments return self:tree_to_nomsu(t, opts) end - local inline, pop_comments - inline, pop_comments = options.inline, options.pop_comments + local inline + inline = options.inline local _exp_0 = tree.type if "FileChunks" == _exp_0 then if inline then error("Cannot inline a FileChunks") end - local nomsu = NomsuCode(tree.source) + local nomsu = NomsuCode(tree.source, pop_comments(tree.source.start)) for i, chunk in ipairs(tree) do if i > 1 then nomsu:append("\n\n" .. tostring(("~"):rep(80)) .. "\n\n") @@ -785,7 +792,7 @@ do top = true })) end - nomsu:append(pop_comments(tree.source.stop)) + nomsu:append(pop_comments(tree.source.stop, '\n')) return nomsu elseif "Action" == _exp_0 then if inline then @@ -803,7 +810,7 @@ do if not (arg_nomsu) then return nil end - if not (i == 1) then + if not (i == 1 or (bit.type == "Block" and not (#bit > 1 or i < #tree))) then nomsu:append(" ") end if bit.type == "Action" or (bit.type == "Block" and (#bit > 1 or i < #tree)) then @@ -814,9 +821,13 @@ do end return nomsu else - local nomsu = NomsuCode(tree.source) + local pos = tree.source.start + local nomsu = NomsuCode(tree.source, pop_comments(pos)) local next_space = "" for i, bit in ipairs(tree) do + if match(next_space, '\n') then + nomsu:append(pop_comments(pos, '\n')) + end if type(bit) == "string" then nomsu:append(next_space, bit) next_space = " " @@ -853,11 +864,13 @@ do end next_space = "\n.." end + pos = bit.source.stop end if next_space == " " and nomsu.trailing_line_len > MAX_LINE then next_space = "\n.." end end + nomsu:append(pop_comments(pos, '\n')) return nomsu end elseif "EscapedNomsu" == _exp_0 then @@ -885,7 +898,7 @@ do end return nomsu end - local nomsu = NomsuCode(tree.source) + local nomsu = NomsuCode(tree.source, pop_comments(tree.source.start)) for i, line in ipairs(tree) do nomsu:append(pop_comments(line.source.start)) line = assert(recurse(line), "Could not convert line to nomsu") @@ -897,6 +910,7 @@ do end end end + nomsu:append(pop_comments(tree.source.stop, '\n')) return options.top and nomsu or NomsuCode(tree.source, ":\n ", nomsu) elseif "Text" == _exp_0 then if inline then diff --git a/nomsu_compiler.moon b/nomsu_compiler.moon index d1b76f4..7e215dd 100644 --- a/nomsu_compiler.moon +++ b/nomsu_compiler.moon @@ -481,36 +481,40 @@ with NomsuCompiler MIN_COLON_LEN = 20 -- For beautification purposes, don't bother using colon syntax for short bits .tree_to_nomsu = (tree, options)=> options or= {} - unless options.pop_comments - comments = [{comment:c, pos:p} for p,c in pairs(tree.comments or {}) when tree.source.start <= p and p <= tree.source.stop] - table.sort comments, (a,b)-> a.pos < b.pos - comment_i = 1 - options.pop_comments = (pos, prefix='')-> - nomsu = NomsuCode(tree.source) - while comments[comment_i] and comments[comment_i].pos <= pos - comment = comments[comment_i].comment - nomsu\append("#"..(gsub(comment, "\n", "\n ")).."\n") - if comment\match("^\n.") then nomsu\append("\n") -- for aesthetics - comment_i += 1 - return '' if #nomsu.bits == 0 - nomsu\prepend prefix - return nomsu + options.consumed_comments or= {} + pop_comments = (pos, prefix='', suffix='')-> + find_comments = (t)-> + if t.comments and t.source.filename == tree.source.filename + for c in *t.comments + coroutine.yield(c) unless options.consumed_comments[c] + for x in *t + find_comments(x) if AST.is_syntax_tree x + nomsu = NomsuCode(tree.source) + for comment in coroutine.wrap(-> find_comments(tree)) + break if comment.pos > pos + options.consumed_comments[comment] = true + nomsu\append("#"..(gsub(comment.comment, "\n", "\n ")).."\n") + if comment.comment\match("^\n.") then nomsu\append("\n") -- for aesthetics + return '' if #nomsu.bits == 0 + nomsu\prepend prefix + nomsu\append suffix + return nomsu recurse = (t, opts)-> opts or= {} - opts.pop_comments = options.pop_comments + opts.consumed_comments = options.consumed_comments return @tree_to_nomsu(t, opts) - {:inline, :pop_comments} = options + {:inline} = options switch tree.type when "FileChunks" error("Cannot inline a FileChunks") if inline - nomsu = NomsuCode(tree.source) + nomsu = NomsuCode(tree.source, pop_comments(tree.source.start)) for i, chunk in ipairs tree nomsu\append "\n\n#{("~")\rep(80)}\n\n" if i > 1 nomsu\append pop_comments(chunk.source.start) nomsu\append recurse(chunk, top:true) - nomsu\append pop_comments(tree.source.stop) + nomsu\append pop_comments(tree.source.stop, '\n') return nomsu when "Action" @@ -523,16 +527,19 @@ with NomsuCompiler else arg_nomsu = recurse(bit,inline:true) return nil unless arg_nomsu - unless i == 1 + unless i == 1 or (bit.type == "Block" and not (#bit > 1 or i < #tree)) nomsu\append " " if bit.type == "Action" or (bit.type == "Block" and (#bit > 1 or i < #tree)) arg_nomsu\parenthesize! nomsu\append arg_nomsu return nomsu else - nomsu = NomsuCode(tree.source) + pos = tree.source.start + nomsu = NomsuCode(tree.source, pop_comments(pos)) next_space = "" for i,bit in ipairs tree + if match(next_space, '\n') + nomsu\append pop_comments(pos, '\n') if type(bit) == "string" nomsu\append next_space, bit next_space = " " @@ -557,9 +564,11 @@ with NomsuCompiler else nomsu\append arg_nomsu next_space = "\n.." + pos = bit.source.stop if next_space == " " and nomsu.trailing_line_len > MAX_LINE next_space = "\n.." + nomsu\append pop_comments(pos, '\n') return nomsu when "EscapedNomsu" @@ -580,7 +589,7 @@ with NomsuCompiler nomsu\append(i == 1 and " " or "; ") nomsu\append assert(recurse(line, inline:true)) return nomsu - nomsu = NomsuCode(tree.source) + nomsu = NomsuCode(tree.source, pop_comments(tree.source.start)) for i, line in ipairs tree nomsu\append pop_comments(line.source.start) line = assert(recurse(line), "Could not convert line to nomsu") @@ -589,6 +598,7 @@ with NomsuCompiler nomsu\append "\n" if match(tostring(line), "\n") nomsu\append "\n" + nomsu\append pop_comments(tree.source.stop, '\n') return options.top and nomsu or NomsuCode(tree.source, ":\n ", nomsu) when "Text" diff --git a/nomsu_tree.lua b/nomsu_tree.lua index 2e733a2..bbb12ab 100644 --- a/nomsu_tree.lua +++ b/nomsu_tree.lua @@ -53,24 +53,52 @@ for _index_0 = 1, #types do end cls.map = function(self, fn) local replacement = fn(self) - if replacement ~= nil then - return replacement or nil + if replacement == false then + return nil end - local replacements = { } - local changes = false - for i, v in ipairs(self) do - if AST.is_syntax_tree(v) then - replacement = v:map(fn) - else - replacement = v + if replacement then + replacement = (replacement.__class)(self.source, unpack(replacement)) + else + local replacements = { } + local changes = false + for i, v in ipairs(self) do + local _continue_0 = false + repeat + replacements[#replacements + 1] = v + if AST.is_syntax_tree(v) then + local r = v:map(fn) + if r == v or r == nil then + _continue_0 = true + break + end + changes = true + replacements[#replacements] = r + end + _continue_0 = true + until true + if not _continue_0 then + break + end end - changes = changes or (replacement ~= v) - replacements[#replacements + 1] = replacement + if not (changes) then + return self + end + replacement = (self.__class)(self.source, unpack(replacements)) end - if not (changes) then - return self + if self.comments then + do + local _accum_0 = { } + local _len_0 = 1 + local _list_0 = self.comments + for _index_1 = 1, #_list_0 do + local c = _list_0[_index_1] + _accum_0[_len_0] = c + _len_0 = _len_0 + 1 + end + replacement.comments = _accum_0 + end end - return (self.__class)(self.source, unpack(replacements)) + return replacement end end AST[name] = setmetatable(cls, { diff --git a/nomsu_tree.moon b/nomsu_tree.moon index 923bd72..9df240a 100644 --- a/nomsu_tree.moon +++ b/nomsu_tree.moon @@ -22,17 +22,24 @@ for name in *types .__tostring = => "#{@type}(#{repr tostring(@source)}, #{concat([repr(v) for v in *@], ', ')})" .map = (fn)=> replacement = fn(@) - if replacement != nil then return replacement or nil - replacements = {} - changes = false - for i,v in ipairs(@) - replacement = if AST.is_syntax_tree(v) - v\map(fn) - else v - changes or= replacement != v - replacements[#replacements+1] = replacement - return @ unless changes - return (@__class)(@source, unpack(replacements)) + if replacement == false then return nil + if replacement + -- Clone the replacement, but give it a proper source + replacement = (replacement.__class)(@source, unpack(replacement)) + else + replacements = {} + changes = false + for i,v in ipairs(@) + replacements[#replacements+1] = v + if AST.is_syntax_tree(v) + r = v\map(fn) + continue if r == v or r == nil + changes = true + replacements[#replacements] = r + return @ unless changes + replacement = (@__class)(@source, unpack(replacements)) + replacement.comments = [c for c in *@comments] if @comments + return replacement AST[name] = setmetatable cls, __tostring: => @name diff --git a/parser.lua b/parser.lua index 4689e16..5841df1 100644 --- a/parser.lua +++ b/parser.lua @@ -100,7 +100,6 @@ setmetatable(NOMSU_DEFS, { end end setmetatable(value, AST[key]) - value.comments = userdata.comments if value.__init then value:__init() end @@ -190,6 +189,43 @@ Parser.parse = function(nomsu_code, source, version) end error("Errors occurred while parsing (v" .. tostring(version) .. "):\n\n" .. table.concat(errors, "\n\n"), 0) end + local comments + do + local _accum_0 = { } + local _len_0 = 1 + for p, c in pairs(userdata.comments) do + _accum_0[_len_0] = { + comment = c, + pos = p + } + _len_0 = _len_0 + 1 + end + comments = _accum_0 + end + table.sort(comments, function(a, b) + return a.pos > b.pos + end) + local comment_i = 1 + local walk_tree + walk_tree = function(t) + local comment_buff = { } + while comments[#comments] and comments[#comments].pos <= t.source.start do + table.insert(comment_buff, table.remove(comments)) + end + for _index_0 = 1, #t do + local x = t[_index_0] + if AST.is_syntax_tree(x) then + walk_tree(x) + end + end + while comments[#comments] and comments[#comments].pos <= t.source.stop do + table.insert(comment_buff, table.remove(comments)) + end + if #comment_buff > 0 then + t.comments = comment_buff + end + end + walk_tree(tree) tree.version = userdata.version return tree end diff --git a/parser.moon b/parser.moon index 39ad93a..6b719df 100644 --- a/parser.moon +++ b/parser.moon @@ -65,7 +65,6 @@ setmetatable(NOMSU_DEFS, {__index:(key)=> with userdata.source value.source = Source(.filename, .start + start-1, .start + stop-1) setmetatable(value, AST[key]) - value.comments = userdata.comments if value.__init then value\__init! return value @@ -119,7 +118,24 @@ Parser.parse = (nomsu_code, source=nil, version=nil)-> table.sort(keys) errors = [userdata.errors[k] for k in *keys] error("Errors occurred while parsing (v#{version}):\n\n"..table.concat(errors, "\n\n"), 0) - + + comments = [{comment:c, pos:p} for p,c in pairs(userdata.comments)] + -- Sort in descending order so we can pop the first comments off the end one at a time + table.sort comments, (a,b)-> a.pos > b.pos + comment_i = 1 + walk_tree = (t)-> + export comment_i + comment_buff = {} + while comments[#comments] and comments[#comments].pos <= t.source.start + table.insert(comment_buff, table.remove(comments)) + for x in *t + if AST.is_syntax_tree x + walk_tree x + while comments[#comments] and comments[#comments].pos <= t.source.stop + table.insert(comment_buff, table.remove(comments)) + t.comments = comment_buff if #comment_buff > 0 + walk_tree tree + tree.version = userdata.version return tree