import Vue, { nextTick } from 'vue'; import { GlButton } from '@gitlab/ui'; import VueApollo from 'vue-apollo'; import createMockApollo from 'helpers/mock_apollo_helper'; import { createMockDirective, getBinding } from 'helpers/vue_mock_directive'; import { shallowMountExtended, mountExtended } from 'helpers/vue_test_utils_helper'; import runnerToggleActiveMutation from '~/runner/graphql/shared/runner_toggle_active.mutation.graphql'; import waitForPromises from 'helpers/wait_for_promises'; import { captureException } from '~/runner/sentry_utils'; import { createAlert } from '~/flash'; import { I18N_PAUSE, I18N_PAUSE_TOOLTIP, I18N_RESUME, I18N_RESUME_TOOLTIP, } from '~/runner/constants'; import RunnerPauseButton from '~/runner/components/runner_pause_button.vue'; import { allRunnersData } from '../mock_data'; const mockRunner = allRunnersData.data.runners.nodes[0]; Vue.use(VueApollo); jest.mock('~/flash'); jest.mock('~/runner/sentry_utils'); describe('RunnerPauseButton', () => { let wrapper; let runnerToggleActiveHandler; const getTooltip = () => getBinding(wrapper.element, 'gl-tooltip').value; const findBtn = () => wrapper.findComponent(GlButton); const createComponent = ({ props = {}, mountFn = shallowMountExtended } = {}) => { const { runner, ...propsData } = props; wrapper = mountFn(RunnerPauseButton, { propsData: { runner: { id: mockRunner.id, active: mockRunner.active, ...runner, }, ...propsData, }, apolloProvider: createMockApollo([[runnerToggleActiveMutation, runnerToggleActiveHandler]]), directives: { GlTooltip: createMockDirective(), }, }); }; const clickAndWait = async () => { findBtn().vm.$emit('click'); await waitForPromises(); }; beforeEach(() => { runnerToggleActiveHandler = jest.fn().mockImplementation(({ input }) => { return Promise.resolve({ data: { runnerUpdate: { runner: { id: input.id, active: input.active, }, errors: [], }, }, }); }); createComponent(); }); afterEach(() => { wrapper.destroy(); }); describe('Pause/Resume action', () => { describe.each` runnerState | icon | content | tooltip | isActive | newActiveValue ${'paused'} | ${'play'} | ${I18N_RESUME} | ${I18N_RESUME_TOOLTIP} | ${false} | ${true} ${'active'} | ${'pause'} | ${I18N_PAUSE} | ${I18N_PAUSE_TOOLTIP} | ${true} | ${false} `('When the runner is $runnerState', ({ icon, content, tooltip, isActive, newActiveValue }) => { beforeEach(() => { createComponent({ props: { runner: { active: isActive, }, }, }); }); it(`Displays a ${icon} button`, () => { expect(findBtn().props('loading')).toBe(false); expect(findBtn().props('icon')).toBe(icon); }); it('Displays button content', () => { expect(findBtn().text()).toBe(content); expect(getTooltip()).toBe(tooltip); }); it('Does not display redundant text for screen readers', () => { expect(findBtn().attributes('aria-label')).toBe(undefined); }); describe(`Before the ${icon} button is clicked`, () => { it('The mutation has not been called', () => { expect(runnerToggleActiveHandler).toHaveBeenCalledTimes(0); }); }); describe(`Immediately after the ${icon} button is clicked`, () => { const setup = async () => { findBtn().vm.$emit('click'); await nextTick(); }; it('The button has a loading state', async () => { await setup(); expect(findBtn().props('loading')).toBe(true); }); it('The stale tooltip is removed', async () => { await setup(); expect(getTooltip()).toBe(''); }); }); describe(`After clicking on the ${icon} button`, () => { beforeEach(async () => { await clickAndWait(); }); it(`The mutation to that sets active to ${newActiveValue} is called`, async () => { expect(runnerToggleActiveHandler).toHaveBeenCalledTimes(1); expect(runnerToggleActiveHandler).toHaveBeenCalledWith({ input: { id: mockRunner.id, active: newActiveValue, }, }); }); it('The button does not have a loading state', () => { expect(findBtn().props('loading')).toBe(false); }); it('The button emits toggledPaused', () => { expect(wrapper.emitted('toggledPaused')).toHaveLength(1); }); }); describe('When update fails', () => { describe('On a network error', () => { const mockErrorMsg = 'Update error!'; beforeEach(async () => { runnerToggleActiveHandler.mockRejectedValueOnce(new Error(mockErrorMsg)); await clickAndWait(); }); it('error is reported to sentry', () => { expect(captureException).toHaveBeenCalledWith({ error: new Error(mockErrorMsg), component: 'RunnerPauseButton', }); }); it('error is shown to the user', () => { expect(createAlert).toHaveBeenCalledTimes(1); }); }); describe('On a validation error', () => { const mockErrorMsg = 'Runner not found!'; const mockErrorMsg2 = 'User not allowed!'; beforeEach(async () => { runnerToggleActiveHandler.mockResolvedValueOnce({ data: { runnerUpdate: { runner: { id: mockRunner.id, active: isActive, }, errors: [mockErrorMsg, mockErrorMsg2], }, }, }); await clickAndWait(); }); it('error is reported to sentry', () => { expect(captureException).toHaveBeenCalledWith({ error: new Error(`${mockErrorMsg} ${mockErrorMsg2}`), component: 'RunnerPauseButton', }); }); it('error is shown to the user', () => { expect(createAlert).toHaveBeenCalledTimes(1); }); }); }); }); }); describe('When displaying a compact button for an active runner', () => { beforeEach(() => { createComponent({ props: { runner: { active: true, }, compact: true, }, mountFn: mountExtended, }); }); it('Displays no text', () => { expect(findBtn().text()).toBe(''); // Note: Use