Fix bug in sidebar when menu item cannot be rendered

When a user is rendering the project sidebar and any menu item
is not possible to render, we return a `nil` value.

This introduces an edge case when we want to insert another menu item
before or after that menu, because it was returned as `nil` and,
therefore not introduced in the menu item list.

In this MR, we switched to a Null object pattern, in which we return
a nil menu item instead of just a nil value.
parent 3036a86c
...@@ -11,5 +11,5 @@ ...@@ -11,5 +11,5 @@
- if sidebar.render_raw_menus_partial - if sidebar.render_raw_menus_partial
= render sidebar.render_raw_menus_partial = render sidebar.render_raw_menus_partial
= render partial: 'shared/nav/sidebar_hidden_menu_item', collection: sidebar.hidden_menu&.items = render partial: 'shared/nav/sidebar_hidden_menu_item', collection: sidebar.hidden_menu&.renderable_items
= render 'shared/sidebar_toggle_button' = render 'shared/sidebar_toggle_button'
...@@ -13,7 +13,7 @@ ...@@ -13,7 +13,7 @@
%span.badge.badge-pill.count{ **sidebar_menu.pill_html_options } %span.badge.badge-pill.count{ **sidebar_menu.pill_html_options }
= number_with_delimiter(sidebar_menu.pill_count) = number_with_delimiter(sidebar_menu.pill_count)
%ul.sidebar-sub-level-items{ class: ('is-fly-out-only' unless sidebar_menu.has_items?) } %ul.sidebar-sub-level-items{ class: ('is-fly-out-only' unless sidebar_menu.has_renderable_items?) }
= nav_link(**sidebar_menu.all_active_routes, html_options: { class: 'fly-out-top-item' } ) do = nav_link(**sidebar_menu.all_active_routes, html_options: { class: 'fly-out-top-item' } ) do
= link_to sidebar_menu.link, **sidebar_menu.collapsed_container_html_options do = link_to sidebar_menu.link, **sidebar_menu.collapsed_container_html_options do
%strong.fly-out-top-item-name %strong.fly-out-top-item-name
...@@ -24,4 +24,4 @@ ...@@ -24,4 +24,4 @@
- if sidebar_menu.has_items? - if sidebar_menu.has_items?
%li.divider.fly-out-top-item %li.divider.fly-out-top-item
= render partial: 'shared/nav/sidebar_menu_item', collection: sidebar_menu.items = render partial: 'shared/nav/sidebar_menu_item', collection: sidebar_menu.renderable_items
...@@ -31,7 +31,7 @@ module EE ...@@ -31,7 +31,7 @@ module EE
return audit_events_menu_item.link if audit_events_menu_item return audit_events_menu_item.link if audit_events_menu_item
return dependencies_menu_item.link if dependencies_menu_item return dependencies_menu_item.link if dependencies_menu_item
items.first.link renderable_items.first.link
end end
override :render? override :render?
......
...@@ -11,7 +11,7 @@ RSpec.describe Sidebars::Projects::Menus::AnalyticsMenu do ...@@ -11,7 +11,7 @@ RSpec.describe Sidebars::Projects::Menus::AnalyticsMenu do
subject { described_class.new(context) } subject { described_class.new(context) }
describe 'Menu items' do describe 'Menu items' do
subject { described_class.new(context).items.index { |e| e.item_id == item_id } } subject { described_class.new(context).renderable_items.index { |e| e.item_id == item_id } }
describe 'Code Review' do describe 'Code Review' do
let(:item_id) { :code_review } let(:item_id) { :code_review }
......
...@@ -8,7 +8,7 @@ RSpec.describe Sidebars::Projects::Menus::CiCdMenu do ...@@ -8,7 +8,7 @@ RSpec.describe Sidebars::Projects::Menus::CiCdMenu do
let(:context) { Sidebars::Projects::Context.new(current_user: user, container: project, can_view_pipeline_editor: true) } let(:context) { Sidebars::Projects::Context.new(current_user: user, container: project, can_view_pipeline_editor: true) }
describe 'Test Cases' do describe 'Test Cases' do
subject { described_class.new(context).items.index { |e| e.item_id == :test_cases} } subject { described_class.new(context).renderable_items.index { |e| e.item_id == :test_cases} }
context 'when licensed feature quality_management is not enabled' do context 'when licensed feature quality_management is not enabled' do
before do before do
......
...@@ -8,7 +8,7 @@ RSpec.describe Sidebars::Projects::Menus::IssuesMenu do ...@@ -8,7 +8,7 @@ RSpec.describe Sidebars::Projects::Menus::IssuesMenu do
let(:context) { Sidebars::Projects::Context.new(current_user: user, container: project) } let(:context) { Sidebars::Projects::Context.new(current_user: user, container: project) }
describe 'Iterations' do describe 'Iterations' do
subject { described_class.new(context).items.index { |e| e.item_id == :iterations} } subject { described_class.new(context).renderable_items.index { |e| e.item_id == :iterations} }
context 'when licensed feature iterations is not enabled' do context 'when licensed feature iterations is not enabled' do
it 'does not include iterations menu item' do it 'does not include iterations menu item' do
......
...@@ -7,35 +7,39 @@ RSpec.describe Sidebars::Projects::Menus::OperationsMenu do ...@@ -7,35 +7,39 @@ RSpec.describe Sidebars::Projects::Menus::OperationsMenu do
let(:user) { project.owner } let(:user) { project.owner }
let(:context) { Sidebars::Projects::Context.new(current_user: user, container: project, show_cluster_hint: true) } let(:context) { Sidebars::Projects::Context.new(current_user: user, container: project, show_cluster_hint: true) }
describe 'On-call Schedules' do describe 'Menu items' do
subject { described_class.new(context).items.index { |e| e.item_id == :on_call_schedules } } subject { described_class.new(context).renderable_items.index { |e| e.item_id == item_id } }
before do describe 'On-call Schedules' do
stub_licensed_features(oncall_schedules: true) let(:item_id) { :on_call_schedules }
end
before do
stub_licensed_features(oncall_schedules: true)
end
specify { is_expected.not_to be_nil } specify { is_expected.not_to be_nil }
describe 'when the user does not have access' do describe 'when the user does not have access' do
let(:user) { nil } let(:user) { nil }
specify { is_expected.to be_nil } specify { is_expected.to be_nil }
end
end end
end
describe 'Escalation policies' do describe 'Escalation policies' do
subject { described_class.new(context).items.index { |e| e.item_id == :escalation_policies } } let(:item_id) { :escalation_policies }
before do before do
stub_licensed_features(oncall_schedules: true, escalation_policies: true) stub_licensed_features(oncall_schedules: true, escalation_policies: true)
end end
specify { is_expected.not_to be_nil } specify { is_expected.not_to be_nil }
describe 'when the user does not have access' do describe 'when the user does not have access' do
let(:user) { nil } let(:user) { nil }
specify { is_expected.to be_nil } specify { is_expected.to be_nil }
end
end end
end end
end end
...@@ -9,7 +9,7 @@ RSpec.describe Sidebars::Projects::Menus::RepositoryMenu do ...@@ -9,7 +9,7 @@ RSpec.describe Sidebars::Projects::Menus::RepositoryMenu do
let(:context) { Sidebars::Projects::Context.new(current_user: user, container: project, current_ref: 'master') } let(:context) { Sidebars::Projects::Context.new(current_user: user, container: project, current_ref: 'master') }
describe 'File Locks' do describe 'File Locks' do
subject { described_class.new(context).items.index { |e| e.item_id == :file_locks} } subject { described_class.new(context).renderable_items.index { |e| e.item_id == :file_locks} }
context 'when licensed feature file locks is not enabled' do context 'when licensed feature file locks is not enabled' do
it 'does not include file locks menu item' do it 'does not include file locks menu item' do
......
...@@ -37,9 +37,9 @@ RSpec.describe Sidebars::Projects::Menus::JiraMenu do ...@@ -37,9 +37,9 @@ RSpec.describe Sidebars::Projects::Menus::JiraMenu do
end end
it 'contains issue list and open jira menu items' do it 'contains issue list and open jira menu items' do
expect(subject.items).not_to be_empty expect(subject.renderable_items).not_to be_empty
expect(subject.items[0].item_id).to eq :issue_list expect(subject.renderable_items[0].item_id).to eq :issue_list
expect(subject.items[1].item_id).to eq :open_jira expect(subject.renderable_items[1].item_id).to eq :open_jira
end end
end end
end end
......
...@@ -33,7 +33,7 @@ RSpec.describe Sidebars::Projects::Menus::RequirementsMenu do ...@@ -33,7 +33,7 @@ RSpec.describe Sidebars::Projects::Menus::RequirementsMenu do
end end
it 'does not contain any menu item' do it 'does not contain any menu item' do
expect(subject.items).to be_empty expect(subject.renderable_items).to be_empty
end end
end end
...@@ -47,7 +47,7 @@ RSpec.describe Sidebars::Projects::Menus::RequirementsMenu do ...@@ -47,7 +47,7 @@ RSpec.describe Sidebars::Projects::Menus::RequirementsMenu do
end end
it 'contains list menu item' do it 'contains list menu item' do
expect(subject.items[0].item_id).to eq :requirements_list expect(subject.renderable_items[0].item_id).to eq :requirements_list
end end
end end
end end
......
...@@ -13,7 +13,7 @@ module Sidebars ...@@ -13,7 +13,7 @@ module Sidebars
include ::Sidebars::Concerns::ContainerWithHtmlOptions include ::Sidebars::Concerns::ContainerWithHtmlOptions
include ::Sidebars::Concerns::HasActiveRoutes include ::Sidebars::Concerns::HasActiveRoutes
attr_reader :context, :items attr_reader :context
delegate :current_user, :container, to: :@context delegate :current_user, :container, to: :@context
def initialize(context) def initialize(context)
...@@ -29,7 +29,7 @@ module Sidebars ...@@ -29,7 +29,7 @@ module Sidebars
override :render? override :render?
def render? def render?
has_items? has_renderable_items?
end end
# Menus might have or not a link # Menus might have or not a link
...@@ -43,7 +43,7 @@ module Sidebars ...@@ -43,7 +43,7 @@ module Sidebars
# This method filters the information and returns: { path: ['foo', 'bar'], controller: :foo } # This method filters the information and returns: { path: ['foo', 'bar'], controller: :foo }
def all_active_routes def all_active_routes
@all_active_routes ||= begin @all_active_routes ||= begin
([active_routes] + items.map(&:active_routes)).flatten.each_with_object({}) do |pairs, hash| ([active_routes] + renderable_items.map(&:active_routes)).flatten.each_with_object({}) do |pairs, hash|
pairs.each do |k, v| pairs.each do |k, v|
hash[k] ||= [] hash[k] ||= []
hash[k] += Array(v) hash[k] += Array(v)
...@@ -55,10 +55,22 @@ module Sidebars ...@@ -55,10 +55,22 @@ module Sidebars
end end
end end
# Returns whether the menu has any menu item, no
# matter whether it is renderable or not
def has_items? def has_items?
@items.any? @items.any?
end end
# Returns all renderable menu items
def renderable_items
@renderable_items ||= @items.select(&:render?)
end
# Returns whether the menu has any renderable menu item
def has_renderable_items?
renderable_items.any?
end
def add_item(item) def add_item(item)
add_element(@items, item) add_element(@items, item)
end end
......
...@@ -18,5 +18,9 @@ module Sidebars ...@@ -18,5 +18,9 @@ module Sidebars
def show_hint? def show_hint?
hint_html_options.present? hint_html_options.present?
end end
def render?
true
end
end end
end end
# frozen_string_literal: true
module Sidebars
class NilMenuItem < MenuItem
extend ::Gitlab::Utils::Override
def initialize(item_id:)
super(item_id: item_id, title: nil, link: nil, active_routes: {})
end
override :render?
def render?
false
end
end
end
...@@ -21,7 +21,7 @@ module Sidebars ...@@ -21,7 +21,7 @@ module Sidebars
def link def link
return cycle_analytics_menu_item.link if cycle_analytics_menu_item return cycle_analytics_menu_item.link if cycle_analytics_menu_item
items.first.link renderable_items.first.link
end end
override :extra_container_html_options override :extra_container_html_options
......
...@@ -15,7 +15,7 @@ module Sidebars ...@@ -15,7 +15,7 @@ module Sidebars
override :link override :link
def link def link
items.first.link renderable_items.first.link
end end
override :title override :title
......
...@@ -5,15 +5,18 @@ require 'spec_helper' ...@@ -5,15 +5,18 @@ require 'spec_helper'
RSpec.describe Sidebars::Menu do RSpec.describe Sidebars::Menu do
let(:menu) { described_class.new(context) } let(:menu) { described_class.new(context) }
let(:context) { Sidebars::Context.new(current_user: nil, container: nil) } let(:context) { Sidebars::Context.new(current_user: nil, container: nil) }
let(:nil_menu_item) { Sidebars::NilMenuItem.new(item_id: :foo) }
describe '#all_active_routes' do describe '#all_active_routes' do
it 'gathers all active routes of items and the current menu' do it 'gathers all active routes of items and the current menu' do
menu.add_item(Sidebars::MenuItem.new(title: 'foo1', link: 'foo1', active_routes: { path: %w(bar test) })) menu.add_item(Sidebars::MenuItem.new(title: 'foo1', link: 'foo1', active_routes: { path: %w(bar test) }))
menu.add_item(Sidebars::MenuItem.new(title: 'foo2', link: 'foo2', active_routes: { controller: 'fooc' })) menu.add_item(Sidebars::MenuItem.new(title: 'foo2', link: 'foo2', active_routes: { controller: 'fooc' }))
menu.add_item(Sidebars::MenuItem.new(title: 'foo3', link: 'foo3', active_routes: { controller: 'barc' })) menu.add_item(Sidebars::MenuItem.new(title: 'foo3', link: 'foo3', active_routes: { controller: 'barc' }))
menu.add_item(nil_menu_item)
allow(menu).to receive(:active_routes).and_return({ path: 'foo' }) allow(menu).to receive(:active_routes).and_return({ path: 'foo' })
expect(menu).to receive(:renderable_items).and_call_original
expect(menu.all_active_routes).to eq({ path: %w(foo bar test), controller: %w(fooc barc) }) expect(menu.all_active_routes).to eq({ path: %w(foo bar test), controller: %w(fooc barc) })
end end
end end
...@@ -31,6 +34,60 @@ RSpec.describe Sidebars::Menu do ...@@ -31,6 +34,60 @@ RSpec.describe Sidebars::Menu do
expect(menu.render?).to be true expect(menu.render?).to be true
end end
context 'when menu items are NilMenuItem' do
it 'returns false' do
menu.add_item(nil_menu_item)
expect(menu.render?).to be false
end
end
end
end
describe '#has_items?' do
it 'returns true when there are regular menu items' do
menu.add_item(Sidebars::MenuItem.new(title: 'foo1', link: 'foo1', active_routes: {}))
expect(menu.has_items?).to be true
end
it 'returns true when there are nil menu items' do
menu.add_item(nil_menu_item)
expect(menu.has_items?).to be true
end
end
describe '#has_renderable_items?' do
it 'returns true when there are regular menu items' do
menu.add_item(Sidebars::MenuItem.new(title: 'foo1', link: 'foo1', active_routes: {}))
expect(menu.has_renderable_items?).to be true
end
it 'returns false when there are nil menu items' do
menu.add_item(nil_menu_item)
expect(menu.has_renderable_items?).to be false
end
it 'returns true when there are both regular and nil menu items' do
menu.add_item(Sidebars::MenuItem.new(title: 'foo1', link: 'foo1', active_routes: {}))
menu.add_item(nil_menu_item)
expect(menu.has_renderable_items?).to be true
end
end
describe '#renderable_items' do
it 'returns only regular menu items' do
item = Sidebars::MenuItem.new(title: 'foo1', link: 'foo1', active_routes: {})
menu.add_item(item)
menu.add_item(nil_menu_item)
expect(menu.renderable_items.size).to eq 1
expect(menu.renderable_items.first).to eq item
end end
end end
......
...@@ -26,7 +26,7 @@ RSpec.describe Sidebars::Projects::Menus::AnalyticsMenu do ...@@ -26,7 +26,7 @@ RSpec.describe Sidebars::Projects::Menus::AnalyticsMenu do
context 'when menu does not have any menu items' do context 'when menu does not have any menu items' do
it 'returns false' do it 'returns false' do
allow(subject).to receive(:has_items?).and_return(false) allow(subject).to receive(:has_renderable_items?).and_return(false)
expect(subject.render?).to be false expect(subject.render?).to be false
end end
...@@ -49,13 +49,13 @@ RSpec.describe Sidebars::Projects::Menus::AnalyticsMenu do ...@@ -49,13 +49,13 @@ RSpec.describe Sidebars::Projects::Menus::AnalyticsMenu do
it 'returns link to the the first visible menu item' do it 'returns link to the the first visible menu item' do
allow(subject).to receive(:cycle_analytics_menu_item).and_return(nil) allow(subject).to receive(:cycle_analytics_menu_item).and_return(nil)
expect(subject.link).to eq subject.items.first.link expect(subject.link).to eq subject.renderable_items.first.link
end end
end end
end end
describe 'Menu items' do describe 'Menu items' do
subject { described_class.new(context).items.index { |e| e.item_id == item_id } } subject { described_class.new(context).renderable_items.index { |e| e.item_id == item_id } }
describe 'CI/CD' do describe 'CI/CD' do
let(:item_id) { :ci_cd_analytics } let(:item_id) { :ci_cd_analytics }
......
...@@ -26,40 +26,44 @@ RSpec.describe Sidebars::Projects::Menus::CiCdMenu do ...@@ -26,40 +26,44 @@ RSpec.describe Sidebars::Projects::Menus::CiCdMenu do
end end
end end
describe 'Pipelines Editor' do describe 'Menu items' do
subject { described_class.new(context).items.index { |e| e.item_id == :pipelines_editor } } subject { described_class.new(context).renderable_items.index { |e| e.item_id == item_id } }
context 'when user cannot view pipeline editor' do describe 'Pipelines Editor' do
let(:can_view_pipeline_editor) { false } let(:item_id) { :pipelines_editor }
it 'does not include pipeline editor menu item' do context 'when user cannot view pipeline editor' do
is_expected.to be_nil let(:can_view_pipeline_editor) { false }
it 'does not include pipeline editor menu item' do
is_expected.to be_nil
end
end end
end
context 'when user can view pipeline editor' do context 'when user can view pipeline editor' do
it 'includes pipeline editor menu item' do it 'includes pipeline editor menu item' do
is_expected.not_to be_nil is_expected.not_to be_nil
end
end end
end end
end
describe 'Artifacts' do describe 'Artifacts' do
subject { described_class.new(context).items.index { |e| e.item_id == :artifacts } } let(:item_id) { :artifacts }
context 'when feature flag :artifacts_management_page is disabled' do context 'when feature flag :artifacts_management_page is disabled' do
it 'does not include artifacts menu item' do it 'does not include artifacts menu item' do
stub_feature_flags(artifacts_management_page: false) stub_feature_flags(artifacts_management_page: false)
is_expected.to be_nil is_expected.to be_nil
end
end end
end
context 'when feature flag :artifacts_management_page is enabled' do context 'when feature flag :artifacts_management_page is enabled' do
it 'includes artifacts menu item' do it 'includes artifacts menu item' do
stub_feature_flags(artifacts_management_page: true) stub_feature_flags(artifacts_management_page: true)
is_expected.not_to be_nil is_expected.not_to be_nil
end
end end
end end
end end
......
...@@ -36,7 +36,7 @@ RSpec.describe Sidebars::Projects::Menus::ConfluenceMenu do ...@@ -36,7 +36,7 @@ RSpec.describe Sidebars::Projects::Menus::ConfluenceMenu do
end end
it 'does not contain any sub menu' do it 'does not contain any sub menu' do
expect(subject.items).to be_empty expect(subject.has_items?).to be false
end end
end end
end end
......
...@@ -11,7 +11,7 @@ RSpec.describe Sidebars::Projects::Menus::ExternalIssueTrackerMenu do ...@@ -11,7 +11,7 @@ RSpec.describe Sidebars::Projects::Menus::ExternalIssueTrackerMenu do
subject { described_class.new(context) } subject { described_class.new(context) }
it 'does not contain any sub menu' do it 'does not contain any sub menu' do
expect(subject.items).to be_empty expect(subject.has_items?).to be false
end end
describe '#render?' do describe '#render?' do
......
...@@ -10,7 +10,7 @@ RSpec.describe Sidebars::Projects::Menus::ExternalWikiMenu do ...@@ -10,7 +10,7 @@ RSpec.describe Sidebars::Projects::Menus::ExternalWikiMenu do
subject { described_class.new(context) } subject { described_class.new(context) }
it 'does not contain any sub menu' do it 'does not contain any sub menu' do
expect(subject.items).to be_empty expect(subject.has_items?).to be false
end end
describe '#render?' do describe '#render?' do
......
...@@ -13,7 +13,7 @@ RSpec.describe Sidebars::Projects::Menus::HiddenMenu do ...@@ -13,7 +13,7 @@ RSpec.describe Sidebars::Projects::Menus::HiddenMenu do
context 'when menu does not have any menu items' do context 'when menu does not have any menu items' do
it 'returns false' do it 'returns false' do
allow(subject).to receive(:has_items?).and_return(false) allow(subject).to receive(:has_renderable_items?).and_return(false)
expect(subject.render?).to be false expect(subject.render?).to be false
end end
...@@ -27,7 +27,7 @@ RSpec.describe Sidebars::Projects::Menus::HiddenMenu do ...@@ -27,7 +27,7 @@ RSpec.describe Sidebars::Projects::Menus::HiddenMenu do
end end
describe 'Menu items' do describe 'Menu items' do
subject { described_class.new(context).items.index { |e| e.item_id == item_id } } subject { described_class.new(context).renderable_items.index { |e| e.item_id == item_id } }
shared_examples 'access rights checks' do shared_examples 'access rights checks' do
specify { is_expected.not_to be_nil } specify { is_expected.not_to be_nil }
......
...@@ -10,7 +10,7 @@ RSpec.describe Sidebars::Projects::Menus::LabelsMenu do ...@@ -10,7 +10,7 @@ RSpec.describe Sidebars::Projects::Menus::LabelsMenu do
subject { described_class.new(context) } subject { described_class.new(context) }
it 'does not contain any sub menu' do it 'does not contain any sub menu' do
expect(subject.items).to be_empty expect(subject.has_items?).to eq false
end end
describe '#render?' do describe '#render?' do
......
...@@ -19,7 +19,7 @@ RSpec.describe Sidebars::Projects::Menus::LearnGitlabMenu do ...@@ -19,7 +19,7 @@ RSpec.describe Sidebars::Projects::Menus::LearnGitlabMenu do
subject { described_class.new(context) } subject { described_class.new(context) }
it 'does not contain any sub menu' do it 'does not contain any sub menu' do
expect(subject.instance_variable_get(:@items)).to be_empty expect(subject.has_items?).to be false
end end
describe '#nav_link_html_options' do describe '#nav_link_html_options' do
......
...@@ -21,9 +21,9 @@ RSpec.describe Sidebars::Projects::Menus::OperationsMenu do ...@@ -21,9 +21,9 @@ RSpec.describe Sidebars::Projects::Menus::OperationsMenu do
end end
context 'when operation feature is enabled' do context 'when operation feature is enabled' do
context 'when menu does not have any menu items' do context 'when menu does not have any renderable menu items' do
it 'returns false' do it 'returns false' do
allow(subject).to receive(:has_items?).and_return(false) allow(subject).to receive(:has_renderable_items?).and_return(false)
expect(subject.render?).to be false expect(subject.render?).to be false
end end
...@@ -54,7 +54,7 @@ RSpec.describe Sidebars::Projects::Menus::OperationsMenu do ...@@ -54,7 +54,7 @@ RSpec.describe Sidebars::Projects::Menus::OperationsMenu do
end end
context 'Menu items' do context 'Menu items' do
subject { described_class.new(context).items.index { |e| e.item_id == item_id } } subject { described_class.new(context).renderable_items.index { |e| e.item_id == item_id } }
describe 'Metrics Dashboard' do describe 'Metrics Dashboard' do
let(:item_id) { :metrics } let(:item_id) { :metrics }
......
...@@ -12,7 +12,7 @@ RSpec.describe Sidebars::Projects::Menus::PackagesRegistriesMenu do ...@@ -12,7 +12,7 @@ RSpec.describe Sidebars::Projects::Menus::PackagesRegistriesMenu do
describe '#render?' do describe '#render?' do
context 'when menu does not have any menu item to show' do context 'when menu does not have any menu item to show' do
it 'returns false' do it 'returns false' do
allow(subject).to receive(:has_items?).and_return(false) allow(subject).to receive(:has_renderable_items?).and_return(false)
expect(subject.render?).to eq false expect(subject.render?).to eq false
end end
...@@ -36,7 +36,7 @@ RSpec.describe Sidebars::Projects::Menus::PackagesRegistriesMenu do ...@@ -36,7 +36,7 @@ RSpec.describe Sidebars::Projects::Menus::PackagesRegistriesMenu do
context 'when Packages Registry is visible' do context 'when Packages Registry is visible' do
it 'menu link points to Packages Registry page' do it 'menu link points to Packages Registry page' do
expect(subject.link).to eq described_class.new(context).items.find { |i| i.item_id == :packages_registry }.link expect(subject.link).to eq described_class.new(context).renderable_items.find { |i| i.item_id == :packages_registry }.link
end end
end end
...@@ -44,95 +44,99 @@ RSpec.describe Sidebars::Projects::Menus::PackagesRegistriesMenu do ...@@ -44,95 +44,99 @@ RSpec.describe Sidebars::Projects::Menus::PackagesRegistriesMenu do
let(:packages_enabled) { false } let(:packages_enabled) { false }
it 'menu link points to Container Registry page' do it 'menu link points to Container Registry page' do
expect(subject.link).to eq described_class.new(context).items.find { |i| i.item_id == :container_registry }.link expect(subject.link).to eq described_class.new(context).renderable_items.find { |i| i.item_id == :container_registry }.link
end end
context 'when Container Registry is not visible' do context 'when Container Registry is not visible' do
let(:registry_enabled) { false } let(:registry_enabled) { false }
it 'menu link points to Infrastructure Registry page' do it 'menu link points to Infrastructure Registry page' do
expect(subject.link).to eq described_class.new(context).items.find { |i| i.item_id == :infrastructure_registry }.link expect(subject.link).to eq described_class.new(context).renderable_items.find { |i| i.item_id == :infrastructure_registry }.link
end end
end end
end end
end end
describe 'Packages Registry' do describe 'Menu items' do
subject { described_class.new(context).items.find { |i| i.item_id == :packages_registry }} subject { described_class.new(context).renderable_items.find { |i| i.item_id == item_id } }
context 'when user can read packages' do describe 'Packages Registry' do
context 'when config package setting is disabled' do let(:item_id) { :packages_registry }
it 'the menu item is not added to list of menu items' do
stub_config(packages: { enabled: false })
is_expected.to be_nil context 'when user can read packages' do
context 'when config package setting is disabled' do
it 'the menu item is not added to list of menu items' do
stub_config(packages: { enabled: false })
is_expected.to be_nil
end
end end
end
context 'when config package setting is enabled' do context 'when config package setting is enabled' do
it 'the menu item is added to list of menu items' do it 'the menu item is added to list of menu items' do
stub_config(packages: { enabled: true }) stub_config(packages: { enabled: true })
is_expected.not_to be_nil is_expected.not_to be_nil
end
end end
end end
end
context 'when user cannot read packages' do context 'when user cannot read packages' do
let(:user) { nil } let(:user) { nil }
it 'the menu item is not added to list of menu items' do it 'the menu item is not added to list of menu items' do
is_expected.to be_nil is_expected.to be_nil
end
end end
end end
end
describe 'Container Registry' do describe 'Container Registry' do
subject { described_class.new(context).items.find { |i| i.item_id == :container_registry }} let(:item_id) { :container_registry }
context 'when user can read container images' do context 'when user can read container images' do
context 'when config registry setting is disabled' do context 'when config registry setting is disabled' do
it 'the menu item is not added to list of menu items' do it 'the menu item is not added to list of menu items' do
stub_container_registry_config(enabled: false) stub_container_registry_config(enabled: false)
is_expected.to be_nil is_expected.to be_nil
end
end end
end
context 'when config registry setting is enabled' do context 'when config registry setting is enabled' do
it 'the menu item is added to list of menu items' do it 'the menu item is added to list of menu items' do
stub_container_registry_config(enabled: true) stub_container_registry_config(enabled: true)
is_expected.not_to be_nil is_expected.not_to be_nil
end
end end
end end
end
context 'when user cannot read container images' do context 'when user cannot read container images' do
let(:user) { nil } let(:user) { nil }
it 'the menu item is not added to list of menu items' do it 'the menu item is not added to list of menu items' do
is_expected.to be_nil is_expected.to be_nil
end
end end
end end
end
describe 'Infrastructure Registry' do describe 'Infrastructure Registry' do
subject { described_class.new(context).items.find { |i| i.item_id == :infrastructure_registry }} let(:item_id) { :infrastructure_registry }
context 'when feature flag :infrastructure_registry_page is enabled' do context 'when feature flag :infrastructure_registry_page is enabled' do
it 'the menu item is added to list of menu items' do it 'the menu item is added to list of menu items' do
stub_feature_flags(infrastructure_registry_page: true) stub_feature_flags(infrastructure_registry_page: true)
is_expected.not_to be_nil is_expected.not_to be_nil
end
end end
end
context 'when feature flag :infrastructure_registry_page is disabled' do context 'when feature flag :infrastructure_registry_page is disabled' do
it 'the menu item is not added to list of menu items' do it 'the menu item is not added to list of menu items' do
stub_feature_flags(infrastructure_registry_page: false) stub_feature_flags(infrastructure_registry_page: false)
is_expected.to be_nil is_expected.to be_nil
end
end end
end end
end end
......
...@@ -9,7 +9,7 @@ RSpec.describe Sidebars::Projects::Menus::ProjectInformationMenu do ...@@ -9,7 +9,7 @@ RSpec.describe Sidebars::Projects::Menus::ProjectInformationMenu do
let(:context) { Sidebars::Projects::Context.new(current_user: user, container: project) } let(:context) { Sidebars::Projects::Context.new(current_user: user, container: project) }
describe 'Releases' do describe 'Releases' do
subject { described_class.new(context).items.index { |e| e.item_id == :releases } } subject { described_class.new(context).renderable_items.index { |e| e.item_id == :releases } }
context 'when project repository is empty' do context 'when project repository is empty' do
it 'does not include releases menu item' do it 'does not include releases menu item' do
......
...@@ -11,14 +11,14 @@ RSpec.describe Sidebars::Projects::Menus::SettingsMenu do ...@@ -11,14 +11,14 @@ RSpec.describe Sidebars::Projects::Menus::SettingsMenu do
describe '#render?' do describe '#render?' do
it 'returns false when menu does not have any menu items' do it 'returns false when menu does not have any menu items' do
allow(subject).to receive(:has_items?).and_return(false) allow(subject).to receive(:has_renderable_items?).and_return(false)
expect(subject.render?).to be false expect(subject.render?).to be false
end end
end end
describe 'Menu items' do describe 'Menu items' do
subject { described_class.new(context).items.index { |e| e.item_id == item_id } } subject { described_class.new(context).renderable_items.index { |e| e.item_id == item_id } }
shared_examples 'access rights checks' do shared_examples 'access rights checks' do
specify { is_expected.not_to be_nil } specify { is_expected.not_to be_nil }
......
...@@ -10,7 +10,7 @@ RSpec.describe Sidebars::Projects::Menus::WikiMenu do ...@@ -10,7 +10,7 @@ RSpec.describe Sidebars::Projects::Menus::WikiMenu do
subject { described_class.new(context) } subject { described_class.new(context) }
it 'does not contain any sub menu' do it 'does not contain any sub menu' do
expect(subject.items).to be_empty expect(subject.has_items?).to be false
end end
describe '#render?' do describe '#render?' do
......
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