Removed dependency on 'immutable' library. This lets LuaJIT do more

aggressive optimizations and generally helps performance. Some safety is
lost, but I think the performance gains, reduced complexity, and reduced
dependencies are worth it.
This commit is contained in:
Bruce Hill 2018-06-12 15:12:27 -07:00
parent 0c9973ff03
commit b5fb8933af
10 changed files with 248 additions and 265 deletions

View File

@ -6,7 +6,7 @@ revolving around natural language rule-making and self modification.
## Dependencies ## Dependencies
Nomsu's dependencies are [Lua 5.2 or later](https://www.lua.org/) (tested with version 5.2.4) (or [Luajit](http://luajit.org/) (tested with version 2.1.0)), [LPEG](http://www.inf.puc-rio.br/~roberto/lpeg/) (`luarocks install lpeg`), and [Lua-Immutable](https://bitbucket.org/spilt/lua-immutable). Nomsu's compiler was written in [Moonscript](http://moonscript.org/), but all of the .moon files have been compiled into lua for convenience, so Moonscript is not a dependency. Nomsu's only dependencies are [Lua 5.2 or later](https://www.lua.org/) (tested with version 5.2.4) (or [Luajit](http://luajit.org/) (tested with version 2.1.0)) and [LPEG](http://www.inf.puc-rio.br/~roberto/lpeg/) (`luarocks install lpeg`). Nomsu's compiler was written in [Moonscript](http://moonscript.org/), but all of the .moon files have been compiled into lua for convenience, so Moonscript is not a dependency.
## Usage ## Usage

View File

@ -3,55 +3,76 @@ 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 immutable = require('immutable')
local Lua, Source local Lua, Source
Source = immutable({ do
"filename", local _class_0
"start", local _base_0 = {
"stop" __tostring = function(self)
}, { if self.stop then
name = "Source", return "@" .. tostring(self.filename) .. "[" .. tostring(self.start) .. ":" .. tostring(self.stop) .. "]"
from_string = function(self, str) else
return "@" .. tostring(self.filename) .. "[" .. tostring(self.start) .. "]"
end
end,
__eq = function(self, other)
return getmetatable(self) == getmetatable(other) and self.filename == other.filename and self.start == other.start and self.stop == other.stop
end,
__lt = function(self, other)
assert(self.filename == other.filename, "Cannot compare sources from different files")
if self.start == other.start then
return (self.stop or self.start) < (other.stop or other.start)
else
return self.start < other.start
end
end,
__le = function(self, other)
assert(self.filename == other.filename, "Cannot compare sources from different files")
if self.start == other.start then
return (self.stop or self.start) <= (other.stop or other.start)
else
return self.start <= other.start
end
end,
__add = function(self, offset)
if type(self) == 'number' then
offset, self = self, offset
else
if type(offset) ~= 'number' then
error("Cannot add Source and " .. tostring(type(offset)))
end
end
return Source(self.filename, self.start + offset, self.stop)
end
}
_base_0.__index = _base_0
_class_0 = setmetatable({
__init = function(self, filename, start, stop)
self.filename, self.start, self.stop = filename, start, stop
end,
__base = _base_0,
__name = "Source"
}, {
__index = _base_0,
__call = function(cls, ...)
local _self_0 = setmetatable({}, _base_0)
cls.__init(_self_0, ...)
return _self_0
end
})
_base_0.__class = _class_0
local self = _class_0
self.from_string = function(self, str)
local filename, start, stop = str:match("^@(.-)%[(%d+):(%d+)%]$") local filename, start, stop = str:match("^@(.-)%[(%d+):(%d+)%]$")
if not (filename) then if not (filename) then
filename, start = str:match("^@(.-)%[(%d+)%]$") filename, start = str:match("^@(.-)%[(%d+)%]$")
end end
return Source(filename or str, tonumber(start or 1), tonumber(stop)) return self(filename or str, tonumber(start or 1), tonumber(stop))
end,
__tostring = function(self)
if self.stop then
return "@" .. tostring(self.filename) .. "[" .. tostring(self.start) .. ":" .. tostring(self.stop) .. "]"
else
return "@" .. tostring(self.filename) .. "[" .. tostring(self.start) .. "]"
end
end,
__lt = function(self, other)
assert(self.filename == other.filename, "Cannot compare sources from different files")
if self.start == other.start then
return (self.stop or self.start) < (other.stop or other.start)
else
return self.start < other.start
end
end,
__le = function(self, other)
assert(self.filename == other.filename, "Cannot compare sources from different files")
if self.start == other.start then
return (self.stop or self.start) <= (other.stop or other.start)
else
return self.start <= other.start
end
end,
__add = function(self, offset)
if type(self) == 'number' then
offset, self = self, offset
else
if type(offset) ~= 'number' then
error("Cannot add Source and " .. tostring(type(offset)))
end
end
return Source(self.filename, self.start + offset, self.stop)
end end
}) self.is_instance = function(self, x)
return type(x) == 'table' and x.__class == self
end
Source = _class_0
end
local Code local Code
do do
local _class_0 local _class_0
@ -150,7 +171,7 @@ do
end end
for _index_0 = 1, #vars do for _index_0 = 1, #vars do
local var = vars[_index_0] local var = vars[_index_0]
assert(type(var) == 'userdata' and var.type == "Var") assert(var.type == "Var")
if not (seen[var]) then if not (seen[var]) then
self.free_vars[#self.free_vars + 1] = var self.free_vars[#self.free_vars + 1] = var
seen[var] = true seen[var] = true
@ -165,7 +186,7 @@ do
local removals = { } local removals = { }
for _index_0 = 1, #vars do for _index_0 = 1, #vars do
local var = vars[_index_0] local var = vars[_index_0]
assert(type(var) == 'userdata' and var.type == "Var") assert(var.type == "Var")
removals[var.value] = true removals[var.value] = true
end end
local stack = { local stack = {

View File

@ -2,38 +2,46 @@
-- build up generated code, while keeping track of where it came from, and managing -- build up generated code, while keeping track of where it came from, and managing
-- indentation levels. -- indentation levels.
{:insert, :remove, :concat} = table {:insert, :remove, :concat} = table
immutable = require 'immutable'
local Lua, Source local Lua, Source
export LINE_STARTS export LINE_STARTS
Source = immutable {"filename","start","stop"}, { class Source
name:"Source" new: (@filename, @start, @stop)=>
from_string: (str)=>
@from_string: (str)=>
filename,start,stop = str\match("^@(.-)%[(%d+):(%d+)%]$") filename,start,stop = str\match("^@(.-)%[(%d+):(%d+)%]$")
unless filename unless filename
filename,start = str\match("^@(.-)%[(%d+)%]$") filename,start = str\match("^@(.-)%[(%d+)%]$")
return Source(filename or str, tonumber(start or 1), tonumber(stop)) return @(filename or str, tonumber(start or 1), tonumber(stop))
@is_instance: (x)=> type(x) == 'table' and x.__class == @
__tostring: => __tostring: =>
if @stop if @stop
"@#{@filename}[#{@start}:#{@stop}]" "@#{@filename}[#{@start}:#{@stop}]"
else else
"@#{@filename}[#{@start}]" "@#{@filename}[#{@start}]"
__eq: (other)=>
getmetatable(@) == getmetatable(other) and @filename == other.filename and @start == other.start and @stop == other.stop
__lt: (other)=> __lt: (other)=>
assert(@filename == other.filename, "Cannot compare sources from different files") assert(@filename == other.filename, "Cannot compare sources from different files")
return if @start == other.start return if @start == other.start
(@stop or @start) < (other.stop or other.start) (@stop or @start) < (other.stop or other.start)
else @start < other.start else @start < other.start
__le: (other)=> __le: (other)=>
assert(@filename == other.filename, "Cannot compare sources from different files") assert(@filename == other.filename, "Cannot compare sources from different files")
return if @start == other.start return if @start == other.start
(@stop or @start) <= (other.stop or other.start) (@stop or @start) <= (other.stop or other.start)
else @start <= other.start else @start <= other.start
__add: (offset)=> __add: (offset)=>
if type(self) == 'number' if type(self) == 'number'
offset, self = self, offset offset, self = self, offset
else if type(offset) != 'number' then error("Cannot add Source and #{type(offset)}") else if type(offset) != 'number' then error("Cannot add Source and #{type(offset)}")
return Source(@filename, @start+offset, @stop) return Source(@filename, @start+offset, @stop)
}
class Code class Code
new: (@source, ...)=> new: (@source, ...)=>
@ -90,7 +98,7 @@ class Lua extends Code
return unless #vars > 0 return unless #vars > 0
seen = {[v]:true for v in *@free_vars} seen = {[v]:true for v in *@free_vars}
for var in *vars for var in *vars
assert(type(var) == 'userdata' and var.type == "Var") assert(var.type == "Var")
unless seen[var] unless seen[var]
@free_vars[#@free_vars+1] = var @free_vars[#@free_vars+1] = var
seen[var] = true seen[var] = true
@ -100,7 +108,7 @@ class Lua extends Code
return unless #vars > 0 return unless #vars > 0
removals = {} removals = {}
for var in *vars for var in *vars
assert(type(var) == 'userdata' and var.type == "Var") assert(var.type == "Var")
removals[var.value] = true removals[var.value] = true
stack = {self} stack = {self}

View File

@ -94,14 +94,14 @@ immediately
return replacements[t.value] return replacements[t.value]
elseif t.type == 'Var' then elseif t.type == 'Var' then
return t.type.."("..repr(tostring(t.source))..", "..repr(t.value.."#"..tostring(MANGLE_INDEX))..")" return t.type.."("..repr(tostring(t.source))..", "..repr(t.value.."#"..tostring(MANGLE_INDEX))..")"
elseif t.is_multi then elseif t.value then
return t.type.."("..repr(tostring(t.source))..", "..repr(t.value)..")"
else
local bits = {repr(tostring(t.source))} local bits = {repr(tostring(t.source))}
for i, entry in ipairs(t) do for i, entry in ipairs(t) do
bits[#bits+1] = make_tree(entry) bits[#bits+1] = make_tree(entry)
end end
return t.type.."("..table.concat(bits, ", ")..")" return t.type.."("..table.concat(bits, ", ")..")"
else
return t.type.."("..repr(tostring(t.source))..", "..repr(t.value)..")"
end end
end end
lua:append(")\n local tree = ", make_tree(\%longhand), "\n return nomsu:tree_to_lua(tree)\nend);") lua:append(")\n local tree = ", make_tree(\%longhand), "\n return nomsu:tree_to_lua(tree)\nend);")

View File

@ -34,10 +34,6 @@ local P, R, V, S, Cg, C, Cp, B, Cmt, Carg
P, R, V, S, Cg, C, Cp, B, Cmt, Carg = lpeg.P, lpeg.R, lpeg.V, lpeg.S, lpeg.Cg, lpeg.C, lpeg.Cp, lpeg.B, lpeg.Cmt, lpeg.Carg P, R, V, S, Cg, C, Cp, B, Cmt, Carg = lpeg.P, lpeg.R, lpeg.V, lpeg.S, lpeg.Cg, lpeg.C, lpeg.Cp, lpeg.B, lpeg.Cmt, lpeg.Carg
local utils = require('utils') local utils = require('utils')
local new_uuid = require('uuid') local new_uuid = require('uuid')
local immutable = require('immutable')
Tuple = immutable(nil, {
name = "Tuple"
})
local repr, stringify, min, max, equivalent, set, is_list, sum local repr, stringify, min, max, equivalent, set, is_list, sum
repr, stringify, min, max, equivalent, set, is_list, sum = utils.repr, utils.stringify, utils.min, utils.max, utils.equivalent, utils.set, utils.is_list, utils.sum repr, stringify, min, max, equivalent, set, is_list, sum = utils.repr, utils.stringify, utils.min, utils.max, utils.equivalent, utils.set, utils.is_list, utils.sum
local colors = setmetatable({ }, { local colors = setmetatable({ }, {
@ -163,7 +159,7 @@ do
end end
end end
end end
local Types = require("nomsu_tree") local AST = require("nomsu_tree")
local NOMSU_DEFS local NOMSU_DEFS
do do
local _with_0 = { } local _with_0 = { }
@ -249,13 +245,12 @@ setmetatable(NOMSU_DEFS, {
local _with_0 = userdata.source local _with_0 = userdata.source
source = Source(_with_0.filename, _with_0.start + start - 1, _with_0.start + stop - 1) source = Source(_with_0.filename, _with_0.start + start - 1, _with_0.start + stop - 1)
end end
local tree value.source = source
if Types[key].is_multi then setmetatable(value, AST[key])
tree = Types[key](source, unpack(value)) if value.__init then
else value:__init()
tree = Types[key](source, value)
end end
return tree return value
end end
self[key] = make_node self[key] = make_node
return make_node return make_node
@ -444,7 +439,7 @@ do
run_lua = function(self, lua) run_lua = function(self, lua)
assert(type(lua) ~= 'string', "Attempt to run lua string instead of Lua (object)") assert(type(lua) ~= 'string', "Attempt to run lua string instead of Lua (object)")
local lua_string = tostring(lua) local lua_string = tostring(lua)
local run_lua_fn, err = load(lua_string, tostring(lua.source), "t", self.environment) local run_lua_fn, err = load(lua_string, nil and tostring(lua.source), "t", self.environment)
if not run_lua_fn then if not run_lua_fn then
local n = 1 local n = 1
local fn local fn
@ -592,10 +587,12 @@ do
elseif "EscapedNomsu" == _exp_0 then elseif "EscapedNomsu" == _exp_0 then
local make_tree local make_tree
make_tree = function(t) make_tree = function(t)
if type(t) ~= 'userdata' then if not (AST.is_syntax_tree(t)) then
return repr(t) return repr(t)
end end
if t.is_multi then if t.value then
return t.type .. "(" .. repr(tostring(t.source)) .. ", " .. repr(t.value) .. ")"
else
local bits local bits
do do
local _accum_0 = { } local _accum_0 = { }
@ -608,8 +605,6 @@ do
bits = _accum_0 bits = _accum_0
end end
return t.type .. "(" .. repr(tostring(t.source)) .. ", " .. table.concat(bits, ", ") .. ")" return t.type .. "(" .. repr(tostring(t.source)) .. ", " .. table.concat(bits, ", ") .. ")"
else
return t.type .. "(" .. repr(tostring(t.source)) .. ", " .. repr(t.value) .. ")"
end end
end end
return Lua.Value(tree.source, make_tree(tree[1])) return Lua.Value(tree.source, make_tree(tree[1]))
@ -1327,10 +1322,9 @@ do
end end
return _pairs(x) return _pairs(x)
end end
for k, v in pairs(Types) do for k, v in pairs(AST) do
self.environment[k] = v self.environment[k] = v
end end
self.environment.Tuple = Tuple
self.environment.Lua = Lua self.environment.Lua = Lua
self.environment.Nomsu = Nomsu self.environment.Nomsu = Nomsu
self.environment.Source = Source self.environment.Source = Source
@ -1346,7 +1340,7 @@ do
__mode = "k" __mode = "k"
}) })
self.environment.LOADED = { } self.environment.LOADED = { }
self.environment.Types = Types self.environment.AST = AST
return self:initialize_core() return self:initialize_core()
end, end,
__base = _base_0, __base = _base_0,

View File

@ -40,9 +40,6 @@ lpeg.setmaxstack 10000
{:P,:R,:V,:S,:Cg,:C,:Cp,:B,:Cmt,:Carg} = lpeg {:P,:R,:V,:S,:Cg,:C,:Cp,:B,:Cmt,:Carg} = lpeg
utils = require 'utils' utils = require 'utils'
new_uuid = require 'uuid' new_uuid = require 'uuid'
immutable = require 'immutable'
export Tuple
Tuple = immutable(nil, {name:"Tuple"})
{:repr, :stringify, :min, :max, :equivalent, :set, :is_list, :sum} = utils {:repr, :stringify, :min, :max, :equivalent, :set, :is_list, :sum} = utils
colors = setmetatable({}, {__index:->""}) colors = setmetatable({}, {__index:->""})
export colored export colored
@ -134,7 +131,7 @@ do
if type(i) == 'number' then return string.sub(@, i, i) if type(i) == 'number' then return string.sub(@, i, i)
elseif type(i) == 'table' then return string.sub(@, i[1], i[2]) elseif type(i) == 'table' then return string.sub(@, i[1], i[2])
Types = require "nomsu_tree" AST = require "nomsu_tree"
NOMSU_DEFS = with {} NOMSU_DEFS = with {}
-- Newline supports either windows-style CR+LF or unix-style LF -- Newline supports either windows-style CR+LF or unix-style LF
@ -205,10 +202,11 @@ setmetatable(NOMSU_DEFS, {__index:(key)=>
local source local source
with userdata.source with userdata.source
source = Source(.filename, .start + start-1, .start + stop-1) source = Source(.filename, .start + start-1, .start + stop-1)
tree = if Types[key].is_multi value.source = source
Types[key](source, unpack(value)) setmetatable(value, AST[key])
else Types[key](source, value) if value.__init then value\__init!
return tree return value
self[key] = make_node self[key] = make_node
return make_node return make_node
}) })
@ -254,6 +252,8 @@ class NomsuCompiler
@[key] = id @[key] = id
return id return id
}) })
-- Mapping from source string (e.g. "@core/metaprogramming.nom[1:100]") to a mapping
-- from lua line number to nomsu line number
@source_map = {} @source_map = {}
_list_mt = _list_mt =
@ -304,8 +304,7 @@ class NomsuCompiler
if mt.__pairs if mt.__pairs
return mt.__pairs(x) return mt.__pairs(x)
return _pairs(x) return _pairs(x)
for k,v in pairs(Types) do @environment[k] = v for k,v in pairs(AST) do @environment[k] = v
@environment.Tuple = Tuple
@environment.Lua = Lua @environment.Lua = Lua
@environment.Nomsu = Nomsu @environment.Nomsu = Nomsu
@environment.Source = Source @environment.Source = Source
@ -316,7 +315,7 @@ class NomsuCompiler
@environment.COMPILE_ACTIONS = {} @environment.COMPILE_ACTIONS = {}
@environment.ARG_ORDERS = setmetatable({}, {__mode:"k"}) @environment.ARG_ORDERS = setmetatable({}, {__mode:"k"})
@environment.LOADED = {} @environment.LOADED = {}
@environment.Types = Types @environment.AST = AST
@initialize_core! @initialize_core!
local stub_defs local stub_defs
@ -427,7 +426,7 @@ class NomsuCompiler
run_lua: (lua)=> run_lua: (lua)=>
assert(type(lua) != 'string', "Attempt to run lua string instead of Lua (object)") assert(type(lua) != 'string', "Attempt to run lua string instead of Lua (object)")
lua_string = tostring(lua) lua_string = tostring(lua)
run_lua_fn, err = load(lua_string, tostring(lua.source), "t", @environment) run_lua_fn, err = load(lua_string, nil and tostring(lua.source), "t", @environment)
if not run_lua_fn if not run_lua_fn
n = 1 n = 1
fn = -> fn = ->
@ -525,13 +524,13 @@ class NomsuCompiler
when "EscapedNomsu" when "EscapedNomsu"
make_tree = (t)-> make_tree = (t)->
if type(t) != 'userdata' unless AST.is_syntax_tree(t)
return repr(t) return repr(t)
if t.is_multi if t.value
return t.type.."("..repr(tostring t.source)..", "..repr(t.value)..")"
else
bits = [make_tree(bit) for bit in *t] bits = [make_tree(bit) for bit in *t]
return t.type.."("..repr(tostring t.source)..", "..table.concat(bits, ", ")..")" return t.type.."("..repr(tostring t.source)..", "..table.concat(bits, ", ")..")"
else
return t.type.."("..repr(tostring t.source)..", "..repr(t.value)..")"
Lua.Value tree.source, make_tree(tree[1]) Lua.Value tree.source, make_tree(tree[1])
when "Block" when "Block"

View File

@ -98,11 +98,11 @@ text_interpolation:
inline_text_interpolation / inline_text_interpolation /
("\" indented_expression nodent "..") ("\" indented_expression nodent "..")
number (Number): (("-"? (([0-9]+ "." [0-9]+) / ("." [0-9]+) / ([0-9]+)))-> tonumber) number (Number): {| {:value: (("-"? (([0-9]+ "." [0-9]+) / ("." [0-9]+) / ([0-9]+)))-> tonumber) :} |}
-- Variables can be nameless (i.e. just %) and can't contain operators like apostrophe -- 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 -- which is a hack to allow %'s to parse as "%" and "' s" separately
variable (Var): "%" { (%ident_char+ ((!"'" %operator_char+) / %ident_char+)*)? } variable (Var): "%" {| {:value: (%ident_char+ ((!"'" %operator_char+) / %ident_char+)*)? :} |}
inline_list (List): inline_list (List):
!('[..]') !('[..]')

View File

@ -1,130 +1,105 @@
local utils = require('utils') local repr
local repr, stringify, min, max, equivalent, set, is_list, sum repr = require('utils').repr
repr, stringify, min, max, equivalent, set, is_list, sum = utils.repr, utils.stringify, utils.min, utils.max, utils.equivalent, utils.set, utils.is_list, utils.sum
local immutable = require('immutable')
local insert, remove, concat local insert, remove, concat
do 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, Source local Source
do Source = require("code_obj").Source
local _obj_0 = require("code_obj") local AST = { }
Lua, Nomsu, Source = _obj_0.Lua, _obj_0.Nomsu, _obj_0.Source AST.is_syntax_tree = function(n)
end return type(n) == 'table' and getmetatable(n) and AST[n.type] == getmetatable(n)
local MAX_LINE = 80
local Types = { }
Types.is_node = function(n)
return type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
end end
local Tree local Tree
Tree = function(name, fields, methods) Tree = function(name, leaf_or_branch, methods)
methods = methods or { } local cls = methods or { }
local is_multi = true local is_multi = leaf_or_branch == 'branch'
for _index_0 = 1, #fields do
local f = fields[_index_0]
is_multi = is_multi and (f ~= "value")
end
do do
methods.type = name cls.type = name
methods.name = name cls.is_instance = function(self, x)
methods.__new = methods.__new or function(self, source, ...) return getmetatable(x) == self
assert(source) end
cls.__index = cls
cls.__tostring = function(self)
return tostring(self.name) .. "(#{@value and repr(@value) or table.concat([repr(v) for v in *@]), ', '})"
end
cls.map = function(self, fn)
do
local replacement = fn(self)
if replacement then
return replacement
end
end
if self.value then
return self
end
local new_vals
do
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1, #self do
local v = self[_index_0]
_accum_0[_len_0] = v.map and v:map(fn) or v
_len_0 = _len_0 + 1
end
new_vals = _accum_0
end
return getmetatable(self)(self.source, unpack(new_vals))
end
end
AST[name] = setmetatable(cls, {
__tostring = function(self)
return self.name
end,
__call = function(self, source, ...)
if type(source) == 'string' then if type(source) == 'string' then
source = Source:from_string(source) source = Source:from_string(source)
end end
return source, ... assert(Source:is_instance(source))
local inst
if is_multi then
inst = {
source = source,
...
}
else
inst = {
source = source,
value = ...
}
end
setmetatable(inst, self)
if inst.__init then
inst:__init()
end
return inst
end end
methods.is_multi = is_multi })
if is_multi then
methods.__tostring = function(self)
return tostring(self.name) .. "(" .. tostring(table.concat((function()
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1, #self do
local v = self[_index_0]
_accum_0[_len_0] = repr(v)
_len_0 = _len_0 + 1
end
return _accum_0
end)(), ', ')) .. ")"
end
methods.map = function(self, fn)
do
local replacement = fn(self)
if replacement then
return replacement
end
end
local new_vals
do
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1, #self do
local v = self[_index_0]
_accum_0[_len_0] = v.map and v:map(fn) or v
_len_0 = _len_0 + 1
end
new_vals = _accum_0
end
return getmetatable(self)(self.source, unpack(new_vals))
end
else
methods.__tostring = function(self)
return tostring(self.name) .. "(" .. tostring(repr(self.value)) .. ")"
end
methods.map = function(self, fn)
return fn(self) or self
end
end
end
Types[name] = immutable(fields, methods)
end end
Tree("Block", { Tree("Number", 'leaf')
"source" Tree("Var", 'leaf')
}) Tree("Block", 'branch')
Tree("EscapedNomsu", { Tree("EscapedNomsu", 'branch')
"source" Tree("Text", 'branch')
}) Tree("List", 'branch')
Tree("Text", { Tree("Dict", 'branch')
"source" Tree("DictEntry", 'branch')
}) Tree("IndexChain", 'branch')
Tree("List", { Tree("Action", 'branch', {
"source" __init = function(self)
}) local stub_bits
Tree("Dict", { do
"source" local _accum_0 = { }
}) local _len_0 = 1
Tree("DictEntry", { for _index_0 = 1, #self do
"source" local a = self[_index_0]
}) _accum_0[_len_0] = type(a) == 'string' and a or '%'
Tree("IndexChain", { _len_0 = _len_0 + 1
"source" end
}) stub_bits = _accum_0
Tree("Number", {
"source",
"value"
})
Tree("Var", {
"source",
"value"
})
Tree("Action", {
"source",
"stub"
}, {
__new = function(self, source, ...)
assert(source)
if type(source) == 'string' then
source = Source:from_string(source)
end end
local stub_bits = { } self.stub = concat(stub_bits, " ")
for i = 1, select("#", ...) do
local a = select(i, ...)
stub_bits[i] = type(a) == 'string' and a or "%"
end
local stub = concat(stub_bits, " ")
return source, stub, ...
end, end,
get_spec = function(self) get_spec = function(self)
return concat((function() return concat((function()
@ -139,4 +114,4 @@ Tree("Action", {
end)(), " ") end)(), " ")
end end
}) })
return Types return AST

View File

@ -1,69 +1,53 @@
-- This file contains the datastructures used to represent parsed Nomsu syntax trees, -- This file contains the datastructures used to represent parsed Nomsu syntax trees,
-- as well as the logic for converting them to Lua code. -- as well as the logic for converting them to Lua code.
utils = require 'utils' {:repr} = require 'utils'
{:repr, :stringify, :min, :max, :equivalent, :set, :is_list, :sum} = utils
immutable = require 'immutable'
{:insert, :remove, :concat} = table {:insert, :remove, :concat} = table
{:Lua, :Nomsu, :Source} = require "code_obj" {:Source} = require "code_obj"
MAX_LINE = 80 -- For beautification purposes, try not to make lines much longer than this value AST = {}
AST.is_syntax_tree = (n)->
Types = {} type(n) == 'table' and getmetatable(n) and AST[n.type] == getmetatable(n)
Types.is_node = (n)->
type(n) == 'userdata' and getmetatable(n) and Types[n.type] == getmetatable(n)
-- Helper method: -- Helper method:
Tree = (name, fields, methods)-> Tree = (name, leaf_or_branch, methods)->
methods or= {} cls = methods or {}
is_multi = true is_multi = leaf_or_branch == 'branch'
for f in *fields do is_multi and= (f != "value") with cls
with methods
.type = name .type = name
.name = name .is_instance = (x)=> getmetatable(x) == @
.__new or= (source, ...)=> .__index = cls
assert source .__tostring = => "#{@name}(#{@value and repr(@value) or table.concat([repr(v) for v in *@]), ', '})"
.map = (fn)=>
if replacement = fn(@) then return replacement
if @value then return @
new_vals = [v.map and v\map(fn) or v for v in *@]
return getmetatable(self)(@source, unpack(new_vals))
AST[name] = setmetatable cls,
__tostring: => @name
__call: (source, ...)=>
if type(source) == 'string' if type(source) == 'string'
source = Source\from_string(source) source = Source\from_string(source)
--assert Source\is_instance(source) assert(Source\is_instance(source))
return source, ... inst = if is_multi then {:source, ...} else {:source, value:...}
.is_multi = is_multi setmetatable(inst, @)
if is_multi if inst.__init then inst\__init!
.__tostring = => "#{@name}(#{table.concat [repr(v) for v in *@], ', '})" return inst
.map = (fn)=>
if replacement = fn(@)
return replacement
new_vals = [v.map and v\map(fn) or v for v in *@]
return getmetatable(self)(@source, unpack(new_vals))
else
.__tostring = => "#{@name}(#{repr(@value)})"
.map = (fn)=>
fn(@) or @
Types[name] = immutable fields, methods Tree "Number", 'leaf'
Tree "Var", 'leaf'
Tree "Block", {"source"} Tree "Block", 'branch'
Tree "EscapedNomsu", {"source"} Tree "EscapedNomsu", 'branch'
Tree "Text", {"source"} Tree "Text", 'branch'
Tree "List", {"source"} Tree "List", 'branch'
Tree "Dict", {"source"} Tree "Dict", 'branch'
Tree "DictEntry", {"source"} Tree "DictEntry", 'branch'
Tree "IndexChain", {"source"} Tree "IndexChain", 'branch'
Tree "Number", {"source", "value"} Tree "Action", 'branch',
Tree "Var", {"source", "value"} __init: =>
stub_bits = [type(a) == 'string' and a or '%' for a in *@]
Tree "Action", {"source", "stub"}, @stub = concat stub_bits, " "
__new: (source, ...)=>
assert source
if type(source) == 'string'
source = Source\from_string(source)
--assert Source\is_instance(source)
stub_bits = {}
for i=1,select("#",...)
a = select(i, ...)
stub_bits[i] = type(a) == 'string' and a or "%"
stub = concat stub_bits, " "
return source, stub, ...
get_spec: => get_spec: =>
concat [type(a) == "string" and a or "%#{a.value}" for a in *@], " " concat [type(a) == "string" and a or "%#{a.value}" for a in *@], " "
return Types return AST

View File

@ -55,7 +55,9 @@ try: foo 99
assume ((\(5 + 5) as value) = 10) or barf "%tree as value failed." assume ((\(5 + 5) as value) = 10) or barf "%tree as value failed."
assume ("\(\(foo %x) as nomsu)" = "foo %x") or barf "source code failed." assume ("\(\(foo %x) as nomsu)" = "foo %x") or barf "action source code failed."
assume ("\(\%x as nomsu)" = "%x") or barf "var source code failed."
assume ((type of {}) = "table") or barf "type of failed." assume ((type of {}) = "table") or barf "type of failed."