facebooker-micah 1.0.74

Sign up to get free protection for your applications and to get access to all the features.
Files changed (156) hide show
  1. data/.autotest +15 -0
  2. data/CHANGELOG.rdoc +24 -0
  3. data/COPYING.rdoc +19 -0
  4. data/Manifest.txt +152 -0
  5. data/README.rdoc +119 -0
  6. data/Rakefile +94 -0
  7. data/TODO.rdoc +4 -0
  8. data/examples/desktop_login.rb +14 -0
  9. data/facebooker.gemspec +42 -0
  10. data/generators/facebook/facebook_generator.rb +14 -0
  11. data/generators/facebook/templates/config/facebooker.yml +49 -0
  12. data/generators/facebook/templates/public/javascripts/facebooker.js +93 -0
  13. data/generators/facebook_controller/USAGE +33 -0
  14. data/generators/facebook_controller/facebook_controller_generator.rb +40 -0
  15. data/generators/facebook_controller/templates/controller.rb +7 -0
  16. data/generators/facebook_controller/templates/functional_test.rb +11 -0
  17. data/generators/facebook_controller/templates/helper.rb +2 -0
  18. data/generators/facebook_controller/templates/view.fbml.erb +2 -0
  19. data/generators/facebook_controller/templates/view.html.erb +2 -0
  20. data/generators/facebook_publisher/facebook_publisher_generator.rb +14 -0
  21. data/generators/facebook_publisher/templates/create_facebook_templates.rb +15 -0
  22. data/generators/facebook_publisher/templates/publisher.rb +3 -0
  23. data/generators/facebook_scaffold/USAGE +27 -0
  24. data/generators/facebook_scaffold/facebook_scaffold_generator.rb +118 -0
  25. data/generators/facebook_scaffold/templates/controller.rb +93 -0
  26. data/generators/facebook_scaffold/templates/facebook_style.css +2579 -0
  27. data/generators/facebook_scaffold/templates/functional_test.rb +89 -0
  28. data/generators/facebook_scaffold/templates/helper.rb +2 -0
  29. data/generators/facebook_scaffold/templates/layout.fbml.erb +6 -0
  30. data/generators/facebook_scaffold/templates/layout.html.erb +17 -0
  31. data/generators/facebook_scaffold/templates/style.css +74 -0
  32. data/generators/facebook_scaffold/templates/view_edit.fbml.erb +13 -0
  33. data/generators/facebook_scaffold/templates/view_edit.html.erb +18 -0
  34. data/generators/facebook_scaffold/templates/view_index.fbml.erb +24 -0
  35. data/generators/facebook_scaffold/templates/view_index.html.erb +24 -0
  36. data/generators/facebook_scaffold/templates/view_new.fbml.erb +12 -0
  37. data/generators/facebook_scaffold/templates/view_new.html.erb +17 -0
  38. data/generators/facebook_scaffold/templates/view_show.fbml.erb +10 -0
  39. data/generators/facebook_scaffold/templates/view_show.html.erb +10 -0
  40. data/generators/publisher/publisher_generator.rb +14 -0
  41. data/generators/xd_receiver/templates/xd_receiver.html +10 -0
  42. data/generators/xd_receiver/templates/xd_receiver_ssl.html +10 -0
  43. data/generators/xd_receiver/xd_receiver_generator.rb +10 -0
  44. data/init.rb +28 -0
  45. data/install.rb +12 -0
  46. data/lib/facebooker.rb +261 -0
  47. data/lib/facebooker/adapters/adapter_base.rb +91 -0
  48. data/lib/facebooker/adapters/bebo_adapter.rb +77 -0
  49. data/lib/facebooker/adapters/facebook_adapter.rb +60 -0
  50. data/lib/facebooker/admin.rb +42 -0
  51. data/lib/facebooker/application.rb +37 -0
  52. data/lib/facebooker/attachment.rb +59 -0
  53. data/lib/facebooker/batch_request.rb +45 -0
  54. data/lib/facebooker/data.rb +57 -0
  55. data/lib/facebooker/feed.rb +78 -0
  56. data/lib/facebooker/logging.rb +44 -0
  57. data/lib/facebooker/mobile.rb +20 -0
  58. data/lib/facebooker/mock/service.rb +50 -0
  59. data/lib/facebooker/mock/session.rb +18 -0
  60. data/lib/facebooker/model.rb +139 -0
  61. data/lib/facebooker/models/affiliation.rb +10 -0
  62. data/lib/facebooker/models/album.rb +11 -0
  63. data/lib/facebooker/models/applicationproperties.rb +39 -0
  64. data/lib/facebooker/models/applicationrestrictions.rb +10 -0
  65. data/lib/facebooker/models/comment.rb +9 -0
  66. data/lib/facebooker/models/cookie.rb +10 -0
  67. data/lib/facebooker/models/education_info.rb +11 -0
  68. data/lib/facebooker/models/event.rb +28 -0
  69. data/lib/facebooker/models/family_relative_info.rb +7 -0
  70. data/lib/facebooker/models/friend_list.rb +16 -0
  71. data/lib/facebooker/models/group.rb +36 -0
  72. data/lib/facebooker/models/info_item.rb +10 -0
  73. data/lib/facebooker/models/info_section.rb +10 -0
  74. data/lib/facebooker/models/location.rb +8 -0
  75. data/lib/facebooker/models/message_thread.rb +89 -0
  76. data/lib/facebooker/models/notifications.rb +17 -0
  77. data/lib/facebooker/models/page.rb +46 -0
  78. data/lib/facebooker/models/photo.rb +19 -0
  79. data/lib/facebooker/models/tag.rb +12 -0
  80. data/lib/facebooker/models/user.rb +761 -0
  81. data/lib/facebooker/models/video.rb +9 -0
  82. data/lib/facebooker/models/work_info.rb +10 -0
  83. data/lib/facebooker/parser.rb +980 -0
  84. data/lib/facebooker/rails/backwards_compatible_param_checks.rb +31 -0
  85. data/lib/facebooker/rails/controller.rb +353 -0
  86. data/lib/facebooker/rails/cucumber.rb +28 -0
  87. data/lib/facebooker/rails/cucumber/world.rb +40 -0
  88. data/lib/facebooker/rails/extensions/action_controller.rb +48 -0
  89. data/lib/facebooker/rails/extensions/rack_setup.rb +16 -0
  90. data/lib/facebooker/rails/extensions/routing.rb +15 -0
  91. data/lib/facebooker/rails/facebook_form_builder.rb +141 -0
  92. data/lib/facebooker/rails/facebook_pretty_errors.rb +22 -0
  93. data/lib/facebooker/rails/facebook_request_fix.rb +28 -0
  94. data/lib/facebooker/rails/facebook_request_fix_2-3.rb +31 -0
  95. data/lib/facebooker/rails/facebook_session_handling.rb +68 -0
  96. data/lib/facebooker/rails/facebook_url_helper.rb +192 -0
  97. data/lib/facebooker/rails/facebook_url_rewriting.rb +60 -0
  98. data/lib/facebooker/rails/helpers.rb +835 -0
  99. data/lib/facebooker/rails/helpers/fb_connect.rb +165 -0
  100. data/lib/facebooker/rails/helpers/stream_publish.rb +22 -0
  101. data/lib/facebooker/rails/integration_session.rb +38 -0
  102. data/lib/facebooker/rails/profile_publisher_extensions.rb +42 -0
  103. data/lib/facebooker/rails/publisher.rb +608 -0
  104. data/lib/facebooker/rails/routing.rb +49 -0
  105. data/lib/facebooker/rails/test_helpers.rb +68 -0
  106. data/lib/facebooker/rails/utilities.rb +22 -0
  107. data/lib/facebooker/server_cache.rb +24 -0
  108. data/lib/facebooker/service.rb +103 -0
  109. data/lib/facebooker/service/base_service.rb +19 -0
  110. data/lib/facebooker/service/curl_service.rb +44 -0
  111. data/lib/facebooker/service/net_http_service.rb +12 -0
  112. data/lib/facebooker/service/typhoeus_multi_service.rb +27 -0
  113. data/lib/facebooker/service/typhoeus_service.rb +17 -0
  114. data/lib/facebooker/session.rb +788 -0
  115. data/lib/facebooker/stream_post.rb +19 -0
  116. data/lib/facebooker/version.rb +9 -0
  117. data/lib/net/http_multipart_post.rb +123 -0
  118. data/lib/rack/facebook.rb +89 -0
  119. data/lib/rack/facebook_session.rb +21 -0
  120. data/lib/tasks/facebooker.rake +19 -0
  121. data/lib/tasks/facebooker.rb +2 -0
  122. data/lib/tasks/tunnel.rake +46 -0
  123. data/rails/init.rb +1 -0
  124. data/setup.rb +1585 -0
  125. data/templates/layout.erb +24 -0
  126. data/test/facebooker/adapters_test.rb +191 -0
  127. data/test/facebooker/admin_test.rb +102 -0
  128. data/test/facebooker/application_test.rb +110 -0
  129. data/test/facebooker/attachment_test.rb +72 -0
  130. data/test/facebooker/batch_request_test.rb +83 -0
  131. data/test/facebooker/data_test.rb +86 -0
  132. data/test/facebooker/logging_test.rb +43 -0
  133. data/test/facebooker/mobile_test.rb +45 -0
  134. data/test/facebooker/model_test.rb +133 -0
  135. data/test/facebooker/models/event_test.rb +15 -0
  136. data/test/facebooker/models/page_test.rb +56 -0
  137. data/test/facebooker/models/photo_test.rb +16 -0
  138. data/test/facebooker/models/user_test.rb +1087 -0
  139. data/test/facebooker/rails/facebook_request_fix_2-3_test.rb +25 -0
  140. data/test/facebooker/rails/facebook_url_rewriting_test.rb +76 -0
  141. data/test/facebooker/rails/integration_session_test.rb +13 -0
  142. data/test/facebooker/rails/publisher_test.rb +538 -0
  143. data/test/facebooker/rails_integration_test.rb +1543 -0
  144. data/test/facebooker/server_cache_test.rb +44 -0
  145. data/test/facebooker/service_test.rb +58 -0
  146. data/test/facebooker/session_test.rb +883 -0
  147. data/test/facebooker_test.rb +1263 -0
  148. data/test/fixtures/multipart_post_body_with_only_parameters.txt +33 -0
  149. data/test/fixtures/multipart_post_body_with_single_file.txt +38 -0
  150. data/test/fixtures/multipart_post_body_with_single_file_that_has_nil_key.txt +38 -0
  151. data/test/net/http_multipart_post_test.rb +52 -0
  152. data/test/rack/facebook_session_test.rb +34 -0
  153. data/test/rack/facebook_test.rb +73 -0
  154. data/test/rails_test_helper.rb +36 -0
  155. data/test/test_helper.rb +74 -0
  156. metadata +300 -0
@@ -0,0 +1,9 @@
1
+ require 'facebooker/model'
2
+ module Facebooker
3
+ class Video
4
+ include Model
5
+ attr_accessor :vid, :owner, :title,
6
+ :link, :description, :created,
7
+ :story_fbid
8
+ end
9
+ end
@@ -0,0 +1,10 @@
1
+ module Facebooker
2
+ class WorkInfo
3
+ include Model
4
+ attr_accessor :end_date, :start_date, :company_name, :description, :position
5
+ attr_reader :location
6
+ def location=(location)
7
+ @location = location.kind_of?(Hash) ? Location.from_hash(location) : location
8
+ end
9
+ end
10
+ end
@@ -0,0 +1,980 @@
1
+ require 'facebooker/session'
2
+
3
+ begin
4
+ require 'nokogiri'
5
+ rescue Exception
6
+ require 'rexml/document'
7
+ end
8
+
9
+ module Facebooker
10
+ class Parser
11
+
12
+ module REXMLElementExtensions # :nodoc:
13
+ def content
14
+ self.text || ''
15
+ end
16
+
17
+ def [] key
18
+ attributes[key]
19
+ end
20
+
21
+ def text?
22
+ false
23
+ end
24
+ end
25
+
26
+ module REXMLTextExtensions # :nodoc:
27
+ def text?
28
+ true
29
+ end
30
+ end
31
+
32
+ if Object.const_defined?(:REXML) && REXML.const_defined?(:Element)
33
+ ::REXML::Element.__send__(:include, REXMLElementExtensions)
34
+ ::REXML::Text.__send__(:include, REXMLTextExtensions)
35
+ end
36
+
37
+ def self.parse(method, data)
38
+ Errors.process(data)
39
+ parser = Parser::PARSERS[method]
40
+ raise "Can't find a parser for '#{method}'" unless parser
41
+ parser.process(data)
42
+ end
43
+
44
+ def self.array_of(response_element, element_name)
45
+ values_to_return = []
46
+ response_element.children.each do |element|
47
+ next if element.text?
48
+ next unless element.name == element_name
49
+ values_to_return << yield(element)
50
+ end
51
+ values_to_return
52
+ end
53
+
54
+ def self.array_of_text_values(response_element, element_name)
55
+ array_of(response_element, element_name) do |element|
56
+ element.content.strip
57
+ end
58
+ end
59
+
60
+ def self.array_of_hashes(response_element, element_name)
61
+ array_of(response_element, element_name) do |element|
62
+ hashinate(element)
63
+ end
64
+ end
65
+
66
+ def self.element(name, data)
67
+ data = data.body rescue data # either data or an HTTP response
68
+ if Object.const_defined?(:Nokogiri)
69
+ xml = Nokogiri::XML(data.strip)
70
+ if node = xml.at(name)
71
+ return node
72
+ end
73
+ if xml.root.name == name
74
+ return xml.root
75
+ end
76
+ else
77
+ doc = REXML::Document.new(data)
78
+ doc.elements.each(name) do |element|
79
+ return element
80
+ end
81
+ end
82
+ raise "Element #{name} not found in #{data}"
83
+ end
84
+
85
+ def self.hash_or_value_for(element)
86
+ if element.children.size == 1 && element.children.first.text?
87
+ element.content.strip
88
+ else
89
+ # We can have lists in not list item
90
+ if element['list'] == 'true'
91
+ element.children.reject{|c| c.text? }.map { |subchild| hash_or_value_for(subchild)}
92
+ else
93
+ hashinate(element)
94
+ end
95
+ end
96
+ end
97
+
98
+ def self.hashinate(response_element)
99
+ response_element.children.reject{|c| c.text? }.inject({}) do |hash, child|
100
+ # If the node hasn't any child, and is not a list, we want empty strings, not empty hashes,
101
+ # except if attributes['nil'] == true
102
+ hash[child.name] =
103
+ if (child['nil'] == 'true')
104
+ nil
105
+ elsif (child.children.size == 1 && child.children.first.text?) || (child.children.size == 0 && child['list'] != 'true')
106
+ anonymous_field_from(child, hash) || child.content.strip
107
+ elsif child['list'] == 'true'
108
+ child.children.reject{|c| c.text? }.map { |subchild| hash_or_value_for(subchild)}
109
+ else
110
+ child.children.reject{|c| c.text? }.inject({}) do |subhash, subchild|
111
+ subhash[subchild.name] = hash_or_value_for(subchild)
112
+ subhash
113
+ end
114
+ end #if (child.attributes)
115
+ hash
116
+ end #do |hash, child|
117
+ end
118
+
119
+ def self.hash_by_key_or_value_for(element, convert_1_to_true=false)
120
+ if element.children.size == 0
121
+ { element['key'] => nil }
122
+ elsif element.children.size == 1 && element.children.first.text?
123
+ { element['key'] => (convert_1_to_true ? element.content.strip == '1' : element.content.strip) }
124
+ else
125
+ hashinate_by_key(element, convert_1_to_true)
126
+ end
127
+ end
128
+
129
+ # A modification to hashinate. The new dashboard API returns XML in a different format than
130
+ # the other calls. What used to be the element name has become an attribute called "key".
131
+ def self.hashinate_by_key(response_element, convert_1_to_true=false)
132
+ response_element.children.reject{|c| c.text? }.inject({}) do |hash, child|
133
+
134
+ # If the node hasn't any child, and is not a list, we want empty strings, not empty hashes,
135
+ # except if attributes['nil'] == true
136
+ hash[child['key']] =
137
+ if (child['nil'] == 'true')
138
+ nil
139
+ elsif (child.children.size == 1 && child.children.first.text?) || (child.children.size == 0 && child['list'] != 'true')
140
+ anonymous_field_from(child, hash) || (convert_1_to_true ? child.content.strip == '1' : child.content.strip)
141
+ elsif child['list'] == 'true' && child.children.reject {|subchild| subchild.text?}.all? { |subchild| !subchild.text? && subchild['key'].nil? }
142
+ child.children.reject{|c| c.text? }.map { |subchild| hash_by_key_or_value_for(subchild, convert_1_to_true)}
143
+ elsif child['list'] == 'true'
144
+ hash_by_key_or_value_for(child, convert_1_to_true)
145
+ else
146
+ child.children.reject{|c| c.text? }.inject({}) do |subhash, subchild|
147
+ subhash[subchild['key']] = hash_by_key_or_value_for(subchild, convert_1_to_true)
148
+ subhash
149
+ end
150
+ end
151
+ hash
152
+ end
153
+ end
154
+
155
+
156
+
157
+ def self.booleanize(response)
158
+ response == "1" ? true : false
159
+ end
160
+
161
+ def self.anonymous_field_from(child, hash)
162
+ if child.name == 'anon'
163
+ (hash[child.name] || []) << child.content.strip
164
+ end
165
+ end
166
+
167
+ end
168
+
169
+ class RevokeAuthorization < Parser#:nodoc:
170
+ def self.process(data)
171
+ booleanize(data)
172
+ end
173
+ end
174
+
175
+ class RevokeExtendedPermission < Parser#:nodoc:
176
+ def self.process(data)
177
+ booleanize(element('auth_revokeExtendedPermission_response', data).content.strip)
178
+ end
179
+ end
180
+
181
+ class CreateToken < Parser#:nodoc:
182
+ def self.process(data)
183
+ element('auth_createToken_response', data).content.strip
184
+ end
185
+ end
186
+
187
+ class RegisterUsers < Parser
188
+ def self.process(data)
189
+ array_of_text_values(element("connect_registerUsers_response", data), "connect_registerUsers_response_elt")
190
+ end
191
+ end
192
+
193
+ class UnregisterUsers < Parser
194
+ def self.process(data)
195
+ array_of_text_values(element("connect_unregisterUsers_response", data), "connect_unregisterUsers_response_elt")
196
+ end
197
+ end
198
+
199
+ class GetUnconnectedFriendsCount < Parser
200
+ def self.process(data)
201
+ hash_or_value_for(element("connect_getUnconnectedFriendsCount_response",data)).to_i
202
+ end
203
+ end
204
+
205
+ class GetSession < Parser#:nodoc:
206
+ def self.process(data)
207
+ hashinate(element('auth_getSession_response', data))
208
+ end
209
+ end
210
+
211
+ class IsAppUser < Parser#:nodoc:
212
+ def self.process(data)
213
+ element('users_isAppUser_response', data).content == '1'
214
+ end
215
+ end
216
+
217
+ class GetFriends < Parser#:nodoc:
218
+ def self.process(data)
219
+ array_of_text_values(element('friends_get_response', data), 'uid')
220
+ end
221
+ end
222
+
223
+ class FriendListsGet < Parser#:nodoc:
224
+ def self.process(data)
225
+ array_of_hashes(element('friends_getLists_response', data), 'friendlist')
226
+ end
227
+ end
228
+
229
+ class UserInfo < Parser#:nodoc:
230
+ def self.process(data)
231
+ array_of_hashes(element('users_getInfo_response', data), 'user')
232
+ end
233
+ end
234
+
235
+ class UserStandardInfo < Parser#:nodoc:
236
+ def self.process(data)
237
+ array_of_hashes(element('users_getStandardInfo_response', data), 'standard_user_info')
238
+ end
239
+ end
240
+
241
+ class GetLoggedInUser < Parser#:nodoc:
242
+ def self.process(data)
243
+ Integer(element('users_getLoggedInUser_response', data).content.strip)
244
+ end
245
+ end
246
+
247
+ class PagesIsAdmin < Parser#:nodoc:
248
+ def self.process(data)
249
+ element('pages_isAdmin_response', data).content.strip == '1'
250
+ end
251
+ end
252
+
253
+ class PagesGetInfo < Parser#:nodoc:
254
+ def self.process(data)
255
+ array_of_hashes(element('pages_getInfo_response', data), 'page')
256
+ end
257
+ end
258
+
259
+ class PagesIsFan < Parser#:nodoc:
260
+ def self.process(data)
261
+ element('pages_isFan_response', data).content.strip == '1'
262
+ end
263
+ end
264
+
265
+ class PublishStoryToUser < Parser#:nodoc:
266
+ def self.process(data)
267
+ element('feed_publishStoryToUser_response', data).content.strip
268
+ end
269
+ end
270
+
271
+ class StreamPublish < Parser#:nodoc:
272
+ def self.process(data)
273
+ element('stream_publish_response', data).content.strip
274
+ end
275
+ end
276
+
277
+ class StreamAddComment < Parser#:nodoc:
278
+ def self.process(data)
279
+ element('stream_addComment_response', data).content.strip
280
+ end
281
+ end
282
+
283
+ class StreamAddLike < Parser#:nodoc:
284
+ def self.process(data)
285
+ element('stream_addLike_response', data).content.strip
286
+ end
287
+ end
288
+
289
+ class StreamRemoveLike < Parser#:nodoc:
290
+ def self.process(data)
291
+ booleanize(element('stream_removeLike_response', data).content.strip)
292
+ end
293
+ end
294
+
295
+ class RegisterTemplateBundle < Parser#:nodoc:
296
+ def self.process(data)
297
+ element('feed_registerTemplateBundle_response', data).content.to_i
298
+ end
299
+ end
300
+
301
+ class GetRegisteredTemplateBundles < Parser
302
+ def self.process(data)
303
+ array_of_hashes(element('feed_getRegisteredTemplateBundles_response',data), 'template_bundle')
304
+ end
305
+ end
306
+
307
+ class DeactivateTemplateBundleByID < Parser#:nodoc:
308
+ def self.process(data)
309
+ element('feed_deactivateTemplateBundleByID_response', data).content.strip == '1'
310
+ end
311
+ end
312
+
313
+ class PublishUserAction < Parser#:nodoc:
314
+ def self.process(data)
315
+ element('feed_publishUserAction_response', data).children[1].content.strip == "1"
316
+ end
317
+ end
318
+
319
+ class UploadNativeStrings < Parser#:nodoc:
320
+ def self.process(data)
321
+ element('intl_uploadNativeStrings_response', data).content.strip
322
+ end
323
+ end
324
+
325
+ class PublishActionOfUser < Parser#:nodoc:
326
+ def self.process(data)
327
+ element('feed_publishActionOfUser_response', data).content.strip
328
+ end
329
+ end
330
+
331
+ class PublishTemplatizedAction < Parser#:nodoc:
332
+ def self.process(data)
333
+ element('feed_publishTemplatizedAction_response', data).children[1].content.strip
334
+ end
335
+ end
336
+
337
+ class SetAppProperties < Parser#:nodoc:
338
+ def self.process(data)
339
+ element('admin_setAppProperties_response', data).content.strip
340
+ end
341
+ end
342
+
343
+ class GetAppProperties < Parser#:nodoc:
344
+ def self.process(data)
345
+ element('admin_getAppProperties_response', data).content.strip
346
+ end
347
+ end
348
+
349
+ class SetRestrictionInfo < Parser#:nodoc:
350
+ def self.process(data)
351
+ element('admin_setRestrictionInfo_response', data).content.strip
352
+ end
353
+ end
354
+
355
+ class GetRestrictionInfo < Parser#:nodoc:
356
+ def self.process(data)
357
+ element('admin_getRestrictionInfo_response', data).content.strip
358
+ end
359
+ end
360
+
361
+ class GetAllocation < Parser#:nodoc:
362
+ def self.process(data)
363
+ element('admin_getAllocation_response', data).content.strip
364
+ end
365
+ end
366
+
367
+ class GetPublicInfo < Parser#:nodoc:
368
+ def self.process(data)
369
+ hashinate(element('application_getPublicInfo_response', data))
370
+ end
371
+ end
372
+
373
+ class CommentsAdd < Parser#:nodoc:
374
+ def self.process(data)
375
+ element('comments_add_response', data).content.strip
376
+ end
377
+ end
378
+
379
+ class CommentsRemove < Parser#:nodoc:
380
+ def self.process(data)
381
+ booleanize(data)
382
+ end
383
+ end
384
+
385
+ class CommentsGet < Parser#:nodoc:
386
+ def self.process(data)
387
+ array_of_hashes(element('comments_get_response', data), 'comment')
388
+ end
389
+ end
390
+
391
+ class BatchRun < Parser #:nodoc:
392
+ class << self
393
+ def current_batch=(current_batch)
394
+ Thread.current[:facebooker_current_batch]=current_batch
395
+ end
396
+ def current_batch
397
+ Thread.current[:facebooker_current_batch]
398
+ end
399
+ end
400
+ def self.process(data)
401
+ array_of_text_values(element('batch_run_response',data),"batch_run_response_elt").each_with_index do |response,i|
402
+ batch_request=current_batch[i]
403
+ body=Struct.new(:body).new
404
+ body.body=response
405
+ begin
406
+ batch_request.result=Parser.parse(batch_request.method,body)
407
+ rescue Exception=>ex
408
+ batch_request.exception_raised=ex
409
+ end
410
+ end
411
+ end
412
+ end
413
+
414
+ class GetAppUsers < Parser#:nodoc:
415
+ def self.process(data)
416
+ array_of_text_values(element('friends_getAppUsers_response', data), 'uid')
417
+ end
418
+ end
419
+
420
+ class MessageGetThreadsInFolder < Parser#:nodoc:
421
+ def self.process(data)
422
+ array_of_hashes(element('message_getThreadsInFolder_response', data), 'thread')
423
+ end
424
+ end
425
+
426
+ class NotificationsGet < Parser#:nodoc:
427
+ def self.process(data)
428
+ hashinate(element('notifications_get_response', data))
429
+ end
430
+ end
431
+
432
+ class NotificationsSend < Parser#:nodoc:
433
+ def self.process(data)
434
+ element('notifications_send_response', data).content.strip
435
+ end
436
+ end
437
+
438
+ class NotificationsSendEmail < Parser#:nodoc:
439
+ def self.process(data)
440
+ element('notifications_sendEmail_response', data).content.strip
441
+ end
442
+ end
443
+
444
+ class GetTags < Parser#nodoc:
445
+ def self.process(data)
446
+ array_of_hashes(element('photos_getTags_response', data), 'photo_tag')
447
+ end
448
+ end
449
+
450
+ class AddTags < Parser#nodoc:
451
+ def self.process(data)
452
+ element('photos_addTag_response', data)
453
+ end
454
+ end
455
+
456
+ class GetPhotos < Parser#nodoc:
457
+ def self.process(data)
458
+ array_of_hashes(element('photos_get_response', data), 'photo')
459
+ end
460
+ end
461
+
462
+ class GetAlbums < Parser#nodoc:
463
+ def self.process(data)
464
+ array_of_hashes(element('photos_getAlbums_response', data), 'album')
465
+ end
466
+ end
467
+
468
+ class GetStream < Parser #:nodoc:
469
+ def self.process(data)
470
+ response = {}
471
+ response[:albums] = array_of_hashes(element('stream_get_response/albums', data), 'album')
472
+ response[:posts] = array_of_hashes(element('stream_get_response/posts', data), 'stream_post')
473
+ response[:profile] = array_of_hashes(element('stream_get_response/profiles', data), 'profile')
474
+ response
475
+ end
476
+ end
477
+
478
+ class CreateAlbum < Parser#:nodoc:
479
+ def self.process(data)
480
+ hashinate(element('photos_createAlbum_response', data))
481
+ end
482
+ end
483
+
484
+ class UploadPhoto < Parser#:nodoc:
485
+ def self.process(data)
486
+ hashinate(element('photos_upload_response', data))
487
+ end
488
+ end
489
+
490
+ class UploadVideo < Parser#:nodoc:
491
+ def self.process(data)
492
+ hashinate(element('video_upload_response', data))
493
+ end
494
+ end
495
+
496
+ class SendRequest < Parser#:nodoc:
497
+ def self.process(data)
498
+ element('notifications_sendRequest_response', data).content.strip
499
+ end
500
+ end
501
+
502
+ class ProfileFBML < Parser#:nodoc:
503
+ def self.process(data)
504
+ element('profile_getFBML_response', data).content.strip
505
+ end
506
+ end
507
+
508
+ class ProfileFBMLSet < Parser#:nodoc:
509
+ def self.process(data)
510
+ element('profile_setFBML_response', data).content.strip
511
+ end
512
+ end
513
+
514
+ class ProfileInfo < Parser#:nodoc:
515
+ def self.process(data)
516
+ hashinate(element('profile_getInfo_response info_fields', data))
517
+ end
518
+ end
519
+
520
+ class ProfileInfoSet < Parser#:nodoc:
521
+ def self.process(data)
522
+ element('profile_setInfo_response', data).content.strip
523
+ end
524
+ end
525
+
526
+ class FqlQuery < Parser#nodoc
527
+ def self.process(data)
528
+ root = element('fql_query_response', data)
529
+ first_child = root.children.reject{|c| c.text? }.first
530
+ first_child.nil? ? [] : [first_child.name, array_of_hashes(root, first_child.name)]
531
+ end
532
+ end
533
+
534
+ class FqlMultiquery < Parser#nodoc
535
+ def self.process(data)
536
+ root = element('fql_multiquery_response', data)
537
+ root.children.reject { |child| child.text? }.map do |elm|
538
+ elm.children.reject { |child| child.text? }.map do |query|
539
+ if 'name' == query.name
540
+ query.text
541
+ else
542
+ list = query.children.reject { |child| child.text? }
543
+ if list.length == 0
544
+ []
545
+ else
546
+ [list.first.name, array_of_hashes(query, list.first.name)]
547
+ end
548
+ end
549
+ end
550
+ end
551
+ end
552
+ end
553
+
554
+ class SetRefHandle < Parser#:nodoc:
555
+ def self.process(data)
556
+ element('fbml_setRefHandle_response', data).content.strip
557
+ end
558
+ end
559
+
560
+ class RefreshRefURL < Parser#:nodoc:
561
+ def self.process(data)
562
+ element('fbml_refreshRefUrl_response', data).content.strip
563
+ end
564
+ end
565
+
566
+ class RefreshImgSrc < Parser#:nodoc:
567
+ def self.process(data)
568
+ element('fbml_refreshImgSrc_response', data).content.strip
569
+ end
570
+ end
571
+
572
+ class SetCookie < Parser#:nodoc:
573
+ def self.process(data)
574
+ element('data_setCookie_response', data).content.strip
575
+ end
576
+ end
577
+
578
+ class GetCookies < Parser#:nodoc:
579
+ def self.process(data)
580
+ array_of_hashes(element('data_getCookies_response', data), 'cookies')
581
+ end
582
+ end
583
+
584
+ class EventsRsvp < Parser#:nodoc:
585
+ def self.process(data)
586
+ element('events_rsvp_response', data).content.strip
587
+ end
588
+ end
589
+
590
+ class EventsCreate < Parser#:nodoc:
591
+ def self.process(data)
592
+ element('events_create_response', data).content.strip
593
+ end
594
+ end
595
+
596
+ class EventsCancel < Parser#:nodoc:
597
+ def self.process(data)
598
+ element('events_cancel_response', data).content.strip
599
+ end
600
+ end
601
+
602
+ class EventsGet < Parser#:nodoc:
603
+ def self.process(data)
604
+ array_of_hashes(element('events_get_response', data), 'event')
605
+ end
606
+ end
607
+
608
+ class GroupGetMembers < Parser#:nodoc:
609
+ def self.process(data)
610
+ root = element('groups_getMembers_response', data)
611
+ result = ['members', 'admins', 'officers', 'not_replied'].map do |position|
612
+ array_of(root, position) {|element| element}.map do |element|
613
+ array_of_text_values(element, 'uid').map do |uid|
614
+ {:position => position}.merge(:uid => uid)
615
+ end
616
+ end
617
+ end.flatten
618
+ end
619
+ end
620
+
621
+ class EventMembersGet < Parser#:nodoc:
622
+ def self.process(data)
623
+ root = element('events_getMembers_response', data)
624
+ result = ['attending', 'declined', 'unsure', 'not_replied'].map do |rsvp_status|
625
+ array_of(root, rsvp_status) {|element| element}.map do |element|
626
+ array_of_text_values(element, 'uid').map do |uid|
627
+ {:rsvp_status => rsvp_status}.merge(:uid => uid)
628
+ end
629
+ end
630
+ end.flatten
631
+ end
632
+ end
633
+
634
+ class GroupsGet < Parser#:nodoc:
635
+ def self.process(data)
636
+ array_of_hashes(element('groups_get_response', data), 'group')
637
+ end
638
+ end
639
+
640
+ class AreFriends < Parser#:nodoc:
641
+ def self.process(data)
642
+ array_of_hashes(element('friends_areFriends_response', data), 'friend_info').inject({}) do |memo, hash|
643
+ memo[[Integer(hash['uid1']), Integer(hash['uid2'])].sort] = are_friends?(hash['are_friends'])
644
+ memo
645
+ end
646
+ end
647
+
648
+ private
649
+ def self.are_friends?(raw_value)
650
+ if raw_value == '1'
651
+ true
652
+ elsif raw_value == '0'
653
+ false
654
+ else
655
+ nil
656
+ end
657
+ end
658
+ end
659
+
660
+ class SetStatus < Parser
661
+ def self.process(data)
662
+ element('users_setStatus_response',data).content.strip == '1'
663
+ end
664
+ end
665
+
666
+ class GetStatus < Parser # :nodoc:
667
+ def self.process(data)
668
+ array_of_hashes(element('status_get_response',data),'user_status')
669
+ end
670
+ end
671
+
672
+ class GetPreference < Parser#:nodoc:
673
+ def self.process(data)
674
+ element('data_getUserPreference_response', data).content.strip
675
+ end
676
+ end
677
+
678
+ class SetPreference < Parser#:nodoc:
679
+ def self.process(data)
680
+ element('data_setUserPreference_response', data).content.strip
681
+ end
682
+ end
683
+
684
+ class UserHasPermission < Parser
685
+ def self.process(data)
686
+ element('users_hasAppPermission_response', data).content.strip
687
+ end
688
+ end
689
+
690
+ class SmsSend < Parser#:nodoc:
691
+ def self.process(data)
692
+ element('sms_send_response', data).content.strip == '0'
693
+ end
694
+ end
695
+
696
+ class SmsCanSend < Parser#:nodoc:
697
+ def self.process(data)
698
+ element('sms_canSend_response', data).content.strip
699
+ end
700
+ end
701
+
702
+ class DashboardGetCount < Parser
703
+ def self.process(data)
704
+ element('dashboard_getCount_response', data).content.strip
705
+ end
706
+ end
707
+
708
+ class DashboardSetCount < Parser
709
+ def self.process(data)
710
+ element('dashboard_setCount_response', data).content.strip == '1'
711
+ end
712
+ end
713
+
714
+ class DashboardIncrementCount < Parser
715
+ def self.process(data)
716
+ element('dashboard_incrementCount_response', data).content.strip == '1'
717
+ end
718
+ end
719
+
720
+ class DashboardDecrementCount < Parser
721
+ def self.process(data)
722
+ element('dashboard_decrementCount_response', data).content.strip == '1'
723
+ end
724
+ end
725
+
726
+ class DashboardMultiGetCount < Parser
727
+ def self.process(data)
728
+ hashinate_by_key(element('dashboard_multiGetCount_response', data))
729
+ end
730
+ end
731
+
732
+ class DashboardMultiSetCount < Parser
733
+ def self.process(data)
734
+ hashinate_by_key(element('dashboard_multiSetCount_response', data), true)
735
+ end
736
+ end
737
+
738
+ class DashboardMultiIncrementCount < Parser
739
+ def self.process(data)
740
+ hashinate_by_key(element('dashboard_multiIncrementCount_response', data), true)
741
+ end
742
+ end
743
+
744
+ class DashboardMultiDecrementCount < Parser
745
+ def self.process(data)
746
+ hashinate_by_key(element('dashboard_multiDecrementCount_response', data), true)
747
+ end
748
+ end
749
+
750
+ class DashboardAddGlobalNews < Parser
751
+ def self.process(data)
752
+ element('dashboard_addGlobalNews_response', data).content.strip
753
+ end
754
+ end
755
+
756
+ # Currently, always returns all
757
+ class DashboardGetGlobalNews < Parser
758
+ def self.process(data)
759
+ hashinate_by_key(element('dashboard_getGlobalNews_response', data))
760
+ end
761
+ end
762
+
763
+ class DashboardClearGlobalNews < Parser
764
+ def self.process(data)
765
+ hashinate_by_key(element('dashboard_clearGlobalNews_response', data), true)
766
+ end
767
+ end
768
+
769
+ class DashboardAddNews < Parser
770
+ def self.process(data)
771
+ element('dashboard_addNews_response', data).content.strip
772
+ end
773
+ end
774
+
775
+ class DashboardGetNews < Parser
776
+ def self.process(data)
777
+ hashinate_by_key(element('dashboard_getNews_response', data))
778
+ end
779
+ end
780
+
781
+ class DashboardClearNews < Parser
782
+ def self.process(data)
783
+ hashinate_by_key(element('dashboard_clearNews_response', data), true)
784
+ end
785
+ end
786
+
787
+ class DashboardMultiAddNews < Parser
788
+ def self.process(data)
789
+ hashinate_by_key(element('dashboard_multiAddNews_response', data))
790
+ end
791
+ end
792
+
793
+ class DashboardMultiClearNews < Parser
794
+ def self.process(data)
795
+ hashinate_by_key(element('dashboard_multiClearNews_response', data), true)
796
+ end
797
+ end
798
+
799
+ class DashboardMultiGetNews < Parser
800
+ def self.process(data)
801
+ hashinate_by_key(element('dashboard_multiGetNews_response', data))
802
+ end
803
+ end
804
+
805
+ class DashboardPublishActivity < Parser
806
+ def self.process(data)
807
+ element('dashboard_publishActivity_response', data).content.strip
808
+ end
809
+ end
810
+
811
+ class DashboardRemoveActivity < Parser
812
+ def self.process(data)
813
+ hashinate_by_key(element('dashboard_removeActivity_response', data), true)
814
+ end
815
+ end
816
+
817
+ class DashboardGetActivity < Parser
818
+ def self.process(data)
819
+ hashinate_by_key(element('dashboard_getActivity_response', data))
820
+ end
821
+ end
822
+
823
+
824
+ class Errors < Parser#:nodoc:
825
+ EXCEPTIONS = {
826
+ 1 => Facebooker::Session::UnknownError,
827
+ 2 => Facebooker::Session::ServiceUnavailable,
828
+ 4 => Facebooker::Session::MaxRequestsDepleted,
829
+ 5 => Facebooker::Session::HostNotAllowed,
830
+ 10 => Facebooker::Session::AppPermissionError,
831
+ 100 => Facebooker::Session::MissingOrInvalidParameter,
832
+ 101 => Facebooker::Session::InvalidAPIKey,
833
+ 102 => Facebooker::Session::SessionExpired,
834
+ 103 => Facebooker::Session::CallOutOfOrder,
835
+ 104 => Facebooker::Session::IncorrectSignature,
836
+ 120 => Facebooker::Session::InvalidAlbumId,
837
+ 200 => Facebooker::Session::PermissionError,
838
+ 250 => Facebooker::Session::ExtendedPermissionRequired,
839
+ 321 => Facebooker::Session::AlbumIsFull,
840
+ 324 => Facebooker::Session::MissingOrInvalidImageFile,
841
+ 325 => Facebooker::Session::TooManyUnapprovedPhotosPending,
842
+ 330 => Facebooker::Session::TemplateDataMissingRequiredTokens,
843
+ 340 => Facebooker::Session::TooManyUserCalls,
844
+ 341 => Facebooker::Session::TooManyUserActionCalls,
845
+ 342 => Facebooker::Session::InvalidFeedTitleLink,
846
+ 343 => Facebooker::Session::InvalidFeedTitleLength,
847
+ 344 => Facebooker::Session::InvalidFeedTitleName,
848
+ 345 => Facebooker::Session::BlankFeedTitle,
849
+ 346 => Facebooker::Session::FeedBodyLengthTooLong,
850
+ 347 => Facebooker::Session::InvalidFeedPhotoSource,
851
+ 348 => Facebooker::Session::InvalidFeedPhotoLink,
852
+ 330 => Facebooker::Session::FeedMarkupInvalid,
853
+ 360 => Facebooker::Session::FeedTitleDataInvalid,
854
+ 361 => Facebooker::Session::FeedTitleTemplateInvalid,
855
+ 362 => Facebooker::Session::FeedBodyDataInvalid,
856
+ 363 => Facebooker::Session::FeedBodyTemplateInvalid,
857
+ 364 => Facebooker::Session::FeedPhotosNotRetrieved,
858
+ 366 => Facebooker::Session::FeedTargetIdsInvalid,
859
+ 601 => Facebooker::Session::FQLParseError,
860
+ 602 => Facebooker::Session::FQLFieldDoesNotExist,
861
+ 603 => Facebooker::Session::FQLTableDoesNotExist,
862
+ 604 => Facebooker::Session::FQLStatementNotIndexable,
863
+ 605 => Facebooker::Session::FQLFunctionDoesNotExist,
864
+ 606 => Facebooker::Session::FQLWrongNumberArgumentsPassedToFunction,
865
+ 612 => Facebooker::Session::ReadMailboxExtendedPermissionRequired,
866
+ 807 => Facebooker::Session::TemplateBundleInvalid
867
+ }
868
+ def self.process(data)
869
+ response_element = element('error_response', data) rescue nil
870
+ if response_element
871
+ hash = hashinate(response_element)
872
+ exception = EXCEPTIONS[Integer(hash['error_code'])] || StandardError
873
+ raise exception, hash['error_msg']
874
+ end
875
+ end
876
+ end
877
+
878
+ class Parser
879
+ PARSERS = {
880
+ 'facebook.auth.revokeAuthorization' => RevokeAuthorization,
881
+ 'facebook.auth.revokeExtendedPermission' => RevokeExtendedPermission,
882
+ 'facebook.auth.createToken' => CreateToken,
883
+ 'facebook.auth.getSession' => GetSession,
884
+ 'facebook.connect.registerUsers' => RegisterUsers,
885
+ 'facebook.connect.unregisterUsers' => UnregisterUsers,
886
+ 'facebook.connect.getUnconnectedFriendsCount' => GetUnconnectedFriendsCount,
887
+ 'facebook.users.getInfo' => UserInfo,
888
+ 'facebook.users.getStandardInfo' => UserStandardInfo,
889
+ 'facebook.users.setStatus' => SetStatus,
890
+ 'facebook.status.get' => GetStatus,
891
+ 'facebook.users.getLoggedInUser' => GetLoggedInUser,
892
+ 'facebook.users.hasAppPermission' => UserHasPermission,
893
+ 'facebook.users.isAppUser' => IsAppUser,
894
+ 'facebook.pages.isAdmin' => PagesIsAdmin,
895
+ 'facebook.pages.getInfo' => PagesGetInfo,
896
+ 'facebook.pages.isFan' => PagesIsFan,
897
+ 'facebook.friends.get' => GetFriends,
898
+ 'facebook.friends.getLists' => FriendListsGet,
899
+ 'facebook.friends.areFriends' => AreFriends,
900
+ 'facebook.friends.getAppUsers' => GetAppUsers,
901
+ 'facebook.feed.publishStoryToUser' => PublishStoryToUser,
902
+ 'facebook.feed.publishActionOfUser' => PublishActionOfUser,
903
+ 'facebook.feed.publishTemplatizedAction' => PublishTemplatizedAction,
904
+ 'facebook.feed.registerTemplateBundle' => RegisterTemplateBundle,
905
+ 'facebook.feed.deactivateTemplateBundleByID' => DeactivateTemplateBundleByID,
906
+ 'facebook.feed.getRegisteredTemplateBundles' => GetRegisteredTemplateBundles,
907
+ 'facebook.feed.publishUserAction' => PublishUserAction,
908
+ 'facebook.message.getThreadsInFolder' => MessageGetThreadsInFolder,
909
+ 'facebook.notifications.get' => NotificationsGet,
910
+ 'facebook.notifications.send' => NotificationsSend,
911
+ 'facebook.notifications.sendRequest' => SendRequest,
912
+ 'facebook.profile.getFBML' => ProfileFBML,
913
+ 'facebook.profile.setFBML' => ProfileFBMLSet,
914
+ 'facebook.profile.getInfo' => ProfileInfo,
915
+ 'facebook.profile.setInfo' => ProfileInfoSet,
916
+ 'facebook.fbml.setRefHandle' => SetRefHandle,
917
+ 'facebook.fbml.refreshRefUrl' => RefreshRefURL,
918
+ 'facebook.fbml.refreshImgSrc' => RefreshImgSrc,
919
+ 'facebook.data.setCookie' => SetCookie,
920
+ 'facebook.data.getCookies' => GetCookies,
921
+ 'facebook.admin.setAppProperties' => SetAppProperties,
922
+ 'facebook.admin.getAppProperties' => GetAppProperties,
923
+ 'facebook.admin.setRestrictionInfo' => SetRestrictionInfo,
924
+ 'facebook.admin.getRestrictionInfo' => GetRestrictionInfo,
925
+ 'facebook.admin.getAllocation' => GetAllocation,
926
+ 'facebook.application.getPublicInfo' => GetPublicInfo,
927
+ 'facebook.batch.run' => BatchRun,
928
+ 'facebook.fql.query' => FqlQuery,
929
+ 'facebook.fql.multiquery' => FqlMultiquery,
930
+ 'facebook.photos.get' => GetPhotos,
931
+ 'facebook.photos.getAlbums' => GetAlbums,
932
+ 'facebook.photos.createAlbum' => CreateAlbum,
933
+ 'facebook.photos.getTags' => GetTags,
934
+ 'facebook.photos.addTag' => AddTags,
935
+ 'facebook.photos.upload' => UploadPhoto,
936
+ 'facebook.stream.get' => GetStream,
937
+ 'facebook.stream.publish' => StreamPublish,
938
+ 'facebook.stream.addComment' => StreamAddComment,
939
+ 'facebook.stream.addLike' => StreamAddLike,
940
+ 'facebook.stream.removeLike' => StreamRemoveLike,
941
+ 'facebook.events.create' => EventsCreate,
942
+ 'facebook.events.cancel' => EventsCancel,
943
+ 'facebook.events.get' => EventsGet,
944
+ 'facebook.events.rsvp' => EventsRsvp,
945
+ 'facebook.groups.get' => GroupsGet,
946
+ 'facebook.events.getMembers' => EventMembersGet,
947
+ 'facebook.groups.getMembers' => GroupGetMembers,
948
+ 'facebook.notifications.sendEmail' => NotificationsSendEmail,
949
+ 'facebook.data.getUserPreference' => GetPreference,
950
+ 'facebook.data.setUserPreference' => SetPreference,
951
+ 'facebook.video.upload' => UploadVideo,
952
+ 'facebook.sms.send' => SmsSend,
953
+ 'facebook.sms.canSend' => SmsCanSend,
954
+ 'facebook.comments.add' => CommentsAdd,
955
+ 'facebook.comments.remove' => CommentsRemove,
956
+ 'facebook.comments.get' => CommentsGet,
957
+ 'facebook.dashboard.setCount' => DashboardSetCount,
958
+ 'facebook.dashboard.getCount' => DashboardGetCount,
959
+ 'facebook.dashboard.incrementCount' => DashboardIncrementCount,
960
+ 'facebook.dashboard.decrementCount' => DashboardDecrementCount,
961
+ 'facebook.dashboard.multiGetCount' => DashboardMultiGetCount,
962
+ 'facebook.dashboard.multiSetCount' => DashboardMultiSetCount,
963
+ 'facebook.dashboard.multiIncrementCount' => DashboardMultiIncrementCount,
964
+ 'facebook.dashboard.multiDecrementCount' => DashboardMultiDecrementCount,
965
+ 'facebook.dashboard.addGlobalNews' => DashboardAddGlobalNews,
966
+ 'facebook.dashboard.getGlobalNews' => DashboardGetGlobalNews,
967
+ 'facebook.dashboard.clearGlobalNews' => DashboardClearGlobalNews,
968
+ 'facebook.dashboard.addNews' => DashboardAddNews,
969
+ 'facebook.dashboard.getNews' => DashboardGetNews,
970
+ 'facebook.dashboard.clearNews' => DashboardClearNews,
971
+ 'facebook.dashboard.multiAddNews' => DashboardMultiAddNews,
972
+ 'facebook.dashboard.multiGetNews' => DashboardMultiGetNews,
973
+ 'facebook.dashboard.multiClearNews' => DashboardMultiClearNews,
974
+ 'facebook.dashboard.publishActivity' => DashboardPublishActivity,
975
+ 'facebook.dashboard.removeActivity' => DashboardRemoveActivity,
976
+ 'facebook.dashboard.getActivity' => DashboardGetActivity,
977
+ 'facebook.intl.uploadNativeStrings' => UploadNativeStrings
978
+ }
979
+ end
980
+ end