Skip to content

Commit 4497ee9

Browse files
authored
feat: add telemetry (#256)
See https://docs.pact.io/telemetry
1 parent b8dbfab commit 4497ee9

File tree

5 files changed

+148
-2
lines changed

5 files changed

+148
-2
lines changed

lib/pact/provider/pact_spec_runner.rb

+3
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@
1212
require 'pact/provider/rspec/json_formatter'
1313
require 'pact/provider/rspec'
1414
require 'pact/provider/rspec/calculate_exit_code'
15+
require 'pact/utils/metrics'
1516

1617
module Pact
1718
module Provider
@@ -130,6 +131,8 @@ def initialize_specs
130131
ignore_failures: options[:ignore_failures],
131132
request_customizer: options[:request_customizer]
132133
}
134+
Pact::Utils::Metrics.report_metric("Pacts verified", "ProviderTest", "Completed")
135+
133136
honour_pactfile pact_source, ordered_pact_json(pact_source.pact_json), spec_options
134137
end
135138
end

lib/pact/utils/metrics.rb

+85
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,85 @@
1+
require 'securerandom'
2+
require 'digest'
3+
require 'socket'
4+
require 'pact/version'
5+
require 'net/http'
6+
7+
module Pact
8+
module Utils
9+
class Metrics
10+
11+
def self.report_metric(event, category, action, value = 1)
12+
in_thread do
13+
begin
14+
if track_events?
15+
Pact.configuration.output_stream.puts "WARN: Please note: we are tracking events anonymously to gather important usage statistics like Pact-Ruby version
16+
and operating system. To disable tracking, set the 'PACT_DO_NOT_TRACK' environment
17+
variable to 'true'."
18+
19+
uri = URI('https://www.google-analytics.com/collect')
20+
req = Net::HTTP::Post.new(uri)
21+
req.set_form_data(create_tracking_event(event, category, action, value))
22+
23+
Net::HTTP.start(uri.hostname, uri.port, read_timeout:2, open_timeout:2, :use_ssl => true ) do |http|
24+
http.request(req)
25+
end
26+
end
27+
rescue StandardError => e
28+
handle_error(e)
29+
end
30+
end
31+
end
32+
33+
private
34+
def self.handle_error e
35+
if ENV['PACT_METRICS_DEBUG'] == 'true'
36+
Pact.configuration.output_stream.puts("DEBUG: #{e.inspect}\n" + e.backtrace.join("\n"))
37+
end
38+
end
39+
40+
def self.in_thread
41+
Thread.new do
42+
yield
43+
end
44+
end
45+
46+
def self.create_tracking_event(event, category, action, value)
47+
{
48+
"v" => 1,
49+
"t" => "event",
50+
"tid" => "UA-117778936-1",
51+
"cid" => calculate_cid,
52+
"an" => "Pact Ruby",
53+
"av" => Pact::VERSION,
54+
"aid" => "pact-ruby",
55+
"aip" => 1,
56+
"ds" => ENV['PACT_EXECUTING_LANGUAGE'] ? "client" : "cli",
57+
"cd2" => ENV['CI'] == "true" ? "CI" : "unknown",
58+
"cd3" => RUBY_PLATFORM,
59+
"cd6" => ENV['PACT_EXECUTING_LANGUAGE'] || "unknown",
60+
"cd7" => ENV['PACT_EXECUTING_LANGUAGE_VERSION'],
61+
"el" => event,
62+
"ec" => category,
63+
"ea" => action,
64+
"ev" => value
65+
}
66+
end
67+
68+
def self.track_events?
69+
ENV['PACT_DO_NOT_TRACK'] != 'true'
70+
end
71+
72+
def self.calculate_cid
73+
if RUBY_PLATFORM.include? "windows"
74+
hostname = ENV['COMPUTERNAME']
75+
else
76+
hostname = ENV['HOSTNAME']
77+
end
78+
if !hostname
79+
hostname = Socket.gethostname
80+
end
81+
Digest::MD5.hexdigest hostname || SecureRandom.urlsafe_base64(5)
82+
end
83+
end
84+
end
85+
end

spec/lib/pact/provider/pact_spec_runner_spec.rb

+8-1
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,7 @@
4444
end
4545
allow(subject).to receive(:configure_rspec)
4646
allow(subject).to receive(:run_specs)
47+
allow(Pact::Utils::Metrics).to receive(:report_metric)
4748

4849
expect(Pact::Provider::PactSource).to receive(:new).with(pact_url).and_return(pact_source)
4950
end
@@ -62,6 +63,13 @@
6263
subject.run
6364
end
6465

66+
it 'reports pacts verified metric' do
67+
allow(subject).to receive(:honour_pactfile).and_return([])
68+
69+
expect(Pact::Utils::Metrics).to receive(:report_metric).with("Pacts verified", "ProviderTest", "Completed")
70+
subject.run
71+
end
72+
6573
context 'and interactions_replay_order option set to random' do
6674
let(:interactions_replay_order) { :random }
6775

@@ -82,6 +90,5 @@
8290
end
8391
end
8492
end
85-
8693
end
8794
end

spec/pact/utils/metrics_spec.rb

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
require 'rspec'
2+
require 'pact/utils/metrics'
3+
4+
describe Pact::Utils::Metrics do
5+
6+
before do
7+
stub_request(:post, "https://www.google-analytics.com/collect").to_return(status: 200, body: "", headers: {})
8+
ENV['COMPUTERNAME'] = 'test'
9+
ENV['HOSTNAME'] = 'test'
10+
end
11+
12+
describe ".report_metric" do
13+
subject { Pact::Utils::Metrics.report_metric("Event", "Category", "Action", "Value") }
14+
context 'when do not track is not set' do
15+
let(:expected_event) { {
16+
"v" => 1,
17+
"t" => "event",
18+
"tid" => "UA-117778936-1",
19+
"cid" => "098f6bcd4621d373cade4e832627b4f6",
20+
"an" => "Pact Ruby",
21+
"av" => Pact::VERSION,
22+
"aid" => "pact-ruby",
23+
"aip" => 1,
24+
"ds" => ENV['PACT_EXECUTING_LANGUAGE'] ? "client" : "cli",
25+
"cd2" => ENV['CI'] == "true" ? "CI" : "unknown",
26+
"cd3" => RUBY_PLATFORM,
27+
"cd6" => ENV['PACT_EXECUTING_LANGUAGE'] || "unknown",
28+
"cd7" => ENV['PACT_EXECUTING_LANGUAGE_VERSION'],
29+
"el" => "Event",
30+
"ec" => "Category",
31+
"ea" => "Action",
32+
"ev" => "Value"
33+
} }
34+
35+
it 'sends metrics' do
36+
expect_any_instance_of(Net::HTTP::Post).to receive(:set_form_data).with(expected_event)
37+
expect(Net::HTTP).to receive(:start)
38+
subject
39+
end
40+
end
41+
context 'when do not track is set to true' do
42+
before do
43+
ENV['PACT_DO_NOT_TRACK'] = "true"
44+
end
45+
it 'does not send metrics' do
46+
expect(Net::HTTP).to_not receive(:post)
47+
subject
48+
end
49+
end
50+
end
51+
end

spec/spec_helper.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77
require 'support/spec_support'
88
require 'pact/provider/rspec'
99

10-
WebMock.disable_net_connect!(allow_localhost: true)
10+
WebMock.disable_net_connect!(allow_localhost: true, allow: 'https://www.google-analytics.com')
1111

1212
require './spec/support/active_support_if_configured'
1313
require './spec/support/warning_silencer'

0 commit comments

Comments
 (0)