root/luci/trunk/libs/uvl/luasrc/uvl/errors.lua @ 5865

Revision 5865, 5.7 KB (checked in by jow, 3 years ago)

libs/uvl: local symbol error shadows error class declaration, drop it

  • Property svn:keywords set to Id
Line 
1--[[
2
3UCI Validation Layer - Error handling
4(c) 2008 Jo-Philipp Wich <xm@leipzig.freifunk.net>
5(c) 2008 Steven Barth <steven@midlink.org>
6
7Licensed under the Apache License, Version 2.0 (the "License");
8you may not use this file except in compliance with the License.
9You may obtain a copy of the License at
10
11        http://www.apache.org/licenses/LICENSE-2.0
12
13$Id$
14
15]]--
16
17local uci = require "luci.model.uci"
18local uvl = require "luci.uvl"
19local util = require "luci.util"
20local string = require "string"
21
22local luci, tonumber, unpack, ipairs, type =
23    luci, tonumber, unpack, ipairs, type
24
25module "luci.uvl.errors"
26
27ERRCODES = {
28    UCILOAD         = 'Unable to load config "%p": %1',
29
30    SCHEME          = 'Error in scheme "%p":\n%c',
31    CONFIG          = 'Error in config "%p":\n%c',
32    SECTION         = 'Error in section "%i" (%I):\n%c',
33    OPTION          = 'Error in option "%i" (%I):\n%c',
34    REFERENCE       = 'Option "%i" has invalid reference specification %1:\n%c',
35    DEPENDENCY      = 'In dependency check for %t "%i":\n%c',
36
37    SME_FIND        = 'Can not find scheme "%p" in "%1"',
38    SME_READ        = 'Can not access file "%1"',
39    SME_REQFLD      = 'Missing required scheme field "%1" in "%i"',
40    SME_INVREF      = 'Illegal reference "%1" to an anonymous section',
41    SME_BADREF      = 'Malformed reference in "%1"',
42    SME_BADDEP      = 'Malformed dependency specification "%1" in "%i"',
43    SME_BADVAL      = 'Malformed validator specification "%1" in "%i"',
44    SME_ERRVAL      = 'External validator "%1" failed: %2',
45    SME_VBADPACK    = 'Variable "%o" in scheme "%p" references unknown package "%1"',
46    SME_VBADSECT    = 'Variable "%o" in scheme "%p" references unknown section "%1"',
47    SME_EBADPACK    = 'Enum "%v" in scheme "%p" references unknown package "%1"',
48    SME_EBADSECT    = 'Enum "%v" in scheme "%p" references unknown section "%1"',
49    SME_EBADOPT     = 'Enum "%v" in scheme "%p" references unknown option "%1"',
50    SME_EBADTYPE    = 'Enum "%v" in scheme "%p" references non-enum option "%I"',
51    SME_EBADDEF     = 'Enum "%v" in scheme "%p" redeclares the default value of "%I"',
52
53    SECT_UNKNOWN    = 'Section "%i" (%I) not found in scheme',
54    SECT_REQUIRED   = 'Required section "%p.%S" not found in config',
55    SECT_UNIQUE     = 'Unique section "%p.%S" occurs multiple times in config',
56    SECT_NAMED      = 'The section of type "%p.%S" is stored anonymously in config but must be named',
57    SECT_NOTFOUND   = 'Section "%p.%s" not found in config',
58
59    OPT_UNKNOWN     = 'Option "%i" (%I) not found in scheme',
60    OPT_REQUIRED    = 'Required option "%i" has no value',
61    OPT_BADVALUE    = 'Value "%1" of option "%i" is not defined in enum %2',
62    OPT_INVVALUE    = 'Value "%1" of option "%i" does not validate as datatype "%2"',
63    OPT_NOTLIST     = 'Option "%i" is defined as list but stored as plain value',
64    OPT_DATATYPE    = 'Option "%i" has unknown datatype "%1"',
65    OPT_NOTFOUND    = 'Option "%p.%s.%o" not found in config',
66    OPT_RANGE       = 'Option "%p.%s.%o" is not within the specified range',
67
68    DEP_NOTEQUAL    = 'Dependency (%1) failed:\nOption "%i" is not eqal "%2"',
69    DEP_NOVALUE     = 'Dependency (%1) failed:\nOption "%i" has no value',
70    DEP_NOTVALID    = 'Dependency (%1) failed:\n%c',
71    DEP_RECURSIVE   = 'Recursive dependency for option "%i" detected',
72    DEP_BADENUM     = 'In dependency check for enum value "%i":\n%c'
73}
74
75function i18n(key)
76    if luci.i18n then
77        return luci.i18n.translate(key)
78    else
79        return key
80    end
81end
82
83
84error = util.class()
85
86function error.__init__(self, code, pso, args)
87
88    self.code = code
89    self.args = ( type(args) == "table" and args or { args } )
90
91    if util.instanceof( pso, uvl.uvlitem ) then
92        self.stype = pso.sref[2]
93        self.package, self.section, self.option, self.value = unpack(pso.cref)
94        self.object = pso
95        self.value  = self.value or ( pso.value and pso:value() )
96    else
97        pso = ( type(pso) == "table" and pso or { pso } )
98
99        if pso[2] then
100            local uci = uci.cursor()
101            self.stype = uci:get(pso[1], pso[2]) or pso[2]
102        end
103
104        self.package, self.section, self.option, self.value = unpack(pso)
105    end
106end
107
108function error.child(self, err)
109    if not self.childs then
110        self.childs = { err }
111    else
112        self.childs[#self.childs+1] = err
113    end
114    return self
115end
116
117function error.string(self,pad)
118    pad = pad or "  "
119
120    local str = i18n(ERRCODES[self.code] or self.code)
121        :gsub("\n", "\n"..pad)
122        :gsub("%%i", self:cid())
123        :gsub("%%I", self:sid())
124        :gsub("%%p", self.package or '(nil)')
125        :gsub("%%s", self.section or '(nil)')
126        :gsub("%%S", self.stype   or '(nil)')
127        :gsub("%%o", self.option  or '(nil)')
128        :gsub("%%v", self.value   or '(nil)')
129        :gsub("%%t", self.object and self.object:type()  or '(nil)' )
130        :gsub("%%T", self.object and self.object:title() or '(nil)' )
131        :gsub("%%([1-9])", function(n) return self.args[tonumber(n)] or '(nil)' end)
132        :gsub("%%c",
133            function()
134                local s = ""
135                for _, err in ipairs(self.childs or {}) do
136                    s = s .. err:string(pad.."  ") .. "\n" .. pad
137                end
138                return s
139            end
140        )
141
142    return (str:gsub("%s+$",""))
143end
144
145function error.cid(self)
146    return self.object and self.object:cid() or self.package ..
147        ( self.section and '.' .. self.section or '' ) ..
148        ( self.option  and '.' .. self.option  or '' ) ..
149        ( self.value   and '.' .. self.value   or '' )
150end
151
152function error.sid(self)
153    return self.object and self.object:sid() or self.package ..
154        ( self.stype   and '.' .. self.stype   or '' ) ..
155        ( self.option  and '.' .. self.option  or '' ) ..
156        ( self.value   and '.' .. self.value   or '' )
157end
158
159function error.is(self, code)
160    if self.code == code then
161        return true
162    elseif self.childs then
163        for _, c in ipairs(self.childs) do
164            if c:is(code) then
165                return true
166            end
167        end
168    end
169    return false
170end
171
172function error.is_all(self, ...)
173    local codes = { ... }
174
175    if util.contains(codes, self.code) then
176        return true
177    else
178        local equal = false
179        for _, c in ipairs(self.childs) do
180            if c.childs then
181                equal = c:is_all(...)
182            else
183                equal = util.contains(codes, c.code)
184            end
185        end
186        return equal
187    end
188end
Note: See TracBrowser for help on using the browser.