Commit 1e26f1e4 authored by Quang-Minh Nguyen's avatar Quang-Minh Nguyen

Add start time to external http items in the performance bar

Issue https://gitlab.com/groups/gitlab-org/-/epics/5590
parent f48f1396
...@@ -37,7 +37,7 @@ module Gitlab ...@@ -37,7 +37,7 @@ module Gitlab
def request(event) def request(event)
payload = event.payload payload = event.payload
add_to_detail_store(payload) add_to_detail_store(event.time, payload)
add_to_request_store(payload) add_to_request_store(payload)
expose_metrics(payload) expose_metrics(payload)
end end
...@@ -48,10 +48,11 @@ module Gitlab ...@@ -48,10 +48,11 @@ module Gitlab
::Gitlab::Metrics::Transaction.current ::Gitlab::Metrics::Transaction.current
end end
def add_to_detail_store(payload) def add_to_detail_store(start, payload)
return unless Gitlab::PerformanceBar.enabled_for_request? return unless Gitlab::PerformanceBar.enabled_for_request?
self.class.detail_store << { self.class.detail_store << {
start: start,
duration: payload[:duration], duration: payload[:duration],
scheme: payload[:scheme], scheme: payload[:scheme],
method: payload[:method], method: payload[:method],
......
...@@ -6,29 +6,45 @@ RSpec.describe Gitlab::Metrics::Subscribers::ExternalHttp, :request_store do ...@@ -6,29 +6,45 @@ RSpec.describe Gitlab::Metrics::Subscribers::ExternalHttp, :request_store do
let(:transaction) { Gitlab::Metrics::Transaction.new } let(:transaction) { Gitlab::Metrics::Transaction.new }
let(:subscriber) { described_class.new } let(:subscriber) { described_class.new }
around do |example|
freeze_time { example.run }
end
let(:event_1) do let(:event_1) do
double(:event, payload: { double(
:event,
payload: {
method: 'POST', code: "200", duration: 0.321, method: 'POST', code: "200", duration: 0.321,
scheme: 'https', host: 'gitlab.com', port: 80, path: '/api/v4/projects', scheme: 'https', host: 'gitlab.com', port: 80, path: '/api/v4/projects',
query: 'current=true' query: 'current=true'
}) },
time: Time.current
)
end end
let(:event_2) do let(:event_2) do
double(:event, payload: { double(
:event,
payload: {
method: 'GET', code: "301", duration: 0.12, method: 'GET', code: "301", duration: 0.12,
scheme: 'http', host: 'gitlab.com', port: 80, path: '/api/v4/projects/2', scheme: 'http', host: 'gitlab.com', port: 80, path: '/api/v4/projects/2',
query: 'current=true' query: 'current=true'
}) },
time: Time.current
)
end end
let(:event_3) do let(:event_3) do
double(:event, payload: { double(
:event,
payload: {
method: 'POST', duration: 5.3, method: 'POST', duration: 5.3,
scheme: 'http', host: 'gitlab.com', port: 80, path: '/api/v4/projects/2/issues', scheme: 'http', host: 'gitlab.com', port: 80, path: '/api/v4/projects/2/issues',
query: 'current=true', query: 'current=true',
exception_object: Net::ReadTimeout.new exception_object: Net::ReadTimeout.new
}) },
time: Time.current
)
end end
describe '.detail_store' do describe '.detail_store' do
...@@ -134,19 +150,22 @@ RSpec.describe Gitlab::Metrics::Subscribers::ExternalHttp, :request_store do ...@@ -134,19 +150,22 @@ RSpec.describe Gitlab::Metrics::Subscribers::ExternalHttp, :request_store do
subscriber.request(event_3) subscriber.request(event_3)
expect(Gitlab::SafeRequestStore[:external_http_detail_store].length).to eq(3) expect(Gitlab::SafeRequestStore[:external_http_detail_store].length).to eq(3)
expect(Gitlab::SafeRequestStore[:external_http_detail_store][0]).to include( expect(Gitlab::SafeRequestStore[:external_http_detail_store][0]).to match a_hash_including(
start: be_like_time(Time.current),
method: 'POST', code: "200", duration: 0.321, method: 'POST', code: "200", duration: 0.321,
scheme: 'https', host: 'gitlab.com', port: 80, path: '/api/v4/projects', scheme: 'https', host: 'gitlab.com', port: 80, path: '/api/v4/projects',
query: 'current=true', exception_object: nil, query: 'current=true', exception_object: nil,
backtrace: be_a(Array) backtrace: be_a(Array)
) )
expect(Gitlab::SafeRequestStore[:external_http_detail_store][1]).to include( expect(Gitlab::SafeRequestStore[:external_http_detail_store][1]).to match a_hash_including(
start: be_like_time(Time.current),
method: 'GET', code: "301", duration: 0.12, method: 'GET', code: "301", duration: 0.12,
scheme: 'http', host: 'gitlab.com', port: 80, path: '/api/v4/projects/2', scheme: 'http', host: 'gitlab.com', port: 80, path: '/api/v4/projects/2',
query: 'current=true', exception_object: nil, query: 'current=true', exception_object: nil,
backtrace: be_a(Array) backtrace: be_a(Array)
) )
expect(Gitlab::SafeRequestStore[:external_http_detail_store][2]).to include( expect(Gitlab::SafeRequestStore[:external_http_detail_store][2]).to match a_hash_including(
start: be_like_time(Time.current),
method: 'POST', duration: 5.3, method: 'POST', duration: 5.3,
scheme: 'http', host: 'gitlab.com', port: 80, path: '/api/v4/projects/2/issues', scheme: 'http', host: 'gitlab.com', port: 80, path: '/api/v4/projects/2/issues',
query: 'current=true', query: 'current=true',
......
...@@ -11,6 +11,10 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do ...@@ -11,6 +11,10 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do
allow(Gitlab::PerformanceBar).to receive(:enabled_for_request?).and_return(true) allow(Gitlab::PerformanceBar).to receive(:enabled_for_request?).and_return(true)
end end
around do |example|
freeze_time { example.run }
end
let(:event_1) do let(:event_1) do
{ {
method: 'POST', code: "200", duration: 0.03, method: 'POST', code: "200", duration: 0.03,
...@@ -44,9 +48,9 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do ...@@ -44,9 +48,9 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do
end end
it 'returns aggregated results' do it 'returns aggregated results' do
subscriber.request(double(:event, payload: event_1)) subscriber.request(double(:event, payload: event_1, time: Time.current))
subscriber.request(double(:event, payload: event_2)) subscriber.request(double(:event, payload: event_2, time: Time.current))
subscriber.request(double(:event, payload: event_3)) subscriber.request(double(:event, payload: event_3, time: Time.current))
results = subject.results results = subject.results
expect(results[:calls]).to eq(3) expect(results[:calls]).to eq(3)
...@@ -55,6 +59,7 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do ...@@ -55,6 +59,7 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do
expected = [ expected = [
{ {
start: be_like_time(Time.current),
duration: 30.0, duration: 30.0,
label: "POST https://gitlab.com:80/api/v4/projects?current=true", label: "POST https://gitlab.com:80/api/v4/projects?current=true",
code: "Response status: 200", code: "Response status: 200",
...@@ -63,6 +68,7 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do ...@@ -63,6 +68,7 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do
warnings: [] warnings: []
}, },
{ {
start: be_like_time(Time.current),
duration: 1300, duration: 1300,
label: "POST http://gitlab.com:80/api/v4/projects/2/issues?current=true", label: "POST http://gitlab.com:80/api/v4/projects/2/issues?current=true",
code: nil, code: nil,
...@@ -71,6 +77,7 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do ...@@ -71,6 +77,7 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do
warnings: ["1300.0 over 100"] warnings: ["1300.0 over 100"]
}, },
{ {
start: be_like_time(Time.current),
duration: 5.0, duration: 5.0,
label: "GET http://gitlab.com:80/api/v4/projects/2?current=true", label: "GET http://gitlab.com:80/api/v4/projects/2?current=true",
code: "Response status: 301", code: "Response status: 301",
...@@ -81,7 +88,7 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do ...@@ -81,7 +88,7 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do
] ]
expect( expect(
results[:details].map { |data| data.slice(:duration, :label, :code, :proxy, :error, :warnings) } results[:details].map { |data| data.slice(:start, :duration, :label, :code, :proxy, :error, :warnings) }
).to match_array(expected) ).to match_array(expected)
end end
...@@ -91,10 +98,11 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do ...@@ -91,10 +98,11 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do
end end
it 'displays IPv4 in the label' do it 'displays IPv4 in the label' do
subscriber.request(double(:event, payload: event_1)) subscriber.request(double(:event, payload: event_1, time: Time.current))
expect(subject.results[:details]).to contain_exactly( expect(subject.results[:details]).to contain_exactly(
a_hash_including( a_hash_including(
start: be_like_time(Time.current),
duration: 30.0, duration: 30.0,
label: "POST https://1.2.3.4:80/api/v4/projects?current=true", label: "POST https://1.2.3.4:80/api/v4/projects?current=true",
code: "Response status: 200", code: "Response status: 200",
...@@ -112,10 +120,11 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do ...@@ -112,10 +120,11 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do
end end
it 'displays IPv6 in the label' do it 'displays IPv6 in the label' do
subscriber.request(double(:event, payload: event_1)) subscriber.request(double(:event, payload: event_1, time: Time.current))
expect(subject.results[:details]).to contain_exactly( expect(subject.results[:details]).to contain_exactly(
a_hash_including( a_hash_including(
start: be_like_time(Time.current),
duration: 30.0, duration: 30.0,
label: "POST https://[2606:4700:90:0:f22e:fbec:5bed:a9b9]:80/api/v4/projects?current=true", label: "POST https://[2606:4700:90:0:f22e:fbec:5bed:a9b9]:80/api/v4/projects?current=true",
code: "Response status: 200", code: "Response status: 200",
...@@ -133,10 +142,11 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do ...@@ -133,10 +142,11 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do
end end
it 'converts query hash into a query string' do it 'converts query hash into a query string' do
subscriber.request(double(:event, payload: event_1)) subscriber.request(double(:event, payload: event_1, time: Time.current))
expect(subject.results[:details]).to contain_exactly( expect(subject.results[:details]).to contain_exactly(
a_hash_including( a_hash_including(
start: be_like_time(Time.current),
duration: 30.0, duration: 30.0,
label: "POST https://gitlab.com:80/api/v4/projects?current=true&item1=string&item2%5B%5D=1&item2%5B%5D=2", label: "POST https://gitlab.com:80/api/v4/projects?current=true&item1=string&item2%5B%5D=1&item2%5B%5D=2",
code: "Response status: 200", code: "Response status: 200",
...@@ -154,10 +164,11 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do ...@@ -154,10 +164,11 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do
end end
it 'displays unknown in the label' do it 'displays unknown in the label' do
subscriber.request(double(:event, payload: event_1)) subscriber.request(double(:event, payload: event_1, time: Time.current))
expect(subject.results[:details]).to contain_exactly( expect(subject.results[:details]).to contain_exactly(
a_hash_including( a_hash_including(
start: be_like_time(Time.current),
duration: 30.0, duration: 30.0,
label: "POST unknown", label: "POST unknown",
code: "Response status: 200", code: "Response status: 200",
...@@ -176,10 +187,11 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do ...@@ -176,10 +187,11 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do
end end
it 'displays unknown in the label' do it 'displays unknown in the label' do
subscriber.request(double(:event, payload: event_1)) subscriber.request(double(:event, payload: event_1, time: Time.current))
expect(subject.results[:details]).to contain_exactly( expect(subject.results[:details]).to contain_exactly(
a_hash_including( a_hash_including(
start: be_like_time(Time.current),
duration: 30.0, duration: 30.0,
label: "POST unknown", label: "POST unknown",
code: "Response status: 200", code: "Response status: 200",
...@@ -198,10 +210,11 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do ...@@ -198,10 +210,11 @@ RSpec.describe Peek::Views::ExternalHttp, :request_store do
end end
it 'displays unknown in the label' do it 'displays unknown in the label' do
subscriber.request(double(:event, payload: event_1)) subscriber.request(double(:event, payload: event_1, time: Time.current))
expect(subject.results[:details]).to contain_exactly( expect(subject.results[:details]).to contain_exactly(
a_hash_including( a_hash_including(
start: be_like_time(Time.current),
duration: 30.0, duration: 30.0,
label: "POST unknown", label: "POST unknown",
code: "Response status: 200", code: "Response status: 200",
......
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