Skip to content

20 API Introduction

Dave Strus edited this page May 18, 2016 · 2 revisions

Creating a REST API with JSON

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.