数据库Recipe
没有数据库的rails
问题描述
就像所有“偏执的软件”一样,Rails一开始就假定你在做一个数据库应用程序的开发。出现这个假定是因为--通常情况都是这样的。但是,当你开发一个基于文件做后台的程序时,或者是为一个外部API提供前台展现时,Rails都显得不那么友好,特别是在做测试时。
解决方案
默认的情况下,Rails假定你运行任何测试的时候都要连接并初始化一个数据库。这就意味着如果你没有数据库的话,测试就变得难以进行。当然,你可以仅仅创建一个没有任何内容的空库,但这也意味着你有了一堆没有意义的基础架构。对于已经生成的Rails程序,有一些小技巧(hacking)可以使它们变成可测试的样子。
为了保持事情简单和可重复,我们将从一个新的程序开始。这样你就能容易的把我们在这里做的东西放到你的程序里边。让我们现在就开始创建这个程序,你可以叫它任意你喜欢的名字,我给它的命名是“DatabaselessApplicatio”。
然后我们在lib中创建个简单的class来写一些测试。让我们创建这个叫Adder的类,它可以使两个数加起来(真的非常容易)。
DatabaselessApplication/lib/adder.rb
class Adder
def initialize(first,second)
@first=first
@second=second
end
def sum
@first+@second
end
end
现在我们来为它创建一个测试用例,也就是test/unit/adder_test.rb:
DatabaselessApplication/test/unit/adder_test.rb
require File.join(File.dirname(__FILE__), "..", "test_helper")
require ' adder'
class AdderTest
def test_simple_addition
assert_equal(4,Adder.new(3,1).sum)
end
end
让我们运行这个测试:
chad> raketest_units
(in/Users/chad/src/FR_RR/Book/code/DatabaselessApplication)
Rake aborted!
#42000Unknown database 'databaselessapplication_development'
看上去Rails的test_units()这个RakeTask做了些数据初始化的工作。事实上,用rake-P 可以证明这一点:
chad> rake-P|tail
prepare_test_database
rakestats
raketest_functional
prepare_test_database
raketest_plugins
environment
raketest_units
prepare_test_database
rakeunfreeze_rails
rakeupdate_javascripts
足够确定test_units()是依赖于prepare_test_database() 这个task的。那我们不通过RakeTest直接运行这个测试会怎么样呢?
chad> rubytest/unit/adder_test.rb
Loaded suite test/unit/adder_test
Started
EE
Finished in 0.052262 seconds.
1)Error:
test_simple_addition(AdderTest):
Mysql::Error:
#42000Unknowndatabase ' databaselessapplication_test'
(abbreviated)
这些输出的结果说明了,因为在运行链路的某个地方引入了test_helper.rb,使得我们继承了数据库为中心的setup()和teardown()方法。我们可以在测试中仅仅手工指定是否需要”test/unit”,但是这样在我们创建的每一个测试里都会有重复这个指定。而且我们还会发现这样将不能初始化Rails环境。所以,我们采用修改test_helper.rb的方法来替代。