log4r-color 1.1.11

Sign up to get free protection for your applications and to get access to all the features.
Files changed (106) hide show
  1. data/INSTALL +11 -0
  2. data/LICENSE +90 -0
  3. data/LICENSE.LGPLv3 +165 -0
  4. data/README +95 -0
  5. data/Rakefile +80 -0
  6. data/TODO +2 -0
  7. data/doc/content/contact.html +22 -0
  8. data/doc/content/contribute.html +21 -0
  9. data/doc/content/index.html +90 -0
  10. data/doc/content/license.html +56 -0
  11. data/doc/content/manual.html +449 -0
  12. data/doc/dev/README.developers +55 -0
  13. data/doc/dev/checklist +23 -0
  14. data/doc/dev/things-to-do +5 -0
  15. data/doc/images/log4r-logo.png +0 -0
  16. data/doc/images/logo2.png +0 -0
  17. data/doc/log4r.css +111 -0
  18. data/doc/rdoc-log4r.css +696 -0
  19. data/doc/templates/main.html +147 -0
  20. data/examples/README +19 -0
  21. data/examples/ancestors.rb +53 -0
  22. data/examples/chainsaw_settings.xml +7 -0
  23. data/examples/color_output.rb +26 -0
  24. data/examples/customlevels.rb +34 -0
  25. data/examples/filelog.rb +25 -0
  26. data/examples/fileroll.rb +40 -0
  27. data/examples/gmail.rb +30 -0
  28. data/examples/gmail.yaml +95 -0
  29. data/examples/log4r_yaml.yaml +0 -0
  30. data/examples/logclient.rb +25 -0
  31. data/examples/logserver.rb +18 -0
  32. data/examples/moderate.xml +29 -0
  33. data/examples/moderateconfig.rb +66 -0
  34. data/examples/myformatter.rb +23 -0
  35. data/examples/outofthebox.rb +21 -0
  36. data/examples/rdoc-gen +2 -0
  37. data/examples/rrconfig.xml +63 -0
  38. data/examples/rrsetup.rb +42 -0
  39. data/examples/simpleconfig.rb +39 -0
  40. data/examples/syslogcustom.rb +52 -0
  41. data/examples/xmlconfig.rb +25 -0
  42. data/examples/yaml.rb +30 -0
  43. data/lib/log4r.rb +20 -0
  44. data/lib/log4r/GDC.rb +41 -0
  45. data/lib/log4r/MDC.rb +59 -0
  46. data/lib/log4r/NDC.rb +86 -0
  47. data/lib/log4r/base.rb +90 -0
  48. data/lib/log4r/config.rb +9 -0
  49. data/lib/log4r/configurator.rb +224 -0
  50. data/lib/log4r/formatter/formatter.rb +105 -0
  51. data/lib/log4r/formatter/log4jxmlformatter.rb +61 -0
  52. data/lib/log4r/formatter/patternformatter.rb +145 -0
  53. data/lib/log4r/lib/drbloader.rb +52 -0
  54. data/lib/log4r/lib/xmlloader.rb +24 -0
  55. data/lib/log4r/logevent.rb +28 -0
  56. data/lib/log4r/logger.rb +199 -0
  57. data/lib/log4r/loggerfactory.rb +89 -0
  58. data/lib/log4r/logserver.rb +28 -0
  59. data/lib/log4r/outputter/consoleoutputters.rb +49 -0
  60. data/lib/log4r/outputter/datefileoutputter.rb +117 -0
  61. data/lib/log4r/outputter/emailoutputter.rb +143 -0
  62. data/lib/log4r/outputter/fileoutputter.rb +56 -0
  63. data/lib/log4r/outputter/iooutputter.rb +55 -0
  64. data/lib/log4r/outputter/outputter.rb +146 -0
  65. data/lib/log4r/outputter/outputterfactory.rb +61 -0
  66. data/lib/log4r/outputter/remoteoutputter.rb +40 -0
  67. data/lib/log4r/outputter/rollingfileoutputter.rb +234 -0
  68. data/lib/log4r/outputter/scribeoutputter.rb +37 -0
  69. data/lib/log4r/outputter/staticoutputter.rb +32 -0
  70. data/lib/log4r/outputter/syslogoutputter.rb +130 -0
  71. data/lib/log4r/outputter/udpoutputter.rb +53 -0
  72. data/lib/log4r/rdoc/GDC +14 -0
  73. data/lib/log4r/rdoc/MDC +16 -0
  74. data/lib/log4r/rdoc/NDC +41 -0
  75. data/lib/log4r/rdoc/configurator +243 -0
  76. data/lib/log4r/rdoc/emailoutputter +103 -0
  77. data/lib/log4r/rdoc/formatter +39 -0
  78. data/lib/log4r/rdoc/log4r +89 -0
  79. data/lib/log4r/rdoc/logger +175 -0
  80. data/lib/log4r/rdoc/logserver +85 -0
  81. data/lib/log4r/rdoc/outputter +108 -0
  82. data/lib/log4r/rdoc/patternformatter +128 -0
  83. data/lib/log4r/rdoc/scribeoutputter +16 -0
  84. data/lib/log4r/rdoc/syslogoutputter +29 -0
  85. data/lib/log4r/rdoc/win32eventoutputter +7 -0
  86. data/lib/log4r/rdoc/yamlconfigurator +20 -0
  87. data/lib/log4r/repository.rb +88 -0
  88. data/lib/log4r/staticlogger.rb +49 -0
  89. data/lib/log4r/yamlconfigurator.rb +196 -0
  90. data/tests/README +10 -0
  91. data/tests/testGDC.rb +26 -0
  92. data/tests/testMDC.rb +42 -0
  93. data/tests/testNDC.rb +27 -0
  94. data/tests/testall.rb +6 -0
  95. data/tests/testbase.rb +49 -0
  96. data/tests/testchainsaw.rb +48 -0
  97. data/tests/testcoloroutput.rb +14 -0
  98. data/tests/testconf.xml +37 -0
  99. data/tests/testcustom.rb +27 -0
  100. data/tests/testformatter.rb +27 -0
  101. data/tests/testlogger.rb +196 -0
  102. data/tests/testoutputter.rb +132 -0
  103. data/tests/testpatternformatter.rb +78 -0
  104. data/tests/testthreads.rb +35 -0
  105. data/tests/testxmlconf.rb +45 -0
  106. metadata +195 -0
data/lib/log4r.rb ADDED
@@ -0,0 +1,20 @@
1
+ # :include: log4r/rdoc/log4r
2
+ #
3
+ # == Other Info
4
+ #
5
+ # Author:: Leon Torres
6
+ # Version:: $Id$
7
+
8
+ require "log4r/outputter/fileoutputter"
9
+ require "log4r/outputter/consoleoutputters"
10
+ require "log4r/outputter/staticoutputter"
11
+ require "log4r/outputter/rollingfileoutputter"
12
+ require "log4r/formatter/patternformatter"
13
+ require "log4r/loggerfactory"
14
+ require "log4r/GDC"
15
+ require "log4r/NDC"
16
+ require "log4r/MDC"
17
+
18
+ module Log4r
19
+ Log4rVersion = [1, 1, 9].join '.'
20
+ end
data/lib/log4r/GDC.rb ADDED
@@ -0,0 +1,41 @@
1
+ # :include: rdoc/GDC
2
+ #
3
+ # == Other Info
4
+ #
5
+ # Version:: $Id$
6
+ # Author:: Colby Gutierrez-Kraybill <colby(at)astro.berkeley.edu>
7
+
8
+ require 'monitor'
9
+
10
+ module Log4r
11
+ GDCNAME = "log4rGDC"
12
+ $globalGDCLock = Monitor.new
13
+
14
+ # See log4r/GDC.rb
15
+ class GDC < Monitor
16
+ private_class_method :new
17
+
18
+ def self.clear()
19
+ Thread.main[GDCNAME] = ""
20
+ end
21
+
22
+ def self.get()
23
+ $globalGDCLock.synchronize do
24
+ if ( Thread.main[GDCNAME] == nil ) then
25
+ Thread.main[GDCNAME] = $0
26
+ end
27
+ end
28
+ return Thread.main[GDCNAME]
29
+ end
30
+
31
+ def self.set( a_name )
32
+ if ( Thread.current != Thread.main ) then
33
+ raise "Can only initialize Global Diagnostic Context from Thread.main"
34
+ end
35
+ $globalGDCLock.synchronize do
36
+ Thread.main[GDCNAME] = a_name
37
+ end
38
+ end
39
+ end
40
+ end
41
+
data/lib/log4r/MDC.rb ADDED
@@ -0,0 +1,59 @@
1
+ # :include: rdoc/MDC
2
+ #
3
+ # == Other Info
4
+ #
5
+ # Version:: $Id$
6
+ # Author:: Colby Gutierrez-Kraybill <colby(at)astro.berkeley.edu>
7
+
8
+ require 'monitor'
9
+
10
+ module Log4r
11
+ MDCNAME = "log4rMDC"
12
+ MDCNAMEMAXDEPTH = "log4rMDCMAXDEPTH"
13
+ $globalMDCLock = Monitor.new
14
+
15
+ # See log4r/MDC.rb
16
+ class MDC < Monitor
17
+ private_class_method :new
18
+
19
+ def self.check_thread_instance()
20
+ # need to interlock here, so that if
21
+ # another thread is entering this section
22
+ # of code before the main thread does,
23
+ # then the main thread copy of the MDC
24
+ # is setup before then attempting to clone
25
+ # it off
26
+ if ( Thread.current[MDCNAME] == nil ) then
27
+ $globalMDCLock.synchronize do
28
+ if ( Thread.main[MDCNAME] == nil ) then
29
+ Thread.main[MDCNAME] = Hash.new
30
+ end
31
+ if ( Thread.current != Thread.main ) then
32
+ Thread.current[MDCNAME] = Hash.new
33
+ Thread.main[MDCNAME].each{ |k,v| Thread.current[MDCNAME][k] = v }
34
+ end
35
+ end
36
+ end
37
+ end
38
+
39
+ def self.get( a_key )
40
+ self.check_thread_instance()
41
+ Thread.current[MDCNAME].fetch(a_key, "");
42
+ end
43
+
44
+ def self.get_context()
45
+ self.check_thread_instance()
46
+ return Thread.current[MDCNAME].clone
47
+ end
48
+
49
+ def self.put( a_key, a_value )
50
+ self.check_thread_instance()
51
+ Thread.current[MDCNAME][a_key] = a_value
52
+ end
53
+
54
+ def self.remove( a_key )
55
+ self.check_thread_instance()
56
+ Thread.current[MDCNAME].delete( a_key )
57
+ end
58
+ end
59
+ end
data/lib/log4r/NDC.rb ADDED
@@ -0,0 +1,86 @@
1
+ # :include: rdoc/NDC
2
+ #
3
+ # == Other Info
4
+ #
5
+ # Version:: $Id$
6
+ # Author:: Colby Gutierrez-Kraybill <colby(at)astro.berkeley.edu>
7
+
8
+ module Log4r
9
+ NDCNAME = "log4rNDC"
10
+ NDCNAMEMAXDEPTH = "log4rNDCMAXDEPTH"
11
+ NDCDEFAULTMAXDEPTH = 256
12
+
13
+ # See log4r/NDC.rb
14
+ class NDC
15
+ private_class_method :new
16
+
17
+ def self.check_thread_instance()
18
+ if ( Thread.current[NDCNAME] == nil ) then
19
+ Thread.current[NDCNAME] = Array.new
20
+ Thread.current[NDCNAMEMAXDEPTH] = NDCDEFAULTMAXDEPTH
21
+ end
22
+ end
23
+
24
+ def self.clear()
25
+ self.check_thread_instance()
26
+ Thread.current[NDCNAME].clear
27
+ end
28
+
29
+ def self.clone_stack()
30
+ self.check_thread_instance()
31
+ return Thread.current[NDCNAME].clone
32
+ end
33
+
34
+ def self.get_depth()
35
+ self.check_thread_instance()
36
+ return Thread.current[NDCNAME].length
37
+ end
38
+
39
+ def self.inherit( a_stack )
40
+ if ( a_stack.class == Array ) then
41
+ if ( Thread.current[NDCNAME] != nil ) then
42
+ Thread.current[NDCNAME].clear
43
+ Thread.current[NDCNAME] = nil
44
+ end
45
+ Thread.current[NDCNAME] = a_stack
46
+ else
47
+ raise "Expecting Array in NDC.inherit"
48
+ end
49
+ end
50
+
51
+ def self.get()
52
+ self.check_thread_instance
53
+ return Thread.current[NDCNAME] * " "
54
+ end
55
+
56
+ def self.peek()
57
+ self.check_thread_instance()
58
+ return Thread.current[NDCNAME].last
59
+ end
60
+
61
+ def self.pop()
62
+ self.check_thread_instance()
63
+ return Thread.current[NDCNAME].pop
64
+ end
65
+
66
+ def self.push( value )
67
+ self.check_thread_instance()
68
+ if ( Thread.current[NDCNAME].length < Thread.current[NDCNAMEMAXDEPTH] ) then
69
+ Thread.current[NDCNAME].push( value )
70
+ end
71
+ end
72
+
73
+ def self.remove()
74
+ self.check_thread_instance()
75
+ Thread.current[NDCNAME].clear
76
+ Thread.current[NDCNAMEMAXDEPTH] = nil
77
+ Thread.current[NDCNAME] = nil
78
+ end
79
+
80
+ def self.set_max_depth( max_depth )
81
+ self.check_thread_instance()
82
+ Thread.current[NDCNAMEMAXDEPTH] = max_depth
83
+ end
84
+ end
85
+ end
86
+
data/lib/log4r/base.rb ADDED
@@ -0,0 +1,90 @@
1
+ # :nodoc:
2
+ require "log4r/config"
3
+
4
+ module Log4r
5
+ ALL = 0
6
+ LNAMES = ['ALL']
7
+
8
+ # Defines the log levels of the Log4r module at runtime. It is given
9
+ # either the default level spec (when root logger is created) or the
10
+ # user-specified level spec (when Logger.custom_levels is called).
11
+ #
12
+ # The last constant defined by this method is OFF. Other level-sensitive
13
+ # parts of the code check to see if OFF is defined before deciding what
14
+ # to do. The typical action would be to force the creation of RootLogger
15
+ # so that the custom levels get loaded and business can proceed as usual.
16
+ #
17
+ # For purposes of formatting, a constant named MaxLevelLength is defined
18
+ # in this method. It stores the max level name string size.
19
+
20
+ def Log4r.define_levels(*levels) #:nodoc:
21
+ return if const_defined? :OFF
22
+ for i in 0...levels.size
23
+ name = levels[i].to_s
24
+ module_eval "#{name} = #{i} + 1; LNAMES.push '#{name}'"
25
+ end
26
+ module_eval %{
27
+ LNAMES.push 'OFF'
28
+ LEVELS = LNAMES.size
29
+ OFF = LEVELS - 1
30
+ MaxLevelLength = Log4rTools.max_level_str_size
31
+ }
32
+ end
33
+
34
+ # Some common functions
35
+ class Log4rTools
36
+ # Raises ArgumentError if level argument is an invalid level. Depth
37
+ # specifies how many trace entries to remove.
38
+ def self.validate_level(level, depth=0)
39
+ unless valid_level?(level)
40
+ raise ArgumentError, "Log level must be in 0..#{LEVELS}",
41
+ caller[1..-(depth + 1)]
42
+ end
43
+ end
44
+
45
+ def self.valid_level?(lev)
46
+ not lev.nil? and lev.kind_of?(Numeric) and lev >= ALL and lev <= OFF
47
+ end
48
+
49
+ def self.max_level_str_size #:nodoc:
50
+ size = 0
51
+ LNAMES.each {|i| size = i.length if i.length > size}
52
+ size
53
+ end
54
+
55
+ def self.validate_colors(colors)
56
+ raise ArgumentError, "Colors option must consist of a hash where each key is the log level to define a color scheme for" if !colors.kind_of? Hash
57
+ invalid_levels = colors.keys.reject {|level| valid_levels.include? level}
58
+ if !invalid_levels.empty?
59
+ raise ArgumentError, "Color schemes can not be defined for these invalid log levels: #{invalid_levels.join(', ')}"
60
+ end
61
+ end
62
+
63
+ def self.level_key level
64
+ valid_levels[level]
65
+ end
66
+
67
+ def self.valid_levels
68
+ [:debug, :info, :debug, :warn, :error, :fatal]
69
+ end
70
+
71
+ # Shortcut for decoding 'true', 'false', true, false or nil into a bool
72
+ # from a hash parameter. E.g., it looks for true/false values for
73
+ # the keys 'symbol' and :symbol.
74
+
75
+ def self.decode_bool(hash, symbol, default)
76
+ data = hash[symbol]
77
+ data = hash[symbol.to_s] if data.nil?
78
+ return case data
79
+ when 'true',true then true
80
+ when 'false',false then false
81
+ else default
82
+ end
83
+ end
84
+
85
+ # Splits comma-delimited lists with arbitrary \s padding
86
+ def self.comma_split(string)
87
+ string.split(/\s*,\s*/).collect {|s| s.strip}
88
+ end
89
+ end
90
+ end
@@ -0,0 +1,9 @@
1
+ # :nodoc:
2
+ # Version:: $Id$
3
+
4
+ module Log4r
5
+ module Log4rConfig #:nodoc:
6
+ LogLevels = ['DEBUG', 'INFO', 'WARN', 'ERROR', 'FATAL']
7
+ LoggerPathDelimiter = '::'
8
+ end
9
+ end
@@ -0,0 +1,224 @@
1
+ # :include: rdoc/configurator
2
+ #
3
+ # == Other Info
4
+ #
5
+ # Version:: $Id$
6
+
7
+ require "log4r/logger"
8
+ require "log4r/outputter/staticoutputter"
9
+ require "log4r/lib/xmlloader"
10
+ require "log4r/logserver"
11
+ require "log4r/outputter/remoteoutputter"
12
+
13
+ # TODO: catch unparsed parameters #{FOO} and die
14
+ module Log4r
15
+ # Gets raised when Configurator encounters bad XML.
16
+ class ConfigError < Exception
17
+ end
18
+
19
+ # See log4r/configurator.rb
20
+ class Configurator
21
+ include REXML if HAVE_REXML
22
+ @@params = Hash.new
23
+
24
+ # Get a parameter's value
25
+ def self.[](param); @@params[param] end
26
+ # Define a parameter with a value
27
+ def self.[]=(param, value); @@params[param] = value end
28
+
29
+ # Sets the custom levels. This method accepts symbols or strings.
30
+ #
31
+ # Configurator.custom_levels('My', 'Custom', :Levels)
32
+ #
33
+ # Alternatively, you can specify custom levels in XML:
34
+ #
35
+ # <log4r_config>
36
+ # <pre_config>
37
+ # <custom_levels>
38
+ # My, Custom, Levels
39
+ # </custom_levels>
40
+ # ...
41
+
42
+ def self.custom_levels(*levels)
43
+ return Logger.root if levels.size == 0
44
+ for i in 0...levels.size
45
+ name = levels[i].to_s
46
+ if name =~ /\s/ or name !~ /^[A-Z]/
47
+ raise TypeError, "#{name} is not a valid Ruby Constant name", caller
48
+ end
49
+ end
50
+ Log4r.define_levels *levels
51
+ end
52
+
53
+ # Given a filename, loads the XML configuration for Log4r.
54
+ def self.load_xml_file(filename)
55
+ detect_rexml
56
+ actual_load Document.new(File.new(filename))
57
+ end
58
+
59
+ # You can load a String XML configuration instead of a file.
60
+ def self.load_xml_string(string)
61
+ detect_rexml
62
+ actual_load Document.new(string)
63
+ end
64
+
65
+ #######
66
+ private
67
+ #######
68
+
69
+ def self.detect_rexml
70
+ unless HAVE_REXML
71
+ raise LoadError,
72
+ "Need REXML to load XML configuration", caller[1..-1]
73
+ end
74
+ end
75
+
76
+ def self.actual_load(doc)
77
+ confignode = doc.elements['//log4r_config']
78
+ if confignode.nil?
79
+ raise ConfigError,
80
+ "<log4r_config> element not defined", caller[1..-1]
81
+ end
82
+ decode_xml(confignode)
83
+ end
84
+
85
+ def self.decode_xml(doc)
86
+ decode_pre_config(doc.elements['pre_config'])
87
+ doc.elements.each('outputter') {|e| decode_outputter(e)}
88
+ doc.elements.each('logger') {|e| decode_logger(e)}
89
+ doc.elements.each('logserver') {|e| decode_logserver(e)}
90
+ end
91
+
92
+ def self.decode_pre_config(e)
93
+ return Logger.root if e.nil?
94
+ decode_custom_levels(e.elements['custom_levels'])
95
+ global_config(e.elements['global'])
96
+ global_config(e.elements['root'])
97
+ decode_parameters(e.elements['parameters'])
98
+ e.elements.each('parameter') {|p| decode_parameter(p)}
99
+ end
100
+
101
+ def self.decode_custom_levels(e)
102
+ return Logger.root if e.nil? or e.text.nil?
103
+ begin custom_levels *Log4rTools.comma_split(e.text)
104
+ rescue TypeError => te
105
+ raise ConfigError, te.message, caller[1..-4]
106
+ end
107
+ end
108
+
109
+ def self.global_config(e)
110
+ return if e.nil?
111
+ globlev = e.value_of 'level'
112
+ return if globlev.nil?
113
+ lev = LNAMES.index(globlev) # find value in LNAMES
114
+ Log4rTools.validate_level(lev, 4) # choke on bad level
115
+ Logger.global.level = lev
116
+ end
117
+
118
+ def self.decode_parameters(e)
119
+ e.elements.each{|p| @@params[p.name] = p.text} unless e.nil?
120
+ end
121
+
122
+ def self.decode_parameter(e)
123
+ @@params[e.value_of('name')] = e.value_of 'value'
124
+ end
125
+
126
+ def self.decode_outputter(e)
127
+ # fields
128
+ name = e.value_of 'name'
129
+ type = e.value_of 'type'
130
+ level = e.value_of 'level'
131
+ only_at = e.value_of 'only_at'
132
+ # validation
133
+ raise ConfigError, "Outputter missing name", caller[1..-3] if name.nil?
134
+ raise ConfigError, "Outputter missing type", caller[1..-3] if type.nil?
135
+ Log4rTools.validate_level(LNAMES.index(level)) unless level.nil?
136
+ only_levels = []
137
+ unless only_at.nil?
138
+ for lev in Log4rTools.comma_split(only_at)
139
+ alev = LNAMES.index(lev)
140
+ Log4rTools.validate_level(alev, 3)
141
+ only_levels.push alev
142
+ end
143
+ end
144
+ formatter = decode_formatter(e.elements['formatter'])
145
+ # build the eval string
146
+ buff = "Outputter[name] = #{type}.new name"
147
+ buff += ",:level=>#{LNAMES.index(level)}" unless level.nil?
148
+ buff += ",:formatter=>formatter" unless formatter.nil?
149
+ params = decode_hash_params(e)
150
+ buff += "," + params.join(',') if params.size > 0
151
+ begin eval buff
152
+ rescue Exception => ae
153
+ raise ConfigError,
154
+ "Problem creating outputter: #{ae.message}", caller[1..-3]
155
+ end
156
+ Outputter[name].only_at *only_levels if only_levels.size > 0
157
+ Outputter[name]
158
+ end
159
+
160
+ def self.decode_formatter(e)
161
+ return nil if e.nil?
162
+ type = e.value_of 'type'
163
+ raise ConfigError, "Formatter missing type", caller[1..-4] if type.nil?
164
+ buff = "#{type}.new " + decode_hash_params(e).join(',')
165
+ begin return eval(buff)
166
+ rescue Exception => ae
167
+ raise ConfigError,
168
+ "Problem creating outputter: #{ae.message}", caller[1..-4]
169
+ end
170
+ end
171
+
172
+ ExcludeParams = %w{formatter level name type}
173
+
174
+ # Does the fancy parameter to hash argument transformation
175
+ def self.decode_hash_params(e)
176
+ buff = []
177
+ e.attributes.each_attribute {|p|
178
+ next if ExcludeParams.include? p.name
179
+ buff << ":" + p.name + "=>" + paramsub(p.value)
180
+ }
181
+ e.elements.each {|p|
182
+ next if ExcludeParams.include? p.name
183
+ buff << ":" + p.name + "=>" + paramsub(p.text)
184
+ }
185
+ buff
186
+ end
187
+
188
+ # Substitues any #{foo} in the XML with Parameter['foo']
189
+ def self.paramsub(str)
190
+ return nil if str.nil?
191
+ @@params.each {|param, value| str.sub! '#{'+param+'}', value}
192
+ "'" + str + "'"
193
+ end
194
+
195
+ def self.decode_logger(e)
196
+ l = Logger.new e.value_of('name')
197
+ decode_logger_common(l, e)
198
+ end
199
+
200
+ def self.decode_logserver(e)
201
+ return unless HAVE_REXML
202
+ name = e.value_of 'name'
203
+ uri = e.value_of 'uri'
204
+ l = LogServer.new name, uri
205
+ decode_logger_common(l, e)
206
+ end
207
+
208
+ def self.decode_logger_common(l, e)
209
+ level = e.value_of 'level'
210
+ additive = e.value_of 'additive'
211
+ trace = e.value_of 'trace'
212
+ l.level = LNAMES.index(level) unless level.nil?
213
+ l.additive = additive unless additive.nil?
214
+ l.trace = trace unless trace.nil?
215
+ # and now for outputters
216
+ outs = e.value_of 'outputters'
217
+ Log4rTools.comma_split(outs).each {|n| l.add n.strip} unless outs.nil?
218
+ e.elements.each('outputter') {|e|
219
+ name = (e.value_of 'name' or e.text)
220
+ l.add Outputter[name]
221
+ }
222
+ end
223
+ end
224
+ end