Skip to content
代码片段 群组 项目
提交 ed39af32 编辑于 作者: Dave Pisek's avatar Dave Pisek
浏览文件

Add AI badge to MR widget vulnerability findings

This code diff introduces a new feature to the
Merge Request widget.

 It adds a badge to vulnerabilities that can be
resolved using AI.

This badge is only shown if the user has the
ability to resolve vulnerabilities with AI and the
vulnerability has AI resolution enabled.
上级 9135f4da
No related branches found
No related tags found
无相关合并请求
<script>
import { GlBadge, GlButton } from '@gitlab/ui';
import { GlBadge, GlButton, GlIcon } from '@gitlab/ui';
import { SEVERITY_LEVELS } from 'ee/security_dashboard/store/constants';
import MrWidget from '~/vue_merge_request_widget/components/widget/widget.vue';
import MrWidgetRow from '~/vue_merge_request_widget/components/widget/widget_content_row.vue';
......@@ -7,6 +7,7 @@ import axios from '~/lib/utils/axios_utils';
import { s__ } from '~/locale';
import SummaryHighlights from 'ee/vue_shared/security_reports/components/summary_highlights.vue';
import glFeatureFlagMixin from '~/vue_shared/mixins/gl_feature_flags_mixin';
import glAbilitiesMixin from '~/vue_shared/mixins/gl_abilities_mixin';
import { EXTENSION_ICONS } from '~/vue_merge_request_widget/constants';
import { capitalizeFirstCharacter, convertToCamelCase } from '~/lib/utils/text_utility';
import { helpPagePath } from '~/helpers/help_page_helper';
......@@ -28,10 +29,11 @@ export default {
SecurityTrainingPromoWidget,
GlBadge,
GlButton,
GlIcon,
DynamicScroller,
DynamicScrollerItem,
},
mixins: [glFeatureFlagMixin()],
mixins: [glAbilitiesMixin(), glFeatureFlagMixin()],
i18n,
props: {
mr: {
......@@ -317,6 +319,14 @@ export default {
clearModalData() {
this.modalData = null;
},
isAiResolvable(vuln) {
return (
vuln.ai_resolution_enabled &&
this.glFeatures.resolveVulnerabilityInMr &&
this.glAbilities.resolveVulnerabilityWithAi
);
},
},
SEVERITY_LEVELS,
widgetHelpPopover: {
......@@ -452,6 +462,14 @@ export default {
<gl-badge v-if="isDismissed(vuln)" class="gl-ml-3">{{
$options.i18n.dismissed
}}</gl-badge>
<gl-badge
v-if="isAiResolvable(vuln)"
variant="info"
class="gl-ml-3"
data-testid="ai-resolvable-badge"
>
<gl-icon :size="12" name="tanuki-ai" />
</gl-badge>
</template>
</mr-widget-row>
</dynamic-scroller-item>
......
......@@ -275,6 +275,84 @@ describe('MR Widget Security Reports', () => {
expect(findDismissedBadge().text()).toBe('Dismissed');
});
it.each`
resolveVulnerabilityWithAi | aiResolutionEnabled | shouldShowAiBadge
${true} | ${true} | ${true}
${false} | ${true} | ${false}
${true} | ${false} | ${false}
`(
'with the resolveVulnerabilityWithAi ability set to "$resolveVulnerabilityWithAi" and the vulnerability has ai_resolution_enabled set to "$aiResolutionEnabled" it should show the AI-Badge: "$shouldShowAiBadge"',
async ({ resolveVulnerabilityWithAi, aiResolutionEnabled, shouldShowAiBadge }) => {
await createComponentAndExpandWidget({
mockDataFn: () =>
mockWithData({
findings: {
sast: {
added: [
{
uuid: '1',
severity: 'critical',
name: 'Password leak',
state: 'dismissed',
ai_resolution_enabled: aiResolutionEnabled,
},
],
},
},
}),
provide: {
glAbilities: {
resolveVulnerabilityWithAi,
},
glFeatures: {
resolveVulnerabilityInMr: true,
},
},
});
expect(wrapper.findByTestId('ai-resolvable-badge').exists()).toBe(shouldShowAiBadge);
},
);
it.each`
resolveVulnerabilityWithAi | aiResolutionEnabled
${true} | ${true}
${false} | ${true}
${true} | ${false}
`(
'with the "resolveVulnerabilityInMr" feature flag set to "false" it should never show the AI-Badge',
async ({ resolveVulnerabilityWithAi, aiResolutionEnabled }) => {
await createComponentAndExpandWidget({
mockDataFn: () =>
mockWithData({
findings: {
sast: {
added: [
{
uuid: '1',
severity: 'critical',
name: 'Password leak',
state: 'dismissed',
ai_resolution_enabled: aiResolutionEnabled,
},
],
},
},
}),
provide: {
glAbilities: {
resolveVulnerabilityWithAi,
},
glFeatures: {
resolveVulnerabilityInMr: false,
},
},
});
expect(wrapper.findByTestId('ai-resolvable-badge').exists()).toBe(false);
},
);
it('should mount the widget component', async () => {
await createComponentWithData();
......
0% 加载中 .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册