Commit 6c403eb4 authored by Florie Guibert's avatar Florie Guibert

Boards - Hide EE components in CE

Use licensed features and add rspec tests
parent 7dc7cb0e
...@@ -16,7 +16,7 @@ export default { ...@@ -16,7 +16,7 @@ export default {
gon.features?.graphqlBoardLists || gon.features?.epicBoards gon.features?.graphqlBoardLists || gon.features?.epicBoards
? BoardColumn ? BoardColumn
: BoardColumnDeprecated, : BoardColumnDeprecated,
BoardContentSidebar: () => import('ee_else_ce/boards/components/board_content_sidebar.vue'), BoardContentSidebar: () => import('~/boards/components/board_content_sidebar.vue'),
EpicBoardContentSidebar: () => EpicBoardContentSidebar: () =>
import('ee_component/boards/components/epic_board_content_sidebar.vue'), import('ee_component/boards/components/epic_board_content_sidebar.vue'),
EpicsSwimlanes: () => import('ee_component/boards/components/epics_swimlanes.vue'), EpicsSwimlanes: () => import('ee_component/boards/components/epics_swimlanes.vue'),
......
...@@ -32,6 +32,12 @@ export default { ...@@ -32,6 +32,12 @@ export default {
import('ee_component/sidebar/components/sidebar_iteration_widget.vue'), import('ee_component/sidebar/components/sidebar_iteration_widget.vue'),
}, },
inject: { inject: {
epicFeatureAvailable: {
default: false,
},
iterationFeatureAvailable: {
default: false,
},
weightFeatureAvailable: { weightFeatureAvailable: {
default: false, default: false,
}, },
...@@ -53,15 +59,6 @@ export default { ...@@ -53,15 +59,6 @@ export default {
fullPath() { fullPath() {
return this.activeBoardItem?.referencePath?.split('#')[0] || ''; return this.activeBoardItem?.referencePath?.split('#')[0] || '';
}, },
showEpicSelector() {
return false;
},
showIterationSelector() {
return false;
},
showWeightSelector() {
return false;
},
}, },
methods: { methods: {
...mapActions(['toggleBoardItem', 'setAssignees', 'setActiveItemConfidential']), ...mapActions(['toggleBoardItem', 'setAssignees', 'setActiveItemConfidential']),
...@@ -89,11 +86,11 @@ export default { ...@@ -89,11 +86,11 @@ export default {
class="assignee" class="assignee"
@assignees-updated="setAssignees" @assignees-updated="setAssignees"
/> />
<board-sidebar-epic-select v-if="showEpicSelector" class="epic" /> <board-sidebar-epic-select v-if="epicFeatureAvailable" class="epic" />
<div> <div>
<board-sidebar-milestone-select /> <board-sidebar-milestone-select />
<sidebar-iteration-widget <sidebar-iteration-widget
v-if="showIterationSelector" v-if="iterationFeatureAvailable"
:iid="activeBoardItem.iid" :iid="activeBoardItem.iid"
:workspace-path="projectPathForActiveIssue" :workspace-path="projectPathForActiveIssue"
:iterations-workspace-path="groupPathForActiveIssue" :iterations-workspace-path="groupPathForActiveIssue"
...@@ -104,10 +101,7 @@ export default { ...@@ -104,10 +101,7 @@ export default {
<board-sidebar-time-tracker class="swimlanes-sidebar-time-tracker" /> <board-sidebar-time-tracker class="swimlanes-sidebar-time-tracker" />
<board-sidebar-due-date /> <board-sidebar-due-date />
<board-sidebar-labels-select class="labels" /> <board-sidebar-labels-select class="labels" />
<board-sidebar-weight-input <board-sidebar-weight-input v-if="weightFeatureAvailable" class="weight" />
v-if="weightFeatureAvailable && showWeightSelector"
class="weight"
/>
<sidebar-confidentiality-widget <sidebar-confidentiality-widget
:iid="activeBoardItem.iid" :iid="activeBoardItem.iid"
:full-path="fullPath" :full-path="fullPath"
......
...@@ -101,6 +101,8 @@ export default () => { ...@@ -101,6 +101,8 @@ export default () => {
labelsManagePath: $boardApp.dataset.labelsManagePath, labelsManagePath: $boardApp.dataset.labelsManagePath,
labelsFilterBasePath: $boardApp.dataset.labelsFilterBasePath, labelsFilterBasePath: $boardApp.dataset.labelsFilterBasePath,
timeTrackingLimitToHours: parseBoolean($boardApp.dataset.timeTrackingLimitToHours), timeTrackingLimitToHours: parseBoolean($boardApp.dataset.timeTrackingLimitToHours),
epicFeatureAvailable: parseBoolean($boardApp.dataset.epicFeatureAvailable),
iterationFeatureAvailable: parseBoolean($boardApp.dataset.iterationFeatureAvailable),
weightFeatureAvailable: parseBoolean($boardApp.dataset.weightFeatureAvailable), weightFeatureAvailable: parseBoolean($boardApp.dataset.weightFeatureAvailable),
boardWeight: $boardApp.dataset.boardWeight boardWeight: $boardApp.dataset.boardWeight
? parseInt($boardApp.dataset.boardWeight, 10) ? parseInt($boardApp.dataset.boardWeight, 10)
......
<script>
// This is a false violation of @gitlab/no-runtime-template-compiler, since it
// extends a valid Vue single file component.
/* eslint-disable @gitlab/no-runtime-template-compiler */
import BoardContentSidebarFoss from '~/boards/components/board_content_sidebar.vue';
export default {
extends: BoardContentSidebarFoss,
computed: {
showEpicSelector() {
return true;
},
showIterationSelector() {
return true;
},
showWeightSelector() {
return true;
},
},
};
</script>
...@@ -29,6 +29,8 @@ module EE ...@@ -29,6 +29,8 @@ module EE
milestone_lists_available: current_board_parent.feature_available?(:board_milestone_lists).to_s, milestone_lists_available: current_board_parent.feature_available?(:board_milestone_lists).to_s,
assignee_lists_available: current_board_parent.feature_available?(:board_assignee_lists).to_s, assignee_lists_available: current_board_parent.feature_available?(:board_assignee_lists).to_s,
iteration_lists_available: current_board_parent.feature_available?(:board_iteration_lists).to_s, iteration_lists_available: current_board_parent.feature_available?(:board_iteration_lists).to_s,
epic_feature_available: current_board_parent.feature_available?(:epics).to_s,
iteration_feature_available: current_board_parent.feature_available?(:iterations).to_s,
show_promotion: show_feature_promotion, show_promotion: show_feature_promotion,
scoped_labels: current_board_parent.feature_available?(:scoped_labels)&.to_s, scoped_labels: current_board_parent.feature_available?(:scoped_labels)&.to_s,
can_update: can_update?.to_s, can_update: can_update?.to_s,
......
// Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`ee/BoardContentSidebar matches the snapshot 1`] = `
<div>
Issue details
<board-sidebar-title-stub />
<sidebar-assignees-widget-stub
allowmultipleassignees="true"
class="assignee"
fullpath="gitlab-org/gitlab-test"
iid="27"
initialassignees="[object Object],[object Object]"
issuabletype="issue"
/>
<board-editable-item-stub
class="epic"
data-testid="sidebar-epic"
handleoffclick="true"
title="Epic"
>
<epics-select-stub
canedit="true"
class="gl-w-full"
epicissueid="0"
groupid="1"
initialepic="[object Object]"
issueid="0"
variant="standalone"
/>
</board-editable-item-stub>
<div>
<board-sidebar-milestone-select-stub />
<div
class="gl-mt-5"
data-qa-selector="iteration_container"
>
<sidebar-editable-item-stub
canedit="true"
data-testid="iteration-edit-link"
title="Iteration"
tracking="[object Object]"
/>
</div>
</div>
<board-sidebar-time-tracker-stub
class="swimlanes-sidebar-time-tracker"
/>
<board-sidebar-due-date-stub />
<board-sidebar-labels-select-stub
class="labels"
/>
<board-editable-item-stub
class="weight"
data-testid="sidebar-weight"
handleoffclick="true"
title="Weight"
>
<gl-form-stub>
<gl-form-input-stub
min="0"
placeholder="Enter a number"
type="number"
/>
</gl-form-stub>
</board-editable-item-stub>
<sidebar-confidentiality-widget-stub
fullpath="gitlab-org/gitlab-test"
iid="27"
issuabletype="issue"
/>
<sidebar-subscriptions-widget-stub
data-testid="sidebar-notifications"
fullpath="gitlab-org/gitlab-test"
iid="27"
issuabletype="issue"
/>
</div>
`;
import { GlDrawer } from '@gitlab/ui'; import { GlDrawer } from '@gitlab/ui';
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import Vuex from 'vuex'; import Vuex from 'vuex';
import BoardContentSidebar from 'ee_component/boards/components/board_content_sidebar.vue';
import BoardSidebarEpicSelect from 'ee_component/boards/components/sidebar/board_sidebar_epic_select.vue';
import BoardSidebarWeightInput from 'ee_component/boards/components/sidebar/board_sidebar_weight_input.vue';
import SidebarIterationWidget from 'ee_component/sidebar/components/sidebar_iteration_widget';
import { stubComponent } from 'helpers/stub_component'; import { stubComponent } from 'helpers/stub_component';
import BoardContentSidebar from '~/boards/components/board_content_sidebar.vue';
import { ISSUABLE, issuableTypes } from '~/boards/constants'; import { ISSUABLE, issuableTypes } from '~/boards/constants';
import { mockIssue, mockIssueGroupPath, mockIssueProjectPath } from '../mock_data'; import { mockIssue, mockIssueGroupPath, mockIssueProjectPath } from '../mock_data';
...@@ -47,6 +44,8 @@ describe('ee/BoardContentSidebar', () => { ...@@ -47,6 +44,8 @@ describe('ee/BoardContentSidebar', () => {
canUpdate: true, canUpdate: true,
rootPath: '/', rootPath: '/',
groupId: 1, groupId: 1,
epicFeatureAvailable: true,
iterationFeatureAvailable: true,
weightFeatureAvailable: true, weightFeatureAvailable: true,
}, },
store, store,
...@@ -83,15 +82,7 @@ describe('ee/BoardContentSidebar', () => { ...@@ -83,15 +82,7 @@ describe('ee/BoardContentSidebar', () => {
wrapper = null; wrapper = null;
}); });
it('renders SidebarIterationWidget', () => { it('matches the snapshot', () => {
expect(wrapper.find(SidebarIterationWidget).exists()).toBe(true); expect(wrapper.element).toMatchSnapshot();
});
it('renders BoardSidebarEpicSelect', () => {
expect(wrapper.find(BoardSidebarEpicSelect).exists()).toBe(true);
});
it('renders BoardSidebarWeightInput', () => {
expect(wrapper.find(BoardSidebarWeightInput).exists()).toBe(true);
}); });
}); });
# frozen_string_literal: true # frozen_string_literal: true
RSpec.shared_examples 'issue boards sidebar EE' do RSpec.shared_examples 'issue boards sidebar EE' do
context 'epic dropdown' do context 'epics' do
let_it_be(:epic1) { create(:epic, group: group) } context 'when epic feature available' do
let_it_be(:epic2) { create(:epic, group: group) } let_it_be(:epic1) { create(:epic, group: group) }
let_it_be(:epic2) { create(:epic, group: group) }
let_it_be(:epic_issue, reload: true) { create(:epic_issue, epic: epic1, issue: issue) } let_it_be(:epic_issue, reload: true) { create(:epic_issue, epic: epic1, issue: issue) }
context 'when the issue is associated with an epic' do context 'when the issue is associated with an epic' do
before do before do
first_card_with_epic.click stub_licensed_features(epics: true)
first_card_with_epic.click
end
it 'displays name of epic and links to it' do
page.within('[data-testid="sidebar-epic"]') do
expect(page).to have_link(epic1.title, href: epic_path(epic1))
end
end
it 'updates the epic associated with the issue' do
page.within('[data-testid="sidebar-epic"]') do
find("[data-testid='edit-button']").click
wait_for_requests
find('.gl-new-dropdown-item', text: epic2.title).click
wait_for_requests
expect(page).to have_link(epic2.title, href: epic_path(epic2))
end
end
context 'when epics feature is not available' do
before do
stub_licensed_features(epics: false)
first_card_with_epic.click
wait_for_all_requests
end
it 'cannot find sidebar-epic' do
expect(page).not_to have_selector('[data-testid="sidebar-epic"]')
end
end
end end
end
context 'iterations' do
context 'when iterations feature available' do
let_it_be(:iteration) { create(:iteration, group: group, start_date: 1.day.from_now, due_date: 2.days.from_now, title: 'Iteration 1') }
let_it_be(:iteration2) { create(:iteration, group: group, start_date: 2.days.ago, due_date: 1.day.ago, title: 'Iteration 2', state: 'closed', skip_future_date_validation: true) }
before do
iteration
stub_licensed_features(iterations: true)
first_card.click
wait_for_all_requests
end
it 'selects and updates the right iteration' do
find_and_click_edit_iteration
select_iteration(iteration.title)
expect(page.find('[data-testid="iteration-edit-link"]')).to have_content('Iteration 1')
find_and_click_edit_iteration
it 'displays name of epic and links to it' do select_iteration('No iteration')
page.within('[data-testid="sidebar-epic"]') do
expect(page).to have_link(epic1.title, href: epic_path(epic1)) expect(page.find('[data-testid="iteration-edit-link"]')).to have_content('None')
end end
context 'when iteration feature is not available' do
before do
stub_licensed_features(iterations: false)
first_card.click
wait_for_all_requests
end
it 'cannot find the iteration-edit-link' do
expect(page).not_to have_selector('[data-testid="iteration-edit-link"]')
end
end
end
end
def find_and_click_edit_iteration
page.find('[data-testid="iteration-edit-link"] [data-testid="edit-button"]').click
wait_for_all_requests
end
def select_iteration(iteration_name)
click_button(iteration_name)
wait_for_all_requests
end
end
context 'weight' do
context 'when issue weight feature available' do
before do
stub_licensed_features(issue_weights: true)
first_card.click
wait_for_all_requests
end end
it 'updates the epic associated with the issue' do it 'selects and updates weight' do
page.within('[data-testid="sidebar-epic"]') do page.within('[data-testid="sidebar-weight"]') do
find("[data-testid='edit-button']").click expect(page).to have_content('None')
click_button('Edit')
wait_for_requests fill_in 'Enter a number', with: '2'
find_field('Enter a number').native.send_keys :enter
find('.gl-new-dropdown-item', text: epic2.title).click wait_for_all_requests
wait_for_requests expect(page).to have_content('2')
end
end
context 'when issue weight feature is not available' do
before do
stub_licensed_features(issue_weights: false)
first_card.click
wait_for_all_requests
end
expect(page).to have_link(epic2.title, href: epic_path(epic2)) it 'cannot find the sidebar-weight' do
expect(page).not_to have_selector('[data-testid="sidebar-weight"]')
end end
end end
end end
......
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