Commit 983eadcf authored by Savas Vedova's avatar Savas Vedova

Merge branch '348711-error-handling-for-configuration-security-training-mutation' into 'master'

Add error handling to security training configuration (GraphQL query and mutation)

See merge request gitlab-org/gitlab!77166
parents 748862ff b405e234
<script> <script>
import { GlCard, GlToggle, GlLink, GlSkeletonLoader } from '@gitlab/ui'; import { GlAlert, GlCard, GlToggle, GlLink, GlSkeletonLoader } from '@gitlab/ui';
import { __ } from '~/locale';
import securityTrainingProvidersQuery from '../graphql/security_training_providers.query.graphql'; import securityTrainingProvidersQuery from '../graphql/security_training_providers.query.graphql';
import configureSecurityTrainingProvidersMutation from '../graphql/configure_security_training_providers.mutation.graphql'; import configureSecurityTrainingProvidersMutation from '../graphql/configure_security_training_providers.mutation.graphql';
const i18n = {
providerQueryErrorMessage: __(
'Could not fetch training providers. Please refresh the page, or try again later.',
),
configMutationErrorMessage: __(
'Could not save configuration. Please refresh the page, or try again later.',
),
};
export default { export default {
components: { components: {
GlAlert,
GlCard, GlCard,
GlToggle, GlToggle,
GlLink, GlLink,
...@@ -14,10 +25,14 @@ export default { ...@@ -14,10 +25,14 @@ export default {
apollo: { apollo: {
securityTrainingProviders: { securityTrainingProviders: {
query: securityTrainingProvidersQuery, query: securityTrainingProvidersQuery,
error() {
this.errorMessage = this.$options.i18n.providerQueryErrorMessage;
},
}, },
}, },
data() { data() {
return { return {
errorMessage: '',
toggleLoading: false, toggleLoading: false,
securityTrainingProviders: [], securityTrainingProviders: [],
}; };
...@@ -34,17 +49,21 @@ export default { ...@@ -34,17 +49,21 @@ export default {
...(provider.id === selectedProviderId && { isEnabled: !provider.isEnabled }), ...(provider.id === selectedProviderId && { isEnabled: !provider.isEnabled }),
})); }));
this.storeEnabledProviders(toggledProviders);
},
storeEnabledProviders(toggledProviders) {
const enabledProviderIds = toggledProviders const enabledProviderIds = toggledProviders
.filter(({ isEnabled }) => isEnabled) .filter(({ isEnabled }) => isEnabled)
.map(({ id }) => id); .map(({ id }) => id);
this.storeEnabledProviders(toggledProviders, enabledProviderIds);
},
async storeEnabledProviders(toggledProviders, enabledProviderIds) {
this.toggleLoading = true; this.toggleLoading = true;
return this.$apollo try {
.mutate({ const {
data: {
configureSecurityTrainingProviders: { errors = [] },
},
} = await this.$apollo.mutate({
mutation: configureSecurityTrainingProvidersMutation, mutation: configureSecurityTrainingProvidersMutation,
variables: { variables: {
input: { input: {
...@@ -52,16 +71,28 @@ export default { ...@@ -52,16 +71,28 @@ export default {
fullPath: this.projectPath, fullPath: this.projectPath,
}, },
}, },
})
.then(() => {
this.toggleLoading = false;
}); });
if (errors.length > 0) {
// throwing an error here means we can handle scenarios within the `catch` block below
throw new Error();
}
} catch {
this.errorMessage = this.$options.i18n.configMutationErrorMessage;
} finally {
this.toggleLoading = false;
}
}, },
}, },
i18n,
}; };
</script> </script>
<template> <template>
<div>
<gl-alert v-if="errorMessage" variant="danger" :dismissible="false" class="gl-mb-6">
{{ errorMessage }}
</gl-alert>
<div <div
v-if="isLoading" v-if="isLoading"
class="gl-bg-white gl-py-6 gl-rounded-base gl-border-1 gl-border-solid gl-border-gray-100" class="gl-bg-white gl-py-6 gl-rounded-base gl-border-1 gl-border-solid gl-border-gray-100"
...@@ -98,4 +129,5 @@ export default { ...@@ -98,4 +129,5 @@ export default {
</gl-card> </gl-card>
</li> </li>
</ul> </ul>
</div>
</template> </template>
mutation configureSecurityTrainingProviders($input: configureSecurityTrainingProvidersInput!) { mutation configureSecurityTrainingProviders($input: configureSecurityTrainingProvidersInput!) {
configureSecurityTrainingProviders(input: $input) @client { configureSecurityTrainingProviders(input: $input) @client {
errors
securityTrainingProviders { securityTrainingProviders {
id id
isEnabled isEnabled
......
...@@ -9853,6 +9853,9 @@ msgstr "" ...@@ -9853,6 +9853,9 @@ msgstr ""
msgid "Could not fetch policy because existing policy YAML is invalid" msgid "Could not fetch policy because existing policy YAML is invalid"
msgstr "" msgstr ""
msgid "Could not fetch training providers. Please refresh the page, or try again later."
msgstr ""
msgid "Could not find design." msgid "Could not find design."
msgstr "" msgstr ""
...@@ -9889,6 +9892,9 @@ msgstr "" ...@@ -9889,6 +9892,9 @@ msgstr ""
msgid "Could not revoke project access token %{project_access_token_name}." msgid "Could not revoke project access token %{project_access_token_name}."
msgstr "" msgstr ""
msgid "Could not save configuration. Please refresh the page, or try again later."
msgstr ""
msgid "Could not save group ID" msgid "Could not save group ID"
msgstr "" msgstr ""
......
import { GlLink, GlToggle, GlCard, GlSkeletonLoader } from '@gitlab/ui'; import { GlAlert, GlLink, GlToggle, GlCard, GlSkeletonLoader } from '@gitlab/ui';
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import Vue from 'vue'; import Vue from 'vue';
import VueApollo from 'vue-apollo'; import VueApollo from 'vue-apollo';
...@@ -8,7 +8,7 @@ import configureSecurityTrainingProvidersMutation from '~/security_configuration ...@@ -8,7 +8,7 @@ import configureSecurityTrainingProvidersMutation from '~/security_configuration
import waitForPromises from 'helpers/wait_for_promises'; import waitForPromises from 'helpers/wait_for_promises';
import { import {
securityTrainingProviders, securityTrainingProviders,
mockResolvers, createMockResolvers,
testProjectPath, testProjectPath,
textProviderIds, textProviderIds,
} from '../mock_data'; } from '../mock_data';
...@@ -17,37 +17,42 @@ Vue.use(VueApollo); ...@@ -17,37 +17,42 @@ Vue.use(VueApollo);
describe('TrainingProviderList component', () => { describe('TrainingProviderList component', () => {
let wrapper; let wrapper;
let mockApollo; let apolloProvider;
let mockSecurityTrainingProvidersData;
const createComponent = () => { const createApolloProvider = ({ resolvers } = {}) => {
mockApollo = createMockApollo([], mockResolvers); apolloProvider = createMockApollo([], createMockResolvers({ resolvers }));
};
const createComponent = () => {
wrapper = shallowMount(TrainingProviderList, { wrapper = shallowMount(TrainingProviderList, {
provide: { provide: {
projectPath: testProjectPath, projectPath: testProjectPath,
}, },
apolloProvider: mockApollo, apolloProvider,
}); });
}; };
const waitForQueryToBeLoaded = () => waitForPromises(); const waitForQueryToBeLoaded = () => waitForPromises();
const waitForMutationToBeLoaded = waitForQueryToBeLoaded;
const findCards = () => wrapper.findAllComponents(GlCard); const findCards = () => wrapper.findAllComponents(GlCard);
const findLinks = () => wrapper.findAllComponents(GlLink); const findLinks = () => wrapper.findAllComponents(GlLink);
const findToggles = () => wrapper.findAllComponents(GlToggle); const findToggles = () => wrapper.findAllComponents(GlToggle);
const findFirstToggle = () => findToggles().at(0);
const findLoader = () => wrapper.findComponent(GlSkeletonLoader); const findLoader = () => wrapper.findComponent(GlSkeletonLoader);
const findErrorAlert = () => wrapper.findComponent(GlAlert);
beforeEach(() => { const toggleFirstProvider = () => findFirstToggle().vm.$emit('change');
mockSecurityTrainingProvidersData = jest.fn();
mockSecurityTrainingProvidersData.mockResolvedValue(securityTrainingProviders);
createComponent();
});
afterEach(() => { afterEach(() => {
wrapper.destroy(); wrapper.destroy();
mockApollo = null; apolloProvider = null;
});
describe('with a successful response', () => {
beforeEach(() => {
createApolloProvider();
createComponent();
}); });
describe('when loading', () => { describe('when loading', () => {
...@@ -95,35 +100,94 @@ describe('TrainingProviderList component', () => { ...@@ -95,35 +100,94 @@ describe('TrainingProviderList component', () => {
}); });
}); });
describe('success mutation', () => { describe('storing training provider settings', () => {
const firstToggle = () => findToggles().at(0);
beforeEach(async () => { beforeEach(async () => {
jest.spyOn(mockApollo.defaultClient, 'mutate'); jest.spyOn(apolloProvider.defaultClient, 'mutate');
await waitForQueryToBeLoaded(); await waitForMutationToBeLoaded();
firstToggle().vm.$emit('change'); toggleFirstProvider();
});
it.each`
loading | wait | desc
${true} | ${false} | ${'enables loading of GlToggle when mutation is called'}
${false} | ${true} | ${'disables loading of GlToggle when mutation is complete'}
`('$desc', async ({ loading, wait }) => {
if (wait) {
await waitForMutationToBeLoaded();
}
expect(findFirstToggle().props('isLoading')).toBe(loading);
}); });
it('calls mutation when toggle is changed', () => { it('calls mutation when toggle is changed', () => {
expect(mockApollo.defaultClient.mutate).toHaveBeenCalledWith( expect(apolloProvider.defaultClient.mutate).toHaveBeenCalledWith(
expect.objectContaining({ expect.objectContaining({
mutation: configureSecurityTrainingProvidersMutation, mutation: configureSecurityTrainingProvidersMutation,
variables: { input: { enabledProviders: textProviderIds, fullPath: testProjectPath } }, variables: { input: { enabledProviders: textProviderIds, fullPath: testProjectPath } },
}), }),
); );
}); });
});
});
it.each` describe('with errors', () => {
loading | wait | desc const expectErrorAlertToExist = () => {
${true} | ${false} | ${'enables loading of GlToggle when mutation is called'} expect(findErrorAlert().props()).toMatchObject({
${false} | ${true} | ${'disables loading of GlToggle when mutation is complete'} dismissible: false,
`('$desc', async ({ loading, wait }) => { variant: 'danger',
if (wait) { });
await waitForPromises(); };
}
expect(firstToggle().props('isLoading')).toBe(loading); describe('when fetching training providers', () => {
beforeEach(async () => {
createApolloProvider({
resolvers: {
Query: {
securityTrainingProviders: jest.fn().mockReturnValue(new Error()),
},
},
});
createComponent();
await waitForQueryToBeLoaded();
});
it('shows an non-dismissible error alert', () => {
expectErrorAlertToExist();
});
it('shows an error description', () => {
expect(findErrorAlert().text()).toBe(TrainingProviderList.i18n.providerQueryErrorMessage);
});
});
describe('when storing training provider configurations', () => {
beforeEach(async () => {
createApolloProvider({
resolvers: {
Mutation: {
configureSecurityTrainingProviders: () => ({
errors: ['something went wrong!'],
securityTrainingProviders: [],
}),
},
},
});
createComponent();
await waitForQueryToBeLoaded();
toggleFirstProvider();
await waitForMutationToBeLoaded();
});
it('shows an non-dismissible error alert', () => {
expectErrorAlertToExist();
});
it('shows an error description', () => {
expect(findErrorAlert().text()).toBe(TrainingProviderList.i18n.configMutationErrorMessage);
});
}); });
}); });
}); });
...@@ -25,10 +25,15 @@ export const securityTrainingProvidersResponse = { ...@@ -25,10 +25,15 @@ export const securityTrainingProvidersResponse = {
}, },
}; };
export const mockResolvers = { const defaultMockResolvers = {
Query: { Query: {
securityTrainingProviders() { securityTrainingProviders() {
return securityTrainingProviders; return securityTrainingProviders;
}, },
}, },
}; };
export const createMockResolvers = ({ resolvers: customMockResolvers = {} } = {}) => ({
...defaultMockResolvers,
...customMockResolvers,
});
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