1
0
mirror of https://github.com/elua/elua.git synced 2025-01-25 01:02:54 +08:00
elua/config/sections.lua

144 lines
5.5 KiB
Lua

-- Sections (collection of elements) management
module( ..., package.seeall )
local sf = string.format
local gen = require "generators"
conf, enabled, required = {}, {}, {}
-- Enables and configures an element (a collection of attributes)
-- section - the section of the element
-- sectname - the name of the section that contains the element (components, config ...)
-- name - name of the element
-- data - values of attributes in the element
-- req: required mode. true if this element's value are forced to its required values, false otherwise
-- Returns true if OK, (false, err) for error
function config_element( section, sectname, name, data, req )
local desc = section[ name ]
local attrs = desc.attrs or {}
-- Process each element in 'data' in turn
for attr, v in pairs( data ) do
local attrmeta = attrs[ attr ]
if not attrmeta then return false, sf( "attribute '%s' is not defined for element '%s' in section '%s'", attr, name, sectname ) end
if attrmeta.validator and not req then
local res, err = attrmeta:validator( attr, v, name, sectname )
if not res then
return false, err
else
-- The validator can also change the attribute's value
v = res
end
end
if conf[ attrmeta.macro ] and tostring( conf[ attrmeta.macro ].value ) ~= tostring( v ) then
print( sf( "WARNING: overriding value of attribute '%s' in element '%s' from '%s' to '%s' in section '%s'", attr, name,
conf[ attrmeta.macro ].value, v, sectname ) )
end
conf[ attrmeta.macro ] = { desc = attrmeta, value = v }
-- print( sf( "SET -> '%s' = '%s'", attrmeta.macro, v ) )
end
-- Set default values where needed
for name, data in pairs( attrs ) do
if not conf[ data.macro ] and data.default then
conf[ data.macro ] = { desc = data, value = data.default }
-- print( sf( "DEFAULT -> '%s' = '%s'", data.macro, data.default ) )
end
end
-- Mark this component as configured
enabled[ name ] = true
if req then required[ name ] = true end
-- print( sf( "ENABLED -> %s", name ) )
return true
end
-- Configures the given section
-- section: the section that will be compiled
-- sectname: the name of the section
-- data: the data corresponding to the section
-- Returns true if OK, (false, errmsg) for error
function configure_section( section, sectname, data )
conf, enabled, required = {}, {}, {}
-- Configure each element in turn, doing validation if required
for elname, elval in pairs( data ) do
if not section[ elname ] then return nil, sf( "unknown element '%s' in section '%s'", elname, sectname ) end
-- Handle the special situation <elname> = true (or anything else that is not false)
if type( elval ) ~= "table" and elval then
data[ elname ] = {}
elval = data[ elname ]
end
if elval then
local cres, cerr = config_element( section, sectname, elname, elval )
if not cres then return false, cerr end
end
end
-- We also need to generated required elements. A required element is an element that
-- is generated every time, even if it was not specified in the configuration file.
for elname, eldesc in pairs( section ) do
if eldesc.required and not enabled[ elname ] then
config_element( section, sectname, elname, eldesc.required, true )
end
end
-- Step 2: basic consistency check
-- For each element, we check that all its required attributes (the ones that don't have
-- an 'optional' key set to true) have a value. An element can overwrite this default
-- verification by specifying its own 'confcheck' function
for elname, _ in pairs( enabled ) do
if not required[ elname ] then
local desc = section[ elname ]
local attrs = desc.attrs or {}
if desc.confcheck then
local d, err = desc:confcheck( conf )
if not d then return false, err end
else
for attr, adesc in pairs( attrs ) do
if not conf[ adesc.macro ] and not adesc.optional then
return false, sf( "required attribute '%s' of component '%s' in section '%s' not specified", attr, elname, sectname )
end
end
end
end
end
return true
end
-- Generate configuration data for the given section (must be called after configure_section!)
-- section: the section that will be compiled
-- sectname: the name of the section
-- data: the data corresponding to the section
-- Returns the generated header if OK, (false, errmsg) for error
function generate_section( section, sectname, data )
-- Actual generation of code
-- The default generator simply adds '#define KEY VALUE' pairs. An element can overwrite this
-- default generation by specifying its own 'gen' function
-- Also, we never generate the same key twice. We ensure this by keeping a table of the
-- keys that were already generated
local generated = {}
local genstr = string.rep( "/", 80 ) .. "\n" .. sf( "// Configuration for section '%s'\n\n", sectname )
for elname, _ in pairs( enabled ) do
local desc = section[ elname ]
local attrs = desc.attrs or {}
genstr = genstr .. sf( "// Configuration for element '%s'\n", elname )
if desc.gen then
genstr = genstr .. desc:gen( conf, generated )
else
for aname, adesc in pairs( attrs ) do
genstr = genstr .. gen.simple_gen( adesc.macro, conf, generated )
end
end
-- Add the "build enable" macro
if desc.macro then
genstr = genstr .. gen.print_define( desc.macro ) .. "\n"
else
genstr = genstr .. "\n"
end
end
-- All done
return genstr
end