Commit 7c1b4c48 authored by George Koltsov's avatar George Koltsov

Emit Bitbucket Server Importer metrics

- For each bitbucket server import,
  increment importer projects counter,
  imported merge requests counter as well as
  measure import duration of imported project
parent 5564f6bf
---
title: Emit Bitbucket Server Importer metrics
merge_request: 33700
author:
type: changed
...@@ -3,8 +3,6 @@ ...@@ -3,8 +3,6 @@
module Gitlab module Gitlab
module BitbucketImport module BitbucketImport
class Importer class Importer
include Gitlab::BitbucketImport::Metrics
LABELS = [{ title: 'bug', color: '#FF0000' }, LABELS = [{ title: 'bug', color: '#FF0000' },
{ title: 'enhancement', color: '#428BCA' }, { title: 'enhancement', color: '#428BCA' },
{ title: 'proposal', color: '#69D100' }, { title: 'proposal', color: '#69D100' },
...@@ -26,6 +24,7 @@ module Gitlab ...@@ -26,6 +24,7 @@ module Gitlab
import_issues import_issues
import_pull_requests import_pull_requests
handle_errors handle_errors
metrics.track_finished_import
true true
end end
...@@ -115,6 +114,8 @@ module Gitlab ...@@ -115,6 +114,8 @@ module Gitlab
updated_at: issue.updated_at updated_at: issue.updated_at
) )
metrics.issues_counter.increment
gitlab_issue.labels << @labels[label_name] gitlab_issue.labels << @labels[label_name]
import_issue_comments(issue, gitlab_issue) if gitlab_issue.persisted? import_issue_comments(issue, gitlab_issue) if gitlab_issue.persisted?
...@@ -195,6 +196,8 @@ module Gitlab ...@@ -195,6 +196,8 @@ module Gitlab
updated_at: pull_request.updated_at updated_at: pull_request.updated_at
) )
metrics.merge_requests_counter.increment
import_pull_request_comments(pull_request, merge_request) if merge_request.persisted? import_pull_request_comments(pull_request, merge_request) if merge_request.persisted?
rescue StandardError => e rescue StandardError => e
store_pull_request_error(pull_request, e) store_pull_request_error(pull_request, e)
...@@ -288,6 +291,10 @@ module Gitlab ...@@ -288,6 +291,10 @@ module Gitlab
project_path: project.full_path project_path: project.full_path
} }
end end
def metrics
@metrics ||= Gitlab::Import::Metrics.new(:bitbucket_importer, @project)
end
end end
end end
end end
# frozen_string_literal: true
module Gitlab
module BitbucketImport
module Metrics
extend ActiveSupport::Concern
IMPORTER = :bitbucket_importer
included do
prepend Gitlab::Import::Metrics
Gitlab::Import::Metrics.measure(:execute, metrics: {
"#{IMPORTER}_imported_projects": {
type: :counter,
description: 'The number of imported Bitbucket projects'
},
"#{IMPORTER}_total_duration_seconds": {
type: :histogram,
labels: { importer: IMPORTER },
description: 'Total time spent importing Bitbucket projects, in seconds'
}
})
Gitlab::Import::Metrics.measure(:import_issue, metrics: {
"#{IMPORTER}_imported_issues": {
type: :counter,
description: 'The number of imported Bitbucket issues'
}
})
Gitlab::Import::Metrics.measure(:import_pull_request, metrics: {
"#{IMPORTER}_imported_pull_requests": {
type: :counter,
description: 'The number of imported Bitbucket pull requests'
}
})
end
end
end
end
...@@ -43,6 +43,7 @@ module Gitlab ...@@ -43,6 +43,7 @@ module Gitlab
import_pull_requests import_pull_requests
delete_temp_branches delete_temp_branches
handle_errors handle_errors
metrics.track_finished_import
log_info(stage: "complete") log_info(stage: "complete")
...@@ -219,7 +220,11 @@ module Gitlab ...@@ -219,7 +220,11 @@ module Gitlab
creator = Gitlab::Import::MergeRequestCreator.new(project) creator = Gitlab::Import::MergeRequestCreator.new(project)
merge_request = creator.execute(attributes) merge_request = creator.execute(attributes)
import_pull_request_comments(pull_request, merge_request) if merge_request.persisted? if merge_request.persisted?
import_pull_request_comments(pull_request, merge_request)
metrics.merge_requests_counter.increment
end
log_info(stage: 'import_bitbucket_pull_requests', message: 'finished', iid: pull_request.iid) log_info(stage: 'import_bitbucket_pull_requests', message: 'finished', iid: pull_request.iid)
end end
...@@ -388,6 +393,10 @@ module Gitlab ...@@ -388,6 +393,10 @@ module Gitlab
project_path: project.full_path project_path: project.full_path
} }
end end
def metrics
@metrics ||= Gitlab::Import::Metrics.new(:bitbucket_server_importer, @project)
end
end end
end end
end end
# frozen_string_literal: true # frozen_string_literal: true
# Prepend `Gitlab::Import::Metrics` to a class in order
# to measure and emit `Gitlab::Metrics` metrics of specified methods.
#
# @example
# class Importer
# prepend Gitlab::Import::Metrics
#
# Gitlab::ImportExport::Metrics.measure :execute, metrics: {
# importer_counter: {
# type: :counter,
# description: 'counter'
# },
# importer_histogram: {
# type: :histogram,
# labels: { importer: 'importer' },
# description: 'histogram'
# }
# }
#
# def execute
# ...
# end
# end
#
# Each call to `#execute` increments `importer_counter` as well as
# measures `#execute` duration and reports histogram `importer_histogram`
module Gitlab module Gitlab
module Import module Import
module Metrics class Metrics
def self.measure(method_name, metrics:) IMPORT_DURATION_BUCKETS = [0.5, 1, 3, 5, 10, 60, 120, 240, 360, 720, 1440].freeze
define_method "#{method_name}" do |*args|
start_time = Time.zone.now
result = super(*args) attr_reader :importer
end_time = Time.zone.now def initialize(importer, project)
@importer = importer
@project = project
end
def track_finished_import
duration = Time.zone.now - @project.created_at
duration_histogram.observe({ importer: importer }, duration)
projects_counter.increment
end
report_measurement_metrics(metrics, end_time - start_time) def projects_counter
@projects_counter ||= Gitlab::Metrics.counter(
:"#{importer}_imported_projects_total",
'The number of imported projects'
)
end
def issues_counter
@issues_counter ||= Gitlab::Metrics.counter(
:"#{importer}_imported_issues_total",
'The number of imported issues'
)
end
result def merge_requests_counter
end @merge_requests_counter ||= Gitlab::Metrics.counter(
:"#{importer}_imported_merge_requests_total",
'The number of imported merge (pull) requests'
)
end end
def report_measurement_metrics(metrics, duration) private
metrics.each do |metric_name, metric_value|
case metric_value[:type] def duration_histogram
when :counter @duration_histogram ||= Gitlab::Metrics.histogram(
Gitlab::Metrics.counter(metric_name, metric_value[:description]).increment :"#{importer}_total_duration_seconds",
when :histogram 'Total time spent importing projects, in seconds',
Gitlab::Metrics.histogram(metric_name, metric_value[:description]).observe(metric_value[:labels], duration) {},
else IMPORT_DURATION_BUCKETS
nil )
end
end
end end
end end
end end
......
...@@ -226,8 +226,8 @@ describe Gitlab::BitbucketImport::Importer do ...@@ -226,8 +226,8 @@ describe Gitlab::BitbucketImport::Importer do
it 'counts imported pull requests' do it 'counts imported pull requests' do
expect(Gitlab::Metrics).to receive(:counter).with( expect(Gitlab::Metrics).to receive(:counter).with(
:bitbucket_importer_imported_pull_requests, :bitbucket_importer_imported_merge_requests_total,
'The number of imported Bitbucket pull requests' 'The number of imported merge (pull) requests'
) )
expect(counter).to receive(:increment) expect(counter).to receive(:increment)
...@@ -369,8 +369,8 @@ describe Gitlab::BitbucketImport::Importer do ...@@ -369,8 +369,8 @@ describe Gitlab::BitbucketImport::Importer do
it 'counts imported issues' do it 'counts imported issues' do
expect(Gitlab::Metrics).to receive(:counter).with( expect(Gitlab::Metrics).to receive(:counter).with(
:bitbucket_importer_imported_issues, :bitbucket_importer_imported_issues_total,
'The number of imported Bitbucket issues' 'The number of imported issues'
) )
expect(counter).to receive(:increment) expect(counter).to receive(:increment)
...@@ -389,23 +389,27 @@ describe Gitlab::BitbucketImport::Importer do ...@@ -389,23 +389,27 @@ describe Gitlab::BitbucketImport::Importer do
allow(subject).to receive(:import_issues) allow(subject).to receive(:import_issues)
allow(subject).to receive(:import_pull_requests) allow(subject).to receive(:import_pull_requests)
allow(Gitlab::Metrics).to receive(:counter) { counter } allow(Gitlab::Metrics).to receive(:counter).and_return(counter)
allow(Gitlab::Metrics).to receive(:histogram) { histogram } allow(Gitlab::Metrics).to receive(:histogram).and_return(histogram)
allow(histogram).to receive(:observe)
allow(counter).to receive(:increment)
end end
it 'counts and measures duration of imported projects' do it 'counts and measures duration of imported projects' do
expect(Gitlab::Metrics).to receive(:counter).with( expect(Gitlab::Metrics).to receive(:counter).with(
:bitbucket_importer_imported_projects, :bitbucket_importer_imported_projects_total,
'The number of imported Bitbucket projects' 'The number of imported projects'
) )
expect(Gitlab::Metrics).to receive(:histogram).with( expect(Gitlab::Metrics).to receive(:histogram).with(
:bitbucket_importer_total_duration_seconds, :bitbucket_importer_total_duration_seconds,
'Total time spent importing Bitbucket projects, in seconds' 'Total time spent importing projects, in seconds',
{},
Gitlab::Import::Metrics::IMPORT_DURATION_BUCKETS
) )
expect(counter).to receive(:increment) expect(counter).to receive(:increment)
expect(histogram).to receive(:observe).with({ importer: described_class::IMPORTER }, anything) expect(histogram).to receive(:observe).with({ importer: :bitbucket_importer }, anything)
subject.execute subject.execute
end end
......
...@@ -5,7 +5,10 @@ require 'spec_helper' ...@@ -5,7 +5,10 @@ require 'spec_helper'
describe Gitlab::BitbucketServerImport::Importer do describe Gitlab::BitbucketServerImport::Importer do
include ImportSpecHelper include ImportSpecHelper
let(:project) { create(:project, :repository, import_url: 'http://my-bitbucket') } let(:import_url) { 'http://my-bitbucket' }
let(:user) { 'bitbucket' }
let(:password) { 'test' }
let(:project) { create(:project, :repository, import_url: import_url) }
let(:now) { Time.now.utc.change(usec: 0) } let(:now) { Time.now.utc.change(usec: 0) }
let(:project_key) { 'TEST' } let(:project_key) { 'TEST' }
let(:repo_slug) { 'rouge' } let(:repo_slug) { 'rouge' }
...@@ -16,7 +19,7 @@ describe Gitlab::BitbucketServerImport::Importer do ...@@ -16,7 +19,7 @@ describe Gitlab::BitbucketServerImport::Importer do
before do before do
data = project.create_or_update_import_data( data = project.create_or_update_import_data(
data: { project_key: project_key, repo_slug: repo_slug }, data: { project_key: project_key, repo_slug: repo_slug },
credentials: { base_uri: 'http://my-bitbucket', user: 'bitbucket', password: 'test' } credentials: { base_uri: import_url, user: user, password: password }
) )
data.save data.save
project.save project.save
...@@ -125,6 +128,48 @@ describe Gitlab::BitbucketServerImport::Importer do ...@@ -125,6 +128,48 @@ describe Gitlab::BitbucketServerImport::Importer do
expect(note.updated_at).to eq(@pr_note.created_at) expect(note.updated_at).to eq(@pr_note.created_at)
end end
context 'metrics' do
let(:histogram) { double(:histogram) }
let(:counter) { double('counter', increment: true) }
before do
allow(Gitlab::Metrics).to receive(:counter) { counter }
allow(Gitlab::Metrics).to receive(:histogram) { histogram }
allow(subject.client).to receive(:activities).and_return([@merge_event])
end
it 'counts and measures duration of imported projects' do
expect(Gitlab::Metrics).to receive(:counter).with(
:bitbucket_server_importer_imported_projects_total,
'The number of imported projects'
)
expect(Gitlab::Metrics).to receive(:histogram).with(
:bitbucket_server_importer_total_duration_seconds,
'Total time spent importing projects, in seconds',
{},
Gitlab::Import::Metrics::IMPORT_DURATION_BUCKETS
)
expect(counter).to receive(:increment)
expect(histogram).to receive(:observe).with({ importer: :bitbucket_server_importer }, anything)
subject.execute
end
it 'counts imported pull requests' do
expect(Gitlab::Metrics).to receive(:counter).with(
:bitbucket_server_importer_imported_merge_requests_total,
'The number of imported merge (pull) requests'
)
expect(counter).to receive(:increment)
allow(histogram).to receive(:observe).with({ importer: :bitbucket_server_importer }, anything)
subject.execute
end
end
it 'imports threaded discussions' do it 'imports threaded discussions' do
reply = instance_double( reply = instance_double(
BitbucketServer::Representation::PullRequestComment, BitbucketServer::Representation::PullRequestComment,
......
...@@ -3,54 +3,38 @@ ...@@ -3,54 +3,38 @@
require 'spec_helper' require 'spec_helper'
describe Gitlab::Import::Metrics do describe Gitlab::Import::Metrics do
let(:importer_stub) do let(:importer) { :test_importer }
Class.new do let(:project) { create(:project) }
prepend Gitlab::Import::Metrics let(:histogram) { double(:histogram) }
let(:counter) { double(:counter) }
Gitlab::Import::Metrics.measure :execute, metrics: {
importer_counter: { subject { described_class.new(importer, project) }
type: :counter,
description: 'description' describe '#report_import_time' do
}, before do
importer_histogram: { allow(Gitlab::Metrics).to receive(:counter) { counter }
type: :histogram, allow(Gitlab::Metrics).to receive(:histogram) { histogram }
labels: { importer: 'importer' }, allow(counter).to receive(:increment)
description: 'description' allow(counter).to receive(:observe)
}
}
def execute
true
end
end end
end
subject { importer_stub.new.execute }
describe '#execute' do it 'emits importer metrics' do
let(:counter) { double(:counter) } expect(Gitlab::Metrics).to receive(:counter).with(
let(:histogram) { double(:histogram) } :test_importer_imported_projects_total,
'The number of imported projects'
)
it 'increments counter metric' do expect(Gitlab::Metrics).to receive(:histogram).with(
expect(Gitlab::Metrics) :test_importer_total_duration_seconds,
.to receive(:counter) 'Total time spent importing projects, in seconds',
.with(:importer_counter, 'description') {},
.and_return(counter) described_class::IMPORT_DURATION_BUCKETS
)
expect(counter).to receive(:increment) expect(counter).to receive(:increment)
expect(histogram).to receive(:observe).with({ importer: :test_importer }, anything)
subject subject.track_finished_import
end
it 'measures method duration and reports histogram metric' do
expect(Gitlab::Metrics)
.to receive(:histogram)
.with(:importer_histogram, 'description')
.and_return(histogram)
expect(histogram).to receive(:observe).with({ importer: 'importer' }, anything)
subject
end end
end end
end end
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