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,21 +3,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 immutable = require('immutable')
local Lua, Source local Lua, Source
Source = immutable({ do
"filename", local _class_0
"start", local _base_0 = {
"stop"
}, {
name = "Source",
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) .. "]"
@ -25,6 +14,9 @@ Source = immutable({
return "@" .. tostring(self.filename) .. "[" .. tostring(self.start) .. "]" return "@" .. tostring(self.filename) .. "[" .. tostring(self.start) .. "]"
end end
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) __lt = function(self, other)
assert(self.filename == other.filename, "Cannot compare sources from different files") assert(self.filename == other.filename, "Cannot compare sources from different files")
if self.start == other.start then if self.start == other.start then
@ -51,7 +43,36 @@ Source = immutable({
end end
return Source(self.filename, self.start + offset, self.stop) return Source(self.filename, self.start + offset, self.stop)
end 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+)%]$")
if not (filename) then
filename, start = str:match("^@(.-)%[(%d+)%]$")
end
return self(filename or str, tonumber(start or 1), tonumber(stop))
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,61 +1,39 @@
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)
if type(source) == 'string' then
source = Source:from_string(source)
end end
return source, ... cls.__index = cls
cls.__tostring = function(self)
return tostring(self.name) .. "(#{@value and repr(@value) or table.concat([repr(v) for v in *@]), ', '})"
end end
methods.is_multi = is_multi cls.map = function(self, fn)
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 do
local replacement = fn(self) local replacement = fn(self)
if replacement then if replacement then
return replacement return replacement
end end
end end
if self.value then
return self
end
local new_vals local new_vals
do do
local _accum_0 = { } local _accum_0 = { }
@ -69,62 +47,59 @@ Tree = function(name, fields, methods)
end end
return getmetatable(self)(self.source, unpack(new_vals)) return getmetatable(self)(self.source, unpack(new_vals))
end end
else
methods.__tostring = function(self)
return tostring(self.name) .. "(" .. tostring(repr(self.value)) .. ")"
end end
methods.map = function(self, fn) AST[name] = setmetatable(cls, {
return fn(self) or self __tostring = function(self)
end return self.name
end end,
end __call = function(self, source, ...)
Types[name] = immutable(fields, methods)
end
Tree("Block", {
"source"
})
Tree("EscapedNomsu", {
"source"
})
Tree("Text", {
"source"
})
Tree("List", {
"source"
})
Tree("Dict", {
"source"
})
Tree("DictEntry", {
"source"
})
Tree("IndexChain", {
"source"
})
Tree("Number", {
"source",
"value"
})
Tree("Var", {
"source",
"value"
})
Tree("Action", {
"source",
"stub"
}, {
__new = function(self, source, ...)
assert(source)
if type(source) == 'string' then if type(source) == 'string' then
source = Source:from_string(source) source = Source:from_string(source)
end end
local stub_bits = { } assert(Source:is_instance(source))
for i = 1, select("#", ...) do local inst
local a = select(i, ...) if is_multi then
stub_bits[i] = type(a) == 'string' and a or "%" inst = {
source = source,
...
}
else
inst = {
source = source,
value = ...
}
end end
local stub = concat(stub_bits, " ") setmetatable(inst, self)
return source, stub, ... if inst.__init then
inst:__init()
end
return inst
end
})
end
Tree("Number", 'leaf')
Tree("Var", 'leaf')
Tree("Block", 'branch')
Tree("EscapedNomsu", 'branch')
Tree("Text", 'branch')
Tree("List", 'branch')
Tree("Dict", 'branch')
Tree("DictEntry", 'branch')
Tree("IndexChain", 'branch')
Tree("Action", 'branch', {
__init = function(self)
local stub_bits
do
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1, #self do
local a = self[_index_0]
_accum_0[_len_0] = type(a) == 'string' and a or '%'
_len_0 = _len_0 + 1
end
stub_bits = _accum_0
end
self.stub = concat(stub_bits, " ")
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 *@]), ', '})"
if type(source) == 'string'
source = Source\from_string(source)
--assert Source\is_instance(source)
return source, ...
.is_multi = is_multi
if is_multi
.__tostring = => "#{@name}(#{table.concat [repr(v) for v in *@], ', '})"
.map = (fn)=> .map = (fn)=>
if replacement = fn(@) if replacement = fn(@) then return replacement
return replacement if @value then return @
new_vals = [v.map and v\map(fn) or v for v in *@] new_vals = [v.map and v\map(fn) or v for v in *@]
return getmetatable(self)(@source, unpack(new_vals)) return getmetatable(self)(@source, unpack(new_vals))
else
.__tostring = => "#{@name}(#{repr(@value)})"
.map = (fn)=>
fn(@) or @
Types[name] = immutable fields, methods AST[name] = setmetatable cls,
__tostring: => @name
Tree "Block", {"source"} __call: (source, ...)=>
Tree "EscapedNomsu", {"source"}
Tree "Text", {"source"}
Tree "List", {"source"}
Tree "Dict", {"source"}
Tree "DictEntry", {"source"}
Tree "IndexChain", {"source"}
Tree "Number", {"source", "value"}
Tree "Var", {"source", "value"}
Tree "Action", {"source", "stub"},
__new: (source, ...)=>
assert 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))
stub_bits = {} inst = if is_multi then {:source, ...} else {:source, value:...}
for i=1,select("#",...) setmetatable(inst, @)
a = select(i, ...) if inst.__init then inst\__init!
stub_bits[i] = type(a) == 'string' and a or "%" return inst
stub = concat stub_bits, " "
return source, stub, ... Tree "Number", 'leaf'
Tree "Var", 'leaf'
Tree "Block", 'branch'
Tree "EscapedNomsu", 'branch'
Tree "Text", 'branch'
Tree "List", 'branch'
Tree "Dict", 'branch'
Tree "DictEntry", 'branch'
Tree "IndexChain", 'branch'
Tree "Action", 'branch',
__init: =>
stub_bits = [type(a) == 'string' and a or '%' for a in *@]
@stub = concat stub_bits, " "
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."