Commit 00be2e6c authored by Sam Beckham's avatar Sam Beckham Committed by Fatih Acet

Allows anyone to comment on a dismissal

- Adds actions/mutations for commenting on a dismissed vulnerability in
the group security dashboard.
- Adds the same actions to the security reports
- Displays the comment box on a dismissed comment
- Sets the dismissal modal in the correct state when the comment box is
focused
- Adds snowplow tracking for dismissal comments

Once a vulnerability has been dismissed, anyone can now comment on it to
explain the dismissal reason.
parent f12c208e
...@@ -63,6 +63,10 @@ ...@@ -63,6 +63,10 @@
margin-left: $grid-size; margin-left: $grid-size;
} }
.btn-group .btn + .btn {
margin-left: -1px;
}
@include media-breakpoint-down(xs) { @include media-breakpoint-down(xs) {
flex-direction: column; flex-direction: column;
...@@ -72,6 +76,11 @@ ...@@ -72,6 +76,11 @@
margin-left: 0; margin-left: 0;
margin-top: $grid-size; margin-top: $grid-size;
} }
.btn-group .btn + .btn {
margin-left: -1px;
margin-top: 0;
}
} }
} }
......
...@@ -96,6 +96,7 @@ export default { ...@@ -96,6 +96,7 @@ export default {
}, },
methods: { methods: {
...mapActions('vulnerabilities', [ ...mapActions('vulnerabilities', [
'addDismissalComment',
'closeDismissalCommentBox', 'closeDismissalCommentBox',
'createIssue', 'createIssue',
'createMergeRequest', 'createMergeRequest',
...@@ -136,6 +137,7 @@ export default { ...@@ -136,6 +137,7 @@ export default {
:can-create-issue="canCreateIssue" :can-create-issue="canCreateIssue"
:can-create-merge-request="canCreateMergeRequest" :can-create-merge-request="canCreateMergeRequest"
:can-dismiss-vulnerability="canDismissVulnerability" :can-dismiss-vulnerability="canDismissVulnerability"
@addDismissalComment="addDismissalComment({ vulnerability, comment: $event })"
@closeDismissalCommentBox="closeDismissalCommentBox()" @closeDismissalCommentBox="closeDismissalCommentBox()"
@createMergeRequest="createMergeRequest({ vulnerability })" @createMergeRequest="createMergeRequest({ vulnerability })"
@createNewIssue="createIssue({ vulnerability })" @createNewIssue="createIssue({ vulnerability })"
......
...@@ -6,6 +6,15 @@ import { parseIntPagination, normalizeHeaders } from '~/lib/utils/common_utils'; ...@@ -6,6 +6,15 @@ import { parseIntPagination, normalizeHeaders } from '~/lib/utils/common_utils';
import { s__ } from '~/locale'; import { s__ } from '~/locale';
import createFlash from '~/flash'; import createFlash from '~/flash';
/**
* A lot of this file has duplicate actions in
* ee/app/assets/javascripts/vue_shared/security_reports/store/actions.js
* This is being addressed in the following issues:
*
* https://gitlab.com/gitlab-org/gitlab-ee/issues/8146
* https://gitlab.com/gitlab-org/gitlab-ee/issues/8519
*/
const hideModal = () => $('#modal-mrwidget-security-issue').modal('hide'); const hideModal = () => $('#modal-mrwidget-security-issue').modal('hide');
export const setVulnerabilitiesEndpoint = ({ commit }, endpoint) => { export const setVulnerabilitiesEndpoint = ({ commit }, endpoint) => {
...@@ -182,6 +191,41 @@ export const receiveDismissVulnerabilityError = ({ commit }, { flashError }) => ...@@ -182,6 +191,41 @@ export const receiveDismissVulnerabilityError = ({ commit }, { flashError }) =>
} }
}; };
export const addDismissalComment = ({ dispatch }, { vulnerability, comment }) => {
dispatch('requestAddDismissalComment');
const { dismissal_feedback } = vulnerability;
const url = `${vulnerability.create_vulnerability_feedback_dismissal_path}/${dismissal_feedback.id}`;
axios
.patch(url, {
project_id: dismissal_feedback.project_id,
id: dismissal_feedback.id,
comment,
})
.then(({ data }) => {
const { id } = vulnerability;
dispatch('closeDismissalCommentBox');
dispatch('receiveAddDismissalCommentSuccess', { id, data });
})
.catch(() => {
dispatch('receiveAddDismissalCommentError');
});
};
export const requestAddDismissalComment = ({ commit }) => {
commit(types.REQUEST_ADD_DISMISSAL_COMMENT);
};
export const receiveAddDismissalCommentSuccess = ({ commit }, payload) => {
commit(types.RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS, payload);
hideModal();
};
export const receiveAddDismissalCommentError = ({ commit }) => {
commit(types.RECEIVE_ADD_DISMISSAL_COMMENT_ERROR);
};
export const undoDismiss = ({ dispatch }, { vulnerability, flashError }) => { export const undoDismiss = ({ dispatch }, { vulnerability, flashError }) => {
const { destroy_vulnerability_feedback_dismissal_path } = vulnerability.dismissal_feedback; const { destroy_vulnerability_feedback_dismissal_path } = vulnerability.dismissal_feedback;
......
...@@ -25,6 +25,10 @@ export const REQUEST_DISMISS_VULNERABILITY = 'REQUEST_DISMISS_VULNERABILITY'; ...@@ -25,6 +25,10 @@ export const REQUEST_DISMISS_VULNERABILITY = 'REQUEST_DISMISS_VULNERABILITY';
export const RECEIVE_DISMISS_VULNERABILITY_SUCCESS = 'RECEIVE_DISMISS_VULNERABILITY_SUCCESS'; export const RECEIVE_DISMISS_VULNERABILITY_SUCCESS = 'RECEIVE_DISMISS_VULNERABILITY_SUCCESS';
export const RECEIVE_DISMISS_VULNERABILITY_ERROR = 'RECEIVE_DISMISS_VULNERABILITY_ERROR'; export const RECEIVE_DISMISS_VULNERABILITY_ERROR = 'RECEIVE_DISMISS_VULNERABILITY_ERROR';
export const REQUEST_ADD_DISMISSAL_COMMENT = 'REQUEST_ADD_DISMISSAL_COMMENT';
export const RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS = 'RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS';
export const RECEIVE_ADD_DISMISSAL_COMMENT_ERROR = 'RECEIVE_ADD_DISMISSAL_COMMENT_ERROR';
export const REQUEST_REVERT_DISMISSAL = 'REQUEST_REVERT_DISMISSAL'; export const REQUEST_REVERT_DISMISSAL = 'REQUEST_REVERT_DISMISSAL';
export const RECEIVE_REVERT_DISMISSAL_SUCCESS = 'RECEIVE_REVERT_DISMISSAL_SUCCESS'; export const RECEIVE_REVERT_DISMISSAL_SUCCESS = 'RECEIVE_REVERT_DISMISSAL_SUCCESS';
export const RECEIVE_REVERT_DISMISSAL_ERROR = 'RECEIVE_REVERT_DISMISSAL_ERROR'; export const RECEIVE_REVERT_DISMISSAL_ERROR = 'RECEIVE_REVERT_DISMISSAL_ERROR';
......
...@@ -179,6 +179,26 @@ export default { ...@@ -179,6 +179,26 @@ export default {
s__('Security Reports|There was an error dismissing the vulnerability.'), s__('Security Reports|There was an error dismissing the vulnerability.'),
); );
}, },
[types.REQUEST_ADD_DISMISSAL_COMMENT](state) {
state.isDismissingVulnerability = true;
Vue.set(state.modal, 'isDismissingVulnerability', true);
Vue.set(state.modal, 'error', null);
},
[types.RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS](state, payload) {
const vulnerability = state.vulnerabilities.find(vuln => vuln.id === payload.id);
if (!vulnerability) {
return;
}
vulnerability.dismissal_feedback = payload.data;
state.isDismissingVulnerability = false;
Vue.set(state.modal, 'isDismissingVulnerability', false);
Vue.set(state.modal.vulnerability, 'isDismissed', true);
},
[types.RECEIVE_ADD_DISMISSAL_COMMENT_ERROR](state) {
state.isDismissingVulnerability = false;
Vue.set(state.modal, 'isDismissingVulnerability', false);
Vue.set(state.modal, 'error', s__('Security Reports|There was an error adding the comment.'));
},
[types.REQUEST_REVERT_DISMISSAL](state) { [types.REQUEST_REVERT_DISMISSAL](state) {
state.isDismissingVulnerability = true; state.isDismissingVulnerability = true;
Vue.set(state.modal, 'isDismissingVulnerability', true); Vue.set(state.modal, 'isDismissingVulnerability', true);
......
...@@ -50,7 +50,7 @@ export default { ...@@ -50,7 +50,7 @@ export default {
:loading="isDismissing" :loading="isDismissing"
:disabled="isDismissing" :disabled="isDismissing"
:label="buttonText" :label="buttonText"
container-class="js-dismiss-btn btn btn-close m-0" container-class="js-dismiss-btn btn btn-close"
@click="handleDismissClick" @click="handleDismissClick"
/> />
<gl-button <gl-button
...@@ -58,7 +58,8 @@ export default { ...@@ -58,7 +58,8 @@ export default {
v-gl-tooltip.hover v-gl-tooltip.hover
v-gl-tooltip.focus v-gl-tooltip.focus
:title="s__('vulnerability|Add comment & dismiss')" :title="s__('vulnerability|Add comment & dismiss')"
class="js-dismiss-with-comment btn-close m-0" variant="close"
class="js-dismiss-with-comment "
@click="$emit('openDismissalCommentBox')" @click="$emit('openDismissalCommentBox')"
> >
<icon name="comment" /> <icon name="comment" />
......
...@@ -3,7 +3,6 @@ ...@@ -3,7 +3,6 @@
// It's dangerous to go alone! take this // It's dangerous to go alone! take this
// https://zaengle.com/blog/using-v-model-on-nested-vue-components // https://zaengle.com/blog/using-v-model-on-nested-vue-components
import { s__ } from '~/locale';
import { GlFormTextarea } from '@gitlab/ui'; import { GlFormTextarea } from '@gitlab/ui';
export default { export default {
...@@ -12,6 +11,11 @@ export default { ...@@ -12,6 +11,11 @@ export default {
GlFormTextarea, GlFormTextarea,
}, },
props: { props: {
placeholder: {
type: String,
required: false,
default: '',
},
value: { value: {
type: String, type: String,
required: false, required: false,
...@@ -23,9 +27,6 @@ export default { ...@@ -23,9 +27,6 @@ export default {
default: '', default: '',
}, },
}, },
data: () => ({
placeholder: s__('vulnerability|Add a comment or reason for dismissal'),
}),
computed: { computed: {
localComment: { localComment: {
get() { get() {
...@@ -59,10 +60,10 @@ export default { ...@@ -59,10 +60,10 @@ export default {
<template> <template>
<div> <div>
<hr />
<gl-form-textarea <gl-form-textarea
ref="dismissalComment" ref="dismissalComment"
v-model="localComment" v-model="localComment"
rows="3"
:state="textAreaState" :state="textAreaState"
:placeholder="placeholder" :placeholder="placeholder"
@keydown.native="handleKeyPress" @keydown.native="handleKeyPress"
......
<script>
// Nested `v-model`s in custom components are weird.
// It's dangerous to go alone! take this
// https://zaengle.com/blog/using-v-model-on-nested-vue-components
import { GlFormTextarea } from '@gitlab/ui';
import { s__ } from '~/locale';
import DismissalCommentBox from 'ee/vue_shared/security_reports/components/dismissal_comment_box.vue';
const PLACEHOLDER = s__('vulnerability|Add a comment or reason for dismissal');
export default {
name: 'DismissalCommentBoxToggle',
components: {
DismissalCommentBox,
GlFormTextarea,
},
props: {
value: {
type: String,
required: false,
default: '',
},
errorMessage: {
type: String,
required: false,
default: '',
},
isActive: {
type: Boolean,
required: false,
default: false,
},
},
PLACEHOLDER,
computed: {
localComment: {
get() {
return this.value;
},
set(localComment) {
this.$emit('input', localComment);
},
},
},
};
</script>
<template>
<div>
<hr class="my-3" />
<dismissal-comment-box
v-if="isActive"
v-model="localComment"
:error-message="errorMessage"
:placeholder="$options.PLACEHOLDER"
@submit="$emit('submit')"
@clearError="$emit('clearError')"
/>
<gl-form-textarea
v-else
:placeholder="$options.PLACEHOLDER"
class="bg-gray-light js-comment-placeholder"
@focus.native="$emit('openDismissalCommentBox')"
/>
</div>
</template>
<script> <script>
import { GlButton } from '@gitlab/ui'; import { GlButton } from '@gitlab/ui';
import Stats from 'ee/stats';
import { s__ } from '~/locale';
import LoadingButton from '~/vue_shared/components/loading_button.vue'; import LoadingButton from '~/vue_shared/components/loading_button.vue';
export default { export default {
...@@ -14,22 +16,52 @@ export default { ...@@ -14,22 +16,52 @@ export default {
required: false, required: false,
default: false, default: false,
}, },
isDismissed: {
type: Boolean,
required: false,
default: false,
},
},
computed: {
submitLabel() {
return this.isDismissed
? s__('vulnerability|Add comment')
: s__('vulnerability|Add comment & dismiss');
},
},
methods: {
addCommentAndDismiss() {
Stats.trackEvent(document.body.dataset.page, 'click_add_comment_and_dismiss');
this.$emit('addCommentAndDismiss');
},
addDismissalComment() {
Stats.trackEvent(document.body.dataset.page, 'click_add_comment');
this.$emit('addDismissalComment');
},
handleSubmit() {
if (this.isDismissed) {
this.addDismissalComment();
} else {
this.addCommentAndDismiss();
}
},
}, },
}; };
</script> </script>
<template> <template>
<div> <div>
<gl-button @click="$emit('cancel')"> <gl-button class="js-cancel" @click="$emit('cancel')">
{{ __('Cancel') }} {{ __('Cancel') }}
</gl-button> </gl-button>
<loading-button <loading-button
:loading="isDismissingVulnerability" :loading="isDismissingVulnerability"
:disabled="isDismissingVulnerability" :disabled="isDismissingVulnerability"
:label="s__('vulnerability|Add comment & dismiss')" :label="submitLabel"
class="js-loading-button"
container-class="btn btn-close" container-class="btn btn-close"
@click="$emit('dismissVulnerability')" @click="handleSubmit"
/> />
</div> </div>
</template> </template>
...@@ -4,7 +4,7 @@ import Modal from '~/vue_shared/components/gl_modal.vue'; ...@@ -4,7 +4,7 @@ import Modal from '~/vue_shared/components/gl_modal.vue';
import ExpandButton from '~/vue_shared/components/expand_button.vue'; import ExpandButton from '~/vue_shared/components/expand_button.vue';
import DismissalNote from 'ee/vue_shared/security_reports/components/dismissal_note.vue'; import DismissalNote from 'ee/vue_shared/security_reports/components/dismissal_note.vue';
import DismissalCommentBox from 'ee/vue_shared/security_reports/components/dismissal_comment_box.vue'; import DismissalCommentBoxToggle from 'ee/vue_shared/security_reports/components/dismissal_comment_box_toggle.vue';
import DismissalCommentModalFooter from 'ee/vue_shared/security_reports/components/dismissal_comment_modal_footer.vue'; import DismissalCommentModalFooter from 'ee/vue_shared/security_reports/components/dismissal_comment_modal_footer.vue';
import EventItem from 'ee/vue_shared/security_reports/components/event_item.vue'; import EventItem from 'ee/vue_shared/security_reports/components/event_item.vue';
import IssueNote from 'ee/vue_shared/security_reports/components/issue_note.vue'; import IssueNote from 'ee/vue_shared/security_reports/components/issue_note.vue';
...@@ -16,7 +16,7 @@ import VulnerabilityDetails from 'ee/vue_shared/security_reports/components/vuln ...@@ -16,7 +16,7 @@ import VulnerabilityDetails from 'ee/vue_shared/security_reports/components/vuln
export default { export default {
components: { components: {
DismissalNote, DismissalNote,
DismissalCommentBox, DismissalCommentBoxToggle,
DismissalCommentModalFooter, DismissalCommentModalFooter,
EventItem, EventItem,
ExpandButton, ExpandButton,
...@@ -88,9 +88,9 @@ export default { ...@@ -88,9 +88,9 @@ export default {
this.vulnerability && this.vulnerability.remediations && this.vulnerability.remediations[0] this.vulnerability && this.vulnerability.remediations && this.vulnerability.remediations[0]
); );
}, },
/** renderSolutionCard() {
* The slot for the footer should be rendered if any of the conditions is true. return this.solution || this.remediation;
*/ },
shouldRenderFooterSection() { shouldRenderFooterSection() {
return !this.modal.isResolved && (this.canCreateIssue || this.canDismissVulnerability); return !this.modal.isResolved && (this.canCreateIssue || this.canDismissVulnerability);
}, },
...@@ -155,13 +155,30 @@ export default { ...@@ -155,13 +155,30 @@ export default {
}, },
}, },
methods: { methods: {
dismissVulnerabilityWithComment() { handleDismissalCommentSubmission() {
if (this.dismissalFeedback) {
this.addDismissalComment();
} else {
this.addCommentAndDismiss();
}
},
addCommentAndDismiss() {
if (this.localDismissalComment.length) { if (this.localDismissalComment.length) {
this.$emit('dismissVulnerability', this.localDismissalComment); this.$emit('dismissVulnerability', this.localDismissalComment);
} else { } else {
this.dismissalCommentErrorMessage = __('Please add a comment in the text area above'); this.addDismissalError();
} }
}, },
addDismissalComment() {
if (this.localDismissalComment.length) {
this.$emit('addDismissalComment', this.localDismissalComment);
} else {
this.addDismissalError();
}
},
addDismissalError() {
this.dismissalCommentErrorMessage = __('Please add a comment in the text area above');
},
clearDismissalError() { clearDismissalError() {
this.dismissalCommentErrorMessage = ''; this.dismissalCommentErrorMessage = '';
}, },
...@@ -203,11 +220,13 @@ export default { ...@@ -203,11 +220,13 @@ export default {
<div v-if="dismissalFeedback || modal.isCommentingOnDismissal" class="card my-4"> <div v-if="dismissalFeedback || modal.isCommentingOnDismissal" class="card my-4">
<div class="card-body"> <div class="card-body">
<dismissal-note :feedback="dismissalFeedbackObject" :project="project" /> <dismissal-note :feedback="dismissalFeedbackObject" :project="project" />
<dismissal-comment-box <dismissal-comment-box-toggle
v-if="modal.isCommentingOnDismissal" v-if="!dismissalFeedback || !dismissalFeedback.comment_details"
v-model="localDismissalComment" v-model="localDismissalComment"
:is-active="modal.isCommentingOnDismissal"
:error-message="dismissalCommentErrorMessage" :error-message="dismissalCommentErrorMessage"
@submit="dismissVulnerabilityWithComment" @openDismissalCommentBox="$emit('openDismissalCommentBox')"
@submit="handleDismissalCommentSubmission"
@clearError="clearDismissalError" @clearError="clearDismissalError"
/> />
</div> </div>
...@@ -218,7 +237,9 @@ export default { ...@@ -218,7 +237,9 @@ export default {
<div slot="footer"> <div slot="footer">
<dismissal-comment-modal-footer <dismissal-comment-modal-footer
v-if="modal.isCommentingOnDismissal" v-if="modal.isCommentingOnDismissal"
@dismissVulnerability="dismissVulnerabilityWithComment" :is-dismissed="vulnerability.isDismissed"
@addCommentAndDismiss="addCommentAndDismiss"
@addDismissalComment="addDismissalComment"
@cancel="$emit('closeDismissalCommentBox')" @cancel="$emit('closeDismissalCommentBox')"
/> />
<modal-footer <modal-footer
......
...@@ -91,7 +91,7 @@ export default { ...@@ -91,7 +91,7 @@ export default {
<dismiss-button <dismiss-button
v-if="canDismissVulnerability" v-if="canDismissVulnerability"
:is-dismissing="modal.isDismissingIssue" :is-dismissing="modal.isDismissingVulnerability"
:is-dismissed="isDismissed" :is-dismissed="isDismissed"
@dismissVulnerability="$emit('dismissVulnerability')" @dismissVulnerability="$emit('dismissVulnerability')"
@openDismissalCommentBox="$emit('openDismissalCommentBox')" @openDismissalCommentBox="$emit('openDismissalCommentBox')"
......
...@@ -257,6 +257,7 @@ export default { ...@@ -257,6 +257,7 @@ export default {
'openDismissalCommentBox', 'openDismissalCommentBox',
'closeDismissalCommentBox', 'closeDismissalCommentBox',
'downloadPatch', 'downloadPatch',
'addDismissalComment',
]), ]),
}, },
}; };
...@@ -363,6 +364,7 @@ export default { ...@@ -363,6 +364,7 @@ export default {
@openDismissalCommentBox="openDismissalCommentBox()" @openDismissalCommentBox="openDismissalCommentBox()"
@revertDismissVulnerability="revertDismissVulnerability" @revertDismissVulnerability="revertDismissVulnerability"
@downloadPatch="downloadPatch" @downloadPatch="downloadPatch"
@addDismissalComment="addDismissalComment({ comment: $event })"
/> />
</div> </div>
</report-section> </report-section>
......
...@@ -237,6 +237,7 @@ export default { ...@@ -237,6 +237,7 @@ export default {
'openDismissalCommentBox', 'openDismissalCommentBox',
'closeDismissalCommentBox', 'closeDismissalCommentBox',
'downloadPatch', 'downloadPatch',
'addDismissalComment',
]), ]),
summaryTextBuilder(reportType, issuesCount = 0) { summaryTextBuilder(reportType, issuesCount = 0) {
if (issuesCount === 0) { if (issuesCount === 0) {
...@@ -327,6 +328,7 @@ export default { ...@@ -327,6 +328,7 @@ export default {
@openDismissalCommentBox="openDismissalCommentBox()" @openDismissalCommentBox="openDismissalCommentBox()"
@revertDismissVulnerability="revertDismissVulnerability" @revertDismissVulnerability="revertDismissVulnerability"
@downloadPatch="downloadPatch" @downloadPatch="downloadPatch"
@addDismissalComment="addDismissalComment({ comment: $event })"
/> />
</div> </div>
</template> </template>
...@@ -5,6 +5,17 @@ import { visitUrl } from '~/lib/utils/url_utility'; ...@@ -5,6 +5,17 @@ import { visitUrl } from '~/lib/utils/url_utility';
import * as types from './mutation_types'; import * as types from './mutation_types';
import downloadPatchHelper from './utils/download_patch_helper'; import downloadPatchHelper from './utils/download_patch_helper';
/**
* A lot of this file has duplicate actions to
* ee/app/assets/javascripts/security_dashboard/store/modules/vulnerabilities/actions.js
* This is being addressed in the following issues:
*
* https://gitlab.com/gitlab-org/gitlab-ee/issues/8146
* https://gitlab.com/gitlab-org/gitlab-ee/issues/8519
*/
const hideModal = () => $('#modal-mrwidget-security-issue').modal('hide');
export const setHeadBlobPath = ({ commit }, blobPath) => commit(types.SET_HEAD_BLOB_PATH, blobPath); export const setHeadBlobPath = ({ commit }, blobPath) => commit(types.SET_HEAD_BLOB_PATH, blobPath);
export const setBaseBlobPath = ({ commit }, blobPath) => commit(types.SET_BASE_BLOB_PATH, blobPath); export const setBaseBlobPath = ({ commit }, blobPath) => commit(types.SET_BASE_BLOB_PATH, blobPath);
...@@ -267,7 +278,7 @@ export const dismissVulnerability = ({ state, dispatch }, comment) => { ...@@ -267,7 +278,7 @@ export const dismissVulnerability = ({ state, dispatch }, comment) => {
default: default:
} }
$('#modal-mrwidget-security-issue').modal('hide'); hideModal();
}) })
.catch(() => { .catch(() => {
dispatch( dispatch(
...@@ -277,6 +288,44 @@ export const dismissVulnerability = ({ state, dispatch }, comment) => { ...@@ -277,6 +288,44 @@ export const dismissVulnerability = ({ state, dispatch }, comment) => {
}); });
}; };
export const addDismissalComment = ({ state, dispatch }, { comment }) => {
dispatch('requestAddDismissalComment');
const { vulnerability } = state.modal;
const { dismissalFeedback } = vulnerability;
const url = `${state.createVulnerabilityFeedbackDismissalPath}/${dismissalFeedback.id}`;
axios
.patch(url, {
project_id: dismissalFeedback.project_id,
id: dismissalFeedback.id,
comment,
})
.then(({ data }) => {
dispatch('closeDismissalCommentBox');
dispatch('receiveAddDismissalCommentSuccess', { data });
})
.catch(() => {
dispatch(
'receiveAddDismissalCommentError',
s__('Security Reports|There was an error adding the comment.'),
);
});
};
export const requestAddDismissalComment = ({ commit }) => {
commit(types.REQUEST_ADD_DISMISSAL_COMMENT);
};
export const receiveAddDismissalCommentSuccess = ({ commit }, payload) => {
commit(types.RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS, payload);
hideModal();
};
export const receiveAddDismissalCommentError = ({ commit }, error) => {
commit(types.RECEIVE_ADD_DISMISSAL_COMMENT_ERROR, error);
};
export const revertDismissVulnerability = ({ state, dispatch }) => { export const revertDismissVulnerability = ({ state, dispatch }) => {
dispatch('requestDismissVulnerability'); dispatch('requestDismissVulnerability');
...@@ -309,7 +358,7 @@ export const revertDismissVulnerability = ({ state, dispatch }) => { ...@@ -309,7 +358,7 @@ export const revertDismissVulnerability = ({ state, dispatch }) => {
default: default:
} }
$('#modal-mrwidget-security-issue').modal('hide'); hideModal();
}) })
.catch(() => .catch(() =>
dispatch( dispatch(
......
...@@ -46,6 +46,10 @@ export const SET_ISSUE_MODAL_DATA = 'SET_ISSUE_MODAL_DATA'; ...@@ -46,6 +46,10 @@ export const SET_ISSUE_MODAL_DATA = 'SET_ISSUE_MODAL_DATA';
export const REQUEST_DISMISS_VULNERABILITY = 'REQUEST_DISMISS_VULNERABILITY'; export const REQUEST_DISMISS_VULNERABILITY = 'REQUEST_DISMISS_VULNERABILITY';
export const RECEIVE_DISMISS_VULNERABILITY_SUCCESS = 'RECEIVE_DISMISS_VULNERABILITY_SUCCESS'; export const RECEIVE_DISMISS_VULNERABILITY_SUCCESS = 'RECEIVE_DISMISS_VULNERABILITY_SUCCESS';
export const RECEIVE_DISMISS_VULNERABILITY_ERROR = 'RECEIVE_DISMISS_VULNERABILITY_ERROR'; export const RECEIVE_DISMISS_VULNERABILITY_ERROR = 'RECEIVE_DISMISS_VULNERABILITY_ERROR';
export const REQUEST_ADD_DISMISSAL_COMMENT = 'REQUEST_ADD_DISMISSAL_COMMENT';
export const RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS = 'RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS';
export const RECEIVE_ADD_DISMISSAL_COMMENT_ERROR = 'RECEIVE_ADD_DISMISSAL_COMMENT_ERROR';
export const REQUEST_CREATE_ISSUE = 'CREATE_DISMISS_VULNERABILITY'; export const REQUEST_CREATE_ISSUE = 'CREATE_DISMISS_VULNERABILITY';
export const RECEIVE_CREATE_ISSUE_SUCCESS = 'CREATE_DISMISS_VULNERABILITY_SUCCESS'; export const RECEIVE_CREATE_ISSUE_SUCCESS = 'CREATE_DISMISS_VULNERABILITY_SUCCESS';
export const RECEIVE_CREATE_ISSUE_ERROR = 'CREATE_DISMISS_VULNERABILITY_ERROR'; export const RECEIVE_CREATE_ISSUE_ERROR = 'CREATE_DISMISS_VULNERABILITY_ERROR';
......
...@@ -309,13 +309,37 @@ export default { ...@@ -309,13 +309,37 @@ export default {
}, },
[types.REQUEST_DISMISS_VULNERABILITY](state) { [types.REQUEST_DISMISS_VULNERABILITY](state) {
Vue.set(state.modal, 'isDismissingIssue', true); Vue.set(state.modal, 'isDismissingVulnerability', true);
// reset error in case previous state was error // reset error in case previous state was error
Vue.set(state.modal, 'error', null); Vue.set(state.modal, 'error', null);
}, },
[types.RECEIVE_DISMISS_VULNERABILITY_SUCCESS](state) { [types.RECEIVE_DISMISS_VULNERABILITY_SUCCESS](state) {
Vue.set(state.modal, 'isDismissingIssue', false); Vue.set(state.modal, 'isDismissingVulnerability', false);
},
[types.RECEIVE_DISMISS_VULNERABILITY_ERROR](state, error) {
Vue.set(state.modal, 'error', error);
Vue.set(state.modal, 'isDismissingVulnerability', false);
},
[types.REQUEST_ADD_DISMISSAL_COMMENT](state) {
state.isDismissingVulnerability = true;
Vue.set(state.modal, 'isDismissingVulnerability', true);
Vue.set(state.modal, 'error', null);
},
[types.RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS](state, payload) {
state.isDismissingVulnerability = false;
Vue.set(state.modal, 'isDismissingVulnerability', false);
Vue.set(state.modal.vulnerability, 'isDismissed', true);
Vue.set(state.modal.vulnerability, 'dismissalFeedback', payload.data);
},
[types.RECEIVE_ADD_DISMISSAL_COMMENT_ERROR](state, error) {
state.isDismissingVulnerability = false;
Vue.set(state.modal, 'isDismissingVulnerability', false);
Vue.set(state.modal, 'error', error);
}, },
[types.UPDATE_SAST_ISSUE](state, issue) { [types.UPDATE_SAST_ISSUE](state, issue) {
...@@ -390,11 +414,6 @@ export default { ...@@ -390,11 +414,6 @@ export default {
} }
}, },
[types.RECEIVE_DISMISS_VULNERABILITY_ERROR](state, error) {
Vue.set(state.modal, 'error', error);
Vue.set(state.modal, 'isDismissingIssue', false);
},
[types.REQUEST_CREATE_ISSUE](state) { [types.REQUEST_CREATE_ISSUE](state) {
Vue.set(state.modal, 'isCreatingNewIssue', true); Vue.set(state.modal, 'isCreatingNewIssue', true);
// reset error in case previous state was error // reset error in case previous state was error
......
...@@ -139,7 +139,7 @@ export default () => ({ ...@@ -139,7 +139,7 @@ export default () => ({
}, },
isCreatingNewIssue: false, isCreatingNewIssue: false,
isDismissingIssue: false, isDismissingVulnerability: false,
error: null, error: null,
}, },
......
---
title: Allows any user to comment on a dismissed vulnerability
merge_request: 12067
author:
type: added
...@@ -31,4 +31,10 @@ describe('DismissalCommentBox', () => { ...@@ -31,4 +31,10 @@ describe('DismissalCommentBox', () => {
wrapper.setProps({ errorMessage }); wrapper.setProps({ errorMessage });
expect(wrapper.find('.js-error').text()).toBe(errorMessage); expect(wrapper.find('.js-error').text()).toBe(errorMessage);
}); });
it('should render the placeholder', () => {
const placeholder = 'Please type into the box';
wrapper.setProps({ placeholder });
expect(wrapper.find(GlFormTextarea).attributes('placeholder')).toBe(placeholder);
});
}); });
import { mount } from '@vue/test-utils';
import DismissalCommentBox from 'ee/vue_shared/security_reports/components/dismissal_comment_box.vue';
import component from 'ee/vue_shared/security_reports/components/dismissal_comment_box_toggle.vue';
describe('DismissalCommentBox', () => {
let wrapper;
describe('when the box is inactive', () => {
beforeEach(() => {
wrapper = mount(component);
});
it('should render the placeholder text box', () => {
expect(wrapper.find('.js-comment-placeholder').exists()).toBeTruthy();
});
it('should not render the dismissal comment box', () => {
expect(wrapper.find(DismissalCommentBox).exists()).toBeFalsy();
});
});
describe('when the box is active', () => {
beforeEach(() => {
wrapper = mount(component, {
propsData: {
isActive: true,
},
});
});
it('should render the dismissal comment box', () => {
expect(wrapper.find(DismissalCommentBox).exists()).toBeTruthy();
});
it('should not render the placeholder text box', () => {
expect(wrapper.find('.js-comment-placeholder').exists()).toBeFalsy();
});
});
});
import { mount } from '@vue/test-utils';
import Stats from 'ee/stats';
import LoadingButton from '~/vue_shared/components/loading_button.vue';
import component from 'ee/vue_shared/security_reports/components/dismissal_comment_modal_footer.vue';
jest.mock('ee/stats');
describe('DismissalCommentModalFooter', () => {
let wrapper;
describe('with an non-dismissed vulnerability', () => {
beforeEach(() => {
wrapper = mount(component, { sync: false });
});
it('should render the "Add comment and dismiss" button', () => {
expect(wrapper.find(LoadingButton).text()).toBe('Add comment & dismiss');
});
it('should emit the "addCommentAndDismiss" event when clicked', () => {
wrapper.find(LoadingButton).trigger('click');
expect(wrapper.emitted().addCommentAndDismiss).toBeTruthy();
expect(Stats.trackEvent).toHaveBeenCalledWith(
document.body.dataset.page,
'click_add_comment_and_dismiss',
);
});
it('should emit the cancel event when the cancel button is clicked', () => {
wrapper.find('.js-cancel').trigger('click');
expect(wrapper.emitted().cancel).toBeTruthy();
});
});
describe('with an already dismissed vulnerability', () => {
beforeEach(() => {
const propsData = {
isDismissed: true,
};
wrapper = mount(component, { propsData });
});
it('should render the "Add comment and dismiss" button', () => {
expect(wrapper.find(LoadingButton).text()).toBe('Add comment');
});
it('should emit the "addCommentAndDismiss" event when clicked', () => {
wrapper.find(LoadingButton).trigger('click');
expect(wrapper.emitted().addDismissalComment).toBeTruthy();
expect(Stats.trackEvent).toHaveBeenCalledWith(
document.body.dataset.page,
'click_add_comment',
);
});
});
});
...@@ -7,10 +7,6 @@ describe('Security Reports modal', () => { ...@@ -7,10 +7,6 @@ describe('Security Reports modal', () => {
const Component = Vue.extend(component); const Component = Vue.extend(component);
let wrapper; let wrapper;
afterEach(() => {
wrapper.destroy();
});
describe('with permissions', () => { describe('with permissions', () => {
describe('with dismissed issue', () => { describe('with dismissed issue', () => {
beforeEach(() => { beforeEach(() => {
...@@ -31,6 +27,10 @@ describe('Security Reports modal', () => { ...@@ -31,6 +27,10 @@ describe('Security Reports modal', () => {
expect(wrapper.text().trim()).toContain('@jsmith'); expect(wrapper.text().trim()).toContain('@jsmith');
expect(wrapper.text().trim()).toContain('#123'); expect(wrapper.text().trim()).toContain('#123');
}); });
it('renders the dismissal comment placeholder', () => {
expect(wrapper.find('.js-comment-placeholder')).not.toBeNull();
});
}); });
describe('with not dismissed issue', () => { describe('with not dismissed issue', () => {
...@@ -225,4 +225,73 @@ describe('Security Reports modal', () => { ...@@ -225,4 +225,73 @@ describe('Security Reports modal', () => {
expect(wrapper.contains('hr')).toBe(false); expect(wrapper.contains('hr')).toBe(false);
}); });
}); });
describe('add dismissal comment', () => {
const comment = "Pirates don't eat the tourists";
let propsData;
beforeEach(() => {
propsData = {
modal: createState().modal,
};
propsData.modal.isCommentingOnDismissal = true;
});
beforeAll(() => {
// https://github.com/vuejs/vue-test-utils/issues/532#issuecomment-398449786
Vue.config.silent = true;
});
afterAll(() => {
Vue.config.silent = false;
});
describe('with a non-dismissed vulnerability', () => {
beforeEach(() => {
wrapper = mount(Component, { propsData });
});
it('creates an error when an empty comment is submitted', () => {
const { vm } = wrapper;
vm.handleDismissalCommentSubmission();
expect(vm.dismissalCommentErrorMessage).toBe('Please add a comment in the text area above');
});
it('submits the comment and dismisses the vulnerability if text has been entered', () => {
const { vm } = wrapper;
vm.addCommentAndDismiss = jasmine.createSpy();
vm.localDismissalComment = comment;
vm.handleDismissalCommentSubmission();
expect(vm.addCommentAndDismiss).toHaveBeenCalled();
expect(vm.dismissalCommentErrorMessage).toBe('');
});
});
describe('with a dismissed vulnerability', () => {
beforeEach(() => {
propsData.modal.vulnerability.dismissal_feedback = { author: {} };
wrapper = mount(Component, { propsData });
});
it('creates an error when an empty comment is submitted', () => {
const { vm } = wrapper;
vm.handleDismissalCommentSubmission();
expect(vm.dismissalCommentErrorMessage).toBe('Please add a comment in the text area above');
});
it('submits the comment if text is entered and the vulnerability is already dismissed', () => {
const { vm } = wrapper;
vm.addDismissalComment = jasmine.createSpy();
vm.localDismissalComment = comment;
vm.handleDismissalCommentSubmission();
expect(vm.addDismissalComment).toHaveBeenCalled();
expect(vm.dismissalCommentErrorMessage).toBe('');
});
});
});
}); });
...@@ -435,7 +435,7 @@ describe('security reports mutations', () => { ...@@ -435,7 +435,7 @@ describe('security reports mutations', () => {
expect(stateCopy.modal.vulnerability.hasIssue).toEqual(false); expect(stateCopy.modal.vulnerability.hasIssue).toEqual(false);
expect(stateCopy.modal.isCreatingNewIssue).toEqual(false); expect(stateCopy.modal.isCreatingNewIssue).toEqual(false);
expect(stateCopy.modal.isDismissingIssue).toEqual(false); expect(stateCopy.modal.isDismissingVulnerability).toEqual(false);
expect(stateCopy.modal.title).toEqual(null); expect(stateCopy.modal.title).toEqual(null);
expect(stateCopy.modal.learnMoreUrl).toEqual(null); expect(stateCopy.modal.learnMoreUrl).toEqual(null);
...@@ -505,31 +505,94 @@ describe('security reports mutations', () => { ...@@ -505,31 +505,94 @@ describe('security reports mutations', () => {
}); });
describe('REQUEST_DISMISS_VULNERABILITY', () => { describe('REQUEST_DISMISS_VULNERABILITY', () => {
it('sets isDismissingIssue prop to true and resets error', () => { it('sets isDismissingVulnerability prop to true and resets error', () => {
mutations[types.REQUEST_DISMISS_VULNERABILITY](stateCopy); mutations[types.REQUEST_DISMISS_VULNERABILITY](stateCopy);
expect(stateCopy.modal.isDismissingIssue).toEqual(true); expect(stateCopy.modal.isDismissingVulnerability).toEqual(true);
expect(stateCopy.modal.error).toBeNull(); expect(stateCopy.modal.error).toBeNull();
}); });
}); });
describe('RECEIVE_DISMISS_VULNERABILITY_SUCCESS', () => { describe('RECEIVE_DISMISS_VULNERABILITY_SUCCESS', () => {
it('sets isDismissingIssue prop to false', () => { it('sets isDismissingVulnerability prop to false', () => {
mutations[types.RECEIVE_DISMISS_VULNERABILITY_SUCCESS](stateCopy); mutations[types.RECEIVE_DISMISS_VULNERABILITY_SUCCESS](stateCopy);
expect(stateCopy.modal.isDismissingIssue).toEqual(false); expect(stateCopy.modal.isDismissingVulnerability).toEqual(false);
}); });
}); });
describe('RECEIVE_DISMISS_VULNERABILITY_ERROR', () => { describe('RECEIVE_DISMISS_VULNERABILITY_ERROR', () => {
it('sets isDismissingIssue prop to false and sets error', () => { it('sets isDismissingVulnerability prop to false and sets error', () => {
mutations[types.RECEIVE_DISMISS_VULNERABILITY_ERROR](stateCopy, 'error'); mutations[types.RECEIVE_DISMISS_VULNERABILITY_ERROR](stateCopy, 'error');
expect(stateCopy.modal.isDismissingIssue).toEqual(false); expect(stateCopy.modal.isDismissingVulnerability).toEqual(false);
expect(stateCopy.modal.error).toEqual('error'); expect(stateCopy.modal.error).toEqual('error');
}); });
}); });
describe(types.REQUEST_ADD_DISMISSAL_COMMENT, () => {
beforeEach(() => {
mutations[types.REQUEST_ADD_DISMISSAL_COMMENT](stateCopy);
});
it('should set isDismissingVulnerability to true', () => {
expect(stateCopy.isDismissingVulnerability).toBe(true);
});
it('should set isDismissingVulnerability in the modal data to true', () => {
expect(stateCopy.modal.isDismissingVulnerability).toBe(true);
});
it('should nullify the error state on the modal', () => {
expect(stateCopy.modal.error).toBeNull();
});
});
describe(types.RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS, () => {
let payload;
let vulnerability;
let data;
beforeEach(() => {
vulnerability = { id: 1 };
data = { name: 'dismissal feedback' };
payload = { id: vulnerability.id, data };
mutations[types.RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS](stateCopy, payload);
});
it('should set isDismissingVulnerability to false', () => {
expect(stateCopy.isDismissingVulnerability).toBe(false);
});
it('should set isDismissingVulnerability on the modal to false', () => {
expect(stateCopy.modal.isDismissingVulnerability).toBe(false);
});
it('shoulfd set isDissmissed on the modal vulnerability to be true', () => {
expect(stateCopy.modal.vulnerability.isDismissed).toBe(true);
});
});
describe(types.RECEIVE_ADD_DISMISSAL_COMMENT_ERROR, () => {
const error = 'There was an error adding the comment.';
beforeEach(() => {
mutations[types.RECEIVE_ADD_DISMISSAL_COMMENT_ERROR](stateCopy, error);
});
it('should set isDismissingVulnerability to false', () => {
expect(stateCopy.isDismissingVulnerability).toBe(false);
});
it('should set isDismissingVulnerability in the modal data to false', () => {
expect(stateCopy.modal.isDismissingVulnerability).toBe(false);
});
it('should set the error state on the modal', () => {
expect(stateCopy.modal.error).toEqual(error);
});
});
describe('REQUEST_CREATE_ISSUE', () => { describe('REQUEST_CREATE_ISSUE', () => {
it('sets isCreatingNewIssue prop to true and resets error', () => { it('sets isCreatingNewIssue prop to true and resets error', () => {
mutations[types.REQUEST_CREATE_ISSUE](stateCopy); mutations[types.REQUEST_CREATE_ISSUE](stateCopy);
......
...@@ -680,7 +680,7 @@ describe('vulnerability dismissal', () => { ...@@ -680,7 +680,7 @@ describe('vulnerability dismissal', () => {
[], [],
[ [
{ type: 'requestDismissVulnerability' }, { type: 'requestDismissVulnerability' },
{ type: 'receiveDismissVulnerabilityError', payload: { flashError: false } }, { type: 'receiveDismissVulnerabilityError', payload: { flashError } },
], ],
done, done,
); );
...@@ -740,6 +740,115 @@ describe('vulnerability dismissal', () => { ...@@ -740,6 +740,115 @@ describe('vulnerability dismissal', () => {
}); });
}); });
describe('add vulnerability dismissal comment', () => {
describe('addDismissalComment', () => {
const vulnerability = mockDataVulnerabilities[2];
const data = { vulnerability };
const url = `${vulnerability.create_vulnerability_feedback_dismissal_path}/${vulnerability.dismissal_feedback.id}`;
const comment = 'Well, we’re back in the car again.';
let mock;
beforeEach(() => {
mock = new MockAdapter(axios);
});
afterEach(() => {
mock.restore();
});
describe('on success', () => {
beforeEach(() => {
mock.onPatch(url).replyOnce(200, data);
});
it('should dispatch the request and success actions', done => {
const checkPassedData = () => {
const { project_id, id } = vulnerability.dismissal_feedback;
const expected = { project_id, id, comment };
expect(mock.history.patch[0].data).toBe(JSON.stringify(expected));
done();
};
testAction(
actions.addDismissalComment,
{ vulnerability, comment },
{},
[],
[
{ type: 'requestAddDismissalComment' },
{ type: 'closeDismissalCommentBox' },
{ type: 'receiveAddDismissalCommentSuccess', payload: { data, id: vulnerability.id } },
],
checkPassedData,
);
});
});
describe('on error', () => {
beforeEach(() => {
mock.onPatch(url).replyOnce(404);
});
it('should dispatch the request and error actions', done => {
testAction(
actions.addDismissalComment,
{ vulnerability, comment },
{},
[],
[{ type: 'requestAddDismissalComment' }, { type: 'receiveAddDismissalCommentError' }],
done,
);
});
});
describe('receiveAddDismissalCommentSuccess', () => {
it('should commit the success mutation', done => {
const state = initialState;
testAction(
actions.receiveAddDismissalCommentSuccess,
{ data },
state,
[{ type: types.RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS, payload: { data } }],
[],
done,
);
});
});
describe('receiveAddDismissalCommentError', () => {
it('should commit the error mutation', done => {
const state = initialState;
testAction(
actions.receiveAddDismissalCommentError,
{},
state,
[{ type: types.RECEIVE_ADD_DISMISSAL_COMMENT_ERROR }],
[],
done,
);
});
});
describe('requestAddDismissalComment', () => {
it('should commit the request mutation', done => {
const state = initialState;
testAction(
actions.requestAddDismissalComment,
{},
state,
[{ type: types.REQUEST_ADD_DISMISSAL_COMMENT }],
[],
done,
);
});
});
});
});
describe('revert vulnerability dismissal', () => { describe('revert vulnerability dismissal', () => {
describe('undoDismiss', () => { describe('undoDismiss', () => {
const vulnerability = mockDataVulnerabilities[2]; const vulnerability = mockDataVulnerabilities[2];
...@@ -789,7 +898,7 @@ describe('revert vulnerability dismissal', () => { ...@@ -789,7 +898,7 @@ describe('revert vulnerability dismissal', () => {
[], [],
[ [
{ type: 'requestUndoDismiss' }, { type: 'requestUndoDismiss' },
{ type: 'receiveUndoDismissError', payload: { flashError: false } }, { type: 'receiveUndoDismissError', payload: { flashError } },
], ],
done, done,
); );
......
...@@ -542,6 +542,80 @@ describe('vulnerabilities module mutations', () => { ...@@ -542,6 +542,80 @@ describe('vulnerabilities module mutations', () => {
}); });
}); });
describe('REQUEST_DISMISSAL_COMMENT', () => {
let state;
beforeEach(() => {
state = createState();
mutations[types.REQUEST_ADD_DISMISSAL_COMMENT](state);
});
it('should set isDismissingVulnerability to true', () => {
expect(state.isDismissingVulnerability).toBe(true);
});
it('should set isDismissingVulnerability in the modal data to true', () => {
expect(state.modal.isDismissingVulnerability).toBe(true);
});
it('should nullify the error state on the modal', () => {
expect(state.modal.error).toBeNull();
});
});
describe('RECEIVE_DISMISSAL_COMMENT_SUCCESS', () => {
let state;
let payload;
let vulnerability;
let data;
beforeEach(() => {
state = createState();
state.vulnerabilities = mockData;
[vulnerability] = mockData;
data = { name: 'dismissal feedback' };
payload = { id: vulnerability.id, data };
mutations[types.RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS](state, payload);
});
it('should set the dismissal feedback on the passed vulnerability', () => {
expect(state.vulnerabilities[0].dismissal_feedback).toEqual(data);
});
it('should set isDismissingVulnerability to false', () => {
expect(state.isDismissingVulnerability).toBe(false);
});
it('should set isDismissingVulnerability on the modal to false', () => {
expect(state.modal.isDismissingVulnerability).toBe(false);
});
it('should set isDissmissed on the modal vulnerability to be true', () => {
expect(state.modal.vulnerability.isDismissed).toBe(true);
});
});
describe('RECEIVE_DISMISSAL_COMMENT_ERROR', () => {
let state;
beforeEach(() => {
state = createState();
mutations[types.RECEIVE_ADD_DISMISSAL_COMMENT_ERROR](state);
});
it('should set isDismissingVulnerability to false', () => {
expect(state.isDismissingVulnerability).toBe(false);
});
it('should set isDismissingVulnerability in the modal data to false', () => {
expect(state.modal.isDismissingVulnerability).toBe(false);
});
it('should set the error state on the modal', () => {
expect(state.modal.error).toEqual('There was an error adding the comment.');
});
});
describe('REQUEST_REVERT_DISMISSAL', () => { describe('REQUEST_REVERT_DISMISSAL', () => {
let state; let state;
......
...@@ -52,6 +52,10 @@ import actions, { ...@@ -52,6 +52,10 @@ import actions, {
updateDependencyScanningIssue, updateDependencyScanningIssue,
updateContainerScanningIssue, updateContainerScanningIssue,
updateDastIssue, updateDastIssue,
addDismissalComment,
receiveAddDismissalCommentError,
receiveAddDismissalCommentSuccess,
requestAddDismissalComment,
} from 'ee/vue_shared/security_reports/store/actions'; } from 'ee/vue_shared/security_reports/store/actions';
import * as types from 'ee/vue_shared/security_reports/store/mutation_types'; import * as types from 'ee/vue_shared/security_reports/store/mutation_types';
import state from 'ee/vue_shared/security_reports/store/state'; import state from 'ee/vue_shared/security_reports/store/state';
...@@ -1300,6 +1304,108 @@ describe('security reports actions', () => { ...@@ -1300,6 +1304,108 @@ describe('security reports actions', () => {
}); });
}); });
describe('addDismissalComment', () => {
const vulnerability = {
id: 0,
vulnerability_feedback_dismissal_path: 'foo',
dismissalFeedback: { id: 1 },
};
const data = { vulnerability };
const url = `${state.createVulnerabilityFeedbackDismissalPath}/${vulnerability.dismissalFeedback.id}`;
const comment = 'Well, we’re back in the car again.';
describe('on success', () => {
beforeEach(() => {
mock.onPatch(url).replyOnce(200, data);
});
it('should dispatch the request and success actions', done => {
testAction(
addDismissalComment,
{ comment },
{ modal: { vulnerability } },
[],
[
{ type: 'requestAddDismissalComment' },
{ type: 'closeDismissalCommentBox' },
{
type: 'receiveAddDismissalCommentSuccess',
payload: { data },
},
],
done,
);
});
});
describe('on error', () => {
beforeEach(() => {
mock.onPatch(url).replyOnce(404);
});
it('should dispatch the request and error actions', done => {
testAction(
addDismissalComment,
{ comment },
{ modal: { vulnerability } },
[],
[
{ type: 'requestAddDismissalComment' },
{
type: 'receiveAddDismissalCommentError',
payload: 'There was an error adding the comment.',
},
],
done,
);
});
});
describe('receiveAddDismissalCommentSuccess', () => {
it('should commit the success mutation', done => {
testAction(
receiveAddDismissalCommentSuccess,
{ data },
state,
[{ type: types.RECEIVE_ADD_DISMISSAL_COMMENT_SUCCESS, payload: { data } }],
[],
done,
);
});
});
describe('receiveAddDismissalCommentError', () => {
it('should commit the error mutation', done => {
testAction(
receiveAddDismissalCommentError,
{},
state,
[
{
type: types.RECEIVE_ADD_DISMISSAL_COMMENT_ERROR,
payload: {},
},
],
[],
done,
);
});
});
describe('requestAddDismissalComment', () => {
it('should commit the request mutation', done => {
testAction(
requestAddDismissalComment,
{},
state,
[{ type: types.REQUEST_ADD_DISMISSAL_COMMENT }],
[],
done,
);
});
});
});
describe('revertDismissVulnerability', () => { describe('revertDismissVulnerability', () => {
describe('with success', () => { describe('with success', () => {
beforeEach(() => { beforeEach(() => {
......
...@@ -12206,6 +12206,9 @@ msgstr "" ...@@ -12206,6 +12206,9 @@ msgstr ""
msgid "Security Reports|Oops, something doesn't seem right." msgid "Security Reports|Oops, something doesn't seem right."
msgstr "" msgstr ""
msgid "Security Reports|There was an error adding the comment."
msgstr ""
msgid "Security Reports|There was an error creating the issue." msgid "Security Reports|There was an error creating the issue."
msgstr "" msgstr ""
...@@ -17561,6 +17564,9 @@ msgstr "" ...@@ -17561,6 +17564,9 @@ msgstr ""
msgid "vulnerability|Add a comment or reason for dismissal" msgid "vulnerability|Add a comment or reason for dismissal"
msgstr "" msgstr ""
msgid "vulnerability|Add comment"
msgstr ""
msgid "vulnerability|Add comment & dismiss" msgid "vulnerability|Add comment & dismiss"
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