robert-shoulda 2.10.3
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.
- data/CONTRIBUTION_GUIDELINES.rdoc +10 -0
- data/MIT-LICENSE +22 -0
- data/README.rdoc +171 -0
- data/Rakefile +72 -0
- data/bin/convert_to_should_syntax +42 -0
- data/lib/shoulda/action_controller/macros.rb +240 -0
- data/lib/shoulda/action_controller/matchers/assign_to_matcher.rb +109 -0
- data/lib/shoulda/action_controller/matchers/filter_param_matcher.rb +57 -0
- data/lib/shoulda/action_controller/matchers/render_with_layout_matcher.rb +81 -0
- data/lib/shoulda/action_controller/matchers/respond_with_content_type_matcher.rb +74 -0
- data/lib/shoulda/action_controller/matchers/respond_with_matcher.rb +81 -0
- data/lib/shoulda/action_controller/matchers/route_matcher.rb +93 -0
- data/lib/shoulda/action_controller/matchers/set_session_matcher.rb +87 -0
- data/lib/shoulda/action_controller/matchers/set_the_flash_matcher.rb +85 -0
- data/lib/shoulda/action_controller/matchers.rb +37 -0
- data/lib/shoulda/action_controller.rb +26 -0
- data/lib/shoulda/action_mailer/assertions.rb +38 -0
- data/lib/shoulda/action_mailer.rb +10 -0
- data/lib/shoulda/action_view/macros.rb +61 -0
- data/lib/shoulda/action_view.rb +10 -0
- data/lib/shoulda/active_record/assertions.rb +69 -0
- data/lib/shoulda/active_record/helpers.rb +27 -0
- data/lib/shoulda/active_record/macros.rb +571 -0
- data/lib/shoulda/active_record/matchers/allow_mass_assignment_of_matcher.rb +83 -0
- data/lib/shoulda/active_record/matchers/allow_value_matcher.rb +102 -0
- data/lib/shoulda/active_record/matchers/association_matcher.rb +226 -0
- data/lib/shoulda/active_record/matchers/ensure_inclusion_of_matcher.rb +87 -0
- data/lib/shoulda/active_record/matchers/ensure_length_of_matcher.rb +141 -0
- data/lib/shoulda/active_record/matchers/have_db_column_matcher.rb +169 -0
- data/lib/shoulda/active_record/matchers/have_db_index_matcher.rb +112 -0
- data/lib/shoulda/active_record/matchers/have_named_scope_matcher.rb +128 -0
- data/lib/shoulda/active_record/matchers/have_readonly_attribute_matcher.rb +59 -0
- data/lib/shoulda/active_record/matchers/validate_acceptance_of_matcher.rb +41 -0
- data/lib/shoulda/active_record/matchers/validate_format_of_matcher.rb +67 -0
- data/lib/shoulda/active_record/matchers/validate_numericality_of_matcher.rb +39 -0
- data/lib/shoulda/active_record/matchers/validate_presence_of_matcher.rb +60 -0
- data/lib/shoulda/active_record/matchers/validate_uniqueness_of_matcher.rb +148 -0
- data/lib/shoulda/active_record/matchers/validation_matcher.rb +57 -0
- data/lib/shoulda/active_record/matchers.rb +43 -0
- data/lib/shoulda/active_record.rb +16 -0
- data/lib/shoulda/assertions.rb +71 -0
- data/lib/shoulda/autoload_macros.rb +46 -0
- data/lib/shoulda/context.rb +413 -0
- data/lib/shoulda/helpers.rb +8 -0
- data/lib/shoulda/macros.rb +133 -0
- data/lib/shoulda/private_helpers.rb +13 -0
- data/lib/shoulda/proc_extensions.rb +14 -0
- data/lib/shoulda/rails.rb +13 -0
- data/lib/shoulda/rspec.rb +11 -0
- data/lib/shoulda/tasks/list_tests.rake +29 -0
- data/lib/shoulda/tasks/yaml_to_shoulda.rake +28 -0
- data/lib/shoulda/tasks.rb +3 -0
- data/lib/shoulda/test_unit.rb +22 -0
- data/lib/shoulda.rb +9 -0
- data/rails/init.rb +7 -0
- data/test/README +36 -0
- data/test/fail_macros.rb +39 -0
- data/test/fixtures/addresses.yml +3 -0
- data/test/fixtures/friendships.yml +0 -0
- data/test/fixtures/posts.yml +5 -0
- data/test/fixtures/products.yml +0 -0
- data/test/fixtures/taggings.yml +0 -0
- data/test/fixtures/tags.yml +9 -0
- data/test/fixtures/users.yml +6 -0
- data/test/functional/posts_controller_test.rb +121 -0
- data/test/functional/users_controller_test.rb +19 -0
- data/test/matchers/active_record/allow_mass_assignment_of_matcher_test.rb +68 -0
- data/test/matchers/active_record/allow_value_matcher_test.rb +64 -0
- data/test/matchers/active_record/association_matcher_test.rb +263 -0
- data/test/matchers/active_record/ensure_inclusion_of_matcher_test.rb +80 -0
- data/test/matchers/active_record/ensure_length_of_matcher_test.rb +158 -0
- data/test/matchers/active_record/have_db_column_matcher_test.rb +169 -0
- data/test/matchers/active_record/have_db_index_matcher_test.rb +91 -0
- data/test/matchers/active_record/have_named_scope_matcher_test.rb +65 -0
- data/test/matchers/active_record/have_readonly_attributes_matcher_test.rb +29 -0
- data/test/matchers/active_record/validate_acceptance_of_matcher_test.rb +44 -0
- data/test/matchers/active_record/validate_format_of_matcher_test.rb +39 -0
- data/test/matchers/active_record/validate_numericality_of_matcher_test.rb +52 -0
- data/test/matchers/active_record/validate_presence_of_matcher_test.rb +86 -0
- data/test/matchers/active_record/validate_uniqueness_of_matcher_test.rb +147 -0
- data/test/matchers/controller/assign_to_matcher_test.rb +35 -0
- data/test/matchers/controller/filter_param_matcher_test.rb +32 -0
- data/test/matchers/controller/render_with_layout_matcher_test.rb +33 -0
- data/test/matchers/controller/respond_with_content_type_matcher_test.rb +32 -0
- data/test/matchers/controller/respond_with_matcher_test.rb +106 -0
- data/test/matchers/controller/route_matcher_test.rb +75 -0
- data/test/matchers/controller/set_session_matcher_test.rb +38 -0
- data/test/matchers/controller/set_the_flash_matcher.rb +41 -0
- data/test/model_builder.rb +106 -0
- data/test/other/autoload_macro_test.rb +18 -0
- data/test/other/context_test.rb +203 -0
- data/test/other/convert_to_should_syntax_test.rb +63 -0
- data/test/other/helpers_test.rb +340 -0
- data/test/other/private_helpers_test.rb +32 -0
- data/test/other/should_test.rb +271 -0
- data/test/rails_root/app/controllers/application_controller.rb +25 -0
- data/test/rails_root/app/controllers/posts_controller.rb +87 -0
- data/test/rails_root/app/controllers/users_controller.rb +84 -0
- data/test/rails_root/app/helpers/application_helper.rb +3 -0
- data/test/rails_root/app/helpers/posts_helper.rb +2 -0
- data/test/rails_root/app/helpers/users_helper.rb +2 -0
- data/test/rails_root/app/models/address.rb +7 -0
- data/test/rails_root/app/models/flea.rb +3 -0
- data/test/rails_root/app/models/friendship.rb +4 -0
- data/test/rails_root/app/models/pets/cat.rb +7 -0
- data/test/rails_root/app/models/pets/dog.rb +10 -0
- data/test/rails_root/app/models/post.rb +12 -0
- data/test/rails_root/app/models/product.rb +12 -0
- data/test/rails_root/app/models/profile.rb +2 -0
- data/test/rails_root/app/models/registration.rb +2 -0
- data/test/rails_root/app/models/tag.rb +8 -0
- data/test/rails_root/app/models/tagging.rb +4 -0
- data/test/rails_root/app/models/treat.rb +3 -0
- data/test/rails_root/app/models/user.rb +32 -0
- data/test/rails_root/app/views/layouts/posts.rhtml +19 -0
- data/test/rails_root/app/views/layouts/users.rhtml +17 -0
- data/test/rails_root/app/views/layouts/wide.html.erb +1 -0
- data/test/rails_root/app/views/posts/edit.rhtml +27 -0
- data/test/rails_root/app/views/posts/index.rhtml +25 -0
- data/test/rails_root/app/views/posts/new.rhtml +26 -0
- data/test/rails_root/app/views/posts/show.rhtml +18 -0
- data/test/rails_root/app/views/users/edit.rhtml +22 -0
- data/test/rails_root/app/views/users/index.rhtml +22 -0
- data/test/rails_root/app/views/users/new.rhtml +21 -0
- data/test/rails_root/app/views/users/show.rhtml +13 -0
- data/test/rails_root/config/boot.rb +110 -0
- data/test/rails_root/config/database.yml +4 -0
- data/test/rails_root/config/environment.rb +18 -0
- data/test/rails_root/config/environments/test.rb +0 -0
- data/test/rails_root/config/initializers/new_rails_defaults.rb +15 -0
- data/test/rails_root/config/initializers/shoulda.rb +8 -0
- data/test/rails_root/config/routes.rb +6 -0
- data/test/rails_root/db/migrate/001_create_users.rb +19 -0
- data/test/rails_root/db/migrate/002_create_posts.rb +13 -0
- data/test/rails_root/db/migrate/003_create_taggings.rb +12 -0
- data/test/rails_root/db/migrate/004_create_tags.rb +11 -0
- data/test/rails_root/db/migrate/005_create_dogs.rb +12 -0
- data/test/rails_root/db/migrate/006_create_addresses.rb +14 -0
- data/test/rails_root/db/migrate/007_create_fleas.rb +11 -0
- data/test/rails_root/db/migrate/008_create_dogs_fleas.rb +12 -0
- data/test/rails_root/db/migrate/009_create_products.rb +17 -0
- data/test/rails_root/db/migrate/010_create_friendships.rb +14 -0
- data/test/rails_root/db/migrate/011_create_treats.rb +12 -0
- data/test/rails_root/db/migrate/20090506203502_create_profiles.rb +12 -0
- data/test/rails_root/db/migrate/20090506203536_create_registrations.rb +14 -0
- data/test/rails_root/db/migrate/20090513104502_create_cats.rb +12 -0
- data/test/rails_root/db/schema.rb +0 -0
- data/test/rails_root/log/test.log +8963 -0
- data/test/rails_root/public/404.html +30 -0
- data/test/rails_root/public/422.html +30 -0
- data/test/rails_root/public/500.html +30 -0
- data/test/rails_root/script/console +3 -0
- data/test/rails_root/script/generate +3 -0
- data/test/rails_root/test/shoulda_macros/custom_macro.rb +6 -0
- data/test/rails_root/vendor/gems/gem_with_macro-0.0.1/shoulda_macros/gem_macro.rb +6 -0
- data/test/rails_root/vendor/plugins/plugin_with_macro/shoulda_macros/plugin_macro.rb +6 -0
- data/test/rspec_test.rb +207 -0
- data/test/test_helper.rb +28 -0
- data/test/unit/address_test.rb +15 -0
- data/test/unit/cat_test.rb +7 -0
- data/test/unit/dog_test.rb +9 -0
- data/test/unit/flea_test.rb +6 -0
- data/test/unit/friendship_test.rb +6 -0
- data/test/unit/post_test.rb +19 -0
- data/test/unit/product_test.rb +23 -0
- data/test/unit/tag_test.rb +15 -0
- data/test/unit/tagging_test.rb +6 -0
- data/test/unit/user_test.rb +80 -0
- metadata +225 -0
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
module Shoulda # :nodoc:
|
|
2
|
+
module ActionController # :nodoc:
|
|
3
|
+
module Matchers
|
|
4
|
+
|
|
5
|
+
# Ensures that filter_parameter_logging is set for the specified key.
|
|
6
|
+
#
|
|
7
|
+
# Example:
|
|
8
|
+
#
|
|
9
|
+
# it { should filter_param(:password) }
|
|
10
|
+
def filter_param(key)
|
|
11
|
+
FilterParamMatcher.new(key)
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
class FilterParamMatcher # :nodoc:
|
|
15
|
+
|
|
16
|
+
def initialize(key)
|
|
17
|
+
@key = key.to_s
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
def matches?(controller)
|
|
21
|
+
@controller = controller
|
|
22
|
+
filters_params? && filters_key?
|
|
23
|
+
end
|
|
24
|
+
|
|
25
|
+
def failure_message
|
|
26
|
+
"Expected #{@key} to be filtered"
|
|
27
|
+
end
|
|
28
|
+
|
|
29
|
+
def negative_failure_message
|
|
30
|
+
"Did not expect #{@key} to be filtered"
|
|
31
|
+
end
|
|
32
|
+
|
|
33
|
+
def description
|
|
34
|
+
"filter #{@key}"
|
|
35
|
+
end
|
|
36
|
+
|
|
37
|
+
private
|
|
38
|
+
|
|
39
|
+
def filters_params?
|
|
40
|
+
@controller.respond_to?(:filter_parameters)
|
|
41
|
+
end
|
|
42
|
+
|
|
43
|
+
def filters_key?
|
|
44
|
+
filtered_value == '[FILTERED]'
|
|
45
|
+
end
|
|
46
|
+
|
|
47
|
+
def filtered_value
|
|
48
|
+
filtered = @controller.send(:filter_parameters,
|
|
49
|
+
@key.to_s => @key.to_s)
|
|
50
|
+
filtered[@key.to_s]
|
|
51
|
+
end
|
|
52
|
+
|
|
53
|
+
end
|
|
54
|
+
|
|
55
|
+
end
|
|
56
|
+
end
|
|
57
|
+
end
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
module Shoulda # :nodoc:
|
|
2
|
+
module ActionController # :nodoc:
|
|
3
|
+
module Matchers
|
|
4
|
+
|
|
5
|
+
# Ensures that the controller rendered with the given layout.
|
|
6
|
+
#
|
|
7
|
+
# Example:
|
|
8
|
+
#
|
|
9
|
+
# it { should render_with_layout }
|
|
10
|
+
# it { should render_with_layout(:special) }
|
|
11
|
+
# it { should_not render_with_layout }
|
|
12
|
+
def render_with_layout(layout = nil)
|
|
13
|
+
RenderWithLayout.new(layout)
|
|
14
|
+
end
|
|
15
|
+
|
|
16
|
+
class RenderWithLayout # :nodoc:
|
|
17
|
+
|
|
18
|
+
def initialize(layout)
|
|
19
|
+
@layout = layout.to_s unless layout.nil?
|
|
20
|
+
end
|
|
21
|
+
|
|
22
|
+
def matches?(controller)
|
|
23
|
+
@controller = controller
|
|
24
|
+
rendered_with_layout? && rendered_with_expected_layout?
|
|
25
|
+
end
|
|
26
|
+
|
|
27
|
+
def failure_message
|
|
28
|
+
"Expected #{expectation}, but #{result}"
|
|
29
|
+
end
|
|
30
|
+
|
|
31
|
+
def negative_failure_message
|
|
32
|
+
"Did not expect #{expectation}, but #{result}"
|
|
33
|
+
end
|
|
34
|
+
|
|
35
|
+
def description
|
|
36
|
+
description = "render with "
|
|
37
|
+
if @layout.nil?
|
|
38
|
+
description << "a layout"
|
|
39
|
+
else
|
|
40
|
+
description << "the #{@layout.inspect} layout"
|
|
41
|
+
end
|
|
42
|
+
description
|
|
43
|
+
end
|
|
44
|
+
|
|
45
|
+
private
|
|
46
|
+
|
|
47
|
+
def rendered_with_layout?
|
|
48
|
+
!layout.blank?
|
|
49
|
+
end
|
|
50
|
+
|
|
51
|
+
def rendered_with_expected_layout?
|
|
52
|
+
return true if @layout.nil?
|
|
53
|
+
layout == @layout
|
|
54
|
+
end
|
|
55
|
+
|
|
56
|
+
def layout
|
|
57
|
+
layout = @controller.response.layout
|
|
58
|
+
if layout.nil?
|
|
59
|
+
nil
|
|
60
|
+
else
|
|
61
|
+
layout.split('/').last
|
|
62
|
+
end
|
|
63
|
+
end
|
|
64
|
+
|
|
65
|
+
def expectation
|
|
66
|
+
"to #{description}"
|
|
67
|
+
end
|
|
68
|
+
|
|
69
|
+
def result
|
|
70
|
+
if rendered_with_layout?
|
|
71
|
+
"rendered with the #{layout.inspect} layout"
|
|
72
|
+
else
|
|
73
|
+
"rendered without a layout"
|
|
74
|
+
end
|
|
75
|
+
end
|
|
76
|
+
|
|
77
|
+
end
|
|
78
|
+
|
|
79
|
+
end
|
|
80
|
+
end
|
|
81
|
+
end
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
module Shoulda # :nodoc:
|
|
2
|
+
module ActionController # :nodoc:
|
|
3
|
+
module Matchers
|
|
4
|
+
|
|
5
|
+
# Ensures a controller responded with expected 'response' content type.
|
|
6
|
+
#
|
|
7
|
+
# You can pass an explicit content type such as 'application/rss+xml'
|
|
8
|
+
# or its symbolic equivalent :rss
|
|
9
|
+
# or a regular expression such as /rss/
|
|
10
|
+
#
|
|
11
|
+
# Example:
|
|
12
|
+
#
|
|
13
|
+
# it { should respond_with_content_type(:xml) }
|
|
14
|
+
# it { should respond_with_content_type(:csv) }
|
|
15
|
+
# it { should respond_with_content_type(:atom) }
|
|
16
|
+
# it { should respond_with_content_type(:yaml) }
|
|
17
|
+
# it { should respond_with_content_type(:text) }
|
|
18
|
+
# it { should respond_with_content_type('application/rss+xml') }
|
|
19
|
+
# it { should respond_with_content_type(/json/) }
|
|
20
|
+
def respond_with_content_type(content_type)
|
|
21
|
+
RespondWithContentTypeMatcher.new(content_type)
|
|
22
|
+
end
|
|
23
|
+
|
|
24
|
+
class RespondWithContentTypeMatcher # :nodoc:
|
|
25
|
+
|
|
26
|
+
def initialize(content_type)
|
|
27
|
+
@content_type = if content_type.is_a?(Symbol)
|
|
28
|
+
lookup_by_extension(content_type)
|
|
29
|
+
else
|
|
30
|
+
content_type
|
|
31
|
+
end
|
|
32
|
+
end
|
|
33
|
+
|
|
34
|
+
def description
|
|
35
|
+
"respond with content type of #{@content_type}"
|
|
36
|
+
end
|
|
37
|
+
|
|
38
|
+
def matches?(controller)
|
|
39
|
+
@controller = controller
|
|
40
|
+
if @content_type.is_a?(Regexp)
|
|
41
|
+
response_content_type =~ @content_type
|
|
42
|
+
else
|
|
43
|
+
response_content_type == @content_type
|
|
44
|
+
end
|
|
45
|
+
end
|
|
46
|
+
|
|
47
|
+
def failure_message
|
|
48
|
+
"Expected #{expectation}"
|
|
49
|
+
end
|
|
50
|
+
|
|
51
|
+
def negative_failure_message
|
|
52
|
+
"Did not expect #{expectation}"
|
|
53
|
+
end
|
|
54
|
+
|
|
55
|
+
protected
|
|
56
|
+
|
|
57
|
+
def response_content_type
|
|
58
|
+
@controller.response.content_type
|
|
59
|
+
end
|
|
60
|
+
|
|
61
|
+
def lookup_by_extension(extension)
|
|
62
|
+
Mime::Type.lookup_by_extension(extension.to_s).to_s
|
|
63
|
+
end
|
|
64
|
+
|
|
65
|
+
def expectation
|
|
66
|
+
"content type to be #{@content_type}, " <<
|
|
67
|
+
"but was #{response_content_type}"
|
|
68
|
+
end
|
|
69
|
+
|
|
70
|
+
end
|
|
71
|
+
|
|
72
|
+
end
|
|
73
|
+
end
|
|
74
|
+
end
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
module Shoulda # :nodoc:
|
|
2
|
+
module ActionController # :nodoc:
|
|
3
|
+
module Matchers
|
|
4
|
+
|
|
5
|
+
# Ensures a controller responded with expected 'response' status code.
|
|
6
|
+
#
|
|
7
|
+
# You can pass an explicit status number like 200, 301, 404, 500
|
|
8
|
+
# or its symbolic equivalent :success, :redirect, :missing, :error.
|
|
9
|
+
# See ActionController::StatusCodes for a full list.
|
|
10
|
+
#
|
|
11
|
+
# Example:
|
|
12
|
+
#
|
|
13
|
+
# it { should respond_with(:success) }
|
|
14
|
+
# it { should respond_with(:redirect) }
|
|
15
|
+
# it { should respond_with(:missing) }
|
|
16
|
+
# it { should respond_with(:error) }
|
|
17
|
+
# it { should respond_with(501) }
|
|
18
|
+
def respond_with(status)
|
|
19
|
+
RespondWithMatcher.new(status)
|
|
20
|
+
end
|
|
21
|
+
|
|
22
|
+
class RespondWithMatcher # :nodoc:
|
|
23
|
+
|
|
24
|
+
def initialize(status)
|
|
25
|
+
@status = symbol_to_status_code(status)
|
|
26
|
+
end
|
|
27
|
+
|
|
28
|
+
def matches?(controller)
|
|
29
|
+
@controller = controller
|
|
30
|
+
correct_status_code? || correct_status_code_range?
|
|
31
|
+
end
|
|
32
|
+
|
|
33
|
+
def failure_message
|
|
34
|
+
"Expected #{expectation}"
|
|
35
|
+
end
|
|
36
|
+
|
|
37
|
+
def negative_failure_message
|
|
38
|
+
"Did not expect #{expectation}"
|
|
39
|
+
end
|
|
40
|
+
|
|
41
|
+
def description
|
|
42
|
+
"respond with #{@status}"
|
|
43
|
+
end
|
|
44
|
+
|
|
45
|
+
protected
|
|
46
|
+
|
|
47
|
+
def correct_status_code?
|
|
48
|
+
response_code == @status
|
|
49
|
+
end
|
|
50
|
+
|
|
51
|
+
def correct_status_code_range?
|
|
52
|
+
@status.is_a?(Range) &&
|
|
53
|
+
@status.include?(response_code)
|
|
54
|
+
end
|
|
55
|
+
|
|
56
|
+
def response_code
|
|
57
|
+
@controller.response.response_code
|
|
58
|
+
end
|
|
59
|
+
|
|
60
|
+
def symbol_to_status_code(potential_symbol)
|
|
61
|
+
case potential_symbol
|
|
62
|
+
when :success then 200
|
|
63
|
+
when :redirect then 300..399
|
|
64
|
+
when :missing then 404
|
|
65
|
+
when :error then 500..599
|
|
66
|
+
when Symbol
|
|
67
|
+
::ActionController::StatusCodes::SYMBOL_TO_STATUS_CODE[potential_symbol]
|
|
68
|
+
else
|
|
69
|
+
potential_symbol
|
|
70
|
+
end
|
|
71
|
+
end
|
|
72
|
+
|
|
73
|
+
def expectation
|
|
74
|
+
"response to be a #{@status}, but was #{response_code}"
|
|
75
|
+
end
|
|
76
|
+
|
|
77
|
+
end
|
|
78
|
+
|
|
79
|
+
end
|
|
80
|
+
end
|
|
81
|
+
end
|
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
module Shoulda # :nodoc:
|
|
2
|
+
module ActionController # :nodoc:
|
|
3
|
+
module Matchers
|
|
4
|
+
|
|
5
|
+
# Ensures that requesting +path+ using +method+ routes to +options+.
|
|
6
|
+
#
|
|
7
|
+
# If you don't specify a controller, it will use the controller from the
|
|
8
|
+
# example group.
|
|
9
|
+
#
|
|
10
|
+
# +to_param+ is called on the +options+ given.
|
|
11
|
+
#
|
|
12
|
+
# Examples:
|
|
13
|
+
#
|
|
14
|
+
# it { should route(:get, "/posts").
|
|
15
|
+
# to(:controller => :posts, :action => :index) }
|
|
16
|
+
# it { should route(:get, "/posts/new").to(:action => :new) }
|
|
17
|
+
# it { should route(:post, "/posts").to(:action => :create) }
|
|
18
|
+
# it { should route(:get, "/posts/1").to(:action => :show, :id => 1) }
|
|
19
|
+
# it { should route(:edit, "/posts/1").to(:action => :show, :id => 1) }
|
|
20
|
+
# it { should route(:put, "/posts/1").to(:action => :update, :id => 1) }
|
|
21
|
+
# it { should route(:delete, "/posts/1").
|
|
22
|
+
# to(:action => :destroy, :id => 1) }
|
|
23
|
+
# it { should route(:get, "/users/1/posts/1").
|
|
24
|
+
# to(:action => :show, :id => 1, :user_id => 1) }
|
|
25
|
+
def route(method, path)
|
|
26
|
+
RouteMatcher.new(method, path, self)
|
|
27
|
+
end
|
|
28
|
+
|
|
29
|
+
class RouteMatcher # :nodoc:
|
|
30
|
+
|
|
31
|
+
def initialize(method, path, context)
|
|
32
|
+
@method = method
|
|
33
|
+
@path = path
|
|
34
|
+
@context = context
|
|
35
|
+
end
|
|
36
|
+
|
|
37
|
+
def to(params)
|
|
38
|
+
@params = params
|
|
39
|
+
self
|
|
40
|
+
end
|
|
41
|
+
|
|
42
|
+
def in_context(context)
|
|
43
|
+
@context = context
|
|
44
|
+
self
|
|
45
|
+
end
|
|
46
|
+
|
|
47
|
+
def matches?(controller)
|
|
48
|
+
@controller = controller
|
|
49
|
+
guess_controller!
|
|
50
|
+
stringify_params!
|
|
51
|
+
route_recognized?
|
|
52
|
+
end
|
|
53
|
+
|
|
54
|
+
attr_reader :failure_message, :negative_failure_message
|
|
55
|
+
|
|
56
|
+
def description
|
|
57
|
+
"route #{@method.to_s.upcase} #{@path} to/from #{@params.inspect}"
|
|
58
|
+
end
|
|
59
|
+
|
|
60
|
+
private
|
|
61
|
+
|
|
62
|
+
def guess_controller!
|
|
63
|
+
@params[:controller] ||= @controller.controller_path
|
|
64
|
+
end
|
|
65
|
+
|
|
66
|
+
def stringify_params!
|
|
67
|
+
@params.each do |key, value|
|
|
68
|
+
@params[key] = value.is_a?(Array) ? value.collect {|v| v.to_param } : value.to_param
|
|
69
|
+
end
|
|
70
|
+
end
|
|
71
|
+
|
|
72
|
+
def route_recognized?
|
|
73
|
+
begin
|
|
74
|
+
@context.send(:assert_routing,
|
|
75
|
+
{ :method => @method, :path => @path },
|
|
76
|
+
@params)
|
|
77
|
+
|
|
78
|
+
@negative_failure_message = "Didn't expect to #{description}"
|
|
79
|
+
true
|
|
80
|
+
rescue ::ActionController::RoutingError => error
|
|
81
|
+
@failure_message = error.message
|
|
82
|
+
false
|
|
83
|
+
rescue Test::Unit::AssertionFailedError => error
|
|
84
|
+
@failure_message = error.message
|
|
85
|
+
false
|
|
86
|
+
end
|
|
87
|
+
end
|
|
88
|
+
|
|
89
|
+
end
|
|
90
|
+
|
|
91
|
+
end
|
|
92
|
+
end
|
|
93
|
+
end
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
module Shoulda # :nodoc:
|
|
2
|
+
module ActionController # :nodoc:
|
|
3
|
+
module Matchers
|
|
4
|
+
|
|
5
|
+
# Ensures that a session key was set to the expected value.
|
|
6
|
+
#
|
|
7
|
+
# Example:
|
|
8
|
+
#
|
|
9
|
+
# it { should set_session(:message) }
|
|
10
|
+
# it { should set_session(:user_id).to(@user.id) }
|
|
11
|
+
# it { should_not set_session(:user_id) }
|
|
12
|
+
def set_session(key)
|
|
13
|
+
SetSessionMatcher.new(key)
|
|
14
|
+
end
|
|
15
|
+
|
|
16
|
+
class SetSessionMatcher # :nodoc:
|
|
17
|
+
|
|
18
|
+
def initialize(key)
|
|
19
|
+
@key = key.to_s
|
|
20
|
+
end
|
|
21
|
+
|
|
22
|
+
def to(value)
|
|
23
|
+
@value = value
|
|
24
|
+
self
|
|
25
|
+
end
|
|
26
|
+
|
|
27
|
+
def matches?(controller)
|
|
28
|
+
@controller = controller
|
|
29
|
+
(assigned_value? && assigned_correct_value?) || cleared_value?
|
|
30
|
+
end
|
|
31
|
+
|
|
32
|
+
def failure_message
|
|
33
|
+
"Expected #{expectation}, but #{result}"
|
|
34
|
+
end
|
|
35
|
+
|
|
36
|
+
def negative_failure_message
|
|
37
|
+
"Didn't expect #{expectation}, but #{result}"
|
|
38
|
+
end
|
|
39
|
+
|
|
40
|
+
def description
|
|
41
|
+
description = "set session variable #{@key.inspect}"
|
|
42
|
+
description << " to #{@value.inspect}" if defined?(@value)
|
|
43
|
+
description
|
|
44
|
+
end
|
|
45
|
+
|
|
46
|
+
private
|
|
47
|
+
|
|
48
|
+
def assigned_value?
|
|
49
|
+
!assigned_value.nil?
|
|
50
|
+
end
|
|
51
|
+
|
|
52
|
+
def cleared_value?
|
|
53
|
+
defined?(@value) && @value.nil? && assigned_value.nil?
|
|
54
|
+
end
|
|
55
|
+
|
|
56
|
+
def assigned_correct_value?
|
|
57
|
+
return true if @value.nil?
|
|
58
|
+
assigned_value == @value
|
|
59
|
+
end
|
|
60
|
+
|
|
61
|
+
def assigned_value
|
|
62
|
+
session[@key]
|
|
63
|
+
end
|
|
64
|
+
|
|
65
|
+
def session
|
|
66
|
+
@controller.response.session.data
|
|
67
|
+
end
|
|
68
|
+
|
|
69
|
+
def expectation
|
|
70
|
+
expectation = "session variable #{@key} to be set"
|
|
71
|
+
expectation << " to #{@value.inspect}" if @value
|
|
72
|
+
expectation
|
|
73
|
+
end
|
|
74
|
+
|
|
75
|
+
def result
|
|
76
|
+
if session.empty?
|
|
77
|
+
"no session variables were set"
|
|
78
|
+
else
|
|
79
|
+
"the session was #{session.inspect}"
|
|
80
|
+
end
|
|
81
|
+
end
|
|
82
|
+
|
|
83
|
+
end
|
|
84
|
+
|
|
85
|
+
end
|
|
86
|
+
end
|
|
87
|
+
end
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
module Shoulda # :nodoc:
|
|
2
|
+
module ActionController # :nodoc:
|
|
3
|
+
module Matchers
|
|
4
|
+
|
|
5
|
+
# Ensures that the flash contains the given value. Can be a String, a
|
|
6
|
+
# Regexp, or nil (indicating that the flash should not be set).
|
|
7
|
+
#
|
|
8
|
+
# Example:
|
|
9
|
+
#
|
|
10
|
+
# it { should set_the_flash }
|
|
11
|
+
# it { should set_the_flash.to("Thank you for placing this order.") }
|
|
12
|
+
# it { should set_the_flash.to(/created/i) }
|
|
13
|
+
# it { should_not set_the_flash }
|
|
14
|
+
def set_the_flash
|
|
15
|
+
SetTheFlashMatcher.new
|
|
16
|
+
end
|
|
17
|
+
|
|
18
|
+
class SetTheFlashMatcher # :nodoc:
|
|
19
|
+
|
|
20
|
+
def to(value)
|
|
21
|
+
@value = value
|
|
22
|
+
self
|
|
23
|
+
end
|
|
24
|
+
|
|
25
|
+
def matches?(controller)
|
|
26
|
+
@controller = controller
|
|
27
|
+
sets_the_flash? && string_value_matches? && regexp_value_matches?
|
|
28
|
+
end
|
|
29
|
+
|
|
30
|
+
attr_reader :failure_message, :negative_failure_message
|
|
31
|
+
|
|
32
|
+
def description
|
|
33
|
+
description = "set the flash"
|
|
34
|
+
description << " to #{@value.inspect}" unless @value.nil?
|
|
35
|
+
description
|
|
36
|
+
end
|
|
37
|
+
|
|
38
|
+
def failure_message
|
|
39
|
+
"Expected #{expectation}"
|
|
40
|
+
end
|
|
41
|
+
|
|
42
|
+
def negative_failure_message
|
|
43
|
+
"Did not expect #{expectation}"
|
|
44
|
+
end
|
|
45
|
+
|
|
46
|
+
private
|
|
47
|
+
|
|
48
|
+
def sets_the_flash?
|
|
49
|
+
!flash.blank?
|
|
50
|
+
end
|
|
51
|
+
|
|
52
|
+
def string_value_matches?
|
|
53
|
+
return true unless String === @value
|
|
54
|
+
flash.values.any? {|value| value == @value }
|
|
55
|
+
end
|
|
56
|
+
|
|
57
|
+
def regexp_value_matches?
|
|
58
|
+
return true unless Regexp === @value
|
|
59
|
+
flash.values.any? {|value| value =~ @value }
|
|
60
|
+
end
|
|
61
|
+
|
|
62
|
+
def flash
|
|
63
|
+
@controller.response.session['flash']
|
|
64
|
+
end
|
|
65
|
+
|
|
66
|
+
def expectation
|
|
67
|
+
expectation = "the flash to be set"
|
|
68
|
+
expectation << " to #{@value.inspect}" unless @value.nil?
|
|
69
|
+
expectation << ", but #{flash_description}"
|
|
70
|
+
expectation
|
|
71
|
+
end
|
|
72
|
+
|
|
73
|
+
def flash_description
|
|
74
|
+
if flash.blank?
|
|
75
|
+
"no flash was set"
|
|
76
|
+
else
|
|
77
|
+
"was #{flash.inspect}"
|
|
78
|
+
end
|
|
79
|
+
end
|
|
80
|
+
|
|
81
|
+
end
|
|
82
|
+
|
|
83
|
+
end
|
|
84
|
+
end
|
|
85
|
+
end
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
require 'shoulda/action_controller/matchers/assign_to_matcher'
|
|
2
|
+
require 'shoulda/action_controller/matchers/filter_param_matcher'
|
|
3
|
+
require 'shoulda/action_controller/matchers/set_the_flash_matcher'
|
|
4
|
+
require 'shoulda/action_controller/matchers/render_with_layout_matcher'
|
|
5
|
+
require 'shoulda/action_controller/matchers/respond_with_matcher'
|
|
6
|
+
require 'shoulda/action_controller/matchers/respond_with_content_type_matcher'
|
|
7
|
+
require 'shoulda/action_controller/matchers/set_session_matcher'
|
|
8
|
+
require 'shoulda/action_controller/matchers/route_matcher'
|
|
9
|
+
|
|
10
|
+
module Shoulda # :nodoc:
|
|
11
|
+
module ActionController # :nodoc:
|
|
12
|
+
|
|
13
|
+
# By using the macro helpers you can quickly and easily create concise and
|
|
14
|
+
# easy to read test suites.
|
|
15
|
+
#
|
|
16
|
+
# This code segment:
|
|
17
|
+
#
|
|
18
|
+
# describe UsersController, "on GET to show with a valid id" do
|
|
19
|
+
# before(:each) do
|
|
20
|
+
# get :show, :id => User.first.to_param
|
|
21
|
+
# end
|
|
22
|
+
#
|
|
23
|
+
# it { should assign_to(:user) }
|
|
24
|
+
# it { should respond_with(:success) }
|
|
25
|
+
# it { should render_template(:show) }
|
|
26
|
+
# it { should not_set_the_flash) }
|
|
27
|
+
#
|
|
28
|
+
# it "should do something else really cool" do
|
|
29
|
+
# assigns[:user].id.should == 1
|
|
30
|
+
# end
|
|
31
|
+
# end
|
|
32
|
+
#
|
|
33
|
+
# Would produce 5 tests for the show action
|
|
34
|
+
module Matchers
|
|
35
|
+
end
|
|
36
|
+
end
|
|
37
|
+
end
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
require 'shoulda'
|
|
2
|
+
require 'shoulda/action_controller/matchers'
|
|
3
|
+
require 'shoulda/action_controller/macros'
|
|
4
|
+
|
|
5
|
+
module Test # :nodoc: all
|
|
6
|
+
module Unit
|
|
7
|
+
class TestCase
|
|
8
|
+
include Shoulda::ActionController::Matchers
|
|
9
|
+
extend Shoulda::ActionController::Macros
|
|
10
|
+
end
|
|
11
|
+
end
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
require 'shoulda/active_record/assertions'
|
|
15
|
+
require 'shoulda/action_mailer/assertions'
|
|
16
|
+
|
|
17
|
+
module ActionController #:nodoc: all
|
|
18
|
+
module Integration
|
|
19
|
+
class Session
|
|
20
|
+
include Shoulda::Assertions
|
|
21
|
+
include Shoulda::Helpers
|
|
22
|
+
include Shoulda::ActiveRecord::Assertions
|
|
23
|
+
include Shoulda::ActionMailer::Assertions
|
|
24
|
+
end
|
|
25
|
+
end
|
|
26
|
+
end
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
module Shoulda # :nodoc:
|
|
2
|
+
module ActionMailer # :nodoc:
|
|
3
|
+
module Assertions
|
|
4
|
+
# Asserts that an email was delivered. Can take a block that can further
|
|
5
|
+
# narrow down the types of emails you're expecting.
|
|
6
|
+
#
|
|
7
|
+
# assert_sent_email
|
|
8
|
+
#
|
|
9
|
+
# Passes if ActionMailer::Base.deliveries has an email
|
|
10
|
+
#
|
|
11
|
+
# assert_sent_email do |email|
|
|
12
|
+
# email.subject =~ /hi there/ && email.to.include?('none@none.com')
|
|
13
|
+
# end
|
|
14
|
+
#
|
|
15
|
+
# Passes if there is an email with subject containing 'hi there' and
|
|
16
|
+
# 'none@none.com' as one of the recipients.
|
|
17
|
+
#
|
|
18
|
+
def assert_sent_email
|
|
19
|
+
emails = ::ActionMailer::Base.deliveries
|
|
20
|
+
assert !emails.empty?, "No emails were sent"
|
|
21
|
+
if block_given?
|
|
22
|
+
matching_emails = emails.select {|email| yield email }
|
|
23
|
+
assert !matching_emails.empty?, "None of the emails matched."
|
|
24
|
+
end
|
|
25
|
+
end
|
|
26
|
+
|
|
27
|
+
# Asserts that no ActionMailer mails were delivered
|
|
28
|
+
#
|
|
29
|
+
# assert_did_not_send_email
|
|
30
|
+
def assert_did_not_send_email
|
|
31
|
+
msg = "Sent #{::ActionMailer::Base.deliveries.size} emails.\n"
|
|
32
|
+
::ActionMailer::Base.deliveries.each { |m| msg << " '#{m.subject}' sent to #{m.to.to_sentence}\n" }
|
|
33
|
+
assert ::ActionMailer::Base.deliveries.empty?, msg
|
|
34
|
+
end
|
|
35
|
+
end
|
|
36
|
+
end
|
|
37
|
+
end
|
|
38
|
+
|