ianheggie/cruisecontrol.rb

View on GitHub
app/models/build_log_parser.rb

Summary

Maintainability
A
0 mins
Test Coverage
# A BuildLogParser understands how to parse Test::Unit and RSpec test errors generated by a build
# run and turn them into a more succinct error representation.
class BuildLogParser

  TEST_ERROR_REGEX = /^\s+\d+\) Error:\n(.*):\n(.*)\n([\s\S]*?)\n\n/
  TEST_FAILURE_REGEX = /^\s+\d+\) Failure:\n([\S\s]*?)\n\n/

  RSPEC_ERROR_REGEX = /^(\S+) in '(.*)'\n((.*\n)+)/
  RSPEC_FAILURE_REGEX = /^'(.*)' FAILED\n((.+\n)+)/
  RSPEC_STACK_TRACE_REGEX = /^.*:\d+:.*$/
  RSPEC_STACK_TRACE_MAYBE_END_REGEX = /\n\nFinished.*$/
  
  TEST_NAME_REGEX = /\S+/
  MESSAGE_REGEX = /\]\:\n([\s\S]+)/
  STACK_TRACE_REGEX = /\[([\s\S]*?)\]\:/

  def initialize(log)
    @log = log
  end

  def errors
    test_errors + rspec_errors
  end

  def test_errors
    test_errors = []
    
    @log.scan(TEST_ERROR_REGEX) do |match|
      test_errors << TestErrorEntry.create_error($1, $2, $3)
    end
  
    return test_errors
  end

  def rspec_errors
    errors = []
    [@log.split(/\d+\)/)[1..-1]].compact.flatten.each do |issue_content|
      issue_content.chop.scan(RSPEC_ERROR_REGEX) do |match|
        exception_name = $1
        spec_name = $2
        content = $3
        rest_of_the_message, stack_trace = rspec_rest_of_message_and_stack_trace(content)
        message = "#{exception_name} in '#{spec_name}'\n#{rest_of_the_message}"
        errors << TestErrorEntry.create_error(spec_name, message, stack_trace)
      end
    end
    errors
  end

  def failures
    test_failures + rspec_failures
  end
   
  def test_failures
    test_failures = []

    @log.scan(TEST_FAILURE_REGEX) do |text|
      content = $1

      begin
        test_name = content.match(TEST_NAME_REGEX).to_s
        message = content.match(MESSAGE_REGEX)[1]
        stack_trace = content.match(STACK_TRACE_REGEX)[1]

        test_failures << TestErrorEntry.create_failure(test_name, message, stack_trace)
      rescue
        # Do Nothing, Pattern does not match
      end
    end

    test_failures
  end
  
  def rspec_failures
    failures = []
    [@log.split(/\d+\)/)[1..-1]].compact.flatten.each do |issue_content|
      issue_content.chop.scan(RSPEC_FAILURE_REGEX) do |match|
        spec_name = $1
        content = $2
        rest_of_the_message, stack_trace = rspec_rest_of_message_and_stack_trace(content)
        failures << TestErrorEntry.create_failure(spec_name, rest_of_the_message, stack_trace)
      end
    end
    failures
  end  

  def failures_and_errors
    failures + errors
  end

  def rspec_rest_of_message_and_stack_trace(content)
    rest_of_the_message = []
    stack_trace = ""
    content_lines = content.split("\n")
    content_lines.each_with_index do |line, index|
      if line =~ RSPEC_STACK_TRACE_REGEX
        stack_trace << content_lines[index..-1].join("\n").gsub(RSPEC_STACK_TRACE_MAYBE_END_REGEX, "")
        break
      else
        rest_of_the_message << line
      end
    end
    [rest_of_the_message.join("\n"), stack_trace]
  end
end