Made iteration easier to work with by using .__inext and .__next for

custom iteration, and a custom ipairs() and pairs() to use that.
This commit is contained in:
Bruce Hill 2019-03-13 20:55:24 -07:00
parent 1e99bbbe0a
commit 783eec9b45
8 changed files with 82 additions and 130 deletions

View File

@ -186,8 +186,11 @@ co_mt = {
return math.huge return math.huge
end, end,
__call = coroutine.resume, __call = coroutine.resume,
__next = function(self, k) __inext = function(self, k)
local ok, val = coroutine.resume(self) local ok, val = coroutine.resume(self)
if coroutine.status(self) == 'dead' then
return
end
if ok then if ok then
return (k or 0) + 1, val return (k or 0) + 1, val
end end
@ -207,6 +210,7 @@ co_mt = {
return co_mt[k] return co_mt[k]
end end
} }
co_mt.__next = co_mt.__inext
debug.setmetatable(coroutine.create(function() end), co_mt) debug.setmetatable(coroutine.create(function() end), co_mt)
local nil_mt = { local nil_mt = {
__type = "Nil", __type = "Nil",

View File

@ -80,8 +80,9 @@ co_mt =
as_text: => (tostring(@)\gsub("thread", "Coroutine")).." ("..coroutine.status(@)..")" as_text: => (tostring(@)\gsub("thread", "Coroutine")).." ("..coroutine.status(@)..")"
__len: => math.huge __len: => math.huge
__call: coroutine.resume __call: coroutine.resume
__next: (k)=> __inext: (k)=>
ok, val = coroutine.resume(@) ok, val = coroutine.resume(@)
return if coroutine.status(@) == 'dead'
if ok then return (k or 0) + 1, val if ok then return (k or 0) + 1, val
__index: (k)=> __index: (k)=>
if k == (_last_co_i[@] or 0) + 1 if k == (_last_co_i[@] or 0) + 1
@ -92,6 +93,7 @@ co_mt =
else else
return nil return nil
return co_mt[k] return co_mt[k]
co_mt.__next = co_mt.__inext
debug.setmetatable(coroutine.create(->), co_mt) debug.setmetatable(coroutine.create(->), co_mt)
nil_mt = nil_mt =

View File

@ -1,4 +1,4 @@
local List, Dict, Undict, DictEntries, _undict_mt, _dict_mt, _dict_entries_mt local List, Dict, Undict, _undict_mt, _dict_mt
local insert, remove, concat local insert, remove, concat
do do
local _obj_0 = table local _obj_0 = table
@ -426,62 +426,6 @@ Undict = function(d)
end)()) end)())
return u return u
end end
local _dict_entries_mt
_dict_entries_mt = {
__type = "a Dict's Entries",
__index = function(self, k)
if type(k) == 'number' then
if k == 0 then
return nil
end
if k < 0 then
if k < 0 then
k = #self.dict + k + 1
end
end
local i, last_k = self._last_i, self._last_k
if k < i then
i, last_k = 0, nil
end
local d = self.dict
for i = i + 1, k do
last_k = next(d, last_k)
if last_k == nil then
return nil
end
end
self._last_i, self._last_k = k, last_k
return Dict({
key = last_k,
d[last_k]
})
else
return _dict_entries_mt[k]
end
end,
__len = function(self)
return #self.dict
end,
__eq = function(self, other)
return type(other) == type(self) and getmetatable(other) == getmetatable(self) and other.dict == self.dict
end,
__tostring = function(self)
return "(entries in " .. tostring(self.dict) .. ")"
end,
as_nomsu = function(self)
return "(entries in " .. _dict_mt.as_nomsu(self.dict) .. ")"
end,
as_lua = function(self)
return "entries_in" .. _dict_mt.as_lua(self.dict)
end
}
DictEntries = function(d)
return setmetatable({
dict = d,
_last_i = 0,
_last_k = nil
}, _dict_entries_mt)
end
_dict_mt = { _dict_mt = {
__type = "a Dict", __type = "a Dict",
__eq = function(self, other) __eq = function(self, other)
@ -540,8 +484,14 @@ _dict_mt = {
return _accum_0 return _accum_0
end)(), ", ") .. "}" end)(), ", ") .. "}"
end, end,
as_iterable = function(self) __inext = function(self, key)
return DictEntries(self) local nextkey, value = next(self, key)
if nextkey ~= nil then
return nextkey, Dict({
key = nextkey,
value = value
})
end
end, end,
__band = function(self, other) __band = function(self, other)
return Dict((function() return Dict((function()
@ -662,6 +612,5 @@ Dict = function(t, more, ...)
end end
return { return {
List = List, List = List,
Dict = Dict, Dict = Dict
DictEntries = DictEntries
} }

View File

@ -1,5 +1,5 @@
-- This file contains container classes, i.e. Lists and Dicts, plus some extended string functionality -- This file contains container classes, i.e. Lists and Dicts, plus some extended string functionality
local List, Dict, Undict, DictEntries, _undict_mt, _dict_mt, _dict_entries_mt local List, Dict, Undict, _undict_mt, _dict_mt
{:insert,:remove,:concat} = table {:insert,:remove,:concat} = table
as_nomsu = => as_nomsu = =>
@ -199,34 +199,6 @@ Undict = (d)->
compliments[u] = Dict{k,true for k,v in pairs(d) when v} compliments[u] = Dict{k,true for k,v in pairs(d) when v}
return u return u
local _dict_entries_mt
_dict_entries_mt =
__type: "a Dict's Entries"
__index: (k)=>
if type(k) == 'number'
return nil if k == 0
if k < 0 then k = #@dict+k+1 if k < 0
i, last_k = @_last_i, @_last_k
if k < i
i, last_k = 0, nil
d = @dict
for i=i+1,k
last_k = next(d, last_k)
return nil if last_k == nil
@_last_i, @_last_k = k, last_k
return Dict{key:last_k, d[last_k]}
else
return _dict_entries_mt[k]
__len: => #@dict
__eq: (other)=>
type(other) == type(@) and getmetatable(other) == getmetatable(@) and other.dict == @dict
__tostring: => "(entries in "..tostring(@dict)..")"
as_nomsu: => "(entries in ".._dict_mt.as_nomsu(@dict)..")"
as_lua: => "entries_in".._dict_mt.as_lua(@dict)
DictEntries = (d)->
setmetatable {dict:d, _last_i:0, _last_k:nil}, _dict_entries_mt
_dict_mt = _dict_mt =
__type: "a Dict" __type: "a Dict"
__eq: (other)=> __eq: (other)=>
@ -247,7 +219,10 @@ _dict_mt =
"{"..concat([v == true and "."..as_nomsu(k) or ".#{as_nomsu(k)} = #{as_nomsu(v)}" for k,v in pairs @], ", ").."}" "{"..concat([v == true and "."..as_nomsu(k) or ".#{as_nomsu(k)} = #{as_nomsu(v)}" for k,v in pairs @], ", ").."}"
as_lua: => as_lua: =>
"a_Dict{"..concat(["[ #{as_lua(k)}]= #{as_lua(v)}" for k,v in pairs @], ", ").."}" "a_Dict{"..concat(["[ #{as_lua(k)}]= #{as_lua(v)}" for k,v in pairs @], ", ").."}"
as_iterable: => DictEntries(@) __inext: (key)=>
nextkey, value = next(@, key)
if nextkey != nil
return nextkey, Dict{key:nextkey, value:value}
__band: (other)=> __band: (other)=>
Dict{k,v for k,v in pairs(@) when other[k]} Dict{k,v for k,v in pairs(@) when other[k]}
__bor: (other)=> __bor: (other)=>
@ -300,4 +275,4 @@ Dict = (t,more,...)->
d[k] = v d[k] = v
return d return d
return {:List, :Dict, :DictEntries} return {:List, :Dict}

View File

@ -180,7 +180,6 @@ test:
for $ in $r: for $ in $r:
$visited, add $ $visited, add $
assume ($visited == [1, 3, 5, 7, 9]) assume ($visited == [1, 3, 5, 7, 9])
$(inext) = (=lua "ipairs({})")
$range_mt = { $range_mt = {
.__type = "a Range" .__type = "a Range"
.__index = .__index =
@ -212,7 +211,8 @@ $range_mt = {
($self.last == $other.last) and ($self.step == $other.step) ($self.last == $other.last) and ($self.step == $other.step)
.backwards = (for $self ($self.last to $self.first by (- $self.step))) .backwards = (for $self ($self.last to $self.first by (- $self.step)))
.__ipairs = (for $self: return $(inext) $self 0) .__inext = $(inext)
.__next = $(inext)
.as_text = .as_text =
for $self: for $self:
if ($self.step == 1): if ($self.step == 1):

View File

@ -167,13 +167,9 @@ test:
# This uses Lua's approach of only allowing loop-scoped variables in a loop # This uses Lua's approach of only allowing loop-scoped variables in a loop
if (($var.type == "Action") and ($var.stub == "1 =")): if (($var.type == "Action") and ($var.stub == "1 =")):
[$key, $value] = [$var.1, $var.3] [$key, $value] = [$var.1, $var.3]
go to (vars set)
if (($var.type == "Action") and ($var.stub == "1 at")):
[$key, $value] = [$var.3, $var.1]
..else: ..else:
[$key, $value] = [nil, $var] [$key, $value] = [nil, $var]
--- (vars set) ---
unless $value: unless $value:
at (this tree) fail "No value here" at (this tree) fail "No value here"
@ -211,21 +207,16 @@ test:
if $key: if $key:
$loop = $loop =
Lua (" Lua ("
local _iterating = \($iterable as lua expr); for \($key as lua identifier),\($value as lua identifier) in pairs(\($iterable as lua expr)) do
local _next = getmetatable(_iterating).__next or next;
for \($key as lua identifier),\($value as lua identifier) in _next,_iterating,nil do
if \($value as lua identifier) == nil and _1_is_a_dead_coroutine(_iterating) then break end
") ")
..else: ..else:
$loop = $loop =
Lua (" Lua ("
local _iterating = _1_as_an_iterable(\($iterable as lua expr)) for _i,\($value as lua identifier) in _ipairs(\($iterable as lua expr)) do
for _i=1,#_iterating do
local \($value as lua identifier) = _iterating[_i]
if \($value as lua identifier) == nil and _1_is_a_dead_coroutine(_iterating) then break end
") ")
--- (loop set) --- --- (loop set) ---
# TODO: don't always wrap in block
$lua = $lua =
Lua (" Lua ("
do -- for-loop do -- for-loop

View File

@ -3,10 +3,10 @@ do
local _obj_0 = require("code_obj") local _obj_0 = require("code_obj")
NomsuCode, LuaCode, Source = _obj_0.NomsuCode, _obj_0.LuaCode, _obj_0.Source NomsuCode, LuaCode, Source = _obj_0.NomsuCode, _obj_0.LuaCode, _obj_0.Source
end end
local List, Dict, DictEntries local List, Dict
do do
local _obj_0 = require('containers') local _obj_0 = require('containers')
List, Dict, DictEntries = _obj_0.List, _obj_0.Dict, _obj_0.DictEntries List, Dict = _obj_0.List, _obj_0.Dict
end end
local Text = require('text') local Text = require('text')
local SyntaxTree = require("syntax_tree") local SyntaxTree = require("syntax_tree")
@ -73,21 +73,39 @@ _1_as_text = function(x)
end end
return tostring(x) return tostring(x)
end end
local _1_as_an_iterable local nomsu_pairs
_1_as_an_iterable = function(x) nomsu_pairs = function(self)
do local mt = getmetatable(self)
local mt = getmetatable(x) if mt and mt.__next then
if mt then return mt.__next, self, nil
if mt.as_iterable then else
return mt.as_iterable(x) return next, self, nil
end end
end
local inext
if _VERSION == "Lua 5.2" or _VERSION == "Lua 5.1" then
inext = function(self, i)
local value = self[i + 1]
if value ~= nil then
return i + 1, value
end end
end end
return x else
inext = ipairs({ })
end
local nomsu_ipairs
nomsu_ipairs = function(self)
local mt = getmetatable(self)
if mt and mt.__inext then
return mt.__inext, self, 0
else
return inext, self, 0
end
end end
local nomsu_environment local nomsu_environment
nomsu_environment = Importer({ nomsu_environment = Importer({
next = next, next = next,
inext = inext,
unpack = unpack or table.unpack, unpack = unpack or table.unpack,
setmetatable = setmetatable, setmetatable = setmetatable,
rawequal = rawequal, rawequal = rawequal,
@ -122,8 +140,9 @@ nomsu_environment = Importer({
math = math, math = math,
io = io, io = io,
load = load, load = load,
pairs = pairs, pairs = nomsu_pairs,
ipairs = ipairs, ipairs = ipairs,
_ipairs = nomsu_ipairs,
jit = jit, jit = jit,
_VERSION = _VERSION, _VERSION = _VERSION,
LUA_VERSION = (jit and jit.version or _VERSION), LUA_VERSION = (jit and jit.version or _VERSION),
@ -132,7 +151,6 @@ nomsu_environment = Importer({
a_List = List, a_List = List,
a_Dict = Dict, a_Dict = Dict,
Text = Text, Text = Text,
Dict_Entries = DictEntries,
lpeg = lpeg, lpeg = lpeg,
re = re, re = re,
Files = Files, Files = Files,

View File

@ -1,7 +1,7 @@
-- This file defines the environment in which Nomsu code runs, including some -- This file defines the environment in which Nomsu code runs, including some
-- basic bootstrapping functionality. -- basic bootstrapping functionality.
{:NomsuCode, :LuaCode, :Source} = require "code_obj" {:NomsuCode, :LuaCode, :Source} = require "code_obj"
{:List, :Dict, :DictEntries} = require 'containers' {:List, :Dict} = require 'containers'
Text = require 'text' Text = require 'text'
SyntaxTree = require "syntax_tree" SyntaxTree = require "syntax_tree"
Files = require "files" Files = require "files"
@ -44,25 +44,38 @@ _1_as_text = (x)->
if x == false then return "no" if x == false then return "no"
return tostring(x) return tostring(x)
_1_as_an_iterable = (x)-> nomsu_pairs = =>
if mt = getmetatable(x) mt = getmetatable(@)
if mt.as_iterable if mt and mt.__next
return mt.as_iterable(x) return mt.__next, @, nil
return x else
return next, @, nil
inext = if _VERSION == "Lua 5.2" or _VERSION == "Lua 5.1"
inext = (i)=>
value = @[i+1]
return i+1, value if value != nil
else ipairs{}
nomsu_ipairs = =>
mt = getmetatable(@)
if mt and mt.__inext
return mt.__inext, @, 0
else
return inext, @, 0
local nomsu_environment local nomsu_environment
nomsu_environment = Importer{ nomsu_environment = Importer{
-- Lua stuff: -- Lua stuff:
:next, unpack: unpack or table.unpack, :setmetatable, :rawequal, :getmetatable, :pcall, :next, :inext, unpack: unpack or table.unpack, :setmetatable, :rawequal, :getmetatable, :pcall,
yield:coroutine.yield, resume:coroutine.resume, coroutine_status_of:coroutine.status, yield:coroutine.yield, resume:coroutine.resume, coroutine_status_of:coroutine.status,
coroutine_wrap:coroutine.wrap, coroutine_from: coroutine.create, coroutine_wrap:coroutine.wrap, coroutine_from: coroutine.create,
:error, :package, :os, :require, :tonumber, :tostring, :string, :xpcall, :error, :package, :os, :require, :tonumber, :tostring, :string, :xpcall,
:print, :loadfile, :rawset, :_VERSION, :collectgarbage, :rawget, :rawlen, :print, :loadfile, :rawset, :_VERSION, :collectgarbage, :rawget, :rawlen,
:table, :assert, :dofile, :loadstring, lua_type_of:type, :select, :math, :io, :load, :table, :assert, :dofile, :loadstring, lua_type_of:type, :select, :math, :io, :load,
:pairs, :ipairs, :jit, :_VERSION, LUA_VERSION: (jit and jit.version or _VERSION), pairs:nomsu_pairs, :ipairs, _ipairs:nomsu_ipairs, :jit,
:_VERSION, LUA_VERSION: (jit and jit.version or _VERSION),
LUA_API: _VERSION, Bit: (jit or _VERSION == "Lua 5.2") and require('bitops') or nil LUA_API: _VERSION, Bit: (jit or _VERSION == "Lua 5.2") and require('bitops') or nil
-- Nomsu types: -- Nomsu types:
a_List:List, a_Dict:Dict, Text:Text, Dict_Entries:DictEntries, a_List:List, a_Dict:Dict, Text:Text,
-- Utilities and misc. -- Utilities and misc.
lpeg:lpeg, re:re, Files:Files, lpeg:lpeg, re:re, Files:Files,
:SyntaxTree, TESTS: Dict({}), globals: Dict({}), :SyntaxTree, TESTS: Dict({}), globals: Dict({}),