|
@@ -1,1648 +1,22 @@
|
|
|
# Copyright (C) 2012-2016 Zammad Foundation, http://zammad-foundation.org/
|
|
|
|
|
|
class ApplicationModel < ActiveRecord::Base
|
|
|
- include ApplicationModel::Assets
|
|
|
- include ApplicationModel::HistoryLogBase
|
|
|
- include ApplicationModel::ActivityStreamBase
|
|
|
- include ApplicationModel::SearchIndexBase
|
|
|
+ include ApplicationModel::ActivityStreamLoggable
|
|
|
+ include ApplicationModel::Cacheable
|
|
|
+ include ApplicationModel::CanLookup
|
|
|
+ include ApplicationModel::CanLookupSearchIndexAttributes
|
|
|
+ include ApplicationModel::ChecksAttributeLength
|
|
|
+ include ApplicationModel::CleansParam
|
|
|
+ include ApplicationModel::CleansRecentView
|
|
|
+ include ApplicationModel::FillsByUserColumns
|
|
|
+ include ApplicationModel::HandlesCreatesAndUpdates
|
|
|
+ include ApplicationModel::HasAssets
|
|
|
+ include ApplicationModel::HasAssociations
|
|
|
+ include ApplicationModel::HasAttachments
|
|
|
+ include ApplicationModel::HasLatestChangeTimestamp
|
|
|
+ include ApplicationModel::Importable
|
|
|
+ include ApplicationModel::HistoryLoggable
|
|
|
+ include ApplicationModel::TouchesReferences
|
|
|
|
|
|
self.abstract_class = true
|
|
|
-
|
|
|
- before_create :check_attributes_protected, :check_limits, :cache_delete, :fill_up_user_create
|
|
|
- before_update :check_limits, :fill_up_user_update
|
|
|
- before_destroy :destroy_dependencies
|
|
|
-
|
|
|
- after_create :cache_delete
|
|
|
- after_update :cache_delete
|
|
|
- after_touch :cache_delete
|
|
|
- after_destroy :cache_delete
|
|
|
-
|
|
|
- after_create :attachments_buffer_check
|
|
|
- after_update :attachments_buffer_check
|
|
|
-
|
|
|
- after_create :activity_stream_create
|
|
|
- after_update :activity_stream_update
|
|
|
- before_destroy :activity_stream_destroy
|
|
|
-
|
|
|
- after_create :history_create
|
|
|
- after_update :history_update
|
|
|
- after_destroy :history_destroy
|
|
|
-
|
|
|
- after_create :search_index_update
|
|
|
- after_update :search_index_update
|
|
|
- after_touch :search_index_update
|
|
|
- after_destroy :search_index_destroy
|
|
|
-
|
|
|
- before_destroy :recent_view_destroy
|
|
|
-
|
|
|
- # create instance accessor
|
|
|
- class << self
|
|
|
- attr_accessor :activity_stream_support_config, :history_support_config, :search_index_support_config, :attributes_with_associations_support_config
|
|
|
- end
|
|
|
-
|
|
|
- attr_accessor :history_changes_last_done
|
|
|
-
|
|
|
- def check_attributes_protected
|
|
|
-
|
|
|
- import_class_list = ['Ticket', 'Ticket::Article', 'History', 'Ticket::State', 'Ticket::StateType', 'Ticket::Priority', 'Group', 'User', 'Role' ]
|
|
|
-
|
|
|
- # do noting, use id as it is
|
|
|
- return if !Setting.get('system_init_done')
|
|
|
- return if Setting.get('import_mode') && import_class_list.include?(self.class.to_s)
|
|
|
-
|
|
|
- self[:id] = nil
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-remove all not used model attributes of params
|
|
|
-
|
|
|
- result = Model.param_cleanup(params)
|
|
|
-
|
|
|
- for object creation, ignore id's
|
|
|
-
|
|
|
- result = Model.param_cleanup(params, true)
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- result = params # params with valid attributes of model
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.param_cleanup(params, new_object = false)
|
|
|
-
|
|
|
- if params.respond_to?('permit!')
|
|
|
- params.permit!
|
|
|
- end
|
|
|
-
|
|
|
- if params.nil?
|
|
|
- raise ArgumentError, "No params for #{self}!"
|
|
|
- end
|
|
|
-
|
|
|
- data = {}
|
|
|
- params.each { |key, value|
|
|
|
- data[key.to_sym] = value
|
|
|
- }
|
|
|
-
|
|
|
- # ignore id for new objects
|
|
|
- if new_object && params[:id]
|
|
|
- data.delete(:id)
|
|
|
- end
|
|
|
-
|
|
|
- # only use object attributes
|
|
|
- clean_params = {}
|
|
|
- new.attributes.each { |attribute, _value|
|
|
|
- next if !data.key?(attribute.to_sym)
|
|
|
-
|
|
|
- # check reference records, referenced by _id attributes
|
|
|
- reflect_on_all_associations.map { |assoc|
|
|
|
- class_name = assoc.options[:class_name]
|
|
|
- next if !class_name
|
|
|
- name = "#{assoc.name}_id".to_sym
|
|
|
- next if !data.key?(name)
|
|
|
- next if data[name].blank?
|
|
|
- next if assoc.klass.lookup(id: data[name])
|
|
|
- raise ArgumentError, "Invalid value for param '#{name}': #{data[name].inspect}"
|
|
|
- }
|
|
|
- clean_params[attribute.to_sym] = data[attribute.to_sym]
|
|
|
- }
|
|
|
-
|
|
|
- # we do want to set this via database
|
|
|
- param_validation(clean_params)
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-set relations of model based on params
|
|
|
-
|
|
|
- model = Model.find(1)
|
|
|
- result = model.param_set_associations(params)
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- result = true|false
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def param_set_associations(params)
|
|
|
-
|
|
|
- # set relations by id/verify if ref exists
|
|
|
- self.class.reflect_on_all_associations.map { |assoc|
|
|
|
- real_ids = assoc.name.to_s[0, assoc.name.to_s.length - 1] + '_ids'
|
|
|
- real_ids = real_ids.to_sym
|
|
|
- next if !params.key?(real_ids)
|
|
|
- list_of_items = params[real_ids]
|
|
|
- if params[real_ids].class != Array
|
|
|
- list_of_items = [ params[real_ids] ]
|
|
|
- end
|
|
|
- list = []
|
|
|
- list_of_items.each { |item_id|
|
|
|
- next if !item_id
|
|
|
- lookup = assoc.klass.lookup(id: item_id)
|
|
|
-
|
|
|
- # complain if we found no reference
|
|
|
- if !lookup
|
|
|
- raise ArgumentError, "No value found for '#{assoc.name}' with id #{item_id.inspect}"
|
|
|
- end
|
|
|
- list.push item_id
|
|
|
- }
|
|
|
- #p "SEND #{real_ids} = #{list.inspect}"
|
|
|
- send("#{real_ids}=", list)
|
|
|
- }
|
|
|
-
|
|
|
- # set relations by name/lookup
|
|
|
- self.class.reflect_on_all_associations.map { |assoc|
|
|
|
- real_ids = assoc.name.to_s[0, assoc.name.to_s.length - 1] + '_ids'
|
|
|
- next if !respond_to?(real_ids)
|
|
|
- real_values = assoc.name.to_s[0, assoc.name.to_s.length - 1] + 's'
|
|
|
- real_values = real_values.to_sym
|
|
|
- next if !respond_to?(real_values)
|
|
|
- next if !params[real_values]
|
|
|
- next if params[real_values].class != Array
|
|
|
- list = []
|
|
|
- class_object = assoc.klass
|
|
|
- params[real_values].each { |value|
|
|
|
- lookup = nil
|
|
|
- if class_object == User
|
|
|
- if !lookup
|
|
|
- lookup = class_object.lookup(login: value)
|
|
|
- end
|
|
|
- if !lookup
|
|
|
- lookup = class_object.lookup(email: value)
|
|
|
- end
|
|
|
- else
|
|
|
- lookup = class_object.lookup(name: value)
|
|
|
- end
|
|
|
-
|
|
|
- # complain if we found no reference
|
|
|
- if !lookup
|
|
|
- raise ArgumentError, "No lookup value found for '#{assoc.name}': #{value.inspect}"
|
|
|
- end
|
|
|
- list.push lookup.id
|
|
|
- }
|
|
|
- #p "SEND #{real_ids} = #{list.inspect}"
|
|
|
- send("#{real_ids}=", list)
|
|
|
- }
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-get relations of model based on params
|
|
|
-
|
|
|
- model = Model.find(1)
|
|
|
- attributes = model.attributes_with_associations
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- hash with attributes and association ids
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def attributes_with_associations
|
|
|
-
|
|
|
- key = "#{self.class}::aws::#{id}"
|
|
|
- cache = Cache.get(key)
|
|
|
- return cache if cache
|
|
|
-
|
|
|
- # get relations
|
|
|
- attributes = self.attributes
|
|
|
- self.class.reflect_on_all_associations.map { |assoc|
|
|
|
- real_ids = assoc.name.to_s[0, assoc.name.to_s.length - 1] + '_ids'
|
|
|
- next if self.class.attributes_with_associations_support_config && self.class.attributes_with_associations_support_config[:ignore][real_ids.to_sym] == true
|
|
|
- next if !respond_to?(real_ids)
|
|
|
- attributes[real_ids] = send(real_ids)
|
|
|
- }
|
|
|
- Cache.write(key, attributes)
|
|
|
- attributes
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-get relation name of model based on params
|
|
|
-
|
|
|
- model = Model.find(1)
|
|
|
- attributes = model.attributes_with_relation_names
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- hash with attributes, association ids, association names and relation name
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def attributes_with_relation_names
|
|
|
-
|
|
|
- # get relations
|
|
|
- attributes = attributes_with_associations
|
|
|
- self.class.reflect_on_all_associations.map { |assoc|
|
|
|
- next if !respond_to?(assoc.name)
|
|
|
- ref = send(assoc.name)
|
|
|
- next if !ref
|
|
|
- if ref.respond_to?(:first)
|
|
|
- attributes[assoc.name.to_s] = []
|
|
|
- ref.each { |item|
|
|
|
- if item[:login]
|
|
|
- attributes[assoc.name.to_s].push item[:login]
|
|
|
- next
|
|
|
- end
|
|
|
- next if !item[:name]
|
|
|
- attributes[assoc.name.to_s].push item[:name]
|
|
|
- }
|
|
|
- if ref.count.positive? && attributes[assoc.name.to_s].empty?
|
|
|
- attributes.delete(assoc.name.to_s)
|
|
|
- end
|
|
|
- next
|
|
|
- end
|
|
|
- if ref[:login]
|
|
|
- attributes[assoc.name.to_s] = ref[:login]
|
|
|
- next
|
|
|
- end
|
|
|
- next if !ref[:name]
|
|
|
- attributes[assoc.name.to_s] = ref[:name]
|
|
|
- }
|
|
|
-
|
|
|
- # fill created_by/updated_by
|
|
|
- {
|
|
|
- 'created_by_id' => 'created_by',
|
|
|
- 'updated_by_id' => 'updated_by',
|
|
|
- }.each { |source, destination|
|
|
|
- next if !attributes[source]
|
|
|
- user = User.lookup(id: attributes[source])
|
|
|
- next if !user
|
|
|
- attributes[destination] = user.login
|
|
|
- }
|
|
|
-
|
|
|
- # remove forbitten attributes
|
|
|
- %w(password token tokens token_ids).each { |item|
|
|
|
- attributes.delete(item)
|
|
|
- }
|
|
|
-
|
|
|
- attributes
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-remove all not used params of object (per default :updated_at, :created_at, :updated_by_id and :created_by_id)
|
|
|
-
|
|
|
- result = Model.param_validation(params)
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- result = params # params without listed attributes
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.param_validation(data)
|
|
|
-
|
|
|
- # we do want to set this via database
|
|
|
- [:action, :controller, :updated_at, :created_at, :updated_by_id, :created_by_id, :updated_by, :created_by].each { |key|
|
|
|
- data.delete(key)
|
|
|
- }
|
|
|
-
|
|
|
- data
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-do name/login/email based lookup for associations
|
|
|
-
|
|
|
- params = {
|
|
|
- login: 'some login',
|
|
|
- firstname: 'some firstname',
|
|
|
- lastname: 'some lastname',
|
|
|
- email: 'some email',
|
|
|
- organization: 'some organization',
|
|
|
- roles: ['Agent', 'Admin'],
|
|
|
- }
|
|
|
-
|
|
|
- attributes = Model.param_association_lookup(params)
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- attributes = params # params with possible lookups
|
|
|
-
|
|
|
- attributes = {
|
|
|
- login: 'some login',
|
|
|
- firstname: 'some firstname',
|
|
|
- lastname: 'some lastname',
|
|
|
- email: 'some email',
|
|
|
- organization_id: 123,
|
|
|
- role_ids: [2,1],
|
|
|
- }
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.param_association_lookup(params)
|
|
|
-
|
|
|
- data = {}
|
|
|
- params.each { |key, value|
|
|
|
- data[key.to_sym] = value
|
|
|
- }
|
|
|
-
|
|
|
- data.symbolize_keys!
|
|
|
- available_attributes = attribute_names
|
|
|
- reflect_on_all_associations.map { |assoc|
|
|
|
- value = data[assoc.name.to_sym]
|
|
|
- next if !value # next if we do not have a value
|
|
|
- ref_name = "#{assoc.name}_id"
|
|
|
-
|
|
|
- # handle _id values
|
|
|
- if available_attributes.include?(ref_name) # if we do have an _id attribute
|
|
|
- next if data[ref_name.to_sym] # next if we have already the _id filled
|
|
|
-
|
|
|
- # get association class and do lookup
|
|
|
- class_object = assoc.klass
|
|
|
- lookup = nil
|
|
|
- if class_object == User
|
|
|
- if value.class == String
|
|
|
- if !lookup
|
|
|
- lookup = class_object.lookup(login: value)
|
|
|
- end
|
|
|
- if !lookup
|
|
|
- lookup = class_object.lookup(email: value)
|
|
|
- end
|
|
|
- else
|
|
|
- raise ArgumentError, "String is needed as ref value #{value.inspect} for '#{assoc.name}'"
|
|
|
- end
|
|
|
- else
|
|
|
- lookup = class_object.lookup(name: value)
|
|
|
- end
|
|
|
-
|
|
|
- # complain if we found no reference
|
|
|
- if !lookup
|
|
|
- raise ArgumentError, "No lookup value found for '#{assoc.name}': #{value.inspect}"
|
|
|
- end
|
|
|
-
|
|
|
- # release data value
|
|
|
- data.delete(assoc.name.to_sym)
|
|
|
-
|
|
|
- # remember id reference
|
|
|
- data[ref_name.to_sym] = lookup.id
|
|
|
- next
|
|
|
- end
|
|
|
-
|
|
|
- next if value.class != Array
|
|
|
- next if value.empty?
|
|
|
- next if value[0].class != String
|
|
|
-
|
|
|
- # handle _ids values
|
|
|
- ref_names = "#{assoc.name[0, assoc.name.length - 1]}_ids"
|
|
|
- generic_object_tmp = new
|
|
|
- next unless generic_object_tmp.respond_to?(ref_names) # if we do have an _ids attribute
|
|
|
- next if data[ref_names.to_sym] # next if we have already the _ids filled
|
|
|
-
|
|
|
- # get association class and do lookup
|
|
|
- class_object = assoc.klass
|
|
|
- lookup_ids = []
|
|
|
- value.each { |item|
|
|
|
- lookup = nil
|
|
|
- if class_object == User
|
|
|
- if item.class == String
|
|
|
- if !lookup
|
|
|
- lookup = class_object.lookup(login: item)
|
|
|
- end
|
|
|
- if !lookup
|
|
|
- lookup = class_object.lookup(email: item)
|
|
|
- end
|
|
|
- else
|
|
|
- raise ArgumentError, "String is needed in array ref as ref value #{value.inspect} for '#{assoc.name}'"
|
|
|
- end
|
|
|
- else
|
|
|
- lookup = class_object.lookup(name: item)
|
|
|
- end
|
|
|
-
|
|
|
- # complain if we found no reference
|
|
|
- if !lookup
|
|
|
- raise ArgumentError, "No lookup value found for '#{assoc.name}': #{item.inspect}"
|
|
|
- end
|
|
|
- lookup_ids.push lookup.id
|
|
|
- }
|
|
|
-
|
|
|
- # release data value
|
|
|
- data.delete(assoc.name.to_sym)
|
|
|
-
|
|
|
- # remember id reference
|
|
|
- data[ref_names.to_sym] = lookup_ids
|
|
|
- }
|
|
|
-
|
|
|
- data
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-reference if association id check
|
|
|
-
|
|
|
- model = Model.find(123)
|
|
|
- attributes = model.association_id_check('attribute_id', value)
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- true | false
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def association_id_check(attribute_id, value)
|
|
|
- return true if value.nil?
|
|
|
-
|
|
|
- attributes.each { |key, _value|
|
|
|
- next if key != attribute_id
|
|
|
-
|
|
|
- # check if id is assigned
|
|
|
- key_short = key[ key.length - 3, key.length ]
|
|
|
- next if key_short != '_id'
|
|
|
- key_short = key[ 0, key.length - 3 ]
|
|
|
-
|
|
|
- self.class.reflect_on_all_associations.map { |assoc|
|
|
|
- next if assoc.name.to_s != key_short
|
|
|
- item = assoc.class_name.constantize
|
|
|
- return false if !item.respond_to?(:find_by)
|
|
|
- ref_object = item.find_by(id: value)
|
|
|
- return false if !ref_object
|
|
|
- return true
|
|
|
- }
|
|
|
- }
|
|
|
- true
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-set created_by_id & updated_by_id if not given based on UserInfo (current session)
|
|
|
-
|
|
|
-Used as before_create callback, no own use needed
|
|
|
-
|
|
|
- result = Model.fill_up_user_create(params)
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- result = params # params with updated_by_id & created_by_id if not given based on UserInfo (current session)
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def fill_up_user_create
|
|
|
- if self.class.column_names.include? 'updated_by_id'
|
|
|
- if UserInfo.current_user_id
|
|
|
- if updated_by_id && updated_by_id != UserInfo.current_user_id
|
|
|
- logger.info "NOTICE create - self.updated_by_id is different: #{updated_by_id}/#{UserInfo.current_user_id}"
|
|
|
- end
|
|
|
- self.updated_by_id = UserInfo.current_user_id
|
|
|
- end
|
|
|
- end
|
|
|
-
|
|
|
- return if !self.class.column_names.include? 'created_by_id'
|
|
|
-
|
|
|
- return if !UserInfo.current_user_id
|
|
|
-
|
|
|
- if created_by_id && created_by_id != UserInfo.current_user_id
|
|
|
- logger.info "NOTICE create - self.created_by_id is different: #{created_by_id}/#{UserInfo.current_user_id}"
|
|
|
- end
|
|
|
- self.created_by_id = UserInfo.current_user_id
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-set updated_by_id if not given based on UserInfo (current session)
|
|
|
-
|
|
|
-Used as before_update callback, no own use needed
|
|
|
-
|
|
|
- result = Model.fill_up_user_update(params)
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- result = params # params with updated_by_id & created_by_id if not given based on UserInfo (current session)
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def fill_up_user_update
|
|
|
- return if !self.class.column_names.include? 'updated_by_id'
|
|
|
- return if !UserInfo.current_user_id
|
|
|
-
|
|
|
- self.updated_by_id = UserInfo.current_user_id
|
|
|
- end
|
|
|
-
|
|
|
- def cache_update(o)
|
|
|
- cache_delete if respond_to?('cache_delete')
|
|
|
- o.cache_delete if o.respond_to?('cache_delete')
|
|
|
- end
|
|
|
-
|
|
|
- def cache_delete
|
|
|
- keys = []
|
|
|
-
|
|
|
- # delete by id caches
|
|
|
- keys.push "#{self.class}::#{id}"
|
|
|
-
|
|
|
- # delete by id with attributes_with_associations caches
|
|
|
- keys.push "#{self.class}::aws::#{id}"
|
|
|
-
|
|
|
- # delete by name caches
|
|
|
- if self[:name]
|
|
|
- keys.push "#{self.class}::#{name}"
|
|
|
- end
|
|
|
-
|
|
|
- # delete by login caches
|
|
|
- if self[:login]
|
|
|
- keys.push "#{self.class}::#{login}"
|
|
|
- end
|
|
|
-
|
|
|
- keys.each { |key|
|
|
|
- Cache.delete(key)
|
|
|
- }
|
|
|
-
|
|
|
- # delete old name / login caches
|
|
|
- if changed?
|
|
|
- if changes.key?('name')
|
|
|
- name = changes['name'][0]
|
|
|
- key = "#{self.class}::#{name}"
|
|
|
- Cache.delete(key)
|
|
|
- end
|
|
|
- if changes.key?('login')
|
|
|
- name = changes['login'][0]
|
|
|
- key = "#{self.class}::#{name}"
|
|
|
- Cache.delete(key)
|
|
|
- end
|
|
|
- end
|
|
|
-
|
|
|
- end
|
|
|
-
|
|
|
- def self.cache_set(data_id, data)
|
|
|
- key = "#{self}::#{data_id}"
|
|
|
- Cache.write(key, data)
|
|
|
- end
|
|
|
-
|
|
|
- def self.cache_get(data_id)
|
|
|
- key = "#{self}::#{data_id}"
|
|
|
- Cache.get(key)
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-generate uniq name (will check name of model and generates _1 sequenze)
|
|
|
-
|
|
|
-Used as before_update callback, no own use needed
|
|
|
-
|
|
|
- name = Model.genrate_uniq_name('some name')
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- result = 'some name_X'
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.genrate_uniq_name(name)
|
|
|
- return name if !find_by(name: name)
|
|
|
- (1..100).each { |counter|
|
|
|
- name = "#{name}_#{counter}"
|
|
|
- exists = find_by(name: name)
|
|
|
- next if exists
|
|
|
- break
|
|
|
- }
|
|
|
- name
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-lookup model from cache (if exists) or retrieve it from db, id, name, login or email possible
|
|
|
-
|
|
|
- result = Model.lookup(id: 123)
|
|
|
- result = Model.lookup(name: 'some name')
|
|
|
- result = Model.lookup(login: 'some login')
|
|
|
- result = Model.lookup(email: 'some login')
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- result = model # with all attributes
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.lookup(data)
|
|
|
- if data[:id]
|
|
|
- cache = cache_get(data[:id])
|
|
|
- return cache if cache
|
|
|
-
|
|
|
- record = find_by(id: data[:id])
|
|
|
- cache_set(data[:id], record)
|
|
|
- return record
|
|
|
- elsif data[:name]
|
|
|
- cache = cache_get(data[:name])
|
|
|
- return cache if cache
|
|
|
-
|
|
|
- # do lookup with == to handle case insensitive databases
|
|
|
- records = if Rails.application.config.db_case_sensitive
|
|
|
- where('LOWER(name) = LOWER(?)', data[:name])
|
|
|
- else
|
|
|
- where(name: data[:name])
|
|
|
- end
|
|
|
- records.each { |loop_record|
|
|
|
- if loop_record.name == data[:name]
|
|
|
- cache_set(data[:name], loop_record)
|
|
|
- return loop_record
|
|
|
- end
|
|
|
- }
|
|
|
- return
|
|
|
- elsif data[:login]
|
|
|
- cache = cache_get(data[:login])
|
|
|
- return cache if cache
|
|
|
-
|
|
|
- # do lookup with == to handle case insensitive databases
|
|
|
- records = if Rails.application.config.db_case_sensitive
|
|
|
- where('LOWER(login) = LOWER(?)', data[:login])
|
|
|
- else
|
|
|
- where(login: data[:login])
|
|
|
- end
|
|
|
- records.each { |loop_record|
|
|
|
- if loop_record.login == data[:login]
|
|
|
- cache_set(data[:login], loop_record)
|
|
|
- return loop_record
|
|
|
- end
|
|
|
- }
|
|
|
- return
|
|
|
- elsif data[:email]
|
|
|
- cache = cache_get(data[:email])
|
|
|
- return cache if cache
|
|
|
-
|
|
|
- # do lookup with == to handle case insensitive databases
|
|
|
- records = if Rails.application.config.db_case_sensitive
|
|
|
- where('LOWER(email) = LOWER(?)', data[:email])
|
|
|
- else
|
|
|
- where(email: data[:email])
|
|
|
- end
|
|
|
- records.each { |loop_record|
|
|
|
- if loop_record.email == data[:email]
|
|
|
- cache_set(data[:email], loop_record)
|
|
|
- return loop_record
|
|
|
- end
|
|
|
- }
|
|
|
- return
|
|
|
- end
|
|
|
-
|
|
|
- raise ArgumentError, 'Need name, id, login or email for lookup()'
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-create model if not exists (check exists based on id, name, login, email or locale)
|
|
|
-
|
|
|
- result = Model.create_if_not_exists(attributes)
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- result = model # with all attributes
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.create_if_not_exists(data)
|
|
|
- if data[:id]
|
|
|
- record = find_by(id: data[:id])
|
|
|
- return record if record
|
|
|
- elsif data[:name]
|
|
|
-
|
|
|
- # do lookup with == to handle case insensitive databases
|
|
|
- records = if Rails.application.config.db_case_sensitive
|
|
|
- where('LOWER(name) = LOWER(?)', data[:name])
|
|
|
- else
|
|
|
- where(name: data[:name])
|
|
|
- end
|
|
|
- records.each { |loop_record|
|
|
|
- return loop_record if loop_record.name == data[:name]
|
|
|
- }
|
|
|
- elsif data[:login]
|
|
|
-
|
|
|
- # do lookup with == to handle case insensitive databases
|
|
|
- records = if Rails.application.config.db_case_sensitive
|
|
|
- where('LOWER(login) = LOWER(?)', data[:login])
|
|
|
- else
|
|
|
- where(login: data[:login])
|
|
|
- end
|
|
|
- records.each { |loop_record|
|
|
|
- return loop_record if loop_record.login == data[:login]
|
|
|
- }
|
|
|
- elsif data[:email]
|
|
|
-
|
|
|
- # do lookup with == to handle case insensitive databases
|
|
|
- records = if Rails.application.config.db_case_sensitive
|
|
|
- where('LOWER(email) = LOWER(?)', data[:email])
|
|
|
- else
|
|
|
- where(email: data[:email])
|
|
|
- end
|
|
|
- records.each { |loop_record|
|
|
|
- return loop_record if loop_record.email == data[:email]
|
|
|
- }
|
|
|
- elsif data[:locale] && data[:source]
|
|
|
-
|
|
|
- # do lookup with == to handle case insensitive databases
|
|
|
- records = if Rails.application.config.db_case_sensitive
|
|
|
- where('LOWER(locale) = LOWER(?) AND LOWER(source) = LOWER(?)', data[:locale], data[:source])
|
|
|
- else
|
|
|
- where(locale: data[:locale], source: data[:source])
|
|
|
- end
|
|
|
- records.each { |loop_record|
|
|
|
- return loop_record if loop_record.source == data[:source]
|
|
|
- }
|
|
|
- end
|
|
|
- create(data)
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-Model.create_if_not_exists with ref lookups
|
|
|
-
|
|
|
- result = Model.create_if_not_exists_with_ref(attributes)
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- result = model # with all attributes
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.create_if_not_exists_with_ref(data)
|
|
|
- data = param_association_lookup(data)
|
|
|
- create_or_update(data)
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-create or update model (check exists based on id, name, login, email or locale)
|
|
|
-
|
|
|
- result = Model.create_or_update(attributes)
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- result = model # with all attributes
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.create_or_update(data)
|
|
|
- if data[:id]
|
|
|
- record = find_by(id: data[:id])
|
|
|
- if record
|
|
|
- record.update_attributes(data)
|
|
|
- return record
|
|
|
- end
|
|
|
- record = new(data)
|
|
|
- record.save
|
|
|
- return record
|
|
|
- elsif data[:name]
|
|
|
-
|
|
|
- # do lookup with == to handle case insensitive databases
|
|
|
- records = if Rails.application.config.db_case_sensitive
|
|
|
- where('LOWER(name) = LOWER(?)', data[:name])
|
|
|
- else
|
|
|
- where(name: data[:name])
|
|
|
- end
|
|
|
- records.each { |loop_record|
|
|
|
- if loop_record.name == data[:name]
|
|
|
- loop_record.update_attributes(data)
|
|
|
- return loop_record
|
|
|
- end
|
|
|
- }
|
|
|
- record = new(data)
|
|
|
- record.save
|
|
|
- return record
|
|
|
- elsif data[:login]
|
|
|
-
|
|
|
- # do lookup with == to handle case insensitive databases
|
|
|
- records = if Rails.application.config.db_case_sensitive
|
|
|
- where('LOWER(login) = LOWER(?)', data[:login])
|
|
|
- else
|
|
|
- where(login: data[:login])
|
|
|
- end
|
|
|
- records.each { |loop_record|
|
|
|
- if loop_record.login.casecmp(data[:login]).zero?
|
|
|
- loop_record.update_attributes(data)
|
|
|
- return loop_record
|
|
|
- end
|
|
|
- }
|
|
|
- record = new(data)
|
|
|
- record.save
|
|
|
- return record
|
|
|
- elsif data[:email]
|
|
|
-
|
|
|
- # do lookup with == to handle case insensitive databases
|
|
|
- records = if Rails.application.config.db_case_sensitive
|
|
|
- where('LOWER(email) = LOWER(?)', data[:email])
|
|
|
- else
|
|
|
- where(email: data[:email])
|
|
|
- end
|
|
|
- records.each { |loop_record|
|
|
|
- if loop_record.email.casecmp(data[:email]).zero?
|
|
|
- loop_record.update_attributes(data)
|
|
|
- return loop_record
|
|
|
- end
|
|
|
- }
|
|
|
- record = new(data)
|
|
|
- record.save
|
|
|
- return record
|
|
|
- elsif data[:locale]
|
|
|
-
|
|
|
- # do lookup with == to handle case insensitive databases
|
|
|
- records = if Rails.application.config.db_case_sensitive
|
|
|
- where('LOWER(locale) = LOWER(?)', data[:locale])
|
|
|
- else
|
|
|
- where(locale: data[:locale])
|
|
|
- end
|
|
|
- records.each { |loop_record|
|
|
|
- if loop_record.locale.casecmp(data[:locale]).zero?
|
|
|
- loop_record.update_attributes(data)
|
|
|
- return loop_record
|
|
|
- end
|
|
|
- }
|
|
|
- record = new(data)
|
|
|
- record.save
|
|
|
- return record
|
|
|
- else
|
|
|
- raise ArgumentError, 'Need name, login, email or locale for create_or_update()'
|
|
|
- end
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-Model.create_or_update with ref lookups
|
|
|
-
|
|
|
- result = Model.create_or_update(attributes)
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- result = model # with all attributes
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.create_or_update_with_ref(data)
|
|
|
- data = param_association_lookup(data)
|
|
|
- create_or_update(data)
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-activate latest change on create, update, touch and destroy
|
|
|
-
|
|
|
-class Model < ApplicationModel
|
|
|
- latest_change_support
|
|
|
-end
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.latest_change_support
|
|
|
- after_create :latest_change_set_from_observer
|
|
|
- after_update :latest_change_set_from_observer
|
|
|
- after_touch :latest_change_set_from_observer
|
|
|
- after_destroy :latest_change_set_from_observer_destroy
|
|
|
- end
|
|
|
-
|
|
|
- def latest_change_set_from_observer
|
|
|
- self.class.latest_change_set(updated_at)
|
|
|
- end
|
|
|
-
|
|
|
- def latest_change_set_from_observer_destroy
|
|
|
- self.class.latest_change_set(nil)
|
|
|
- end
|
|
|
-
|
|
|
- def self.latest_change_set(updated_at)
|
|
|
- key = "#{new.class.name}_latest_change"
|
|
|
- expires_in = 31_536_000 # 1 year
|
|
|
-
|
|
|
- if updated_at.nil?
|
|
|
- Cache.delete(key)
|
|
|
- else
|
|
|
- Cache.write(key, updated_at, { expires_in: expires_in })
|
|
|
- end
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
- get latest updated_at object timestamp
|
|
|
-
|
|
|
- latest_change = Ticket.latest_change
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- result = timestamp
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.latest_change
|
|
|
- key = "#{new.class.name}_latest_change"
|
|
|
- updated_at = Cache.get(key)
|
|
|
-
|
|
|
- # if we do not have it cached, do lookup
|
|
|
- if !updated_at
|
|
|
- o = select(:updated_at).order(updated_at: :desc).limit(1).first
|
|
|
- if o
|
|
|
- updated_at = o.updated_at
|
|
|
- latest_change_set(updated_at)
|
|
|
- end
|
|
|
- end
|
|
|
- updated_at
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-activate client notify support on create, update, touch and destroy
|
|
|
-
|
|
|
-class Model < ApplicationModel
|
|
|
- notify_clients_support
|
|
|
-end
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.notify_clients_support
|
|
|
- after_create :notify_clients_after_create
|
|
|
- after_update :notify_clients_after_update
|
|
|
- after_touch :notify_clients_after_touch
|
|
|
- after_destroy :notify_clients_after_destroy
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-notify_clients_after_create after model got created
|
|
|
-
|
|
|
-used as callback in model file
|
|
|
-
|
|
|
-class OwnModel < ApplicationModel
|
|
|
- after_create :notify_clients_after_create
|
|
|
- after_update :notify_clients_after_update
|
|
|
- after_touch :notify_clients_after_touch
|
|
|
- after_destroy :notify_clients_after_destroy
|
|
|
-
|
|
|
- [...]
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def notify_clients_after_create
|
|
|
-
|
|
|
- # return if we run import mode
|
|
|
- return if Setting.get('import_mode')
|
|
|
- logger.debug "#{self.class.name}.find(#{id}) notify created " + created_at.to_s
|
|
|
- class_name = self.class.name
|
|
|
- class_name.gsub!(/::/, '')
|
|
|
- PushMessages.send(
|
|
|
- message: {
|
|
|
- event: class_name + ':create',
|
|
|
- data: { id: id, updated_at: updated_at }
|
|
|
- },
|
|
|
- type: 'authenticated',
|
|
|
- )
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-notify_clients_after_update after model got updated
|
|
|
-
|
|
|
-used as callback in model file
|
|
|
-
|
|
|
-class OwnModel < ApplicationModel
|
|
|
- after_create :notify_clients_after_create
|
|
|
- after_update :notify_clients_after_update
|
|
|
- after_touch :notify_clients_after_touch
|
|
|
- after_destroy :notify_clients_after_destroy
|
|
|
-
|
|
|
- [...]
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def notify_clients_after_update
|
|
|
-
|
|
|
- # return if we run import mode
|
|
|
- return if Setting.get('import_mode')
|
|
|
- logger.debug "#{self.class.name}.find(#{id}) notify UPDATED " + updated_at.to_s
|
|
|
- class_name = self.class.name
|
|
|
- class_name.gsub!(/::/, '')
|
|
|
- PushMessages.send(
|
|
|
- message: {
|
|
|
- event: class_name + ':update',
|
|
|
- data: { id: id, updated_at: updated_at }
|
|
|
- },
|
|
|
- type: 'authenticated',
|
|
|
- )
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-notify_clients_after_touch after model got touched
|
|
|
-
|
|
|
-used as callback in model file
|
|
|
-
|
|
|
-class OwnModel < ApplicationModel
|
|
|
- after_create :notify_clients_after_create
|
|
|
- after_update :notify_clients_after_update
|
|
|
- after_touch :notify_clients_after_touch
|
|
|
- after_destroy :notify_clients_after_destroy
|
|
|
-
|
|
|
- [...]
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def notify_clients_after_touch
|
|
|
-
|
|
|
- # return if we run import mode
|
|
|
- return if Setting.get('import_mode')
|
|
|
- logger.debug "#{self.class.name}.find(#{id}) notify TOUCH " + updated_at.to_s
|
|
|
- class_name = self.class.name
|
|
|
- class_name.gsub!(/::/, '')
|
|
|
- PushMessages.send(
|
|
|
- message: {
|
|
|
- event: class_name + ':touch',
|
|
|
- data: { id: id, updated_at: updated_at }
|
|
|
- },
|
|
|
- type: 'authenticated',
|
|
|
- )
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-notify_clients_after_destroy after model got destroyed
|
|
|
-
|
|
|
-used as callback in model file
|
|
|
-
|
|
|
-class OwnModel < ApplicationModel
|
|
|
- after_create :notify_clients_after_create
|
|
|
- after_update :notify_clients_after_update
|
|
|
- after_touch :notify_clients_after_touch
|
|
|
- after_destroy :notify_clients_after_destroy
|
|
|
-
|
|
|
- [...]
|
|
|
-
|
|
|
-=end
|
|
|
- def notify_clients_after_destroy
|
|
|
-
|
|
|
- # return if we run import mode
|
|
|
- return if Setting.get('import_mode')
|
|
|
- logger.debug "#{self.class.name}.find(#{id}) notify DESTOY " + updated_at.to_s
|
|
|
- class_name = self.class.name
|
|
|
- class_name.gsub!(/::/, '')
|
|
|
- PushMessages.send(
|
|
|
- message: {
|
|
|
- event: class_name + ':destroy',
|
|
|
- data: { id: id, updated_at: updated_at }
|
|
|
- },
|
|
|
- type: 'authenticated',
|
|
|
- )
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-serve methode to configure and enable search index support for this model
|
|
|
-
|
|
|
-class Model < ApplicationModel
|
|
|
- search_index_support
|
|
|
- ignore_attributes: {
|
|
|
- create_article_type_id: true,
|
|
|
- create_article_sender_id: true,
|
|
|
- article_count: true,
|
|
|
- },
|
|
|
- ignore_ids: [1,2,4]
|
|
|
-
|
|
|
-end
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.search_index_support(data = {})
|
|
|
- @search_index_support_config = data
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-update search index, if configured - will be executed automatically
|
|
|
-
|
|
|
- model = Model.find(123)
|
|
|
- model.search_index_update
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def search_index_update
|
|
|
- config = self.class.search_index_support_config
|
|
|
- return if !config
|
|
|
- return if config[:ignore_ids] && config[:ignore_ids].include?(id)
|
|
|
-
|
|
|
- # start background job to transfer data to search index
|
|
|
- return if !SearchIndexBackend.enabled?
|
|
|
- Delayed::Job.enqueue(ApplicationModel::BackgroundJobSearchIndex.new(self.class.to_s, id))
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-delete search index object, will be executed automatically
|
|
|
-
|
|
|
- model = Model.find(123)
|
|
|
- model.search_index_destroy
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def search_index_destroy
|
|
|
- config = self.class.search_index_support_config
|
|
|
- return if !config
|
|
|
- return if config[:ignore_ids] && config[:ignore_ids].include?(id)
|
|
|
-
|
|
|
- SearchIndexBackend.remove(self.class.to_s, id)
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-reload search index with full data
|
|
|
-
|
|
|
- Model.search_index_reload
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.search_index_reload
|
|
|
- config = @search_index_support_config
|
|
|
- return if !config
|
|
|
- tolerance = 5
|
|
|
- tolerance_count = 0
|
|
|
- all_ids = select('id').all.order('created_at DESC')
|
|
|
- all_ids.each { |item_with_id|
|
|
|
- next if config[:ignore_ids] && config[:ignore_ids].include?(item_with_id.id)
|
|
|
- item = find(item_with_id.id)
|
|
|
- begin
|
|
|
- item.search_index_update_backend
|
|
|
- rescue => e
|
|
|
- logger.error "Unable to send #{item.class}.find(#{item.id}) backend: #{e.inspect}"
|
|
|
- tolerance_count += 1
|
|
|
- raise "Unable to send #{item.class}.find(#{item.id}) backend: #{e.inspect}" if tolerance_count == tolerance
|
|
|
- end
|
|
|
- }
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-serve methode to configure and enable activity stream support for this model
|
|
|
-
|
|
|
-class Model < ApplicationModel
|
|
|
- activity_stream_support permission: 'admin.user'
|
|
|
-end
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.activity_stream_support(data = {})
|
|
|
- @activity_stream_support_config = data
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-log object create activity stream, if configured - will be executed automatically
|
|
|
-
|
|
|
- model = Model.find(123)
|
|
|
- model.activity_stream_create
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def activity_stream_create
|
|
|
- return if !self.class.activity_stream_support_config
|
|
|
- activity_stream_log('create', self['created_by_id'])
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-log object update activity stream, if configured - will be executed automatically
|
|
|
-
|
|
|
- model = Model.find(123)
|
|
|
- model.activity_stream_update
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def activity_stream_update
|
|
|
- return if !self.class.activity_stream_support_config
|
|
|
-
|
|
|
- return if !changed?
|
|
|
-
|
|
|
- # default ignored attributes
|
|
|
- ignore_attributes = {
|
|
|
- created_at: true,
|
|
|
- updated_at: true,
|
|
|
- created_by_id: true,
|
|
|
- updated_by_id: true,
|
|
|
- }
|
|
|
- if self.class.activity_stream_support_config[:ignore_attributes]
|
|
|
- self.class.activity_stream_support_config[:ignore_attributes].each { |key, value|
|
|
|
- ignore_attributes[key] = value
|
|
|
- }
|
|
|
- end
|
|
|
-
|
|
|
- log = false
|
|
|
- changes.each { |key, _value|
|
|
|
-
|
|
|
- # do not log created_at and updated_at attributes
|
|
|
- next if ignore_attributes[key.to_sym] == true
|
|
|
-
|
|
|
- log = true
|
|
|
- }
|
|
|
-
|
|
|
- return if !log
|
|
|
-
|
|
|
- activity_stream_log('update', self['updated_by_id'])
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-delete object activity stream, will be executed automatically
|
|
|
-
|
|
|
- model = Model.find(123)
|
|
|
- model.activity_stream_destroy
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def activity_stream_destroy
|
|
|
- return if !self.class.activity_stream_support_config
|
|
|
- ActivityStream.remove(self.class.to_s, id)
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-serve methode to configure and enable history support for this model
|
|
|
-
|
|
|
-class Model < ApplicationModel
|
|
|
- history_support
|
|
|
-end
|
|
|
-
|
|
|
-class Model < ApplicationModel
|
|
|
- history_support ignore_attributes: { article_count: true }
|
|
|
-end
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.history_support(data = {})
|
|
|
- @history_support_config = data
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-log object create history, if configured - will be executed automatically
|
|
|
-
|
|
|
- model = Model.find(123)
|
|
|
- model.history_create
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def history_create
|
|
|
- return if !self.class.history_support_config
|
|
|
- #logger.debug 'create ' + self.changes.inspect
|
|
|
- history_log('created', created_by_id)
|
|
|
-
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-log object update history with all updated attributes, if configured - will be executed automatically
|
|
|
-
|
|
|
- model = Model.find(123)
|
|
|
- model.history_update
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def history_update
|
|
|
- return if !self.class.history_support_config
|
|
|
-
|
|
|
- return if !changed?
|
|
|
-
|
|
|
- # return if it's no update
|
|
|
- return if new_record?
|
|
|
-
|
|
|
- # new record also triggers update, so ignore new records
|
|
|
- changes = self.changes
|
|
|
- if history_changes_last_done
|
|
|
- history_changes_last_done.each { |key, value|
|
|
|
- if changes.key?(key) && changes[key] == value
|
|
|
- changes.delete(key)
|
|
|
- end
|
|
|
- }
|
|
|
- end
|
|
|
- self.history_changes_last_done = changes
|
|
|
- #logger.info 'updated ' + self.changes.inspect
|
|
|
-
|
|
|
- return if changes['id'] && !changes['id'][0]
|
|
|
-
|
|
|
- # default ignored attributes
|
|
|
- ignore_attributes = {
|
|
|
- created_at: true,
|
|
|
- updated_at: true,
|
|
|
- created_by_id: true,
|
|
|
- updated_by_id: true,
|
|
|
- }
|
|
|
- if self.class.history_support_config[:ignore_attributes]
|
|
|
- self.class.history_support_config[:ignore_attributes].each { |key, value|
|
|
|
- ignore_attributes[key] = value
|
|
|
- }
|
|
|
- end
|
|
|
-
|
|
|
- changes.each { |key, value|
|
|
|
-
|
|
|
- # do not log created_at and updated_at attributes
|
|
|
- next if ignore_attributes[key.to_sym] == true
|
|
|
-
|
|
|
- # get attribute name
|
|
|
- attribute_name = key.to_s
|
|
|
- if attribute_name[-3, 3] == '_id'
|
|
|
- attribute_name = attribute_name[ 0, attribute_name.length - 3 ]
|
|
|
- end
|
|
|
-
|
|
|
- value_id = []
|
|
|
- value_str = [ value[0], value[1] ]
|
|
|
- if key.to_s[-3, 3] == '_id'
|
|
|
- value_id[0] = value[0]
|
|
|
- value_id[1] = value[1]
|
|
|
-
|
|
|
- if respond_to?(attribute_name) && send(attribute_name)
|
|
|
- relation_class = send(attribute_name).class
|
|
|
- if relation_class && value_id[0]
|
|
|
- relation_model = relation_class.lookup(id: value_id[0])
|
|
|
- if relation_model
|
|
|
- if relation_model['name']
|
|
|
- value_str[0] = relation_model['name']
|
|
|
- elsif relation_model.respond_to?('fullname')
|
|
|
- value_str[0] = relation_model.send('fullname')
|
|
|
- end
|
|
|
- end
|
|
|
- end
|
|
|
- if relation_class && value_id[1]
|
|
|
- relation_model = relation_class.lookup(id: value_id[1])
|
|
|
- if relation_model
|
|
|
- if relation_model['name']
|
|
|
- value_str[1] = relation_model['name']
|
|
|
- elsif relation_model.respond_to?('fullname')
|
|
|
- value_str[1] = relation_model.send('fullname')
|
|
|
- end
|
|
|
- end
|
|
|
- end
|
|
|
- end
|
|
|
- end
|
|
|
- data = {
|
|
|
- history_attribute: attribute_name,
|
|
|
- value_from: value_str[0].to_s,
|
|
|
- value_to: value_str[1].to_s,
|
|
|
- id_from: value_id[0],
|
|
|
- id_to: value_id[1],
|
|
|
- }
|
|
|
- #logger.info "HIST NEW #{self.class.to_s}.find(#{self.id}) #{data.inspect}"
|
|
|
- history_log('updated', updated_by_id, data)
|
|
|
- }
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-delete object history, will be executed automatically
|
|
|
-
|
|
|
- model = Model.find(123)
|
|
|
- model.history_destroy
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def history_destroy
|
|
|
- return if !self.class.history_support_config
|
|
|
- History.remove(self.class.to_s, id)
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-serve methode to configure and attributes_with_associations support for this model
|
|
|
-
|
|
|
-class Model < ApplicationModel
|
|
|
- attributes_with_associations(ignore: { user_ids: => true })
|
|
|
-end
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.attributes_with_associations_support(data = {})
|
|
|
- @attributes_with_associations_support_config = data
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-get list of attachments of this object
|
|
|
-
|
|
|
- item = Model.find(123)
|
|
|
- list = item.attachments
|
|
|
-
|
|
|
-returns
|
|
|
-
|
|
|
- # array with Store model objects
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def attachments
|
|
|
- Store.list(object: self.class.to_s, o_id: id)
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-store attachments for this object
|
|
|
-
|
|
|
- item = Model.find(123)
|
|
|
- item.attachments = [ Store-Object1, Store-Object2 ]
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def attachments=(attachments)
|
|
|
- self.attachments_buffer = attachments
|
|
|
-
|
|
|
- # update if object already exists
|
|
|
- return if !(id && id.nonzero?)
|
|
|
-
|
|
|
- attachments_buffer_check
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-return object and assets
|
|
|
-
|
|
|
- data = Model.full(123)
|
|
|
- data = {
|
|
|
- id: 123,
|
|
|
- assets: assets,
|
|
|
- }
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.full(id)
|
|
|
- object = find(id)
|
|
|
- assets = object.assets({})
|
|
|
- {
|
|
|
- id: id,
|
|
|
- assets: assets,
|
|
|
- }
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-get assets of object list
|
|
|
-
|
|
|
- list = [
|
|
|
- {
|
|
|
- object => 'Ticket',
|
|
|
- o_id => 1,
|
|
|
- },
|
|
|
- {
|
|
|
- object => 'User',
|
|
|
- o_id => 121,
|
|
|
- },
|
|
|
- ]
|
|
|
-
|
|
|
- assets = Model.assets_of_object_list(list, assets)
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.assets_of_object_list(list, assets = {})
|
|
|
- list.each { |item|
|
|
|
- require item['object'].to_filename
|
|
|
- record = Kernel.const_get(item['object']).find(item['o_id'])
|
|
|
- assets = record.assets(assets)
|
|
|
- if item['created_by_id']
|
|
|
- user = User.find(item['created_by_id'])
|
|
|
- assets = user.assets(assets)
|
|
|
- end
|
|
|
- if item['updated_by_id']
|
|
|
- user = User.find(item['updated_by_id'])
|
|
|
- assets = user.assets(assets)
|
|
|
- end
|
|
|
- }
|
|
|
- assets
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-get assets and record_ids of selector
|
|
|
-
|
|
|
- model = Model.find(123)
|
|
|
-
|
|
|
- assets = model.assets_of_selector('attribute_name_of_selector', assets)
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def assets_of_selector(selector, assets = {})
|
|
|
-
|
|
|
- # get assets of condition
|
|
|
- models = Models.all
|
|
|
- send(selector).each { |item, content|
|
|
|
- attribute = item.split(/\./)
|
|
|
- next if !attribute[1]
|
|
|
- begin
|
|
|
- attribute_class = attribute[0].to_classname.constantize
|
|
|
- rescue => e
|
|
|
- logger.error "Unable to get asset for '#{attribute[0]}': #{e.inspect}"
|
|
|
- next
|
|
|
- end
|
|
|
- reflection = attribute[1].sub(/_id$/, '')
|
|
|
- #reflection = reflection.to_sym
|
|
|
- next if !models[attribute_class]
|
|
|
- next if !models[attribute_class][:reflections]
|
|
|
- next if !models[attribute_class][:reflections][reflection]
|
|
|
- next if !models[attribute_class][:reflections][reflection].klass
|
|
|
- attribute_ref_class = models[attribute_class][:reflections][reflection].klass
|
|
|
- if content['value'].class == Array
|
|
|
- content['value'].each { |item_id|
|
|
|
- attribute_object = attribute_ref_class.find_by(id: item_id)
|
|
|
- if attribute_object
|
|
|
- assets = attribute_object.assets(assets)
|
|
|
- end
|
|
|
- }
|
|
|
- else
|
|
|
- attribute_object = attribute_ref_class.find_by(id: content['value'])
|
|
|
- if attribute_object
|
|
|
- assets = attribute_object.assets(assets)
|
|
|
- end
|
|
|
- end
|
|
|
- }
|
|
|
- assets
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-touch references by params
|
|
|
-
|
|
|
- Model.touch_reference_by_params(
|
|
|
- object: 'Ticket',
|
|
|
- o_id: 123,
|
|
|
- )
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def self.touch_reference_by_params(data)
|
|
|
-
|
|
|
- object_class = Kernel.const_get(data[:object])
|
|
|
- object = object_class.lookup(id: data[:o_id])
|
|
|
- return if !object
|
|
|
- object.touch
|
|
|
- rescue => e
|
|
|
- logger.error e.message
|
|
|
- logger.error e.backtrace.inspect
|
|
|
-
|
|
|
- end
|
|
|
-
|
|
|
- private
|
|
|
-
|
|
|
- def attachments_buffer
|
|
|
- @attachments_buffer_data
|
|
|
- end
|
|
|
-
|
|
|
- def attachments_buffer=(attachments)
|
|
|
- @attachments_buffer_data = attachments
|
|
|
- end
|
|
|
-
|
|
|
- def attachments_buffer_check
|
|
|
-
|
|
|
- # do nothing if no attachment exists
|
|
|
- return 1 if attachments_buffer.nil?
|
|
|
-
|
|
|
- # store attachments
|
|
|
- article_store = []
|
|
|
- attachments_buffer.each do |attachment|
|
|
|
- article_store.push Store.add(
|
|
|
- object: self.class.to_s,
|
|
|
- o_id: id,
|
|
|
- data: attachment.content,
|
|
|
- filename: attachment.filename,
|
|
|
- preferences: attachment.preferences,
|
|
|
- created_by_id: created_by_id,
|
|
|
- )
|
|
|
- end
|
|
|
- attachments_buffer = nil
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-delete object recent viewed list, will be executed automatically
|
|
|
-
|
|
|
- model = Model.find(123)
|
|
|
- model.recent_view_destroy
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def recent_view_destroy
|
|
|
- RecentView.log_destroy(self.class.to_s, id)
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-check string/varchar size and cut them if needed
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def check_limits
|
|
|
- attributes.each { |attribute|
|
|
|
- next if !self[ attribute[0] ]
|
|
|
- next if self[ attribute[0] ].class != String
|
|
|
- next if self[ attribute[0] ].empty?
|
|
|
- column = self.class.columns_hash[ attribute[0] ]
|
|
|
- next if !column
|
|
|
- limit = column.limit
|
|
|
- if column && limit
|
|
|
- current_length = attribute[1].to_s.length
|
|
|
- if limit < current_length
|
|
|
- logger.warn "WARNING: cut string because of database length #{self.class}.#{attribute[0]}(#{limit} but is #{current_length}:#{attribute[1]})"
|
|
|
- self[ attribute[0] ] = attribute[1][ 0, limit ]
|
|
|
- end
|
|
|
- end
|
|
|
-
|
|
|
- # strip 4 bytes utf8 chars if needed
|
|
|
- if column && self[ attribute[0] ]
|
|
|
- self[attribute[0]] = self[ attribute[0] ].utf8_to_3bytesutf8
|
|
|
- end
|
|
|
- }
|
|
|
- end
|
|
|
-
|
|
|
-=begin
|
|
|
-
|
|
|
-destroy object dependencies, will be executed automatically
|
|
|
-
|
|
|
-=end
|
|
|
-
|
|
|
- def destroy_dependencies
|
|
|
- end
|
|
|
-
|
|
|
end
|