DEV: Enable color CI output and tweak formatting (#21527)

* Color for turbo_rspec in CI (`progress` and `documentation` formats)
* Show "DONE" only when `documentation` formatter is used
* Fix formatting
* Collapse RSpec commands
* Add line wrapping to the `progress` formatter (to mitigate GH Actions issue)
This commit is contained in:
Jarek Radosz 2023-05-12 18:22:15 +02:00 committed by GitHub
parent b32cdb0880
commit eec10efc3d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 83 additions and 7 deletions

View File

@ -180,7 +180,9 @@ def spec(plugin, parallel: false, argv: nil)
Dir.glob("./plugins/#{plugin}/spec/**/*_spec.rb").reject { |f| f.include?("spec/system/") }.sort Dir.glob("./plugins/#{plugin}/spec/**/*_spec.rb").reject { |f| f.include?("spec/system/") }.sort
if files.length > 0 if files.length > 0
cmd = parallel ? "bin/turbo_rspec" : "bin/rspec" cmd = parallel ? "bin/turbo_rspec" : "bin/rspec"
sh "LOAD_PLUGINS=1 #{cmd} #{files.join(" ")} #{params.join(" ")}" puts cmd if !parallel
system("LOAD_PLUGINS=1 #{cmd} #{files.join(" ")} #{params.join(" ")}")
else else
abort "No specs found." abort "No specs found."
end end

View File

@ -4,7 +4,7 @@ RSpec::Support.require_rspec_core "formatters/base_text_formatter"
module TurboTests module TurboTests
# An RSpec formatter that prepends the process id to all messages # An RSpec formatter that prepends the process id to all messages
class DocumentationFormatter < ::RSpec::Core::Formatters::BaseTextFormatter class DocumentationFormatter < RSpec::Core::Formatters::BaseTextFormatter
RSpec::Core::Formatters.register(self, :example_failed, :example_passed, :example_pending) RSpec::Core::Formatters.register(self, :example_failed, :example_passed, :example_pending)
def example_passed(notification) def example_passed(notification)
@ -19,7 +19,7 @@ module TurboTests
message = notification.example.execution_result.pending_message message = notification.example.execution_result.pending_message
output.puts RSpec::Core::Formatters::ConsoleCodes.wrap( output.puts RSpec::Core::Formatters::ConsoleCodes.wrap(
"[#{notification.example.process_id}] #{notification.example.full_description}" \ "[#{notification.example.process_id}] #{notification.example.full_description}" \
"(PENDING: #{message})", " (PENDING: #{message})",
:pending, :pending,
) )
output.flush output.flush
@ -28,7 +28,7 @@ module TurboTests
def example_failed(notification) def example_failed(notification)
output.puts RSpec::Core::Formatters::ConsoleCodes.wrap( output.puts RSpec::Core::Formatters::ConsoleCodes.wrap(
"[#{notification.example.process_id}] #{notification.example.full_description}" \ "[#{notification.example.process_id}] #{notification.example.full_description}" \
"(FAILED - #{next_failure_index})", " (FAILED - #{next_failure_index})",
:failure, :failure,
) )
output.flush output.flush

View File

@ -49,7 +49,6 @@ module TurboTests
{ {
execution_result: execution_result_to_json(example.execution_result), execution_result: execution_result_to_json(example.execution_result),
location: example.location, location: example.location,
description: example.description,
full_description: example.full_description, full_description: example.full_description,
metadata: { metadata: {
shared_group_inclusion_backtrace: shared_group_inclusion_backtrace:

View File

@ -0,0 +1,52 @@
# frozen_string_literal: true
RSpec::Support.require_rspec_core "formatters/base_text_formatter"
module TurboTests
class ProgressFormatter < RSpec::Core::Formatters::BaseTextFormatter
LINE_LENGTH = 80
RSpec::Core::Formatters.register(
self,
:example_passed,
:example_pending,
:example_failed,
:start_dump,
)
def initialize(*args)
super
@examples = 0
end
def example_passed(_notification)
output.print RSpec::Core::Formatters::ConsoleCodes.wrap(".", :success)
wrap
end
def example_pending(_notification)
output.print RSpec::Core::Formatters::ConsoleCodes.wrap("*", :pending)
wrap
end
def example_failed(_notification)
output.print RSpec::Core::Formatters::ConsoleCodes.wrap("F", :failure)
wrap
end
def start_dump(_notification)
output.puts
end
private
def wrap
@examples += 1
if @examples == LINE_LENGTH
output.print "\n"
@examples = 0
end
end
end
end

View File

@ -18,6 +18,7 @@ module TurboTests
attr_reader :pending_examples attr_reader :pending_examples
attr_reader :failed_examples attr_reader :failed_examples
attr_reader :formatters
def initialize(start_time) def initialize(start_time)
@formatters = [] @formatters = []
@ -34,7 +35,7 @@ module TurboTests
formatter_class = formatter_class =
case name case name
when "p", "progress" when "p", "progress"
RSpec::Core::Formatters::ProgressFormatter TurboTests::ProgressFormatter
when "d", "documentation" when "d", "documentation"
TurboTests::DocumentationFormatter TurboTests::DocumentationFormatter
else else

View File

@ -14,6 +14,15 @@ module TurboTests
reporter = Reporter.from_config(formatters, start_time) reporter = Reporter.from_config(formatters, start_time)
if ENV["GITHUB_ACTIONS"]
RSpec.configure do |config|
# Enable color output in GitHub Actions
# This eventually will be `config.color_mode = :on` in RSpec 4?
config.tty = true
config.color = true
end
end
new( new(
reporter: reporter, reporter: reporter,
files: files, files: files,
@ -166,7 +175,9 @@ module TurboTests
[env.map { |k, v| "#{k}=#{v}" }.join(" "), command.join(" ")].select { |x| x.size > 0 } [env.map { |k, v| "#{k}=#{v}" }.join(" "), command.join(" ")].select { |x| x.size > 0 }
.join(" ") .join(" ")
STDERR.puts "::group::[#{process_id}] Run RSpec" if ENV["CI"]
STDERR.puts "Process #{process_id}: #{command_str}" STDERR.puts "Process #{process_id}: #{command_str}"
STDERR.puts "::endgroup::" if ENV["CI"]
end end
stdin, stdout, stderr, wait_thr = Open3.popen3(env, *command) stdin, stdout, stderr, wait_thr = Open3.popen3(env, *command)
@ -237,7 +248,11 @@ module TurboTests
@error = true @error = true
when "exit" when "exit"
exited += 1 exited += 1
@reporter.message("[#{message[:process_id]}] DONE (#{exited}/#{@num_processes + 1})")
if @reporter.formatters.any? { |f| f.is_a?(DocumentationFormatter) }
@reporter.message("[#{message[:process_id]}] DONE (#{exited}/#{@num_processes + 1})")
end
break if exited == @num_processes + 1 break if exited == @num_processes + 1
else else
STDERR.puts("Unhandled message in main process: #{message}") STDERR.puts("Unhandled message in main process: #{message}")

View File

@ -182,6 +182,13 @@ RSpec.configure do |config|
config.order = "random" config.order = "random"
config.infer_spec_type_from_file_location! config.infer_spec_type_from_file_location!
if ENV["GITHUB_ACTIONS"]
# Enable color output in GitHub Actions
# This eventually will be `config.color_mode = :on` in RSpec 4?
config.tty = true
config.color = true
end
# If you're not using ActiveRecord, or you'd prefer not to run each of your # If you're not using ActiveRecord, or you'd prefer not to run each of your
# examples within a transaction, remove the following line or assign false # examples within a transaction, remove the following line or assign false
# instead of true. # instead of true.