thorero 0.9.4.4 → 0.9.4.5

Sign up to get free protection for your applications and to get access to all the features.
Files changed (298) hide show
  1. data/LICENSE +1 -1
  2. data/README +21 -0
  3. data/Rakefile +275 -108
  4. data/TODO +0 -0
  5. data/bin/merb +12 -0
  6. data/bin/merb-specs +5 -0
  7. data/docs/bootloading.dox +58 -0
  8. data/docs/documentation_standards +40 -0
  9. data/docs/merb-core-call-stack-diagram.mmap +0 -0
  10. data/docs/merb-core-call-stack-diagram.pdf +0 -0
  11. data/docs/merb-core-call-stack-diagram.png +0 -0
  12. data/docs/new_render_api +51 -0
  13. data/lib/merb-core.rb +603 -0
  14. data/lib/merb-core/autoload.rb +32 -0
  15. data/lib/merb-core/bootloader.rb +708 -0
  16. data/lib/merb-core/config.rb +303 -0
  17. data/lib/merb-core/constants.rb +43 -0
  18. data/lib/merb-core/controller/abstract_controller.rb +578 -0
  19. data/lib/merb-core/controller/exceptions.rb +302 -0
  20. data/lib/merb-core/controller/merb_controller.rb +256 -0
  21. data/lib/merb-core/controller/mime.rb +106 -0
  22. data/lib/merb-core/controller/mixins/authentication.rb +87 -0
  23. data/lib/merb-core/controller/mixins/controller.rb +290 -0
  24. data/lib/merb-core/controller/mixins/render.rb +481 -0
  25. data/lib/merb-core/controller/mixins/responder.rb +472 -0
  26. data/lib/merb-core/controller/template.rb +254 -0
  27. data/lib/merb-core/core_ext.rb +8 -0
  28. data/lib/merb-core/core_ext/kernel.rb +319 -0
  29. data/lib/merb-core/dispatch/cookies.rb +91 -0
  30. data/lib/merb-core/dispatch/dispatcher.rb +278 -0
  31. data/lib/merb-core/dispatch/exceptions.html.erb +303 -0
  32. data/lib/merb-core/dispatch/request.rb +603 -0
  33. data/lib/merb-core/dispatch/router.rb +179 -0
  34. data/lib/merb-core/dispatch/router/behavior.rb +867 -0
  35. data/lib/merb-core/dispatch/router/cached_proc.rb +52 -0
  36. data/lib/merb-core/dispatch/router/route.rb +321 -0
  37. data/lib/merb-core/dispatch/session.rb +78 -0
  38. data/lib/merb-core/dispatch/session/cookie.rb +168 -0
  39. data/lib/merb-core/dispatch/session/memcached.rb +184 -0
  40. data/lib/merb-core/dispatch/session/memory.rb +241 -0
  41. data/lib/merb-core/dispatch/worker.rb +28 -0
  42. data/lib/merb-core/gem_ext/erubis.rb +77 -0
  43. data/lib/{extlib → merb-core}/logger.rb +2 -2
  44. data/lib/merb-core/plugins.rb +59 -0
  45. data/lib/merb-core/rack.rb +21 -0
  46. data/lib/merb-core/rack/adapter.rb +44 -0
  47. data/lib/merb-core/rack/adapter/ebb.rb +25 -0
  48. data/lib/merb-core/rack/adapter/evented_mongrel.rb +26 -0
  49. data/lib/merb-core/rack/adapter/fcgi.rb +17 -0
  50. data/lib/merb-core/rack/adapter/irb.rb +118 -0
  51. data/lib/merb-core/rack/adapter/mongrel.rb +26 -0
  52. data/lib/merb-core/rack/adapter/runner.rb +28 -0
  53. data/lib/merb-core/rack/adapter/swiftiplied_mongrel.rb +26 -0
  54. data/lib/merb-core/rack/adapter/thin.rb +39 -0
  55. data/lib/merb-core/rack/adapter/thin_turbo.rb +24 -0
  56. data/lib/merb-core/rack/adapter/webrick.rb +36 -0
  57. data/lib/merb-core/rack/application.rb +18 -0
  58. data/lib/merb-core/rack/handler/mongrel.rb +97 -0
  59. data/lib/merb-core/rack/middleware.rb +26 -0
  60. data/lib/merb-core/rack/middleware/path_prefix.rb +31 -0
  61. data/lib/merb-core/rack/middleware/profiler.rb +19 -0
  62. data/lib/merb-core/rack/middleware/static.rb +45 -0
  63. data/lib/merb-core/server.rb +252 -0
  64. data/lib/merb-core/tasks/audit.rake +68 -0
  65. data/lib/merb-core/tasks/merb.rb +1 -0
  66. data/lib/merb-core/tasks/merb_rake_helper.rb +12 -0
  67. data/lib/merb-core/test.rb +11 -0
  68. data/lib/merb-core/test/helpers.rb +9 -0
  69. data/lib/merb-core/test/helpers/controller_helper.rb +8 -0
  70. data/lib/merb-core/test/helpers/multipart_request_helper.rb +175 -0
  71. data/lib/merb-core/test/helpers/request_helper.rb +344 -0
  72. data/lib/merb-core/test/helpers/route_helper.rb +33 -0
  73. data/lib/merb-core/test/helpers/view_helper.rb +121 -0
  74. data/lib/merb-core/test/matchers.rb +9 -0
  75. data/lib/merb-core/test/matchers/controller_matchers.rb +319 -0
  76. data/lib/merb-core/test/matchers/route_matchers.rb +136 -0
  77. data/lib/merb-core/test/matchers/view_matchers.rb +335 -0
  78. data/lib/merb-core/test/run_specs.rb +47 -0
  79. data/lib/merb-core/test/tasks/spectasks.rb +68 -0
  80. data/lib/merb-core/test/test_ext/hpricot.rb +32 -0
  81. data/lib/merb-core/test/test_ext/object.rb +14 -0
  82. data/lib/merb-core/test/test_ext/string.rb +14 -0
  83. data/lib/merb-core/vendor/facets.rb +2 -0
  84. data/lib/merb-core/vendor/facets/dictionary.rb +433 -0
  85. data/lib/merb-core/vendor/facets/inflect.rb +345 -0
  86. data/lib/merb-core/version.rb +11 -0
  87. data/spec/private/config/adapter_spec.rb +32 -0
  88. data/spec/private/config/config_spec.rb +202 -0
  89. data/spec/private/config/environment_spec.rb +13 -0
  90. data/spec/private/config/spec_helper.rb +1 -0
  91. data/spec/private/core_ext/kernel_spec.rb +169 -0
  92. data/spec/private/dispatch/bootloader_spec.rb +24 -0
  93. data/spec/private/dispatch/cookies_spec.rb +107 -0
  94. data/spec/private/dispatch/dispatch_spec.rb +35 -0
  95. data/spec/private/dispatch/fixture/app/controllers/application.rb +4 -0
  96. data/spec/private/dispatch/fixture/app/controllers/exceptions.rb +27 -0
  97. data/spec/private/dispatch/fixture/app/controllers/foo.rb +21 -0
  98. data/spec/private/dispatch/fixture/app/helpers/global_helpers.rb +8 -0
  99. data/spec/private/dispatch/fixture/app/views/exeptions/client_error.html.erb +37 -0
  100. data/spec/private/dispatch/fixture/app/views/exeptions/internal_server_error.html.erb +216 -0
  101. data/spec/private/dispatch/fixture/app/views/exeptions/not_acceptable.html.erb +38 -0
  102. data/spec/private/dispatch/fixture/app/views/exeptions/not_found.html.erb +40 -0
  103. data/spec/private/dispatch/fixture/app/views/foo/bar.html.erb +0 -0
  104. data/spec/private/dispatch/fixture/app/views/layout/application.html.erb +11 -0
  105. data/spec/private/dispatch/fixture/config/black_hole.rb +12 -0
  106. data/spec/private/dispatch/fixture/config/environments/development.rb +6 -0
  107. data/spec/private/dispatch/fixture/config/environments/production.rb +5 -0
  108. data/spec/private/dispatch/fixture/config/environments/test.rb +6 -0
  109. data/spec/private/dispatch/fixture/config/init.rb +45 -0
  110. data/spec/private/dispatch/fixture/config/rack.rb +11 -0
  111. data/spec/private/dispatch/fixture/config/router.rb +35 -0
  112. data/spec/private/dispatch/fixture/log/merb_test.log +1874 -0
  113. data/spec/private/dispatch/fixture/public/images/merb.jpg +0 -0
  114. data/spec/private/dispatch/fixture/public/merb.fcgi +4 -0
  115. data/spec/private/dispatch/fixture/public/stylesheets/master.css +119 -0
  116. data/spec/private/dispatch/route_params_spec.rb +24 -0
  117. data/spec/private/dispatch/session_mixin_spec.rb +47 -0
  118. data/spec/private/dispatch/spec_helper.rb +1 -0
  119. data/spec/private/plugins/plugin_spec.rb +166 -0
  120. data/spec/private/rack/application_spec.rb +49 -0
  121. data/spec/private/router/behavior_spec.rb +60 -0
  122. data/spec/private/router/fixture/log/merb_test.log +139 -0
  123. data/spec/private/router/route_spec.rb +414 -0
  124. data/spec/private/router/router_spec.rb +175 -0
  125. data/spec/private/vendor/facets/plural_spec.rb +564 -0
  126. data/spec/private/vendor/facets/singular_spec.rb +489 -0
  127. data/spec/public/DEFINITIONS +11 -0
  128. data/spec/public/abstract_controller/controllers/alt_views/layout/application.erb +1 -0
  129. data/spec/public/abstract_controller/controllers/alt_views/layout/merb/test/fixtures/abstract/render_string_controller_layout.erb +1 -0
  130. data/spec/public/abstract_controller/controllers/alt_views/layout/merb/test/fixtures/abstract/render_template_controller_layout.erb +1 -0
  131. data/spec/public/abstract_controller/controllers/alt_views/merb/test/fixtures/abstract/display_object_with_multiple_roots/index.erb +1 -0
  132. data/spec/public/abstract_controller/controllers/alt_views/merb/test/fixtures/abstract/display_object_with_multiple_roots/show.erb +1 -0
  133. data/spec/public/abstract_controller/controllers/alt_views/merb/test/fixtures/abstract/render_template_multiple_roots/index.erb +1 -0
  134. data/spec/public/abstract_controller/controllers/alt_views/partial/basic_partial_with_multiple_roots/_partial.erb +1 -0
  135. data/spec/public/abstract_controller/controllers/alt_views/render_template_multiple_roots_and_custom_location/index.erb +1 -0
  136. data/spec/public/abstract_controller/controllers/alt_views/render_template_multiple_roots_inherited/index.erb +1 -0
  137. data/spec/public/abstract_controller/controllers/cousins.rb +41 -0
  138. data/spec/public/abstract_controller/controllers/display.rb +54 -0
  139. data/spec/public/abstract_controller/controllers/filters.rb +193 -0
  140. data/spec/public/abstract_controller/controllers/helpers.rb +41 -0
  141. data/spec/public/abstract_controller/controllers/partial.rb +121 -0
  142. data/spec/public/abstract_controller/controllers/render.rb +113 -0
  143. data/spec/public/abstract_controller/controllers/views/helpers/capture/index.erb +1 -0
  144. data/spec/public/abstract_controller/controllers/views/helpers/capture_eq/index.erb +1 -0
  145. data/spec/public/abstract_controller/controllers/views/helpers/capture_with_args/index.erb +1 -0
  146. data/spec/public/abstract_controller/controllers/views/helpers/concat/index.erb +1 -0
  147. data/spec/public/abstract_controller/controllers/views/layout/alt.erb +1 -0
  148. data/spec/public/abstract_controller/controllers/views/layout/custom.erb +1 -0
  149. data/spec/public/abstract_controller/controllers/views/merb/test/fixtures/abstract/display_object/index.erb +1 -0
  150. data/spec/public/abstract_controller/controllers/views/merb/test/fixtures/abstract/display_object_with_action/new.erb +1 -0
  151. data/spec/public/abstract_controller/controllers/views/merb/test/fixtures/abstract/render_template/index.erb +1 -0
  152. data/spec/public/abstract_controller/controllers/views/merb/test/fixtures/abstract/render_template_app_layout/index.erb +0 -0
  153. data/spec/public/abstract_controller/controllers/views/merb/test/fixtures/abstract/render_template_custom_layout/index.erb +1 -0
  154. data/spec/public/abstract_controller/controllers/views/merb/test/fixtures/abstract/render_template_multiple_roots/index.erb +1 -0
  155. data/spec/public/abstract_controller/controllers/views/merb/test/fixtures/abstract/render_template_multiple_roots/show.erb +1 -0
  156. data/spec/public/abstract_controller/controllers/views/merb/test/fixtures/abstract/render_two_throw_contents/index.erb +1 -0
  157. data/spec/public/abstract_controller/controllers/views/partial/another_directory/_partial.erb +1 -0
  158. data/spec/public/abstract_controller/controllers/views/partial/basic_partial/_partial.erb +1 -0
  159. data/spec/public/abstract_controller/controllers/views/partial/basic_partial/index.erb +1 -0
  160. data/spec/public/abstract_controller/controllers/views/partial/basic_partial_with_multiple_roots/index.erb +1 -0
  161. data/spec/public/abstract_controller/controllers/views/partial/nested_partial/_first.erb +1 -0
  162. data/spec/public/abstract_controller/controllers/views/partial/nested_partial/_second.erb +1 -0
  163. data/spec/public/abstract_controller/controllers/views/partial/nested_partial/index.erb +1 -0
  164. data/spec/public/abstract_controller/controllers/views/partial/partial_in_another_directory/index.erb +1 -0
  165. data/spec/public/abstract_controller/controllers/views/partial/partial_with_both/_collection.erb +1 -0
  166. data/spec/public/abstract_controller/controllers/views/partial/partial_with_both/index.erb +1 -0
  167. data/spec/public/abstract_controller/controllers/views/partial/partial_with_collections/_collection.erb +1 -0
  168. data/spec/public/abstract_controller/controllers/views/partial/partial_with_collections/index.erb +1 -0
  169. data/spec/public/abstract_controller/controllers/views/partial/partial_with_collections_and_as/_collection.erb +1 -0
  170. data/spec/public/abstract_controller/controllers/views/partial/partial_with_collections_and_as/index.erb +1 -0
  171. data/spec/public/abstract_controller/controllers/views/partial/partial_with_collections_and_counter/_collection.erb +1 -0
  172. data/spec/public/abstract_controller/controllers/views/partial/partial_with_collections_and_counter/index.erb +1 -0
  173. data/spec/public/abstract_controller/controllers/views/partial/partial_with_locals/_variables.erb +1 -0
  174. data/spec/public/abstract_controller/controllers/views/partial/partial_with_locals/index.erb +1 -0
  175. data/spec/public/abstract_controller/controllers/views/partial/partial_with_with_and_locals/_both.erb +1 -0
  176. data/spec/public/abstract_controller/controllers/views/partial/partial_with_with_and_locals/index.erb +1 -0
  177. data/spec/public/abstract_controller/controllers/views/partial/with_absolute_partial/_partial.erb +1 -0
  178. data/spec/public/abstract_controller/controllers/views/partial/with_absolute_partial/index.erb +1 -0
  179. data/spec/public/abstract_controller/controllers/views/partial/with_as_partial/_with_partial.erb +1 -0
  180. data/spec/public/abstract_controller/controllers/views/partial/with_as_partial/index.erb +1 -0
  181. data/spec/public/abstract_controller/controllers/views/partial/with_nil_partial/_with_partial.erb +1 -0
  182. data/spec/public/abstract_controller/controllers/views/partial/with_nil_partial/index.erb +1 -0
  183. data/spec/public/abstract_controller/controllers/views/partial/with_partial/_with_partial.erb +1 -0
  184. data/spec/public/abstract_controller/controllers/views/partial/with_partial/index.erb +1 -0
  185. data/spec/public/abstract_controller/controllers/views/test_display/foo.html.erb +1 -0
  186. data/spec/public/abstract_controller/controllers/views/test_render/foo.html.erb +0 -0
  187. data/spec/public/abstract_controller/controllers/views/wonderful/index.erb +1 -0
  188. data/spec/public/abstract_controller/display_spec.rb +33 -0
  189. data/spec/public/abstract_controller/filter_spec.rb +106 -0
  190. data/spec/public/abstract_controller/helper_spec.rb +21 -0
  191. data/spec/public/abstract_controller/partial_spec.rb +61 -0
  192. data/spec/public/abstract_controller/render_spec.rb +90 -0
  193. data/spec/public/abstract_controller/spec_helper.rb +31 -0
  194. data/spec/public/boot_loader/boot_loader_spec.rb +33 -0
  195. data/spec/public/boot_loader/spec_helper.rb +1 -0
  196. data/spec/public/controller/authentication_spec.rb +103 -0
  197. data/spec/public/controller/base_spec.rb +36 -0
  198. data/spec/public/controller/controllers/authentication.rb +45 -0
  199. data/spec/public/controller/controllers/base.rb +36 -0
  200. data/spec/public/controller/controllers/display.rb +118 -0
  201. data/spec/public/controller/controllers/redirect.rb +30 -0
  202. data/spec/public/controller/controllers/responder.rb +93 -0
  203. data/spec/public/controller/controllers/url.rb +7 -0
  204. data/spec/public/controller/controllers/views/layout/custom.html.erb +1 -0
  205. data/spec/public/controller/controllers/views/layout/custom_arg.html.erb +1 -0
  206. data/spec/public/controller/controllers/views/layout/custom_arg.json.erb +1 -0
  207. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/class_and_local_provides/index.html.erb +1 -0
  208. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/class_and_local_provides/index.xml.erb +1 -0
  209. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/class_provides/index.html.erb +1 -0
  210. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/class_provides/index.xml.erb +1 -0
  211. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/display_with_template/index.html.erb +1 -0
  212. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/display_with_template/no_layout.html.erb +1 -0
  213. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/display_with_template_argument/index.html.erb +1 -0
  214. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/html_default/index.html.erb +1 -0
  215. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/layout/custom.html.erb +1 -0
  216. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/local_provides/index.html.erb +1 -0
  217. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/local_provides/index.xml.erb +1 -0
  218. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/multi_provides/index.html.erb +1 -0
  219. data/spec/public/controller/controllers/views/merb/test/fixtures/controllers/multi_provides/index.js.erb +1 -0
  220. data/spec/public/controller/display_spec.rb +84 -0
  221. data/spec/public/controller/redirect_spec.rb +27 -0
  222. data/spec/public/controller/responder_spec.rb +163 -0
  223. data/spec/public/controller/spec_helper.rb +11 -0
  224. data/spec/public/controller/url_spec.rb +180 -0
  225. data/spec/public/core/merb_core_spec.rb +45 -0
  226. data/spec/public/core_ext/class_spec.rb +91 -0
  227. data/spec/public/core_ext/fixtures/core_ext_dependency.rb +2 -0
  228. data/spec/public/core_ext/kernel_spec.rb +9 -0
  229. data/spec/public/core_ext/spec_helper.rb +1 -0
  230. data/spec/public/directory_structure/directory/app/controllers/application.rb +3 -0
  231. data/spec/public/directory_structure/directory/app/controllers/base.rb +13 -0
  232. data/spec/public/directory_structure/directory/app/controllers/custom.rb +19 -0
  233. data/spec/public/directory_structure/directory/app/views/base/template.html.erb +1 -0
  234. data/spec/public/directory_structure/directory/app/views/wonderful/template.erb +1 -0
  235. data/spec/public/directory_structure/directory/config/router.rb +3 -0
  236. data/spec/public/directory_structure/directory/log/merb_test.log +562 -0
  237. data/spec/public/directory_structure/directory_spec.rb +44 -0
  238. data/spec/public/logger/logger_spec.rb +181 -0
  239. data/spec/public/logger/spec_helper.rb +1 -0
  240. data/spec/public/reloading/directory/app/controllers/application.rb +3 -0
  241. data/spec/public/reloading/directory/app/controllers/reload.rb +6 -0
  242. data/spec/public/reloading/directory/config/init.rb +2 -0
  243. data/spec/public/reloading/directory/log/merb_test.log +138 -0
  244. data/spec/public/reloading/reload_spec.rb +103 -0
  245. data/spec/public/request/multipart_spec.rb +41 -0
  246. data/spec/public/request/request_spec.rb +228 -0
  247. data/spec/public/router/default_spec.rb +21 -0
  248. data/spec/public/router/deferred_spec.rb +22 -0
  249. data/spec/public/router/fixation_spec.rb +27 -0
  250. data/spec/public/router/fixture/log/merb_test.log +1556 -0
  251. data/spec/public/router/namespace_spec.rb +113 -0
  252. data/spec/public/router/nested_matches_spec.rb +97 -0
  253. data/spec/public/router/nested_resources_spec.rb +41 -0
  254. data/spec/public/router/resource_spec.rb +37 -0
  255. data/spec/public/router/resources_spec.rb +82 -0
  256. data/spec/public/router/spec_helper.rb +90 -0
  257. data/spec/public/router/special_spec.rb +61 -0
  258. data/spec/public/router/string_spec.rb +61 -0
  259. data/spec/public/template/template_spec.rb +104 -0
  260. data/spec/public/template/templates/error.html.erb +2 -0
  261. data/spec/public/template/templates/template.html.erb +1 -0
  262. data/spec/public/template/templates/template.html.myt +1 -0
  263. data/spec/public/test/controller_matchers_spec.rb +402 -0
  264. data/spec/public/test/controllers/controller_assertion_mock.rb +7 -0
  265. data/spec/public/test/controllers/dispatch_controller.rb +11 -0
  266. data/spec/public/test/controllers/spec_helper_controller.rb +38 -0
  267. data/spec/public/test/multipart_request_helper_spec.rb +159 -0
  268. data/spec/public/test/multipart_upload_text_file.txt +1 -0
  269. data/spec/public/test/request_helper_spec.rb +221 -0
  270. data/spec/public/test/route_helper_spec.rb +71 -0
  271. data/spec/public/test/route_matchers_spec.rb +162 -0
  272. data/spec/public/test/view_helper_spec.rb +96 -0
  273. data/spec/public/test/view_matchers_spec.rb +183 -0
  274. data/spec/spec_helper.rb +68 -0
  275. metadata +493 -41
  276. data/README.txt +0 -3
  277. data/lib/extlib.rb +0 -32
  278. data/lib/extlib/assertions.rb +0 -8
  279. data/lib/extlib/blank.rb +0 -42
  280. data/lib/extlib/class.rb +0 -175
  281. data/lib/extlib/hash.rb +0 -410
  282. data/lib/extlib/hook.rb +0 -366
  283. data/lib/extlib/inflection.rb +0 -141
  284. data/lib/extlib/lazy_array.rb +0 -106
  285. data/lib/extlib/mash.rb +0 -143
  286. data/lib/extlib/module.rb +0 -37
  287. data/lib/extlib/object.rb +0 -165
  288. data/lib/extlib/object_space.rb +0 -13
  289. data/lib/extlib/pathname.rb +0 -5
  290. data/lib/extlib/pooling.rb +0 -233
  291. data/lib/extlib/rubygems.rb +0 -38
  292. data/lib/extlib/simple_set.rb +0 -39
  293. data/lib/extlib/string.rb +0 -132
  294. data/lib/extlib/struct.rb +0 -8
  295. data/lib/extlib/tasks/release.rb +0 -9
  296. data/lib/extlib/time.rb +0 -12
  297. data/lib/extlib/version.rb +0 -3
  298. data/lib/extlib/virtual_file.rb +0 -10
@@ -0,0 +1,472 @@
1
+ require 'enumerator'
2
+ require 'merb-core/controller/mime'
3
+ require "merb-core/vendor/facets/dictionary"
4
+
5
+ module Merb
6
+ # The ResponderMixin adds methods that help you manage what
7
+ # formats your controllers have available, determine what format(s)
8
+ # the client requested and is capable of handling, and perform
9
+ # content negotiation to pick the proper content format to
10
+ # deliver.
11
+ #
12
+ # If you hear someone say "Use provides" they're talking about the
13
+ # Responder. If you hear someone ask "What happened to respond_to?"
14
+ # it was replaced by provides and the other Responder methods.
15
+ #
16
+ # == A simple example
17
+ #
18
+ # The best way to understand how all of these pieces fit together is
19
+ # with an example. Here's a simple web-service ready resource that
20
+ # provides a list of all the widgets we know about. The widget list is
21
+ # available in 3 formats: :html (the default), plus :xml and :text.
22
+ #
23
+ # class Widgets < Application
24
+ # provides :html # This is the default, but you can
25
+ # # be explicit if you like.
26
+ # provides :xml, :text
27
+ #
28
+ # def index
29
+ # @widgets = Widget.fetch
30
+ # render @widgets
31
+ # end
32
+ # end
33
+ #
34
+ # Let's look at some example requests for this list of widgets. We'll
35
+ # assume they're all GET requests, but that's only to make the examples
36
+ # easier; this works for the full set of RESTful methods.
37
+ #
38
+ # 1. The simplest case, /widgets.html
39
+ # Since the request includes a specific format (.html) we know
40
+ # what format to return. Since :html is in our list of provided
41
+ # formats, that's what we'll return. +render+ will look
42
+ # for an index.html.erb (or another template format
43
+ # like index.html.mab; see the documentation on Template engines)
44
+ #
45
+ # 2. Almost as simple, /widgets.xml
46
+ # This is very similar. They want :xml, we have :xml, so
47
+ # that's what they get. If +render+ doesn't find an
48
+ # index.xml.builder or similar template, it will call +to_xml+
49
+ # on @widgets. This may or may not do something useful, but you can
50
+ # see how it works.
51
+ #
52
+ # 3. A browser request for /widgets
53
+ # This time the URL doesn't say what format is being requested, so
54
+ # we'll look to the HTTP Accept: header. If it's '*/*' (anything),
55
+ # we'll use the first format on our list, :html by default.
56
+ #
57
+ # If it parses to a list of accepted formats, we'll look through
58
+ # them, in order, until we find one we have available. If we find
59
+ # one, we'll use that. Otherwise, we can't fulfill the request:
60
+ # they asked for a format we don't have. So we raise
61
+ # 406: Not Acceptable.
62
+ #
63
+ # == A more complex example
64
+ #
65
+ # Sometimes you don't have the same code to handle each available
66
+ # format. Sometimes you need to load different data to serve
67
+ # /widgets.xml versus /widgets.txt. In that case, you can use
68
+ # +content_type+ to determine what format will be delivered.
69
+ #
70
+ # class Widgets < Application
71
+ # def action1
72
+ # if content_type == :text
73
+ # Widget.load_text_formatted(params[:id])
74
+ # else
75
+ # render
76
+ # end
77
+ # end
78
+ #
79
+ # def action2
80
+ # case content_type
81
+ # when :html
82
+ # handle_html()
83
+ # when :xml
84
+ # handle_xml()
85
+ # when :text
86
+ # handle_text()
87
+ # else
88
+ # render
89
+ # end
90
+ # end
91
+ # end
92
+ #
93
+ # You can do any standard Ruby flow control using +content_type+. If
94
+ # you don't call it yourself, it will be called (triggering content
95
+ # negotiation) by +render+.
96
+ #
97
+ # Once +content_type+ has been called, the output format is frozen,
98
+ # and none of the provides methods can be used.
99
+ module ResponderMixin
100
+
101
+ TYPES = Dictionary.new
102
+ MIMES = {}
103
+
104
+ class ContentTypeAlreadySet < StandardError; end
105
+
106
+ # ==== Parameters
107
+ # base<Module>:: The module that ResponderMixin was mixed into
108
+ def self.included(base)
109
+ base.extend(ClassMethods)
110
+ base.class_eval do
111
+ class_inheritable_accessor :class_provided_formats
112
+ self.class_provided_formats = []
113
+ end
114
+ base.reset_provides
115
+ end
116
+
117
+ module ClassMethods
118
+
119
+ # Adds symbols representing formats to the controller's default list of
120
+ # provided_formats. These will apply to every action in the controller,
121
+ # unless modified in the action. If the last argument is a Hash or an
122
+ # Array, these are regarded as arguments to pass to the to_<mime_type>
123
+ # method as needed.
124
+ #
125
+ # ==== Parameters
126
+ # *formats<Symbol>::
127
+ # A list of mime-types that the controller should provide.
128
+ #
129
+ # ==== Returns
130
+ # Array[Symbol]:: List of formats passed in.
131
+ #
132
+ # ==== Examples
133
+ # provides :html, :xml
134
+ #---
135
+ # @public
136
+ def provides(*formats)
137
+ self.class_provided_formats |= formats
138
+ end
139
+
140
+ # This class should only provide the formats listed here, despite any
141
+ # other definitions previously or in superclasses.
142
+ #
143
+ # ==== Parameters
144
+ # *formats<Symbol>:: Registered mime-types.
145
+ #
146
+ # ==== Returns
147
+ # Array[Symbol]:: List of formats passed in.
148
+ #
149
+ #---
150
+ # @public
151
+ def only_provides(*formats)
152
+ clear_provides
153
+ provides(*formats)
154
+ end
155
+
156
+ # This class should not provide any of this list of formats, despite any.
157
+ # other definitions previously or in superclasses.
158
+ #
159
+ # ==== Parameters
160
+ # *formats<Symbol>:: Registered mime-types.
161
+ #
162
+ # ==== Returns
163
+ # Array[Symbol]::
164
+ # List of formats that remain after removing the ones not to provide.
165
+ #
166
+ #---
167
+ # @public
168
+ def does_not_provide(*formats)
169
+ self.class_provided_formats -= formats
170
+ end
171
+
172
+ # Clear the list of provides.
173
+ #
174
+ # ==== Returns
175
+ # Array:: An empty Array.
176
+ def clear_provides
177
+ self.class_provided_formats.clear
178
+ end
179
+
180
+ # Reset the list of provides to include only :html.
181
+ #
182
+ # ==== Returns
183
+ # Array[Symbol]:: [:html].
184
+ def reset_provides
185
+ only_provides(:html)
186
+ end
187
+ end
188
+
189
+ # ==== Returns
190
+ # Array[Symbol]::
191
+ # The current list of formats provided for this instance of the
192
+ # controller. It starts with what has been set in the controller (or
193
+ # :html by default) but can be modifed on a per-action basis.
194
+ def _provided_formats
195
+ @_provided_formats ||= class_provided_formats.dup
196
+ end
197
+
198
+ # Adds formats to the list of provided formats for this particular request.
199
+ # Usually used to add formats to a single action. See also the
200
+ # controller-level provides that affects all actions in a controller.
201
+ #
202
+ # ==== Parameters
203
+ # *formats<Symbol>::
204
+ # A list of formats to add to the per-action list of provided formats.
205
+ #
206
+ # ==== Raises
207
+ # Merb::ResponderMixin::ContentTypeAlreadySet::
208
+ # Content negotiation already occured, and the content_type is set.
209
+ #
210
+ # ==== Returns
211
+ # Array[Symbol]:: List of formats passed in.
212
+ #
213
+ #---
214
+ # @public
215
+ def provides(*formats)
216
+ if @_content_type
217
+ raise ContentTypeAlreadySet, "Cannot modify provided_formats because content_type has already been set"
218
+ end
219
+ @_provided_formats = self._provided_formats | formats # merges with class_provided_formats if not already
220
+ end
221
+
222
+ # Sets list of provided formats for this particular request. Usually used
223
+ # to limit formats to a single action. See also the controller-level
224
+ # only_provides that affects all actions in a controller.
225
+ #
226
+ # ==== Parameters
227
+ # *formats<Symbol>::
228
+ # A list of formats to use as the per-action list of provided formats.
229
+ #
230
+ # ==== Returns
231
+ # Array[Symbol]:: List of formats passed in.
232
+ #
233
+ #---
234
+ # @public
235
+ def only_provides(*formats)
236
+ @_provided_formats = []
237
+ provides(*formats)
238
+ end
239
+
240
+ # Removes formats from the list of provided formats for this particular
241
+ # request. Usually used to remove formats from a single action. See
242
+ # also the controller-level does_not_provide that affects all actions in a
243
+ # controller.
244
+ #
245
+ # ==== Parameters
246
+ # *formats<Symbol>:: Registered mime-type
247
+ #
248
+ # ==== Returns
249
+ # Array[Symbol]::
250
+ # List of formats that remain after removing the ones not to provide.
251
+ #
252
+ #---
253
+ # @public
254
+ def does_not_provide(*formats)
255
+ @_provided_formats -= formats.flatten
256
+ end
257
+
258
+ # Do the content negotiation:
259
+ # 1. if params[:format] is there, and provided, use it
260
+ # 2. Parse the Accept header
261
+ # 3. If it's */*, use the first provided format
262
+ # 4. Look for one that is provided, in order of request
263
+ # 5. Raise 406 if none found
264
+ def _perform_content_negotiation
265
+ if (fmt = params[:format]) && !fmt.empty?
266
+ accepts = [fmt.to_sym]
267
+ elsif request.accept =~ %r{^(text/html|\*/\*)} && _provided_formats.first == :html
268
+ # Handle the common case of text/html and :html provided after checking :format
269
+ return :html
270
+ else
271
+ accepts = Responder.parse(request.accept).map {|t| t.to_sym}.compact
272
+ end
273
+
274
+ # no need to make a bunch of method calls to _provided_formats
275
+ provided_formats = _provided_formats
276
+
277
+ specifics = accepts & provided_formats
278
+ return specifics.first unless specifics.length == 0
279
+ return provided_formats.first if accepts.include?(:all) && !provided_formats.empty?
280
+
281
+ message = "A format (%s) that isn't provided (%s) has been requested. "
282
+ message += "Make sure the action provides the format, and be "
283
+ message += "careful of before filters which won't recognize "
284
+ message += "formats provided within actions."
285
+ raise Merb::ControllerExceptions::NotAcceptable,
286
+ (message % [accepts.join(', '), provided_formats.join(', ')])
287
+ end
288
+
289
+ # Returns the output format for this request, based on the
290
+ # provided formats, <tt>params[:format]</tt> and the client's HTTP
291
+ # Accept header.
292
+ #
293
+ # The first time this is called, it triggers content negotiation
294
+ # and caches the value. Once you call +content_type+ you can
295
+ # not set or change the list of provided formats.
296
+ #
297
+ # Called automatically by +render+, so you should only call it if
298
+ # you need the value, not to trigger content negotiation.
299
+ #
300
+ # ==== Parameters
301
+ # fmt<String>::
302
+ # An optional format to use instead of performing content negotiation.
303
+ # This can be used to pass in the values of opts[:format] from the
304
+ # render function to short-circuit content-negotiation when it's not
305
+ # necessary. This optional parameter should not be considered part
306
+ # of the public API.
307
+ #
308
+ # ==== Returns
309
+ # Symbol:: The content-type that will be used for this controller.
310
+ #
311
+ #---
312
+ # @public
313
+ def content_type(fmt = nil)
314
+ self.content_type = (fmt || _perform_content_negotiation) unless @_content_type
315
+ @_content_type
316
+ end
317
+
318
+ # Sets the content type of the current response to a value based on
319
+ # a passed in key. The Content-Type header will be set to the first
320
+ # registered header for the mime-type.
321
+ #
322
+ # ==== Parameters
323
+ # type<Symbol>:: The content type.
324
+ #
325
+ # ==== Raises
326
+ # ArgumentError:: type is not in the list of registered mime-types.
327
+ #
328
+ # ==== Returns
329
+ # Symbol:: The content-type that was passed in.
330
+ #
331
+ #---
332
+ # @semipublic
333
+ def content_type=(type)
334
+ unless Merb.available_mime_types.has_key?(type)
335
+ raise Merb::ControllerExceptions::NotAcceptable.new("Unknown content_type for response: #{type}")
336
+ end
337
+
338
+ mime = Merb.available_mime_types[type]
339
+
340
+ headers["Content-Type"] = mime[:content_type]
341
+
342
+ # merge any format specific response headers
343
+ mime[:response_headers].each { |k,v| headers[k] ||= v }
344
+
345
+ # if given, use a block to finetune any runtime headers
346
+ mime[:response_block].call(self) if mime[:response_block]
347
+
348
+ @_content_type = type
349
+ end
350
+
351
+ end
352
+
353
+ class Responder
354
+
355
+ protected
356
+
357
+ # Parses the raw accept header into an array of sorted AcceptType objects.
358
+ #
359
+ # ==== Parameters
360
+ # accept_header<~to_s>:: The raw accept header.
361
+ #
362
+ # ==== Returns
363
+ # Array[AcceptType]:: The accepted types.
364
+ def self.parse(accept_header)
365
+ # FF2 is broken. If we get FF2 headers, use FF3 headers instead.
366
+ if accept_header == "text/xml,application/xml,application/xhtml+xml,text/html;q=0.9,text/plain;q=0.8,image/png,*/*;q=0.5"
367
+ accept_header = "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8"
368
+ end
369
+
370
+ headers = accept_header.split(/,/)
371
+ idx, list = 0, []
372
+ while idx < headers.size
373
+ list << AcceptType.new(headers[idx], idx)
374
+ idx += 1
375
+ end
376
+ list.sort
377
+ end
378
+
379
+ end
380
+
381
+ class AcceptType
382
+
383
+ attr_reader :media_range, :quality, :index, :type, :sub_type
384
+
385
+ # ==== Parameters
386
+ # entry<String>:: The accept type pattern
387
+ # index<Fixnum>::
388
+ # The index used for sorting accept types. A lower value indicates higher
389
+ # priority.
390
+ def initialize(entry,index)
391
+ @index = index
392
+
393
+ entry =~ /\s*([^;\s]*)\s*(;\s*q=\s*(.*))?/
394
+ @media_range, quality = $1, $3
395
+
396
+ @type, @sub_type = @media_range.split(%r{/})
397
+ (quality ||= 0.0) if @media_range == "*/*"
398
+ @quality = quality ? (quality.to_f * 100).to_i : 100
399
+ end
400
+
401
+ # Compares two accept types for sorting purposes.
402
+ #
403
+ # ==== Parameters
404
+ # entry<AcceptType>:: The accept type to compare.
405
+ #
406
+ # ==== Returns
407
+ # Fixnum::
408
+ # -1, 0 or 1, depending on whether entry has a lower, equal or higher
409
+ # priority than the accept type being compared.
410
+ def <=>(entry)
411
+ if entry.quality == quality
412
+ @index <=> entry.index
413
+ else
414
+ entry.quality <=> @quality
415
+ end
416
+ end
417
+
418
+
419
+ # ==== Parameters
420
+ # entry<AcceptType>:: The accept type to compare.
421
+ #
422
+ # ==== Returns
423
+ # Boolean::
424
+ # True if the accept types are equal, i.e. if the synonyms for this
425
+ # accept type includes the entry media range.
426
+ def eql?(entry)
427
+ synonyms.include?(entry.media_range)
428
+ end
429
+
430
+ # An alias for eql?.
431
+ def ==(entry); eql?(entry); end
432
+
433
+ # ==== Returns
434
+ # Fixnum:: A hash based on the super range.
435
+ def hash; super_range.hash; end
436
+
437
+ # ==== Returns
438
+ # Array[String]::
439
+ # All Accept header values, such as "text/html", that match this type.
440
+ def synonyms
441
+ return @syns if @syns
442
+ if mime = Merb.available_mime_types[Merb::ResponderMixin::MIMES[@media_range]]
443
+ @syns = mime[:accepts]
444
+ else
445
+ @syns = []
446
+ end
447
+ end
448
+
449
+ # ==== Returns
450
+ # String::
451
+ # The primary media range for this accept type, i.e. either the first
452
+ # synonym or, if none exist, the media range.
453
+ def super_range
454
+ synonyms.first || @media_range
455
+ end
456
+
457
+ # ==== Returns
458
+ # Symbol: The type as a symbol, e.g. :html.
459
+ def to_sym
460
+ Merb.available_mime_types.select{|k,v|
461
+ v[:accepts] == synonyms || v[:accepts][0] == synonyms[0]}.flatten.first
462
+ end
463
+
464
+ # ==== Returns
465
+ # String:: The accept type as a string, i.e. the media range.
466
+ def to_s
467
+ @media_range
468
+ end
469
+
470
+ end
471
+
472
+ end