Commit eaadbcfc authored by Natalia Tepluhina's avatar Natalia Tepluhina

Merge branch '32778-change-retries-approach-promql-queries' into 'master'

Metrics dashboard: Fix error handling by using a backoff timeout and a flash notice

Closes #32778

See merge request gitlab-org/gitlab!20687
parents 8e7d3c67 d87b19bb
...@@ -4,28 +4,22 @@ import createFlash from '~/flash'; ...@@ -4,28 +4,22 @@ import createFlash from '~/flash';
import trackDashboardLoad from '../monitoring_tracking_helper'; import trackDashboardLoad from '../monitoring_tracking_helper';
import statusCodes from '../../lib/utils/http_status'; import statusCodes from '../../lib/utils/http_status';
import { backOff } from '../../lib/utils/common_utils'; import { backOff } from '../../lib/utils/common_utils';
import { s__, __ } from '../../locale'; import { s__ } from '../../locale';
const MAX_REQUESTS = 3; const TWO_MINUTES = 120000;
export function backOffRequest(makeRequestCallback) { function backOffRequest(makeRequestCallback) {
let requestCounter = 0;
return backOff((next, stop) => { return backOff((next, stop) => {
makeRequestCallback() makeRequestCallback()
.then(resp => { .then(resp => {
if (resp.status === statusCodes.NO_CONTENT) { if (resp.status === statusCodes.NO_CONTENT) {
requestCounter += 1; next();
if (requestCounter < MAX_REQUESTS) {
next();
} else {
stop(new Error(__('Failed to connect to the prometheus server')));
}
} else { } else {
stop(resp); stop(resp);
} }
}) })
.catch(stop); .catch(stop);
}); }, TWO_MINUTES);
} }
export const setGettingStartedEmptyState = ({ commit }) => { export const setGettingStartedEmptyState = ({ commit }) => {
...@@ -52,11 +46,6 @@ export const receiveMetricsDashboardFailure = ({ commit }, error) => { ...@@ -52,11 +46,6 @@ export const receiveMetricsDashboardFailure = ({ commit }, error) => {
commit(types.RECEIVE_METRICS_DATA_FAILURE, error); commit(types.RECEIVE_METRICS_DATA_FAILURE, error);
}; };
export const requestMetricsData = ({ commit }) => commit(types.REQUEST_METRICS_DATA);
export const receiveMetricsDataSuccess = ({ commit }, data) =>
commit(types.RECEIVE_METRICS_DATA_SUCCESS, data);
export const receiveMetricsDataFailure = ({ commit }, error) =>
commit(types.RECEIVE_METRICS_DATA_FAILURE, error);
export const receiveDeploymentsDataSuccess = ({ commit }, data) => export const receiveDeploymentsDataSuccess = ({ commit }, data) =>
commit(types.RECEIVE_DEPLOYMENTS_DATA_SUCCESS, data); commit(types.RECEIVE_DEPLOYMENTS_DATA_SUCCESS, data);
export const receiveDeploymentsDataFailure = ({ commit }) => export const receiveDeploymentsDataFailure = ({ commit }) =>
...@@ -149,11 +138,15 @@ export const fetchPrometheusMetrics = ({ state, commit, dispatch }, params) => { ...@@ -149,11 +138,15 @@ export const fetchPrometheusMetrics = ({ state, commit, dispatch }, params) => {
}); });
}); });
return Promise.all(promises).then(() => { return Promise.all(promises)
if (state.metricsWithData.length === 0) { .then(() => {
commit(types.SET_NO_DATA_EMPTY_STATE); if (state.metricsWithData.length === 0) {
} commit(types.SET_NO_DATA_EMPTY_STATE);
}); }
})
.catch(() => {
createFlash(s__(`Metrics|There was an error while retrieving metrics`), 'warning');
});
}; };
export const fetchDeploymentsData = ({ state, dispatch }) => { export const fetchDeploymentsData = ({ state, dispatch }) => {
......
---
title: Improve the way the metrics dashboard waits for data
merge_request: 20687
author:
type: fixed
...@@ -7181,9 +7181,6 @@ msgstr "" ...@@ -7181,9 +7181,6 @@ msgstr ""
msgid "Failed to check related branches." msgid "Failed to check related branches."
msgstr "" msgstr ""
msgid "Failed to connect to the prometheus server"
msgstr ""
msgid "Failed to create Merge Request. Please try again." msgid "Failed to create Merge Request. Please try again."
msgstr "" msgstr ""
......
import MockAdapter from 'axios-mock-adapter'; import MockAdapter from 'axios-mock-adapter';
import Tracking from '~/tracking'; import Tracking from '~/tracking';
import { TEST_HOST } from 'helpers/test_constants';
import testAction from 'helpers/vuex_action_helper'; import testAction from 'helpers/vuex_action_helper';
import axios from '~/lib/utils/axios_utils'; import axios from '~/lib/utils/axios_utils';
import statusCodes from '~/lib/utils/http_status'; import statusCodes from '~/lib/utils/http_status';
import { backOff } from '~/lib/utils/common_utils'; import { backOff } from '~/lib/utils/common_utils';
import createFlash from '~/flash';
import store from '~/monitoring/stores'; import store from '~/monitoring/stores';
import * as types from '~/monitoring/stores/mutation_types'; import * as types from '~/monitoring/stores/mutation_types';
import { import {
backOffRequest,
fetchDashboard, fetchDashboard,
receiveMetricsDashboardSuccess, receiveMetricsDashboardSuccess,
receiveMetricsDashboardFailure, receiveMetricsDashboardFailure,
...@@ -17,7 +16,6 @@ import { ...@@ -17,7 +16,6 @@ import {
fetchEnvironmentsData, fetchEnvironmentsData,
fetchPrometheusMetrics, fetchPrometheusMetrics,
fetchPrometheusMetric, fetchPrometheusMetric,
requestMetricsData,
setEndpoints, setEndpoints,
setGettingStartedEmptyState, setGettingStartedEmptyState,
} from '~/monitoring/stores/actions'; } from '~/monitoring/stores/actions';
...@@ -31,6 +29,7 @@ import { ...@@ -31,6 +29,7 @@ import {
} from '../mock_data'; } from '../mock_data';
jest.mock('~/lib/utils/common_utils'); jest.mock('~/lib/utils/common_utils');
jest.mock('~/flash');
const resetStore = str => { const resetStore = str => {
str.replaceState({ str.replaceState({
...@@ -40,71 +39,36 @@ const resetStore = str => { ...@@ -40,71 +39,36 @@ const resetStore = str => {
}); });
}; };
const MAX_REQUESTS = 3; describe('Monitoring store actions', () => {
describe('Monitoring store helpers', () => {
let mock; let mock;
// Mock underlying `backOff` function to remove in-built delay.
backOff.mockImplementation(
callback =>
new Promise((resolve, reject) => {
const stop = arg => (arg instanceof Error ? reject(arg) : resolve(arg));
const next = () => callback(next, stop);
callback(next, stop);
}),
);
beforeEach(() => { beforeEach(() => {
mock = new MockAdapter(axios); mock = new MockAdapter(axios);
});
afterEach(() => {
mock.restore();
});
describe('backOffRequest', () => {
it('returns immediately when recieving a 200 status code', () => {
mock.onGet(TEST_HOST).reply(200);
return backOffRequest(() => axios.get(TEST_HOST)).then(() => {
expect(mock.history.get.length).toBe(1);
});
});
it(`repeats the network call ${MAX_REQUESTS} times when receiving a 204 response`, done => { // Mock `backOff` function to remove exponential algorithm delay.
mock.onGet(TEST_HOST).reply(statusCodes.NO_CONTENT, {}); jest.useFakeTimers();
backOffRequest(() => axios.get(TEST_HOST)) backOff.mockImplementation(callback => {
.then(done.fail) const q = new Promise((resolve, reject) => {
.catch(() => { const stop = arg => (arg instanceof Error ? reject(arg) : resolve(arg));
expect(mock.history.get.length).toBe(MAX_REQUESTS); const next = () => callback(next, stop);
done(); // Define a timeout based on a mock timer
setTimeout(() => {
callback(next, stop);
}); });
});
// Run all resolved promises in chain
jest.runOnlyPendingTimers();
return q;
}); });
}); });
});
describe('Monitoring store actions', () => {
let mock;
beforeEach(() => {
mock = new MockAdapter(axios);
});
afterEach(() => { afterEach(() => {
resetStore(store); resetStore(store);
mock.restore(); mock.reset();
});
describe('requestMetricsData', () => { backOff.mockReset();
it('sets emptyState to loading', () => { createFlash.mockReset();
const commit = jest.fn();
const { state } = store;
requestMetricsData({
state,
commit,
});
expect(commit).toHaveBeenCalledWith(types.REQUEST_METRICS_DATA);
});
}); });
describe('fetchDeploymentsData', () => { describe('fetchDeploymentsData', () => {
it('commits RECEIVE_DEPLOYMENTS_DATA_SUCCESS on error', done => { it('commits RECEIVE_DEPLOYMENTS_DATA_SUCCESS on error', done => {
const dispatch = jest.fn(); const dispatch = jest.fn();
...@@ -362,17 +326,11 @@ describe('Monitoring store actions', () => { ...@@ -362,17 +326,11 @@ describe('Monitoring store actions', () => {
it('commits empty state when state.groups is empty', done => { it('commits empty state when state.groups is empty', done => {
const state = storeState(); const state = storeState();
const params = {}; const params = {};
fetchPrometheusMetrics( fetchPrometheusMetrics({ state, commit, dispatch }, params)
{
state,
commit,
dispatch,
},
params,
)
.then(() => { .then(() => {
expect(commit).toHaveBeenCalledWith(types.SET_NO_DATA_EMPTY_STATE); expect(commit).toHaveBeenCalledWith(types.SET_NO_DATA_EMPTY_STATE);
expect(dispatch).not.toHaveBeenCalled(); expect(dispatch).not.toHaveBeenCalled();
expect(createFlash).not.toHaveBeenCalled();
done(); done();
}) })
.catch(done.fail); .catch(done.fail);
...@@ -382,20 +340,42 @@ describe('Monitoring store actions', () => { ...@@ -382,20 +340,42 @@ describe('Monitoring store actions', () => {
const state = storeState(); const state = storeState();
state.dashboard.panel_groups = metricsDashboardResponse.dashboard.panel_groups; state.dashboard.panel_groups = metricsDashboardResponse.dashboard.panel_groups;
const metric = state.dashboard.panel_groups[0].panels[0].metrics[0]; const metric = state.dashboard.panel_groups[0].panels[0].metrics[0];
fetchPrometheusMetrics( fetchPrometheusMetrics({ state, commit, dispatch }, params)
{ .then(() => {
state, expect(dispatch).toHaveBeenCalledTimes(3);
commit, expect(dispatch).toHaveBeenCalledWith('fetchPrometheusMetric', {
dispatch, metric,
}, params,
params, });
)
expect(createFlash).not.toHaveBeenCalled();
done();
})
.catch(done.fail);
done();
});
it('dispatches fetchPrometheusMetric for each panel query, handles an error', done => {
const params = {};
const state = storeState();
state.dashboard.panel_groups = metricsDashboardResponse.dashboard.panel_groups;
const metric = state.dashboard.panel_groups[0].panels[0].metrics[0];
// Mock having one out of three metrics failing
dispatch.mockRejectedValueOnce(new Error('Error fetching this metric'));
dispatch.mockResolvedValue();
fetchPrometheusMetrics({ state, commit, dispatch }, params)
.then(() => { .then(() => {
expect(dispatch).toHaveBeenCalledTimes(3); expect(dispatch).toHaveBeenCalledTimes(3);
expect(dispatch).toHaveBeenCalledWith('fetchPrometheusMetric', { expect(dispatch).toHaveBeenCalledWith('fetchPrometheusMetric', {
metric, metric,
params, params,
}); });
expect(createFlash).toHaveBeenCalledTimes(1);
done(); done();
}) })
.catch(done.fail); .catch(done.fail);
...@@ -403,28 +383,75 @@ describe('Monitoring store actions', () => { ...@@ -403,28 +383,75 @@ describe('Monitoring store actions', () => {
}); });
}); });
describe('fetchPrometheusMetric', () => { describe('fetchPrometheusMetric', () => {
it('commits prometheus query result', done => { const params = {
const commit = jest.fn(); start: '2019-08-06T12:40:02.184Z',
const params = { end: '2019-08-06T20:40:02.184Z',
start: '2019-08-06T12:40:02.184Z', };
end: '2019-08-06T20:40:02.184Z', let commit;
}; let metric;
const metric = metricsDashboardResponse.dashboard.panel_groups[0].panels[0].metrics[0]; let state;
const state = storeState(); let data;
const data = metricsGroupsAPIResponse[0].panels[0].metrics[0];
const response = { beforeEach(() => {
data, commit = jest.fn();
}; state = storeState();
mock.onGet('http://test').reply(200, response); [metric] = metricsDashboardResponse.dashboard.panel_groups[0].panels[0].metrics;
[data] = metricsGroupsAPIResponse[0].panels[0].metrics;
});
it('commits result', done => {
mock.onGet('http://test').reply(200, { data }); // One attempt
fetchPrometheusMetric({ state, commit }, { metric, params }) fetchPrometheusMetric({ state, commit }, { metric, params })
.then(() => { .then(() => {
expect(commit).toHaveBeenCalledWith(types.SET_QUERY_RESULT, { expect(commit).toHaveBeenCalledWith(types.SET_QUERY_RESULT, {
metricId: metric.metric_id, metricId: metric.metric_id,
result: data.result, result: data.result,
}); });
expect(mock.history.get).toHaveLength(1);
done(); done();
}) })
.catch(done.fail); .catch(done.fail);
}); });
it('commits result, when waiting for results', done => {
// Mock multiple attempts while the cache is filling up
mock.onGet('http://test').replyOnce(statusCodes.NO_CONTENT);
mock.onGet('http://test').replyOnce(statusCodes.NO_CONTENT);
mock.onGet('http://test').replyOnce(statusCodes.NO_CONTENT);
mock.onGet('http://test').reply(200, { data }); // 4th attempt
const fetch = fetchPrometheusMetric({ state, commit }, { metric, params });
fetch
.then(() => {
expect(commit).toHaveBeenCalledWith(types.SET_QUERY_RESULT, {
metricId: metric.metric_id,
result: data.result,
});
expect(mock.history.get).toHaveLength(4);
done();
})
.catch(done.fail);
});
it('commits failure, when waiting for results and getting a server error', done => {
// Mock multiple attempts while the cache is filling up and fails
mock.onGet('http://test').replyOnce(statusCodes.NO_CONTENT);
mock.onGet('http://test').replyOnce(statusCodes.NO_CONTENT);
mock.onGet('http://test').replyOnce(statusCodes.NO_CONTENT);
mock.onGet('http://test').reply(500); // 4th attempt
fetchPrometheusMetric({ state, commit }, { metric, params })
.then(() => {
done.fail();
})
.catch(() => {
expect(commit).not.toHaveBeenCalled();
expect(mock.history.get).toHaveLength(4);
done();
});
});
}); });
}); });
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