Cucumber + Capybara – What we need for rails integration test

RubyOnRails 24 三月 2010 | View Comments


h2. What’s Capybara

Capybara is a webrat alternative which aims to support all browser simulators.

As you know, webrat can not run javascript on the webpage. In order to test javascript and AJAX based website we need install Selenium-clientgithub.com/ph7/selenium-client and learn extra API and also some configuratons.

Capybara give your a full stack solution.
You can use similar API drive webrat and selenium. And don’t need worry about configuraton.

What’s more, Capybara integrated Celerity which is a JRuby wrapper around HtmlUnit - a headless Java browser with JavaScript support. As your expect it using the same API.

And the killer feature of Capybara is you can easily change driver it use for testing. Not only in config file but also in runtime.

Why we need Cucumber

Before I introduce what is Cucumber, you need know what is BDD.

BDD is an evolution thinking behind TestDrivenDevelopment and AcceptanceTestDrivenPlanning

You don’t need worry about what the hell BDD is. After you start using Cucumber you will know. I promise.

Cucumber is allow you execute plain-text which written by BDD like format as automated tests.

There are plenty of materials on the internet which you can learn Cucumber

First place your need go is Ryan Bates’ RailsCasts Beginning with Cucumber
Then official Wiki of Cucumber will very helpful!!, and there also list some blog posts written by community

I won’t involve more about it. In my later blog post I will give you some tricks about using Cucumber.

Using Cucumber and Capybara

Install

Follow the instruction on Capybara:

Install as a gem
sudo gem install capybara

On OSX you may have to install libffi, you can install it via MacPorts with:
sudo port install libffi

And you also need install Cucumber
sudo gem install cucumber

Generate basic Cucumber folder structure and configuratons

Capybara is built to work nicely with Cucumber. You can easily generate Capybara style cucumber structure and configuraton.
script/generate cucumber --capybara

And maybe you also need install another gem named launchy which is helper class for launching cross-platform applications in a fire and forget manner.
sudo gem install launchy

Configuration

Available Configuration

Actually after you run the generator, you don’t need much more configuraton. Here I’ll list some of configuraton you can set.

You can specify it in features/support/env.rb file

Capybara.run_server = true #Whether start server when testing
Capybara.default_selector = :xpath #default selector , you can change to :css
Capybara.default_wait_time = 2 #When we testing AJAX, we can set a default wait time
Capybara.ignore_hidden_elements = false #Ignore hidden elements when testing, make helpful when you hide or show elements using javascript
Capybara.javascript_driver = :culerity #default driver when you using @javascript tag

Load rails test features

You can put codes below in features/support/env.rb file

Before do
  Fixtures.reset_cache
  fixtures_folder = File.join(RAILS_ROOT, 'test', 'fixtures')
  fixtures = Dir[File.join(fixtures_folder, '*.yml')].map {|f| File.basename(f, '.yml') }
  Fixtures.create_fixtures(fixtures_folder, fixtures)
end

And change

Cucumber::Rails::World.use_transactional_fixtures = false
DatabaseCleaner.strategy = :truncation

OK. For now we already finish configuration. we can start writing cucumber test

How to run test under different testing driver

As you know cucumber support tags
Capybara using tag to specify different driver, it supports @javascript, @selenium, @celerity, @culerity and @rack_test tags
You can use it like:

  @javascript
  Scenario: do something AJAXy
  When I click the AJAX link
  ...

About how to write cucumber, you can check out Cucumber Wiki

About Capybara API

What I want to show you is how’s Capybara API look like.

This is all support Webrat like APIs in Capybara

  DSL_METHODS = [
      :all, :attach_file, :body, :check, :choose, :click, :click_button, :click_link, :current_url, :drag, :evaluate_script,
      :field_labeled, :fill_in, :find, :find_button, :find_by_id, :find_field, :find_link, :has_content?, :has_css?,
      :has_no_content?, :has_no_css?, :has_no_xpath?, :has_xpath?, :locate, :save_and_open_page, :select, :source, :uncheck,
      :visit, :wait_until, :within, :within_fieldset, :within_table, :has_link?, :has_no_link?, :has_button?, :has_no_button?,
      :has_field?, :has_no_field?, :has_checked_field?, :has_unchecked_field?, :has_no_table?, :has_table?, :unselect,
      :has_select?, :has_no_select?
    ]

Here are some examples which are not mentioned in Capybara Wiki.

  # we can get page object from cucumber steps, page is an instance of @Capybara::Session@
  page.has_css? "ul.error_messages li", :count => 5, :text => "error"
  page.has_xpath? "//ul[@class='error_messages']/li", :count => 5, :text => "error"

  #Equivalent
  page.find(:css,"ul.error_messages li", :count => 5, :text => "error"
  page.find(:xpath,"//ul[@class='error_messages']/li", :count => 5, :text => "error"

  #Iterate all elements you found
  all(:xpath,"//ul[@class='error_messages']/input").each do |node|
    puts node.value
    puts node.[:attribute_name]
    puts node.click
    puts node.set("aa") #set value
    puts node.text
  end

XPath

If you want to use XPath in Capybara, you need caution string escape.
Capybara give us a good example

def s(string)
  if string.include?("'")
    string = string.split("'", -1).map do |substr|
    "'#{substr}'"
    end.join(%q{,"'",})
    "concat(#{string})"
  else
    "'#{string}'"
  end
end

Summarize

Cucumber + Capybara will make your integration test easilier.
Finally we find a full stack integration test.
Have fun with it!

Thanks aslakhellesoy and jnicklas

PS:In this blog post I did memtion Celerity because I haven't tried it.



Tagged in , , , , ,

  • http://boliviaonrails.com/ Boris Barroso

    I've been using rails 3 beta and I see that by default cucumber uses capybara, but I would like some help to config capybara to test javascript

  • allenwei

    What do you mean test javascript, test javascript behavior on pages or test javascript like unittest?

  • thelinuxlich

    Capybara looks nice, but how can I access session variables(example: user_session from devise) within it?

  • allenwei

    As we using selenium simulate the behavior of browser, you never get chance to get session. I did little googling, but can't find a solution.

    Actually when we using BDD style testing. We should only simulate what you can do and assert what user should see. Like black box test.

  • Justin Ko

    Not sure why you turn transactions off. You can load the fixtures before cucumber runs by placing the fixture code with no hooks and turning transactions on. This is much faster.

  • Justin Ko

    Actually, I take that back. Getting inconsistent database state with the above method.

  • http://www.adhost.dk/sogemaskineoptimering.shtml søgemaskineoptimering

    Those particular words, written in 1849, summarize the simple truth that the power of the individual and self-reliance in our free society are what has driven the development of American Exceptionalism. Unfortunately in…

  • http://pulse.yahoo.com/_FKJGB3WFXOZAJH2RZRL22ZXHRQ Bhe Jane

    I am just satisfied and content to uncover this post extremely helpful for me, as it features number of important information. I continually opt for to look over the good quality content and this thing I uncovered in your site. I am grateful for discussing this one.Mac Poker News | Play Poker On Mac | Mac Poker Gaming

  • http://pulse.yahoo.com/_WKFQJETCFBWPBZRJPZQN23JPC4 Jeff Dave

    You actually have a way of ideas. Great style of delivering the information and I could relate to it. Thanks a lot for this.

  • http://www.facebook.com/people/Raghu-Sangars/100000557598337 Raghu Sangars

    How to test Double click event using Capybara

  • Jacek Helper

    It’s not exacly as you say. Cucumber scenario we have chance to test some cases, and you must have a changes to create some background for that cases like create some user in Given block or add somethink to db and etc. Sessions is the same resoures like db and i thnik you should have the chance to prepare it for tests.

  • http://www.eusuperpharmacy.com/ Lowest price kamagra

    This one, called the Solar Cucumber, carries out the desalination of sea water and produces drinkable freshwater. The Solar Cucumber turns sea water to …

blog comments powered by Disqus

Ads Plugin created by Jake Ruston's Wordpress Plugins - Powered by and football database.