提问



当我加载script/console时,有时我想要使用控制器的输出或视图助手方法。


有办法:



  • 模拟请求?

  • 根据上述请求从控制器实例调用方法?

  • 通过所述控制器实例或其他方式测试助手方法?


最佳参考


要调用助手,请使用helper对象:


$ ./script/console
>> helper.number_to_currency('123.45')
=> "R$ 123,45"


如果你想使用一个默认不包含的帮助器(比如说,因为你从ApplicationController中删除了helper :all),只需要包含帮助器。


>> include BogusHelper
>> helper.bogus
=> "bogus output"


至于处理控制器,我引用尼克的回答:



> app.get '/posts/1'
> response = app.response
# you now have a rails response object much like the integration tests

> response.body            # get you the HTML
> response.cookies         # hash of the cookies

# etc, etc


其它参考1


从脚本/控制台调用控制器操作并查看/操作响应对象的简单方法是:


> app.get '/posts/1'
> response = app.response
# you now have a rails response object much like the integration tests

> response.body            # get you the HTML
> response.cookies         # hash of the cookies

# etc, etc


app对象是ActionController :: Integration :: Session [41]的一个实例


这适用于我使用Rails 2.1和2.3,我没有尝试早期版本。

其它参考2


如果您需要从控制台进行测试(在Rails 3.1和4.1上测试):


呼叫控制器操作:


app.get '/'              
   app.response            
   app.response.headers  # => { "Content-Type"=>"text/html", ... }
   app.response.body     # => "<!DOCTYPE html>\n<html>\n\n<head>\n..." 


ApplicationController方法:


foo = ActionController::Base::ApplicationController.new
foo.public_methods(true||false).sort
foo.some_method 


路线助手:


app.myresource_path     # => "/myresource" 
app.myresource_url      # => "http://www.example.com/myresource"


查看助手:


foo = ActionView::Base.new

foo.javascript_include_tag 'myscript' #=> "<script src=\"/javascripts/myscript.js\"></script>"

helper.link_to "foo", "bar" #=> "<a href=\"bar\">foo</a>"

ActionController::Base.helpers.image_tag('logo.png')  #=> "<img alt=\"Logo\" src=\"/images/logo.png\" />"


渲染:


views = Rails::Application::Configuration.new(Rails.root).paths["app/views"]
views_helper = ActionView::Base.new views
views_helper.render 'myview/mytemplate'
views_helper.render file: 'myview/_mypartial', locals: {my_var: "display:block;"}
views_helper.assets_prefix  #=> '/assets'


ActiveSupport方法:


require 'active_support/all'
1.week.ago
=> 2013-08-31 10:07:26 -0300
a = {'a'=>123}
a.symbolize_keys
=> {:a=>123}


Lib模块:


> require 'my_utils'
 => true 
> include MyUtils
 => Object 
> MyUtils.say "hi"
evaluate: hi
 => true 

其它参考3


这是通过控制台执行此操作的一种方法:


>> foo = ActionView::Base.new
=> #<ActionView::Base:0x2aaab0ac2af8 @assigns_added=nil, @assigns={}, @helpers=#<ActionView::Base::ProxyModule:0x2aaab0ac2a58>, @controller=nil, @view_paths=[]>

>> foo.extend YourHelperModule
=> #<ActionView::Base:0x2aaab0ac2af8 @assigns_added=nil, @assigns={}, @helpers=#<ActionView::Base::ProxyModule:0x2aaab0ac2a58>, @controller=nil, @view_paths=[]>

>> foo.your_helper_method(args)
=> "<html>created by your helper</html>"


创建ActionView::Base的新实例使您可以访问助手可能使用的普通视图方法。然后扩展YourHelperModule将其方法混合到您的对象中,让您查看它们的返回值。

其它参考4


另一种方法是使用rails调试器。在http://guides.rubyonrails.org/debugging_rails_applications.html [42]上有一个关于调试的Rails指南]]


基本上,使用-u选项启动服务器:


./script/server -u


然后在您希望有权访问控制器/帮助程序/等的脚本中插入断点。


class EventsController < ApplicationController
  def index
    debugger
  end
end


当您发出请求并在代码中点击该部分时,服务器控制台将返回一个提示,您可以从命令提示符处创建请求,查看对象等。完成后,只需输入cont继续执行。还有扩展调试的选项,但这至少应该让你开始。

其它参考5


如果方法是POST方法那么


app.post 'controller/action?parameter1=value1&parameter2=value2'


[[此处参数将根据您的适用性]]


否则如果是GET方法那么


app.get 'controller/action'

其它参考6


您可以在Rails控制台中访问您的方法,如下所示


controller.method_name
helper.method_name

其它参考7


以下是使用Refinery作为示例进行经过身份验证的POST请求的方法:


# Start Rails console
rails console
# Get the login form
app.get '/community_members/sign_in'
# View the session
app.session.to_hash
# Copy the CSRF token "_csrf_token" and place it in the login request.
# Log in from the console to create a session
app.post '/community_members/login', {"authenticity_token"=>"gT7G17RNFaWUDLC6PJGapwHk/OEyYfI1V8yrlg0lHpM=",  "refinery_user[login]"=>'chloe', 'refinery_user[password]'=>'test'}
# View the session to verify CSRF token is the same
app.session.to_hash
# Copy the CSRF token "_csrf_token" and place it in the request. It's best to edit this in Notepad++
app.post '/refinery/blog/posts', {"authenticity_token"=>"gT7G17RNFaWUDLC6PJGapwHk/OEyYfI1V8yrlg0lHpM=", "switch_locale"=>"en", "post"=>{"title"=>"Test", "homepage"=>"0", "featured"=>"0", "magazine"=>"0", "refinery_category_ids"=>["1282"], "body"=>"Tests do a body good.", "custom_teaser"=>"", "draft"=>"0", "tag_list"=>"", "published_at(1i)"=>"2014", "published_at(2i)"=>"5", "published_at(3i)"=>"27", "published_at(4i)"=>"21", "published_at(5i)"=>"20", "custom_url"=>"", "source_url_title"=>"", "source_url"=>"", "user_id"=>"56", "browser_title"=>"", "meta_description"=>""}, "continue_editing"=>"false", "locale"=>:en}


如果出现错误,您可能会发现这些有用:


app.cookies.to_hash
app.flash.to_hash
app.response # long, raw, HTML

其它参考8


在rails 3中,试试这个:


session = ActionDispatch::Integration::Session.new(Rails.application)
session.get(url)
body = session.response.body


正文将包含网址的HTML。


如何在Rails 3 中从模型路由和渲染(调度)

其它参考9


之前的答案是调用助手,但以下内容将有助于调用控制器方法。我在rails 2.3.2上使用过它。


首先将以下代码添加到.irbrc文件(可以在您的主目录中)


class Object
   def request(options = {})
     url=app.url_for(options)
     app.get(url)
     puts app.html_document.root.to_s    
  end
end


然后在rails控制台中你可以键入类似......


request(:controller => :show, :action => :show_frontpage)


...并且html将被转储到控制台。

其它参考10


在rails控制台中进行Helper方法测试的一个可能方法是


Struct.new(:t).extend(YourHelper).your_method(*arg)


并重新加载和做


reload!; Struct.new(:t).extend(YourHelper).your_method(*arg)

其它参考11


在任何控制器操作或视图中,您可以通过调用控制台方法来调用控制台。


例如,在控制器中:


class PostsController < ApplicationController
  def new
    console
    @post = Post.new
  end
end


或者在视图中:


<% console %>

<h2>New Post</h2>


这将在视图中呈现控制台。您不需要关心控制台调用的位置;它不会在其调用的位置呈现,而是在HTML内容旁边呈现。


见:http://guides.rubyonrails.org/debugging_rails_applications.html [44]

其它参考12


如果你已经添加了自己的帮助器,并且希望它的方法在控制台中可用,请执行以下操作:



  1. 在控制台中执行include YourHelperName

  2. 您的帮助方法现在可以在控制台中使用,在控制台中使用它们调用method_name(args)



示例:假设您在app/helpers/my_helper.rb中有MyHelper(使用方法my_method),然后在控制台中执行:



  1. include MyHelper

  2. my_helper.my_method