252 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Ruby
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			252 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Ruby
		
	
	
		
			Executable File
		
	
	
	
	
| #!/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 |o|
 | |
|       o.banner = 'Usage: unity_to_junit.rb [options]'
 | |
| 
 | |
|       o.separator ''
 | |
|       o.separator 'Specific options:'
 | |
| 
 | |
|       o.on('-r', '--results <dir>', 'Look for Unity Results files here.') do |results|
 | |
|         # puts "results #{results}"
 | |
|         options.results_dir = results
 | |
|       end
 | |
| 
 | |
|       o.on('-p', '--root_path <path>', 'Prepend this path to files in results.') do |root_path|
 | |
|         options.root_path = root_path
 | |
|       end
 | |
| 
 | |
|       o.on('-o', '--output <filename>', 'XML file to generate.') do |out_file|
 | |
|         # puts "out_file: #{out_file}"
 | |
|         options.out_file = out_file
 | |
|       end
 | |
| 
 | |
|       o.separator ''
 | |
|       o.separator 'Common options:'
 | |
| 
 | |
|       # No argument, shows at tail.  This will print an options summary.
 | |
|       o.on_tail('-h', '--help', 'Show this message') do
 | |
|         puts o
 | |
|         exit
 | |
|       end
 | |
| 
 | |
|       # Another typical switch to print the version.
 | |
|       o.on_tail('--version', 'Show version') do
 | |
|         puts "unity_to_junit.rb version #{VERSION}"
 | |
|         exit
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     opts.parse!(args)
 | |
|     options
 | |
|   end
 | |
| end
 | |
| 
 | |
| class UnityToJUnit
 | |
|   include FileUtils::Verbose
 | |
|   attr_reader :report, :total_tests, :failures, :ignored
 | |
|   attr_writer :targets, :root, :out_file
 | |
| 
 | |
|   def initialize
 | |
|     @report = ''
 | |
|     @unit_name = ''
 | |
|   end
 | |
| 
 | |
|   def run
 | |
|     # Clean up result file names
 | |
|     results = @targets.map { |target| target.tr('\\', '/') }
 | |
|     # 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(&:chomp)
 | |
| 
 | |
|       raise "Empty test result file: #{result_file}" if lines.empty?
 | |
| 
 | |
|       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])
 | |
| 
 | |
|       # use line[0] from the test output to get the test_file path and name
 | |
|       test_file_str = lines[0].tr('\\', '/')
 | |
|       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 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 = results_structure
 | |
|     lines.each do |line|
 | |
|       line = line.tr('\\', '/')
 | |
|       _src_file, src_line, test_name, status, msg = line.split(/:/)
 | |
|       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
 | |
|     results
 | |
|   end
 | |
| 
 | |
|   def parse_test_summary(summary)
 | |
|     raise "Couldn't parse test results: #{summary}" unless summary.find { |v| v =~ /(\d+) Tests (\d+) Failures (\d+) Ignored/ }
 | |
| 
 | |
|     [Regexp.last_match(1).to_i, Regexp.last_match(2).to_i, Regexp.last_match(3).to_i]
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def results_structure
 | |
|     {
 | |
|       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|
 | |
|       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
 | |
| 
 | |
| if $0 == __FILE__
 | |
|   # 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.tr('\\', '/')}**/*.test*"
 | |
| 
 | |
|     results = Dir[targets]
 | |
| 
 | |
|     raise "No *.testpass, *.testfail, or *.testresults files found in '#{targets}'" if results.empty?
 | |
| 
 | |
|     utj.targets = results
 | |
| 
 | |
|     # set the root path
 | |
|     utj.root = options.root_path
 | |
| 
 | |
|     # set the output XML file name
 | |
|     # puts "Output File from options: #{options.out_file}"
 | |
|     utj.out_file = options.out_file
 | |
| 
 | |
|     # run the summarizer
 | |
|     puts utj.run
 | |
|   rescue StandardError => e
 | |
|     utj.usage e.message
 | |
|   end
 | |
| end
 |