Commit 87f73e36 authored by Sanad Liaquat's avatar Sanad Liaquat

Merge branch 'acunskis-reliable-report' into 'master'

E2E: Update reliable spec report creation

See merge request gitlab-org/gitlab!76335
parents e6305bfd 7c0af33d
# frozen_string_literal: true # frozen_string_literal: true
require_relative "../../qa"
require "influxdb-client" require "influxdb-client"
require "terminal-table" require "terminal-table"
require "slack-notifier" require "slack-notifier"
require "colorize"
module QA module QA
module Tools module Tools
class ReliableReport class ReliableReport
def initialize(run_type, range = 30) include Support::API
@results = 2
@slack_channel = "#quality-reports" # Project for report creation: https://gitlab.com/gitlab-org/gitlab
PROJECT_ID = 278964
def initialize(range)
@range = range @range = range
@run_type = run_type @influxdb_bucket = "e2e-test-stats"
@stable_title = "Top #{results} stable specs for past #{@range} days in '#{run_type}' runs" @slack_channel = "#quality-reports"
@unstable_title = "Top #{results} unstable reliable specs for past #{@range} days in '#{run_type}' runs" @influxdb_url = ENV["QA_INFLUXDB_URL"] || raise("Missing QA_INFLUXDB_URL env variable")
@influxdb_token = ENV["QA_INFLUXDB_TOKEN"] || raise("Missing QA_INFLUXDB_TOKEN env variable")
end end
# Print top stable specs # Run reliable reporter
# #
# @param [Integer] range amount of days for results range
# @param [String] report_in_issue_and_slack
# @return [void] # @return [void]
def show_top_stable def self.run(range: 14, report_in_issue_and_slack: "false")
results_table(:stable).each { |table| puts "#{table}\n\n" } reporter = new(range)
reporter.print_report
reporter.report_in_issue_and_slack if report_in_issue_and_slack == "true"
rescue StandardError => e
puts "Report creation failed! Error: '#{e}'".colorize(:red)
reporter.notify_failure(e)
exit(1)
end end
# Post top stable spec report to slack # Print top stable specs
# #
# @return [void] # @return [void]
def notify_top_stable def print_report
puts "\nSending top stable spec report to #{slack_channel} slack channel" puts "#{stable_summary_table}\n\n"
slack_args = { icon_emoji: ":mtg_green:", username: "Stable Spec Report" } stable_results_tables.each { |stage, table| puts "#{table}\n\n" }
notifier.post(text: "*#{stable_title}*", **slack_args) return puts("No unstable reliable tests present!".colorize(:yellow)) if unstable_reliable_test_runs.empty?
results_table(:stable).each { |table| notifier.post(text: "```#{table}```", **slack_args) }
puts "#{unstable_summary_table}\n\n"
unstable_reliable_results_tables.each { |stage, table| puts "#{table}\n\n" }
end end
# Print top unstable specs # Create report issue
# #
# @return [void] # @return [void]
def show_top_unstable def report_in_issue_and_slack
return puts("No unstable tests present!") if top_unstable_reliable.empty? puts "Creating report".colorize(:green)
response = post(
"#{gitlab_api_url}/projects/#{PROJECT_ID}/issues",
{ title: "Reliable spec report", description: report_issue_body, labels: "Quality,test" },
headers: { "PRIVATE-TOKEN" => gitlab_access_token }
)
web_url = parse_body(response)[:web_url]
puts "Created report issue: #{web_url}"
results_table(:unstable).each { |table| puts "#{table}\n\n" } puts "Sending slack notification".colorize(:green)
notifier.post(
icon_emoji: ":tanuki-protect:",
text: <<~TEXT
```#{stable_summary_table}```
```#{unstable_summary_table}```
#{web_url}
TEXT
)
puts "Done!"
end end
# Post top unstable reliable spec report to slack # Notify failure
# #
# @param [StandardError] error
# @return [void] # @return [void]
def notify_top_unstable def notify_failure(error)
return puts("No unstable tests present!") if top_unstable_reliable.empty? notifier.post(
text: "Reliable reporter failed to create report. Error: ```#{error}```",
puts "\nSending top unstable reliable spec report to #{slack_channel} slack channel" icon_emoji: ":sadpanda:"
slack_args = { icon_emoji: ":sadpanda:", username: "Unstable Spec Report" } )
notifier.post(text: "*#{unstable_title}*", **slack_args)
results_table(:unstable).each { |table| notifier.post(text: "```#{table}```", **slack_args) }
end end
private private
attr_reader :results, attr_reader :range, :influxdb_bucket, :slack_channel, :influxdb_url, :influxdb_token
:slack_channel,
:range, # Markdown formatted report issue body
:run_type, #
:stable_title, # @return [String]
:unstable_title def report_issue_body
issue = []
issue << "[[_TOC_]]"
issue << "# Candidates for promotion to reliable\n\n```\n#{stable_summary_table}\n```"
issue << results_markdown(stable_results_tables)
return issue.join("\n\n") if unstable_reliable_test_runs.empty?
issue << "# Reliable specs with failures\n\n```\n#{unstable_summary_table}\n```"
issue << results_markdown(unstable_reliable_results_tables)
issue.join("\n\n")
end
# Stable spec summary table
#
# @return [Terminal::Table]
def stable_summary_table
@stable_summary_table ||= terminal_table(
rows: stable_test_runs.map { |stage, specs| [stage, specs.length] },
title: "Stable spec summary for past #{range} days".ljust(50),
headings: %w[STAGE COUNT]
)
end
# Unstable reliable summary table
#
# @return [Terminal::Table]
def unstable_summary_table
@unstable_summary_table ||= terminal_table(
rows: unstable_reliable_test_runs.map { |stage, specs| [stage, specs.length] },
title: "Unstable spec summary for past #{range} days".ljust(50),
headings: %w[STAGE COUNT]
)
end
# Result tables for stable specs
#
# @return [Hash]
def stable_results_tables
@stable_results ||= results_tables(:stable)
end
# Result table for unstable specs
#
# @return [Hash]
def unstable_reliable_results_tables
@unstable_results ||= results_tables(:unstable)
end
# Markdown formatted tables
#
# @param [Hash] results
# @return [String]
def results_markdown(results)
results.map do |stage, table|
<<~STAGE.strip
## #{stage}
<details>
<summary>Executions table</summary>
```
#{table}
```
</details>
STAGE
end.join("\n\n")
end
# Results table # Results table
# #
# @param [Symbol] type result type - :stable, :unstable # @param [Symbol] type result type - :stable, :unstable
# @return [Hash] # @return [Hash<Symbol, Terminal::Table>]
def results_table(type) def results_tables(type)
(type == :stable ? top_stable : top_unstable_reliable).map do |stage, specs| (type == :stable ? stable_test_runs : unstable_reliable_test_runs).to_h do |stage, specs|
terminal_table( headings = ["name", "runs", "failures", "failure rate"]
[stage, terminal_table(
rows: specs.map { |k, v| [name_column(k, v[:file]), *table_params(v.values)] }, rows: specs.map { |k, v| [name_column(k, v[:file]), *table_params(v.values)] },
title: "Top #{type} specs in '#{stage}' stage" title: "Top #{type} specs in '#{stage}' stage for past #{range} days",
) headings: headings.map(&:upcase)
)]
end end
end end
# Top stable specs # Stable specs
# #
# @return [Hash] # @return [Hash]
def top_stable def stable_test_runs
@top_stable ||= runs(reliable: false).transform_values do |specs| @top_stable ||= begin
specs.sort_by { |k, v| [v[:failure_rate], -v[:runs]] }[0..results - 1].to_h stable_specs = test_runs(reliable: false).transform_values do |specs|
specs
.reject { |k, v| v[:failure_rate] != 0 }
.sort_by { |k, v| -v[:runs] }
.to_h
end
stable_specs.reject { |k, v| v.empty? }
end end
end end
# Top unstable reliable specs # Unstable reliable specs
# #
# @return [Hash] # @return [Hash]
def top_unstable_reliable def unstable_reliable_test_runs
@top_unstable_reliable ||= begin @top_unstable_reliable ||= begin
unstable = runs(reliable: true).transform_values do |specs| unstable = test_runs(reliable: true).transform_values do |specs|
specs specs
.reject { |k, v| v[:failure_rate] == 0 } .reject { |k, v| v[:failure_rate] == 0 }
.sort_by { |k, v| -v[:failure_rate] }[0..results - 1] .sort_by { |k, v| -v[:failure_rate] }
.to_h .to_h
end end
...@@ -104,9 +215,9 @@ module QA ...@@ -104,9 +215,9 @@ module QA
# Terminal table for result formatting # Terminal table for result formatting
# #
# @return [Terminal::Table] # @return [Terminal::Table]
def terminal_table(rows:, title: nil) def terminal_table(rows:, headings:, title: nil)
Terminal::Table.new( Terminal::Table.new(
headings: ["name", "runs", "failed", "failure rate"], headings: headings,
style: { all_separators: true }, style: { all_separators: true },
title: title, title: title,
rows: rows rows: rows
...@@ -127,20 +238,19 @@ module QA ...@@ -127,20 +238,19 @@ module QA
# @param [String] file # @param [String] file
# @return [String] # @return [String]
def name_column(name, file) def name_column(name, file)
spec_name = name.length > 100 ? "#{name} ".scan(/.{1,100} /).map(&:strip).join("\n") : name spec_name = name.length > 150 ? "#{name} ".scan(/.{1,150} /).map(&:strip).join("\n") : name
name_line = "name: '#{spec_name}'" name_line = "name: '#{spec_name}'"
file_line = "file: '#{file}'" file_line = "file: '#{file}'"
"#{name_line}\n#{file_line.ljust(110)}" "#{name_line}\n#{file_line.ljust(160)}"
end end
# Test executions grouped by name # Test executions grouped by name
# #
# @param [Boolean] reliable # @param [Boolean] reliable
# @return [Hash<String, Hash>] # @return [Hash<String, Hash>]
def runs(reliable:) def test_runs(reliable:)
puts("Fetching data on #{reliable ? 'reliable ' : ''}test execution for past 30 days in '#{run_type}' runs") puts("Fetching data on #{reliable ? 'reliable ' : ''}test execution for past #{range} days\n".colorize(:green))
puts
all_runs = query_api.query(query: query(reliable)).values all_runs = query_api.query(query: query(reliable)).values
all_runs.each_with_object(Hash.new { |hsh, key| hsh[key] = {} }) do |table, result| all_runs.each_with_object(Hash.new { |hsh, key| hsh[key] = {} }) do |table, result|
...@@ -168,17 +278,24 @@ module QA ...@@ -168,17 +278,24 @@ module QA
# @return [String] # @return [String]
def query(reliable) def query(reliable)
<<~QUERY <<~QUERY
from(bucket: "e2e-test-stats") from(bucket: "#{influxdb_bucket}")
|> range(start: -#{range}d) |> range(start: -#{range}d)
|> filter(fn: (r) => r._measurement == "test-stats" and |> filter(fn: (r) => r._measurement == "test-stats")
r.run_type == "#{run_type}" and |> filter(fn: (r) => r.run_type == "staging-full" or
r.status != "pending" and r.run_type == "staging-sanity" or
r.merge_request == "false" and r.run_type == "staging-sanity-no-admin" or
r.quarantined == "false" and r.run_type == "production-full" or
r.reliable == "#{reliable}" and r.run_type == "production-sanity" or
r._field == "id" r.run_type == "package-and-qa" or
) r.run_type == "nightly"
|> group(columns: ["name"]) )
|> filter(fn: (r) => r.status != "pending" and
r.merge_request == "false" and
r.quarantined == "false" and
r.reliable == "#{reliable}" and
r._field == "id"
)
|> group(columns: ["name"])
QUERY QUERY
end end
...@@ -196,7 +313,7 @@ module QA ...@@ -196,7 +313,7 @@ module QA
@influx_client ||= InfluxDB2::Client.new( @influx_client ||= InfluxDB2::Client.new(
influxdb_url, influxdb_url,
influxdb_token, influxdb_token,
bucket: "e2e-test-stats", bucket: influxdb_bucket,
org: "gitlab-qa", org: "gitlab-qa",
precision: InfluxDB2::WritePrecision::NANOSECOND precision: InfluxDB2::WritePrecision::NANOSECOND
) )
...@@ -209,29 +326,29 @@ module QA ...@@ -209,29 +326,29 @@ module QA
@notifier ||= Slack::Notifier.new( @notifier ||= Slack::Notifier.new(
slack_webhook_url, slack_webhook_url,
channel: slack_channel, channel: slack_channel,
username: "Reliable spec reporter" username: "Reliable Spec Report"
) )
end end
# InfluxDb instance url # Gitlab access token
# #
# @return [String] # @return [String]
def influxdb_url def gitlab_access_token
@influxdb_url ||= ENV["QA_INFLUXDB_URL"] || raise("Missing QA_INFLUXDB_URL environment variable") @gitlab_access_token ||= ENV["GITLAB_ACCESS_TOKEN"] || raise("Missing GITLAB_ACCESS_TOKEN env variable")
end end
# Influxdb token # Gitlab api url
# #
# @return [String] # @return [String]
def influxdb_token def gitlab_api_url
@influxdb_token ||= ENV["QA_INFLUXDB_TOKEN"] || raise("Missing QA_INFLUXDB_TOKEN environment variable") @gitlab_api_url ||= ENV["CI_API_V4_URL"] || raise("Missing CI_API_V4_URL env variable")
end end
# Slack webhook url # Slack webhook url
# #
# @return [String] # @return [String]
def slack_webhook_url def slack_webhook_url
@slack_webhook_url ||= ENV["CI_SLACK_WEBHOOK_URL"] || raise("Missing CI_SLACK_WEBHOOK_URL environment variable") @slack_webhook_url ||= ENV["SLACK_WEBHOOK"] || raise("Missing SLACK_WEBHOOK env variable")
end end
end end
end end
......
...@@ -3,63 +3,94 @@ ...@@ -3,63 +3,94 @@
describe QA::Tools::ReliableReport do describe QA::Tools::ReliableReport do
include QA::Support::Helpers::StubEnv include QA::Support::Helpers::StubEnv
subject(:reporter) { described_class.new(run_type, range) } subject(:run) { described_class.run(range: range, report_in_issue_and_slack: create_issue) }
let(:gitlab_response) { instance_double("RestClient::Response", code: 200, body: { web_url: issue_url }.to_json) }
let(:slack_notifier) { instance_double("Slack::Notifier", post: nil) } let(:slack_notifier) { instance_double("Slack::Notifier", post: nil) }
let(:influx_client) { instance_double("InfluxDB2::Client", create_query_api: query_api) } let(:influx_client) { instance_double("InfluxDB2::Client", create_query_api: query_api) }
let(:query_api) { instance_double("InfluxDB2::QueryApi") } let(:query_api) { instance_double("InfluxDB2::QueryApi") }
let(:slack_channel) { "#quality-reports" } let(:slack_channel) { "#quality-reports" }
let(:run_type) { "package-and-qa" } let(:range) { 14 }
let(:range) { 30 } let(:issue_url) { "https://gitlab.com/issue/1" }
let(:results) { 2 }
let(:runs) do
let(:runs) { { 0 => stable_spec, 1 => unstable_spec } } values = { "name" => "stable spec", "status" => "passed", "file_path" => "some/spec.rb", "stage" => "manage" }
{
let(:spec_values) { { "file_path" => "some/spec.rb", "stage" => "manage" } } 0 => instance_double(
let(:stable_spec) do "InfluxDB2::FluxTable",
values = { "name" => "stable spec", "status" => "passed", **spec_values } records: [
instance_double( instance_double("InfluxDB2::FluxRecord", values: values),
"InfluxDB2::FluxTable", instance_double("InfluxDB2::FluxRecord", values: values),
records: [ instance_double("InfluxDB2::FluxRecord", values: values)
instance_double("InfluxDB2::FluxRecord", values: values), ]
instance_double("InfluxDB2::FluxRecord", values: values), )
instance_double("InfluxDB2::FluxRecord", values: values) }
]
)
end end
let(:unstable_spec) do let(:reliable_runs) do
values = { "name" => "unstable spec", "status" => "failed", **spec_values } values = { "name" => "unstable spec", "status" => "failed", "file_path" => "some/spec.rb", "stage" => "create" }
instance_double( {
"InfluxDB2::FluxTable", 0 => instance_double(
records: [ "InfluxDB2::FluxTable",
instance_double("InfluxDB2::FluxRecord", values: { **values, "status" => "passed" }), records: [
instance_double("InfluxDB2::FluxRecord", values: values), instance_double("InfluxDB2::FluxRecord", values: { **values, "status" => "passed" }),
instance_double("InfluxDB2::FluxRecord", values: values) instance_double("InfluxDB2::FluxRecord", values: values),
] instance_double("InfluxDB2::FluxRecord", values: values)
) ]
)
}
end end
def flux_query(reliable) def flux_query(reliable:)
<<~QUERY <<~QUERY
from(bucket: "e2e-test-stats") from(bucket: "e2e-test-stats")
|> range(start: -#{range}d) |> range(start: -#{range}d)
|> filter(fn: (r) => r._measurement == "test-stats" and |> filter(fn: (r) => r._measurement == "test-stats")
r.run_type == "#{run_type}" and |> filter(fn: (r) => r.run_type == "staging-full" or
r.status != "pending" and r.run_type == "staging-sanity" or
r.merge_request == "false" and r.run_type == "staging-sanity-no-admin" or
r.quarantined == "false" and r.run_type == "production-full" or
r.reliable == "#{reliable}" and r.run_type == "production-sanity" or
r._field == "id" r.run_type == "package-and-qa" or
) r.run_type == "nightly"
|> group(columns: ["name"]) )
|> filter(fn: (r) => r.status != "pending" and
r.merge_request == "false" and
r.quarantined == "false" and
r.reliable == "#{reliable}" and
r._field == "id"
)
|> group(columns: ["name"])
QUERY QUERY
end end
def table(rows, title = nil) def markdown_section(summary, result, stage, type)
<<~SECTION.strip
```
#{summary_table(summary, type)}
```
## #{stage}
<details>
<summary>Executions table</summary>
```
#{table(result, ['NAME', 'RUNS', 'FAILURES', 'FAILURE RATE'], "Top #{type} specs in '#{stage}' stage for past #{range} days")}
```
</details>
SECTION
end
def summary_table(summary, type)
table(summary, %w[STAGE COUNT], "#{type.capitalize} spec summary for past #{range} days".ljust(50))
end
def table(rows, headings, title)
Terminal::Table.new( Terminal::Table.new(
headings: ["name", "runs", "failed", "failure rate"], headings: headings,
style: { all_separators: true }, style: { all_separators: true },
title: title, title: title,
rows: rows rows: rows
...@@ -68,7 +99,7 @@ describe QA::Tools::ReliableReport do ...@@ -68,7 +99,7 @@ describe QA::Tools::ReliableReport do
def name_column(spec_name) def name_column(spec_name)
name = "name: '#{spec_name}'" name = "name: '#{spec_name}'"
file = "file: 'spec.rb'".ljust(110) file = "file: 'spec.rb'".ljust(160)
"#{name}\n#{file}" "#{name}\n#{file}"
end end
...@@ -76,73 +107,85 @@ describe QA::Tools::ReliableReport do ...@@ -76,73 +107,85 @@ describe QA::Tools::ReliableReport do
before do before do
stub_env("QA_INFLUXDB_URL", "url") stub_env("QA_INFLUXDB_URL", "url")
stub_env("QA_INFLUXDB_TOKEN", "token") stub_env("QA_INFLUXDB_TOKEN", "token")
stub_env("CI_SLACK_WEBHOOK_URL", "slack_url") stub_env("SLACK_WEBHOOK", "slack_url")
stub_env("CI_API_V4_URL", "gitlab_api_url")
stub_env("GITLAB_ACCESS_TOKEN", "gitlab_token")
allow(RestClient::Request).to receive(:execute).and_return(gitlab_response)
allow(Slack::Notifier).to receive(:new).and_return(slack_notifier) allow(Slack::Notifier).to receive(:new).and_return(slack_notifier)
allow(InfluxDB2::Client).to receive(:new).and_return(influx_client) allow(InfluxDB2::Client).to receive(:new).and_return(influx_client)
allow(query_api).to receive(:query).with(query: query).and_return(runs)
end
context "with stable spec report" do allow(query_api).to receive(:query).with(query: flux_query(reliable: false)).and_return(runs)
let(:query) { flux_query(false) } allow(query_api).to receive(:query).with(query: flux_query(reliable: true)).and_return(reliable_runs)
let(:fetch_message) { "Fetching data on test execution for past #{range} days in '#{run_type}' runs" } end
let(:slack_send_message) { "Sending top stable spec report to #{slack_channel} slack channel" }
let(:message_title) { "Top #{results} stable specs for past #{range} days in '#{run_type}' runs" }
let(:table_title) { "Top stable specs in 'manage' stage" }
let(:rows) do
[
[name_column("stable spec"), 3, 0, "0%"],
[name_column("unstable spec"), 3, 2, "66.67%"]
]
end
it "prints top stable spec report to console" do context "without report creation" do
expect { reporter.show_top_stable }.to output("#{fetch_message}\n\n#{table(rows, table_title)}\n\n").to_stdout let(:create_issue) { "false" }
end
it "sends top stable spec report to slack" do it "does not create report issue", :aggregate_failures do
slack_args = { icon_emoji: ":mtg_green:", username: "Stable Spec Report" } expect { run }.to output.to_stdout
expect { reporter.notify_top_stable }.to output("\n#{slack_send_message}\n#{fetch_message}\n\n").to_stdout expect(RestClient::Request).not_to have_received(:execute)
expect(slack_notifier).to have_received(:post).with(text: "*#{message_title}*", **slack_args) expect(slack_notifier).not_to have_received(:post)
expect(slack_notifier).to have_received(:post).with(text: "```#{table(rows, table_title)}```", **slack_args)
end end
end end
context "with unstable spec report" do context "with report creation" do
let(:query) { flux_query(true) } let(:create_issue) { "true" }
let(:fetch_message) { "Fetching data on reliable test execution for past #{range} days in '#{run_type}' runs" } let(:issue_body) do
let(:slack_send_message) { "Sending top unstable reliable spec report to #{slack_channel} slack channel" } <<~TXT.strip
let(:message_title) { "Top #{results} unstable reliable specs for past #{range} days in '#{run_type}' runs" } [[_TOC_]]
let(:table_title) { "Top unstable specs in 'manage' stage" }
let(:rows) { [[name_column("unstable spec"), 3, 2, "66.67%"]] }
it "prints top unstable spec report to console" do # Candidates for promotion to reliable
expect { reporter.show_top_unstable }.to output("#{fetch_message}\n\n#{table(rows, table_title)}\n\n").to_stdout
end
it "sends top unstable reliable spec report to slack" do #{markdown_section([['manage', 1]], [[name_column('stable spec'), 3, 0, '0%']], 'manage', 'stable')}
slack_args = { icon_emoji: ":sadpanda:", username: "Unstable Spec Report" }
expect { reporter.notify_top_unstable }.to output("#{fetch_message}\n\n\n#{slack_send_message}\n").to_stdout # Reliable specs with failures
expect(slack_notifier).to have_received(:post).with(text: "*#{message_title}*", **slack_args)
expect(slack_notifier).to have_received(:post).with(text: "```#{table(rows, table_title)}```", **slack_args) #{markdown_section([['create', 1]], [[name_column('unstable spec'), 3, 2, '66.67%']], 'create', 'unstable')}
TXT
end
it "creates report issue", :aggregate_failures do
expect { run }.to output.to_stdout
expect(RestClient::Request).to have_received(:execute).with(
method: :post,
url: "gitlab_api_url/projects/278964/issues",
verify_ssl: false,
headers: { "PRIVATE-TOKEN" => "gitlab_token" },
payload: {
title: "Reliable spec report",
description: issue_body,
labels: "Quality,test"
}
)
expect(slack_notifier).to have_received(:post).with(
icon_emoji: ":tanuki-protect:",
text: <<~TEXT
```#{summary_table([['manage', 1]], 'stable')}```
```#{summary_table([['create', 1]], 'unstable')}```
#{issue_url}
TEXT
)
end end
end end
context "without unstable reliable specs" do context "with failure" do
let(:query) { flux_query(true) } let(:create_issue) { "true" }
let(:runs) { { 0 => stable_spec } }
let(:fetch_message) { "Fetching data on reliable test execution for past #{range} days in '#{run_type}' runs" }
let(:no_result_message) { "No unstable tests present!" }
it "prints no result message to console" do before do
expect { reporter.show_top_unstable }.to output("#{fetch_message}\n\n#{no_result_message}\n").to_stdout allow(query_api).to receive(:query).and_raise("Connection error!")
end end
it "skips slack notification" do it "notifies failure", :aggregate_failures do
expect { reporter.notify_top_unstable }.to output("#{fetch_message}\n\n#{no_result_message}\n").to_stdout expect { expect { run }.to raise_error(SystemExit) }.to output.to_stdout
expect(slack_notifier).not_to have_received(:post)
expect(slack_notifier).to have_received(:post).with(
icon_emoji: ":sadpanda:",
text: "Reliable reporter failed to create report. Error: ```Connection error!```"
)
end end
end end
end end
...@@ -3,19 +3,8 @@ ...@@ -3,19 +3,8 @@
require_relative "../qa/tools/reliable_report" require_relative "../qa/tools/reliable_report"
desc "Fetch top most reliable specs" desc "Fetch reliable and unreliable spec data and create report"
task :reliable_spec_report, [:run_type, :range, :create_slack_report] do |_task, args| task :reliable_spec_report, [:range, :report_in_issue_and_slack] do |_task, args|
report = QA::Tools::ReliableReport.new(args[:run_type] || "package-and-qa", args[:range]) QA::Tools::ReliableReport.run(**args)
report.show_top_stable
report.notify_top_stable if args[:create_slack_report] == 'true'
end
desc "Fetch top most unstable reliable specs"
task :unreliable_spec_report, [:run_type, :range, :create_slack_report] do |_task, args|
report = QA::Tools::ReliableReport.new(args[:run_type] || "package-and-qa", args[:range])
report.show_top_unstable
report.notify_top_unstable if args[:create_slack_report] == 'true'
end end
# rubocop:enable Rails/RakeEnvironment # rubocop:enable Rails/RakeEnvironment
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment