active_resource_test_helper

active_resource_test_helper makes it easier to use ActiveResouce::HttpMock.

Instead of declaring manually all the request-response pairs, it’s possible to use dynamically generated contents. These contents are defined using factory_girl and are stored into a Redis database using ohm.

Usually ActiveResource tests look like that:

def setup
  @matz  = { :id => 1, :name => "Matz" }.to_xml(:root => "person")
  ActiveResource::HttpMock.respond_to do |mock|
    mock.post   "/people.xml",   {}, @matz, 201, "Location" => "/people/1.xml"
    mock.get    "/people/1.xml", {}, @matz
    mock.put    "/people/1.xml", {}, nil, 204
    mock.delete "/people/1.xml", {}, nil, 200
  end
end

def test_get_matz
  person = Person.find(1)
  assert_equal "Matz", person.name
end

While using active_resource_test_helper an common ActiveResource test would be something like that:

class RemoteUserTest < Test::Unit::TestCase
  include ActiveResourceTestHelper
  active_resource_factories :user

  def setup
    assert_equal 0, User.count
    @users = []
    20.times do |i|
      @users << Factory.create(:basic_user)
    end
  end

  def test_find_by_id
    @users.each do |expected_user|
      user = UserResource.find(expected_user.id)
      assert_not_nil user
      assert_equal expected_user.first_name, user.first_name
    end
  end
end

HTTP methods supported

Currently active_resource_test_helper dynamic contents are served only by get requests. In the future they will be used also by post, put, delete and head operations.

Get requests

The following operations are currently fully supported:

Person.find(1)
Person.find(:all)
Person.find(:first)
Person.find(:last)
Person.find(:all, :params => { :title => "CEO" })
Person.find(:first, :params => { :first_name => "flavio", :last_name => "castelli" })

Requirements

Install Redis. On most platforms it’s as easy as grabbing the sources, running make and then putting the redis-server binary in the PATH.

Once you have it installed, you can execute redis-server and it will run on localhost:6379 by default. Check the redis.conf file that comes with the sources if you want to change some settings.

Then install the active_resource_test_helper gem:

sudo gem install active_resource_test_helper

Usage

In order to use active_resource_test_helper inside of your tests you have to:

  • require ‘active_resource_test_helper’

  • include the ActiveResourceTestHelper module inside of your test

This is a small example:

require  'active_resource_test_helper'

class MyTest < Test::Unit::TestCase
  include ActiveResourceTestHelper
  active_resource_factories :user, :post, :comment
end

Obviously you have also to define the factories used by your test (see below).

You can find more examples under the test directory.

Defining factories

Factories used by ActiveResource::HttpMock are defined with the Factory.define_active_resource_factory method:

Factory.define_active_resource_factory(:basic_user, :class => "User") do |u|
  u.sequence(:first_name) {|n| "first_name#{n}"}
  u.sequence(:last_name) {|n| "last_name#{n}"}
  u.admin false
  u.email {|u| "#{u.first_name}.#{u.last_name}@example.com" }
  u.age {rand(30) + 18}
end

If you are already familiar with factory_girl you will probably have already noticed we are using the same syntax.

What happens behind the scenes

active_resource_test_helper will automatically generate an ohm model for each build class used by the active resource factories.

These models will have one attribute and one index per each attribute declared inside of the factory.

The factory defined into the previous example will generate this model:

class User < Ohm::Model
    index :id

    attribute :first_name
    index :first_name

    attribute :last_name
    index :last_name

    attribute :admin
    index :admin

    attribute :email
    index :email

    attribute :age
    index :age

    alias :save! :save
end

Transactional factories

Currently redis doesn’t have a complete transaction support. active_resource_test_helper removes from the redis database all the instances of the models defined by the ActiveResource factories that are used by the test. This operation is executed before and after each test execution.

Statically defined responses

It’s possible to use dynamic contents and static pairs of request-response at the same time. However the responses defined in the static way have precedence over those generated in the dynamic way.