I initially tried to use HttpMock which is used to test Active Resource framework within Rails. It is undocumented and you need to dig through web and Rails documentation it self to see how it works. Recently I came across excellent FakeWeb gem and ever since that I have been hooked on it. I am impressed by simplicity with which we I can mock RESTFul Rails web services.
Here is how I use Test::Unit and Shoulda with Rails to Mock RESTful web services for Active Resource classes. I am using XML based web services so I needed sample XML responses that Mock service should return. I create XML files containing those sample responses and store them under test/fixtures folder. After that I edited test_helper.rb file in test folder to have this utility method which loads sample responses from XML files located in fixtures folder.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
# Add more helper methods to be used by all tests here... | |
def load_xml_fixture_file(name) | |
path = RAILS_ROOT + "/test/fixtures/#{name}.xml" | |
if File.exists?(path) | |
contents = "" | |
File.open(path).each do |line| | |
contents << line | |
end | |
return contents | |
else | |
return nil | |
end | |
end |
After this lets jump on to unit test that we are going to write for our Active Resource class. Lets say we are testing an Active Resource class named User. Here is how I write unit test with FakeWeb and Shoulda gem.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
require 'test_helper' | |
require 'fakeweb' | |
require 'shoulda' | |
class UserTest < ActiveSupport::TestCase | |
def setup | |
FakeWeb.allow_net_connect = false | |
end | |
#All test cases go here | |
def teardown | |
FakeWeb.allow_net_connect = true | |
FakeWeb.clean_registry | |
end | |
end |
First I am requiring both FakeWeb and Shoulda a gem. I would also add these gems to config/environments/test.rb as required gem for test environment.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
# Specify gems that this application depends on in test env and have them installed with rake gems:install RAILS_ENV="test" | |
config.gem "fakeweb" | |
config.gem "shoulda" |
After requiring these two gems I would disable real HTTP connections so not to mix real Web services calls with mock RESTful Web services by setting allow_net_connect property of FakeWeb class to false. Similarly in teardown method I would re-enable net HTTP connections by setting allow_net_connect property to true.
Now I am going to write actual test using FakeWeb within Shoulda context blocks.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
context "logged in user" do | |
setup do | |
@user_resp = load_xml_fixture_file('user') | |
FakeWeb.register_uri(:get, "http://api.sample.com/users/123.xml", :body => @user_resp, :status => ["200", "OK"]) | |
end | |
should "be able to retrieve user object with given id" do | |
user = User.find(123) | |
assert_equal 123, user.id | |
assert_equal "test@example.com", user.email | |
end | |
end |
Here within the context block I am setting up url to register as GET request to return response that is contained in XML file. After that a simple test checks the returned response object with some assertions. This shows how easy it is to create mock RESTful web services with FakeWeb and test your Active Resource classes with RSpec or Test::Unit