在线时间:8:00-16:00
迪恩网络APP
随时随地掌握行业动态
扫描二维码
关注迪恩网络微信公众号
上次测试Modeul的问题还没有解决,但是下面的还要继续,这次来测试Controller。
1. 在test\functional目录下,rails已经为我们的controller生成了对应的测试文件,要注意application_controller不会生成测试文件。我们以控制登录的LoginController为例,打开login_controller_test.rb,内容如下: require File.dirname(__FILE__) + '/../test_helper' require 'login_controller' # Re-raise errors caught by the controller. class LoginController; def rescue_action(e) raise e end; end class LoginControllerTest < Test::Unit::TestCase def setup @controller = LoginController.new @request = ActionController::TestRequest.new @response = ActionController::TestResponse.new end # Replace this with your real tests. def test_truth assert true end end 我们看到,在setup方法里,定义了三个对象@controller和@request和@response,这样,我们就可以在不接入webserver或network的情况下进行测试了。 2. 我们来把其中的test_truth方法替换成下面的代码: def test_index get :index assert_response :success end 其中,get方法模拟发出一个web请求,请求的action是index,并且捕捉响应(response),然后由assert_response断言来判断响应是否成功。 现在运行测试:depot>ruby test/functional/login_controller_test.rb 会看到测试失败了,命令行的输出: Expected response to be a <:success>, but was <302> 1 tests, 1 assertions, 1 failures, 0 errors 为什么会这样呢?回想一下,我们在前面的程序里,在访问index页面时,要先判断用户是不是管理员。如果不是,就要跳转到login页面,在login_controller.rb里: before_filter :authorize, :except => :login 在application.rb文件里,有authorize方法: def authorize unless session[:user_id] flash[:notice] = "Please log in" redirect_to(:controller => "login", :action => "login") end end 3. 好了,知道了原因,现在再写一个测试: def test_index_without_user get :index assert_redirected_to :action => "login" assert_equal "Please log in", flash[:notice] end 上面的代码里,我们尝试请求index这个action,并且使用断言判断是否重定向到login,再判断flash[:notice]里的内容。 再次运行测试,命令行的输出如下: Loaded suite test/functional/login_controller_test Started . Finished in 0.062 seconds. 1 tests, 3 assertions, 0 failures, 0 errors 可以看到所有的断言都通过了。 但是还有一个问题,就是根据代码,我们只使用了两个断言,但是提示信息却显示有三个断言,这个问题是什么原因还不太清楚,但是现在并不影响我们继续下面的内容。 4. 在我们的login页面上,用户输入名字和密码后,点击login按钮,这些信息将会发送个login这个action,然后创建一个User的实例来让用户登入。 在login_controller.rb里的login方法里: @user = User.new(params[:user]) logged_in_user = @user.try_to_login 现在,我们来测试login这个action,在login_controller_test.rb中添加方法: def test_login_with_invalid_user post :login, :user => {:name => 'fred', :password => 'opensesame'} assert_response :success assert_equal "Invalid user/password combination", flash[:notice] end 大家一定看到了,这是使用了一个不存在的用户名和密码来测试login。 运行测试,输出结果如下: Finished in 0.609 seconds. 2 tests, 5 assertions, 0 failures, 0 errors 和上一个测试方法相对的,这里代码里是两个断言,根据输出也是两个,挺奇怪的事情。 5. 测试了不存在的用户,现在我们来测试用户存在的情况。 修改fixtures目录下的users.yml文件,我们还使用上面的用户名和密码: # Read about fixtures at http://ar.rubyonrails.org/classes/Fixtures.html fred: id: 1 name: fred hashed_password: <%= Digest::SHA1.hexdigest('abracadabra') %> 然后给login_controller_test.rb添加fixtures :users,指定使用users.yml文件。 再添加方法: def test_login_with_valid_user post :login, :user => {:name => 'fred', :password => 'abracadabra'} assert_redirected_to :action => "index" assert_not_nil(session[:user_id]) user = User.find(session[:user_id]) assert_equal 'fred', user.name end 在这里方法里,我们指定了和yml文件中定义的用户名相一致的数据,并且判断是否定位到index,再判断用户的id是否已经保存在session中。 根据前面测试Model时候的内容,我们在yml文件中定义的数据会被加入到数据库中,这样,这个方法里的三个断言都应该通过才对。 OK了,运行测试,输出结果: Finished in 0.125 seconds. 3 tests, 8 assertions, 0 failures, 0 errors 嗯,所有的断言都通过了。 好了,这次就到这里。 |
2023-10-27
2022-08-15
2022-08-17
2022-09-23
2022-08-13
请发表评论