Commit 0041c815 authored by Jan Provaznik's avatar Jan Provaznik

Merge branch...

Merge branch '212497-update-data-sent-to-vulnerabilities-page-to-include-less-information-and-use-the-new-data' into 'master'

Send less data to Vulnerabilites page

See merge request gitlab-org/gitlab!31616
parents 7292be58 696b84b2
import Vue from 'vue'; import Vue from 'vue';
import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils';
import HeaderApp from 'ee/vulnerabilities/components/header.vue'; import HeaderApp from 'ee/vulnerabilities/components/header.vue';
import DetailsApp from 'ee/vulnerabilities/components/details.vue'; import DetailsApp from 'ee/vulnerabilities/components/details.vue';
import FooterApp from 'ee/vulnerabilities/components/footer.vue'; import FooterApp from 'ee/vulnerabilities/components/footer.vue';
import { VULNERABILITY_STATE_OBJECTS } from 'ee/vulnerabilities/constants';
function createHeaderApp() { function createHeaderApp() {
const el = document.getElementById('js-vulnerability-header'); const el = document.getElementById('js-vulnerability-header');
const initialVulnerability = JSON.parse(el.dataset.vulnerabilityJson); const vulnerability = JSON.parse(el.dataset.vulnerability);
const pipeline = JSON.parse(el.dataset.pipelineJson);
const finding = JSON.parse(el.dataset.findingJson);
const { projectFingerprint, createIssueUrl, createMrUrl } = el.dataset;
return new Vue({ return new Vue({
el, el,
...@@ -17,12 +15,7 @@ function createHeaderApp() { ...@@ -17,12 +15,7 @@ function createHeaderApp() {
render: h => render: h =>
h(HeaderApp, { h(HeaderApp, {
props: { props: {
createMrUrl, initialVulnerability: vulnerability,
initialVulnerability,
finding,
pipeline,
projectFingerprint,
createIssueUrl,
}, },
}), }),
}); });
...@@ -30,12 +23,11 @@ function createHeaderApp() { ...@@ -30,12 +23,11 @@ function createHeaderApp() {
function createDetailsApp() { function createDetailsApp() {
const el = document.getElementById('js-vulnerability-details'); const el = document.getElementById('js-vulnerability-details');
const vulnerability = JSON.parse(el.dataset.vulnerabilityJson); const vulnerability = JSON.parse(el.dataset.vulnerability);
const finding = JSON.parse(el.dataset.findingJson);
return new Vue({ return new Vue({
el, el,
render: h => h(DetailsApp, { props: { vulnerability, finding } }), render: h => h(DetailsApp, { props: { vulnerability } }),
}); });
} }
...@@ -46,25 +38,43 @@ function createFooterApp() { ...@@ -46,25 +38,43 @@ function createFooterApp() {
return false; return false;
} }
const { vulnerabilityFeedbackHelpPath, hasMr, discussionsUrl, notesUrl } = el.dataset; const {
const vulnerability = JSON.parse(el.dataset.vulnerabilityJson); vulnerabilityFeedbackHelpPath,
const finding = JSON.parse(el.dataset.findingJson); hasMr,
discussionsUrl,
state,
issueFeedback,
mergeRequestFeedback,
notesUrl,
project,
remediations,
solution,
} = convertObjectPropsToCamelCase(JSON.parse(el.dataset.vulnerability));
const remediation = remediations?.length ? remediations[0] : null;
const hasDownload = Boolean( const hasDownload = Boolean(
vulnerability.state !== 'resolved' && finding.remediation?.diff?.length && !hasMr, state !== VULNERABILITY_STATE_OBJECTS.resolved.state && remediation?.diff?.length && !hasMr,
); );
const hasRemediation = Boolean(remediation);
const props = { const props = {
discussionsUrl, discussionsUrl,
notesUrl, notesUrl,
finding,
solutionInfo: { solutionInfo: {
solution: finding.solution, solution,
remediation: finding.remediation, remediation,
hasDownload, hasDownload,
hasMr, hasMr,
hasRemediation,
vulnerabilityFeedbackHelpPath, vulnerabilityFeedbackHelpPath,
isStandaloneVulnerability: true, isStandaloneVulnerability: true,
}, },
issueFeedback,
mergeRequestFeedback,
project: {
url: project.full_path,
value: project.full_name,
},
}; };
return new Vue({ return new Vue({
......
...@@ -11,17 +11,13 @@ export default { ...@@ -11,17 +11,13 @@ export default {
type: Object, type: Object,
required: true, required: true,
}, },
finding: {
type: Object,
required: true,
},
}, },
computed: { computed: {
location() { location() {
return this.finding.location || {}; return this.vulnerability.location || {};
}, },
scanner() { scanner() {
return this.finding.scanner || {}; return this.vulnerability.scanner || {};
}, },
fileText() { fileText() {
return (this.location.file || '') + (this.lineNumber ? `:${this.lineNumber}` : ''); return (this.location.file || '') + (this.lineNumber ? `:${this.lineNumber}` : '');
...@@ -57,19 +53,27 @@ export default { ...@@ -57,19 +53,27 @@ export default {
</script> </script>
<template> <template>
<div class="md"> <div class="md" data-qa-selector="vulnerability_details">
<h1 class="mt-3 mb-2 border-bottom-0" data-testid="title">{{ vulnerability.title }}</h1> <h1
class="mt-3 mb-2 border-bottom-0"
data-testid="title"
data-qa-selector="vulnerability_title"
>
{{ vulnerability.title }}
</h1>
<h3 class="mt-0">{{ __('Description') }}</h3> <h3 class="mt-0">{{ __('Description') }}</h3>
<p data-testid="description">{{ finding.description }}</p> <p data-testid="description" data-qa-selector="vulnerability_description">
{{ vulnerability.description }}
</p>
<ul> <ul>
<detail-item :sprintf-message="__('%{labelStart}Severity:%{labelEnd} %{severity}')"> <detail-item :sprintf-message="__('%{labelStart}Severity:%{labelEnd} %{severity}')">
<severity-badge :severity="vulnerability.severity" class="gl-display-inline ml-1" /> <severity-badge :severity="vulnerability.severity" class="gl-display-inline ml-1" />
</detail-item> </detail-item>
<detail-item <detail-item
v-if="finding.evidence" v-if="vulnerability.evidence"
:sprintf-message="__('%{labelStart}Evidence:%{labelEnd} %{evidence}')" :sprintf-message="__('%{labelStart}Evidence:%{labelEnd} %{evidence}')"
>{{ finding.evidence }} >{{ vulnerability.evidence }}
</detail-item> </detail-item>
<detail-item :sprintf-message="__('%{labelStart}Report Type:%{labelEnd} %{reportType}')" <detail-item :sprintf-message="__('%{labelStart}Report Type:%{labelEnd} %{reportType}')"
>{{ vulnerability.report_type }} >{{ vulnerability.report_type }}
...@@ -125,10 +129,10 @@ export default { ...@@ -125,10 +129,10 @@ export default {
</ul> </ul>
</template> </template>
<template v-if="finding.links && finding.links.length"> <template v-if="vulnerability.links && vulnerability.links.length">
<h3>{{ __('Links') }}</h3> <h3>{{ __('Links') }}</h3>
<ul> <ul>
<li v-for="link in finding.links" :key="link.url"> <li v-for="link in vulnerability.links" :key="link.url">
<gl-link <gl-link
:href="link.url" :href="link.url"
data-testid="link" data-testid="link"
...@@ -142,10 +146,10 @@ export default { ...@@ -142,10 +146,10 @@ export default {
</ul> </ul>
</template> </template>
<template v-if="finding.identifiers && finding.identifiers.length"> <template v-if="vulnerability.identifiers && vulnerability.identifiers.length">
<h3>{{ __('Identifiers') }}</h3> <h3>{{ __('Identifiers') }}</h3>
<ul> <ul>
<li v-for="identifier in finding.identifiers" :key="identifier.url"> <li v-for="identifier in vulnerability.identifiers" :key="identifier.url">
<gl-link :href="identifier.url" data-testid="identifier" target="_blank"> <gl-link :href="identifier.url" data-testid="identifier" target="_blank">
{{ identifier.name }} {{ identifier.name }}
</gl-link> </gl-link>
......
...@@ -22,7 +22,7 @@ export default { ...@@ -22,7 +22,7 @@ export default {
type: String, type: String,
required: true, required: true,
}, },
finding: { project: {
type: Object, type: Object,
required: true, required: true,
}, },
...@@ -30,6 +30,16 @@ export default { ...@@ -30,6 +30,16 @@ export default {
type: Object, type: Object,
required: true, required: true,
}, },
issueFeedback: {
type: Object,
required: false,
default: () => null,
},
mergeRequestFeedback: {
type: Object,
required: false,
default: () => null,
},
}, },
data: () => ({ data: () => ({
...@@ -52,12 +62,6 @@ export default { ...@@ -52,12 +62,6 @@ export default {
hasSolution() { hasSolution() {
return Boolean(this.solutionInfo.solution || this.solutionInfo.remediation); return Boolean(this.solutionInfo.solution || this.solutionInfo.remediation);
}, },
project() {
return {
url: this.finding.project?.full_path,
value: this.finding.project?.full_name,
};
},
}, },
created() { created() {
...@@ -153,19 +157,19 @@ export default { ...@@ -153,19 +157,19 @@ export default {
}; };
</script> </script>
<template> <template>
<div> <div data-qa-selector="vulnerability_footer">
<solution-card v-if="hasSolution" v-bind="solutionInfo" /> <solution-card v-if="hasSolution" v-bind="solutionInfo" />
<div v-if="finding.issue_feedback || finding.merge_request_feedback" class="card"> <div v-if="issueFeedback || mergeRequestFeedback" class="card">
<issue-note <issue-note
v-if="finding.issue_feedback" v-if="issueFeedback"
:feedback="finding.issue_feedback" :feedback="issueFeedback"
:project="project" :project="project"
class="card-body" class="card-body"
/> />
<merge-request-note <merge-request-note
v-if="finding.merge_request_feedback" v-if="mergeRequestFeedback"
:feedback="finding.merge_request_feedback" :feedback="mergeRequestFeedback"
:project="project" :project="project"
class="card-body" class="card-body"
/> />
......
...@@ -26,30 +26,10 @@ export default { ...@@ -26,30 +26,10 @@ export default {
}, },
props: { props: {
createMrUrl: {
type: String,
required: true,
},
initialVulnerability: { initialVulnerability: {
type: Object, type: Object,
required: true, required: true,
}, },
finding: {
type: Object,
required: true,
},
pipeline: {
type: Object,
required: true,
},
createIssueUrl: {
type: String,
required: true,
},
projectFingerprint: {
type: String,
required: true,
},
}, },
data() { data() {
...@@ -88,16 +68,16 @@ export default { ...@@ -88,16 +68,16 @@ export default {
); );
}, },
hasIssue() { hasIssue() {
return Boolean(this.finding.issue_feedback?.issue_iid); return Boolean(this.vulnerability.issue_feedback?.issue_iid);
}, },
hasRemediation() { hasRemediation() {
const { remediations } = this.finding; const { remediations } = this.vulnerability;
return Boolean(remediations && remediations[0]?.diff?.length > 0); return Boolean(remediations && remediations[0]?.diff?.length > 0);
}, },
canCreateMergeRequest() { canCreateMergeRequest() {
return ( return (
!this.finding.merge_request_feedback?.merge_request_path && !this.vulnerability.merge_request_feedback?.merge_request_path &&
Boolean(this.createMrUrl) && Boolean(this.vulnerability.create_mr_url) &&
this.hasRemediation this.hasRemediation
); );
}, },
...@@ -163,17 +143,23 @@ export default { ...@@ -163,17 +143,23 @@ export default {
}, },
createIssue() { createIssue() {
this.isProcessingAction = true; this.isProcessingAction = true;
const {
report_type: category,
project_fingerprint: projectFingerprint,
id,
} = this.vulnerability;
axios axios
.post(this.createIssueUrl, { .post(this.vulnerability.create_issue_url, {
vulnerability_feedback: { vulnerability_feedback: {
feedback_type: FEEDBACK_TYPES.ISSUE, feedback_type: FEEDBACK_TYPES.ISSUE,
category: this.vulnerability.report_type, category,
project_fingerprint: this.projectFingerprint, project_fingerprint: projectFingerprint,
vulnerability_data: { vulnerability_data: {
...this.vulnerability, ...this.vulnerability,
...this.finding, category,
category: this.vulnerability.report_type, vulnerability_id: id,
vulnerability_id: this.vulnerability.id,
}, },
}, },
}) })
...@@ -189,17 +175,23 @@ export default { ...@@ -189,17 +175,23 @@ export default {
}, },
createMergeRequest() { createMergeRequest() {
this.isProcessingAction = true; this.isProcessingAction = true;
const {
report_type: category,
pipeline: { sourceBranch },
project_fingerprint: projectFingerprint,
} = this.vulnerability;
axios axios
.post(this.createMrUrl, { .post(this.vulnerability.create_mr_url, {
vulnerability_feedback: { vulnerability_feedback: {
feedback_type: FEEDBACK_TYPES.MERGE_REQUEST, feedback_type: FEEDBACK_TYPES.MERGE_REQUEST,
category: this.vulnerability.report_type, category,
project_fingerprint: this.projectFingerprint, project_fingerprint: projectFingerprint,
vulnerability_data: { vulnerability_data: {
...this.vulnerability, ...this.vulnerability,
...this.finding, category,
category: this.vulnerability.report_type, target_branch: sourceBranch,
target_branch: this.pipeline.sourceBranch,
}, },
}, },
}) })
...@@ -214,14 +206,17 @@ export default { ...@@ -214,14 +206,17 @@ export default {
}); });
}, },
downloadPatch() { downloadPatch() {
download({ fileData: this.finding.remediations[0].diff, fileName: `remediation.patch` }); download({
fileData: this.vulnerability.remediations[0].diff,
fileName: `remediation.patch`,
});
}, },
}, },
}; };
</script> </script>
<template> <template>
<div> <div data-qa-selector="vulnerability_header">
<resolution-alert <resolution-alert
v-if="showResolutionAlert" v-if="showResolutionAlert"
:vulnerability-id="vulnerability.id" :vulnerability-id="vulnerability.id"
...@@ -243,7 +238,6 @@ export default { ...@@ -243,7 +238,6 @@ export default {
<status-description <status-description
class="issuable-meta" class="issuable-meta"
:vulnerability="vulnerability" :vulnerability="vulnerability"
:pipeline="pipeline"
:user="user" :user="user"
:is-loading-vulnerability="isLoadingVulnerability" :is-loading-vulnerability="isLoadingVulnerability"
:is-loading-user="isLoadingUser" :is-loading-user="isLoadingUser"
......
...@@ -19,10 +19,6 @@ export default { ...@@ -19,10 +19,6 @@ export default {
type: Object, type: Object,
required: true, required: true,
}, },
pipeline: {
type: Object,
required: true,
},
user: { user: {
type: Object, type: Object,
required: false, required: false,
...@@ -42,7 +38,7 @@ export default { ...@@ -42,7 +38,7 @@ export default {
time() { time() {
const { state } = this.vulnerability; const { state } = this.vulnerability;
return state === 'detected' return state === 'detected'
? this.pipeline.created_at ? this.vulnerability.pipeline.created_at
: this.vulnerability[`${this.vulnerability.state}_at`]; : this.vulnerability[`${this.vulnerability.state}_at`];
}, },
...@@ -86,9 +82,9 @@ export default { ...@@ -86,9 +82,9 @@ export default {
img-css-classes="avatar-inline" img-css-classes="avatar-inline"
/> />
</template> </template>
<template v-if="pipeline" #pipelineLink> <template v-if="vulnerability.pipeline" #pipelineLink>
<gl-link :href="pipeline.url" target="_blank" class="link"> <gl-link :href="vulnerability.pipeline.url" target="_blank" class="link">
{{ pipeline.id }} {{ vulnerability.pipeline.id }}
</gl-link> </gl-link>
</template> </template>
</gl-sprintf> </gl-sprintf>
......
...@@ -199,7 +199,12 @@ export default { ...@@ -199,7 +199,12 @@ export default {
<template #cell(title)="{ item }"> <template #cell(title)="{ item }">
<div class="d-flex flex-column flex-sm-row align-items-end align-items-sm-start"> <div class="d-flex flex-column flex-sm-row align-items-end align-items-sm-start">
<gl-link class="text-body js-description" :href="item.vulnerabilityPath"> <gl-link
class="text-body js-description"
:href="item.vulnerabilityPath"
data-qa-selector="vulnerability"
:data-qa-vulnerability-description="`${item.title}`"
>
{{ item.title }} {{ item.title }}
</gl-link> </gl-link>
<issue-link v-if="issue(item)" :issue="issue(item)" /> <issue-link v-if="issue(item)" :issue="issue(item)" />
......
# frozen_string_literal: true # frozen_string_literal: true
module VulnerabilitiesHelper module VulnerabilitiesHelper
def vulnerability_data(vulnerability, pipeline) def vulnerability_details_json(vulnerability, pipeline)
vulnerability_details(vulnerability, pipeline).to_json
end
def vulnerability_details(vulnerability, pipeline)
return unless vulnerability return unless vulnerability
{ result = {
vulnerability_json: VulnerabilitySerializer.new.represent(vulnerability).to_json, timestamp: Time.now.to_i,
project_fingerprint: vulnerability.finding.project_fingerprint,
create_issue_url: create_vulnerability_feedback_issue_path(vulnerability.finding.project), create_issue_url: create_vulnerability_feedback_issue_path(vulnerability.finding.project),
notes_url: project_security_vulnerability_notes_path(vulnerability.project, vulnerability),
discussions_url: discussions_project_security_vulnerability_path(vulnerability.project, vulnerability),
pipeline_json: vulnerability_pipeline_data(pipeline).to_json,
has_mr: !!vulnerability.finding.merge_request_feedback.try(:merge_request_iid), has_mr: !!vulnerability.finding.merge_request_feedback.try(:merge_request_iid),
vulnerability_feedback_help_path: help_page_path('user/application_security/index', anchor: 'interacting-with-the-vulnerabilities'),
finding_json: vulnerability_finding_data(vulnerability).to_json,
create_mr_url: create_vulnerability_feedback_merge_request_path(vulnerability.finding.project), create_mr_url: create_vulnerability_feedback_merge_request_path(vulnerability.finding.project),
timestamp: Time.now.to_i discussions_url: discussions_project_security_vulnerability_path(vulnerability.project, vulnerability),
notes_url: project_security_vulnerability_notes_path(vulnerability.project, vulnerability),
vulnerability_feedback_help_path: help_page_path('user/application_security/index', anchor: 'interacting-with-the-vulnerabilities'),
pipeline: vulnerability_pipeline_data(pipeline)
} }
result.merge(vulnerability_data(vulnerability), vulnerability_finding_data(vulnerability))
end end
def vulnerability_pipeline_data(pipeline) def vulnerability_pipeline_data(pipeline)
...@@ -30,9 +33,12 @@ module VulnerabilitiesHelper ...@@ -30,9 +33,12 @@ module VulnerabilitiesHelper
} }
end end
def vulnerability_data(vulnerability)
VulnerabilitySerializer.new.represent(vulnerability)
end
def vulnerability_finding_data(vulnerability) def vulnerability_finding_data(vulnerability)
finding = Vulnerabilities::FindingSerializer.new(current_user: current_user).represent(vulnerability.finding) finding = Vulnerabilities::FindingSerializer.new(current_user: current_user).represent(vulnerability.finding)
remediation = finding[:remediations]&.first
data = finding.slice( data = finding.slice(
:description, :description,
...@@ -43,11 +49,11 @@ module VulnerabilitiesHelper ...@@ -43,11 +49,11 @@ module VulnerabilitiesHelper
:issue_feedback, :issue_feedback,
:merge_request_feedback, :merge_request_feedback,
:project, :project,
:project_fingerprint,
:remediations, :remediations,
:evidence, :evidence,
:scanner :scanner,
).merge( :solution
solution: remediation ? remediation['summary'] : finding[:solution]
) )
if data[:location]['file'] if data[:location]['file']
......
...@@ -3,8 +3,8 @@ ...@@ -3,8 +3,8 @@
- breadcrumb_title @vulnerability.id - breadcrumb_title @vulnerability.id
- page_title @vulnerability.title - page_title @vulnerability.title
- page_description @vulnerability.description - page_description @vulnerability.description
- vulnerability_data = vulnerability_data(@vulnerability, @pipeline) - vulnerability_init_details = { vulnerability: vulnerability_details_json(@vulnerability, @pipeline)}
#js-vulnerability-header{ data: vulnerability_data } #js-vulnerability-header{ data: vulnerability_init_details }
#js-vulnerability-details{ data: vulnerability_data } #js-vulnerability-details{ data: vulnerability_init_details }
#js-vulnerability-footer{ data: vulnerability_data } #js-vulnerability-footer{ data: vulnerability_init_details }
...@@ -11,16 +11,12 @@ describe('Vulnerability Details', () => { ...@@ -11,16 +11,12 @@ describe('Vulnerability Details', () => {
severity: 'bad severity', severity: 'bad severity',
confidence: 'high confidence', confidence: 'high confidence',
report_type: 'nice report_type', report_type: 'nice report_type',
description: 'vulnerability description',
}; };
const finding = { const createWrapper = vulnerabilityOverrides => {
description: 'finding description',
};
const createWrapper = findingOverrides => {
const propsData = { const propsData = {
vulnerability, vulnerability: { ...vulnerability, ...vulnerabilityOverrides },
finding: { ...finding, ...findingOverrides },
}; };
wrapper = mount(VulnerabilityDetails, { propsData }); wrapper = mount(VulnerabilityDetails, { propsData });
...@@ -37,7 +33,7 @@ describe('Vulnerability Details', () => { ...@@ -37,7 +33,7 @@ describe('Vulnerability Details', () => {
it('shows the properties that should always be shown', () => { it('shows the properties that should always be shown', () => {
createWrapper(); createWrapper();
expect(getText('title')).toBe(vulnerability.title); expect(getText('title')).toBe(vulnerability.title);
expect(getText('description')).toBe(finding.description); expect(getText('description')).toBe(vulnerability.description);
expect(wrapper.find(SeverityBadge).props('severity')).toBe(vulnerability.severity); expect(wrapper.find(SeverityBadge).props('severity')).toBe(vulnerability.severity);
expect(getText('reportType')).toBe(`Report Type: ${vulnerability.report_type}`); expect(getText('reportType')).toBe(`Report Type: ${vulnerability.report_type}`);
...@@ -62,12 +58,12 @@ describe('Vulnerability Details', () => { ...@@ -62,12 +58,12 @@ describe('Vulnerability Details', () => {
expect(getText('namespace')).toBe(`Namespace: linux`); expect(getText('namespace')).toBe(`Namespace: linux`);
}); });
it('shows the finding class if it exists', () => { it('shows the vulnerability class if it exists', () => {
createWrapper({ location: { file: 'file', class: 'class name' } }); createWrapper({ location: { file: 'file', class: 'class name' } });
expect(getText('class')).toBe(`Class: class name`); expect(getText('class')).toBe(`Class: class name`);
}); });
it('shows the finding method if it exists', () => { it('shows the vulnerability method if it exists', () => {
createWrapper({ location: { file: 'file', method: 'method name' } }); createWrapper({ location: { file: 'file', method: 'method name' } });
expect(getText('method')).toBe(`Method: method name`); expect(getText('method')).toBe(`Method: method name`);
}); });
...@@ -89,7 +85,7 @@ describe('Vulnerability Details', () => { ...@@ -89,7 +85,7 @@ describe('Vulnerability Details', () => {
}); });
}); });
it('shows the finding identifiers if they exist', () => { it('shows the vulnerability identifiers if they exist', () => {
createWrapper({ createWrapper({
identifiers: [{ url: '0', name: '00' }, { url: '1', name: '11' }, { url: '2', name: '22' }], identifiers: [{ url: '0', name: '00' }, { url: '1', name: '11' }, { url: '2', name: '22' }],
}); });
......
...@@ -29,11 +29,10 @@ describe('Vulnerability Footer', () => { ...@@ -29,11 +29,10 @@ describe('Vulnerability Footer', () => {
}, },
finding: {}, finding: {},
notesUrl: '/notes', notesUrl: '/notes',
}; project: {
const project = {
full_path: '/root/security-reports', full_path: '/root/security-reports',
full_name: 'Administrator / Security Reports', full_name: 'Administrator / Security Reports',
},
}; };
const solutionInfoProp = { const solutionInfoProp = {
...@@ -82,21 +81,17 @@ describe('Vulnerability Footer', () => { ...@@ -82,21 +81,17 @@ describe('Vulnerability Footer', () => {
describe.each` describe.each`
type | prop | component type | prop | component
${'issue'} | ${'issue_feedback'} | ${IssueNote} ${'issue'} | ${'issueFeedback'} | ${IssueNote}
${'merge request'} | ${'merge_request_feedback'} | ${MergeRequestNote} ${'merge request'} | ${'mergeRequestFeedback'} | ${MergeRequestNote}
`('$type note', ({ prop, component }) => { `('$type note', ({ prop, component }) => {
// The object itself does not matter, we just want to make sure it's passed to the issue note. // The object itself does not matter, we just want to make sure it's passed to the issue note.
const feedback = {}; const feedback = {};
it('shows issue note when an issue exists for the vulnerability', () => { it('shows issue note when an issue exists for the vulnerability', () => {
createWrapper({ ...minimumProps, finding: { project, [prop]: feedback } }); createWrapper({ ...minimumProps, [prop]: feedback });
expect(wrapper.contains(component)).toBe(true); expect(wrapper.contains(component)).toBe(true);
expect(wrapper.find(component).props()).toMatchObject({ expect(wrapper.find(component).props()).toMatchObject({
feedback, feedback,
project: {
url: project.full_path,
value: project.full_name,
},
}); });
}); });
......
...@@ -29,20 +29,26 @@ describe('Vulnerability Header', () => { ...@@ -29,20 +29,26 @@ describe('Vulnerability Header', () => {
created_at: new Date().toISOString(), created_at: new Date().toISOString(),
report_type: 'sast', report_type: 'sast',
state: 'detected', state: 'detected',
}; create_mr_url: '/create_mr_url',
create_issue_url: '/create_issue_url',
const diff = 'some diff to download'; project_fingerprint: 'abc123',
pipeline: {
const getFinding = ({ id: 2,
shouldShowCreateIssueButton = false, created_at: new Date().toISOString(),
shouldShowMergeRequestButton = false, url: 'pipeline_url',
}) => { sourceBranch: 'master',
return { },
description: 'description', description: 'description',
identifiers: 'identifiers', identifiers: 'identifiers',
links: 'links', links: 'links',
location: 'location', location: 'location',
name: 'name', name: 'name',
};
const diff = 'some diff to download';
const getVulnerability = ({ shouldShowCreateIssueButton, shouldShowMergeRequestButton }) => {
return {
issue_feedback: shouldShowCreateIssueButton ? null : { issue_iid: 12 }, issue_feedback: shouldShowCreateIssueButton ? null : { issue_iid: 12 },
remediations: shouldShowMergeRequestButton ? [{ diff }] : null, remediations: shouldShowMergeRequestButton ? [{ diff }] : null,
merge_request_feedback: { merge_request_feedback: {
...@@ -51,18 +57,6 @@ describe('Vulnerability Header', () => { ...@@ -51,18 +57,6 @@ describe('Vulnerability Header', () => {
}; };
}; };
const dataset = {
createMrUrl: '/create_mr_url',
createIssueUrl: '/create_issue_url',
projectFingerprint: 'abc123',
pipeline: {
id: 2,
created_at: new Date().toISOString(),
url: 'pipeline_url',
sourceBranch: 'master',
},
};
const createRandomUser = () => { const createRandomUser = () => {
const user = UsersMockHelper.createRandomUser(); const user = UsersMockHelper.createRandomUser();
const url = Api.buildUrl(Api.userPath).replace(':id', user.id); const url = Api.buildUrl(Api.userPath).replace(':id', user.id);
...@@ -77,13 +71,10 @@ describe('Vulnerability Header', () => { ...@@ -77,13 +71,10 @@ describe('Vulnerability Header', () => {
const findResolutionAlert = () => wrapper.find(ResolutionAlert); const findResolutionAlert = () => wrapper.find(ResolutionAlert);
const findStatusDescription = () => wrapper.find(StatusDescription); const findStatusDescription = () => wrapper.find(StatusDescription);
const createWrapper = ({ vulnerability = {}, finding = getFinding({}), props = {} }) => { const createWrapper = (vulnerability = {}) => {
wrapper = shallowMount(Header, { wrapper = shallowMount(Header, {
propsData: { propsData: {
...dataset,
...props,
initialVulnerability: { ...defaultVulnerability, ...vulnerability }, initialVulnerability: { ...defaultVulnerability, ...vulnerability },
finding,
}, },
}); });
}; };
...@@ -96,7 +87,7 @@ describe('Vulnerability Header', () => { ...@@ -96,7 +87,7 @@ describe('Vulnerability Header', () => {
}); });
describe('state dropdown', () => { describe('state dropdown', () => {
beforeEach(() => createWrapper({})); beforeEach(() => createWrapper());
it('the vulnerability state dropdown is rendered', () => { it('the vulnerability state dropdown is rendered', () => {
expect(wrapper.find(VulnerabilityStateDropdown).exists()).toBe(true); expect(wrapper.find(VulnerabilityStateDropdown).exists()).toBe(true);
...@@ -158,12 +149,12 @@ describe('Vulnerability Header', () => { ...@@ -158,12 +149,12 @@ describe('Vulnerability Header', () => {
describe('split button', () => { describe('split button', () => {
it('does render the create merge request and issue button as a split button', () => { it('does render the create merge request and issue button as a split button', () => {
createWrapper({ createWrapper(
finding: getFinding({ getVulnerability({
shouldShowCreateIssueButton: true, shouldShowCreateIssueButton: true,
shouldShowMergeRequestButton: true, shouldShowMergeRequestButton: true,
}), }),
}); );
expect(findSplitButton().exists()).toBe(true); expect(findSplitButton().exists()).toBe(true);
const buttons = findSplitButton().props('buttons'); const buttons = findSplitButton().props('buttons');
expect(buttons).toHaveLength(3); expect(buttons).toHaveLength(3);
...@@ -173,21 +164,19 @@ describe('Vulnerability Header', () => { ...@@ -173,21 +164,19 @@ describe('Vulnerability Header', () => {
}); });
it('does not render the split button if there is only one action', () => { it('does not render the split button if there is only one action', () => {
createWrapper({ finding: getFinding({ shouldShowCreateIssueButton: true }) }); createWrapper(getVulnerability({ shouldShowCreateIssueButton: true }));
expect(findSplitButton().exists()).toBe(false); expect(findSplitButton().exists()).toBe(false);
}); });
}); });
describe('single action button', () => { describe('single action button', () => {
it('does not display if there are no actions', () => { it('does not display if there are no actions', () => {
createWrapper({}); createWrapper(getVulnerability({}));
expect(findGlDeprecatedButton().exists()).toBe(false); expect(findGlDeprecatedButton().exists()).toBe(false);
}); });
describe('create issue', () => { describe('create issue', () => {
beforeEach(() => beforeEach(() => createWrapper(getVulnerability({ shouldShowCreateIssueButton: true })));
createWrapper({ finding: getFinding({ shouldShowCreateIssueButton: true }) }),
);
it('does display if there is only one action and not an issue already created', () => { it('does display if there is only one action and not an issue already created', () => {
expect(findGlDeprecatedButton().exists()).toBe(true); expect(findGlDeprecatedButton().exists()).toBe(true);
...@@ -197,22 +186,21 @@ describe('Vulnerability Header', () => { ...@@ -197,22 +186,21 @@ describe('Vulnerability Header', () => {
it('calls create issue endpoint on click and redirects to new issue', () => { it('calls create issue endpoint on click and redirects to new issue', () => {
const issueUrl = '/group/project/issues/123'; const issueUrl = '/group/project/issues/123';
const spy = jest.spyOn(urlUtility, 'redirectTo'); const spy = jest.spyOn(urlUtility, 'redirectTo');
mockAxios.onPost(dataset.createIssueUrl).reply(200, { mockAxios.onPost(defaultVulnerability.create_issue_url).reply(200, {
issue_url: issueUrl, issue_url: issueUrl,
}); });
findGlDeprecatedButton().vm.$emit('click'); findGlDeprecatedButton().vm.$emit('click');
return waitForPromises().then(() => { return waitForPromises().then(() => {
expect(mockAxios.history.post).toHaveLength(1); expect(mockAxios.history.post).toHaveLength(1);
const [postRequest] = mockAxios.history.post; const [postRequest] = mockAxios.history.post;
expect(postRequest.url).toBe(dataset.createIssueUrl); expect(postRequest.url).toBe(defaultVulnerability.create_issue_url);
expect(JSON.parse(postRequest.data)).toMatchObject({ expect(JSON.parse(postRequest.data)).toMatchObject({
vulnerability_feedback: { vulnerability_feedback: {
feedback_type: FEEDBACK_TYPES.ISSUE, feedback_type: FEEDBACK_TYPES.ISSUE,
category: defaultVulnerability.report_type, category: defaultVulnerability.report_type,
project_fingerprint: dataset.projectFingerprint, project_fingerprint: defaultVulnerability.project_fingerprint,
vulnerability_data: { vulnerability_data: {
...defaultVulnerability, ...getVulnerability({ shouldShowCreateIssueButton: true }),
...getFinding({ shouldShowCreateIssueButton: true }),
category: defaultVulnerability.report_type, category: defaultVulnerability.report_type,
vulnerability_id: defaultVulnerability.id, vulnerability_id: defaultVulnerability.id,
}, },
...@@ -223,7 +211,7 @@ describe('Vulnerability Header', () => { ...@@ -223,7 +211,7 @@ describe('Vulnerability Header', () => {
}); });
it('shows an error message when issue creation fails', () => { it('shows an error message when issue creation fails', () => {
mockAxios.onPost(dataset.createIssueUrl).reply(500); mockAxios.onPost(defaultVulnerability.create_issue_url).reply(500);
findGlDeprecatedButton().vm.$emit('click'); findGlDeprecatedButton().vm.$emit('click');
return waitForPromises().then(() => { return waitForPromises().then(() => {
expect(mockAxios.history.post).toHaveLength(1); expect(mockAxios.history.post).toHaveLength(1);
...@@ -237,8 +225,8 @@ describe('Vulnerability Header', () => { ...@@ -237,8 +225,8 @@ describe('Vulnerability Header', () => {
describe('create merge request', () => { describe('create merge request', () => {
beforeEach(() => { beforeEach(() => {
createWrapper({ createWrapper({
vulnerability: { state: 'resolved' }, ...getVulnerability({ shouldShowMergeRequestButton: true }),
finding: getFinding({ shouldShowMergeRequestButton: true }), state: 'resolved',
}); });
}); });
...@@ -250,22 +238,21 @@ describe('Vulnerability Header', () => { ...@@ -250,22 +238,21 @@ describe('Vulnerability Header', () => {
it('emits createMergeRequest when create merge request button is clicked', () => { it('emits createMergeRequest when create merge request button is clicked', () => {
const mergeRequestPath = '/group/project/merge_request/123'; const mergeRequestPath = '/group/project/merge_request/123';
const spy = jest.spyOn(urlUtility, 'redirectTo'); const spy = jest.spyOn(urlUtility, 'redirectTo');
mockAxios.onPost(dataset.createMRUrl).reply(200, { mockAxios.onPost(defaultVulnerability.create_mr_url).reply(200, {
merge_request_path: mergeRequestPath, merge_request_path: mergeRequestPath,
}); });
findGlDeprecatedButton().vm.$emit('click'); findGlDeprecatedButton().vm.$emit('click');
return waitForPromises().then(() => { return waitForPromises().then(() => {
expect(mockAxios.history.post).toHaveLength(1); expect(mockAxios.history.post).toHaveLength(1);
const [postRequest] = mockAxios.history.post; const [postRequest] = mockAxios.history.post;
expect(postRequest.url).toBe(dataset.createMrUrl); expect(postRequest.url).toBe(defaultVulnerability.create_mr_url);
expect(JSON.parse(postRequest.data)).toMatchObject({ expect(JSON.parse(postRequest.data)).toMatchObject({
vulnerability_feedback: { vulnerability_feedback: {
feedback_type: FEEDBACK_TYPES.MERGE_REQUEST, feedback_type: FEEDBACK_TYPES.MERGE_REQUEST,
category: defaultVulnerability.report_type, category: defaultVulnerability.report_type,
project_fingerprint: dataset.projectFingerprint, project_fingerprint: defaultVulnerability.project_fingerprint,
vulnerability_data: { vulnerability_data: {
...defaultVulnerability, ...getVulnerability({ shouldShowMergeRequestButton: true }),
...getFinding({ shouldShowMergeRequestButton: true }),
category: defaultVulnerability.report_type, category: defaultVulnerability.report_type,
state: 'resolved', state: 'resolved',
}, },
...@@ -276,7 +263,7 @@ describe('Vulnerability Header', () => { ...@@ -276,7 +263,7 @@ describe('Vulnerability Header', () => {
}); });
it('shows an error message when merge request creation fails', () => { it('shows an error message when merge request creation fails', () => {
mockAxios.onPost(dataset.createMRUrl).reply(500); mockAxios.onPost(defaultVulnerability.create_mr_url).reply(500);
findGlDeprecatedButton().vm.$emit('click'); findGlDeprecatedButton().vm.$emit('click');
return waitForPromises().then(() => { return waitForPromises().then(() => {
expect(mockAxios.history.post).toHaveLength(1); expect(mockAxios.history.post).toHaveLength(1);
...@@ -290,8 +277,8 @@ describe('Vulnerability Header', () => { ...@@ -290,8 +277,8 @@ describe('Vulnerability Header', () => {
describe('can download patch', () => { describe('can download patch', () => {
beforeEach(() => { beforeEach(() => {
createWrapper({ createWrapper({
finding: getFinding({ shouldShowMergeRequestButton: true }), ...getVulnerability({ shouldShowMergeRequestButton: true }),
props: { createMrUrl: '' }, create_mr_url: '',
}); });
}); });
...@@ -314,7 +301,7 @@ describe('Vulnerability Header', () => { ...@@ -314,7 +301,7 @@ describe('Vulnerability Header', () => {
test.each(vulnerabilityStateEntries)( test.each(vulnerabilityStateEntries)(
'the vulnerability state badge has the correct style for the %s state', 'the vulnerability state badge has the correct style for the %s state',
(state, stateObject) => { (state, stateObject) => {
createWrapper({ vulnerability: { state } }); createWrapper({ state });
expect(findBadge().classes()).toContain(`status-box-${stateObject.statusBoxStyle}`); expect(findBadge().classes()).toContain(`status-box-${stateObject.statusBoxStyle}`);
expect(findBadge().text()).toBe(state); expect(findBadge().text()).toBe(state);
...@@ -327,16 +314,16 @@ describe('Vulnerability Header', () => { ...@@ -327,16 +314,16 @@ describe('Vulnerability Header', () => {
const user = createRandomUser(); const user = createRandomUser();
const vulnerability = { const vulnerability = {
...defaultVulnerability, ...defaultVulnerability,
...{ state: 'confirmed', confirmed_by_id: user.id }, state: 'confirmed',
confirmed_by_id: user.id,
}; };
createWrapper({ vulnerability }); createWrapper(vulnerability);
return waitForPromises().then(() => { return waitForPromises().then(() => {
expect(findStatusDescription().exists()).toBe(true); expect(findStatusDescription().exists()).toBe(true);
expect(findStatusDescription().props()).toEqual({ expect(findStatusDescription().props()).toEqual({
vulnerability, vulnerability,
pipeline: dataset.pipeline,
user, user,
isLoadingVulnerability: wrapper.vm.isLoadingVulnerability, isLoadingVulnerability: wrapper.vm.isLoadingVulnerability,
isLoadingUser: wrapper.vm.isLoadingUser, isLoadingUser: wrapper.vm.isLoadingUser,
...@@ -350,10 +337,8 @@ describe('Vulnerability Header', () => { ...@@ -350,10 +337,8 @@ describe('Vulnerability Header', () => {
beforeEach(() => { beforeEach(() => {
createWrapper({ createWrapper({
vulnerability: {
resolved_on_default_branch: true, resolved_on_default_branch: true,
project_default_branch: branchName, project_default_branch: branchName,
},
}); });
}); });
...@@ -372,10 +357,8 @@ describe('Vulnerability Header', () => { ...@@ -372,10 +357,8 @@ describe('Vulnerability Header', () => {
describe('when the vulnerability is already resolved', () => { describe('when the vulnerability is already resolved', () => {
beforeEach(() => { beforeEach(() => {
createWrapper({ createWrapper({
vulnerability: {
resolved_on_default_branch: true, resolved_on_default_branch: true,
state: 'resolved', state: 'resolved',
},
}); });
}); });
...@@ -392,7 +375,7 @@ describe('Vulnerability Header', () => { ...@@ -392,7 +375,7 @@ describe('Vulnerability Header', () => {
`loads the correct user for the vulnerability state "%s"`, `loads the correct user for the vulnerability state "%s"`,
state => { state => {
const user = createRandomUser(); const user = createRandomUser();
createWrapper({ vulnerability: { state, [`${state}_by_id`]: user.id } }); createWrapper({ state, [`${state}_by_id`]: user.id });
return waitForPromises().then(() => { return waitForPromises().then(() => {
expect(mockAxios.history.get).toHaveLength(1); expect(mockAxios.history.get).toHaveLength(1);
...@@ -402,7 +385,7 @@ describe('Vulnerability Header', () => { ...@@ -402,7 +385,7 @@ describe('Vulnerability Header', () => {
); );
it('does not load a user if there is no user ID', () => { it('does not load a user if there is no user ID', () => {
createWrapper({ vulnerability: { state: 'detected' } }); createWrapper({ state: 'detected' });
return waitForPromises().then(() => { return waitForPromises().then(() => {
expect(mockAxios.history.get).toHaveLength(0); expect(mockAxios.history.get).toHaveLength(0);
...@@ -411,7 +394,7 @@ describe('Vulnerability Header', () => { ...@@ -411,7 +394,7 @@ describe('Vulnerability Header', () => {
}); });
it('will show an error when the user cannot be loaded', () => { it('will show an error when the user cannot be loaded', () => {
createWrapper({ vulnerability: { state: 'confirmed', confirmed_by_id: 1 } }); createWrapper({ state: 'confirmed', confirmed_by_id: 1 });
mockAxios.onGet().replyOnce(500); mockAxios.onGet().replyOnce(500);
...@@ -423,7 +406,7 @@ describe('Vulnerability Header', () => { ...@@ -423,7 +406,7 @@ describe('Vulnerability Header', () => {
it('will set the isLoadingUser property correctly when the user is loading and finished loading', () => { it('will set the isLoadingUser property correctly when the user is loading and finished loading', () => {
const user = createRandomUser(); const user = createRandomUser();
createWrapper({ vulnerability: { state: 'confirmed', confirmed_by_id: user.id } }); createWrapper({ state: 'confirmed', confirmed_by_id: user.id });
expect(findStatusDescription().props('isLoadingUser')).toBe(true); expect(findStatusDescription().props('isLoadingUser')).toBe(true);
......
...@@ -27,32 +27,30 @@ describe('Vulnerability status description component', () => { ...@@ -27,32 +27,30 @@ describe('Vulnerability status description component', () => {
const createDate = value => (value ? new Date(value) : new Date()).toISOString(); const createDate = value => (value ? new Date(value) : new Date()).toISOString();
const createWrapper = ({ const createWrapper = ({
vulnerability = {}, vulnerability = { pipeline: {} },
pipeline = {},
user, user,
isLoadingVulnerability = false, isLoadingVulnerability = false,
isLoadingUser = false, isLoadingUser = false,
} = {}) => { } = {}) => {
const v = vulnerability; const v = vulnerability;
const p = pipeline;
// Automatically create the ${v.state}_at property if it doesn't exist. Otherwise, every test would need to create // Automatically create the ${v.state}_at property if it doesn't exist. Otherwise, every test would need to create
// it manually for the component to mount properly. // it manually for the component to mount properly.
if (v.state === 'detected') { if (v.state === 'detected') {
p.created_at = p.created_at || createDate(); v.pipeline.created_at = v.pipeline.created_at || createDate();
} else { } else {
const propertyName = `${v.state}_at`; const propertyName = `${v.state}_at`;
v[propertyName] = v[propertyName] || createDate(); v[propertyName] = v[propertyName] || createDate();
} }
wrapper = mount(StatusText, { wrapper = mount(StatusText, {
propsData: { vulnerability, pipeline, user, isLoadingVulnerability, isLoadingUser }, propsData: { vulnerability, user, isLoadingVulnerability, isLoadingUser },
}); });
}; };
describe('state text', () => { describe('state text', () => {
it.each(ALL_STATES)('shows the correct string for the vulnerability state "%s"', state => { it.each(ALL_STATES)('shows the correct string for the vulnerability state "%s"', state => {
createWrapper({ vulnerability: { state } }); createWrapper({ vulnerability: { state, pipeline: {} } });
expect(wrapper.text()).toMatch(new RegExp(`^${capitalize(state)}`)); expect(wrapper.text()).toMatch(new RegExp(`^${capitalize(state)}`));
}); });
...@@ -62,8 +60,7 @@ describe('Vulnerability status description component', () => { ...@@ -62,8 +60,7 @@ describe('Vulnerability status description component', () => {
it('uses the pipeline created date when the vulnerability state is "detected"', () => { it('uses the pipeline created date when the vulnerability state is "detected"', () => {
const pipelineDateString = createDate('2001'); const pipelineDateString = createDate('2001');
createWrapper({ createWrapper({
vulnerability: { state: 'detected' }, vulnerability: { state: 'detected', pipeline: { created_at: pipelineDateString } },
pipeline: { created_at: pipelineDateString },
}); });
expect(timeAgo().props('time')).toBe(pipelineDateString); expect(timeAgo().props('time')).toBe(pipelineDateString);
...@@ -75,8 +72,11 @@ describe('Vulnerability status description component', () => { ...@@ -75,8 +72,11 @@ describe('Vulnerability status description component', () => {
state => { state => {
const expectedDate = createDate(); const expectedDate = createDate();
createWrapper({ createWrapper({
vulnerability: { state, [`${state}_at`]: expectedDate }, vulnerability: {
state,
pipeline: { created_at: 'pipeline_created_at' }, pipeline: { created_at: 'pipeline_created_at' },
[`${state}_at`]: expectedDate,
},
}); });
expect(timeAgo().props('time')).toBe(expectedDate); expect(timeAgo().props('time')).toBe(expectedDate);
...@@ -87,8 +87,7 @@ describe('Vulnerability status description component', () => { ...@@ -87,8 +87,7 @@ describe('Vulnerability status description component', () => {
describe('pipeline link', () => { describe('pipeline link', () => {
it('shows the pipeline link when the vulnerability state is "detected"', () => { it('shows the pipeline link when the vulnerability state is "detected"', () => {
createWrapper({ createWrapper({
vulnerability: { state: 'detected' }, vulnerability: { state: 'detected', pipeline: { url: 'pipeline/url' } },
pipeline: { url: 'pipeline/url' },
}); });
expect(pipelineLink().attributes('href')).toBe('pipeline/url'); expect(pipelineLink().attributes('href')).toBe('pipeline/url');
...@@ -98,8 +97,7 @@ describe('Vulnerability status description component', () => { ...@@ -98,8 +97,7 @@ describe('Vulnerability status description component', () => {
'does not show the pipeline link when the vulnerability state is "%s"', 'does not show the pipeline link when the vulnerability state is "%s"',
state => { state => {
createWrapper({ createWrapper({
vulnerability: { state }, vulnerability: { state, pipeline: { url: 'pipeline/url' } },
pipeline: { url: 'pipeline/url' },
}); });
expect(pipelineLink().exists()).toBe(false); // The user avatar should be shown instead, those tests are handled separately. expect(pipelineLink().exists()).toBe(false); // The user avatar should be shown instead, those tests are handled separately.
...@@ -152,7 +150,7 @@ describe('Vulnerability status description component', () => { ...@@ -152,7 +150,7 @@ describe('Vulnerability status description component', () => {
}); });
it('hides the skeleton loader and shows everything else when the vulnerability is not loading', () => { it('hides the skeleton loader and shows everything else when the vulnerability is not loading', () => {
createWrapper({ vulnerability: { state: 'detected' } }); createWrapper({ vulnerability: { state: 'detected', pipeline: {} } });
expect(skeletonLoader().exists()).toBe(false); expect(skeletonLoader().exists()).toBe(false);
expect(timeAgo().exists()).toBe(true); expect(timeAgo().exists()).toBe(true);
......
...@@ -58,35 +58,31 @@ RSpec.describe VulnerabilitiesHelper do ...@@ -58,35 +58,31 @@ RSpec.describe VulnerabilitiesHelper do
it 'has expected vulnerability properties' do it 'has expected vulnerability properties' do
expect(subject).to include( expect(subject).to include(
vulnerability_json: kind_of(String), timestamp: Time.now.to_i,
project_fingerprint: vulnerability.finding.project_fingerprint,
create_issue_url: "/#{project.full_path}/-/vulnerability_feedback", create_issue_url: "/#{project.full_path}/-/vulnerability_feedback",
notes_url: "/#{project.full_path}/-/security/vulnerabilities/#{vulnerability.id}/notes",
discussions_url: "/#{project.full_path}/-/security/vulnerabilities/#{vulnerability.id}/discussions",
has_mr: anything, has_mr: anything,
vulnerability_feedback_help_path: kind_of(String),
finding_json: kind_of(String),
create_mr_url: "/#{project.full_path}/-/vulnerability_feedback", create_mr_url: "/#{project.full_path}/-/vulnerability_feedback",
timestamp: Time.now.to_i discussions_url: "/#{project.full_path}/-/security/vulnerabilities/#{vulnerability.id}/discussions",
notes_url: "/#{project.full_path}/-/security/vulnerabilities/#{vulnerability.id}/notes",
vulnerability_feedback_help_path: kind_of(String),
pipeline: anything
) )
end end
end end
describe '#vulnerability_data' do describe '#vulnerability_details' do
subject { helper.vulnerability_data(vulnerability, pipeline) } subject { helper.vulnerability_details(vulnerability, pipeline) }
describe 'when pipeline exists' do describe 'when pipeline exists' do
let(:pipeline) { create(:ci_pipeline) } let(:pipeline) { create(:ci_pipeline) }
let(:pipelineData) { Gitlab::Json.parse(subject[:pipeline_json]) }
include_examples 'vulnerability properties' include_examples 'vulnerability properties'
it 'returns expected pipeline data' do it 'returns expected pipeline data' do
expect(pipelineData).to include( expect(subject[:pipeline]).to include(
'id' => pipeline.id, id: pipeline.id,
'created_at' => pipeline.created_at.iso8601, created_at: pipeline.created_at.iso8601,
'url' => be_present, url: be_present
'source_branch' => pipeline.ref
) )
end end
end end
...@@ -110,11 +106,12 @@ RSpec.describe VulnerabilitiesHelper do ...@@ -110,11 +106,12 @@ RSpec.describe VulnerabilitiesHelper do
description: finding.description, description: finding.description,
identifiers: kind_of(Array), identifiers: kind_of(Array),
issue_feedback: anything, issue_feedback: anything,
merge_request_feedback: anything,
links: finding.links, links: finding.links,
location: finding.location, location: finding.location,
name: finding.name, name: finding.name,
merge_request_feedback: anything,
project: kind_of(Grape::Entity::Exposure::NestingExposure::OutputBuilder), project: kind_of(Grape::Entity::Exposure::NestingExposure::OutputBuilder),
project_fingerprint: finding.project_fingerprint,
remediations: finding.remediations, remediations: finding.remediations,
solution: kind_of(String), solution: kind_of(String),
evidence: kind_of(String), evidence: kind_of(String),
......
...@@ -153,6 +153,8 @@ module QA ...@@ -153,6 +153,8 @@ module QA
module Secure module Secure
autoload :Show, 'qa/ee/page/project/secure/show' autoload :Show, 'qa/ee/page/project/secure/show'
autoload :DependencyList, 'qa/ee/page/project/secure/dependency_list' autoload :DependencyList, 'qa/ee/page/project/secure/dependency_list'
autoload :SecurityDashboard, 'qa/ee/page/project/secure/security_dashboard'
autoload :VulnerabilityDetails, 'qa/ee/page/project/secure/vulnerability_details'
end end
module PathLocks module PathLocks
......
...@@ -4,7 +4,7 @@ ...@@ -4,7 +4,7 @@
{ {
"category": "container_scanning", "category": "container_scanning",
"message": "CVE-2017-18269 in glibc", "message": "CVE-2017-18269 in glibc",
"description": "An SSE2-optimized memmove implementation for i386 in sysdeps/i386/i686/multiarch/memcpy-sse2-unaligned.S in the GNU C Library (aka glibc or libc6) 2.21 through 2.27 does not correctly perform the overlapping memory check if the source memory range spans the middle of the address space, resulting in corrupt data being produced by the copy operation. This may disclose information to context-dependent attackers, or result in a denial of service, or, possibly, code execution.", "description": "Short description to match in specs",
"cve": "debian:9:glibc:CVE-2017-18269", "cve": "debian:9:glibc:CVE-2017-18269",
"severity": "Critical", "severity": "Critical",
"confidence": "Unknown", "confidence": "Unknown",
......
# frozen_string_literal: true
module QA
module EE
module Page
module Project
module Secure
class SecurityDashboard < QA::Page::Base
view 'ee/app/assets/javascripts/vulnerabilities/components/vulnerability_list.vue' do
element :vulnerability
end
def has_vulnerability?(description:)
has_element?(:vulnerability, vulnerability_description: description)
end
def click_vulnerability(description:)
return false unless has_vulnerability?(description: description)
click_element(:vulnerability, vulnerability_description: description)
end
end
end
end
end
end
end
# frozen_string_literal: true
module QA
module EE
module Page
module Project
module Secure
class VulnerabilityDetails < QA::Page::Base
view 'ee/app/assets/javascripts/vulnerabilities/components/header.vue' do
element :vulnerability_header
end
view 'ee/app/assets/javascripts/vulnerabilities/components/details.vue' do
element :vulnerability_details
element :vulnerability_title
element :vulnerability_description
end
view 'ee/app/assets/javascripts/vulnerabilities/components/footer.vue' do
element :vulnerability_footer
end
def has_component?(component_name:)
has_element?(component_name.to_sym)
end
def has_vulnerability_title?(title:)
has_element?(:vulnerability_title, text: title)
end
def has_vulnerability_description?(description:)
has_element?(:vulnerability_description, text: description)
end
end
end
end
end
end
end
...@@ -29,6 +29,30 @@ module QA ...@@ -29,6 +29,30 @@ module QA
@add_files = files @add_files = files
end end
def add_directory(dir)
raise "Must set directory as a Pathname" unless dir.is_a?(Pathname)
files_to_add = []
dir.each_child do |child|
case child.ftype?
when "file"
files_to_add.append({
file_path: child.to_s,
content: child.read
})
when "directory"
add_directory(child)
else
continue
end
end
validate_files!(files_to_add)
@add_files.merge(files_to_add)
end
def update_files(files) def update_files(files)
validate_files!(files) validate_files!(files)
......
# frozen_string_literal: true
module QA
context 'Secure', :docker, :runner do
describe 'Security Dashboard in a Project' do
let(:vulnerability_name) { "CVE-2017-18269 in glibc" }
let(:vulnerability_description) { "Short description to match in specs" }
before(:all) do
@executor = "qa-runner-#{Time.now.to_i}"
Flow::Login.sign_in
@project = Resource::Project.fabricate_via_api! do |p|
p.name = Runtime::Env.auto_devops_project_name || 'project-with-secure'
p.description = 'Project with Secure'
p.auto_devops_enabled = false
p.initialize_with_readme = true
end
@runner = Resource::Runner.fabricate! do |runner|
runner.project = @project
runner.name = @executor
runner.tags = %w[qa test]
end
# Push fixture to generate Secure reports
@source = Resource::Repository::ProjectPush.fabricate! do |push|
push.project = @project
push.directory = Pathname
.new(__dir__)
.join('../../../../../ee/fixtures/secure_premade_reports')
push.commit_message = 'Create Secure compatible application to serve premade reports'
push.branch_name = 'secure-mr'
end
@merge_request = Resource::MergeRequest.fabricate_via_api! do |mr|
mr.project = @project
mr.source_branch = 'secure-mr'
mr.target_branch = 'master'
mr.source = @source
mr.target = 'master'
mr.target_new_branch = false
end
@merge_request.visit!
Page::MergeRequest::Show.perform do |merge_request|
merge_request.merge!
end
Page::Project::Menu.perform(&:click_ci_cd_pipelines)
Page::Project::Pipeline::Index.perform(&:wait_for_latest_pipeline_success)
end
after(:all) do
@runner.remove_via_api!
end
it 'shows vulnerability details', quarantine: { type: :investigating } do
@project.visit!
Page::Project::Menu.perform(&:click_on_security_dashboard)
EE::Page::Project::Secure::SecurityDashboard.perform do |security_dashboard|
expect(security_dashboard).to have_vulnerability(description: vulnerability_name)
security_dashboard.click_vulnerability(description: vulnerability_name)
end
EE::Page::Project::Secure::VulnerabilityDetails.perform do |vulnerability_details|
expect(vulnerability_details).to have_component(component_name: :vulnerability_header)
expect(vulnerability_details).to have_component(component_name: :vulnerability_details)
expect(vulnerability_details).to have_vulnerability_title(title: vulnerability_name)
expect(vulnerability_details).to have_vulnerability_description(description: vulnerability_description)
expect(vulnerability_details).to have_component(component_name: :vulnerability_footer)
end
end
end
end
end
...@@ -25,8 +25,6 @@ module QA ...@@ -25,8 +25,6 @@ module QA
@project = Resource::Project.fabricate_via_api! do |p| @project = Resource::Project.fabricate_via_api! do |p|
p.name = Runtime::Env.auto_devops_project_name || 'project-with-secure' p.name = Runtime::Env.auto_devops_project_name || 'project-with-secure'
p.description = 'Project with Secure' p.description = 'Project with Secure'
p.auto_devops_enabled = false
p.initialize_with_readme = true
end end
@runner = Resource::Runner.fabricate! do |runner| @runner = Resource::Runner.fabricate! do |runner|
...@@ -35,15 +33,16 @@ module QA ...@@ -35,15 +33,16 @@ module QA
runner.tags = %w[qa test] runner.tags = %w[qa test]
end end
# Push fixture to generate Secure reports commit_message = 'Add premade security reports'
@source = Resource::Repository::ProjectPush.fabricate! do |push| @commit = Resource::Repository::Commit.fabricate_via_api! do |commit|
push.project = @project commit.project = @project
push.directory = Pathname commit.commit_message = commit_message
.new(__dir__) commit.branch = 'secure-mr'
.join('../../../../../ee/fixtures/secure_premade_reports') commit.add_directory(
push.commit_message = 'Create Secure compatible application to serve premade reports' Pathname.new(__dir__).join('../../../../../ee/fixtures/secure_premade_reports')
push.branch_name = 'secure-mr' )
end end
@project.wait_for_push(commit_message)
@merge_request = Resource::MergeRequest.fabricate_via_api! do |mr| @merge_request = Resource::MergeRequest.fabricate_via_api! do |mr|
mr.project = @project mr.project = @project
......
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