Commit 7bc7d423 authored by Robert Hunt's avatar Robert Hunt

Added the drawer to the new report app

- Added the drawer component and trigger methods
- Updated the drawer component to accept the new data format
- Updated the existing dashboard to pass in the new data format
- Added mapping functions to convert to a unified format
- Added another dummy GraphQL object to have two rows in the table
parent dfa84df8
...@@ -2,7 +2,9 @@ ...@@ -2,7 +2,9 @@
import { GlTabs, GlTab } from '@gitlab/ui'; import { GlTabs, GlTab } from '@gitlab/ui';
import Cookies from 'js-cookie'; import Cookies from 'js-cookie';
import { __ } from '~/locale'; import { __ } from '~/locale';
import { DRAWER_Z_INDEX } from '~/lib/utils/constants';
import { COMPLIANCE_TAB_COOKIE_KEY } from '../constants'; import { COMPLIANCE_TAB_COOKIE_KEY } from '../constants';
import { mapDashboardToDrawerData } from '../utils';
import MergeRequestDrawer from './drawer.vue'; import MergeRequestDrawer from './drawer.vue';
import EmptyState from './empty_state.vue'; import EmptyState from './empty_state.vue';
import MergeRequestsGrid from './merge_requests/grid.vue'; import MergeRequestsGrid from './merge_requests/grid.vue';
...@@ -41,7 +43,8 @@ export default { ...@@ -41,7 +43,8 @@ export default {
data() { data() {
return { return {
showDrawer: false, showDrawer: false,
drawerData: {}, drawerMergeRequest: {},
drawerProject: {},
}; };
}, },
computed: { computed: {
...@@ -51,27 +54,33 @@ export default { ...@@ -51,27 +54,33 @@ export default {
hasMergeCommitsCsvExportPath() { hasMergeCommitsCsvExportPath() {
return this.mergeCommitsCsvExportPath !== ''; return this.mergeCommitsCsvExportPath !== '';
}, },
drawerMergeRequests() {
return this.mergeRequests.map(mapDashboardToDrawerData);
},
}, },
methods: { methods: {
showTabs() { showTabs() {
return Cookies.get(COMPLIANCE_TAB_COOKIE_KEY) === 'true'; return Cookies.get(COMPLIANCE_TAB_COOKIE_KEY) === 'true';
}, },
toggleDrawer(mergeRequest) { toggleDrawer(mergeRequest) {
if (this.showDrawer && mergeRequest.id === this.drawerData.id) { if (this.showDrawer && mergeRequest.id === this.drawerMergeRequest.id) {
this.closeDrawer(); this.closeDrawer();
} else { } else {
this.openDrawer(mergeRequest); this.openDrawer(this.drawerMergeRequests.find((mr) => mr.id === mergeRequest.id));
} }
}, },
openDrawer(mergeRequest) { openDrawer(data) {
this.showDrawer = true; this.showDrawer = true;
this.drawerData = mergeRequest; this.drawerMergeRequest = data.mergeRequest;
this.drawerProject = data.project;
}, },
closeDrawer() { closeDrawer() {
this.showDrawer = false; this.showDrawer = false;
this.drawerData = {}; this.drawerMergeRequest = {};
this.drawerProject = {};
}, },
}, },
DRAWER_Z_INDEX,
strings: { strings: {
heading: __('Compliance report'), heading: __('Compliance report'),
subheading: __('Here you will find recent merge request activity'), subheading: __('Here you will find recent merge request activity'),
...@@ -113,8 +122,9 @@ export default { ...@@ -113,8 +122,9 @@ export default {
/> />
<merge-request-drawer <merge-request-drawer
:show-drawer="showDrawer" :show-drawer="showDrawer"
:merge-request="drawerData" :merge-request="drawerMergeRequest"
z-index="252" :project="drawerProject"
:z-index="$options.DRAWER_Z_INDEX"
@close="closeDrawer" @close="closeDrawer"
/> />
</div> </div>
......
<script> <script>
import { GlDrawer } from '@gitlab/ui'; import { GlDrawer } from '@gitlab/ui';
import { DRAWER_Z_INDEX } from '~/lib/utils/constants'; import { DRAWER_Z_INDEX } from '~/lib/utils/constants';
import { convertArrayOfObjectsToCamelCase } from '~/lib/utils/common_utils';
import { COMPLIANCE_DRAWER_CONTAINER_CLASS } from '../constants'; import { COMPLIANCE_DRAWER_CONTAINER_CLASS } from '../constants';
import { getContentWrapperHeight } from '../../threat_monitoring/utils'; import { getContentWrapperHeight } from '../../threat_monitoring/utils';
import BranchPath from './drawer_sections/branch_path.vue'; import BranchPath from './drawer_sections/branch_path.vue';
...@@ -26,6 +25,10 @@ export default { ...@@ -26,6 +25,10 @@ export default {
type: Object, type: Object,
required: true, required: true,
}, },
project: {
type: Object,
required: true,
},
showDrawer: { showDrawer: {
type: Boolean, type: Boolean,
required: false, required: false,
...@@ -34,20 +37,11 @@ export default { ...@@ -34,20 +37,11 @@ export default {
}, },
computed: { computed: {
hasBranchDetails() { hasBranchDetails() {
return this.mergeRequest.source_branch && this.mergeRequest.target_branch; return this.mergeRequest.sourceBranch && this.mergeRequest.targetBranch;
}, },
drawerHeaderHeight() { drawerHeaderHeight() {
return getContentWrapperHeight(COMPLIANCE_DRAWER_CONTAINER_CLASS); return getContentWrapperHeight(COMPLIANCE_DRAWER_CONTAINER_CLASS);
}, },
committers() {
return convertArrayOfObjectsToCamelCase(this.mergeRequest.committers);
},
approvedByUsers() {
return convertArrayOfObjectsToCamelCase(this.mergeRequest.approved_by_users);
},
commenters() {
return convertArrayOfObjectsToCamelCase(this.mergeRequest.participants);
},
}, },
DRAWER_Z_INDEX, DRAWER_Z_INDEX,
}; };
...@@ -64,22 +58,25 @@ export default { ...@@ -64,22 +58,25 @@ export default {
</template> </template>
<template v-if="showDrawer" #default> <template v-if="showDrawer" #default>
<project <project
:avatar-url="mergeRequest.project.avatar_url" :avatar-url="project.avatarUrl"
:compliance-framework="mergeRequest.compliance_management_framework" :compliance-framework="project.complianceFramework"
:name="mergeRequest.project.name" :name="project.name"
:url="mergeRequest.project.web_url" :url="project.webUrl"
/> />
<reference :path="mergeRequest.path" :reference="mergeRequest.reference" /> <reference :path="mergeRequest.webUrl" :reference="mergeRequest.reference" />
<branch-path <branch-path
v-if="hasBranchDetails" v-if="hasBranchDetails"
:source-branch="mergeRequest.source_branch" :source-branch="mergeRequest.sourceBranch"
:source-branch-uri="mergeRequest.source_branch_uri" :source-branch-uri="mergeRequest.sourceBranchUri"
:target-branch="mergeRequest.target_branch" :target-branch="mergeRequest.targetBranch"
:target-branch-uri="mergeRequest.target_branch_uri" :target-branch-uri="mergeRequest.targetBranchUri"
/>
<committers :committers="mergeRequest.committers" />
<reviewers
:approvers="mergeRequest.approvedByUsers"
:commenters="mergeRequest.participants"
/> />
<committers :committers="committers" /> <merged-by :merged-by="mergeRequest.mergedBy" />
<reviewers :approvers="approvedByUsers" :commenters="commenters" />
<merged-by :merged-by="mergeRequest.merged_by" />
</template> </template>
</gl-drawer> </gl-drawer>
</template> </template>
<script> <script>
import { GlAvatarLabeled, GlAvatarLink } from '@gitlab/ui'; import { GlAvatarLabeled, GlAvatarLink } from '@gitlab/ui';
import { isEmpty } from 'lodash';
import ComplianceFrameworkLabel from 'ee/vue_shared/components/compliance_framework_label/compliance_framework_label.vue'; import ComplianceFrameworkLabel from 'ee/vue_shared/components/compliance_framework_label/compliance_framework_label.vue';
import { __ } from '~/locale'; import { __ } from '~/locale';
import { DRAWER_AVATAR_SIZE } from '../../constants'; import { DRAWER_AVATAR_SIZE } from '../../constants';
...@@ -32,6 +33,11 @@ export default { ...@@ -32,6 +33,11 @@ export default {
required: true, required: true,
}, },
}, },
computed: {
hasComplianceFramework() {
return !isEmpty(this.complianceFramework);
},
},
i18n: { i18n: {
header: __('Project'), header: __('Project'),
}, },
...@@ -52,7 +58,7 @@ export default { ...@@ -52,7 +58,7 @@ export default {
/> />
</gl-avatar-link> </gl-avatar-link>
<compliance-framework-label <compliance-framework-label
v-if="complianceFramework" v-if="hasComplianceFramework"
class="gl-ml-3" class="gl-ml-3"
:name="complianceFramework.name" :name="complianceFramework.name"
:color="complianceFramework.color" :color="complianceFramework.color"
......
...@@ -4,9 +4,12 @@ import * as Sentry from '@sentry/browser'; ...@@ -4,9 +4,12 @@ import * as Sentry from '@sentry/browser';
import { __ } from '~/locale'; import { __ } from '~/locale';
import { thWidthClass } from '~/lib/utils/table_utility'; import { thWidthClass } from '~/lib/utils/table_utility';
import TimeAgoTooltip from '~/vue_shared/components/time_ago_tooltip.vue'; import TimeAgoTooltip from '~/vue_shared/components/time_ago_tooltip.vue';
import { DRAWER_Z_INDEX } from '~/lib/utils/constants';
import complianceViolationsQuery from '../graphql/compliance_violations.query.graphql'; import complianceViolationsQuery from '../graphql/compliance_violations.query.graphql';
import { mapViolations } from '../graphql/mappers';
import EmptyState from './empty_state.vue'; import EmptyState from './empty_state.vue';
import MergeCommitsExportButton from './merge_requests/merge_commits_export_button.vue'; import MergeCommitsExportButton from './merge_requests/merge_commits_export_button.vue';
import MergeRequestDrawer from './drawer.vue';
import ViolationReason from './violations/reason.vue'; import ViolationReason from './violations/reason.vue';
export default { export default {
...@@ -17,6 +20,7 @@ export default { ...@@ -17,6 +20,7 @@ export default {
GlLoadingIcon, GlLoadingIcon,
GlTable, GlTable,
MergeCommitsExportButton, MergeCommitsExportButton,
MergeRequestDrawer,
ViolationReason, ViolationReason,
TimeAgoTooltip, TimeAgoTooltip,
}, },
...@@ -35,6 +39,9 @@ export default { ...@@ -35,6 +39,9 @@ export default {
return { return {
queryError: false, queryError: false,
violations: [], violations: [],
showDrawer: false,
drawerMergeRequest: {},
drawerProject: {},
}; };
}, },
apollo: { apollo: {
...@@ -46,7 +53,7 @@ export default { ...@@ -46,7 +53,7 @@ export default {
}; };
}, },
update(data) { update(data) {
return data?.group?.mergeRequestViolations?.nodes; return mapViolations(data?.group?.mergeRequestViolations?.nodes);
}, },
error(e) { error(e) {
Sentry.captureException(e); Sentry.captureException(e);
...@@ -65,6 +72,30 @@ export default { ...@@ -65,6 +72,30 @@ export default {
return this.mergeCommitsCsvExportPath !== ''; return this.mergeCommitsCsvExportPath !== '';
}, },
}, },
methods: {
toggleDrawer(rows) {
const { id, mergeRequest, project } = rows[0] || {};
if (!mergeRequest || (this.showDrawer && id === this.drawerMergeRequest.id)) {
this.closeDrawer();
} else {
this.openDrawer(id, mergeRequest, project);
}
},
openDrawer(id, mergeRequest, project) {
this.showDrawer = true;
this.drawerMergeRequest = mergeRequest;
this.drawerProject = project;
},
closeDrawer() {
this.showDrawer = false;
// Refs are required by BTable to manipulate the selection
// issue: https://gitlab.com/gitlab-org/gitlab-ui/-/issues/1531
this.$refs.table.$children[0].clearSelected();
this.drawerMergeRequest = {};
this.drawerProject = {};
},
},
fields: [ fields: [
{ {
key: 'severity', key: 'severity',
...@@ -96,6 +127,7 @@ export default { ...@@ -96,6 +127,7 @@ export default {
'Retrieving the compliance report failed. Please refresh the page and try again.', 'Retrieving the compliance report failed. Please refresh the page and try again.',
), ),
}, },
DRAWER_Z_INDEX,
}; };
</script> </script>
...@@ -120,11 +152,17 @@ export default { ...@@ -120,11 +152,17 @@ export default {
/> />
<gl-table <gl-table
v-else-if="hasViolations" v-else-if="hasViolations"
ref="table"
:fields="$options.fields" :fields="$options.fields"
:items="violations" :items="violations"
head-variant="white" head-variant="white"
stacked="lg" stacked="lg"
select-mode="single"
selectable
hover
selected-variant="primary"
thead-class="gl-border-b-solid gl-border-b-1 gl-border-b-gray-100" thead-class="gl-border-b-solid gl-border-b-1 gl-border-b-gray-100"
@row-selected="toggleDrawer"
> >
<template #cell(reason)="{ item: { reason, violatingUser } }"> <template #cell(reason)="{ item: { reason, violatingUser } }">
<violation-reason :reason="reason" :user="violatingUser" /> <violation-reason :reason="reason" :user="violatingUser" />
...@@ -137,5 +175,12 @@ export default { ...@@ -137,5 +175,12 @@ export default {
</template> </template>
</gl-table> </gl-table>
<empty-state v-else :image-path="emptyStateSvgPath" /> <empty-state v-else :image-path="emptyStateSvgPath" />
<merge-request-drawer
:show-drawer="showDrawer"
:merge-request="drawerMergeRequest"
:project="drawerProject"
:z-index="$options.DRAWER_Z_INDEX"
@close="closeDrawer"
/>
</section> </section>
</template> </template>
...@@ -59,7 +59,7 @@ query getComplianceViolations($fullPath: ID!) { ...@@ -59,7 +59,7 @@ query getComplianceViolations($fullPath: ID!) {
webUrl webUrl
} }
} }
ref: reference reference
fullRef: reference(full: true) fullRef: reference(full: true)
sourceBranch sourceBranch
sourceBranchExists sourceBranchExists
......
export const mapViolations = (nodes = []) => {
return nodes.map((node) => ({
...node,
mergeRequest: {
...node.mergeRequest,
committers: node.mergeRequest.committers?.nodes || [],
approvedByUsers: node.mergeRequest.approvedBy?.nodes || [],
participants: node.mergeRequest.participants?.nodes || [],
},
project: {
...node.project,
complianceFramework: node.project?.complianceFrameworks?.nodes[0] || null,
},
}));
};
...@@ -32,7 +32,7 @@ export default { ...@@ -32,7 +32,7 @@ export default {
title: title:
'Officiis architecto voluptas ut sit qui qui quisquam sequi consectetur porro.', 'Officiis architecto voluptas ut sit qui qui quisquam sequi consectetur porro.',
mergedAt: '2021-11-25T11:56:52.215Z', mergedAt: '2021-11-25T11:56:52.215Z',
webUrl: 'https://gdk.localhost:3443/gitlab-org/gitlab-shell/-/merge_requests/2', webUrl: 'https://gdk.localhost:3443/gitlab-org/gitlab-shell/-/merge_requests/1',
author: { author: {
__typename: 'Author', __typename: 'Author',
id: 50, id: 50,
...@@ -92,8 +92,8 @@ export default { ...@@ -92,8 +92,8 @@ export default {
}, },
], ],
}, },
ref: 'gitlab-shell!2', fullRef: 'gitlab-shell!1',
fullRef: '!2', reference: '!1',
sourceBranch: 'ut-171ad4e263', sourceBranch: 'ut-171ad4e263',
sourceBranchExists: false, sourceBranchExists: false,
targetBranch: 'master', targetBranch: 'master',
...@@ -111,6 +111,104 @@ export default { ...@@ -111,6 +111,104 @@ export default {
{ {
__typename: 'ComplianceFrameworks', __typename: 'ComplianceFrameworks',
id: 1, id: 1,
name: 'GDPR',
description: 'General Data Protection Regulation',
color: '#009966',
},
],
},
},
},
{
__typename: 'MergeRequestViolation',
id: 2,
severity: 2,
reason: 2,
violatingUser: {
__typename: 'Violator',
id: 50,
name: 'John Doe6',
username: 'user6',
avatarUrl:
'https://secure.gravatar.com/avatar/7ff9b8111da2e2109e7b66f37aa632cc?s=80&d=identicon',
webUrl: 'https://gdk.localhost:3443/user6',
},
mergeRequest: {
__typename: 'MergeRequest',
id: 25,
title:
'Officiis architecto voluptas ut sit qui qui quisquam sequi consectetur porro.',
mergedAt: '2021-11-25T11:56:52.215Z',
webUrl: 'https://gdk.localhost:3443/gitlab-org/gitlab-test/-/merge_requests/2',
author: {
__typename: 'Author',
id: 50,
name: 'John Doe6',
username: 'user6',
avatarUrl:
'https://secure.gravatar.com/avatar/7ff9b8111da2e2109e7b66f37aa632cc?s=80&d=identicon',
webUrl: 'https://gdk.localhost:3443/user6',
},
mergedBy: {
__typename: 'MergedBy',
id: 50,
name: 'John Doe6',
username: 'user6',
avatarUrl:
'https://secure.gravatar.com/avatar/7ff9b8111da2e2109e7b66f37aa632cc?s=80&d=identicon',
webUrl: 'https://gdk.localhost:3443/user6',
},
committers: {
__typename: 'Committers',
nodes: [],
},
participants: {
__typename: 'Participants',
nodes: [
{
__typename: 'User',
id: 50,
name: 'John Doe6',
username: 'user6',
avatarUrl:
'https://secure.gravatar.com/avatar/7ff9b8111da2e2109e7b66f37aa632cc?s=80&d=identicon',
webUrl: 'https://gdk.localhost:3443/user6',
},
],
},
approvedBy: {
__typename: 'ApprovedBy',
nodes: [
{
__typename: 'User',
id: 49,
name: 'John Doe5',
username: 'user5',
avatarUrl:
'https://secure.gravatar.com/avatar/eaafc9b0f704edaf23cd5cf7727df560?s=80&d=identicon',
webUrl: 'https://gdk.localhost:3443/user5',
},
],
},
fullRef: 'gitlab-test!2',
reference: '!2',
sourceBranch: 'ut-171ad4e264',
sourceBranchExists: false,
targetBranch: 'master',
targetBranchExists: true,
},
project: {
__typename: 'Project',
id: 2,
avatarUrl: null,
name: 'Gitlab Test',
webUrl: 'https://gdk.localhost:3443/gitlab-org/gitlab-test',
complianceFrameworks: {
__typename: 'ComplianceFrameworks',
nodes: [
{
__typename: 'ComplianceFrameworks',
id: 2,
name: 'SOX', name: 'SOX',
description: 'A framework', description: 'A framework',
color: '#00FF00', color: '#00FF00',
......
import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils';
export const mapDashboardToDrawerData = (mergeRequest) => ({
id: mergeRequest.id,
mergeRequest: {
...convertObjectPropsToCamelCase(mergeRequest, { deep: true }),
webUrl: mergeRequest.path,
},
project: {
...convertObjectPropsToCamelCase(mergeRequest.project, { deep: true }),
complianceFramework: convertObjectPropsToCamelCase(
mergeRequest.compliance_management_framework,
),
},
});
...@@ -46,6 +46,7 @@ exports[`ComplianceDashboard component when there are merge requests and the sho ...@@ -46,6 +46,7 @@ exports[`ComplianceDashboard component when there are merge requests and the sho
<merge-request-drawer-stub <merge-request-drawer-stub
mergerequest="[object Object]" mergerequest="[object Object]"
project="[object Object]"
z-index="252" z-index="252"
/> />
</div> </div>
...@@ -81,6 +82,7 @@ exports[`ComplianceDashboard component when there are merge requests matches the ...@@ -81,6 +82,7 @@ exports[`ComplianceDashboard component when there are merge requests matches the
<merge-request-drawer-stub <merge-request-drawer-stub
mergerequest="[object Object]" mergerequest="[object Object]"
project="[object Object]"
z-index="252" z-index="252"
/> />
</div> </div>
......
...@@ -7,6 +7,7 @@ import MergeRequestDrawer from 'ee/compliance_dashboard/components/drawer.vue'; ...@@ -7,6 +7,7 @@ import MergeRequestDrawer from 'ee/compliance_dashboard/components/drawer.vue';
import MergeRequestGrid from 'ee/compliance_dashboard/components/merge_requests/grid.vue'; import MergeRequestGrid from 'ee/compliance_dashboard/components/merge_requests/grid.vue';
import MergeCommitsExportButton from 'ee/compliance_dashboard/components/merge_requests/merge_commits_export_button.vue'; import MergeCommitsExportButton from 'ee/compliance_dashboard/components/merge_requests/merge_commits_export_button.vue';
import { COMPLIANCE_TAB_COOKIE_KEY } from 'ee/compliance_dashboard/constants'; import { COMPLIANCE_TAB_COOKIE_KEY } from 'ee/compliance_dashboard/constants';
import { mapDashboardToDrawerData } from 'ee/compliance_dashboard/utils';
import { createMergeRequests } from '../mock_data'; import { createMergeRequests } from '../mock_data';
describe('ComplianceDashboard component', () => { describe('ComplianceDashboard component', () => {
...@@ -109,23 +110,29 @@ describe('ComplianceDashboard component', () => { ...@@ -109,23 +110,29 @@ describe('ComplianceDashboard component', () => {
}); });
}); });
describe('with the merge requests drawer', () => { describe('with the merge request drawer', () => {
beforeEach(() => { beforeEach(() => {
wrapper = createComponent(); wrapper = createComponent();
}); });
it('opens the drawer', async () => { it('opens the drawer', async () => {
const drawerData = mapDashboardToDrawerData(mergeRequests[0]);
await findMergeRequestsGrid().vm.$emit('toggleDrawer', mergeRequests[0]); await findMergeRequestsGrid().vm.$emit('toggleDrawer', mergeRequests[0]);
expect(findMergeRequestsDrawer().props('showDrawer')).toBe(true); expect(findMergeRequestsDrawer().props('showDrawer')).toBe(true);
expect(findMergeRequestsDrawer().props('mergeRequest')).toStrictEqual(mergeRequests[0]); expect(findMergeRequestsDrawer().props('mergeRequest')).toStrictEqual(
drawerData.mergeRequest,
);
expect(findMergeRequestsDrawer().props('project')).toStrictEqual(drawerData.project);
}); });
it('closes the drawer via the drawer close event', async () => { it('closes the drawer via the drawer close event', async () => {
await findMergeRequestsDrawer().vm.$emit('close'); await findMergeRequestsDrawer().vm.$emit('close');
expect(findMergeRequestsDrawer().props('showDrawer')).toBe(false); expect(findMergeRequestsDrawer().props('showDrawer')).toBe(false);
expect(findMergeRequestsDrawer().props('mergeRequest')).toEqual({}); expect(findMergeRequestsDrawer().props('mergeRequest')).toStrictEqual({});
expect(findMergeRequestsDrawer().props('project')).toStrictEqual({});
}); });
it('closes the drawer via the grid toggle event', async () => { it('closes the drawer via the grid toggle event', async () => {
...@@ -133,15 +140,21 @@ describe('ComplianceDashboard component', () => { ...@@ -133,15 +140,21 @@ describe('ComplianceDashboard component', () => {
await findMergeRequestsGrid().vm.$emit('toggleDrawer', mergeRequests[0]); await findMergeRequestsGrid().vm.$emit('toggleDrawer', mergeRequests[0]);
expect(findMergeRequestsDrawer().props('showDrawer')).toBe(false); expect(findMergeRequestsDrawer().props('showDrawer')).toBe(false);
expect(findMergeRequestsDrawer().props('mergeRequest')).toEqual({}); expect(findMergeRequestsDrawer().props('mergeRequest')).toStrictEqual({});
expect(findMergeRequestsDrawer().props('project')).toStrictEqual({});
}); });
it('swaps the drawer when a new merge request is selected', async () => { it('swaps the drawer when a new merge request is selected', async () => {
const drawerData = mapDashboardToDrawerData(mergeRequests[1]);
await findMergeRequestsGrid().vm.$emit('toggleDrawer', mergeRequests[0]); await findMergeRequestsGrid().vm.$emit('toggleDrawer', mergeRequests[0]);
await findMergeRequestsGrid().vm.$emit('toggleDrawer', mergeRequests[1]); await findMergeRequestsGrid().vm.$emit('toggleDrawer', mergeRequests[1]);
expect(findMergeRequestsDrawer().props('showDrawer')).toBe(true); expect(findMergeRequestsDrawer().props('showDrawer')).toBe(true);
expect(findMergeRequestsDrawer().props('mergeRequest')).toStrictEqual(mergeRequests[1]); expect(findMergeRequestsDrawer().props('mergeRequest')).toStrictEqual(
drawerData.mergeRequest,
);
expect(findMergeRequestsDrawer().props('project')).toStrictEqual(drawerData.project);
}); });
}); });
}); });
...@@ -21,6 +21,7 @@ describe('Project component', () => { ...@@ -21,6 +21,7 @@ describe('Project component', () => {
propsData: { propsData: {
name: projectName, name: projectName,
url, url,
complianceFramework: {},
...props, ...props,
}, },
}); });
......
import { GlDrawer } from '@gitlab/ui'; import { GlDrawer } from '@gitlab/ui';
import { convertArrayOfObjectsToCamelCase } from '~/lib/utils/common_utils';
import MergeRequestDrawer from 'ee/compliance_dashboard/components/drawer.vue'; import MergeRequestDrawer from 'ee/compliance_dashboard/components/drawer.vue';
import BranchPath from 'ee/compliance_dashboard/components/drawer_sections/branch_path.vue'; import BranchPath from 'ee/compliance_dashboard/components/drawer_sections/branch_path.vue';
import Committers from 'ee/compliance_dashboard/components/drawer_sections/committers.vue'; import Committers from 'ee/compliance_dashboard/components/drawer_sections/committers.vue';
...@@ -7,10 +6,11 @@ import MergedBy from 'ee/compliance_dashboard/components/drawer_sections/merged_ ...@@ -7,10 +6,11 @@ import MergedBy from 'ee/compliance_dashboard/components/drawer_sections/merged_
import Project from 'ee/compliance_dashboard/components/drawer_sections/project.vue'; import Project from 'ee/compliance_dashboard/components/drawer_sections/project.vue';
import Reference from 'ee/compliance_dashboard/components/drawer_sections/reference.vue'; import Reference from 'ee/compliance_dashboard/components/drawer_sections/reference.vue';
import Reviewers from 'ee/compliance_dashboard/components/drawer_sections/reviewers.vue'; import Reviewers from 'ee/compliance_dashboard/components/drawer_sections/reviewers.vue';
import { complianceFramework } from 'ee_jest/vue_shared/components/compliance_framework_label/mock_data';
import { getContentWrapperHeight } from 'ee/threat_monitoring/utils'; import { getContentWrapperHeight } from 'ee/threat_monitoring/utils';
import { shallowMountExtended } from 'helpers/vue_test_utils_helper'; import { shallowMountExtended } from 'helpers/vue_test_utils_helper';
import { createApprovers, createMergeRequests } from '../mock_data'; import resolvers from 'ee/compliance_dashboard/graphql/resolvers';
import { DRAWER_Z_INDEX } from '~/lib/utils/constants';
import { mapViolations } from 'ee/compliance_dashboard/graphql/mappers';
jest.mock('ee/threat_monitoring/utils', () => ({ jest.mock('ee/threat_monitoring/utils', () => ({
getContentWrapperHeight: jest.fn(), getContentWrapperHeight: jest.fn(),
...@@ -18,15 +18,18 @@ jest.mock('ee/threat_monitoring/utils', () => ({ ...@@ -18,15 +18,18 @@ jest.mock('ee/threat_monitoring/utils', () => ({
describe('MergeRequestDrawer component', () => { describe('MergeRequestDrawer component', () => {
let wrapper; let wrapper;
const mergeRequest = createMergeRequests({ const defaultData = mapViolations(resolvers.Query.group().mergeRequestViolations.nodes)[0];
count: 1, const data = {
props: { id: defaultData.id,
compliance_management_framework: complianceFramework, mergeRequest: {
committers: createApprovers(3), ...defaultData.mergeRequest,
approved_by_users: createApprovers(2), sourceBranch: null,
participants: createApprovers(3), sourceBranchUri: null,
targetBranch: null,
targetBranchUri: null,
}, },
})[0]; project: defaultData.project,
};
const findTitle = () => wrapper.findByTestId('dashboard-drawer-title'); const findTitle = () => wrapper.findByTestId('dashboard-drawer-title');
const findDrawer = () => wrapper.findComponent(GlDrawer); const findDrawer = () => wrapper.findComponent(GlDrawer);
...@@ -40,7 +43,8 @@ describe('MergeRequestDrawer component', () => { ...@@ -40,7 +43,8 @@ describe('MergeRequestDrawer component', () => {
const createComponent = (props) => { const createComponent = (props) => {
return shallowMountExtended(MergeRequestDrawer, { return shallowMountExtended(MergeRequestDrawer, {
propsData: { propsData: {
mergeRequest, mergeRequest: data.mergeRequest,
project: data.project,
...props, ...props,
}, },
}); });
...@@ -61,7 +65,7 @@ describe('MergeRequestDrawer component', () => { ...@@ -61,7 +65,7 @@ describe('MergeRequestDrawer component', () => {
it('configures the drawer with header height and z-index', () => { it('configures the drawer with header height and z-index', () => {
expect(findDrawer().props()).toMatchObject({ expect(findDrawer().props()).toMatchObject({
headerHeight: mockHeaderHeight, headerHeight: mockHeaderHeight,
zIndex: 252, zIndex: DRAWER_Z_INDEX,
}); });
}); });
}); });
...@@ -90,22 +94,22 @@ describe('MergeRequestDrawer component', () => { ...@@ -90,22 +94,22 @@ describe('MergeRequestDrawer component', () => {
}); });
it('has the drawer title', () => { it('has the drawer title', () => {
expect(findTitle().text()).toEqual(mergeRequest.title); expect(findTitle().text()).toEqual(data.mergeRequest.title);
}); });
it('has the project section', () => { it('has the project section', () => {
expect(findProject().props()).toStrictEqual({ expect(findProject().props()).toStrictEqual({
avatarUrl: mergeRequest.project.avatar_url, avatarUrl: data.project.avatarUrl,
complianceFramework, complianceFramework: data.project.complianceFramework,
name: mergeRequest.project.name, name: data.project.name,
url: mergeRequest.project.web_url, url: data.project.webUrl,
}); });
}); });
it('has the reference section', () => { it('has the reference section', () => {
expect(findReference().props()).toStrictEqual({ expect(findReference().props()).toStrictEqual({
path: mergeRequest.path, path: data.mergeRequest.webUrl,
reference: mergeRequest.reference, reference: data.mergeRequest.reference,
}); });
}); });
...@@ -115,20 +119,20 @@ describe('MergeRequestDrawer component', () => { ...@@ -115,20 +119,20 @@ describe('MergeRequestDrawer component', () => {
it('has the committers section with users array converted to camel case', () => { it('has the committers section with users array converted to camel case', () => {
expect(findCommitters().props()).toStrictEqual({ expect(findCommitters().props()).toStrictEqual({
committers: convertArrayOfObjectsToCamelCase(mergeRequest.committers), committers: data.mergeRequest.committers,
}); });
}); });
it('has the reviewers section with users array converted to camel case', () => { it('has the reviewers section with users array converted to camel case', () => {
expect(findReviewers().props()).toStrictEqual({ expect(findReviewers().props()).toStrictEqual({
approvers: convertArrayOfObjectsToCamelCase(mergeRequest.approved_by_users), approvers: data.mergeRequest.approvedByUsers,
commenters: convertArrayOfObjectsToCamelCase(mergeRequest.participants), commenters: data.mergeRequest.participants,
}); });
}); });
it('has the merged by section', () => { it('has the merged by section', () => {
expect(findMergedBy().props()).toStrictEqual({ expect(findMergedBy().props()).toStrictEqual({
mergedBy: mergeRequest.merged_by, mergedBy: data.mergeRequest.mergedBy,
}); });
}); });
}); });
...@@ -143,11 +147,11 @@ describe('MergeRequestDrawer component', () => { ...@@ -143,11 +147,11 @@ describe('MergeRequestDrawer component', () => {
wrapper = createComponent({ wrapper = createComponent({
showDrawer: true, showDrawer: true,
mergeRequest: { mergeRequest: {
...mergeRequest, ...data.mergeRequest,
source_branch: sourceBranch, sourceBranch,
source_branch_uri: sourceBranchUri, sourceBranchUri,
target_branch: targetBranch, targetBranch,
target_branch_uri: targetBranchUri, targetBranchUri,
}, },
}); });
}); });
......
import { GlAlert, GlLoadingIcon, GlTable } from '@gitlab/ui'; import { GlAlert, GlLoadingIcon, GlTable } from '@gitlab/ui';
import { mount, shallowMount } from '@vue/test-utils'; import { mount, shallowMount } from '@vue/test-utils';
import VueApollo from 'vue-apollo'; import VueApollo from 'vue-apollo';
import Vue from 'vue'; import Vue, { nextTick } from 'vue';
import * as Sentry from '@sentry/browser'; import * as Sentry from '@sentry/browser';
import ComplianceReport from 'ee/compliance_dashboard/components/report.vue'; import ComplianceReport from 'ee/compliance_dashboard/components/report.vue';
import EmptyState from 'ee/compliance_dashboard/components/empty_state.vue'; import EmptyState from 'ee/compliance_dashboard/components/empty_state.vue';
import MergeRequestDrawer from 'ee/compliance_dashboard/components/drawer.vue';
import MergeCommitsExportButton from 'ee/compliance_dashboard/components/merge_requests/merge_commits_export_button.vue'; import MergeCommitsExportButton from 'ee/compliance_dashboard/components/merge_requests/merge_commits_export_button.vue';
import ViolationReason from 'ee/compliance_dashboard/components/violations/reason.vue'; import ViolationReason from 'ee/compliance_dashboard/components/violations/reason.vue';
import resolvers from 'ee/compliance_dashboard/graphql/resolvers'; import resolvers from 'ee/compliance_dashboard/graphql/resolvers';
import { mapViolations } from 'ee/compliance_dashboard/graphql/mappers';
import { stripTypenames } from 'helpers/graphql_helpers';
import waitForPromises from 'helpers/wait_for_promises'; import waitForPromises from 'helpers/wait_for_promises';
import createMockApollo from 'helpers/mock_apollo_helper'; import createMockApollo from 'helpers/mock_apollo_helper';
import TimeAgoTooltip from '~/vue_shared/components/time_ago_tooltip.vue'; import TimeAgoTooltip from '~/vue_shared/components/time_ago_tooltip.vue';
...@@ -25,6 +28,7 @@ describe('ComplianceReport component', () => { ...@@ -25,6 +28,7 @@ describe('ComplianceReport component', () => {
const findLoadingIcon = () => wrapper.findComponent(GlLoadingIcon); const findLoadingIcon = () => wrapper.findComponent(GlLoadingIcon);
const findErrorMessage = () => wrapper.findComponent(GlAlert); const findErrorMessage = () => wrapper.findComponent(GlAlert);
const findViolationsTable = () => wrapper.findComponent(GlTable); const findViolationsTable = () => wrapper.findComponent(GlTable);
const findMergeRequestDrawer = () => wrapper.findComponent(MergeRequestDrawer);
const findEmptyState = () => wrapper.findComponent(EmptyState); const findEmptyState = () => wrapper.findComponent(EmptyState);
const findMergeCommitsExportButton = () => wrapper.findComponent(MergeCommitsExportButton); const findMergeCommitsExportButton = () => wrapper.findComponent(MergeCommitsExportButton);
const findViolationReason = () => wrapper.findComponent(ViolationReason); const findViolationReason = () => wrapper.findComponent(ViolationReason);
...@@ -33,6 +37,12 @@ describe('ComplianceReport component', () => { ...@@ -33,6 +37,12 @@ describe('ComplianceReport component', () => {
const findTableHeaders = () => findViolationsTable().findAll('th'); const findTableHeaders = () => findViolationsTable().findAll('th');
const findTablesFirstRowData = () => const findTablesFirstRowData = () =>
findViolationsTable().findAll('tbody > tr').at(0).findAll('td'); findViolationsTable().findAll('tbody > tr').at(0).findAll('td');
const findSelectedRows = () => findViolationsTable().findAll('tr.b-table-row-selected');
const selectRow = async (idx) => {
await findViolationsTable().findAll('tbody > tr').at(idx).trigger('click');
await nextTick();
};
function createMockApolloProvider() { function createMockApolloProvider() {
return createMockApollo([], { Query: { group: mockResolver } }); return createMockApollo([], { Query: { group: mockResolver } });
...@@ -144,6 +154,61 @@ describe('ComplianceReport component', () => { ...@@ -144,6 +154,61 @@ describe('ComplianceReport component', () => {
expect(findTimeAgoTooltip().props('time')).toBe(mergedAt); expect(findTimeAgoTooltip().props('time')).toBe(mergedAt);
}); });
describe('with the merge request drawer', () => {
it('opens the drawer', async () => {
const drawerData = mapViolations(mockResolver().mergeRequestViolations.nodes)[0];
await selectRow(0);
expect(findMergeRequestDrawer().props('showDrawer')).toBe(true);
expect(findMergeRequestDrawer().props('mergeRequest')).toStrictEqual(
stripTypenames(drawerData.mergeRequest),
);
expect(findMergeRequestDrawer().props('project')).toStrictEqual(
stripTypenames(drawerData.project),
);
});
it('closes the drawer via the drawer close event', async () => {
await selectRow(0);
expect(findSelectedRows()).toHaveLength(1);
await findMergeRequestDrawer().vm.$emit('close');
expect(findMergeRequestDrawer().props('showDrawer')).toBe(false);
expect(findSelectedRows()).toHaveLength(0);
expect(findMergeRequestDrawer().props('mergeRequest')).toStrictEqual({});
expect(findMergeRequestDrawer().props('project')).toStrictEqual({});
});
it('closes the drawer via the row-selected event', async () => {
await selectRow(0);
expect(findSelectedRows()).toHaveLength(1);
await selectRow(0);
expect(findMergeRequestDrawer().props('showDrawer')).toBe(false);
expect(findMergeRequestDrawer().props('mergeRequest')).toStrictEqual({});
expect(findMergeRequestDrawer().props('project')).toStrictEqual({});
});
it('swaps the drawer when a new row is selected', async () => {
const drawerData = mapViolations(mockResolver().mergeRequestViolations.nodes)[1];
await selectRow(0);
await selectRow(1);
expect(findMergeRequestDrawer().props('showDrawer')).toBe(true);
expect(findMergeRequestDrawer().props('mergeRequest')).toStrictEqual(
stripTypenames(drawerData.mergeRequest),
);
expect(findMergeRequestDrawer().props('project')).toStrictEqual(
stripTypenames(drawerData.project),
);
});
});
}); });
describe('when there are no violations', () => { describe('when there are no violations', () => {
......
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