1
0
mirror of https://github.com/elua/elua.git synced 2025-01-08 20:56:17 +08:00
elua/config/sections.lua
2012-07-14 13:18:29 +03:00

175 lines
7.0 KiB
Lua

-- Sections (collection of elements) management
module( ..., package.seeall )
local sf = string.format
local gen = require "generators"
local utils = require "utils"
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 ) and not conf[ attrmeta.macro ].from_default then
print( utils.col_yellow( sf( "[CONFIG] 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 ] = { name = attr, desc = attrmeta, value = v, sectname = sectname, elname = name, from_default = false }
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 ] = { name = name, desc = data, value = data.default, sectname = sectname, elname = name, from_default = true }
end
end
-- Mark this component as configured
enabled[ name ] = true
if req then required[ name ] = true end
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. There is another function called
-- 'auxcheck' that is called AFTER the basic consistency check (if it exists)
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, enabled )
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
if desc.auxcheck then
local d, err = desc:auxcheck( conf, enabled )
if not d then return false, err 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. If the element has an 'auxgen'
-- function, it will be called after the default attribute generation
-- 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
if desc.auxgen then genstr = genstr .. desc:auxgen( 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
-- Finally, check for dependencies
-- For each attribute that has a 'needs' element, check if the dependency is met
-- The attribute can also define the 'depcheck' function for a default dependency check
for elname, _ in pairs( enabled ) do
local desc = section[ elname ]
if desc.depcheck then
local res, err = desc:depcheck( conf, generated )
if not res then return false, err end
elseif desc.needs then
local needs = type( desc.needs ) == "table" and desc.needs or { desc.needs }
for _, v in pairs( needs ) do
-- Look for negative expressions (not enabled)
local neg
if v:sub( 1, 1 ) == "!" then
neg = true
v = v:sub( 2 )
end
if not neg and not enabled[ v ] then
return false, sf( "element '%s' in section '%s' needs element '%s' to be enabled", elname, sectname, v )
elseif neg and enabled[ v ] then
return false, sf( "element '%s' in section '%s' needs element '%s' to be disabled", elname, sectname, v )
end
end
end
end
-- All done
return genstr
end