123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- module HasSearchIndexBackend
- extend ActiveSupport::Concern
- included do
- after_commit :search_index_update, if: :persisted?
- after_destroy :search_index_destroy
- end
- =begin
- update search index, if configured - will be executed automatically
- model = Model.find(123)
- model.search_index_update
- =end
- def search_index_update
- return true if ignore_search_indexing?(:update)
- # start background job to transfer data to search index
- return true if !SearchIndexBackend.enabled?
- return true if previous_changes.blank?
- SearchIndexJob.perform_later(self.class.to_s, id)
- SearchIndexAssociationsJob.perform_later(self.class.to_s, id)
- true
- end
- def search_index_indexable
- Models.indexable.reject { |local_class| local_class == self.class }
- end
- def search_index_indexable_attributes(index_class)
- result = []
- index_class.new.attributes.each_key do |key|
- attribute_name = key.to_s
- next if attribute_name.blank?
- # due to performance reasons, we only want to process some attributes for specific classes (e.g. tickets)
- next if !index_class.search_index_attribute_relevant?(attribute_name)
- attribute_ref_name = index_class.search_index_attribute_ref_name(attribute_name)
- next if attribute_ref_name.blank?
- association = index_class.reflect_on_association(attribute_ref_name)
- next if association.blank?
- next if association.options[:polymorphic]
- attribute_class = association.klass
- next if attribute_class.blank?
- next if attribute_class != self.class
- result << {
- name: attribute_name,
- ref_name: attribute_ref_name,
- }
- end
- result
- end
- def search_index_update_delta(index_class:, value:, attribute:)
- raise "Attribute lookup data needs updated_at for delta updates (object: #{self.class}, #{id})" if value['updated_at'].blank?
- data = {
- attribute[:ref_name] => value,
- }
- where = {
- bool: {
- must: [
- {
- term: {
- attribute[:name] => id,
- },
- },
- {
- range: {
- "#{attribute[:ref_name]}.updated_at" => {
- lt: value['updated_at'].strftime('%Y-%m-%dT%H:%M:%S.%LZ'),
- },
- },
- },
- ],
- },
- }
- SearchIndexBackend.update_by_query(index_class.to_s, data, where)
- end
- =begin
- update search index, if configured - will be executed automatically
- model = Organizations.find(123)
- result = model.search_index_update_associations
- returns
- # Updates asscociation data for users and tickets of the organization in this example
- result = true
- =end
- def search_index_update_associations
- return if !SearchIndexBackend.enabled?
- new_search_index_value = search_index_attribute_lookup(include_references: false)
- return if new_search_index_value.blank?
- search_index_indexable.each_with_object([]) do |index_class, result|
- search_index_indexable_attributes(index_class).each do |attribute|
- result << search_index_update_delta(index_class: index_class, value: new_search_index_value, attribute: attribute)
- end
- end
- end
- =begin
- delete search index object, will be executed automatically
- model = Model.find(123)
- model.search_index_destroy
- =end
- def search_index_destroy
- return true if ignore_search_indexing?(:destroy)
- SearchIndexBackend.remove(self.class.to_s, id)
- true
- end
- =begin
- collect data to index and send to backend
- ticket = Ticket.find(123)
- result = ticket.search_index_update_backend
- returns
- result = true # false
- =end
- def search_index_update_backend
- # fill up with search data
- attributes = search_index_attribute_lookup
- return true if !attributes
- # update backend
- SearchIndexBackend.add(self.class.to_s, attributes)
- true
- end
- def ignore_search_indexing?(_action)
- false
- end
- # methods defined here are going to extend the class, not the instance of it
- class_methods do # rubocop:disable Metrics/BlockLength
- =begin
- serve method to ignore model attributes in search index
- class Model < ApplicationModel
- include HasSearchIndexBackend
- search_index_attributes_ignored :password, :image
- end
- =end
- def search_index_attributes_ignored(*attributes)
- @search_index_attributes_ignored = attributes
- end
- def search_index_attributes_relevant(*attributes)
- @search_index_attributes_relevant = attributes
- end
- =begin
- reload search index with full data
- Model.search_index_reload
- =end
- def search_index_reload(silent: false, worker: 0)
- tolerance = 10
- tolerance_count = 0
- query = reorder(created_at: :desc)
- total = query.count
- record_count = 0
- offset = 0
- batch_size = 200
- while query.offset(offset).limit(batch_size).count.positive?
- records = query.offset(offset).limit(batch_size)
- Parallel.map(records, { in_processes: worker }) do |record|
- next if record.ignore_search_indexing?(:destroy)
- begin
- record.search_index_update_backend
- rescue => e
- logger.error "Unable to send #{record.class}.find(#{record.id}).search_index_update_backend backend: #{e.inspect}"
- tolerance_count += 1
- sleep 15
- raise "Unable to send #{record.class}.find(#{record.id}).search_index_update_backend backend: #{e.inspect}" if tolerance_count == tolerance
- end
- end
- offset += batch_size
- next if silent
- record_count += records.count
- if (record_count % batch_size).zero? || record_count == total
- print "\r #{record_count}/#{total}" # rubocop:disable Rails/Output
- end
- end
- end
- end
- end
|