`
darkbaby123
  • 浏览: 103289 次
  • 性别: Icon_minigender_1
  • 来自: 武汉
社区版块
存档分类
最新评论

如何在测试代码中设置子域名,一点小心得

阅读更多

事情的起因:前段时间闲赋在家(这段时间也是……),想练练手。觉得JavaEye的子域名挺有意思,就想仿照做个博客,把子域名用进去。子域名插件用SubdomainFu。结果代码写好了,测试时发现问题了。

 

我要写一个before_filter方法,在进入控制器之前校验一下子域名。方法名叫check_subdomain,因为比较通用,放在application.rb中。

先看控制器application的代码

# RAILS_ROOT/app/controllers/application.rb
class ApplicationController < ActionController::Base
  # 为了简洁,其他代码都删了,只留了关键部分

  # 这个设置可以使session跨域,但只能是 anysubdomain.myblog.com
  session :session_domain => "myblog.com"

  include AuthenticatedSystem

  before_filter :login_required

  protected

  # 对用户的后台管理页面,检查子域名必须和用户login相同,否则跳转到404页面
  def check_subdomain
    unless current_user && current_user.login == current_subdomain
      render_404
      return false
    end
  end

  # 简陋的处理,返回404页面
  def render_404
    render :file => "#{RAILS_ROOT}/public/404.html", :status => 404
  end
end

上面的check_subdomain方法要在个人的博客管理控制台相关的控制器中,作为before_filter使用。比如,在categories控制器中,它处理“分类管理”的功能。

# RAILS_ROOT/app/controllers/categories_controller.rb
class CategoriesController < ApplicationController
  # 这里做一个校验
  before_filter :check_subdomain

  # 相信有博客的朋友都会熟悉这个url吧,这是“分类管理”
  # GET /admin/categories
  def index
    @categories = current_user.categories.ordered
  end
end

这个逻辑很简单,在进入“分类管理” 页面之前,做个校验,判断子域名和当前用户的login相同。校验通过后,根据当前用户,找到所有分类,然后显示页面。

 

好了,现在我想测试CategoriesController的index方法,这需要使用功能测试(Functional Test),代码如下:

# RAILS_ROOT/test/functional/categories_controller_test.rb
class CategoriesControllerTest < ActionController::TestCase
  fixtures :users, :categories

  def setup
    # 假设当前用户为alex
    @alex = users(:alex)
  end

  test "index" do
    # 这里设置下session,以便通过登录校验
    get :index, {}, :user_id => @alex.id
    assert_response :success
    assert_template "index"
  end
end

本该显示index页面,但是因为check_subdomain,跳转到404页面去了

ruby -I test test/functional/categories_controller_test.rb -n test_index
...
Expected response to be a <:success>, but was <404>

错误原因在于使用get方法发起提交时,默认是没有子域名的,而且get方法中也没有办法设置url。找了很多地方,最后在一个国外帖子里发现了解决办法。就是用 request.host 来设置域名。

 

改进的测试代码如下:

# RAILS_ROOT/test/functional/categories_controller_test.rb
class CategoriesControllerTest < ActionController::TestCase

  def setup
    @alex = users(:alex)
    # 利用Request对象的host属性来设定子域名,奇怪的是Rails API中没说明host是个属性,也没找到host=这个方法……
    #
    # 敏捷开发3rd居然没有说可以用@request,@controller之类的实例变量
    # 如果哪位知道这方面的介绍,还请告之,谢谢。
    @request.host = "#{@alex.login}.myblog.com"
  end

  # 测试是否可以正确获取子域名
  test "subdomain" do
    get :index
    assert_equal @alex.login, @controller.send(:current_subdomain)
  end

  test "index" do
    get :index, {}, :user_id => @alex.id
    assert_response :success
    assert_template "index"
  end
end

改进后测试通过,一切正常。

 

另外,子域名的问题在集成测试(Integration Test)中也有,但集成测试中设置子域名是一件很简单的事情,代码如下:

class AddBlogTest < ActionController::IntegrationTest
  fixtures :all

  # 测试子域名
  test "subdomain" do
    # 第一种方法,指定host,host!方法是集成测试专有的方法,功能测试则没有提供
    host! "alex.myblog.com"
    get "/admin/blogs"

    # 第二种方法,在url中加入子域名 
    # get "http://alex.myblog.com/admin/blogs"

    assert_equal "alex", controller.send(:current_subdomain)
  end
end

顺便补上Rspec版的controller测试代码,个人觉得Rspec换汤不换药,但写法倒是很舒服:

# RAILS_ROOT/spec/controllers/categories_controller_spec.rb
require 'spec_helper'

describe CategoriesController do
  fixtures :users

  before :each do
    @alex = users(:alex)
    # 和Rails的功能测试一样,只是Rspec中用request而非@request
    request.host = "#{@alex.login}.myblog.com"
  end

  it "should tell you the subdomain" do
    controller.send(:current_subdomain).should == @alex.login
  end
end

 

参考资料

Testing subdomains as account keys

How to simulate a login into a specific subdomain

分享到:
评论
5 楼 darkbaby123 2010-02-27  
笨笨狗 写道
抛砖引玉:)

其实我想表述的意思是,应该严格按照TDD的流程来实施开发过程……


受教了。原来做项目时测试用的不多,只在关键地方用过单元测试和性能测试。最近才开始学Rspec。看来我还是欠缺一些测试的思路。
Cucumber大概看了一下,很有创意的东西啊。不过个人觉得,其最大的价值在于降低和业务人员的沟通成本,就看能降低到什么程度了。相对的开发人员的学习成本有所增加,不过碰到这种有趣的点子,喜欢编程的人应该很乐意研究吧。
4 楼 llfzy 2010-02-26  
好久也学学这种语言!
3 楼 笨笨狗 2010-02-25  
抛砖引玉:)

其实我想表述的意思是,应该严格按照TDD的流程来实施开发过程,主贴里的例子,实际上应该先写测试,再来实现。在设计测试用例的时候,开发人员不需要也不应该考虑太多外部依赖,每次只需要关注目前需要解决的问题,也就是说,步进越小越合适。比如,你在构思check_subdomain这个filter的时候,可以不用去考虑如何实现current_subdomain,此时就可以借助mock来模拟依赖的外部接口:
controller.should_receive(:current_subdomain).and_return('ooxx')
这样能分离关注点,而且不需要依赖某个严格的代码编写顺序(此时都不需要去实现current_subdomain)。

上面说的主要针对传统的“单元测试”,如果需要做集成测试或者是功能测试,可以利用webrat等方便的gem来实现(就这个例子来说,可以直接去请求某个真实的子域名),此时引入cucumber就是再合适不过了:)
2 楼 darkbaby123 2010-02-25  
笨笨狗 写道
你的测试代码牵涉了太多的外部环境,其实可以不用这么麻烦的。以Rspec风格为例,当设计一个功能时,要尽量隔离现实依赖,你可以这么写:……

btw,那几个以@开头的实例变量,已经不赞成使用了,具体的介绍可以参看《the rails way》,good luck!

确实,你这段话提醒了我,原来Mock还可以这样用
1 楼 笨笨狗 2010-02-25  
你的测试代码牵涉了太多的外部环境,其实可以不用这么麻烦的。以Rspec风格为例,当设计一个功能时,要尽量隔离现实依赖,你可以这么写:

# RAILS_ROOT/spec/controllers/categories_controller_spec.rb
require 'spec_helper'

describe CategoriesController do
  #fixtures :users  我不喜欢用fixtues这种笨重的东西,你可以用Factory或者mock来代替

  before :each do
    @alex = mock_model(User, :login=>'alex')
    #如果你想验证算法,可以修改stub!为should_receive  
    request.stub!(:host).and_return("#{@alex.login}.myblog.com")
  end

  it "should tell you the subdomain" do
    #我不喜欢在测试代码里见到大堆的send调用非public的hack,你可以用hide_action :current_subdomain来实现非action公有方法
    controller.current_subdomain.should == @alex.login
  end
end


btw,那几个以@开头的实例变量,已经不赞成使用了,具体的介绍可以参看《the rails way》,good luck!

相关推荐

Global site tag (gtag.js) - Google Analytics