* luci/libs: uvl: implement scheme loading on demand, add inline documentation

This commit is contained in:
Jo-Philipp Wich 2008-08-17 18:47:32 +00:00
parent 97ce1ad8ad
commit a966a3b2b4
2 changed files with 131 additions and 24 deletions

View file

@ -14,6 +14,11 @@ $Id$
]]--
--- UVL - UCI Validation Layer
-- @class module
-- @cstyle instance
module( "luci.uvl", package.seeall )
require("luci.fs")
@ -24,12 +29,21 @@ require("luci.uvl.datatypes")
require("luci.uvl.validation")
require("luci.uvl.dependencies")
TYPE_SECTION = 0x01
TYPE_VARIABLE = 0x02
TYPE_ENUM = 0x03
--- Boolean; default true;
-- treat sections found in config but not in scheme as error
STRICT_UNKNOWN_SECTIONS = true
--- Boolean; default true;
-- treat options found in config but not in scheme as error
STRICT_UNKNOWN_OPTIONS = true
--- Boolean; default true;
-- treat failed external validators as error
STRICT_EXTERNAL_VALIDATORS = true
@ -43,6 +57,12 @@ local function _assert( condition, fmt, ... )
end
end
--- Object constructor
-- @class function
-- @name UVL
-- @param schemedir Path to the scheme directory (optional)
-- @return Instance object
UVL = luci.util.class()
function UVL.__init__( self, schemedir )
@ -57,10 +77,17 @@ end
--- Validate given configuration.
-- @param config Name of the configuration to validate
-- @param scheme Scheme to validate against (optional)
-- @return Boolean indicating weather the given config validates
-- @return Boolean indicating whether the given config validates
-- @return String containing the reason for errors (if any)
function UVL.validate( self, config )
if not self.packages[config] then
local ok, err = pcall( self.read_scheme, self, config )
if not ok then
return false, self.log.scheme_error( config, err )
end
end
self.uci.load_config( config )
self.beenthere = { }
@ -118,7 +145,20 @@ function UVL.validate( self, config )
return true, nil
end
--- Validate given config section.
-- @param config Name of the configuration to validate
-- @param section Name of the section to validate
-- @return Boolean indicating whether the given config validates
-- @return String containing the reason for errors (if any)
function UVL.validate_section( self, config, section )
if not self.packages[config] then
local ok, err = pcall( self.read_scheme, self, config )
if not ok then
return false, self.log.scheme_error( config, err )
end
end
self.uci.load_config( config )
self.beenthere = { }
@ -133,7 +173,21 @@ function UVL.validate_section( self, config, section )
end
end
--- Validate given config option.
-- @param config Name of the configuration to validate
-- @param section Name of the section to validate
-- @param option Name of the option to validate
-- @return Boolean indicating whether the given config validates
-- @return String containing the reason for errors (if any)
function UVL.validate_option( self, config, section, option )
if not self.packages[config] then
local ok, err = pcall( self.read_scheme, self, config )
if not ok then
return false, self.log.scheme_error( config, err )
end
end
self.uci.load_config( config )
self.beenthere = { }
@ -149,12 +203,7 @@ function UVL.validate_option( self, config, section, option )
end
end
--- Validate given section of given configuration.
-- @param config Name of the configuration to validate
-- @param section Key of the section to validate
-- @param scheme Scheme to validate against
-- @return Boolean indicating weather the given config validates
-- @return String containing the reason for errors (if any)
function UVL._validate_section( self, section )
if section:values() then
@ -190,13 +239,6 @@ function UVL._validate_section( self, section )
return true, nil
end
--- Validate given option within section of given configuration.
-- @param config Name of the configuration to validate
-- @param section Key of the section to validate
-- @param option Name of the option to validate
-- @param scheme Scheme to validate against
-- @return Boolean indicating weather the given config validates
-- @return String containing the reason for errors (if any)
function UVL._validate_option( self, option, nodeps )
if not option:option() and
@ -253,22 +295,31 @@ function UVL._validate_option( self, option, nodeps )
return true, nil
end
--- Find all parts of given scheme and construct validation tree
--- Find all parts of given scheme and construct validation tree.
-- This is normally done on demand, so you don't have to call this function
-- by yourself.
-- @param scheme Name of the scheme to parse
-- @return Parsed scheme
function UVL.read_scheme( self, scheme )
local schemes = { }
local files = luci.fs.glob(self.schemedir .. '/*/' .. scheme)
for i, file in ipairs( luci.fs.glob(self.schemedir .. '/*/' .. scheme) ) do
_assert( luci.fs.access(file), "Can't access file '%s'", file )
if files then
for i, file in ipairs( files ) do
_assert( luci.fs.access(file), "Can't access file '%s'", file )
self.uci.set_confdir( luci.fs.dirname(file) )
self.uci.load( luci.fs.basename(file) )
self.uci.set_confdir( luci.fs.dirname(file) )
self.uci.load( luci.fs.basename(file) )
table.insert( schemes, self.uci.get_all( luci.fs.basename(file) ) )
table.insert( schemes, self.uci.get_all( luci.fs.basename(file) ) )
end
return self:_read_scheme_parts( scheme, schemes )
else
error(
'Can\'t find scheme "' .. scheme ..
'" in "' .. self.schemedir .. '"'
)
end
return self:_read_scheme_parts( scheme, schemes )
end
-- Process all given parts and construct validation tree
@ -542,6 +593,20 @@ function UVL._resolve_function( self, value )
end
--- Object representation of a scheme/config section.
-- @class module
-- @cstyle instance
-- @name luci.uvl.section
--- Section instance constructor.
-- @class function
-- @name section
-- @param scheme Scheme instance
-- @param co Configuration data
-- @param st Section type
-- @param c Configuration name
-- @param s Section name
-- @return Section instance
section = luci.util.class()
function section.__init__(self, scheme, co, st, c, s)
@ -554,22 +619,32 @@ function section.__init__(self, scheme, co, st, c, s)
self.type = luci.uvl.TYPE_SECTION
end
--- Get the config path of this section.
-- @return String containing the identifier
function section.cid(self)
return ( self.cref[1] or '?' ) .. '.' .. ( self.cref[2] or '?' )
end
--- Get the scheme path of this section.
-- @return String containing the identifier
function section.sid(self)
return ( self.sref[1] or '?' ) .. '.' .. ( self.sref[2] or '?' )
end
--- Get all configuration values within this section.
-- @return Table containing the values
function section.values(self)
return self.csection
end
--- Get the associated section information in scheme.
-- @return Table containing the scheme properties
function section.section(self)
return self.ssection
end
--- Get all option objects associated with this section.
-- @return Table containing all associated luci.uvl.option instances
function section.variables(self)
local v = { }
if self.scheme.packages[self.sref[1]].variables[self.sref[2]] then
@ -586,6 +661,21 @@ function section.variables(self)
end
--- Object representation of a scheme/config option.
-- @class module
-- @cstyle instance
-- @name luci.uvl.option
--- Section instance constructor.
-- @class function
-- @name option
-- @param scheme Scheme instance
-- @param co Configuration data
-- @param st Section type
-- @param c Configuration name
-- @param s Section name
-- @param o Option name
-- @return Option instance
option = luci.util.class()
function option.__init__(self, scheme, co, st, c, s, o)
@ -598,26 +688,36 @@ function option.__init__(self, scheme, co, st, c, s, o)
self.type = luci.uvl.TYPE_OPTION
end
--- Get the config path of this option.
-- @return String containing the identifier
function option.cid(self)
return ( self.cref[1] or '?' ) .. '.' ..
( self.cref[2] or '?' ) .. '.' ..
( self.cref[3] or '?' )
end
--- Get the scheme path of this option.
-- @return String containing the identifier
function option.sid(self)
return ( self.sref[1] or '?' ) .. '.' ..
( self.sref[2] or '?' ) .. '.' ..
( self.sref[3] or '?' )
end
--- Get the value of this option.
-- @return The associated configuration value
function option.value(self)
return self.coption
end
--- Get the associated option information in scheme.
-- @return Table containing the scheme properties
function option.option(self)
return self.soption
end
--- Get the associated section information in scheme.
-- @return Table containing the scheme properties
function option.section(self)
return self.scheme.packages[self.sref[1]].sections[self.sref[2]]
end

View file

@ -37,6 +37,13 @@ function validator_error( option, message )
)
end
function scheme_error( scheme, message )
return string.format(
'Error while loading scheme "%s":\n%s',
scheme, message:gsub("^.-:.-: ","")
)
end
function dump_dependency( dep, ref, v, e )
local str = nil