Skip to content

abstract struct Athena::Framework::Spec::APITestCase
inherits Athena::Framework::Spec::WebTestCase #

A WebTestCase implementation with the intent of testing API controllers. Can be extended to add additional application specific configuration, such as setting up an authenticated user to make the request as.

Usage#

Say we want to test the following controller:

class ExampleController < ATH::Controller
  @[ARTA::Get("/add/{value1}/{value2}")]
  def add(value1 : Int32, value2 : Int32, @[ATHA::MapQueryParameter] negative : Bool = false) : Int32
    sum = value1 + value2
    negative ? -sum : sum
  end
end

We can define a struct inheriting from self to implement our test logic:

struct ExampleControllerTest < ATH::Spec::APITestCase
  def test_add_positive : Nil
    self.get("/add/5/3").body.should eq "8"
  end

  def test_add_negative : Nil
    self.get("/add/5/3?negative=true").body.should eq "-8"
  end
end

The #request method is used to make our requests to the API, then we run are assertions against the resulting HTTP::Server::Response. A key thing to point out is that there is no HTTP::Server involved, thus resulting in more performant specs.

Tip

Checkout the built in expectations to make testing easier.

Attention

Be sure to call Athena::Spec.run_all to your spec_helper.cr to ensure all test case instances are executed.

Mocking External Dependencies#

The previous example was quite simple. However, most likely a controller is going to have dependencies on various other services; such as an API client to make requests to a third party API. By default each test will be executed with the same services as it would normally, i.e. those requests to the third party API would actually be made. To solve this we can create a mock implementation of the API client and make it so that implementation is injected when the test runs.

# Create an example API client.
@[ADI::Register]
class APIClient
  def fetch_latest_data : String
    # Assume this method actually makes an `HTTP` request to get the latest data.
    "DATA"
  end
end

# Define a mock implementation of our APIClient that does not make a request and just returns mock data.
class MockAPIClient < APIClient
  def fetch_latest_data : String
    # This could also be an instance variable that gets set when this mock is created.
    "MOCK_DATA"
  end
end

# Enable our API client to be replaced in the service container.
class ADI::Spec::MockableServiceContainer
  # Use the block version of the `property` macro to use our mocked client by default, while still allowing it to be replaced at runtime.
  #
  # The block version of `getter` could also be used if you don't need to set it at runtime.
  # The `setter` macro could be also if you only want to allow replacing it at runtime.
  property(api_client) { MockAPIClient.new }
end

@[ADI::Register]
class ExampleServiceController < ATH::Controller
  def initialize(@api_client : APIClient); end

  @[ARTA::Post("/sync")]
  def sync_data : String
    # Use the injected api client to get the latest data to sync.
    data = @api_client.fetch_latest_data

    # ...

    data
  end
end

struct ExampleServiceControllerTest < ATH::Spec::APITestCase
  def initialize
    super

    # Our API client could also have been replaced at runtime;
    # such as if you wanted provide it what data it should return on a test by test basis.
    # self.client.container.api_client = MockAPIClient.new
  end

  def test_sync_data : Nil
    self.post("/sync").body.should eq %("MOCK_DATA")
  end
end

Tip

See ADI::Spec::MockableServiceContainer for more details on mocking services.

Each test_* method has its own service container instance. Any services that are mutated/replaced within the initialize method will affect all test_* methods. However, services can also be mutated/replaced within specific test_* methods to scope it that particular test; just be sure that you do it before calling #request.

Constructors#

.new#

Methods#

#client : ATH::Spec::HTTPBrowser#

Returns a reference to the AbstractBrowser being used for the test.

#delete(path : String, headers : HTTP::Headers = HTTP::Headers.new) : HTTP::Server::Response#

Makes a DELETE request to the provided path, optionally with the provided headers.

#get(path : String, headers : HTTP::Headers = HTTP::Headers.new) : HTTP::Server::Response#

Makes a GET request to the provided path, optionally with the provided headers.

#head(path : String, headers : HTTP::Headers = HTTP::Headers.new) : HTTP::Server::Response#

Makes a HEAD request to the provided path, optionally with the provided headers.

#link(path : String, headers : HTTP::Headers = HTTP::Headers.new) : HTTP::Server::Response#

Makes a LINK request to the provided path, optionally with the provided headers.

#patch(path : String, body : String | Bytes | IO | Nil = nil, headers : HTTP::Headers = HTTP::Headers.new) : HTTP::Server::Response#

Makes a PATCH request to the provided path, optionally with the provided body and headers.

#post(path : String, body : String | Bytes | IO | Nil = nil, headers : HTTP::Headers = HTTP::Headers.new) : HTTP::Server::Response#

Makes a POST request to the provided path, optionally with the provided body and headers.

#put(path : String, body : String | Bytes | IO | Nil = nil, headers : HTTP::Headers = HTTP::Headers.new) : HTTP::Server::Response#

Makes a PUT request to the provided path, optionally with the provided body and headers.

#request(method : String, path : String, body : String | Bytes | IO | Nil = nil, headers : HTTP::Headers = HTTP::Headers.new) : HTTP::Server::Response#

#request(request : HTTP::Request | ATH::Request) : HTTP::Server::Response#

#unlink(path : String, headers : HTTP::Headers = HTTP::Headers.new) : HTTP::Server::Response#

Makes a UNLINK request to the provided path, optionally with the provided headers.