access creates constant and sealed tables with the following guarantees:
- Constant tables throw errors when a value is assigned to a field, even if the field doesn't exist
- Sealed tables throw errors when a value is assigned to a field that doesn't exist, but accepts changes to existing fields
Also, in both cases accessing an nonexistent field will throw an error.
access is a pure Lua module.
Example:
local access = require 'access'
-- Create a constant table.
local c = access.const {
pi = math.pi
}
-- Throws an error, c.huge doesn't exist.
print(c.huge)
-- Throws an error, the table is constant.
c.pi = math.huge
-- Also throws an error.
c.huge = math.huge
Example:
local access = require 'access'
-- Create a sealed table.
local s = access.seal {
pi = math.pi
}
-- Throws an error, c.huge doesn't exist.
print(s.huge)
-- Ok, s.pi exists and will be equal to math.huge.
s.pi = math.huge
-- Throws an error, the table is sealed and the field doesn't exist.
s.huge = math.huge
Example:
local access = require 'access'
-- Create a record.
local s = access.record {
pi = math.pi
}
-- Throws an error, c.huge doesn't exist.
print(s.huge)
-- Throws an error, s.pi exists but is a number.
s.pi = 'pi'
-- Throws an error, the table is a record and the field doesn't exist.
s.huge = math.huge
When run, the code below will install a different require
function that will make all loaded modules constant if the access module is available.
local require0 = require
local found, access = pcall(require, 'access')
if found then
require = function(modname)
local mod = require0(modname)
local const = access.const(mod)
package.loaded[modname] = const
return const
end
end
The code below will make the object on the top of the stack constant if the access module is available, otherwise it will leave the object untouched. It returns LUA_OK
if successful, or an error code from luaL_loadstring
if it fails (which it shouldn't), in which case an error message is left at the top of the stack.
static int make_const(lua_State* const L) {
// Compile the code and leave the chunk at the top of the stack.
int const load_res = luaL_loadstring(L,
// This code will return its argument as a constant object only if
// the access module is available.
"return function(module)\n"
" local found, access = pcall(require, 'access')\n"
" return found and access.const(module) or module\n"
"end\n"
);
// Oops.
if (load_res != LUA_OK) {
return load_res;
}
// Run the chunk and leave the function that it returns at the top of the
// stack.
lua_call(L, 0, 1);
// Move the function to below the object.
lua_insert(L, -2);
// Run the Lua code to make the module table constant.
lua_call(L, 1, 1);
// All is good.
return LUA_OK;
}
access depends on the proxyud module to create full userdata objects used as proxies to the real values.
__close
metamethod not proxiedipairs
doesn't work with proxied tables because it will try to access one element past the end of the array, which throws an error for all constant, sealed, and record tables created with this module
- 1.2.0
- Added support to more metamethods
- Added unit tests
- Cached some additional globals
- Some refactoring to make sure all metamethods are copied from the
const
metatable toseal
andrecord
ones
- 1.1.0
- Added
record
, which acts likeseal
but errors if the new value being assigned to a field doesn't have the same type as the previous value
- Added
- 1.0.0
- First public release
MIT, enjoy.