Moving back to capturing tables in LPEG and everything is a Source, not
string.
This commit is contained in:
parent
8cb2788e0d
commit
0c7c06beab
22
code_obj.lua
22
code_obj.lua
@ -20,6 +20,13 @@ Source = immutable({
|
|||||||
end
|
end
|
||||||
return filename, start, stop
|
return filename, start, stop
|
||||||
end,
|
end,
|
||||||
|
from_string = function(self, str)
|
||||||
|
local filename, start, stop = str:match("^(.-)%[(%d+):(%d+)%]$")
|
||||||
|
if not (filename) then
|
||||||
|
filename, start = str:match("^(.-)%[(%d+)%]$")
|
||||||
|
end
|
||||||
|
return Source(filename or str, tonumber(start or 1), tonumber(stop))
|
||||||
|
end,
|
||||||
__tostring = function(self)
|
__tostring = function(self)
|
||||||
if self.stop then
|
if self.stop then
|
||||||
return "\"" .. tostring(self.filename) .. "[" .. tostring(self.start) .. ":" .. tostring(self.stop) .. "]\""
|
return "\"" .. tostring(self.filename) .. "[" .. tostring(self.start) .. ":" .. tostring(self.stop) .. "]\""
|
||||||
@ -94,11 +101,6 @@ local Code
|
|||||||
do
|
do
|
||||||
local _class_0
|
local _class_0
|
||||||
local _base_0 = {
|
local _base_0 = {
|
||||||
sub = function(self, start, stop)
|
|
||||||
local str = tostring(self):sub(start, stop)
|
|
||||||
local cls = self.__class
|
|
||||||
return cls(self.source:sub(start, stop), str)
|
|
||||||
end,
|
|
||||||
append = function(self, ...)
|
append = function(self, ...)
|
||||||
local n = select("#", ...)
|
local n = select("#", ...)
|
||||||
local bits, indents = self.bits, self.indents
|
local bits, indents = self.bits, self.indents
|
||||||
@ -156,15 +158,7 @@ do
|
|||||||
self.bits, self.indents, self.current_indent = { }, { }, 0
|
self.bits, self.indents, self.current_indent = { }, { }, 0
|
||||||
self:append(...)
|
self:append(...)
|
||||||
if type(self.source) == 'string' then
|
if type(self.source) == 'string' then
|
||||||
local filename, start, stop = self.source:match("^(.-)%[(%d+):(%d+)%]$")
|
self.source = Source:from_string(self.source)
|
||||||
if not (filename) then
|
|
||||||
filename, start = self.source:match("^(.-)%[(%d+)%]$")
|
|
||||||
end
|
|
||||||
if start or stop then
|
|
||||||
self.source = Source(filename, tonumber(start), tonumber(stop))
|
|
||||||
else
|
|
||||||
self.source = Source(self.source, 1, #tostring(self) + 1)
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
return assert(self.source)
|
return assert(self.source)
|
||||||
end,
|
end,
|
||||||
|
@ -10,6 +10,11 @@ Source = immutable {"filename","start","stop"}, {
|
|||||||
start, stop = 1, #FILE_CACHE[filename]
|
start, stop = 1, #FILE_CACHE[filename]
|
||||||
if stop then assert(start <= stop+1, "Invalid range: #{start}, #{stop}")
|
if stop then assert(start <= stop+1, "Invalid range: #{start}, #{stop}")
|
||||||
return filename, start, stop
|
return filename, start, stop
|
||||||
|
from_string: (str)=>
|
||||||
|
filename,start,stop = str\match("^(.-)%[(%d+):(%d+)%]$")
|
||||||
|
unless filename
|
||||||
|
filename,start = str\match("^(.-)%[(%d+)%]$")
|
||||||
|
return Source(filename or str, tonumber(start or 1), tonumber(stop))
|
||||||
__tostring: =>
|
__tostring: =>
|
||||||
if @stop
|
if @stop
|
||||||
"\"#{@filename}[#{@start}:#{@stop}]\""
|
"\"#{@filename}[#{@start}:#{@stop}]\""
|
||||||
@ -65,21 +70,9 @@ class Code
|
|||||||
@bits, @indents, @current_indent = {}, {}, 0
|
@bits, @indents, @current_indent = {}, {}, 0
|
||||||
@append(...)
|
@append(...)
|
||||||
if type(@source) == 'string'
|
if type(@source) == 'string'
|
||||||
filename,start,stop = @source\match("^(.-)%[(%d+):(%d+)%]$")
|
@source = Source\from_string(@source)
|
||||||
unless filename
|
|
||||||
filename,start = @source\match("^(.-)%[(%d+)%]$")
|
|
||||||
if start or stop
|
|
||||||
@source = Source(filename, tonumber(start), tonumber(stop))
|
|
||||||
else
|
|
||||||
@source = Source(@source, 1, #tostring(self)+1)
|
|
||||||
assert(@source)
|
assert(@source)
|
||||||
|
|
||||||
sub: (start,stop)=>
|
|
||||||
-- TODO: implement this better
|
|
||||||
str = tostring(self)\sub(start,stop)
|
|
||||||
cls = @__class
|
|
||||||
return cls(@source\sub(start,stop), str)
|
|
||||||
|
|
||||||
append: (...)=>
|
append: (...)=>
|
||||||
n = select("#",...)
|
n = select("#",...)
|
||||||
bits, indents = @bits, @indents
|
bits, indents = @bits, @indents
|
||||||
|
15
nomsu.lua
15
nomsu.lua
@ -211,14 +211,14 @@ do
|
|||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
local err_pos = start_pos
|
local err_pos = start_pos
|
||||||
local text_loc = lpeg.userdata.source_code.source:sub(err_pos, err_pos)
|
local text_loc = lpeg.userdata.source:sub(err_pos, err_pos)
|
||||||
local line_no = text_loc:get_line_number()
|
local line_no = text_loc:get_line_number()
|
||||||
src = FILE_CACHE[text_loc.filename]
|
src = FILE_CACHE[text_loc.filename]
|
||||||
local prev_line = line_no == 1 and "" or src:sub(LINE_STARTS[src][line_no - 1] or 1, LINE_STARTS[src][line_no] - 2)
|
local prev_line = line_no == 1 and "" or src:sub(LINE_STARTS[src][line_no - 1] or 1, LINE_STARTS[src][line_no] - 2)
|
||||||
local err_line = src:sub(LINE_STARTS[src][line_no], (LINE_STARTS[src][line_no + 1] or 0) - 2)
|
local err_line = src:sub(LINE_STARTS[src][line_no], (LINE_STARTS[src][line_no + 1] or 0) - 2)
|
||||||
local next_line = src:sub(LINE_STARTS[src][line_no + 1] or -1, (LINE_STARTS[src][line_no + 2] or 0) - 2)
|
local next_line = src:sub(LINE_STARTS[src][line_no + 1] or -1, (LINE_STARTS[src][line_no + 2] or 0) - 2)
|
||||||
local pointer = ("-"):rep(err_pos - LINE_STARTS[src][line_no]) .. "^"
|
local pointer = ("-"):rep(err_pos - LINE_STARTS[src][line_no]) .. "^"
|
||||||
err_msg = (err_msg or "Parse error") .. " at " .. tostring(lpeg.userdata.source_code.source.filename) .. ":" .. tostring(line_no) .. ":\n"
|
err_msg = (err_msg or "Parse error") .. " at " .. tostring(lpeg.userdata.source.filename) .. ":" .. tostring(line_no) .. ":\n"
|
||||||
if #prev_line > 0 then
|
if #prev_line > 0 then
|
||||||
err_msg = err_msg .. ("\n" .. prev_line)
|
err_msg = err_msg .. ("\n" .. prev_line)
|
||||||
end
|
end
|
||||||
@ -234,18 +234,13 @@ end
|
|||||||
setmetatable(NOMSU_DEFS, {
|
setmetatable(NOMSU_DEFS, {
|
||||||
__index = function(self, key)
|
__index = function(self, key)
|
||||||
local make_node
|
local make_node
|
||||||
make_node = function(start, ...)
|
make_node = function(start, value, stop)
|
||||||
local args = {
|
|
||||||
...
|
|
||||||
}
|
|
||||||
local stop = args[#args]
|
|
||||||
local source = lpeg.userdata.source:sub(start, stop)
|
local source = lpeg.userdata.source:sub(start, stop)
|
||||||
args[#args] = nil
|
|
||||||
local tree
|
local tree
|
||||||
if Types[key].is_multi then
|
if Types[key].is_multi then
|
||||||
tree = Types[key](Tuple(unpack(args)), source)
|
tree = Types[key](Tuple(unpack(value)), source)
|
||||||
else
|
else
|
||||||
tree = Types[key](args[1], source)
|
tree = Types[key](value, source)
|
||||||
end
|
end
|
||||||
return tree
|
return tree
|
||||||
end
|
end
|
||||||
|
13
nomsu.moon
13
nomsu.moon
@ -184,14 +184,14 @@ NOMSU_DEFS = with {}
|
|||||||
err_pos = start_pos
|
err_pos = start_pos
|
||||||
--if src\sub(err_pos,err_pos)\match("[\r\n]")
|
--if src\sub(err_pos,err_pos)\match("[\r\n]")
|
||||||
-- err_pos += #src\match("[ \t\n\r]*", err_pos)
|
-- err_pos += #src\match("[ \t\n\r]*", err_pos)
|
||||||
text_loc = lpeg.userdata.source_code.source\sub(err_pos,err_pos)
|
text_loc = lpeg.userdata.source\sub(err_pos,err_pos)
|
||||||
line_no = text_loc\get_line_number!
|
line_no = text_loc\get_line_number!
|
||||||
src = FILE_CACHE[text_loc.filename]
|
src = FILE_CACHE[text_loc.filename]
|
||||||
prev_line = line_no == 1 and "" or src\sub(LINE_STARTS[src][line_no-1] or 1, LINE_STARTS[src][line_no]-2)
|
prev_line = line_no == 1 and "" or src\sub(LINE_STARTS[src][line_no-1] or 1, LINE_STARTS[src][line_no]-2)
|
||||||
err_line = src\sub(LINE_STARTS[src][line_no], (LINE_STARTS[src][line_no+1] or 0)-2)
|
err_line = src\sub(LINE_STARTS[src][line_no], (LINE_STARTS[src][line_no+1] or 0)-2)
|
||||||
next_line = src\sub(LINE_STARTS[src][line_no+1] or -1, (LINE_STARTS[src][line_no+2] or 0)-2)
|
next_line = src\sub(LINE_STARTS[src][line_no+1] or -1, (LINE_STARTS[src][line_no+2] or 0)-2)
|
||||||
pointer = ("-")\rep(err_pos-LINE_STARTS[src][line_no]) .. "^"
|
pointer = ("-")\rep(err_pos-LINE_STARTS[src][line_no]) .. "^"
|
||||||
err_msg = (err_msg or "Parse error").." at #{lpeg.userdata.source_code.source.filename}:#{line_no}:\n"
|
err_msg = (err_msg or "Parse error").." at #{lpeg.userdata.source.filename}:#{line_no}:\n"
|
||||||
if #prev_line > 0 then err_msg ..= "\n"..prev_line
|
if #prev_line > 0 then err_msg ..= "\n"..prev_line
|
||||||
err_msg ..= "\n#{err_line}\n#{pointer}"
|
err_msg ..= "\n#{err_line}\n#{pointer}"
|
||||||
if #next_line > 0 then err_msg ..= "\n"..next_line
|
if #next_line > 0 then err_msg ..= "\n"..next_line
|
||||||
@ -200,14 +200,11 @@ NOMSU_DEFS = with {}
|
|||||||
return true
|
return true
|
||||||
|
|
||||||
setmetatable(NOMSU_DEFS, {__index:(key)=>
|
setmetatable(NOMSU_DEFS, {__index:(key)=>
|
||||||
make_node = (start, ...)->
|
make_node = (start, value, stop)->
|
||||||
args = {...}
|
|
||||||
stop = args[#args]
|
|
||||||
source = lpeg.userdata.source\sub(start, stop)
|
source = lpeg.userdata.source\sub(start, stop)
|
||||||
args[#args] = nil
|
|
||||||
tree = if Types[key].is_multi
|
tree = if Types[key].is_multi
|
||||||
Types[key](Tuple(unpack(args)), source)
|
Types[key](Tuple(unpack(value)), source)
|
||||||
else Types[key](args[1], source)
|
else Types[key](value, source)
|
||||||
return tree
|
return tree
|
||||||
self[key] = make_node
|
self[key] = make_node
|
||||||
return make_node
|
return make_node
|
||||||
|
34
nomsu.peg
34
nomsu.peg
@ -8,9 +8,9 @@ file:
|
|||||||
shebang: "#!" [^%nl]* (!. / %nl)
|
shebang: "#!" [^%nl]* (!. / %nl)
|
||||||
|
|
||||||
inline_block (Block):
|
inline_block (Block):
|
||||||
inline_statement (%ws* ";" %ws* inline_statement)+
|
{| inline_statement (%ws* ";" %ws* inline_statement)+ |}
|
||||||
block (Block):
|
block (Block):
|
||||||
statement (nodent (statement / (({} ([^%nl]* -> "Error while parsing block line")) => error)))+
|
{| statement (nodent (statement / (({} ([^%nl]* -> "Error while parsing block line")) => error)))+ |}
|
||||||
|
|
||||||
statement: (action / expression) (eol / (({} ([^%nl]* -> "Error while parsing line")) => error))
|
statement: (action / expression) (eol / (({} ([^%nl]* -> "Error while parsing line")) => error))
|
||||||
inline_statement: inline_action / inline_expression
|
inline_statement: inline_action / inline_expression
|
||||||
@ -35,19 +35,21 @@ indented_expression:
|
|||||||
expression:
|
expression:
|
||||||
inline_expression / (":" %ws* (inline_block / inline_action / inline_expression) eol) / indented_expression
|
inline_expression / (":" %ws* (inline_block / inline_action / inline_expression) eol) / indented_expression
|
||||||
|
|
||||||
inline_nomsu (EscapedNomsu): "\" inline_expression
|
inline_nomsu (EscapedNomsu): "\" {| inline_expression |}
|
||||||
indented_nomsu (EscapedNomsu):
|
indented_nomsu (EscapedNomsu):
|
||||||
"\" (noindex_inline_expression / (":" %ws* (inline_block / inline_action / inline_expression) eol) / indented_expression)
|
"\" {| (noindex_inline_expression / (":" %ws* (inline_block / inline_action / inline_expression) eol) / indented_expression) |}
|
||||||
|
|
||||||
index_chain (IndexChain):
|
index_chain (IndexChain):
|
||||||
noindex_inline_expression ("." (text_word / noindex_inline_expression))+
|
{| noindex_inline_expression ("." (text_word / noindex_inline_expression))+ |}
|
||||||
|
|
||||||
-- Actions need at least one word in them
|
-- Actions need at least one word in them
|
||||||
inline_action (Action):
|
inline_action (Action):
|
||||||
|
{|
|
||||||
(inline_expression %ws*)* word (%ws* (inline_expression / word))*
|
(inline_expression %ws*)* word (%ws* (inline_expression / word))*
|
||||||
(%ws* ":" %ws* (inline_block / inline_action / inline_expression))?
|
(%ws* ":" %ws* (inline_block / inline_action / inline_expression))?
|
||||||
|
|}
|
||||||
action (Action):
|
action (Action):
|
||||||
(expression (dotdot? %ws*))* word ((dotdot? %ws*) (expression / word))*
|
{| (expression (dotdot? %ws*))* word ((dotdot? %ws*) (expression / word))* |}
|
||||||
|
|
||||||
word: { %operator / (!number plain_word) }
|
word: { %operator / (!number plain_word) }
|
||||||
|
|
||||||
@ -55,19 +57,19 @@ text_word (Text): {%operator / (!number plain_word)}
|
|||||||
|
|
||||||
inline_text (Text):
|
inline_text (Text):
|
||||||
!('".."' eol)
|
!('".."' eol)
|
||||||
'"' (
|
'"' {|
|
||||||
({~ (('\"' -> '"') / ('\\' -> '\') / %escaped_char / [^%nl\"])+ ~}
|
({~ (('\"' -> '"') / ('\\' -> '\') / %escaped_char / [^%nl\"])+ ~}
|
||||||
/ inline_text_interpolation)*
|
/ inline_text_interpolation)*
|
||||||
) ('"' / (({} ([^%nl]*->'Failed to find a closing " mark on the same line')) => error))
|
|} ('"' / (({} ([^%nl]*->'Failed to find a closing " mark on the same line')) => error))
|
||||||
|
|
||||||
-- Have to use "%indent" instead of "indent" etc. to avoid messing up text lines that start with "#"
|
-- Have to use "%indent" instead of "indent" etc. to avoid messing up text lines that start with "#"
|
||||||
indented_text (Text):
|
indented_text (Text):
|
||||||
'".."' eol %nl (
|
'".."' eol %nl {|
|
||||||
{~ (%nl*) (%indent -> "") ~}
|
{~ (%nl*) (%indent -> "") ~}
|
||||||
({~
|
({~
|
||||||
(("\\" -> "\") / (("\" nodent "..") -> "")/ (%nl+ {~ %nodent -> "" ~}) / [^%nl\] / (!text_interpolation "\"))+
|
(("\\" -> "\") / (("\" nodent "..") -> "")/ (%nl+ {~ %nodent -> "" ~}) / [^%nl\] / (!text_interpolation "\"))+
|
||||||
~} / text_interpolation)*
|
~} / text_interpolation)*
|
||||||
) (((!.) &%dedent) / (&(%nl %dedent)) / (({} (non_dedent_error -> "Error while parsing Text")) => error))
|
|} (((!.) &%dedent) / (&(%nl %dedent)) / (({} (non_dedent_error -> "Error while parsing Text")) => error))
|
||||||
inline_text_interpolation:
|
inline_text_interpolation:
|
||||||
"\" (
|
"\" (
|
||||||
variable / inline_list / inline_dict / inline_text
|
variable / inline_list / inline_dict / inline_text
|
||||||
@ -90,11 +92,11 @@ variable (Var): "%" { ((!"'" %operator) / plain_word)* }
|
|||||||
|
|
||||||
inline_list (List):
|
inline_list (List):
|
||||||
!('[..]')
|
!('[..]')
|
||||||
"[" %ws* (inline_list_item (comma inline_list_item)* comma?)? %ws*
|
"[" %ws* {| (inline_list_item (comma inline_list_item)* comma?)? |} %ws*
|
||||||
("]" / (({} ([^%nl]*->"Failed to find a closing ] on the same line")) => error))
|
("]" / (({} ([^%nl]*->"Failed to find a closing ] on the same line")) => error))
|
||||||
indented_list (List):
|
indented_list (List):
|
||||||
"[..]" indent
|
"[..]" indent
|
||||||
list_line (nodent list_line)*
|
{| list_line (nodent list_line)* |}
|
||||||
(dedent / (({} (non_dedent_error -> "Error while parsing list")) => error))
|
(dedent / (({} (non_dedent_error -> "Error while parsing list")) => error))
|
||||||
list_line:
|
list_line:
|
||||||
((action / expression) !comma)
|
((action / expression) !comma)
|
||||||
@ -103,20 +105,20 @@ inline_list_item: inline_block / inline_action / inline_expression
|
|||||||
|
|
||||||
inline_dict (Dict):
|
inline_dict (Dict):
|
||||||
!('{..}')
|
!('{..}')
|
||||||
"{" %ws* (inline_dict_entry (comma inline_dict_entry)*)? %ws*
|
"{" %ws* {| (inline_dict_entry (comma inline_dict_entry)*)? |} %ws*
|
||||||
("}"
|
("}"
|
||||||
/ (({} (%ws* comma? (!. / &%nl)->"Failed to find a closing } on the same line")) => error)
|
/ (({} (%ws* comma? (!. / &%nl)->"Failed to find a closing } on the same line")) => error)
|
||||||
/ (({} ([^%nl]*->"Error while parsing dictionary")) => error))
|
/ (({} ([^%nl]*->"Error while parsing dictionary")) => error))
|
||||||
indented_dict (Dict):
|
indented_dict (Dict):
|
||||||
"{..}" indent
|
"{..}" indent
|
||||||
dict_line (nodent dict_line)*
|
{| dict_line (nodent dict_line)* |}
|
||||||
(dedent / (({} (non_dedent_error -> "Error while parsing dict")) => error))
|
(dedent / (({} (non_dedent_error -> "Error while parsing dict")) => error))
|
||||||
dict_line:
|
dict_line:
|
||||||
(dict_entry !comma) / (inline_dict_entry (comma dict_line?)?)
|
(dict_entry !comma) / (inline_dict_entry (comma dict_line?)?)
|
||||||
dict_entry(DictEntry):
|
dict_entry(DictEntry):
|
||||||
dict_key %ws* ":" %ws* (action / expression)
|
{| dict_key %ws* ":" %ws* (action / expression) |}
|
||||||
inline_dict_entry(DictEntry):
|
inline_dict_entry(DictEntry):
|
||||||
dict_key %ws* (":" %ws* (inline_block / inline_action / inline_expression)?)
|
{| dict_key %ws* (":" %ws* (inline_block / inline_action / inline_expression)?) |}
|
||||||
dict_key:
|
dict_key:
|
||||||
text_word / inline_expression
|
text_word / inline_expression
|
||||||
|
|
||||||
|
@ -7,10 +7,10 @@ do
|
|||||||
local _obj_0 = table
|
local _obj_0 = table
|
||||||
insert, remove, concat = _obj_0.insert, _obj_0.remove, _obj_0.concat
|
insert, remove, concat = _obj_0.insert, _obj_0.remove, _obj_0.concat
|
||||||
end
|
end
|
||||||
local Lua, Nomsu, Location
|
local Lua, Nomsu, Source
|
||||||
do
|
do
|
||||||
local _obj_0 = require("code_obj")
|
local _obj_0 = require("code_obj")
|
||||||
Lua, Nomsu, Location = _obj_0.Lua, _obj_0.Nomsu, _obj_0.Location
|
Lua, Nomsu, Source = _obj_0.Lua, _obj_0.Nomsu, _obj_0.Source
|
||||||
end
|
end
|
||||||
local MAX_LINE = 80
|
local MAX_LINE = 80
|
||||||
local Types = { }
|
local Types = { }
|
||||||
@ -27,6 +27,9 @@ Tree = function(name, kind, methods)
|
|||||||
methods.name = name
|
methods.name = name
|
||||||
methods.__new = function(self, value, source)
|
methods.__new = function(self, value, source)
|
||||||
assert(source)
|
assert(source)
|
||||||
|
if type(source) == 'string' then
|
||||||
|
source = Source:from_string(source)
|
||||||
|
end
|
||||||
return value, source
|
return value, source
|
||||||
end
|
end
|
||||||
methods.is_multi = is_multi
|
methods.is_multi = is_multi
|
||||||
|
@ -4,7 +4,7 @@ utils = require 'utils'
|
|||||||
{:repr, :stringify, :min, :max, :equivalent, :set, :is_list, :sum} = utils
|
{:repr, :stringify, :min, :max, :equivalent, :set, :is_list, :sum} = utils
|
||||||
immutable = require 'immutable'
|
immutable = require 'immutable'
|
||||||
{:insert, :remove, :concat} = table
|
{:insert, :remove, :concat} = table
|
||||||
{:Lua, :Nomsu, :Location} = require "code_obj"
|
{:Lua, :Nomsu, :Source} = require "code_obj"
|
||||||
|
|
||||||
MAX_LINE = 80 -- For beautification purposes, try not to make lines much longer than this value
|
MAX_LINE = 80 -- For beautification purposes, try not to make lines much longer than this value
|
||||||
|
|
||||||
@ -22,6 +22,8 @@ Tree = (name, kind, methods)->
|
|||||||
.name = name
|
.name = name
|
||||||
.__new = (value, source)=>
|
.__new = (value, source)=>
|
||||||
assert source
|
assert source
|
||||||
|
if type(source) == 'string'
|
||||||
|
source = Source\from_string(source)
|
||||||
return value, source
|
return value, source
|
||||||
.is_multi = is_multi
|
.is_multi = is_multi
|
||||||
.map = (fn)=>
|
.map = (fn)=>
|
||||||
|
Loading…
Reference in New Issue
Block a user