log4r-color 1.1.11
Sign up to get free protection for your applications and to get access to all the features.
- data/INSTALL +11 -0
- data/LICENSE +90 -0
- data/LICENSE.LGPLv3 +165 -0
- data/README +95 -0
- data/Rakefile +80 -0
- data/TODO +2 -0
- data/doc/content/contact.html +22 -0
- data/doc/content/contribute.html +21 -0
- data/doc/content/index.html +90 -0
- data/doc/content/license.html +56 -0
- data/doc/content/manual.html +449 -0
- data/doc/dev/README.developers +55 -0
- data/doc/dev/checklist +23 -0
- data/doc/dev/things-to-do +5 -0
- data/doc/images/log4r-logo.png +0 -0
- data/doc/images/logo2.png +0 -0
- data/doc/log4r.css +111 -0
- data/doc/rdoc-log4r.css +696 -0
- data/doc/templates/main.html +147 -0
- data/examples/README +19 -0
- data/examples/ancestors.rb +53 -0
- data/examples/chainsaw_settings.xml +7 -0
- data/examples/color_output.rb +26 -0
- data/examples/customlevels.rb +34 -0
- data/examples/filelog.rb +25 -0
- data/examples/fileroll.rb +40 -0
- data/examples/gmail.rb +30 -0
- data/examples/gmail.yaml +95 -0
- data/examples/log4r_yaml.yaml +0 -0
- data/examples/logclient.rb +25 -0
- data/examples/logserver.rb +18 -0
- data/examples/moderate.xml +29 -0
- data/examples/moderateconfig.rb +66 -0
- data/examples/myformatter.rb +23 -0
- data/examples/outofthebox.rb +21 -0
- data/examples/rdoc-gen +2 -0
- data/examples/rrconfig.xml +63 -0
- data/examples/rrsetup.rb +42 -0
- data/examples/simpleconfig.rb +39 -0
- data/examples/syslogcustom.rb +52 -0
- data/examples/xmlconfig.rb +25 -0
- data/examples/yaml.rb +30 -0
- data/lib/log4r.rb +20 -0
- data/lib/log4r/GDC.rb +41 -0
- data/lib/log4r/MDC.rb +59 -0
- data/lib/log4r/NDC.rb +86 -0
- data/lib/log4r/base.rb +90 -0
- data/lib/log4r/config.rb +9 -0
- data/lib/log4r/configurator.rb +224 -0
- data/lib/log4r/formatter/formatter.rb +105 -0
- data/lib/log4r/formatter/log4jxmlformatter.rb +61 -0
- data/lib/log4r/formatter/patternformatter.rb +145 -0
- data/lib/log4r/lib/drbloader.rb +52 -0
- data/lib/log4r/lib/xmlloader.rb +24 -0
- data/lib/log4r/logevent.rb +28 -0
- data/lib/log4r/logger.rb +199 -0
- data/lib/log4r/loggerfactory.rb +89 -0
- data/lib/log4r/logserver.rb +28 -0
- data/lib/log4r/outputter/consoleoutputters.rb +49 -0
- data/lib/log4r/outputter/datefileoutputter.rb +117 -0
- data/lib/log4r/outputter/emailoutputter.rb +143 -0
- data/lib/log4r/outputter/fileoutputter.rb +56 -0
- data/lib/log4r/outputter/iooutputter.rb +55 -0
- data/lib/log4r/outputter/outputter.rb +146 -0
- data/lib/log4r/outputter/outputterfactory.rb +61 -0
- data/lib/log4r/outputter/remoteoutputter.rb +40 -0
- data/lib/log4r/outputter/rollingfileoutputter.rb +234 -0
- data/lib/log4r/outputter/scribeoutputter.rb +37 -0
- data/lib/log4r/outputter/staticoutputter.rb +32 -0
- data/lib/log4r/outputter/syslogoutputter.rb +130 -0
- data/lib/log4r/outputter/udpoutputter.rb +53 -0
- data/lib/log4r/rdoc/GDC +14 -0
- data/lib/log4r/rdoc/MDC +16 -0
- data/lib/log4r/rdoc/NDC +41 -0
- data/lib/log4r/rdoc/configurator +243 -0
- data/lib/log4r/rdoc/emailoutputter +103 -0
- data/lib/log4r/rdoc/formatter +39 -0
- data/lib/log4r/rdoc/log4r +89 -0
- data/lib/log4r/rdoc/logger +175 -0
- data/lib/log4r/rdoc/logserver +85 -0
- data/lib/log4r/rdoc/outputter +108 -0
- data/lib/log4r/rdoc/patternformatter +128 -0
- data/lib/log4r/rdoc/scribeoutputter +16 -0
- data/lib/log4r/rdoc/syslogoutputter +29 -0
- data/lib/log4r/rdoc/win32eventoutputter +7 -0
- data/lib/log4r/rdoc/yamlconfigurator +20 -0
- data/lib/log4r/repository.rb +88 -0
- data/lib/log4r/staticlogger.rb +49 -0
- data/lib/log4r/yamlconfigurator.rb +196 -0
- data/tests/README +10 -0
- data/tests/testGDC.rb +26 -0
- data/tests/testMDC.rb +42 -0
- data/tests/testNDC.rb +27 -0
- data/tests/testall.rb +6 -0
- data/tests/testbase.rb +49 -0
- data/tests/testchainsaw.rb +48 -0
- data/tests/testcoloroutput.rb +14 -0
- data/tests/testconf.xml +37 -0
- data/tests/testcustom.rb +27 -0
- data/tests/testformatter.rb +27 -0
- data/tests/testlogger.rb +196 -0
- data/tests/testoutputter.rb +132 -0
- data/tests/testpatternformatter.rb +78 -0
- data/tests/testthreads.rb +35 -0
- data/tests/testxmlconf.rb +45 -0
- 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
|
data/lib/log4r/config.rb
ADDED
@@ -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
|