Commit 93f16f3d authored by Kushal Pandya's avatar Kushal Pandya

Merge branch 'winh-fix-flaky-dashboard_spec' into 'master'

Move monitoring dashboard_spec to Jest

Closes #66922

See merge request gitlab-org/gitlab-ce!32571
parents 913c87c6 d8f24afa
...@@ -40,6 +40,7 @@ class CustomEnvironment extends JSDOMEnvironment { ...@@ -40,6 +40,7 @@ class CustomEnvironment extends JSDOMEnvironment {
this.global.fixturesBasePath = `${ROOT_PATH}/tmp/tests/frontend/fixtures${IS_EE ? '-ee' : ''}`; this.global.fixturesBasePath = `${ROOT_PATH}/tmp/tests/frontend/fixtures${IS_EE ? '-ee' : ''}`;
this.global.staticFixturesBasePath = `${ROOT_PATH}/spec/frontend/fixtures`; this.global.staticFixturesBasePath = `${ROOT_PATH}/spec/frontend/fixtures`;
this.global.IS_EE = IS_EE;
// Not yet supported by JSDOM: https://github.com/jsdom/jsdom/issues/317 // Not yet supported by JSDOM: https://github.com/jsdom/jsdom/issues/317
this.global.document.createRange = () => ({ this.global.document.createRange = () => ({
......
import Vue from 'vue'; import Vue from 'vue';
import { shallowMount, createLocalVue } from '@vue/test-utils'; import { shallowMount, createLocalVue } from '@vue/test-utils';
import { GlToast } from '@gitlab/ui'; import { GlToast, GlDropdownItem } from '@gitlab/ui';
import MockAdapter from 'axios-mock-adapter'; import MockAdapter from 'axios-mock-adapter';
import Dashboard from '~/monitoring/components/dashboard.vue'; import Dashboard from '~/monitoring/components/dashboard.vue';
import { timeWindows, timeWindowsKeyNames } from '~/monitoring/constants'; import GraphGroup from '~/monitoring/components/graph_group.vue';
import EmptyState from '~/monitoring/components/empty_state.vue';
import { timeWindows } from '~/monitoring/constants';
import * as types from '~/monitoring/stores/mutation_types'; import * as types from '~/monitoring/stores/mutation_types';
import { createStore } from '~/monitoring/stores'; import { createStore } from '~/monitoring/stores';
import axios from '~/lib/utils/axios_utils'; import axios from '~/lib/utils/axios_utils';
// TODO: replace with dynamic fixture
// https://gitlab.com/gitlab-org/gitlab-ce/issues/62785
import MonitoringMock, { import MonitoringMock, {
metricsGroupsAPIResponse, metricsGroupsAPIResponse,
mockApiEndpoint, mockApiEndpoint,
environmentData, environmentData,
singleGroupResponse, singleGroupResponse,
dashboardGitResponse, dashboardGitResponse,
} from '../mock_data'; } from '../../../../spec/javascripts/monitoring/mock_data';
/* eslint-disable no-unused-vars */
/* eslint-disable no-undef */
// see https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/32571#note_211860465
function setupComponentStore(component) {
component.$store.commit(
`monitoringDashboard/${types.RECEIVE_METRICS_DATA_SUCCESS}`,
metricsGroupsAPIResponse,
);
component.$store.commit(
`monitoringDashboard/${types.SET_QUERY_RESULT}`,
mockedQueryResultPayload,
);
component.$store.commit(
`monitoringDashboard/${types.RECEIVE_ENVIRONMENTS_DATA_SUCCESS}`,
environmentData,
);
}
// Mock imported files while retaining the original behaviour
// See https://github.com/facebook/jest/issues/936#issuecomment-265074320
function mockMonitoringUtils() {
const original = require.requireActual('~/monitoring/utils');
return {
...original, // Pass down all the exported objects
getTimeDiff: jest.spyOn(original, 'getTimeDiff'),
};
}
jest.mock('~/monitoring/utils', () => mockMonitoringUtils());
const monitoringUtils = require.requireMock('~/monitoring/utils');
function mockUrlUtility() {
const original = require.requireActual('~/lib/utils/url_utility');
return {
...original, // Pass down all the exported objects
getParameterValues: jest.spyOn(original, 'getParameterValues'),
};
}
jest.mock('~/lib/utils/url_utility', () => mockUrlUtility());
const urlUtility = require.requireMock('~/lib/utils/url_utility');
const localVue = createLocalVue(); const localVue = createLocalVue();
const propsData = { const propsData = {
...@@ -83,7 +128,7 @@ describe('Dashboard', () => { ...@@ -83,7 +128,7 @@ describe('Dashboard', () => {
}); });
it('shows the environment selector', () => { it('shows the environment selector', () => {
expect(component.$el.querySelector('.js-environments-dropdown')).toBeTruthy(); expect(component.$el.querySelector('#monitor-environments-dropdown')).toBeTruthy();
}); });
}); });
...@@ -95,7 +140,7 @@ describe('Dashboard', () => { ...@@ -95,7 +140,7 @@ describe('Dashboard', () => {
store, store,
}); });
expect(component.$el.querySelector('.js-environments-dropdown')).toBeTruthy(); expect(component.$el.querySelector('#monitor-environments-dropdown')).toBeTruthy();
}); });
}); });
...@@ -117,47 +162,27 @@ describe('Dashboard', () => { ...@@ -117,47 +162,27 @@ describe('Dashboard', () => {
}); });
}); });
it('hides the legend when showLegend is false', done => {
component = new DashboardComponent({
el: document.querySelector('.prometheus-graphs'),
propsData: {
...propsData,
hasMetrics: true,
showLegend: false,
},
store,
});
setTimeout(() => {
expect(component.showEmptyState).toEqual(false);
expect(component.$el.querySelector('.legend-group')).toEqual(null);
expect(component.$el.querySelector('.prometheus-graph-group')).toBeTruthy();
done();
});
});
it('hides the group panels when showPanels is false', done => { it('hides the group panels when showPanels is false', done => {
component = new DashboardComponent({ const wrapper = shallowMount(DashboardComponent, {
el: document.querySelector('.prometheus-graphs'),
propsData: { propsData: {
...propsData, ...propsData,
hasMetrics: true, hasMetrics: true,
showPanels: false, showPanels: false,
}, },
store, store,
sync: false,
localVue,
}); });
setImmediate(() => {
setTimeout(() => { expect(wrapper.find(EmptyState).exists()).toBe(false);
expect(component.showEmptyState).toEqual(false); expect(wrapper.find(GraphGroup).exists()).toBe(true);
expect(component.$el.querySelector('.prometheus-panel')).toEqual(null); expect(wrapper.find(GraphGroup).props().showPanels).toBe(false);
expect(component.$el.querySelector('.prometheus-graph-group')).toBeTruthy();
done(); done();
}); });
}); });
it('renders the environments dropdown with a number of environments', done => { it('renders the environments dropdown with a number of environments', () => {
component = new DashboardComponent({ const wrapper = shallowMount(DashboardComponent, {
el: document.querySelector('.prometheus-graphs'),
propsData: { propsData: {
...propsData, ...propsData,
hasMetrics: true, hasMetrics: true,
...@@ -166,38 +191,30 @@ describe('Dashboard', () => { ...@@ -166,38 +191,30 @@ describe('Dashboard', () => {
store, store,
}); });
component.$store.commit( store.commit(
`monitoringDashboard/${types.RECEIVE_ENVIRONMENTS_DATA_SUCCESS}`, `monitoringDashboard/${types.RECEIVE_ENVIRONMENTS_DATA_SUCCESS}`,
environmentData, environmentData,
); );
component.$store.commit( store.commit(
`monitoringDashboard/${types.RECEIVE_METRICS_DATA_SUCCESS}`, `monitoringDashboard/${types.RECEIVE_METRICS_DATA_SUCCESS}`,
singleGroupResponse, singleGroupResponse,
); );
Vue.nextTick() Vue.nextTick(() => {
.then(() => { const dropdownMenuEnvironments = wrapper
const dropdownMenuEnvironments = component.$el.querySelectorAll( .find('.js-environments-dropdown')
'.js-environments-dropdown .dropdown-item', .findAll(GlDropdownItem);
);
expect(component.environments.length).toEqual(environmentData.length);
expect(dropdownMenuEnvironments.length).toEqual(component.environments.length);
Array.from(dropdownMenuEnvironments).forEach((value, index) => {
if (environmentData[index].metrics_path) {
expect(value).toHaveAttr('href', environmentData[index].metrics_path);
}
});
done(); expect(environmentData.length).toBeGreaterThan(0);
}) expect(dropdownMenuEnvironments.length).toEqual(environmentData.length);
.catch(done.fail); dropdownMenuEnvironments.wrappers.forEach((value, index) => {
expect(value.attributes('href')).toEqual(environmentData[index].metrics_path);
});
});
}); });
it('hides the environments dropdown list when there is no environments', done => { it('hides the environments dropdown list when there is no environments', () => {
component = new DashboardComponent({ const wrapper = shallowMount(DashboardComponent, {
el: document.querySelector('.prometheus-graphs'),
propsData: { propsData: {
...propsData, ...propsData,
hasMetrics: true, hasMetrics: true,
...@@ -206,54 +223,48 @@ describe('Dashboard', () => { ...@@ -206,54 +223,48 @@ describe('Dashboard', () => {
store, store,
}); });
component.$store.commit(`monitoringDashboard/${types.RECEIVE_ENVIRONMENTS_DATA_SUCCESS}`, []); const findEnvironmentsDropdownItems = () => wrapper.find('#monitor-environments-wrapper');
component.$store.commit(
store.commit(`monitoringDashboard/${types.RECEIVE_ENVIRONMENTS_DATA_SUCCESS}`, []);
store.commit(
`monitoringDashboard/${types.RECEIVE_METRICS_DATA_SUCCESS}`, `monitoringDashboard/${types.RECEIVE_METRICS_DATA_SUCCESS}`,
singleGroupResponse, singleGroupResponse,
); );
Vue.nextTick() return Vue.nextTick(() => {
.then(() => { expect(findEnvironmentsDropdownItems(wrapper).exists()).toEqual(false);
const dropdownMenuEnvironments = component.$el.querySelectorAll( });
'.js-environments-dropdown .dropdown-item',
);
expect(dropdownMenuEnvironments.length).toEqual(0);
done();
})
.catch(done.fail);
}); });
it('renders the environments dropdown with a single active element', done => { it('renders the environments dropdown with a single active element', () => {
component = new DashboardComponent({ const wrapper = shallowMount(DashboardComponent, {
el: document.querySelector('.prometheus-graphs'),
propsData: { propsData: {
...propsData, ...propsData,
hasMetrics: true, hasMetrics: true,
showPanels: false, showPanels: false,
}, },
store, store,
sync: false,
localVue,
}); });
component.$store.commit( store.commit(
`monitoringDashboard/${types.RECEIVE_ENVIRONMENTS_DATA_SUCCESS}`, `monitoringDashboard/${types.RECEIVE_ENVIRONMENTS_DATA_SUCCESS}`,
environmentData, environmentData,
); );
component.$store.commit( store.commit(
`monitoringDashboard/${types.RECEIVE_METRICS_DATA_SUCCESS}`, `monitoringDashboard/${types.RECEIVE_METRICS_DATA_SUCCESS}`,
singleGroupResponse, singleGroupResponse,
); );
Vue.nextTick() Vue.nextTick(() => {
.then(() => { const activeDropdownMenuEnvironments = wrapper
const dropdownItems = component.$el.querySelectorAll( .find('#monitor-environments-dropdown')
'.js-environments-dropdown .dropdown-item.active', .findAll(GlDropdownItem)
); .filter(item => item.attributes('active') === 'true');
expect(dropdownItems.length).toEqual(1); expect(activeDropdownMenuEnvironments.length).toEqual(1);
done(); });
})
.catch(done.fail);
}); });
it('hides the dropdown', done => { it('hides the dropdown', done => {
...@@ -277,33 +288,40 @@ describe('Dashboard', () => { ...@@ -277,33 +288,40 @@ describe('Dashboard', () => {
}); });
it('renders the time window dropdown with a set of options', done => { it('renders the time window dropdown with a set of options', done => {
component = new DashboardComponent({ const wrapper = shallowMount(DashboardComponent, {
el: document.querySelector('.prometheus-graphs'),
propsData: { propsData: {
...propsData, ...propsData,
hasMetrics: true, hasMetrics: true,
showPanels: false, showPanels: false,
sync: false,
}, },
store, store,
}); });
const numberOfTimeWindows = Object.keys(timeWindows).length; const numberOfTimeWindows = Object.keys(timeWindows).length;
setTimeout(() => { setImmediate(() => {
const timeWindowDropdown = component.$el.querySelector('.js-time-window-dropdown'); const timeWindowDropdown = wrapper.find('.js-time-window-dropdown');
const timeWindowDropdownEls = component.$el.querySelectorAll( const timeWindowDropdownEls = wrapper
'.js-time-window-dropdown .dropdown-item', .find('.js-time-window-dropdown')
); .findAll(GlDropdownItem);
expect(timeWindowDropdown).not.toBeNull(); expect(timeWindowDropdown.exists()).toBe(true);
expect(timeWindowDropdownEls.length).toEqual(numberOfTimeWindows); expect(timeWindowDropdownEls.length).toEqual(numberOfTimeWindows);
done(); done();
}); });
}); });
it('fetches the metrics data with proper time window', done => { it('fetches the metrics data with proper time window', () => {
component = new DashboardComponent({ jest.spyOn(store, 'dispatch').mockImplementationOnce(() => {});
el: document.querySelector('.prometheus-graphs'),
store.commit(
`monitoringDashboard/${types.RECEIVE_ENVIRONMENTS_DATA_SUCCESS}`,
environmentData,
);
shallowMount(DashboardComponent, {
propsData: { propsData: {
...propsData, ...propsData,
hasMetrics: true, hasMetrics: true,
...@@ -312,75 +330,49 @@ describe('Dashboard', () => { ...@@ -312,75 +330,49 @@ describe('Dashboard', () => {
store, store,
}); });
spyOn(component.$store, 'dispatch').and.stub(); const defaultRange = monitoringUtils.getTimeDiff();
const getTimeDiffSpy = spyOnDependency(Dashboard, 'getTimeDiff').and.callThrough(); return Vue.nextTick().then(() => {
expect(store.dispatch).toHaveBeenCalledWith('monitoringDashboard/fetchData', defaultRange);
component.$store.commit( });
`monitoringDashboard/${types.RECEIVE_ENVIRONMENTS_DATA_SUCCESS}`,
environmentData,
);
component.$mount();
Vue.nextTick()
.then(() => {
expect(component.$store.dispatch).toHaveBeenCalled();
expect(getTimeDiffSpy).toHaveBeenCalled();
done();
})
.catch(done.fail);
}); });
it('shows a specific time window selected from the url params', done => { it('shows a specific time window selected from the url params', done => {
const start = 1564439536; const start = 1564439536;
const end = 1564441336; const end = 1564441336;
spyOnDependency(Dashboard, 'getTimeDiff').and.returnValue({ monitoringUtils.getTimeDiff.mockReturnValueOnce({
start, start,
end, end,
}); });
spyOnDependency(Dashboard, 'getParameterValues').and.callFake(param => { urlUtility.getParameterValues.mockImplementationOnce(param => {
if (param === 'start') return [start]; if (param === 'start') return [start];
if (param === 'end') return [end]; if (param === 'end') return [end];
return []; return [];
}); });
component = new DashboardComponent({ const wrapper = shallowMount(DashboardComponent, {
el: document.querySelector('.prometheus-graphs'), propsData: {
propsData: { ...propsData, hasMetrics: true }, ...propsData,
hasMetrics: true,
showPanels: false,
},
store, store,
}); });
setTimeout(() => { setImmediate(() => {
const selectedTimeWindow = component.$el.querySelector('.js-time-window-dropdown .active'); const activeTimeWindowItems = wrapper
.find('.js-time-window-dropdown')
expect(selectedTimeWindow.textContent.trim()).toEqual('30 minutes'); .findAll(GlDropdownItem)
done(); .filter(item => item.attributes('active') === 'true');
});
});
it('defaults to the eight hours time window for non valid url parameters', done => { expect(activeTimeWindowItems.length).toEqual(1);
spyOnDependency(Dashboard, 'getParameterValues').and.returnValue([ expect(activeTimeWindowItems.wrappers[0].text().trim()).toEqual('30 minutes');
'<script>alert("XSS")</script>',
]);
component = new DashboardComponent({
el: document.querySelector('.prometheus-graphs'),
propsData: { ...propsData, hasMetrics: true },
store,
});
Vue.nextTick(() => {
expect(component.selectedTimeWindowKey).toEqual(timeWindowsKeyNames.eightHours);
done(); done();
}); });
}); });
}); });
// https://gitlab.com/gitlab-org/gitlab-ce/issues/66922 describe('link to chart', () => {
// eslint-disable-next-line jasmine/no-disabled-tests
xdescribe('link to chart', () => {
let wrapper; let wrapper;
const currentDashboard = 'TEST_DASHBOARD'; const currentDashboard = 'TEST_DASHBOARD';
localVue.use(GlToast); localVue.use(GlToast);
...@@ -392,13 +384,13 @@ describe('Dashboard', () => { ...@@ -392,13 +384,13 @@ describe('Dashboard', () => {
wrapper = shallowMount(DashboardComponent, { wrapper = shallowMount(DashboardComponent, {
localVue, localVue,
sync: false,
attachToDocument: true,
propsData: { ...propsData, hasMetrics: true, currentDashboard }, propsData: { ...propsData, hasMetrics: true, currentDashboard },
store, store,
}); });
setTimeout(done); setImmediate(() => {
done();
});
}); });
afterEach(() => { afterEach(() => {
...@@ -437,7 +429,7 @@ describe('Dashboard', () => { ...@@ -437,7 +429,7 @@ describe('Dashboard', () => {
}); });
it('creates a toast when clicked', () => { it('creates a toast when clicked', () => {
spyOn(wrapper.vm.$toast, 'show').and.stub(); jest.spyOn(wrapper.vm.$toast, 'show').mockImplementation(() => {});
link().vm.$emit('click'); link().vm.$emit('click');
...@@ -448,11 +440,6 @@ describe('Dashboard', () => { ...@@ -448,11 +440,6 @@ describe('Dashboard', () => {
describe('when the window resizes', () => { describe('when the window resizes', () => {
beforeEach(() => { beforeEach(() => {
mock.onGet(mockApiEndpoint).reply(200, metricsGroupsAPIResponse); mock.onGet(mockApiEndpoint).reply(200, metricsGroupsAPIResponse);
jasmine.clock().install();
});
afterEach(() => {
jasmine.clock().uninstall();
}); });
it('sets elWidth to page width when the sidebar is resized', done => { it('sets elWidth to page width when the sidebar is resized', done => {
...@@ -473,7 +460,7 @@ describe('Dashboard', () => { ...@@ -473,7 +460,7 @@ describe('Dashboard', () => {
Vue.nextTick() Vue.nextTick()
.then(() => { .then(() => {
jasmine.clock().tick(1000); jest.advanceTimersByTime(1000);
return Vue.nextTick(); return Vue.nextTick();
}) })
.then(() => { .then(() => {
...@@ -485,11 +472,12 @@ describe('Dashboard', () => { ...@@ -485,11 +472,12 @@ describe('Dashboard', () => {
}); });
describe('external dashboard link', () => { describe('external dashboard link', () => {
beforeEach(() => { let wrapper;
beforeEach(done => {
mock.onGet(mockApiEndpoint).reply(200, metricsGroupsAPIResponse); mock.onGet(mockApiEndpoint).reply(200, metricsGroupsAPIResponse);
component = new DashboardComponent({ wrapper = shallowMount(DashboardComponent, {
el: document.querySelector('.prometheus-graphs'),
propsData: { propsData: {
...propsData, ...propsData,
hasMetrics: true, hasMetrics: true,
...@@ -498,62 +486,54 @@ describe('Dashboard', () => { ...@@ -498,62 +486,54 @@ describe('Dashboard', () => {
externalDashboardUrl: '/mockUrl', externalDashboardUrl: '/mockUrl',
}, },
store, store,
sync: false,
localVue,
}); });
setImmediate(done);
}); });
it('shows the link', done => { it('shows the link', () => {
setTimeout(() => { expect(wrapper.find('.js-external-dashboard-link').text()).toContain('View full dashboard');
expect(component.$el.querySelector('.js-external-dashboard-link').innerText).toContain(
'View full dashboard',
);
done();
});
}); });
}); });
describe('Dashboard dropdown', () => { describe('Dashboard dropdown', () => {
beforeEach(() => { let wrapper;
beforeEach(done => {
mock.onGet(mockApiEndpoint).reply(200, metricsGroupsAPIResponse); mock.onGet(mockApiEndpoint).reply(200, metricsGroupsAPIResponse);
component = new DashboardComponent({ wrapper = shallowMount(DashboardComponent, {
el: document.querySelector('.prometheus-graphs'), propsData: { ...propsData, hasMetrics: true, showPanels: false },
propsData: {
...propsData,
hasMetrics: true,
showPanels: false,
},
store, store,
sync: false,
localVue,
}); });
component.$store.dispatch('monitoringDashboard/setFeatureFlags', { setImmediate(() => {
prometheusEndpoint: false, store.dispatch('monitoringDashboard/setFeatureFlags', {
multipleDashboardsEnabled: true, prometheusEndpoint: false,
}); multipleDashboardsEnabled: true,
});
component.$store.commit( store.commit(
`monitoringDashboard/${types.RECEIVE_ENVIRONMENTS_DATA_SUCCESS}`, `monitoringDashboard/${types.RECEIVE_ENVIRONMENTS_DATA_SUCCESS}`,
environmentData, environmentData,
); );
store.commit(
component.$store.commit( `monitoringDashboard/${types.RECEIVE_METRICS_DATA_SUCCESS}`,
`monitoringDashboard/${types.RECEIVE_METRICS_DATA_SUCCESS}`, singleGroupResponse,
singleGroupResponse, );
);
component.$store.commit(
`monitoringDashboard/${types.SET_ALL_DASHBOARDS}`,
dashboardGitResponse,
);
});
it('shows the dashboard dropdown', done => {
setTimeout(() => {
const dashboardDropdown = component.$el.querySelector('.js-dashboards-dropdown');
expect(dashboardDropdown).not.toEqual(null); store.commit(`monitoringDashboard/${types.SET_ALL_DASHBOARDS}`, dashboardGitResponse);
done(); done();
}); });
}); });
it('shows the dashboard dropdown', () => {
expect(wrapper.find('.js-dashboards-dropdown').exists()).toEqual(true);
});
}); });
describe('when downloading metrics data as CSV', () => { describe('when downloading metrics data as CSV', () => {
...@@ -577,13 +557,25 @@ describe('Dashboard', () => { ...@@ -577,13 +557,25 @@ describe('Dashboard', () => {
const data = mockGraphData.queries[0].result[0].values; const data = mockGraphData.queries[0].result[0].values;
const firstRow = `${data[0][0]},${data[0][1]}`; const firstRow = `${data[0][0]},${data[0][1]}`;
expect(component.csvText(mockGraphData)).toMatch(`^${header}\r\n${firstRow}`); expect(component.csvText(mockGraphData)).toContain(`${header}\r\n${firstRow}`);
}); });
}); });
describe('downloadCsv', () => { describe('downloadCsv', () => {
it('produces a link with a Blob', () => { let spy;
expect(component.downloadCsv(mockGraphData)).toContain(`blob:`);
beforeEach(() => {
spy = jest.spyOn(window.URL, 'createObjectURL');
});
afterEach(() => {
spy.mockRestore();
});
it('creates a string containing a URL that represents the object', () => {
component.downloadCsv(mockGraphData);
expect(spy).toHaveBeenCalled();
}); });
}); });
}); });
......
...@@ -73,6 +73,9 @@ expect.extend(customMatchers); ...@@ -73,6 +73,9 @@ expect.extend(customMatchers);
// Tech debt issue TBD // Tech debt issue TBD
testUtilsConfig.logModifiedComponents = false; testUtilsConfig.logModifiedComponents = false;
// Stub for URL.createObjectURL
window.URL.createObjectURL = function createObjectURL() {};
// Basic stub for MutationObserver // Basic stub for MutationObserver
global.MutationObserver = () => ({ global.MutationObserver = () => ({
disconnect: () => {}, disconnect: () => {},
......
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