123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264 |
- #!/usr/bin/ruby
- #
- # unity_to_junit.rb
- #
- require 'fileutils'
- require 'optparse'
- require 'ostruct'
- require 'set'
- require 'pp'
- VERSION = 1.0
- class ArgvParser
- #
- # Return a structure describing the options.
- #
- def self.parse(args)
- # The options specified on the command line will be collected in *options*.
- # We set default values here.
- options = OpenStruct.new
- options.results_dir = "."
- options.root_path = "."
- options.out_file = "results.xml"
- opts = OptionParser.new do |opts|
- opts.banner = "Usage: unity_to_junit.rb [options]"
- opts.separator ""
- opts.separator "Specific options:"
- opts.on("-r", "--results <dir>", "Look for Unity Results files here.") do |results|
- #puts "results #{results}"
- options.results_dir = results
- end
- opts.on("-p", "--root_path <path>", "Prepend this path to files in results.") do |root_path|
- options.root_path = root_path
- end
- opts.on("-o", "--output <filename>", "XML file to generate.") do |out_file|
- #puts "out_file: #{out_file}"
- options.out_file = out_file
- end
- opts.separator ""
- opts.separator "Common options:"
- # No argument, shows at tail. This will print an options summary.
- opts.on_tail("-h", "--help", "Show this message") do
- puts opts
- exit
- end
- # Another typical switch to print the version.
- opts.on_tail("--version", "Show version") do
- puts "unity_to_junit.rb version #{VERSION}"
- exit
- end
- end
- opts.parse!(args)
- options
- end # parse()
- end # class OptparseExample
- class UnityToJUnit
- include FileUtils::Verbose
- attr_reader :report, :total_tests, :failures, :ignored
- def initialize
- @report = ''
- @unit_name = ''
- end
- def run
- # Clean up result file names
- results = @targets.map {|target| target.gsub(/\\/,"/")}
- #puts "Output File: #{@out_file}"
- f = File.new(@out_file, "w")
- write_xml_header(f)
- write_suites_header( f )
- results.each do |result_file|
- lines = File.readlines(result_file).map { |line| line.chomp }
- if lines.length == 0
- raise "Empty test result file: #{result_file}"
- else
- result_output = get_details(result_file, lines)
- tests,failures,ignored = parse_test_summary(lines)
- result_output[:counts][:total] = tests
- result_output[:counts][:failed] = failures
- result_output[:counts][:ignored] = ignored
- result_output[:counts][:passed] = (result_output[:counts][:total] - result_output[:counts][:failed] - result_output[:counts][:ignored])
- end
- #use line[0] from the test output to get the test_file path and name
- test_file_str = lines[0].gsub("\\","/")
- test_file_str = test_file_str.split(":")
- test_file = if (test_file_str.length < 2)
- result_file
- else
- test_file_str[0] + ':' + test_file_str[1]
- end
- result_output[:source][:path] = File.dirname(test_file)
- result_output[:source][:file] = File.basename(test_file)
- # save result_output
- @unit_name = File.basename(test_file, ".*")
- write_suite_header( result_output[:counts], f)
- write_failures( result_output, f )
- write_tests( result_output, f )
- write_ignored( result_output, f )
- write_suite_footer( f )
- end
- write_suites_footer( f )
- f.close
- end
- def set_targets(target_array)
- @targets = target_array
- end
- def set_root_path(path)
- @root = path
- end
- def set_out_file(filename)
- @out_file = filename
- end
- def usage(err_msg=nil)
- puts "\nERROR: "
- puts err_msg if err_msg
- puts "Usage: unity_to_junit.rb [options]"
- puts ""
- puts "Specific options:"
- puts " -r, --results <dir> Look for Unity Results files here."
- puts " -p, --root_path <path> Prepend this path to files in results."
- puts " -o, --output <filename> XML file to generate."
- puts ""
- puts "Common options:"
- puts " -h, --help Show this message"
- puts " --version Show version"
- exit 1
- end
- protected
- def get_details(result_file, lines)
- results = get_results_structure
- lines.each do |line|
- line = line.gsub("\\","/")
- src_file,src_line,test_name,status,msg = line.split(/:/)
- line_out = ((@root and (@root != 0)) ? "#{@root}#{line}" : line ).gsub(/\//, "\\")
- case(status)
- when 'IGNORE' then results[:ignores] << {:test => test_name, :line => src_line, :message => msg}
- when 'FAIL' then results[:failures] << {:test => test_name, :line => src_line, :message => msg}
- when 'PASS' then results[:successes] << {:test => test_name, :line => src_line, :message => msg}
- end
- end
- return results
- end
- def parse_test_summary(summary)
- if summary.find { |v| v =~ /(\d+) Tests (\d+) Failures (\d+) Ignored/ }
- [$1.to_i,$2.to_i,$3.to_i]
- else
- raise "Couldn't parse test results: #{summary}"
- end
- end
- def here; File.expand_path(File.dirname(__FILE__)); end
- private
- def get_results_structure
- return {
- :source => {:path => '', :file => ''},
- :successes => [],
- :failures => [],
- :ignores => [],
- :counts => {:total => 0, :passed => 0, :failed => 0, :ignored => 0},
- :stdout => [],
- }
- end
- def write_xml_header( stream )
- stream.puts "<?xml version='1.0' encoding='utf-8' ?>"
- end
- def write_suites_header( stream )
- stream.puts "<testsuites>"
- end
- def write_suite_header( counts, stream )
- stream.puts "\t<testsuite errors=\"0\" skipped=\"#{counts[:ignored]}\" failures=\"#{counts[:failed]}\" tests=\"#{counts[:total]}\" name=\"unity\">"
- end
- def write_failures( results, stream )
- result = results[:failures]
- result.each do |item|
- filename = File.join(results[:source][:path], File.basename(results[:source][:file], '.*'))
- stream.puts "\t\t<testcase classname=\"#{@unit_name}\" name=\"#{item[:test]}\" time=\"0\">"
- stream.puts "\t\t\t<failure message=\"#{item[:message]}\" type=\"Assertion\"/>"
- stream.puts "\t\t\t<system-err>
[File] #{filename}
[Line] #{item[:line]}
</system-err>"
- stream.puts "\t\t</testcase>"
- end
- end
- def write_tests( results, stream )
- result = results[:successes]
- result.each do |item|
- filename = File.join(results[:source][:path], File.basename(results[:source][:file], '.*'))
- stream.puts "\t\t<testcase classname=\"#{@unit_name}\" name=\"#{item[:test]}\" time=\"0\" />"
- end
- end
- def write_ignored( results, stream )
- result = results[:ignores]
- result.each do |item|
- filename = File.join(results[:source][:path], File.basename(results[:source][:file], '.*'))
- puts "Writing ignored tests for test harness: #{filename}"
- stream.puts "\t\t<testcase classname=\"#{@unit_name}\" name=\"#{item[:test]}\" time=\"0\">"
- stream.puts "\t\t\t<skipped message=\"#{item[:message]}\" type=\"Assertion\"/>"
- stream.puts "\t\t\t<system-err>
[File] #{filename}
[Line] #{item[:line]}
</system-err>"
- stream.puts "\t\t</testcase>"
- end
- end
- def write_suite_footer( stream )
- stream.puts "\t</testsuite>"
- end
- def write_suites_footer( stream )
- stream.puts "</testsuites>"
- end
- end #UnityToJUnit
- if __FILE__ == $0
- #parse out the command options
- options = ArgvParser.parse(ARGV)
- #create an instance to work with
- utj = UnityToJUnit.new
- begin
- #look in the specified or current directory for result files
- targets = "#{options.results_dir.gsub(/\\/, '/')}**/*.test*"
- results = Dir[targets]
- raise "No *.testpass, *.testfail, or *.testresults files found in '#{targets}'" if results.empty?
- utj.set_targets(results)
- #set the root path
- utj.set_root_path(options.root_path)
- #set the output XML file name
- #puts "Output File from options: #{options.out_file}"
- utj.set_out_file(options.out_file)
- #run the summarizer
- puts utj.run
- rescue Exception => e
- utj.usage e.message
- end
- end
|