import { mount, shallowMount } from '@vue/test-utils'; import Vue, { nextTick } from 'vue'; import VueApollo from 'vue-apollo'; import createMockApollo from 'helpers/mock_apollo_helper'; import waitForPromises from 'helpers/wait_for_promises'; import getPipelineDetails from 'shared_queries/pipelines/get_pipeline_details.query.graphql'; import { DOWNSTREAM, UPSTREAM, LAYER_VIEW, STAGE_VIEW, } from '~/pipelines/components/graph/constants'; import PipelineGraph from '~/pipelines/components/graph/graph_component.vue'; import LinkedPipeline from '~/pipelines/components/graph/linked_pipeline.vue'; import LinkedPipelinesColumn from '~/pipelines/components/graph/linked_pipelines_column.vue'; import * as parsingUtils from '~/pipelines/components/parsing_utils'; import { LOAD_FAILURE } from '~/pipelines/constants'; import { mockPipelineResponse, pipelineWithUpstreamDownstream, wrappedPipelineReturn, } from './mock_data'; const processedPipeline = pipelineWithUpstreamDownstream(mockPipelineResponse); describe('Linked Pipelines Column', () => { const defaultProps = { columnTitle: 'Downstream', linkedPipelines: processedPipeline.downstream, showLinks: false, type: DOWNSTREAM, viewType: STAGE_VIEW, configPaths: { metricsPath: '', graphqlResourceEtag: 'this/is/a/path', }, }; let wrapper; const findLinkedColumnTitle = () => wrapper.find('[data-testid="linked-column-title"]'); const findLinkedPipelineElements = () => wrapper.findAll(LinkedPipeline); const findPipelineGraph = () => wrapper.find(PipelineGraph); const findExpandButton = () => wrapper.find('[data-testid="expand-pipeline-button"]'); Vue.use(VueApollo); const createComponent = ({ apolloProvider, mountFn = shallowMount, props = {} } = {}) => { wrapper = mountFn(LinkedPipelinesColumn, { apolloProvider, propsData: { ...defaultProps, ...props, }, }); }; const createComponentWithApollo = ({ mountFn = shallowMount, getPipelineDetailsHandler = jest.fn().mockResolvedValue(wrappedPipelineReturn), props = {}, } = {}) => { const requestHandlers = [[getPipelineDetails, getPipelineDetailsHandler]]; const apolloProvider = createMockApollo(requestHandlers); createComponent({ apolloProvider, mountFn, props }); }; afterEach(() => { wrapper.destroy(); wrapper = null; }); describe('it renders correctly', () => { beforeEach(() => { createComponentWithApollo(); }); it('renders the pipeline title', () => { expect(findLinkedColumnTitle().text()).toBe(defaultProps.columnTitle); }); it('renders the correct number of linked pipelines', () => { expect(findLinkedPipelineElements()).toHaveLength(defaultProps.linkedPipelines.length); }); }); describe('click action', () => { const clickExpandButton = async () => { await findExpandButton().trigger('click'); await waitForPromises(); }; describe('layer type rendering', () => { let layersFn; beforeEach(() => { layersFn = jest.spyOn(parsingUtils, 'listByLayers'); createComponentWithApollo({ mountFn: mount }); }); it('calls listByLayers only once no matter how many times view is switched', async () => { expect(layersFn).not.toHaveBeenCalled(); await clickExpandButton(); await wrapper.setProps({ viewType: LAYER_VIEW }); await nextTick(); expect(layersFn).toHaveBeenCalledTimes(1); await wrapper.setProps({ viewType: STAGE_VIEW }); await wrapper.setProps({ viewType: LAYER_VIEW }); await wrapper.setProps({ viewType: STAGE_VIEW }); expect(layersFn).toHaveBeenCalledTimes(1); }); }); describe('when graph does not use needs', () => { beforeEach(() => { const nonNeedsResponse = { ...wrappedPipelineReturn }; nonNeedsResponse.data.project.pipeline.usesNeeds = false; createComponentWithApollo({ props: { viewType: LAYER_VIEW, }, getPipelineDetailsHandler: jest.fn().mockResolvedValue(nonNeedsResponse), mountFn: mount, }); }); it('shows the stage view, even when the main graph view type is layers', async () => { await clickExpandButton(); expect(findPipelineGraph().props('viewType')).toBe(STAGE_VIEW); }); }); describe('downstream', () => { describe('when successful', () => { beforeEach(() => { createComponentWithApollo({ mountFn: mount }); }); it('toggles the pipeline visibility', async () => { expect(findPipelineGraph().exists()).toBe(false); await clickExpandButton(); expect(findPipelineGraph().exists()).toBe(true); await clickExpandButton(); expect(findPipelineGraph().exists()).toBe(false); }); }); describe('on error', () => { beforeEach(() => { createComponentWithApollo({ mountFn: mount, getPipelineDetailsHandler: jest.fn().mockRejectedValue(new Error('GraphQL error')), }); }); it('emits the error', async () => { await clickExpandButton(); expect(wrapper.emitted().error).toEqual([[{ type: LOAD_FAILURE, skipSentry: true }]]); }); it('does not show the pipeline', async () => { expect(findPipelineGraph().exists()).toBe(false); await clickExpandButton(); expect(findPipelineGraph().exists()).toBe(false); }); }); }); describe('upstream', () => { const upstreamProps = { columnTitle: 'Upstream', /* Because the IDs need to match to work, rather than make new mock data, we are representing the upstream pipeline with the downstream data. */ linkedPipelines: processedPipeline.downstream, type: UPSTREAM, }; describe('when successful', () => { beforeEach(() => { createComponentWithApollo({ mountFn: mount, props: upstreamProps, }); }); it('toggles the pipeline visibility', async () => { expect(findPipelineGraph().exists()).toBe(false); await clickExpandButton(); expect(findPipelineGraph().exists()).toBe(true); await clickExpandButton(); expect(findPipelineGraph().exists()).toBe(false); }); }); describe('on error', () => { beforeEach(() => { createComponentWithApollo({ mountFn: mount, getPipelineDetailsHandler: jest.fn().mockRejectedValue(new Error('GraphQL error')), props: upstreamProps, }); }); it('emits the error', async () => { await clickExpandButton(); expect(wrapper.emitted().error).toEqual([[{ type: LOAD_FAILURE, skipSentry: true }]]); }); it('does not show the pipeline', async () => { expect(findPipelineGraph().exists()).toBe(false); await clickExpandButton(); expect(findPipelineGraph().exists()).toBe(false); }); }); }); }); });