Commit 4529939f authored by Mark Florian's avatar Mark Florian

Merge branch '214990-license-check-mr-ui' into 'master'

Add MR widget support for license check

See merge request gitlab-org/gitlab!34034
parents 7340161a e747eb95
<script>
import { GlButton } from '@gitlab/ui';
import StatusIcon from '~/vue_merge_request_widget/components/mr_widget_status_icon.vue';
export default {
name: 'MRWidgetPolicyViolation',
components: {
GlButton,
StatusIcon,
},
};
</script>
<template>
<div class="mr-widget-body media">
<div class="space-children gl-display-flex">
<status-icon status="warning" />
<gl-button category="primary" variant="success" disabled size="small">
{{ s__('mrWidget|Merge') }}
</gl-button>
</div>
<div class="media-body">
<strong class="gl-font-weight-bold gl-text-gray-800 gl-pl-2">
{{ s__('mrWidget|You can only merge once the denied license is removed') }}
</strong>
</div>
</div>
</template>
...@@ -12,6 +12,7 @@ import { n__, s__, __, sprintf } from '~/locale'; ...@@ -12,6 +12,7 @@ import { n__, s__, __, sprintf } from '~/locale';
import CEWidgetOptions from '~/vue_merge_request_widget/mr_widget_options.vue'; import CEWidgetOptions from '~/vue_merge_request_widget/mr_widget_options.vue';
import MrWidgetApprovals from './components/approvals/approvals.vue'; import MrWidgetApprovals from './components/approvals/approvals.vue';
import MrWidgetGeoSecondaryNode from './components/states/mr_widget_secondary_geo_node.vue'; import MrWidgetGeoSecondaryNode from './components/states/mr_widget_secondary_geo_node.vue';
import MrWidgetPolicyViolation from './components/states/mr_widget_policy_violation.vue';
import MergeTrainHelperText from './components/merge_train_helper_text.vue'; import MergeTrainHelperText from './components/merge_train_helper_text.vue';
import { MTWPS_MERGE_STRATEGY } from '~/vue_merge_request_widget/constants'; import { MTWPS_MERGE_STRATEGY } from '~/vue_merge_request_widget/constants';
...@@ -21,6 +22,7 @@ export default { ...@@ -21,6 +22,7 @@ export default {
MrWidgetLicenses, MrWidgetLicenses,
MrWidgetApprovals, MrWidgetApprovals,
MrWidgetGeoSecondaryNode, MrWidgetGeoSecondaryNode,
MrWidgetPolicyViolation,
BlockingMergeRequestsReport, BlockingMergeRequestsReport,
GroupedSecurityReportsApp, GroupedSecurityReportsApp,
GroupedMetricsReportsApp, GroupedMetricsReportsApp,
...@@ -442,6 +444,7 @@ export default { ...@@ -442,6 +444,7 @@ export default {
<mr-widget-licenses <mr-widget-licenses
v-if="shouldRenderLicenseReport" v-if="shouldRenderLicenseReport"
:api-url="mr.licenseScanning.managed_licenses_path" :api-url="mr.licenseScanning.managed_licenses_path"
:approvals-api-path="mr.apiApprovalsPath"
:licenses-api-path="licensesApiPath" :licenses-api-path="licensesApiPath"
:pipeline-path="mr.pipeline.path" :pipeline-path="mr.pipeline.path"
:can-manage-licenses="mr.licenseScanning.can_manage_licenses" :can-manage-licenses="mr.licenseScanning.can_manage_licenses"
......
import CEGetStateKey from '~/vue_merge_request_widget/stores/get_state_key'; import CEGetStateKey from '~/vue_merge_request_widget/stores/get_state_key';
import { stateKey } from './state_maps';
export default function(data) { export default function(data) {
if (this.isGeoSecondaryNode) { if (this.isGeoSecondaryNode) {
return 'geoSecondaryNode'; return 'geoSecondaryNode';
} }
if (data.policy_violation) {
return stateKey.policyViolation;
}
return CEGetStateKey.call(this, data); return CEGetStateKey.call(this, data);
} }
import stateMaps from '~/vue_merge_request_widget/stores/state_maps'; import stateMaps from '~/vue_merge_request_widget/stores/state_maps';
stateMaps.stateToComponentMap.geoSecondaryNode = 'mr-widget-geo-secondary-node'; stateMaps.stateToComponentMap.geoSecondaryNode = 'mr-widget-geo-secondary-node';
stateMaps.stateToComponentMap.policyViolation = 'mr-widget-policy-violation';
export const stateKey = {
policyViolation: 'policyViolation',
};
export default { export default {
stateToComponentMap: stateMaps.stateToComponentMap, stateToComponentMap: stateMaps.stateToComponentMap,
......
...@@ -46,6 +46,11 @@ export default { ...@@ -46,6 +46,11 @@ export default {
required: false, required: false,
default: '', default: '',
}, },
approvalsApiPath: {
type: String,
required: false,
default: '',
},
canManageLicenses: { canManageLicenses: {
type: Boolean, type: Boolean,
required: true, required: true,
...@@ -94,18 +99,24 @@ export default { ...@@ -94,18 +99,24 @@ export default {
}, },
}, },
mounted() { mounted() {
const { apiUrl, canManageLicenses, licensesApiPath } = this; const { apiUrl, canManageLicenses, licensesApiPath, approvalsApiPath } = this;
this.setAPISettings({ this.setAPISettings({
apiUrlManageLicenses: apiUrl, apiUrlManageLicenses: apiUrl,
canManageLicenses, canManageLicenses,
licensesApiPath, licensesApiPath,
approvalsApiPath,
}); });
this.fetchParsedLicenseReport(); this.fetchParsedLicenseReport();
this.fetchLicenseCheckApprovalRule();
}, },
methods: { methods: {
...mapActions(LICENSE_MANAGEMENT, ['setAPISettings', 'fetchParsedLicenseReport']), ...mapActions(LICENSE_MANAGEMENT, [
'setAPISettings',
'fetchParsedLicenseReport',
'fetchLicenseCheckApprovalRule',
]),
}, },
}; };
</script> </script>
......
...@@ -2,6 +2,7 @@ import axios from '~/lib/utils/axios_utils'; ...@@ -2,6 +2,7 @@ import axios from '~/lib/utils/axios_utils';
import pollUntilComplete from '~/lib/utils/poll_until_complete'; import pollUntilComplete from '~/lib/utils/poll_until_complete';
import * as types from './mutation_types'; import * as types from './mutation_types';
import { LICENSE_APPROVAL_STATUS } from '../constants'; import { LICENSE_APPROVAL_STATUS } from '../constants';
import { LICENSE_CHECK_NAME } from 'ee/approvals/constants';
import { convertToOldReportFormat } from './utils'; import { convertToOldReportFormat } from './utils';
export const setAPISettings = ({ commit }, data) => { export const setAPISettings = ({ commit }, data) => {
...@@ -111,6 +112,47 @@ export const receiveSetLicenseApprovalError = ({ commit }, error) => { ...@@ -111,6 +112,47 @@ export const receiveSetLicenseApprovalError = ({ commit }, error) => {
commit(types.RECEIVE_SET_LICENSE_APPROVAL_ERROR, error); commit(types.RECEIVE_SET_LICENSE_APPROVAL_ERROR, error);
}; };
export const fetchLicenseCheckApprovalRule = ({ dispatch, state }) => {
dispatch('requestLicenseCheckApprovalRule');
/*
If we call this action from the "License" tab in the pipeline view,
then we don't fetch the approvals since we aren't in the Merge request context.
Pipelines cannot have approval rules.
*/
if (!state.approvalsApiPath) {
return dispatch(
'receiveLicenseCheckApprovalRuleError',
new Error('approvalsApiPath not provided'),
);
}
return axios
.get(state.approvalsApiPath)
.then(({ data }) => {
const hasLicenseCheckApprovalRule = data.approval_rules_left.some(rule => {
return rule.name === LICENSE_CHECK_NAME;
});
dispatch('receiveLicenseCheckApprovalRuleSuccess', { hasLicenseCheckApprovalRule });
})
.catch(error => {
dispatch('receiveLicenseCheckApprovalRuleError', error);
});
};
export const requestLicenseCheckApprovalRule = ({ commit }) => {
commit(types.REQUEST_LICENSE_CHECK_APPROVAL_RULE);
};
export const receiveLicenseCheckApprovalRuleSuccess = ({ commit }, rule) => {
commit(types.RECEIVE_LICENSE_CHECK_APPROVAL_RULE_SUCCESS, rule);
};
export const receiveLicenseCheckApprovalRuleError = ({ commit }, error) => {
commit(types.RECEIVE_LICENSE_CHECK_APPROVAL_RULE_ERROR, error);
};
export const setIsAdmin = ({ commit }, payload) => { export const setIsAdmin = ({ commit }, payload) => {
commit(types.SET_IS_ADMIN, payload); commit(types.SET_IS_ADMIN, payload);
}; };
......
...@@ -2,7 +2,10 @@ import { n__, s__, sprintf } from '~/locale'; ...@@ -2,7 +2,10 @@ import { n__, s__, sprintf } from '~/locale';
import { addLicensesMatchingReportGroupStatus, reportGroupHasAtLeastOneLicense } from './utils'; import { addLicensesMatchingReportGroupStatus, reportGroupHasAtLeastOneLicense } from './utils';
import { LICENSE_APPROVAL_STATUS, REPORT_GROUPS } from '../constants'; import { LICENSE_APPROVAL_STATUS, REPORT_GROUPS } from '../constants';
export const isLoading = state => state.isLoadingManagedLicenses || state.isLoadingLicenseReport; export const isLoading = state =>
state.isLoadingManagedLicenses ||
state.isLoadingLicenseReport ||
state.isLoadingLicenseCheckApprovalRule;
export const isLicenseBeingUpdated = state => (id = null) => state.pendingLicenses.includes(id); export const isLicenseBeingUpdated = state => (id = null) => state.pendingLicenses.includes(id);
...@@ -17,10 +20,19 @@ export const licenseReportGroups = state => ...@@ -17,10 +20,19 @@ export const licenseReportGroups = state =>
reportGroupHasAtLeastOneLicense, reportGroupHasAtLeastOneLicense,
); );
export const licenseSummaryText = (state, getters) => { export const hasReportItems = (_, getters) => {
const hasReportItems = getters.licenseReport && getters.licenseReport.length; return Boolean(getters.licenseReportLength);
const baseReportHasLicenses = state.existingLicenses.length; };
export const baseReportHasLicenses = state => {
return Boolean(state.existingLicenses.length);
};
export const licenseReportLength = (_, getters) => {
return getters.licenseReport.length;
};
export const licenseSummaryText = (state, getters) => {
if (getters.isLoading) { if (getters.isLoading) {
return sprintf(s__('ciReport|Loading %{reportName} report'), { return sprintf(s__('ciReport|Loading %{reportName} report'), {
reportName: s__('License Compliance'), reportName: s__('License Compliance'),
...@@ -33,20 +45,33 @@ export const licenseSummaryText = (state, getters) => { ...@@ -33,20 +45,33 @@ export const licenseSummaryText = (state, getters) => {
}); });
} }
if (hasReportItems) { if (getters.hasReportItems) {
const licenseReportLength = getters.licenseReport.length; return state.hasLicenseCheckApprovalRule
? getters.summaryTextWithLicenseCheck
: getters.summaryTextWithoutLicenseCheck;
}
if (!baseReportHasLicenses) { if (!getters.baseReportHasLicenses) {
return s__(
'LicenseCompliance|License Compliance detected no licenses for the source branch only',
);
}
return s__('LicenseCompliance|License Compliance detected no new licenses');
};
export const summaryTextWithLicenseCheck = (_, getters) => {
if (!getters.baseReportHasLicenses) {
return getters.reportContainsBlacklistedLicense return getters.reportContainsBlacklistedLicense
? n__( ? n__(
'LicenseCompliance|License Compliance detected %d license and policy violation for the source branch only; approval required', 'LicenseCompliance|License Compliance detected %d license and policy violation for the source branch only; approval required',
'LicenseCompliance|License Compliance detected %d licenses and policy violations for the source branch only; approval required', 'LicenseCompliance|License Compliance detected %d licenses and policy violations for the source branch only; approval required',
licenseReportLength, getters.licenseReportLength,
) )
: n__( : n__(
'LicenseCompliance|License Compliance detected %d license for the source branch only', 'LicenseCompliance|License Compliance detected %d license for the source branch only',
'LicenseCompliance|License Compliance detected %d licenses for the source branch only', 'LicenseCompliance|License Compliance detected %d licenses for the source branch only',
licenseReportLength, getters.licenseReportLength,
); );
} }
...@@ -54,22 +79,41 @@ export const licenseSummaryText = (state, getters) => { ...@@ -54,22 +79,41 @@ export const licenseSummaryText = (state, getters) => {
? n__( ? n__(
'LicenseCompliance|License Compliance detected %d new license and policy violation; approval required', 'LicenseCompliance|License Compliance detected %d new license and policy violation; approval required',
'LicenseCompliance|License Compliance detected %d new licenses and policy violations; approval required', 'LicenseCompliance|License Compliance detected %d new licenses and policy violations; approval required',
licenseReportLength, getters.licenseReportLength,
) )
: n__( : n__(
'LicenseCompliance|License Compliance detected %d new license', 'LicenseCompliance|License Compliance detected %d new license',
'LicenseCompliance|License Compliance detected %d new licenses', 'LicenseCompliance|License Compliance detected %d new licenses',
licenseReportLength, getters.licenseReportLength,
); );
} };
if (!baseReportHasLicenses) { export const summaryTextWithoutLicenseCheck = (_, getters) => {
return s__( if (!getters.baseReportHasLicenses) {
'LicenseCompliance|License Compliance detected no licenses for the source branch only', return getters.reportContainsBlacklistedLicense
? n__(
'LicenseCompliance|License Compliance detected %d license and policy violation for the source branch only',
'LicenseCompliance|License Compliance detected %d licenses and policy violations for the source branch only',
getters.licenseReportLength,
)
: n__(
'LicenseCompliance|License Compliance detected %d license for the source branch only',
'LicenseCompliance|License Compliance detected %d licenses for the source branch only',
getters.licenseReportLength,
); );
} }
return s__('LicenseCompliance|License Compliance detected no new licenses'); return getters.reportContainsBlacklistedLicense
? n__(
'LicenseCompliance|License Compliance detected %d new license and policy violation',
'LicenseCompliance|License Compliance detected %d new licenses and policy violations',
getters.licenseReportLength,
)
: n__(
'LicenseCompliance|License Compliance detected %d new license',
'LicenseCompliance|License Compliance detected %d new licenses',
getters.licenseReportLength,
);
}; };
export const reportContainsBlacklistedLicense = (_, getters) => export const reportContainsBlacklistedLicense = (_, getters) =>
......
...@@ -16,6 +16,11 @@ export const SET_LICENSE_IN_MODAL = 'SET_LICENSE_IN_MODAL'; ...@@ -16,6 +16,11 @@ export const SET_LICENSE_IN_MODAL = 'SET_LICENSE_IN_MODAL';
export const SET_IS_ADMIN = 'SET_IS_ADMIN'; export const SET_IS_ADMIN = 'SET_IS_ADMIN';
export const ADD_PENDING_LICENSE = 'ADD_PENDING_LICENSE'; export const ADD_PENDING_LICENSE = 'ADD_PENDING_LICENSE';
export const REMOVE_PENDING_LICENSE = 'REMOVE_PENDING_LICENSE'; export const REMOVE_PENDING_LICENSE = 'REMOVE_PENDING_LICENSE';
export const REQUEST_LICENSE_CHECK_APPROVAL_RULE = 'REQUEST_LICENSE_CHECK_APPROVAL_RULE';
export const RECEIVE_LICENSE_CHECK_APPROVAL_RULE_SUCCESS =
'RECEIVE_LICENSE_CHECK_APPROVAL_RULE_SUCCESS';
export const RECEIVE_LICENSE_CHECK_APPROVAL_RULE_ERROR =
'RECEIVE_LICENSE_CHECK_APPROVAL_RULE_ERROR';
// prevent babel-plugin-rewire from generating an invalid default during karma tests // prevent babel-plugin-rewire from generating an invalid default during karma tests
export default () => {}; export default () => {};
...@@ -96,6 +96,22 @@ export default { ...@@ -96,6 +96,22 @@ export default {
currentLicenseInModal: null, currentLicenseInModal: null,
}); });
}, },
[types.REQUEST_LICENSE_CHECK_APPROVAL_RULE](state) {
Object.assign(state, {
isLoadingLicenseCheckApprovalRule: true,
});
},
[types.RECEIVE_LICENSE_CHECK_APPROVAL_RULE_SUCCESS](state, { hasLicenseCheckApprovalRule }) {
Object.assign(state, {
isLoadingLicenseCheckApprovalRule: false,
hasLicenseCheckApprovalRule,
});
},
[types.RECEIVE_LICENSE_CHECK_APPROVAL_RULE_ERROR](state) {
Object.assign(state, {
isLoadingLicenseCheckApprovalRule: false,
});
},
[types.ADD_PENDING_LICENSE](state, id) { [types.ADD_PENDING_LICENSE](state, id) {
state.pendingLicenses.push(id); state.pendingLicenses.push(id);
}, },
......
export default () => ({ export default () => ({
apiUrlManageLicenses: null, apiUrlManageLicenses: null,
approvalsApiPath: null,
licensesApiPath: null, licensesApiPath: null,
canManageLicenses: false, canManageLicenses: false,
currentLicenseInModal: null, currentLicenseInModal: null,
...@@ -14,4 +15,6 @@ export default () => ({ ...@@ -14,4 +15,6 @@ export default () => ({
managedLicenses: [], managedLicenses: [],
newLicenses: [], newLicenses: [],
existingLicenses: [], existingLicenses: [],
hasLicenseCheckApprovalRule: false,
isLoadingLicenseCheckApprovalRule: false,
}); });
import { shallowMount } from '@vue/test-utils';
import { GlButton } from '@gitlab/ui';
import MrWidgetPolicyViolation from 'ee/vue_merge_request_widget/components/states/mr_widget_policy_violation.vue';
describe('EE MrWidgetPolicyViolation', () => {
let wrapper;
const findButton = () => wrapper.find(GlButton);
const createComponent = () => {
wrapper = shallowMount(MrWidgetPolicyViolation, {});
};
afterEach(() => {
wrapper.destroy();
wrapper = null;
});
beforeEach(() => {
createComponent();
});
it('shows the disabled merge button', () => {
expect(wrapper.text()).toContain('Merge');
expect(findButton().props().disabled).toBe(true);
});
it('shows the disabled reason', () => {
expect(wrapper.text()).toContain('You can only merge once the denied license is removed');
});
});
...@@ -48,6 +48,7 @@ describe('License Report MR Widget', () => { ...@@ -48,6 +48,7 @@ describe('License Report MR Widget', () => {
loadingText: 'LOADING', loadingText: 'LOADING',
errorText: 'ERROR', errorText: 'ERROR',
licensesApiPath: `${TEST_HOST}/parsed_license_report.json`, licensesApiPath: `${TEST_HOST}/parsed_license_report.json`,
approvalsApiPath: `${TEST_HOST}/path/to/approvals`,
canManageLicenses: true, canManageLicenses: true,
licenseManagementSettingsPath: `${TEST_HOST}/lm_settings`, licenseManagementSettingsPath: `${TEST_HOST}/lm_settings`,
fullReportPath: `${TEST_HOST}/path/to/the/full/report`, fullReportPath: `${TEST_HOST}/path/to/the/full/report`,
...@@ -59,6 +60,7 @@ describe('License Report MR Widget', () => { ...@@ -59,6 +60,7 @@ describe('License Report MR Widget', () => {
setAPISettings: () => {}, setAPISettings: () => {},
fetchManagedLicenses: () => {}, fetchManagedLicenses: () => {},
fetchParsedLicenseReport: () => {}, fetchParsedLicenseReport: () => {},
fetchLicenseCheckApprovalRule: () => {},
}; };
const mountComponent = ({ const mountComponent = ({
...@@ -339,8 +341,9 @@ describe('License Report MR Widget', () => { ...@@ -339,8 +341,9 @@ describe('License Report MR Widget', () => {
it('should init store after mount', () => { it('should init store after mount', () => {
const actions = { const actions = {
setAPISettings: jest.fn(() => {}), setAPISettings: jest.fn(),
fetchParsedLicenseReport: jest.fn(() => {}), fetchParsedLicenseReport: jest.fn(),
fetchLicenseCheckApprovalRule: jest.fn(),
}; };
mountComponent({ actions }); mountComponent({ actions });
...@@ -349,6 +352,7 @@ describe('License Report MR Widget', () => { ...@@ -349,6 +352,7 @@ describe('License Report MR Widget', () => {
{ {
apiUrlManageLicenses: apiUrl, apiUrlManageLicenses: apiUrl,
licensesApiPath: defaultProps.licensesApiPath, licensesApiPath: defaultProps.licensesApiPath,
approvalsApiPath: defaultProps.approvalsApiPath,
canManageLicenses: true, canManageLicenses: true,
}, },
undefined, undefined,
...@@ -359,6 +363,12 @@ describe('License Report MR Widget', () => { ...@@ -359,6 +363,12 @@ describe('License Report MR Widget', () => {
undefined, undefined,
undefined, undefined,
); );
expect(actions.fetchLicenseCheckApprovalRule).toHaveBeenCalledWith(
expect.any(Object),
undefined,
undefined,
);
}); });
describe('approval status', () => { describe('approval status', () => {
......
...@@ -3,6 +3,7 @@ import * as actions from 'ee/vue_shared/license_compliance/store/actions'; ...@@ -3,6 +3,7 @@ import * as actions from 'ee/vue_shared/license_compliance/store/actions';
import * as mutationTypes from 'ee/vue_shared/license_compliance/store/mutation_types'; import * as mutationTypes from 'ee/vue_shared/license_compliance/store/mutation_types';
import createState from 'ee/vue_shared/license_compliance/store/state'; import createState from 'ee/vue_shared/license_compliance/store/state';
import { LICENSE_APPROVAL_STATUS } from 'ee/vue_shared/license_compliance/constants'; import { LICENSE_APPROVAL_STATUS } from 'ee/vue_shared/license_compliance/constants';
import { LICENSE_CHECK_NAME } from 'ee/approvals/constants';
import { TEST_HOST } from 'spec/test_constants'; import { TEST_HOST } from 'spec/test_constants';
import testAction from 'helpers/vuex_action_helper'; import testAction from 'helpers/vuex_action_helper';
import { approvedLicense, blacklistedLicense } from '../mock_data'; import { approvedLicense, blacklistedLicense } from '../mock_data';
...@@ -10,6 +11,7 @@ import axios from '~/lib/utils/axios_utils'; ...@@ -10,6 +11,7 @@ import axios from '~/lib/utils/axios_utils';
describe('License store actions', () => { describe('License store actions', () => {
const apiUrlManageLicenses = `${TEST_HOST}/licenses/management`; const apiUrlManageLicenses = `${TEST_HOST}/licenses/management`;
const approvalsApiPath = `${TEST_HOST}/approvalsApiPath`;
const licensesApiPath = `${TEST_HOST}/licensesApiPath`; const licensesApiPath = `${TEST_HOST}/licensesApiPath`;
let axiosMock; let axiosMock;
...@@ -26,6 +28,7 @@ describe('License store actions', () => { ...@@ -26,6 +28,7 @@ describe('License store actions', () => {
state = { state = {
...createState(), ...createState(),
apiUrlManageLicenses, apiUrlManageLicenses,
approvalsApiPath,
currentLicenseInModal: approvedLicense, currentLicenseInModal: approvedLicense,
}; };
licenseId = approvedLicense.id; licenseId = approvedLicense.id;
...@@ -480,6 +483,139 @@ describe('License store actions', () => { ...@@ -480,6 +483,139 @@ describe('License store actions', () => {
}); });
}); });
describe('fetchLicenseCheckApprovalRule ', () => {
it('dispatches request/receive with detected approval rule', done => {
const APPROVAL_RULE_RESPONSE = {
approval_rules_left: [{ name: LICENSE_CHECK_NAME }],
};
axiosMock.onGet(approvalsApiPath).replyOnce(200, APPROVAL_RULE_RESPONSE);
testAction(
actions.fetchLicenseCheckApprovalRule,
null,
state,
[],
[
{ type: 'requestLicenseCheckApprovalRule' },
{
type: 'receiveLicenseCheckApprovalRuleSuccess',
payload: { hasLicenseCheckApprovalRule: true },
},
],
done,
);
});
it('dispatches request/receive without detected approval rule', done => {
const APPROVAL_RULE_RESPONSE = {
approval_rules_left: [{ name: 'Another Approval Rule' }],
};
axiosMock.onGet(approvalsApiPath).replyOnce(200, APPROVAL_RULE_RESPONSE);
testAction(
actions.fetchLicenseCheckApprovalRule,
null,
state,
[],
[
{ type: 'requestLicenseCheckApprovalRule' },
{
type: 'receiveLicenseCheckApprovalRuleSuccess',
payload: { hasLicenseCheckApprovalRule: false },
},
],
done,
);
});
it('dispatches request/receive error when no approvalsAPiPath is provided', done => {
const error = new Error('approvalsApiPath not provided');
axiosMock.onGet(approvalsApiPath).replyOnce(500);
testAction(
actions.fetchLicenseCheckApprovalRule,
null,
{ ...state, approvalsApiPath: '' },
[],
[
{ type: 'requestLicenseCheckApprovalRule' },
{ type: 'receiveLicenseCheckApprovalRuleError', payload: error },
],
done,
);
});
it('dispatches request/receive on error', done => {
const error = new Error('Request failed with status code 500');
axiosMock.onGet(approvalsApiPath).replyOnce(500);
testAction(
actions.fetchLicenseCheckApprovalRule,
null,
state,
[],
[
{ type: 'requestLicenseCheckApprovalRule' },
{ type: 'receiveLicenseCheckApprovalRuleError', payload: error },
],
done,
);
});
});
describe('requestLicenseCheckApprovalRule', () => {
it('commits REQUEST_LICENSE_CHECK_APPROVAL_RULE', done => {
testAction(
actions.requestLicenseCheckApprovalRule,
null,
state,
[{ type: mutationTypes.REQUEST_LICENSE_CHECK_APPROVAL_RULE }],
[],
)
.then(done)
.catch(done.fail);
});
});
describe('receiveLicenseCheckApprovalRuleSuccess', () => {
it('commits REQUEST_LICENSE_CHECK_APPROVAL_RULE', done => {
const hasLicenseCheckApprovalRule = true;
testAction(
actions.receiveLicenseCheckApprovalRuleSuccess,
{ hasLicenseCheckApprovalRule },
state,
[
{
type: mutationTypes.RECEIVE_LICENSE_CHECK_APPROVAL_RULE_SUCCESS,
payload: { hasLicenseCheckApprovalRule },
},
],
[],
)
.then(done)
.catch(done.fail);
});
});
describe('receiveLicenseCheckApprovalRuleError', () => {
it('commits RECEIVE_LICENSE_CHECK_APPROVAL_RULE_ERROR', done => {
const error = new Error('Error');
testAction(
actions.receiveLicenseCheckApprovalRuleError,
error,
state,
[{ type: mutationTypes.RECEIVE_LICENSE_CHECK_APPROVAL_RULE_ERROR, payload: error }],
[],
)
.then(done)
.catch(done.fail);
});
});
describe('requestParsedLicenseReport', () => { describe('requestParsedLicenseReport', () => {
it(`should commit ${mutationTypes.REQUEST_PARSED_LICENSE_REPORT}`, done => { it(`should commit ${mutationTypes.REQUEST_PARSED_LICENSE_REPORT}`, done => {
testAction( testAction(
......
...@@ -145,7 +145,6 @@ describe('getters', () => { ...@@ -145,7 +145,6 @@ describe('getters', () => {
}); });
describe('licenseSummaryText', () => { describe('licenseSummaryText', () => {
describe('when licenses exist on both the HEAD and the BASE', () => {
beforeEach(() => { beforeEach(() => {
state = { state = {
...createState(), ...createState(),
...@@ -173,52 +172,322 @@ describe('getters', () => { ...@@ -173,52 +172,322 @@ describe('getters', () => {
); );
}); });
it.each` it('should call summaryTextWithLicenseCheck if new license are detected and license-check approval group is enabled', () => {
givenLicenseReport | givenReportContainsBlacklistedLicense | expectedSummaryText
${[]} | ${false} | ${'License Compliance detected no new licenses'}
${[licenseReportMock[0]]} | ${false} | ${'License Compliance detected 1 new license'}
${[licenseReportMock[0], licenseReportMock[0]]} | ${false} | ${'License Compliance detected 2 new licenses'}
${[licenseReportMock[0]]} | ${true} | ${'License Compliance detected 1 new license and policy violation; approval required'}
${[licenseReportMock[0], licenseReportMock[0]]} | ${true} | ${'License Compliance detected 2 new licenses and policy violations; approval required'}
`(
`should show "$expectedSummaryText" if the report contains $givenLicenseReport.length license(s) and contains blacklisted licenses is: $givenReportContainsBlacklistedLicense`,
({ givenLicenseReport, givenReportContainsBlacklistedLicense, expectedSummaryText }) => {
const mockGetters = { const mockGetters = {
licenseReport: givenLicenseReport, hasReportItems: true,
reportContainsBlacklistedLicense: givenReportContainsBlacklistedLicense, summaryTextWithLicenseCheck: 'summary text with license check',
}; };
expect(
getters.licenseSummaryText({ state, hasLicenseCheckApprovalRule: true }, mockGetters),
).toBe('summary text with license check');
});
expect(getters.licenseSummaryText(state, mockGetters)).toBe(expectedSummaryText); it('should call summaryTextWithOutLicenseCheck if new license are detected and license-check approval group is disabled', () => {
}, const mockGetters = {
hasReportItems: true,
summaryTextWithoutLicenseCheck: 'summary text without license check',
};
expect(
getters.licenseSummaryText({ state, hasLicenseCheckApprovalRule: false }, mockGetters),
).toBe('summary text without license check');
});
it('should show "License Compliance detected no licenses for the source branch only" if there are no existing licenses', () => {
const mockGetters = {
baseReportHasLicenses: false,
};
expect(getters.licenseSummaryText(state, mockGetters)).toBe(
'License Compliance detected no licenses for the source branch only',
); );
}); });
it('should show "License Compliance detected no new licenses" if there are no new licenses, but existing licenses', () => {
const mockGetters = {
baseReportHasLicenses: true,
};
expect(getters.licenseSummaryText(state, mockGetters)).toBe(
'License Compliance detected no new licenses',
);
});
});
describe('summaryTextWithLicenseCheck', () => {
describe('when licenses exist on both the HEAD and the BASE', () => {
beforeEach(() => {
state = {
...createState(),
};
});
describe('when blacklisted licenses exist on the HEAD', () => {
describe('when a single license is detected', () => {
it('should return "License Compliance detected 1 new license and policy violation; approval required"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: true,
baseReportHasLicenses: true,
licenseReportLength: 1,
};
expect(getters.summaryTextWithLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 1 new license and policy violation; approval required',
);
});
});
describe('when multiple licenses are detected', () => {
it('should return License Compliance detected 2 new licenses and policy violations; approval required', () => {
const mockGetters = {
reportContainsBlacklistedLicense: true,
baseReportHasLicenses: true,
licenseReportLength: 2,
};
expect(getters.summaryTextWithLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 2 new licenses and policy violations; approval required',
);
});
});
});
describe('when blacklisted licenses are not detected on the HEAD', () => {
describe('when a single license is detected', () => {
it('should return "License Compliance detected 1 new license"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: false,
baseReportHasLicenses: true,
licenseReportLength: 1,
};
expect(getters.summaryTextWithLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 1 new license',
);
});
});
describe('when multiple licenses are detected', () => {
it('should return "License Compliance detected 2 new licenses"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: false,
baseReportHasLicenses: true,
licenseReportLength: 2,
};
expect(getters.summaryTextWithLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 2 new licenses',
);
});
});
});
});
describe('when there are no licenses on the BASE', () => { describe('when there are no licenses on the BASE', () => {
beforeEach(() => { beforeEach(() => {
state = createState(); state = {
...createState(),
};
}); });
it.each` describe('when blacklisted licenses exist on the HEAD', () => {
givenLicenseReport | givenReportContainsBlacklistedLicense | expectedSummaryText describe('when a single license is detected', () => {
${[]} | ${false} | ${'License Compliance detected no licenses for the source branch only'} it('should return "License Compliance detected 1 license and policy violation for the source branch only; approval required"', () => {
${[licenseReportMock[0]]} | ${false} | ${'License Compliance detected 1 license for the source branch only'}
${[licenseReportMock[0], licenseReportMock[0]]} | ${false} | ${'License Compliance detected 2 licenses for the source branch only'}
${[licenseReportMock[0]]} | ${true} | ${'License Compliance detected 1 license and policy violation for the source branch only; approval required'}
${[licenseReportMock[0], licenseReportMock[0]]} | ${true} | ${'License Compliance detected 2 licenses and policy violations for the source branch only; approval required'}
`(
`should show "$expectedSummaryText" if the report contains $givenLicenseReport.length license(s) and contains blacklisted licenses is: $givenReportContainsBlacklistedLicense`,
({ givenLicenseReport, givenReportContainsBlacklistedLicense, expectedSummaryText }) => {
const mockGetters = { const mockGetters = {
licenseReport: givenLicenseReport, reportContainsBlacklistedLicense: true,
reportContainsBlacklistedLicense: givenReportContainsBlacklistedLicense, baseReportHasLicenses: false,
licenseReportLength: 1,
}; };
expect(getters.licenseSummaryText(state, mockGetters)).toBe(expectedSummaryText); expect(getters.summaryTextWithLicenseCheck(state, mockGetters)).toBe(
}, 'License Compliance detected 1 license and policy violation for the source branch only; approval required',
); );
}); });
}); });
describe('when multiple licenses are detected', () => {
it('should return "License Compliance detected 2 licenses and policy violations for the source branch only; approval required"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: true,
baseReportHasLicenses: false,
licenseReportLength: 2,
};
expect(getters.summaryTextWithLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 2 licenses and policy violations for the source branch only; approval required',
);
});
});
});
describe('when blacklisted licenses are not detected on the HEAD', () => {
describe('when a single license is detected', () => {
it('should return "License Compliance detected 1 license for the source branch only"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: false,
baseReportHasLicenses: false,
licenseReportLength: 1,
};
expect(getters.summaryTextWithLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 1 license for the source branch only',
);
});
});
describe('when multiple licenses are detected', () => {
it('should return "License Compliance detected 2 licenses for the source branch only"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: false,
baseReportHasLicenses: false,
licenseReportLength: 2,
};
expect(getters.summaryTextWithLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 2 licenses for the source branch only',
);
});
});
});
});
});
describe('summaryTextWithoutLicenseCheck', () => {
describe('when licenses exist on both the HEAD and the BASE', () => {
beforeEach(() => {
state = {
...createState(),
};
});
describe('when blacklisted licenses exist on the HEAD', () => {
describe('when a single license is detected', () => {
it('should return "License Compliance detected 1 new license and policy violation"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: true,
baseReportHasLicenses: true,
licenseReportLength: 1,
};
expect(getters.summaryTextWithoutLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 1 new license and policy violation',
);
});
});
describe('when multiple licenses are detected', () => {
it('should return "License Compliance detected 2 new licenses and policy violations"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: true,
baseReportHasLicenses: true,
licenseReportLength: 2,
};
expect(getters.summaryTextWithoutLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 2 new licenses and policy violations',
);
});
});
});
describe('when blacklisted licenses are not detected on the HEAD', () => {
describe('when a single license is detected', () => {
it('should return "License Compliance detected 1 new license"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: false,
baseReportHasLicenses: true,
licenseReportLength: 1,
};
expect(getters.summaryTextWithoutLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 1 new license',
);
});
});
describe('when multiple licenses are detected', () => {
it('should return "License Compliance detected 2 new licenses"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: false,
baseReportHasLicenses: true,
licenseReportLength: 2,
};
expect(getters.summaryTextWithoutLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 2 new licenses',
);
});
});
});
});
describe('when there are no licenses on the BASE', () => {
beforeEach(() => {
state = {
...createState(),
};
});
describe('when blacklisted licenses exist on the HEAD', () => {
describe('when a single license is detected', () => {
it('should return "License Compliance detected 1 license and policy violation for the source branch only"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: true,
baseReportHasLicenses: false,
licenseReportLength: 1,
};
expect(getters.summaryTextWithoutLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 1 license and policy violation for the source branch only',
);
});
});
describe('when multiple licenses are detected', () => {
it('should return "License Compliance detected 2 licenses and policy violations for the source branch only"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: true,
baseReportHasLicenses: false,
licenseReportLength: 2,
};
expect(getters.summaryTextWithoutLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 2 licenses and policy violations for the source branch only',
);
});
});
});
describe('when blacklisted licenses are not detected on the HEAD', () => {
describe('when a single license is detected', () => {
it('should return "License Compliance detected 1 license for the source branch only"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: false,
baseReportHasLicenses: false,
licenseReportLength: 1,
};
expect(getters.summaryTextWithoutLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 1 license for the source branch only',
);
});
});
describe('when multiple licenses are detected', () => {
it('should return "License Compliance detected 2 licenses for the source branch only"', () => {
const mockGetters = {
reportContainsBlacklistedLicense: false,
baseReportHasLicenses: false,
licenseReportLength: 2,
};
expect(getters.summaryTextWithoutLicenseCheck(state, mockGetters)).toBe(
'License Compliance detected 2 licenses for the source branch only',
);
});
});
});
});
});
describe('reportContainsBlacklistedLicense', () => { describe('reportContainsBlacklistedLicense', () => {
it('should be false if the report does not contain blacklisted licenses', () => { it('should be false if the report does not contain blacklisted licenses', () => {
const mockGetters = { const mockGetters = {
......
...@@ -153,6 +153,54 @@ describe('License store mutations', () => { ...@@ -153,6 +153,54 @@ describe('License store mutations', () => {
}); });
}); });
describe('REQUEST_LICENSE_CHECK_APPROVAL_RULE', () => {
it('sets isLoadingLicenseCheckApprovalRule to true', () => {
store.replaceState({
...store.state,
licenseManagement: {
isLoadingLicenseCheckApprovalRule: true,
},
});
store.commit(`licenseManagement/${types.REQUEST_LICENSE_CHECK_APPROVAL_RULE}`);
expect(store.state.licenseManagement.isLoadingLicenseCheckApprovalRule).toBe(true);
});
});
describe('RECEIVE_LICENSE_CHECK_APPROVAL_RULE_SUCCESS', () => {
it('sets isLoadingLicenseCheckApprovalRule to false and hasLicenseCheckApprovalRule to true', () => {
store.replaceState({
...store.state,
licenseManagement: {
isLoadingLicenseCheckApprovalRule: true,
},
});
store.commit(`licenseManagement/${types.RECEIVE_LICENSE_CHECK_APPROVAL_RULE_SUCCESS}`, {
hasLicenseCheckApprovalRule: true,
});
expect(store.state.licenseManagement.isLoadingLicenseCheckApprovalRule).toBe(false);
expect(store.state.licenseManagement.hasLicenseCheckApprovalRule).toBe(true);
});
});
describe('RECEIVE_LICENSE_CHECK_APPROVAL_RULE_ERROR', () => {
it('sets isLoadingLicenseCheckApprovalRule to false', () => {
store.replaceState({
...store.state,
licenseManagement: {
isLoadingLicenseCheckApprovalRule: true,
},
});
store.commit(`licenseManagement/${types.RECEIVE_LICENSE_CHECK_APPROVAL_RULE_ERROR}`);
expect(store.state.licenseManagement.isLoadingLicenseCheckApprovalRule).toBe(false);
});
});
describe('RECEIVE_MANAGED_LICENSES_SUCCESS', () => { describe('RECEIVE_MANAGED_LICENSES_SUCCESS', () => {
it('sets isLoadingManagedLicenses and loadManagedLicensesError to false and saves managed licenses', () => { it('sets isLoadingManagedLicenses and loadManagedLicensesError to false and saves managed licenses', () => {
store.replaceState({ store.replaceState({
......
...@@ -13640,6 +13640,11 @@ msgstr "" ...@@ -13640,6 +13640,11 @@ msgstr ""
msgid "LicenseCompliance|License Approvals" msgid "LicenseCompliance|License Approvals"
msgstr "" msgstr ""
msgid "LicenseCompliance|License Compliance detected %d license and policy violation for the source branch only"
msgid_plural "LicenseCompliance|License Compliance detected %d licenses and policy violations for the source branch only"
msgstr[0] ""
msgstr[1] ""
msgid "LicenseCompliance|License Compliance detected %d license and policy violation for the source branch only; approval required" msgid "LicenseCompliance|License Compliance detected %d license and policy violation for the source branch only; approval required"
msgid_plural "LicenseCompliance|License Compliance detected %d licenses and policy violations for the source branch only; approval required" msgid_plural "LicenseCompliance|License Compliance detected %d licenses and policy violations for the source branch only; approval required"
msgstr[0] "" msgstr[0] ""
...@@ -13655,6 +13660,11 @@ msgid_plural "LicenseCompliance|License Compliance detected %d new licenses" ...@@ -13655,6 +13660,11 @@ msgid_plural "LicenseCompliance|License Compliance detected %d new licenses"
msgstr[0] "" msgstr[0] ""
msgstr[1] "" msgstr[1] ""
msgid "LicenseCompliance|License Compliance detected %d new license and policy violation"
msgid_plural "LicenseCompliance|License Compliance detected %d new licenses and policy violations"
msgstr[0] ""
msgstr[1] ""
msgid "LicenseCompliance|License Compliance detected %d new license and policy violation; approval required" msgid "LicenseCompliance|License Compliance detected %d new license and policy violation; approval required"
msgid_plural "LicenseCompliance|License Compliance detected %d new licenses and policy violations; approval required" msgid_plural "LicenseCompliance|License Compliance detected %d new licenses and policy violations; approval required"
msgstr[0] "" msgstr[0] ""
...@@ -28173,6 +28183,9 @@ msgstr "" ...@@ -28173,6 +28183,9 @@ msgstr ""
msgid "mrWidget|You can merge this merge request manually using the" msgid "mrWidget|You can merge this merge request manually using the"
msgstr "" msgstr ""
msgid "mrWidget|You can only merge once the denied license is removed"
msgstr ""
msgid "mrWidget|Your password" msgid "mrWidget|Your password"
msgstr "" msgstr ""
......
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