-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtypes.lua
210 lines (179 loc) · 7.06 KB
/
types.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
local ErrorBuilder = import("errors.lua")
local tables = import("tables.lua")
--------------------
-- Type Utilities --
--------------------
-- TODO: Optimize by not calling stuff like getmetatable repeatedly
-- Returns the type of some value, giving classes first-level types.
local function get(value)
if type(value) == "table" and getmetatable(value)
and type(getmetatable(value).__class_name) == "string" then
return getmetatable(value).__class_name
end
return type(value)
end
-- Checks if some value has the expected type.
local function check(value, expected)
-- if a `|` character is found the in expected string, this is a complex comparison
if expected:find("|") then
for str in expected:gmatch("([^|]+)") do
if type(value) == str or get(value) == str then return true end
end
-- otherwise, we can assume that this is a simple comparison
elseif type(value) == expected or get(value) == expected then return true end
return false
end
local force_err = ErrorBuilder:new("types->force", 3)
-- Forces a specific set of types for function arguments. `types` is an array-equivalent and should include then names
-- of valid types. Arguments may be made optional by appending `?` (a question mark) to the type name. Arguments to
-- check are passed to the end of the function. The order of items in `types` corresponds to the order of the function
-- arguments passed to this function.
local function force(rules, ...)
if not DEBUG then return end
local arg = {...}
force_err:set_postfix(function()
return ("Rules = %s; Arguments = %s"):format(tables.dump(rules), tables.dump(arg))
end)
-- Check if argument is required
local function is_required(rule)
local required = true
-- Check for optional marker
if rule:sub(-1, -1) == "?" then
required = false
rule = rule:sub(0, -2)
end
return rule, required
end
-- Check rules
for key, raw_rule in ipairs(rules) do
-- Validate rule structure
force_err:assert(type(key) == "number", "table must be array-equivalent, only numbers may be used as rule "
.. "keys (found %s '%s')", type(key), key)
force_err:assert(type(raw_rule) == "string", "expected rules value to be a string (found rules[%s] = '%s')",
key, raw_rule)
local value = arg[key]
local rule, required = is_required(raw_rule)
-- if the argument is required and nil, error
if required and value == nil then
force_err:throw("argument #%d is required", key)
-- elseif the argument is not nil and does not match the rule, error
elseif value ~= nil and not check(value, rule) then
force_err:throw("argument #%d must be a %s (found %s)", key, rule, get(value))
end
end
-- if there are more arguments than rules, error
if #arg > #rules then
force_err:throw("found %d argument(s) and only %d rule(s)", #arg, #rules)
end
end
local force_array_err = ErrorBuilder:new("types->force_array", 2)
-- Ensures that a table contains only numerical indexes and optionally checks the type of each item within the table.
local function force_array(tbl, expected)
if not DEBUG then return end
force({"table", "string?"}, tbl, expected)
if expected then
force_array_err:set_postfix(function() return ("Expected Item Type = %s; Table = %s")
:format(expected, tables.dump(tbl)) end)
else
force_array_err:set_postfix(function() return ("Table = %s"):format(tables.dump(tbl)) end)
end
-- Check table
for key, value in pairs(tbl) do
if type(key) ~= "number" then
force_array_err:throw("found non-numerically indexed entry at %s (contains: %s)", tables.dump(key),
tables.dump(value))
end
if expected and not check(value, expected) then
force_array_err:throw("entry #%d must be a %s (found %s)", key, expected, get(value))
end
end
end
local validate_rules_err = ErrorBuilder:new("types->validate_rules", 3)
-- Validates `rules` table used by `types.constrain`. An error is raised if invalid.
--[[ Rules Example Structure: {
{
"show", -- This is the name of a table key.
"boolean", -- This is the name of the type expected. If nil any type is allowed.
required = false, -- Overrides the default of true to make the key optional.
}
} ]]
local function validate_rules(rules)
force({"table"}, rules)
validate_rules_err:set_postfix(function() return ("Rules = %s"):format(tables.dump(rules)) end)
validate_rules_err:assert(tables.count(rules) > 0, "'rules' argument (no. 1), a table, must contain at least one rule")
-- Verify that rules are valid
for index, rule in ipairs(rules) do
validate_rules_err:assert(type(rule) == "table", "rule[%d] must be a table (found %s)", index, type(rule))
validate_rules_err:assert(type(rule[1]) == "string", "rule[%d][1] must be a string (found %s)", index, type(rule[1]))
validate_rules_err:assert(rule[2] == nil or type(rule[2]) == "string", "rule[%d][2] must be a string or nil "
.. "(found %s)", index, type(rule[2]))
validate_rules_err:assert(rule.required == nil or type(rule.required) == "boolean",
"rule[%d].required must be a boolean or nil (found %s)", index, type(rule.required))
end
end
local constrain_err = ErrorBuilder:new("types->constrain", 2)
-- Constrains the keys within a table to meet specific requirements. Unless strict is false, an error is thrown if any
-- keys are found that are not in the rules table.
local function constrain(tbl, rules, strict)
if not DEBUG then return end
force({"table", "table", "boolean?"}, tbl, rules, strict)
-- Returns a rule by key name.
local function get_rule(key)
for _, rule in ipairs(rules) do
if rule[1] == key then
return rule
end
end
end
constrain_err:set_postfix(function()
return ("Table = %s; Rules = %s; Strict Mode = %s"):format(tables.dump(tbl), tables.dump(rules), tostring(strict))
end)
-- Validate rules
validate_rules(rules)
-- Compare table to rules
for key, value in pairs(tbl) do
local rule = get_rule(key)
-- if no rule exists for this key and strict mode hasn't been disabled, error
if not rule and strict ~= true then
constrain_err:throw("key '%s' is not allowed in strict mode", key)
end
-- if rule exists, make comparison
if rule then
-- if the type is controlled and it is not valid, error
if rule[2] and not check(value, rule[2]) then
constrain_err:throw("key '%s' must be of type %s (found %s)", key, rule[2], get(value))
end
end
end
-- if the rules table contains more entries than the table we are processing, check which keys are required
if tables.count(rules) > tables.count(tbl) then
local missing_keys = ""
for _, rule in ipairs(rules) do
if rule.required ~= false and not tbl[rule[1]] then
missing_keys = missing_keys .. "(" .. rule[1] .. (rule[2] == nil and "" or ": " .. rule[2]) .. "), "
end
end
if missing_keys ~= "" then
missing_keys = missing_keys:sub(1, -3)
constrain_err:throw("key(s) %s are required", missing_keys)
end
end
end
-- Returns a new type with name attached.
local function new_type(name)
local class = {}
class.__index = class
class.__class_name = name
return class
end
-------------
-- Exports --
--------------------
return {
get = get,
check = check,
force = force,
force_array = force_array,
constrain = constrain,
type = new_type
}