diff options
author | Steve Kemp <steve@steve.org.uk> | 2012-11-14 16:41:25 +0000 |
---|---|---|
committer | Steve Kemp <steve@steve.org.uk> | 2012-11-14 16:41:25 +0000 |
commit | fc75adb85d83cebc21bc7e6831d1eddebf6ce4bd (patch) | |
tree | 75dee415c06da7e0bac61c5aa9360fea69c39be9 | |
parent | 11286fd4a12f018589e3e16059a4a6f7a165c5ed (diff) |
Moved parser code into its own file, and updated main script + test suite to use it.
-rwxr-xr-x | bin/custodian-enqueue | 453 | ||||
-rwxr-xr-x | lib/custodian/parser.rb | 451 | ||||
-rwxr-xr-x | t/test-parser.rb | 5 |
3 files changed, 458 insertions, 451 deletions
diff --git a/bin/custodian-enqueue b/bin/custodian-enqueue index 23121d3..23083c3 100755 --- a/bin/custodian-enqueue +++ b/bin/custodian-enqueue @@ -45,457 +45,11 @@ # -require 'beanstalk-client' -require 'getoptlong' -require 'json' -require 'net/http' -require 'net/https' -require 'uri' - - - - - - - -# -# This is a simple class which will parse a sentinel configuration file. -# -# Unlike Sentinel it is not using a real parser, instead it peels off lines -# via a small number of very simple regular expressions - this should be flaky, -# but in practice it manages to successfully parse each of the configuration -# files that we currently maintain @ Bytemark. -# -# TODO: # -# 1. Explicitly abort and panic on malformed lines. +# Implementation of our parser. # -# -# Steve -# -- -# -class MonitorConfig - - # - # A hash of macros we found. - # - attr_reader :MACROS - - # - # A handle to the beanstalkd queue. - # - attr_reader :queue - - # - # The filename that we're going to parse. - # - attr_reader :filename - - # - # Timeout period, in seconds, that we encode into test objects. - # - attr_reader :timeout - - - - # - # Constructor - # - def initialize( filename ) - - - @MACROS = Hash.new() - @queue = Beanstalk::Pool.new(['127.0.0.1:11300']) - @file = filename - @timeout = 3 - - raise ArgumentError, "Missing configuration file!" if ( @file.nil? ) - raise ArgumentError, "File not found: #{@file}" unless ( File.exists?( @file) ) - end - - - def get_timeout() - @timeout - end - - def set_timeout( new_val ) - @timeout = new_val - end - - - # - # Retrieve a HTTP page from the web - this is used for macro-expansion - # - # NOTE: This came from sentinel. - # - def getURL (uri_str) - begin - uri_str = 'http://' + uri_str unless uri_str.match(/^http/) - url = URI.parse(uri_str) - http = Net::HTTP.new(url.host, url.port) - http.open_timeout = @timeout - http.read_timeout = @timeout - - if (url.scheme == "https") - http.use_ssl = true - http.verify_mode = OpenSSL::SSL::VERIFY_NONE - end - - response = nil - - if nil == url.query - response = http.start { http.get(url.path) } - else - response = http.start { http.get("#{url.path}?#{url.query}") } - end - - - if ( response.code.to_i != 200 ) - puts "Status code of #{uri_str} was #{response.code}" - puts "ABORTING" - exit( 0 ) - end - - case response - when Net::HTTPRedirection - then - newURL = response['location'].match(/^http/)? - response['Location']:uri_str+response['Location'] - return( getURL(newURL) ) - else - return response.body - end - - rescue Errno::EHOSTUNREACH => ex - raise ex, "no route to host" - rescue Timeout::Error => ex - raise ex, "timeout" - rescue Errno::ECONNREFUSED => ex - raise ex, "connection refused" - end - end - - - - # - # Define a macro, from the configuration file. - # - def define_macro( line ) - name = nil - val = Array.new - - # - # Get the name - # - name = $1.dup if ( line =~ /^([1-2A-Z_]+)\s+/ ) - - - # - # Get the value - # - if ( line =~ /fetched\s+from\s+(.*)[\r\n\.]*$/ ) - - # - # HTTP-fetch - # - uri = $1.dup.chomp(".") - - text = getURL(uri) - text.split( /[\r\n]/ ).each do |line| - val.push( line ) if ( line.length() > 0) - end - - elsif ( line =~ /\s(is|are)\s+(.*)\.+$/ ) - - # - # Literal list of hosts - # - hosts = $2.dup - - # - # If there is " and " then tokenize - # - if ( hosts =~ /\s+and\s+/ ) - tmp = hosts.split( /\s+and\s+/ ) - tmp.each do |entry| - val.push( entry ) - end - else - # - # Otherwise a single host - # - val.push( hosts ) - end - - - end - - @MACROS[name] = val - end - - - - - # - # Return a hash of our current macro-definitions. - # - # This is used only by the test-suite. - # - def macros - @MACROS - end - - - - - # - # Is the given string of text a macro? - # - def is_macro?( name ) - !(@MACROS[name]).nil? - end - - - - # - # Return an array of hosts if the given string was a macro identifier. - # - def get_macro_targets( name ) - @MACROS[name] - end - - - - - # - # Parse a single line from the configuration file. - # - def parse_line( line ) - - line.chomp! if ( !line.nil? ) - - # - # A blank line, or a comment may be skipped. - # - return nil if ( ( line.nil? ) || ( line =~ /^#/ ) || ( line.length < 1 ) ) - - # - # The specification of mauve-server to which we should raise our alerts to. - # - return nil if ( line =~ /Mauve\s+server(.*)source/ ) - - - # - # Look for macro definitions, inline - # - if ( line =~ /^([0-9A-Z]_+)\s+are\s+fetched\s+from\s+([^\s]+)\.?/ ) - define_macro( line ) - - elsif ( line =~ /^([0-9A-Z_]+)\s+(is|are)\s+/ ) - define_macro( line ) - - elsif ( line =~ /(.*)\s+must\s+ping(.*)/ ) - - # - # Ping is a special case because the configuration file entry - # would read: - # - # $FOO must ping otherwise ... - # - # All other tests are of the form: - # - # $FOO must run XXX ... otherwise ... - # - # If we clevery rewrite the line into: - # - # ... must run ping ... - # - # We can avoid duplicating the macro-expansion, etc. - # - pre = $1.dup - post = $2.dup - new_line = "#{pre} must run ping #{post}" - return( parse_line( new_line ) ) - - elsif ( line =~ /\s+must\s+run\s+([^\s]+)(\s+|\.)/i ) - - # - # Get the service we're testing, and remove any trailing "." - # - # This handles the case of: - # - # LINN_HOSTS must run ssh. - # - service = $1.dup - service.chomp!(".") - - # - # Target of the service-test. - # - targets = Array.new - target = line.split( /\s+/)[0] - - # - # If the target is a macro then get the list of hosts to - # which the test will apply. - # - if ( is_macro?( target ) ) - targets = get_macro_targets( target ) - else - - # - # Otherwise a list of one, literal, entry. - # - targets.push( target ) - end - - # - # Alert text will have a default, which may be overridden. - # - alert = "#{service} failed" - if ( line =~ /otherwise '([^']+)'/ ) - alert=$1.dup - end - - # - # All our service tests, except ping, require a port - we setup the defaults here, - # but the configuration file will allow users to specify an alternative - # via " on XXX ". - # - case service - when /ssh/ then - port=22 - when /jabber/ then - port=5222 - when /ldap/ then - port=389 - when /^https$/ then - port=443 - when /^http$/ then - port=80 - when /rsync/i then - port=873 - when /ftp/i then - port=21 - when /telnet/i then - port=23 - when /smtp/i then - port=25 - when /dns/i then - port=53 - end - - # - # Allow the port to be changed, for example: - # - # must run ssh on 33 otherwise .. - # must run ftp on 44 otherwise .. - # must run http on 8000 otherwise .. - # - if ( line =~ /\s+on\s+([0-9]+)/ ) - port = $1.dup - end - - - ret = Array.new() - - # - # For each host in our possibly-macro-expanded list: - # - targets.each do |host| - - # - # The test we'll apply. - # - test = { - :target_host => host, - :test_type => service, - :test_port => port, - :test_alert => alert, - :timeout => @timeout - } - - - # - # HTTP-tests will include the expected result in one of two forms: - # - # must run http with status 200 - # - # must run http with content 'text' - # - # If those are sepcified then include them here. - # - # Note we're deliberately fast and loose here - which allows both to - # be specified: - # - # http://example.vm/ must run http with status 200 and content 'OK'. - # - # - if ( line =~ /\s+with\s+status\s+([0-9]+)\s+/ ) - test[:http_status]=$1.dup - end - if ( line =~ /\s+with\s+content\s+'([^']+)'/ ) - test[:http_text]=$1.dup - end - - # - # These are special cased for the DNS types - # - if ( test[:test_type] =~ /dns/ ) - - # - # Sample line: - # - # DNSHOSTS must run dns for www.bytemark.co.uk resolving A as '212.110.161.144'. - # - # - if ( line =~ /for\s+([^\s]+)\sresolving\s([A-Z]+)\s+as\s'([^']+)'/ ) - test[:resolve_name] = $1.dup - test[:resolve_type] = $2.dup - test[:resolve_expected] = $3.dup - end - end - - - # - # We've now parsed the line. Either output the JSON to the console - # or add to the queue. - # - if ( !ENV['DUMP'].nil? ) - puts ( test.to_json ) - else - @queue.put( test.to_json ) - end - - ret.push( test.to_json ) - end - - ret - else - puts "Unknown line: '#{line}'" - end - end - - - - - # - # Parse the configuration file which was named in our constructor. - # - def parse_file() - # - # Parse the configuration file on the command line - # - File.open( @file, "r").each_line do |line| - parse_line( line) - end - end - - -end - - +require 'custodian/parser' @@ -580,7 +134,8 @@ if __FILE__ == $0 then # # Run # - mon.parse_file(); + mon.parse_file() + end diff --git a/lib/custodian/parser.rb b/lib/custodian/parser.rb new file mode 100755 index 0000000..a5ce584 --- /dev/null +++ b/lib/custodian/parser.rb @@ -0,0 +1,451 @@ + +require 'beanstalk-client' +require 'getoptlong' +require 'json' +require 'net/http' +require 'net/https' +require 'uri' + + + + + + + + +# +# This is a simple class which will parse a sentinel configuration file. +# +# Unlike Sentinel it is not using a real parser, instead it peels off lines +# via a small number of very simple regular expressions - this should be flaky, +# but in practice it manages to successfully parse each of the configuration +# files that we currently maintain @ Bytemark. +# +# TODO: +# +# 1. Explicitly abort and panic on malformed lines. +# +# +# Steve +# -- +# +class MonitorConfig + + # + # A hash of macros we found. + # + attr_reader :MACROS + + # + # A handle to the beanstalkd queue. + # + attr_reader :queue + + # + # The filename that we're going to parse. + # + attr_reader :filename + + # + # Timeout period, in seconds, that we encode into test objects. + # + attr_reader :timeout + + + + # + # Constructor + # + def initialize( filename ) + + + @MACROS = Hash.new() + @queue = Beanstalk::Pool.new(['127.0.0.1:11300']) + @file = filename + @timeout = 3 + + raise ArgumentError, "Missing configuration file!" if ( @file.nil? ) + raise ArgumentError, "File not found: #{@file}" unless ( File.exists?( @file) ) + end + + + def get_timeout() + @timeout + end + + def set_timeout( new_val ) + @timeout = new_val + end + + + # + # Retrieve a HTTP page from the web - this is used for macro-expansion + # + # NOTE: This came from sentinel. + # + def getURL (uri_str) + begin + uri_str = 'http://' + uri_str unless uri_str.match(/^http/) + url = URI.parse(uri_str) + http = Net::HTTP.new(url.host, url.port) + http.open_timeout = @timeout + http.read_timeout = @timeout + + if (url.scheme == "https") + http.use_ssl = true + http.verify_mode = OpenSSL::SSL::VERIFY_NONE + end + + response = nil + + if nil == url.query + response = http.start { http.get(url.path) } + else + response = http.start { http.get("#{url.path}?#{url.query}") } + end + + + if ( response.code.to_i != 200 ) + puts "Status code of #{uri_str} was #{response.code}" + puts "ABORTING" + exit( 0 ) + end + + case response + when Net::HTTPRedirection + then + newURL = response['location'].match(/^http/)? + response['Location']:uri_str+response['Location'] + return( getURL(newURL) ) + else + return response.body + end + + rescue Errno::EHOSTUNREACH => ex + raise ex, "no route to host" + rescue Timeout::Error => ex + raise ex, "timeout" + rescue Errno::ECONNREFUSED => ex + raise ex, "connection refused" + end + end + + + + # + # Define a macro, from the configuration file. + # + def define_macro( line ) + name = nil + val = Array.new + + # + # Get the name + # + name = $1.dup if ( line =~ /^([1-2A-Z_]+)\s+/ ) + + + # + # Get the value + # + if ( line =~ /fetched\s+from\s+(.*)[\r\n\.]*$/ ) + + # + # HTTP-fetch + # + uri = $1.dup.chomp(".") + + text = getURL(uri) + text.split( /[\r\n]/ ).each do |line| + val.push( line ) if ( line.length() > 0) + end + + elsif ( line =~ /\s(is|are)\s+(.*)\.+$/ ) + + # + # Literal list of hosts + # + hosts = $2.dup + + # + # If there is " and " then tokenize + # + if ( hosts =~ /\s+and\s+/ ) + tmp = hosts.split( /\s+and\s+/ ) + tmp.each do |entry| + val.push( entry ) + end + else + # + # Otherwise a single host + # + val.push( hosts ) + end + + + end + + @MACROS[name] = val + end + + + + + # + # Return a hash of our current macro-definitions. + # + # This is used only by the test-suite. + # + def macros + @MACROS + end + + + + + # + # Is the given string of text a macro? + # + def is_macro?( name ) + !(@MACROS[name]).nil? + end + + + + # + # Return an array of hosts if the given string was a macro identifier. + # + def get_macro_targets( name ) + @MACROS[name] + end + + + + + # + # Parse a single line from the configuration file. + # + def parse_line( line ) + + line.chomp! if ( !line.nil? ) + + # + # A blank line, or a comment may be skipped. + # + return nil if ( ( line.nil? ) || ( line =~ /^#/ ) || ( line.length < 1 ) ) + + # + # The specification of mauve-server to which we should raise our alerts to. + # + return nil if ( line =~ /Mauve\s+server(.*)source/ ) + + + # + # Look for macro definitions, inline + # + if ( line =~ /^([0-9A-Z]_+)\s+are\s+fetched\s+from\s+([^\s]+)\.?/ ) + define_macro( line ) + + elsif ( line =~ /^([0-9A-Z_]+)\s+(is|are)\s+/ ) + define_macro( line ) + + elsif ( line =~ /(.*)\s+must\s+ping(.*)/ ) + + # + # Ping is a special case because the configuration file entry + # would read: + # + # $FOO must ping otherwise ... + # + # All other tests are of the form: + # + # $FOO must run XXX ... otherwise ... + # + # If we clevery rewrite the line into: + # + # ... must run ping ... + # + # We can avoid duplicating the macro-expansion, etc. + # + pre = $1.dup + post = $2.dup + new_line = "#{pre} must run ping #{post}" + return( parse_line( new_line ) ) + + elsif ( line =~ /\s+must\s+run\s+([^\s]+)(\s+|\.)/i ) + + # + # Get the service we're testing, and remove any trailing "." + # + # This handles the case of: + # + # LINN_HOSTS must run ssh. + # + service = $1.dup + service.chomp!(".") + + # + # Target of the service-test. + # + targets = Array.new + target = line.split( /\s+/)[0] + + # + # If the target is a macro then get the list of hosts to + # which the test will apply. + # + if ( is_macro?( target ) ) + targets = get_macro_targets( target ) + else + + # + # Otherwise a list of one, literal, entry. + # + targets.push( target ) + end + + # + # Alert text will have a default, which may be overridden. + # + alert = "#{service} failed" + if ( line =~ /otherwise '([^']+)'/ ) + alert=$1.dup + end + + # + # All our service tests, except ping, require a port - we setup the defaults here, + # but the configuration file will allow users to specify an alternative + # via " on XXX ". + # + case service + when /ssh/ then + port=22 + when /jabber/ then + port=5222 + when /ldap/ then + port=389 + when /^https$/ then + port=443 + when /^http$/ then + port=80 + when /rsync/i then + port=873 + when /ftp/i then + port=21 + when /telnet/i then + port=23 + when /smtp/i then + port=25 + when /dns/i then + port=53 + end + + # + # Allow the port to be changed, for example: + # + # must run ssh on 33 otherwise .. + # must run ftp on 44 otherwise .. + # must run http on 8000 otherwise .. + # + if ( line =~ /\s+on\s+([0-9]+)/ ) + port = $1.dup + end + + + ret = Array.new() + + # + # For each host in our possibly-macro-expanded list: + # + targets.each do |host| + + # + # The test we'll apply. + # + test = { + :target_host => host, + :test_type => service, + :test_port => port, + :test_alert => alert, + :timeout => @timeout + } + + + # + # HTTP-tests will include the expected result in one of two forms: + # + # must run http with status 200 + # + # must run http with content 'text' + # + # If those are sepcified then include them here. + # + # Note we're deliberately fast and loose here - which allows both to + # be specified: + # + # http://example.vm/ must run http with status 200 and content 'OK'. + # + # + if ( line =~ /\s+with\s+status\s+([0-9]+)\s+/ ) + test[:http_status]=$1.dup + end + if ( line =~ /\s+with\s+content\s+'([^']+)'/ ) + test[:http_text]=$1.dup + end + + # + # These are special cased for the DNS types + # + if ( test[:test_type] =~ /dns/ ) + + # + # Sample line: + # + # DNSHOSTS must run dns for www.bytemark.co.uk resolving A as '212.110.161.144'. + # + # + if ( line =~ /for\s+([^\s]+)\sresolving\s([A-Z]+)\s+as\s'([^']+)'/ ) + test[:resolve_name] = $1.dup + test[:resolve_type] = $2.dup + test[:resolve_expected] = $3.dup + end + end + + + # + # We've now parsed the line. Either output the JSON to the console + # or add to the queue. + # + if ( !ENV['DUMP'].nil? ) + puts ( test.to_json ) + else + @queue.put( test.to_json ) + end + + ret.push( test.to_json ) + end + + ret + else + puts "Unknown line: '#{line}'" + end + end + + + + + # + # Parse the configuration file which was named in our constructor. + # + def parse_file() + # + # Parse the configuration file on the command line + # + File.open( @file, "r").each_line do |line| + parse_line( line) + end + end + + +end + diff --git a/t/test-parser.rb b/t/test-parser.rb index e92c5e8..dea0548 100755 --- a/t/test-parser.rb +++ b/t/test-parser.rb @@ -1,9 +1,10 @@ -#!/usr/bin/ruby1.8 -I./bin/ -I../bin/ +#!/usr/bin/ruby1.8 -I./lib/ -I../lib/ require 'json' require 'test/unit' -load 'custodian-enqueue' +require 'custodian/parser' + |