Compare commits

...

402 Commits

Author SHA1 Message Date
John Bintz 5e4ffeae9f ensure query strings are built correctly for mechanize 2010-06-01 14:29:57 -04:00
John Bintz 0856d8d405 potential fix for invalid query string construction 2010-06-01 14:10:39 -04:00
Bryan Helmkamp 8f637e6785 Bumping version to prerelease until next release 2010-04-26 22:15:41 -04:00
Bryan Helmkamp 312c175384 Releasing v0.7.1 2010-04-26 22:10:16 -04:00
Damian Janowski 7cde0573c0 Update changelog. 2010-04-23 13:19:45 -03:00
Luismi Cavalle 43b19eeafa Fix have_xpath not matching negative expectation in the block.
[#182 state:resolved]
2010-04-23 13:17:49 -03:00
Damian Janowski 107bdf7cf9 Update Gemfile to the newest Bundler. 2010-04-23 13:15:30 -03:00
Damian Janowski 71a281066e Update changelog. 2010-04-23 13:09:05 -03:00
Damian Janowski 438e53c295 Fix broken integration test in 1.9.1-p378. 2010-04-22 19:09:45 -03:00
Noah Davis 80ea640901 Fix selenium webrat tests in FF/Safari -- locatedLabel.htmlFor returns empty string in FF/Safari which breaks location strategy for label 2010-04-20 23:46:43 -04:00
Noah Davis 68d5fde17a Move verbose selenium output that can clutter build output behind setting 2010-04-20 11:44:55 -04:00
Damian Janowski 1979a7fc7f Use `htmlFor` on IE.
[#317 state:resolved]
2010-04-15 15:40:03 -03:00
Álvaro Gil 06bb72d24f fix post method in mechanize adapter to handle data as array or hash
it fixes spec/integration/mechanize/sample_app.rb:20
2010-04-13 15:40:24 -03:00
Damian Janowski c49123c772 Fix Mechanize spec. 2010-04-13 13:25:28 -03:00
Damian Janowski 2c9de38129 Add missing spec that shows a regression after e3bcf5. 2010-04-13 12:57:25 -03:00
Damian Janowski 2a6c67b17b Fix delegation of `response_headers` method.
[#332 state:resolved]
2010-04-13 12:04:36 -03:00
Damian Janowski 38e699820f Merge remote branch 'wemrysi/master' into mechanize_redirects
Conflicts:
	lib/webrat/adapters/mechanize.rb
2010-04-13 12:01:22 -03:00
Damian Janowski 4c62279fc1 The WWW namespace is deprecated as of Mechanize 1.0.
[#368 state:resolved]
2010-04-12 23:38:04 -03:00
Damian Janowski deccefa2c3 Some explicit requires of RubyGems are necessary for integration tests.
Should not affect the way Webrat behaves as a library.
2010-04-12 23:30:56 -03:00
Simon Rozet 10c70a37de Get rid of needless require "rubygems" 2010-04-12 22:28:14 -03:00
Damian Janowski 19fc2ccb89 Remove warnings [#372 state:resolved].
On why this is important:
http://eigenclass.org/hiki/ruby-warnings-SEX-and-stds
2010-04-12 22:25:53 -03:00
Larry Marburger fd944cccb5 Webrat::Core::Session#redirect? should not mistake a 304 for a redirect. 2010-04-12 17:52:09 -03:00
Damian Janowski 0bbf163d4f Better stack traces when using class_eval (Alex Chaffee). 2010-04-12 17:45:10 -03:00
Damian Janowski ef6874c855 Allow submitting forms by CSS selectors too.
Priority is given to selecting by ID for backwards compatibility.

Also add pending `submit_form` specs.
2010-04-12 17:35:12 -03:00
Damian Janowski e3bcf5c599 Merge branch 'nested_params_support' 2010-04-12 17:22:54 -03:00
Damian Janowski bd6cd436de Point users in the right direction when selenium is not available. 2010-04-12 16:39:27 -03:00
Álvaro Gil e36487458f test added for nested params files upload 2010-04-09 11:38:33 -03:00
Álvaro Gil c49e23d81d fixes #341 attach_file with nested attributes 2010-04-05 11:31:38 -03:00
Álvaro Gil 15102bd60d !trivial: define and call class methods in From appropriate 2010-04-05 11:07:35 -03:00
Álvaro Gil fa881a88c8 build querystring with all form elements and then parse to get query params
* Basically Field#to_param was replaced for Field#to_query_string and
some methods related to build params were moved to Form class.

Before this commit the params hash was made by parsing each element
querystring to param and then merge, now we build the whole querystring
first and then parse it to get params with Rack or Rails depending of
the configure mode.
2010-04-03 12:07:16 -03:00
Noah Davis 2f12422ae8 New webrat configuration option selenium_firefox_profile which is passed to selenium server 2010-02-26 15:54:31 -05:00
Emrys Ingersoll 67febe9072 Normalize access to response headers in adapters 2010-02-20 17:06:30 -05:00
Emrys Ingersoll 56b2f342da Adding spec for disabling automatic redirects in Mechanize 2010-02-18 19:04:11 -05:00
Emrys Ingersoll b2dddf2f09 Disable the automatic following of redirects in the Mechanize instance so they can be handled by Webrat::Session 2010-02-18 18:33:22 -05:00
lukemelia d9d9cc1fcc Added application_port_for_selenium to webrat configuration. The use case is when you want to test through a web server sitting in front of your application server. 2010-01-18 09:49:44 -05:00
Noah Davis 0387d9eacc Bumping version to prerelease until next official release 2010-01-17 20:48:23 -05:00
Noah Davis d73eca71ee Releasing v 0.7.0 2010-01-17 20:44:20 -05:00
Noah Davis 8d2fa24b4a Added missing dependency "rack-test" to gemspec (LH #339) 2010-01-17 20:42:37 -05:00
Bryan Helmkamp eb3a4ff5d6 Whitespace 2010-01-17 20:35:42 -05:00
hpoydar 4ac8456a48 Replaced selenium standalone jar to fix Snow Leopard + Firefox 3.5 issue 2010-01-17 20:30:55 -05:00
Noah Davis 3159ae081f changelog update 2010-01-17 20:23:48 -05:00
Noah Davis ccaf41160c Save and open page storage directory now defaults to "tmp" under current directory if exists, otherwise current directory 2010-01-17 18:29:35 -05:00
Noah Davis d50d24e09e Documentation fix for application environment 2010-01-17 18:08:40 -05:00
Noah Davis 1874f6d395 Removed save_and_open_page's rewriting of static asset paths before saving (was not actually working) 2010-01-16 19:28:29 -05:00
Noah Davis 71dcfb327d Save and open page directory specified via configuration 2010-01-16 19:18:28 -05:00
Noah Davis db168bedec Make "should contain" ignore extra whitespace when doing string comparisons 2010-01-13 16:47:35 -05:00
lukemelia f9985cd488 Make selenium matchers handle negative match more consistently with positive match 2010-01-05 18:33:45 -05:00
Bryan Helmkamp b64d68a2a8 Bump VERSION to prerelease 2009-11-28 00:24:47 -05:00
Bryan Helmkamp 4081fe5538 Prep for 0.6.0 release 2009-11-28 00:18:58 -05:00
Bryan Helmkamp 3bd7266ab9 Merge branch 'lh_292'
Conflicts:
	History.txt
2009-11-15 16:51:23 -05:00
Bryan Helmkamp a9ff3efcd7 History 2009-11-15 16:50:46 -05:00
Michael Klett 61b0fce828 Base64 encoding of HTTP basic auth credentials should omit newlines (do not enforce a 76 character line limit)
According to RFC 2617 (http://www.ietf.org/rfc/rfc2617.txt), the 76 character line limit normally enforced by Base64 encoding should not apply to the encoding of HTTP basic authentication credentials.
2009-11-15 16:50:40 -05:00
Bryan Helmkamp f9f2a98189 History 2009-11-15 16:50:39 -05:00
David Chelimsky 51bdded2ce require 'nokogiri' in rspec-rails include file 2009-11-15 16:50:39 -05:00
Bryan Helmkamp 4de7d64776 History 2009-11-15 16:42:39 -05:00
Bryan Helmkamp 13d513837a Merge remote branch 'dchelimsky/require_nokogiri' into lh_293 2009-11-15 16:41:37 -05:00
Bryan Helmkamp ba95363486 Spec fix: Don't depend on Nokogiri #inspect output 2009-11-15 16:38:09 -05:00
Bryan Helmkamp 672bd583af Whitespace 2009-11-15 16:29:44 -05:00
Bryan Helmkamp 0536360773 History 2009-11-15 16:29:27 -05:00
Jonathan Weiss ea4a52ccf1 add support for Rails 2.3.4 form authenticity tokens 2009-11-15 16:24:56 -05:00
Bryan Helmkamp cbb07306d4 Regenerate gemspec with thor tasks 2009-11-15 16:07:38 -05:00
Bryan Helmkamp 5a05d860ec Use simple Thor tasks for release management instead of Jeweler 2009-11-15 15:49:38 -05:00
Bryan Helmkamp 05af4dd2a0 Switching to Safari until Snow Leopard + Firefox + Selenium is fixed
http://jira.openqa.org/browse/SRC-743
2009-11-13 21:23:27 -05:00
Bryan Helmkamp 7812f25741 Don't blow up if PID doesn't exist at the end 2009-11-13 21:22:41 -05:00
Bryan Helmkamp fcde80f43f Avoid Selenium test bleed by ending with an assertion about the next page 2009-11-13 21:17:41 -05:00
Bryan Helmkamp 1ce72b6912 Ending selenium tests with assertions avoids "test bleed" 2009-11-13 21:10:25 -05:00
Bryan Helmkamp bb282d992e Remove a couple more bogus tests
<input type="button" /> doesn't submit forms
2009-11-13 21:07:21 -05:00
Bryan Helmkamp f6dd24f7c5 Remove bogus integration tests that were passing (sometimes) by coincidence
An element of <button type="button" /> is not supposed to submit a form
2009-11-13 20:57:55 -05:00
Bryan Helmkamp d0cadf8a76 Update file location for Rails 2.3.4 2009-11-13 20:52:14 -05:00
Bryan Helmkamp 1c28425583 Use Bundler for managing dependencies needed to run the tests 2009-11-13 17:41:33 -05:00
David Chelimsky 9a92afedff require 'nokogiri' in rspec-rails include file 2009-09-23 23:04:43 -05:00
Bryan Helmkamp f51b745e89 When a timeout occurs in wait_for, include the HTML from Selenium in the exception 2009-09-22 15:23:49 -04:00
Bryan Helmkamp 8dacd973aa Regenerated gemspec for version 0.6.rc1 2009-09-22 12:44:56 -04:00
Bryan Helmkamp 194d3250cc Prep for 0.6.rc1 release 2009-09-22 12:41:57 -04:00
Bryan Helmkamp b68c6859e8 History 2009-09-22 12:41:41 -04:00
Bryan Helmkamp 87c54dde2e Fix typo in Rakefile 2009-09-19 17:57:35 -04:00
Bryan Helmkamp 042b416092 Whitespace 2009-09-19 17:34:09 -04:00
Bryan Helmkamp 40d7f970e0 Gracefully handle cases where RSpec is not installed in Rakefile 2009-09-19 17:32:42 -04:00
Bryan Helmkamp 03f2023117 Make link location in Selenium more reliable and consistent with non-Selenium 2009-09-18 15:49:06 -04:00
Bryan Helmkamp b51ba029db Fix support for passing in an ActiveRecord model to within (which uses dom_id) LB/BH 2009-09-18 11:28:07 -04:00
Bryan Helmkamp 862471476f Run Selenium tests during spec:integration 2009-09-17 22:13:10 -04:00
Bryan Helmkamp c689166c48 Fix "element.getAttribute is not a function" Selenium errors when filling in fields
The root cause was the locator strategy was naively returning an element that was not a form field, causing Selenium's internals to blow up
2009-09-17 21:32:56 -04:00
Bryan Helmkamp 3a9e7d3999 Fix bugs when dealing with special characters in labels 2009-09-17 21:18:52 -04:00
Bryan Helmkamp 63e1053afa Fix bug in Selenium when dealing with special characters in link text 2009-09-17 20:28:19 -04:00
Bryan Helmkamp aae82b5111 Whitespace 2009-09-17 20:15:40 -04:00
Bryan Helmkamp e7a3ac865c Guard tests for things that aren't supported in Selenium yet 2009-09-17 20:13:04 -04:00
Bryan Helmkamp 8772505322 History 2009-09-07 14:16:23 -04:00
Bryan Helmkamp b0ef59f5ec Properly handle multiline param values (previously, subsequent lines were lost) 2009-09-07 14:07:05 -04:00
Bryan Helmkamp ad06a777fb Re-introduce VERSION constant, but don't make it dependent on the VERSION file 2009-09-07 13:25:33 -04:00
Bryan Helmkamp 756a0c62ba Make build task depend on gemspec. Closes GH issue #15 2009-09-07 13:22:57 -04:00
Bryan Helmkamp 63591a8c30 History 2009-09-07 13:18:57 -04:00
Bryan Helmkamp ecaed5ff93 Merge commit 'm3talsmith/master' into gh_18 2009-09-07 13:16:53 -04:00
Bryan Helmkamp b0a8922eb4 History 2009-09-07 13:08:34 -04:00
Bryan Helmkamp 2490c247d2 More clarification of select option locating behavior.
TODO: Ensure the other locators are all matching on text, not HTML
2009-09-07 12:53:06 -04:00
Bryan Helmkamp 00c49a0521 Simplify select option matching code. Minor behavior change:
Valid: select "Peanut Butter & Jelly"
  Invalid: select "Peanut Butter &amp; Jelly"
2009-09-07 12:44:54 -04:00
Bryan Helmkamp b7ea26841e Add some development dependencies (needed for running the specs) 2009-09-07 12:44:42 -04:00
Kieran Pilkington a8d0cebde3 Various improvements, refinements, and bugfixes for select field usage.
Multiple select boxes are now handled a lot better.
* You can now select more than one field (where as previously, the last one select would override previous selections)
* You can now unselect a value. It works like select, takes the same params, but removes the selection value

Issues regarding ampersands in values have been resolved.
* Values are now unescaped when the to_param method is run on elements, to make assertions less complicated
* Locating an option with ampersand values now works as expected (it will check for both escaped and unescaped occurrences)

Bunch of new specs and 3 broken pending ones have now been fixed. [#145 status:resolved]
2009-08-31 18:34:04 +12:00
Bryan Helmkamp 98efa54900 Whitespace 2009-08-29 16:09:30 -04:00
Bryan Helmkamp 49c25027e9 History 2009-08-29 16:05:56 -04:00
Bryan Helmkamp 6fd9732b4e Spec giving me trouble -- removing for now 2009-08-29 16:04:04 -04:00
Bryan Helmkamp b9466df773 Merge commit 'djanowski/fix_logger' 2009-08-29 15:57:58 -04:00
Michael Christenson II 3255c9edf0 Forces an integer to fix Issue #18 2009-08-27 11:54:03 -04:00
Bryan Helmkamp 5b90488f29 Revert "Regenerated gemspec for version 0.5.2"
This reverts commit f1cd50d615.
2009-08-27 01:02:01 -04:00
Bryan Helmkamp f1cd50d615 Regenerated gemspec for version 0.5.2 2009-08-27 00:59:53 -04:00
Bryan Helmkamp e18ef197b5 Move selenium integration code to webrat/integrations/selenium so it's automatically required 2009-08-20 00:09:03 -04:00
Bryan Helmkamp 755abb7e07 History 2009-08-19 23:44:33 -04:00
Bryan Helmkamp 723f4356b9 Move rspec-rails integration code to webrat/integrations/rspec-rails 2009-08-19 23:42:58 -04:00
Bryan Helmkamp 2fc9f3f7f9 Move adapters to webrat/adapters, and Rails and Merb integration code into webrat/integrations 2009-08-19 23:42:54 -04:00
Bryan Helmkamp 610f0fc217 Webrat doesn't need to worry about StringIOs
...or, we're missing a test and this is going to help us find it
2009-08-19 23:14:13 -04:00
Bryan Helmkamp 11f30d1d2e Dropping support for Hpricot and REXML 2009-08-19 23:14:13 -04:00
Simon Rozet 997ff97405 MerbSession now use RackSession under the hood 2009-08-19 23:14:08 -04:00
Bryan Helmkamp ab745d786c Require rubygems in rakefile for convenience
This does not break the "Don't require rubygems" spirit,
because it's not creating a dependency in the Webrat runtime code
2009-08-19 22:02:19 -04:00
Bryan Helmkamp 759d7e4581 Regenerated gemspec for version 0.5.1 2009-08-18 00:42:57 -04:00
Bryan Helmkamp 35deb84a18 Version bump to 0.5.1 2009-08-18 00:42:31 -04:00
Bryan Helmkamp 887d2b07f3 Prep for 0.5.1 release 2009-08-18 00:42:24 -04:00
Bryan Helmkamp ad0c846f17 History 2009-08-18 00:39:53 -04:00
Damian Janowski 4a949e0e5a Fix logger issue when running inside Cucumber. 2009-08-14 13:05:20 -03:00
Bryan Helmkamp 66251b66ae Cleanup session initializaton in Webrat::Methods 2009-08-12 22:05:04 -04:00
Bryan Helmkamp ec43bb9546 Selenium is a session type, not an adapter type 2009-08-12 21:44:02 -04:00
Bryan Helmkamp 2e6bef3b57 Rename TestSession to TestAdapter 2009-08-12 21:42:01 -04:00
Bryan Helmkamp 17713c9036 MechanizeAdapter doesn't need to inherit from Session 2009-08-12 21:39:32 -04:00
Bryan Helmkamp 0412543e5a Add very basic spec for Mechanize 2009-08-12 21:37:15 -04:00
Bryan Helmkamp e4ea9134a7 Rename MechanizeSession to MechanizeAdapter 2009-08-12 20:54:29 -04:00
Bryan Helmkamp 89ec303db5 Rename RailsSession to RailsAdapter 2009-08-12 20:52:57 -04:00
Bryan Helmkamp 986682488e Rename MerbSession to MerbAdapter 2009-08-12 20:50:55 -04:00
Bryan Helmkamp 22a06000f9 Introduce session_class method 2009-08-12 20:44:32 -04:00
Bryan Helmkamp a8c5af2b7a Rename session_class method to adapter_class 2009-08-12 20:43:44 -04:00
Bryan Helmkamp b0b88a0d82 Rename SinatraSession to SinatraAdapter 2009-08-12 20:40:44 -04:00
Bryan Helmkamp 561525dd81 Rename RackSession to RackAdapter 2009-08-12 20:39:35 -04:00
Bryan Helmkamp 3670b19f66 Remove duplicate branch for :sinatra mode 2009-08-12 20:25:25 -04:00
Bryan Helmkamp d55802d26e History 2009-08-12 20:17:16 -04:00
Bryan Helmkamp 74d470ff10 Merge commit 'sr/rack-upload' 2009-08-12 20:16:57 -04:00
Bryan Helmkamp 25e46a00f6 History 2009-08-12 20:14:01 -04:00
Bryan Helmkamp 27a773e6b0 Merge commit 'sr/mime' into sr-mime
Conflicts:
	Rakefile
	lib/webrat.rb
2009-08-12 20:12:14 -04:00
Bryan Helmkamp 3cdf378dd7 Add description to gemspec 2009-08-12 12:22:06 -04:00
Bryan Helmkamp 6a85b6e444 Regenerated gemspec for version 0.5.0 2009-08-12 12:15:14 -04:00
Bryan Helmkamp b48f9d622f Version bump to 0.5.0 2009-08-12 12:14:32 -04:00
Bryan Helmkamp 083332cdd3 Prep for 0.5.0 release 2009-08-12 12:14:18 -04:00
Bryan Helmkamp 3952d46979 History 2009-08-12 11:50:17 -04:00
Bryan Helmkamp 25c1d7c783 Merge branch 'master' into rack 2009-08-12 09:40:57 -04:00
Kamal Fariz Mahyuddin d9ebabf461 Ruby 1.9-compat - test for MiniTest::Assertion instead of T::U::AssertionFailedError 2009-08-12 14:52:01 +08:00
Bryan Helmkamp 0c7fa65880 Update gemspec 2009-08-10 20:21:38 -04:00
Bryan Helmkamp 19ca271bbd Merge branch 'master' into rack 2009-08-10 20:18:24 -04:00
Bryan Helmkamp 987766b10d Don't require rubygems at runtime (Simon Rozet) 2009-08-10 20:14:48 -04:00
Bryan Helmkamp 29186616f5 Regenerated gemspec for version 0.4.5 2009-08-10 20:01:31 -04:00
Bryan Helmkamp 796318d1ed Prep for 0.4.5 release 2009-08-10 20:01:06 -04:00
Bryan Helmkamp 8a81f16e44 Use Jeweler for gem management 2009-08-10 19:50:42 -04:00
Bryan Helmkamp 273e8c541a History 2009-07-08 14:06:03 -04:00
Brian Landau dd98b33713 Make compatable with latest version of "selenium-client" gem, version 1.2.16 2009-07-06 13:50:00 -04:00
Brian Landau 7f36cf472a Update Selenium RC Server to version 1.0.1 2009-07-06 13:43:31 -04:00
Bryan Helmkamp 11996a15dc Updating to leverage latest rack-test 2009-06-25 21:13:54 -04:00
Simon Rozet 195f9c3544 Add Rack >= 1.0 as a dependency to gemspec 2009-06-26 01:43:12 +02:00
Simon Rozet 0a2b77c9c4 Require rack in lib/webrat.rb 2009-06-26 01:42:51 +02:00
Simon Rozet 6cda9c79b4 Implement Webrat::MIME on top of Rack::Mime 2009-06-26 01:39:22 +02:00
Simon Rozet 3760867d0f Implement file uploading for :rack 2009-06-25 23:41:51 +02:00
Simon Rozet f2882ef8b2 Deprecate :rack_test 2009-06-25 22:41:05 +02:00
Simon Rozet a5a91b32e9 Deprecate the :sinatra mode
* I took the conservative approach here: the sinatra code could
  be removed but that'd cause a regression. (using Sinatra::Application
  when `app` is not defined)

* I left the sinatra tests for now; they should be removed
  once we're fully confident with rack session ones.
2009-06-25 00:38:39 +02:00
Simon Rozet 051dfe188c Make sure Webrat is usable without the Methods mixin 2009-06-25 00:09:03 +02:00
Simon Rozet 042268c6a6 Simplify Methods#webrat_adapter
It is unecessary to have a special case for the :rack mode.
2009-06-25 00:07:38 +02:00
Simon Rozet f2b3b9891b Refactor Webrat::Session
* Webrat::Session.new now take a session adapter class as it's argument and
  delegates methods such as #response_body and #response_code to an instance
  of it.

* Some of these methods will go away in the future. However, *a lot*
  of specs depends on them so I've left them for now. The plan is to
  strip down the session adapter API to these three methods:

    * request
    * response_code
    * response_body

* I had to mark a spec as pending: spec/private/rails/rails_session_spec.rb
2009-06-25 00:07:30 +02:00
Damian Janowski cbc447223c Fix form fields being double-escaped with :rack
Signed-off-by: Simon Rozet <simon@rozet.name>
2009-06-24 18:57:55 +02:00
Simon Rozet cf8d891302 More integration specs for the rack mode
One test is still pending because form fields are double-escaped.
2009-06-24 18:57:55 +02:00
Simon Rozet 4c010d1c65 Use Rack::Lint to test the rack session 2009-06-24 18:57:55 +02:00
Simon Rozet 8cda77318c Fix wrong HTTP_HOST in rack session
Webrat relies on www.example.com while Rack::Test expects example.org
2009-06-24 18:57:38 +02:00
Simon Rozet 92ec1c3d92 Require rack/test in rack session 2009-06-24 15:14:08 +02:00
Simon Rozet 0a3b979772 Rename :rack_test mode to :rack 2009-06-24 15:13:27 +02:00
Simon Rozet 3e7886ecab Get rid of RackSession
Looks like dead code
2009-06-24 15:13:16 +02:00
Bryan Helmkamp 5eeceff9e8 Removing some vestigial code from Merb session 2009-06-17 00:19:26 -04:00
Ryan Carver c11f4868a9 Don't require merb-core/two-oh for multipart support. Instead, copy the code into Webrat 2009-06-17 00:06:32 -04:00
Bryan Helmkamp 1769075c25 History 2009-06-16 17:47:55 -04:00
Bryan Helmkamp 28f19616d6 Wait for app server to boot on 0.0.0.0 instead of application_address
application_address is used to know where Selenium should navitate to,
but we always are booting app servers on 0.0.0.0 so that's where we
should verify they exist
2009-06-16 17:44:42 -04:00
Mike Gaffney 9b5b9fad62 Added attribution
[#168 state:resolved]
2009-06-15 22:44:02 -05:00
Mike Gaffney 0c9944c4d8 Merge commit 'eddd97936'
Conflicts:
	spec/integration/rails/test/integration/webrat_test.rb
2009-06-15 22:40:02 -05:00
Bryan Helmkamp 9a42304513 Revert "make label location strategy work with unescaped strings"
This reverts commit 4daf037146.

This was causing failures in the Weplay selenium suite, where we
we're using partial matches. For example:

  fill_in "Write a message", ...

When the markup was:

  <label>Write a message to Bryan ...</label>
2009-06-15 17:59:13 -04:00
Bryan Helmkamp b4ae28c48c Add JRuby to "rake multiruby" 2009-06-15 00:46:56 -04:00
Bryan Helmkamp 953d35b65a Commenting out failing Rack spec for now 2009-06-15 00:17:44 -04:00
Bryan Helmkamp ae6e3dbfc1 Adding "rake multiruby" task 2009-06-15 00:16:42 -04:00
Bryan Helmkamp d9de87c41f Remove redgreen from rails integration tests too 2009-06-15 00:16:23 -04:00
Bryan Helmkamp 9b971a15c0 Require English so we can use $LAST_MATCH_INFO instead of $~ 2009-06-15 00:08:25 -04:00
Bryan Helmkamp 6720bbb476 History 2009-06-15 00:07:09 -04:00
Bryan Helmkamp 1b5cf4b0e6 Merge commit 'origin/master' 2009-06-15 00:04:59 -04:00
Bryan Helmkamp a015931578 Comment out redgreen because it has issues on Ruby 1.9 2009-06-15 00:03:46 -04:00
Mike Gaffney a0b10d9916 requiring specific rspec version 2009-06-14 22:58:33 -05:00
Mike Gaffney 4cce77ef0d removing rspec warnings 2009-06-14 22:37:35 -05:00
Bryan Helmkamp 45e19867d6 Merge commit 'manveru/dev' into ruby19
Conflicts:
	lib/webrat/core/elements/field.rb
2009-06-14 23:31:42 -04:00
Bryan Helmkamp f8af783185 Add spec for needing to ensure Fixnum#/ return an integer in redirect checking 2009-06-14 22:03:49 -04:00
Bryan Helmkamp c2409d78a7 History 2009-06-14 21:59:32 -04:00
Bryan Helmkamp 6231e6e4d1 Merge commit 'pdcawley/master' 2009-06-14 21:58:18 -04:00
Bryan Helmkamp d620e66bd8 Use Launchy to handle opening pages in the browser with cross-platform compatibility 2009-06-14 21:38:04 -04:00
Bryan Helmkamp d60671cd3d History 2009-06-14 21:20:01 -04:00
Bryan Helmkamp 3bc2d4d1b8 Merge commit 'rcarver/merb-uploads' 2009-06-14 21:18:47 -04:00
Jakub Kuźma 8b98540aef Ruby 1.9 compatibility: Avoid calling #to_s on Arrays
[#249 state:resolved]
2009-06-14 21:05:38 -04:00
Erin Staniland b75ff6221b Added missing quote from rewrite_css_and_image_references and added support for single quotes 2009-06-14 21:01:06 -04:00
Bryan Helmkamp 78a23abb6e Merge commit 'dchelimsky/rspec-fix' 2009-06-14 20:54:02 -04:00
Mike Gaffney f4141f787d [#261 state:awaiting-patch]
added a test that shows that selenium can't click a link with text that has parens in it
2009-06-08 12:08:53 -05:00
Michael Fellinger 8d2c027089 Fix replacing of &nbsp;, aka &#160; so it works on 1.9 2009-06-07 17:03:13 +09:00
Michael Fellinger 520081c93e Don't use $LAST_MATCH_INFO since we don't require English.rb, use $~ instead 2009-06-07 17:02:32 +09:00
Michael Fellinger 48a4ec905e Make sure Field#escaped_value also works when @value is an Array on 1.9 2009-06-07 16:45:19 +09:00
Michael Fellinger 9134b09b87 Make mechanize_session_spec pass on 1.9 2009-06-07 16:34:35 +09:00
Mike Gaffney 6bfdcc54cc Fixed Spec in Test Unit more thoroghly 2009-06-05 13:51:55 -05:00
Mike Gaffney dd8688f371 moved application_server to appservers/base 2009-06-04 15:19:05 -05:00
Mike Gaffney 1910204974 refactor>moved the app servers to a directory 2009-06-04 15:13:54 -05:00
Mike Gaffney 70fadbe7fc added attribution 2009-06-04 15:05:08 -05:00
Mike Gaffney 6d74cd935e [#246 state:resolved] added external application server if you have your own started 2009-06-04 15:03:00 -05:00
Mike Gaffney a6a4a7cfd9 moving app server lookup out into a factory 2009-06-04 14:44:06 -05:00
Mike Gaffney 6816c46d47 refactored tests to be more rspeccy 2009-06-04 14:24:17 -05:00
Mike Gaffney f3a12adaa9 whitespace changes 2009-06-04 12:18:31 -05:00
Mike Gaffney e46df8c6f9 made tests cleaner and better coverage 2009-06-03 22:06:07 -05:00
Mike Gaffney 977f643dca catches spec failed exception again 2009-06-03 18:49:32 -05:00
Mike Gaffney abac2023bc removed reference to Rspec from common code 2009-06-03 17:53:19 -05:00
mike.gaffney b2c423c49c [#50 state:resolved] added deprecation warning 2009-06-03 10:01:27 -05:00
mike.gaffney 796ede6cea expand name refactor 2009-06-02 18:45:02 -05:00
mike.gaffney 28c676b4fc [#234 state:resolved] added attribution 2009-06-02 18:44:44 -05:00
Thomas Jack 4daf037146 make label location strategy work with unescaped strings 2009-06-02 18:31:33 -05:00
Thomas Jack 2ae14f78b1 expose regexp escape problem in fill_in_test 2009-06-02 17:56:12 -05:00
Thomas Jack 57326e5846 Merge branch 'master' of git://github.com/brynary/webrat 2009-06-02 17:52:14 -05:00
mike.gaffney f8f254d517 [#257 state:resolved] fixed issue with rails encodd fields in webrat 2009-06-02 17:35:29 -05:00
mike.gaffney bd1655c8e2 [#257 state:awaiting-patch] here is a test proving the issue 2009-06-02 17:23:12 -05:00
mike.gaffney 2704609224 added clicking link by title in selenium mode 2009-06-01 12:26:37 -05:00
mike.gaffney 2219ab62d3 refactoring buttons to their own controller 2009-06-01 12:25:36 -05:00
mike.gaffney 5430930bac added test for clicking link by title 2009-06-01 12:17:29 -05:00
mike.gaffney b6d26480ea [#221 state:resolved] added attribution 2009-06-01 12:01:47 -05:00
mike.gaffney b85aacae1c From larrytheliquid
Merge commit '4d05a2cf2b2de34dda0a3e805ab11b60540578df'
2009-06-01 11:59:21 -05:00
mike.gaffney 0394a1fc15 [#254 state:resolved] Added attribution 2009-05-29 17:09:55 -05:00
Matthias Marschall 9b85b6d7e0 added integration tests for all cases of click_button (by id, by html, by value, by alt for buttons and inputs of types image, button and submit) 2009-05-29 17:05:35 -05:00
Matthias Marschall 6cd734aec9 made selenium integration tests run applying larrytheliquid's patch: 627913708c 2009-05-29 17:05:35 -05:00
Matthias Marschall 73dc59cc29 click_button in selenium works now same as in headless mode 2009-05-29 17:05:34 -05:00
mike.gaffney 6fc6530a6b [#256 state:resolved] Added attribution 2009-05-29 16:36:46 -05:00
Josh Lubaway 846a90e561 Passing TextField values as plaintext to Mechanize.
Mechanize takes input as plaintext and escapes field values on its own.
If the values are escaped before they are passed to Mechanize, the values will be escaped twice.

These doubly-escaped values will result in incorrect behavior on the server side.
2009-05-29 10:42:30 -07:00
Larry Diehl 4d05a2cf2b click_link matches anchor id's with Selenium 2009-05-28 23:23:09 -07:00
Larry Diehl 627913708c Made Selenium Rails integration tests at least runnable, although
there are failures.
2009-05-28 22:40:52 -07:00
mike.gaffney c3f067b551 added missing attribution 2009-05-28 18:05:28 -05:00
mike.gaffney e0bdab3236 [#194 state:resolved] updated history 2009-05-28 18:04:40 -05:00
mike.gaffney f6b0a763a0 Merge branch 'lh_194_delegate_field_named' of git://github.com/pivotal/webrat into lh194
Conflicts:
	lib/webrat/core/methods.rb
2009-05-28 18:01:53 -05:00
Piers Cawley 58601c4653 Make Webrat::Core::Session#redirect? work in the face of rational maths.
Such as when someone uses `ruby-units' and suddenly 302/100 becomes 151/50.
2009-05-27 12:35:36 +01:00
Thomas Jack 956b43d72c Merge branch 'master' of git://github.com/brynary/webrat 2009-05-19 14:24:41 -05:00
mike.gaffney fd431f2ce8 [#242 state:resolved]
allowed changing the default timeout for browser startup in selenium
2009-05-18 16:37:45 -05:00
Mike Gaffney c6fd28d37a [#238 state:resolved] merged 2009-05-13 23:15:16 -05:00
Mike Gaffney f342142a71 Merge commit 'snusnu/master' 2009-05-13 23:12:32 -05:00
Mike Gaffney fa3b2c8bb2 [#215 state:resolved] merged 2009-05-13 23:06:26 -05:00
Luke Amdor 7f13a70b4b Added current_url to SeleniumSession to have the same api for both sessions 2009-05-13 22:33:20 -05:00
Thomas Jack 232ed36379 spec for descendant selectors in have_xpath block 2009-05-13 14:52:53 -05:00
Thomas Jack db9c5bdc77 Merge branch 'master' of git://github.com/brynary/webrat 2009-05-13 14:34:16 -05:00
snusnu 77168fd29d only require silence_stream if active_support hasn't already defined it 2009-05-13 05:01:19 +02:00
snusnu 926bcc6c66 extracted silence_stream into its own file under webrat/selenium
I don't think silence_stream.rb should be stored under
core_extensions, because the way it's implemented, it simply is
no core extension. Also, a grep through webrat source shows that
silence_stream is only used inside the selenium support. This is
why I added webrat/selenium/silence_stream.rb and require it
*before* all other selenium related files in webrat/selenium.rb.
It's necessary to include it this early, because if mode is set
to :selenium, webrat/selenium.rb is required, which in turn
requires webrat/selenium/selenium_session.rb and silence_stream
must be available before selenium_session gets required because
selenium_session already wants to silence the stream.
2009-05-13 03:35:32 +02:00
snusnu bfa250e7af Provide an implementation for silence_stream inside webrat.
I added the implementation to webrat/core/configuration.rb because
it seems to me that this always gets read first. I decided to put
it inside a util module Webrat::SilentStream and include this where
appropriate, instead of altering Kernel.
2009-05-12 18:32:30 +02:00
Bryan Helmkamp 77fa1bcdb6 Ignore *.log 2009-05-11 16:51:57 -04:00
Bryan Helmkamp 7b6b3168a2 Revert "Correct some specs that need full URLs"
This reverts commit 3e71ae3733.

Conflicts:

	spec/public/select_spec.rb
2009-05-11 16:51:42 -04:00
Bryan Helmkamp b439d7f807 Revert "Canonicalize all URLs (Shalon Wood)"
This reverts commit 755cf6e508.
2009-05-11 16:48:28 -04:00
Thomas Jack 8bedf2235d Merge branch 'master' of git://github.com/brynary/webrat 2009-05-11 00:29:49 -05:00
Bryan Helmkamp 453cb4b3eb Stripping whitespace 2009-05-11 01:27:04 -04:00
Bryan Helmkamp 3e71ae3733 Correct some specs that need full URLs 2009-05-11 00:16:38 -04:00
Bryan Helmkamp 35cbfd9643 Use Nokogiri on JRuby -- ~2x faster than REXML on JRuby for me 2009-05-11 00:13:00 -04:00
Bryan Helmkamp b5254109f1 Start on Rack::Test integration 2009-05-11 00:12:27 -04:00
Bryan Helmkamp 755cf6e508 Canonicalize all URLs (Shalon Wood) 2009-05-11 00:12:27 -04:00
Thomas Jack 9a3668be92 match descendents in have_selector/have_xpath blocks 2009-04-29 01:33:26 -05:00
David Chelimsky ffb56aab90 config for rspec in one line 2009-04-27 11:54:46 -05:00
Mutwin Kraus eddd979361 Fixed following of internal redirects when using a custom Host header
Webrat didn't properly follow internal redirects when using the Host
  header, this lets Webrat check the Host header before falling back
  to www.example.com as the current_host.
2009-04-24 23:49:11 -07:00
Ryan Carver 1a110fe908 Improve forward and backward compatibility for Merb uploads 2009-04-24 14:23:00 -07:00
Ryan Carver 26edfbc7cb Integration tests for Merb file uploads 2009-04-24 14:11:49 -07:00
Ryan Carver 09509025c3 Allow Merb to do file uploads 2009-04-24 13:56:28 -07:00
mike.gaffney 481bfe03c0 updated history 2009-04-22 12:27:24 -05:00
mike.gaffney 7d4a2035f7 Merge commit '742db2d910d7baa51487400cb339a20bd7079418' 2009-04-22 12:15:27 -05:00
mike.gaffney b1dfc1c01f updating history for Zach Dennis's fix 2009-04-16 12:37:33 -05:00
Zach Dennis 7c08390bb8 Updated regex for LabelLocator and FieldLabeledLocator to work with labels whose text ends in a non-word character like \? or # 2009-04-16 12:25:45 -05:00
Noah Davis ff42db076f Select fields with duplicate selected options sent an incorrect value 2009-04-11 13:19:59 -04:00
Bryan Helmkamp 03914fd293 Whitespace 2009-04-07 20:30:12 -04:00
Bryan Helmkamp f3f81dabdc Add "rake whitespace" task 2009-04-07 20:29:59 -04:00
Bryan Helmkamp b69b4acec4 Change setup => before 2009-04-07 20:14:59 -04:00
Bryan Helmkamp 6697ecd2d3 Prep for 0.4.4 release 2009-04-06 14:03:58 -04:00
Bryan Helmkamp 5bb587dde1 History 2009-04-06 14:03:47 -04:00
Bryan Helmkamp 996484c890 Refactoring selenium RC and app server boot code 2009-04-06 14:02:09 -04:00
Bryan Helmkamp 979418e303 History 2009-04-06 12:40:28 -04:00
Bryan Helmkamp 644478f6f0 Merge commit 'MarkMenard/master' 2009-04-06 12:39:31 -04:00
Bryan Helmkamp 50d8a7ec51 History 2009-04-06 12:33:45 -04:00
Bryan Helmkamp 91ea8cfa54 Make selenium process management code more robust and informative 2009-04-06 12:33:24 -04:00
Bryan Helmkamp 304baeb754 Change default app environment from "selenium" to "test" 2009-04-06 12:33:24 -04:00
Bryan Helmkamp aa9049953d Bump selenium-client dependency to 1.2.14 2009-03-29 16:35:27 -04:00
Balint Erdi cd49c2d939 bugfix: selenium's wait_for_element method takes the number of seconds before timing out from an option hash
Signed-off-by: Balint Erdi <balint.erdi@gmail.com>
2009-03-29 15:57:14 -04:00
pivotal 0fe3998f64 field_named correctly delegated 2009-03-17 18:21:06 -04:00
Bryan Helmkamp e2a672a767 Prep 0.4.3 release 2009-03-17 01:26:38 -04:00
Andrew Premdas 742db2d910 added spec for click links by title - functionality is already done 2009-03-13 12:09:24 +00:00
Mark Menard 0c2261d869 Add support for Rails javascript post links. 2009-03-12 15:42:20 -04:00
Mark Menard 73f4c441c1 Merge commit 'upstream/master' 2009-03-12 15:41:37 -04:00
Bryan Helmkamp 62491121c1 History 2009-03-08 18:59:15 -04:00
Bryan Helmkamp d1a2c80ab8 Merge commit 'sr/sinatra' into sr_sinatra 2009-03-08 18:54:21 -04:00
Simon Rozet 90baf3311c Sinatra now provides sensible options for testing
In the lastest release, 0.9.1:

* Set sensible defaults on Default for test env [1]
  5a33a9586f

* Set :environment to :test in TestHarness
  9af37724e0

* Set :environment to :test when Sinatra::Test is included
  93b19a3492

[1] Only Sinatra::Default provides special options for testing. If you subclass
    Sinatra::Base, you'll have to set them yourself.

foo
2009-03-08 23:41:34 +01:00
Bryan Helmkamp 5dc24c613e Merge branch 'master' into sr_sinatra 2009-03-08 14:59:54 -04:00
Mark Menard c647ae98bb Memoize WWW::Mechanize instance. 2009-03-05 17:45:56 -05:00
Luke Melia 72123c1cec When faced with a label with no for attribute, that contains a hidden field and another field, as can be the case in Rails 2.3's checkbox view, webrat now locates the non-hidden field. 2009-03-01 01:44:35 -05:00
Matthew Ford 740bb293e3 Fix for #161 When using Rails 2.3 it uses Rack::Utils to parse params 2009-02-28 23:08:07 -05:00
Simon Rozet 4e07f5b654 Fix following of absolute redirect URL in Sinatra
What's going on when the app redirects to an absolute URL?

* Sinatra relies on Rack::MockSession which sets SERVER_NAME to
  "example.org" [1] on request it makes. However, Webrat expects
  it to be "www.example.com"

* In MyClassyApp, the redirect URL is made out of Rack::Request#url [2]
  which uses  SERVER_NAME, which is set to "example.org" by Rack::MockSession.
  As a result, Webrat see it as an external redirect and don't follow it.

NOTE:

* SERVER_NAME is stricly equivalent to HTTP_HOST [3]
* This could have been fixed in Webrat::Session too. I'am not sure
  that it won't affect other frameworks so I left it intact.

      def request_page(url, http_method, data) #:nodoc:
         h = headers
         h['HTTP_REFERER'] = @current_url if @current_url
  +      h['HTTP_HOST']    = 'www.example.com'

[1] 5c00dd698e/lib/rack/mock.rb (L79)
[2] 5c00dd698e/lib/rack/request.rb (L201)
[3] 5c00dd698e/lib/rack/request.rb (L72)
2009-02-28 03:30:26 +01:00
Simon Rozet 7d7c32256d Support for "modular" Sinatra app style
Since Sinatra 0.9, apps can be ancapsulated in a class ("modular" style)
or be on the top level, just like before ("classic" style)

* http://wiki.github.com/brynary/webrat/sinatra
* http://www.sinatrarb.com/extensions.html
2009-02-28 03:30:25 +01:00
Bryan Helmkamp 00433bafe5 History 2009-02-27 20:45:51 -05:00
Bryan Helmkamp 4115c99c9b Merge branch 'master' into atmos 2009-02-27 20:41:05 -05:00
Bryan Helmkamp b7e0d6f8e6 Prep for 0.4.2 release 2009-02-24 12:29:05 -05:00
Bryan Helmkamp e32a208b7f Fix warning caused by Nokogiri deprecating CSS::Parser.parse (Aaron Patterson) [#176 state:resolved] 2009-02-23 22:19:02 -05:00
Bryan Helmkamp aec889a14d Always deal with one query selector (a String, not an Array) 2009-02-23 22:14:58 -05:00
Bryan Helmkamp dad3da195c Bumping nokogiri dependency version to 1.2.0 2009-02-23 21:18:17 -05:00
Bryan Helmkamp 18c60f4e51 Refactoring XPath manipulation 2009-02-23 20:52:17 -05:00
Bryan Helmkamp 29c40bd73c Pulling attribute conditions up from have_selector to have_xpath 2009-02-23 20:47:02 -05:00
Bryan Helmkamp 1625e3e9ba Pulling :content functionality up from have_selector into have_xpath 2009-02-23 20:22:36 -05:00
Bryan Helmkamp a688c28b19 Extracting xpath_escape method 2009-02-23 20:21:23 -05:00
Bryan Helmkamp 9761b64b60 Pulling :count option up from have_selector to have_xpath 2009-02-23 20:18:36 -05:00
Bryan Helmkamp 66d39b851c Allow have_xpath to accept options 2009-02-23 20:15:29 -05:00
Bryan Helmkamp 2296cadb93 Pulling features up from have_tag into have_selector 2009-02-23 20:11:25 -05:00
Corey Donohoe 120a53dced small doc fixup to explain how to use different application frameworks with selenium 2009-02-23 14:21:42 -07:00
Bryan Helmkamp d96899be8c Remove extra whitespace from contain failure message 2009-02-23 11:19:09 -05:00
Corey Donohoe ec06c76b91 use bin/merb if it's available for appserver startup 2009-02-18 12:31:52 -07:00
Corey Donohoe 92565d3bc4 use application_environment in merb also 2009-02-18 12:16:19 -07:00
Bryan Helmkamp b4687c29d5 Break out specs for matchers into separate files 2009-02-15 19:47:12 -05:00
Bryan Helmkamp 17a8bc7b66 Adding :count option to have_tag 2009-02-15 19:41:44 -05:00
Bryan Helmkamp 932fdab884 Get quoting working for content 2009-02-15 19:32:25 -05:00
Bryan Helmkamp 7ba620f38b Whitespace 2009-02-15 18:57:53 -05:00
Corey Donohoe 4fb9406154 remove commented line that serves no purpose 2009-02-15 15:54:04 -07:00
Corey Donohoe 63337d4476 it helps if i make sure the spec suite still passes :\ 2009-02-15 14:41:54 -07:00
Corey Donohoe 69dfa5022a Webrat.configuration.application_type => Webrat.configuration.application_framework 2009-02-15 14:40:17 -07:00
Corey Donohoe 7692930769 clean up constant definitions for rails and fix merb port fubar on selenium appserver stop 2009-02-15 14:37:38 -07:00
Corey Donohoe 9066d2a9cd add specs for starting and stopping appservers, covers rails, sinatra and merb 2009-02-13 20:26:44 -07:00
Corey Donohoe 4bf49a5163 allow picking your appserver type via a configuration option, not discovering framework specific files in the current working directory. also start rack apps with the application port parameter 2009-02-11 11:17:43 -07:00
Corey Donohoe 8fbcbef180 better handling for starting/stopping the server 2009-02-11 10:09:23 -07:00
Corey Donohoe 82eabc31ee small fixup for sinatra + selenium, use rackup instead of mongrel_rails if config.ru exists 2009-02-11 09:25:39 -07:00
Peter Jaros 24eab77ecd Accept do/end blocks in matchers. [#157 state:resolved] 2009-02-09 00:31:17 -05:00
Bryan Helmkamp 7fe667da73 Removing HashWithIndifferentAccess
Doesn't seem to be required, according to our integraton specs
2009-02-09 00:13:41 -05:00
Bryan Helmkamp a89e3ee612 History [#159 state:resolved] 2009-02-08 21:05:58 -05:00
Bryan Helmkamp 7fc6a79bee Merge commit 'teejayvanslyke/master' into lh_159 2009-02-08 21:03:40 -05:00
Bryan Helmkamp e97f7fafc5 Add a spec for two checkboxes with the same name (array style) 2009-02-08 20:58:43 -05:00
Bryan Helmkamp 4769a5f90b Detect infinite redirects and raise a Webrat::InfiniteRedirectError (Daniel Lucraft) 2009-02-08 20:54:00 -05:00
Bryan Helmkamp e5ae16367c Prep for 0.4.1 release 2009-01-31 17:58:17 -05:00
Bryan Helmkamp d9de0c4e78 Remove gemspec. I don't want GitHub to build gems... 2009-01-31 17:57:35 -05:00
Bryan Helmkamp aa97b063f7 History 2009-01-31 17:31:35 -05:00
Lee Bankewitz 56e6d15811 add uncheck to selenium_session 2009-01-30 15:15:22 -05:00
T.J. VanSlyke ea8e1910d1 Added single-quotes around --chdir argument to mongrel_rails call, as
webrat bombs with a file not found error if RAILS_ROOT contains a space.
2009-01-30 11:37:43 -08:00
Noah Davis 49e86d103f comment out have_tag from selenium matcher which overrides rails' have_tag
partially reverts 483559f279
2009-01-28 14:34:24 -05:00
Bryan Helmkamp c25885afc0 History 2009-01-27 19:16:32 -05:00
Bryan Helmkamp 7f9df54838 Merge commit 'lenalena/master' into lh_139 2009-01-27 19:13:53 -05:00
Bryan Helmkamp d2d8de7963 History 2009-01-27 19:11:28 -05:00
Bryan Helmkamp 91edb22b81 History 2009-01-25 20:02:40 -05:00
Bryan Helmkamp 33509ef575 Remove stray require of ruby-debug 2009-01-25 19:50:58 -05:00
Bryan Helmkamp 238bfb469e Use redgreen when running rails integraton tests if available 2009-01-25 19:48:02 -05:00
Bryan Helmkamp 0039bf4d4b Check exit code of spec:jruby when running precommit 2009-01-25 19:46:22 -05:00
Bryan Helmkamp 6836115308 History 2009-01-25 19:45:58 -05:00
Mike Gaffney 3d5dd1343c turning off have_tag test for now until we figure out whether we want it or not. Making integration not run the selenium tests for now. Need a eventually framework for test::unit 2009-01-25 18:30:03 -06:00
Mike Gaffney 8d9d907e73 have_tag is built into rails and rspec and we don't want to overload those.
Please use assert_contain or assert_select
Revert "Make have_tag work in rails as well as Merb"

This reverts commit 6e7609e01c.
2009-01-24 13:01:41 -06:00
Noah Davis f956bea7dd Adding rescue from Webrat::TimeoutError in selenium matchers which allows NegativeMatchers to behave correctly 2009-01-24 01:44:05 -05:00
Mike Gaffney 18fa3ab11c Merge branch 'master' of git@github.com:gaffo/webrat
Conflicts:
	History.txt
2009-01-23 22:55:32 -06:00
Amos King 40280a4a0d fix history to be in correct version. 2009-01-23 22:53:29 -06:00
Mike Gaffney fc4ba75e22 Merge branch 'master' of git@github.com:gaffo/webrat 2009-01-23 22:46:16 -06:00
Amos King 099341cede Forgot to add Mike 2009-01-23 22:42:34 -06:00
Mike Gaffney f4f0e3b6a7 Merge branch 'master' of git@github.com:brynary/webrat 2009-01-23 22:35:46 -06:00
Amos King 483559f279 Makes have_tag work outside of Merb. 2009-01-23 21:32:54 -06:00
Amos King 0a021059d6 Added to rails test for have_tag 2009-01-23 21:13:52 -06:00
Amos King 6e7609e01c Make have_tag work in rails as well as Merb 2009-01-23 21:13:13 -06:00
Amos King 118b522662 moved have content 2009-01-23 21:03:19 -06:00
Amos King 3a5d34e759 Moved have selector 2009-01-23 21:01:21 -06:00
Amos King 5a91f10a99 Began moving selenium matchers to their own files to be consistent with core matchers and for maintainability. 2009-01-23 20:59:30 -06:00
Amos King 0b8e378fb2 change to use rails label tag for test. 2009-01-23 20:56:16 -06:00
Amos King 5b8e451272 forgot to update History file 2009-01-23 20:37:35 -06:00
mike.gaffney cbddebd01c Merge branch 'lh_82' of git@github.com:gaffo/webrat 2009-01-23 16:23:12 -06:00
Amos King a1ef306838 update Rakefile to run test_units with the new test task 2009-01-23 16:13:43 -06:00
Noah Davis 996eedb0d2 Switch to using selenium.click instead of .check when checking a checkbox 2009-01-23 17:08:04 -05:00
Amos King 8c0facc5af cleanup and tests closes LH #82 2009-01-23 16:05:33 -06:00
Amos King 813adcc238 Remove the rails selenium test directory and change rake tasks for selenium and rails modes to run the same tests. The end goal should be that the same tests run in both Seleium and Webrat. 2009-01-23 15:37:26 -06:00
Amos King da59584cdf add the ability for selenium mode to create pids dirs 2009-01-23 15:34:51 -06:00
Amos King 2c9b546a83 Merge branch 'lh_120_tests_with_selenium' of git@github.com:gaffo/webrat into lh_82
Conflicts:
	spec/integration/rails/test/integration/webrat_test.rb
2009-01-23 15:11:16 -06:00
Zach Dennis 0272e81847 Fixed issue where webrat was only using relative paths when making requests for Rails apps. This borked Rails apps that use subdomains [John Hwang/Zach Dennis] 2009-01-20 21:48:48 -05:00
Josh Knowles 7a59353c78 Unfortunately this commit breaks any Rails specs which pass anchor tags. I like the concept, but I'd prefer to work on this post 0.4.1 once I have a bit more time to work out the failures in our referance app.
This reverts commit 4fc2b7eb7e.
2009-01-20 20:08:57 -05:00
Adam Greene 4fc2b7eb7e removing the normalization of url's for the rails session. stipping them of host, port, and anchors is not really needed, especially since josh's redirect change to stop when going to a different host. See comments here: http://webrat.lighthouseapp.com/projects/10503-webrat/tickets/132 2009-01-20 19:30:28 -05:00
Adam Greene 4e3cf59920 adding a helper method to make it easier to see where the user was redirected_to 2009-01-20 19:30:22 -05:00
Adam Greene ce36e5890f lets follow redirects that are on the same domain but a different subdomain... 2009-01-20 19:30:16 -05:00
Bryan Helmkamp c06fa90864 History 2009-01-20 00:37:03 -05:00
Bryan Helmkamp e53c64f763 Setup deprecated writers for the selenium_environment= and selenium_port= config methods 2009-01-20 00:32:19 -05:00
Bryan Helmkamp 9a2302e9af History 2009-01-20 00:27:38 -05:00
Bryan Helmkamp d60f524df1 Merge commit '1bfe7634f8bb93472f4f53715d43916a8ff1569f' 2009-01-20 00:26:36 -05:00
Bryan Helmkamp 0204c2766d Change query params parser to work with Edge Rails 2009-01-20 00:25:52 -05:00
Bryan Helmkamp d64bba627e History 2009-01-19 20:03:21 -05:00
ichverstehe 1bfe7634f8 sinatra 0.9 support 2009-01-19 23:51:01 +01:00
Kyle Hargraves 6529a016d1 Labels should only search for fields within the current scope 2009-01-19 13:45:36 -06:00
Bryan Helmkamp 14d114ce1d Ensure the previous pages params aren't passed through redirect 2009-01-19 13:56:22 -05:00
Bryan Helmkamp ced63f6e5a Don't install RDoc when installing locally built gem 2009-01-19 13:48:58 -05:00
Bryan Helmkamp b8b4a614c4 Use reload instead of deprecated reloads in spec 2009-01-19 13:40:58 -05:00
Amos King 652610381a added selenium test for assert_selector 2009-01-13 22:48:57 -06:00
Amos King fd860a2e6d added rails test for assert_selector 2009-01-13 22:48:06 -06:00
Amos King a32ec70d7f added xpath test to selenium 2009-01-13 22:45:26 -06:00
Amos King 647eb4cb6f add test for assert_xpath 2009-01-13 22:43:31 -06:00
Amos King 2d065e8a2d add test for click_link with id to selenium 2009-01-13 22:43:14 -06:00
Amos King 0aa5fda61d added click link test to selenium 2009-01-13 22:35:58 -06:00
Amos King 2510e7d43d added clicking link by id 2009-01-13 22:34:47 -06:00
Amos King 3902573d3e add a link to the rendered text and add a test for clicking it 2009-01-13 22:33:47 -06:00
Amos King 9745673f1c Revert "remove commented code and make assert_contain use a true assertion"
This reverts commit 114db434182bbf9ad5b2737d181048bab363f055.
2009-01-13 22:30:52 -06:00
Amos King cdd390a19d remove commented code and make assert_contain use a true assertion 2009-01-13 22:30:52 -06:00
Amos King a396758da4 fixed rakefile so the tasks will run the correct mode, and so there is a task to run both modes. 2009-01-13 07:57:20 -06:00
Mike Gaffney 758e323f89 breaking up selenium from rails_webrat tests for the moment 2009-01-13 00:26:49 -06:00
Mike Gaffney c0220232c7 broke selenium mode tests out into a seperate file. 2009-01-13 00:13:17 -06:00
Mike Gaffney c1b49f36ea working on getting standardized for the specs 2009-01-12 23:31:14 -06:00
Mike Gaffney 86089d90b6 Merge commit 'origin/lh_116_selenium_asserts' into test_unit_integrations 2009-01-12 23:14:37 -06:00
Mike Gaffney f7f8c91cb7 getting selenium up and running 2009-01-12 22:54:33 -06:00
Mike Gaffney f2a202ae5e adding pids directories 2009-01-12 22:44:53 -06:00
Mike Gaffney f835bac31b cleaning out app and getting selenium mode to work 2009-01-12 22:42:35 -06:00
Mike Gaffney bdf660aecb Renamed the task for seleniumw 2009-01-12 22:30:36 -06:00
Mike Gaffney a8a63472bc breaking the test mode out into a specific test modes 2009-01-12 22:30:03 -06:00
Lena b4d85cb50b Added recognition of input tags with type button, which are generated by the Rails submit_to_remote helper 2009-01-07 12:27:47 +01:00
186 changed files with 5029 additions and 2731 deletions

6
.gitignore vendored
View File

@ -11,4 +11,8 @@ log
*.swp *.swp
results results
test_apps test_apps
*.tmproj *.tmproj
*.log
*.pid
bin
vendor/gems

13
Gemfile Normal file
View File

@ -0,0 +1,13 @@
gem "rake", "0.8.7"
gem "rspec", "1.2.9"
gem "diff-lcs", "1.1.2"
gem "rack", "1.0.1"
gem "nokogiri", "1.4.0"
gem "merb-core", "1.0.15"
gem "selenium-client", "1.2.17"
gem "mechanize", "0.9.3"
gem "rails", "2.3.4"
gem "launchy", "0.3.3"
gem "rack-test", "0.5.2"
gem "sinatra", "0.9.4"
gem "mongrel", "1.1.5"

View File

@ -1,3 +1,216 @@
== 0.7.1 / 2010-04-26
* Minor enhancements
* Move verbose selenium output that can clutter build output behind setting
* Added application_port_for_selenium to webrat configuration. The use case is when you want to test through a web server sitting in front of your application server. (Luke Melia)
* New webrat configuration option selenium_firefox_profile which is passed to selenium server
* Allow submit_form to select by CSS too (Damian Janowski)
* Bug fixes
* Fix that current_url wasn't reflecting redirects in Mechanize [#332] (Emrys Ingersoll)
* Fix attach_file with nested params [#341] (Álvaro Gil)
* Fix that a 304 was considered a redirect (Larry Marburger)
* Fix selection of LABEL elements in Selenium mode under IE [#317] (Damian Janowski, Noah Davis)
* Fix have_xpath not matching negative expectation in the block [#182] (Luismi Cavallé)
== 0.7.0 / 2010-01-17
* Major enhancements
* Upgrade bundled Selenium server JAR to 2.0a1 (Henry Poydar and Jake Scruggs)
* Minor enhancements
* Save and open page directory specified via configuration, defaults to tmp dir otherwise current dir (Noah Davis)
* Bug fixes
* Added missing dependency "rack-test" to gemspec (LH #339) (Noah Davis)
* Removed save_and_open_page's rewriting of static asset paths before saving (was not actually working) (Noah Davis)
* Make "should contain" ignore extra whitespace when doing string comparisons (Noah Davis)
* Make selenium matchers handle negative match more consistently with positive match (Luke Melia)
== 0.6.0 / 2009-11-28
REMOVED: Support for Hpricot + REXML as an alternative to Nokogiri.
Hpricot and REXML were difficult to work with, REXML is terribly slow,
and Nokogiri is recommended even by the author of Hpricot (_why). Now
that Nokogiri works on JRuby, Webrat is going to use it as its sole
XML backend.
CHANGED: Due to a reorganization, if you're currently requiring "webrat/rspec-rails",
please change your code to require "webrat/integrations/rspec-rails"
* Minor enhancements
* Support Rails 2.3.4 JavaScript form authenticity tokens in simulated mode (Jonathan Weiss)
* When a timeout occurs in wait_for, include the HTML from Selenium in the exception
* Update the Merb support to be based directly on Rack (Simon Rozet)
* Support multiple select fields (Kieran P)
* When locating select options, always match against text, not HTML
* Bug fixes
* Remove newlines from HTTP Basic authentication credentials (Michael Klett)
* Require nokogiri form rspec-rails.rb (David Chelimsky)
* Fix logger issue when running inside Cucumber (Damian Janowski)
* Fix various issues related to submitting values with HTML entities (Kieran P)
* Call #to_i on the :count option in matchers (Michael Christenson II)
* Fix bug where multiline param values were getting truncated
== 0.5.3 / 2009-08-27
* Minor enhancements
* Remove unnecessary requires which are to the wrong paths on Edge Rails
== 0.5.1 / 2009-08-18
* Minor enhancements
* Base Webrat::MIME on Rack::Mime (Simon Rozet)
* Support file uploads in Rack mode (Simon Rozet)
* Bug fixes
* Fix bug in Webrat::Methods preventing Selenium mode from working [#277]
== 0.5.0 / 2009-08-12
* Major enhancements
* Depreacate :rack_test and :sinatra in favor of :rack, which uses Rack::Test (Simon Rozet)
* Minor enhancements
* Don't require rubygems at runtime (Simon Rozet)
== 0.4.5 / 2009-08-10
* Major enhancements
* Ruby 1.9 compatibility (Michael Fellinger, Jakub Kuźma)
* Improve performance (~2x) on JRuby by supporting Nokogiri
* Added support for external app servers in selenium mode (basically don't start one) (Mike Gaffney)
* Added support for uploading files for Merb (Ryan Carver)
* Minor enhancements
* Upgrade to selenium-client to 1.2.16 (Brian Landau)
* Upgrade selenium-server.jar to 1.0.1 (Brian Landau)
* Make redirect detection work in the face of rational maths (like when ruby-units is active) (Piers Cawley)
* Use Launchy to handle opening pages in the browser with cross-platform compatibility (Bryan Helmkamp)
* Added support for field_labeled_locators ending in non word characters
lh 148 (Zach Dennis)
* Filled in tests on click link lh 195 (diabolo)
* Added current_url to selenium session lh 215 (Luke Amdor)
* Added silence spec to selenium lh 238 (Martin Gamsjaeger aka snusnu)
* Added ability to configure the browser startup timeout for selenium lh 242 (Mike Gaffney, Mark Dodwell)
* Added delegation for field_named lh194 (pivotal labs)
* Added fix to keep from escaping field values in mechanize mode lh256 (jish)
* Adding fixes for click button/link and made the integration specs pass for the same in selenium lh254 (Ldiehl, Matthias Marschall)
* Adding clicking link by id in selenium mode lh221 (larrytheliquid)
* Adding fix for have_selector and have_xpath in descendent blocks lh234 (Thomas Jack)
* Adding fix for fields with labels with special characters (Thomas Jack, Mike Gaffney, Bryan Hemlkamp)
* Deprecated current_page lh50 (Mike Gaffney)
* Fixed issue with redirects and multiple hosts lh168 (Mutwin Kraus)
* Bug fixes
* Wait for application servers socket on 0.0.0.0 instead of the application_address
* Translate CSS and image paths with single quotes in save_and_open_page (Erin Staniland)
== 0.4.4 / 2009-04-06
* Major enhancements
* Make selenium process management code more robust and informative
* Minor enhancements
* Add support for Rails javascript post links (Mark Menard)
* Upgrade selenium-client dependency to 1.2.14, and update for new waiting
API (Balint Erdi)
* Change default app environment from "selenium" to "test"
* Bug fixes
* Don't create a new instance of WWW::Mechanize for each request
(Mark Menard)
* Select fields with duplicate selected options sent an incorrect value (Noah Davis)
== 0.4.3 / 2009-03-17
* Minor enhancements
* Support Rails 2.3. Use Rack::Utils to parse params (Matthew Ford)
* Support for "modular" Sinatra app style (Simon Rozet)
* Initial Merb and Sinatra compatibility for Selenium mode (Corey Donohoe)
* When faced with a label with no for attribute, that contains a hidden field
and another field, as can be the case in Rails 2.3's checkbox view,
webrat now locates the non-hidden field. (Luke Melia)
* Add application_framework config for Selenium mode to determine how to
start and stop the app server (Corey Donohoe)
* Bug fixes
* Fix following of absolute redirect URL in Sinatra (Simon Rozet)
== 0.4.2 / 2009-02-24
* Major enhancements
* Significant improvements to have_selector. It now supports specifying
attributes in a hash and :count and :content options. See
have_selector_spec.rb for more.
* Add the same functionality mentioned above to have_xpath
* Minor enhancements
* Squeeze extra whitespace out of failures messages from contain
matcher
* Detect infinite redirects and raise a Webrat::InfiniteRedirectError
(Daniel Lucraft)
* Bug fixes
* Properly quote single and double quotes strings in XPath
* Fix warning caused by Nokogiri deprecating CSS::Parser.parse
(Aaron Patterson)
* Accept do/end blocks in matchers. [#157] (Peter Jaros)
* Quote --chdir option to mongrel_rails to support RAILS_ROOTs with spaces
(T.J. VanSlyke)
== 0.4.1 / 2009-01-31
* Minor enhancements
* Support Sinatra 0.9 (Harry Vangberg)
* Update query param parsing to work with latest Edge Rails
* Added #redirected_to method to easily check where an external redirect was
redirected to (Adam Greene)
* Recognize input tags with type button (Lena Herrmann)
* Add uncheck method to Selenium mode (Lee Bankewitz)
* Bug fixes
* Make requests to a Rails app using a full URL instead of a relative path. This change
is helpful for Rails apps that use subdomains. (John Hwang and Zach Dennis)
* Follow redirects that are on the same domain but a different subdomain
(Adam Greene)
* rescue from Webrat::TimeoutError in selenium matchers which allows NegativeMatchers
to behave correctly (Noah Davis)
* Switch to using selenium.click instead of .check when checking a checkbox
(Noah Davis)
* Create tmp/pids directory if directory does not exist. (Amos King and Mike Gaffney)
* Setup deprecated writers for the selenium_environment= and selenium_port= config
* Ensure the previous pages params aren't passed through redirect (Daniel Lucraft and
Bryan Helmkamp)
* Labels should only search for fields within the current scope (Kyle Hargraves)
== 0.4.0 / 2009-01-18 == 0.4.0 / 2009-01-18
* _IMPORTANT_ Breaking change: * _IMPORTANT_ Breaking change:
@ -68,7 +281,7 @@
* Added save_and_open_screengrab for Selenium mode (Luke Melia) * Added save_and_open_screengrab for Selenium mode (Luke Melia)
* Bug fixes * Bug fixes
* field_labeled should disregard labels without matching fields (Kyle Hargraves) * field_labeled should disregard labels without matching fields (Kyle Hargraves)
* Fixed bug where Scope was creating a new DOM rather than re-using the existing DOM. * Fixed bug where Scope was creating a new DOM rather than re-using the existing DOM.
[#105] (Zach Dennis) [#105] (Zach Dennis)

198
Rakefile
View File

@ -1,93 +1,67 @@
# require 'rubygems' begin
require "rake/gempackagetask" require 'spec/rake/spectask'
require 'rake/rdoctask' rescue LoadError
require "rake/clean" desc "Run specs"
require 'spec' task(:spec) { $stderr.puts '`gem install rspec` to run specs' }
require 'spec/rake/spectask' else
require 'spec/rake/verify_rcov' desc "Run API and Core specs"
require File.expand_path('./lib/webrat.rb') Spec::Rake::SpecTask.new do |t|
t.spec_opts = ['--options', "\"#{File.dirname(__FILE__)}/spec/spec.opts\""]
t.spec_files = FileList['spec/public/**/*_spec.rb'] + FileList['spec/private/**/*_spec.rb']
end
############################################################################## desc "Run all specs in spec directory with RCov"
# Package && release Spec::Rake::SpecTask.new(:rcov) do |t|
############################################################################## t.spec_opts = ['--options', "\"#{File.dirname(__FILE__)}/spec/spec.opts\""]
spec = Gem::Specification.new do |s| t.spec_files = FileList['spec/public/**/*_spec.rb'] + FileList['spec/private/**/*_spec.rb']
s.name = "webrat" t.rcov = true
s.version = Webrat::VERSION t.rcov_opts = lambda do
s.platform = Gem::Platform::RUBY IO.readlines(File.dirname(__FILE__) + "/spec/rcov.opts").map {|l| l.chomp.split " "}.flatten
s.author = "Bryan Helmkamp" end
s.email = "bryan" + "@" + "brynary.com"
s.homepage = "http://github.com/brynary/webrat"
s.summary = "Webrat. Ruby Acceptance Testing for Web applications"
s.bindir = "bin"
s.description = s.summary
s.require_path = "lib"
s.files = %w(History.txt install.rb MIT-LICENSE.txt README.rdoc Rakefile) + Dir["lib/**/*"] + Dir["vendor/**/*"]
# rdoc
s.has_rdoc = true
s.extra_rdoc_files = %w(README.rdoc MIT-LICENSE.txt)
# Dependencies
s.add_dependency "nokogiri", ">= 1.1.0"
s.rubyforge_project = "webrat"
end
Rake::GemPackageTask.new(spec) do |package|
package.gem_spec = spec
end
desc 'Show information about the gem.'
task :debug_gem do
puts spec.to_ruby
end
CLEAN.include ["pkg", "*.gem", "doc", "ri", "coverage"]
desc "Upload rdoc to brynary.com"
task :publish_rdoc => :docs do
sh "scp -r doc/ brynary.com:/apps/uploads/webrat"
end
desc "Run API and Core specs"
Spec::Rake::SpecTask.new do |t|
t.spec_opts = ['--options', "\"#{File.dirname(__FILE__)}/spec/spec.opts\""]
t.spec_files = FileList['spec/public/**/*_spec.rb'] + FileList['spec/private/**/*_spec.rb']
end
desc "Run all specs in spec directory with RCov"
Spec::Rake::SpecTask.new(:rcov) do |t|
t.spec_opts = ['--options', "\"#{File.dirname(__FILE__)}/spec/spec.opts\""]
t.spec_files = FileList['spec/public/**/*_spec.rb'] + FileList['spec/private/**/*_spec.rb']
t.rcov = true
t.rcov_opts = lambda do
IO.readlines(File.dirname(__FILE__) + "/spec/rcov.opts").map {|l| l.chomp.split " "}.flatten
end end
end end
RCov::VerifyTask.new(:verify_rcov => :rcov) do |t| desc "Run everything against multiruby"
t.threshold = 96.2 # Make sure you have rcov 0.7 or higher! task :multiruby do
end result = system "multiruby -S rake spec"
raise "Multiruby tests failed" unless result
result = system "jruby -S rake spec"
raise "JRuby tests failed" unless result
desc 'Install the package as a gem.' Dir.chdir "spec/integration/rails" do
task :install_gem => [:clean, :package] do result = system "multiruby -S rake test_unit:rails"
gem_filename = Dir['pkg/*.gem'].first raise "Rails integration tests failed" unless result
sh "sudo gem install --local #{gem_filename}"
end
desc "Delete generated RDoc" result = system "jruby -S rake test_unit:rails"
task :clobber_docs do raise "Rails integration tests failed" unless result
FileUtils.rm_rf("doc") end
end
desc "Generate RDoc" Dir.chdir "spec/integration/merb" do
task :docs => :clobber_docs do result = system "multiruby -S rake spec"
system "hanna --title 'Webrat #{Webrat::VERSION} API Documentation'" raise "Merb integration tests failed" unless result
end
desc "Run specs using jruby" result = system "jruby -S rake spec"
task "spec:jruby" do raise "Rails integration tests failed" unless result
system "jruby -S rake spec" end
Dir.chdir "spec/integration/sinatra" do
result = system "multiruby -S rake test"
raise "Sinatra integration tests failed" unless result
result = system "jruby -S rake test"
raise "Sinatra integration tests failed" unless result
end
Dir.chdir "spec/integration/rack" do
result = system "multiruby -S rake test"
raise "Rack integration tests failed" unless result
result = system "jruby -S rake test"
raise "Rack integration tests failed" unless result
end
puts
puts "Multiruby OK!"
end end
desc "Run each spec in isolation to test for dependency issues" desc "Run each spec in isolation to test for dependency issues"
@ -99,27 +73,38 @@ task :spec_deps do
end end
end end
task :prepare do
system "ln -s ../../../../.. ./spec/integration/rails/vendor/plugins/webrat"
end
namespace :spec do namespace :spec do
desc "Run the integration specs" desc "Run the integration specs"
task :integration => ["integration:rails", "integration:merb", "integration:sinatra"] task :integration => [
"integration:rack",
"integration:sinatra",
"integration:merb",
"integration:mechanize",
"integration:rails:webrat",
"integration:rails:selenium",
]
namespace :integration do namespace :integration do
desc "Run the Rails integration specs" namespace :rails do
task :rails do task :selenium do
Dir.chdir "spec/integration/rails" do Dir.chdir "spec/integration/rails" do
result = system "rake test:integration" result = system "rake -rubygems test_unit:selenium"
raise "Rails integration tests failed" unless result raise "Rails integration tests failed" unless result
end
end
task :webrat do
Dir.chdir "spec/integration/rails" do
result = system "rake -rubygems test_unit:rails"
raise "Rails integration tests failed" unless result
end
end end
end end
desc "Run the Merb integration specs" desc "Run the Merb integration specs"
task :merb do task :merb do
Dir.chdir "spec/integration/merb" do Dir.chdir "spec/integration/merb" do
result = system "rake spec" result = system "rake -rubygems spec"
raise "Merb integration tests failed" unless result raise "Merb integration tests failed" unless result
end end
end end
@ -127,13 +112,32 @@ namespace :spec do
desc "Run the Sinatra integration specs" desc "Run the Sinatra integration specs"
task :sinatra do task :sinatra do
Dir.chdir "spec/integration/sinatra" do Dir.chdir "spec/integration/sinatra" do
result = system "rake test" result = system "rake -rubygems test"
raise "Sinatra tntegration tests failed" unless result raise "Sinatra integration tests failed" unless result
end
end
desc "Run the Sinatra integration specs"
task :rack do
Dir.chdir "spec/integration/rack" do
result = system "rake -rubygems test"
raise "Rack integration tests failed" unless result
end
end
desc "Run the Mechanize integration specs"
task :mechanize do
Dir.chdir "spec/integration/mechanize" do
result = system "rake -rubygems spec"
raise "Mechanize integration tests failed" unless result
end end
end end
end end
end end
task :default => :spec desc 'Removes trailing whitespace'
task :whitespace do
sh %{find . -name '*.rb' -exec sed -i '' 's/ *$//g' {} \\;}
end
task :precommit => ["spec", "spec:jruby", "spec:integration"] task :default => :spec

118
Thorfile Normal file
View File

@ -0,0 +1,118 @@
module GemHelpers
def generate_gemspec
$LOAD_PATH.unshift(File.expand_path(File.join(File.dirname(__FILE__), "lib")))
require "webrat"
Gem::Specification.new do |s|
s.name = "webrat"
s.version = Webrat::VERSION
s.author = "Bryan Helmkamp"
s.email = "bryan@brynary.com"
s.homepage = "http://github.com/brynary/webrat"
s.summary = "Ruby Acceptance Testing for Web applications"
s.description = <<-EOS.strip
Webrat lets you quickly write expressive and robust acceptance tests
for a Ruby web application. It supports simulating a browser inside
a Ruby process to avoid the performance hit and browser dependency of
Selenium or Watir, but the same API can also be used to drive real
Selenium tests when necessary (eg. for testing AJAX interactions).
Most Ruby web frameworks and testing frameworks are supported.
EOS
s.rubyforge_project = "webrat"
require "git"
repo = Git.open(".")
s.files = normalize_files(repo.ls_files.keys - repo.lib.ignored_files)
s.test_files = normalize_files(Dir['spec/**/*.rb'] - repo.lib.ignored_files)
s.has_rdoc = true
s.extra_rdoc_files = %w[README.rdoc MIT-LICENSE.txt History.txt]
s.add_dependency "nokogiri", ">= 1.2.0"
s.add_dependency "rack", ">= 1.0"
s.add_dependency "rack-test", ">= 0.5.3"
end
end
def normalize_files(array)
# only keep files, no directories, and sort
array.select do |path|
File.file?(path)
end.sort
end
# Adds extra space when outputting an array. This helps create better version
# control diffs, because otherwise it is all on the same line.
def prettyify_array(gemspec_ruby, array_name)
gemspec_ruby.gsub(/s\.#{array_name.to_s} = \[.+?\]/) do |match|
leadin, files = match[0..-2].split("[")
leadin + "[\n #{files.split(",").join(",\n ")}\n ]"
end
end
def read_gemspec
@read_gemspec ||= eval(File.read("webrat.gemspec"))
end
def sh(command)
puts command
system command
end
end
class Default < Thor
include GemHelpers
desc "gemspec", "Regenerate webrat.gemspec"
def gemspec
File.open("webrat.gemspec", "w") do |file|
gemspec_ruby = generate_gemspec.to_ruby
gemspec_ruby = prettyify_array(gemspec_ruby, :files)
gemspec_ruby = prettyify_array(gemspec_ruby, :test_files)
gemspec_ruby = prettyify_array(gemspec_ruby, :extra_rdoc_files)
file.write gemspec_ruby
end
puts "Wrote gemspec to webrat.gemspec"
read_gemspec.validate
end
desc "build", "Build a webrat gem"
def build
sh "gem build webrat.gemspec"
FileUtils.mkdir_p "pkg"
FileUtils.mv read_gemspec.file_name, "pkg"
end
desc "install", "Install the latest built gem"
def install
sh "gem install --local pkg/#{read_gemspec.file_name}"
end
desc "release", "Release the current branch to GitHub and Gemcutter"
def release
gemspec
build
Release.new.tag
Release.new.gem
end
end
class Release < Thor
include GemHelpers
desc "tag", "Tag the gem on the origin server"
def tag
release_tag = "v#{read_gemspec.version}"
sh "git tag -a #{release_tag} -m 'Tagging #{release_tag}'"
sh "git push origin #{release_tag}"
end
desc "gem", "Push the gem to Gemcutter"
def gem
sh "gem push pkg/#{read_gemspec.file_name}"
end
end

View File

@ -1,34 +1,18 @@
require "rubygems" require "rack"
require "nokogiri"
$LOAD_PATH.unshift(File.expand_path(File.dirname(__FILE__))) unless $LOAD_PATH.include?(File.expand_path(File.dirname(__FILE__)))
module Webrat module Webrat
VERSION = "0.7.2.pre"
autoload :MechanizeAdapter, "webrat/adapters/mechanize"
autoload :MerbAdapter, "webrat/adapters/merb"
autoload :RackAdapter, "webrat/adapters/rack"
autoload :RailsAdapter, "webrat/adapters/rails"
autoload :SinatraAdapter, "webrat/adapters/sinatra"
# The common base class for all exceptions raised by Webrat. # The common base class for all exceptions raised by Webrat.
class WebratError < StandardError class WebratError < StandardError
end end
VERSION = '0.4.0'
def self.require_xml
gem "nokogiri", ">= 1.0.6"
if on_java?
# We need Nokogiri's CSS to XPath support, even if using REXML and Hpricot for parsing and searching
require "nokogiri/css"
require "hpricot"
require "rexml/document"
else
require "nokogiri"
require "webrat/core/xml/nokogiri"
end
end
def self.on_java?
RUBY_PLATFORM =~ /java/
end
end end
Webrat.require_xml
require "webrat/core" require "webrat/core"

View File

@ -1,32 +1,44 @@
require "mechanize" require "mechanize"
module Webrat #:nodoc: module Webrat #:nodoc:
class MechanizeSession < Session #:nodoc: class MechanizeAdapter #:nodoc:
extend Forwardable
Mechanize = WWW::Mechanize if defined?(WWW::Mechanize)
attr_accessor :response attr_accessor :response
alias :page :response alias :page :response
def initialize(*args)
end
def request_page(url, http_method, data) #:nodoc: def request_page(url, http_method, data) #:nodoc:
super(absolute_url(url), http_method, data) super(absolute_url(url), http_method, data)
end end
def get(url, data, headers_argument_not_used = nil) def get(url, data, headers_argument_not_used = nil)
@response = mechanize.get(url, data) @response = mechanize.get(url, data)
end end
def post(url, data, headers_argument_not_used = nil) def post(url, data, headers_argument_not_used = nil)
post_data = data.inject({}) do |memo, param| post_data = data.inject({}) do |memo, param|
case param.last case param
when Hash when Hash
param.last.each {|attribute, value| memo["#{param.first}[#{attribute}]"] = value } param.each {|attribute, value| memo[attribute] = value }
else memo
memo[param.first] = param.last when Array
case param.last
when Hash
param.last.each {|attribute, value| memo["#{param.first}[#{attribute}]"] = value }
else
memo[param.first] = param.last
end
memo
end end
memo
end end
@response = mechanize.post(url, post_data) @response = mechanize.post(url, post_data)
end end
def response_body def response_body
@response.content @response.content
end end
@ -34,13 +46,21 @@ module Webrat #:nodoc:
def response_code def response_code
@response.code.to_i @response.code.to_i
end end
def response_headers
@response.header
end
def mechanize def mechanize
@mechanize = WWW::Mechanize.new @mechanize ||= begin
mechanize = Mechanize.new
mechanize.redirect_ok = false
mechanize
end
end end
def_delegators :mechanize, :basic_auth def_delegators :mechanize, :basic_auth
def absolute_url(url) #:nodoc: def absolute_url(url) #:nodoc:
current_host, current_path = split_current_url current_host, current_path = split_current_url
if url =~ Regexp.new('^https?://') if url =~ Regexp.new('^https?://')
@ -53,13 +73,13 @@ module Webrat #:nodoc:
url url
end end
end end
private private
def split_current_url def split_current_url
current_url =~ Regexp.new('^(https?://[^/]+)(/.*)?') current_url =~ Regexp.new('^(https?://[^/]+)(/.*)?')
[Regexp.last_match(1), Regexp.last_match(2)] [Regexp.last_match(1), Regexp.last_match(2)]
end end
def absolute_path(current_path, url) def absolute_path(current_path, url)
levels_up = url.split('/').find_all { |x| x == '..' }.size levels_up = url.split('/').find_all { |x| x == '..' }.size
ancestor = if current_path.nil? ancestor = if current_path.nil?
@ -67,7 +87,7 @@ module Webrat #:nodoc:
else else
current_path.split("/")[0..(-1 - levels_up)].join("/") current_path.split("/")[0..(-1 - levels_up)].join("/")
end end
descendent = url.split("/")[levels_up..-1] descendent = url.split("/")[levels_up..-1].join
"#{ancestor}/#{descendent}" "#{ancestor}/#{descendent}"
end end
end end

View File

@ -0,0 +1,11 @@
require "webrat/integrations/merb"
module Webrat
class MerbAdapter < RackAdapter #:nodoc:
def initialize(context=nil)
app = context.respond_to?(:app) ?
context.app : Merb::Rack::Application.new
super(Rack::Test::Session.new(Rack::MockSession.new(app, "www.example.com")))
end
end
end

View File

@ -0,0 +1,29 @@
require "rack/test"
module Webrat
class RackAdapter
extend Forwardable
def_delegators :@session, :get, :post, :put, :delete
def initialize(session) #:nodoc:
@session = session
end
def response_body
response.body
end
def response_code
response.status
end
def response_headers
response.headers
end
def response
@session.last_response
end
end
end

View File

@ -1,34 +1,14 @@
require "webrat" require "webrat/integrations/rails"
require "action_controller"
require "action_controller/integration"
require "action_controller/record_identifier" require "action_controller/record_identifier"
module Webrat module Webrat
class RailsSession < Session #:nodoc: class RailsAdapter #:nodoc:
include ActionController::RecordIdentifier include ActionController::RecordIdentifier
# The Rails version of within supports passing in a model and Webrat
# will apply a scope based on Rails' dom_id for that model.
#
# Example:
# within User.last do
# click_link "Delete"
# end
def within(selector_or_object, &block)
if selector_or_object.is_a?(String)
super
else
super('#' + dom_id(selector_or_object), &block)
end
end
def doc_root
File.expand_path(File.join(RAILS_ROOT, 'public'))
end
def saved_page_dir attr_reader :integration_session
File.expand_path(File.join(RAILS_ROOT, "tmp"))
def initialize(session)
@integration_session = session
end end
def get(url, data, headers = nil) def get(url, data, headers = nil)
@ -55,16 +35,16 @@ module Webrat
response.code.to_i response.code.to_i
end end
def response_headers
response.headers
end
def xml_content_type? def xml_content_type?
response.headers["Content-Type"].to_s =~ /xml/ response.headers["Content-Type"].to_s =~ /xml/
end end
protected protected
def integration_session
@context
end
def do_request(http_method, url, data, headers) #:nodoc: def do_request(http_method, url, data, headers) #:nodoc:
update_protocol(url) update_protocol(url)
integration_session.send(http_method, normalize_url(url), data, headers) integration_session.send(http_method, normalize_url(url), data, headers)
@ -73,11 +53,13 @@ module Webrat
# remove protocol, host and anchor # remove protocol, host and anchor
def normalize_url(href) #:nodoc: def normalize_url(href) #:nodoc:
uri = URI.parse(href) uri = URI.parse(href)
normalized_url = uri.path normalized_url = []
if uri.query normalized_url << "#{uri.scheme}://" if uri.scheme
normalized_url += "?" + uri.query normalized_url << uri.host if uri.host
end normalized_url << ":#{uri.port}" if uri.port && ![80,443].include?(uri.port)
normalized_url normalized_url << uri.path if uri.path
normalized_url << "?#{uri.query}" if uri.query
normalized_url.join
end end
def update_protocol(href) #:nodoc: def update_protocol(href) #:nodoc:
@ -91,13 +73,5 @@ module Webrat
def response #:nodoc: def response #:nodoc:
integration_session.response integration_session.response
end end
end
end
module ActionController #:nodoc:
IntegrationTest.class_eval do
include Webrat::Methods
include Webrat::Matchers
end end
end end

View File

@ -0,0 +1,9 @@
module Webrat
class SinatraAdapter < RackAdapter
def initialize(context)
app = context.respond_to?(:app) ? context.app : Sinatra::Application
super(Rack::Test::Session.new(Rack::MockSession.new(app, "www.example.com")))
end
end
end

View File

@ -1,6 +1,5 @@
require "webrat/core/configuration" require "webrat/core/configuration"
require "webrat/core/xml" require "webrat/core/xml"
require "webrat/core/xml/nokogiri"
require "webrat/core/logging" require "webrat/core/logging"
require "webrat/core/elements/form" require "webrat/core/elements/form"
require "webrat/core/scope" require "webrat/core/scope"

View File

@ -1,4 +1,5 @@
require "webrat/core_extensions/deprecate" require "webrat/core_extensions/deprecate"
require "pathname"
module Webrat module Webrat
@ -16,26 +17,38 @@ module Webrat
# Webrat can be configured using the Webrat.configure method. For example: # Webrat can be configured using the Webrat.configure method. For example:
# #
# Webrat.configure do |config| # Webrat.configure do |config|
# config.parse_with_nokogiri = false # config.mode = :sinatra
# end # end
class Configuration class Configuration
# Should XHTML be parsed with Nokogiri? Defaults to true, except on JRuby. When false, Hpricot and REXML are used
attr_writer :parse_with_nokogiri
# Webrat's mode, set automatically when requiring webrat/rails, webrat/merb, etc. # Webrat's mode, set automatically when requiring webrat/rails, webrat/merb, etc.
attr_reader :mode # :nodoc: attr_reader :mode # :nodoc:
# Save and open pages with error status codes (500-599) in a browser? Defualts to true. # Save and open pages with error status codes (500-599) in a browser? Defualts to true.
attr_writer :open_error_files attr_writer :open_error_files
# Which rails environment should the selenium tests be run in? Defaults to selenium. # Save and open page storage directory, defaults to "tmp" under current directory if exists, otherwise current directory
attr_accessor :saved_pages_dir
# Which rails environment should the selenium tests be run in? Defaults to test.
attr_accessor :application_environment attr_accessor :application_environment
webrat_deprecate :selenium_environment, :application_environment webrat_deprecate :selenium_environment, :application_environment
webrat_deprecate :selenium_environment=, :application_environment=
# Which port is the application running on for selenium testing? Defaults to 3001. # Which port is the application running on for selenium testing? Defaults to 3001.
attr_accessor :application_port attr_accessor :application_port
webrat_deprecate :selenium_port, :application_port webrat_deprecate :selenium_port, :application_port
webrat_deprecate :selenium_port=, :application_port=
# Which port should selenium use to access the application. Defaults to application_port
attr_writer :application_port_for_selenium
def application_port_for_selenium
@application_port_for_selenium || self.application_port
end
# Which underlying app framework we're testing with selenium
attr_accessor :application_framework
# Which server the application is running on for selenium testing? Defaults to localhost # Which server the application is running on for selenium testing? Defaults to localhost
attr_accessor :application_address attr_accessor :application_address
@ -49,20 +62,36 @@ module Webrat
# Set the key that Selenium uses to determine the browser running. Default *firefox # Set the key that Selenium uses to determine the browser running. Default *firefox
attr_accessor :selenium_browser_key attr_accessor :selenium_browser_key
# Set the timeout for waiting for the browser process to start
attr_accessor :selenium_browser_startup_timeout
# Set the firefox profile for selenium to use
attr_accessor :selenium_firefox_profile
# How many redirects to the same URL should be halted as an infinite redirect
# loop? Defaults to 10
attr_accessor :infinite_redirect_limit
# Print out the full HTML on wait failure
# Defaults to false
attr_accessor :selenium_verbose_output
def initialize # :nodoc: def initialize # :nodoc:
self.open_error_files = true self.open_error_files = true
self.parse_with_nokogiri = !Webrat.on_java? self.application_environment = :test
self.application_environment = :selenium
self.application_port = 3001 self.application_port = 3001
self.application_address = 'localhost' self.application_address = 'localhost'
self.application_framework = :rails
self.selenium_server_port = 4444 self.selenium_server_port = 4444
self.infinite_redirect_limit = 10
self.selenium_browser_key = '*firefox' self.selenium_browser_key = '*firefox'
end self.selenium_browser_startup_timeout = 5
self.selenium_verbose_output = false
def parse_with_nokogiri? #:nodoc: tmp_dir = Pathname.new(Dir.pwd).join("tmp")
@parse_with_nokogiri ? true : false self.saved_pages_dir = tmp_dir.exist? ? tmp_dir : Dir.pwd
end end
def open_error_files? #:nodoc: def open_error_files? #:nodoc:
@open_error_files ? true : false @open_error_files ? true : false
end end
@ -71,17 +100,15 @@ module Webrat
# :rails, :selenium, :rack, :sinatra, :mechanize, :merb # :rails, :selenium, :rack, :sinatra, :mechanize, :merb
def mode=(mode) def mode=(mode)
@mode = mode.to_sym @mode = mode.to_sym
# This is a temporary hack to support backwards compatibility begin
# with Merb 1.0.8 until it's updated to use the new Webrat.configure require("webrat/integrations/#{mode}")
# syntax rescue LoadError
if @mode == :merb # Only some modes have integration code that needs to
require("webrat/merb_session") # be loaded, so this is OK
else
require("webrat/#{mode}")
end end
end end
end end
end end

View File

@ -2,21 +2,21 @@ require "webrat/core/elements/element"
module Webrat module Webrat
class Area < Element #:nodoc: class Area < Element #:nodoc:
def self.xpath_search def self.xpath_search
".//area" [".//area"]
end end
def click(method = nil, options = {}) def click(method = nil, options = {})
@session.request_page(absolute_href, :get, {}) @session.request_page(absolute_href, :get, {})
end end
protected protected
def href def href
Webrat::XML.attribute(@element, "href") @element["href"]
end end
def absolute_href def absolute_href
if href =~ /^\?/ if href =~ /^\?/
"#{@session.current_url}#{href}" "#{@session.current_url}#{href}"
@ -26,6 +26,6 @@ module Webrat
href href
end end
end end
end end
end end

View File

@ -1,33 +1,33 @@
module Webrat module Webrat
class Element # :nodoc: class Element # :nodoc:
def self.load_all(session, dom) def self.load_all(session, dom)
Webrat::XML.xpath_search(dom, xpath_search).map do |element| dom.xpath(*xpath_search).map do |element|
load(session, element) load(session, element)
end end
end end
def self.load(session, element) def self.load(session, element)
return nil if element.nil? return nil if element.nil?
session.elements[Webrat::XML.xpath_to(element)] ||= self.new(session, element) session.elements[element.path] ||= self.new(session, element)
end end
attr_reader :element attr_reader :element
def initialize(session, element) def initialize(session, element)
@session = session @session = session
@element = element @element = element
end end
def path def path
Webrat::XML.xpath_to(@element) @element.path
end end
def inspect def inspect
"#<#{self.class} @element=#{element.inspect}>" "#<#{self.class} @element=#{element.inspect}>"
end end
end end
end end

View File

@ -1,6 +1,7 @@
require "cgi" require "cgi"
require "digest/md5"
require "webrat/core_extensions/blank" require "webrat/core_extensions/blank"
require "webrat/core_extensions/nil_to_param" require "webrat/core_extensions/nil_to_query_string"
require "webrat/core/elements/element" require "webrat/core/elements/element"
@ -8,36 +9,45 @@ module Webrat
# Raised when Webrat is asked to manipulate a disabled form field # Raised when Webrat is asked to manipulate a disabled form field
class DisabledFieldError < WebratError class DisabledFieldError < WebratError
end end
class Field < Element #:nodoc: class Field < Element #:nodoc:
attr_reader :value attr_reader :value
def self.xpath_search def self.xpath_search
[".//button", ".//input", ".//textarea", ".//select"] ".//button|.//input|.//textarea|.//select"
end end
def self.xpath_search_excluding_hidden
[".//button", ".//input[ @type != 'hidden']", ".//textarea", ".//select"]
end
def self.field_classes def self.field_classes
@field_classes || [] @field_classes || []
end end
def self.inherited(klass) def self.inherited(klass)
@field_classes ||= [] @field_classes ||= []
@field_classes << klass @field_classes << klass
# raise args.inspect # raise args.inspect
end end
def self.load(session, element) def self.load(session, element)
return nil if element.nil? return nil if element.nil?
session.elements[Webrat::XML.xpath_to(element)] ||= field_class(element).new(session, element) session.elements[element.path] ||= field_class(element).new(session, element)
end end
def self.field_class(element) def self.field_class(element)
case element.name case element.name
when "button" then ButtonField when "button" then ButtonField
when "select" then SelectField when "select"
if element.attributes["multiple"].nil?
SelectField
else
MultipleSelectField
end
when "textarea" then TextareaField when "textarea" then TextareaField
else else
case Webrat::XML.attribute(element, "type") case element["type"]
when "checkbox" then CheckboxField when "checkbox" then CheckboxField
when "hidden" then HiddenField when "hidden" then HiddenField
when "radio" then RadioField when "radio" then RadioField
@ -45,12 +55,13 @@ module Webrat
when "file" then FileField when "file" then FileField
when "reset" then ResetField when "reset" then ResetField
when "submit" then ButtonField when "submit" then ButtonField
when "button" then ButtonField
when "image" then ButtonField when "image" then ButtonField
else TextField else TextField
end end
end end
end end
def initialize(*args) def initialize(*args)
super super
@value = default_value @value = default_value
@ -60,82 +71,86 @@ module Webrat
return nil if labels.empty? return nil if labels.empty?
labels.first.text labels.first.text
end end
def id def id
Webrat::XML.attribute(@element, "id") @element["id"]
end end
def disabled? def disabled?
@element.attributes.has_key?("disabled") && Webrat::XML.attribute(@element, "disabled") != 'false' @element.attributes.has_key?("disabled") && @element["disabled"] != 'false'
end end
def raise_error_if_disabled def raise_error_if_disabled
return unless disabled? return unless disabled?
raise DisabledFieldError.new("Cannot interact with disabled form element (#{self})") raise DisabledFieldError.new("Cannot interact with disabled form element (#{self})")
end end
def to_param def to_query_string
return nil if disabled? return nil if disabled?
case Webrat.configuration.mode query_string = case Webrat.configuration.mode
when :rails when :rails, :merb, :rack, :sinatra
rails_request_parser.parse_query_parameters("#{name}=#{escaped_value}") build_query_string
when :merb when :mechanize
::Merb::Parse.query("#{name}=#{escaped_value}") build_query_string(false)
else
{ name => escaped_value }
end end
query_string
end end
def set(value) def set(value)
@value = value @value = value
end end
def unset def unset
@value = default_value @value = default_value
end end
protected protected
def rails_request_parser
if defined?(ActionController::RequestParser) # For Rails > 2.2
ActionController::RequestParser
else
ActionController::AbstractRequest
end
end
def form def form
Form.load(@session, form_element) Form.load(@session, form_element)
end end
def form_element def form_element
parent = @element.parent parent = @element.parent
while parent.respond_to?(:parent) while parent.respond_to?(:parent)
return parent if parent.name == 'form' return parent if parent.name == 'form'
parent = parent.parent parent = parent.parent
end end
end end
def name def name
Webrat::XML.attribute(@element, "name") @element["name"]
end end
def build_query_string(escape_value=true)
if @value.is_a?(Array)
@value.collect {|value| "#{name}=#{ escape_value ? escape(value) : value }" }.join("&")
else
"#{name}=#{ escape_value ? escape(value) : value }"
end
end
def escape(value)
CGI.escape(value.to_s)
end
def escaped_value def escaped_value
CGI.escape(@value.to_s) CGI.escape(@value.to_s)
end end
def labels def labels
@labels ||= label_elements.map do |element| @labels ||= label_elements.map do |element|
Label.load(@session, element) Label.load(@session, element)
end end
end end
def label_elements def label_elements
return @label_elements unless @label_elements.nil? return @label_elements unless @label_elements.nil?
@label_elements = [] @label_elements = []
parent = @element.parent parent = @element.parent
while parent.respond_to?(:parent) while parent.respond_to?(:parent)
if parent.name == 'label' if parent.name == 'label'
@ -144,42 +159,26 @@ module Webrat
end end
parent = parent.parent parent = parent.parent
end end
unless id.blank? unless id.blank?
@label_elements += Webrat::XML.xpath_search(form.element, ".//label[@for = '#{id}']") @label_elements += form.element.xpath(".//label[@for = '#{id}']")
end end
@label_elements @label_elements
end end
def default_value def default_value
Webrat::XML.attribute(@element, "value") @element["value"]
end
def replace_param_value(params, oval, nval)
output = Hash.new
params.each do |key, value|
case value
when Hash
value = replace_param_value(value, oval, nval)
when Array
value = value.map { |o| o == oval ? nval : oval }
when oval
value = nval
end
output[key] = value
end
output
end end
end end
class ButtonField < Field #:nodoc: class ButtonField < Field #:nodoc:
def self.xpath_search def self.xpath_search
[".//button", ".//input[@type = 'submit']", ".//input[@type = 'image']"] [".//button", ".//input[@type = 'submit']", ".//input[@type = 'button']", ".//input[@type = 'image']"]
end end
def to_param def to_query_string
return nil if @value.nil? return nil if @value.nil?
super super
end end
@ -190,7 +189,7 @@ module Webrat
def click def click
raise_error_if_disabled raise_error_if_disabled
set(Webrat::XML.attribute(@element, "value")) unless Webrat::XML.attribute(@element, "name").blank? set(@element["value"]) unless @element["name"].blank?
form.submit form.submit
end end
@ -201,14 +200,14 @@ module Webrat
def self.xpath_search def self.xpath_search
".//input[@type = 'hidden']" ".//input[@type = 'hidden']"
end end
def to_param def to_query_string
if collection_name? if collection_name?
super super
else else
checkbox_with_same_name = form.field_named(name, CheckboxField) checkbox_with_same_name = form.field_named(name, CheckboxField)
if checkbox_with_same_name.to_param.blank? if checkbox_with_same_name.to_query_string.blank?
super super
else else
nil nil
@ -229,19 +228,19 @@ module Webrat
def self.xpath_search def self.xpath_search
".//input[@type = 'checkbox']" ".//input[@type = 'checkbox']"
end end
def to_param def to_query_string
return nil if @value.nil? return nil if @value.nil?
super super
end end
def check def check
raise_error_if_disabled raise_error_if_disabled
set(Webrat::XML.attribute(@element, "value") || "on") set(@element["value"] || "on")
end end
def checked? def checked?
Webrat::XML.attribute(@element, "checked") == "checked" @element["checked"] == "checked"
end end
def uncheck def uncheck
@ -252,8 +251,8 @@ module Webrat
protected protected
def default_value def default_value
if Webrat::XML.attribute(@element, "checked") == "checked" if @element["checked"] == "checked"
Webrat::XML.attribute(@element, "value") || "on" @element["value"] || "on"
else else
nil nil
end end
@ -262,11 +261,11 @@ module Webrat
end end
class PasswordField < Field #:nodoc: class PasswordField < Field #:nodoc:
def self.xpath_search def self.xpath_search
".//input[@type = 'password']" ".//input[@type = 'password']"
end end
end end
class RadioField < Field #:nodoc: class RadioField < Field #:nodoc:
@ -274,34 +273,34 @@ module Webrat
def self.xpath_search def self.xpath_search
".//input[@type = 'radio']" ".//input[@type = 'radio']"
end end
def to_param def to_query_string
return nil if @value.nil? return nil if @value.nil?
super super
end end
def choose def choose
raise_error_if_disabled raise_error_if_disabled
other_options.each do |option| other_options.each do |option|
option.set(nil) option.set(nil)
end end
set(Webrat::XML.attribute(@element, "value") || "on") set(@element["value"] || "on")
end end
def checked? def checked?
Webrat::XML.attribute(@element, "checked") == "checked" @element["checked"] == "checked"
end end
protected protected
def other_options def other_options
form.fields.select { |f| f.name == name } form.fields.select { |f| f.name == name }
end end
def default_value def default_value
if Webrat::XML.attribute(@element, "checked") == "checked" if @element["checked"] == "checked"
Webrat::XML.attribute(@element, "value") || "on" @element["value"] || "on"
else else
nil nil
end end
@ -314,43 +313,50 @@ module Webrat
def self.xpath_search def self.xpath_search
".//textarea" ".//textarea"
end end
protected protected
def default_value def default_value
Webrat::XML.inner_html(@element) @element.inner_html
end end
end end
class FileField < Field #:nodoc: class FileField < Field #:nodoc:
def self.xpath_search def self.xpath_search
".//input[@type = 'file']" ".//input[@type = 'file']"
end end
attr_accessor :content_type attr_accessor :content_type
def set(value, content_type = nil) def set(value, content_type = nil)
@original_value = @value
@content_type ||= content_type
super(value) super(value)
@content_type = content_type
end end
def to_param def digest_value
if @value.nil? @value ? Digest::MD5.hexdigest(self.object_id.to_s) : ""
super
else
replace_param_value(super, @value, test_uploaded_file)
end
end end
protected def to_query_string
@value.nil? ? set("") : set(digest_value)
super
end
def test_uploaded_file def test_uploaded_file
if content_type return "" if @original_value.blank?
ActionController::TestUploadedFile.new(@value, content_type)
else case Webrat.configuration.mode
ActionController::TestUploadedFile.new(@value) when :rails
if content_type
ActionController::TestUploadedFile.new(@original_value, content_type)
else
ActionController::TestUploadedFile.new(@original_value)
end
when :rack, :merb
Rack::Test::UploadedFile.new(@original_value, content_type)
end end
end end
@ -364,31 +370,69 @@ module Webrat
class ResetField < Field #:nodoc: class ResetField < Field #:nodoc:
def self.xpath_search def self.xpath_search
".//input[@type = 'reset']" [".//input[@type = 'reset']"]
end end
end end
class SelectField < Field #:nodoc: class SelectField < Field #:nodoc:
def self.xpath_search def self.xpath_search
".//select" [".//select[not(@multiple)]"]
end end
def options def options
@options ||= SelectOption.load_all(@session, @element) @options ||= SelectOption.load_all(@session, @element)
end end
def unset(value)
@value = nil
end
protected protected
def default_value def default_value
selected_options = Webrat::XML.xpath_search(@element, ".//option[@selected = 'selected']") selected_options = @element.xpath(".//option[@selected = 'selected']")
selected_options = Webrat::XML.xpath_search(@element, ".//option[position() = 1]") if selected_options.empty? selected_options = @element.xpath(".//option[position() = 1]") if selected_options.empty?
selected_options.map do |option| selected_options.map do |option|
return "" if option.nil? return "" if option.nil?
Webrat::XML.attribute(option, "value") || Webrat::XML.inner_html(option) option["value"] || option.inner_html
end end.uniq.first
end end
end end
class MultipleSelectField < Field #:nodoc:
def self.xpath_search
[".//select[@multiple='multiple']"]
end
def options
@options ||= SelectOption.load_all(@session, @element)
end
def set(value)
@value << value
end
def unset(value)
@value.delete(value)
end
protected
# Overwrite SelectField definition because we don't want to select the first option
# (mutliples don't select the first option unlike their non multiple versions)
def default_value
selected_options = @element.xpath(".//option[@selected = 'selected']")
selected_options.map do |option|
return "" if option.nil?
option["value"] || option.inner_html
end.uniq
end
end
end end

View File

@ -7,29 +7,29 @@ require "webrat/core/locators/field_named_locator"
module Webrat module Webrat
class Form < Element #:nodoc: class Form < Element #:nodoc:
attr_reader :element attr_reader :element
def self.xpath_search def self.xpath_search
".//form" [".//form"]
end end
def fields def fields
@fields ||= Field.load_all(@session, @element) @fields ||= Field.load_all(@session, @element)
end end
def submit def submit
@session.request_page(form_action, form_method, params) @session.request_page(form_action, form_method, params)
end end
def field_named(name, *field_types) def field_named(name, *field_types)
Webrat::Locators::FieldNamedLocator.new(@session, dom, name, *field_types).locate Webrat::Locators::FieldNamedLocator.new(@session, dom, name, *field_types).locate
end end
protected protected
def dom def dom
Webrat::XML.xpath_at(@session.dom, path) @session.dom.xpath(path).first
end end
def fields_by_type(field_types) def fields_by_type(field_types)
if field_types.any? if field_types.any?
fields.select { |f| field_types.include?(f.class) } fields.select { |f| field_types.include?(f.class) }
@ -37,67 +37,91 @@ module Webrat
fields fields
end end
end end
# iterate over all form fields to build a request querystring to get params from it,
# for file_field we made a work around to pass a digest as value to later replace it
# in params hash with the real file.
def params def params
all_params = {} query_string = []
replaces = {}
fields.each do |field| fields.each do |field|
next if field.to_param.nil? next if field.to_query_string.nil?
merge(all_params, field.to_param) replaces.merge!({field.digest_value => field.test_uploaded_file}) if field.is_a?(FileField)
query_string << field.to_query_string
end end
all_params query_params = self.class.query_string_to_params(query_string.join('&'))
query_params = self.class.replace_params_values(query_params, replaces)
self.class.unescape_params(query_params)
end end
def form_method def form_method
Webrat::XML.attribute(@element, "method").blank? ? :get : Webrat::XML.attribute(@element, "method").downcase @element["method"].blank? ? :get : @element["method"].downcase
end end
def form_action def form_action
Webrat::XML.attribute(@element, "action").blank? ? @session.current_url : Webrat::XML.attribute(@element, "action") @element["action"].blank? ? @session.current_url : @element["action"]
end end
def merge(all_params, new_param) def self.replace_param_value(params, oval, nval)
new_param.each do |key, value| output = Hash.new
case all_params[key] params.each do |key, value|
when *hash_classes case value
merge_hash_values(all_params[key], value) when Hash
value = replace_param_value(value, oval, nval)
when Array when Array
all_params[key] += value value = value.map { |o| o == oval ? nval : ( o.is_a?(Hash) ? replace_param_value(o, oval, nval) : o) }
else when oval
all_params[key] = value value = nval
end end
output[key] = value
end
output
end
def self.replace_params_values(params, values)
values.each do |key, value|
params = replace_param_value(params, key, value)
end
params
end
def self.unescape_params(params)
case params.class.name
when 'Hash', 'Mash'
params.each { |key,value| params[key] = unescape_params(value) }
params
when 'Array'
params.collect { |value| unescape_params(value) }
else
params.is_a?(String) ? CGI.unescapeHTML(params) : params
end end
end end
def merge_hash_values(a, b) # :nodoc: def self.query_string_to_params(query_string)
a.keys.each do |k|
if b.has_key?(k)
case [a[k], b[k]].map{|value| value.class}
when *hash_classes.zip(hash_classes)
a[k] = merge_hash_values(a[k], b[k])
b.delete(k)
when [Array, Array]
a[k] += b[k]
b.delete(k)
end
end
end
a.merge!(b)
end
def hash_classes
klasses = [Hash]
case Webrat.configuration.mode case Webrat.configuration.mode
when :rails when :rails
klasses << HashWithIndifferentAccess parse_rails_request_params(query_string)
when :merb when :merb
klasses << Mash ::Merb::Parse.query(query_string)
when :rack, :sinatra
Rack::Utils.parse_nested_query(query_string)
else
Hash[query_string.split('&').map {|query| [ query.split('=').first, query.split('=').last ]}]
end
end
def self.parse_rails_request_params(query_string)
if defined?(ActionController::AbstractRequest)
ActionController::AbstractRequest.parse_query_parameters(query_string)
elsif defined?(ActionController::UrlEncodedPairParser)
ActionController::UrlEncodedPairParser.parse_query_parameters(query_string)
else
Rack::Utils.parse_nested_query(query_string)
end end
klasses
end end
end end
end end

View File

@ -2,30 +2,30 @@ require "webrat/core/elements/element"
module Webrat module Webrat
class Label < Element #:nodoc: class Label < Element #:nodoc:
attr_reader :element attr_reader :element
def self.xpath_search def self.xpath_search
".//label" [".//label"]
end end
def for_id def for_id
Webrat::XML.attribute(@element, "for") @element["for"]
end end
def field def field
Field.load(@session, field_element) Field.load(@session, field_element)
end end
protected protected
def field_element def field_element
if for_id.blank? if for_id.blank?
Webrat::XML.xpath_at(@element, *Field.xpath_search) @element.xpath(*Field.xpath_search_excluding_hidden).first
else else
Webrat::XML.css_search(@session.dom, "#" + for_id).first @session.current_dom.css("#" + for_id).first
end end
end end
end end
end end

View File

@ -1,43 +1,44 @@
require "webrat/core_extensions/blank" require "English"
require "webrat/core_extensions/blank"
require "webrat/core/elements/element" require "webrat/core/elements/element"
module Webrat module Webrat
class Link < Element #:nodoc: class Link < Element #:nodoc:
def self.xpath_search def self.xpath_search
".//a[@href]" [".//a[@href]"]
end end
def click(options = {}) def click(options = {})
method = options[:method] || http_method method = options[:method] || http_method
return if href =~ /^#/ && method == :get return if href =~ /^#/ && method == :get
options[:javascript] = true if options[:javascript].nil? options[:javascript] = true if options[:javascript].nil?
if options[:javascript] if options[:javascript]
@session.request_page(absolute_href, method, data) @session.request_page(absolute_href, method, data)
else else
@session.request_page(absolute_href, :get, {}) @session.request_page(absolute_href, :get, {})
end end
end end
protected protected
def id def id
Webrat::XML.attribute(@element, "id") @element["id"]
end end
def data def data
authenticity_token.blank? ? {} : {"authenticity_token" => authenticity_token} authenticity_token.blank? ? {} : {"authenticity_token" => authenticity_token}
end end
def title def title
Webrat::XML.attribute(@element, "title") @element["title"]
end end
def href def href
Webrat::XML.attribute(@element, "href") @element["href"]
end end
def absolute_href def absolute_href
@ -49,17 +50,17 @@ module Webrat
href href
end end
end end
def authenticity_token def authenticity_token
return unless onclick && onclick.include?("s.setAttribute('name', 'authenticity_token');") && return unless onclick && onclick.include?("s.setAttribute('name', 'authenticity_token');") &&
onclick =~ /s\.setAttribute\('value', '([a-f0-9]{40})'\);/ ( onclick =~ /s\.setAttribute\('value', '([a-f0-9]{40})'\);/ || onclick =~ /s\.setAttribute\('value', '(.{44})'\);/ )
$LAST_MATCH_INFO.captures.first $LAST_MATCH_INFO.captures.first
end end
def onclick def onclick
Webrat::XML.attribute(@element, "onclick") @element["onclick"]
end end
def http_method def http_method
if !onclick.blank? && onclick.include?("f.submit()") if !onclick.blank? && onclick.include?("f.submit()")
http_method_from_js_form http_method_from_js_form
@ -81,10 +82,12 @@ module Webrat
:delete :delete
elsif onclick.include?("m.setAttribute('value', 'put')") elsif onclick.include?("m.setAttribute('value', 'put')")
:put :put
elsif onclick.include?("m.setAttribute('value', 'post')")
:post
else else
raise Webrat::WebratError.new("No HTTP method for _method param in #{onclick.inspect}") raise Webrat::WebratError.new("No HTTP method for _method param in #{onclick.inspect}")
end end
end end
end end
end end

View File

@ -2,34 +2,47 @@ require "webrat/core/elements/element"
module Webrat module Webrat
class SelectOption < Element #:nodoc: class SelectOption < Element #:nodoc:
def self.xpath_search def self.xpath_search
".//option" [".//option"]
end end
def choose def choose
select.raise_error_if_disabled select.raise_error_if_disabled
select.set(value) select.set(value)
end end
def unchoose
select.raise_error_if_disabled
select.unset(value)
end
def inner_text
@element.inner_text
end
protected protected
def select def select
SelectField.load(@session, select_element) SelectField.load(@session, select_element)
end end
def select_element def select_element
parent = @element.parent parent = @element.parent
while parent.respond_to?(:parent) while parent.respond_to?(:parent)
return parent if parent.name == 'select' return parent if parent.name == 'select'
parent = parent.parent parent = parent.parent
end end
end end
def value def value
Webrat::XML.attribute(@element, "value") || Webrat::XML.inner_html(@element) @element["value"] || @element.inner_html
end end
def label
@element.inner_html
end
end end
end end

View File

@ -11,10 +11,10 @@ require "webrat/core/locators/form_locator"
module Webrat module Webrat
module Locators module Locators
def field_by_xpath(xpath) def field_by_xpath(xpath)
Field.load(@session, Webrat::XML.xpath_at(dom, xpath)) Field.load(@session, dom.xpath(xpath).first)
end end
end end
end end

View File

@ -2,37 +2,37 @@ require "webrat/core/locators/locator"
module Webrat module Webrat
module Locators module Locators
class AreaLocator < Locator # :nodoc: class AreaLocator < Locator # :nodoc:
def locate def locate
Area.load(@session, area_element) Area.load(@session, area_element)
end end
def area_element def area_element
area_elements.detect do |area_element| area_elements.detect do |area_element|
Webrat::XML.attribute(area_element, "title") =~ matcher || area_element["title"] =~ matcher ||
Webrat::XML.attribute(area_element, "id") =~ matcher area_element["id"] =~ matcher
end end
end end
def matcher def matcher
/#{Regexp.escape(@value.to_s)}/i /#{Regexp.escape(@value.to_s)}/i
end end
def area_elements def area_elements
Webrat::XML.xpath_search(@dom, Area.xpath_search) @dom.xpath(*Area.xpath_search)
end end
def error_message def error_message
"Could not find area with name #{@value}" "Could not find area with name #{@value}"
end end
end end
def find_area(id_or_title) #:nodoc: def find_area(id_or_title) #:nodoc:
AreaLocator.new(@session, dom, id_or_title).locate! AreaLocator.new(@session, dom, id_or_title).locate!
end end
end end
end end

View File

@ -2,13 +2,13 @@ require "webrat/core/locators/locator"
module Webrat module Webrat
module Locators module Locators
class ButtonLocator < Locator # :nodoc: class ButtonLocator < Locator # :nodoc:
def locate def locate
ButtonField.load(@session, button_element) ButtonField.load(@session, button_element)
end end
def button_element def button_element
button_elements.detect do |element| button_elements.detect do |element|
@value.nil? || @value.nil? ||
@ -18,37 +18,37 @@ module Webrat
matches_alt?(element) matches_alt?(element)
end end
end end
def matches_id?(element) def matches_id?(element)
(@value.is_a?(Regexp) && Webrat::XML.attribute(element, "id") =~ @value) || (@value.is_a?(Regexp) && element["id"] =~ @value) ||
(!@value.is_a?(Regexp) && Webrat::XML.attribute(element, "id") == @value.to_s) (!@value.is_a?(Regexp) && element["id"] == @value.to_s)
end end
def matches_value?(element) def matches_value?(element)
Webrat::XML.attribute(element, "value") =~ /^\W*#{Regexp.escape(@value.to_s)}/i element["value"] =~ /^\W*#{Regexp.escape(@value.to_s)}/i
end end
def matches_html?(element) def matches_html?(element)
Webrat::XML.inner_html(element) =~ /#{Regexp.escape(@value.to_s)}/i element.inner_html =~ /#{Regexp.escape(@value.to_s)}/i
end end
def matches_alt?(element) def matches_alt?(element)
Webrat::XML.attribute(element, "alt") =~ /^\W*#{Regexp.escape(@value.to_s)}/i element["alt"] =~ /^\W*#{Regexp.escape(@value.to_s)}/i
end end
def button_elements def button_elements
Webrat::XML.xpath_search(@dom, *ButtonField.xpath_search) @dom.xpath(*ButtonField.xpath_search)
end end
def error_message def error_message
"Could not find button #{@value.inspect}" "Could not find button #{@value.inspect}"
end end
end end
def find_button(value) #:nodoc: def find_button(value) #:nodoc:
ButtonLocator.new(@session, dom, value).locate! ButtonLocator.new(@session, dom, value).locate!
end end
end end
end end

View File

@ -2,36 +2,36 @@ require "webrat/core/locators/locator"
module Webrat module Webrat
module Locators module Locators
class FieldByIdLocator < Locator # :nodoc: class FieldByIdLocator < Locator # :nodoc:
def locate def locate
Field.load(@session, field_element) Field.load(@session, field_element)
end end
def field_element def field_element
field_elements.detect do |field_element| field_elements.detect do |field_element|
if @value.is_a?(Regexp) if @value.is_a?(Regexp)
Webrat::XML.attribute(field_element, "id") =~ @value field_element["id"] =~ @value
else else
Webrat::XML.attribute(field_element, "id") == @value.to_s field_element["id"] == @value.to_s
end end
end end
end end
def field_elements def field_elements
Webrat::XML.xpath_search(@dom, *Field.xpath_search) @dom.xpath(*Field.xpath_search)
end end
def error_message def error_message
"Could not find field with id #{@value.inspect}" "Could not find field with id #{@value.inspect}"
end end
end end
def field_with_id(id, *field_types) def field_with_id(id, *field_types)
FieldByIdLocator.new(@session, dom, id, *field_types).locate! FieldByIdLocator.new(@session, dom, id, *field_types).locate!
end end
end end
end end

View File

@ -3,13 +3,13 @@ require "webrat/core/locators/locator"
module Webrat module Webrat
module Locators module Locators
class FieldLabeledLocator < Locator # :nodoc: class FieldLabeledLocator < Locator # :nodoc:
def locate def locate
matching_labels.any? && matching_labels.detect_mapped { |label| label.field } matching_labels.any? && matching_labels.detect_mapped { |label| label.field }
end end
def matching_labels def matching_labels
matching_label_elements.sort_by do |label_element| matching_label_elements.sort_by do |label_element|
text(label_element).length text(label_element).length
@ -17,31 +17,31 @@ module Webrat
Label.load(@session, label_element) Label.load(@session, label_element)
end end
end end
def matching_label_elements def matching_label_elements
label_elements.select do |label_element| label_elements.select do |label_element|
text(label_element) =~ /^\W*#{Regexp.escape(@value.to_s)}\b/i text(label_element) =~ /^\W*#{Regexp.escape(@value.to_s)}(\b|\Z)/i
end end
end end
def label_elements def label_elements
Webrat::XML.xpath_search(@dom, Label.xpath_search) @dom.xpath(*Label.xpath_search)
end end
def error_message def error_message
"Could not find field labeled #{@value.inspect}" "Could not find field labeled #{@value.inspect}"
end end
def text(element) def text(element)
str = Webrat::XML.all_inner_text(element) str = element.inner_text
str.gsub!("\n","") str.gsub!("\n","")
str.strip! str.strip!
str.squeeze!(" ") str.squeeze!(" ")
str str
end end
end end
# Locates a form field based on a <tt>label</tt> element in the HTML source. # Locates a form field based on a <tt>label</tt> element in the HTML source.
# This can be useful in order to verify that a field is pre-filled with the # This can be useful in order to verify that a field is pre-filled with the
# correct value. # correct value.
@ -51,6 +51,6 @@ module Webrat
def field_labeled(label, *field_types) def field_labeled(label, *field_types)
FieldLabeledLocator.new(@session, dom, label, *field_types).locate! FieldLabeledLocator.new(@session, dom, label, *field_types).locate!
end end
end end
end end

View File

@ -2,24 +2,24 @@ require "webrat/core/locators/locator"
module Webrat module Webrat
module Locators module Locators
class FieldLocator < Locator # :nodoc: class FieldLocator < Locator # :nodoc:
def locate def locate
FieldByIdLocator.new(@session, @dom, @value).locate || FieldByIdLocator.new(@session, @dom, @value).locate ||
FieldNamedLocator.new(@session, @dom, @value, *@field_types).locate || FieldNamedLocator.new(@session, @dom, @value, *@field_types).locate ||
FieldLabeledLocator.new(@session, @dom, @value, *@field_types).locate FieldLabeledLocator.new(@session, @dom, @value, *@field_types).locate
end end
def error_message def error_message
"Could not find field: #{@value.inspect}" "Could not find field: #{@value.inspect}"
end end
end end
def field(*args) # :nodoc: def field(*args) # :nodoc:
FieldLocator.new(@session, dom, *args).locate! FieldLocator.new(@session, dom, *args).locate!
end end
end end
end end

View File

@ -2,40 +2,40 @@ require "webrat/core/locators/locator"
module Webrat module Webrat
module Locators module Locators
class FieldNamedLocator < Locator # :nodoc: class FieldNamedLocator < Locator # :nodoc:
def locate def locate
Field.load(@session, field_element) Field.load(@session, field_element)
end end
def field_element def field_element
field_elements.detect do |field_element| field_elements.detect do |field_element|
Webrat::XML.attribute(field_element, "name") == @value.to_s field_element["name"] == @value.to_s
end end
end end
def field_elements def field_elements
Webrat::XML.xpath_search(@dom, *xpath_searches) @dom.xpath(*xpath_searches)
end end
def xpath_searches def xpath_searches
if @field_types.any? if @field_types.any?
@field_types.map { |field_type| field_type.xpath_search }.flatten @field_types.map { |field_type| field_type.xpath_search }.flatten
else else
Array(Field.xpath_search) Field.xpath_search
end end
end end
def error_message def error_message
"Could not find field named #{@value.inspect}" "Could not find field named #{@value.inspect}"
end end
end end
def field_named(name, *field_types) def field_named(name, *field_types)
FieldNamedLocator.new(@session, dom, name, *field_types).locate! FieldNamedLocator.new(@session, dom, name, *field_types).locate!
end end
end end
end end

View File

@ -2,18 +2,18 @@ require "webrat/core/locators/locator"
module Webrat module Webrat
module Locators module Locators
class FormLocator < Locator # :nodoc: class FormLocator < Locator # :nodoc:
def locate def locate
Form.load(@session, form_element) Form.load(@session, form_element)
end end
def form_element def form_element
Webrat::XML.css_at(@dom, "#" + @value) @dom.css("#" + @value).first || @dom.css(@value).first
end end
end end
end end
end end

View File

@ -3,32 +3,32 @@ require "webrat/core/locators/locator"
module Webrat module Webrat
module Locators module Locators
class LabelLocator < Locator # :nodoc: class LabelLocator < Locator # :nodoc:
def locate def locate
Label.load(@session, label_element) Label.load(@session, label_element)
end end
def label_element def label_element
label_elements.detect do |label_element| label_elements.detect do |label_element|
text(label_element) =~ /^\W*#{Regexp.escape(@value.to_s)}\b/i text(label_element) =~ /^\W*#{Regexp.escape(@value.to_s)}(\b|\Z)/i
end end
end end
def label_elements def label_elements
Webrat::XML.xpath_search(@dom, Label.xpath_search) @dom.xpath(*Label.xpath_search)
end end
def text(label_element) def text(label_element)
str = Webrat::XML.all_inner_text(label_element) str = label_element.inner_text
str.gsub!("\n","") str.gsub!("\n","")
str.strip! str.strip!
str.squeeze!(" ") str.squeeze!(" ")
str str
end end
end end
end end
end end

View File

@ -2,24 +2,24 @@ require "webrat/core/locators/locator"
module Webrat module Webrat
module Locators module Locators
class LinkLocator < Locator # :nodoc: class LinkLocator < Locator # :nodoc:
def locate def locate
Link.load(@session, link_element) Link.load(@session, link_element)
end end
def link_element def link_element
matching_links.min { |a, b| Webrat::XML.all_inner_text(a).length <=> Webrat::XML.all_inner_text(b).length } matching_links.min { |a, b| a.inner_text.length <=> b.inner_text.length }
end end
def matching_links def matching_links
@matching_links ||= link_elements.select do |link_element| @matching_links ||= link_elements.select do |link_element|
matches_text?(link_element) || matches_text?(link_element) ||
matches_id?(link_element) matches_id?(link_element)
end end
end end
def matches_text?(link) def matches_text?(link)
if @value.is_a?(Regexp) if @value.is_a?(Regexp)
matcher = @value matcher = @value
@ -27,40 +27,48 @@ module Webrat
matcher = /#{Regexp.escape(@value.to_s)}/i matcher = /#{Regexp.escape(@value.to_s)}/i
end end
replace_nbsp(Webrat::XML.all_inner_text(link)) =~ matcher || replace_nbsp(link.inner_text) =~ matcher ||
replace_nbsp_ref(Webrat::XML.inner_html(link)) =~ matcher || replace_nbsp_ref(link.inner_html) =~ matcher ||
Webrat::XML.attribute(link, "title")=~ matcher link["title"] =~ matcher
end end
def matches_id?(link) def matches_id?(link)
if @value.is_a?(Regexp) if @value.is_a?(Regexp)
(Webrat::XML.attribute(link, "id") =~ @value) ? true : false link["id"] =~ @value ? true : false
else else
(Webrat::XML.attribute(link, "id") == @value) ? true : false link["id"] == @value ? true : false
end end
end end
def link_elements def link_elements
Webrat::XML.xpath_search(@dom, *Link.xpath_search) @dom.xpath(*Link.xpath_search)
end end
def replace_nbsp(str) def replace_nbsp(str)
str.gsub([0xA0].pack('U'), ' ') if str.respond_to?(:valid_encoding?)
if str.valid_encoding?
str.gsub(/\xc2\xa0/u, ' ')
else
str.force_encoding('UTF-8').gsub(/\xc2\xa0/u, ' ')
end
else
str.gsub(/\xc2\xa0/u, ' ')
end
end end
def replace_nbsp_ref(str) def replace_nbsp_ref(str)
str.gsub('&#xA0;',' ').gsub('&nbsp;', ' ') str.gsub('&#xA0;',' ').gsub('&nbsp;', ' ')
end end
def error_message def error_message
"Could not find link with text or title or id #{@value.inspect}" "Could not find link with text or title or id #{@value.inspect}"
end end
end end
def find_link(text_or_title_or_id) #:nodoc: def find_link(text_or_title_or_id) #:nodoc:
LinkLocator.new(@session, dom, text_or_title_or_id).locate! LinkLocator.new(@session, dom, text_or_title_or_id).locate!
end end
end end
end end

View File

@ -1,6 +1,6 @@
module Webrat module Webrat
module Locators module Locators
class Locator # :nodoc: class Locator # :nodoc:
def initialize(session, dom, value, *field_types) def initialize(session, dom, value, *field_types)
@ -9,12 +9,12 @@ module Webrat
@value = value @value = value
@field_types = field_types @field_types = field_types
end end
def locate! def locate!
locate || raise(NotFoundError.new(error_message)) locate || raise(NotFoundError.new(error_message))
end end
end end
end end
end end

View File

@ -3,44 +3,44 @@ require "webrat/core/locators/locator"
module Webrat module Webrat
module Locators module Locators
class SelectOptionLocator < Locator # :nodoc: class SelectOptionLocator < Locator # :nodoc:
def initialize(session, dom, option_text, id_or_name_or_label) def initialize(session, dom, option_text, id_or_name_or_label)
@session = session @session = session
@dom = dom @dom = dom
@option_text = option_text @option_text = option_text
@id_or_name_or_label = id_or_name_or_label @id_or_name_or_label = id_or_name_or_label
end end
def locate def locate
if @id_or_name_or_label if @id_or_name_or_label
field = FieldLocator.new(@session, @dom, @id_or_name_or_label, SelectField).locate! field = FieldLocator.new(@session, @dom, @id_or_name_or_label, SelectField).locate!
field.options.detect do |o| field.options.detect do |o|
if @option_text.is_a?(Regexp) if @option_text.is_a?(Regexp)
Webrat::XML.inner_html(o.element) =~ @option_text o.element.inner_text =~ @option_text
else else
Webrat::XML.inner_html(o.element) == @option_text.to_s o.inner_text == @option_text.to_s
end end
end end
else else
option_element = option_elements.detect do |o| option_element = option_elements.detect do |o|
if @option_text.is_a?(Regexp) if @option_text.is_a?(Regexp)
Webrat::XML.inner_html(o) =~ @option_text o.inner_text =~ @option_text
else else
Webrat::XML.inner_html(o) == @option_text.to_s o.inner_text == @option_text.to_s
end end
end end
SelectOption.load(@session, option_element) SelectOption.load(@session, option_element)
end end
end end
def option_elements def option_elements
Webrat::XML.xpath_search(@dom, *SelectOption.xpath_search) @dom.xpath(*SelectOption.xpath_search)
end end
def error_message def error_message
if @id_or_name_or_label if @id_or_name_or_label
"The '#{@option_text}' option was not found in the #{@id_or_name_or_label.inspect} select box" "The '#{@option_text}' option was not found in the #{@id_or_name_or_label.inspect} select box"
@ -48,12 +48,12 @@ module Webrat
"Could not find option #{@option_text.inspect}" "Could not find option #{@option_text.inspect}"
end end
end end
end end
def select_option(option_text, id_or_name_or_label = nil) #:nodoc: def select_option(option_text, id_or_name_or_label = nil) #:nodoc:
SelectOptionLocator.new(@session, dom, option_text, id_or_name_or_label).locate! SelectOptionLocator.new(@session, dom, option_text, id_or_name_or_label).locate!
end end
end end
end end

View File

@ -1,6 +1,8 @@
require "logger"
module Webrat module Webrat
module Logging #:nodoc: module Logging #:nodoc:
def debug_log(message) # :nodoc: def debug_log(message) # :nodoc:
return unless logger return unless logger
logger.debug message logger.debug message
@ -11,11 +13,11 @@ module Webrat
when :rails when :rails
defined?(RAILS_DEFAULT_LOGGER) ? RAILS_DEFAULT_LOGGER : nil defined?(RAILS_DEFAULT_LOGGER) ? RAILS_DEFAULT_LOGGER : nil
when :merb when :merb
Merb.logger ::Merb.logger
else else
nil @logger ||= ::Logger.new("webrat.log")
end end
end end
end end
end end

View File

@ -1,40 +1,39 @@
module Webrat module Webrat
module Matchers module Matchers
class HasContent #:nodoc: class HasContent #:nodoc:
def initialize(content) def initialize(content)
@content = content @content = content
end end
def matches?(stringlike) def matches?(stringlike)
if Webrat.configuration.parse_with_nokogiri? @document = Webrat::XML.document(stringlike)
@document = Webrat.nokogiri_document(stringlike) @element = @document.inner_text
else
@document = Webrat.hpricot_document(stringlike)
end
@element = Webrat::XML.inner_text(@document)
case @content case @content
when String when String
@element.include?(@content) @element.gsub(/\s+/, ' ').include?(@content)
when Regexp when Regexp
@element.match(@content) @element.match(@content)
end end
end end
# ==== Returns # ==== Returns
# String:: The failure message. # String:: The failure message.
def failure_message def failure_message
"expected the following element's content to #{content_message}:\n#{@element}" "expected the following element's content to #{content_message}:\n#{squeeze_space(@element)}"
end end
# ==== Returns # ==== Returns
# String:: The failure message to be displayed in negative matches. # String:: The failure message to be displayed in negative matches.
def negative_failure_message def negative_failure_message
"expected the following element's content to not #{content_message}:\n#{@element}" "expected the following element's content to not #{content_message}:\n#{squeeze_space(@element)}"
end end
def squeeze_space(inner_text)
inner_text.gsub(/^\s*$/, "").squeeze("\n")
end
def content_message def content_message
case @content case @content
when String when String
@ -44,26 +43,26 @@ module Webrat
end end
end end
end end
# Matches the contents of an HTML document with # Matches the contents of an HTML document with
# whatever string is supplied # whatever string is supplied
def contain(content) def contain(content)
HasContent.new(content) HasContent.new(content)
end end
# Asserts that the body of the response contain # Asserts that the body of the response contain
# the supplied string or regexp # the supplied string or regexp
def assert_contain(content) def assert_contain(content)
hc = HasContent.new(content) hc = HasContent.new(content)
assert hc.matches?(response_body), hc.failure_message assert hc.matches?(response_body), hc.failure_message
end end
# Asserts that the body of the response # Asserts that the body of the response
# does not contain the supplied string or regepx # does not contain the supplied string or regepx
def assert_not_contain(content) def assert_not_contain(content)
hc = HasContent.new(content) hc = HasContent.new(content)
assert !hc.matches?(response_body), hc.negative_failure_message assert !hc.matches?(response_body), hc.negative_failure_message
end end
end end
end end

View File

@ -1,26 +1,48 @@
require "webrat/core/matchers/have_xpath"
module Webrat module Webrat
module Matchers module Matchers
class HaveSelector < HaveXpath #:nodoc:
class HaveSelector < HaveXpath #:nodoc:
# ==== Returns # ==== Returns
# String:: The failure message. # String:: The failure message.
def failure_message def failure_message
"expected following text to match selector #{@expected}:\n#{@document}" "expected following output to contain a #{tag_inspect} tag:\n#{@document}"
end end
# ==== Returns # ==== Returns
# String:: The failure message to be displayed in negative matches. # String:: The failure message to be displayed in negative matches.
def negative_failure_message def negative_failure_message
"expected following text to not match selector #{@expected}:\n#{@document}" "expected following output to omit a #{tag_inspect}:\n#{@document}"
end end
def tag_inspect
options = @options.dup
count = options.delete(:count)
content = options.delete(:content)
html = "<#{@expected}"
options.each do |k,v|
html << " #{k}='#{v}'"
end
if content
html << ">#{content}</#{@expected}>"
else
html << "/>"
end
html
end
def query def query
Nokogiri::CSS::Parser.parse(*super).map { |ast| ast.to_xpath } Nokogiri::CSS.parse(@expected.to_s).map do |ast|
ast.to_xpath
end.first
end end
end end
# Matches HTML content against a CSS 3 selector. # Matches HTML content against a CSS 3 selector.
# #
# ==== Parameters # ==== Parameters
@ -28,25 +50,25 @@ module Webrat
# #
# ==== Returns # ==== Returns
# HaveSelector:: A new have selector matcher. # HaveSelector:: A new have selector matcher.
def have_selector(expected, &block) def have_selector(name, attributes = {}, &block)
HaveSelector.new(expected, &block) HaveSelector.new(name, attributes, &block)
end end
alias_method :match_selector, :have_selector alias_method :match_selector, :have_selector
# Asserts that the body of the response contains # Asserts that the body of the response contains
# the supplied selector # the supplied selector
def assert_have_selector(expected) def assert_have_selector(name, attributes = {}, &block)
hs = HaveSelector.new(expected) matcher = HaveSelector.new(name, attributes, &block)
assert hs.matches?(response_body), hs.failure_message assert matcher.matches?(response_body), matcher.failure_message
end end
# Asserts that the body of the response # Asserts that the body of the response
# does not contain the supplied string or regepx # does not contain the supplied string or regepx
def assert_have_no_selector(expected) def assert_have_no_selector(name, attributes = {}, &block)
hs = HaveSelector.new(expected) matcher = HaveSelector.new(name, attributes, &block)
assert !hs.matches?(response_body), hs.negative_failure_message assert !matcher.matches?(response_body), matcher.negative_failure_message
end end
end end
end end

View File

@ -1,71 +1,21 @@
require "webrat/core/matchers/have_selector"
module Webrat module Webrat
module HaveTagMatcher module HaveTagMatcher
class HaveTag < ::Webrat::Matchers::HaveSelector #:nodoc: def have_tag(*args, &block)
# ==== Returns have_selector(*args, &block)
# String:: The failure message.
def failure_message
"expected following output to contain a #{tag_inspect} tag:\n#{@document}"
end
# ==== Returns
# String:: The failure message to be displayed in negative matches.
def negative_failure_message
"expected following output to omit a #{tag_inspect}:\n#{@document}"
end
def tag_inspect
options = @expected.last.dup
content = options.delete(:content)
html = "<#{@expected.first}"
options.each do |k,v|
html << " #{k}='#{v}'"
end
if content
html << ">#{content}</#{@expected.first}>"
else
html << "/>"
end
html
end
def query
options = @expected.last.dup
selector = @expected.first.to_s
selector << ":contains('#{options.delete(:content)}')" if options[:content]
options.each do |key, value|
selector << "[#{key}='#{value}']"
end
Nokogiri::CSS::Parser.parse(selector).map { |ast| ast.to_xpath }
end
end end
def have_tag(name, attributes = {}, &block)
HaveTag.new([name, attributes], &block)
end
alias_method :match_tag, :have_tag alias_method :match_tag, :have_tag
# Asserts that the body of the response contains def assert_have_tag(*args, &block)
# the supplied tag with the associated selectors assert_have_selector(*args, &block)
def assert_have_tag(name, attributes = {})
ht = HaveTag.new([name, attributes])
assert ht.matches?(response_body), ht.failure_message
end end
# Asserts that the body of the response def assert_have_no_tag(*args, &block)
# does not contain the supplied string or regepx assert_have_no_selector(*args, &block)
def assert_have_no_tag(name, attributes = {})
ht = HaveTag.new([name, attributes])
assert !ht.matches?(response_body), ht.negative_failure_message
end end
end end
end end

View File

@ -1,59 +1,73 @@
require "webrat/core/xml/nokogiri" require "webrat/core/xml"
require "webrat/core/xml/rexml"
module Webrat module Webrat
module Matchers module Matchers
class HaveXpath #:nodoc: class HaveXpath #:nodoc:
def initialize(expected, &block) def initialize(expected, options = {}, &block)
@expected = expected @expected = expected
@options = options
@block = block @block = block
end end
def matches?(stringlike) def matches?(stringlike, &block)
if Webrat.configuration.parse_with_nokogiri? @block ||= block
matches_nokogiri?(stringlike) matched = matches(stringlike)
@block.call(matched) if @block
if @options[:count]
matched.size == @options[:count].to_i
else else
matches_rexml?(stringlike) matched.any?
end end
end end
def matches_rexml?(stringlike)
if REXML::Node === stringlike || Array === stringlike
@query = query.map { |q| q.gsub(%r'//', './') }
else
@query = query
end
@document = Webrat.rexml_document(stringlike) def matches(stringlike)
nokogiri_matches(stringlike)
matched = @query.map do |q|
if @document.is_a?(Array)
@document.map { |d| REXML::XPath.match(d, q) }
else
REXML::XPath.match(@document, q)
end
end.flatten.compact
matched.any? && (!@block || @block.call(matched))
end end
def matches_nokogiri?(stringlike) def nokogiri_matches(stringlike)
if Nokogiri::XML::NodeSet === stringlike if Nokogiri::XML::NodeSet === stringlike
@query = query.map { |q| q.gsub(%r'//', './') } @query = query.gsub(%r'^//', './/')
else else
@query = query @query = query
end end
add_options_conditions_to(@query)
@document = Webrat::XML.document(stringlike) @document = Webrat::XML.document(stringlike)
matched = @document.xpath(*@query) @document.xpath(*@query)
matched.any? && (!@block || @block.call(matched))
end end
def add_options_conditions_to(query)
add_attributes_conditions_to(query)
add_content_condition_to(query)
end
def add_attributes_conditions_to(query)
attribute_conditions = []
@options.each do |key, value|
next if [:content, :count].include?(key)
attribute_conditions << "@#{key} = #{xpath_escape(value)}"
end
if attribute_conditions.any?
query << "[#{attribute_conditions.join(' and ')}]"
end
end
def add_content_condition_to(query)
if @options[:content]
query << "[contains(., #{xpath_escape(@options[:content])})]"
end
end
def query def query
[@expected].flatten.compact @expected
end end
# ==== Returns # ==== Returns
# String:: The failure message. # String:: The failure message.
def failure_message def failure_message
@ -64,9 +78,26 @@ module Webrat
# String:: The failure message to be displayed in negative matches. # String:: The failure message to be displayed in negative matches.
def negative_failure_message def negative_failure_message
"expected following text to not match xpath #{@expected}:\n#{@document}" "expected following text to not match xpath #{@expected}:\n#{@document}"
end end
protected
def xpath_escape(string)
if string.include?("'") && string.include?('"')
parts = string.split("'").map do |part|
"'#{part}'"
end
"concat(" + parts.join(", \"'\", ") + ")"
elsif string.include?("'")
"\"#{string}\""
else
"'#{string}'"
end
end
end end
# Matches HTML content against an XPath query # Matches HTML content against an XPath query
# #
# ==== Parameters # ==== Parameters
@ -74,20 +105,20 @@ module Webrat
# #
# ==== Returns # ==== Returns
# HaveXpath:: A new have xpath matcher. # HaveXpath:: A new have xpath matcher.
def have_xpath(expected, &block) def have_xpath(expected, options = {}, &block)
HaveXpath.new(expected, &block) HaveXpath.new(expected, options, &block)
end end
alias_method :match_xpath, :have_xpath alias_method :match_xpath, :have_xpath
def assert_have_xpath(expected, &block) def assert_have_xpath(expected, options = {}, &block)
hs = HaveXpath.new(expected, &block) hs = HaveXpath.new(expected, options, &block)
assert hs.matches?(response_body), hs.failure_message assert hs.matches?(response_body), hs.failure_message
end end
def assert_have_no_xpath(expected, &block) def assert_have_no_xpath(expected, options = {}, &block)
hs = HaveXpath.new(expected, &block) hs = HaveXpath.new(expected, options, &block)
assert !hs.matches?(response_body), hs.negative_failure_message assert !hs.matches?(response_body), hs.negative_failure_message
end end
end end
end end

View File

@ -3,22 +3,26 @@ module Webrat
def self.delegate_to_session(*meths) def self.delegate_to_session(*meths)
meths.each do |meth| meths.each do |meth|
self.class_eval <<-RUBY self.class_eval(<<-RUBY, __FILE__, __LINE__)
def #{meth}(*args, &blk) def #{meth}(*args, &blk)
webrat_session.#{meth}(*args, &blk) webrat_session.#{meth}(*args, &blk)
end end
RUBY RUBY
end end
end end
def webrat def webrat
webrat_session webrat_session
end end
def webrat_session def webrat_session
@_webrat_session ||= ::Webrat.session_class.new(self) @_webrat_session ||= begin
session = Webrat.session_class.new
session.adapter = Webrat.adapter_class.new(self) if session.respond_to?(:adapter=)
session
end
end end
# all of these methods delegate to the @session, which should # all of these methods delegate to the @session, which should
# be created transparently. # be created transparently.
# #
@ -31,10 +35,11 @@ module Webrat
:header, :http_accept, :basic_auth, :header, :http_accept, :basic_auth,
:save_and_open_page, :save_and_open_page,
:fills_in, :fill_in, :fills_in, :fill_in,
:checks, :check, :checks, :check,
:unchecks, :uncheck, :unchecks, :uncheck,
:chooses, :choose, :chooses, :choose,
:selects, :select, :selects, :select,
:unselects, :unselect,
:attaches_file, :attach_file, :attaches_file, :attach_file,
:current_page, :current_page,
:current_url, :current_url,
@ -47,15 +52,13 @@ module Webrat
:select_option, :select_option,
:set_hidden_field, :submit_form, :set_hidden_field, :submit_form,
:request_page, :current_dom, :request_page, :current_dom,
:response_body, :response_body,
:selects_date, :selects_time, :selects_datetime, :selects_date, :selects_time, :selects_datetime,
:select_date, :select_time, :select_datetime, :select_date, :select_time, :select_datetime,
:field_by_xpath, :field_by_xpath,
:field_with_id, :field_with_id,
:selenium, :selenium,
:simulate, :automate :simulate, :automate,
:field_named
end end
end end

View File

@ -1,29 +1,18 @@
module Webrat #:nodoc: module Webrat #:nodoc:
module MIME #:nodoc: module MIME #:nodoc:
MIME_TYPES = Rack::Mime::MIME_TYPES.dup.merge(
def self.mime_type(string_or_symbol) #:nodoc: ".multipart_form" => "multipart/form-data",
if string_or_symbol.is_a?(String) ".url_encoded_form" => "application/x-www-form-urlencoded"
string_or_symbol ).freeze
else
case string_or_symbol def mime_type(type)
when :text then "text/plain" return type if type.nil? || type.to_s.include?("/")
when :html then "text/html" type = ".#{type}" unless type.to_s[0] == ?.
when :js then "text/javascript" MIME_TYPES.fetch(type) { |invalid_type|
when :css then "text/css" raise ArgumentError.new("Invalid Mime type: #{invalid_type}")
when :ics then "text/calendar" }
when :csv then "text/csv"
when :xml then "application/xml"
when :rss then "application/rss+xml"
when :atom then "application/atom+xml"
when :yaml then "application/x-yaml"
when :multipart_form then "multipart/form-data"
when :url_encoded_form then "application/x-www-form-urlencoded"
when :json then "application/json"
else
raise ArgumentError.new("Invalid Mime type: #{string_or_symbol.inspect}")
end
end
end end
module_function :mime_type
end end
end end

View File

@ -2,49 +2,27 @@ module Webrat
module SaveAndOpenPage module SaveAndOpenPage
# Saves the page out to RAILS_ROOT/tmp/ and opens it in the default # Saves the page out to RAILS_ROOT/tmp/ and opens it in the default
# web browser if on OS X. Useful for debugging. # web browser if on OS X. Useful for debugging.
# #
# Example: # Example:
# save_and_open_page # save_and_open_page
def save_and_open_page def save_and_open_page
return unless File.exist?(saved_page_dir) return unless File.exist?(Webrat.configuration.saved_pages_dir)
filename = "#{Webrat.configuration.saved_pages_dir}/webrat-#{Time.now.to_i}.html"
filename = "#{saved_page_dir}/webrat-#{Time.now.to_i}.html"
File.open(filename, "w") do |f| File.open(filename, "w") do |f|
f.write rewrite_css_and_image_references(response_body) f.write response_body
end end
open_in_browser(filename) open_in_browser(filename)
end end
def open_in_browser(path) # :nodoc
platform = ruby_platform
if platform =~ /cygwin/ || platform =~ /win32/
`rundll32 url.dll,FileProtocolHandler #{path.gsub("/", "\\\\")}`
elsif platform =~ /darwin/
`open #{path}`
end
end
def rewrite_css_and_image_references(response_html) # :nodoc:
return response_html unless doc_root
response_html.gsub(/"\/(stylesheets|images)/, doc_root + '/\1')
end
def saved_page_dir #:nodoc:
File.expand_path(".")
end
def doc_root #:nodoc:
nil
end
private
# accessor for testing def open_in_browser(path) # :nodoc
def ruby_platform require "launchy"
RUBY_PLATFORM Launchy::Browser.run(path)
rescue LoadError
warn "Sorry, you need to install launchy to open pages: `gem install launchy`"
end end
end end
end end

View File

@ -6,36 +6,37 @@ module Webrat
# An HTML element (link, button, field, etc.) that Webrat expected was not found on the page # An HTML element (link, button, field, etc.) that Webrat expected was not found on the page
class NotFoundError < WebratError class NotFoundError < WebratError
end end
class Scope class Scope
include Logging include Logging
include Locators include Locators
def self.from_page(session, response, response_body) #:nodoc: def self.from_page(session, response, response_body) #:nodoc:
new(session) do new(session) do
@response = response @response = response
@response_body = response_body @response_body = response_body
end end
end end
def self.from_scope(session, scope, selector) #:nodoc: def self.from_scope(session, scope, selector) #:nodoc:
new(session) do new(session) do
@scope = scope @scope = scope
@selector = selector @selector = selector
end end
end end
attr_reader :session attr_reader :session
def initialize(session, &block) #:nodoc: def initialize(session, &block) #:nodoc:
@selector, @dom = nil
@session = session @session = session
instance_eval(&block) if block_given? instance_eval(&block) if block_given?
if @selector && scoped_dom.nil? if @selector && scoped_dom.nil?
raise Webrat::NotFoundError.new("The scope was not found on the page: #{@selector.inspect}") raise Webrat::NotFoundError.new("The scope was not found on the page: #{@selector.inspect}")
end end
end end
# Verifies an input field or textarea exists on the current page, and stores a value for # Verifies an input field or textarea exists on the current page, and stores a value for
# it which will be sent when the form is submitted. # it which will be sent when the form is submitted.
# #
@ -53,7 +54,7 @@ module Webrat
end end
webrat_deprecate :fills_in, :fill_in webrat_deprecate :fills_in, :fill_in
# Verifies that a hidden field exists on the current page and sets # Verifies that a hidden field exists on the current page and sets
# the value to that given by the <tt>:to</tt> option. # the value to that given by the <tt>:to</tt> option.
# #
@ -63,7 +64,7 @@ module Webrat
field = locate_field(field_locator, HiddenField) field = locate_field(field_locator, HiddenField)
field.set(options[:to]) field.set(options[:to])
end end
# Verifies that an input checkbox exists on the current page and marks it # Verifies that an input checkbox exists on the current page and marks it
# as checked, so that the value will be submitted with the form. # as checked, so that the value will be submitted with the form.
# #
@ -74,7 +75,7 @@ module Webrat
end end
webrat_deprecate :checks, :check webrat_deprecate :checks, :check
# Verifies that an input checkbox exists on the current page and marks it # Verifies that an input checkbox exists on the current page and marks it
# as unchecked, so that the value will not be submitted with the form. # as unchecked, so that the value will not be submitted with the form.
# #
@ -85,7 +86,7 @@ module Webrat
end end
webrat_deprecate :unchecks, :uncheck webrat_deprecate :unchecks, :uncheck
# Verifies that an input radio button exists on the current page and marks it # Verifies that an input radio button exists on the current page and marks it
# as checked, so that the value will be submitted with the form. # as checked, so that the value will be submitted with the form.
# #
@ -96,7 +97,7 @@ module Webrat
end end
webrat_deprecate :chooses, :choose webrat_deprecate :chooses, :choose
# Verifies that a an option element exists on the current page with the specified # Verifies that a an option element exists on the current page with the specified
# text. You can optionally restrict the search to a specific select list by # text. You can optionally restrict the search to a specific select list by
# assigning <tt>options[:from]</tt> the value of the select list's name or # assigning <tt>options[:from]</tt> the value of the select list's name or
@ -111,7 +112,22 @@ module Webrat
end end
webrat_deprecate :selects, :select webrat_deprecate :selects, :select
# Verifies that a an option element exists on the current page with the specified
# text. You can optionally restrict the search to a specific select list by
# assigning <tt>options[:from]</tt> the value of the select list's name or
# a label. Remove the option's value before the form is submitted.
#
# Examples:
# unselect "January"
# unselect "February", :from => "event_month"
# unselect "February", :from => "Event Month"
def unselect(option_text, options={})
select_option(option_text, options[:from]).unchoose
end
webrat_deprecate :unselects, :unselect
DATE_TIME_SUFFIXES = { DATE_TIME_SUFFIXES = {
:year => '1i', :year => '1i',
:month => '2i', :month => '2i',
@ -120,9 +136,9 @@ module Webrat
:minute => '5i' :minute => '5i'
} }
# Verifies that date elements (year, month, day) exist on the current page # Verifies that date elements (year, month, day) exist on the current page
# with the specified values. You can optionally restrict the search to a specific # with the specified values. You can optionally restrict the search to a specific
# date's elements by assigning <tt>options[:from]</tt> the value of the date's # date's elements by assigning <tt>options[:from]</tt> the value of the date's
# label. Selects all the date elements with date provided. The date provided may # label. Selects all the date elements with date provided. The date provided may
# be a string or a Date/Time object. # be a string or a Date/Time object.
# #
@ -136,15 +152,15 @@ module Webrat
# select_date Date.parse("December 25, 2000"), :from => "Event" # select_date Date.parse("December 25, 2000"), :from => "Event"
# select_date "April 26, 1982", :id_prefix => 'birthday' # select_date "April 26, 1982", :id_prefix => 'birthday'
def select_date(date_to_select, options ={}) def select_date(date_to_select, options ={})
date = date_to_select.is_a?(Date) || date_to_select.is_a?(Time) ? date = date_to_select.is_a?(Date) || date_to_select.is_a?(Time) ?
date_to_select : Date.parse(date_to_select) date_to_select : Date.parse(date_to_select)
id_prefix = locate_id_prefix(options) do id_prefix = locate_id_prefix(options) do
year_field = FieldByIdLocator.new(@session, dom, /(.*?)_#{DATE_TIME_SUFFIXES[:year]}$/).locate year_field = FieldByIdLocator.new(@session, dom, /(.*?)_#{DATE_TIME_SUFFIXES[:year]}$/).locate
raise NotFoundError.new("No date fields were found") unless year_field && year_field.id =~ /(.*?)_1i/ raise NotFoundError.new("No date fields were found") unless year_field && year_field.id =~ /(.*?)_1i/
$1 $1
end end
select date.year, :from => "#{id_prefix}_#{DATE_TIME_SUFFIXES[:year]}" select date.year, :from => "#{id_prefix}_#{DATE_TIME_SUFFIXES[:year]}"
select date.strftime('%B'), :from => "#{id_prefix}_#{DATE_TIME_SUFFIXES[:month]}" select date.strftime('%B'), :from => "#{id_prefix}_#{DATE_TIME_SUFFIXES[:month]}"
select date.day, :from => "#{id_prefix}_#{DATE_TIME_SUFFIXES[:day]}" select date.day, :from => "#{id_prefix}_#{DATE_TIME_SUFFIXES[:day]}"
@ -152,9 +168,9 @@ module Webrat
webrat_deprecate :selects_date, :select_date webrat_deprecate :selects_date, :select_date
# Verifies that time elements (hour, minute) exist on the current page # Verifies that time elements (hour, minute) exist on the current page
# with the specified values. You can optionally restrict the search to a specific # with the specified values. You can optionally restrict the search to a specific
# time's elements by assigning <tt>options[:from]</tt> the value of the time's # time's elements by assigning <tt>options[:from]</tt> the value of the time's
# label. Selects all the time elements with date provided. The time provided may # label. Selects all the time elements with date provided. The time provided may
# be a string or a Time object. # be a string or a Time object.
# #
@ -164,28 +180,28 @@ module Webrat
# #
# Note: Just like Rails' time_select helper this assumes the form is using # Note: Just like Rails' time_select helper this assumes the form is using
# 24 hour select boxes, and not 12 hours with AM/PM. # 24 hour select boxes, and not 12 hours with AM/PM.
# #
# Examples: # Examples:
# select_time "9:30" # select_time "9:30"
# select_date "3:30PM", :from => "Party Time" # select_date "3:30PM", :from => "Party Time"
# select_date Time.parse("10:00PM"), :from => "Event" # select_date Time.parse("10:00PM"), :from => "Event"
# select_date "10:30AM", :id_prefix => 'meeting' # select_date "10:30AM", :id_prefix => 'meeting'
def select_time(time_to_select, options ={}) def select_time(time_to_select, options ={})
time = time_to_select.is_a?(Time) ? time_to_select : Time.parse(time_to_select) time = time_to_select.is_a?(Time) ? time_to_select : Time.parse(time_to_select)
id_prefix = locate_id_prefix(options) do id_prefix = locate_id_prefix(options) do
hour_field = FieldByIdLocator.new(@session, dom, /(.*?)_#{DATE_TIME_SUFFIXES[:hour]}$/).locate hour_field = FieldByIdLocator.new(@session, dom, /(.*?)_#{DATE_TIME_SUFFIXES[:hour]}$/).locate
raise NotFoundError.new("No time fields were found") unless hour_field && hour_field.id =~ /(.*?)_4i/ raise NotFoundError.new("No time fields were found") unless hour_field && hour_field.id =~ /(.*?)_4i/
$1 $1
end end
select time.hour.to_s.rjust(2,'0'), :from => "#{id_prefix}_#{DATE_TIME_SUFFIXES[:hour]}" select time.hour.to_s.rjust(2,'0'), :from => "#{id_prefix}_#{DATE_TIME_SUFFIXES[:hour]}"
select time.min.to_s.rjust(2,'0'), :from => "#{id_prefix}_#{DATE_TIME_SUFFIXES[:minute]}" select time.min.to_s.rjust(2,'0'), :from => "#{id_prefix}_#{DATE_TIME_SUFFIXES[:minute]}"
end end
webrat_deprecate :selects_time, :select_time webrat_deprecate :selects_time, :select_time
# Verifies and selects all the date and time elements on the current page. # Verifies and selects all the date and time elements on the current page.
# See #select_time and #select_date for more details and available options. # See #select_time and #select_date for more details and available options.
# #
# Examples: # Examples:
@ -194,23 +210,23 @@ module Webrat
# select_datetime Time.parse("December 25, 2000 15:30"), :from => "Event" # select_datetime Time.parse("December 25, 2000 15:30"), :from => "Event"
# select_datetime "April 26, 1982 5:50PM", :id_prefix => 'birthday' # select_datetime "April 26, 1982 5:50PM", :id_prefix => 'birthday'
def select_datetime(time_to_select, options ={}) def select_datetime(time_to_select, options ={})
time = time_to_select.is_a?(Time) ? time_to_select : Time.parse(time_to_select) time = time_to_select.is_a?(Time) ? time_to_select : Time.parse(time_to_select)
options[:id_prefix] ||= (options[:from] ? FieldByIdLocator.new(@session, dom, options[:from]).locate : nil) options[:id_prefix] ||= (options[:from] ? FieldByIdLocator.new(@session, dom, options[:from]).locate : nil)
select_date time, options select_date time, options
select_time time, options select_time time, options
end end
webrat_deprecate :selects_datetime, :select_datetime webrat_deprecate :selects_datetime, :select_datetime
# Verifies that an input file field exists on the current page and sets # Verifies that an input file field exists on the current page and sets
# its value to the given +file+, so that the file will be uploaded # its value to the given +file+, so that the file will be uploaded
# along with the form. An optional <tt>content_type</tt> may be given. # along with the form. An optional <tt>content_type</tt> may be given.
# #
# Example: # Example:
# attaches_file "Resume", "/path/to/the/resume.txt" # attach_file "Resume", "/path/to/the/resume.txt"
# attaches_file "Photo", "/path/to/the/image.png", "image/png" # attach_file "Photo", "/path/to/the/image.png", "image/png"
def attach_file(field_locator, path, content_type = nil) def attach_file(field_locator, path, content_type = nil)
locate_field(field_locator, FileField).set(path, content_type) locate_field(field_locator, FileField).set(path, content_type)
end end
@ -229,13 +245,13 @@ module Webrat
def click_area(area_name) def click_area(area_name)
find_area(area_name).click find_area(area_name).click
end end
webrat_deprecate :clicks_area, :click_area webrat_deprecate :clicks_area, :click_area
# Issues a request for the URL pointed to by a link on the current page, # Issues a request for the URL pointed to by a link on the current page,
# follows any redirects, and verifies the final page load was successful. # follows any redirects, and verifies the final page load was successful.
# #
# click_link has very basic support for detecting Rails-generated # click_link has very basic support for detecting Rails-generated
# JavaScript onclick handlers for PUT, POST and DELETE links, as well as # JavaScript onclick handlers for PUT, POST and DELETE links, as well as
# CSRF authenticity tokens if they are present. # CSRF authenticity tokens if they are present.
# #
@ -243,15 +259,15 @@ module Webrat
# #
# Passing a :method in the options hash overrides the HTTP method used # Passing a :method in the options hash overrides the HTTP method used
# for making the link request # for making the link request
# #
# It will try to find links by (in order of precedence): # It will try to find links by (in order of precedence):
# innerHTML, with simple &nbsp; handling # innerHTML, with simple &nbsp; handling
# title # title
# id # id
# #
# innerHTML and title are matchable by text subtring or Regexp # innerHTML and title are matchable by text subtring or Regexp
# id is matchable by full text equality or Regexp # id is matchable by full text equality or Regexp
# #
# Example: # Example:
# click_link "Sign up" # click_link "Sign up"
# click_link "Sign up", :javascript => false # click_link "Sign up", :javascript => false
@ -261,7 +277,7 @@ module Webrat
end end
webrat_deprecate :clicks_link, :click_link webrat_deprecate :clicks_link, :click_link
# Verifies that a submit button exists for the form, then submits the form, follows # Verifies that a submit button exists for the form, then submits the form, follows
# any redirects, and verifies the final page was successful. # any redirects, and verifies the final page was successful.
# #
@ -288,38 +304,38 @@ module Webrat
def submit_form(id) def submit_form(id)
FormLocator.new(@session, dom, id).locate.submit FormLocator.new(@session, dom, id).locate.submit
end end
def dom # :nodoc: def dom # :nodoc:
return @dom if @dom return @dom if @dom
if @selector if @selector
@dom = scoped_dom @dom = scoped_dom
else else
@dom = page_dom @dom = page_dom
end end
return @dom return @dom
end end
protected protected
def page_dom #:nodoc: def page_dom #:nodoc:
return @response.dom if @response.respond_to?(:dom) return @response.dom if @response.respond_to?(:dom)
if @session.xml_content_type? if @session.xml_content_type?
dom = Webrat::XML.xml_document(@response_body) dom = Webrat::XML.xml_document(@response_body)
else else
dom = Webrat::XML.html_document(@response_body) dom = Webrat::XML.html_document(@response_body)
end end
Webrat.define_dom_method(@response, dom) Webrat::XML.define_dom_method(@response, dom)
return dom return dom
end end
def scoped_dom def scoped_dom
Webrat::XML.css_at(@scope.dom, @selector) @scope.dom.css(@selector).first
end end
def locate_field(field_locator, *field_types) #:nodoc: def locate_field(field_locator, *field_types) #:nodoc:
if field_locator.is_a?(Field) if field_locator.is_a?(Field)
field_locator field_locator
@ -327,10 +343,10 @@ module Webrat
field(field_locator, *field_types) field(field_locator, *field_types)
end end
end end
def locate_id_prefix(options, &location_strategy) #:nodoc: def locate_id_prefix(options, &location_strategy) #:nodoc:
return options[:id_prefix] if options[:id_prefix] return options[:id_prefix] if options[:id_prefix]
if options[:from] if options[:from]
if (label = LabelLocator.new(@session, dom, options[:from]).locate) if (label = LabelLocator.new(@session, dom, options[:from]).locate)
label.for_id label.for_id
@ -341,10 +357,10 @@ module Webrat
yield yield
end end
end end
def forms #:nodoc: def forms #:nodoc:
@forms ||= Form.load_all(@session, dom) @forms ||= Form.load_all(@session, dom)
end end
end end
end end

View File

@ -9,20 +9,34 @@ module Webrat
class PageLoadError < WebratError class PageLoadError < WebratError
end end
class InfiniteRedirectError < WebratError
end
def self.session_class def self.session_class
if Webrat.configuration.mode == :selenium
SeleniumSession
else
Session
end
end
def self.adapter_class
case Webrat.configuration.mode case Webrat.configuration.mode
when :rails when :rails
RailsSession RailsAdapter
when :merb when :merb
MerbSession MerbAdapter
when :selenium
SeleniumSession
when :rack when :rack
RackSession RackAdapter
when :rack_test
warn("The :rack_test mode is deprecated. Please use :rack instead")
require "webrat/rack"
RackAdapter
when :sinatra when :sinatra
SinatraSession warn("The :sinatra mode is deprecated. Please use :rack instead")
SinatraAdapter
when :mechanize when :mechanize
MechanizeSession MechanizeAdapter
else else
raise WebratError.new(<<-STR) raise WebratError.new(<<-STR)
Unknown Webrat mode: #{Webrat.configuration.mode.inspect} Unknown Webrat mode: #{Webrat.configuration.mode.inspect}
@ -45,16 +59,23 @@ For example:
extend Forwardable extend Forwardable
include Logging include Logging
include SaveAndOpenPage include SaveAndOpenPage
attr_accessor :adapter
attr_reader :current_url attr_reader :current_url
attr_reader :elements attr_reader :elements
def initialize(context = nil) #:nodoc: def_delegators :@adapter, :response, :response_code, :response_body, :response_headers,
:response_body=, :response_code=,
:get, :post, :put, :delete
def initialize(adapter = nil)
@adapter = adapter
@http_method = :get @http_method = :get
@data = {} @data = {}
@default_headers = {} @default_headers = {}
@custom_headers = {} @custom_headers = {}
@context = context @current_url = nil
reset reset
end end
@ -64,6 +85,7 @@ For example:
# For backwards compatibility -- removing in 1.0 # For backwards compatibility -- removing in 1.0
def current_page #:nodoc: def current_page #:nodoc:
warn "current_page is deprecated and will be going away in the next release. Use current_url instead."
page = OpenStruct.new page = OpenStruct.new
page.url = @current_url page.url = @current_url
page.http_method = @http_method page.http_method = @http_method
@ -71,10 +93,6 @@ For example:
page page
end end
def doc_root #:nodoc:
nil
end
def header(key, value) def header(key, value)
@custom_headers[key] = value @custom_headers[key] = value
end end
@ -84,7 +102,7 @@ For example:
end end
def basic_auth(user, pass) def basic_auth(user, pass)
encoded_login = ["#{user}:#{pass}"].pack("m*") encoded_login = ["#{user}:#{pass}"].pack("m*").gsub(/\n/, '')
header('HTTP_AUTHORIZATION', "Basic #{encoded_login}") header('HTTP_AUTHORIZATION', "Basic #{encoded_login}")
end end
@ -97,11 +115,8 @@ For example:
h['HTTP_REFERER'] = @current_url if @current_url h['HTTP_REFERER'] = @current_url if @current_url
debug_log "REQUESTING PAGE: #{http_method.to_s.upcase} #{url} with #{data.inspect} and HTTP headers #{h.inspect}" debug_log "REQUESTING PAGE: #{http_method.to_s.upcase} #{url} with #{data.inspect} and HTTP headers #{h.inspect}"
if h.empty?
send "#{http_method}", url, data || {} process_request(http_method, url, data, h)
else
send "#{http_method}", url, data || {}, h
end
save_and_open_page if exception_caught? && Webrat.configuration.open_error_files? save_and_open_page if exception_caught? && Webrat.configuration.open_error_files?
raise PageLoadError.new("Page load was not successful (Code: #{response_code.inspect}):\n#{formatted_error}") unless success_code? raise PageLoadError.new("Page load was not successful (Code: #{response_code.inspect}):\n#{formatted_error}") unless success_code?
@ -112,21 +127,49 @@ For example:
@http_method = http_method @http_method = http_method
@data = data @data = data
request_page(response_location, :get, data) if internal_redirect? if internal_redirect?
check_for_infinite_redirects
request_page(response_location, :get, {})
end
return response return response
end end
def check_for_infinite_redirects
if current_url == response_location
@_identical_redirect_count ||= 0
@_identical_redirect_count += 1
end
if infinite_redirect_limit_exceeded?
raise InfiniteRedirectError.new("#{Webrat.configuration.infinite_redirect_limit} redirects to the same URL (#{current_url.inspect})")
end
end
def infinite_redirect_limit_exceeded?
Webrat.configuration.infinite_redirect_limit &&
(@_identical_redirect_count || 0) > Webrat.configuration.infinite_redirect_limit
end
def success_code? #:nodoc: def success_code? #:nodoc:
(200..499).include?(response_code) (200..499).include?(response_code)
end end
def redirect? #:nodoc: def redirect? #:nodoc:
response_code / 100 == 3 [301, 302, 303, 307].include?(response_code)
end end
def internal_redirect? #:nodoc: def internal_redirect?
redirect? && current_host == response_location_host return false unless redirect?
#should keep internal_redirects if the subdomain changes
current_host_domain = current_host.split('.')[-2..-1].join('.') rescue current_host
response_location_host_domain = response_location_host.split('.')[-2..-1].join('.') rescue response_location_host
current_host_domain == response_location_host_domain
end
#easy helper to pull out where we were redirected to
def redirected_to
redirect? ? response_location : nil
end end
def exception_caught? #:nodoc: def exception_caught? #:nodoc:
@ -214,6 +257,7 @@ For example:
def_delegators :current_scope, :uncheck, :unchecks def_delegators :current_scope, :uncheck, :unchecks
def_delegators :current_scope, :choose, :chooses def_delegators :current_scope, :choose, :chooses
def_delegators :current_scope, :select, :selects def_delegators :current_scope, :select, :selects
def_delegators :current_scope, :unselect, :unselects
def_delegators :current_scope, :select_datetime, :selects_datetime def_delegators :current_scope, :select_datetime, :selects_datetime
def_delegators :current_scope, :select_date, :selects_date def_delegators :current_scope, :select_date, :selects_date
def_delegators :current_scope, :select_time, :selects_time def_delegators :current_scope, :select_time, :selects_time
@ -225,15 +269,24 @@ For example:
def_delegators :current_scope, :field_by_xpath def_delegators :current_scope, :field_by_xpath
def_delegators :current_scope, :field_with_id def_delegators :current_scope, :field_with_id
def_delegators :current_scope, :select_option def_delegators :current_scope, :select_option
def_delegators :current_scope, :field_named
private private
def process_request(http_method, url, data, headers)
if headers.empty?
send "#{http_method}", url, data || {}
else
send "#{http_method}", url, data || {}, headers
end
end
def response_location def response_location
response.headers["Location"] response_headers['Location']
end end
def current_host def current_host
URI.parse(current_url).host || "www.example.com" URI.parse(current_url).host || @custom_headers["Host"] || "www.example.com"
end end
def response_location_host def response_location_host
@ -245,6 +298,6 @@ For example:
@_scopes = nil @_scopes = nil
@_page_scope = nil @_page_scope = nil
end end
end end
end end

View File

@ -1,115 +1,72 @@
require "webrat/core/xml/nokogiri" require "webrat/core_extensions/meta_class"
require "webrat/core/xml/hpricot"
require "webrat/core/xml/rexml"
module Webrat #:nodoc: module Webrat #:nodoc:
module XML #:nodoc: module XML #:nodoc:
def self.document(stringlike) #:nodoc: def self.document(stringlike) #:nodoc:
if Webrat.configuration.parse_with_nokogiri? return stringlike.dom if stringlike.respond_to?(:dom)
Webrat.nokogiri_document(stringlike)
if Nokogiri::HTML::Document === stringlike
stringlike
elsif Nokogiri::XML::NodeSet === stringlike
stringlike
elsif stringlike.respond_to?(:body)
Nokogiri::HTML(stringlike.body.to_s)
else else
Webrat.rexml_document(Webrat.hpricot_document(stringlike).to_html) Nokogiri::HTML(stringlike.to_s)
end
end
def self.html_document(stringlike) #:nodoc:
if Webrat.configuration.parse_with_nokogiri?
Webrat.html_nokogiri_document(stringlike)
else
Webrat.rexml_document(Webrat.hpricot_document(stringlike).to_html)
end
end
def self.xml_document(stringlike) #:nodoc:
if Webrat.configuration.parse_with_nokogiri?
Webrat.xml_nokogiri_document(stringlike)
else
Webrat.rexml_document(Webrat.hpricot_document(stringlike).to_html)
end end
end end
def self.to_html(element) def self.html_document(stringlike) #:nodoc:
if Webrat.configuration.parse_with_nokogiri? return stringlike.dom if stringlike.respond_to?(:dom)
element.to_html
if Nokogiri::HTML::Document === stringlike
stringlike
elsif Nokogiri::XML::NodeSet === stringlike
stringlike
elsif stringlike.respond_to?(:body)
Nokogiri::HTML(stringlike.body.to_s)
else else
element.to_s Nokogiri::HTML(stringlike.to_s)
end end
end end
def self.inner_html(element) def self.xml_document(stringlike) #:nodoc:
if Webrat.configuration.parse_with_nokogiri? return stringlike.dom if stringlike.respond_to?(:dom)
element.inner_html
if Nokogiri::HTML::Document === stringlike
stringlike
elsif Nokogiri::XML::NodeSet === stringlike
stringlike
elsif stringlike.respond_to?(:body)
Nokogiri::XML(stringlike.body.to_s)
else else
element.text Nokogiri::XML(stringlike.to_s)
end end
end end
def self.all_inner_text(element) def self.define_dom_method(object, dom) #:nodoc:
if Webrat.configuration.parse_with_nokogiri? object.meta_class.send(:define_method, :dom) do
element.inner_text dom
else
Hpricot(element.to_s).children.first.inner_text
end end
end end
def self.inner_text(element)
if Webrat.configuration.parse_with_nokogiri?
element.inner_text
else
if defined?(Hpricot::Doc) && element.is_a?(Hpricot::Doc)
element.inner_text
else
element.text
end
end
end
def self.xpath_to(element)
if Webrat.configuration.parse_with_nokogiri?
element.path
else
element.xpath
end
end
def self.attribute(element, attribute_name)
return element[attribute_name] if element.is_a?(Hash)
if Webrat.configuration.parse_with_nokogiri?
element[attribute_name]
else
element.attributes[attribute_name]
end
end
def self.xpath_at(*args)
xpath_search(*args).first
end
def self.css_at(*args)
css_search(*args).first
end
def self.xpath_search(element, *searches)
searches.flatten.map do |search|
if Webrat.configuration.parse_with_nokogiri?
element.xpath(search)
else
REXML::XPath.match(element, search)
end
end.flatten.compact
end
def self.css_search(element, *searches) #:nodoc:
xpath_search(element, css_to_xpath(*searches))
end
def self.css_to_xpath(*selectors)
selectors.map do |rule|
Nokogiri::CSS.xpath_for(rule, :prefix => ".//")
end.flatten.uniq
end
end end
end end
module Nokogiri #:nodoc:
module CSS #:nodoc:
class XPathVisitor #:nodoc:
def visit_pseudo_class_text(node) #:nodoc:
"@type='text'"
end
def visit_pseudo_class_password(node) #:nodoc:
"@type='password'"
end
end
end
end

View File

@ -1,19 +0,0 @@
module Webrat
def self.hpricot_document(stringlike)
return stringlike.dom if stringlike.respond_to?(:dom)
if Hpricot::Doc === stringlike
stringlike
elsif Hpricot::Elements === stringlike
stringlike
elsif StringIO === stringlike
Hpricot(stringlike.string)
elsif stringlike.respond_to?(:body)
Hpricot(stringlike.body.to_s)
else
Hpricot(stringlike.to_s)
end
end
end

View File

@ -1,76 +0,0 @@
require "webrat/core_extensions/meta_class"
module Webrat
def self.nokogiri_document(stringlike) #:nodoc:
return stringlike.dom if stringlike.respond_to?(:dom)
if Nokogiri::HTML::Document === stringlike
stringlike
elsif Nokogiri::XML::NodeSet === stringlike
stringlike
elsif StringIO === stringlike
Nokogiri::HTML(stringlike.string)
elsif stringlike.respond_to?(:body)
Nokogiri::HTML(stringlike.body.to_s)
else
Nokogiri::HTML(stringlike.to_s)
end
end
def self.html_nokogiri_document(stringlike) #:nodoc:
return stringlike.dom if stringlike.respond_to?(:dom)
if Nokogiri::HTML::Document === stringlike
stringlike
elsif Nokogiri::XML::NodeSet === stringlike
stringlike
elsif StringIO === stringlike
Nokogiri::HTML(stringlike.string)
elsif stringlike.respond_to?(:body)
Nokogiri::HTML(stringlike.body.to_s)
else
Nokogiri::HTML(stringlike.to_s)
end
end
def self.xml_nokogiri_document(stringlike) #:nodoc:
return stringlike.dom if stringlike.respond_to?(:dom)
if Nokogiri::HTML::Document === stringlike
stringlike
elsif Nokogiri::XML::NodeSet === stringlike
stringlike
elsif StringIO === stringlike
Nokogiri::XML(stringlike.string)
elsif stringlike.respond_to?(:body)
Nokogiri::XML(stringlike.body.to_s)
else
Nokogiri::XML(stringlike.to_s)
end
end
def self.define_dom_method(object, dom) #:nodoc:
object.meta_class.send(:define_method, :dom) do
dom
end
end
end
module Nokogiri #:nodoc:
module CSS #:nodoc:
class XPathVisitor #:nodoc:
def visit_pseudo_class_text(node) #:nodoc:
"@type='text'"
end
def visit_pseudo_class_password(node) #:nodoc:
"@type='password'"
end
end
end
end

View File

@ -1,24 +0,0 @@
module Webrat
def self.rexml_document(stringlike)
stringlike = stringlike.body.to_s if stringlike.respond_to?(:body)
case stringlike
when REXML::Document
stringlike.root
when REXML::Node, Array
stringlike
else
begin
REXML::Document.new(stringlike.to_s).root
rescue REXML::ParseException => e
if e.message.include?("second root element")
REXML::Document.new("<fake-root-element>#{stringlike}</fake-root-element>").root
else
raise e
end
end
end
end
end

View File

@ -12,7 +12,7 @@ class Object #:nodoc:
def blank? def blank?
respond_to?(:empty?) ? empty? : !self respond_to?(:empty?) ? empty? : !self
end end
# An object is present if it's not blank. # An object is present if it's not blank.
def present? def present?
!blank? !blank?

View File

@ -5,4 +5,4 @@ class Module #:nodoc:
__send__(new_method_name, *args) __send__(new_method_name, *args)
end end
end end
end end

View File

@ -1,12 +1,12 @@
class Array #:nodoc: class Array #:nodoc:
def detect_mapped def detect_mapped
each do |element| each do |element|
result = yield element result = yield element
return result if result return result if result
end end
return nil return nil
end end
end end

View File

@ -1,131 +0,0 @@
# This class has dubious semantics and we only have it so that
# people can write params[:key] instead of params['key']
# and they get the same value for both keys.
class HashWithIndifferentAccess < Hash #:nodoc:
def initialize(constructor = {})
if constructor.is_a?(Hash)
super()
update(constructor)
else
super(constructor)
end
end
def default(key = nil)
if key.is_a?(Symbol) && include?(key = key.to_s)
self[key]
else
super
end
end
alias_method :regular_writer, :[]= unless method_defined?(:regular_writer)
alias_method :regular_update, :update unless method_defined?(:regular_update)
#
# Assigns a new value to the hash.
#
# Example:
#
# hash = HashWithIndifferentAccess.new
# hash[:key] = "value"
#
def []=(key, value)
regular_writer(convert_key(key), convert_value(value))
end
#
# Updates the instantized hash with values from the second.
#
# Example:
#
# >> hash_1 = HashWithIndifferentAccess.new
# => {}
#
# >> hash_1[:key] = "value"
# => "value"
#
# >> hash_2 = HashWithIndifferentAccess.new
# => {}
#
# >> hash_2[:key] = "New Value!"
# => "New Value!"
#
# >> hash_1.update(hash_2)
# => {"key"=>"New Value!"}
#
def update(other_hash)
other_hash.each_pair { |key, value| regular_writer(convert_key(key), convert_value(value)) }
self
end
alias_method :merge!, :update
# Checks the hash for a key matching the argument passed in
def key?(key)
super(convert_key(key))
end
alias_method :include?, :key?
alias_method :has_key?, :key?
alias_method :member?, :key?
# Fetches the value for the specified key, same as doing hash[key]
def fetch(key, *extras)
super(convert_key(key), *extras)
end
# Returns an array of the values at the specified indicies.
def values_at(*indices)
indices.collect {|key| self[convert_key(key)]}
end
# Returns an exact copy of the hash.
def dup
HashWithIndifferentAccess.new(self)
end
# Merges the instantized and the specified hashes together, giving precedence to the values from the second hash
# Does not overwrite the existing hash.
def merge(hash)
self.dup.update(hash)
end
# Removes a specified key from the hash.
def delete(key)
super(convert_key(key))
end
def stringify_keys!; self end
def symbolize_keys!; self end
def to_options!; self end
# Convert to a Hash with String keys.
def to_hash
Hash.new(default).merge(self)
end
protected
def convert_key(key)
key.kind_of?(Symbol) ? key.to_s : key
end
def convert_value(value)
case value
when Hash
value.with_indifferent_access
when Array
value.collect { |e| e.is_a?(Hash) ? e.with_indifferent_access : e }
else
value
end
end
end
class Hash #:nodoc:
def with_indifferent_access
hash = HashWithIndifferentAccess.new(self)
hash.default = self.default
hash
end
end

View File

@ -3,4 +3,4 @@ class ::Object #:nodoc:
class << self; self end class << self; self end
end end
end end

View File

@ -1,5 +1,5 @@
class NilClass #:nodoc: class NilClass #:nodoc:
def to_param def to_query_string
nil nil
end end
end end

View File

@ -0,0 +1,27 @@
class TCPSocket
def self.wait_for_service_with_timeout(options)
start_time = Time.now
until listening_service?(options)
verbose_wait
if options[:timeout] && (Time.now > start_time + options[:timeout])
raise SocketError.new("Socket did not open within #{options[:timeout]} seconds")
end
end
end
def self.wait_for_service_termination_with_timeout(options)
start_time = Time.now
while listening_service?(options)
verbose_wait
if options[:timeout] && (Time.now > start_time + options[:timeout])
raise SocketError.new("Socket did not terminate within #{options[:timeout]} seconds")
end
end
end
end

View File

@ -0,0 +1,10 @@
module Merb #:nodoc:
module Test #:nodoc:
module RequestHelper #:nodoc:
def request(uri, env = {})
@_webrat_session ||= Webrat::MerbAdapter.new
@_webrat_session.response = @_webrat_session.request(uri, env)
end
end
end
end

View File

@ -0,0 +1,25 @@
require "action_controller"
require "action_controller/integration"
module ActionController #:nodoc:
IntegrationTest.class_eval do
include Webrat::Methods
include Webrat::Matchers
# The Rails version of within supports passing in a model and Webrat
# will apply a scope based on Rails' dom_id for that model.
#
# Example:
# within User.last do
# click_link "Delete"
# end
def within(selector_or_object, &block)
if selector_or_object.is_a?(String)
super
else
super('#' + RecordIdentifier.dom_id(selector_or_object), &block)
end
end
end
end

View File

@ -0,0 +1,11 @@
# Supports using the matchers in controller, helper, and view specs if you're
# using rspec-rails. Just add a require statement to spec/spec_helper.rb or env.rb:
#
# require 'webrat/integrations/rspec-rails'
#
require "nokogiri"
require "webrat/core/matchers"
Spec::Runner.configure do |config|
config.include(Webrat::Matchers, :type => [:controller, :helper, :view])
end

View File

@ -0,0 +1,11 @@
require "webrat/selenium"
if defined?(ActionController::IntegrationTest)
module ActionController #:nodoc:
IntegrationTest.class_eval do
include Webrat::Methods
include Webrat::Selenium::Methods
include Webrat::Selenium::Matchers
end
end
end

View File

@ -6,4 +6,4 @@ require "webrat"
Webrat.configure do |config| Webrat.configure do |config|
config.mode = :merb config.mode = :merb
end end

View File

@ -1,65 +0,0 @@
require "webrat"
require "cgi"
gem "extlib"
require "extlib"
require "merb-core"
HashWithIndifferentAccess = Mash
module Webrat
class MerbSession < Session #:nodoc:
include Merb::Test::MakeRequest
attr_accessor :response
def get(url, data, headers = nil)
do_request(url, data, headers, "GET")
end
def post(url, data, headers = nil)
do_request(url, data, headers, "POST")
end
def put(url, data, headers = nil)
do_request(url, data, headers, "PUT")
end
def delete(url, data, headers = nil)
do_request(url, data, headers, "DELETE")
end
def response_body
@response.body.to_s
end
def response_code
@response.status
end
def do_request(url, data, headers, method)
@response = request(url,
:params => (data && data.any?) ? data : nil,
:headers => headers,
:method => method)
end
end
end
module Merb #:nodoc:
module Test #:nodoc:
module RequestHelper #:nodoc:
def request(uri, env = {})
@_webrat_session ||= Webrat::MerbSession.new
@_webrat_session.response = @_webrat_session.request(uri, env)
end
end
end
end
class Merb::Test::RspecStory #:nodoc:
def browser
@browser ||= Webrat::MerbSession.new
end
end

View File

@ -1,24 +0,0 @@
require 'webrat'
class CGIMethods #:nodoc:
def self.parse_query_parameters(params)
hash = {}
params.split('&').each do |p|
pair = p.split('=')
hash[pair[0]] = pair[1]
end
hash
end
end
module Webrat
class RackSession < Session #:nodoc:
def response_body
@response.body
end
def response_code
@response.status
end
end
end

View File

@ -1,13 +1,2 @@
# Supports using the matchers in controller, helper, and view specs if you're warn("Requiring 'webrat/rspec-rails' is deprecated. Please require 'webrat/integrations/rspec-rails' instead")
# using rspec-rails. Just add a require statement to spec/spec_helper.rb or env.rb: require "webrat/integrations/rspec-rails"
#
# require 'webrat/rspec-rails'
#
require "webrat/core/matchers"
Spec::Runner.configure do |config|
# rspec should support :type => [:controller, :helper, :view] - but until it does ...
config.include(Webrat::Matchers, :type => :controller)
config.include(Webrat::Matchers, :type => :helper)
config.include(Webrat::Matchers, :type => :view)
end

View File

@ -1,41 +1,11 @@
require "webrat" require "webrat"
gem "selenium-client", ">=1.2.9"
require "selenium/client" require "selenium/client"
require "webrat/selenium/silence_stream"
require "webrat/selenium/selenium_session" require "webrat/selenium/selenium_session"
require "webrat/selenium/matchers" require "webrat/selenium/matchers"
require "webrat/core_extensions/tcp_socket"
module Webrat module Webrat
def self.with_selenium_server #:nodoc:
start_selenium_server
yield
stop_selenium_server
end
def self.start_selenium_server #:nodoc:
unless Webrat.configuration.selenium_server_address
remote_control = ::Selenium::RemoteControl::RemoteControl.new("0.0.0.0", Webrat.configuration.selenium_server_port, 5)
remote_control.jar_file = File.expand_path(__FILE__ + "../../../../vendor/selenium-server.jar")
remote_control.start :background => true
end
TCPSocket.wait_for_service :host => (Webrat.configuration.selenium_server_address || "0.0.0.0"), :port => Webrat.configuration.selenium_server_port
end
def self.stop_selenium_server #:nodoc:
::Selenium::RemoteControl::RemoteControl.new("0.0.0.0", Webrat.configuration.selenium_server_port, 5).stop unless Webrat.configuration.selenium_server_address
end
def self.start_app_server #:nodoc:
pid_file = File.expand_path(RAILS_ROOT + "/tmp/pids/mongrel_selenium.pid")
system("mongrel_rails start -d --chdir=#{RAILS_ROOT} --port=#{Webrat.configuration.application_port} --environment=#{Webrat.configuration.application_environment} --pid #{pid_file} &")
TCPSocket.wait_for_service :host => Webrat.configuration.application_address, :port => Webrat.configuration.application_port.to_i
end
def self.stop_app_server #:nodoc:
pid_file = File.expand_path(RAILS_ROOT + "/tmp/pids/mongrel_selenium.pid")
system "mongrel_rails stop -c #{RAILS_ROOT} --pid #{pid_file}"
end
# To use Webrat's Selenium support, you'll need the selenium-client gem installed. # To use Webrat's Selenium support, you'll need the selenium-client gem installed.
# Activate it with (for example, in your <tt>env.rb</tt>): # Activate it with (for example, in your <tt>env.rb</tt>):
# #
@ -55,11 +25,25 @@ module Webrat
# selenium.dragdrop("id=photo_123", "+350, 0") # selenium.dragdrop("id=photo_123", "+350, 0")
# end # end
# #
# == Auto-starting of the mongrel and java server # == Choosing the underlying framework to test
#
# Webrat assumes you're using rails by default but it can also work with sinatra
# and merb. To take advantage of this you can use the configuration block to
# set the application_framework variable.
# require "webrat"
#
# Webrat.configure do |config|
# config.mode = :selenium
# config.application_port = 4567
# config.application_framework = :sinatra # could also be :merb
# end
#
# == Auto-starting of the appserver and java server
# #
# Webrat will automatically start the Selenium Java server process and an instance # Webrat will automatically start the Selenium Java server process and an instance
# of Mongrel when a test is run. The Mongrel will run in the "selenium" environment # of Mongrel when a test is run. The Mongrel will run in the "selenium" environment
# instead of "test", so ensure you've got that defined, and will run on port 3001. # instead of "test", so ensure you've got that defined, and will run on port
# Webrat.configuration.application_port.
# #
# == Waiting # == Waiting
# #
@ -84,11 +68,3 @@ module Webrat
end end
end end
end end
module ActionController #:nodoc:
IntegrationTest.class_eval do
include Webrat::Methods
include Webrat::Selenium::Methods
include Webrat::Selenium::Matchers
end
end

View File

@ -0,0 +1,40 @@
module Webrat
module Selenium
class ApplicationServerFactory
def self.app_server_instance
case Webrat.configuration.application_framework
when :sinatra
require "webrat/selenium/application_servers/sinatra"
return Webrat::Selenium::ApplicationServers::Sinatra.new
when :merb
require "webrat/selenium/application_servers/merb"
return Webrat::Selenium::ApplicationServers::Merb.new
when :rails
require "webrat/selenium/application_servers/rails"
return Webrat::Selenium::ApplicationServers::Rails.new
when :external
require "webrat/selenium/application_servers/external"
return Webrat::Selenium::ApplicationServers::External.new
else
raise WebratError.new(<<-STR)
Unknown Webrat application_framework: #{Webrat.configuration.application_framework.inspect}
Please ensure you have a Webrat configuration block that specifies an application_framework
in your test_helper.rb, spec_helper.rb, or env.rb (for Cucumber).
For example:
Webrat.configure do |config|
# ...
config.application_framework = :rails
end
STR
end
end
end
end
end

View File

@ -0,0 +1,5 @@
require "webrat/selenium/application_servers/base"
require "webrat/selenium/application_servers/sinatra"
require "webrat/selenium/application_servers/merb"
require "webrat/selenium/application_servers/rails"
require "webrat/selenium/application_servers/external"

View File

@ -0,0 +1,46 @@
require "webrat/selenium/silence_stream"
module Webrat
module Selenium
module ApplicationServers
class Base
include Webrat::Selenium::SilenceStream
def boot
start
wait
stop_at_exit
end
def stop_at_exit
at_exit do
stop
end
end
def wait
$stderr.print "==> Waiting for #{Webrat.configuration.application_framework} application server on port #{Webrat.configuration.application_port}... "
wait_for_socket
$stderr.print "Ready!\n"
end
def wait_for_socket
silence_stream(STDOUT) do
TCPSocket.wait_for_service_with_timeout \
:host => "0.0.0.0",
:port => Webrat.configuration.application_port.to_i,
:timeout => 30 # seconds
end
rescue SocketError
fail
end
def prepare_pid_file(file_path, pid_file_name)
FileUtils.mkdir_p File.expand_path(file_path)
File.expand_path("#{file_path}/#{pid_file_name}")
end
end
end
end
end

View File

@ -0,0 +1,26 @@
require "webrat/selenium/application_servers/base"
module Webrat
module Selenium
module ApplicationServers
class External < Webrat::Selenium::ApplicationServers::Base
def start
warn "Webrat Ignoring Start Of Application Server Due to External Mode"
end
def stop
end
def fail
end
def pid_file
end
def wait
end
end
end
end
end

View File

@ -0,0 +1,50 @@
require "webrat/selenium/application_servers/base"
module Webrat
module Selenium
module ApplicationServers
class Merb < Webrat::Selenium::ApplicationServers::Base
def start
system start_command
end
def stop
silence_stream(STDOUT) do
pid = File.read(pid_file)
system("kill -9 #{pid}")
FileUtils.rm_f pid_file
end
end
def fail
$stderr.puts
$stderr.puts
$stderr.puts "==> Failed to boot the Merb application server... exiting!"
$stderr.puts
$stderr.puts "Verify you can start a Merb server on port #{Webrat.configuration.application_port} with the following command:"
$stderr.puts
$stderr.puts " #{start_command}"
exit
end
def pid_file
"log/merb.#{Webrat.configuration.application_port}.pid"
end
def start_command
"#{merb_command} -d -p #{Webrat.configuration.application_port} -e #{Webrat.configuration.application_environment}"
end
def merb_command
if File.exist?('bin/merb')
merb_cmd = 'bin/merb'
else
merb_cmd = 'merb'
end
end
end
end
end
end

View File

@ -0,0 +1,44 @@
require "webrat/selenium/application_servers/base"
module Webrat
module Selenium
module ApplicationServers
class Rails < Webrat::Selenium::ApplicationServers::Base
def start
system start_command
end
def stop
silence_stream(STDOUT) do
system stop_command
end
end
def fail
$stderr.puts
$stderr.puts
$stderr.puts "==> Failed to boot the Rails application server... exiting!"
$stderr.puts
$stderr.puts "Verify you can start a Rails server on port #{Webrat.configuration.application_port} with the following command:"
$stderr.puts
$stderr.puts " #{start_command}"
exit
end
def pid_file
prepare_pid_file("#{RAILS_ROOT}/tmp/pids", "mongrel_selenium.pid")
end
def start_command
"mongrel_rails start -d --chdir='#{RAILS_ROOT}' --port=#{Webrat.configuration.application_port} --environment=#{Webrat.configuration.application_environment} --pid #{pid_file} &"
end
def stop_command
"mongrel_rails stop -c #{RAILS_ROOT} --pid #{pid_file}"
end
end
end
end
end

View File

@ -0,0 +1,37 @@
require "webrat/selenium/application_servers/base"
module Webrat
module Selenium
module ApplicationServers
class Sinatra < Webrat::Selenium::ApplicationServers::Base
def start
fork do
File.open('rack.pid', 'w') { |fp| fp.write Process.pid }
exec 'rackup', File.expand_path(Dir.pwd + '/config.ru'), '-p', Webrat.configuration.application_port.to_s
end
end
def stop
silence_stream(STDOUT) do
pid = File.read(pid_file)
system("kill -9 #{pid}")
FileUtils.rm_f pid_file
end
end
def fail
$stderr.puts
$stderr.puts
$stderr.puts "==> Failed to boot the Sinatra application server... exiting!"
exit
end
def pid_file
prepare_pid_file(Dir.pwd, 'rack.pid')
end
end
end
end
end

View File

@ -1,12 +1,19 @@
if (locator == '*') { if (locator == '*') {
return selenium.browserbot.locationStrategies['xpath'].call(this, "//input[@type='submit']", inDocument, inWindow) return selenium.browserbot.locationStrategies['xpath'].call(this, "//input[@type='submit']", inDocument, inWindow)
} }
var buttons = inDocument.getElementsByTagName('button');
var inputs = inDocument.getElementsByTagName('input'); var inputs = inDocument.getElementsByTagName('input');
return $A(inputs).find(function(candidate){ var result = $A(inputs).concat($A(buttons)).find(function(candidate){
inputType = candidate.getAttribute('type'); var type = candidate.getAttribute('type');
if (inputType == 'submit' || inputType == 'image') { if (type == 'submit' || type == 'image' || type == 'button') {
var buttonText = $F(candidate); var matches_id = PatternMatcher.matches(locator, candidate.id);
return (PatternMatcher.matches(locator, buttonText)); var matches_value = PatternMatcher.matches(locator, candidate.value);
} var matches_html = PatternMatcher.matches(locator, candidate.innerHTML);
return false; var matches_alt = PatternMatcher.matches(locator, candidate.alt);
if (matches_id || matches_value || matches_html || matches_alt) {
return true;
}
}
return false;
}); });
return result;

View File

@ -1,16 +1,48 @@
// Credit to: http://simonwillison.net/2006/Jan/20/escape/
RegExp.escape = function(text) {
if (!arguments.callee.sRE) {
var specials = [
'/', '.', '*', '+', '?', '|',
'(', ')', '[', ']', '{', '}', '\\'
];
arguments.callee.sRE = new RegExp(
'(\\' + specials.join('|\\') + ')', 'g'
);
}
return text.replace(arguments.callee.sRE, '\\$1');
};
var allLabels = inDocument.getElementsByTagName("label"); var allLabels = inDocument.getElementsByTagName("label");
var regExp = new RegExp('^\\W*' + RegExp.escape(locator) + '(\\b|$)', 'i');
var candidateLabels = $A(allLabels).select(function(candidateLabel){ var candidateLabels = $A(allLabels).select(function(candidateLabel){
var regExp = new RegExp('^' + locator + '\\b', 'i');
var labelText = getText(candidateLabel).strip(); var labelText = getText(candidateLabel).strip();
return (labelText.search(regExp) >= 0); return (labelText.search(regExp) >= 0);
}); });
if (candidateLabels.length == 0) { if (candidateLabels.length == 0) {
return null; return null;
} }
candidateLabels = candidateLabels.sortBy(function(s) { return s.length * -1; }); //reverse length sort
//reverse length sort
candidateLabels = candidateLabels.sortBy(function(s) {
return s.length * -1;
});
var locatedLabel = candidateLabels.first(); var locatedLabel = candidateLabels.first();
var labelFor = locatedLabel.getAttribute('for'); var labelFor = null;
if ((labelFor == null) && (locatedLabel.hasChildNodes())) {
return locatedLabel.firstChild; //TODO: should find the first form field, not just any node if (locatedLabel.getAttribute('for')) {
labelFor = locatedLabel.getAttribute('for');
} else if (locatedLabel.attributes['for']) { // IE
labelFor = locatedLabel.attributes['for'].nodeValue;
} }
if ((labelFor == null) && (locatedLabel.hasChildNodes())) {
return locatedLabel.getElementsByTagName('button')[0]
|| locatedLabel.getElementsByTagName('input')[0]
|| locatedLabel.getElementsByTagName('textarea')[0]
|| locatedLabel.getElementsByTagName('select')[0];
}
return selenium.browserbot.locationStrategies['id'].call(this, labelFor, inDocument, inWindow); return selenium.browserbot.locationStrategies['id'].call(this, labelFor, inDocument, inWindow);

View File

@ -1,4 +1,5 @@
var locationStrategies = selenium.browserbot.locationStrategies; var locationStrategies = selenium.browserbot.locationStrategies;
return locationStrategies['id'].call(this, locator, inDocument, inWindow) return locationStrategies['id'].call(this, locator, inDocument, inWindow)
|| locationStrategies['name'].call(this, locator, inDocument, inWindow) || locationStrategies['name'].call(this, locator, inDocument, inWindow)
|| locationStrategies['label'].call(this, locator, inDocument, inWindow) || locationStrategies['label'].call(this, locator, inDocument, inWindow)

View File

@ -1,9 +1,32 @@
var links = inDocument.getElementsByTagName('a'); var links = inDocument.getElementsByTagName('a');
var candidateLinks = $A(links).select(function(candidateLink) { var candidateLinks = $A(links).select(function(candidateLink) {
return PatternMatcher.matches(locator, getText(candidateLink)); var textMatched = false;
var titleMatched = false;
var idMatched = false;
if (getText(candidateLink).toLowerCase().indexOf(locator.toLowerCase()) != -1) {
textMatched = true;
}
if (candidateLink.title.toLowerCase().indexOf(locator.toLowerCase()) != -1) {
titleMatched = true;
}
if (candidateLink.id.toLowerCase().indexOf(locator.toLowerCase()) != -1) {
idMatched = true;
}
return textMatched || idMatched || titleMatched;
}); });
if (candidateLinks.length == 0) { if (candidateLinks.length == 0) {
return null; return null;
} }
candidateLinks = candidateLinks.sortBy(function(s) { return s.length * -1; }); //reverse length sort
//reverse length sort
candidateLinks = candidateLinks.sortBy(function(s) {
return s.length * -1;
});
return candidateLinks.first(); return candidateLinks.first();

View File

@ -1,146 +1,4 @@
module Webrat require "webrat/selenium/matchers/have_xpath"
module Selenium require "webrat/selenium/matchers/have_selector"
module Matchers # require "webrat/selenium/matchers/have_tag"
require "webrat/selenium/matchers/have_content"
class HaveXpath
def initialize(expected)
@expected = expected
end
def matches?(response)
response.session.wait_for do
response.selenium.is_element_present("xpath=#{@expected}")
end
end
# ==== Returns
# String:: The failure message.
def failure_message
"expected following text to match xpath #{@expected}:\n#{@document}"
end
# ==== Returns
# String:: The failure message to be displayed in negative matches.
def negative_failure_message
"expected following text to not match xpath #{@expected}:\n#{@document}"
end
end
def have_xpath(xpath)
HaveXpath.new(xpath)
end
def assert_have_xpath(expected)
hs = HaveXpath.new(expected)
assert hs.matches?(response), hs.failure_message
end
def assert_have_no_xpath(expected)
hs = HaveXpath.new(expected)
assert !hs.matches?(response), hs.negative_failure_message
end
class HaveSelector
def initialize(expected)
@expected = expected
end
def matches?(response)
response.session.wait_for do
response.selenium.is_element_present("css=#{@expected}")
end
end
# ==== Returns
# String:: The failure message.
def failure_message
"expected following text to match selector #{@expected}:\n#{@document}"
end
# ==== Returns
# String:: The failure message to be displayed in negative matches.
def negative_failure_message
"expected following text to not match selector #{@expected}:\n#{@document}"
end
end
def have_selector(content)
HaveSelector.new(content)
end
# Asserts that the body of the response contains
# the supplied selector
def assert_have_selector(expected)
hs = HaveSelector.new(expected)
assert hs.matches?(response), hs.failure_message
end
# Asserts that the body of the response
# does not contain the supplied string or regepx
def assert_have_no_selector(expected)
hs = HaveSelector.new(expected)
assert !hs.matches?(response), hs.negative_failure_message
end
class HasContent #:nodoc:
def initialize(content)
@content = content
end
def matches?(response)
if @content.is_a?(Regexp)
text_finder = "regexp:#{@content.source}"
else
text_finder = @content
end
response.session.wait_for do
response.selenium.is_text_present(text_finder)
end
end
# ==== Returns
# String:: The failure message.
def failure_message
"expected the following element's content to #{content_message}:\n#{@element}"
end
# ==== Returns
# String:: The failure message to be displayed in negative matches.
def negative_failure_message
"expected the following element's content to not #{content_message}:\n#{@element}"
end
def content_message
case @content
when String
"include \"#{@content}\""
when Regexp
"match #{@content.inspect}"
end
end
end
# Matches the contents of an HTML document with
# whatever string is supplied
def contain(content)
HasContent.new(content)
end
# Asserts that the body of the response contain
# the supplied string or regexp
def assert_contain(content)
hc = HasContent.new(content)
assert hc.matches?(response), hc.failure_message
end
# Asserts that the body of the response
# does not contain the supplied string or regepx
def assert_not_contain(content)
hc = HasContent.new(content)
assert !hc.matches?(response), hc.negative_failure_message
end
end
end
end

View File

@ -0,0 +1,78 @@
module Webrat
module Selenium
module Matchers
class HasContent #:nodoc:
def initialize(content)
@content = content
end
def matches?(response)
response.session.wait_for do
response.selenium.is_text_present(text_finder)
end
rescue Webrat::TimeoutError => e
@error_message = e.message
false
end
def does_not_match?(response)
response.session.wait_for do
!response.selenium.is_text_present(text_finder)
end
rescue Webrat::TimeoutError => e
@error_message = e.message
false
end
# ==== Returns
# String:: The failure message.
def failure_message
"expected the response to #{content_message}:\n#{@error_message}"
end
# ==== Returns
# String:: The failure message to be displayed in negative matches.
def negative_failure_message
"expected the response to not #{content_message}"
end
def content_message
case @content
when String
"include \"#{@content}\""
when Regexp
"match #{@content.inspect}"
end
end
def text_finder
if @content.is_a?(Regexp)
"regexp:#{@content.source}"
else
@content
end
end
end
# Matches the contents of an HTML document with
# whatever string is supplied
def contain(content)
HasContent.new(content)
end
# Asserts that the body of the response contain
# the supplied string or regexp
def assert_contain(content)
hc = HasContent.new(content)
assert hc.matches?(response), hc.failure_message
end
# Asserts that the body of the response
# does not contain the supplied string or regepx
def assert_not_contain(content)
hc = HasContent.new(content)
assert !hc.matches?(response), hc.negative_failure_message
end
end
end
end

View File

@ -0,0 +1,57 @@
module Webrat
module Selenium
module Matchers
class HaveSelector
def initialize(expected)
@expected = expected
end
def matches?(response)
response.session.wait_for do
response.selenium.is_element_present("css=#{@expected}")
end
rescue Webrat::TimeoutError
false
end
def does_not_match?(response)
response.session.wait_for do
!response.selenium.is_element_present("css=#{@expected}")
end
rescue Webrat::TimeoutError
false
end
# ==== Returns
# String:: The failure message.
def failure_message
"expected following text to match selector #{@expected}:\n#{@document}"
end
# ==== Returns
# String:: The failure message to be displayed in negative matches.
def negative_failure_message
"expected following text to not match selector #{@expected}:\n#{@document}"
end
end
def have_selector(content)
HaveSelector.new(content)
end
# Asserts that the body of the response contains
# the supplied selector
def assert_have_selector(expected)
hs = HaveSelector.new(expected)
assert hs.matches?(response), hs.failure_message
end
# Asserts that the body of the response
# does not contain the supplied string or regepx
def assert_have_no_selector(expected)
hs = HaveSelector.new(expected)
assert !hs.matches?(response), hs.negative_failure_message
end
end
end
end

View File

@ -0,0 +1,72 @@
module Webrat
module Selenium
module Matchers
class HaveTag < HaveSelector #:nodoc:
# ==== Returns
# String:: The failure message.
def failure_message
"expected following output to contain a #{tag_inspect} tag:\n#{@document}"
end
# ==== Returns
# String:: The failure message to be displayed in negative matches.
def negative_failure_message
"expected following output to omit a #{tag_inspect}:\n#{@document}"
end
def tag_inspect
options = @expected.last.dup
content = options.delete(:content)
html = "<#{@expected.first}"
options.each do |k,v|
html << " #{k}='#{v}'"
end
if content
html << ">#{content}</#{@expected.first}>"
else
html << "/>"
end
html
end
def query
options = @expected.last.dup
selector = @expected.first.to_s
selector << ":contains('#{options.delete(:content)}')" if options[:content]
options.each do |key, value|
selector << "[#{key}='#{value}']"
end
Nokogiri::CSS.parse(selector).map { |ast| ast.to_xpath }
end
end
def have_tag(name, attributes = {}, &block)
HaveTag.new([name, attributes], &block)
end
alias_method :match_tag, :have_tag
# Asserts that the body of the response contains
# the supplied tag with the associated selectors
def assert_have_tag(name, attributes = {})
ht = HaveTag.new([name, attributes])
assert ht.matches?(response), ht.failure_message
end
# Asserts that the body of the response
# does not contain the supplied string or regepx
def assert_have_no_tag(name, attributes = {})
ht = HaveTag.new([name, attributes])
assert !ht.matches?(response), ht.negative_failure_message
end
end
end
end

View File

@ -0,0 +1,53 @@
module Webrat
module Selenium
module Matchers
class HaveXpath
def initialize(expected)
@expected = expected
end
def matches?(response)
response.session.wait_for do
response.selenium.is_element_present("xpath=#{@expected}")
end
rescue Webrat::TimeoutError
false
end
def does_not_match?(response)
response.session.wait_for do
!response.selenium.is_element_present("xpath=#{@expected}")
end
rescue Webrat::TimeoutError
false
end
# ==== Returns
# String:: The failure message.
def failure_message
"expected following text to match xpath #{@expected}:\n#{@document}"
end
# ==== Returns
# String:: The failure message to be displayed in negative matches.
def negative_failure_message
"expected following text to not match xpath #{@expected}:\n#{@document}"
end
end
def have_xpath(xpath)
HaveXpath.new(xpath)
end
def assert_have_xpath(expected)
hs = HaveXpath.new(expected)
assert hs.matches?(response), hs.failure_message
end
def assert_have_no_xpath(expected)
hs = HaveXpath.new(expected)
assert !hs.matches?(response), hs.negative_failure_message
end
end
end
end

View File

@ -0,0 +1,89 @@
module Webrat
module Selenium
class SeleniumRCServer
include Webrat::Selenium::SilenceStream
def self.boot
new.boot
end
def boot
return if selenium_grid?
start
wait
stop_at_exit
end
def start
silence_stream(STDOUT) do
remote_control.start :background => true
end
end
def stop_at_exit
at_exit do
stop
end
end
def remote_control
return @remote_control if @remote_control
server_options = { :timeout => Webrat.configuration.selenium_browser_startup_timeout }
server_options[:firefox_profile] = Webrat.configuration.selenium_firefox_profile if Webrat.configuration.selenium_firefox_profile
@remote_control = ::Selenium::RemoteControl::RemoteControl.new("0.0.0.0",
Webrat.configuration.selenium_server_port,
server_options)
@remote_control.jar_file = jar_path
return @remote_control
end
def jar_path
File.expand_path(__FILE__ + "../../../../../vendor/selenium-server.jar")
end
def selenium_grid?
Webrat.configuration.selenium_server_address
end
def wait
$stderr.print "==> Waiting for Selenium RC server on port #{Webrat.configuration.selenium_server_port}... "
wait_for_socket
$stderr.print "Ready!\n"
rescue SocketError
fail
end
def wait_for_socket
silence_stream(STDOUT) do
TCPSocket.wait_for_service_with_timeout \
:host => (Webrat.configuration.selenium_server_address || "0.0.0.0"),
:port => Webrat.configuration.selenium_server_port,
:timeout => 45 # seconds
end
end
def fail
$stderr.puts
$stderr.puts
$stderr.puts "==> Failed to boot the Selenium RC server... exiting!"
exit
end
def stop
silence_stream(STDOUT) do
::Selenium::RemoteControl::RemoteControl.new("0.0.0.0",
Webrat.configuration.selenium_server_port,
:timeout => 5).stop
end
end
end
end
end

View File

@ -1,4 +1,14 @@
require "webrat/core/save_and_open_page" require "webrat/core/save_and_open_page"
require "webrat/selenium/selenium_rc_server"
require "webrat/selenium/application_server_factory"
require "webrat/selenium/application_servers/base"
begin
require "selenium"
rescue LoadError => e
e.message << " (You may need to install the selenium-rc gem)"
raise e
end
module Webrat module Webrat
class TimeoutError < WebratError class TimeoutError < WebratError
@ -20,6 +30,7 @@ module Webrat
class SeleniumSession class SeleniumSession
include Webrat::SaveAndOpenPage include Webrat::SaveAndOpenPage
include Webrat::Selenium::SilenceStream
def initialize(*args) # :nodoc: def initialize(*args) # :nodoc:
end end
@ -38,8 +49,8 @@ module Webrat
webrat_deprecate :visits, :visit webrat_deprecate :visits, :visit
def fill_in(field_identifier, options) def fill_in(field_identifier, options)
locator = "webrat=#{Regexp.escape(field_identifier)}" locator = "webrat=#{field_identifier}"
selenium.wait_for_element locator, 5 selenium.wait_for_element locator, :timeout_in_seconds => 5
selenium.type(locator, "#{options[:with]}") selenium.type(locator, "#{options[:with]}")
end end
@ -53,6 +64,10 @@ module Webrat
selenium.get_html_source selenium.get_html_source
end end
def current_url
selenium.location
end
def click_button(button_text_or_regexp = nil, options = {}) def click_button(button_text_or_regexp = nil, options = {})
if button_text_or_regexp.is_a?(Hash) && options == {} if button_text_or_regexp.is_a?(Hash) && options == {}
pattern, options = nil, button_text_or_regexp pattern, options = nil, button_text_or_regexp
@ -62,16 +77,21 @@ module Webrat
pattern ||= '*' pattern ||= '*'
locator = "button=#{pattern}" locator = "button=#{pattern}"
selenium.wait_for_element locator, 5 selenium.wait_for_element locator, :timeout_in_seconds => 5
selenium.click locator selenium.click locator
end end
webrat_deprecate :clicks_button, :click_button webrat_deprecate :clicks_button, :click_button
def click_link(link_text_or_regexp, options = {}) def click_link(link_text_or_regexp, options = {})
pattern = adjust_if_regexp(link_text_or_regexp) if link_text_or_regexp.is_a?(Regexp)
pattern = "evalregex:#{link_text_or_regexp.inspect}"
else
pattern = link_text_or_regexp.to_s
end
locator = "webratlink=#{pattern}" locator = "webratlink=#{pattern}"
selenium.wait_for_element locator, 5 selenium.wait_for_element locator, :timeout_in_seconds => 5
selenium.click locator selenium.click locator
end end
@ -79,7 +99,7 @@ module Webrat
def click_link_within(selector, link_text, options = {}) def click_link_within(selector, link_text, options = {})
locator = "webratlinkwithin=#{selector}|#{link_text}" locator = "webratlinkwithin=#{selector}|#{link_text}"
selenium.wait_for_element locator, 5 selenium.wait_for_element locator, :timeout_in_seconds => 5
selenium.click locator selenium.click locator
end end
@ -94,7 +114,7 @@ module Webrat
select_locator = "webratselectwithoption=#{option_text}" select_locator = "webratselectwithoption=#{option_text}"
end end
selenium.wait_for_element select_locator, 5 selenium.wait_for_element select_locator, :timeout_in_seconds => 5
selenium.select(select_locator, option_text) selenium.select(select_locator, option_text)
end end
@ -102,7 +122,7 @@ module Webrat
def choose(label_text) def choose(label_text)
locator = "webrat=#{label_text}" locator = "webrat=#{label_text}"
selenium.wait_for_element locator, 5 selenium.wait_for_element locator, :timeout_in_seconds => 5
selenium.click locator selenium.click locator
end end
@ -110,9 +130,10 @@ module Webrat
def check(label_text) def check(label_text)
locator = "webrat=#{label_text}" locator = "webrat=#{label_text}"
selenium.wait_for_element locator, 5 selenium.wait_for_element locator, :timeout_in_seconds => 5
selenium.check locator selenium.click locator
end end
alias_method :uncheck, :check
webrat_deprecate :checks, :check webrat_deprecate :checks, :check
@ -142,8 +163,10 @@ module Webrat
begin begin
value = yield value = yield
rescue ::Spec::Expectations::ExpectationNotMetError, ::Selenium::CommandError, Webrat::WebratError rescue Exception => e
value = nil unless is_ignorable_wait_for_exception?(e)
raise e
end
end end
return value if value return value if value
@ -151,7 +174,19 @@ module Webrat
sleep 0.25 sleep 0.25
end end
raise Webrat::TimeoutError.new(message + " (after #{timeout} sec)") error_message = "#{message} (after #{timeout} sec)"
if $browser && Webrat.configuration.selenium_verbose_output
error_message += <<-EOS
HTML of the page was:
#{selenium.get_html_source}"
EOS
end
raise Webrat::TimeoutError.new(error_message)
true true
end end
@ -165,29 +200,34 @@ module Webrat
def save_and_open_screengrab def save_and_open_screengrab
return unless File.exist?(saved_page_dir) return unless File.exist?(Webrat.configuration.saved_pages_dir)
filename = "#{saved_page_dir}/webrat-#{Time.now.to_i}.png" filename = "#{Webrat.configuration.saved_pages_dir}/webrat-#{Time.now.to_i}.png"
if $browser.chrome_backend? if $browser.chrome_backend?
$browser.capture_entire_page_screenshot(filename, '') $browser.capture_entire_page_screenshot(filename, '')
else else
$browser.capture_screenshot(filename) $browser.capture_screenshot(filename)
end end
open_in_browser(filename) open_in_browser(filename)
end end
protected protected
def is_ignorable_wait_for_exception?(exception) #:nodoc:
if defined?(::Spec::Expectations::ExpectationNotMetError)
return true if exception.class == ::Spec::Expectations::ExpectationNotMetError
end
return true if [::Selenium::CommandError, Webrat::WebratError].include?(exception.class)
return false
end
def setup #:nodoc: def setup #:nodoc:
silence_stream(STDOUT) do Webrat::Selenium::SeleniumRCServer.boot
Webrat.start_selenium_server Webrat::Selenium::ApplicationServerFactory.app_server_instance.boot
Webrat.start_app_server
end
create_browser create_browser
$browser.start $browser.start
teardown_at_exit
extend_selenium extend_selenium
define_location_strategies define_location_strategies
@ -197,16 +237,12 @@ module Webrat
def create_browser def create_browser
$browser = ::Selenium::Client::Driver.new(Webrat.configuration.selenium_server_address || "localhost", $browser = ::Selenium::Client::Driver.new(Webrat.configuration.selenium_server_address || "localhost",
Webrat.configuration.selenium_server_port, Webrat.configuration.selenium_browser_key, "http://#{Webrat.configuration.application_address}:#{Webrat.configuration.application_port}") Webrat.configuration.selenium_server_port, Webrat.configuration.selenium_browser_key, "http://#{Webrat.configuration.application_address}:#{Webrat.configuration.application_port_for_selenium}")
$browser.set_speed(0) unless Webrat.configuration.selenium_server_address $browser.set_speed(0) unless Webrat.configuration.selenium_server_address
end
def teardown_at_exit #:nodoc:
at_exit do at_exit do
silence_stream(STDOUT) do silence_stream(STDOUT) do
$browser.stop $browser.stop
Webrat.stop_app_server
Webrat.stop_selenium_server
end end
end end
end end
@ -233,4 +269,4 @@ module Webrat
end end
end end
end end
end end

View File

@ -0,0 +1,18 @@
module Webrat
module Selenium
module SilenceStream
# active_support already defines silence_stream, no need to do that again if it's already present.
# http://github.com/rails/rails/blob/master/activesupport/lib/active_support/core_ext/kernel/reporting.rb
unless Kernel.respond_to?(:silence_stream)
def silence_stream(stream)
old_stream = stream.dup
stream.reopen(RUBY_PLATFORM =~ /mswin/ ? 'NUL:' : '/dev/null')
stream.sync = true
yield
ensure
stream.reopen(old_stream)
end
end
end
end
end

View File

@ -1,29 +0,0 @@
require 'webrat/rack'
require 'sinatra'
require 'sinatra/test/methods'
class Sinatra::Application
# Override this to prevent Sinatra from barfing on the options passed from RSpec
def self.load_default_options_from_command_line!
end
end
disable :run
disable :reload
module Webrat
class SinatraSession < RackSession #:nodoc:
include Sinatra::Test::Methods
attr_reader :request, :response
%w(get head post put delete).each do |verb|
define_method(verb) do |*args| # (path, data, headers = nil)
path, data, headers = *args
data = data.inject({}) {|data, (key,value)| data[key] = Rack::Utils.unescape(value); data }
params = data.merge(:env => headers || {})
self.__send__("#{verb}_it", path, params)
end
end
end
end

View File

@ -1,14 +1,13 @@
module Webrat #:nodoc: module Webrat #:nodoc:
def self.session_class #:nodoc: def self.adapter_class #:nodoc:
TestSession TestAdapter
end end
class TestSession < Session #:nodoc: class TestAdapter #:nodoc:
attr_accessor :response_body attr_accessor :response_body
attr_writer :response_code attr_writer :response_code
def doc_root def initialize(*args)
File.expand_path(File.join(".", "public"))
end end
def response def response
@ -16,7 +15,7 @@ module Webrat #:nodoc:
end end
def response_code def response_code
@response_code || 200 @response_code ||= 200
end end
def get(url, data, headers = nil) def get(url, data, headers = nil)
@ -31,4 +30,4 @@ module Webrat #:nodoc:
def delete(url, data, headers = nil) def delete(url, data, headers = nil)
end end
end end
end end

View File

@ -0,0 +1,7 @@
require 'rubygems'
require 'spec/rake/spectask'
Spec::Rake::SpecTask.new do |t|
t.spec_opts = ['--color']
t.spec_files = FileList['spec/**/*_spec.rb']
end

View File

@ -0,0 +1,2 @@
require "sample_app"
run SampleApp

View File

@ -0,0 +1,35 @@
require "sinatra/base"
class SampleApp < Sinatra::Default
get "/" do
"Hello World"
end
get "/internal_redirect" do
redirect URI.join(request.url, "redirected").to_s
end
get "/external_redirect" do
redirect "http://example.tst/"
end
get "/redirected" do
"Redirected"
end
get "/form" do
<<-EOS
<html>
<form action="/form" method="post">
<input type="hidden" name="_method" value="put" />
<label for="email">Email:</label> <input type="text" id="email" name="email" /></label>
<input type="submit" value="Add" />
</form>
</html>
EOS
end
put "/form" do
"Welcome #{params[:email]}"
end
end

View File

@ -0,0 +1,30 @@
require File.dirname(__FILE__) + "/spec_helper"
describe "Webrat's Mechanize mode" do
it "should work" do
response = visit("http://localhost:9292/")
response.should contain("Hello World")
end
it "should follow redirects" do
response = visit("http://localhost:9292/internal_redirect")
response.should contain("Redirected")
end
it "should follow links"
it "should submit forms" do
visit "http://localhost:9292/form"
fill_in "Email", :with => "albert@example.com"
response = click_button "Add"
response.should contain("Welcome albert@example.com")
end
it "should not follow external redirects" do
pending do
response = visit("http://localhost:9292/external_redirect")
response.should contain("Foo")
end
end
end

View File

@ -0,0 +1,29 @@
require "rubygems"
require "spec"
$LOAD_PATH.unshift File.dirname(__FILE__) + "/../../../../lib"
require "webrat"
Webrat.configure do |config|
config.mode = :mechanize
end
Spec::Runner.configure do |config|
config.include Webrat::Methods
config.include Webrat::Matchers
config.before :suite do
if File.exists?("rack.pid")
Process.kill("TERM", File.read("rack.pid").to_i)
end
system "rackup --daemonize --pid rack.pid config.ru"
end
config.after :suite do
if File.exists?("rack.pid")
Process.kill("TERM", File.read("rack.pid").to_i)
end
end
end

View File

@ -1,2 +1,2 @@
class Application < Merb::Controller class Application < Merb::Controller
end end

View File

@ -1,5 +1,5 @@
class Exceptions < Merb::Controller class Exceptions < Merb::Controller
# handle NotFound exceptions (404) # handle NotFound exceptions (404)
def not_found def not_found
render :format => :html render :format => :html
@ -10,4 +10,4 @@ class Exceptions < Merb::Controller
render :format => :html render :format => :html
end end
end end

View File

@ -1,18 +1,27 @@
class Testing < Application class Testing < Application
def show_form def show_form
render render
end end
def upload
case request.method
when :get then render
when :post then
uploaded_file = params[:uploaded_file]
render [uploaded_file[:filename], uploaded_file[:tempfile].class.name].inspect
end
end
def submit_form def submit_form
end end
def internal_redirect def internal_redirect
redirect "/" redirect "/"
end end
def external_redirect def external_redirect
redirect "http://google.com" redirect "http://google.com"
end end
end end

View File

@ -0,0 +1,9 @@
<h1>Webrat Form</h1>
<form action="/upload" method="post">
<label>
File
<input type="file" name="uploaded_file" />
</label>
<input type="submit" value="Upload">
</form>

View File

@ -1,25 +1,25 @@
# Go to http://wiki.merbivore.com/pages/init-rb # Go to http://wiki.merbivore.com/pages/init-rb
# Specify a specific version of a dependency # Specify a specific version of a dependency
# dependency "RedCloth", "> 3.0" # dependency "RedCloth", "> 3.0"
# use_orm :none # use_orm :none
use_test :rspec use_test :rspec
use_template_engine :erb use_template_engine :erb
Merb::Config.use do |c| Merb::Config.use do |c|
c[:use_mutex] = false c[:use_mutex] = false
c[:session_store] = 'cookie' # can also be 'memory', 'memcache', 'container', 'datamapper c[:session_store] = 'cookie' # can also be 'memory', 'memcache', 'container', 'datamapper
# cookie session store configuration # cookie session store configuration
c[:session_secret_key] = 'adb9ea7a0e94b5513503f58623a393c5efe18851' # required for cookie session store c[:session_secret_key] = 'adb9ea7a0e94b5513503f58623a393c5efe18851' # required for cookie session store
c[:session_id_key] = '_merb_session_id' # cookie session id key, defaults to "_session_id" c[:session_id_key] = '_merb_session_id' # cookie session id key, defaults to "_session_id"
end end
Merb::BootLoader.before_app_loads do Merb::BootLoader.before_app_loads do
# This will get executed after dependencies have been loaded but before your app's classes have loaded. # This will get executed after dependencies have been loaded but before your app's classes have loaded.
end end
Merb::BootLoader.after_app_loads do Merb::BootLoader.after_app_loads do
# This will get executed after your app's classes have been loaded. # This will get executed after your app's classes have been loaded.
end end

View File

@ -8,4 +8,4 @@ end
use Merb::Rack::Static, Merb.dir_for(:public) use Merb::Rack::Static, Merb.dir_for(:public)
# this is our main merb application # this is our main merb application
run Merb::Rack::Application.new run Merb::Rack::Application.new

View File

@ -10,7 +10,7 @@
# #
# match("/books/:book_id/:action"). # match("/books/:book_id/:action").
# to(:controller => "books") # to(:controller => "books")
# #
# Or, use placeholders in the "to" results for more complicated routing, e.g.: # Or, use placeholders in the "to" results for more complicated routing, e.g.:
# #
# match("/admin/:module/:controller/:action/:id"). # match("/admin/:module/:controller/:action/:id").
@ -28,6 +28,7 @@
Merb.logger.info("Compiling routes...") Merb.logger.info("Compiling routes...")
Merb::Router.prepare do Merb::Router.prepare do
match("/").to(:controller => "testing", :action => "show_form") match("/").to(:controller => "testing", :action => "show_form")
match("/upload").to(:controller => "testing", :action => "upload")
match("/internal_redirect").to(:controller => "testing", :action => "internal_redirect") match("/internal_redirect").to(:controller => "testing", :action => "internal_redirect")
match("/external_redirect").to(:controller => "testing", :action => "external_redirect") match("/external_redirect").to(:controller => "testing", :action => "external_redirect")
end end

View File

@ -1,5 +1,7 @@
require "rubygems" require "rubygems"
$LOAD_PATH.unshift File.dirname(__FILE__) + "/../../../../lib"
# Add the local gems dir if found within the app root; any dependencies loaded # Add the local gems dir if found within the app root; any dependencies loaded
# hereafter will try to load from the local gems before loading system gems. # hereafter will try to load from the local gems before loading system gems.
if (local_gem_dir = File.join(File.dirname(__FILE__), '..', 'gems')) && $BUNDLE.nil? if (local_gem_dir = File.join(File.dirname(__FILE__), '..', 'gems')) && $BUNDLE.nil?
@ -21,4 +23,4 @@ end
Webrat.configure do |config| Webrat.configure do |config|
config.mode = :merb config.mode = :merb
end end

View File

@ -19,7 +19,7 @@ describe "Webrat" do
response.status.should == 200 response.status.should == 200
response.should contain("Webrat Form") response.should contain("Webrat Form")
end end
it "should check the value of a field" do it "should check the value of a field" do
visit "/" visit "/"
field_labeled("Prefilled").value.should == "text" field_labeled("Prefilled").value.should == "text"
@ -29,4 +29,11 @@ describe "Webrat" do
response = visit "/external_redirect" response = visit "/external_redirect"
response.status.should == 302 response.status.should == 302
end end
end
it "should upload files" do
visit "/upload"
attach_file "File", __FILE__
response = click_button "Upload"
response.should contain(%(["webrat_spec.rb", "Tempfile"]))
end
end

View File

@ -12,9 +12,9 @@ if File.directory?(gems_dir)
Gem.clear_paths Gem.clear_paths
Gem.path.replace([File.expand_path(gems_dir)]) Gem.path.replace([File.expand_path(gems_dir)])
ENV["PATH"] = "#{File.dirname(__FILE__)}:#{ENV["PATH"]}" ENV["PATH"] = "#{File.dirname(__FILE__)}:#{ENV["PATH"]}"
gem_file = File.join(gems_dir, "specifications", "<%= spec.name %>-*.gemspec") gem_file = File.join(gems_dir, "specifications", "<%= spec.name %>-*.gemspec")
if local_gem = Dir[gem_file].last if local_gem = Dir[gem_file].last
version = File.basename(local_gem)[/-([\.\d]+)\.gemspec$/, 1] version = File.basename(local_gem)[/-([\.\d]+)\.gemspec$/, 1]
end end
@ -28,4 +28,4 @@ if ARGV.first =~ /^_(.*)_$/ and Gem::Version.correct? $1 then
end end
gem '<%= @spec.name %>', version gem '<%= @spec.name %>', version
load '<%= bin_file_name %>' load '<%= bin_file_name %>'

View File

@ -7,21 +7,21 @@ module Gem
BUNDLED_SPECS = File.join(Dir.pwd, "gems", "specifications") BUNDLED_SPECS = File.join(Dir.pwd, "gems", "specifications")
MAIN_INDEX = Gem::SourceIndex.from_gems_in(BUNDLED_SPECS) MAIN_INDEX = Gem::SourceIndex.from_gems_in(BUNDLED_SPECS)
FALLBACK_INDEX = Gem::SourceIndex.from_installed_gems FALLBACK_INDEX = Gem::SourceIndex.from_installed_gems
def self.source_index def self.source_index
MultiSourceIndex.new MultiSourceIndex.new
end end
def self.searcher def self.searcher
MultiPathSearcher.new MultiPathSearcher.new
end end
class ArbitrarySearcher < GemPathSearcher class ArbitrarySearcher < GemPathSearcher
def initialize(source_index) def initialize(source_index)
@source_index = source_index @source_index = source_index
super() super()
end end
def init_gemspecs def init_gemspecs
@source_index.map { |_, spec| spec }.sort { |a,b| @source_index.map { |_, spec| spec }.sort { |a,b|
(a.name <=> b.name).nonzero? || (b.version <=> a.version) (a.name <=> b.name).nonzero? || (b.version <=> a.version)
@ -34,31 +34,31 @@ module Gem
@main_searcher = ArbitrarySearcher.new(MAIN_INDEX) @main_searcher = ArbitrarySearcher.new(MAIN_INDEX)
@fallback_searcher = ArbitrarySearcher.new(FALLBACK_INDEX) @fallback_searcher = ArbitrarySearcher.new(FALLBACK_INDEX)
end end
def find(path) def find(path)
try = @main_searcher.find(path) try = @main_searcher.find(path)
return try if try return try if try
@fallback_searcher.find(path) @fallback_searcher.find(path)
end end
def find_all(path) def find_all(path)
try = @main_searcher.find_all(path) try = @main_searcher.find_all(path)
return try unless try.empty? return try unless try.empty?
@fallback_searcher.find_all(path) @fallback_searcher.find_all(path)
end end
end end
class MultiSourceIndex class MultiSourceIndex
def search(*args) def search(*args)
try = MAIN_INDEX.search(*args) try = MAIN_INDEX.search(*args)
return try unless try.empty? return try unless try.empty?
FALLBACK_INDEX.search(*args) FALLBACK_INDEX.search(*args)
end end
def find_name(*args) def find_name(*args)
try = MAIN_INDEX.find_name(*args) try = MAIN_INDEX.find_name(*args)
return try unless try.empty? return try unless try.empty?
FALLBACK_INDEX.find_name(*args) FALLBACK_INDEX.find_name(*args)
end end
end end
end end

View File

@ -2,12 +2,12 @@ require "erb"
Gem.pre_install_hooks.push(proc do |installer| Gem.pre_install_hooks.push(proc do |installer|
$INSTALLING << installer.spec $INSTALLING << installer.spec
unless File.file?(installer.bin_dir / "common.rb") unless File.file?(installer.bin_dir / "common.rb")
FileUtils.mkdir_p(installer.bin_dir) FileUtils.mkdir_p(installer.bin_dir)
FileUtils.cp(File.dirname(__FILE__) / "common.rb", installer.bin_dir / "common.rb") FileUtils.cp(File.dirname(__FILE__) / "common.rb", installer.bin_dir / "common.rb")
end end
include ColorfulMessages include ColorfulMessages
name = installer.spec.name name = installer.spec.name
if $GEMS && versions = ($GEMS.assoc(name) || [])[1] if $GEMS && versions = ($GEMS.assoc(name) || [])[1]
@ -25,19 +25,19 @@ end)
class ::Gem::Uninstaller class ::Gem::Uninstaller
def self._with_silent_ui def self._with_silent_ui
ui = Gem::DefaultUserInteraction.ui ui = Gem::DefaultUserInteraction.ui
def ui.say(str) def ui.say(str)
puts "- #{str}" puts "- #{str}"
end end
yield yield
class << Gem::DefaultUserInteraction.ui class << Gem::DefaultUserInteraction.ui
remove_method :say remove_method :say
end end
end end
def self._uninstall(source_index, name, op, version) def self._uninstall(source_index, name, op, version)
unless source_index.find_name(name, "#{op} #{version}").empty? unless source_index.find_name(name, "#{op} #{version}").empty?
uninstaller = Gem::Uninstaller.new( uninstaller = Gem::Uninstaller.new(
@ -50,7 +50,7 @@ class ::Gem::Uninstaller
_with_silent_ui { uninstaller.uninstall } _with_silent_ui { uninstaller.uninstall }
end end
end end
def self._uninstall_others(source_index, name, version) def self._uninstall_others(source_index, name, version)
_uninstall(source_index, name, "<", version) _uninstall(source_index, name, "<", version)
_uninstall(source_index, name, ">", version) _uninstall(source_index, name, ">", version)
@ -67,14 +67,14 @@ end)
class ::Gem::DependencyInstaller class ::Gem::DependencyInstaller
alias old_fg find_gems_with_sources alias old_fg find_gems_with_sources
def find_gems_with_sources(dep) def find_gems_with_sources(dep)
if @source_index.any? { |_, installed_spec| if @source_index.any? { |_, installed_spec|
installed_spec.satisfies_requirement?(dep) installed_spec.satisfies_requirement?(dep)
} }
return [] return []
end end
old_fg(dep) old_fg(dep)
end end
end end
@ -83,9 +83,9 @@ class ::Gem::SpecFetcher
alias old_fetch fetch alias old_fetch fetch
def fetch(dependency, all = false, matching_platform = true) def fetch(dependency, all = false, matching_platform = true)
idx = Gem::SourceIndex.from_installed_gems idx = Gem::SourceIndex.from_installed_gems
dep = idx.search(dependency).sort.last dep = idx.search(dependency).sort.last
if dep if dep
file = dep.loaded_from.dup file = dep.loaded_from.dup
file.gsub!(/specifications/, "cache") file.gsub!(/specifications/, "cache")
@ -121,4 +121,4 @@ class ::Gem::Specification
end end
specs + specs.map {|s| s.recursive_dependencies(self, index)}.flatten.uniq specs + specs.map {|s| s.recursive_dependencies(self, index)}.flatten.uniq
end end
end end

Some files were not shown because too many files have changed in this diff Show More