nileshtrivedi-lp_resource_builder 0.5.1

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 (240) hide show
  1. data/LICENSE +22 -0
  2. data/README.rdoc +326 -0
  3. data/Rakefile +35 -0
  4. data/generators/scaffold_resource/USAGE +29 -0
  5. data/generators/scaffold_resource/scaffold_resource_generator.rb +183 -0
  6. data/generators/scaffold_resource/templates/controller.rb +69 -0
  7. data/generators/scaffold_resource/templates/fixtures.yml +10 -0
  8. data/generators/scaffold_resource/templates/functional_test.rb +57 -0
  9. data/generators/scaffold_resource/templates/helper.rb +22 -0
  10. data/generators/scaffold_resource/templates/migration.rb +15 -0
  11. data/generators/scaffold_resource/templates/model.rb +60 -0
  12. data/generators/scaffold_resource/templates/old_migration.rb +13 -0
  13. data/generators/scaffold_resource/templates/rspec/functional_spec.rb +255 -0
  14. data/generators/scaffold_resource/templates/rspec/helper_spec.rb +11 -0
  15. data/generators/scaffold_resource/templates/rspec/routing_spec.rb +61 -0
  16. data/generators/scaffold_resource/templates/rspec/unit_spec.rb +11 -0
  17. data/generators/scaffold_resource/templates/rspec/views/edit_spec.rb +28 -0
  18. data/generators/scaffold_resource/templates/rspec/views/index_spec.rb +26 -0
  19. data/generators/scaffold_resource/templates/rspec/views/new_spec.rb +30 -0
  20. data/generators/scaffold_resource/templates/rspec/views/show_spec.rb +25 -0
  21. data/generators/scaffold_resource/templates/shoulda_functional_test.rb +19 -0
  22. data/generators/scaffold_resource/templates/unit_test.rb +7 -0
  23. data/generators/scaffold_resource/templates/view__form.erb +6 -0
  24. data/generators/scaffold_resource/templates/view__form.haml +5 -0
  25. data/generators/scaffold_resource/templates/view_edit.erb +2 -0
  26. data/generators/scaffold_resource/templates/view_edit.haml +11 -0
  27. data/generators/scaffold_resource/templates/view_index.erb +2 -0
  28. data/generators/scaffold_resource/templates/view_index.haml +19 -0
  29. data/generators/scaffold_resource/templates/view_new.erb +2 -0
  30. data/generators/scaffold_resource/templates/view_new.haml +9 -0
  31. data/generators/scaffold_resource/templates/view_partial_edit.html.erb +18 -0
  32. data/generators/scaffold_resource/templates/view_partial_index.html.erb +28 -0
  33. data/generators/scaffold_resource/templates/view_partial_new.html.erb +13 -0
  34. data/generators/scaffold_resource/templates/view_partial_show.html.erb +9 -0
  35. data/generators/scaffold_resource/templates/view_show.erb +2 -0
  36. data/generators/scaffold_resource/templates/view_show.haml +9 -0
  37. data/init.rb +1 -0
  38. data/lib/resource_controller/accessors.rb +77 -0
  39. data/lib/resource_controller/action_options.rb +40 -0
  40. data/lib/resource_controller/actions.rb +75 -0
  41. data/lib/resource_controller/base.rb +15 -0
  42. data/lib/resource_controller/class_methods.rb +24 -0
  43. data/lib/resource_controller/controller.rb +69 -0
  44. data/lib/resource_controller/failable_action_options.rb +25 -0
  45. data/lib/resource_controller/helpers/current_objects.rb +69 -0
  46. data/lib/resource_controller/helpers/internal.rb +76 -0
  47. data/lib/resource_controller/helpers/nested.rb +63 -0
  48. data/lib/resource_controller/helpers/singleton_customizations.rb +60 -0
  49. data/lib/resource_controller/helpers/urls.rb +128 -0
  50. data/lib/resource_controller/helpers.rb +28 -0
  51. data/lib/resource_controller/response_collector.rb +27 -0
  52. data/lib/resource_controller/singleton.rb +15 -0
  53. data/lib/resource_controller/version.rb +9 -0
  54. data/lib/resource_controller.rb +20 -0
  55. data/lib/urligence.rb +50 -0
  56. data/rails/init.rb +6 -0
  57. data/test/Rakefile +10 -0
  58. data/test/app/controllers/accounts_controller.rb +6 -0
  59. data/test/app/controllers/application.rb +7 -0
  60. data/test/app/controllers/cms/options_controller.rb +3 -0
  61. data/test/app/controllers/cms/products_controller.rb +3 -0
  62. data/test/app/controllers/comments_controller.rb +3 -0
  63. data/test/app/controllers/images_controller.rb +4 -0
  64. data/test/app/controllers/options_controller.rb +8 -0
  65. data/test/app/controllers/people_controller.rb +9 -0
  66. data/test/app/controllers/photos_controller.rb +11 -0
  67. data/test/app/controllers/posts_controller.rb +10 -0
  68. data/test/app/controllers/projects_controller.rb +3 -0
  69. data/test/app/controllers/somethings_controller.rb +3 -0
  70. data/test/app/controllers/tags_controller.rb +13 -0
  71. data/test/app/controllers/users_controller.rb +12 -0
  72. data/test/app/helpers/accounts_helper.rb +2 -0
  73. data/test/app/helpers/application_helper.rb +3 -0
  74. data/test/app/helpers/cms/products_helper.rb +2 -0
  75. data/test/app/helpers/comments_helper.rb +2 -0
  76. data/test/app/helpers/images_helper.rb +2 -0
  77. data/test/app/helpers/options_helper.rb +2 -0
  78. data/test/app/helpers/people_helper.rb +2 -0
  79. data/test/app/helpers/photos_helper.rb +2 -0
  80. data/test/app/helpers/posts_helper.rb +2 -0
  81. data/test/app/helpers/projects_helper.rb +2 -0
  82. data/test/app/helpers/somethings_helper.rb +2 -0
  83. data/test/app/helpers/tags_helper.rb +2 -0
  84. data/test/app/helpers/users_helper.rb +2 -0
  85. data/test/app/models/account.rb +4 -0
  86. data/test/app/models/comment.rb +3 -0
  87. data/test/app/models/image.rb +3 -0
  88. data/test/app/models/option.rb +3 -0
  89. data/test/app/models/photo.rb +4 -0
  90. data/test/app/models/post.rb +3 -0
  91. data/test/app/models/product.rb +3 -0
  92. data/test/app/models/project.rb +2 -0
  93. data/test/app/models/something.rb +2 -0
  94. data/test/app/models/tag.rb +3 -0
  95. data/test/app/models/user.rb +3 -0
  96. data/test/app/views/accounts/_form.html.erb +4 -0
  97. data/test/app/views/accounts/edit.html.erb +14 -0
  98. data/test/app/views/accounts/new.html.erb +12 -0
  99. data/test/app/views/accounts/show.html.erb +5 -0
  100. data/test/app/views/cms/options/edit.rhtml +17 -0
  101. data/test/app/views/cms/options/index.rhtml +20 -0
  102. data/test/app/views/cms/options/new.rhtml +16 -0
  103. data/test/app/views/cms/options/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 +13 -0
  159. data/test/config/environment.rb +64 -0
  160. data/test/config/environments/development.rb +21 -0
  161. data/test/config/environments/test.rb +19 -0
  162. data/test/config/routes.rb +58 -0
  163. data/test/db/migrate/001_create_posts.rb +12 -0
  164. data/test/db/migrate/002_create_products.rb +11 -0
  165. data/test/db/migrate/003_create_comments.rb +13 -0
  166. data/test/db/migrate/004_create_options.rb +13 -0
  167. data/test/db/migrate/005_create_photos.rb +11 -0
  168. data/test/db/migrate/006_create_tags.rb +17 -0
  169. data/test/db/migrate/007_create_somethings.rb +11 -0
  170. data/test/db/migrate/008_create_accounts.rb +11 -0
  171. data/test/db/migrate/009_add_account_id_to_photos.rb +9 -0
  172. data/test/db/migrate/010_create_projects.rb +11 -0
  173. data/test/db/migrate/011_create_images.rb +12 -0
  174. data/test/db/migrate/012_create_users.rb +11 -0
  175. data/test/db/schema.rb +72 -0
  176. data/test/script/console +3 -0
  177. data/test/script/destroy +3 -0
  178. data/test/script/generate +3 -0
  179. data/test/script/server +3 -0
  180. data/test/test/fixtures/accounts.yml +7 -0
  181. data/test/test/fixtures/comments.yml +11 -0
  182. data/test/test/fixtures/images.yml +6 -0
  183. data/test/test/fixtures/options.yml +9 -0
  184. data/test/test/fixtures/photos.yml +9 -0
  185. data/test/test/fixtures/photos_tags.yml +3 -0
  186. data/test/test/fixtures/posts.yml +9 -0
  187. data/test/test/fixtures/products.yml +7 -0
  188. data/test/test/fixtures/projects.yml +7 -0
  189. data/test/test/fixtures/somethings.yml +7 -0
  190. data/test/test/fixtures/tags.yml +7 -0
  191. data/test/test/fixtures/users.yml +5 -0
  192. data/test/test/functional/cms/options_controller_test.rb +23 -0
  193. data/test/test/functional/cms/products_controller_test.rb +23 -0
  194. data/test/test/functional/comments_controller_test.rb +26 -0
  195. data/test/test/functional/images_controller_test.rb +37 -0
  196. data/test/test/functional/people_controller_test.rb +34 -0
  197. data/test/test/functional/photos_controller_test.rb +130 -0
  198. data/test/test/functional/posts_controller_test.rb +34 -0
  199. data/test/test/functional/projects_controller_test.rb +18 -0
  200. data/test/test/functional/somethings_controller_test.rb +28 -0
  201. data/test/test/functional/tags_controller_test.rb +64 -0
  202. data/test/test/functional/users_controller_test.rb +24 -0
  203. data/test/test/test_helper.rb +12 -0
  204. data/test/test/unit/accessors_test.rb +110 -0
  205. data/test/test/unit/account_test.rb +7 -0
  206. data/test/test/unit/action_options_test.rb +109 -0
  207. data/test/test/unit/base_test.rb +11 -0
  208. data/test/test/unit/comment_test.rb +10 -0
  209. data/test/test/unit/failable_action_options_test.rb +77 -0
  210. data/test/test/unit/helpers/current_objects_test.rb +133 -0
  211. data/test/test/unit/helpers/internal_test.rb +106 -0
  212. data/test/test/unit/helpers/nested_test.rb +86 -0
  213. data/test/test/unit/helpers/singleton_current_objects_test.rb +68 -0
  214. data/test/test/unit/helpers/singleton_nested_test.rb +77 -0
  215. data/test/test/unit/helpers/singleton_urls_test.rb +67 -0
  216. data/test/test/unit/helpers/urls_test.rb +75 -0
  217. data/test/test/unit/helpers_test.rb +25 -0
  218. data/test/test/unit/image_test.rb +7 -0
  219. data/test/test/unit/option_test.rb +10 -0
  220. data/test/test/unit/photo_test.rb +10 -0
  221. data/test/test/unit/post_test.rb +10 -0
  222. data/test/test/unit/project_test.rb +10 -0
  223. data/test/test/unit/response_collector_test.rb +49 -0
  224. data/test/test/unit/something_test.rb +10 -0
  225. data/test/test/unit/tag_test.rb +10 -0
  226. data/test/test/unit/urligence_test.rb +203 -0
  227. data/test/vendor/plugins/shoulda/Rakefile +32 -0
  228. data/test/vendor/plugins/shoulda/bin/convert_to_should_syntax +40 -0
  229. data/test/vendor/plugins/shoulda/init.rb +3 -0
  230. data/test/vendor/plugins/shoulda/lib/shoulda/active_record_helpers.rb +580 -0
  231. data/test/vendor/plugins/shoulda/lib/shoulda/color.rb +77 -0
  232. data/test/vendor/plugins/shoulda/lib/shoulda/controller_tests/controller_tests.rb +467 -0
  233. data/test/vendor/plugins/shoulda/lib/shoulda/controller_tests/formats/html.rb +201 -0
  234. data/test/vendor/plugins/shoulda/lib/shoulda/controller_tests/formats/xml.rb +170 -0
  235. data/test/vendor/plugins/shoulda/lib/shoulda/gem/proc_extensions.rb +14 -0
  236. data/test/vendor/plugins/shoulda/lib/shoulda/gem/shoulda.rb +239 -0
  237. data/test/vendor/plugins/shoulda/lib/shoulda/general.rb +118 -0
  238. data/test/vendor/plugins/shoulda/lib/shoulda/private_helpers.rb +22 -0
  239. data/test/vendor/plugins/shoulda/lib/shoulda.rb +43 -0
  240. metadata +342 -0
data/LICENSE ADDED
@@ -0,0 +1,22 @@
1
+ Copyright (c) 2008 James Golick
2
+
3
+ Permission is hereby granted, free of charge, to any person
4
+ obtaining a copy of this software and associated documentation
5
+ files (the "Software"), to deal in the Software without
6
+ restriction, including without limitation the rights to use,
7
+ copy, modify, merge, publish, distribute, sublicense, and/or sell
8
+ copies of the Software, and to permit persons to whom the
9
+ Software is furnished to do so, subject to the following
10
+ conditions:
11
+
12
+ The above copyright notice and this permission notice shall be
13
+ included in all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
16
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
17
+ OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
18
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
19
+ HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
20
+ WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21
+ FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
22
+ OTHER DEALINGS IN THE SOFTWARE.
data/README.rdoc ADDED
@@ -0,0 +1,326 @@
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
+ Add it as a gem dependency
8
+
9
+ config.gem 'resource_controller'
10
+
11
+ ...or for the development version
12
+
13
+ config.gem 'giraffesoft-resource_controller', :lib => 'resource_controller', :source => 'http://gems.github.com'
14
+
15
+ Or install it as a gem manually
16
+
17
+ sudo gem install resource_controller
18
+
19
+ ...or for the development version
20
+
21
+ sudo gem install giraffesoft-resource_controller
22
+
23
+ Or grab the source
24
+
25
+ git clone git://github.com/giraffesoft/resource_controller.git
26
+
27
+ = Usage
28
+
29
+ Creating a basic RESTful controller is as easy as...
30
+
31
+ class PostsController < ResourceController::Base
32
+ end
33
+
34
+ ...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:
35
+
36
+ class PostsController < ApplicationController
37
+ resource_controller
38
+ end
39
+
40
+ 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.
41
+
42
+
43
+ Nobody just uses the default RESTful controller, though. resource_controller provides a simple API for customizations.
44
+
45
+ == Action Lifecycle
46
+
47
+ It's really easy to make changes to the lifecycle of your actions.
48
+
49
+ Note: We had to call the new accessor "new_action", since new is somewhat reserved in ruby.
50
+
51
+ === Before and After
52
+
53
+ class ProjectsController < ResourceController::Base
54
+
55
+ new_action.before do
56
+ 3.times { object.tasks.build }
57
+ end
58
+
59
+ create.after do
60
+ object.creator = current_user
61
+ end
62
+
63
+ end
64
+
65
+ === Flash
66
+
67
+ class ProjectsController < ResourceController::Base
68
+ create.flash "Can you believe how easy it is to use resource_controller? Neither could I!"
69
+ end
70
+
71
+ === respond_to
72
+
73
+ You can add to what's already there...
74
+
75
+ class ProjectsController < ResourceController::Base
76
+ create.wants.js { render :template => "show.rjs" }
77
+ end
78
+
79
+ Or you can create a whole new block. This syntax destroys everything that's there, and starts again...
80
+
81
+ class ProjectsController < ResourceController::Base
82
+ create.response do |wants|
83
+ wants.html
84
+ wants.js { render :template => "show.rjs" }
85
+ end
86
+ end
87
+
88
+ === Scoping
89
+
90
+ 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.
91
+
92
+ With actions that can fail, the scoping defaults to success. That means that create.flash == create.success.flash.
93
+
94
+ class ProjectsController < ResourceController::Base
95
+
96
+ create do
97
+ flash "Object successfully created!"
98
+ wants.js { render :template => "show.rjs" }
99
+
100
+ failure.wants.js { render :template => "display_errors.rjs" }
101
+ end
102
+
103
+ destroy do
104
+ flash "You destroyed your project. Good work."
105
+
106
+ failure do
107
+ flash "You cannot destroy that project. Stop trying!"
108
+ wants.js { render :template => "display_errors.rjs" }
109
+ end
110
+ end
111
+
112
+ end
113
+
114
+ == Singleton Resource
115
+
116
+ If you want to create a singleton RESTful controller inherit from ResourceController::Singleton.
117
+
118
+ class AccountsController < ResourceController::Singleton
119
+ end
120
+
121
+ *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.
122
+
123
+ 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.
124
+
125
+ class AccountsController < ResourceController::Singleton
126
+ private
127
+ def object
128
+ @object ||= Account.find(session[:account_id])
129
+ end
130
+ end
131
+
132
+ In other cases you can use the default logic and override it only if you use permalinks or anything special.
133
+
134
+ Singleton nesting with both :has_many and :has_one associations is provided...
135
+
136
+ map.resource :account, :has_many => :options # /account/options, account is a singleton parent
137
+ map.resources :users, :has_one => :image # /users/1/image, image is a singleton child
138
+
139
+ 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.
140
+
141
+ class OptionsController < ResourceController::Base
142
+ belongs_to :account
143
+
144
+ protected
145
+ def parent_object
146
+ Account.find(session[:account_id])
147
+ end
148
+ end
149
+
150
+ == Helpers (ResourceController::Helpers)
151
+
152
+ === Loading objects
153
+
154
+ You want to add something like pagination to your controller...
155
+
156
+ class PostsController < ResourceController::Base
157
+ private
158
+ def collection
159
+ @collection ||= end_of_association_chain.find(:all, :page => {:size => 10, :current => params[:page]})
160
+ end
161
+ end
162
+
163
+ Or maybe you used a permalink...
164
+
165
+ class PostsController < ResourceController::Base
166
+ private
167
+ def object
168
+ @object ||= end_of_association_chain.find_by_permalink(param)
169
+ end
170
+ end
171
+
172
+ === Building objects
173
+
174
+ Maybe you have some alternative way of building objects...
175
+
176
+ class PostsController < ResourceController::Base
177
+ private
178
+ def build_object
179
+ @object ||= end_of_association_chain.build_my_object_some_funky_way object_params
180
+ end
181
+ end
182
+
183
+ ...and there are tons more helpers in the ResourceController::Helpers
184
+
185
+ == Nested Resources
186
+
187
+ 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.
188
+
189
+ class CommentsController < ResourceController::Base
190
+ belongs_to :post
191
+ end
192
+
193
+ All of the finding, and creation, and everything will be done at the scope of the post automatically.
194
+
195
+ == Namespaced Resources
196
+
197
+ ...are handled automatically, and any namespaces are always available, symbolized, in array form @ ResourceController::Helpers#namespaces
198
+
199
+ == Polymorphic Resources
200
+
201
+ Everything, including url generation is handled completely automatically. Take this example...
202
+
203
+ ## comment.rb
204
+ class Comment
205
+ belongs_to :commentable, :polymorphic => true
206
+ end
207
+
208
+ ## comments_controller.rb
209
+ class CommentsController < ResourceController::Base
210
+ belongs_to :post, :product, :user
211
+ end
212
+ *Note:* Your model doesn't have to be polymorphic in the ActiveRecord sense. It can be associated in whichever way you want.
213
+
214
+ ## routes.rb
215
+ map.resources :posts, :has_many => :comments
216
+ map.resources :products, :has_many => :comments
217
+ map.resources :users, :has_many => :comments
218
+
219
+ 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.
220
+
221
+ === Parent Helpers
222
+
223
+ You also get some helpers for reflecting on your parent.
224
+
225
+ parent? # => true/false is there a parent present?
226
+ parent_type # => :post
227
+ parent_model # => Post
228
+ parent_object # => @post
229
+
230
+ === Non-standard resource names
231
+
232
+ resource_controller supports overrides for every non-standard configuration of resources.
233
+
234
+ 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.
235
+
236
+ map.resources :tags
237
+ ...
238
+ class PhotoTag < ActiveRecord::Base
239
+ ...
240
+ class TagsController < ResourceController::Base
241
+ private
242
+ def model_name
243
+ 'photo_tag'
244
+ end
245
+ end
246
+
247
+ 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.
248
+
249
+ def object_name
250
+ 'photo_tag'
251
+ end
252
+
253
+ 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.
254
+
255
+ map.resources :tags, :controller => "somethings"
256
+ ...
257
+ class Tag < ActiveRecord::Base
258
+ ...
259
+ class SomethingsController < ResourceController::Base
260
+ private
261
+ def resource_name
262
+ 'tag'
263
+ end
264
+ end
265
+
266
+ 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.
267
+
268
+ map.resources :tags, :controller => "taggings"
269
+ ...
270
+ class Taggings < ActiveRecord::Base
271
+ ...
272
+ class TaggingsController < ResourceController::Base
273
+ private
274
+ def route_name
275
+ 'tag'
276
+ end
277
+ end
278
+
279
+ == Url Helpers
280
+
281
+ Thanks to Urligence, you also get some free url helpers.
282
+
283
+ No matter what your controller looks like...
284
+
285
+ [edit_|new_]object_url # is the equivalent of saying [edit_|new_]post_url(@post)
286
+ [edit_|new_]object_url(some_other_object) # allows you to specify an object, but still maintain any paths or namespaces that are present
287
+
288
+ collection_url # is like saying posts_url
289
+
290
+ Url helpers are especially useful when working with polymorphic controllers.
291
+
292
+ # /posts/1/comments
293
+ object_url # => /posts/1/comments/#{@comment.to_param}
294
+ object_url(comment) # => /posts/1/comments/#{comment.to_param}
295
+ edit_object_url # => /posts/1/comments/#{@comment.to_param}/edit
296
+ collection_url # => /posts/1/comments
297
+
298
+ # /products/1/comments
299
+ object_url # => /products/1/comments/#{@comment.to_param}
300
+ object_url(comment) # => /products/1/comments/#{comment.to_param}
301
+ edit_object_url # => /products/1/comments/#{@comment.to_param}/edit
302
+ collection_url # => /products/1/comments
303
+
304
+ # /comments
305
+ object_url # => /comments/#{@comment.to_param}
306
+ object_url(comment) # => /comments/#{comment.to_param}
307
+ edit_object_url # => /comments/#{@comment.to_param}/edit
308
+ collection_url # => /comments
309
+
310
+ Or with namespaced, nested controllers...
311
+
312
+ # /admin/products/1/options
313
+ object_url # => /admin/products/1/options/#{@option.to_param}
314
+ object_url(option) # => /admin/products/1/options/#{option.to_param}
315
+ edit_object_url # => /admin/products/1/options/#{@option.to_param}/edit
316
+ collection_url # => /admin/products/1/options
317
+
318
+ You get the idea. Everything is automagical! All parameters are inferred.
319
+
320
+ == Credits
321
+
322
+ resource_controller was created, and is maintained by {James Golick}[http://jamesgolick.com].
323
+
324
+ == License
325
+
326
+ resource_controller is available under the {MIT License}[http://en.wikipedia.org/wiki/MIT_License]
data/Rakefile ADDED
@@ -0,0 +1,35 @@
1
+ require 'rake'
2
+ require 'rake/testtask'
3
+ require 'rake/rdoctask'
4
+ require File.dirname(__FILE__)+'/lib/resource_controller/version'
5
+ Dir['tasks/**.rake'].each { |tasks| load tasks }
6
+
7
+ desc 'Default: run unit tests.'
8
+ task :default => :test
9
+
10
+ desc 'Test the ResourceController plugin.'
11
+ Rake::TestTask.new(:test) do |t|
12
+ t.libs << 'lib'
13
+ t.pattern = 'test/test/**/*_test.rb'
14
+ t.verbose = true
15
+ end
16
+
17
+ desc 'Generate documentation for the ResourceController plugin.'
18
+ Rake::RDocTask.new(:rdoc) do |rdoc|
19
+ rdoc.rdoc_dir = 'rdoc'
20
+ rdoc.title = 'ResourceController'
21
+ rdoc.options << '--line-numbers' << '--inline-source'
22
+ rdoc.rdoc_files.include('README.rdoc')
23
+ rdoc.rdoc_files.include('lib/**/*.rb')
24
+ end
25
+
26
+ task :upload_docs => :rdoc do
27
+ puts 'Deleting previous rdoc'
28
+ `ssh jamesgolick.com 'rm -Rf /home/apps/jamesgolick.com/public/resource_controller/rdoc'`
29
+
30
+ puts "Uploading current rdoc"
31
+ `scp -r rdoc jamesgolick.com:/home/apps/jamesgolick.com/public/resource_controller`
32
+
33
+ puts "Deleting rdoc"
34
+ `rm -Rf rdoc`
35
+ end
@@ -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,183 @@
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
+ m.template(
77
+ "view_partial_#{action}.html.erb",
78
+ File.join('app/views', controller_class_path, controller_file_name, "_#{action}.html.erb")
79
+ ) unless action == "_form"
80
+ end
81
+
82
+ m.template('model.rb', File.join('app/models', class_path, "#{file_name}.rb"))
83
+ m.template('controller.rb', File.join('app/controllers', controller_class_path, "#{controller_file_name}_controller.rb"))
84
+ m.template('helper.rb', File.join('app/helpers', controller_class_path, "#{controller_file_name}_helper.rb"))
85
+
86
+ if @rspec
87
+ m.template('rspec/functional_spec.rb', File.join('spec/controllers', controller_class_path, "#{controller_file_name}_controller_spec.rb"))
88
+ m.template('rspec/routing_spec.rb', File.join('spec/controllers', controller_class_path, "#{controller_file_name}_routing_spec.rb"))
89
+ m.template('rspec/helper_spec.rb', File.join('spec/helpers', class_path, "#{controller_file_name}_helper_spec.rb"))
90
+ m.template('rspec/unit_spec.rb', File.join('spec/models', class_path, "#{file_name}_spec.rb"))
91
+ m.template('fixtures.yml', File.join('spec/fixtures', "#{table_name}.yml"))
92
+
93
+ rspec_views.each do |action|
94
+ m.template(
95
+ "rspec/views/#{action}_spec.rb",
96
+ File.join('spec/views', controller_class_path, controller_file_name, "#{action}_spec.rb")
97
+ )
98
+ end
99
+
100
+ else
101
+ functional_test = (defined? ThoughtBot::Shoulda) ? "shoulda_functional_test.rb" : "functional_test.rb"
102
+
103
+ m.template("#{functional_test}", File.join('test/functional', controller_class_path, "#{controller_file_name}_controller_test.rb"))
104
+ m.template('unit_test.rb', File.join('test/unit', class_path, "#{file_name}_test.rb"))
105
+ m.template('fixtures.yml', File.join('test/fixtures', "#{table_name}.yml"))
106
+ end
107
+
108
+
109
+ unless options[:skip_migration]
110
+ migration_template = RAILS_GEM_VERSION.to_i == 1 ? 'old_migration.rb' : 'migration.rb'
111
+
112
+ m.migration_template(
113
+ migration_template, 'db/migrate',
114
+ :assigns => {
115
+ :migration_name => "Create#{class_name.pluralize.gsub(/::/, '')}",
116
+ :attributes => attributes
117
+ },
118
+ :migration_file_name => "create_#{file_path.gsub(/\//, '_').pluralize}"
119
+ )
120
+ end
121
+
122
+ m.route_resources controller_file_name
123
+ end
124
+ end
125
+
126
+ # Lifted from Rick Olson's restful_authentication
127
+ def has_rspec?
128
+ options[:rspec] || (File.exist?('spec') && File.directory?('spec'))
129
+ end
130
+
131
+ protected
132
+ # Override with your own usage banner.
133
+ def banner
134
+ "Usage: #{$0} scaffold_resource ModelName [field:type, field:type]"
135
+ end
136
+
137
+ def rspec_views
138
+ %w[ index show new edit ]
139
+ end
140
+
141
+ def scaffold_views
142
+ rspec_views + %w[ _form ]
143
+ end
144
+
145
+ def model_name
146
+ class_name.demodulize
147
+ end
148
+
149
+ def add_options!(opt)
150
+ opt.separator ''
151
+ opt.separator 'Options:'
152
+ opt.on("--rspec", "Force rspec mode (checks for RAILS_ROOT/spec by default)") { |v| options[:rspec] = true }
153
+ end
154
+ end
155
+
156
+ module Rails
157
+ module Generator
158
+ class GeneratedAttribute
159
+ def default_value
160
+ @default_value ||= case type
161
+ when :int, :integer then "\"1\""
162
+ when :float then "\"1.5\""
163
+ when :decimal then "\"9.99\""
164
+ when :datetime, :timestamp, :time then "Time.now"
165
+ when :date then "Date.today"
166
+ when :string then "\"MyString\""
167
+ when :text then "\"MyText\""
168
+ when :boolean then "false"
169
+ else
170
+ ""
171
+ end
172
+ end
173
+
174
+ def input_type
175
+ @input_type ||= case type
176
+ when :text then "textarea"
177
+ else
178
+ "input"
179
+ end
180
+ end
181
+ end
182
+ end
183
+ end
@@ -0,0 +1,69 @@
1
+ class <%= controller_class_name %>Controller < ResourceController::Base
2
+ before_filter :has_edit_permission, :only => [:edit]
3
+ before_filter :has_update_permission, :only => [:update]
4
+ before_filter :has_create_permission, :only => [:new, :create]
5
+ before_filter :has_view_permission, :only => [:show]
6
+ before_filter :has_delete_permission, :only => [:destroy]
7
+ before_filter :select_viewable_objects, :only => [:index]
8
+
9
+ private
10
+
11
+ def select_viewable_objects
12
+ #TODO: instead of Article.all, this should be chain.articles || Article.all as the case is
13
+ @viewable_<%= plural_name %> = end_of_association_chain.find(:all).select {|v| (v.viewable_by?(current_user, "id")) }
14
+ @viewable_<%= plural_name %> = @viewable_<%= plural_name %>.paginate(:page => params[:page], :per_page => 4)
15
+ end
16
+
17
+ def has_edit_permission
18
+ #TODO: infer permissions from updatable_by? whether to display the field for a given attribute or not
19
+ load_object
20
+ if(!@<%= singular_name %>.editable_by?(current_user))
21
+ flash[:notice] = "Permision denied."
22
+ redirect_to collection_url
23
+ return false
24
+ end
25
+ true
26
+ end
27
+
28
+ def has_create_permission
29
+ build_object
30
+ load_object
31
+ if(!@<%= singular_name %>.creatable_by?(current_user))
32
+ flash[:notice] = "Permision denied."
33
+ redirect_to collection_url
34
+ return
35
+ end
36
+ end
37
+
38
+ def has_view_permission
39
+ load_object
40
+ if(!@<%= singular_name %>.viewable_by?(current_user,"id"))
41
+ flash[:notice] = "Permision denied."
42
+ redirect_to collection_url
43
+ return
44
+ end
45
+ end
46
+
47
+ def has_delete_permission
48
+ load_object
49
+ if(!@<%= singular_name %>.deletable_by?(current_user))
50
+ flash[:notice] = "Permision denied."
51
+ redirect_to collection_url
52
+ return
53
+ end
54
+ end
55
+
56
+ def has_update_permission
57
+ load_object
58
+ #update attributes without saving to db so that we can call updatable_by?
59
+ #TODO: r_c will call update_attributes again even though simple save would be sufficient after the following line
60
+ @<%= singular_name %>.attributes = params[:article]
61
+ if(!@<%= singular_name %>.updatable_by?(current_user,@<%= singular_name %>))
62
+ flash[:notice] = "Permision denied."
63
+ redirect_to collection_url
64
+ return
65
+ end
66
+ end
67
+ end
68
+
69
+
@@ -0,0 +1,10 @@
1
+ one:
2
+ id: 1
3
+ <% for attribute in attributes -%>
4
+ <%= attribute.name %>: <%= attribute.default %>
5
+ <% end -%>
6
+ two:
7
+ id: 2
8
+ <% for attribute in attributes -%>
9
+ <%= attribute.name %>: <%= attribute.default %>
10
+ <% end -%>