Write-once table in lua?
Asked Answered
A

1

5

I'd like to have a write-once table in Lua (specifically LuaJIT 2.0.3), so that:

local tbl = write_once_tbl()
tbl["a"] = 'foo'
tbl["b"] = 'bar'
tbl["a"] = 'baz'  -- asserts false

Ideally, this would otherwise function like a regular table (pairs() and ipairs() work).

__newindex is basically the opposite of what I'd want for implementing this easily, and I am unaware of any techniques for making a proxy table pattern work with pairs() and ipairs().

Astred answered 3/11, 2017 at 0:29 Comment(0)
I
7

You need to use a proxy table, that is, an empty table that catches all access to the actual table:

function write_once_tbl()
    local T={}
    return setmetatable({},{
        __index=T,
        __newindex=
            function (t,k,v)
                if T[k]==nil then
                    T[k]=v
                else
                    error("table is write-once")
                end
            end,
        __pairs=  function (t) return  pairs(T) end,
        __ipairs= function (t) return ipairs(T) end,
        })
end

Note that __pairs and __ipairs only work from Lua 5.2 onwards.

Incursive answered 3/11, 2017 at 15:55 Comment(5)
I know of proxy tables, but they do not work with pairs() or ipairs(). I just tried your code snippet and verified that pairs() and ipairs() do not work with it.Astred
@MikeMx7f, they do in Lua 5.2 and 5.3.Incursive
I tested your code snippet here: lua.org/cgi-bin/demo which is using Lua 5.3.Astred
@MikeMx7f, you need to set __pairs and __ipairs metamethods.Incursive
Thanks. Can you edit that into the answer so I can accept it (since the question specifies pairs() and ipairs() works)? Cheers.Astred

© 2022 - 2024 — McMap. All rights reserved.