I have included a step by step guide in this README on how I set this up. But if you just want to get up and running, here are the steps:
- Clone this repository
- Rename the repo
mv rails5-api-guide/ my_app
- Inside your database.yml rename the database to whatever you want, otherwise it will just be named
generic_api_development
,generic_api_test
,generic_api_production
for each environment respectively. - Rename the top level module inside
application.rb
to whatever you want otherwise it will be namedGenericApi
- Run
bundle install
- Run
bundle exec rake db:create
- Run
bundle exec rake db:migrate; RAILS_ENV=test bundle exec rake db:migrate
- Boot up the server with
rails s
and navigate to localhost:3000 to see your new rails application
If you are interested in how this app was setup and what is included read on.
This guide shows you how I setup my default Rails architecture, now updated for Rails 5. Our Rails layer will do nothing but provide an API which serves JSON, and process background jobs. Here are some of the things we will be setting up:
- Token based authentication with the
devise_token_auth
gem. - A namespaced API
- Serialization with Active Model Serializers
- Testing with rspec and factory girl
- Useful rack middleares like rack-attack for rate limiting and throttling with rack-cors.
- Local development niceties such as mailcatcher, pry/pry-nav, and useful git commit hooks
- Background jobs with Sidekiq
- Caching with Redis
Install Rails
gem install rails
Create a new API only project with Postgres as the selected database
rails new project_name --api --database=postgresql
Setup the database
cd project_name
rails db:setup
rails db:migrate
Confirm that the server is working
Boot up the server with the command rails s
inside the project directory. Visit localhost:3000 and you should see a page that greets you with the message "Yay! You're on Rails!"
Add rspec, factory girl, and pry to test and development
Lets go ahead and a add a couple gems that will be very useful for testing and development. I prefer to use rspec for testing purposes, but some of you might wan't to stick with minitest, if that's the case you can just ignore the rspec-rails
gem. If not add rspec-rails
and factory_bot_rails
. FactoryBot will let us easily create and mock test objects. The gem pry-rails
provides a very handy interactive console when using rails c, pry-byebug
gives you powerful break points and pry-stack_expolorer
rounds out the package with a very solid stack explorer.
Open Gemfile, add the following. You can get rid of whatever else was in the :development, :test group
group :development, :test do
gem 'rspec-rails'
gem 'factory_bot_rails'
gem 'pry-rails'
gem 'pry-byebug'
gem 'pry-stack_explorer'
end
Install the gems by running bundle install
.
Finish the rspec installation by running the command rails generate rspec:install
.
You can go ahead and remove the test directory since we will be using /spec
rm -rf test
Now would be a good time to put everything under version control
git init
git add --all
git commit -m 'Initial Commit'
Set up your git remotes if you would like. I trust you know how to do this.
Add devise_token_auth
and omniauth
gem to the Gemfile:
gem 'devise_token_auth'
gem 'omniauth'
Run bundle install
to install the gems.
Generating the user model with devise concerns and routes
rails g devise_token_auth:install User auth
The devise generator accepts two arguments the user class (in our case User) which is the name of the class to use for user authentication. The second argument is the mount path (in our case auth) which is path at which to mount the authentication routes.
The generator will create an initializer at config/initializers/devise_token_auth.rb
, a User model at app/models/user.rb
, a concern will be included in app/controllers/application_controller.rb
, routes defined in config/routes.rb
, and a migration to create the users
table. You may want to tweak the migration to add columns to your liking. See the github page for more information.
Run rails db:migrate
to add the users table as defined by the devise migrations.
Now run rails routes
to see the authentication routes that have been setup.
Mailer Configuration
Since devise uses mailers for account registration, deletion, password changing, etcetera, we need to configure our development environment to send mail. The first thing is we need to add the following to config/initializers/devise.rb
Devise.setup do |config|
# Using rails-api, tell devise to not use ActionDispatch::Flash
# middleware b/c rails-api does not include it.
config.navigational_formats = [:json]
end
The comment explains it all. We wan't devise to ignore the ActionDispatch::Flash
middleware since it is not bundled in the Rails API mode middleware stack.
Next Add the following to config/environments/development.rb
config.action_mailer.delivery_method = :smtp
config.action_mailer.smtp_settings = { address: 'localhost', port: 1025 }
config.action_mailer.default_url_options = { :host => 'localhost', port: '3000' }
Token based authentication in action
Let's see some token authentication in action. Boot up your rails server with the rails s
command.
Examining the output of rails routes
we can see that in order to regiser a new user we have to submit a POST request to the /auth
routes. Let's do that using curl
:
curl --data "[email protected]&password=testpassword1&password_confirmation=testpassword1&confirm_success_url=http://localhost:3000" http://localhost:3000/auth
You should see a response that looks like the following:
{"status":"success","data":{"id":1,"provider":"email","uid":"[email protected]","name":null,"nickname":null,"image":null,"email":"[email protected]","created_at":"2016-07-23T23:16:10.064Z","updated_at":"2016-07-23T23:16:10.064Z"}}
If you examine your server logs, you should see that the action was processed and an email response was generated:
Started POST "/auth" for ::1 at 2016-07-23 19:16:09 -0400
Processing by DeviseTokenAuth::RegistrationsController#create as */*
Parameters: {"email"=>"[email protected]", "password"=>"[FILTERED]", "password_confirmation"=>"[FILTERED]", "confirm_success_url"=>"http://localhost:3000"}
-- snip --
Devise::Mailer#confirmation_instructions: processed outbound mail in 170.2ms
Sent mail to [email protected] (8.4ms)
Date: Sat, 23 Jul 2016 19:16:10 -0400
From: [email protected]
Reply-To: [email protected]
To: [email protected]
Message-ID:
Subject: Confirmation instructions
Mime-Version: 1.0
Content-Type: text/html;
charset=UTF-8
Content-Transfer-Encoding: 7bit
client-config: default
redirect-url: http://localhost:3000
<p>Welcome [email protected]!</p>
<p>You can confirm your account email through the link below: </p>
<p><a href="http://localhost:3000/auth/confirmation?config=default&confirmation_token=c9UbzsguMJ3BZkKZYC1A&redirect_url=http%3A%2F%2Flocalhost%3A3000">Confirm my account</a></p>
Completed 200 OK in 666ms (Views: 0.6ms | ActiveRecord: 35.4ms)
Go ahead and copy the link into your broswer. Then, boot up a rails console using the rails c
command, and run User.first
. This should return the new test user you just created.
Setting up mailcatcher
Reading the server logs to see what emails were rendered is not a great dev experience. Let's set up mailcatcher so that we can intercept emails delivered locally.
Install mailcatcher with the command gem install mailcatcher
. Start mailcatcher by running mailcatcher
. This will boot it up as a background daemon. Go to http://localhost:1080/ to see the mailcatcher UI.
Now lets try registering a new user.
curl --data "[email protected]&password=testpassword1&password_confirmation=testpassword1&confirm_success_url=http://localhost:3000" http://localhost:3000/auth
This time we should be able to see the email in the mailcatcher UI.
Our API will be responsible for rendering JSON responses, and that's about it. In order to build these JSON responses, we will use a gem called ActiveModelSerializers.
Install ActiveModelSerializers
Add gem 'active_model_serializers'
to the Gemfile and run bundle install
.
Add a config file in the location config/initializers/active_model_serializer.rb
with the code:
ActiveModelSerializers.config.adapter = :json_api
Generating a user serializer
Run the command rails g serializer user
. In the generated user serializer add the attributes :name, and :email so that the final code looks like this:
class UserSerializer < ActiveModel::Serializer
attributes :id, :name, :email
end
Setup an endpoint for serving user objects
Now that we have a user serializer, let's actually setup some endpoints that will serve us some user objects. This will be the begining of our API.
Let's start by setting up the routes. We will scope these api routes to a module 'api' and then provide a v1 namespace for all our routes. Here is what we will add to the routes file:
scope module: 'api' do
namespace :v1 do
resources :users, only: [:index, :show]
end
end
Run rails routes
to see the routes that have been set up. You should see something like this:
v1_users GET /v1/users(.:format) api/v1/users#index
v1_user GET /v1/users/:id(.:format) api/v1/users#show
Now let's add an api controller from which all our other controllers in the API will inherit. The file should live inapp/controllers/api/v1/api_controller.rb
.
module Api::V1
class ApiController < ApplicationController
# before_action :authenticate_user!
end
end
Next add a users controller, with two actions index and show: app/controllers/api/v1/users_controller.rb
module Api::V1
class UsersController < ApiController
# GET /v1/users
def index
render json: User.all
end
# GET /v1/users/{id}
def show
render json: User.find(params[:id])
end
end
end
Now lets confirm that our controllers, routes, and serialization is working as expected:
Run curl http://localhost:3000/v1/users
, which will be routed to our index action, and you should see a response that looks something like this:
{"data":[{"id":"1","type":"users","attributes":{"name":null,"email":"[email protected]"}},{"id":"2","type":"users","attributes":{"name":null,"email":"[email protected]"}}]}
Run curl http://localhost:3000/v1/users/1
, which will be routed to our show action, and you should see a response like this:
{"data":{"id":"1","type":"users","attributes":{"name":null,"email":"[email protected]"}}}
Great! We now have a bare minimum working API that serves some JSON. We can see that the attributes being returned are only the ones we specified in the serializer.
We setup devise, but we are not actually authenticating anywhere in our API! Let's do that now.
Devise makes it easy for us. All we have to do is add the following before_action
to inapp/controllers/api/v1/api_controller.rb
:
before_action :authenticate_user!
Since we are poking around in our base api controller, and since we will exclusively be serving JSON to our clients, we can add the following to api controller:
respond_to :json
Great! Now if we try to hit the users controller with the same curl commands we used above we should get an authentication error. Let's test it out. Run curl http://localhost:3000/v1/users/1
, and you should get the response
{"errors":["Authorized users only."]}
Using tokens for authentication
When using token based authentication, we have to generate a token for every single request. Tokens are invalidated after each request to the API. During each request, a new token is generated. The access-token header that should be used in the next request is returned in the access-token header of the response to the previous request. For more information see the devise_token_auth documentation.
When we run rails routes
we should see a devise route that looks like this:
user_session POST /auth/sign_in(.:format) devise_token_auth/sessions#create
This is our login route. It return the initial auth token for when a user signs in. Let's test that by sending a post request to the endpoint with the appropriate payload:
curl -XPOST -v -H 'Content-Type: application/json' localhost:3000/auth/sign_in -d '{"email": "[email protected]", "password": "testpassword1" }'
We should get a response that looks like this:
* Connected to localhost (::1) port 3000 (#0)
> POST /auth/sign_in HTTP/1.1
-- snip --
< Content-Type: application/json; charset=utf-8
< access-token: Avj8j2wQ4JAlFUDuPbS3fQ
< token-type: Bearer
< client: r4Pn4MLXvpCFTkwSc0HD7w
< expiry: 1470579487
< uid: [email protected]
-- snip --
* Connection #0 to host localhost left intact
{"data":{"id":1,"email":"[email protected]","provider":"email","uid":"[email protected]","name":null,"nickname":null,"image":null,"type":"user"}}
As you can see, we get the access-token, client, and uid in the headers of the response. The JSON response provides us with some attributes on the user object. Since this is a default devise route, it does not go through the user serializer we setup.
Now we can use the validate_token route to confirm that our token is, well, valid:
curl -XGET -v -H 'Content-Type: application/json' -H 'access-token: Avj8j2wQ4JAlFUDuPbS3fQ' -H 'client: r4Pn4MLXvpCFTkwSc0HD7w' -H "uid: [email protected]" localhost:3000/auth/validate_token`
The success response:
* Connected to localhost (::1) port 3000 (#0)
> GET /auth/validate_token HTTP/1.1
-- snip --
< Content-Type: application/json; charset=utf-8
< access-token: 2J6ygFVQrYHGy6aSH25D_g
< token-type: Bearer
< client: r4Pn4MLXvpCFTkwSc0HD7w
< expiry: 1470579646
< uid: [email protected]
-- snip --
{"success":true,"data":{"id":1,"provider":"email","uid":"[email protected]","name":null,"nickname":null,"image":null,"email":"[email protected]","type":"user"}}
You can see the new access was generated and sent back in the headers of the previous response. So let's hit the API with the token from the last request!
curl -XGET -v -H 'Content-Type: application/json' -H 'access-token: 2J6ygFVQrYHGy6aSH25D_g' -H 'client: r4Pn4MLXvpCFTkwSc0HD7w' -H "uid: [email protected]" localhost:3000/v1/users/
And the sucessful response should look as follows:
{"data":[{"id":"2","type":"users","attributes":{"name":null,"email":"[email protected]"}},{"id":"1","type":"users","attributes":{"name":null,"email":"[email protected]"}}]}
Fantastic. At this point we have a bare minimum rails 5 api. Our routes are namespaced, users are authenticated with token based auth, and can sign up with email. We serialize our responses using active_model_serializers
. This is most of what we need to get up and running.
We sent some time setting up rspec and some useful debugging tools like pry-nav and factory girl, but we haven't actually written any tests yet. Why don't we go ahead and set up our first spec.
First we need to set up factory bot. At the top of our spec/spec_helper.rb
we need to explicitly require it as follows: require 'factory_girl_rails'
. Then to gain access to the FactoryBot DSL inside our specs we need to add the following line config.include FactoryBot::Syntax::Methods
within our config block. Then inside of our .rspec
file we need to add the following line --require rails_helper
. We can go ahead and get rid of the line --require spec_helper
, since the rails_helper
includes the spec_helper
. If we don't do this RSpec Cannot find the Controllers and throws an Uninitialized Constant error. See here for more information.
Now lets set up a very simple test for the users controller in the location spec/controllers/api/v1/users_controller_spec.rb
.
require 'spec_helper'
describe Api::V1::UsersController do
describe "GET #show" do
before(:each) do
@user = FactoryBot.create :user
auth_headers = @user.create_new_auth_token
request.headers.merge!(auth_headers)
get :show, id: @user.id
end
it 'responds with 200 status code' do
expect(response.code).to eq('200')
end
end
end
As you can see in the before block we hadd to generate an auth token for our user, and merge it into the request headers. See here for more information. Since you will likely be doing this in all your controller tests, it might be useful to abstract this to a convienience method inside spec_helper
, for now I will leave it.
When I tried running the test by executing rspec spec/controllers/api/v1/users_controller_spec.rb
I got the following error:
Devise::MissingWarden:
Devise could not find the `Warden::Proxy` instance on your request environment.
Make sure that your application is loading Devise and Warden as expected and that the `Warden::Manager` middleware is present in your middleware stack.
If you are seeing this on one of your tests, ensure that your tests are either executing the Rails middleware stack or that your tests are using the `Devise::Test::ControllerHelpers` module to inject the `request.env['warden']` object for you.
What a useful error message! All I had to do was add the devise test helpers to rails_helper.rb
inside the config block: config.include Devise::Test::ControllerHelpers, type: :controller
.
Now running the test should pass!
Let's flesh this test out by adding another expectation for checking the response body:
it "returns the serialized user attributes" do
expect(JSON.parse(response.body)['data']['attributes']).to eq({"name"=>"John Doe", "email"=>"[email protected]"})
end
And now let's add some tests for the index action:
describe Api::V1::UsersController do
-- snip --
describe 'GET #index' do
before(:each) do
@user = FactoryBot.create :user
auth_headers = @user.create_new_auth_token
request.headers.merge!(auth_headers)
get :index, id: @user.id
end
it 'responds with 200 status code' do
expect(response.code).to eq('200')
end
it 'returns the serialized user attributes' do
expect(JSON.parse(response.body)['data'].length).to eq(1)
expect(JSON.parse(response.body)['data'].first['attributes']).to eq({'name'=>'John Doe', 'email'=>'[email protected]'})
end
end
end
Use Rack CORS to allows cross origin resource sharing
If your API will be public, you will need to enable Cross-Origin Resource Sharing. A resource makes a cross-origin HTTP request when it requests a resource from a different domain than the one which the first resource itself serves.
Add gem 'rack-cors'
to the Gemfile and run bundle install
Then add this to application.rb
:
config.middleware.use Rack::Cors do
allow do
origins '*'
resource '*',
:headers => :any,
:expose => ['access-token', 'expiry', 'token-type', 'uid', 'client'],
:methods => [:get, :post, :options, :delete, :put]
end
end
WARNING: This will whitelist requests from any domain. Make sure to whitelist only the needed domains.
Rate limiting/throttling, blacklisting/whitelisting of IP's with rack-attack
rack-attack
is a middleware that allows us rate limit, throttle, white list, and black list IP's. Trust me, you want this.
Add gem 'rack-attack'
to the Gemfile and then run bundle install
. Inside of config/application.rb
add config.middleware.use Rack::Attack
.
Next, create a new initializer config/initializers/rack_attack.rb
. I plucked the basic config from this super helpful guide
class Rack::Attack
# `Rack::Attack` is configured to use the `Rails.cache` value by default,
# but you can override that by setting the `Rack::Attack.cache.store` value
Rack::Attack.cache.store = ActiveSupport::Cache::MemoryStore.new
# Allow all local traffic
safelist('allow-localhost') do |req|
'127.0.0.1' == req.ip || '::1' == req.ip
end
# Allow an IP address to make 5 requests every 5 seconds
throttle('req/ip', limit: 5, period: 5) do |req|
req.ip
end
# Send the following response to throttled clients
self.throttled_response = ->(env) {
retry_after = (env['rack.attack.match_data'] || {})[:period]
[
429,
{'Content-Type' => 'application/json', 'Retry-After' => retry_after.to_s},
[{error: "Throttle limit reached. Retry later."}.to_json]
]
}
end
Depending on your needs, the 5 requests every 5 seconds may be a little too strict. For instance, if you are building a web client that makes several async requests, you may want to loosen the limit a bit.
You may have noticed that we are using the ActiveSupport::Cache::MemoryStore.new
to back rack-attack. This is no good! This particular store puts everything into memory in the same process, so while the cache might be blazingly fast, it disappears when the process dissapears, and it's cache data cannot be shared across processes.
I like to use Redis for cacheing. Some prefer Memcached. Both have their merits. But since we will be using sidekiq for job processing which requires Redis, and we may want to use action-cable, we might as well stick with Redis.
First add gem 'redis-rails'
to your Gemfile, and then run bundle install
. Install and start redis using instructions here.
In development.rb
get rid of the block of code that looks like this:
# Enable/disable caching. By default caching is disabled.
if Rails.root.join('tmp/caching-dev.txt').exist?
config.action_controller.perform_caching = true
config.cache_store = :memory_store
config.public_file_server.headers = {
'Cache-Control' => 'public, max-age=172800'
}
else
config.action_controller.perform_caching = false
config.cache_store = :null_store
end
And then replace with this:
if ENV['REDIS_URL']
config.action_controller.perform_caching = true
config.cache_store = :redis_store, ENV['REDIS_URL']
else
config.action_controller.perform_caching = false
config.cache_store = :null_store
end
This way we can enable cacheing locally if we wish to test it. Usually in development cacheing is turned off, but if you want to enable it all you have to do is set the environment variable REDIS_URL
to redis://localhost:6379
.
In production.rb
add
config.cache_store = :redis_store, ENV['REDIS_URL']
This is sort of optional, but Sideiq is my go to for background job processing in ruby. It is extremely robust and efficient.
Add gem 'sidekiq'
to the Gemfile and bundle.
Create an app/workers
directory, and try running a sample job:
class SampleWorker
include Sidekiq::Worker
def perform()
# do something
end
end
You can kick it off in your console with SampleWorker.perform_async
which will return a job ID.
http://www.victorareba.com/tutorials/speed-your-rails-app-with-model-caching-using-redis https://medium.com/ruby-on-rails/easy-caching-with-rails-4-heroku-redis-5fb36381628#.o1a12hbzb
Setup Rack-attack to use Default Cache
Now that we have enabled redis we can enable rack-attack to use the default cache. In config/initializers/rack_attack.rb
remove the line Rack::Attack.cache.store = ActiveSupport::Cache::MemoryStore.new
. By default, Rack::Attack will use whichever caching backend is configured as Rails.cache.
Set up a git commit hook to stop from committing binding.pry
I use binding.pry all over the place, and sometimes I forget to take it out before commiting. Here's a little pre-commit hook taken from the following gist to prevent that from happening.
Inside the /.git/hooks/pre-commit
file
#!/bin/bash
FILES_PATTERN='\.(rb)(\..+)?$'
FORBIDDEN='binding.pry'
git diff --cached --name-only | \
grep -E $FILES_PATTERN | \
GREP_COLOR='4;5;37;41' xargs grep --color --with-filename -n $FORBIDDEN && \
echo 'COMMIT REJECTED' && \
exit 1
exit 0
Make it executable by running chmod +x /.git/hooks/pre-commit
Now if you try to commit a binding pry it will reject the commit! Awesome!
Get rid of the "Yay! You're on Rails!" splash page
I like rails a whole bunch. But I don't need the world to know that when they go to my root url. For now let's just replace that with an empty JSON response.
Inside our routes.rb
file lets add a root:
root to: 'home#show'
And then add the following controller: app/controllers/home_controller.rb
with
class HomeController < ApplicationController
def show
render json: {}
end
end
Change the sent from email in devise mailers
You may have noticed the "[email protected]" address in the from field of our devise mailers. We can change that by specifying the mailer_sender
inside the devise config file:
Devise.setup do |config|
--snip--
config.mailer_sender = "[email protected]"
end