spec-kemal
Testing helpers for the Kemal web framework. Write expressive and readable tests for your Kemal applications using Crystal's built-in spec library.
Table of Contents
- Installation
- Quick Start
- API Reference
- Testing Patterns
- Configuration
- Troubleshooting
- Contributing
- License
Installation
Add spec-kemal to your shard.yml as a development dependency:
name: your-kemal-app
version: 0.1.0
dependencies:
kemal:
github: kemalcr/kemal
development_dependencies:
spec-kemal:
github: kemalcr/spec-kemal
Then run:
shards install
Quick Start
1. Set up your spec helper
Create or update spec/spec_helper.cr:
require "spec"
require "spec-kemal"
require "../src/your-kemal-app"
Spec.before_each do
Kemal.config.env = "test"
end
Spec.after_each do
Kemal.config.clear
end
2. Write your tests
# spec/your-kemal-app_spec.cr
require "./spec_helper"
describe "My Kemal App" do
it "renders the homepage" do
get "/"
response.status_code.should eq 200
response.body.should contain "Welcome"
end
it "creates a new user" do
post "/users", body: {name: "Crystal"}.to_json,
headers: HTTP::Headers{"Content-Type" => "application/json"}
response.status_code.should eq 201
end
end
3. Run your tests
KEMAL_ENV=test crystal spec
API Reference
HTTP Methods
spec-kemal provides helper methods for all standard HTTP verbs:
| Method | Description |
|--------|-------------|
| get(path, headers?, body?) | Sends a GET request |
| post(path, headers?, body?) | Sends a POST request |
| put(path, headers?, body?) | Sends a PUT request |
| patch(path, headers?, body?) | Sends a PATCH request |
| delete(path, headers?, body?) | Sends a DELETE request |
| head(path, headers?, body?) | Sends a HEAD request |
Parameters:
path : String- The request path (e.g.,"/users","/api/v1/posts?page=2")headers : HTTP::Headers?- Optional HTTP headersbody : String?- Optional request body
Response Object
After making a request, access the response using the response method:
get "/users"
# Status
response.status_code # => 200
response.status # => HTTP::Status::OK
response.success? # => true
# Body
response.body # => "{\"users\": []}"
# Headers
response.headers # => HTTP::Headers
response.headers["Content-Type"] # => "application/json"
response.content_type # => "application/json"
# Cookies
response.cookies # => HTTP::Cookies
response.cookies["session"] # => HTTP::Cookie
Headers
Pass custom headers to your requests:
headers = HTTP::Headers{
"Content-Type" => "application/json",
"Authorization" => "Bearer token123",
"Accept" => "application/json"
}
get "/protected", headers: headers
Request Body
Send data in the request body:
# JSON body
post "/api/users",
headers: HTTP::Headers{"Content-Type" => "application/json"},
body: {name: "John", email: "[email protected]"}.to_json
# Form-encoded body
post "/login",
headers: HTTP::Headers{"Content-Type" => "application/x-www-form-urlencoded"},
body: "username=john&password=secret"
Testing Patterns
JSON APIs
describe "Users API" do
it "returns users as JSON" do
get "/api/users",
headers: HTTP::Headers{"Accept" => "application/json"}
response.status_code.should eq 200
response.content_type.should eq "application/json"
users = JSON.parse(response.body)
users.as_a.size.should eq 3
end
it "creates a user" do
payload = {
name: "Alice",
email: "[email protected]"
}
post "/api/users",
headers: HTTP::Headers{"Content-Type" => "application/json"},
body: payload.to_json
response.status_code.should eq 201
user = JSON.parse(response.body)
user["name"].should eq "Alice"
end
it "handles validation errors" do
post "/api/users",
headers: HTTP::Headers{"Content-Type" => "application/json"},
body: {name: ""}.to_json
response.status_code.should eq 422
end
end
Form Data
describe "Login" do
it "authenticates with valid credentials" do
post "/login",
headers: HTTP::Headers{"Content-Type" => "application/x-www-form-urlencoded"},
body: "[email protected]&password=secret123"
response.status_code.should eq 302
response.headers["Location"].should eq "/dashboard"
end
end
Authentication
describe "Protected Routes" do
it "requires authentication" do
get "/admin/dashboard"
response.status_code.should eq 401
end
it "allows access with valid token" do
headers = HTTP::Headers{
"Authorization" => "Bearer valid-jwt-token"
}
get "/admin/dashboard", headers: headers
response.status_code.should eq 200
end
end
Sessions
For testing session-based features, require the session module:
require "spec-kemal/session"
Important: Configure your session secret before tests:
Spec.before_each do
Kemal::Session.config.secret = "your-test-secret"
end
Use with_session to create an authenticated session:
describe "Dashboard" do
it "shows user data from session" do
with_session do |session|
session.int("user_id", 42)
session.string("username", "alice")
get "/dashboard"
response.body.should contain "Welcome, alice"
end
end
it "handles session expiry" do
with_session do |session|
session.int("user_id", 42)
# Session is automatically destroyed after the block
end
get "/dashboard"
response.status_code.should eq 401
end
end
Available session methods:
session.string("key", "value") # String
session.int("key", 42) # Int32
session.bigint("key", 12345_i64) # Int64
session.float("key", 3.14) # Float64
session.bool("key", true) # Bool
session.object("key", my_object) # Any serializable object
Configuration
Disable Logging
Logging is disabled by default in spec-kemal. To enable it:
Kemal.config.logging = true
Error Handling
By default, Kemal rescues errors and renders an error page. For testing, you may want exceptions to propagate:
Spec.before_each do
Kemal.config.always_rescue = false
end
This is useful when testing error handling:
it "raises on invalid input" do
expect_raises(JSON::ParseException) do
post "/api/data",
headers: HTTP::Headers{"Content-Type" => "application/json"},
body: "invalid json"
end
end
Test Environment
Always run tests with KEMAL_ENV=test:
KEMAL_ENV=test crystal spec
Or set it in your spec helper:
ENV["KEMAL_ENV"] = "test"
Troubleshooting
"response is nil" Error
Make sure you've made a request before accessing response:
# Wrong
response.body # Error: response is nil
# Correct
get "/"
response.body # Works!
Tests Interfering with Each Other
Clear Kemal's configuration between tests:
Spec.after_each do
Kemal.config.clear
end
Session Not Working
-
Ensure you've required the session module:
require "spec-kemal/session" -
Set the session secret:
Kemal::Session.config.secret = "test-secret"
Handlers Not Being Called
Make sure Kemal.config.setup is called:
Spec.before_each do
Kemal.config.env = "test"
Kemal.config.setup
end
Contributing
We welcome contributions! Please see CONTRIBUTING.md for guidelines.
- Fork it (https://github.com/kemalcr/spec-kemal/fork)
- Create your feature branch (
git checkout -b my-new-feature) - Write tests for your changes
- Ensure all tests pass (
crystal spec) - Ensure code is formatted (
crystal tool format) - Commit your changes (
git commit -am 'Add some feature') - Push to the branch (
git push origin my-new-feature) - Create a new Pull Request
License
This project is licensed under the MIT License - see the LICENSE file for details.
Contributors
- sdogruyol - Creator and maintainer