application_model.rb 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  1. # Copyright (C) 2013-2013 Zammad Foundation, http://zammad-foundation.org/
  2. class ApplicationModel < ActiveRecord::Base
  3. include ApplicationModel::HistoryLogBase
  4. self.abstract_class = true
  5. before_create :check_attributes_protected, :cache_delete, :fill_up_user_create
  6. before_create :cache_delete, :fill_up_user_create
  7. before_update :cache_delete_before, :fill_up_user_update
  8. before_destroy :cache_delete_before, :destroy_dependencies
  9. after_create :cache_delete
  10. after_update :cache_delete
  11. after_destroy :cache_delete
  12. @@import_class_list = ['Ticket', 'Ticket::Article', 'History', 'Ticket::State', 'Ticket::Priority', 'Group', 'User' ]
  13. def check_attributes_protected
  14. if Setting.get('import_mode') && @@import_class_list.include?( self.name.to_s )
  15. # do noting, use id as it is
  16. else
  17. self[:id] = nil
  18. end
  19. end
  20. =begin
  21. remove all not used model attributes of params
  22. result = Model.param_cleanup(params)
  23. returns
  24. result = params # params with valid attributes of model
  25. =end
  26. def self.param_cleanup(params)
  27. if params == nil
  28. raise "No params for #{self.to_s}!"
  29. end
  30. # only use object attributes
  31. data = {}
  32. self.new.attributes.each {|item|
  33. if params.has_key?(item[0])
  34. # puts 'use ' + item[0].to_s + '-' + params[item[0]].to_s
  35. data[item[0].to_sym] = params[item[0]]
  36. end
  37. }
  38. # we do want to set this via database
  39. self.param_validation(data)
  40. end
  41. =begin
  42. remove all not used params of object (per default :updated_at, :created_at, :updated_by_id and :created_by_id)
  43. result = Model.param_validation(params)
  44. returns
  45. result = params # params without listed attributes
  46. =end
  47. def self.param_validation(data)
  48. # we do want to set this via database
  49. data.delete( :updated_at )
  50. data.delete( :created_at )
  51. data.delete( :updated_by_id )
  52. data.delete( :created_by_id )
  53. if data.respond_to?('permit!')
  54. data.permit!
  55. end
  56. data
  57. end
  58. =begin
  59. set created_by_id & updated_by_id if not given based on UserInfo (current session)
  60. Used as before_create callback, no own use needed
  61. result = Model.fill_up_user_create(params)
  62. returns
  63. result = params # params with updated_by_id & created_by_id if not given based on UserInfo (current session)
  64. =end
  65. def fill_up_user_create
  66. if self.class.column_names.include? 'updated_by_id'
  67. if UserInfo.current_user_id
  68. if self.updated_by_id && self.updated_by_id != UserInfo.current_user_id
  69. puts "NOTICE create - self.updated_by_id is different: #{self.updated_by_id.to_s}/#{UserInfo.current_user_id.to_s}"
  70. end
  71. self.updated_by_id = UserInfo.current_user_id
  72. end
  73. end
  74. if self.class.column_names.include? 'created_by_id'
  75. if UserInfo.current_user_id
  76. if self.created_by_id && self.created_by_id != UserInfo.current_user_id
  77. puts "NOTICE create - self.created_by_id is different: #{self.created_by_id.to_s}/#{UserInfo.current_user_id.to_s}"
  78. end
  79. self.created_by_id = UserInfo.current_user_id
  80. end
  81. end
  82. end
  83. =begin
  84. set updated_by_id if not given based on UserInfo (current session)
  85. Used as before_update callback, no own use needed
  86. result = Model.fill_up_user_update(params)
  87. returns
  88. result = params # params with updated_by_id & created_by_id if not given based on UserInfo (current session)
  89. =end
  90. def fill_up_user_update
  91. return if !self.class.column_names.include? 'updated_by_id'
  92. if UserInfo.current_user_id
  93. self.updated_by_id = UserInfo.current_user_id
  94. end
  95. end
  96. def cache_update(o)
  97. # puts 'u ' + self.class.to_s
  98. if self.respond_to?('cache_delete') then self.cache_delete end
  99. # puts 'g ' + group.class.to_s
  100. if o.respond_to?('cache_delete') then o.cache_delete end
  101. end
  102. def cache_delete_before
  103. old_object = self.class.where( :id => self.id ).first
  104. if old_object
  105. old_object.cache_delete
  106. end
  107. self.cache_delete
  108. end
  109. def cache_delete
  110. key = self.class.to_s + '::' + self.id.to_s
  111. Cache.delete( key.to_s )
  112. key = self.class.to_s + ':f:' + self.id.to_s
  113. Cache.delete( key.to_s )
  114. if self[:name]
  115. key = self.class.to_s + '::' + self.name.to_s
  116. Cache.delete( key.to_s )
  117. key = self.class.to_s + ':f:' + self.name.to_s
  118. Cache.delete( key.to_s )
  119. end
  120. if self[:login]
  121. key = self.class.to_s + '::' + self.login.to_s
  122. Cache.delete( key.to_s )
  123. key = self.class.to_s + ':f:' + self.login.to_s
  124. Cache.delete( key.to_s )
  125. end
  126. end
  127. def self.cache_set(data_id, data, full = false)
  128. if !full
  129. key = self.to_s + '::' + data_id.to_s
  130. else
  131. key = self.to_s + ':f:' + data_id.to_s
  132. end
  133. Cache.write( key.to_s, data )
  134. end
  135. def self.cache_get(data_id, full = false)
  136. if !full
  137. key = self.to_s + '::' + data_id.to_s
  138. else
  139. key = self.to_s + ':f:' + data_id.to_s
  140. end
  141. Cache.get( key.to_s )
  142. end
  143. =begin
  144. lookup model from cache (if exists) or retrieve it from db, id, name or login possible
  145. result = Model.lookup( :id => 123 )
  146. result = Model.lookup( :name => 'some name' )
  147. result = Model.lookup( :login => 'some login' )
  148. returns
  149. result = model # with all attributes
  150. =end
  151. def self.lookup(data)
  152. if data[:id]
  153. # puts "GET- + #{self.to_s}.#{data[:id].to_s}"
  154. cache = self.cache_get( data[:id] )
  155. return cache if cache
  156. # puts "Fillup- + #{self.to_s}.#{data[:id].to_s}"
  157. record = self.where( :id => data[:id] ).first
  158. self.cache_set( data[:id], record )
  159. return record
  160. elsif data[:name]
  161. cache = self.cache_get( data[:name] )
  162. return cache if cache
  163. records = self.where( :name => data[:name] )
  164. records.each {|record|
  165. if record.name == data[:name]
  166. self.cache_set( data[:name], record )
  167. return record
  168. end
  169. }
  170. return
  171. elsif data[:login]
  172. cache = self.cache_get( data[:login] )
  173. return cache if cache
  174. records = self.where( :login => data[:login] )
  175. records.each {|record|
  176. if record.login == data[:login]
  177. self.cache_set( data[:login], record )
  178. return record
  179. end
  180. }
  181. return
  182. else
  183. raise "Need name, id or login for lookup()"
  184. end
  185. end
  186. =begin
  187. create model if not exists (check exists based on id, name, login or locale)
  188. result = Model.create_if_not_exists( attributes )
  189. returns
  190. result = model # with all attributes
  191. =end
  192. def self.create_if_not_exists(data)
  193. if data[:id]
  194. record = self.where( :id => data[:id] ).first
  195. return record if record
  196. elsif data[:name]
  197. records = self.where( :name => data[:name] )
  198. records.each {|record|
  199. return record if record.name == data[:name]
  200. }
  201. elsif data[:login]
  202. records = self.where( :login => data[:login] )
  203. records.each {|record|
  204. return record if record.login == data[:login]
  205. }
  206. elsif data[:locale] && data[:source]
  207. records = self.where( :locale => data[:locale], :source => data[:source] )
  208. records.each {|record|
  209. return record if record.source == data[:source]
  210. }
  211. end
  212. self.create(data)
  213. end
  214. =begin
  215. create or update model (check exists based on name, login or locale)
  216. result = Model.create_or_update( attributes )
  217. returns
  218. result = model # with all attributes
  219. =end
  220. def self.create_or_update(data)
  221. if data[:name]
  222. records = self.where( :name => data[:name] )
  223. records.each {|record|
  224. if record.name == data[:name]
  225. record.update_attributes( data )
  226. return record
  227. end
  228. }
  229. record = self.new( data )
  230. record.save
  231. return record
  232. elsif data[:login]
  233. records = self.where( :login => data[:login] )
  234. records.each {|record|
  235. if record.login.downcase == data[:login].downcase
  236. record.update_attributes( data )
  237. return record
  238. end
  239. }
  240. record = self.new( data )
  241. record.save
  242. return record
  243. elsif data[:locale]
  244. records = self.where( :locale => data[:locale] )
  245. records.each {|record|
  246. if record.locale.downcase == data[:locale].downcase
  247. record.update_attributes( data )
  248. return record
  249. end
  250. }
  251. record = self.new( data )
  252. record.save
  253. return record
  254. else
  255. raise "Need name, login or locale for create_or_update()"
  256. end
  257. end
  258. =begin
  259. notify_clients_after_create after model got created
  260. used as callback in model file
  261. class OwnModel < ApplicationModel
  262. after_create :notify_clients_after_create
  263. after_update :notify_clients_after_update
  264. after_destroy :notify_clients_after_destroy
  265. [...]
  266. =end
  267. def notify_clients_after_create
  268. # return if we run import mode
  269. return if Setting.get('import_mode')
  270. class_name = self.class.name
  271. class_name.gsub!(/::/, '')
  272. Sessions.broadcast(
  273. :event => class_name + ':created',
  274. :data => { :id => self.id, :updated_at => self.updated_at }
  275. )
  276. end
  277. =begin
  278. notify_clients_after_update after model got updated
  279. used as callback in model file
  280. class OwnModel < ApplicationModel
  281. after_create :notify_clients_after_create
  282. after_update :notify_clients_after_update
  283. after_destroy :notify_clients_after_destroy
  284. [...]
  285. =end
  286. def notify_clients_after_update
  287. # return if we run import mode
  288. return if Setting.get('import_mode')
  289. puts "#{self.class.name.downcase} UPDATED " + self.updated_at.to_s
  290. class_name = self.class.name
  291. class_name.gsub!(/::/, '')
  292. Sessions.broadcast(
  293. :event => class_name + ':updated',
  294. :data => { :id => self.id, :updated_at => self.updated_at }
  295. )
  296. end
  297. =begin
  298. notify_clients_after_destroy after model got destroyed
  299. used as callback in model file
  300. class OwnModel < ApplicationModel
  301. after_create :notify_clients_after_create
  302. after_update :notify_clients_after_update
  303. after_destroy :notify_clients_after_destroy
  304. [...]
  305. =end
  306. def notify_clients_after_destroy
  307. # return if we run import mode
  308. return if Setting.get('import_mode')
  309. puts "#{self.class.name.downcase} DESTOY " + self.updated_at.to_s
  310. class_name = self.class.name
  311. class_name.gsub!(/::/, '')
  312. Sessions.broadcast(
  313. :event => class_name + ':destroy',
  314. :data => { :id => self.id, :updated_at => self.updated_at }
  315. )
  316. end
  317. private
  318. =begin
  319. destory object dependencies, will be executed automatically
  320. =end
  321. def destroy_dependencies
  322. # delete history
  323. History.remove( self.class.to_s, self.id )
  324. end
  325. end