Commit e699bb7f authored by Allison Browne's avatar Allison Browne Committed by Kamil Trzciński

Add sorting to Sentry error tracking

Allow a sort param on the issue index of the
error tracking controller
parent 5eb21d86
...@@ -111,7 +111,7 @@ class Projects::ErrorTrackingController < Projects::ApplicationController ...@@ -111,7 +111,7 @@ class Projects::ErrorTrackingController < Projects::ApplicationController
end end
def list_issues_params def list_issues_params
params.permit(:search_term) params.permit([:search_term, :sort])
end end
def list_projects_params def list_projects_params
......
...@@ -5,6 +5,7 @@ module ErrorTracking ...@@ -5,6 +5,7 @@ module ErrorTracking
include Gitlab::Utils::StrongMemoize include Gitlab::Utils::StrongMemoize
include ReactiveCaching include ReactiveCaching
SENTRY_API_ERROR_TYPE_BAD_REQUEST = 'bad_request_for_sentry_api'
SENTRY_API_ERROR_TYPE_MISSING_KEYS = 'missing_keys_in_sentry_response' SENTRY_API_ERROR_TYPE_MISSING_KEYS = 'missing_keys_in_sentry_response'
SENTRY_API_ERROR_TYPE_NON_20X_RESPONSE = 'non_20x_response_from_sentry' SENTRY_API_ERROR_TYPE_NON_20X_RESPONSE = 'non_20x_response_from_sentry'
SENTRY_API_ERROR_INVALID_SIZE = 'invalid_size_of_sentry_response' SENTRY_API_ERROR_INVALID_SIZE = 'invalid_size_of_sentry_response'
...@@ -119,6 +120,8 @@ module ErrorTracking ...@@ -119,6 +120,8 @@ module ErrorTracking
{ error: e.message, error_type: SENTRY_API_ERROR_TYPE_MISSING_KEYS } { error: e.message, error_type: SENTRY_API_ERROR_TYPE_MISSING_KEYS }
rescue Sentry::Client::ResponseInvalidSizeError => e rescue Sentry::Client::ResponseInvalidSizeError => e
{ error: e.message, error_type: SENTRY_API_ERROR_INVALID_SIZE } { error: e.message, error_type: SENTRY_API_ERROR_INVALID_SIZE }
rescue Sentry::Client::BadRequestError => e
{ error: e.message, error_type: SENTRY_API_ERROR_TYPE_BAD_REQUEST }
end end
# http://HOST/api/0/projects/ORG/PROJECT # http://HOST/api/0/projects/ORG/PROJECT
......
...@@ -4,6 +4,7 @@ module ErrorTracking ...@@ -4,6 +4,7 @@ module ErrorTracking
class ListIssuesService < ErrorTracking::BaseService class ListIssuesService < ErrorTracking::BaseService
DEFAULT_ISSUE_STATUS = 'unresolved' DEFAULT_ISSUE_STATUS = 'unresolved'
DEFAULT_LIMIT = 20 DEFAULT_LIMIT = 20
DEFAULT_SORT = 'last_seen'
def execute def execute
return error('Error Tracking is not enabled') unless enabled? return error('Error Tracking is not enabled') unless enabled?
...@@ -12,7 +13,8 @@ module ErrorTracking ...@@ -12,7 +13,8 @@ module ErrorTracking
result = project_error_tracking_setting.list_sentry_issues( result = project_error_tracking_setting.list_sentry_issues(
issue_status: issue_status, issue_status: issue_status,
limit: limit, limit: limit,
search_term: search_term search_term: search_term,
sort: sort
) )
# our results are not yet ready # our results are not yet ready
...@@ -33,10 +35,6 @@ module ErrorTracking ...@@ -33,10 +35,6 @@ module ErrorTracking
private private
def fetch
project_error_tracking_setting.list_sentry_issues(issue_status: issue_status, limit: limit)
end
def parse_response(response) def parse_response(response)
{ issues: response[:issues] } { issues: response[:issues] }
end end
...@@ -60,5 +58,9 @@ module ErrorTracking ...@@ -60,5 +58,9 @@ module ErrorTracking
def can_read? def can_read?
can?(current_user, :read_sentry_issue, project) can?(current_user, :read_sentry_issue, project)
end end
def sort
params[:sort] || DEFAULT_SORT
end
end end
end end
---
title: Add sort param to error tracking issue index
merge_request: 20101
author:
type: changed
...@@ -5,6 +5,14 @@ module Sentry ...@@ -5,6 +5,14 @@ module Sentry
Error = Class.new(StandardError) Error = Class.new(StandardError)
MissingKeysError = Class.new(StandardError) MissingKeysError = Class.new(StandardError)
ResponseInvalidSizeError = Class.new(StandardError) ResponseInvalidSizeError = Class.new(StandardError)
BadRequestError = Class.new(StandardError)
SENTRY_API_SORT_VALUE_MAP = {
# <accepted_by_client> => <accepted_by_sentry_api>
'frequency' => 'freq',
'first_seen' => 'new',
'last_seen' => nil
}.freeze
attr_accessor :url, :token attr_accessor :url, :token
...@@ -25,12 +33,8 @@ module Sentry ...@@ -25,12 +33,8 @@ module Sentry
map_to_event(latest_event) map_to_event(latest_event)
end end
def list_issues(issue_status:, limit:, search_term: '') def list_issues(**keyword_args)
issues = get_issues( issues = get_issues(keyword_args)
issue_status: issue_status,
limit: limit,
search_term: search_term
)
validate_size(issues) validate_size(issues)
...@@ -52,14 +56,14 @@ module Sentry ...@@ -52,14 +56,14 @@ module Sentry
def validate_size(issues) def validate_size(issues)
return if Gitlab::Utils::DeepSize.new(issues).valid? return if Gitlab::Utils::DeepSize.new(issues).valid?
raise Client::ResponseInvalidSizeError, "Sentry API response is too big. Limit is #{Gitlab::Utils::DeepSize.human_default_max_size}." raise ResponseInvalidSizeError, "Sentry API response is too big. Limit is #{Gitlab::Utils::DeepSize.human_default_max_size}."
end end
def handle_mapping_exceptions(&block) def handle_mapping_exceptions(&block)
yield yield
rescue KeyError => e rescue KeyError => e
Gitlab::Sentry.track_acceptable_exception(e) Gitlab::Sentry.track_acceptable_exception(e)
raise Client::MissingKeysError, "Sentry API response is missing keys. #{e.message}" raise MissingKeysError, "Sentry API response is missing keys. #{e.message}"
end end
def request_params def request_params
...@@ -78,13 +82,25 @@ module Sentry ...@@ -78,13 +82,25 @@ module Sentry
handle_response(response) handle_response(response)
end end
def get_issues(issue_status:, limit:, search_term: '') def get_issues(**keyword_args)
query = "is:#{issue_status} #{search_term}".strip http_get(
issues_api_url,
query: list_issue_sentry_query(keyword_args)
)
end
def list_issue_sentry_query(issue_status:, limit:, sort: nil, search_term: '')
unless SENTRY_API_SORT_VALUE_MAP.key?(sort)
raise BadRequestError, 'Invalid value for sort param'
end
query_params = {
query: "is:#{issue_status} #{search_term}".strip,
limit: limit,
sort: SENTRY_API_SORT_VALUE_MAP[sort]
}
http_get(issues_api_url, query: { query_params.compact
query: query,
limit: limit
})
end end
def get_issue(issue_id:) def get_issue(issue_id:)
......
...@@ -48,20 +48,17 @@ describe Projects::ErrorTrackingController do ...@@ -48,20 +48,17 @@ describe Projects::ErrorTrackingController do
describe 'format json' do describe 'format json' do
let(:list_issues_service) { spy(:list_issues_service) } let(:list_issues_service) { spy(:list_issues_service) }
let(:external_url) { 'http://example.com' } let(:external_url) { 'http://example.com' }
let(:search_term) do
ActionController::Parameters.new(
search_term: 'something'
).permit!
end
context 'no data' do context 'no data' do
let(:search_term) do let(:params) { project_params(format: :json) }
let(:permitted_params) do
ActionController::Parameters.new({}).permit! ActionController::Parameters.new({}).permit!
end end
before do before do
expect(ErrorTracking::ListIssuesService) expect(ErrorTracking::ListIssuesService)
.to receive(:new).with(project, user, search_term) .to receive(:new).with(project, user, permitted_params)
.and_return(list_issues_service) .and_return(list_issues_service)
expect(list_issues_service).to receive(:execute) expect(list_issues_service).to receive(:execute)
...@@ -75,10 +72,16 @@ describe Projects::ErrorTrackingController do ...@@ -75,10 +72,16 @@ describe Projects::ErrorTrackingController do
end end
end end
context 'with a search_term param' do context 'with a search_term and sort params' do
let(:params) { project_params(format: :json, search_term: 'something', sort: 'last_seen') }
let(:permitted_params) do
ActionController::Parameters.new(search_term: 'something', sort: 'last_seen').permit!
end
before do before do
expect(ErrorTracking::ListIssuesService) expect(ErrorTracking::ListIssuesService)
.to receive(:new).with(project, user, search_term) .to receive(:new).with(project, user, permitted_params)
.and_return(list_issues_service) .and_return(list_issues_service)
end end
...@@ -93,7 +96,7 @@ describe Projects::ErrorTrackingController do ...@@ -93,7 +96,7 @@ describe Projects::ErrorTrackingController do
let(:error) { build(:error_tracking_error) } let(:error) { build(:error_tracking_error) }
it 'returns a list of errors' do it 'returns a list of errors' do
get :index, params: project_params(format: :json, search_term: 'something') get :index, params: params
expect(response).to have_gitlab_http_status(:ok) expect(response).to have_gitlab_http_status(:ok)
expect(response).to match_response_schema('error_tracking/index') expect(response).to match_response_schema('error_tracking/index')
...@@ -103,7 +106,7 @@ describe Projects::ErrorTrackingController do ...@@ -103,7 +106,7 @@ describe Projects::ErrorTrackingController do
end end
end end
context 'without a search_term param' do context 'without params' do
before do before do
expect(ErrorTracking::ListIssuesService) expect(ErrorTracking::ListIssuesService)
.to receive(:new).with(project, user, {}) .to receive(:new).with(project, user, {})
......
...@@ -5,6 +5,12 @@ require 'spec_helper' ...@@ -5,6 +5,12 @@ require 'spec_helper'
describe Sentry::Client do describe Sentry::Client do
let(:sentry_url) { 'https://sentrytest.gitlab.com/api/0/projects/sentry-org/sentry-project' } let(:sentry_url) { 'https://sentrytest.gitlab.com/api/0/projects/sentry-org/sentry-project' }
let(:token) { 'test-token' } let(:token) { 'test-token' }
let(:default_httparty_options) do
{
follow_redirects: false,
headers: { "Authorization" => "Bearer test-token" }
}
end
let(:issues_sample_response) do let(:issues_sample_response) do
Gitlab::Utils.deep_indifferent_access( Gitlab::Utils.deep_indifferent_access(
...@@ -94,7 +100,7 @@ describe Sentry::Client do ...@@ -94,7 +100,7 @@ describe Sentry::Client do
let!(:sentry_api_request) { stub_sentry_request(sentry_request_url, body: sentry_api_response) } let!(:sentry_api_request) { stub_sentry_request(sentry_request_url, body: sentry_api_response) }
subject { client.list_issues(issue_status: issue_status, limit: limit, search_term: search_term) } subject { client.list_issues(issue_status: issue_status, limit: limit, search_term: search_term, sort: 'last_seen') }
it_behaves_like 'calls sentry api' it_behaves_like 'calls sentry api'
...@@ -165,6 +171,35 @@ describe Sentry::Client do ...@@ -165,6 +171,35 @@ describe Sentry::Client do
end end
end end
context 'requests with sort parameter in sentry api' do
let(:sentry_request_url) do
'https://sentrytest.gitlab.com/api/0/projects/sentry-org/sentry-project/' \
'issues/?limit=20&query=is:unresolved&sort=freq'
end
let!(:sentry_api_request) { stub_sentry_request(sentry_request_url, body: sentry_api_response) }
subject { client.list_issues(issue_status: issue_status, limit: limit, sort: 'frequency') }
it 'calls the sentry api with sort params' do
expect(Gitlab::HTTP).to receive(:get).with(
URI("#{sentry_url}/issues/"),
default_httparty_options.merge(query: { limit: 20, query: "is:unresolved", sort: "freq" })
).and_call_original
subject
expect(sentry_api_request).to have_been_requested
end
end
context 'with invalid sort params' do
subject { client.list_issues(issue_status: issue_status, limit: limit, sort: 'fish') }
it 'throws an error' do
expect { subject }.to raise_error(Sentry::Client::BadRequestError, 'Invalid value for sort param')
end
end
context 'Older sentry versions where keys are not present' do context 'Older sentry versions where keys are not present' do
let(:sentry_api_response) do let(:sentry_api_response) do
issues_sample_response[0...1].map do |issue| issues_sample_response[0...1].map do |issue|
......
...@@ -5,12 +5,13 @@ require 'spec_helper' ...@@ -5,12 +5,13 @@ require 'spec_helper'
describe ErrorTracking::ListIssuesService do describe ErrorTracking::ListIssuesService do
set(:user) { create(:user) } set(:user) { create(:user) }
set(:project) { create(:project) } set(:project) { create(:project) }
let(:params) { { search_term: 'something' } } let(:params) { { search_term: 'something', sort: 'last_seen' } }
let(:list_sentry_issues_args) do let(:list_sentry_issues_args) do
{ {
issue_status: 'unresolved', issue_status: 'unresolved',
limit: 20, limit: 20,
search_term: params[:search_term] search_term: params[:search_term],
sort: params[:sort]
} }
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