123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293 |
- # 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, <KnowledgeBase>] :knowledge_base (nil) knowledge base instance
- # @option params [KnowledgeBase::Locale, <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<String>] :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 [Hash<String=>String>, Hash<Symbol=>Symbol>] :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
|