Skip to content

Commit 2c0d20d

Browse files
committed
feat: support pending pacts in json formatter
1 parent e8d4a55 commit 2c0d20d

File tree

2 files changed

+87
-0
lines changed

2 files changed

+87
-0
lines changed

lib/pact/provider/pact_spec_runner.rb

+1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@
1212

1313
require_relative 'rspec'
1414

15+
require 'pact/provider/rspec/json_formatter'
1516

1617
module Pact
1718
module Provider
+86
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,86 @@
1+
require 'rspec/core/formatters/json_formatter'
2+
3+
module Pact
4+
module Provider
5+
module RSpec
6+
class JsonFormatter < ::RSpec::Core::Formatters::JsonFormatter
7+
::RSpec::Core::Formatters.register self, :message, :dump_summary, :dump_profile, :stop, :seed, :close
8+
9+
def dump_summary(summary)
10+
super(create_custom_summary(summary))
11+
output_hash[:summary][:notices] = pact_broker_notices(summary)
12+
end
13+
14+
def format_example(example)
15+
{
16+
:id => example.id,
17+
:description => example.description,
18+
:full_description => example.full_description,
19+
:status => calculate_status(example),
20+
:file_path => example.metadata[:file_path],
21+
:line_number => example.metadata[:line_number],
22+
:run_time => example.execution_result.run_time,
23+
:mismatches => extract_differences(example)
24+
}
25+
end
26+
27+
def stop(notification)
28+
output_hash[:examples] = notification.examples.map do |example|
29+
format_example(example).tap do |hash|
30+
e = example.exception
31+
# No point providing a backtrace for a mismatch, too much noise
32+
if e && ! e.is_a?(::RSpec::Expectations::ExpectationNotMetError)
33+
hash[:exception] = {
34+
:class => e.class.name,
35+
:message => e.message,
36+
:backtrace => e.backtrace,
37+
}
38+
end
39+
end
40+
end
41+
end
42+
43+
def calculate_status(example)
44+
if example.execution_result.status == :failed && example.metadata[:pact_ignore_failures]
45+
'pending'
46+
else
47+
example.execution_result.status.to_s
48+
end
49+
end
50+
51+
# There will most likely be only one pact associated with this RSpec execution, because
52+
# the most likely user of this formatter is the Go implementation that parses the JSON
53+
# and builds Go tests from them.
54+
# If the JSON formatter is used by someone else and they have multiple pacts, all the notices
55+
# for the pacts will be mushed together in one collection, so it will be hard to know which notice
56+
# belongs to which pact.
57+
def pact_broker_notices(summary)
58+
pact_uris(summary).collect{ |pact_uri| pact_uri.metadata[:notices] }.compact.flatten
59+
end
60+
61+
def pact_uris(summary)
62+
summary.examples.collect(&:metadata).collect{ |metadata| metadata[:pact_uri] }.uniq
63+
end
64+
65+
def create_custom_summary(summary)
66+
::RSpec::Core::Notifications::SummaryNotification.new(
67+
summary.duration,
68+
summary.examples,
69+
summary.examples.select{ | example | example.execution_result.status == :failed && !example.metadata[:pact_ignore_failures] },
70+
summary.examples.select{ | example | example.execution_result.status == :failed && example.metadata[:pact_ignore_failures] },
71+
summary.load_time,
72+
summary.errors_outside_of_examples_count
73+
)
74+
end
75+
76+
def extract_differences(example)
77+
if example.metadata[:pact_diff]
78+
Pact::Matchers::ExtractDiffMessages.call(example.metadata[:pact_diff]).to_a
79+
else
80+
[]
81+
end
82+
end
83+
end
84+
end
85+
end
86+
end

0 commit comments

Comments
 (0)