# Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ class SearchKnowledgeBaseBackend attr_reader :knowledge_base # @param [Hash] params the paramsused to initialize search instance # @option params [KnowledgeBase, ] :knowledge_base (nil) knowledge base instance # @option params [KnowledgeBase::Locale, , String] :locale (nil) KB Locale or string identifier # @option params [KnowledgeBase::Category] :scope (nil) optional search scope # @option params [Symbol] :flavor (:public) agent or public to indicate source and narrow down to internal or public answers accordingly # @option params [String, Array] :index (nil) indexes to limit search to, searches all indexes if nil # @option params [Integer] :limit per page param for paginatin # @option params [Boolean] :highlight_enabled (true) highlight matching text # @option params [HashString>, HashSymbol>] :order_by hash with column => asc/desc def initialize(params) @params = params.compact prepare_scope_ids end def use_internal_assets? flavor == :agent && KnowledgeBase.granular_permissions? end def search(query, user: nil, pagination: nil) if use_internal_assets? # cache for later use @granular_permissions_handler = KnowledgeBase::InternalAssets.new(user) end raw_results = raw_results(query, pagination: pagination) filtered = filter_results raw_results, user if pagination filtered = filtered.slice pagination.offset, pagination.limit elsif @params[:limit] filtered = filtered.slice 0, @params[:limit] end filtered end def search_fallback(query, indexes) indexes.flat_map { |index| search_fallback_for_index(query, index) } end def search_fallback_for_index(query, index) index .constantize .search_sql_text_fallback("%#{query}%") .apply_kb_scope(@cached_scope_ids) .where(kb_locale: kb_locales) .reorder(**search_fallback_order) .pluck(:id) .map { |id| { id: id, type: index } } end def search_fallback_order @params[:order_by].presence || { updated_at: :desc } end def raw_results(query, pagination: nil) return search_fallback(query, indexes) if !SearchIndexBackend.enabled? SearchIndexBackend .search(query, indexes, options(pagination: pagination)) .map do |hash| hash[:id] = hash[:id].to_i hash end end def filter_results(raw_results, user) raw_results .group_by { |result| result[:type] } .map { |group_name, grouped_results| filter_type(group_name, grouped_results, user) } .flatten end def filter_type(type, grouped_results, user) translation_ids = translation_ids_for_type(type, user) if !translation_ids return [] end grouped_results.select { |result| translation_ids&.include? result[:id].to_i } end def translation_ids_for_type(type, user) case type when KnowledgeBase::Answer::Translation.name translation_ids_for_answers(user) when KnowledgeBase::Category::Translation.name translation_ids_for_categories(user) when KnowledgeBase::Translation.name translation_ids_for_kbs(user) end end def translation_ids_for_answers(user) scope = KnowledgeBase::Answer .joins(:category) .where(knowledge_base_categories: { knowledge_base_id: knowledge_bases }) .then do |relation| if use_internal_assets? # cache for later use relation.where(id: @granular_permissions_handler.all_answer_ids) elsif user&.permissions?('knowledge_base.editor') relation elsif user&.permissions?('knowledge_base.reader') && flavor == :agent relation.internal else relation.published end end flatten_translation_ids(scope) end def translation_ids_for_categories(user) scope = KnowledgeBase::Category.where(knowledge_base_id: knowledge_bases) if use_internal_assets? flatten_translation_ids scope.where(id: @granular_permissions_handler.all_category_ids) elsif user&.permissions?('knowledge_base.editor') flatten_translation_ids scope elsif user&.permissions?('knowledge_base.reader') && flavor == :agent flatten_answer_translation_ids(scope, :internal) else flatten_answer_translation_ids(scope, :public) end end def translation_ids_for_kbs(_user) flatten_translation_ids KnowledgeBase.active.where(id: knowledge_bases) end def indexes return Array(@params.fetch(:index)) if @params.key?(:index) %w[ KnowledgeBase::Answer::Translation KnowledgeBase::Category::Translation KnowledgeBase::Translation ] end def kb_locales @kb_locales ||= begin case @params.fetch(:locale, nil) when KnowledgeBase::Locale Array(@params.fetch(:locale)) when String KnowledgeBase::Locale .joins(:system_locale) .where(knowledge_base_id: knowledge_bases, locales: { locale: @params.fetch(:locale) }) else KnowledgeBase::Locale .where(knowledge_base_id: knowledge_bases) end end end def kb_locales_in(knowledge_base_id) @kb_locales_in ||= {} @kb_locales_in[knowledge_base_id] ||= @kb_locales.select { |locale| locale.knowledge_base_id == knowledge_base_id } end def kb_locale_ids @kb_locale_ids ||= kb_locales.pluck(:id) end def knowledge_bases @knowledge_bases ||= begin if @params.key? :knowledge_base Array(@params.fetch(:knowledge_base)) else KnowledgeBase.active end end end def flavor @params.fetch(:flavor, :public).to_sym end def base_options { query_extension: { bool: { must: [ { terms: { kb_locale_id: kb_locale_ids } } ] } } } end def options_apply_query_fields(hash) return if flavor == :agent hash[:query_fields_by_indexes] = { 'KnowledgeBase::Answer::Translation': %w[title content.body attachment.content tags], 'KnowledgeBase::Category::Translation': %w[title], 'KnowledgeBase::Translation': %w[title] } end def options_apply_highlight(hash) return if !@params.fetch(:highlight_enabled, true) hash[:highlight_fields_by_indexes] = { 'KnowledgeBase::Answer::Translation': %w[title content.body tags], 'KnowledgeBase::Category::Translation': %w[title], 'KnowledgeBase::Translation': %w[title] } end def options_apply_scope(hash) return if !@params.fetch(:scope, nil) hash[:query_extension][:bool][:must].push({ terms: { scope_id: @cached_scope_ids } }) end def options_apply_pagination(hash, pagination) if @params[:from] && @params[:limit] hash[:from] = @params[:from] hash[:limit] = @params[:limit] elsif pagination hash[:from] = 0 hash[:limit] = pagination.limit * 99 end end def options_apply_order(hash) return if @params[:order_by].blank? hash[:sort_by] = @params[:order_by].keys hash[:order_by] = @params[:order_by].values end def options_apply_fulltext(hash) hash[:fulltext] = true end def options(pagination: nil) output = base_options options_apply_query_fields(output) options_apply_highlight(output) options_apply_scope(output) options_apply_pagination(output, pagination) options_apply_order(output) options_apply_fulltext(output) output end def flatten_translation_ids(collection) collection .eager_load(:translations) .map { |elem| elem.translations.pluck(:id) } .flatten end def flatten_answer_translation_ids(collection, visibility) collection .eager_load(:translations) .map { |elem| visible_category_translation_ids(elem, visibility) } .flatten end def visible_category_translation_ids(category, visibility) category .translations .to_a .select { |elem| visible_translation?(elem, visibility) } .pluck(:id) end def visible_translation?(translation, visibility) if kb_locales_in(translation.category.knowledge_base_id).exclude?(translation.kb_locale) return false end translation.category.send(:"#{visibility}_content?", translation.kb_locale) end def prepare_scope_ids return if !@params.key? :scope @cached_scope_ids = @params.fetch(:scope).self_with_children_ids end end