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:
parent
0c9973ff03
commit
b5fb8933af
@ -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
|
||||||
|
|
||||||
|
111
code_obj.lua
111
code_obj.lua
@ -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 = {
|
||||||
|
@ -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}
|
||||||
|
@ -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);")
|
||||||
|
32
nomsu.lua
32
nomsu.lua
@ -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,
|
||||||
|
31
nomsu.moon
31
nomsu.moon
@ -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"
|
||||||
|
@ -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):
|
||||||
!('[..]')
|
!('[..]')
|
||||||
|
203
nomsu_tree.lua
203
nomsu_tree.lua
@ -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
|
||||||
|
@ -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
|
||||||
|
@ -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."
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user