avatar.rb 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417
  1. # Copyright (C) 2012-2022 Zammad Foundation, https://zammad-foundation.org/
  2. class Avatar < ApplicationModel
  3. belongs_to :object_lookup, optional: true
  4. =begin
  5. add an avatar based on auto detection (email address)
  6. Avatar.auto_detection(
  7. object: 'User',
  8. o_id: user.id,
  9. url: 'somebody@example.com',
  10. updated_by_id: 1,
  11. created_by_id: 1,
  12. )
  13. =end
  14. def self.auto_detection(data)
  15. # return if we run import mode
  16. return if Setting.get('import_mode')
  17. return if data[:url].blank?
  18. Avatar.add(
  19. object: data[:object],
  20. o_id: data[:o_id],
  21. url: data[:url],
  22. source: 'zammad.com',
  23. deletable: false,
  24. updated_by_id: 1,
  25. created_by_id: 1,
  26. )
  27. end
  28. =begin
  29. add avatar by upload
  30. Avatar.add(
  31. object: 'User',
  32. o_id: user.id,
  33. default: true,
  34. full: {
  35. content: '...',
  36. mime_type: 'image/png',
  37. },
  38. resize: {
  39. content: '...',
  40. mime_type: 'image/png',
  41. },
  42. source: 'web',
  43. deletable: true,
  44. updated_by_id: 1,
  45. created_by_id: 1,
  46. )
  47. add avatar by url
  48. Avatar.add(
  49. object: 'User',
  50. o_id: user.id,
  51. default: true,
  52. url: ...,
  53. source: 'web',
  54. deletable: true,
  55. updated_by_id: 1,
  56. created_by_id: 1,
  57. )
  58. =end
  59. def self.add(data)
  60. # lookups
  61. if data[:object]
  62. object_id = ObjectLookup.by_name(data[:object])
  63. end
  64. # add initial avatar
  65. _add_init_avatar(object_id, data[:o_id])
  66. record = {
  67. o_id: data[:o_id],
  68. object_lookup_id: object_id,
  69. default: true,
  70. deletable: data[:deletable],
  71. initial: false,
  72. source: data[:source],
  73. source_url: data[:url],
  74. updated_by_id: data[:updated_by_id],
  75. created_by_id: data[:created_by_id],
  76. }
  77. # check if avatar with url already exists
  78. avatar_already_exists = nil
  79. if data[:source].present?
  80. avatar_already_exists = Avatar.find_by(
  81. object_lookup_id: object_id,
  82. o_id: data[:o_id],
  83. source: data[:source],
  84. )
  85. end
  86. # fetch image based on http url
  87. if data[:url].present?
  88. if data[:url].instance_of?(Tempfile)
  89. logger.info "Reading image from tempfile '#{data[:url].inspect}'"
  90. content = data[:url].read
  91. filename = data[:url].path
  92. mime_type = 'image'
  93. if filename.match?(%r{\.png}i)
  94. mime_type = 'image/png'
  95. end
  96. if filename.match?(%r{\.(jpg|jpeg)}i)
  97. mime_type = 'image/jpeg'
  98. end
  99. data[:resize] ||= {}
  100. data[:resize][:content] = content
  101. data[:resize][:mime_type] = mime_type
  102. data[:full] ||= {}
  103. data[:full][:content] = content
  104. data[:full][:mime_type] = mime_type
  105. elsif data[:url].to_s.match?(%r{^https?://})
  106. url = data[:url].to_s
  107. # check if source was updated within last 2 minutes
  108. return if avatar_already_exists&.source_url == url && avatar_already_exists.updated_at > 2.minutes.ago
  109. # twitter workaround to get bigger avatar images
  110. # see also https://dev.twitter.com/overview/general/user-profile-images-and-banners
  111. if url.match?(%r{//pbs.twimg.com/}i)
  112. url.sub!(%r{normal\.(png|jpg|gif)$}, 'bigger.\1')
  113. end
  114. # fetch image
  115. response = UserAgent.get(
  116. url,
  117. {},
  118. {
  119. open_timeout: 4,
  120. read_timeout: 6,
  121. total_timeout: 6,
  122. },
  123. )
  124. if !response.success?
  125. logger.info "Can't fetch '#{url}' (maybe no avatar available), http code: #{response.code}"
  126. return
  127. end
  128. logger.info "Fetched image '#{url}', http code: #{response.code}"
  129. mime_type = 'image'
  130. if url.match?(%r{\.png}i)
  131. mime_type = 'image/png'
  132. end
  133. if url.match?(%r{\.(jpg|jpeg)}i)
  134. mime_type = 'image/jpeg'
  135. end
  136. data[:resize] ||= {}
  137. data[:resize][:content] = response.body
  138. data[:resize][:mime_type] = mime_type
  139. data[:full] ||= {}
  140. data[:full][:content] = response.body
  141. data[:full][:mime_type] = mime_type
  142. # try zammad backend to find image based on email
  143. elsif data[:url].to_s.match?(URI::MailTo::EMAIL_REGEXP)
  144. url = data[:url].to_s
  145. # check if source ist already updated within last 3 minutes
  146. return if avatar_already_exists&.source_url == url && avatar_already_exists.updated_at > 2.minutes.ago
  147. # fetch image
  148. image = Service::Image.user(url)
  149. return if !image
  150. data[:resize] = image
  151. data[:full] = image
  152. end
  153. end
  154. # check if avatar needs to be updated
  155. if data[:resize].present? && data[:resize][:content].present?
  156. record[:store_hash] = Digest::MD5.hexdigest(data[:resize][:content])
  157. if avatar_already_exists&.store_hash == record[:store_hash]
  158. avatar_already_exists.touch # rubocop:disable Rails/SkipsModelValidations
  159. return avatar_already_exists
  160. end
  161. end
  162. # store images
  163. object_name = "Avatar::#{data[:object]}"
  164. if data[:full].present?
  165. store_full = Store.add(
  166. object: "#{object_name}::Full",
  167. o_id: data[:o_id],
  168. data: data[:full][:content],
  169. filename: 'avatar_full',
  170. preferences: {
  171. 'Mime-Type' => data[:full][:mime_type]
  172. },
  173. created_by_id: data[:created_by_id],
  174. )
  175. record[:store_full_id] = store_full.id
  176. record[:store_hash] = Digest::MD5.hexdigest(data[:full][:content])
  177. end
  178. if data[:resize].present?
  179. store_resize = Store.add(
  180. object: "#{object_name}::Resize",
  181. o_id: data[:o_id],
  182. data: data[:resize][:content],
  183. filename: 'avatar',
  184. preferences: {
  185. 'Mime-Type' => data[:resize][:mime_type]
  186. },
  187. created_by_id: data[:created_by_id],
  188. )
  189. record[:store_resize_id] = store_resize.id
  190. record[:store_hash] = Digest::MD5.hexdigest(data[:resize][:content])
  191. end
  192. return if record[:store_resize_id].blank? || record[:store_hash].blank?
  193. # update existing
  194. if avatar_already_exists
  195. avatar_already_exists.update!(record)
  196. avatar = avatar_already_exists
  197. # add new one and set it as default
  198. else
  199. avatar = Avatar.create(record)
  200. set_default_items(object_id, data[:o_id], avatar.id)
  201. end
  202. avatar
  203. end
  204. =begin
  205. set avatars as default
  206. Avatar.set_default('User', 123, avatar_id)
  207. =end
  208. def self.set_default(object_name, o_id, avatar_id)
  209. object_id = ObjectLookup.by_name(object_name)
  210. avatar = Avatar.find_by(
  211. object_lookup_id: object_id,
  212. o_id: o_id,
  213. id: avatar_id,
  214. )
  215. avatar.default = true
  216. avatar.save!
  217. # set all other to default false
  218. set_default_items(object_id, o_id, avatar_id)
  219. avatar
  220. end
  221. =begin
  222. remove all avatars of an object
  223. Avatar.remove('User', 123)
  224. =end
  225. def self.remove(object_name, o_id)
  226. object_id = ObjectLookup.by_name(object_name)
  227. Avatar.where(
  228. object_lookup_id: object_id,
  229. o_id: o_id,
  230. ).destroy_all
  231. object_name_store = "Avatar::#{object_name}"
  232. Store.remove(
  233. object: "#{object_name_store}::Full",
  234. o_id: o_id,
  235. )
  236. Store.remove(
  237. object: "#{object_name_store}::Resize",
  238. o_id: o_id,
  239. )
  240. end
  241. =begin
  242. remove one avatars of an object
  243. Avatar.remove_one('User', 123, avatar_id)
  244. =end
  245. def self.remove_one(object_name, o_id, avatar_id)
  246. object_id = ObjectLookup.by_name(object_name)
  247. Avatar.where(
  248. object_lookup_id: object_id,
  249. o_id: o_id,
  250. id: avatar_id,
  251. ).destroy_all
  252. end
  253. =begin
  254. return all avatars of an user
  255. avatars = Avatar.list('User', 123)
  256. avatars = Avatar.list('User', 123, no_init_add_as_boolean) # per default true
  257. =end
  258. def self.list(object_name, o_id, no_init_add_as_boolean = true)
  259. object_id = ObjectLookup.by_name(object_name)
  260. avatars = Avatar.where(
  261. object_lookup_id: object_id,
  262. o_id: o_id,
  263. ).order(initial: :desc, deletable: :asc, created_at: :asc)
  264. # add initial avatar
  265. if no_init_add_as_boolean
  266. _add_init_avatar(object_id, o_id)
  267. end
  268. avatar_list = []
  269. avatars.each do |avatar|
  270. data = avatar.attributes
  271. if avatar.store_resize_id
  272. file = Store.find(avatar.store_resize_id)
  273. data['content'] = "data:#{file.preferences['Mime-Type']};base64,#{Base64.strict_encode64(file.content)}"
  274. end
  275. avatar_list.push data
  276. end
  277. avatar_list
  278. end
  279. =begin
  280. get default avatar image of user by hash
  281. store = Avatar.get_by_hash(hash)
  282. returns:
  283. store object
  284. =end
  285. def self.get_by_hash(hash)
  286. avatar = Avatar.find_by(
  287. store_hash: hash,
  288. )
  289. return if !avatar
  290. Store.find(avatar.store_resize_id)
  291. end
  292. =begin
  293. get default avatar of user by user id
  294. avatar = Avatar.get_default('User', user_id)
  295. returns:
  296. avatar object
  297. =end
  298. def self.get_default(object_name, o_id)
  299. object_id = ObjectLookup.by_name(object_name)
  300. Avatar.find_by(
  301. object_lookup_id: object_id,
  302. o_id: o_id,
  303. default: true,
  304. )
  305. end
  306. def self.set_default_items(object_id, o_id, avatar_id)
  307. avatars = Avatar.where(
  308. object_lookup_id: object_id,
  309. o_id: o_id,
  310. ).order(created_at: :asc)
  311. avatars.each do |avatar|
  312. next if avatar.id == avatar_id
  313. avatar.default = false
  314. avatar.save!
  315. end
  316. end
  317. def self._add_init_avatar(object_id, o_id)
  318. count = Avatar.where(
  319. object_lookup_id: object_id,
  320. o_id: o_id,
  321. ).count
  322. return if count.positive?
  323. object_name = ObjectLookup.by_id(object_id)
  324. return if !object_name.constantize.exists?(id: o_id)
  325. Avatar.create!(
  326. o_id: o_id,
  327. object_lookup_id: object_id,
  328. default: true,
  329. source: 'init',
  330. initial: true,
  331. deletable: false,
  332. updated_by_id: 1,
  333. created_by_id: 1,
  334. )
  335. end
  336. end