Ruby on rails开发从头来(windows)(二十)-测试model(CRUD)
上篇随笔简单了解了rails的测试和测试用数据的使用,这次来看看怎样对一个model进行添删查改的测试。
1. 还是使用上次写的products_test.rb,修改test_turth方法的名字为test_create,并且使其内容为:
def test_create
assert_kind_of Product, @product
assert_equal 1, @product.id
assert_equal "Pragmatic Version Control", @product.title
assert_equal "How to use version control", @product.description
assert_equal "http://.../sk_svn_small.jpg", @product.image_url
assert_equal 29.95,@product.price
assert_equal "2005-01-26 00:00:00",
@product.date_available_before_type_cast
end
然后运行测试命令:depot>ruby test/unit/product_test.rb,屏幕上会显示信息:
Loaded suite test/unit/product_test
Started
F
Finished in 0.109 seconds.
1) Failure:
test_create(ProductTest) [test/unit/product_test.rb:16]:
<29> expected but was
<#<BigDecimal:4aad7b0,'0.2995E2',8(8)>>.
1 tests, 6 assertions, 1 failures, 0 errors
我们看到,是assert_equal 29.95,@product.price断言失败了。根据《Agile Web Development with Rails》里的内容,这句断言应该是正常通过的。但是不知道是不是版本或环境的问题,我自己写的时候总是不行。为了能够使断言通过,我们修改一下,把
assert_equal 29.95,@product.price
改为:assert_equal "29.95",@product.price_before_type_cast
我们看到了,product对象的每个属性都有对应的_before_type_cast版本,其内容是一个字符串。
现在再次运行测试命令,得到的结果如下:
Loaded suite test/unit/product_test
Started
.
Finished in 0.078 seconds.
1 tests, 7 assertions, 0 failures, 0 errors
从上面的测试中看到,我们在setup方法中,从数据库中查找了id为1的记录,然后在test_create方法中对其的属性逐个判断测试。
2. 对创建和读取的测试完成了,我们来进行对Update的测试,添加方法test_update,在这个方法里我们要对price字段进行更新,并通过断言判断是否成功更新:
def test_update
assert_equal "29.95", @product.price_before_type_cast
@product.price = 99.99
assert @product.save, @product.errors.full_messages.join("; ")
@product.reload
assert_equal "99.99", @product.price_before_type_cast
end
基于上面的test_create方法中对price测试的时候出现的问题,这里我们在断言里还是使用price的before_type_cast版本。
再次运行测试命令,屏幕上的输出如下:
Loaded suite test/unit/product_test
Started
..
Finished in 0.078 seconds.
2 tests, 10 assertions, 0 failures, 0 errors
证明所有的断言都成功了,现在到数据库里看看,id为1的product的price字段已经更新为99.99了。
3. 最后我们来测试删除,添加test_destroy方法,内容如下:
def test_destroy
@product.destroy
assert_raise(ActiveRecord::RecordNotFound) { Product.find(@product.id) }
end
运行测试命令,屏幕上输出如下:
Loaded suite test/unit/product_test
Started
..E
Finished in 0.078 seconds.
1) Error:
test_update(ProductTest):
ActiveRecord::RecordNotFound: Couldn't find Product with ID=1
……
3 tests, 8 assertions, 0 failures, 1 errors
上面的测试里,先删除掉id为1的记录,然后使用断言,如果失败,就抛出一个ActiveRecord::RecordNotFound的错误。
写到这里,从上面的信息里显示是test_update方法中显示了异常信息,按照直觉,应该是test_destroy方法才对,这个问题现在自己也还没有搞明白,还请rails高人指点。