facebook.rb 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359
  1. # Copyright (C) 2012-2015 Zammad Foundation, http://zammad-foundation.org/
  2. require 'koala'
  3. class Facebook
  4. attr_accessor :client, :account
  5. =begin
  6. client = Facebook.new('user_or_page_access_token')
  7. =end
  8. def initialize(access_token)
  9. connect(access_token)
  10. end
  11. =begin
  12. reconnect with other access_token
  13. client.connect('user_or_page_access_token')
  14. =end
  15. def connect(access_token)
  16. @client = Koala::Facebook::API.new(access_token)
  17. end
  18. =begin
  19. disconnect client
  20. client.disconnect
  21. =end
  22. def disconnect
  23. return if !@client
  24. @client = nil
  25. end
  26. =begin
  27. get pages of user
  28. pages = client.pages
  29. result
  30. [
  31. {
  32. id: '12345',
  33. name: 'Some Page Name',
  34. access_token, 'some_access_token_for_page',
  35. },
  36. ]
  37. =end
  38. def pages
  39. pages = []
  40. @client.get_connections('me', 'accounts').each { |page|
  41. pages.push(
  42. id: page['id'],
  43. name: page['name'],
  44. access_token: page['access_token'],
  45. )
  46. }
  47. pages
  48. end
  49. =begin
  50. get current user
  51. pages = current_user
  52. result
  53. {
  54. 'id' => '1234567890123456',
  55. 'name' => 'Page/User Name',
  56. 'access_token' => 'some_acces_token'
  57. }
  58. =end
  59. def current_user
  60. @client.get_object('me')
  61. end
  62. =begin
  63. get user of comment/post
  64. pages = user(comment_or_post)
  65. result
  66. ???
  67. =end
  68. def user(item)
  69. return if !item['from']
  70. return if !item['from']['id']
  71. cache_key = "FB:User:Lookup:#{item['from']['id']}"
  72. cache = Cache.get(cache_key)
  73. return cache if cache
  74. begin
  75. result = @client.get_object(item['from']['id'], fields: 'first_name,last_name,email')
  76. rescue
  77. result = @client.get_object(item['from']['id'], fields: 'name')
  78. end
  79. if result
  80. Cache.write(cache_key, result, { expires_in: 15.minutes })
  81. end
  82. result
  83. end
  84. def to_user(item)
  85. Rails.logger.debug 'Create user from item...'
  86. Rails.logger.debug item.inspect
  87. # do item_user lookup
  88. item_user = user(item)
  89. return if !item_user
  90. auth = Authorization.find_by(uid: item_user['id'], provider: 'facebook')
  91. # create or update user
  92. user_data = {
  93. image_source: "https://graph.facebook.com/#{item_user['id']}/picture?type=large",
  94. }
  95. if auth
  96. user = User.find(auth.user_id)
  97. map = {
  98. #note: 'description',
  99. }
  100. # ignore if value is already set
  101. map.each {|target, source|
  102. next if user[target] && !user[target].empty?
  103. new_value = tweet_user.send(source).to_s
  104. next if !new_value || new_value.empty?
  105. user_data[target] = new_value
  106. }
  107. user.update_attributes(user_data)
  108. else
  109. user_data[:login] = item_user['id']
  110. if item_user['first_name'] && item_user['last_name']
  111. user_data[:firstname] = item_user['first_name']
  112. user_data[:lastname] = item_user['last_name']
  113. else
  114. user_data[:firstname] = item_user['name']
  115. end
  116. user_data[:active] = true
  117. user_data[:roles] = Role.where(name: 'Customer')
  118. user = User.create(user_data)
  119. end
  120. if user_data[:image_source]
  121. avatar = Avatar.add(
  122. object: 'User',
  123. o_id: user.id,
  124. url: user_data[:image_source],
  125. source: 'facebook',
  126. deletable: true,
  127. updated_by_id: user.id,
  128. created_by_id: user.id,
  129. )
  130. # update user link
  131. if avatar && user.image != avatar.store_hash
  132. user.image = avatar.store_hash
  133. user.save
  134. end
  135. end
  136. # create authorization
  137. if !auth
  138. auth_data = {
  139. uid: item_user['id'],
  140. username: item_user['id'],
  141. user_id: user.id,
  142. provider: 'facebook'
  143. }
  144. Authorization.create(auth_data)
  145. end
  146. UserInfo.current_user_id = user.id
  147. user
  148. end
  149. def to_ticket(post, group_id, channel, page)
  150. Rails.logger.debug 'Create ticket from post...'
  151. Rails.logger.debug post.inspect
  152. Rails.logger.debug group_id.inspect
  153. user = to_user(post)
  154. return if !user
  155. # prepare title
  156. title = post['message']
  157. if title.length > 80
  158. title = "#{title[0, 80]}..."
  159. end
  160. Ticket.create(
  161. customer_id: user.id,
  162. title: title,
  163. group_id: group_id,
  164. state: Ticket::State.find_by(name: 'new'),
  165. priority: Ticket::Priority.find_by(name: '2 normal'),
  166. preferences: {
  167. channel_id: channel.id,
  168. channel_fb_object_id: page['id'],
  169. },
  170. )
  171. end
  172. def to_article(post, ticket, _page)
  173. Rails.logger.debug 'Create article from post...'
  174. Rails.logger.debug post.inspect
  175. Rails.logger.debug ticket.inspect
  176. # set ticket state to open if not new
  177. if ticket.state.name != 'new'
  178. ticket.state = Ticket::State.find_by(name: 'open')
  179. ticket.save
  180. end
  181. user = to_user(post)
  182. return if !user
  183. to = nil
  184. if post['to'] && post['to']['data']
  185. post['to']['data'].each {|to_entry|
  186. if !to
  187. to = ''
  188. else
  189. to += ', '
  190. end
  191. to += to_entry['name']
  192. }
  193. end
  194. feed_post = {
  195. from: post['from']['name'],
  196. to: to,
  197. body: post['message'],
  198. message_id: post['id'],
  199. type_id: Ticket::Article::Type.find_by(name: 'facebook feed post').id,
  200. }
  201. articles = []
  202. articles.push(feed_post)
  203. if post['comments'] && post['comments']['data']
  204. articles += nested_comments(post['comments']['data'], post['id'])
  205. end
  206. articles.each { |article|
  207. next if Ticket::Article.find_by(message_id: article[:message_id])
  208. article = {
  209. #to: @account['name'],
  210. ticket_id: ticket.id,
  211. internal: false,
  212. sender_id: Ticket::Article::Sender.lookup(name: 'Customer').id,
  213. created_by_id: 1,
  214. updated_by_id: 1,
  215. }.merge(article)
  216. Ticket::Article.create(article)
  217. }
  218. end
  219. def to_group(post, group_id, channel, page)
  220. Rails.logger.debug 'import post'
  221. return if !post['message']
  222. ticket = nil
  223. # use transaction
  224. ActiveRecord::Base.transaction do
  225. UserInfo.current_user_id = 1
  226. existing_article = Ticket::Article.find_by(message_id: post['id'])
  227. ticket = if existing_article
  228. existing_article.ticket
  229. else
  230. to_ticket(post, group_id, channel, page)
  231. end
  232. to_article(post, ticket, page)
  233. # execute ticket events
  234. Observer::Ticket::Notification.transaction
  235. end
  236. ticket
  237. end
  238. def from_article(article)
  239. post = nil
  240. if article[:type] == 'facebook feed comment'
  241. Rails.logger.debug 'Create feed comment from article...'
  242. post = @client.put_comment(article[:in_reply_to], article[:body])
  243. else
  244. raise "Can't handle unknown facebook article type '#{article[:type]}'."
  245. end
  246. Rails.logger.debug post.inspect
  247. @client.get_object(post['id'])
  248. end
  249. private
  250. def access_token_for_page(lookup)
  251. access_token = nil
  252. pages.each { |page|
  253. next if !lookup[:page_id] && !lookup[:page]
  254. next if lookup[:page_id] && lookup[:page_id].to_s != page[:id]
  255. next if lookup[:page] && lookup[:page] != page[:name]
  256. access_token = page[:access_token]
  257. break
  258. }
  259. access_token
  260. end
  261. def nested_comments(comments, in_reply_to)
  262. Rails.logger.debug 'Fetching nested comments...'
  263. Rails.logger.debug comments.inspect
  264. result = []
  265. return result if comments.empty?
  266. comments.each { |comment|
  267. user = to_user(comment)
  268. next if !user
  269. article_data = {
  270. from: "#{user.firstname} #{user.lastname}",
  271. body: comment['message'],
  272. message_id: comment['id'],
  273. type_id: Ticket::Article::Type.find_by( name: 'facebook feed comment' ).id,
  274. in_reply_to: in_reply_to
  275. }
  276. result.push(article_data)
  277. sub_comments = @client.get_object( "#{comment['id']}/comments" )
  278. sub_articles = nested_comments(sub_comments, comment['id'])
  279. result += sub_articles
  280. }
  281. result
  282. end
  283. end