thoughtbot-shoulda 2.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (101) hide show
  1. data/CONTRIBUTION_GUIDELINES.rdoc +12 -0
  2. data/MIT-LICENSE +22 -0
  3. data/README.rdoc +132 -0
  4. data/Rakefile +72 -0
  5. data/bin/convert_to_should_syntax +42 -0
  6. data/lib/shoulda.rb +16 -0
  7. data/lib/shoulda/action_mailer.rb +10 -0
  8. data/lib/shoulda/action_mailer/assertions.rb +39 -0
  9. data/lib/shoulda/active_record.rb +12 -0
  10. data/lib/shoulda/active_record/assertions.rb +85 -0
  11. data/lib/shoulda/active_record/macros.rb +703 -0
  12. data/lib/shoulda/assertions.rb +45 -0
  13. data/lib/shoulda/context.rb +309 -0
  14. data/lib/shoulda/controller.rb +30 -0
  15. data/lib/shoulda/controller/formats/html.rb +201 -0
  16. data/lib/shoulda/controller/formats/xml.rb +170 -0
  17. data/lib/shoulda/controller/helpers.rb +64 -0
  18. data/lib/shoulda/controller/macros.rb +287 -0
  19. data/lib/shoulda/controller/resource_options.rb +236 -0
  20. data/lib/shoulda/helpers.rb +10 -0
  21. data/lib/shoulda/macros.rb +80 -0
  22. data/lib/shoulda/private_helpers.rb +22 -0
  23. data/lib/shoulda/proc_extensions.rb +14 -0
  24. data/lib/shoulda/rails.rb +19 -0
  25. data/lib/shoulda/tasks.rb +3 -0
  26. data/lib/shoulda/tasks/list_tests.rake +24 -0
  27. data/lib/shoulda/tasks/yaml_to_shoulda.rake +28 -0
  28. data/test/README +36 -0
  29. data/test/fixtures/addresses.yml +3 -0
  30. data/test/fixtures/friendships.yml +0 -0
  31. data/test/fixtures/posts.yml +5 -0
  32. data/test/fixtures/products.yml +0 -0
  33. data/test/fixtures/taggings.yml +0 -0
  34. data/test/fixtures/tags.yml +9 -0
  35. data/test/fixtures/users.yml +6 -0
  36. data/test/functional/posts_controller_test.rb +104 -0
  37. data/test/functional/users_controller_test.rb +36 -0
  38. data/test/other/context_test.rb +145 -0
  39. data/test/other/convert_to_should_syntax_test.rb +63 -0
  40. data/test/other/helpers_test.rb +183 -0
  41. data/test/other/private_helpers_test.rb +34 -0
  42. data/test/other/should_test.rb +266 -0
  43. data/test/rails_root/app/controllers/application.rb +25 -0
  44. data/test/rails_root/app/controllers/posts_controller.rb +85 -0
  45. data/test/rails_root/app/controllers/users_controller.rb +81 -0
  46. data/test/rails_root/app/helpers/application_helper.rb +3 -0
  47. data/test/rails_root/app/helpers/posts_helper.rb +2 -0
  48. data/test/rails_root/app/helpers/users_helper.rb +2 -0
  49. data/test/rails_root/app/models/address.rb +7 -0
  50. data/test/rails_root/app/models/dog.rb +5 -0
  51. data/test/rails_root/app/models/flea.rb +3 -0
  52. data/test/rails_root/app/models/friendship.rb +4 -0
  53. data/test/rails_root/app/models/post.rb +12 -0
  54. data/test/rails_root/app/models/product.rb +12 -0
  55. data/test/rails_root/app/models/tag.rb +8 -0
  56. data/test/rails_root/app/models/tagging.rb +4 -0
  57. data/test/rails_root/app/models/user.rb +28 -0
  58. data/test/rails_root/app/views/layouts/posts.rhtml +17 -0
  59. data/test/rails_root/app/views/layouts/users.rhtml +17 -0
  60. data/test/rails_root/app/views/posts/edit.rhtml +27 -0
  61. data/test/rails_root/app/views/posts/index.rhtml +25 -0
  62. data/test/rails_root/app/views/posts/new.rhtml +26 -0
  63. data/test/rails_root/app/views/posts/show.rhtml +18 -0
  64. data/test/rails_root/app/views/users/edit.rhtml +22 -0
  65. data/test/rails_root/app/views/users/index.rhtml +22 -0
  66. data/test/rails_root/app/views/users/new.rhtml +21 -0
  67. data/test/rails_root/app/views/users/show.rhtml +13 -0
  68. data/test/rails_root/config/boot.rb +109 -0
  69. data/test/rails_root/config/database.yml +4 -0
  70. data/test/rails_root/config/environment.rb +14 -0
  71. data/test/rails_root/config/environments/sqlite3.rb +0 -0
  72. data/test/rails_root/config/initializers/new_rails_defaults.rb +15 -0
  73. data/test/rails_root/config/initializers/shoulda.rb +8 -0
  74. data/test/rails_root/config/routes.rb +6 -0
  75. data/test/rails_root/db/migrate/001_create_users.rb +18 -0
  76. data/test/rails_root/db/migrate/002_create_posts.rb +13 -0
  77. data/test/rails_root/db/migrate/003_create_taggings.rb +12 -0
  78. data/test/rails_root/db/migrate/004_create_tags.rb +11 -0
  79. data/test/rails_root/db/migrate/005_create_dogs.rb +12 -0
  80. data/test/rails_root/db/migrate/006_create_addresses.rb +14 -0
  81. data/test/rails_root/db/migrate/007_create_fleas.rb +11 -0
  82. data/test/rails_root/db/migrate/008_create_dogs_fleas.rb +12 -0
  83. data/test/rails_root/db/migrate/009_create_products.rb +17 -0
  84. data/test/rails_root/db/migrate/010_create_friendships.rb +14 -0
  85. data/test/rails_root/db/schema.rb +0 -0
  86. data/test/rails_root/public/404.html +30 -0
  87. data/test/rails_root/public/422.html +30 -0
  88. data/test/rails_root/public/500.html +30 -0
  89. data/test/rails_root/script/console +3 -0
  90. data/test/rails_root/script/generate +3 -0
  91. data/test/test_helper.rb +33 -0
  92. data/test/unit/address_test.rb +10 -0
  93. data/test/unit/dog_test.rb +7 -0
  94. data/test/unit/flea_test.rb +6 -0
  95. data/test/unit/friendship_test.rb +6 -0
  96. data/test/unit/post_test.rb +15 -0
  97. data/test/unit/product_test.rb +27 -0
  98. data/test/unit/tag_test.rb +14 -0
  99. data/test/unit/tagging_test.rb +6 -0
  100. data/test/unit/user_test.rb +52 -0
  101. metadata +197 -0
@@ -0,0 +1,34 @@
1
+ require File.join(File.dirname(__FILE__), '..', 'test_helper')
2
+
3
+ class PrivateHelpersTest < Test::Unit::TestCase # :nodoc:
4
+ include ThoughtBot::Shoulda::Private
5
+ context "get_options!" do
6
+ should "remove opts from args" do
7
+ args = [:a, :b, {}]
8
+ get_options!(args)
9
+ assert_equal [:a, :b], args
10
+ end
11
+
12
+ should "return wanted opts in order" do
13
+ args = [{:one => 1, :two => 2}]
14
+ one, two = get_options!(args, :one, :two)
15
+ assert_equal 1, one
16
+ assert_equal 2, two
17
+ end
18
+
19
+ should "raise ArgumentError if given unwanted option" do
20
+ args = [{:one => 1, :two => 2}]
21
+ assert_raises ArgumentError do
22
+ get_options!(args, :one)
23
+ end
24
+ end
25
+ end
26
+
27
+ class ::SomeModel; end
28
+ context "model_class" do
29
+ should "sniff the class constant from the test class" do
30
+ self.expects(:name).returns("SomeModelTest")
31
+ assert_equal SomeModel, model_class
32
+ end
33
+ end
34
+ end
@@ -0,0 +1,266 @@
1
+ require File.join(File.dirname(__FILE__), '..', 'test_helper')
2
+
3
+ class ShouldTest < Test::Unit::TestCase # :nodoc:
4
+ should "be able to define a should statement outside of a context" do
5
+ assert true
6
+ end
7
+
8
+ should "see the name of my class as ShouldTest" do
9
+ assert_equal "ShouldTest", self.class.name
10
+ end
11
+
12
+ def self.should_see_class_methods
13
+ should "be able to see class methods" do
14
+ assert true
15
+ end
16
+ end
17
+
18
+ def self.should_see_a_context_block_like_a_Test_Unit_class
19
+ should "see a context block as a Test::Unit class" do
20
+ assert_equal "ShouldTest", self.class.name
21
+ end
22
+ end
23
+
24
+ def self.should_see_blah
25
+ should "see @blah through a macro" do
26
+ assert @blah
27
+ end
28
+ end
29
+
30
+ def self.should_not_see_blah
31
+ should "not see @blah through a macro" do
32
+ assert_nil @blah
33
+ end
34
+ end
35
+
36
+ def self.should_be_able_to_make_context_macros(prefix = nil)
37
+ context "a macro" do
38
+ should "have the tests named correctly" do
39
+ assert_match(/^test: #{prefix}a macro should have the tests named correctly/, self.to_s)
40
+ end
41
+ end
42
+ end
43
+
44
+ context "Context" do
45
+
46
+ should_see_class_methods
47
+ should_see_a_context_block_like_a_Test_Unit_class
48
+ should_be_able_to_make_context_macros("Context ")
49
+
50
+ should "not define @blah" do
51
+ assert ! self.instance_variables.include?("@blah")
52
+ end
53
+
54
+ should_not_see_blah
55
+
56
+ should "be able to define a should statement" do
57
+ assert true
58
+ end
59
+
60
+ should "see the name of my class as ShouldTest" do
61
+ assert_equal "ShouldTest", self.class.name
62
+ end
63
+
64
+ context "with a subcontext" do
65
+ should_be_able_to_make_context_macros("Context with a subcontext ")
66
+ end
67
+ end
68
+
69
+ context "Context with setup block" do
70
+ setup do
71
+ @blah = "blah"
72
+ end
73
+
74
+ should "have @blah == 'blah'" do
75
+ assert_equal "blah", @blah
76
+ end
77
+ should_see_blah
78
+
79
+ should "have name set right" do
80
+ assert_match(/^test: Context with setup block/, self.to_s)
81
+ end
82
+
83
+ context "and a subcontext" do
84
+ setup do
85
+ @blah = "#{@blah} twice"
86
+ end
87
+
88
+ should "be named correctly" do
89
+ assert_match(/^test: Context with setup block and a subcontext should be named correctly/, self.to_s)
90
+ end
91
+
92
+ should "run the setup methods in order" do
93
+ assert_equal @blah, "blah twice"
94
+ end
95
+ should_see_blah
96
+ end
97
+ end
98
+
99
+ context "Another context with setup block" do
100
+ setup do
101
+ @blah = "foo"
102
+ end
103
+
104
+ should "have @blah == 'foo'" do
105
+ assert_equal "foo", @blah
106
+ end
107
+
108
+ should "have name set right" do
109
+ assert_match(/^test: Another context with setup block/, self.to_s)
110
+ end
111
+ should_see_blah
112
+ end
113
+
114
+ should_eventually "pass, since it's a should_eventually" do
115
+ flunk "what?"
116
+ end
117
+
118
+ # Context creation and naming
119
+
120
+ def test_should_create_a_new_context
121
+ assert_nothing_raised do
122
+ Thoughtbot::Shoulda::Context.new("context name", self) do; end
123
+ end
124
+ end
125
+
126
+ def test_should_create_a_nested_context
127
+ assert_nothing_raised do
128
+ parent = Thoughtbot::Shoulda::Context.new("Parent", self) do; end
129
+ child = Thoughtbot::Shoulda::Context.new("Child", parent) do; end
130
+ end
131
+ end
132
+
133
+ def test_should_name_a_contexts_correctly
134
+ parent = Thoughtbot::Shoulda::Context.new("Parent", self) do; end
135
+ child = Thoughtbot::Shoulda::Context.new("Child", parent) do; end
136
+ grandchild = Thoughtbot::Shoulda::Context.new("GrandChild", child) do; end
137
+
138
+ assert_equal "Parent", parent.full_name
139
+ assert_equal "Parent Child", child.full_name
140
+ assert_equal "Parent Child GrandChild", grandchild.full_name
141
+ end
142
+
143
+ # Should statements
144
+
145
+ def test_should_have_should_hashes_when_given_should_statements
146
+ context = Thoughtbot::Shoulda::Context.new("name", self) do
147
+ should "be good" do; end
148
+ should "another" do; end
149
+ end
150
+
151
+ names = context.shoulds.map {|s| s[:name]}
152
+ assert_equal ["another", "be good"], names.sort
153
+ end
154
+
155
+ # setup and teardown
156
+
157
+ def test_should_capture_setup_and_teardown_blocks
158
+ context = Thoughtbot::Shoulda::Context.new("name", self) do
159
+ setup do; "setup"; end
160
+ teardown do; "teardown"; end
161
+ end
162
+
163
+ assert_equal "setup", context.setup_blocks.first.call
164
+ assert_equal "teardown", context.teardown_blocks.first.call
165
+ end
166
+
167
+ # building
168
+
169
+ def test_should_create_shoulda_test_for_each_should_on_build
170
+ context = Thoughtbot::Shoulda::Context.new("name", self) do
171
+ should "one" do; end
172
+ should "two" do; end
173
+ end
174
+ context.expects(:create_test_from_should_hash).with(has_entry(:name => "one"))
175
+ context.expects(:create_test_from_should_hash).with(has_entry(:name => "two"))
176
+ context.build
177
+ end
178
+
179
+ def test_should_create_test_methods_on_build
180
+ tu_class = Test::Unit::TestCase
181
+ context = Thoughtbot::Shoulda::Context.new("A Context", tu_class) do
182
+ should "define the test" do; end
183
+ end
184
+
185
+ tu_class.expects(:define_method).with(:"test: A Context should define the test. ")
186
+ context.build
187
+ end
188
+
189
+ def test_should_create_test_methods_on_build_when_subcontext
190
+ tu_class = Test::Unit::TestCase
191
+ context = Thoughtbot::Shoulda::Context.new("A Context", tu_class) do
192
+ context "with a child" do
193
+ should "define the test" do; end
194
+ end
195
+ end
196
+
197
+ tu_class.expects(:define_method).with(:"test: A Context with a child should define the test. ")
198
+ context.build
199
+ end
200
+
201
+ # Test::Unit integration
202
+
203
+ def test_should_create_a_new_context_and_build_it_on_Test_Unit_context
204
+ c = mock("context")
205
+ c.expects(:build)
206
+ Thoughtbot::Shoulda::Context.expects(:new).with("foo", kind_of(Class)).returns(c)
207
+ self.class.context "foo" do; end
208
+ end
209
+
210
+ def test_should_create_a_one_off_context_and_build_it_on_Test_Unit_should
211
+ s = mock("test")
212
+ Thoughtbot::Shoulda::Context.any_instance.expects(:should).with("rock", {}).returns(s)
213
+ Thoughtbot::Shoulda::Context.any_instance.expects(:build)
214
+ self.class.should "rock" do; end
215
+ end
216
+
217
+ def test_should_create_a_one_off_context_and_build_it_on_Test_Unit_should_eventually
218
+ s = mock("test")
219
+ Thoughtbot::Shoulda::Context.any_instance.expects(:should_eventually).with("rock").returns(s)
220
+ Thoughtbot::Shoulda::Context.any_instance.expects(:build)
221
+ self.class.should_eventually "rock" do; end
222
+ end
223
+
224
+ should "run a :before proc", :before => lambda { @value = "before" } do
225
+ assert_equal "before", @value
226
+ end
227
+
228
+ context "A :before proc" do
229
+ setup do
230
+ assert_equal "before", @value
231
+ @value = "setup"
232
+ end
233
+
234
+ should "run before the current setup", :before => lambda { @value = "before" } do
235
+ assert_equal "setup", @value
236
+ end
237
+ end
238
+
239
+ context "a before statement" do
240
+ setup do
241
+ assert_equal "before", @value
242
+ @value = "setup"
243
+ end
244
+
245
+ before_should "run before the current setup" do
246
+ @value = "before"
247
+ end
248
+ end
249
+
250
+ context "A context" do
251
+ setup do
252
+ @value = "outer"
253
+ end
254
+
255
+ context "with a subcontext and a :before proc" do
256
+ before = lambda do
257
+ assert "outer", @value
258
+ @value = "before"
259
+ end
260
+ should "run after the parent setup", :before => before do
261
+ assert_equal "before", @value
262
+ end
263
+ end
264
+ end
265
+
266
+ end
@@ -0,0 +1,25 @@
1
+ # Filters added to this controller apply to all controllers in the application.
2
+ # Likewise, all the methods added will be available for all controllers.
3
+
4
+ class ApplicationController < ActionController::Base
5
+ # Pick a unique cookie name to distinguish our session data from others'
6
+ session :session_key => '_rails_root_session_id'
7
+
8
+ def ensure_logged_in
9
+ unless session[:logged_in]
10
+ respond_to do |accepts|
11
+ accepts.html do
12
+ flash[:error] = 'What do you think you\'re doing?'
13
+ redirect_to '/'
14
+ end
15
+ accepts.xml do
16
+ headers["Status"] = "Unauthorized"
17
+ headers["WWW-Authenticate"] = %(Basic realm="Web Password")
18
+ render :text => "Couldn't authenticate you", :status => '401 Unauthorized'
19
+ end
20
+ end
21
+ return false
22
+ end
23
+ return true
24
+ end
25
+ end
@@ -0,0 +1,85 @@
1
+ class PostsController < ApplicationController
2
+ before_filter :ensure_logged_in
3
+ before_filter :load_user
4
+
5
+ def index
6
+ @posts = @user.posts
7
+
8
+ respond_to do |format|
9
+ format.html # index.rhtml
10
+ format.xml { render :xml => @posts.to_xml }
11
+ format.rss do
12
+ headers['Content-Type'] = 'application/rss+xml'
13
+ session[:special] = '$2 off your next purchase'
14
+ session[:special_user_id] = @user.id
15
+ head :ok
16
+ end
17
+ end
18
+ end
19
+
20
+ def show
21
+ @post = @user.posts.find(params[:id])
22
+
23
+ respond_to do |format|
24
+ format.html { render :layout => 'wide' }
25
+ format.xml { render :xml => @post.to_xml }
26
+ end
27
+ end
28
+
29
+ def new
30
+ @post = @user.posts.build
31
+ render :layout => false
32
+ end
33
+
34
+ def edit
35
+ @post = @user.posts.find(params[:id])
36
+ end
37
+
38
+ def create
39
+ @post = @user.posts.build(params[:post])
40
+
41
+ respond_to do |format|
42
+ if @post.save
43
+ flash[:notice] = 'Post was successfully created.'
44
+ format.html { redirect_to user_post_url(@post.user, @post) }
45
+ format.xml { head :created, :location => user_post_url(@post.user, @post) }
46
+ else
47
+ format.html { render :action => "new" }
48
+ format.xml { render :xml => @post.errors.to_xml }
49
+ end
50
+ end
51
+ end
52
+
53
+ def update
54
+ @post = @user.posts.find(params[:id])
55
+
56
+ respond_to do |format|
57
+ if @post.update_attributes(params[:post])
58
+ flash[:notice] = 'Post was successfully updated.'
59
+ format.html { redirect_to user_post_url(@post.user, @post) }
60
+ format.xml { head :ok }
61
+ else
62
+ format.html { render :action => "edit" }
63
+ format.xml { render :xml => @post.errors.to_xml }
64
+ end
65
+ end
66
+ end
67
+
68
+ def destroy
69
+ @post = @user.posts.find(params[:id])
70
+ @post.destroy
71
+
72
+ flash[:notice] = "Post was removed"
73
+
74
+ respond_to do |format|
75
+ format.html { redirect_to user_posts_url(@post.user) }
76
+ format.xml { head :ok }
77
+ end
78
+ end
79
+
80
+ private
81
+
82
+ def load_user
83
+ @user = User.find(params[:user_id])
84
+ end
85
+ end
@@ -0,0 +1,81 @@
1
+ class UsersController < ApplicationController
2
+ # GET /users
3
+ # GET /users.xml
4
+ def index
5
+ @users = User.find(:all)
6
+
7
+ respond_to do |format|
8
+ format.html # index.rhtml
9
+ format.xml { render :xml => @users.to_xml }
10
+ end
11
+ end
12
+
13
+ # GET /users/1
14
+ # GET /users/1.xml
15
+ def show
16
+ @user = User.find(params[:id])
17
+
18
+ respond_to do |format|
19
+ format.html # show.rhtml
20
+ format.xml { render :xml => @user.to_xml }
21
+ end
22
+ end
23
+
24
+ # GET /users/new
25
+ def new
26
+ @user = User.new
27
+ end
28
+
29
+ # GET /users/1;edit
30
+ def edit
31
+ @user = User.find(params[:id])
32
+ end
33
+
34
+ # POST /users
35
+ # POST /users.xml
36
+ def create
37
+ @user = User.new(params[:user])
38
+
39
+ respond_to do |format|
40
+ if @user.save
41
+ flash[:notice] = 'User was successfully created.'
42
+ format.html { redirect_to user_url(@user) }
43
+ format.xml { head :created, :location => user_url(@user) }
44
+ else
45
+ format.html { render :action => "new" }
46
+ format.xml { render :xml => @user.errors.to_xml }
47
+ end
48
+ end
49
+ end
50
+
51
+ # PUT /users/1
52
+ # PUT /users/1.xml
53
+ def update
54
+ @user = User.find(params[:id])
55
+
56
+ respond_to do |format|
57
+ if @user.update_attributes(params[:user])
58
+ flash[:notice] = 'User was successfully updated.'
59
+ format.html { redirect_to user_url(@user) }
60
+ format.xml { head :ok }
61
+ else
62
+ format.html { render :action => "edit" }
63
+ format.xml { render :xml => @user.errors.to_xml }
64
+ end
65
+ end
66
+ end
67
+
68
+ # DELETE /users/1
69
+ # DELETE /users/1.xml
70
+ def destroy
71
+ @user = User.find(params[:id])
72
+ @user.destroy
73
+
74
+ flash[:notice] = "User was removed"
75
+
76
+ respond_to do |format|
77
+ format.html { redirect_to users_url }
78
+ format.xml { head :ok }
79
+ end
80
+ end
81
+ end