Skip to content
代码片段 群组 项目
提交 89a3da03 编辑于 作者: Kushal Pandya's avatar Kushal Pandya
浏览文件

Merge branch '372362-add-approval-rules-section' into 'master'

No related branches found
No related tags found
无相关合并请求
显示
160 个添加31 个删除
...@@ -19,9 +19,14 @@ export const I18N = { ...@@ -19,9 +19,14 @@ export const I18N = {
), ),
disallowForcePushDescription: s__('BranchRules|Force push is not allowed.'), disallowForcePushDescription: s__('BranchRules|Force push is not allowed.'),
approvalsTitle: s__('BranchRules|Approvals'), approvalsTitle: s__('BranchRules|Approvals'),
manageApprovalsLinkTitle: s__('BranchRules|Manage in Merge Request Approvals'),
approvalsDescription: s__(
'BranchRules|Approvals to ensure separation of duties for new merge requests. %{linkStart}Lean more.%{linkEnd}',
),
statusChecksTitle: s__('BranchRules|Status checks'), statusChecksTitle: s__('BranchRules|Status checks'),
allowedToPushHeader: s__('BranchRules|Allowed to push (%{total})'), allowedToPushHeader: s__('BranchRules|Allowed to push (%{total})'),
allowedToMergeHeader: s__('BranchRules|Allowed to merge (%{total})'), allowedToMergeHeader: s__('BranchRules|Allowed to merge (%{total})'),
approvalsHeader: s__('BranchRules|Required approvals (%{total})'),
noData: s__('BranchRules|No data to display'), noData: s__('BranchRules|No data to display'),
}; };
...@@ -33,3 +38,5 @@ export const WILDCARDS_HELP_PATH = ...@@ -33,3 +38,5 @@ export const WILDCARDS_HELP_PATH =
'user/project/protected_branches#configure-multiple-protected-branches-by-using-a-wildcard'; 'user/project/protected_branches#configure-multiple-protected-branches-by-using-a-wildcard';
export const PROTECTED_BRANCHES_HELP_PATH = 'user/project/protected_branches'; export const PROTECTED_BRANCHES_HELP_PATH = 'user/project/protected_branches';
export const APPROVALS_HELP_PATH = 'user/project/merge_requests/approvals/index.md';
...@@ -11,16 +11,19 @@ import { ...@@ -11,16 +11,19 @@ import {
BRANCH_PARAM_NAME, BRANCH_PARAM_NAME,
WILDCARDS_HELP_PATH, WILDCARDS_HELP_PATH,
PROTECTED_BRANCHES_HELP_PATH, PROTECTED_BRANCHES_HELP_PATH,
APPROVALS_HELP_PATH,
} from './constants'; } from './constants';
const wildcardsHelpDocLink = helpPagePath(WILDCARDS_HELP_PATH); const wildcardsHelpDocLink = helpPagePath(WILDCARDS_HELP_PATH);
const protectedBranchesHelpDocLink = helpPagePath(PROTECTED_BRANCHES_HELP_PATH); const protectedBranchesHelpDocLink = helpPagePath(PROTECTED_BRANCHES_HELP_PATH);
const approvalsHelpDocLink = helpPagePath(APPROVALS_HELP_PATH);
export default { export default {
name: 'RuleView', name: 'RuleView',
i18n: I18N, i18n: I18N,
wildcardsHelpDocLink, wildcardsHelpDocLink,
protectedBranchesHelpDocLink, protectedBranchesHelpDocLink,
approvalsHelpDocLink,
components: { Protection, GlSprintf, GlLink, GlLoadingIcon }, components: { Protection, GlSprintf, GlLink, GlLoadingIcon },
inject: { inject: {
projectPath: { projectPath: {
...@@ -29,6 +32,9 @@ export default { ...@@ -29,6 +32,9 @@ export default {
protectedBranchesPath: { protectedBranchesPath: {
default: '', default: '',
}, },
approvalRulesPath: {
default: '',
},
}, },
apollo: { apollo: {
project: { project: {
...@@ -48,7 +54,9 @@ export default { ...@@ -48,7 +54,9 @@ export default {
data() { data() {
return { return {
branch: getParameterByName(BRANCH_PARAM_NAME), branch: getParameterByName(BRANCH_PARAM_NAME),
branchProtection: {}, branchProtection: {
approvalRules: {},
},
}; };
}, },
computed: { computed: {
...@@ -75,6 +83,15 @@ export default { ...@@ -75,6 +83,15 @@ export default {
total: this.pushAccessLevels.total, total: this.pushAccessLevels.total,
}); });
}, },
approvalsHeader() {
const total = this.approvals.reduce(
(sum, { approvalsRequired }) => sum + approvalsRequired,
0,
);
return sprintf(this.$options.i18n.approvalsHeader, {
total,
});
},
allBranches() { allBranches() {
return this.branch === ALL_BRANCHES_WILDCARD; return this.branch === ALL_BRANCHES_WILDCARD;
}, },
...@@ -86,6 +103,9 @@ export default { ...@@ -86,6 +103,9 @@ export default {
? this.$options.i18n.targetBranch ? this.$options.i18n.targetBranch
: this.$options.i18n.branchNameOrPattern; : this.$options.i18n.branchNameOrPattern;
}, },
approvals() {
return this.branchProtection?.approvalRules?.nodes || [];
},
}, },
methods: { methods: {
getAccessLevels(accessLevels = {}) { getAccessLevels(accessLevels = {}) {
...@@ -164,7 +184,22 @@ export default { ...@@ -164,7 +184,22 @@ export default {
/> />
<!-- Approvals --> <!-- Approvals -->
<!-- Follow-up: add approval section (https://gitlab.com/gitlab-org/gitlab/-/issues/372362) --> <h4 class="gl-mb-1 gl-mt-5">{{ $options.i18n.approvalsTitle }}</h4>
<gl-sprintf :message="$options.i18n.approvalsDescription">
<template #link="{ content }">
<gl-link :href="$options.approvalsHelpDocLink">
{{ content }}
</gl-link>
</template>
</gl-sprintf>
<protection
class="gl-mt-3"
:header="approvalsHeader"
:header-link-title="$options.i18n.manageApprovalsLinkTitle"
:header-link-href="approvalRulesPath"
:approvals="approvals"
/>
<!-- Status checks --> <!-- Status checks -->
<!-- Follow-up: add status checks section (https://gitlab.com/gitlab-org/gitlab/-/issues/372362) --> <!-- Follow-up: add status checks section (https://gitlab.com/gitlab-org/gitlab/-/issues/372362) -->
......
...@@ -41,6 +41,11 @@ export default { ...@@ -41,6 +41,11 @@ export default {
required: false, required: false,
default: () => [], default: () => [],
}, },
approvals: {
type: Array,
required: false,
default: () => [],
},
}, },
computed: { computed: {
showUsersDivider() { showUsersDivider() {
...@@ -80,5 +85,15 @@ export default { ...@@ -80,5 +85,15 @@ export default {
:title="$options.i18n.groupsTitle" :title="$options.i18n.groupsTitle"
:access-levels="groups" :access-levels="groups"
/> />
<!-- Approvals -->
<protection-row
v-for="(approval, index) in approvals"
:key="approval.name"
:show-divider="index !== 0"
:title="approval.name"
:users="approval.eligibleApprovers.nodes"
:approvals-required="approval.approvalsRequired"
/>
</gl-card> </gl-card>
</template> </template>
...@@ -36,6 +36,11 @@ export default { ...@@ -36,6 +36,11 @@ export default {
required: false, required: false,
default: () => [], default: () => [],
}, },
approvalsRequired: {
type: Number,
required: false,
default: 0,
},
}, },
computed: { computed: {
avatarBadgeSrOnlyText() { avatarBadgeSrOnlyText() {
...@@ -48,6 +53,11 @@ export default { ...@@ -48,6 +53,11 @@ export default {
commaSeparateList() { commaSeparateList() {
return this.accessLevels.length > 1; return this.accessLevels.length > 1;
}, },
approvalsRequiredTitle() {
return this.approvalsRequired
? n__('%d approval required', '%d approvals required', this.approvalsRequired)
: null;
},
}, },
}; };
</script> </script>
...@@ -57,34 +67,44 @@ export default { ...@@ -57,34 +67,44 @@ export default {
class="gl-display-flex gl-align-items-center gl-border-gray-100 gl-mb-4 gl-pt-4" class="gl-display-flex gl-align-items-center gl-border-gray-100 gl-mb-4 gl-pt-4"
:class="{ 'gl-border-t-solid': showDivider }" :class="{ 'gl-border-t-solid': showDivider }"
> >
<div class="gl-mr-7">{{ title }}</div> <div class="gl-display-flex gl-w-half gl-justify-content-space-between">
<div class="gl-mr-7 gl-w-quarter">{{ title }}</div>
<gl-avatars-inline
v-if="users.length"
class="gl-w-quarter!"
:avatars="users"
:collapsed="true"
:max-visible="$options.MAX_VISIBLE_AVATARS"
:avatar-size="$options.AVATAR_SIZE"
badge-tooltip-prop="name"
:badge-tooltip-max-chars="$options.AVATAR_TOOLTIP_MAX_CHARS"
:badge-sr-only-text="avatarBadgeSrOnlyText"
>
<template #avatar="{ avatar }">
<gl-avatar-link
:key="avatar.username"
v-gl-tooltip
target="_blank"
:href="avatar.webUrl"
:title="avatar.name"
>
<gl-avatar :src="avatar.avatarUrl" :label="avatar.name" :size="$options.AVATAR_SIZE" />
</gl-avatar-link>
</template>
</gl-avatars-inline>
<gl-avatars-inline <div
v-if="users.length" v-for="(item, index) in accessLevels"
:avatars="users" :key="index"
:collapsed="true" data-testid="access-level"
:max-visible="$options.MAX_VISIBLE_AVATARS" class="gl-w-quarter"
:avatar-size="$options.AVATAR_SIZE" >
badge-tooltip-prop="name" <span v-if="commaSeparateList && index > 0" data-testid="comma-separator">,</span>
:badge-tooltip-max-chars="$options.AVATAR_TOOLTIP_MAX_CHARS" {{ item.accessLevelDescription }}
:badge-sr-only-text="avatarBadgeSrOnlyText" </div>
>
<template #avatar="{ avatar }">
<gl-avatar-link
:key="avatar.username"
v-gl-tooltip
target="_blank"
:href="avatar.webUrl"
:title="avatar.name"
>
<gl-avatar :src="avatar.avatarUrl" :label="avatar.name" :size="$options.AVATAR_SIZE" />
</gl-avatar-link>
</template>
</gl-avatars-inline>
<div v-for="(item, index) in accessLevels" :key="index" data-testid="access-level"> <div class="gl-ml-7 gl-w-quarter">{{ approvalsRequiredTitle }}</div>
<span v-if="commaSeparateList && index > 0" data-testid="comma-separator">,</span>
{{ item.accessLevelDescription }}
</div> </div>
</div> </div>
</template> </template>
...@@ -14,7 +14,7 @@ export default function mountBranchRules(el) { ...@@ -14,7 +14,7 @@ export default function mountBranchRules(el) {
defaultClient: createDefaultClient(), defaultClient: createDefaultClient(),
}); });
const { projectPath, protectedBranchesPath } = el.dataset; const { projectPath, protectedBranchesPath, approvalRulesPath } = el.dataset;
return new Vue({ return new Vue({
el, el,
...@@ -22,6 +22,7 @@ export default function mountBranchRules(el) { ...@@ -22,6 +22,7 @@ export default function mountBranchRules(el) {
provide: { provide: {
projectPath, projectPath,
protectedBranchesPath, protectedBranchesPath,
approvalRulesPath,
}, },
render(h) { render(h) {
return h(View); return h(View);
......
...@@ -3,4 +3,4 @@ ...@@ -3,4 +3,4 @@
%h3.gl-mb-5= s_('BranchRules|Branch rules details') %h3.gl-mb-5= s_('BranchRules|Branch rules details')
#js-branch-rules{ data: { project_path: @project.full_path, protected_branches_path: project_settings_repository_path(@project, anchor: 'js-protected-branches-settings') } } #js-branch-rules{ data: { project_path: @project.full_path, protected_branches_path: project_settings_repository_path(@project, anchor: 'js-protected-branches-settings'), approval_rules_path: project_settings_merge_requests_path(@project, anchor: 'js-merge-request-approval-settings') } }
...@@ -143,6 +143,11 @@ msgid_plural "%d additional users" ...@@ -143,6 +143,11 @@ msgid_plural "%d additional users"
msgstr[0] "" msgstr[0] ""
msgstr[1] "" msgstr[1] ""
   
msgid "%d approval required"
msgid_plural "%d approvals required"
msgstr[0] ""
msgstr[1] ""
msgid "%d approver" msgid "%d approver"
msgid_plural "%d approvers" msgid_plural "%d approvers"
msgstr[0] "" msgstr[0] ""
...@@ -6826,6 +6831,9 @@ msgstr "" ...@@ -6826,6 +6831,9 @@ msgstr ""
msgid "BranchRules|Approvals" msgid "BranchRules|Approvals"
msgstr "" msgstr ""
   
msgid "BranchRules|Approvals to ensure separation of duties for new merge requests. %{linkStart}Lean more.%{linkEnd}"
msgstr ""
msgid "BranchRules|Branch" msgid "BranchRules|Branch"
msgstr "" msgstr ""
   
...@@ -6853,6 +6861,9 @@ msgstr "" ...@@ -6853,6 +6861,9 @@ msgstr ""
msgid "BranchRules|Keep stable branches secure and force developers to use merge requests. %{linkStart}What are protected branches?%{linkEnd}" msgid "BranchRules|Keep stable branches secure and force developers to use merge requests. %{linkStart}What are protected branches?%{linkEnd}"
msgstr "" msgstr ""
   
msgid "BranchRules|Manage in Merge Request Approvals"
msgstr ""
msgid "BranchRules|Manage in Protected Branches" msgid "BranchRules|Manage in Protected Branches"
msgstr "" msgstr ""
   
...@@ -6874,6 +6885,9 @@ msgstr "" ...@@ -6874,6 +6885,9 @@ msgstr ""
msgid "BranchRules|Require approval from code owners." msgid "BranchRules|Require approval from code owners."
msgstr "" msgstr ""
   
msgid "BranchRules|Required approvals (%{total})"
msgstr ""
msgid "BranchRules|Roles" msgid "BranchRules|Roles"
msgstr "" msgstr ""
   
...@@ -33,6 +33,7 @@ describe('View branch rules', () => { ...@@ -33,6 +33,7 @@ describe('View branch rules', () => {
let fakeApollo; let fakeApollo;
const projectPath = 'test/testing'; const projectPath = 'test/testing';
const protectedBranchesPath = 'protected/branches'; const protectedBranchesPath = 'protected/branches';
const approvalRulesPath = 'approval/rules';
const branchProtectionsMockRequestHandler = jest const branchProtectionsMockRequestHandler = jest
.fn() .fn()
.mockResolvedValue(branchProtectionsMockResponse); .mockResolvedValue(branchProtectionsMockResponse);
...@@ -42,7 +43,7 @@ describe('View branch rules', () => { ...@@ -42,7 +43,7 @@ describe('View branch rules', () => {
wrapper = shallowMountExtended(RuleView, { wrapper = shallowMountExtended(RuleView, {
apolloProvider: fakeApollo, apolloProvider: fakeApollo,
provide: { projectPath, protectedBranchesPath }, provide: { projectPath, protectedBranchesPath, approvalRulesPath },
}); });
await waitForPromises(); await waitForPromises();
...@@ -57,6 +58,7 @@ describe('View branch rules', () => { ...@@ -57,6 +58,7 @@ describe('View branch rules', () => {
const findBranchProtectionTitle = () => wrapper.findByText(I18N.protectBranchTitle); const findBranchProtectionTitle = () => wrapper.findByText(I18N.protectBranchTitle);
const findBranchProtections = () => wrapper.findAllComponents(Protection); const findBranchProtections = () => wrapper.findAllComponents(Protection);
const findForcePushTitle = () => wrapper.findByText(I18N.allowForcePushDescription); const findForcePushTitle = () => wrapper.findByText(I18N.allowForcePushDescription);
const findApprovalsTitle = () => wrapper.findByText(I18N.approvalsTitle);
it('gets the branch param from url and renders it in the view', () => { it('gets the branch param from url and renders it in the view', () => {
expect(util.getParameterByName).toHaveBeenCalledWith('branch'); expect(util.getParameterByName).toHaveBeenCalledWith('branch');
...@@ -98,4 +100,14 @@ describe('View branch rules', () => { ...@@ -98,4 +100,14 @@ describe('View branch rules', () => {
...protectionMockProps, ...protectionMockProps,
}); });
}); });
it('renders a branch protection component for approvals', () => {
expect(findApprovalsTitle().exists()).toBe(true);
expect(findBranchProtections().at(2).props()).toMatchObject({
header: sprintf(I18N.approvalsHeader, { total: 0 }),
headerLinkHref: approvalRulesPath,
headerLinkTitle: I18N.manageApprovalsLinkTitle,
});
});
}); });
...@@ -36,6 +36,8 @@ const accessLevelsMock = [ ...@@ -36,6 +36,8 @@ const accessLevelsMock = [
{ accessLevelDescription: 'Maintainer' }, { accessLevelDescription: 'Maintainer' },
]; ];
const approvalsRequired = 3;
const groupsMock = [{ name: 'test_group_1' }, { name: 'test_group_2' }]; const groupsMock = [{ name: 'test_group_1' }, { name: 'test_group_2' }];
export const protectionPropsMock = { export const protectionPropsMock = {
...@@ -45,12 +47,20 @@ export const protectionPropsMock = { ...@@ -45,12 +47,20 @@ export const protectionPropsMock = {
roles: accessLevelsMock, roles: accessLevelsMock,
users: usersMock, users: usersMock,
groups: groupsMock, groups: groupsMock,
approvals: [
{
name: 'test',
eligibleApprovers: { nodes: usersMock },
approvalsRequired,
},
],
}; };
export const protectionRowPropsMock = { export const protectionRowPropsMock = {
title: 'Test title', title: 'Test title',
users: usersMock, users: usersMock,
accessLevels: accessLevelsMock, accessLevels: accessLevelsMock,
approvalsRequired,
}; };
export const accessLevelsMockResponse = [ export const accessLevelsMockResponse = [
......
...@@ -25,6 +25,8 @@ describe('Branch rule protection row', () => { ...@@ -25,6 +25,8 @@ describe('Branch rule protection row', () => {
const findAvatarLinks = () => wrapper.findAllComponents(GlAvatarLink); const findAvatarLinks = () => wrapper.findAllComponents(GlAvatarLink);
const findAvatars = () => wrapper.findAllComponents(GlAvatar); const findAvatars = () => wrapper.findAllComponents(GlAvatar);
const findAccessLevels = () => wrapper.findAllByTestId('access-level'); const findAccessLevels = () => wrapper.findAllByTestId('access-level');
const findApprovalsRequired = () =>
wrapper.findByText(`${protectionRowPropsMock.approvalsRequired} approvals required`);
it('renders a title', () => { it('renders a title', () => {
expect(findTitle().exists()).toBe(true); expect(findTitle().exists()).toBe(true);
...@@ -62,4 +64,8 @@ describe('Branch rule protection row', () => { ...@@ -62,4 +64,8 @@ describe('Branch rule protection row', () => {
protectionRowPropsMock.accessLevels[1].accessLevelDescription, protectionRowPropsMock.accessLevels[1].accessLevelDescription,
); );
}); });
it('renders the number of approvals required', () => {
expect(findApprovalsRequired().exists()).toBe(true);
});
}); });
...@@ -56,4 +56,13 @@ describe('Branch rule protection', () => { ...@@ -56,4 +56,13 @@ describe('Branch rule protection', () => {
title: i18n.groupsTitle, title: i18n.groupsTitle,
}); });
}); });
it('renders a protection row for approvals', () => {
const approval = protectionPropsMock.approvals[0];
expect(findProtectionRows().at(3).props()).toMatchObject({
title: approval.name,
users: approval.eligibleApprovers.nodes,
approvalsRequired: approval.approvalsRequired,
});
});
}); });
0% 加载中 .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册