mmurphy-webrat 0.4.3 → 0.4.4

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 (65) hide show
  1. data/lib/webrat.rb +34 -0
  2. data/lib/webrat/core.rb +14 -0
  3. data/lib/webrat/core/configuration.rb +98 -0
  4. data/lib/webrat/core/elements/area.rb +31 -0
  5. data/lib/webrat/core/elements/element.rb +33 -0
  6. data/lib/webrat/core/elements/field.rb +403 -0
  7. data/lib/webrat/core/elements/form.rb +103 -0
  8. data/lib/webrat/core/elements/label.rb +31 -0
  9. data/lib/webrat/core/elements/link.rb +90 -0
  10. data/lib/webrat/core/elements/select_option.rb +35 -0
  11. data/lib/webrat/core/locators.rb +20 -0
  12. data/lib/webrat/core/locators/area_locator.rb +38 -0
  13. data/lib/webrat/core/locators/button_locator.rb +54 -0
  14. data/lib/webrat/core/locators/field_by_id_locator.rb +37 -0
  15. data/lib/webrat/core/locators/field_labeled_locator.rb +56 -0
  16. data/lib/webrat/core/locators/field_locator.rb +25 -0
  17. data/lib/webrat/core/locators/field_named_locator.rb +41 -0
  18. data/lib/webrat/core/locators/form_locator.rb +19 -0
  19. data/lib/webrat/core/locators/label_locator.rb +34 -0
  20. data/lib/webrat/core/locators/link_locator.rb +66 -0
  21. data/lib/webrat/core/locators/locator.rb +20 -0
  22. data/lib/webrat/core/locators/select_option_locator.rb +59 -0
  23. data/lib/webrat/core/logging.rb +21 -0
  24. data/lib/webrat/core/matchers.rb +4 -0
  25. data/lib/webrat/core/matchers/have_content.rb +73 -0
  26. data/lib/webrat/core/matchers/have_selector.rb +74 -0
  27. data/lib/webrat/core/matchers/have_tag.rb +21 -0
  28. data/lib/webrat/core/matchers/have_xpath.rb +147 -0
  29. data/lib/webrat/core/methods.rb +61 -0
  30. data/lib/webrat/core/mime.rb +29 -0
  31. data/lib/webrat/core/save_and_open_page.rb +50 -0
  32. data/lib/webrat/core/scope.rb +350 -0
  33. data/lib/webrat/core/session.rb +281 -0
  34. data/lib/webrat/core/xml.rb +115 -0
  35. data/lib/webrat/core/xml/hpricot.rb +19 -0
  36. data/lib/webrat/core/xml/nokogiri.rb +76 -0
  37. data/lib/webrat/core/xml/rexml.rb +24 -0
  38. data/lib/webrat/core_extensions/blank.rb +58 -0
  39. data/lib/webrat/core_extensions/deprecate.rb +8 -0
  40. data/lib/webrat/core_extensions/detect_mapped.rb +12 -0
  41. data/lib/webrat/core_extensions/meta_class.rb +6 -0
  42. data/lib/webrat/core_extensions/nil_to_param.rb +5 -0
  43. data/lib/webrat/mechanize.rb +74 -0
  44. data/lib/webrat/merb.rb +9 -0
  45. data/lib/webrat/merb_session.rb +65 -0
  46. data/lib/webrat/rack.rb +24 -0
  47. data/lib/webrat/rails.rb +105 -0
  48. data/lib/webrat/rspec-rails.rb +13 -0
  49. data/lib/webrat/selenium.rb +154 -0
  50. data/lib/webrat/selenium/location_strategy_javascript/button.js +12 -0
  51. data/lib/webrat/selenium/location_strategy_javascript/label.js +16 -0
  52. data/lib/webrat/selenium/location_strategy_javascript/webrat.js +5 -0
  53. data/lib/webrat/selenium/location_strategy_javascript/webratlink.js +9 -0
  54. data/lib/webrat/selenium/location_strategy_javascript/webratlinkwithin.js +15 -0
  55. data/lib/webrat/selenium/location_strategy_javascript/webratselectwithoption.js +5 -0
  56. data/lib/webrat/selenium/matchers.rb +4 -0
  57. data/lib/webrat/selenium/matchers/have_content.rb +66 -0
  58. data/lib/webrat/selenium/matchers/have_selector.rb +49 -0
  59. data/lib/webrat/selenium/matchers/have_tag.rb +72 -0
  60. data/lib/webrat/selenium/matchers/have_xpath.rb +45 -0
  61. data/lib/webrat/selenium/selenium_extensions.js +6 -0
  62. data/lib/webrat/selenium/selenium_session.rb +247 -0
  63. data/lib/webrat/sinatra.rb +44 -0
  64. data/vendor/selenium-server.jar +0 -0
  65. metadata +76 -2
@@ -0,0 +1,24 @@
1
+ module Webrat
2
+
3
+ def self.rexml_document(stringlike)
4
+ stringlike = stringlike.body.to_s if stringlike.respond_to?(:body)
5
+
6
+ case stringlike
7
+ when REXML::Document
8
+ stringlike.root
9
+ when REXML::Node, Array
10
+ stringlike
11
+ else
12
+ begin
13
+ REXML::Document.new(stringlike.to_s).root
14
+ rescue REXML::ParseException => e
15
+ if e.message.include?("second root element")
16
+ REXML::Document.new("<fake-root-element>#{stringlike}</fake-root-element>").root
17
+ else
18
+ raise e
19
+ end
20
+ end
21
+ end
22
+ end
23
+
24
+ end
@@ -0,0 +1,58 @@
1
+ class Object #:nodoc:
2
+ # An object is blank if it's false, empty, or a whitespace string.
3
+ # For example, "", " ", +nil+, [], and {} are blank.
4
+ #
5
+ # This simplifies
6
+ #
7
+ # if !address.nil? && !address.empty?
8
+ #
9
+ # to
10
+ #
11
+ # if !address.blank?
12
+ def blank?
13
+ respond_to?(:empty?) ? empty? : !self
14
+ end
15
+
16
+ # An object is present if it's not blank.
17
+ def present?
18
+ !blank?
19
+ end
20
+ end
21
+
22
+ class NilClass #:nodoc:
23
+ def blank?
24
+ true
25
+ end
26
+ end
27
+
28
+ class FalseClass #:nodoc:
29
+ def blank?
30
+ true
31
+ end
32
+ end
33
+
34
+ class TrueClass #:nodoc:
35
+ def blank?
36
+ false
37
+ end
38
+ end
39
+
40
+ class Array #:nodoc:
41
+ alias_method :blank?, :empty?
42
+ end
43
+
44
+ class Hash #:nodoc:
45
+ alias_method :blank?, :empty?
46
+ end
47
+
48
+ class String #:nodoc:
49
+ def blank?
50
+ self !~ /\S/
51
+ end
52
+ end
53
+
54
+ class Numeric #:nodoc:
55
+ def blank?
56
+ false
57
+ end
58
+ end
@@ -0,0 +1,8 @@
1
+ class Module #:nodoc:
2
+ def webrat_deprecate(old_method_name, new_method_name)
3
+ define_method old_method_name do |*args|
4
+ warn "#{old_method_name} is deprecated. Use #{new_method_name} instead."
5
+ __send__(new_method_name, *args)
6
+ end
7
+ end
8
+ end
@@ -0,0 +1,12 @@
1
+ class Array #:nodoc:
2
+
3
+ def detect_mapped
4
+ each do |element|
5
+ result = yield element
6
+ return result if result
7
+ end
8
+
9
+ return nil
10
+ end
11
+
12
+ end
@@ -0,0 +1,6 @@
1
+ class ::Object #:nodoc:
2
+ def meta_class
3
+ class << self; self end
4
+ end
5
+ end
6
+
@@ -0,0 +1,5 @@
1
+ class NilClass #:nodoc:
2
+ def to_param
3
+ nil
4
+ end
5
+ end
@@ -0,0 +1,74 @@
1
+ require "mechanize"
2
+
3
+ module Webrat #:nodoc:
4
+ class MechanizeSession < Session #:nodoc:
5
+
6
+ attr_accessor :response
7
+ alias :page :response
8
+
9
+ def request_page(url, http_method, data) #:nodoc:
10
+ super(absolute_url(url), http_method, data)
11
+ end
12
+
13
+ def get(url, data, headers_argument_not_used = nil)
14
+ @response = mechanize.get(url, data)
15
+ end
16
+
17
+ def post(url, data, headers_argument_not_used = nil)
18
+ post_data = data.inject({}) do |memo, param|
19
+ case param.last
20
+ when Hash
21
+ param.last.each {|attribute, value| memo["#{param.first}[#{attribute}]"] = value }
22
+ else
23
+ memo[param.first] = param.last
24
+ end
25
+ memo
26
+ end
27
+ @response = mechanize.post(url, post_data)
28
+ end
29
+
30
+ def response_body
31
+ @response.content
32
+ end
33
+
34
+ def response_code
35
+ @response.code.to_i
36
+ end
37
+
38
+ def mechanize
39
+ @mechanize = WWW::Mechanize.new
40
+ end
41
+
42
+ def_delegators :mechanize, :basic_auth
43
+
44
+ def absolute_url(url) #:nodoc:
45
+ current_host, current_path = split_current_url
46
+ if url =~ Regexp.new('^https?://')
47
+ url
48
+ elsif url =~ Regexp.new('^/')
49
+ current_host + url
50
+ elsif url =~ Regexp.new('^\.')
51
+ current_host + absolute_path(current_path, url)
52
+ else
53
+ url
54
+ end
55
+ end
56
+
57
+ private
58
+ def split_current_url
59
+ current_url =~ Regexp.new('^(https?://[^/]+)(/.*)?')
60
+ [Regexp.last_match(1), Regexp.last_match(2)]
61
+ end
62
+
63
+ def absolute_path(current_path, url)
64
+ levels_up = url.split('/').find_all { |x| x == '..' }.size
65
+ ancestor = if current_path.nil?
66
+ ""
67
+ else
68
+ current_path.split("/")[0..(-1 - levels_up)].join("/")
69
+ end
70
+ descendent = url.split("/")[levels_up..-1]
71
+ "#{ancestor}/#{descendent}"
72
+ end
73
+ end
74
+ end
@@ -0,0 +1,9 @@
1
+ require "webrat"
2
+
3
+ # This is a temporary hack to support backwards compatibility
4
+ # with Merb 1.0.8 until it's updated to use the new Webrat.configure
5
+ # syntax
6
+
7
+ Webrat.configure do |config|
8
+ config.mode = :merb
9
+ end
@@ -0,0 +1,65 @@
1
+ require "webrat"
2
+
3
+ require "cgi"
4
+ gem "extlib"
5
+ require "extlib"
6
+ require "merb-core"
7
+
8
+ # HashWithIndifferentAccess = Mash
9
+
10
+ module Webrat
11
+ class MerbSession < Session #:nodoc:
12
+ include Merb::Test::MakeRequest
13
+
14
+ attr_accessor :response
15
+
16
+ def get(url, data, headers = nil)
17
+ do_request(url, data, headers, "GET")
18
+ end
19
+
20
+ def post(url, data, headers = nil)
21
+ do_request(url, data, headers, "POST")
22
+ end
23
+
24
+ def put(url, data, headers = nil)
25
+ do_request(url, data, headers, "PUT")
26
+ end
27
+
28
+ def delete(url, data, headers = nil)
29
+ do_request(url, data, headers, "DELETE")
30
+ end
31
+
32
+ def response_body
33
+ @response.body.to_s
34
+ end
35
+
36
+ def response_code
37
+ @response.status
38
+ end
39
+
40
+ def do_request(url, data, headers, method)
41
+ @response = request(url,
42
+ :params => (data && data.any?) ? data : nil,
43
+ :headers => headers,
44
+ :method => method)
45
+ end
46
+
47
+ end
48
+ end
49
+
50
+ module Merb #:nodoc:
51
+ module Test #:nodoc:
52
+ module RequestHelper #:nodoc:
53
+ def request(uri, env = {})
54
+ @_webrat_session ||= Webrat::MerbSession.new
55
+ @_webrat_session.response = @_webrat_session.request(uri, env)
56
+ end
57
+ end
58
+ end
59
+ end
60
+
61
+ class Merb::Test::RspecStory #:nodoc:
62
+ def browser
63
+ @browser ||= Webrat::MerbSession.new
64
+ end
65
+ end
@@ -0,0 +1,24 @@
1
+ require 'webrat'
2
+
3
+ class CGIMethods #:nodoc:
4
+ def self.parse_query_parameters(params)
5
+ hash = {}
6
+ params.split('&').each do |p|
7
+ pair = p.split('=')
8
+ hash[pair[0]] = pair[1]
9
+ end
10
+ hash
11
+ end
12
+ end
13
+
14
+ module Webrat
15
+ class RackSession < Session #:nodoc:
16
+ def response_body
17
+ @response.body
18
+ end
19
+
20
+ def response_code
21
+ @response.status
22
+ end
23
+ end
24
+ end
@@ -0,0 +1,105 @@
1
+ require "webrat"
2
+
3
+ require "action_controller"
4
+ require "action_controller/integration"
5
+ require "action_controller/record_identifier"
6
+
7
+ module Webrat
8
+ class RailsSession < Session #:nodoc:
9
+ include ActionController::RecordIdentifier
10
+
11
+ # The Rails version of within supports passing in a model and Webrat
12
+ # will apply a scope based on Rails' dom_id for that model.
13
+ #
14
+ # Example:
15
+ # within User.last do
16
+ # click_link "Delete"
17
+ # end
18
+ def within(selector_or_object, &block)
19
+ if selector_or_object.is_a?(String)
20
+ super
21
+ else
22
+ super('#' + dom_id(selector_or_object), &block)
23
+ end
24
+ end
25
+
26
+ def doc_root
27
+ File.expand_path(File.join(RAILS_ROOT, 'public'))
28
+ end
29
+
30
+ def saved_page_dir
31
+ File.expand_path(File.join(RAILS_ROOT, "tmp"))
32
+ end
33
+
34
+ def get(url, data, headers = nil)
35
+ do_request(:get, url, data, headers)
36
+ end
37
+
38
+ def post(url, data, headers = nil)
39
+ do_request(:post, url, data, headers)
40
+ end
41
+
42
+ def put(url, data, headers = nil)
43
+ do_request(:put, url, data, headers)
44
+ end
45
+
46
+ def delete(url, data, headers = nil)
47
+ do_request(:delete, url, data, headers)
48
+ end
49
+
50
+ def response_body
51
+ response.body
52
+ end
53
+
54
+ def response_code
55
+ response.code.to_i
56
+ end
57
+
58
+ def xml_content_type?
59
+ response.headers["Content-Type"].to_s =~ /xml/
60
+ end
61
+
62
+ protected
63
+
64
+ def integration_session
65
+ @context
66
+ end
67
+
68
+ def do_request(http_method, url, data, headers) #:nodoc:
69
+ update_protocol(url)
70
+ integration_session.send(http_method, normalize_url(url), data, headers)
71
+ end
72
+
73
+ # remove protocol, host and anchor
74
+ def normalize_url(href) #:nodoc:
75
+ uri = URI.parse(href)
76
+ normalized_url = []
77
+ normalized_url << "#{uri.scheme}://" if uri.scheme
78
+ normalized_url << uri.host if uri.host
79
+ normalized_url << ":#{uri.port}" if uri.port && ![80,443].include?(uri.port)
80
+ normalized_url << uri.path if uri.path
81
+ normalized_url << "?#{uri.query}" if uri.query
82
+ normalized_url.join
83
+ end
84
+
85
+ def update_protocol(href) #:nodoc:
86
+ if href =~ /^https:/
87
+ integration_session.https!(true)
88
+ elsif href =~ /^http:/
89
+ integration_session.https!(false)
90
+ end
91
+ end
92
+
93
+ def response #:nodoc:
94
+ integration_session.response
95
+ end
96
+
97
+ end
98
+ end
99
+
100
+ module ActionController #:nodoc:
101
+ IntegrationTest.class_eval do
102
+ include Webrat::Methods
103
+ include Webrat::Matchers
104
+ end
105
+ end
@@ -0,0 +1,13 @@
1
+ # Supports using the matchers in controller, helper, and view specs if you're
2
+ # using rspec-rails. Just add a require statement to spec/spec_helper.rb or env.rb:
3
+ #
4
+ # require 'webrat/rspec-rails'
5
+ #
6
+ require "webrat/core/matchers"
7
+
8
+ Spec::Runner.configure do |config|
9
+ # rspec should support :type => [:controller, :helper, :view] - but until it does ...
10
+ config.include(Webrat::Matchers, :type => :controller)
11
+ config.include(Webrat::Matchers, :type => :helper)
12
+ config.include(Webrat::Matchers, :type => :view)
13
+ end
@@ -0,0 +1,154 @@
1
+ require "webrat"
2
+ gem "selenium-client", ">=1.2.9"
3
+ require "selenium/client"
4
+ require "webrat/selenium/selenium_session"
5
+ require "webrat/selenium/matchers"
6
+
7
+ module Webrat
8
+
9
+ def self.with_selenium_server #:nodoc:
10
+ start_selenium_server
11
+ yield
12
+ stop_selenium_server
13
+ end
14
+
15
+ def self.start_selenium_server #:nodoc:
16
+ unless Webrat.configuration.selenium_server_address
17
+ remote_control = ::Selenium::RemoteControl::RemoteControl.new("0.0.0.0", Webrat.configuration.selenium_server_port, 5)
18
+ remote_control.jar_file = File.expand_path(__FILE__ + "../../../../vendor/selenium-server.jar")
19
+ remote_control.start :background => true
20
+ end
21
+ TCPSocket.wait_for_service :host => (Webrat.configuration.selenium_server_address || "0.0.0.0"), :port => Webrat.configuration.selenium_server_port
22
+ end
23
+
24
+ def self.stop_selenium_server #:nodoc:
25
+ ::Selenium::RemoteControl::RemoteControl.new("0.0.0.0", Webrat.configuration.selenium_server_port, 5).stop unless Webrat.configuration.selenium_server_address
26
+ end
27
+
28
+ def self.pid_file
29
+ if File.exists?('config.ru')
30
+ prepare_pid_file(Dir.pwd, 'rack.pid')
31
+ else
32
+ prepare_pid_file("#{RAILS_ROOT}/tmp/pids", "mongrel_selenium.pid")
33
+ end
34
+ end
35
+ # Start the appserver for the underlying framework to test
36
+ #
37
+ # Sinatra: requires a config.ru in the root of your sinatra app to use this
38
+ # Merb: Attempts to use bin/merb and falls back to the system merb
39
+ # Rails: Calls mongrel_rails to startup the appserver
40
+ def self.start_app_server
41
+ case Webrat.configuration.application_framework
42
+ when :sinatra
43
+ fork do
44
+ File.open('rack.pid', 'w') { |fp| fp.write Process.pid }
45
+ exec 'rackup', File.expand_path(Dir.pwd + '/config.ru'), '-p', Webrat.configuration.application_port.to_s
46
+ end
47
+ when :merb
48
+ cmd = 'merb'
49
+ if File.exist?('bin/merb')
50
+ cmd = 'bin/merb'
51
+ end
52
+ system("#{cmd} -d -p #{Webrat.configuration.application_port} -e #{Webrat.configuration.application_environment}")
53
+ else # rails
54
+ system("mongrel_rails start -d --chdir='#{RAILS_ROOT}' --port=#{Webrat.configuration.application_port} --environment=#{Webrat.configuration.application_environment} --pid #{pid_file} &")
55
+ end
56
+ TCPSocket.wait_for_service :host => Webrat.configuration.application_address, :port => Webrat.configuration.application_port.to_i
57
+ end
58
+
59
+ # Stop the appserver for the underlying framework under test
60
+ #
61
+ # Sinatra: Reads and kills the pid from the pid file created on startup
62
+ # Merb: Reads and kills the pid from the pid file created on startup
63
+ # Rails: Calls mongrel_rails stop to kill the appserver
64
+ def self.stop_app_server
65
+ case Webrat.configuration.application_framework
66
+ when :sinatra
67
+ pid = File.read('rack.pid')
68
+ system("kill -9 #{pid}")
69
+ FileUtils.rm_f 'rack.pid'
70
+ when :merb
71
+ pid = File.read("log/merb.#{Webrat.configuration.application_port}.pid")
72
+ system("kill -9 #{pid}")
73
+ FileUtils.rm_f "log/merb.#{Webrat.configuration.application_port}.pid"
74
+ else # rails
75
+ system "mongrel_rails stop -c #{RAILS_ROOT} --pid #{pid_file}"
76
+ end
77
+ end
78
+
79
+ def self.prepare_pid_file(file_path, pid_file_name)
80
+ FileUtils.mkdir_p File.expand_path(file_path)
81
+ File.expand_path("#{file_path}/#{pid_file_name}")
82
+ end
83
+
84
+ # To use Webrat's Selenium support, you'll need the selenium-client gem installed.
85
+ # Activate it with (for example, in your <tt>env.rb</tt>):
86
+ #
87
+ # require "webrat"
88
+ #
89
+ # Webrat.configure do |config|
90
+ # config.mode = :selenium
91
+ # end
92
+ #
93
+ # == Dropping down to the selenium-client API
94
+ #
95
+ # If you ever need to do something with Selenium not provided in the Webrat API,
96
+ # you can always drop down to the selenium-client API using the <tt>selenium</tt> method.
97
+ # For example:
98
+ #
99
+ # When "I drag the photo to the left" do
100
+ # selenium.dragdrop("id=photo_123", "+350, 0")
101
+ # end
102
+ #
103
+ # == Choosing the underlying framework to test
104
+ #
105
+ # Webrat assumes you're using rails by default but it can also work with sinatra
106
+ # and merb. To take advantage of this you can use the configuration block to
107
+ # set the application_framework variable.
108
+ # require "webrat"
109
+ #
110
+ # Webrat.configure do |config|
111
+ # config.mode = :selenium
112
+ # config.application_port = 4567
113
+ # config.application_framework = :sinatra # could also be :merb
114
+ # end
115
+ #
116
+ # == Auto-starting of the appserver and java server
117
+ #
118
+ # Webrat will automatically start the Selenium Java server process and an instance
119
+ # of Mongrel when a test is run. The Mongrel will run in the "selenium" environment
120
+ # instead of "test", so ensure you've got that defined, and will run on port
121
+ # Webrat.configuration.application_port.
122
+ #
123
+ # == Waiting
124
+ #
125
+ # In order to make writing Selenium tests as easy as possible, Webrat will automatically
126
+ # wait for the correct elements to exist on the page when trying to manipulate them
127
+ # with methods like <tt>fill_in</tt>, etc. In general, this means you should be able to write
128
+ # your Webrat::Selenium tests ignoring the concurrency issues that can plague in-browser
129
+ # testing, so long as you're using the Webrat API.
130
+ module Selenium
131
+ module Methods
132
+ def response
133
+ webrat_session.response
134
+ end
135
+
136
+ def wait_for(*args, &block)
137
+ webrat_session.wait_for(*args, &block)
138
+ end
139
+
140
+ def save_and_open_screengrab
141
+ webrat_session.save_and_open_screengrab
142
+ end
143
+ end
144
+ end
145
+ end
146
+ if defined?(ActionController::IntegrationTest)
147
+ module ActionController #:nodoc:
148
+ IntegrationTest.class_eval do
149
+ include Webrat::Methods
150
+ include Webrat::Selenium::Methods
151
+ include Webrat::Selenium::Matchers
152
+ end
153
+ end
154
+ end