123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- module ApplicationModel::CanLookupSearchIndexAttributes
- extend ActiveSupport::Concern
- class RequestCache < ActiveSupport::CurrentAttributes
- attribute :integer_attribute_names
- def integer_fields(class_name)
- self.integer_attribute_names ||= {}
- updated_at = ObjectManager::Attribute.maximum('updated_at')
- return self.integer_attribute_names[class_name][:data] if self.integer_attribute_names[class_name].present? && self.integer_attribute_names[class_name][:updated_at] == updated_at
- self.integer_attribute_names[class_name] = {
- updated_at: updated_at,
- data: ObjectManager::Attribute.where(object_lookup: ObjectLookup.find_by(name: class_name), data_type: 'integer', editable: true).pluck(:name),
- }
- self.integer_attribute_names[class_name][:data]
- end
- end
- =begin
- This function return the attributes for the elastic search with relation hash values.
- It can be run with parameter include_references: false to skip the relational hashes to prevent endless loops.
- ticket = Ticket.find(3)
- attributes = ticket.search_index_attribute_lookup
- attributes = ticket.search_index_attribute_lookup(include_references: false)
- returns
- attributes # object with lookup data
- =end
- def search_index_attribute_lookup(include_references: true)
- attributes = self.attributes
- self.attributes.each do |key, value|
- break if !include_references
- attribute_name = key.to_s
- # ignore standard attribute if needed
- if self.class.search_index_attribute_ignored?(attribute_name)
- attributes.delete(attribute_name)
- next
- end
- # need value for reference data
- next if !value
- # check if we have a referenced object which we could include here
- next if !search_index_attribute_method(attribute_name)
- # get referenced attribute name
- attribute_ref_name = self.class.search_index_attribute_ref_name(attribute_name)
- next if !attribute_ref_name
- # ignore referenced attributes if needed
- next if self.class.search_index_attribute_ignored?(attribute_ref_name)
- # get referenced attribute value
- value = search_index_value_by_attribute(attribute_name)
- next if !value
- # save name of ref object
- attributes[ attribute_ref_name ] = value
- end
- if is_a? HasObjectManagerAttributes
- RequestCache.integer_fields(self.class.to_s).each do |field|
- next if attributes[field].blank?
- attributes["#{field}_text"] = attributes[field].to_s
- end
- end
- attributes
- end
- =begin
- This function returns the relational search index value based on the attribute name.
- organization = Organization.find(1)
- value = organization.search_index_value_by_attribute('organization_id')
- returns
- value = {"name"=>"Zammad Foundation"}
- =end
- def search_index_value_by_attribute(attribute_name = '')
- # get attribute name
- relation_class = search_index_attribute_method(attribute_name)
- return if !relation_class
- # lookup ref object
- relation_model = relation_class.lookup(id: attributes[attribute_name])
- return if !relation_model
- relation_model.search_index_attribute_lookup(include_references: false)
- end
- =begin
- This function returns the method for the relational search index attribute.
- method = Ticket.new.search_index_attribute_method('organization_id')
- returns
- method = Organization (class)
- =end
- def search_index_attribute_method(attribute_name = '')
- return if attribute_name[-3, 3] != '_id'
- attribute_name = attribute_name[ 0, attribute_name.length - 3 ]
- return if !respond_to?(attribute_name)
- send(attribute_name).class
- end
- class_methods do
- =begin
- This function returns the relational search index attribute name for the given class.
- attribute_ref_name = Organization.search_index_attribute_ref_name('user_id')
- returns
- attribute_ref_name = 'user'
- =end
- def search_index_attribute_ref_name(attribute_name)
- attribute_name[ 0, attribute_name.length - 3 ]
- end
- =begin
- This function returns if a search index attribute should be ignored.
- ignored = Ticket.search_index_attribute_ignored?('organization_id')
- returns
- ignored = false
- =end
- def search_index_attribute_ignored?(attribute_name = '')
- ignored_attributes = instance_variable_get(:@search_index_attributes_ignored) || []
- return if ignored_attributes.blank?
- ignored_attributes.include?(attribute_name.to_sym)
- end
- =begin
- This function returns if a search index attribute is relevant for creating/updating the search index for this object.
- relevant = Ticket.search_index_attribute_relevant?('organization_id')
- returns
- relevant = true
- =end
- def search_index_attribute_relevant?(attribute_name = '')
- relevant_attributes = instance_variable_get(:@search_index_attributes_relevant) || []
- return true if relevant_attributes.blank?
- relevant_attributes.include?(attribute_name.to_sym)
- end
- end
- end
|