2019-12-21 20:55:43 +05:30
|
|
|
import Visibility from 'visibilityjs';
|
|
|
|
import { GlLoadingIcon } from '@gitlab/ui';
|
2020-01-01 13:55:28 +05:30
|
|
|
import { shallowMount } from '@vue/test-utils';
|
2019-12-21 20:55:43 +05:30
|
|
|
import Poll from '~/lib/utils/poll';
|
2020-10-24 23:57:45 +05:30
|
|
|
import { deprecatedCreateFlash as flash } from '~/flash';
|
2019-12-21 20:55:43 +05:30
|
|
|
import CiIcon from '~/vue_shared/components/ci_icon.vue';
|
|
|
|
import CommitPipelineStatus from '~/projects/tree/components/commit_pipeline_status_component.vue';
|
|
|
|
import { getJSONFixture } from '../helpers/fixtures';
|
|
|
|
|
|
|
|
jest.mock('~/lib/utils/poll');
|
|
|
|
jest.mock('visibilityjs');
|
|
|
|
jest.mock('~/flash');
|
|
|
|
|
|
|
|
const mockFetchData = jest.fn();
|
|
|
|
jest.mock('~/projects/tree/services/commit_pipeline_service', () =>
|
|
|
|
jest.fn().mockImplementation(() => ({
|
|
|
|
fetchData: mockFetchData.mockReturnValue(Promise.resolve()),
|
|
|
|
})),
|
|
|
|
);
|
|
|
|
|
|
|
|
describe('Commit pipeline status component', () => {
|
|
|
|
let wrapper;
|
|
|
|
const { pipelines } = getJSONFixture('pipelines/pipelines.json');
|
|
|
|
const { status: mockCiStatus } = pipelines[0].details;
|
|
|
|
|
|
|
|
const defaultProps = {
|
|
|
|
endpoint: 'endpoint',
|
|
|
|
};
|
|
|
|
|
|
|
|
const createComponent = (props = {}) => {
|
|
|
|
wrapper = shallowMount(CommitPipelineStatus, {
|
|
|
|
propsData: {
|
|
|
|
...defaultProps,
|
|
|
|
...props,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const findLoader = () => wrapper.find(GlLoadingIcon);
|
|
|
|
const findLink = () => wrapper.find('a');
|
|
|
|
const findCiIcon = () => findLink().find(CiIcon);
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
wrapper.destroy();
|
|
|
|
wrapper = null;
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('Visibility management', () => {
|
|
|
|
describe('when component is hidden', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
Visibility.hidden.mockReturnValue(true);
|
|
|
|
createComponent();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does not start polling', () => {
|
|
|
|
const [pollInstance] = Poll.mock.instances;
|
|
|
|
expect(pollInstance.makeRequest).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('requests pipeline data', () => {
|
|
|
|
expect(mockFetchData).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('when component is visible', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
Visibility.hidden.mockReturnValue(false);
|
|
|
|
createComponent();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('starts polling', () => {
|
|
|
|
const [pollInstance] = [...Poll.mock.instances].reverse();
|
|
|
|
expect(pollInstance.makeRequest).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('when component changes its visibility', () => {
|
|
|
|
it.each`
|
|
|
|
visibility | action
|
|
|
|
${false} | ${'restart'}
|
|
|
|
${true} | ${'stop'}
|
|
|
|
`(
|
|
|
|
'$action polling when component visibility becomes $visibility',
|
|
|
|
({ visibility, action }) => {
|
|
|
|
Visibility.hidden.mockReturnValue(!visibility);
|
|
|
|
createComponent();
|
|
|
|
const [pollInstance] = Poll.mock.instances;
|
|
|
|
expect(pollInstance[action]).not.toHaveBeenCalled();
|
|
|
|
Visibility.hidden.mockReturnValue(visibility);
|
|
|
|
const [visibilityHandler] = Visibility.change.mock.calls[0];
|
|
|
|
visibilityHandler();
|
|
|
|
expect(pollInstance[action]).toHaveBeenCalled();
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('stops polling when component is destroyed', () => {
|
|
|
|
createComponent();
|
|
|
|
wrapper.destroy();
|
|
|
|
const [pollInstance] = Poll.mock.instances;
|
|
|
|
expect(pollInstance.stop).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('when polling', () => {
|
|
|
|
let pollConfig;
|
|
|
|
beforeEach(() => {
|
|
|
|
Poll.mockImplementation(config => {
|
|
|
|
pollConfig = config;
|
|
|
|
return { makeRequest: jest.fn(), restart: jest.fn(), stop: jest.fn() };
|
|
|
|
});
|
|
|
|
createComponent();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('shows the loading icon at start', () => {
|
|
|
|
createComponent();
|
|
|
|
expect(findLoader().exists()).toBe(true);
|
|
|
|
|
|
|
|
pollConfig.successCallback({
|
|
|
|
data: { pipelines: [] },
|
|
|
|
});
|
|
|
|
|
|
|
|
return wrapper.vm.$nextTick().then(() => {
|
|
|
|
expect(findLoader().exists()).toBe(false);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('is successful', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
pollConfig.successCallback({
|
|
|
|
data: { pipelines: [{ details: { status: mockCiStatus } }] },
|
|
|
|
});
|
|
|
|
return wrapper.vm.$nextTick();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does not render loader', () => {
|
|
|
|
expect(findLoader().exists()).toBe(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('renders link with href', () => {
|
|
|
|
expect(findLink().attributes('href')).toEqual(mockCiStatus.details_path);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('renders CI icon', () => {
|
2021-01-03 14:25:43 +05:30
|
|
|
expect(findCiIcon().attributes('title')).toEqual('Pipeline: pending');
|
2019-12-21 20:55:43 +05:30
|
|
|
expect(findCiIcon().props('status')).toEqual(mockCiStatus);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('is not successful', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
pollConfig.errorCallback();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does not render loader', () => {
|
|
|
|
expect(findLoader().exists()).toBe(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('renders link with href', () => {
|
|
|
|
expect(findLink().attributes('href')).toBeUndefined();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('renders not found CI icon', () => {
|
2021-01-03 14:25:43 +05:30
|
|
|
expect(findCiIcon().attributes('title')).toEqual('Pipeline: not found');
|
2019-12-21 20:55:43 +05:30
|
|
|
expect(findCiIcon().props('status')).toEqual({
|
|
|
|
text: 'not found',
|
|
|
|
icon: 'status_notfound',
|
|
|
|
group: 'notfound',
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('displays flash error message', () => {
|
|
|
|
expect(flash).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|