本章主要介绍了如何对功能进行验证和单元测试。这两项工作的目的,主要是为了让模型更加坚固,以确保数据中发生的错误永远都不会提交给数据库。
验证
应该把验证代码放置在模型类中
因为模型层扮演了代码和数据库之间的看门人的角色。那些应用程序要用到的数据,无论是从数据库中读取的,还是要往数据库中写入的,都首先会进入到模型层。所以,只要把验证代码放在模型类中,在把数据写入到数据库中之前,对数据进行了验证,就可以避免坏数据对数据库造成破坏。
验证代码使用validates方法
就之前的创建商品的例子而言,我们只需要往模型类Product类的源码app/models/products.rb中,插入以下代码,就可以在使用页面上的功能时获得相应的验证信息:
class Product < ApplicationRecord
validates :title, :description, :image_url, presence: true
validates :price, numericality: {
greater_than_or_equal_to: 0.01}
validates :title, uniqueness: true
validates :image_url, allow_blank: true, format:{
with: %r{\.(gif|jpg|png)\Z}i,
message: 'must be a URL for GIF, JPG or PNG image.'
}
end
validates
这个方法是标准的Rails验证器,它可以根据一个或多个条件来验证模型的一个或多个字段。presence: true
是让验证器核实每个已命名的字段都必须存在,而且内容不能为空。numericality
是用来判断输入的数字是不是有效的数字。greater_than_or_equal_to
选项的值设为0.01而不是0,是因为数据库只能存储小数点后的两位数字,如果将这个比较值设为0,当输入0.001时,数据库也将把0.001保存为0,从而导致功能上的错误。uniqueness: true
是对字段进行唯一性的验证。format
选项可以判断字段是否和正则表达式相匹配。allow_blank: true
书中表示该选项可以允许某个字段为空(但本人实测无效)
用rails test命令进行简单的测试
利用rails test
命令可以测试一下刚才加入的那些验证代码,是否会给整个程序的代码带来一个错误。
在运行后该命令后,terminal报出了两个错误,分别是在test/controllers/products_controller_test.rb文件里的should create product处和should update product处。这是因为我们在Product模型类中增加了上面的验证方法后,还没有增减相应的测试数据,所以暂时还无法执行测试脚本中,与“创建商品”功能和“更新商品”功能相关的这两部分测试代码。在products_controller_test.rb中增加具体的测试数据后,便可解决掉这个问题。增加测试数据后的products_controller_test.rb代码如下:
require 'test_helper'
class ProductsControllerTest < ActionDispatch::IntegrationTest
setup do
@product = products(:one)
@update = {
title: 'Lorem Ipsum',
description: 'Wibbles are fun!',
image_url: 'lorem.jpg',
price: 19.95
}
end
test "should get index" do
get products_url
assert_res