-
Notifications
You must be signed in to change notification settings - Fork 0
20 API Introduction
We want to be able to get at our notes programmatically through other applications—say, a different web app or a mobile app. Rails gives you all the pieces you need to do that with a minimum of pain.
Because Rails encourages use of a RESTful pattern for resource-based controllers with standardized CRUD actions, it makes it easy to accept & return JSON, or even XML.
Looking back at Mutant Registration, we can see that the scaffold generator takes advantage of this built-in functionality. If you were to make requests to http://localhost:3000/mutants.json, you should see a JSON response in your browser.
If viewing JSON in a web browser is something you do often, try installing JSON Formatter for Chrome.
If you take a look back at the scaffolded mutants_controller.rb
in Mutant Registration, you'll notice that the create
, update
, and destroy
actions each have a respond_to
block, and some of the actions have .json.jbuilder
views.
Let's take a look at the source in Mutant Registration for a minute to see what it's doing.
app/controllers/mutants_controller.rb
class MutantsController < ApplicationController
before_action :set_mutant, only: [:show, :edit, :update, :destroy]
# GET /mutants
# GET /mutants.json
def index
@mutants = Mutant.all
end
# GET /mutants/1
# GET /mutants/1.json
def show
end
# GET /mutants/new
def new
@mutant = Mutant.new
end
# GET /mutants/1/edit
def edit
end
# POST /mutants
# POST /mutants.json
def create
@mutant = Mutant.new(mutant_params)
respond_to do |format|
if @mutant.save
format.html { redirect_to @mutant, notice: 'Mutant was successfully created.' }
format.json { render :show, status: :created, location: @mutant }
else
format.html { render :new }
format.json { render json: @mutant.errors, status: :unprocessable_entity }
end
end
end
# PATCH/PUT /mutants/1
# PATCH/PUT /mutants/1.json
def update
respond_to do |format|
if @mutant.update(mutant_params)
format.html { redirect_to @mutant, notice: 'Mutant was successfully updated.' }
format.json { render :show, status: :ok, location: @mutant }
else
format.html { render :edit }
format.json { render json: @mutant.errors, status: :unprocessable_entity }
end
end
end
# DELETE /mutants/1
# DELETE /mutants/1.json
def destroy
@mutant.destroy
respond_to do |format|
format.html { redirect_to mutants_url, notice: 'Mutant was successfully destroyed.' }
format.json { head :no_content }
end
end
private
# Use callbacks to share common setup or constraints between actions.
def set_mutant
@mutant = Mutant.find(params[:id])
end
# Never trust parameters from the scary internet, only allow the white list through.
def mutant_params
params.require(:mutant).permit(:name, :power, :age, :real_name)
end
end
It's great to be able to have different responses to the same controller action/route, but if you need to have a standardized interface for third-party integrations, then it's best not to have API code mixed in with your web application code.
As we go through our codebase to make future changes and enhancements, it becomes hard to keep ourselves from breaking or changing our API. Even if we manage (with help from automated tests) never to break that API, we can still end up with little islands of code we don't like that must be there just to avoid breaking the API.
We're not going to follow the scaffolded example when writing the API for Nevernote.