#!/usr/bin/lua --[[ UCI Validation Layer - Command Line Utility (c) 2008 Jo-Philipp Wich <xm@leipzig.freifunk.net> (c) 2008 Steven Barth <steven@midlink.org> Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 $Id: uvl 4085 2009-01-17 23:51:28Z jow $ ]]-- require("luci.uvl") require("luci.util") function getopt( arg, options ) options = options or "" local tab = {} local args = {} for k, v in ipairs(arg) do if v:sub(1, 2) == "--" then local x = v:find( "=", 1, true ) if x then tab[ v:sub( 3, x-1 ) ] = v:sub( x+1 ) else tab[ v:sub( 3 ) ] = true end elseif v:sub( 1, 1 ) == "-" then local y = 2 local l = #v local jopt while ( y <= l ) do jopt = v:sub( y, y ) if options:find( jopt, 1, true ) then if y < l then tab[ jopt ] = v:sub( y+1 ) y = l else tab[ jopt ] = arg[ k + 1 ] arg[ k + 1 ] = "" end else tab[ jopt ] = true end y = y + 1 end elseif #v > 0 then table.insert(args, v) end end return tab, args end function genspec(conf) require("luci.model.uci") require("luci.uvl.datatypes") local uci = luci.model.uci.cursor() local ok, err = uci:load(conf) if not ok then print("Can not load config:", err) os.exit(1) else local function _guess_datatype(v) if type(v) == "table" then v = v[1] end for _, type in ipairs({ "boolean", "integer", "float", "ip4addr", "ip6addr", "macaddr", "directory", "file" }) do if luci.uvl.datatypes[type](v) then return type end end return "string" end local co = uci:get_all(conf) local ct = { } local ca = { } local so = { } local to = { } -- count section types for _, section in pairs(co) do ct[section['.type']] = ( ct[section['.type']] or 0 ) + 1 ca[section['.type']] = section['.anonymous'] so[section['.type']] = so[section['.type']] or { } to[section['.type']] = to[section['.type']] or { } for option, value in pairs(section) do if option:sub(1,1) ~= "." then so[section['.type']][option] = _guess_datatype(value) to[section['.type']][option] = ( type(value) == "table" and "list" or "variable" ) end end end -- package name print( "package %s" % conf ) -- write section schemes for type, count in luci.util.kspairs(ct) do print( "\nconfig section" ) print( "\toption name '%s'" % type ) print( "\toption title 'Section %s'" % type ) print( "\toption package '%s'"% conf ) print( "\toption named %s" % ( ca[type] and 'false' or 'true' ) ) print( "\toption unique %s" % ( ct[type] > 1 and 'false' or ( ca[type] and 'false' or 'true' ) ) ) print( "\toption dynamic false" ) print( "\toption required false" ) -- write option schemes for opt, val in luci.util.kspairs(so[type]) do print( "\nconfig variable" ) print( "\toption name '%s'" % opt ) print( "\toption title 'Option %s'" % opt ) print( "\toption section '%s.%s'" %{ conf, type } ) print( "\toption datatype '%s'" % so[type][opt] ) if to[type][opt] ~= "variable" then print( "\toption type '%s'" % to[type][opt] ) end end print("") end end end local options, arguments = getopt( arg ) if #arguments ~= 2 or options.help then print([=[ uvl - UCI Validation Layer $Id: uvl 4085 2009-01-17 23:51:28Z jow $ (c) 2008 Jo-Philipp Wich, Steven Barth Usage: uvl --help uvl [--silent] [--schemedir=DIR] [--configdir=DIR] [--no-strict-sections] \ [--no-strict-options] [--no-strict-validators] [--no-strict-lists] \ {verify|verify-scheme|genspec} config[.section[.option]] Options: --help Display this help message. --silent Don't produce any output. --schemedir=DIR Use DIR as scheme directory. --configdir=DIR Use DIR as config directory. --no-strict-sections Don't treat sections found in config but not in scheme as error. --no-strict-options Don't treat options found in config but not in scheme as error. --no-strict-validators Don't invalidate config if an external validator fails. --no-strict-lists Don't invalidate lists that are stored options. Actions: verify Validate given configuration, section or option. verify-scheme Validate given scheme against the reference meta scheme. genspec Generate a scheme skeleton from given configuration. ]=]) os.exit(255) elseif arguments[1] == "verify" or arguments[1] == "verify-scheme" then luci.uvl.STRICT_UNKNOWN_SECTIONS = ( not options['no-strict-sections'] and true or false ) luci.uvl.STRICT_UNKNOWN_OPTIONS = ( not options['no-strict-options'] and true or false ) luci.uvl.STRICT_EXTERNAL_VALIDATORS = ( not options['no-strict-validators'] and true or false ) luci.uvl.STRICT_LIST_TYPE = ( not options['no-strict-lists'] and true or false ) local uvl = luci.uvl.UVL( type(options.schemedir) == "string" and options.schemedir, type(options.configdir) == "string" and options.configdir ) local cso = luci.util.split( arguments[2], "." ) local ok, err if arguments[1] == "verify-scheme" then uvl:read_scheme( 'schema', cso[1] ) local uci = uvl.uci.cursor( type(options.configdir) == "string" and options.configdir or uvl.schemedir .. '/default' ) ok, err = uvl:validate_config( cso[1], uci:get_all(cso[1]) ) if err then err.code = luci.uvl.errors.ERR_SCHEME end else ok, err = uvl:validate( unpack(cso) ) end if ok then if not options.silent then print( string.format( '%s "%s" validates fine!', ( arguments[1] == "verify-scheme" and "Scheme" or ( #cso == 1 and "Config" or ( #cso == 2 and "Section" or "Option" ) ) ), table.concat(cso, ".") ) ) end os.exit( 0 ) else if not options.silent then print( err and err:string() or "Unknown error" ) end os.exit( 1 ) end else genspec( arguments[2] ) end