jeffrafter-resource_controller 0.5.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (233) hide show
  1. data/README.rdoc +314 -0
  2. data/VERSION.yml +4 -0
  3. data/generators/scaffold_resource/USAGE +29 -0
  4. data/generators/scaffold_resource/scaffold_resource_generator.rb +179 -0
  5. data/generators/scaffold_resource/templates/controller.rb +2 -0
  6. data/generators/scaffold_resource/templates/fixtures.yml +10 -0
  7. data/generators/scaffold_resource/templates/functional_test.rb +57 -0
  8. data/generators/scaffold_resource/templates/helper.rb +2 -0
  9. data/generators/scaffold_resource/templates/migration.rb +15 -0
  10. data/generators/scaffold_resource/templates/model.rb +2 -0
  11. data/generators/scaffold_resource/templates/old_migration.rb +13 -0
  12. data/generators/scaffold_resource/templates/rspec/functional_spec.rb +255 -0
  13. data/generators/scaffold_resource/templates/rspec/helper_spec.rb +11 -0
  14. data/generators/scaffold_resource/templates/rspec/routing_spec.rb +61 -0
  15. data/generators/scaffold_resource/templates/rspec/unit_spec.rb +11 -0
  16. data/generators/scaffold_resource/templates/rspec/views/edit_spec.rb +28 -0
  17. data/generators/scaffold_resource/templates/rspec/views/index_spec.rb +26 -0
  18. data/generators/scaffold_resource/templates/rspec/views/new_spec.rb +30 -0
  19. data/generators/scaffold_resource/templates/rspec/views/show_spec.rb +25 -0
  20. data/generators/scaffold_resource/templates/shoulda_functional_test.rb +19 -0
  21. data/generators/scaffold_resource/templates/unit_test.rb +7 -0
  22. data/generators/scaffold_resource/templates/view__form.erb +6 -0
  23. data/generators/scaffold_resource/templates/view__form.haml +5 -0
  24. data/generators/scaffold_resource/templates/view_edit.erb +16 -0
  25. data/generators/scaffold_resource/templates/view_edit.haml +11 -0
  26. data/generators/scaffold_resource/templates/view_index.erb +22 -0
  27. data/generators/scaffold_resource/templates/view_index.haml +19 -0
  28. data/generators/scaffold_resource/templates/view_new.erb +12 -0
  29. data/generators/scaffold_resource/templates/view_new.haml +9 -0
  30. data/generators/scaffold_resource/templates/view_show.erb +9 -0
  31. data/generators/scaffold_resource/templates/view_show.haml +9 -0
  32. data/lib/resource_controller.rb +26 -0
  33. data/lib/resource_controller/accessors.rb +77 -0
  34. data/lib/resource_controller/action_options.rb +40 -0
  35. data/lib/resource_controller/actions.rb +73 -0
  36. data/lib/resource_controller/base.rb +15 -0
  37. data/lib/resource_controller/class_methods.rb +65 -0
  38. data/lib/resource_controller/controller.rb +69 -0
  39. data/lib/resource_controller/failable_action_options.rb +25 -0
  40. data/lib/resource_controller/helpers.rb +28 -0
  41. data/lib/resource_controller/helpers/current_objects.rb +73 -0
  42. data/lib/resource_controller/helpers/internal.rb +80 -0
  43. data/lib/resource_controller/helpers/nested.rb +67 -0
  44. data/lib/resource_controller/helpers/singleton_customizations.rb +64 -0
  45. data/lib/resource_controller/helpers/urls.rb +132 -0
  46. data/lib/resource_controller/response_collector.rb +27 -0
  47. data/lib/resource_controller/singleton.rb +15 -0
  48. data/lib/urligence.rb +50 -0
  49. data/rails/init.rb +6 -0
  50. data/test/Rakefile +10 -0
  51. data/test/app/controllers/accounts_controller.rb +6 -0
  52. data/test/app/controllers/application.rb +7 -0
  53. data/test/app/controllers/cms/options_controller.rb +3 -0
  54. data/test/app/controllers/cms/personnel_controller.rb +2 -0
  55. data/test/app/controllers/cms/photos_controller.rb +6 -0
  56. data/test/app/controllers/cms/products_controller.rb +3 -0
  57. data/test/app/controllers/comments_controller.rb +3 -0
  58. data/test/app/controllers/images_controller.rb +4 -0
  59. data/test/app/controllers/options_controller.rb +8 -0
  60. data/test/app/controllers/people_controller.rb +9 -0
  61. data/test/app/controllers/photos_controller.rb +12 -0
  62. data/test/app/controllers/posts_controller.rb +10 -0
  63. data/test/app/controllers/projects_controller.rb +3 -0
  64. data/test/app/controllers/somethings_controller.rb +3 -0
  65. data/test/app/controllers/tags_controller.rb +13 -0
  66. data/test/app/controllers/users_controller.rb +12 -0
  67. data/test/app/helpers/accounts_helper.rb +2 -0
  68. data/test/app/helpers/application_helper.rb +3 -0
  69. data/test/app/helpers/cms/products_helper.rb +2 -0
  70. data/test/app/helpers/comments_helper.rb +2 -0
  71. data/test/app/helpers/images_helper.rb +2 -0
  72. data/test/app/helpers/options_helper.rb +2 -0
  73. data/test/app/helpers/people_helper.rb +2 -0
  74. data/test/app/helpers/photos_helper.rb +2 -0
  75. data/test/app/helpers/posts_helper.rb +2 -0
  76. data/test/app/helpers/projects_helper.rb +2 -0
  77. data/test/app/helpers/somethings_helper.rb +2 -0
  78. data/test/app/helpers/tags_helper.rb +2 -0
  79. data/test/app/helpers/users_helper.rb +2 -0
  80. data/test/app/models/account.rb +4 -0
  81. data/test/app/models/comment.rb +3 -0
  82. data/test/app/models/image.rb +3 -0
  83. data/test/app/models/option.rb +3 -0
  84. data/test/app/models/personnel.rb +3 -0
  85. data/test/app/models/photo.rb +5 -0
  86. data/test/app/models/post.rb +3 -0
  87. data/test/app/models/product.rb +3 -0
  88. data/test/app/models/project.rb +2 -0
  89. data/test/app/models/something.rb +2 -0
  90. data/test/app/models/tag.rb +3 -0
  91. data/test/app/models/user.rb +3 -0
  92. data/test/app/views/accounts/_form.html.erb +4 -0
  93. data/test/app/views/accounts/edit.html.erb +14 -0
  94. data/test/app/views/accounts/new.html.erb +12 -0
  95. data/test/app/views/accounts/show.html.erb +5 -0
  96. data/test/app/views/cms/options/edit.rhtml +17 -0
  97. data/test/app/views/cms/options/index.rhtml +20 -0
  98. data/test/app/views/cms/options/new.rhtml +16 -0
  99. data/test/app/views/cms/options/show.rhtml +8 -0
  100. data/test/app/views/cms/photos/edit.rhtml +17 -0
  101. data/test/app/views/cms/photos/index.rhtml +20 -0
  102. data/test/app/views/cms/photos/new.rhtml +16 -0
  103. data/test/app/views/cms/photos/show.rhtml +8 -0
  104. data/test/app/views/cms/products/edit.rhtml +17 -0
  105. data/test/app/views/cms/products/index.rhtml +20 -0
  106. data/test/app/views/cms/products/new.rhtml +16 -0
  107. data/test/app/views/cms/products/show.rhtml +8 -0
  108. data/test/app/views/comments/edit.rhtml +27 -0
  109. data/test/app/views/comments/index.rhtml +24 -0
  110. data/test/app/views/comments/new.rhtml +26 -0
  111. data/test/app/views/comments/show.rhtml +18 -0
  112. data/test/app/views/images/_form.html.erb +4 -0
  113. data/test/app/views/images/edit.html.erb +14 -0
  114. data/test/app/views/images/new.html.erb +12 -0
  115. data/test/app/views/layouts/application.rhtml +17 -0
  116. data/test/app/views/layouts/comments.rhtml +17 -0
  117. data/test/app/views/layouts/options.rhtml +17 -0
  118. data/test/app/views/layouts/people.rhtml +17 -0
  119. data/test/app/views/layouts/photos.rhtml +17 -0
  120. data/test/app/views/layouts/projects.rhtml +17 -0
  121. data/test/app/views/layouts/somethings.rhtml +17 -0
  122. data/test/app/views/layouts/tags.rhtml +17 -0
  123. data/test/app/views/options/_form.html.erb +8 -0
  124. data/test/app/views/options/edit.html.erb +16 -0
  125. data/test/app/views/options/index.html.erb +21 -0
  126. data/test/app/views/options/new.html.erb +12 -0
  127. data/test/app/views/options/show.html.erb +10 -0
  128. data/test/app/views/people/edit.rhtml +17 -0
  129. data/test/app/views/people/index.rhtml +20 -0
  130. data/test/app/views/people/new.rhtml +16 -0
  131. data/test/app/views/people/show.rhtml +8 -0
  132. data/test/app/views/photos/edit.rhtml +17 -0
  133. data/test/app/views/photos/index.rhtml +20 -0
  134. data/test/app/views/photos/new.rhtml +16 -0
  135. data/test/app/views/photos/show.rhtml +8 -0
  136. data/test/app/views/posts/edit.rhtml +22 -0
  137. data/test/app/views/posts/index.rhtml +22 -0
  138. data/test/app/views/posts/new.rhtml +21 -0
  139. data/test/app/views/posts/show.rhtml +13 -0
  140. data/test/app/views/projects/edit.rhtml +17 -0
  141. data/test/app/views/projects/index.rhtml +20 -0
  142. data/test/app/views/projects/new.rhtml +16 -0
  143. data/test/app/views/projects/show.rhtml +8 -0
  144. data/test/app/views/somethings/edit.rhtml +17 -0
  145. data/test/app/views/somethings/index.rhtml +20 -0
  146. data/test/app/views/somethings/new.rhtml +16 -0
  147. data/test/app/views/somethings/show.rhtml +8 -0
  148. data/test/app/views/tags/edit.rhtml +17 -0
  149. data/test/app/views/tags/index.rhtml +20 -0
  150. data/test/app/views/tags/index.rjs +0 -0
  151. data/test/app/views/tags/new.rhtml +16 -0
  152. data/test/app/views/tags/show.rhtml +8 -0
  153. data/test/app/views/users/edit.rhtml +17 -0
  154. data/test/app/views/users/index.rhtml +20 -0
  155. data/test/app/views/users/new.rhtml +16 -0
  156. data/test/app/views/users/show.rhtml +8 -0
  157. data/test/config/boot.rb +109 -0
  158. data/test/config/database.yml +9 -0
  159. data/test/config/environment.rb +49 -0
  160. data/test/config/environments/development.rb +20 -0
  161. data/test/config/environments/test.rb +19 -0
  162. data/test/config/initializers/inflections.rb +14 -0
  163. data/test/config/routes.rb +61 -0
  164. data/test/db/migrate/001_create_posts.rb +12 -0
  165. data/test/db/migrate/002_create_products.rb +11 -0
  166. data/test/db/migrate/003_create_comments.rb +13 -0
  167. data/test/db/migrate/004_create_options.rb +13 -0
  168. data/test/db/migrate/005_create_photos.rb +11 -0
  169. data/test/db/migrate/006_create_tags.rb +17 -0
  170. data/test/db/migrate/007_create_somethings.rb +11 -0
  171. data/test/db/migrate/008_create_accounts.rb +11 -0
  172. data/test/db/migrate/009_add_account_id_to_photos.rb +9 -0
  173. data/test/db/migrate/010_create_projects.rb +11 -0
  174. data/test/db/migrate/011_create_images.rb +12 -0
  175. data/test/db/migrate/012_create_users.rb +11 -0
  176. data/test/db/migrate/013_create_personnel.rb +11 -0
  177. data/test/db/migrate/014_add_personnel_id_to_photos.rb +9 -0
  178. data/test/db/schema.rb +78 -0
  179. data/test/script/console +3 -0
  180. data/test/script/destroy +3 -0
  181. data/test/script/generate +3 -0
  182. data/test/script/server +3 -0
  183. data/test/test/fixtures/accounts.yml +7 -0
  184. data/test/test/fixtures/comments.yml +11 -0
  185. data/test/test/fixtures/images.yml +6 -0
  186. data/test/test/fixtures/options.yml +9 -0
  187. data/test/test/fixtures/personnel.yml +5 -0
  188. data/test/test/fixtures/photos.yml +9 -0
  189. data/test/test/fixtures/photos_tags.yml +3 -0
  190. data/test/test/fixtures/posts.yml +9 -0
  191. data/test/test/fixtures/products.yml +7 -0
  192. data/test/test/fixtures/projects.yml +7 -0
  193. data/test/test/fixtures/somethings.yml +7 -0
  194. data/test/test/fixtures/tags.yml +7 -0
  195. data/test/test/fixtures/users.yml +5 -0
  196. data/test/test/functional/cms/options_controller_test.rb +23 -0
  197. data/test/test/functional/cms/photos_controller_test.rb +43 -0
  198. data/test/test/functional/cms/products_controller_test.rb +23 -0
  199. data/test/test/functional/comments_controller_test.rb +26 -0
  200. data/test/test/functional/images_controller_test.rb +37 -0
  201. data/test/test/functional/people_controller_test.rb +34 -0
  202. data/test/test/functional/photos_controller_test.rb +130 -0
  203. data/test/test/functional/posts_controller_test.rb +34 -0
  204. data/test/test/functional/projects_controller_test.rb +18 -0
  205. data/test/test/functional/somethings_controller_test.rb +28 -0
  206. data/test/test/functional/tags_controller_test.rb +64 -0
  207. data/test/test/functional/users_controller_test.rb +24 -0
  208. data/test/test/test_helper.rb +12 -0
  209. data/test/test/unit/accessors_test.rb +110 -0
  210. data/test/test/unit/account_test.rb +7 -0
  211. data/test/test/unit/action_options_test.rb +109 -0
  212. data/test/test/unit/base_test.rb +11 -0
  213. data/test/test/unit/comment_test.rb +10 -0
  214. data/test/test/unit/failable_action_options_test.rb +77 -0
  215. data/test/test/unit/helpers/current_objects_test.rb +133 -0
  216. data/test/test/unit/helpers/internal_test.rb +106 -0
  217. data/test/test/unit/helpers/nested_test.rb +86 -0
  218. data/test/test/unit/helpers/singleton_current_objects_test.rb +68 -0
  219. data/test/test/unit/helpers/singleton_nested_test.rb +77 -0
  220. data/test/test/unit/helpers/singleton_urls_test.rb +67 -0
  221. data/test/test/unit/helpers/urls_test.rb +75 -0
  222. data/test/test/unit/helpers_test.rb +25 -0
  223. data/test/test/unit/image_test.rb +7 -0
  224. data/test/test/unit/option_test.rb +10 -0
  225. data/test/test/unit/photo_test.rb +10 -0
  226. data/test/test/unit/post_test.rb +10 -0
  227. data/test/test/unit/project_test.rb +10 -0
  228. data/test/test/unit/response_collector_test.rb +49 -0
  229. data/test/test/unit/something_test.rb +10 -0
  230. data/test/test/unit/tag_test.rb +10 -0
  231. data/test/test/unit/urligence_test.rb +203 -0
  232. data/test/vendor/gems/resource_controller-10.0.0 +1 -0
  233. metadata +329 -0
data/README.rdoc ADDED
@@ -0,0 +1,314 @@
1
+ = Resource Controller
2
+
3
+ resource_controller makes RESTful controllers easier, more maintainable, and super readable. With the RESTful controller pattern hidden away, you can focus on what makes your controller special.
4
+
5
+ == Get It
6
+
7
+ Install it as a plugin:
8
+
9
+ script/plugin install git://github.com/giraffesoft/resource_controller.git
10
+
11
+ Or grab the source
12
+
13
+ git clone git://github.com/giraffesoft/resource_controller.git
14
+
15
+ = Usage
16
+
17
+ Creating a basic RESTful controller is as easy as...
18
+
19
+ class PostsController < ResourceController::Base
20
+ end
21
+
22
+ ...or if you prefer, you can use the method-call syntax. If you need to inherit from some other class, this syntax is definitely for you:
23
+
24
+ class PostsController < ApplicationController
25
+ resource_controller
26
+ end
27
+
28
+ Both syntaxes are identical in their behavior. Just make sure you call resource_controller before you use any other r_c functionality in your controller.
29
+
30
+
31
+ Nobody just uses the default RESTful controller, though. resource_controller provides a simple API for customizations.
32
+
33
+ == Action Lifecycle
34
+
35
+ It's really easy to make changes to the lifecycle of your actions.
36
+
37
+ Note: We had to call the new accessor "new_action", since new is somewhat reserved in ruby.
38
+
39
+ === Before and After
40
+
41
+ class ProjectsController < ResourceController::Base
42
+
43
+ new_action.before do
44
+ 3.times { object.tasks.build }
45
+ end
46
+
47
+ create.after do
48
+ object.creator = current_user
49
+ end
50
+
51
+ end
52
+
53
+ === Flash
54
+
55
+ class ProjectsController < ResourceController::Base
56
+ create.flash "Can you believe how easy it is to use resource_controller? Neither could I!"
57
+ end
58
+
59
+ === respond_to
60
+
61
+ You can add to what's already there...
62
+
63
+ class ProjectsController < ResourceController::Base
64
+ create.wants.js { render :template => "show.rjs" }
65
+ end
66
+
67
+ Or you can create a whole new block. This syntax destroys everything that's there, and starts again...
68
+
69
+ class ProjectsController < ResourceController::Base
70
+ create.response do |wants|
71
+ wants.html
72
+ wants.js { render :template => "show.rjs" }
73
+ end
74
+ end
75
+
76
+ === Scoping
77
+
78
+ Because sometimes you want to make a bunch of customizations at once, most of the helpers accept blocks that make grouping calls really easy. Is it a DSL? Maybe; maybe not. But, it's definitely awesome.
79
+
80
+ With actions that can fail, the scoping defaults to success. That means that create.flash == create.success.flash.
81
+
82
+ class ProjectsController < ResourceController::Base
83
+
84
+ create do
85
+ flash "Object successfully created!"
86
+ wants.js { render :template => "show.rjs" }
87
+
88
+ failure.wants.js { render :template => "display_errors.rjs" }
89
+ end
90
+
91
+ destroy do
92
+ flash "You destroyed your project. Good work."
93
+
94
+ failure do
95
+ flash "You cannot destroy that project. Stop trying!"
96
+ wants.js { render :template => "display_errors.rjs" }
97
+ end
98
+ end
99
+
100
+ end
101
+
102
+ == Singleton Resource
103
+
104
+ If you want to create a singleton RESTful controller inherit from ResourceController::Singleton.
105
+
106
+ class AccountsController < ResourceController::Singleton
107
+ end
108
+
109
+ *Note:* This type of controllers handle a single resource only so the index action and all the collection helpers (collection_url, collection_path...) are not available for them.
110
+
111
+ Loading objects in singletons is similar to plural controllers with one exception. For non-nested singleton controllers you should override the object method as it defaults to nil for them.
112
+
113
+ class AccountsController < ResourceController::Singleton
114
+ private
115
+ def object
116
+ @object ||= Account.find(session[:account_id])
117
+ end
118
+ end
119
+
120
+ In other cases you can use the default logic and override it only if you use permalinks or anything special.
121
+
122
+ Singleton nesting with both :has_many and :has_one associations is provided...
123
+
124
+ map.resource :account, :has_many => :options # /account/options, account is a singleton parent
125
+ map.resources :users, :has_one => :image # /users/1/image, image is a singleton child
126
+
127
+ If you have the :has_many association with a singleton parent remember to override parent_object for your :has_many controller as it returns nil by default in this case.
128
+
129
+ class OptionsController < ResourceController::Base
130
+ belongs_to :account
131
+
132
+ protected
133
+ def parent_object
134
+ Account.find(session[:account_id])
135
+ end
136
+ end
137
+
138
+ == Helpers (ResourceController::Helpers)
139
+
140
+ === Loading objects
141
+
142
+ You want to add something like pagination to your controller...
143
+
144
+ class PostsController < ResourceController::Base
145
+ private
146
+ def collection
147
+ @collection ||= end_of_association_chain.find(:all, :page => {:size => 10, :current => params[:page]})
148
+ end
149
+ end
150
+
151
+ Or maybe you used a permalink...
152
+
153
+ class PostsController < ResourceController::Base
154
+ private
155
+ def object
156
+ @object ||= end_of_association_chain.find_by_permalink(param)
157
+ end
158
+ end
159
+
160
+ === Building objects
161
+
162
+ Maybe you have some alternative way of building objects...
163
+
164
+ class PostsController < ResourceController::Base
165
+ private
166
+ def build_object
167
+ @object ||= end_of_association_chain.build_my_object_some_funky_way object_params
168
+ end
169
+ end
170
+
171
+ ...and there are tons more helpers in the ResourceController::Helpers
172
+
173
+ == Nested Resources
174
+
175
+ Nested controllers can be a pain, especially if routing is such that you may or may not have a parent. Not so with Resource Controller.
176
+
177
+ class CommentsController < ResourceController::Base
178
+ belongs_to :post
179
+ end
180
+
181
+ All of the finding, and creation, and everything will be done at the scope of the post automatically.
182
+
183
+ == Namespaced Resources
184
+
185
+ ...are handled automatically, and any namespaces are always available, symbolized, in array form @ ResourceController::Helpers#namespaces
186
+
187
+ == Polymorphic Resources
188
+
189
+ Everything, including url generation is handled completely automatically. Take this example...
190
+
191
+ ## comment.rb
192
+ class Comment
193
+ belongs_to :commentable, :polymorphic => true
194
+ end
195
+
196
+ ## comments_controller.rb
197
+ class CommentsController < ResourceController::Base
198
+ belongs_to :post, :product, :user
199
+ end
200
+ *Note:* Your model doesn't have to be polymorphic in the ActiveRecord sense. It can be associated in whichever way you want.
201
+
202
+ ## routes.rb
203
+ map.resources :posts, :has_many => :comments
204
+ map.resources :products, :has_many => :comments
205
+ map.resources :users, :has_many => :comments
206
+
207
+ All you have to do is that, and r_c will infer whichever relationship is present, and perform all the actions at the scope of the parent object.
208
+
209
+ === Parent Helpers
210
+
211
+ You also get some helpers for reflecting on your parent.
212
+
213
+ parent? # => true/false is there a parent present?
214
+ parent_type # => :post
215
+ parent_model # => Post
216
+ parent_object # => @post
217
+
218
+ === Non-standard resource names
219
+
220
+ resource_controller supports overrides for every non-standard configuration of resources.
221
+
222
+ The most common example is where the resource has a different name than the associated model. Simply overriding the model_name helper will get resource_controller working with your model.
223
+
224
+ map.resources :tags
225
+ ...
226
+ class PhotoTag < ActiveRecord::Base
227
+ ...
228
+ class TagsController < ResourceController::Base
229
+ private
230
+ def model_name
231
+ 'photo_tag'
232
+ end
233
+ end
234
+
235
+ In the above example, the variable, and params will be set to @tag, @tags, and params[:tag]. If you'd like to change that, override object_name.
236
+
237
+ def object_name
238
+ 'photo_tag'
239
+ end
240
+
241
+ If you're using a non-standard controller name, but everything else is standard, overriding resource_name will propagate through all of the other helpers.
242
+
243
+ map.resources :tags, :controller => "somethings"
244
+ ...
245
+ class Tag < ActiveRecord::Base
246
+ ...
247
+ class SomethingsController < ResourceController::Base
248
+ private
249
+ def resource_name
250
+ 'tag'
251
+ end
252
+ end
253
+
254
+ Finally, the route_name helper is used by Urligence to determine which url helper to call, so if you have non-standard route names, override it.
255
+
256
+ map.resources :tags, :controller => "taggings"
257
+ ...
258
+ class Taggings < ActiveRecord::Base
259
+ ...
260
+ class TaggingsController < ResourceController::Base
261
+ private
262
+ def route_name
263
+ 'tag'
264
+ end
265
+ end
266
+
267
+ == Url Helpers
268
+
269
+ Thanks to Urligence, you also get some free url helpers.
270
+
271
+ No matter what your controller looks like...
272
+
273
+ [edit_|new_]object_url # is the equivalent of saying [edit_|new_]post_url(@post)
274
+ [edit_|new_]object_url(some_other_object) # allows you to specify an object, but still maintain any paths or namespaces that are present
275
+
276
+ collection_url # is like saying posts_url
277
+
278
+ Url helpers are especially useful when working with polymorphic controllers.
279
+
280
+ # /posts/1/comments
281
+ object_url # => /posts/1/comments/#{@comment.to_param}
282
+ object_url(comment) # => /posts/1/comments/#{comment.to_param}
283
+ edit_object_url # => /posts/1/comments/#{@comment.to_param}/edit
284
+ collection_url # => /posts/1/comments
285
+
286
+ # /products/1/comments
287
+ object_url # => /products/1/comments/#{@comment.to_param}
288
+ object_url(comment) # => /products/1/comments/#{comment.to_param}
289
+ edit_object_url # => /products/1/comments/#{@comment.to_param}/edit
290
+ collection_url # => /products/1/comments
291
+
292
+ # /comments
293
+ object_url # => /comments/#{@comment.to_param}
294
+ object_url(comment) # => /comments/#{comment.to_param}
295
+ edit_object_url # => /comments/#{@comment.to_param}/edit
296
+ collection_url # => /comments
297
+
298
+ Or with namespaced, nested controllers...
299
+
300
+ # /admin/products/1/options
301
+ object_url # => /admin/products/1/options/#{@option.to_param}
302
+ object_url(option) # => /admin/products/1/options/#{option.to_param}
303
+ edit_object_url # => /admin/products/1/options/#{@option.to_param}/edit
304
+ collection_url # => /admin/products/1/options
305
+
306
+ You get the idea. Everything is automagical! All parameters are inferred.
307
+
308
+ == Credits
309
+
310
+ resource_controller was created, and is maintained by {James Golick}[http://jamesgolick.com].
311
+
312
+ == License
313
+
314
+ resource_controller is available under the {MIT License}[http://en.wikipedia.org/wiki/MIT_License]
data/VERSION.yml ADDED
@@ -0,0 +1,4 @@
1
+ ---
2
+ :patch: 5
3
+ :major: 0
4
+ :minor: 5
@@ -0,0 +1,29 @@
1
+ Description:
2
+ The scaffold resource generator creates a model, a controller, and a set of templates that's ready to use as the
3
+ starting point for your REST-like, resource-oriented application. This basically means that it follows a set of
4
+ conventions to exploit the full set of HTTP verbs (GET/POST/PUT/DELETE) and is prepared for multi-client access
5
+ (like one view for HTML, one for an XML API, one for ATOM, etc). Everything comes with sample unit and functional
6
+ tests as well.
7
+
8
+ The generator takes the name of the model as its first argument. This model name is then pluralized to get the
9
+ controller name. So "scaffold_resource post" will generate a Post model and a PostsController and will be intended
10
+ for URLs like /posts and /posts/45.
11
+
12
+ As additional parameters, the generator will take attribute pairs described by name and type. These attributes will
13
+ be used to prepopulate the migration to create the table for the model and to give you a set of templates for the
14
+ view. For example, "scaffold_resource post title:string created_on:date body:text published:boolean" will give
15
+ you a model with those four attributes, forms to create and edit those models from, and an index that'll list them
16
+ all.
17
+
18
+ You don't have to think up all attributes up front, but it's a good idea of adding just the baseline of what's
19
+ needed to start really working with the resource.
20
+
21
+ Once the generator has run, you'll need to add a declaration to your config/routes.rb file to hook up the rules
22
+ that'll point URLs to this new resource. If you create a resource like "scaffold_resource post", you'll need to
23
+ add "map.resources :posts" (notice the plural form) in the routes file. Then your new resource is accessible from
24
+ /posts.
25
+
26
+ Examples:
27
+ ./script/generate scaffold_resource post # no attributes, view will be anemic
28
+ ./script/generate scaffold_resource post title:string created_on:date body:text published:boolean
29
+ ./script/generate scaffold_resource purchase order_id:integer created_at:datetime amount:decimal
@@ -0,0 +1,179 @@
1
+ class ScaffoldResourceGenerator < Rails::Generator::NamedBase
2
+ attr_reader :controller_name,
3
+ :controller_class_path,
4
+ :controller_file_path,
5
+ :controller_class_nesting,
6
+ :controller_class_nesting_depth,
7
+ :controller_class_name,
8
+ :controller_singular_name,
9
+ :controller_plural_name,
10
+ :resource_edit_path,
11
+ :default_file_extension,
12
+ :generator_default_file_extension
13
+ alias_method :controller_file_name, :controller_singular_name
14
+ alias_method :controller_table_name, :controller_plural_name
15
+
16
+ def initialize(runtime_args, runtime_options = {})
17
+ super
18
+
19
+ if @rspec = has_rspec?
20
+ if ActionController::Base.respond_to?(:resource_action_separator)
21
+ @resource_edit_path = "/edit"
22
+ else
23
+ @resource_edit_path = ";edit"
24
+ end
25
+ end
26
+
27
+ @generator_default_file_extension = (defined? Haml )? "haml" : "erb"
28
+
29
+ # we want to call erb templates .rhtml or .haml if this is rails 1
30
+ if RAILS_GEM_VERSION.to_i == 1
31
+ @default_file_extension = @generator_default_file_extension == 'erb' ? 'rhtml' : @generator_default_file_extension
32
+ else
33
+ @default_file_extension = "html.#{@generator_default_file_extension}"
34
+ end
35
+
36
+ @controller_name = @name.pluralize
37
+
38
+ base_name, @controller_class_path, @controller_file_path, @controller_class_nesting, @controller_class_nesting_depth = extract_modules(@controller_name)
39
+ @controller_class_name_without_nesting, @controller_singular_name, @controller_plural_name = inflect_names(base_name)
40
+
41
+ if @controller_class_nesting.empty?
42
+ @controller_class_name = @controller_class_name_without_nesting
43
+ else
44
+ @controller_class_name = "#{@controller_class_nesting}::#{@controller_class_name_without_nesting}"
45
+ end
46
+ end
47
+
48
+ def manifest
49
+ record do |m|
50
+ # Check for class naming collisions.
51
+ m.class_collisions(controller_class_path, "#{controller_class_name}Controller", "#{controller_class_name}Helper")
52
+ m.class_collisions(class_path, "#{class_name}")
53
+
54
+ # Controller, helper, views, and test directories.
55
+ m.directory(File.join('app/models', class_path))
56
+ m.directory(File.join('app/controllers', controller_class_path))
57
+ m.directory(File.join('app/helpers', controller_class_path))
58
+ m.directory(File.join('app/views', controller_class_path, controller_file_name))
59
+
60
+ if @rspec
61
+ m.directory(File.join('spec/controllers', controller_class_path))
62
+ m.directory(File.join('spec/helpers', class_path))
63
+ m.directory(File.join('spec/models', class_path))
64
+ m.directory File.join('spec/views', controller_class_path, controller_file_name)
65
+ m.directory(File.join('spec/fixtures', class_path))
66
+ else
67
+ m.directory(File.join('test/functional', controller_class_path))
68
+ m.directory(File.join('test/unit', class_path))
69
+ end
70
+
71
+ scaffold_views.each do |action|
72
+ m.template(
73
+ "view_#{action}.#{generator_default_file_extension}",
74
+ File.join('app/views', controller_class_path, controller_file_name, "#{action}.#{default_file_extension}")
75
+ )
76
+ end
77
+
78
+ m.template('model.rb', File.join('app/models', class_path, "#{file_name}.rb"))
79
+ m.template('controller.rb', File.join('app/controllers', controller_class_path, "#{controller_file_name}_controller.rb"))
80
+ m.template('helper.rb', File.join('app/helpers', controller_class_path, "#{controller_file_name}_helper.rb"))
81
+
82
+ if @rspec
83
+ m.template('rspec/functional_spec.rb', File.join('spec/controllers', controller_class_path, "#{controller_file_name}_controller_spec.rb"))
84
+ m.template('rspec/routing_spec.rb', File.join('spec/controllers', controller_class_path, "#{controller_file_name}_routing_spec.rb"))
85
+ m.template('rspec/helper_spec.rb', File.join('spec/helpers', class_path, "#{controller_file_name}_helper_spec.rb"))
86
+ m.template('rspec/unit_spec.rb', File.join('spec/models', class_path, "#{file_name}_spec.rb"))
87
+ m.template('fixtures.yml', File.join('spec/fixtures', "#{table_name}.yml"))
88
+
89
+ rspec_views.each do |action|
90
+ m.template(
91
+ "rspec/views/#{action}_spec.rb",
92
+ File.join('spec/views', controller_class_path, controller_file_name, "#{action}_spec.rb")
93
+ )
94
+ end
95
+
96
+ else
97
+ functional_test = (defined? ThoughtBot::Shoulda) ? "shoulda_functional_test.rb" : "functional_test.rb"
98
+
99
+ m.template("#{functional_test}", File.join('test/functional', controller_class_path, "#{controller_file_name}_controller_test.rb"))
100
+ m.template('unit_test.rb', File.join('test/unit', class_path, "#{file_name}_test.rb"))
101
+ m.template('fixtures.yml', File.join('test/fixtures', "#{table_name}.yml"))
102
+ end
103
+
104
+
105
+ unless options[:skip_migration]
106
+ migration_template = RAILS_GEM_VERSION.to_i == 1 ? 'old_migration.rb' : 'migration.rb'
107
+
108
+ m.migration_template(
109
+ migration_template, 'db/migrate',
110
+ :assigns => {
111
+ :migration_name => "Create#{class_name.pluralize.gsub(/::/, '')}",
112
+ :attributes => attributes
113
+ },
114
+ :migration_file_name => "create_#{file_path.gsub(/\//, '_').pluralize}"
115
+ )
116
+ end
117
+
118
+ m.route_resources controller_file_name
119
+ end
120
+ end
121
+
122
+ # Lifted from Rick Olson's restful_authentication
123
+ def has_rspec?
124
+ options[:rspec] || (File.exist?('spec') && File.directory?('spec'))
125
+ end
126
+
127
+ protected
128
+ # Override with your own usage banner.
129
+ def banner
130
+ "Usage: #{$0} scaffold_resource ModelName [field:type, field:type]"
131
+ end
132
+
133
+ def rspec_views
134
+ %w[ index show new edit ]
135
+ end
136
+
137
+ def scaffold_views
138
+ rspec_views + %w[ _form ]
139
+ end
140
+
141
+ def model_name
142
+ class_name.demodulize
143
+ end
144
+
145
+ def add_options!(opt)
146
+ opt.separator ''
147
+ opt.separator 'Options:'
148
+ opt.on("--rspec", "Force rspec mode (checks for RAILS_ROOT/spec by default)") { |v| options[:rspec] = true }
149
+ end
150
+ end
151
+
152
+ module Rails
153
+ module Generator
154
+ class GeneratedAttribute
155
+ def default_value
156
+ @default_value ||= case type
157
+ when :int, :integer then "\"1\""
158
+ when :float then "\"1.5\""
159
+ when :decimal then "\"9.99\""
160
+ when :datetime, :timestamp, :time then "Time.now"
161
+ when :date then "Date.today"
162
+ when :string then "\"MyString\""
163
+ when :text then "\"MyText\""
164
+ when :boolean then "false"
165
+ else
166
+ ""
167
+ end
168
+ end
169
+
170
+ def input_type
171
+ @input_type ||= case type
172
+ when :text then "textarea"
173
+ else
174
+ "input"
175
+ end
176
+ end
177
+ end
178
+ end
179
+ end