Added list/dict metatables to make comparison and string representations
simpler. Also deleted Counters.
This commit is contained in:
parent
2d88c68d71
commit
810ae220bc
@ -134,10 +134,6 @@ immediately
|
|||||||
self[\(%key as lua expr)] = value
|
self[\(%key as lua expr)] = value
|
||||||
return value
|
return value
|
||||||
end})
|
end})
|
||||||
|
|
||||||
immediately
|
|
||||||
parse [new counter] as: {} with fallback % -> 0
|
|
||||||
parse [new default dict] as: {} with fallback % -> {}
|
|
||||||
|
|
||||||
# Sorting
|
# Sorting
|
||||||
immediately
|
immediately
|
||||||
|
@ -132,7 +132,7 @@ immediately
|
|||||||
lua> ".."
|
lua> ".."
|
||||||
local lua = nomsu:tree_to_lua(\%tree)
|
local lua = nomsu:tree_to_lua(\%tree)
|
||||||
if not lua.is_value then
|
if not lua.is_value then
|
||||||
error("Invalid thing to convert to lua expr: "..\%tree)
|
error("Invalid thing to convert to lua expr: "..tostring(\%tree))
|
||||||
end
|
end
|
||||||
return lua
|
return lua
|
||||||
|
|
||||||
@ -164,8 +164,9 @@ immediately
|
|||||||
parse [to %var write %code] as: lua> "\%var:append(\%code);"
|
parse [to %var write %code] as: lua> "\%var:append(\%code);"
|
||||||
|
|
||||||
immediately
|
immediately
|
||||||
compile [repr %obj] to: Lua value "repr(\(%obj as lua expr))"
|
compile [quote %s] to
|
||||||
compile [%obj as text] to: Lua value "tostring(\(%obj as lua expr))"
|
Lua value ".."
|
||||||
|
('"'..\(%s as lua expr):gsub("\\\\", "\\\\\\\\"):gsub("\n","\\\\n"):gsub('"', '\\\\"')..'"')
|
||||||
compile [type of %obj] to: Lua value "type(\(%obj as lua expr))"
|
compile [type of %obj] to: Lua value "type(\(%obj as lua expr))"
|
||||||
|
|
||||||
immediately
|
immediately
|
||||||
@ -175,7 +176,7 @@ immediately
|
|||||||
# Compiler tools
|
# Compiler tools
|
||||||
immediately
|
immediately
|
||||||
compile [run %code] to
|
compile [run %code] to
|
||||||
Lua "nomsu:run(Nomsu(\"\(%code.source as text)\", \(%code as lua expr)))"
|
Lua value "nomsu:run(Nomsu(\(quote "\(%code.source)"), \(%code as lua expr)))"
|
||||||
|
|
||||||
immediately
|
immediately
|
||||||
compile [show lua %block] to
|
compile [show lua %block] to
|
||||||
@ -189,7 +190,7 @@ immediately
|
|||||||
if \%message.type == "Text" then
|
if \%message.type == "Text" then
|
||||||
return Lua(tree.source, "print(", \(%message as lua expr), ");");
|
return Lua(tree.source, "print(", \(%message as lua expr), ");");
|
||||||
else
|
else
|
||||||
return Lua(tree.source, "print(stringify(", \(%message as lua expr), "));");
|
return Lua(tree.source, "print(tostring(", \(%message as lua expr), "));");
|
||||||
end
|
end
|
||||||
|
|
||||||
# Return
|
# Return
|
||||||
@ -208,7 +209,7 @@ immediately
|
|||||||
return
|
return
|
||||||
Lua ".."
|
Lua ".."
|
||||||
if not \(%condition as lua expr) then
|
if not \(%condition as lua expr) then
|
||||||
error(\(repr %assumption), 0);
|
error(\(quote "\%assumption"), 0);
|
||||||
end
|
end
|
||||||
|
|
||||||
compile [assume %condition or barf %message] to
|
compile [assume %condition or barf %message] to
|
||||||
|
@ -23,23 +23,9 @@ immediately
|
|||||||
compile [%x <= %y] to: Lua value "(\(%x as lua expr) <= \(%y as lua expr))"
|
compile [%x <= %y] to: Lua value "(\(%x as lua expr) <= \(%y as lua expr))"
|
||||||
compile [%x >= %y] to: Lua value "(\(%x as lua expr) >= \(%y as lua expr))"
|
compile [%x >= %y] to: Lua value "(\(%x as lua expr) >= \(%y as lua expr))"
|
||||||
compile [%a is %b, %a = %b, %a == %b] to
|
compile [%a is %b, %a = %b, %a == %b] to
|
||||||
lua> ".."
|
Lua value "(\(%a as lua expr) == \(%b as lua expr))"
|
||||||
local safe = {Text=true, Number=true}
|
|
||||||
local a_lua, b_lua = \(%a as lua), \(%b as lua)
|
|
||||||
if safe[\%a.type] or safe[\%b.type] then
|
|
||||||
return Lua.Value(tree.source, "(", a_lua, " == ", b_lua, ")")
|
|
||||||
else
|
|
||||||
return Lua.Value(tree.source, "utils.equivalent(", a_lua, ", ", b_lua, ")")
|
|
||||||
end
|
|
||||||
compile [%a isn't %b, %a is not %b, %a not= %b, %a != %b] to
|
compile [%a isn't %b, %a is not %b, %a not= %b, %a != %b] to
|
||||||
lua> ".."
|
Lua value "(\(%a as lua expr) ~= \(%b as lua expr))"
|
||||||
local safe = {Text=true, Number=true}
|
|
||||||
local a_lua, b_lua = \(%a as lua), \(%b as lua)
|
|
||||||
if safe[\%a.type] or safe[\%b.type] then
|
|
||||||
return Lua.Value(tree.source, "(", a_lua, " ~= ", b_lua, ")")
|
|
||||||
else
|
|
||||||
return Lua.Value(tree.source, "(not utils.equivalent(", a_lua, ", ", b_lua, "))")
|
|
||||||
end
|
|
||||||
# For strict identity checking, use (%x's id) is (%y's id)
|
# For strict identity checking, use (%x's id) is (%y's id)
|
||||||
compile [%'s id, id of %] to: Lua value "nomsu.ids[\(% as lua expr)]"
|
compile [%'s id, id of %] to: Lua value "nomsu.ids[\(% as lua expr)]"
|
||||||
|
|
||||||
|
51
nomsu.lua
51
nomsu.lua
@ -677,7 +677,7 @@ do
|
|||||||
end
|
end
|
||||||
return lua
|
return lua
|
||||||
elseif "List" == _exp_0 then
|
elseif "List" == _exp_0 then
|
||||||
local lua = Lua.Value(tree.source, "{")
|
local lua = Lua.Value(tree.source, "list{")
|
||||||
local line_length = 0
|
local line_length = 0
|
||||||
for i, item in ipairs(tree) do
|
for i, item in ipairs(tree) do
|
||||||
local item_lua = self:tree_to_lua(item)
|
local item_lua = self:tree_to_lua(item)
|
||||||
@ -707,7 +707,7 @@ do
|
|||||||
lua:append("}")
|
lua:append("}")
|
||||||
return lua
|
return lua
|
||||||
elseif "Dict" == _exp_0 then
|
elseif "Dict" == _exp_0 then
|
||||||
local lua = Lua.Value(tree.source, "{")
|
local lua = Lua.Value(tree.source, "dict{")
|
||||||
local line_length = 0
|
local line_length = 0
|
||||||
for i, entry in ipairs(tree) do
|
for i, entry in ipairs(tree) do
|
||||||
local entry_lua = self:tree_to_lua(entry)
|
local entry_lua = self:tree_to_lua(entry)
|
||||||
@ -1216,6 +1216,43 @@ do
|
|||||||
end
|
end
|
||||||
})
|
})
|
||||||
self.source_map = { }
|
self.source_map = { }
|
||||||
|
local _list_mt = {
|
||||||
|
__eq = utils.equivalent,
|
||||||
|
__tostring = function(self)
|
||||||
|
return "[" .. concat((function()
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
for _index_0 = 1, #self do
|
||||||
|
local b = self[_index_0]
|
||||||
|
_accum_0[_len_0] = repr(b)
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
return _accum_0
|
||||||
|
end)(), ", ") .. "]"
|
||||||
|
end
|
||||||
|
}
|
||||||
|
local list
|
||||||
|
list = function(t)
|
||||||
|
return setmetatable(t, _list_mt)
|
||||||
|
end
|
||||||
|
local _dict_mt = {
|
||||||
|
__eq = utils.equivalent,
|
||||||
|
__tostring = function(self)
|
||||||
|
return "{" .. concat((function()
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
for k, v in pairs(self) do
|
||||||
|
_accum_0[_len_0] = tostring(repr(k)) .. ": " .. tostring(repr(v))
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
return _accum_0
|
||||||
|
end)(), ", ") .. "}"
|
||||||
|
end
|
||||||
|
}
|
||||||
|
local dict
|
||||||
|
dict = function(t)
|
||||||
|
return setmetatable(t, _dict_mt)
|
||||||
|
end
|
||||||
self.environment = {
|
self.environment = {
|
||||||
nomsu = self,
|
nomsu = self,
|
||||||
repr = repr,
|
repr = repr,
|
||||||
@ -1256,9 +1293,9 @@ do
|
|||||||
debug = debug,
|
debug = debug,
|
||||||
math = math,
|
math = math,
|
||||||
io = io,
|
io = io,
|
||||||
pairs = pairs,
|
|
||||||
load = load,
|
load = load,
|
||||||
ipairs = ipairs
|
list = list,
|
||||||
|
dict = dict
|
||||||
}
|
}
|
||||||
if jit then
|
if jit then
|
||||||
self.environment.len = function(x)
|
self.environment.len = function(x)
|
||||||
@ -1289,11 +1326,10 @@ do
|
|||||||
if mt.__ipairs then
|
if mt.__ipairs then
|
||||||
return mt.__ipairs(x)
|
return mt.__ipairs(x)
|
||||||
end
|
end
|
||||||
else
|
|
||||||
return _ipairs(x)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
return _ipairs(x)
|
||||||
end
|
end
|
||||||
self.environment.pairs = function(x)
|
self.environment.pairs = function(x)
|
||||||
if type(x) == 'function' then
|
if type(x) == 'function' then
|
||||||
@ -1307,11 +1343,10 @@ do
|
|||||||
if mt.__pairs then
|
if mt.__pairs then
|
||||||
return mt.__pairs(x)
|
return mt.__pairs(x)
|
||||||
end
|
end
|
||||||
else
|
|
||||||
return _pairs(x)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
return _pairs(x)
|
||||||
end
|
end
|
||||||
for k, v in pairs(Types) do
|
for k, v in pairs(Types) do
|
||||||
self.environment[k] = v
|
self.environment[k] = v
|
||||||
|
24
nomsu.moon
24
nomsu.moon
@ -245,6 +245,17 @@ class NomsuCompiler
|
|||||||
})
|
})
|
||||||
@source_map = {}
|
@source_map = {}
|
||||||
|
|
||||||
|
_list_mt =
|
||||||
|
__eq:utils.equivalent
|
||||||
|
-- Could consider adding a __newindex to enforce list-ness, but would hurt performance
|
||||||
|
__tostring: =>
|
||||||
|
"["..concat([repr(b) for b in *@], ", ").."]"
|
||||||
|
list = (t)-> setmetatable(t, _list_mt)
|
||||||
|
_dict_mt =
|
||||||
|
__eq:utils.equivalent
|
||||||
|
__tostring: =>
|
||||||
|
"{"..concat(["#{repr(k)}: #{repr(v)}" for k,v in pairs @], ", ").."}"
|
||||||
|
dict = (t)-> setmetatable(t, _dict_mt)
|
||||||
@environment = {
|
@environment = {
|
||||||
-- Discretionary/convenience stuff
|
-- Discretionary/convenience stuff
|
||||||
nomsu:self, repr:repr, stringify:stringify, utils:utils, lpeg:lpeg, re:re,
|
nomsu:self, repr:repr, stringify:stringify, utils:utils, lpeg:lpeg, re:re,
|
||||||
@ -252,8 +263,9 @@ class NomsuCompiler
|
|||||||
:next, :unpack, :setmetatable, :coroutine, :rawequal, :getmetatable, :pcall,
|
:next, :unpack, :setmetatable, :coroutine, :rawequal, :getmetatable, :pcall,
|
||||||
:error, :package, :os, :require, :tonumber, :tostring, :string, :xpcall, :module,
|
:error, :package, :os, :require, :tonumber, :tostring, :string, :xpcall, :module,
|
||||||
:print, :loadfile, :rawset, :_VERSION, :collectgarbage, :rawget, :bit32, :rawlen,
|
:print, :loadfile, :rawset, :_VERSION, :collectgarbage, :rawget, :bit32, :rawlen,
|
||||||
:table, :assert, :dofile, :loadstring, :type, :select, :debug, :math, :io, :pairs,
|
:table, :assert, :dofile, :loadstring, :type, :select, :debug, :math, :io, :load,
|
||||||
:load, :ipairs,
|
-- Nomsu types:
|
||||||
|
:list, :dict,
|
||||||
}
|
}
|
||||||
@environment.len = if jit
|
@environment.len = if jit
|
||||||
(x)->
|
(x)->
|
||||||
@ -270,7 +282,7 @@ class NomsuCompiler
|
|||||||
elseif mt = getmetatable(x)
|
elseif mt = getmetatable(x)
|
||||||
if mt.__ipairs
|
if mt.__ipairs
|
||||||
return mt.__ipairs(x)
|
return mt.__ipairs(x)
|
||||||
else return _ipairs(x)
|
return _ipairs(x)
|
||||||
@environment.pairs = (x)->
|
@environment.pairs = (x)->
|
||||||
if type(x) == 'function'
|
if type(x) == 'function'
|
||||||
return coroutine.wrap(x)
|
return coroutine.wrap(x)
|
||||||
@ -279,7 +291,7 @@ class NomsuCompiler
|
|||||||
elseif mt = getmetatable(x)
|
elseif mt = getmetatable(x)
|
||||||
if mt.__pairs
|
if mt.__pairs
|
||||||
return mt.__pairs(x)
|
return mt.__pairs(x)
|
||||||
else return _pairs(x)
|
return _pairs(x)
|
||||||
for k,v in pairs(Types) do @environment[k] = v
|
for k,v in pairs(Types) do @environment[k] = v
|
||||||
@environment.Tuple = Tuple
|
@environment.Tuple = Tuple
|
||||||
@environment.Lua = Lua
|
@environment.Lua = Lua
|
||||||
@ -558,7 +570,7 @@ class NomsuCompiler
|
|||||||
return lua
|
return lua
|
||||||
|
|
||||||
when "List"
|
when "List"
|
||||||
lua = Lua.Value tree.source, "{"
|
lua = Lua.Value tree.source, "list{"
|
||||||
line_length = 0
|
line_length = 0
|
||||||
for i, item in ipairs tree
|
for i, item in ipairs tree
|
||||||
item_lua = @tree_to_lua(item)
|
item_lua = @tree_to_lua(item)
|
||||||
@ -583,7 +595,7 @@ class NomsuCompiler
|
|||||||
return lua
|
return lua
|
||||||
|
|
||||||
when "Dict"
|
when "Dict"
|
||||||
lua = Lua.Value tree.source, "{"
|
lua = Lua.Value tree.source, "dict{"
|
||||||
line_length = 0
|
line_length = 0
|
||||||
for i, entry in ipairs tree
|
for i, entry in ipairs tree
|
||||||
entry_lua = @tree_to_lua(entry)
|
entry_lua = @tree_to_lua(entry)
|
||||||
|
@ -37,9 +37,5 @@ assume (%x = [3,2,1])
|
|||||||
sort %x by % = %keys.%
|
sort %x by % = %keys.%
|
||||||
assume (%x = [2,3,1])
|
assume (%x = [2,3,1])
|
||||||
assume ((unique [1,2,1,3,2,3]) = [1,2,3])
|
assume ((unique [1,2,1,3,2,3]) = [1,2,3])
|
||||||
%c <- (new counter)
|
|
||||||
for % in ["x","y","x","x","y"]
|
|
||||||
%c.% +<- 1
|
|
||||||
assume (%c = {x:3,y:2})
|
|
||||||
|
|
||||||
say "Collections test passed."
|
say "Collections test passed."
|
||||||
|
@ -55,15 +55,13 @@ 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) as text) = "foo %x") or barf "source code failed."
|
assume ("\(\(foo %x) as nomsu)" = "foo %x") or barf "source code failed."
|
||||||
|
|
||||||
assume ((repr [1,2]) = "{1, 2}") or barf "repr failed."
|
|
||||||
|
|
||||||
assume ((type of {}) = "table") or barf "type of failed."
|
assume ((type of {}) = "table") or barf "type of failed."
|
||||||
|
|
||||||
assume ((nomsu) = (=lua "nomsu")) or barf "nomsu failed"
|
assume ((nomsu) = (=lua "nomsu")) or barf "nomsu failed"
|
||||||
|
|
||||||
assume (((\%x as lua identifier) as text) = "_x") or barf "converting to identifier failed."
|
assume ("\(\%x as lua identifier)" = "_x") or barf "converting to identifier failed."
|
||||||
|
|
||||||
assume ((run "return 99") = 99) or barf "run % failed."
|
assume ((run "return 99") = 99) or barf "run % failed."
|
||||||
|
|
||||||
|
@ -30,7 +30,7 @@ assume (%s = "one two\\nthreefour")
|
|||||||
list:\[..]
|
list:\[..]
|
||||||
1,2,3
|
1,2,3
|
||||||
..
|
..
|
||||||
assume (%s = "list:{1, 2, 3}")
|
assume (%s = "list:[1, 2, 3]")
|
||||||
|
|
||||||
assume
|
assume
|
||||||
".."
|
".."
|
||||||
|
15
utils.lua
15
utils.lua
@ -241,30 +241,29 @@ local function sort(list, keyFn, reverse)
|
|||||||
end
|
end
|
||||||
|
|
||||||
local function equivalent(x, y, depth)
|
local function equivalent(x, y, depth)
|
||||||
depth = depth or -1
|
depth = depth or 0
|
||||||
if x == y then
|
if rawequal(x, y) then
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
if type(x) ~= type(y) then
|
if type(x) ~= type(y) then
|
||||||
return false
|
return false
|
||||||
end
|
end
|
||||||
if type(x) ~= 'table' then
|
if type(x) ~= 'table' then return false end
|
||||||
|
if getmetatable(x) ~= getmetatable(y) then
|
||||||
return false
|
return false
|
||||||
end
|
end
|
||||||
if depth == 0 then
|
if depth >= 99 then
|
||||||
return false
|
|
||||||
elseif depth < -999 then
|
|
||||||
error("Exceeded maximum comparison depth")
|
error("Exceeded maximum comparison depth")
|
||||||
end
|
end
|
||||||
local checked = {}
|
local checked = {}
|
||||||
for k, v in pairs(x) do
|
for k, v in pairs(x) do
|
||||||
if not equivalent(y[k], v, depth - 1) then
|
if not equivalent(y[k], v, depth + 1) then
|
||||||
return false
|
return false
|
||||||
end
|
end
|
||||||
checked[k] = true
|
checked[k] = true
|
||||||
end
|
end
|
||||||
for k, v in pairs(y) do
|
for k, v in pairs(y) do
|
||||||
if not checked[k] and not equivalent(x[k], v, depth - 1) then
|
if not checked[k] and not equivalent(x[k], v, depth + 1) then
|
||||||
return false
|
return false
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
Loading…
Reference in New Issue
Block a user