123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- require 'rails_helper'
- require 'models/concerns/checks_kb_client_notification_examples'
- require 'models/contexts/factory_context'
- RSpec.describe KnowledgeBase::Category, current_user_id: 1, type: :model do
- subject(:kb_category) { create(:knowledge_base_category) }
- include_context 'factory'
- it_behaves_like 'ChecksKbClientNotification'
- it { is_expected.to validate_presence_of(:category_icon) }
- it { is_expected.not_to validate_presence_of(:parent_id) }
- it { is_expected.to have_many(:answers) }
- it { is_expected.to have_many(:children) }
- it { is_expected.to have_many(:permissions) }
- it { is_expected.to belong_to(:parent).optional }
- it { is_expected.to belong_to(:knowledge_base) }
- context 'in multilevel tree' do
- subject(:kb_category_with_tree) { create(:kb_category_with_tree) }
- let(:knowledge_base) { kb_category_with_tree.knowledge_base }
- let(:child_category) { kb_category_with_tree.children.sorted.last }
- let(:grandchild_category) { child_category.children.sorted.first }
- it 'tests to fetch all categories in KB' do
- expect(knowledge_base.categories.count).to eq(7)
- end
- it 'fetches root categories' do
- expect(knowledge_base.categories.root).to contain_exactly(kb_category_with_tree)
- end
- it 'fetches direct children' do
- expect(kb_category_with_tree.children.count).to eq 2
- end
- it 'fetches all children' do
- expect(kb_category_with_tree.self_with_children.count).to eq 7
- end
- it 'fetches all parents' do
- expect(grandchild_category.self_with_parents.count).to eq 3
- end
- it 'root category has no parent' do
- expect(kb_category_with_tree.parent).to be_blank
- end
- it 'children category has to have a parent' do
- expect(child_category.parent).to be_present
- end
- context 'when fetching self with children' do
- it 'root category has multiple layers children and matches all KB categories' do
- expect(kb_category_with_tree.self_with_children).to match_array(knowledge_base.categories)
- end
- it 'child category has multiple layers of children' do
- expect(child_category.self_with_children.count).to eq 5
- end
- it 'grandchild category has single layer of children' do
- expect(grandchild_category.self_with_children.count).to eq 3
- end
- end
- context 'when fetchching self with children ids' do
- it 'root category has multiple layers children ids' do
- expect(kb_category_with_tree.self_with_children_ids).to match_array(knowledge_base.category_ids)
- end
- it 'child category has with multiple layers of children ids' do
- expect(child_category.self_with_children_ids.count).to eq 5
- end
- it 'grandchild category has single layer of children ids count' do
- expect(grandchild_category.self_with_children_ids.count).to eq 3
- end
- it 'grandchild category children ids matches direct children ids' do
- expect(grandchild_category.self_with_children_ids).to match_array([grandchild_category.id] + grandchild_category.child_ids)
- end
- end
- context 'when checking if item is a parent of' do
- it 'root category is indirect (and direct) parent of child' do
- expect(child_category).to be_self_parent(kb_category_with_tree)
- end
- it 'root category is indirect parent of grandchild' do
- expect(grandchild_category).to be_self_parent(kb_category_with_tree)
- end
- it 'child category is not a parent of root category' do
- expect(kb_category_with_tree).not_to be_self_parent(grandchild_category)
- end
- end
- end
- describe '#public_content?' do
- shared_examples 'verify visibility in given state' do |state:, is_visible:|
- it "returns #{is_visible} when contains #{state} answer" do
- object = create(:knowledge_base_category, "containing_#{state}")
- expect(object).send is_visible ? :to : :not_to, be_public_content(object.translations.first.kb_locale) # rubocop:disable RSpec/MissingExpectationTargetMethod
- end
- end
- include_examples 'verify visibility in given state', state: :published, is_visible: true
- include_examples 'verify visibility in given state', state: :internal, is_visible: false
- include_examples 'verify visibility in given state', state: :draft, is_visible: false
- include_examples 'verify visibility in given state', state: :archived, is_visible: false
- end
- describe '#internal_content?' do
- shared_examples 'verify visibility in given state' do |state:, is_visible:|
- it "returns #{is_visible} when contains #{state} answer" do
- object = create(:knowledge_base_category, "containing_#{state}")
- expect(object).send is_visible ? :to : :not_to, be_internal_content(object.translations.first.kb_locale) # rubocop:disable RSpec/MissingExpectationTargetMethod
- end
- end
- include_examples 'verify visibility in given state', state: :published, is_visible: true
- include_examples 'verify visibility in given state', state: :internal, is_visible: true
- include_examples 'verify visibility in given state', state: :draft, is_visible: false
- include_examples 'verify visibility in given state', state: :archived, is_visible: false
- end
- describe '#assets', current_user_id: -> { user.id } do
- subject(:assets) { another_category_answer && internal_answer && category.assets }
- include_context 'basic Knowledge Base'
- let(:user) { create(:agent) }
- let(:another_category) { create(:knowledge_base_category, knowledge_base: knowledge_base) }
- let(:another_category_answer) { create(:knowledge_base_answer, :internal, category: another_category) }
- context 'without permissions' do
- it { expect(assets).to include_assets_of category }
- end
- context 'with readable another category' do
- before do
- KnowledgeBase::PermissionsUpdate
- .new(another_category)
- .update! user.roles.first => 'reader'
- end
- it { expect(assets).to include_assets_of category }
- end
- context 'with hidden another category' do
- before do
- KnowledgeBase::PermissionsUpdate
- .new(another_category)
- .update! user.roles.first => 'none'
- end
- it { expect(assets).to include_assets_of category }
- it { expect(assets).not_to include_assets_of another_category }
- context 'with published answer' do
- let(:another_category_published_answer) { create(:knowledge_base_answer, :published, category: another_category) }
- before { another_category_published_answer }
- it { expect(assets).to include_assets_of category }
- end
- end
- end
- end
|