Added list/dict metatables to make comparison and string representations

simpler. Also deleted Counters.
This commit is contained in:
Bruce Hill 2018-06-06 13:25:01 -07:00
parent 2d88c68d71
commit 810ae220bc
9 changed files with 80 additions and 57 deletions

View File

@ -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

View File

@ -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

View File

@ -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)]"

View File

@ -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

View File

@ -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)

View File

@ -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."

View File

@ -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."

View File

@ -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
".." ".."

View File

@ -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