书要反复提及《哪里有天才》在说,大多数所谓的天才是通过反复刻意练习获得。

当你的练习时间达到10000几个小时后,。你将成为该领域的专家。

近期在学习rails怎样实现RESTful Web Service。

自己想给自己设计一个练习的模板,进行重复练习。

开发过程採用的是TDD的方式进行开发。

练习背景:

我们涉及三个Domain对象,Products、Orders、Payment

1.新建projectrails-rest-practice

rails new rails-rest-practice

cd !$

bundle install

2.安装rspec-rails

在Gemfile中加入

gem "rspec-rails", :group => [:development, :test]

然后

bundle install

rails g rspec:install

在.rspec 中去掉 --warnings

3.GET /products => user get list of products

step 1:创建controller,返回HTTP Status 200

user products api例如以下:

GET   /products       user get list of products

创建文件:spec/products/products_controller_spec.rb

require 'rails_helper'



describe ProductsController, :type => :controller do

   describe 'products controller' do

      it 'get index of products' do

         get :index

         expect(response).to have_http_status(200)

      end

   end

end

have_http_status:http://rubydoc.info/gems/rspec-rails/RSpec/Rails/Matchers#have_http_status-instance_method

创建文件:app/controllers/products_controller.rb

class ProductsController < ApplicationController

   def index

   end

end

执行rake spec。得到错误:

ActionController::UrlGenerationError:

       No route matches {:action=>"index", :controller=>"products"}

配置相关config/routes.rb

resources :products do

   collection do

      get :index

   end

end

执行rake spec,得到错误:

Failure/Error: get :index

     ActionView::MissingTemplate:

改动app/controllers/products_controller.rb

class ProductsController < ApplicationController

   def index

      render :nothing => true

   end

end

这样就完毕我们的第一个步骤。尽管看似这个步骤什么都没測,事实上不然,在这一步中。我们搭建好了routes。同一时候创建了必要的controller类和其相应的方法。

step 2:返回JSON

安装rabl

在Gemfile中加入rabl

gem 'rabl'

bundle install

參考Rails + rabl

改动測试:spec/products/products_controller_spec.rb

render_views



describe 'products controller' do

   before(:all) do

      @products = [

         Product.new({:id => 1, :name => 'apple juice', :description => 'good'}),

         Product.new({:id => 2, :name => 'banana juice', :description => 'just so so'})

      ]

   end



   it 'get index of products' do

      expect(Product).to receive(:all).and_return(@products).once



      get :index, {:format => :json}

      expect(response).to have_http_status(200)



      products_json = JSON.parse(response.body)

      expect(products_json.size).to eq(2)

   end

end

执行測试rake spec

得到错误:

NameError:

       uninitialized constant Product

创建model Product:

rails g model Product name:string description:text

rake db:migrate

执行測试rake spec

得到错误:

Failure/Error: products_json = JSON.parse(response.body)

     JSON::ParserError:

       A JSON text must at least contain two octets!

这是由于我们的response不正确,而且我们没有配置怎么获取json格式的输出。

创建文件: app/views/products/index.json.rabl

collection @products, :object_root => false

attributes :name

再次执行測试rake spec,測试通过

step3: 加入很多其它的字段

在 spec/products/products_controller_spec.rb中

products_json = JSON.parse(response.body)

expect(products_json.size).to eq(2)



expect(products_json[0]['id']).to eq(1)

expect(products_json[1]['id']).to eq(2)



expect(products_json[0]['name']).to eq('apple juice')

expect(products_json[1]['name']).to eq('banana juice')



expect(products_json[0]['description']).to eq('good')

expect(products_json[1]['description']).to eq('just so so')



expect(products_json[0]['uri']).to end_with('/products/1')

expect(products_json[1]['uri']).to end_with('/products/2')

在app/views/products/index.json.rabl中

collection @products, :object_root=>false

attributes :id, :name, :description



node :uri do |product|

   product_url product

end

4.GET /products => user get a product of specified id

step 1: 创建相应的controller方法。返回HTTP 200

加入測试:spec/products/products_controller_spec.rb

it 'get product by product id' do

   get :show, {:id => 1}

   expect(response).to have_http_status(200)

end

相应改动:app/controllers/products_controller.rb

def show

   render :nothing => true

end

相应改动:config/routes.rb

resources :products do

   collection do

      get :index

   end



   member do

      get :show

   end

end

rake spec測试通过

step 2:创建相应的JSON显示

加入測试:spec/products/products_controller_spec.rb

before(:all) do

   #... ...

   @product = Product.new({:id => 1, :name => 'apple juice', :description => 'good'})

end

it 'get product by product id' do

   expect(Product).to receive(:find).with(1).and_return(@product).once



   get :show, {:id => 1, :format => :json}

   expect(response).to have_http_status(200)



   product_json = JSON.parse(response.body)

   expect(product_json['id']).to eq(1)

   expect(product_json['name']).to eq('apple juice')

   expect(product_json['description']).to eq('good')

   expect(product_json['uri']).to end_with('/products/1')

end

相应改动:app/controllers/products_controller.rb

def show

   @product = Product.find(params[:id].to_i)

end

Q:params[:id].to_i,为什么这里从測试代码过来的params[:id]它使一个string类型呢

加入JSON显示:app/views/products/show.json.rabl

object false



node(:id) { |product| @product.id }

node(:name) { |product| @product.name }

node(:description) { |product| @product.description }

node(:uri) { |product| product_url @product }

执行測试,通过

step 3:重构rabl

改动app/views/products/show.json.rabl

object @product



attributes :id, :name, :description



node(:uri) { |product| product_url product }

改动app/views/products/index.json.rabl

collection @products



extends 'products/show'

配置rabl:创建文件config/initializers/rabl_config.rb

Rabl.configure do |config|

   config.include_json_root = false

end

执行測试,通过,这样降低了rabl间的反复代码

step 4:HTTP 404

加入測试:spec/products/products_controller_spec.rb

it 'get 404 when product not found' do

   expect(Product).to receive(:find).with(100).and_raise(ActiveRecord::RecordNotFound)



   get :show, {:id => 100, :format => :json}

   expect(response).to have_http_status(404)

end

相应改动:

class ProductsController < ApplicationController

   rescue_from ActiveRecord::RecordNotFound, with: :product_not_found



   #... ... 



   def show

      @product = Product.find(params[:id])

   end



   protected

      def product_not_found

         response.status = :not_found

      end

end

參考rescue_from

(更新中,欢迎不吝赐教)

将会要改动的部分是怎样笔试rspec,参考:http://betterspecs.org/

版权声明:本文博主原创文章,博客,未经同意不得转载。

有意练习--Rails RESTful(一)的更多相关文章

  1. 从0使用Ruby on Rails打造企业级RESTful API项目实战之我的云音乐

    本节对我们项目实现的功能和知识点做一个简单的介绍,因为是RESTful API项目,所以对于后端来说基本上没有什么UI界面可展示,那我们就在关键的点,使用客户端(Android)实现的效果图. 课程简 ...

  2. Rails ---> routes.rb 详解

    理解路由的目的 看懂routes.rb文件中的代码 使用经典的hash风格或者现在比较流行的Restful风格构造你自己的路径 断定一个路径会映射到哪一个controller和action 路由的双重 ...

  3. Rails : 产品环境(生产环境)的部署

    bundle install rails server (默认为开发环境) rails server -p80 -e production (指定为生产环境 ,并自定义指定站点端口) rake RAI ...

  4. RESTFUL接口

    原文地址:http://kb.cnblogs.com/page/512047/ 1. 什么是REST REST全称是Representational State Transfer,中文意思是表述(编者 ...

  5. Building a RESTful Web Service Using Spring Boot In Eclipse

    一.构建restful web service 创建Maven的java web工程,maven的pom文件加入依赖包 创建包hello Greeting.java package hello; pu ...

  6. RESTful架构详解(转)

    1. 什么是REST REST全称是Representational State Transfer,中文意思是表述(编者注:通常译为表征)性状态转移. 它首次出现在2000年Roy Fielding的 ...

  7. 虚拟研讨会:如何设计好的RESTful API?

    http://www.infoq.com/cn/articles/how-to-design-a-good-restful-api/ REST架构风格最初由Roy T. Fielding(HTTP/1 ...

  8. 使用 Struts 2 开发 RESTful 服务

    REST 简介 REST 是英文 Representational State Transfer 的缩写,这个术语由 Roy Thomas Fielding 博士在他的论文<Architectu ...

  9. 【转】REST on Rails指南

    REST on Rails指南1:理解资源 这是来自http://www.softiesonrails.com的REST简明指南. PART I 在理解REST on Rails之前,有必要先思考一下 ...

随机推荐

  1. SSH是什么?Linux如何修改SSH端口号?

    通过SSH连接可以远程管理Linux等设备,默认linuxssh端口是22端口,如何修改SSH默认端口,如何增加SSH端口呢?,下面小编给大家演示一下   工具/原料 Xshell   putty 等 ...

  2. 从零开始学Xamarin.Forms(三) Android 制作启动画面

    原文:从零开始学Xamarin.Forms(三) Android 制作启动画面     Xamarin.Forms 在启动的时候相当慢,必须添加一个启动界面,步骤如下: 1.将启动画面的图片命名为:s ...

  3. (hdu step 7.1.3)Lifting the Stone(求凸多边形的重心)

    题目: Lifting the Stone Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Other ...

  4. 什么是IT

    这个是同事总结的,我补充了若干项,算不上原创,但这个没有在其他地方看到,在这儿权且当原创了.后面再配个软件架构图吧.看到缺的同学能够补充 什么是IT:Information-信息Technology- ...

  5. 使用JustDecompile修改程序集

    原文:使用JustDecompile修改程序集        JustDecompile是Telerik公司推出一个免费的.net反编译工具,支持插件,与Visual Studio 2010,2012 ...

  6. 很实用的FTP操作类

    using System; using System.Net; using System.Net.Sockets; using System.Text; using System.IO; using ...

  7. 给节点设置tag【从零開始cocos3.0final 】

    在cocos中通过tag来管理节点是非经常常使用的:以下介绍一个关于在cocos中使用tag的实例: typedef enum{ tag1 }Tag; 这里能够使用枚举类型,来为多个节点设置tag: ...

  8. [java面试题]最长的回文字符串中出现确定

    <span style="font-family: Arial, Helvetica, sans-serif;">package com.wzw.util;</s ...

  9. Java 并发专题 :闭锁 CountDownLatch 之一家人一起吃个饭

    最近一直整并发这块东西,顺便写点Java并发的例子,给大家做个分享,也强化下自己记忆. 每天起早贪黑的上班,父母每天也要上班,话说今天定了个饭店,一家人一起吃个饭,通知大家下班去饭店集合.假设:3个人 ...

  10. java.nio分析软件包(三)---Charset理解力

    前面的分析后,2一个基本的封装类型.现在我们就来揭开Java.nio魔法知识的最后一块,CharsetEncoding类,他的主要功能是实现字节Unicode之间的转换转码. 让我们来看看他同样的封装 ...