mirror of
https://github.com/github/rails.git
synced 2026-01-28 07:48:00 -05:00
merge with local tweaks
This commit is contained in:
@@ -426,7 +426,7 @@ module ActionMailer #:nodoc:
|
||||
end
|
||||
|
||||
def template_root=(root)
|
||||
write_inheritable_attribute(:template_root, ActionView::ViewLoadPaths.new(Array(root)))
|
||||
write_inheritable_attribute(:template_root, ActionView::PathSet.new(Array(root)))
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -32,8 +32,7 @@ end
|
||||
|
||||
# Wrap tests that use Mocha and skip if unavailable.
|
||||
def uses_mocha(test_name)
|
||||
gem 'mocha', ">=0.5"
|
||||
require 'stubba'
|
||||
gem 'mocha', ">=0.9.0"
|
||||
yield
|
||||
rescue Gem::LoadError
|
||||
$stderr.puts "Skipping #{test_name} tests (Mocha >= 0.5 is required). `gem install mocha` and try again."
|
||||
|
||||
@@ -1,14 +1,25 @@
|
||||
*Edge*
|
||||
|
||||
* Disable the Accept header by default [Michael Koziarski]
|
||||
* Set config.action_view.warn_cache_misses = true to receive a warning if you perform an action that results in an expensive disk operation that could be cached [Josh Peek]
|
||||
|
||||
* Refactor template preloading. New abstractions include Renderable mixins and a refactored Template class [Josh Peek]
|
||||
|
||||
* Changed ActionView::TemplateHandler#render API method signature to render(template, local_assigns = {}) [Josh Peek]
|
||||
|
||||
* Changed PrototypeHelper#submit_to_remote to PrototypeHelper#button_to_remote to stay consistent with link_to_remote (submit_to_remote still works as an alias) #8994 [clemens]
|
||||
|
||||
* Add :recursive option to javascript_include_tag and stylesheet_link_tag to be used along with :all. #480 [Damian Janowski]
|
||||
|
||||
* Allow users to disable the use of the Accept header [Michael Koziarski]
|
||||
|
||||
The accept header is poorly implemented by browsers and causes strange
|
||||
errors when used on public sites where crawlers make requests too. You
|
||||
should use formatted urls (e.g. /people/1.xml) to support API clients.
|
||||
can use formatted urls (e.g. /people/1.xml) to support API clients in a
|
||||
much simpler way.
|
||||
|
||||
Alternatively to re-enable it you need to set:
|
||||
To disable the header you need to set:
|
||||
|
||||
config.action_controller.use_accept_header = true
|
||||
config.action_controller.use_accept_header = false
|
||||
|
||||
* Do not stat template files in production mode before rendering. You will no longer be able to modify templates in production mode without restarting the server [Josh Peek]
|
||||
|
||||
|
||||
@@ -63,11 +63,18 @@ module ActionController
|
||||
clean_backtrace do
|
||||
assert_response(:redirect, message)
|
||||
return true if options == @response.redirected_to
|
||||
|
||||
# Support partial arguments for hash redirections
|
||||
if options.is_a?(Hash) && @response.redirected_to.is_a?(Hash)
|
||||
return true if options.all? {|(key, value)| @response.redirected_to[key] == value}
|
||||
end
|
||||
|
||||
redirected_to_after_normalisation = normalize_argument_to_redirection(@response.redirected_to)
|
||||
options_after_normalisation = normalize_argument_to_redirection(options)
|
||||
|
||||
assert_equal options_after_normalisation, redirected_to_after_normalisation,
|
||||
"Expected response to be a redirect to <#{options_after_normalisation}> but was a redirect to <#{redirected_to_after_normalisation}>"
|
||||
if redirected_to_after_normalisation != options_after_normalisation
|
||||
flunk "Expected response to be a redirect to <#{options_after_normalisation}> but was a redirect to <#{redirected_to_after_normalisation}>"
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -80,13 +87,13 @@ module ActionController
|
||||
#
|
||||
def assert_template(expected = nil, message=nil)
|
||||
clean_backtrace do
|
||||
rendered = expected ? @response.rendered_file(!expected.include?('/')) : @response.rendered_file
|
||||
rendered = @response.rendered_template
|
||||
msg = build_message(message, "expecting <?> but rendering with <?>", expected, rendered)
|
||||
assert_block(msg) do
|
||||
if expected.nil?
|
||||
!@response.rendered_with_file?
|
||||
@response.rendered_template.nil?
|
||||
else
|
||||
expected == rendered
|
||||
rendered.to_s.match(expected)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -343,12 +343,12 @@ module ActionController #:nodoc:
|
||||
# Indicates whether the response format should be determined by examining the Accept HTTP header,
|
||||
# or by using the simpler params + ajax rules.
|
||||
#
|
||||
# If this is set to +true+ then +respond_to+ and +Request#format+ will take the Accept header into
|
||||
# account. If it is set to false (the default) then the request format will be determined solely
|
||||
# If this is set to +true+ (the default) then +respond_to+ and +Request#format+ will take the Accept
|
||||
# header into account. If it is set to false then the request format will be determined solely
|
||||
# by examining params[:format]. If params format is missing, the format will be either HTML or
|
||||
# Javascript depending on whether the request is an AJAX request.
|
||||
cattr_accessor :use_accept_header
|
||||
self.use_accept_header = false
|
||||
self.use_accept_header = true
|
||||
|
||||
# Controls whether request forgergy protection is turned on or not. Turned off by default only in test mode.
|
||||
class_inheritable_accessor :allow_forgery_protection
|
||||
@@ -412,7 +412,7 @@ module ActionController #:nodoc:
|
||||
# More methods can be hidden using <tt>hide_actions</tt>.
|
||||
def hidden_actions
|
||||
unless read_inheritable_attribute(:hidden_actions)
|
||||
write_inheritable_attribute(:hidden_actions, ActionController::Base.public_instance_methods.map(&:to_s))
|
||||
write_inheritable_attribute(:hidden_actions, ActionController::Base.public_instance_methods.map { |m| m.to_s })
|
||||
end
|
||||
|
||||
read_inheritable_attribute(:hidden_actions)
|
||||
@@ -420,18 +420,18 @@ module ActionController #:nodoc:
|
||||
|
||||
# Hide each of the given methods from being callable as actions.
|
||||
def hide_action(*names)
|
||||
write_inheritable_attribute(:hidden_actions, hidden_actions | names.map(&:to_s))
|
||||
write_inheritable_attribute(:hidden_actions, hidden_actions | names.map { |name| name.to_s })
|
||||
end
|
||||
|
||||
## View load paths determine the bases from which template references can be made. So a call to
|
||||
## render("test/template") will be looked up in the view load paths array and the closest match will be
|
||||
## returned.
|
||||
# View load paths determine the bases from which template references can be made. So a call to
|
||||
# render("test/template") will be looked up in the view load paths array and the closest match will be
|
||||
# returned.
|
||||
def view_paths
|
||||
@view_paths || superclass.view_paths
|
||||
end
|
||||
|
||||
def view_paths=(value)
|
||||
@view_paths = ActionView::ViewLoadPaths.new(Array(value)) if value
|
||||
@view_paths = ActionView::Base.process_view_paths(value) if value
|
||||
end
|
||||
|
||||
# Adds a view_path to the front of the view_paths array.
|
||||
@@ -613,7 +613,8 @@ module ActionController #:nodoc:
|
||||
#
|
||||
# This takes the current URL as is and only exchanges the action. In contrast, <tt>url_for :action => 'print'</tt>
|
||||
# would have slashed-off the path components after the changed action.
|
||||
def url_for(options = {}) #:doc:
|
||||
def url_for(options = {})
|
||||
options ||= {}
|
||||
case options
|
||||
when String
|
||||
options
|
||||
@@ -651,7 +652,7 @@ module ActionController #:nodoc:
|
||||
end
|
||||
|
||||
def view_paths=(value)
|
||||
@template.view_paths = ViewLoadPaths.new(value)
|
||||
@template.view_paths = ActionView::Base.process_view_paths(value)
|
||||
end
|
||||
|
||||
# Adds a view_path to the front of the view_paths array.
|
||||
@@ -1200,7 +1201,7 @@ module ActionController #:nodoc:
|
||||
end
|
||||
|
||||
def self.action_methods
|
||||
@action_methods ||= Set.new(public_instance_methods.map(&:to_s)) - hidden_actions
|
||||
@action_methods ||= Set.new(public_instance_methods.map { |m| m.to_s }) - hidden_actions
|
||||
end
|
||||
|
||||
def add_variables_to_assigns
|
||||
@@ -1247,9 +1248,8 @@ module ActionController #:nodoc:
|
||||
end
|
||||
|
||||
def template_exempt_from_layout?(template_name = default_template_name)
|
||||
extension = @template && @template.pick_template_extension(template_name)
|
||||
name_with_extension = !template_name.include?('.') && extension ? "#{template_name}.#{extension}" : template_name
|
||||
@@exempt_from_layout.any? { |ext| name_with_extension =~ ext }
|
||||
template_name = @template.pick_template(template_name).to_s if @template
|
||||
@@exempt_from_layout.any? { |ext| template_name =~ ext }
|
||||
end
|
||||
|
||||
def default_template_name(action_name = self.action_name)
|
||||
|
||||
@@ -94,7 +94,7 @@ module ActionController #:nodoc:
|
||||
map! do |filter|
|
||||
if filters.include?(filter)
|
||||
new_filter = filter.dup
|
||||
new_filter.options.merge!(options)
|
||||
new_filter.update_options!(options)
|
||||
new_filter
|
||||
else
|
||||
filter
|
||||
@@ -104,6 +104,11 @@ module ActionController #:nodoc:
|
||||
end
|
||||
|
||||
class Filter < ActiveSupport::Callbacks::Callback #:nodoc:
|
||||
def initialize(kind, method, options = {})
|
||||
super
|
||||
update_options! options
|
||||
end
|
||||
|
||||
def before?
|
||||
self.class == BeforeFilter
|
||||
end
|
||||
@@ -116,6 +121,18 @@ module ActionController #:nodoc:
|
||||
self.class == AroundFilter
|
||||
end
|
||||
|
||||
# Make sets of strings from :only/:except options
|
||||
def update_options!(other)
|
||||
if other
|
||||
convert_only_and_except_options_to_sets_of_strings(other)
|
||||
if other[:skip]
|
||||
convert_only_and_except_options_to_sets_of_strings(other[:skip])
|
||||
end
|
||||
end
|
||||
|
||||
options.update(other)
|
||||
end
|
||||
|
||||
private
|
||||
def should_not_skip?(controller)
|
||||
if options[:skip]
|
||||
@@ -127,9 +144,9 @@ module ActionController #:nodoc:
|
||||
|
||||
def included_in_action?(controller, options)
|
||||
if options[:only]
|
||||
Array(options[:only]).map(&:to_s).include?(controller.action_name)
|
||||
options[:only].include?(controller.action_name)
|
||||
elsif options[:except]
|
||||
!Array(options[:except]).map(&:to_s).include?(controller.action_name)
|
||||
!options[:except].include?(controller.action_name)
|
||||
else
|
||||
true
|
||||
end
|
||||
@@ -138,6 +155,14 @@ module ActionController #:nodoc:
|
||||
def should_run_callback?(controller)
|
||||
should_not_skip?(controller) && included_in_action?(controller, options) && super
|
||||
end
|
||||
|
||||
def convert_only_and_except_options_to_sets_of_strings(opts)
|
||||
[:only, :except].each do |key|
|
||||
if values = opts[key]
|
||||
opts[key] = Array(values).map(&:to_s).to_set
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
class AroundFilter < Filter #:nodoc:
|
||||
|
||||
@@ -304,7 +304,7 @@ module ActionController #:nodoc:
|
||||
end
|
||||
|
||||
def layout_directory?(layout_name)
|
||||
@template.view_paths.find_template_file_for_path("#{File.join('layouts', layout_name)}.#{@template.template_format}.erb") ? true : false
|
||||
@template.file_exists?("#{File.join('layouts', layout_name)}.#{@template.template_format}")
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -72,57 +72,61 @@ module Mime
|
||||
end
|
||||
|
||||
def parse(accept_header)
|
||||
# keep track of creation order to keep the subsequent sort stable
|
||||
list = []
|
||||
accept_header.split(/,/).each_with_index do |header, index|
|
||||
params, q = header.split(/;\s*q=/)
|
||||
if params
|
||||
params.strip!
|
||||
list << AcceptItem.new(index, params, q) unless params.empty?
|
||||
end
|
||||
end
|
||||
list.sort!
|
||||
|
||||
# Take care of the broken text/xml entry by renaming or deleting it
|
||||
text_xml = list.index("text/xml")
|
||||
app_xml = list.index(Mime::XML.to_s)
|
||||
|
||||
if text_xml && app_xml
|
||||
# set the q value to the max of the two
|
||||
list[app_xml].q = [list[text_xml].q, list[app_xml].q].max
|
||||
|
||||
# make sure app_xml is ahead of text_xml in the list
|
||||
if app_xml > text_xml
|
||||
list[app_xml], list[text_xml] = list[text_xml], list[app_xml]
|
||||
app_xml, text_xml = text_xml, app_xml
|
||||
end
|
||||
|
||||
# delete text_xml from the list
|
||||
list.delete_at(text_xml)
|
||||
|
||||
elsif text_xml
|
||||
list[text_xml].name = Mime::XML.to_s
|
||||
end
|
||||
|
||||
# Look for more specific XML-based types and sort them ahead of app/xml
|
||||
|
||||
if app_xml
|
||||
idx = app_xml
|
||||
app_xml_type = list[app_xml]
|
||||
|
||||
while(idx < list.length)
|
||||
type = list[idx]
|
||||
break if type.q < app_xml_type.q
|
||||
if type.name =~ /\+xml$/
|
||||
list[app_xml], list[idx] = list[idx], list[app_xml]
|
||||
app_xml = idx
|
||||
if accept_header !~ /,/
|
||||
[Mime::Type.lookup(accept_header)]
|
||||
else
|
||||
# keep track of creation order to keep the subsequent sort stable
|
||||
list = []
|
||||
accept_header.split(/,/).each_with_index do |header, index|
|
||||
params, q = header.split(/;\s*q=/)
|
||||
if params
|
||||
params.strip!
|
||||
list << AcceptItem.new(index, params, q) unless params.empty?
|
||||
end
|
||||
idx += 1
|
||||
end
|
||||
end
|
||||
list.sort!
|
||||
|
||||
list.map! { |i| Mime::Type.lookup(i.name) }.uniq!
|
||||
list
|
||||
# Take care of the broken text/xml entry by renaming or deleting it
|
||||
text_xml = list.index("text/xml")
|
||||
app_xml = list.index(Mime::XML.to_s)
|
||||
|
||||
if text_xml && app_xml
|
||||
# set the q value to the max of the two
|
||||
list[app_xml].q = [list[text_xml].q, list[app_xml].q].max
|
||||
|
||||
# make sure app_xml is ahead of text_xml in the list
|
||||
if app_xml > text_xml
|
||||
list[app_xml], list[text_xml] = list[text_xml], list[app_xml]
|
||||
app_xml, text_xml = text_xml, app_xml
|
||||
end
|
||||
|
||||
# delete text_xml from the list
|
||||
list.delete_at(text_xml)
|
||||
|
||||
elsif text_xml
|
||||
list[text_xml].name = Mime::XML.to_s
|
||||
end
|
||||
|
||||
# Look for more specific XML-based types and sort them ahead of app/xml
|
||||
|
||||
if app_xml
|
||||
idx = app_xml
|
||||
app_xml_type = list[app_xml]
|
||||
|
||||
while(idx < list.length)
|
||||
type = list[idx]
|
||||
break if type.q < app_xml_type.q
|
||||
if type.name =~ /\+xml$/
|
||||
list[app_xml], list[idx] = list[idx], list[app_xml]
|
||||
app_xml = idx
|
||||
end
|
||||
idx += 1
|
||||
end
|
||||
end
|
||||
|
||||
list.map! { |i| Mime::Type.lookup(i.name) }.uniq!
|
||||
list
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -82,10 +82,14 @@ module ActionController
|
||||
# Returns the accepted MIME type for the request
|
||||
def accepts
|
||||
@accepts ||=
|
||||
if @env['HTTP_ACCEPT'].to_s.strip.empty?
|
||||
[ content_type, Mime::ALL ].compact # make sure content_type being nil is not included
|
||||
else
|
||||
Mime::Type.parse(@env['HTTP_ACCEPT'])
|
||||
begin
|
||||
header = @env['HTTP_ACCEPT'].to_s.strip
|
||||
|
||||
if header.empty?
|
||||
[content_type, Mime::ALL].compact
|
||||
else
|
||||
Mime::Type.parse(header)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -88,6 +88,10 @@ module ActionController
|
||||
#
|
||||
# map.connect ':controller/:action/:id', :action => 'show', :defaults => { :page => 'Dashboard' }
|
||||
#
|
||||
# Note: The default routes, as provided by the Rails generator, make all actions in every
|
||||
# controller accessible via GET requests. You should consider removing them or commenting
|
||||
# them out if you're using named routes and resources.
|
||||
#
|
||||
# == Named routes
|
||||
#
|
||||
# Routes can be named with the syntax <tt>map.name_of_route options</tt>,
|
||||
|
||||
@@ -205,21 +205,10 @@ module ActionController #:nodoc:
|
||||
p.match(redirect_url) != nil
|
||||
end
|
||||
|
||||
# Returns the template path of the file which was used to
|
||||
# render this response (or nil)
|
||||
def rendered_file(with_controller=false)
|
||||
unless template.first_render.nil?
|
||||
unless with_controller
|
||||
template.first_render
|
||||
else
|
||||
template.first_render.split('/').last || template.first_render
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
# Was this template rendered by a file?
|
||||
def rendered_with_file?
|
||||
!rendered_file.nil?
|
||||
# Returns the template of the file which was used to
|
||||
# render this response (or nil)
|
||||
def rendered_template
|
||||
template._first_render
|
||||
end
|
||||
|
||||
# A shortcut to the flash. Returns an empyt hash if no session flash exists.
|
||||
|
||||
@@ -21,14 +21,14 @@
|
||||
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
#++
|
||||
|
||||
require 'action_view/template_handlers'
|
||||
require 'action_view/template_file'
|
||||
require 'action_view/view_load_paths'
|
||||
|
||||
require 'action_view/template_handlers'
|
||||
require 'action_view/renderable'
|
||||
require 'action_view/renderable_partial'
|
||||
|
||||
require 'action_view/template'
|
||||
require 'action_view/partial_template'
|
||||
require 'action_view/inline_template'
|
||||
require 'action_view/paths'
|
||||
|
||||
require 'action_view/base'
|
||||
require 'action_view/partials'
|
||||
|
||||
@@ -3,6 +3,12 @@ module ActionView #:nodoc:
|
||||
end
|
||||
|
||||
class MissingTemplate < ActionViewError #:nodoc:
|
||||
def initialize(paths, path, template_format = nil)
|
||||
full_template_path = path.include?('.') ? path : "#{path}.erb"
|
||||
display_paths = paths.join(':')
|
||||
template_type = (path =~ /layouts/i) ? 'layout' : 'template'
|
||||
super("Missing #{template_type} #{full_template_path} in view path #{display_paths}")
|
||||
end
|
||||
end
|
||||
|
||||
# Action View templates can be written in three ways. If the template file has a <tt>.erb</tt> (or <tt>.rhtml</tt>) extension then it uses a mixture of ERb
|
||||
@@ -153,11 +159,11 @@ module ActionView #:nodoc:
|
||||
class Base
|
||||
include ERB::Util
|
||||
|
||||
attr_accessor :base_path, :assigns, :template_extension, :first_render
|
||||
attr_accessor :base_path, :assigns, :template_extension
|
||||
attr_accessor :controller
|
||||
attr_accessor :_first_render, :_last_render
|
||||
|
||||
attr_writer :template_format
|
||||
attr_accessor :current_render_extension
|
||||
|
||||
attr_accessor :output_buffer
|
||||
|
||||
@@ -179,6 +185,10 @@ module ActionView #:nodoc:
|
||||
@@debug_rjs = false
|
||||
cattr_accessor :debug_rjs
|
||||
|
||||
# A warning will be displayed whenever an action results in a cache miss on your view paths.
|
||||
@@warn_cache_misses = false
|
||||
cattr_accessor :warn_cache_misses
|
||||
|
||||
attr_internal :request
|
||||
|
||||
delegate :request_forgery_protection_token, :template, :params, :session, :cookies, :response, :headers,
|
||||
@@ -206,6 +216,10 @@ module ActionView #:nodoc:
|
||||
return helpers
|
||||
end
|
||||
|
||||
def self.process_view_paths(value)
|
||||
ActionView::PathSet.new(Array(value))
|
||||
end
|
||||
|
||||
def initialize(view_paths = [], assigns_for_first_render = {}, controller = nil)#:nodoc:
|
||||
@assigns = assigns_for_first_render
|
||||
@assigns_added = nil
|
||||
@@ -216,12 +230,14 @@ module ActionView #:nodoc:
|
||||
attr_reader :view_paths
|
||||
|
||||
def view_paths=(paths)
|
||||
@view_paths = ViewLoadPaths.new(Array(paths))
|
||||
@view_paths = self.class.process_view_paths(paths)
|
||||
end
|
||||
|
||||
# Renders the template present at <tt>template_path</tt> (relative to the view_paths array).
|
||||
# The hash in <tt>local_assigns</tt> is made available as local variables.
|
||||
def render(options = {}, local_assigns = {}, &block) #:nodoc:
|
||||
local_assigns ||= {}
|
||||
|
||||
if options.is_a?(String)
|
||||
render_file(options, nil, local_assigns)
|
||||
elsif options == :update
|
||||
@@ -270,21 +286,50 @@ module ActionView #:nodoc:
|
||||
end
|
||||
|
||||
def file_exists?(template_path)
|
||||
view_paths.template_exists?(template_file_from_name(template_path))
|
||||
pick_template(template_path) ? true : false
|
||||
rescue MissingTemplate
|
||||
false
|
||||
end
|
||||
|
||||
# Gets the extension for an existing template with the given template_path.
|
||||
# Returns the format with the extension if that template exists.
|
||||
#
|
||||
# pick_template_extension('users/show')
|
||||
# # => 'html.erb'
|
||||
# pick_template('users/show')
|
||||
# # => 'users/show.html.erb'
|
||||
#
|
||||
# pick_template_extension('users/legacy')
|
||||
# # => "rhtml"
|
||||
# pick_template('users/legacy')
|
||||
# # => 'users/legacy.rhtml'
|
||||
#
|
||||
def pick_template_extension(template_path)
|
||||
if template = template_file_from_name(template_path)
|
||||
template.extension
|
||||
def pick_template(template_path)
|
||||
path = template_path.sub(/^\//, '')
|
||||
if m = path.match(/(.*)\.(\w+)$/)
|
||||
template_file_name, template_file_extension = m[1], m[2]
|
||||
else
|
||||
template_file_name = path
|
||||
end
|
||||
|
||||
# OPTIMIZE: Checks to lookup template in view path
|
||||
if template = self.view_paths["#{template_file_name}.#{template_format}"]
|
||||
template
|
||||
elsif template = self.view_paths[template_file_name]
|
||||
template
|
||||
elsif _first_render && template = self.view_paths["#{template_file_name}.#{_first_render.format_and_extension}"]
|
||||
template
|
||||
elsif template_format == :js && template = self.view_paths["#{template_file_name}.html"]
|
||||
@template_format = :html
|
||||
template
|
||||
else
|
||||
template = Template.new(template_path, view_paths)
|
||||
|
||||
if self.class.warn_cache_misses && logger = ActionController::Base.logger
|
||||
logger.debug "[PERFORMANCE] Rendering a template that was " +
|
||||
"not found in view path. Templates outside the view path are " +
|
||||
"not cached and result in expensive disk operations. Move this " +
|
||||
"file into #{view_paths.join(':')} or add the folder to your " +
|
||||
"view path list"
|
||||
end
|
||||
|
||||
template
|
||||
end
|
||||
end
|
||||
|
||||
@@ -292,6 +337,10 @@ module ActionView #:nodoc:
|
||||
# Renders the template present at <tt>template_path</tt>. The hash in <tt>local_assigns</tt>
|
||||
# is made available as local variables.
|
||||
def render_file(template_path, use_full_path = nil, local_assigns = {}) #:nodoc:
|
||||
unless use_full_path == nil
|
||||
ActiveSupport::Deprecation.warn("use_full_path option has been deprecated and has no affect.", caller)
|
||||
end
|
||||
|
||||
if defined?(ActionMailer) && defined?(ActionMailer::Base) && controller.is_a?(ActionMailer::Base) && !template_path.include?("/")
|
||||
raise ActionViewError, <<-END_ERROR
|
||||
Due to changes in ActionMailer, you need to provide the mailer_name along with the template name.
|
||||
@@ -305,11 +354,12 @@ module ActionView #:nodoc:
|
||||
END_ERROR
|
||||
end
|
||||
|
||||
Template.new(self, template_path, use_full_path, local_assigns).render_template
|
||||
template = pick_template(template_path)
|
||||
template.render_template(self, local_assigns)
|
||||
end
|
||||
|
||||
def render_inline(text, local_assigns = {}, type = nil)
|
||||
InlineTemplate.new(self, text, local_assigns, type).render
|
||||
InlineTemplate.new(text, type).render(self, local_assigns)
|
||||
end
|
||||
|
||||
def wrap_content_for_layout(content)
|
||||
@@ -332,33 +382,10 @@ module ActionView #:nodoc:
|
||||
@assigns.each { |key, value| instance_variable_set("@#{key}", value) }
|
||||
end
|
||||
|
||||
def execute(template)
|
||||
send(template.method, template.locals) do |*names|
|
||||
def execute(template, local_assigns = {})
|
||||
send(template.method(local_assigns), local_assigns) do |*names|
|
||||
instance_variable_get "@content_for_#{names.first || 'layout'}"
|
||||
end
|
||||
end
|
||||
|
||||
def template_file_from_name(template_name)
|
||||
template_name = TemplateFile.from_path(template_name)
|
||||
pick_template(template_name) unless template_name.extension
|
||||
end
|
||||
|
||||
def pick_template(file)
|
||||
if f = self.view_paths.find_template_file_for_path(file.dup_with_extension(template_format)) || file_from_first_render(file)
|
||||
f
|
||||
elsif template_format == :js && f = self.view_paths.find_template_file_for_path(file.dup_with_extension(:html))
|
||||
@template_format = :html
|
||||
f
|
||||
else
|
||||
nil
|
||||
end
|
||||
end
|
||||
|
||||
# Determine the template extension from the <tt>@first_render</tt> filename
|
||||
def file_from_first_render(file)
|
||||
if extension = File.basename(@first_render.to_s)[/^[^.]+\.(.+)$/, 1]
|
||||
file.dup_with_extension(extension)
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -209,6 +209,10 @@ module ActionView
|
||||
# Note that the default javascript files will be included first. So Prototype and Scriptaculous are available to
|
||||
# all subsequently included files.
|
||||
#
|
||||
# If you want Rails to search in all the subdirectories under javascripts, you should explicitly set <tt>:recursive</tt>:
|
||||
#
|
||||
# javascript_include_tag :all, :recursive => true
|
||||
#
|
||||
# == Caching multiple javascripts into one
|
||||
#
|
||||
# You can also cache multiple javascripts into one file, which requires less HTTP connections to download and can better be
|
||||
@@ -235,18 +239,23 @@ module ActionView
|
||||
#
|
||||
# javascript_include_tag "prototype", "cart", "checkout", :cache => "shop" # when ActionController::Base.perform_caching is true =>
|
||||
# <script type="text/javascript" src="/javascripts/shop.js"></script>
|
||||
#
|
||||
# The <tt>:recursive</tt> option is also available for caching:
|
||||
#
|
||||
# javascript_include_tag :all, :cache => true, :recursive => true
|
||||
def javascript_include_tag(*sources)
|
||||
options = sources.extract_options!.stringify_keys
|
||||
cache = options.delete("cache")
|
||||
recursive = options.delete("recursive")
|
||||
|
||||
if ActionController::Base.perform_caching && cache
|
||||
joined_javascript_name = (cache == true ? "all" : cache) + ".js"
|
||||
joined_javascript_path = File.join(JAVASCRIPTS_DIR, joined_javascript_name)
|
||||
|
||||
write_asset_file_contents(joined_javascript_path, compute_javascript_paths(sources)) unless File.exists?(joined_javascript_path)
|
||||
write_asset_file_contents(joined_javascript_path, compute_javascript_paths(sources, recursive)) unless File.exists?(joined_javascript_path)
|
||||
javascript_src_tag(joined_javascript_name, options)
|
||||
else
|
||||
expand_javascript_sources(sources).collect { |source| javascript_src_tag(source, options) }.join("\n")
|
||||
expand_javascript_sources(sources, recursive).collect { |source| javascript_src_tag(source, options) }.join("\n")
|
||||
end
|
||||
end
|
||||
|
||||
@@ -332,13 +341,17 @@ module ActionView
|
||||
# <link href="/stylesheets/random.styles" media="screen" rel="stylesheet" type="text/css" />
|
||||
# <link href="/css/stylish.css" media="screen" rel="stylesheet" type="text/css" />
|
||||
#
|
||||
# You can also include all styles in the stylesheet directory using <tt>:all</tt> as the source:
|
||||
# You can also include all styles in the stylesheets directory using <tt>:all</tt> as the source:
|
||||
#
|
||||
# stylesheet_link_tag :all # =>
|
||||
# <link href="/stylesheets/style1.css" media="screen" rel="stylesheet" type="text/css" />
|
||||
# <link href="/stylesheets/styleB.css" media="screen" rel="stylesheet" type="text/css" />
|
||||
# <link href="/stylesheets/styleX2.css" media="screen" rel="stylesheet" type="text/css" />
|
||||
#
|
||||
# If you want Rails to search in all the subdirectories under stylesheets, you should explicitly set <tt>:recursive</tt>:
|
||||
#
|
||||
# stylesheet_link_tag :all, :recursive => true
|
||||
#
|
||||
# == Caching multiple stylesheets into one
|
||||
#
|
||||
# You can also cache multiple stylesheets into one file, which requires less HTTP connections and can better be
|
||||
@@ -362,18 +375,23 @@ module ActionView
|
||||
#
|
||||
# stylesheet_link_tag "shop", "cart", "checkout", :cache => "payment" # when ActionController::Base.perform_caching is true =>
|
||||
# <link href="/stylesheets/payment.css" media="screen" rel="stylesheet" type="text/css" />
|
||||
#
|
||||
# The <tt>:recursive</tt> option is also available for caching:
|
||||
#
|
||||
# stylesheet_link_tag :all, :cache => true, :recursive => true
|
||||
def stylesheet_link_tag(*sources)
|
||||
options = sources.extract_options!.stringify_keys
|
||||
cache = options.delete("cache")
|
||||
recursive = options.delete("recursive")
|
||||
|
||||
if ActionController::Base.perform_caching && cache
|
||||
joined_stylesheet_name = (cache == true ? "all" : cache) + ".css"
|
||||
joined_stylesheet_path = File.join(STYLESHEETS_DIR, joined_stylesheet_name)
|
||||
|
||||
write_asset_file_contents(joined_stylesheet_path, compute_stylesheet_paths(sources)) unless File.exists?(joined_stylesheet_path)
|
||||
write_asset_file_contents(joined_stylesheet_path, compute_stylesheet_paths(sources, recursive)) unless File.exists?(joined_stylesheet_path)
|
||||
stylesheet_tag(joined_stylesheet_name, options)
|
||||
else
|
||||
expand_stylesheet_sources(sources).collect { |source| stylesheet_tag(source, options) }.join("\n")
|
||||
expand_stylesheet_sources(sources, recursive).collect { |source| stylesheet_tag(source, options) }.join("\n")
|
||||
end
|
||||
end
|
||||
|
||||
@@ -556,18 +574,19 @@ module ActionView
|
||||
tag("link", { "rel" => "stylesheet", "type" => Mime::CSS, "media" => "screen", "href" => html_escape(path_to_stylesheet(source)) }.merge(options), false, false)
|
||||
end
|
||||
|
||||
def compute_javascript_paths(sources)
|
||||
expand_javascript_sources(sources).collect { |source| compute_public_path(source, 'javascripts', 'js', false) }
|
||||
def compute_javascript_paths(*args)
|
||||
expand_javascript_sources(*args).collect { |source| compute_public_path(source, 'javascripts', 'js', false) }
|
||||
end
|
||||
|
||||
def compute_stylesheet_paths(sources)
|
||||
expand_stylesheet_sources(sources).collect { |source| compute_public_path(source, 'stylesheets', 'css', false) }
|
||||
def compute_stylesheet_paths(*args)
|
||||
expand_stylesheet_sources(*args).collect { |source| compute_public_path(source, 'stylesheets', 'css', false) }
|
||||
end
|
||||
|
||||
def expand_javascript_sources(sources)
|
||||
def expand_javascript_sources(sources, recursive = false)
|
||||
if sources.include?(:all)
|
||||
all_javascript_files = Dir[File.join(JAVASCRIPTS_DIR, '*.js')].collect { |file| File.basename(file).gsub(/\.\w+$/, '') }.sort
|
||||
@@all_javascript_sources ||= ((determine_source(:defaults, @@javascript_expansions).dup & all_javascript_files) + all_javascript_files).uniq
|
||||
all_javascript_files = collect_asset_files(JAVASCRIPTS_DIR, ('**' if recursive), '*.js')
|
||||
@@all_javascript_sources ||= {}
|
||||
@@all_javascript_sources[recursive] ||= ((determine_source(:defaults, @@javascript_expansions).dup & all_javascript_files) + all_javascript_files).uniq
|
||||
else
|
||||
expanded_sources = sources.collect do |source|
|
||||
determine_source(source, @@javascript_expansions)
|
||||
@@ -577,9 +596,10 @@ module ActionView
|
||||
end
|
||||
end
|
||||
|
||||
def expand_stylesheet_sources(sources)
|
||||
def expand_stylesheet_sources(sources, recursive)
|
||||
if sources.first == :all
|
||||
@@all_stylesheet_sources ||= Dir[File.join(STYLESHEETS_DIR, '*.css')].collect { |file| File.basename(file).gsub(/\.\w+$/, '') }.sort
|
||||
@@all_stylesheet_sources ||= {}
|
||||
@@all_stylesheet_sources[recursive] ||= collect_asset_files(STYLESHEETS_DIR, ('**' if recursive), '*.css')
|
||||
else
|
||||
sources.collect do |source|
|
||||
determine_source(source, @@stylesheet_expansions)
|
||||
@@ -604,6 +624,14 @@ module ActionView
|
||||
FileUtils.mkdir_p(File.dirname(joined_asset_path))
|
||||
File.open(joined_asset_path, "w+") { |cache| cache.write(join_asset_file_contents(asset_paths)) }
|
||||
end
|
||||
|
||||
def collect_asset_files(*path)
|
||||
dir = path.first
|
||||
|
||||
Dir[File.join(*path.compact)].collect do |file|
|
||||
file[-(file.size - dir.size - 1)..-1].sub(/\.\w+$/, '')
|
||||
end.sort
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -32,8 +32,7 @@ module ActionView
|
||||
# <i>Topics listed alphabetically</i>
|
||||
# <% end %>
|
||||
def cache(name = {}, options = nil, &block)
|
||||
handler = Template.handler_class_for_extension(current_render_extension.to_sym)
|
||||
handler.new(@controller).cache_fragment(block, name, options)
|
||||
_last_render.handler.new(@controller).cache_fragment(block, name, options)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -34,9 +34,8 @@ module ActionView
|
||||
# Return captured buffer in erb.
|
||||
if block_called_from_erb?(block)
|
||||
with_output_buffer { block.call(*args) }
|
||||
|
||||
# Return block result otherwise, but protect buffer also.
|
||||
else
|
||||
# Return block result otherwise, but protect buffer also.
|
||||
with_output_buffer { return block.call(*args) }
|
||||
end
|
||||
end
|
||||
|
||||
@@ -445,19 +445,19 @@ module ActionView
|
||||
|
||||
class FormBuilder
|
||||
def select(method, choices, options = {}, html_options = {})
|
||||
@template.select(@object_name, method, choices, options.merge(:object => @object), html_options)
|
||||
@template.select(@object_name, method, choices, objectify_options(options), @default_options.merge(html_options))
|
||||
end
|
||||
|
||||
def collection_select(method, collection, value_method, text_method, options = {}, html_options = {})
|
||||
@template.collection_select(@object_name, method, collection, value_method, text_method, options.merge(:object => @object), html_options)
|
||||
@template.collection_select(@object_name, method, collection, value_method, text_method, objectify_options(options), @default_options.merge(html_options))
|
||||
end
|
||||
|
||||
def country_select(method, priority_countries = nil, options = {}, html_options = {})
|
||||
@template.country_select(@object_name, method, priority_countries, options.merge(:object => @object), html_options)
|
||||
@template.country_select(@object_name, method, priority_countries, objectify_options(options), @default_options.merge(html_options))
|
||||
end
|
||||
|
||||
def time_zone_select(method, priority_zones = nil, options = {}, html_options = {})
|
||||
@template.time_zone_select(@object_name, method, priority_zones, options.merge(:object => @object), html_options)
|
||||
@template.time_zone_select(@object_name, method, priority_zones, objectify_options(options), @default_options.merge(html_options))
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -397,7 +397,7 @@ module ActionView
|
||||
# # Generates: <input name="create_btn" onclick="new Ajax.Request('/testing/create',
|
||||
# # {asynchronous:true, evalScripts:true, parameters:Form.serialize(this.form)});
|
||||
# # return false;" type="button" value="Create" />
|
||||
# <%= submit_to_remote 'create_btn', 'Create', :url => { :action => 'create' } %>
|
||||
# <%= button_to_remote 'create_btn', 'Create', :url => { :action => 'create' } %>
|
||||
#
|
||||
# # Submit to the remote action update and update the DIV succeed or fail based
|
||||
# # on the success or failure of the request
|
||||
@@ -405,11 +405,13 @@ module ActionView
|
||||
# # Generates: <input name="update_btn" onclick="new Ajax.Updater({success:'succeed',failure:'fail'},
|
||||
# # '/testing/update', {asynchronous:true, evalScripts:true, parameters:Form.serialize(this.form)});
|
||||
# # return false;" type="button" value="Update" />
|
||||
# <%= submit_to_remote 'update_btn', 'Update', :url => { :action => 'update' },
|
||||
# <%= button_to_remote 'update_btn', 'Update', :url => { :action => 'update' },
|
||||
# :update => { :success => "succeed", :failure => "fail" }
|
||||
#
|
||||
# <tt>options</tt> argument is the same as in form_remote_tag.
|
||||
def submit_to_remote(name, value, options = {})
|
||||
#
|
||||
# Note: This method used to be called submit_to_remote, but that's now just an alias for button_to_remote
|
||||
def button_to_remote(name, value, options = {})
|
||||
options[:with] ||= 'Form.serialize(this.form)'
|
||||
|
||||
options[:html] ||= {}
|
||||
@@ -420,6 +422,7 @@ module ActionView
|
||||
|
||||
tag("input", options[:html], false)
|
||||
end
|
||||
alias_method :submit_to_remote, :button_to_remote
|
||||
|
||||
# Returns '<tt>eval(request.responseText)</tt>' which is the JavaScript function
|
||||
# that +form_remote_tag+ can call in <tt>:complete</tt> to evaluate a multiple
|
||||
|
||||
@@ -27,7 +27,7 @@ module ActionView
|
||||
# %>
|
||||
def concat(string, unused_binding = nil)
|
||||
if unused_binding
|
||||
ActiveSupport::Deprecation.warn("The binding argument of #concat is no longer needed. Please remove it from your views and helpers.")
|
||||
ActiveSupport::Deprecation.warn("The binding argument of #concat is no longer needed. Please remove it from your views and helpers.", caller)
|
||||
end
|
||||
|
||||
output_buffer << string
|
||||
|
||||
@@ -63,17 +63,15 @@ module ActionView
|
||||
# # calls @workshop.to_s
|
||||
# # => /workshops/5
|
||||
def url_for(options = {})
|
||||
options ||= {}
|
||||
case options
|
||||
when Hash
|
||||
show_path = options[:host].nil? ? true : false
|
||||
options = { :only_path => show_path }.update(options.symbolize_keys)
|
||||
options = { :only_path => options[:host].nil? }.update(options.symbolize_keys)
|
||||
escape = options.key?(:escape) ? options.delete(:escape) : true
|
||||
url = @controller.send(:url_for, options)
|
||||
when String
|
||||
escape = true
|
||||
url = options
|
||||
when NilClass
|
||||
url = @controller.send(:url_for, nil)
|
||||
else
|
||||
escape = false
|
||||
url = polymorphic_path(options)
|
||||
@@ -468,7 +466,7 @@ module ActionView
|
||||
email_address_obfuscated.gsub!(/\./, html_options.delete("replace_dot")) if html_options.has_key?("replace_dot")
|
||||
|
||||
if encode == "javascript"
|
||||
"document.write('#{content_tag("a", name || email_address, html_options.merge({ "href" => "mailto:"+email_address+extras }))}');".each_byte do |c|
|
||||
"document.write('#{content_tag("a", name || email_address_obfuscated, html_options.merge({ "href" => "mailto:"+email_address+extras }))}');".each_byte do |c|
|
||||
string << sprintf("%%%x", c)
|
||||
end
|
||||
"<script type=\"#{Mime::JS}\">eval(unescape('#{string}'))</script>"
|
||||
|
||||
@@ -2,15 +2,18 @@ module ActionView #:nodoc:
|
||||
class InlineTemplate #:nodoc:
|
||||
include Renderable
|
||||
|
||||
def initialize(view, source, locals = {}, type = nil)
|
||||
@view = view
|
||||
attr_reader :source, :extension, :method_segment
|
||||
|
||||
def initialize(source, type = nil)
|
||||
@source = source
|
||||
@extension = type
|
||||
@locals = locals || {}
|
||||
|
||||
@method_segment = "inline_#{@source.hash.abs}"
|
||||
@handler = Template.handler_class_for_extension(@extension).new(@view)
|
||||
end
|
||||
|
||||
private
|
||||
# Always recompile inline templates
|
||||
def recompile?(local_assigns)
|
||||
true
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -1,69 +0,0 @@
|
||||
module ActionView #:nodoc:
|
||||
class PartialTemplate < Template #:nodoc:
|
||||
attr_reader :variable_name, :object, :as
|
||||
|
||||
def initialize(view, partial_path, object = nil, locals = {}, as = nil)
|
||||
@view_controller = view.controller if view.respond_to?(:controller)
|
||||
@as = as
|
||||
set_path_and_variable_name!(partial_path)
|
||||
super(view, @path, nil, locals)
|
||||
add_object_to_local_assigns!(object)
|
||||
|
||||
# This is needed here in order to compile template with knowledge of 'counter'
|
||||
initialize_counter!
|
||||
|
||||
# Prepare early. This is a performance optimization for partial collections
|
||||
prepare!
|
||||
end
|
||||
|
||||
def render
|
||||
ActionController::Base.benchmark("Rendered #{@path.path_without_format_and_extension}", Logger::DEBUG, false) do
|
||||
super
|
||||
end
|
||||
end
|
||||
|
||||
def render_member(object)
|
||||
@locals[:object] = @locals[@variable_name] = object
|
||||
@locals[as] = object if as
|
||||
|
||||
template = render_template
|
||||
@locals[@counter_name] += 1
|
||||
@locals.delete(as)
|
||||
@locals.delete(@variable_name)
|
||||
@locals.delete(:object)
|
||||
|
||||
template
|
||||
end
|
||||
|
||||
def counter=(num)
|
||||
@locals[@counter_name] = num
|
||||
end
|
||||
|
||||
private
|
||||
def add_object_to_local_assigns!(object)
|
||||
@locals[:object] ||=
|
||||
@locals[@variable_name] ||= object || @view_controller.instance_variable_get("@#{variable_name}")
|
||||
@locals[as] ||= @locals[:object] if as
|
||||
end
|
||||
|
||||
def set_path_and_variable_name!(partial_path)
|
||||
if partial_path.include?('/')
|
||||
@variable_name = File.basename(partial_path)
|
||||
@path = "#{File.dirname(partial_path)}/_#{@variable_name}"
|
||||
elsif @view_controller
|
||||
@variable_name = partial_path
|
||||
@path = "#{@view_controller.class.controller_path}/_#{@variable_name}"
|
||||
else
|
||||
@variable_name = partial_path
|
||||
@path = "_#{@variable_name}"
|
||||
end
|
||||
|
||||
@variable_name = @variable_name.sub(/\..*$/, '').to_sym
|
||||
end
|
||||
|
||||
def initialize_counter!
|
||||
@counter_name ||= "#{@variable_name}_counter".to_sym
|
||||
@locals[@counter_name] = 0
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -104,10 +104,11 @@ module ActionView
|
||||
module Partials
|
||||
private
|
||||
def render_partial(partial_path, object_assigns = nil, local_assigns = {}) #:nodoc:
|
||||
local_assigns ||= {}
|
||||
|
||||
case partial_path
|
||||
when String, Symbol, NilClass
|
||||
# Render the template
|
||||
ActionView::PartialTemplate.new(self, partial_path, object_assigns, local_assigns).render_template
|
||||
pick_template(find_partial_path(partial_path)).render_partial(self, object_assigns, local_assigns)
|
||||
when ActionView::Helpers::FormBuilder
|
||||
builder_partial_path = partial_path.class.to_s.demodulize.underscore.sub(/_builder$/, '')
|
||||
render_partial(builder_partial_path, object_assigns, (local_assigns || {}).merge(builder_partial_path.to_sym => partial_path))
|
||||
@@ -128,30 +129,28 @@ module ActionView
|
||||
|
||||
local_assigns = local_assigns ? local_assigns.clone : {}
|
||||
spacer = partial_spacer_template ? render(:partial => partial_spacer_template) : ''
|
||||
_paths = {}
|
||||
_templates = {}
|
||||
|
||||
if partial_path.nil?
|
||||
render_partial_collection_with_unknown_partial_path(collection, local_assigns, as)
|
||||
else
|
||||
render_partial_collection_with_known_partial_path(collection, partial_path, local_assigns, as)
|
||||
index = 0
|
||||
collection.map do |object|
|
||||
_partial_path ||= partial_path || ActionController::RecordIdentifier.partial_path(object, controller.class.controller_path)
|
||||
path = _paths[_partial_path] ||= find_partial_path(_partial_path)
|
||||
template = _templates[path] ||= pick_template(path)
|
||||
local_assigns[template.counter_name] = index
|
||||
result = template.render_partial(self, object, local_assigns, as)
|
||||
index += 1
|
||||
result
|
||||
end.join(spacer)
|
||||
end
|
||||
|
||||
def render_partial_collection_with_known_partial_path(collection, partial_path, local_assigns, as)
|
||||
template = ActionView::PartialTemplate.new(self, partial_path, nil, local_assigns, as)
|
||||
collection.map do |element|
|
||||
template.render_member(element)
|
||||
end
|
||||
end
|
||||
|
||||
def render_partial_collection_with_unknown_partial_path(collection, local_assigns, as)
|
||||
templates = Hash.new
|
||||
i = 0
|
||||
collection.map do |element|
|
||||
partial_path = ActionController::RecordIdentifier.partial_path(element, controller.class.controller_path)
|
||||
template = templates[partial_path] ||= ActionView::PartialTemplate.new(self, partial_path, nil, local_assigns, as)
|
||||
template.counter = i
|
||||
i += 1
|
||||
template.render_member(element)
|
||||
def find_partial_path(partial_path)
|
||||
if partial_path.include?('/')
|
||||
"#{File.dirname(partial_path)}/_#{File.basename(partial_path)}"
|
||||
elsif respond_to?(:controller)
|
||||
"#{controller.class.controller_path}/_#{partial_path}"
|
||||
else
|
||||
"_#{partial_path}"
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
96
actionpack/lib/action_view/paths.rb
Normal file
96
actionpack/lib/action_view/paths.rb
Normal file
@@ -0,0 +1,96 @@
|
||||
module ActionView #:nodoc:
|
||||
class PathSet < Array #:nodoc:
|
||||
def self.type_cast(obj)
|
||||
if obj.is_a?(String)
|
||||
if Base.warn_cache_misses && defined?(Rails) && Rails.initialized?
|
||||
Rails.logger.debug "[PERFORMANCE] Processing view path during a " +
|
||||
"request. This an expense disk operation that should be done at " +
|
||||
"boot. You can manually process this view path with " +
|
||||
"ActionView::Base.process_view_paths(#{obj.inspect}) and set it " +
|
||||
"as your view path"
|
||||
end
|
||||
Path.new(obj)
|
||||
else
|
||||
obj
|
||||
end
|
||||
end
|
||||
|
||||
class Path #:nodoc:
|
||||
attr_reader :path, :paths
|
||||
delegate :to_s, :to_str, :inspect, :to => :path
|
||||
|
||||
def initialize(path)
|
||||
@path = path.freeze
|
||||
reload!
|
||||
end
|
||||
|
||||
def ==(path)
|
||||
to_str == path.to_str
|
||||
end
|
||||
|
||||
def [](path)
|
||||
@paths[path]
|
||||
end
|
||||
|
||||
# Rebuild load path directory cache
|
||||
def reload!
|
||||
@paths = {}
|
||||
|
||||
templates_in_path do |template|
|
||||
@paths[template.path] = template
|
||||
@paths[template.path_without_extension] ||= template
|
||||
end
|
||||
|
||||
@paths.freeze
|
||||
end
|
||||
|
||||
private
|
||||
def templates_in_path
|
||||
(Dir.glob("#{@path}/**/*/**") | Dir.glob("#{@path}/**")).each do |file|
|
||||
unless File.directory?(file)
|
||||
template = Template.new(file.split("#{self}/").last, self)
|
||||
# Eager load memoized methods and freeze cached template
|
||||
template.freeze if Base.cache_template_loading
|
||||
yield template
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
def initialize(*args)
|
||||
super(*args).map! { |obj| self.class.type_cast(obj) }
|
||||
end
|
||||
|
||||
def reload!
|
||||
each { |path| path.reload! }
|
||||
end
|
||||
|
||||
def <<(obj)
|
||||
super(self.class.type_cast(obj))
|
||||
end
|
||||
|
||||
def push(*objs)
|
||||
delete_paths!(objs)
|
||||
super(*objs.map { |obj| self.class.type_cast(obj) })
|
||||
end
|
||||
|
||||
def unshift(*objs)
|
||||
delete_paths!(objs)
|
||||
super(*objs.map { |obj| self.class.type_cast(obj) })
|
||||
end
|
||||
|
||||
def [](template_path)
|
||||
each do |path|
|
||||
if template = path[template_path]
|
||||
return template
|
||||
end
|
||||
end
|
||||
nil
|
||||
end
|
||||
|
||||
private
|
||||
def delete_paths!(paths)
|
||||
paths.each { |p1| delete_if { |p2| p1.to_s == p2.to_s } }
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -1,38 +1,78 @@
|
||||
module ActionView
|
||||
module Renderable
|
||||
# TODO: Local assigns should not be tied to template instance
|
||||
attr_accessor :locals
|
||||
# NOTE: The template that this mixin is beening include into is frozen
|
||||
# So you can not set or modify any instance variables
|
||||
|
||||
# TODO: These readers should be private
|
||||
attr_reader :filename, :source, :handler
|
||||
|
||||
def render
|
||||
prepare!
|
||||
@handler.render(self)
|
||||
def self.included(base)
|
||||
@@mutex = Mutex.new
|
||||
end
|
||||
|
||||
def method
|
||||
['_run', @extension, @method_segment, local_assigns_keys].compact.join('_').to_sym
|
||||
include ActiveSupport::Memoizable
|
||||
|
||||
def handler
|
||||
Template.handler_class_for_extension(extension)
|
||||
end
|
||||
memoize :handler
|
||||
|
||||
def compiled_source
|
||||
handler.new(nil).compile(self) if handler.compilable?
|
||||
end
|
||||
memoize :compiled_source
|
||||
|
||||
def render(view, local_assigns = {})
|
||||
view._first_render ||= self
|
||||
view._last_render = self
|
||||
view.send(:evaluate_assigns)
|
||||
compile(local_assigns) if handler.compilable?
|
||||
handler.new(view).render(self, local_assigns)
|
||||
end
|
||||
|
||||
def method(local_assigns)
|
||||
if local_assigns && local_assigns.any?
|
||||
local_assigns_keys = "locals_#{local_assigns.keys.map { |k| k.to_s }.sort.join('_')}"
|
||||
end
|
||||
['_run', extension, method_segment, local_assigns_keys].compact.join('_').to_sym
|
||||
end
|
||||
|
||||
private
|
||||
def prepare!
|
||||
unless @prepared
|
||||
@view.send(:evaluate_assigns)
|
||||
@view.current_render_extension = @extension
|
||||
# Compile and evaluate the template's code
|
||||
def compile(local_assigns)
|
||||
render_symbol = method(local_assigns)
|
||||
|
||||
if @handler.compilable?
|
||||
@handler.compile_template(self) # compile the given template, if necessary
|
||||
@@mutex.synchronize do
|
||||
return false unless recompile?(render_symbol)
|
||||
|
||||
locals_code = local_assigns.keys.map { |key| "#{key} = local_assigns[:#{key}];" }.join
|
||||
|
||||
source = <<-end_src
|
||||
def #{render_symbol}(local_assigns)
|
||||
old_output_buffer = output_buffer;#{locals_code};#{compiled_source}
|
||||
ensure
|
||||
self.output_buffer = old_output_buffer
|
||||
end
|
||||
end_src
|
||||
|
||||
begin
|
||||
file_name = respond_to?(:filename) ? filename : 'compiled-template'
|
||||
ActionView::Base::CompiledTemplates.module_eval(source, file_name, 0)
|
||||
rescue Exception => e # errors from template code
|
||||
if logger = ActionController::Base.logger
|
||||
logger.debug "ERROR: compiling #{render_symbol} RAISED #{e}"
|
||||
logger.debug "Function body: #{source}"
|
||||
logger.debug "Backtrace: #{e.backtrace.join("\n")}"
|
||||
end
|
||||
|
||||
raise ActionView::TemplateError.new(self, {}, e)
|
||||
end
|
||||
|
||||
@prepared = true
|
||||
end
|
||||
end
|
||||
|
||||
def local_assigns_keys
|
||||
if @locals && @locals.any?
|
||||
"locals_#{@locals.keys.map { |k| k.to_s }.sort.join('_')}"
|
||||
end
|
||||
# Method to check whether template compilation is necessary.
|
||||
# The template will be compiled if the file has not been compiled yet, or
|
||||
# if local_assigns has a new key, which isn't supported by the compiled code yet.
|
||||
def recompile?(symbol)
|
||||
meth = Base::CompiledTemplates.instance_method(template.method) rescue nil
|
||||
!(meth && Base.cache_template_loading)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
36
actionpack/lib/action_view/renderable_partial.rb
Normal file
36
actionpack/lib/action_view/renderable_partial.rb
Normal file
@@ -0,0 +1,36 @@
|
||||
module ActionView
|
||||
module RenderablePartial
|
||||
# NOTE: The template that this mixin is beening include into is frozen
|
||||
# So you can not set or modify any instance variables
|
||||
|
||||
include ActiveSupport::Memoizable
|
||||
|
||||
def variable_name
|
||||
name.sub(/\A_/, '').to_sym
|
||||
end
|
||||
memoize :variable_name
|
||||
|
||||
def counter_name
|
||||
"#{variable_name}_counter".to_sym
|
||||
end
|
||||
memoize :counter_name
|
||||
|
||||
def render(view, local_assigns = {})
|
||||
ActionController::Base.benchmark("Rendered #{path_without_format_and_extension}", Logger::DEBUG, false) do
|
||||
super
|
||||
end
|
||||
end
|
||||
|
||||
def render_partial(view, object = nil, local_assigns = {}, as = nil)
|
||||
object ||= local_assigns[:object] ||
|
||||
local_assigns[variable_name] ||
|
||||
view.controller.instance_variable_get("@#{variable_name}") if view.respond_to?(:controller)
|
||||
|
||||
# Ensure correct object is reassigned to other accessors
|
||||
local_assigns[:object] = local_assigns[variable_name] = object
|
||||
local_assigns[as] = object if as
|
||||
|
||||
render_template(view, local_assigns)
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -1,82 +1,96 @@
|
||||
module ActionView #:nodoc:
|
||||
class Template #:nodoc:
|
||||
class Template
|
||||
extend TemplateHandlers
|
||||
include ActiveSupport::Memoizable
|
||||
include Renderable
|
||||
|
||||
attr_reader :path, :extension
|
||||
attr_accessor :filename, :load_path, :base_path, :name, :format, :extension
|
||||
delegate :to_s, :to => :path
|
||||
|
||||
def initialize(view, path, use_full_path = nil, locals = {})
|
||||
unless use_full_path == nil
|
||||
ActiveSupport::Deprecation.warn("use_full_path option has been deprecated and has no affect.", caller)
|
||||
end
|
||||
def initialize(template_path, load_paths = [])
|
||||
template_path = template_path.dup
|
||||
@base_path, @name, @format, @extension = split(template_path)
|
||||
@base_path.to_s.gsub!(/\/$/, '') # Push to split method
|
||||
@load_path, @filename = find_full_path(template_path, load_paths)
|
||||
|
||||
@view = view
|
||||
@paths = view.view_paths
|
||||
|
||||
@original_path = path
|
||||
@path = TemplateFile.from_path(path)
|
||||
@view.first_render ||= @path.to_s
|
||||
|
||||
set_extension_and_file_name
|
||||
|
||||
@method_segment = compiled_method_name_file_path_segment
|
||||
@locals = (locals && locals.dup) || {}
|
||||
@handler = self.class.handler_class_for_extension(@extension).new(@view)
|
||||
# Extend with partial super powers
|
||||
extend RenderablePartial if @name =~ /^_/
|
||||
end
|
||||
|
||||
def render_template
|
||||
render
|
||||
def format_and_extension
|
||||
(extensions = [format, extension].compact.join(".")).blank? ? nil : extensions
|
||||
end
|
||||
memoize :format_and_extension
|
||||
|
||||
def path
|
||||
[base_path, [name, format, extension].compact.join('.')].compact.join('/')
|
||||
end
|
||||
memoize :path
|
||||
|
||||
def path_without_extension
|
||||
[base_path, [name, format].compact.join('.')].compact.join('/')
|
||||
end
|
||||
memoize :path_without_extension
|
||||
|
||||
def path_without_format_and_extension
|
||||
[base_path, name].compact.join('/')
|
||||
end
|
||||
memoize :path_without_format_and_extension
|
||||
|
||||
def source
|
||||
File.read(filename)
|
||||
end
|
||||
memoize :source
|
||||
|
||||
def method_segment
|
||||
segment = File.expand_path(filename)
|
||||
segment.sub!(/^#{Regexp.escape(File.expand_path(RAILS_ROOT))}/, '') if defined?(RAILS_ROOT)
|
||||
segment.gsub!(/([^a-zA-Z0-9_])/) { $1.ord }
|
||||
end
|
||||
memoize :method_segment
|
||||
|
||||
def render_template(view, local_assigns = {})
|
||||
render(view, local_assigns)
|
||||
rescue Exception => e
|
||||
raise e unless filename
|
||||
if TemplateError === e
|
||||
e.sub_template_of(filename)
|
||||
raise e
|
||||
else
|
||||
raise TemplateError.new(self, @view.assigns, e)
|
||||
raise TemplateError.new(self, view.assigns, e)
|
||||
end
|
||||
end
|
||||
|
||||
def source
|
||||
@source ||= File.read(@filename)
|
||||
end
|
||||
|
||||
def base_path_for_exception
|
||||
(@paths.find_load_path_for_path(@path) || @paths.first).to_s
|
||||
end
|
||||
|
||||
private
|
||||
def set_extension_and_file_name
|
||||
@extension = @path.extension
|
||||
|
||||
unless @extension
|
||||
@path = @view.send(:template_file_from_name, @path)
|
||||
raise_missing_template_exception unless @path
|
||||
@extension = @path.extension
|
||||
end
|
||||
|
||||
if p = @paths.find_template_file_for_path(path)
|
||||
@path = p
|
||||
@filename = @path.full_path
|
||||
@extension = @path.extension
|
||||
raise_missing_template_exception if @filename.blank?
|
||||
else
|
||||
@filename = @original_path
|
||||
raise_missing_template_exception unless File.exist?(@filename)
|
||||
end
|
||||
def valid_extension?(extension)
|
||||
Template.template_handler_extensions.include?(extension)
|
||||
end
|
||||
|
||||
def raise_missing_template_exception
|
||||
full_template_path = @original_path.include?('.') ? @original_path : "#{@original_path}.#{@view.template_format}.erb"
|
||||
display_paths = @paths.join(':')
|
||||
template_type = (@original_path =~ /layouts/i) ? 'layout' : 'template'
|
||||
raise MissingTemplate, "Missing #{template_type} #{full_template_path} in view path #{display_paths}"
|
||||
def find_full_path(path, load_paths)
|
||||
load_paths = Array(load_paths) + [nil]
|
||||
load_paths.each do |load_path|
|
||||
file = [load_path, path].compact.join('/')
|
||||
return load_path, file if File.exist?(file)
|
||||
end
|
||||
raise MissingTemplate.new(load_paths, path)
|
||||
end
|
||||
|
||||
def compiled_method_name_file_path_segment
|
||||
s = File.expand_path(@filename)
|
||||
s.sub!(/^#{Regexp.escape(File.expand_path(RAILS_ROOT))}/, '') if defined?(RAILS_ROOT)
|
||||
s.gsub!(/([^a-zA-Z0-9_])/) { $1.ord }
|
||||
s
|
||||
# Returns file split into an array
|
||||
# [base_path, name, format, extension]
|
||||
def split(file)
|
||||
if m = file.match(/^(.*\/)?([^\.]+)\.?(\w+)?\.?(\w+)?\.?(\w+)?$/)
|
||||
if m[5] # Mulipart formats
|
||||
[m[1], m[2], "#{m[3]}.#{m[4]}", m[5]]
|
||||
elsif m[4] # Single format
|
||||
[m[1], m[2], m[3], m[4]]
|
||||
else
|
||||
if valid_extension?(m[3]) # No format
|
||||
[m[1], m[2], nil, m[3]]
|
||||
else # No extension
|
||||
[m[1], m[2], m[3], nil]
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -7,7 +7,7 @@ module ActionView
|
||||
attr_reader :original_exception
|
||||
|
||||
def initialize(template, assigns, original_exception)
|
||||
@base_path = template.base_path_for_exception
|
||||
@base_path = template.base_path
|
||||
@assigns, @source, @original_exception = assigns.dup, template.source, original_exception
|
||||
@file_path = template.filename
|
||||
@backtrace = compute_backtrace
|
||||
|
||||
@@ -1,88 +0,0 @@
|
||||
module ActionView #:nodoc:
|
||||
# TemplateFile abstracts the pattern of querying a file path for its
|
||||
# path with or without its extension. The path is only the partial path
|
||||
# from the load path root e.g. "hello/index.html.erb" not
|
||||
# "app/views/hello/index.html.erb"
|
||||
class TemplateFile
|
||||
def self.from_path(path)
|
||||
path.is_a?(self) ? path : new(path)
|
||||
end
|
||||
|
||||
def self.from_full_path(load_path, full_path)
|
||||
file = new(full_path.split(load_path).last)
|
||||
file.load_path = load_path
|
||||
file.freeze
|
||||
end
|
||||
|
||||
attr_accessor :load_path, :base_path, :name, :format, :extension
|
||||
delegate :to_s, :inspect, :to => :path
|
||||
|
||||
def initialize(path)
|
||||
path = path.dup
|
||||
|
||||
# Clear the forward slash in the beginning
|
||||
trim_forward_slash!(path)
|
||||
|
||||
@base_path, @name, @format, @extension = split(path)
|
||||
end
|
||||
|
||||
def freeze
|
||||
@load_path.freeze
|
||||
@base_path.freeze
|
||||
@name.freeze
|
||||
@format.freeze
|
||||
@extension.freeze
|
||||
super
|
||||
end
|
||||
|
||||
def format_and_extension
|
||||
extensions = [format, extension].compact.join(".")
|
||||
extensions.blank? ? nil : extensions
|
||||
end
|
||||
|
||||
def full_path
|
||||
if load_path
|
||||
"#{load_path}/#{path}"
|
||||
else
|
||||
path
|
||||
end
|
||||
end
|
||||
|
||||
def path
|
||||
base_path.to_s + [name, format, extension].compact.join(".")
|
||||
end
|
||||
|
||||
def path_without_extension
|
||||
base_path.to_s + [name, format].compact.join(".")
|
||||
end
|
||||
|
||||
def path_without_format_and_extension
|
||||
"#{base_path}#{name}"
|
||||
end
|
||||
|
||||
def dup_with_extension(extension)
|
||||
file = dup
|
||||
file.extension = extension ? extension.to_s : nil
|
||||
file
|
||||
end
|
||||
|
||||
private
|
||||
def trim_forward_slash!(path)
|
||||
path.sub!(/^\//, '')
|
||||
end
|
||||
|
||||
# Returns file split into an array
|
||||
# [base_path, name, format, extension]
|
||||
def split(file)
|
||||
if m = file.match(/^(.*\/)?(\w+)\.?(\w+)?\.?(\w+)?\.?(\w+)?$/)
|
||||
if m[5] # Mulipart formats
|
||||
[m[1], m[2], "#{m[3]}.#{m[4]}", m[5]]
|
||||
elsif m[4] # Single format
|
||||
[m[1], m[2], m[3], m[4]]
|
||||
else # No format
|
||||
[m[1], m[2], nil, m[3]]
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -8,7 +8,7 @@ module ActionView
|
||||
@view = view
|
||||
end
|
||||
|
||||
def render(template)
|
||||
def render(template, local_assigns = {})
|
||||
end
|
||||
|
||||
def compile(template)
|
||||
|
||||
@@ -6,7 +6,8 @@ module ActionView
|
||||
include Compilable
|
||||
|
||||
def compile(template)
|
||||
"controller.response.content_type ||= Mime::XML;" +
|
||||
# ActionMailer does not have a response
|
||||
"controller.respond_to?(:response) && controller.response.content_type ||= Mime::XML;" +
|
||||
"xml = ::Builder::XmlMarkup.new(:indent => 2);" +
|
||||
template.source +
|
||||
";xml.target!;"
|
||||
|
||||
@@ -3,8 +3,6 @@ module ActionView
|
||||
module Compilable
|
||||
def self.included(base)
|
||||
base.extend ClassMethod
|
||||
|
||||
@@mutex = Mutex.new
|
||||
end
|
||||
|
||||
module ClassMethod
|
||||
@@ -14,57 +12,9 @@ module ActionView
|
||||
end
|
||||
end
|
||||
|
||||
def render(template)
|
||||
@view.send(:execute, template)
|
||||
def render(template, local_assigns = {})
|
||||
@view.send(:execute, template, local_assigns)
|
||||
end
|
||||
|
||||
# Compile and evaluate the template's code
|
||||
def compile_template(template)
|
||||
return false unless recompile_template?(template)
|
||||
|
||||
@@mutex.synchronize do
|
||||
locals_code = template.locals.keys.map { |key| "#{key} = local_assigns[:#{key}];" }.join
|
||||
|
||||
source = <<-end_src
|
||||
def #{template.method}(local_assigns)
|
||||
old_output_buffer = output_buffer;#{locals_code};#{compile(template)}
|
||||
ensure
|
||||
self.output_buffer = old_output_buffer
|
||||
end
|
||||
end_src
|
||||
|
||||
begin
|
||||
file_name = template.filename || 'compiled-template'
|
||||
ActionView::Base::CompiledTemplates.module_eval(source, file_name, 0)
|
||||
rescue Exception => e # errors from template code
|
||||
if Base.logger
|
||||
Base.logger.debug "ERROR: compiling #{template.method} RAISED #{e}"
|
||||
Base.logger.debug "Function body: #{source}"
|
||||
Base.logger.debug "Backtrace: #{e.backtrace.join("\n")}"
|
||||
end
|
||||
|
||||
raise ActionView::TemplateError.new(template, @view.assigns, e)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
private
|
||||
# Method to check whether template compilation is necessary.
|
||||
# The template will be compiled if the inline template or file has not been compiled yet,
|
||||
# if local_assigns has a new key, which isn't supported by the compiled code yet.
|
||||
def recompile_template?(template)
|
||||
# Unless the template has been complied yet, compile
|
||||
return true unless Base::CompiledTemplates.instance_methods.include?(template.method.to_s)
|
||||
|
||||
# If template caching is disabled, compile
|
||||
return true unless Base.cache_template_loading
|
||||
|
||||
# Always recompile inline templates
|
||||
return true if template.is_a?(InlineTemplate)
|
||||
|
||||
# Otherwise, use compiled method
|
||||
return false
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -5,7 +5,7 @@ module ActionView
|
||||
|
||||
def compile(template)
|
||||
"controller.response.content_type ||= Mime::JS;" +
|
||||
"update_page do |page|;#{template.source};end"
|
||||
"update_page do |page|;#{template.source}\nend"
|
||||
end
|
||||
|
||||
def cache_fragment(block, name = {}, options = nil) #:nodoc:
|
||||
|
||||
@@ -1,103 +0,0 @@
|
||||
module ActionView #:nodoc:
|
||||
class ViewLoadPaths < Array #:nodoc:
|
||||
def self.type_cast(obj)
|
||||
obj.is_a?(String) ? LoadPath.new(obj) : obj
|
||||
end
|
||||
|
||||
class LoadPath #:nodoc:
|
||||
attr_reader :path, :paths
|
||||
delegate :to_s, :to_str, :inspect, :to => :path
|
||||
|
||||
def initialize(path)
|
||||
@path = path.freeze
|
||||
reload!
|
||||
end
|
||||
|
||||
def ==(path)
|
||||
to_str == path.to_str
|
||||
end
|
||||
|
||||
# Rebuild load path directory cache
|
||||
def reload!
|
||||
@paths = {}
|
||||
|
||||
files.each do |file|
|
||||
@paths[file.path] = file
|
||||
@paths[file.path_without_extension] ||= file
|
||||
end
|
||||
|
||||
@paths.freeze
|
||||
end
|
||||
|
||||
def find_template_file_for_partial_path(template_path, template_format)
|
||||
@paths["#{template_path}.#{template_format}"] ||
|
||||
@paths[template_path] ||
|
||||
@paths[template_path.gsub(/\..*$/, '')]
|
||||
end
|
||||
|
||||
private
|
||||
# Get all the files and directories in the path
|
||||
def files_in_path
|
||||
Dir.glob("#{@path}/**/*/**") | Dir.glob("#{@path}/**")
|
||||
end
|
||||
|
||||
# Create an array of all the files within the path
|
||||
def files
|
||||
files_in_path.map do |file|
|
||||
TemplateFile.from_full_path(@path, file) unless File.directory?(file)
|
||||
end.compact
|
||||
end
|
||||
end
|
||||
|
||||
def initialize(*args)
|
||||
super(*args).map! { |obj| self.class.type_cast(obj) }
|
||||
end
|
||||
|
||||
def reload!
|
||||
each { |path| path.reload! }
|
||||
end
|
||||
|
||||
def <<(obj)
|
||||
super(self.class.type_cast(obj))
|
||||
end
|
||||
|
||||
def push(*objs)
|
||||
delete_paths!(objs)
|
||||
super(*objs.map { |obj| self.class.type_cast(obj) })
|
||||
end
|
||||
|
||||
def unshift(*objs)
|
||||
delete_paths!(objs)
|
||||
super(*objs.map { |obj| self.class.type_cast(obj) })
|
||||
end
|
||||
|
||||
def template_exists?(file)
|
||||
find_load_path_for_path(file) ? true : false
|
||||
end
|
||||
|
||||
def find_load_path_for_path(file)
|
||||
find { |path| path.paths[file.to_s] }
|
||||
end
|
||||
|
||||
def find_template_file_for_path(template_path)
|
||||
template_path_without_extension, template_extension = path_and_extension(template_path.to_s)
|
||||
each do |path|
|
||||
if f = path.find_template_file_for_partial_path(template_path_without_extension, template_extension)
|
||||
return f
|
||||
end
|
||||
end
|
||||
nil
|
||||
end
|
||||
|
||||
private
|
||||
def delete_paths!(paths)
|
||||
paths.each { |p1| delete_if { |p2| p1.to_s == p2.to_s } }
|
||||
end
|
||||
|
||||
# Splits the path and extension from the given template_path and returns as an array.
|
||||
def path_and_extension(template_path)
|
||||
template_path_without_extension = template_path.sub(/\.(\w+)$/, '')
|
||||
[template_path_without_extension, $1]
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -22,7 +22,9 @@ ActiveSupport::Deprecation.debug = true
|
||||
ActionController::Base.logger = nil
|
||||
ActionController::Routing::Routes.reload rescue nil
|
||||
|
||||
FIXTURE_LOAD_PATH = ActionView::ViewLoadPaths::LoadPath.new(File.join(File.dirname(__FILE__), 'fixtures'))
|
||||
ActionView::Base.cache_template_loading = true
|
||||
FIXTURE_LOAD_PATH = File.join(File.dirname(__FILE__), 'fixtures')
|
||||
ActionController::Base.view_paths = FIXTURE_LOAD_PATH
|
||||
|
||||
# Wrap tests that use Mocha and skip if unavailable.
|
||||
def uses_mocha(test_name)
|
||||
|
||||
@@ -41,8 +41,6 @@ class RenderPartialWithRecordIdentificationController < ActionController::Base
|
||||
end
|
||||
end
|
||||
|
||||
RenderPartialWithRecordIdentificationController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
class RenderPartialWithRecordIdentificationTest < ActiveRecordTestCase
|
||||
fixtures :developers, :projects, :developers_projects, :topics, :replies, :companies, :mascots
|
||||
|
||||
@@ -56,26 +54,31 @@ class RenderPartialWithRecordIdentificationTest < ActiveRecordTestCase
|
||||
def test_rendering_partial_with_has_many_and_belongs_to_association
|
||||
get :render_with_has_many_and_belongs_to_association
|
||||
assert_template 'projects/_project'
|
||||
assert_equal 'Active RecordActive Controller', @response.body
|
||||
end
|
||||
|
||||
def test_rendering_partial_with_has_many_association
|
||||
get :render_with_has_many_association
|
||||
assert_template 'replies/_reply'
|
||||
assert_equal 'Birdman is better!', @response.body
|
||||
end
|
||||
|
||||
def test_rendering_partial_with_named_scope
|
||||
get :render_with_named_scope
|
||||
assert_template 'replies/_reply'
|
||||
assert_equal 'Birdman is better!Nuh uh!', @response.body
|
||||
end
|
||||
|
||||
def test_render_with_record
|
||||
get :render_with_record
|
||||
assert_template 'developers/_developer'
|
||||
assert_equal 'David', @response.body
|
||||
end
|
||||
|
||||
def test_render_with_record_collection
|
||||
get :render_with_record_collection
|
||||
assert_template 'developers/_developer'
|
||||
assert_equal 'DavidJamisfixture_3fixture_4fixture_5fixture_6fixture_7fixture_8fixture_9fixture_10Jamis', @response.body
|
||||
end
|
||||
|
||||
def test_rendering_partial_with_has_one_association
|
||||
@@ -118,8 +121,6 @@ class RenderPartialWithRecordIdentificationController < ActionController::Base
|
||||
end
|
||||
end
|
||||
|
||||
RenderPartialWithRecordIdentificationController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
class Game < Struct.new(:name, :id)
|
||||
def to_param
|
||||
id.to_s
|
||||
@@ -137,8 +138,6 @@ module Fun
|
||||
end
|
||||
end
|
||||
|
||||
NestedController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
module Serious
|
||||
class NestedDeeperController < ActionController::Base
|
||||
def render_with_record_in_deeper_nested_controller
|
||||
@@ -149,8 +148,6 @@ module Fun
|
||||
render :partial => [ Game.new("Chess"), Game.new("Sudoku"), Game.new("Solitaire") ]
|
||||
end
|
||||
end
|
||||
|
||||
NestedDeeperController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
end
|
||||
end
|
||||
|
||||
@@ -165,11 +162,13 @@ class RenderPartialWithRecordIdentificationAndNestedControllersTest < ActiveReco
|
||||
def test_render_with_record_in_nested_controller
|
||||
get :render_with_record_in_nested_controller
|
||||
assert_template 'fun/games/_game'
|
||||
assert_equal 'Pong', @response.body
|
||||
end
|
||||
|
||||
def test_render_with_record_collection_in_nested_controller
|
||||
get :render_with_record_collection_in_nested_controller
|
||||
assert_template 'fun/games/_game'
|
||||
assert_equal 'PongTank', @response.body
|
||||
end
|
||||
end
|
||||
|
||||
@@ -184,10 +183,12 @@ class RenderPartialWithRecordIdentificationAndNestedDeeperControllersTest < Acti
|
||||
def test_render_with_record_in_deeper_nested_controller
|
||||
get :render_with_record_in_deeper_nested_controller
|
||||
assert_template 'fun/serious/games/_game'
|
||||
assert_equal 'Chess', @response.body
|
||||
end
|
||||
|
||||
def test_render_with_record_collection_in_deeper_nested_controller
|
||||
get :render_with_record_collection_in_deeper_nested_controller
|
||||
assert_template 'fun/serious/games/_game'
|
||||
assert_equal 'ChessSudokuSolitaire', @response.body
|
||||
end
|
||||
end
|
||||
|
||||
@@ -164,14 +164,6 @@ module Admin
|
||||
end
|
||||
end
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
|
||||
# tell the controller where to find its templates but start from parent
|
||||
# directory of test_request_response to simulate the behaviour of a
|
||||
# production environment
|
||||
ActionPackAssertionsController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
# a test case to exercise the new capabilities TestRequest & TestResponse
|
||||
class ActionPackAssertionsControllerTest < Test::Unit::TestCase
|
||||
# let's get this party started
|
||||
@@ -336,11 +328,11 @@ class ActionPackAssertionsControllerTest < Test::Unit::TestCase
|
||||
# check if we were rendered by a file-based template?
|
||||
def test_rendered_action
|
||||
process :nothing
|
||||
assert !@response.rendered_with_file?
|
||||
assert_nil @response.rendered_template
|
||||
|
||||
process :hello_world
|
||||
assert @response.rendered_with_file?
|
||||
assert 'hello_world', @response.rendered_file
|
||||
assert @response.rendered_template
|
||||
assert 'hello_world', @response.rendered_template.to_s
|
||||
end
|
||||
|
||||
# check the redirection location
|
||||
|
||||
@@ -19,8 +19,6 @@ class AddressesTestController < ActionController::Base
|
||||
def self.controller_path; "addresses"; end
|
||||
end
|
||||
|
||||
AddressesTestController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
class AddressesTest < Test::Unit::TestCase
|
||||
def setup
|
||||
@controller = AddressesTestController.new
|
||||
|
||||
@@ -6,7 +6,6 @@ CACHE_DIR = 'test_cache'
|
||||
FILE_STORE_PATH = File.join(File.dirname(__FILE__), '/../temp/', CACHE_DIR)
|
||||
ActionController::Base.page_cache_directory = FILE_STORE_PATH
|
||||
ActionController::Base.cache_store = :file_store, FILE_STORE_PATH
|
||||
ActionController::Base.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
class PageCachingTestController < ActionController::Base
|
||||
caches_page :ok, :no_content, :if => Proc.new { |c| !c.request.format.json? }
|
||||
@@ -285,9 +284,12 @@ class ActionCacheTest < Test::Unit::TestCase
|
||||
end
|
||||
|
||||
def test_action_cache_conditional_options
|
||||
old_use_accept_header = ActionController::Base.use_accept_header
|
||||
ActionController::Base.use_accept_header = true
|
||||
@request.env['HTTP_ACCEPT'] = 'application/json'
|
||||
get :index
|
||||
assert !fragment_exist?('hostname.com/action_caching_test')
|
||||
ActionController::Base.use_accept_header = old_use_accept_header
|
||||
end
|
||||
|
||||
def test_action_cache_with_store_options
|
||||
@@ -633,8 +635,6 @@ class FunctionalCachingController < ActionController::Base
|
||||
end
|
||||
end
|
||||
|
||||
FunctionalCachingController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
class FunctionalFragmentCachingTest < Test::Unit::TestCase
|
||||
def setup
|
||||
ActionController::Base.perform_caching = true
|
||||
@@ -664,6 +664,14 @@ CACHED
|
||||
assert_match "Fragment caching in a partial", @store.read('views/test.host/functional_caching/html_fragment_cached_with_partial')
|
||||
end
|
||||
|
||||
def test_render_inline_before_fragment_caching
|
||||
get :inline_fragment_cached
|
||||
assert_response :success
|
||||
assert_match /Some inline content/, @response.body
|
||||
assert_match /Some cached content/, @response.body
|
||||
assert_match "Some cached content", @store.read('views/test.host/functional_caching/inline_fragment_cached')
|
||||
end
|
||||
|
||||
def test_fragment_caching_in_rjs_partials
|
||||
xhr :get, :js_fragment_cached_with_partial
|
||||
assert_response :success
|
||||
|
||||
@@ -23,8 +23,6 @@ class CaptureController < ActionController::Base
|
||||
def rescue_action(e) raise end
|
||||
end
|
||||
|
||||
CaptureController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
class CaptureTest < Test::Unit::TestCase
|
||||
def setup
|
||||
@controller = CaptureController.new
|
||||
|
||||
@@ -45,8 +45,6 @@ class ContentTypeController < ActionController::Base
|
||||
def rescue_action(e) raise end
|
||||
end
|
||||
|
||||
ContentTypeController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
class ContentTypeTest < Test::Unit::TestCase
|
||||
def setup
|
||||
@controller = ContentTypeController.new
|
||||
@@ -124,7 +122,7 @@ class AcceptBasedContentTypeTest < ActionController::TestCase
|
||||
ActionController::Base.use_accept_header = true
|
||||
end
|
||||
|
||||
def tear_down
|
||||
def teardown
|
||||
ActionController::Base.use_accept_header = false
|
||||
end
|
||||
|
||||
|
||||
@@ -13,8 +13,6 @@ class DeprecatedBaseMethodsTest < Test::Unit::TestCase
|
||||
def rescue_action(e) raise e end
|
||||
end
|
||||
|
||||
Target.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
def setup
|
||||
@request = ActionController::TestRequest.new
|
||||
@response = ActionController::TestResponse.new
|
||||
|
||||
@@ -34,8 +34,8 @@ end
|
||||
class MabView < ActionView::TemplateHandler
|
||||
def initialize(view)
|
||||
end
|
||||
|
||||
def render(template)
|
||||
|
||||
def render(template, local_assigns)
|
||||
template.source
|
||||
end
|
||||
end
|
||||
@@ -63,6 +63,7 @@ class LayoutAutoDiscoveryTest < Test::Unit::TestCase
|
||||
end
|
||||
|
||||
def test_third_party_template_library_auto_discovers_layout
|
||||
ThirdPartyTemplateLibraryController.view_paths.reload!
|
||||
@controller = ThirdPartyTemplateLibraryController.new
|
||||
get :hello
|
||||
assert_equal 'layouts/third_party_template_library', @controller.active_layout
|
||||
|
||||
@@ -162,8 +162,6 @@ class RespondToController < ActionController::Base
|
||||
end
|
||||
end
|
||||
|
||||
RespondToController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
class MimeControllerTest < Test::Unit::TestCase
|
||||
def setup
|
||||
ActionController::Base.use_accept_header = true
|
||||
|
||||
@@ -465,9 +465,6 @@ class NewRenderTestController < ActionController::Base
|
||||
end
|
||||
end
|
||||
|
||||
NewRenderTestController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
Fun::GamesController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
class NewRenderTest < Test::Unit::TestCase
|
||||
def setup
|
||||
@controller = NewRenderTestController.new
|
||||
@@ -489,6 +486,11 @@ class NewRenderTest < Test::Unit::TestCase
|
||||
assert_equal "<html>Hello world!</html>", @response.body
|
||||
end
|
||||
|
||||
def test_renders_default_template_for_missing_action
|
||||
get :'hyphen-ated'
|
||||
assert_template 'test/hyphen-ated'
|
||||
end
|
||||
|
||||
def test_do_with_render
|
||||
get :render_hello_world
|
||||
assert_template "test/hello_world"
|
||||
|
||||
@@ -227,6 +227,11 @@ class RedirectTest < Test::Unit::TestCase
|
||||
assert_redirected_to Workshop.new(5, true)
|
||||
end
|
||||
|
||||
def test_redirect_with_partial_params
|
||||
get :module_redirect
|
||||
assert_redirected_to :action => 'hello_world'
|
||||
end
|
||||
|
||||
def test_redirect_to_nil
|
||||
assert_raises(ActionController::ActionControllerError) do
|
||||
get :redirect_to_nil
|
||||
|
||||
@@ -217,9 +217,6 @@ class TestController < ActionController::Base
|
||||
end
|
||||
end
|
||||
|
||||
TestController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
Fun::GamesController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
class RenderTest < Test::Unit::TestCase
|
||||
def setup
|
||||
@request = ActionController::TestRequest.new
|
||||
|
||||
@@ -28,18 +28,16 @@ module Backoffice
|
||||
end
|
||||
|
||||
class ResourcesTest < Test::Unit::TestCase
|
||||
|
||||
|
||||
# The assertions in these tests are incompatible with the hash method
|
||||
# optimisation. This could indicate user level problems
|
||||
def setup
|
||||
ActionController::Base.optimise_named_routes = false
|
||||
end
|
||||
|
||||
def tear_down
|
||||
|
||||
def teardown
|
||||
ActionController::Base.optimise_named_routes = true
|
||||
end
|
||||
|
||||
|
||||
def test_should_arrange_actions
|
||||
resource = ActionController::Resources::Resource.new(:messages,
|
||||
:collection => { :rss => :get, :reorder => :post, :csv => :post },
|
||||
@@ -159,14 +157,14 @@ class ResourcesTest < Test::Unit::TestCase
|
||||
|
||||
def test_with_collection_actions_and_name_prefix
|
||||
actions = { 'a' => :get, 'b' => :put, 'c' => :post, 'd' => :delete }
|
||||
|
||||
|
||||
with_restful_routing :messages, :path_prefix => '/threads/:thread_id', :name_prefix => "thread_", :collection => actions do
|
||||
assert_restful_routes_for :messages, :path_prefix => 'threads/1/', :name_prefix => 'thread_', :options => { :thread_id => '1' } do |options|
|
||||
actions.each do |action, method|
|
||||
assert_recognizes(options.merge(:action => action), :path => "/threads/1/messages/#{action}", :method => method)
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
assert_restful_named_routes_for :messages, :path_prefix => 'threads/1/', :name_prefix => 'thread_', :options => { :thread_id => '1' } do |options|
|
||||
actions.keys.each do |action|
|
||||
assert_named_route "/threads/1/messages/#{action}", "#{action}_thread_messages_path", :action => action
|
||||
@@ -177,14 +175,14 @@ class ResourcesTest < Test::Unit::TestCase
|
||||
|
||||
def test_with_collection_action_and_name_prefix_and_formatted
|
||||
actions = { 'a' => :get, 'b' => :put, 'c' => :post, 'd' => :delete }
|
||||
|
||||
|
||||
with_restful_routing :messages, :path_prefix => '/threads/:thread_id', :name_prefix => "thread_", :collection => actions do
|
||||
assert_restful_routes_for :messages, :path_prefix => 'threads/1/', :name_prefix => 'thread_', :options => { :thread_id => '1' } do |options|
|
||||
actions.each do |action, method|
|
||||
assert_recognizes(options.merge(:action => action, :format => 'xml'), :path => "/threads/1/messages/#{action}.xml", :method => method)
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
assert_restful_named_routes_for :messages, :path_prefix => 'threads/1/', :name_prefix => 'thread_', :options => { :thread_id => '1' } do |options|
|
||||
actions.keys.each do |action|
|
||||
assert_named_route "/threads/1/messages/#{action}.xml", "formatted_#{action}_thread_messages_path", :action => action, :format => 'xml'
|
||||
@@ -279,7 +277,7 @@ class ResourcesTest < Test::Unit::TestCase
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
def test_with_new_action_with_name_prefix
|
||||
with_restful_routing :messages, :new => { :preview => :post }, :path_prefix => '/threads/:thread_id', :name_prefix => 'thread_' do
|
||||
preview_options = {:action => 'preview', :thread_id => '1'}
|
||||
@@ -293,7 +291,7 @@ class ResourcesTest < Test::Unit::TestCase
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
def test_with_formatted_new_action_with_name_prefix
|
||||
with_restful_routing :messages, :new => { :preview => :post }, :path_prefix => '/threads/:thread_id', :name_prefix => 'thread_' do
|
||||
preview_options = {:action => 'preview', :thread_id => '1', :format => 'xml'}
|
||||
@@ -307,7 +305,7 @@ class ResourcesTest < Test::Unit::TestCase
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
def test_override_new_method
|
||||
with_restful_routing :messages do
|
||||
assert_restful_routes_for :messages do |options|
|
||||
@@ -524,9 +522,9 @@ class ResourcesTest < Test::Unit::TestCase
|
||||
map.resources :messages, :collection => {:search => :get}, :new => {:preview => :any}, :name_prefix => 'thread_', :path_prefix => '/threads/:thread_id'
|
||||
map.resource :account, :member => {:login => :get}, :new => {:preview => :any}, :name_prefix => 'admin_', :path_prefix => '/admin'
|
||||
end
|
||||
|
||||
|
||||
action_separator = ActionController::Base.resource_action_separator
|
||||
|
||||
|
||||
assert_simply_restful_for :messages, :name_prefix => 'thread_', :path_prefix => 'threads/1/', :options => { :thread_id => '1' }
|
||||
assert_named_route "/threads/1/messages#{action_separator}search", "search_thread_messages_path", {}
|
||||
assert_named_route "/threads/1/messages/new", "new_thread_message_path", {}
|
||||
@@ -623,7 +621,7 @@ class ResourcesTest < Test::Unit::TestCase
|
||||
assert_simply_restful_for :products, :controller => "backoffice/products"
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
def test_nested_resources_using_namespace
|
||||
with_routing do |set|
|
||||
set.draw do |map|
|
||||
@@ -795,7 +793,7 @@ class ResourcesTest < Test::Unit::TestCase
|
||||
|
||||
yield options[:options] if block_given?
|
||||
end
|
||||
|
||||
|
||||
def assert_singleton_routes_for(singleton_name, options = {})
|
||||
options[:options] ||= {}
|
||||
options[:options][:controller] = options[:controller] || singleton_name.to_s.pluralize
|
||||
@@ -855,7 +853,7 @@ class ResourcesTest < Test::Unit::TestCase
|
||||
actual = @controller.send(route, options) rescue $!.class.name
|
||||
assert_equal expected, actual, "Error on route: #{route}(#{options.inspect})"
|
||||
end
|
||||
|
||||
|
||||
def assert_resource_methods(expected, resource, action_method, method)
|
||||
assert_equal expected.length, resource.send("#{action_method}_methods")[method].size, "#{resource.send("#{action_method}_methods")[method].inspect}"
|
||||
expected.each do |action|
|
||||
|
||||
@@ -19,8 +19,6 @@ class SendFileController < ActionController::Base
|
||||
def rescue_action(e) raise end
|
||||
end
|
||||
|
||||
SendFileController.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
class SendFileTest < Test::Unit::TestCase
|
||||
include TestFileUtils
|
||||
|
||||
|
||||
@@ -1,8 +1,6 @@
|
||||
require 'abstract_unit'
|
||||
|
||||
class ViewLoadPathsTest < Test::Unit::TestCase
|
||||
ActionController::Base.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
class TestController < ActionController::Base
|
||||
def self.controller_path() "test" end
|
||||
def rescue_action(e) raise end
|
||||
@@ -146,18 +144,4 @@ class ViewLoadPathsTest < Test::Unit::TestCase
|
||||
assert_nothing_raised { C.view_paths << 'c/path' }
|
||||
assert_equal ['c/path'], C.view_paths
|
||||
end
|
||||
|
||||
def test_find_template_file_for_path
|
||||
assert_equal "test/hello_world.erb", @controller.view_paths.find_template_file_for_path("test/hello_world.erb").to_s
|
||||
assert_equal "test/hello.builder", @controller.view_paths.find_template_file_for_path("test/hello.builder").to_s
|
||||
assert_equal nil, @controller.view_paths.find_template_file_for_path("test/missing.erb")
|
||||
end
|
||||
|
||||
def test_view_paths_find_template_file_for_path
|
||||
assert_equal "test/formatted_html_erb.html.erb", @controller.view_paths.find_template_file_for_path("test/formatted_html_erb.html").to_s
|
||||
assert_equal "test/formatted_xml_erb.xml.erb", @controller.view_paths.find_template_file_for_path("test/formatted_xml_erb.xml").to_s
|
||||
assert_equal "test/hello_world.erb", @controller.view_paths.find_template_file_for_path("test/hello_world.html").to_s
|
||||
assert_equal "test/hello_world.erb", @controller.view_paths.find_template_file_for_path("test/hello_world.xml").to_s
|
||||
assert_equal nil, @controller.view_paths.find_template_file_for_path("test/missing.html")
|
||||
end
|
||||
end
|
||||
|
||||
1
actionpack/test/fixtures/developers/_developer.erb
vendored
Normal file
1
actionpack/test/fixtures/developers/_developer.erb
vendored
Normal file
@@ -0,0 +1 @@
|
||||
<%= developer.name %>
|
||||
1
actionpack/test/fixtures/fun/games/_game.erb
vendored
Normal file
1
actionpack/test/fixtures/fun/games/_game.erb
vendored
Normal file
@@ -0,0 +1 @@
|
||||
<%= game.name %>
|
||||
1
actionpack/test/fixtures/fun/serious/games/_game.erb
vendored
Normal file
1
actionpack/test/fixtures/fun/serious/games/_game.erb
vendored
Normal file
@@ -0,0 +1 @@
|
||||
<%= game.name %>
|
||||
2
actionpack/test/fixtures/functional_caching/inline_fragment_cached.html.erb
vendored
Normal file
2
actionpack/test/fixtures/functional_caching/inline_fragment_cached.html.erb
vendored
Normal file
@@ -0,0 +1,2 @@
|
||||
<%= render :inline => 'Some inline content' %>
|
||||
<% cache do %>Some cached content<% end %>
|
||||
1
actionpack/test/fixtures/projects/_project.erb
vendored
Normal file
1
actionpack/test/fixtures/projects/_project.erb
vendored
Normal file
@@ -0,0 +1 @@
|
||||
<%= project.name %>
|
||||
1
actionpack/test/fixtures/public/javascripts/subdir/subdir.js
vendored
Normal file
1
actionpack/test/fixtures/public/javascripts/subdir/subdir.js
vendored
Normal file
@@ -0,0 +1 @@
|
||||
// subdir js
|
||||
1
actionpack/test/fixtures/public/stylesheets/subdir/subdir.css
vendored
Normal file
1
actionpack/test/fixtures/public/stylesheets/subdir/subdir.css
vendored
Normal file
@@ -0,0 +1 @@
|
||||
/* subdir.css */
|
||||
1
actionpack/test/fixtures/replies/_reply.erb
vendored
Normal file
1
actionpack/test/fixtures/replies/_reply.erb
vendored
Normal file
@@ -0,0 +1 @@
|
||||
<%= reply.content %>
|
||||
1
actionpack/test/fixtures/test/hyphen-ated.erb
vendored
Normal file
1
actionpack/test/fixtures/test/hyphen-ated.erb
vendored
Normal file
@@ -0,0 +1 @@
|
||||
Hello world!
|
||||
@@ -83,6 +83,7 @@ class AssetTagHelperTest < ActionView::TestCase
|
||||
%(javascript_include_tag("common.javascript", "/elsewhere/cools")) => %(<script src="/javascripts/common.javascript" type="text/javascript"></script>\n<script src="/elsewhere/cools.js" type="text/javascript"></script>),
|
||||
%(javascript_include_tag(:defaults)) => %(<script src="/javascripts/prototype.js" type="text/javascript"></script>\n<script src="/javascripts/effects.js" type="text/javascript"></script>\n<script src="/javascripts/dragdrop.js" type="text/javascript"></script>\n<script src="/javascripts/controls.js" type="text/javascript"></script>\n<script src="/javascripts/application.js" type="text/javascript"></script>),
|
||||
%(javascript_include_tag(:all)) => %(<script src="/javascripts/prototype.js" type="text/javascript"></script>\n<script src="/javascripts/effects.js" type="text/javascript"></script>\n<script src="/javascripts/dragdrop.js" type="text/javascript"></script>\n<script src="/javascripts/controls.js" type="text/javascript"></script>\n<script src="/javascripts/application.js" type="text/javascript"></script>\n<script src="/javascripts/bank.js" type="text/javascript"></script>\n<script src="/javascripts/robber.js" type="text/javascript"></script>\n<script src="/javascripts/version.1.0.js" type="text/javascript"></script>),
|
||||
%(javascript_include_tag(:all, :recursive => true)) => %(<script src="/javascripts/prototype.js" type="text/javascript"></script>\n<script src="/javascripts/effects.js" type="text/javascript"></script>\n<script src="/javascripts/dragdrop.js" type="text/javascript"></script>\n<script src="/javascripts/controls.js" type="text/javascript"></script>\n<script src="/javascripts/application.js" type="text/javascript"></script>\n<script src="/javascripts/bank.js" type="text/javascript"></script>\n<script src="/javascripts/robber.js" type="text/javascript"></script>\n<script src="/javascripts/subdir/subdir.js" type="text/javascript"></script>\n<script src="/javascripts/version.1.0.js" type="text/javascript"></script>),
|
||||
%(javascript_include_tag(:defaults, "test")) => %(<script src="/javascripts/prototype.js" type="text/javascript"></script>\n<script src="/javascripts/effects.js" type="text/javascript"></script>\n<script src="/javascripts/dragdrop.js" type="text/javascript"></script>\n<script src="/javascripts/controls.js" type="text/javascript"></script>\n<script src="/javascripts/test.js" type="text/javascript"></script>\n<script src="/javascripts/application.js" type="text/javascript"></script>),
|
||||
%(javascript_include_tag("test", :defaults)) => %(<script src="/javascripts/test.js" type="text/javascript"></script>\n<script src="/javascripts/prototype.js" type="text/javascript"></script>\n<script src="/javascripts/effects.js" type="text/javascript"></script>\n<script src="/javascripts/dragdrop.js" type="text/javascript"></script>\n<script src="/javascripts/controls.js" type="text/javascript"></script>\n<script src="/javascripts/application.js" type="text/javascript"></script>)
|
||||
}
|
||||
@@ -108,6 +109,7 @@ class AssetTagHelperTest < ActionView::TestCase
|
||||
%(stylesheet_link_tag("dir/file")) => %(<link href="/stylesheets/dir/file.css" media="screen" rel="stylesheet" type="text/css" />),
|
||||
%(stylesheet_link_tag("style", :media => "all")) => %(<link href="/stylesheets/style.css" media="all" rel="stylesheet" type="text/css" />),
|
||||
%(stylesheet_link_tag(:all)) => %(<link href="/stylesheets/bank.css" media="screen" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/robber.css" media="screen" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/version.1.0.css" media="screen" rel="stylesheet" type="text/css" />),
|
||||
%(stylesheet_link_tag(:all, :recursive => true)) => %(<link href="/stylesheets/bank.css" media="screen" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/robber.css" media="screen" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/subdir/subdir.css" media="screen" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/version.1.0.css" media="screen" rel="stylesheet" type="text/css" />),
|
||||
%(stylesheet_link_tag(:all, :media => "all")) => %(<link href="/stylesheets/bank.css" media="all" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/robber.css" media="all" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/version.1.0.css" media="all" rel="stylesheet" type="text/css" />),
|
||||
%(stylesheet_link_tag("random.styles", "/css/stylish")) => %(<link href="/stylesheets/random.styles" media="screen" rel="stylesheet" type="text/css" />\n<link href="/css/stylish.css" media="screen" rel="stylesheet" type="text/css" />),
|
||||
%(stylesheet_link_tag("http://www.example.com/styles/style")) => %(<link href="http://www.example.com/styles/style.css" media="screen" rel="stylesheet" type="text/css" />)
|
||||
@@ -343,6 +345,27 @@ class AssetTagHelperTest < ActionView::TestCase
|
||||
FileUtils.rm_f(File.join(ActionView::Helpers::AssetTagHelper::JAVASCRIPTS_DIR, 'cache', 'money.js'))
|
||||
end
|
||||
|
||||
def test_caching_javascript_include_tag_with_all_and_recursive_puts_defaults_at_the_start_of_the_file
|
||||
ENV["RAILS_ASSET_ID"] = ""
|
||||
ActionController::Base.asset_host = 'http://a0.example.com'
|
||||
ActionController::Base.perform_caching = true
|
||||
|
||||
assert_dom_equal(
|
||||
%(<script src="http://a0.example.com/javascripts/combined.js" type="text/javascript"></script>),
|
||||
javascript_include_tag(:all, :cache => "combined", :recursive => true)
|
||||
)
|
||||
|
||||
assert File.exist?(File.join(ActionView::Helpers::AssetTagHelper::JAVASCRIPTS_DIR, 'combined.js'))
|
||||
|
||||
assert_equal(
|
||||
%(// prototype js\n\n// effects js\n\n// dragdrop js\n\n// controls js\n\n// application js\n\n// bank js\n\n// robber js\n\n// subdir js\n\n\n// version.1.0 js),
|
||||
IO.read(File.join(ActionView::Helpers::AssetTagHelper::JAVASCRIPTS_DIR, 'combined.js'))
|
||||
)
|
||||
|
||||
ensure
|
||||
FileUtils.rm_f(File.join(ActionView::Helpers::AssetTagHelper::JAVASCRIPTS_DIR, 'combined.js'))
|
||||
end
|
||||
|
||||
def test_caching_javascript_include_tag_with_all_puts_defaults_at_the_start_of_the_file
|
||||
ENV["RAILS_ASSET_ID"] = ""
|
||||
ActionController::Base.asset_host = 'http://a0.example.com'
|
||||
@@ -373,6 +396,11 @@ class AssetTagHelperTest < ActionView::TestCase
|
||||
javascript_include_tag(:all, :cache => true)
|
||||
)
|
||||
|
||||
assert_dom_equal(
|
||||
%(<script src="/javascripts/prototype.js" type="text/javascript"></script>\n<script src="/javascripts/effects.js" type="text/javascript"></script>\n<script src="/javascripts/dragdrop.js" type="text/javascript"></script>\n<script src="/javascripts/controls.js" type="text/javascript"></script>\n<script src="/javascripts/application.js" type="text/javascript"></script>\n<script src="/javascripts/bank.js" type="text/javascript"></script>\n<script src="/javascripts/robber.js" type="text/javascript"></script>\n<script src="/javascripts/subdir/subdir.js" type="text/javascript"></script>\n<script src="/javascripts/version.1.0.js" type="text/javascript"></script>),
|
||||
javascript_include_tag(:all, :cache => true, :recursive => true)
|
||||
)
|
||||
|
||||
assert !File.exist?(File.join(ActionView::Helpers::AssetTagHelper::JAVASCRIPTS_DIR, 'all.js'))
|
||||
|
||||
assert_dom_equal(
|
||||
@@ -380,6 +408,11 @@ class AssetTagHelperTest < ActionView::TestCase
|
||||
javascript_include_tag(:all, :cache => "money")
|
||||
)
|
||||
|
||||
assert_dom_equal(
|
||||
%(<script src="/javascripts/prototype.js" type="text/javascript"></script>\n<script src="/javascripts/effects.js" type="text/javascript"></script>\n<script src="/javascripts/dragdrop.js" type="text/javascript"></script>\n<script src="/javascripts/controls.js" type="text/javascript"></script>\n<script src="/javascripts/application.js" type="text/javascript"></script>\n<script src="/javascripts/bank.js" type="text/javascript"></script>\n<script src="/javascripts/robber.js" type="text/javascript"></script>\n<script src="/javascripts/subdir/subdir.js" type="text/javascript"></script>\n<script src="/javascripts/version.1.0.js" type="text/javascript"></script>),
|
||||
javascript_include_tag(:all, :cache => "money", :recursive => true)
|
||||
)
|
||||
|
||||
assert !File.exist?(File.join(ActionView::Helpers::AssetTagHelper::JAVASCRIPTS_DIR, 'money.js'))
|
||||
end
|
||||
|
||||
@@ -432,6 +465,11 @@ class AssetTagHelperTest < ActionView::TestCase
|
||||
stylesheet_link_tag(:all, :cache => true)
|
||||
)
|
||||
|
||||
assert_dom_equal(
|
||||
%(<link href="/stylesheets/bank.css" media="screen" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/robber.css" media="screen" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/subdir/subdir.css" media="screen" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/version.1.0.css" media="screen" rel="stylesheet" type="text/css" />),
|
||||
stylesheet_link_tag(:all, :cache => true, :recursive => true)
|
||||
)
|
||||
|
||||
assert !File.exist?(File.join(ActionView::Helpers::AssetTagHelper::STYLESHEETS_DIR, 'all.css'))
|
||||
|
||||
assert_dom_equal(
|
||||
@@ -439,6 +477,11 @@ class AssetTagHelperTest < ActionView::TestCase
|
||||
stylesheet_link_tag(:all, :cache => "money")
|
||||
)
|
||||
|
||||
assert_dom_equal(
|
||||
%(<link href="/stylesheets/bank.css" media="screen" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/robber.css" media="screen" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/subdir/subdir.css" media="screen" rel="stylesheet" type="text/css" />\n<link href="/stylesheets/version.1.0.css" media="screen" rel="stylesheet" type="text/css" />),
|
||||
stylesheet_link_tag(:all, :cache => "money", :recursive => true)
|
||||
)
|
||||
|
||||
assert !File.exist?(File.join(ActionView::Helpers::AssetTagHelper::STYLESHEETS_DIR, 'money.css'))
|
||||
end
|
||||
end
|
||||
|
||||
@@ -231,6 +231,35 @@ uses_mocha "FormOptionsHelperTest" do
|
||||
)
|
||||
end
|
||||
|
||||
def test_select_under_fields_for_with_index
|
||||
@post = Post.new
|
||||
@post.category = "<mus>"
|
||||
|
||||
fields_for :post, @post, :index => 108 do |f|
|
||||
concat f.select(:category, %w( abe <mus> hest))
|
||||
end
|
||||
|
||||
assert_dom_equal(
|
||||
"<select id=\"post_108_category\" name=\"post[108][category]\"><option value=\"abe\">abe</option>\n<option value=\"<mus>\" selected=\"selected\"><mus></option>\n<option value=\"hest\">hest</option></select>",
|
||||
output_buffer
|
||||
)
|
||||
end
|
||||
|
||||
def test_select_under_fields_for_with_auto_index
|
||||
@post = Post.new
|
||||
@post.category = "<mus>"
|
||||
def @post.to_param; 108; end
|
||||
|
||||
fields_for "post[]", @post do |f|
|
||||
concat f.select(:category, %w( abe <mus> hest))
|
||||
end
|
||||
|
||||
assert_dom_equal(
|
||||
"<select id=\"post_108_category\" name=\"post[108][category]\"><option value=\"abe\">abe</option>\n<option value=\"<mus>\" selected=\"selected\"><mus></option>\n<option value=\"hest\">hest</option></select>",
|
||||
output_buffer
|
||||
)
|
||||
end
|
||||
|
||||
def test_select_with_blank
|
||||
@post = Post.new
|
||||
@post.category = "<mus>"
|
||||
@@ -351,6 +380,47 @@ uses_mocha "FormOptionsHelperTest" do
|
||||
)
|
||||
end
|
||||
|
||||
def test_collection_select_under_fields_for_with_index
|
||||
@posts = [
|
||||
Post.new("<Abe> went home", "<Abe>", "To a little house", "shh!"),
|
||||
Post.new("Babe went home", "Babe", "To a little house", "shh!"),
|
||||
Post.new("Cabe went home", "Cabe", "To a little house", "shh!")
|
||||
]
|
||||
|
||||
@post = Post.new
|
||||
@post.author_name = "Babe"
|
||||
|
||||
fields_for :post, @post, :index => 815 do |f|
|
||||
concat f.collection_select(:author_name, @posts, :author_name, :author_name)
|
||||
end
|
||||
|
||||
assert_dom_equal(
|
||||
"<select id=\"post_815_author_name\" name=\"post[815][author_name]\"><option value=\"<Abe>\"><Abe></option>\n<option value=\"Babe\" selected=\"selected\">Babe</option>\n<option value=\"Cabe\">Cabe</option></select>",
|
||||
output_buffer
|
||||
)
|
||||
end
|
||||
|
||||
def test_collection_select_under_fields_for_with_auto_index
|
||||
@posts = [
|
||||
Post.new("<Abe> went home", "<Abe>", "To a little house", "shh!"),
|
||||
Post.new("Babe went home", "Babe", "To a little house", "shh!"),
|
||||
Post.new("Cabe went home", "Cabe", "To a little house", "shh!")
|
||||
]
|
||||
|
||||
@post = Post.new
|
||||
@post.author_name = "Babe"
|
||||
def @post.to_param; 815; end
|
||||
|
||||
fields_for "post[]", @post do |f|
|
||||
concat f.collection_select(:author_name, @posts, :author_name, :author_name)
|
||||
end
|
||||
|
||||
assert_dom_equal(
|
||||
"<select id=\"post_815_author_name\" name=\"post[815][author_name]\"><option value=\"<Abe>\"><Abe></option>\n<option value=\"Babe\" selected=\"selected\">Babe</option>\n<option value=\"Cabe\">Cabe</option></select>",
|
||||
output_buffer
|
||||
)
|
||||
end
|
||||
|
||||
def test_collection_select_with_blank_and_style
|
||||
@posts = [
|
||||
Post.new("<Abe> went home", "<Abe>", "To a little house", "shh!"),
|
||||
@@ -1165,6 +1235,782 @@ uses_mocha "FormOptionsHelperTest" do
|
||||
assert_dom_equal(expected_select[0..-2], country_select("post", "origin", ["New Zealand", "Nicaragua"]))
|
||||
end
|
||||
|
||||
def test_country_select_under_fields_for
|
||||
@post = Post.new
|
||||
@post.origin = "Australia"
|
||||
expected_select = <<-COUNTRIES
|
||||
<select id="post_origin" name="post[origin]"><option value="Afghanistan">Afghanistan</option>
|
||||
<option value="Aland Islands">Aland Islands</option>
|
||||
<option value="Albania">Albania</option>
|
||||
<option value="Algeria">Algeria</option>
|
||||
<option value="American Samoa">American Samoa</option>
|
||||
<option value="Andorra">Andorra</option>
|
||||
<option value="Angola">Angola</option>
|
||||
<option value="Anguilla">Anguilla</option>
|
||||
<option value="Antarctica">Antarctica</option>
|
||||
<option value="Antigua And Barbuda">Antigua And Barbuda</option>
|
||||
<option value="Argentina">Argentina</option>
|
||||
<option value="Armenia">Armenia</option>
|
||||
<option value="Aruba">Aruba</option>
|
||||
<option selected="selected" value="Australia">Australia</option>
|
||||
<option value="Austria">Austria</option>
|
||||
<option value="Azerbaijan">Azerbaijan</option>
|
||||
<option value="Bahamas">Bahamas</option>
|
||||
<option value="Bahrain">Bahrain</option>
|
||||
<option value="Bangladesh">Bangladesh</option>
|
||||
<option value="Barbados">Barbados</option>
|
||||
<option value="Belarus">Belarus</option>
|
||||
<option value="Belgium">Belgium</option>
|
||||
<option value="Belize">Belize</option>
|
||||
<option value="Benin">Benin</option>
|
||||
<option value="Bermuda">Bermuda</option>
|
||||
<option value="Bhutan">Bhutan</option>
|
||||
<option value="Bolivia">Bolivia</option>
|
||||
<option value="Bosnia and Herzegowina">Bosnia and Herzegowina</option>
|
||||
<option value="Botswana">Botswana</option>
|
||||
<option value="Bouvet Island">Bouvet Island</option>
|
||||
<option value="Brazil">Brazil</option>
|
||||
<option value="British Indian Ocean Territory">British Indian Ocean Territory</option>
|
||||
<option value="Brunei Darussalam">Brunei Darussalam</option>
|
||||
<option value="Bulgaria">Bulgaria</option>
|
||||
<option value="Burkina Faso">Burkina Faso</option>
|
||||
<option value="Burundi">Burundi</option>
|
||||
<option value="Cambodia">Cambodia</option>
|
||||
<option value="Cameroon">Cameroon</option>
|
||||
<option value="Canada">Canada</option>
|
||||
<option value="Cape Verde">Cape Verde</option>
|
||||
<option value="Cayman Islands">Cayman Islands</option>
|
||||
<option value="Central African Republic">Central African Republic</option>
|
||||
<option value="Chad">Chad</option>
|
||||
<option value="Chile">Chile</option>
|
||||
<option value="China">China</option>
|
||||
<option value="Christmas Island">Christmas Island</option>
|
||||
<option value="Cocos (Keeling) Islands">Cocos (Keeling) Islands</option>
|
||||
<option value="Colombia">Colombia</option>
|
||||
<option value="Comoros">Comoros</option>
|
||||
<option value="Congo">Congo</option>
|
||||
<option value="Congo, the Democratic Republic of the">Congo, the Democratic Republic of the</option>
|
||||
<option value="Cook Islands">Cook Islands</option>
|
||||
<option value="Costa Rica">Costa Rica</option>
|
||||
<option value="Cote d'Ivoire">Cote d'Ivoire</option>
|
||||
<option value="Croatia">Croatia</option>
|
||||
<option value="Cuba">Cuba</option>
|
||||
<option value="Cyprus">Cyprus</option>
|
||||
<option value="Czech Republic">Czech Republic</option>
|
||||
<option value="Denmark">Denmark</option>
|
||||
<option value="Djibouti">Djibouti</option>
|
||||
<option value="Dominica">Dominica</option>
|
||||
<option value="Dominican Republic">Dominican Republic</option>
|
||||
<option value="Ecuador">Ecuador</option>
|
||||
<option value="Egypt">Egypt</option>
|
||||
<option value="El Salvador">El Salvador</option>
|
||||
<option value="Equatorial Guinea">Equatorial Guinea</option>
|
||||
<option value="Eritrea">Eritrea</option>
|
||||
<option value="Estonia">Estonia</option>
|
||||
<option value="Ethiopia">Ethiopia</option>
|
||||
<option value="Falkland Islands (Malvinas)">Falkland Islands (Malvinas)</option>
|
||||
<option value="Faroe Islands">Faroe Islands</option>
|
||||
<option value="Fiji">Fiji</option>
|
||||
<option value="Finland">Finland</option>
|
||||
<option value="France">France</option>
|
||||
<option value="French Guiana">French Guiana</option>
|
||||
<option value="French Polynesia">French Polynesia</option>
|
||||
<option value="French Southern Territories">French Southern Territories</option>
|
||||
<option value="Gabon">Gabon</option>
|
||||
<option value="Gambia">Gambia</option>
|
||||
<option value="Georgia">Georgia</option>
|
||||
<option value="Germany">Germany</option>
|
||||
<option value="Ghana">Ghana</option>
|
||||
<option value="Gibraltar">Gibraltar</option>
|
||||
<option value="Greece">Greece</option>
|
||||
<option value="Greenland">Greenland</option>
|
||||
<option value="Grenada">Grenada</option>
|
||||
<option value="Guadeloupe">Guadeloupe</option>
|
||||
<option value="Guam">Guam</option>
|
||||
<option value="Guatemala">Guatemala</option>
|
||||
<option value="Guernsey">Guernsey</option>
|
||||
<option value="Guinea">Guinea</option>
|
||||
<option value="Guinea-Bissau">Guinea-Bissau</option>
|
||||
<option value="Guyana">Guyana</option>
|
||||
<option value="Haiti">Haiti</option>
|
||||
<option value="Heard and McDonald Islands">Heard and McDonald Islands</option>
|
||||
<option value="Holy See (Vatican City State)">Holy See (Vatican City State)</option>
|
||||
<option value="Honduras">Honduras</option>
|
||||
<option value="Hong Kong">Hong Kong</option>
|
||||
<option value="Hungary">Hungary</option>
|
||||
<option value="Iceland">Iceland</option>
|
||||
<option value="India">India</option>
|
||||
<option value="Indonesia">Indonesia</option>
|
||||
<option value="Iran, Islamic Republic of">Iran, Islamic Republic of</option>
|
||||
<option value="Iraq">Iraq</option>
|
||||
<option value="Ireland">Ireland</option>
|
||||
<option value="Isle of Man">Isle of Man</option>
|
||||
<option value="Israel">Israel</option>
|
||||
<option value="Italy">Italy</option>
|
||||
<option value="Jamaica">Jamaica</option>
|
||||
<option value="Japan">Japan</option>
|
||||
<option value="Jersey">Jersey</option>
|
||||
<option value="Jordan">Jordan</option>
|
||||
<option value="Kazakhstan">Kazakhstan</option>
|
||||
<option value="Kenya">Kenya</option>
|
||||
<option value="Kiribati">Kiribati</option>
|
||||
<option value="Korea, Democratic People's Republic of">Korea, Democratic People's Republic of</option>
|
||||
<option value="Korea, Republic of">Korea, Republic of</option>
|
||||
<option value="Kuwait">Kuwait</option>
|
||||
<option value="Kyrgyzstan">Kyrgyzstan</option>
|
||||
<option value="Lao People's Democratic Republic">Lao People's Democratic Republic</option>
|
||||
<option value="Latvia">Latvia</option>
|
||||
<option value="Lebanon">Lebanon</option>
|
||||
<option value="Lesotho">Lesotho</option>
|
||||
<option value="Liberia">Liberia</option>
|
||||
<option value="Libyan Arab Jamahiriya">Libyan Arab Jamahiriya</option>
|
||||
<option value="Liechtenstein">Liechtenstein</option>
|
||||
<option value="Lithuania">Lithuania</option>
|
||||
<option value="Luxembourg">Luxembourg</option>
|
||||
<option value="Macao">Macao</option>
|
||||
<option value="Macedonia, The Former Yugoslav Republic Of">Macedonia, The Former Yugoslav Republic Of</option>
|
||||
<option value="Madagascar">Madagascar</option>
|
||||
<option value="Malawi">Malawi</option>
|
||||
<option value="Malaysia">Malaysia</option>
|
||||
<option value="Maldives">Maldives</option>
|
||||
<option value="Mali">Mali</option>
|
||||
<option value="Malta">Malta</option>
|
||||
<option value="Marshall Islands">Marshall Islands</option>
|
||||
<option value="Martinique">Martinique</option>
|
||||
<option value="Mauritania">Mauritania</option>
|
||||
<option value="Mauritius">Mauritius</option>
|
||||
<option value="Mayotte">Mayotte</option>
|
||||
<option value="Mexico">Mexico</option>
|
||||
<option value="Micronesia, Federated States of">Micronesia, Federated States of</option>
|
||||
<option value="Moldova, Republic of">Moldova, Republic of</option>
|
||||
<option value="Monaco">Monaco</option>
|
||||
<option value="Mongolia">Mongolia</option>
|
||||
<option value="Montenegro">Montenegro</option>
|
||||
<option value="Montserrat">Montserrat</option>
|
||||
<option value="Morocco">Morocco</option>
|
||||
<option value="Mozambique">Mozambique</option>
|
||||
<option value="Myanmar">Myanmar</option>
|
||||
<option value="Namibia">Namibia</option>
|
||||
<option value="Nauru">Nauru</option>
|
||||
<option value="Nepal">Nepal</option>
|
||||
<option value="Netherlands">Netherlands</option>
|
||||
<option value="Netherlands Antilles">Netherlands Antilles</option>
|
||||
<option value="New Caledonia">New Caledonia</option>
|
||||
<option value="New Zealand">New Zealand</option>
|
||||
<option value="Nicaragua">Nicaragua</option>
|
||||
<option value="Niger">Niger</option>
|
||||
<option value="Nigeria">Nigeria</option>
|
||||
<option value="Niue">Niue</option>
|
||||
<option value="Norfolk Island">Norfolk Island</option>
|
||||
<option value="Northern Mariana Islands">Northern Mariana Islands</option>
|
||||
<option value="Norway">Norway</option>
|
||||
<option value="Oman">Oman</option>
|
||||
<option value="Pakistan">Pakistan</option>
|
||||
<option value="Palau">Palau</option>
|
||||
<option value="Palestinian Territory, Occupied">Palestinian Territory, Occupied</option>
|
||||
<option value="Panama">Panama</option>
|
||||
<option value="Papua New Guinea">Papua New Guinea</option>
|
||||
<option value="Paraguay">Paraguay</option>
|
||||
<option value="Peru">Peru</option>
|
||||
<option value="Philippines">Philippines</option>
|
||||
<option value="Pitcairn">Pitcairn</option>
|
||||
<option value="Poland">Poland</option>
|
||||
<option value="Portugal">Portugal</option>
|
||||
<option value="Puerto Rico">Puerto Rico</option>
|
||||
<option value="Qatar">Qatar</option>
|
||||
<option value="Reunion">Reunion</option>
|
||||
<option value="Romania">Romania</option>
|
||||
<option value="Russian Federation">Russian Federation</option>
|
||||
<option value="Rwanda">Rwanda</option>
|
||||
<option value="Saint Barthelemy">Saint Barthelemy</option>
|
||||
<option value="Saint Helena">Saint Helena</option>
|
||||
<option value="Saint Kitts and Nevis">Saint Kitts and Nevis</option>
|
||||
<option value="Saint Lucia">Saint Lucia</option>
|
||||
<option value="Saint Pierre and Miquelon">Saint Pierre and Miquelon</option>
|
||||
<option value="Saint Vincent and the Grenadines">Saint Vincent and the Grenadines</option>
|
||||
<option value="Samoa">Samoa</option>
|
||||
<option value="San Marino">San Marino</option>
|
||||
<option value="Sao Tome and Principe">Sao Tome and Principe</option>
|
||||
<option value="Saudi Arabia">Saudi Arabia</option>
|
||||
<option value="Senegal">Senegal</option>
|
||||
<option value="Serbia">Serbia</option>
|
||||
<option value="Seychelles">Seychelles</option>
|
||||
<option value="Sierra Leone">Sierra Leone</option>
|
||||
<option value="Singapore">Singapore</option>
|
||||
<option value="Slovakia">Slovakia</option>
|
||||
<option value="Slovenia">Slovenia</option>
|
||||
<option value="Solomon Islands">Solomon Islands</option>
|
||||
<option value="Somalia">Somalia</option>
|
||||
<option value="South Africa">South Africa</option>
|
||||
<option value="South Georgia and the South Sandwich Islands">South Georgia and the South Sandwich Islands</option>
|
||||
<option value="Spain">Spain</option>
|
||||
<option value="Sri Lanka">Sri Lanka</option>
|
||||
<option value="Sudan">Sudan</option>
|
||||
<option value="Suriname">Suriname</option>
|
||||
<option value="Svalbard and Jan Mayen">Svalbard and Jan Mayen</option>
|
||||
<option value="Swaziland">Swaziland</option>
|
||||
<option value="Sweden">Sweden</option>
|
||||
<option value="Switzerland">Switzerland</option>
|
||||
<option value="Syrian Arab Republic">Syrian Arab Republic</option>
|
||||
<option value="Taiwan, Province of China">Taiwan, Province of China</option>
|
||||
<option value="Tajikistan">Tajikistan</option>
|
||||
<option value="Tanzania, United Republic of">Tanzania, United Republic of</option>
|
||||
<option value="Thailand">Thailand</option>
|
||||
<option value="Timor-Leste">Timor-Leste</option>
|
||||
<option value="Togo">Togo</option>
|
||||
<option value="Tokelau">Tokelau</option>
|
||||
<option value="Tonga">Tonga</option>
|
||||
<option value="Trinidad and Tobago">Trinidad and Tobago</option>
|
||||
<option value="Tunisia">Tunisia</option>
|
||||
<option value="Turkey">Turkey</option>
|
||||
<option value="Turkmenistan">Turkmenistan</option>
|
||||
<option value="Turks and Caicos Islands">Turks and Caicos Islands</option>
|
||||
<option value="Tuvalu">Tuvalu</option>
|
||||
<option value="Uganda">Uganda</option>
|
||||
<option value="Ukraine">Ukraine</option>
|
||||
<option value="United Arab Emirates">United Arab Emirates</option>
|
||||
<option value="United Kingdom">United Kingdom</option>
|
||||
<option value="United States">United States</option>
|
||||
<option value="United States Minor Outlying Islands">United States Minor Outlying Islands</option>
|
||||
<option value="Uruguay">Uruguay</option>
|
||||
<option value="Uzbekistan">Uzbekistan</option>
|
||||
<option value="Vanuatu">Vanuatu</option>
|
||||
<option value="Venezuela">Venezuela</option>
|
||||
<option value="Viet Nam">Viet Nam</option>
|
||||
<option value="Virgin Islands, British">Virgin Islands, British</option>
|
||||
<option value="Virgin Islands, U.S.">Virgin Islands, U.S.</option>
|
||||
<option value="Wallis and Futuna">Wallis and Futuna</option>
|
||||
<option value="Western Sahara">Western Sahara</option>
|
||||
<option value="Yemen">Yemen</option>
|
||||
<option value="Zambia">Zambia</option>
|
||||
<option value="Zimbabwe">Zimbabwe</option></select>
|
||||
COUNTRIES
|
||||
|
||||
fields_for :post, @post do |f|
|
||||
concat f.country_select("origin")
|
||||
end
|
||||
|
||||
assert_dom_equal(expected_select[0..-2], output_buffer)
|
||||
end
|
||||
|
||||
def test_country_select_under_fields_for_with_index
|
||||
@post = Post.new
|
||||
@post.origin = "United States"
|
||||
expected_select = <<-COUNTRIES
|
||||
<select id="post_325_origin" name="post[325][origin]"><option value="Afghanistan">Afghanistan</option>
|
||||
<option value="Aland Islands">Aland Islands</option>
|
||||
<option value="Albania">Albania</option>
|
||||
<option value="Algeria">Algeria</option>
|
||||
<option value="American Samoa">American Samoa</option>
|
||||
<option value="Andorra">Andorra</option>
|
||||
<option value="Angola">Angola</option>
|
||||
<option value="Anguilla">Anguilla</option>
|
||||
<option value="Antarctica">Antarctica</option>
|
||||
<option value="Antigua And Barbuda">Antigua And Barbuda</option>
|
||||
<option value="Argentina">Argentina</option>
|
||||
<option value="Armenia">Armenia</option>
|
||||
<option value="Aruba">Aruba</option>
|
||||
<option value="Australia">Australia</option>
|
||||
<option value="Austria">Austria</option>
|
||||
<option value="Azerbaijan">Azerbaijan</option>
|
||||
<option value="Bahamas">Bahamas</option>
|
||||
<option value="Bahrain">Bahrain</option>
|
||||
<option value="Bangladesh">Bangladesh</option>
|
||||
<option value="Barbados">Barbados</option>
|
||||
<option value="Belarus">Belarus</option>
|
||||
<option value="Belgium">Belgium</option>
|
||||
<option value="Belize">Belize</option>
|
||||
<option value="Benin">Benin</option>
|
||||
<option value="Bermuda">Bermuda</option>
|
||||
<option value="Bhutan">Bhutan</option>
|
||||
<option value="Bolivia">Bolivia</option>
|
||||
<option value="Bosnia and Herzegowina">Bosnia and Herzegowina</option>
|
||||
<option value="Botswana">Botswana</option>
|
||||
<option value="Bouvet Island">Bouvet Island</option>
|
||||
<option value="Brazil">Brazil</option>
|
||||
<option value="British Indian Ocean Territory">British Indian Ocean Territory</option>
|
||||
<option value="Brunei Darussalam">Brunei Darussalam</option>
|
||||
<option value="Bulgaria">Bulgaria</option>
|
||||
<option value="Burkina Faso">Burkina Faso</option>
|
||||
<option value="Burundi">Burundi</option>
|
||||
<option value="Cambodia">Cambodia</option>
|
||||
<option value="Cameroon">Cameroon</option>
|
||||
<option value="Canada">Canada</option>
|
||||
<option value="Cape Verde">Cape Verde</option>
|
||||
<option value="Cayman Islands">Cayman Islands</option>
|
||||
<option value="Central African Republic">Central African Republic</option>
|
||||
<option value="Chad">Chad</option>
|
||||
<option value="Chile">Chile</option>
|
||||
<option value="China">China</option>
|
||||
<option value="Christmas Island">Christmas Island</option>
|
||||
<option value="Cocos (Keeling) Islands">Cocos (Keeling) Islands</option>
|
||||
<option value="Colombia">Colombia</option>
|
||||
<option value="Comoros">Comoros</option>
|
||||
<option value="Congo">Congo</option>
|
||||
<option value="Congo, the Democratic Republic of the">Congo, the Democratic Republic of the</option>
|
||||
<option value="Cook Islands">Cook Islands</option>
|
||||
<option value="Costa Rica">Costa Rica</option>
|
||||
<option value="Cote d'Ivoire">Cote d'Ivoire</option>
|
||||
<option value="Croatia">Croatia</option>
|
||||
<option value="Cuba">Cuba</option>
|
||||
<option value="Cyprus">Cyprus</option>
|
||||
<option value="Czech Republic">Czech Republic</option>
|
||||
<option value="Denmark">Denmark</option>
|
||||
<option value="Djibouti">Djibouti</option>
|
||||
<option value="Dominica">Dominica</option>
|
||||
<option value="Dominican Republic">Dominican Republic</option>
|
||||
<option value="Ecuador">Ecuador</option>
|
||||
<option value="Egypt">Egypt</option>
|
||||
<option value="El Salvador">El Salvador</option>
|
||||
<option value="Equatorial Guinea">Equatorial Guinea</option>
|
||||
<option value="Eritrea">Eritrea</option>
|
||||
<option value="Estonia">Estonia</option>
|
||||
<option value="Ethiopia">Ethiopia</option>
|
||||
<option value="Falkland Islands (Malvinas)">Falkland Islands (Malvinas)</option>
|
||||
<option value="Faroe Islands">Faroe Islands</option>
|
||||
<option value="Fiji">Fiji</option>
|
||||
<option value="Finland">Finland</option>
|
||||
<option value="France">France</option>
|
||||
<option value="French Guiana">French Guiana</option>
|
||||
<option value="French Polynesia">French Polynesia</option>
|
||||
<option value="French Southern Territories">French Southern Territories</option>
|
||||
<option value="Gabon">Gabon</option>
|
||||
<option value="Gambia">Gambia</option>
|
||||
<option value="Georgia">Georgia</option>
|
||||
<option value="Germany">Germany</option>
|
||||
<option value="Ghana">Ghana</option>
|
||||
<option value="Gibraltar">Gibraltar</option>
|
||||
<option value="Greece">Greece</option>
|
||||
<option value="Greenland">Greenland</option>
|
||||
<option value="Grenada">Grenada</option>
|
||||
<option value="Guadeloupe">Guadeloupe</option>
|
||||
<option value="Guam">Guam</option>
|
||||
<option value="Guatemala">Guatemala</option>
|
||||
<option value="Guernsey">Guernsey</option>
|
||||
<option value="Guinea">Guinea</option>
|
||||
<option value="Guinea-Bissau">Guinea-Bissau</option>
|
||||
<option value="Guyana">Guyana</option>
|
||||
<option value="Haiti">Haiti</option>
|
||||
<option value="Heard and McDonald Islands">Heard and McDonald Islands</option>
|
||||
<option value="Holy See (Vatican City State)">Holy See (Vatican City State)</option>
|
||||
<option value="Honduras">Honduras</option>
|
||||
<option value="Hong Kong">Hong Kong</option>
|
||||
<option value="Hungary">Hungary</option>
|
||||
<option value="Iceland">Iceland</option>
|
||||
<option value="India">India</option>
|
||||
<option value="Indonesia">Indonesia</option>
|
||||
<option value="Iran, Islamic Republic of">Iran, Islamic Republic of</option>
|
||||
<option value="Iraq">Iraq</option>
|
||||
<option value="Ireland">Ireland</option>
|
||||
<option value="Isle of Man">Isle of Man</option>
|
||||
<option value="Israel">Israel</option>
|
||||
<option value="Italy">Italy</option>
|
||||
<option value="Jamaica">Jamaica</option>
|
||||
<option value="Japan">Japan</option>
|
||||
<option value="Jersey">Jersey</option>
|
||||
<option value="Jordan">Jordan</option>
|
||||
<option value="Kazakhstan">Kazakhstan</option>
|
||||
<option value="Kenya">Kenya</option>
|
||||
<option value="Kiribati">Kiribati</option>
|
||||
<option value="Korea, Democratic People's Republic of">Korea, Democratic People's Republic of</option>
|
||||
<option value="Korea, Republic of">Korea, Republic of</option>
|
||||
<option value="Kuwait">Kuwait</option>
|
||||
<option value="Kyrgyzstan">Kyrgyzstan</option>
|
||||
<option value="Lao People's Democratic Republic">Lao People's Democratic Republic</option>
|
||||
<option value="Latvia">Latvia</option>
|
||||
<option value="Lebanon">Lebanon</option>
|
||||
<option value="Lesotho">Lesotho</option>
|
||||
<option value="Liberia">Liberia</option>
|
||||
<option value="Libyan Arab Jamahiriya">Libyan Arab Jamahiriya</option>
|
||||
<option value="Liechtenstein">Liechtenstein</option>
|
||||
<option value="Lithuania">Lithuania</option>
|
||||
<option value="Luxembourg">Luxembourg</option>
|
||||
<option value="Macao">Macao</option>
|
||||
<option value="Macedonia, The Former Yugoslav Republic Of">Macedonia, The Former Yugoslav Republic Of</option>
|
||||
<option value="Madagascar">Madagascar</option>
|
||||
<option value="Malawi">Malawi</option>
|
||||
<option value="Malaysia">Malaysia</option>
|
||||
<option value="Maldives">Maldives</option>
|
||||
<option value="Mali">Mali</option>
|
||||
<option value="Malta">Malta</option>
|
||||
<option value="Marshall Islands">Marshall Islands</option>
|
||||
<option value="Martinique">Martinique</option>
|
||||
<option value="Mauritania">Mauritania</option>
|
||||
<option value="Mauritius">Mauritius</option>
|
||||
<option value="Mayotte">Mayotte</option>
|
||||
<option value="Mexico">Mexico</option>
|
||||
<option value="Micronesia, Federated States of">Micronesia, Federated States of</option>
|
||||
<option value="Moldova, Republic of">Moldova, Republic of</option>
|
||||
<option value="Monaco">Monaco</option>
|
||||
<option value="Mongolia">Mongolia</option>
|
||||
<option value="Montenegro">Montenegro</option>
|
||||
<option value="Montserrat">Montserrat</option>
|
||||
<option value="Morocco">Morocco</option>
|
||||
<option value="Mozambique">Mozambique</option>
|
||||
<option value="Myanmar">Myanmar</option>
|
||||
<option value="Namibia">Namibia</option>
|
||||
<option value="Nauru">Nauru</option>
|
||||
<option value="Nepal">Nepal</option>
|
||||
<option value="Netherlands">Netherlands</option>
|
||||
<option value="Netherlands Antilles">Netherlands Antilles</option>
|
||||
<option value="New Caledonia">New Caledonia</option>
|
||||
<option value="New Zealand">New Zealand</option>
|
||||
<option value="Nicaragua">Nicaragua</option>
|
||||
<option value="Niger">Niger</option>
|
||||
<option value="Nigeria">Nigeria</option>
|
||||
<option value="Niue">Niue</option>
|
||||
<option value="Norfolk Island">Norfolk Island</option>
|
||||
<option value="Northern Mariana Islands">Northern Mariana Islands</option>
|
||||
<option value="Norway">Norway</option>
|
||||
<option value="Oman">Oman</option>
|
||||
<option value="Pakistan">Pakistan</option>
|
||||
<option value="Palau">Palau</option>
|
||||
<option value="Palestinian Territory, Occupied">Palestinian Territory, Occupied</option>
|
||||
<option value="Panama">Panama</option>
|
||||
<option value="Papua New Guinea">Papua New Guinea</option>
|
||||
<option value="Paraguay">Paraguay</option>
|
||||
<option value="Peru">Peru</option>
|
||||
<option value="Philippines">Philippines</option>
|
||||
<option value="Pitcairn">Pitcairn</option>
|
||||
<option value="Poland">Poland</option>
|
||||
<option value="Portugal">Portugal</option>
|
||||
<option value="Puerto Rico">Puerto Rico</option>
|
||||
<option value="Qatar">Qatar</option>
|
||||
<option value="Reunion">Reunion</option>
|
||||
<option value="Romania">Romania</option>
|
||||
<option value="Russian Federation">Russian Federation</option>
|
||||
<option value="Rwanda">Rwanda</option>
|
||||
<option value="Saint Barthelemy">Saint Barthelemy</option>
|
||||
<option value="Saint Helena">Saint Helena</option>
|
||||
<option value="Saint Kitts and Nevis">Saint Kitts and Nevis</option>
|
||||
<option value="Saint Lucia">Saint Lucia</option>
|
||||
<option value="Saint Pierre and Miquelon">Saint Pierre and Miquelon</option>
|
||||
<option value="Saint Vincent and the Grenadines">Saint Vincent and the Grenadines</option>
|
||||
<option value="Samoa">Samoa</option>
|
||||
<option value="San Marino">San Marino</option>
|
||||
<option value="Sao Tome and Principe">Sao Tome and Principe</option>
|
||||
<option value="Saudi Arabia">Saudi Arabia</option>
|
||||
<option value="Senegal">Senegal</option>
|
||||
<option value="Serbia">Serbia</option>
|
||||
<option value="Seychelles">Seychelles</option>
|
||||
<option value="Sierra Leone">Sierra Leone</option>
|
||||
<option value="Singapore">Singapore</option>
|
||||
<option value="Slovakia">Slovakia</option>
|
||||
<option value="Slovenia">Slovenia</option>
|
||||
<option value="Solomon Islands">Solomon Islands</option>
|
||||
<option value="Somalia">Somalia</option>
|
||||
<option value="South Africa">South Africa</option>
|
||||
<option value="South Georgia and the South Sandwich Islands">South Georgia and the South Sandwich Islands</option>
|
||||
<option value="Spain">Spain</option>
|
||||
<option value="Sri Lanka">Sri Lanka</option>
|
||||
<option value="Sudan">Sudan</option>
|
||||
<option value="Suriname">Suriname</option>
|
||||
<option value="Svalbard and Jan Mayen">Svalbard and Jan Mayen</option>
|
||||
<option value="Swaziland">Swaziland</option>
|
||||
<option value="Sweden">Sweden</option>
|
||||
<option value="Switzerland">Switzerland</option>
|
||||
<option value="Syrian Arab Republic">Syrian Arab Republic</option>
|
||||
<option value="Taiwan, Province of China">Taiwan, Province of China</option>
|
||||
<option value="Tajikistan">Tajikistan</option>
|
||||
<option value="Tanzania, United Republic of">Tanzania, United Republic of</option>
|
||||
<option value="Thailand">Thailand</option>
|
||||
<option value="Timor-Leste">Timor-Leste</option>
|
||||
<option value="Togo">Togo</option>
|
||||
<option value="Tokelau">Tokelau</option>
|
||||
<option value="Tonga">Tonga</option>
|
||||
<option value="Trinidad and Tobago">Trinidad and Tobago</option>
|
||||
<option value="Tunisia">Tunisia</option>
|
||||
<option value="Turkey">Turkey</option>
|
||||
<option value="Turkmenistan">Turkmenistan</option>
|
||||
<option value="Turks and Caicos Islands">Turks and Caicos Islands</option>
|
||||
<option value="Tuvalu">Tuvalu</option>
|
||||
<option value="Uganda">Uganda</option>
|
||||
<option value="Ukraine">Ukraine</option>
|
||||
<option value="United Arab Emirates">United Arab Emirates</option>
|
||||
<option value="United Kingdom">United Kingdom</option>
|
||||
<option selected="selected" value="United States">United States</option>
|
||||
<option value="United States Minor Outlying Islands">United States Minor Outlying Islands</option>
|
||||
<option value="Uruguay">Uruguay</option>
|
||||
<option value="Uzbekistan">Uzbekistan</option>
|
||||
<option value="Vanuatu">Vanuatu</option>
|
||||
<option value="Venezuela">Venezuela</option>
|
||||
<option value="Viet Nam">Viet Nam</option>
|
||||
<option value="Virgin Islands, British">Virgin Islands, British</option>
|
||||
<option value="Virgin Islands, U.S.">Virgin Islands, U.S.</option>
|
||||
<option value="Wallis and Futuna">Wallis and Futuna</option>
|
||||
<option value="Western Sahara">Western Sahara</option>
|
||||
<option value="Yemen">Yemen</option>
|
||||
<option value="Zambia">Zambia</option>
|
||||
<option value="Zimbabwe">Zimbabwe</option></select>
|
||||
COUNTRIES
|
||||
|
||||
fields_for :post, @post, :index => 325 do |f|
|
||||
concat f.country_select("origin")
|
||||
end
|
||||
|
||||
assert_dom_equal(expected_select[0..-2], output_buffer)
|
||||
end
|
||||
|
||||
def test_country_select_under_fields_for_with_auto_index
|
||||
@post = Post.new
|
||||
@post.origin = "Iraq"
|
||||
def @post.to_param; 325; end
|
||||
|
||||
expected_select = <<-COUNTRIES
|
||||
<select id="post_325_origin" name="post[325][origin]"><option value="Afghanistan">Afghanistan</option>
|
||||
<option value="Aland Islands">Aland Islands</option>
|
||||
<option value="Albania">Albania</option>
|
||||
<option value="Algeria">Algeria</option>
|
||||
<option value="American Samoa">American Samoa</option>
|
||||
<option value="Andorra">Andorra</option>
|
||||
<option value="Angola">Angola</option>
|
||||
<option value="Anguilla">Anguilla</option>
|
||||
<option value="Antarctica">Antarctica</option>
|
||||
<option value="Antigua And Barbuda">Antigua And Barbuda</option>
|
||||
<option value="Argentina">Argentina</option>
|
||||
<option value="Armenia">Armenia</option>
|
||||
<option value="Aruba">Aruba</option>
|
||||
<option value="Australia">Australia</option>
|
||||
<option value="Austria">Austria</option>
|
||||
<option value="Azerbaijan">Azerbaijan</option>
|
||||
<option value="Bahamas">Bahamas</option>
|
||||
<option value="Bahrain">Bahrain</option>
|
||||
<option value="Bangladesh">Bangladesh</option>
|
||||
<option value="Barbados">Barbados</option>
|
||||
<option value="Belarus">Belarus</option>
|
||||
<option value="Belgium">Belgium</option>
|
||||
<option value="Belize">Belize</option>
|
||||
<option value="Benin">Benin</option>
|
||||
<option value="Bermuda">Bermuda</option>
|
||||
<option value="Bhutan">Bhutan</option>
|
||||
<option value="Bolivia">Bolivia</option>
|
||||
<option value="Bosnia and Herzegowina">Bosnia and Herzegowina</option>
|
||||
<option value="Botswana">Botswana</option>
|
||||
<option value="Bouvet Island">Bouvet Island</option>
|
||||
<option value="Brazil">Brazil</option>
|
||||
<option value="British Indian Ocean Territory">British Indian Ocean Territory</option>
|
||||
<option value="Brunei Darussalam">Brunei Darussalam</option>
|
||||
<option value="Bulgaria">Bulgaria</option>
|
||||
<option value="Burkina Faso">Burkina Faso</option>
|
||||
<option value="Burundi">Burundi</option>
|
||||
<option value="Cambodia">Cambodia</option>
|
||||
<option value="Cameroon">Cameroon</option>
|
||||
<option value="Canada">Canada</option>
|
||||
<option value="Cape Verde">Cape Verde</option>
|
||||
<option value="Cayman Islands">Cayman Islands</option>
|
||||
<option value="Central African Republic">Central African Republic</option>
|
||||
<option value="Chad">Chad</option>
|
||||
<option value="Chile">Chile</option>
|
||||
<option value="China">China</option>
|
||||
<option value="Christmas Island">Christmas Island</option>
|
||||
<option value="Cocos (Keeling) Islands">Cocos (Keeling) Islands</option>
|
||||
<option value="Colombia">Colombia</option>
|
||||
<option value="Comoros">Comoros</option>
|
||||
<option value="Congo">Congo</option>
|
||||
<option value="Congo, the Democratic Republic of the">Congo, the Democratic Republic of the</option>
|
||||
<option value="Cook Islands">Cook Islands</option>
|
||||
<option value="Costa Rica">Costa Rica</option>
|
||||
<option value="Cote d'Ivoire">Cote d'Ivoire</option>
|
||||
<option value="Croatia">Croatia</option>
|
||||
<option value="Cuba">Cuba</option>
|
||||
<option value="Cyprus">Cyprus</option>
|
||||
<option value="Czech Republic">Czech Republic</option>
|
||||
<option value="Denmark">Denmark</option>
|
||||
<option value="Djibouti">Djibouti</option>
|
||||
<option value="Dominica">Dominica</option>
|
||||
<option value="Dominican Republic">Dominican Republic</option>
|
||||
<option value="Ecuador">Ecuador</option>
|
||||
<option value="Egypt">Egypt</option>
|
||||
<option value="El Salvador">El Salvador</option>
|
||||
<option value="Equatorial Guinea">Equatorial Guinea</option>
|
||||
<option value="Eritrea">Eritrea</option>
|
||||
<option value="Estonia">Estonia</option>
|
||||
<option value="Ethiopia">Ethiopia</option>
|
||||
<option value="Falkland Islands (Malvinas)">Falkland Islands (Malvinas)</option>
|
||||
<option value="Faroe Islands">Faroe Islands</option>
|
||||
<option value="Fiji">Fiji</option>
|
||||
<option value="Finland">Finland</option>
|
||||
<option value="France">France</option>
|
||||
<option value="French Guiana">French Guiana</option>
|
||||
<option value="French Polynesia">French Polynesia</option>
|
||||
<option value="French Southern Territories">French Southern Territories</option>
|
||||
<option value="Gabon">Gabon</option>
|
||||
<option value="Gambia">Gambia</option>
|
||||
<option value="Georgia">Georgia</option>
|
||||
<option value="Germany">Germany</option>
|
||||
<option value="Ghana">Ghana</option>
|
||||
<option value="Gibraltar">Gibraltar</option>
|
||||
<option value="Greece">Greece</option>
|
||||
<option value="Greenland">Greenland</option>
|
||||
<option value="Grenada">Grenada</option>
|
||||
<option value="Guadeloupe">Guadeloupe</option>
|
||||
<option value="Guam">Guam</option>
|
||||
<option value="Guatemala">Guatemala</option>
|
||||
<option value="Guernsey">Guernsey</option>
|
||||
<option value="Guinea">Guinea</option>
|
||||
<option value="Guinea-Bissau">Guinea-Bissau</option>
|
||||
<option value="Guyana">Guyana</option>
|
||||
<option value="Haiti">Haiti</option>
|
||||
<option value="Heard and McDonald Islands">Heard and McDonald Islands</option>
|
||||
<option value="Holy See (Vatican City State)">Holy See (Vatican City State)</option>
|
||||
<option value="Honduras">Honduras</option>
|
||||
<option value="Hong Kong">Hong Kong</option>
|
||||
<option value="Hungary">Hungary</option>
|
||||
<option value="Iceland">Iceland</option>
|
||||
<option value="India">India</option>
|
||||
<option value="Indonesia">Indonesia</option>
|
||||
<option value="Iran, Islamic Republic of">Iran, Islamic Republic of</option>
|
||||
<option selected="selected" value="Iraq">Iraq</option>
|
||||
<option value="Ireland">Ireland</option>
|
||||
<option value="Isle of Man">Isle of Man</option>
|
||||
<option value="Israel">Israel</option>
|
||||
<option value="Italy">Italy</option>
|
||||
<option value="Jamaica">Jamaica</option>
|
||||
<option value="Japan">Japan</option>
|
||||
<option value="Jersey">Jersey</option>
|
||||
<option value="Jordan">Jordan</option>
|
||||
<option value="Kazakhstan">Kazakhstan</option>
|
||||
<option value="Kenya">Kenya</option>
|
||||
<option value="Kiribati">Kiribati</option>
|
||||
<option value="Korea, Democratic People's Republic of">Korea, Democratic People's Republic of</option>
|
||||
<option value="Korea, Republic of">Korea, Republic of</option>
|
||||
<option value="Kuwait">Kuwait</option>
|
||||
<option value="Kyrgyzstan">Kyrgyzstan</option>
|
||||
<option value="Lao People's Democratic Republic">Lao People's Democratic Republic</option>
|
||||
<option value="Latvia">Latvia</option>
|
||||
<option value="Lebanon">Lebanon</option>
|
||||
<option value="Lesotho">Lesotho</option>
|
||||
<option value="Liberia">Liberia</option>
|
||||
<option value="Libyan Arab Jamahiriya">Libyan Arab Jamahiriya</option>
|
||||
<option value="Liechtenstein">Liechtenstein</option>
|
||||
<option value="Lithuania">Lithuania</option>
|
||||
<option value="Luxembourg">Luxembourg</option>
|
||||
<option value="Macao">Macao</option>
|
||||
<option value="Macedonia, The Former Yugoslav Republic Of">Macedonia, The Former Yugoslav Republic Of</option>
|
||||
<option value="Madagascar">Madagascar</option>
|
||||
<option value="Malawi">Malawi</option>
|
||||
<option value="Malaysia">Malaysia</option>
|
||||
<option value="Maldives">Maldives</option>
|
||||
<option value="Mali">Mali</option>
|
||||
<option value="Malta">Malta</option>
|
||||
<option value="Marshall Islands">Marshall Islands</option>
|
||||
<option value="Martinique">Martinique</option>
|
||||
<option value="Mauritania">Mauritania</option>
|
||||
<option value="Mauritius">Mauritius</option>
|
||||
<option value="Mayotte">Mayotte</option>
|
||||
<option value="Mexico">Mexico</option>
|
||||
<option value="Micronesia, Federated States of">Micronesia, Federated States of</option>
|
||||
<option value="Moldova, Republic of">Moldova, Republic of</option>
|
||||
<option value="Monaco">Monaco</option>
|
||||
<option value="Mongolia">Mongolia</option>
|
||||
<option value="Montenegro">Montenegro</option>
|
||||
<option value="Montserrat">Montserrat</option>
|
||||
<option value="Morocco">Morocco</option>
|
||||
<option value="Mozambique">Mozambique</option>
|
||||
<option value="Myanmar">Myanmar</option>
|
||||
<option value="Namibia">Namibia</option>
|
||||
<option value="Nauru">Nauru</option>
|
||||
<option value="Nepal">Nepal</option>
|
||||
<option value="Netherlands">Netherlands</option>
|
||||
<option value="Netherlands Antilles">Netherlands Antilles</option>
|
||||
<option value="New Caledonia">New Caledonia</option>
|
||||
<option value="New Zealand">New Zealand</option>
|
||||
<option value="Nicaragua">Nicaragua</option>
|
||||
<option value="Niger">Niger</option>
|
||||
<option value="Nigeria">Nigeria</option>
|
||||
<option value="Niue">Niue</option>
|
||||
<option value="Norfolk Island">Norfolk Island</option>
|
||||
<option value="Northern Mariana Islands">Northern Mariana Islands</option>
|
||||
<option value="Norway">Norway</option>
|
||||
<option value="Oman">Oman</option>
|
||||
<option value="Pakistan">Pakistan</option>
|
||||
<option value="Palau">Palau</option>
|
||||
<option value="Palestinian Territory, Occupied">Palestinian Territory, Occupied</option>
|
||||
<option value="Panama">Panama</option>
|
||||
<option value="Papua New Guinea">Papua New Guinea</option>
|
||||
<option value="Paraguay">Paraguay</option>
|
||||
<option value="Peru">Peru</option>
|
||||
<option value="Philippines">Philippines</option>
|
||||
<option value="Pitcairn">Pitcairn</option>
|
||||
<option value="Poland">Poland</option>
|
||||
<option value="Portugal">Portugal</option>
|
||||
<option value="Puerto Rico">Puerto Rico</option>
|
||||
<option value="Qatar">Qatar</option>
|
||||
<option value="Reunion">Reunion</option>
|
||||
<option value="Romania">Romania</option>
|
||||
<option value="Russian Federation">Russian Federation</option>
|
||||
<option value="Rwanda">Rwanda</option>
|
||||
<option value="Saint Barthelemy">Saint Barthelemy</option>
|
||||
<option value="Saint Helena">Saint Helena</option>
|
||||
<option value="Saint Kitts and Nevis">Saint Kitts and Nevis</option>
|
||||
<option value="Saint Lucia">Saint Lucia</option>
|
||||
<option value="Saint Pierre and Miquelon">Saint Pierre and Miquelon</option>
|
||||
<option value="Saint Vincent and the Grenadines">Saint Vincent and the Grenadines</option>
|
||||
<option value="Samoa">Samoa</option>
|
||||
<option value="San Marino">San Marino</option>
|
||||
<option value="Sao Tome and Principe">Sao Tome and Principe</option>
|
||||
<option value="Saudi Arabia">Saudi Arabia</option>
|
||||
<option value="Senegal">Senegal</option>
|
||||
<option value="Serbia">Serbia</option>
|
||||
<option value="Seychelles">Seychelles</option>
|
||||
<option value="Sierra Leone">Sierra Leone</option>
|
||||
<option value="Singapore">Singapore</option>
|
||||
<option value="Slovakia">Slovakia</option>
|
||||
<option value="Slovenia">Slovenia</option>
|
||||
<option value="Solomon Islands">Solomon Islands</option>
|
||||
<option value="Somalia">Somalia</option>
|
||||
<option value="South Africa">South Africa</option>
|
||||
<option value="South Georgia and the South Sandwich Islands">South Georgia and the South Sandwich Islands</option>
|
||||
<option value="Spain">Spain</option>
|
||||
<option value="Sri Lanka">Sri Lanka</option>
|
||||
<option value="Sudan">Sudan</option>
|
||||
<option value="Suriname">Suriname</option>
|
||||
<option value="Svalbard and Jan Mayen">Svalbard and Jan Mayen</option>
|
||||
<option value="Swaziland">Swaziland</option>
|
||||
<option value="Sweden">Sweden</option>
|
||||
<option value="Switzerland">Switzerland</option>
|
||||
<option value="Syrian Arab Republic">Syrian Arab Republic</option>
|
||||
<option value="Taiwan, Province of China">Taiwan, Province of China</option>
|
||||
<option value="Tajikistan">Tajikistan</option>
|
||||
<option value="Tanzania, United Republic of">Tanzania, United Republic of</option>
|
||||
<option value="Thailand">Thailand</option>
|
||||
<option value="Timor-Leste">Timor-Leste</option>
|
||||
<option value="Togo">Togo</option>
|
||||
<option value="Tokelau">Tokelau</option>
|
||||
<option value="Tonga">Tonga</option>
|
||||
<option value="Trinidad and Tobago">Trinidad and Tobago</option>
|
||||
<option value="Tunisia">Tunisia</option>
|
||||
<option value="Turkey">Turkey</option>
|
||||
<option value="Turkmenistan">Turkmenistan</option>
|
||||
<option value="Turks and Caicos Islands">Turks and Caicos Islands</option>
|
||||
<option value="Tuvalu">Tuvalu</option>
|
||||
<option value="Uganda">Uganda</option>
|
||||
<option value="Ukraine">Ukraine</option>
|
||||
<option value="United Arab Emirates">United Arab Emirates</option>
|
||||
<option value="United Kingdom">United Kingdom</option>
|
||||
<option value="United States">United States</option>
|
||||
<option value="United States Minor Outlying Islands">United States Minor Outlying Islands</option>
|
||||
<option value="Uruguay">Uruguay</option>
|
||||
<option value="Uzbekistan">Uzbekistan</option>
|
||||
<option value="Vanuatu">Vanuatu</option>
|
||||
<option value="Venezuela">Venezuela</option>
|
||||
<option value="Viet Nam">Viet Nam</option>
|
||||
<option value="Virgin Islands, British">Virgin Islands, British</option>
|
||||
<option value="Virgin Islands, U.S.">Virgin Islands, U.S.</option>
|
||||
<option value="Wallis and Futuna">Wallis and Futuna</option>
|
||||
<option value="Western Sahara">Western Sahara</option>
|
||||
<option value="Yemen">Yemen</option>
|
||||
<option value="Zambia">Zambia</option>
|
||||
<option value="Zimbabwe">Zimbabwe</option></select>
|
||||
COUNTRIES
|
||||
|
||||
fields_for "post[]", @post do |f|
|
||||
concat f.country_select("origin")
|
||||
end
|
||||
|
||||
assert_dom_equal(expected_select[0..-2], output_buffer)
|
||||
end
|
||||
|
||||
def test_time_zone_select
|
||||
@firm = Firm.new("D")
|
||||
html = time_zone_select( "firm", "time_zone" )
|
||||
@@ -1197,6 +2043,45 @@ uses_mocha "FormOptionsHelperTest" do
|
||||
)
|
||||
end
|
||||
|
||||
def test_time_zone_select_under_fields_for_with_index
|
||||
@firm = Firm.new("D")
|
||||
|
||||
fields_for :firm, @firm, :index => 305 do |f|
|
||||
concat f.time_zone_select(:time_zone)
|
||||
end
|
||||
|
||||
assert_dom_equal(
|
||||
"<select id=\"firm_305_time_zone\" name=\"firm[305][time_zone]\">" +
|
||||
"<option value=\"A\">A</option>\n" +
|
||||
"<option value=\"B\">B</option>\n" +
|
||||
"<option value=\"C\">C</option>\n" +
|
||||
"<option value=\"D\" selected=\"selected\">D</option>\n" +
|
||||
"<option value=\"E\">E</option>" +
|
||||
"</select>",
|
||||
output_buffer
|
||||
)
|
||||
end
|
||||
|
||||
def test_time_zone_select_under_fields_for_with_auto_index
|
||||
@firm = Firm.new("D")
|
||||
def @firm.to_param; 305; end
|
||||
|
||||
fields_for "firm[]", @firm do |f|
|
||||
concat f.time_zone_select(:time_zone)
|
||||
end
|
||||
|
||||
assert_dom_equal(
|
||||
"<select id=\"firm_305_time_zone\" name=\"firm[305][time_zone]\">" +
|
||||
"<option value=\"A\">A</option>\n" +
|
||||
"<option value=\"B\">B</option>\n" +
|
||||
"<option value=\"C\">C</option>\n" +
|
||||
"<option value=\"D\" selected=\"selected\">D</option>\n" +
|
||||
"<option value=\"E\">E</option>" +
|
||||
"</select>",
|
||||
output_buffer
|
||||
)
|
||||
end
|
||||
|
||||
def test_time_zone_select_with_blank
|
||||
@firm = Firm.new("D")
|
||||
html = time_zone_select("firm", "time_zone", nil, :include_blank => true)
|
||||
|
||||
@@ -201,9 +201,9 @@ class PrototypeHelperTest < PrototypeHelperBaseTest
|
||||
|
||||
end
|
||||
|
||||
def test_submit_to_remote
|
||||
def test_button_to_remote
|
||||
assert_dom_equal %(<input name=\"More beer!\" onclick=\"new Ajax.Updater('empty_bottle', 'http://www.example.com/', {asynchronous:true, evalScripts:true, parameters:Form.serialize(this.form)}); return false;\" type=\"button\" value=\"1000000\" />),
|
||||
submit_to_remote("More beer!", 1_000_000, :update => "empty_bottle")
|
||||
button_to_remote("More beer!", 1_000_000, :update => "empty_bottle")
|
||||
end
|
||||
|
||||
def test_observe_field
|
||||
|
||||
@@ -4,7 +4,7 @@ require 'controller/fake_models'
|
||||
class ViewRenderTest < Test::Unit::TestCase
|
||||
def setup
|
||||
@assigns = { :secret => 'in the sauce' }
|
||||
@view = ActionView::Base.new([FIXTURE_LOAD_PATH], @assigns)
|
||||
@view = ActionView::Base.new(ActionController::Base.view_paths, @assigns)
|
||||
end
|
||||
|
||||
def test_render_file
|
||||
@@ -95,8 +95,8 @@ class ViewRenderTest < Test::Unit::TestCase
|
||||
end
|
||||
|
||||
class CustomHandler < ActionView::TemplateHandler
|
||||
def render(template)
|
||||
[template.source, template.locals].inspect
|
||||
def render(template, local_assigns)
|
||||
[template.source, local_assigns].inspect
|
||||
end
|
||||
end
|
||||
|
||||
@@ -115,18 +115,17 @@ class ViewRenderTest < Test::Unit::TestCase
|
||||
|
||||
def compile(template)
|
||||
"@output_buffer = ''\n" +
|
||||
"@output_buffer << 'locals: #{template.locals.inspect}, '\n" +
|
||||
"@output_buffer << 'source: #{template.source.inspect}'\n"
|
||||
end
|
||||
end
|
||||
|
||||
def test_render_inline_with_compilable_custom_type
|
||||
ActionView::Template.register_template_handler :foo, CompilableCustomHandler
|
||||
assert_equal 'locals: {}, source: "Hello, World!"', @view.render(:inline => "Hello, World!", :type => :foo)
|
||||
assert_equal 'source: "Hello, World!"', @view.render(:inline => "Hello, World!", :type => :foo)
|
||||
end
|
||||
|
||||
def test_render_inline_with_locals_and_compilable_custom_type
|
||||
ActionView::Template.register_template_handler :foo, CompilableCustomHandler
|
||||
assert_equal 'locals: {:name=>"Josh"}, source: "Hello, <%= name %>!"', @view.render(:inline => "Hello, <%= name %>!", :locals => { :name => "Josh" }, :type => :foo)
|
||||
assert_equal 'source: "Hello, <%= name %>!"', @view.render(:inline => "Hello, <%= name %>!", :locals => { :name => "Josh" }, :type => :foo)
|
||||
end
|
||||
end
|
||||
|
||||
@@ -292,6 +292,7 @@ class UrlHelperTest < ActionView::TestCase
|
||||
assert_dom_equal "<a href=\"mailto:%6d%65@%64%6f%6d%61%69%6e.%63%6f%6d\">My email</a>", mail_to("me@domain.com", "My email", :encode => "hex", :replace_at => "(at)")
|
||||
assert_dom_equal "<a href=\"mailto:%6d%65@%64%6f%6d%61%69%6e.%63%6f%6d\">me(at)domain(dot)com</a>", mail_to("me@domain.com", nil, :encode => "hex", :replace_at => "(at)", :replace_dot => "(dot)")
|
||||
assert_dom_equal "<script type=\"text/javascript\">eval(unescape('%64%6f%63%75%6d%65%6e%74%2e%77%72%69%74%65%28%27%3c%61%20%68%72%65%66%3d%22%6d%61%69%6c%74%6f%3a%6d%65%40%64%6f%6d%61%69%6e%2e%63%6f%6d%22%3e%4d%79%20%65%6d%61%69%6c%3c%2f%61%3e%27%29%3b'))</script>", mail_to("me@domain.com", "My email", :encode => "javascript", :replace_at => "(at)", :replace_dot => "(dot)")
|
||||
assert_dom_equal "<script type=\"text/javascript\">eval(unescape('%64%6f%63%75%6d%65%6e%74%2e%77%72%69%74%65%28%27%3c%61%20%68%72%65%66%3d%22%6d%61%69%6c%74%6f%3a%6d%65%40%64%6f%6d%61%69%6e%2e%63%6f%6d%22%3e%6d%65%28%61%74%29%64%6f%6d%61%69%6e%28%64%6f%74%29%63%6f%6d%3c%2f%61%3e%27%29%3b'))</script>", mail_to("me@domain.com", nil, :encode => "javascript", :replace_at => "(at)", :replace_dot => "(dot)")
|
||||
end
|
||||
|
||||
def protect_against_forgery?
|
||||
@@ -301,8 +302,6 @@ end
|
||||
|
||||
class UrlHelperWithControllerTest < ActionView::TestCase
|
||||
class UrlHelperController < ActionController::Base
|
||||
self.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
def self.controller_path; 'url_helper_with_controller' end
|
||||
|
||||
def show_url_for
|
||||
@@ -313,6 +312,10 @@ class UrlHelperWithControllerTest < ActionView::TestCase
|
||||
render :inline => "<%= show_named_route_#{params[:kind]} %>"
|
||||
end
|
||||
|
||||
def nil_url_for
|
||||
render :inline => '<%= url_for(nil) %>'
|
||||
end
|
||||
|
||||
def rescue_action(e) raise e end
|
||||
end
|
||||
|
||||
@@ -329,7 +332,7 @@ class UrlHelperWithControllerTest < ActionView::TestCase
|
||||
assert_equal '/url_helper_with_controller/show_url_for', @response.body
|
||||
end
|
||||
|
||||
def test_named_route_shows_host_and_path
|
||||
def test_named_route_url_shows_host_and_path
|
||||
with_url_helper_routing do
|
||||
get :show_named_route, :kind => 'url'
|
||||
assert_equal 'http://test.host/url_helper_with_controller/show_named_route', @response.body
|
||||
@@ -343,6 +346,11 @@ class UrlHelperWithControllerTest < ActionView::TestCase
|
||||
end
|
||||
end
|
||||
|
||||
def test_url_for_nil_returns_current_path
|
||||
get :nil_url_for
|
||||
assert_equal '/url_helper_with_controller/nil_url_for', @response.body
|
||||
end
|
||||
|
||||
protected
|
||||
def with_url_helper_routing
|
||||
with_routing do |set|
|
||||
@@ -356,8 +364,6 @@ end
|
||||
|
||||
class LinkToUnlessCurrentWithControllerTest < ActionView::TestCase
|
||||
class TasksController < ActionController::Base
|
||||
self.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
def self.controller_path; 'tasks' end
|
||||
|
||||
def index
|
||||
@@ -448,8 +454,6 @@ end
|
||||
|
||||
class PolymorphicControllerTest < ActionView::TestCase
|
||||
class WorkshopsController < ActionController::Base
|
||||
self.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
def self.controller_path; 'workshops' end
|
||||
|
||||
def index
|
||||
@@ -466,8 +470,6 @@ class PolymorphicControllerTest < ActionView::TestCase
|
||||
end
|
||||
|
||||
class SessionsController < ActionController::Base
|
||||
self.view_paths = [FIXTURE_LOAD_PATH]
|
||||
|
||||
def self.controller_path; 'sessions' end
|
||||
|
||||
def index
|
||||
|
||||
@@ -1,5 +1,27 @@
|
||||
*Edge*
|
||||
|
||||
* change_column_default preserves the not-null constraint. #617 [Tarmo Tänav]
|
||||
|
||||
* Fixed that create database statements would always include "DEFAULT NULL" (Nick Sieger) [#334]
|
||||
|
||||
* Add :accessible option to associations for allowing (opt-in) mass assignment. #474. [David Dollar] Example :
|
||||
|
||||
class Post < ActiveRecord::Base
|
||||
belongs_to :author, :accessible => true
|
||||
has_many :comments, :accessible => true
|
||||
end
|
||||
|
||||
post = Post.create({
|
||||
:title => 'Accessible Attributes',
|
||||
:author => { :name => 'David Dollar' },
|
||||
:comments => [
|
||||
{ :body => 'First Post!' },
|
||||
{ :body => 'Nested Hashes are great!' }
|
||||
]
|
||||
})
|
||||
|
||||
post.comments << { :body => 'Another Comment' }
|
||||
|
||||
* Add :tokenizer option to validates_length_of to specify how to split up the attribute string. #507. [David Lowenfels] Example :
|
||||
|
||||
# Ensure essay contains at least 100 words.
|
||||
|
||||
@@ -188,7 +188,6 @@ module ActiveRecord
|
||||
through_records
|
||||
end
|
||||
|
||||
# FIXME: quoting
|
||||
def preload_belongs_to_association(records, reflection, preload_options={})
|
||||
options = reflection.options
|
||||
primary_key_name = reflection.primary_key_name
|
||||
@@ -227,9 +226,19 @@ module ActiveRecord
|
||||
|
||||
table_name = klass.quoted_table_name
|
||||
primary_key = klass.primary_key
|
||||
conditions = "#{table_name}.#{primary_key} IN (?)"
|
||||
conditions = "#{table_name}.#{connection.quote_column_name(primary_key)} IN (?)"
|
||||
conditions << append_conditions(options, preload_options)
|
||||
associated_records = klass.find(:all, :conditions => [conditions, id_map.keys.uniq],
|
||||
column_type = klass.columns.detect{|c| c.name == primary_key}.type
|
||||
ids = id_map.keys.uniq.map do |id|
|
||||
if column_type == :integer
|
||||
id.to_i
|
||||
elsif column_type == :float
|
||||
id.to_f
|
||||
else
|
||||
id
|
||||
end
|
||||
end
|
||||
associated_records = klass.find(:all, :conditions => [conditions, ids],
|
||||
:include => options[:include],
|
||||
:select => options[:select],
|
||||
:joins => options[:joins],
|
||||
|
||||
@@ -692,6 +692,7 @@ module ActiveRecord
|
||||
# * <tt>:uniq</tt> - If true, duplicates will be omitted from the collection. Useful in conjunction with <tt>:through</tt>.
|
||||
# * <tt>:readonly</tt> - If true, all the associated objects are readonly through the association.
|
||||
# * <tt>:validate</tt> - If false, don't validate the associated objects when saving the parent object. true by default.
|
||||
# * <tt>:accessible</tt> - Mass assignment is allowed for this assocation (similar to <tt>ActiveRecord::Base#attr_accessible</tt>).
|
||||
#
|
||||
# Option examples:
|
||||
# has_many :comments, :order => "posted_on"
|
||||
@@ -774,6 +775,7 @@ module ActiveRecord
|
||||
# association is a polymorphic +belongs_to+.
|
||||
# * <tt>:readonly</tt> - If true, the associated object is readonly through the association.
|
||||
# * <tt>:validate</tt> - If false, don't validate the associated object when saving the parent object. +false+ by default.
|
||||
# * <tt>:accessible</tt> - Mass assignment is allowed for this assocation (similar to <tt>ActiveRecord::Base#attr_accessible</tt>).
|
||||
#
|
||||
# Option examples:
|
||||
# has_one :credit_card, :dependent => :destroy # destroys the associated credit card
|
||||
@@ -863,6 +865,7 @@ module ActiveRecord
|
||||
# to the +attr_readonly+ list in the associated classes (e.g. <tt>class Post; attr_readonly :comments_count; end</tt>).
|
||||
# * <tt>:readonly</tt> - If true, the associated object is readonly through the association.
|
||||
# * <tt>:validate</tt> - If false, don't validate the associated objects when saving the parent object. +false+ by default.
|
||||
# * <tt>:accessible</tt> - Mass assignment is allowed for this assocation (similar to <tt>ActiveRecord::Base#attr_accessible</tt>).
|
||||
#
|
||||
# Option examples:
|
||||
# belongs_to :firm, :foreign_key => "client_of"
|
||||
@@ -1034,6 +1037,7 @@ module ActiveRecord
|
||||
# but not include the joined columns. Do not forget to include the primary and foreign keys, otherwise it will raise an error.
|
||||
# * <tt>:readonly</tt> - If true, all the associated objects are readonly through the association.
|
||||
# * <tt>:validate</tt> - If false, don't validate the associated objects when saving the parent object. +true+ by default.
|
||||
# * <tt>:accessible</tt> - Mass assignment is allowed for this assocation (similar to <tt>ActiveRecord::Base#attr_accessible</tt>).
|
||||
#
|
||||
# Option examples:
|
||||
# has_and_belongs_to_many :projects
|
||||
@@ -1109,6 +1113,8 @@ module ActiveRecord
|
||||
association = association_proxy_class.new(self, reflection)
|
||||
end
|
||||
|
||||
new_value = reflection.klass.new(new_value) if reflection.options[:accessible] && new_value.is_a?(Hash)
|
||||
|
||||
if association_proxy_class == HasOneThroughAssociation
|
||||
association.create_through_record(new_value)
|
||||
self.send(reflection.name, new_value)
|
||||
@@ -1145,7 +1151,7 @@ module ActiveRecord
|
||||
end
|
||||
|
||||
define_method("#{reflection.name.to_s.singularize}_ids") do
|
||||
send(reflection.name).map(&:id)
|
||||
send(reflection.name).map { |record| record.id }
|
||||
end
|
||||
end
|
||||
|
||||
@@ -1357,7 +1363,7 @@ module ActiveRecord
|
||||
:finder_sql, :counter_sql,
|
||||
:before_add, :after_add, :before_remove, :after_remove,
|
||||
:extend, :readonly,
|
||||
:validate
|
||||
:validate, :accessible
|
||||
)
|
||||
|
||||
options[:extend] = create_extension_modules(association_id, extension, options[:extend])
|
||||
@@ -1367,7 +1373,7 @@ module ActiveRecord
|
||||
|
||||
def create_has_one_reflection(association_id, options)
|
||||
options.assert_valid_keys(
|
||||
:class_name, :foreign_key, :remote, :select, :conditions, :order, :include, :dependent, :counter_cache, :extend, :as, :readonly, :validate, :primary_key
|
||||
:class_name, :foreign_key, :remote, :select, :conditions, :order, :include, :dependent, :counter_cache, :extend, :as, :readonly, :validate, :primary_key, :accessible
|
||||
)
|
||||
|
||||
create_reflection(:has_one, association_id, options, self)
|
||||
@@ -1383,7 +1389,7 @@ module ActiveRecord
|
||||
def create_belongs_to_reflection(association_id, options)
|
||||
options.assert_valid_keys(
|
||||
:class_name, :foreign_key, :foreign_type, :remote, :select, :conditions, :include, :dependent,
|
||||
:counter_cache, :extend, :polymorphic, :readonly, :validate
|
||||
:counter_cache, :extend, :polymorphic, :readonly, :validate, :accessible
|
||||
)
|
||||
|
||||
reflection = create_reflection(:belongs_to, association_id, options, self)
|
||||
@@ -1403,7 +1409,7 @@ module ActiveRecord
|
||||
:finder_sql, :delete_sql, :insert_sql,
|
||||
:before_add, :after_add, :before_remove, :after_remove,
|
||||
:extend, :readonly,
|
||||
:validate
|
||||
:validate, :accessible
|
||||
)
|
||||
|
||||
options[:extend] = create_extension_modules(association_id, extension, options[:extend])
|
||||
@@ -1480,25 +1486,30 @@ module ActiveRecord
|
||||
join_dependency.joins_for_table_name(table)
|
||||
}.flatten.compact.uniq
|
||||
|
||||
order = options[:order]
|
||||
if scoped_order = (scope && scope[:order])
|
||||
order = order ? "#{order}, #{scoped_order}" : scoped_order
|
||||
end
|
||||
|
||||
is_distinct = !options[:joins].blank? || include_eager_conditions?(options, tables_from_conditions) || include_eager_order?(options, tables_from_order)
|
||||
sql = "SELECT "
|
||||
if is_distinct
|
||||
sql << connection.distinct("#{connection.quote_table_name table_name}.#{primary_key}", options[:order])
|
||||
sql << connection.distinct("#{connection.quote_table_name table_name}.#{primary_key}", order)
|
||||
else
|
||||
sql << primary_key
|
||||
end
|
||||
sql << " FROM #{connection.quote_table_name table_name} "
|
||||
|
||||
if is_distinct
|
||||
sql << distinct_join_associations.collect(&:association_join).join
|
||||
sql << distinct_join_associations.collect { |assoc| assoc.association_join }.join
|
||||
add_joins!(sql, options, scope)
|
||||
end
|
||||
|
||||
add_conditions!(sql, options[:conditions], scope)
|
||||
add_group!(sql, options[:group], scope)
|
||||
|
||||
if options[:order] && is_distinct
|
||||
connection.add_order_by_for_association_limiting!(sql, options)
|
||||
if order && is_distinct
|
||||
connection.add_order_by_for_association_limiting!(sql, :order => order)
|
||||
else
|
||||
add_order!(sql, options[:order], scope)
|
||||
end
|
||||
|
||||
@@ -14,7 +14,7 @@ module ActiveRecord
|
||||
# If using a custom finder_sql, scan the entire collection.
|
||||
if @reflection.options[:finder_sql]
|
||||
expects_array = args.first.kind_of?(Array)
|
||||
ids = args.flatten.compact.uniq.map(&:to_i)
|
||||
ids = args.flatten.compact.uniq.map { |arg| arg.to_i }
|
||||
|
||||
if ids.size == 1
|
||||
id = ids.first
|
||||
@@ -78,11 +78,14 @@ module ActiveRecord
|
||||
@loaded = false
|
||||
end
|
||||
|
||||
def build(attributes = {})
|
||||
def build(attributes = {}, &block)
|
||||
if attributes.is_a?(Array)
|
||||
attributes.collect { |attr| build(attr) }
|
||||
attributes.collect { |attr| build(attr, &block) }
|
||||
else
|
||||
build_record(attributes) { |record| set_belongs_to_association_for(record) }
|
||||
build_record(attributes) do |record|
|
||||
block.call(record) if block_given?
|
||||
set_belongs_to_association_for(record)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -94,6 +97,8 @@ module ActiveRecord
|
||||
|
||||
@owner.transaction do
|
||||
flatten_deeper(records).each do |record|
|
||||
record = @reflection.klass.new(record) if @reflection.options[:accessible] && record.is_a?(Hash)
|
||||
|
||||
raise_on_type_mismatch(record)
|
||||
add_record_to_target_with_callbacks(record) do |r|
|
||||
result &&= insert_record(record) unless @owner.new_record?
|
||||
@@ -226,6 +231,10 @@ module ActiveRecord
|
||||
# Replace this collection with +other_array+
|
||||
# This will perform a diff and delete/add only records that have changed.
|
||||
def replace(other_array)
|
||||
other_array.map! do |val|
|
||||
val.is_a?(Hash) ? @reflection.klass.new(val) : val
|
||||
end if @reflection.options[:accessible]
|
||||
|
||||
other_array.each { |val| raise_on_type_mismatch(val) }
|
||||
|
||||
load_target
|
||||
|
||||
@@ -61,9 +61,9 @@ module ActiveRecord
|
||||
def delete_records(records)
|
||||
case @reflection.options[:dependent]
|
||||
when :destroy
|
||||
records.each(&:destroy)
|
||||
records.each { |r| r.destroy }
|
||||
when :delete_all
|
||||
@reflection.klass.delete(records.map(&:id))
|
||||
@reflection.klass.delete(records.map { |record| record.id })
|
||||
else
|
||||
ids = quoted_record_ids(records)
|
||||
@reflection.klass.update_all(
|
||||
|
||||
@@ -1,4 +1,5 @@
|
||||
require 'date'
|
||||
require 'set'
|
||||
require 'bigdecimal'
|
||||
require 'bigdecimal/util'
|
||||
|
||||
@@ -6,6 +7,8 @@ module ActiveRecord
|
||||
module ConnectionAdapters #:nodoc:
|
||||
# An abstract definition of a column in a table.
|
||||
class Column
|
||||
TRUE_VALUES = [true, 1, '1', 't', 'T', 'true', 'TRUE'].to_set
|
||||
|
||||
module Format
|
||||
ISO_DATE = /\A(\d{4})-(\d\d)-(\d\d)\z/
|
||||
ISO_DATETIME = /\A(\d{4})-(\d\d)-(\d\d) (\d\d):(\d\d):(\d\d)(\.\d+)?\z/
|
||||
@@ -135,11 +138,7 @@ module ActiveRecord
|
||||
|
||||
# convert something to a boolean
|
||||
def value_to_boolean(value)
|
||||
if value == true || value == false
|
||||
value
|
||||
else
|
||||
%w(true t 1).include?(value.to_s.downcase)
|
||||
end
|
||||
TRUE_VALUES.include?(value)
|
||||
end
|
||||
|
||||
# convert something to a BigDecimal
|
||||
@@ -257,7 +256,10 @@ module ActiveRecord
|
||||
|
||||
def to_sql
|
||||
column_sql = "#{base.quote_column_name(name)} #{sql_type}"
|
||||
add_column_options!(column_sql, :null => null, :default => default) unless type.to_sym == :primary_key
|
||||
column_options = {}
|
||||
column_options[:null] = null unless null.nil?
|
||||
column_options[:default] = default unless default.nil?
|
||||
add_column_options!(column_sql, column_options) unless type.to_sym == :primary_key
|
||||
column_sql
|
||||
end
|
||||
alias to_s :to_sql
|
||||
|
||||
@@ -69,7 +69,7 @@ module ActiveRecord
|
||||
MysqlCompat.define_all_hashes_method!
|
||||
|
||||
mysql = Mysql.init
|
||||
mysql.ssl_set(config[:sslkey], config[:sslcert], config[:sslca], config[:sslcapath], config[:sslcipher]) if config[:sslkey]
|
||||
mysql.ssl_set(config[:sslkey], config[:sslcert], config[:sslca], config[:sslcapath], config[:sslcipher]) if config[:sslca] || config[:sslkey]
|
||||
|
||||
ConnectionAdapters::MysqlAdapter.new(mysql, logger, [host, username, password, database, port, socket], config)
|
||||
end
|
||||
@@ -145,6 +145,7 @@ module ActiveRecord
|
||||
# * <tt>:password</tt> - Defaults to nothing.
|
||||
# * <tt>:database</tt> - The name of the database. No default, must be provided.
|
||||
# * <tt>:encoding</tt> - (Optional) Sets the client encoding by executing "SET NAMES <encoding>" after connection.
|
||||
# * <tt>:sslca</tt> - Necessary to use MySQL with an SSL connection.
|
||||
# * <tt>:sslkey</tt> - Necessary to use MySQL with an SSL connection.
|
||||
# * <tt>:sslcert</tt> - Necessary to use MySQL with an SSL connection.
|
||||
# * <tt>:sslcapath</tt> - Necessary to use MySQL with an SSL connection.
|
||||
@@ -436,18 +437,29 @@ module ActiveRecord
|
||||
end
|
||||
|
||||
def change_column_default(table_name, column_name, default) #:nodoc:
|
||||
current_type = select_one("SHOW COLUMNS FROM #{quote_table_name(table_name)} LIKE '#{column_name}'")["Type"]
|
||||
column = column_for(table_name, column_name)
|
||||
change_column table_name, column_name, column.sql_type, :default => default
|
||||
end
|
||||
|
||||
execute("ALTER TABLE #{quote_table_name(table_name)} CHANGE #{quote_column_name(column_name)} #{quote_column_name(column_name)} #{current_type} DEFAULT #{quote(default)}")
|
||||
def change_column_null(table_name, column_name, null, default = nil)
|
||||
column = column_for(table_name, column_name)
|
||||
|
||||
unless null || default.nil?
|
||||
execute("UPDATE #{quote_table_name(table_name)} SET #{quote_column_name(column_name)}=#{quote(default)} WHERE #{quote_column_name(column_name)} IS NULL")
|
||||
end
|
||||
|
||||
change_column table_name, column_name, column.sql_type, :null => null
|
||||
end
|
||||
|
||||
def change_column(table_name, column_name, type, options = {}) #:nodoc:
|
||||
column = column_for(table_name, column_name)
|
||||
|
||||
unless options_include_default?(options)
|
||||
if column = columns(table_name).find { |c| c.name == column_name.to_s }
|
||||
options[:default] = column.default
|
||||
else
|
||||
raise "No such column: #{table_name}.#{column_name}"
|
||||
end
|
||||
options[:default] = column.default
|
||||
end
|
||||
|
||||
unless options.has_key?(:null)
|
||||
options[:null] = column.null
|
||||
end
|
||||
|
||||
change_column_sql = "ALTER TABLE #{quote_table_name(table_name)} CHANGE #{quote_column_name(column_name)} #{quote_column_name(column_name)} #{type_to_sql(type, options[:limit], options[:precision], options[:scale])}"
|
||||
@@ -459,6 +471,7 @@ module ActiveRecord
|
||||
options = {}
|
||||
if column = columns(table_name).find { |c| c.name == column_name.to_s }
|
||||
options[:default] = column.default
|
||||
options[:null] = column.null
|
||||
else
|
||||
raise ActiveRecordError, "No such column: #{table_name}.#{column_name}"
|
||||
end
|
||||
@@ -507,7 +520,9 @@ module ActiveRecord
|
||||
@connection.options(Mysql::SET_CHARSET_NAME, encoding) rescue nil
|
||||
end
|
||||
|
||||
@connection.ssl_set(@config[:sslkey], @config[:sslcert], @config[:sslca], @config[:sslcapath], @config[:sslcipher]) if @config[:sslkey]
|
||||
if @config[:sslca] || @config[:sslkey]
|
||||
@connection.ssl_set(@config[:sslkey], @config[:sslcert], @config[:sslca], @config[:sslcapath], @config[:sslcipher])
|
||||
end
|
||||
|
||||
@connection.real_connect(*@connection_options)
|
||||
|
||||
@@ -533,6 +548,13 @@ module ActiveRecord
|
||||
def version
|
||||
@version ||= @connection.server_info.scan(/^(\d+)\.(\d+)\.(\d+)/).flatten.map { |v| v.to_i }
|
||||
end
|
||||
|
||||
def column_for(table_name, column_name)
|
||||
unless column = columns(table_name).find { |c| c.name == column_name.to_s }
|
||||
raise "No such column: #{table_name}.#{column_name}"
|
||||
end
|
||||
column
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -49,7 +49,6 @@ module ActiveRecord
|
||||
private
|
||||
def extract_limit(sql_type)
|
||||
case sql_type
|
||||
when /^integer/i; 4
|
||||
when /^bigint/i; 8
|
||||
when /^smallint/i; 2
|
||||
else super
|
||||
@@ -623,6 +622,19 @@ module ActiveRecord
|
||||
end
|
||||
end
|
||||
|
||||
# Returns the current database name.
|
||||
def current_database
|
||||
query('select current_database()')[0][0]
|
||||
end
|
||||
|
||||
# Returns the current database encoding format.
|
||||
def encoding
|
||||
query(<<-end_sql)[0][0]
|
||||
SELECT pg_encoding_to_char(pg_database.encoding) FROM pg_database
|
||||
WHERE pg_database.datname LIKE '#{current_database}'
|
||||
end_sql
|
||||
end
|
||||
|
||||
# Sets the schema search path to a string of comma-separated schema names.
|
||||
# Names beginning with $ have to be quoted (e.g. $user => '$user').
|
||||
# See: http://www.postgresql.org/docs/current/static/ddl-schemas.html
|
||||
|
||||
@@ -238,6 +238,15 @@ module ActiveRecord
|
||||
end
|
||||
end
|
||||
|
||||
def change_column_null(table_name, column_name, null, default = nil)
|
||||
unless null || default.nil?
|
||||
execute("UPDATE #{quote_table_name(table_name)} SET #{quote_column_name(column_name)}=#{quote(default)} WHERE #{quote_column_name(column_name)} IS NULL")
|
||||
end
|
||||
alter_table(table_name) do |definition|
|
||||
definition[column_name].null = null
|
||||
end
|
||||
end
|
||||
|
||||
def change_column(table_name, column_name, type, options = {}) #:nodoc:
|
||||
alter_table(table_name) do |definition|
|
||||
include_default = options_include_default?(options)
|
||||
@@ -251,6 +260,9 @@ module ActiveRecord
|
||||
end
|
||||
|
||||
def rename_column(table_name, column_name, new_column_name) #:nodoc:
|
||||
unless columns(table_name).detect{|c| c.name == column_name.to_s }
|
||||
raise ActiveRecord::ActiveRecordError, "Missing column #{table_name}.#{column_name}"
|
||||
end
|
||||
alter_table(table_name, :rename => {column_name.to_s => new_column_name.to_s})
|
||||
end
|
||||
|
||||
|
||||
@@ -541,10 +541,11 @@ class Fixtures < (RUBY_VERSION < '1.9' ? YAML::Omap : Hash)
|
||||
label.to_s.hash.abs
|
||||
end
|
||||
|
||||
attr_reader :table_name
|
||||
attr_reader :table_name, :name
|
||||
|
||||
def initialize(connection, table_name, class_name, fixture_path, file_filter = DEFAULT_FILTER_RE)
|
||||
@connection, @table_name, @fixture_path, @file_filter = connection, table_name, fixture_path, file_filter
|
||||
@name = table_name # preserve fixture base name
|
||||
@class_name = class_name ||
|
||||
(ActiveRecord::Base.pluralize_table_names ? @table_name.singularize.camelize : @table_name.camelize)
|
||||
@table_name = "#{ActiveRecord::Base.table_name_prefix}#{@table_name}#{ActiveRecord::Base.table_name_suffix}"
|
||||
@@ -963,9 +964,9 @@ module Test #:nodoc:
|
||||
fixtures = Fixtures.create_fixtures(fixture_path, fixture_table_names, fixture_class_names)
|
||||
unless fixtures.nil?
|
||||
if fixtures.instance_of?(Fixtures)
|
||||
@loaded_fixtures[fixtures.table_name] = fixtures
|
||||
@loaded_fixtures[fixtures.name] = fixtures
|
||||
else
|
||||
fixtures.each { |f| @loaded_fixtures[f.table_name] = f }
|
||||
fixtures.each { |f| @loaded_fixtures[f.name] = f }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -150,7 +150,8 @@ module ActiveRecord
|
||||
if scopes.include?(method)
|
||||
scopes[method].call(self, *args)
|
||||
else
|
||||
with_scope :find => proxy_options do
|
||||
with_scope :find => proxy_options, :create => proxy_options[:conditions].is_a?(Hash) ? proxy_options[:conditions] : {} do
|
||||
method = :new if method == :build
|
||||
proxy_scope.send(method, *args, &block)
|
||||
end
|
||||
end
|
||||
|
||||
@@ -22,11 +22,22 @@ module ActiveRecord
|
||||
end
|
||||
end
|
||||
|
||||
def assert_queries(num = 1)
|
||||
$query_count = 0
|
||||
def assert_sql(*patterns_to_match)
|
||||
$queries_executed = []
|
||||
yield
|
||||
ensure
|
||||
assert_equal num, $query_count, "#{$query_count} instead of #{num} queries were executed."
|
||||
failed_patterns = []
|
||||
patterns_to_match.each do |pattern|
|
||||
failed_patterns << pattern unless $queries_executed.any?{ |sql| pattern === sql }
|
||||
end
|
||||
assert failed_patterns.empty?, "Query pattern(s) #{failed_patterns.map(&:inspect).join(', ')} not found."
|
||||
end
|
||||
|
||||
def assert_queries(num = 1)
|
||||
$queries_executed = []
|
||||
yield
|
||||
ensure
|
||||
assert_equal num, $queries_executed.size, "#{$queries_executed.size} instead of #{num} queries were executed."
|
||||
end
|
||||
|
||||
def assert_no_queries(&block)
|
||||
|
||||
@@ -854,7 +854,7 @@ module ActiveRecord
|
||||
raw_value = raw_value.to_i
|
||||
else
|
||||
begin
|
||||
raw_value = Kernel.Float(raw_value.to_s)
|
||||
raw_value = Kernel.Float(raw_value)
|
||||
rescue ArgumentError, TypeError
|
||||
record.errors.add(attr_name, configuration[:message] || ActiveRecord::Errors.default_error_messages[:not_a_number])
|
||||
next
|
||||
|
||||
@@ -65,6 +65,12 @@ class AdapterTest < ActiveRecord::TestCase
|
||||
end
|
||||
end
|
||||
|
||||
if current_adapter?(:PostgreSQLAdapter)
|
||||
def test_encoding
|
||||
assert_not_nil @connection.encoding
|
||||
end
|
||||
end
|
||||
|
||||
def test_table_alias
|
||||
def @connection.test_table_alias_length() 10; end
|
||||
class << @connection
|
||||
|
||||
@@ -425,6 +425,37 @@ class HasManyAssociationsTest < ActiveRecord::TestCase
|
||||
assert_equal 2, first_topic.replies.to_ary.size
|
||||
end
|
||||
|
||||
def test_build_via_block
|
||||
company = companies(:first_firm)
|
||||
new_client = assert_no_queries { company.clients_of_firm.build {|client| client.name = "Another Client" } }
|
||||
assert !company.clients_of_firm.loaded?
|
||||
|
||||
assert_equal "Another Client", new_client.name
|
||||
assert new_client.new_record?
|
||||
assert_equal new_client, company.clients_of_firm.last
|
||||
company.name += '-changed'
|
||||
assert_queries(2) { assert company.save }
|
||||
assert !new_client.new_record?
|
||||
assert_equal 2, company.clients_of_firm(true).size
|
||||
end
|
||||
|
||||
def test_build_many_via_block
|
||||
company = companies(:first_firm)
|
||||
new_clients = assert_no_queries do
|
||||
company.clients_of_firm.build([{"name" => "Another Client"}, {"name" => "Another Client II"}]) do |client|
|
||||
client.name = "changed"
|
||||
end
|
||||
end
|
||||
|
||||
assert_equal 2, new_clients.size
|
||||
assert_equal "changed", new_clients.first.name
|
||||
assert_equal "changed", new_clients.last.name
|
||||
|
||||
company.name += '-changed'
|
||||
assert_queries(3) { assert company.save }
|
||||
assert_equal 3, company.clients_of_firm(true).size
|
||||
end
|
||||
|
||||
def test_create_without_loading_association
|
||||
first_firm = companies(:first_firm)
|
||||
Firm.column_names
|
||||
|
||||
@@ -189,6 +189,114 @@ class AssociationProxyTest < ActiveRecord::TestCase
|
||||
end
|
||||
end
|
||||
|
||||
def test_belongs_to_mass_assignment
|
||||
post_attributes = { :title => 'Associations', :body => 'Are They Accessible?' }
|
||||
author_attributes = { :name => 'David Dollar' }
|
||||
|
||||
assert_no_difference 'Author.count' do
|
||||
assert_raise(ActiveRecord::AssociationTypeMismatch) do
|
||||
Post.create(post_attributes.merge({:author => author_attributes}))
|
||||
end
|
||||
end
|
||||
|
||||
assert_difference 'Author.count' do
|
||||
post = Post.create(post_attributes.merge({:creatable_author => author_attributes}))
|
||||
assert_equal post.creatable_author.name, author_attributes[:name]
|
||||
end
|
||||
end
|
||||
|
||||
def test_has_one_mass_assignment
|
||||
post_attributes = { :title => 'Associations', :body => 'Are They Accessible?' }
|
||||
comment_attributes = { :body => 'Setter Takes Hash' }
|
||||
|
||||
assert_no_difference 'Comment.count' do
|
||||
assert_raise(ActiveRecord::AssociationTypeMismatch) do
|
||||
Post.create(post_attributes.merge({:uncreatable_comment => comment_attributes}))
|
||||
end
|
||||
end
|
||||
|
||||
assert_difference 'Comment.count' do
|
||||
post = Post.create(post_attributes.merge({:creatable_comment => comment_attributes}))
|
||||
assert_equal post.creatable_comment.body, comment_attributes[:body]
|
||||
end
|
||||
end
|
||||
|
||||
def test_has_many_mass_assignment
|
||||
post = posts(:welcome)
|
||||
post_attributes = { :title => 'Associations', :body => 'Are They Accessible?' }
|
||||
comment_attributes = { :body => 'Setter Takes Hash' }
|
||||
|
||||
assert_no_difference 'Comment.count' do
|
||||
assert_raise(ActiveRecord::AssociationTypeMismatch) do
|
||||
Post.create(post_attributes.merge({:comments => [comment_attributes]}))
|
||||
end
|
||||
assert_raise(ActiveRecord::AssociationTypeMismatch) do
|
||||
post.comments << comment_attributes
|
||||
end
|
||||
end
|
||||
|
||||
assert_difference 'Comment.count' do
|
||||
post = Post.create(post_attributes.merge({:creatable_comments => [comment_attributes]}))
|
||||
assert_equal post.creatable_comments.last.body, comment_attributes[:body]
|
||||
end
|
||||
|
||||
assert_difference 'Comment.count' do
|
||||
post.creatable_comments << comment_attributes
|
||||
assert_equal post.comments.last.body, comment_attributes[:body]
|
||||
end
|
||||
|
||||
post.creatable_comments = [comment_attributes, comment_attributes]
|
||||
assert_equal post.creatable_comments.count, 2
|
||||
end
|
||||
|
||||
def test_has_and_belongs_to_many_mass_assignment
|
||||
post = posts(:welcome)
|
||||
post_attributes = { :title => 'Associations', :body => 'Are They Accessible?' }
|
||||
category_attributes = { :name => 'Accessible Association', :type => 'Category' }
|
||||
|
||||
assert_no_difference 'Category.count' do
|
||||
assert_raise(ActiveRecord::AssociationTypeMismatch) do
|
||||
Post.create(post_attributes.merge({:categories => [category_attributes]}))
|
||||
end
|
||||
assert_raise(ActiveRecord::AssociationTypeMismatch) do
|
||||
post.categories << category_attributes
|
||||
end
|
||||
end
|
||||
|
||||
assert_difference 'Category.count' do
|
||||
post = Post.create(post_attributes.merge({:creatable_categories => [category_attributes]}))
|
||||
assert_equal post.creatable_categories.last.name, category_attributes[:name]
|
||||
end
|
||||
|
||||
assert_difference 'Category.count' do
|
||||
post.creatable_categories << category_attributes
|
||||
assert_equal post.creatable_categories.last.name, category_attributes[:name]
|
||||
end
|
||||
|
||||
post.creatable_categories = [category_attributes, category_attributes]
|
||||
assert_equal post.creatable_categories.count, 2
|
||||
end
|
||||
|
||||
def test_association_proxy_setter_can_take_hash
|
||||
special_comment_attributes = { :body => 'Setter Takes Hash' }
|
||||
|
||||
post = posts(:welcome)
|
||||
post.creatable_comment = { :body => 'Setter Takes Hash' }
|
||||
|
||||
assert_equal post.creatable_comment.body, special_comment_attributes[:body]
|
||||
end
|
||||
|
||||
def test_association_collection_can_take_hash
|
||||
post_attributes = { :title => 'Setter Takes', :body => 'Hash' }
|
||||
david = authors(:david)
|
||||
|
||||
post = (david.posts << post_attributes).last
|
||||
assert_equal post.title, post_attributes[:title]
|
||||
|
||||
david.posts = [post_attributes, post_attributes]
|
||||
assert_equal david.posts.count, 2
|
||||
end
|
||||
|
||||
def setup_dangling_association
|
||||
josh = Author.create(:name => "Josh")
|
||||
p = Post.create(:title => "New on Edge", :body => "More cool stuff!", :author => josh)
|
||||
|
||||
36
activerecord/test/cases/column_definition_test.rb
Normal file
36
activerecord/test/cases/column_definition_test.rb
Normal file
@@ -0,0 +1,36 @@
|
||||
require "cases/helper"
|
||||
|
||||
class ColumnDefinitionTest < ActiveRecord::TestCase
|
||||
def setup
|
||||
@adapter = ActiveRecord::ConnectionAdapters::AbstractAdapter.new(nil)
|
||||
def @adapter.native_database_types
|
||||
{:string => "varchar"}
|
||||
end
|
||||
end
|
||||
|
||||
# Avoid column definitions in create table statements like:
|
||||
# `title` varchar(255) DEFAULT NULL NULL
|
||||
def test_should_not_include_default_clause_when_default_is_null
|
||||
column = ActiveRecord::ConnectionAdapters::Column.new("title", nil, "varchar(20)")
|
||||
column_def = ActiveRecord::ConnectionAdapters::ColumnDefinition.new(
|
||||
@adapter, column.name, "string",
|
||||
column.limit, column.precision, column.scale, column.default, column.null)
|
||||
assert_equal "title varchar(20) NULL", column_def.to_sql
|
||||
end
|
||||
|
||||
def test_should_include_default_clause_when_default_is_present
|
||||
column = ActiveRecord::ConnectionAdapters::Column.new("title", "Hello", "varchar(20)")
|
||||
column_def = ActiveRecord::ConnectionAdapters::ColumnDefinition.new(
|
||||
@adapter, column.name, "string",
|
||||
column.limit, column.precision, column.scale, column.default, column.null)
|
||||
assert_equal %Q{title varchar(20) DEFAULT 'Hello' NULL}, column_def.to_sql
|
||||
end
|
||||
|
||||
def test_should_specify_not_null_if_null_option_is_false
|
||||
column = ActiveRecord::ConnectionAdapters::Column.new("title", "Hello", "varchar(20)", false)
|
||||
column_def = ActiveRecord::ConnectionAdapters::ColumnDefinition.new(
|
||||
@adapter, column.name, "string",
|
||||
column.limit, column.precision, column.scale, column.default, column.null)
|
||||
assert_equal %Q{title varchar(20) DEFAULT 'Hello' NOT NULL}, column_def.to_sql
|
||||
end
|
||||
end
|
||||
@@ -15,6 +15,7 @@ require 'models/pirate'
|
||||
require 'models/treasure'
|
||||
require 'models/matey'
|
||||
require 'models/ship'
|
||||
require 'models/book'
|
||||
|
||||
class FixturesTest < ActiveRecord::TestCase
|
||||
self.use_instantiated_fixtures = true
|
||||
@@ -373,6 +374,34 @@ class CheckSetTableNameFixturesTest < ActiveRecord::TestCase
|
||||
end
|
||||
end
|
||||
|
||||
class FixtureNameIsNotTableNameFixturesTest < ActiveRecord::TestCase
|
||||
set_fixture_class :items => Book
|
||||
fixtures :items
|
||||
# Set to false to blow away fixtures cache and ensure our fixtures are loaded
|
||||
# and thus takes into account our set_fixture_class
|
||||
self.use_transactional_fixtures = false
|
||||
|
||||
def test_named_accessor
|
||||
assert_kind_of Book, items(:dvd)
|
||||
end
|
||||
end
|
||||
|
||||
class FixtureNameIsNotTableNameMultipleFixturesTest < ActiveRecord::TestCase
|
||||
set_fixture_class :items => Book, :funny_jokes => Joke
|
||||
fixtures :items, :funny_jokes
|
||||
# Set to false to blow away fixtures cache and ensure our fixtures are loaded
|
||||
# and thus takes into account our set_fixture_class
|
||||
self.use_transactional_fixtures = false
|
||||
|
||||
def test_named_accessor_of_differently_named_fixture
|
||||
assert_kind_of Book, items(:dvd)
|
||||
end
|
||||
|
||||
def test_named_accessor_of_same_named_fixture
|
||||
assert_kind_of Joke, funny_jokes(:a_joke)
|
||||
end
|
||||
end
|
||||
|
||||
class CustomConnectionFixturesTest < ActiveRecord::TestCase
|
||||
set_fixture_class :courses => Course
|
||||
fixtures :courses
|
||||
|
||||
@@ -32,13 +32,13 @@ end
|
||||
ActiveRecord::Base.connection.class.class_eval do
|
||||
IGNORED_SQL = [/^PRAGMA/, /^SELECT currval/, /^SELECT CAST/, /^SELECT @@IDENTITY/, /^SELECT @@ROWCOUNT/]
|
||||
|
||||
def execute_with_counting(sql, name = nil, &block)
|
||||
$query_count ||= 0
|
||||
$query_count += 1 unless IGNORED_SQL.any? { |r| sql =~ r }
|
||||
execute_without_counting(sql, name, &block)
|
||||
def execute_with_query_record(sql, name = nil, &block)
|
||||
$queries_executed ||= []
|
||||
$queries_executed << sql unless IGNORED_SQL.any? { |r| sql =~ r }
|
||||
execute_without_query_record(sql, name, &block)
|
||||
end
|
||||
|
||||
alias_method_chain :execute, :counting
|
||||
alias_method_chain :execute, :query_record
|
||||
end
|
||||
|
||||
# Make with_scope public for tests
|
||||
|
||||
@@ -191,6 +191,13 @@ class InheritanceTest < ActiveRecord::TestCase
|
||||
assert_not_nil account.instance_variable_get("@firm"), "nil proves eager load failed"
|
||||
end
|
||||
|
||||
def test_eager_load_belongs_to_primary_key_quoting
|
||||
con = Account.connection
|
||||
assert_sql(/\(#{con.quote_table_name('companies')}.#{con.quote_column_name('id')} IN \(1\)\)/) do
|
||||
Account.find(1, :include => :firm)
|
||||
end
|
||||
end
|
||||
|
||||
def test_alt_eager_loading
|
||||
switch_to_alt_inheritance_column
|
||||
test_eager_load_belongs_to_something_inherited
|
||||
|
||||
@@ -3,6 +3,7 @@ require 'bigdecimal/util'
|
||||
|
||||
require 'models/person'
|
||||
require 'models/topic'
|
||||
require 'models/developer'
|
||||
|
||||
require MIGRATIONS_ROOT + "/valid/1_people_have_last_names"
|
||||
require MIGRATIONS_ROOT + "/valid/2_we_need_reminders"
|
||||
@@ -511,7 +512,12 @@ if ActiveRecord::Base.connection.supports_migrations?
|
||||
ActiveRecord::Base.connection.create_table(:hats) do |table|
|
||||
table.column :hat_name, :string, :default => nil
|
||||
end
|
||||
assert_raises(ActiveRecord::ActiveRecordError) do
|
||||
exception = if current_adapter?(:PostgreSQLAdapter)
|
||||
ActiveRecord::StatementInvalid
|
||||
else
|
||||
ActiveRecord::ActiveRecordError
|
||||
end
|
||||
assert_raises(exception) do
|
||||
Person.connection.rename_column "hats", "nonexistent", "should_fail"
|
||||
end
|
||||
ensure
|
||||
@@ -697,6 +703,55 @@ if ActiveRecord::Base.connection.supports_migrations?
|
||||
Person.connection.drop_table :testings rescue nil
|
||||
end
|
||||
|
||||
def test_keeping_default_and_notnull_constaint_on_change
|
||||
Person.connection.create_table :testings do |t|
|
||||
t.column :title, :string
|
||||
end
|
||||
person_klass = Class.new(Person)
|
||||
person_klass.set_table_name 'testings'
|
||||
|
||||
person_klass.connection.add_column "testings", "wealth", :integer, :null => false, :default => 99
|
||||
person_klass.reset_column_information
|
||||
assert_equal 99, person_klass.columns_hash["wealth"].default
|
||||
assert_equal false, person_klass.columns_hash["wealth"].null
|
||||
assert_nothing_raised {person_klass.connection.execute("insert into testings (title) values ('tester')")}
|
||||
|
||||
# change column default to see that column doesn't lose its not null definition
|
||||
person_klass.connection.change_column_default "testings", "wealth", 100
|
||||
person_klass.reset_column_information
|
||||
assert_equal 100, person_klass.columns_hash["wealth"].default
|
||||
assert_equal false, person_klass.columns_hash["wealth"].null
|
||||
|
||||
# rename column to see that column doesn't lose its not null and/or default definition
|
||||
person_klass.connection.rename_column "testings", "wealth", "money"
|
||||
person_klass.reset_column_information
|
||||
assert_nil person_klass.columns_hash["wealth"]
|
||||
assert_equal 100, person_klass.columns_hash["money"].default
|
||||
assert_equal false, person_klass.columns_hash["money"].null
|
||||
|
||||
# change column
|
||||
person_klass.connection.change_column "testings", "money", :integer, :null => false, :default => 1000
|
||||
person_klass.reset_column_information
|
||||
assert_equal 1000, person_klass.columns_hash["money"].default
|
||||
assert_equal false, person_klass.columns_hash["money"].null
|
||||
|
||||
# change column, make it nullable and clear default
|
||||
person_klass.connection.change_column "testings", "money", :integer, :null => true, :default => nil
|
||||
person_klass.reset_column_information
|
||||
assert_nil person_klass.columns_hash["money"].default
|
||||
assert_equal true, person_klass.columns_hash["money"].null
|
||||
|
||||
# change_column_null, make it not nullable and set null values to a default value
|
||||
person_klass.connection.execute('UPDATE testings SET money = NULL')
|
||||
person_klass.connection.change_column_null "testings", "money", false, 2000
|
||||
person_klass.reset_column_information
|
||||
assert_nil person_klass.columns_hash["money"].default
|
||||
assert_equal false, person_klass.columns_hash["money"].null
|
||||
assert_equal [2000], Person.connection.select_values("SELECT money FROM testings").map { |s| s.to_i }.sort
|
||||
ensure
|
||||
Person.connection.drop_table :testings rescue nil
|
||||
end
|
||||
|
||||
def test_change_column_default_to_null
|
||||
Person.connection.change_column_default "people", "first_name", nil
|
||||
Person.reset_column_information
|
||||
|
||||
@@ -183,4 +183,30 @@ class NamedScopeTest < ActiveRecord::TestCase
|
||||
topics.empty? # use loaded (no query)
|
||||
end
|
||||
end
|
||||
|
||||
def test_should_build_with_proxy_options
|
||||
topic = Topic.approved.build({})
|
||||
assert topic.approved
|
||||
end
|
||||
|
||||
def test_should_build_new_with_proxy_options
|
||||
topic = Topic.approved.new
|
||||
assert topic.approved
|
||||
end
|
||||
|
||||
def test_should_create_with_proxy_options
|
||||
topic = Topic.approved.create({})
|
||||
assert topic.approved
|
||||
end
|
||||
|
||||
def test_should_create_with_bang_with_proxy_options
|
||||
topic = Topic.approved.create!({})
|
||||
assert topic.approved
|
||||
end
|
||||
|
||||
def test_should_build_with_proxy_options_chained
|
||||
topic = Topic.approved.by_lifo.build({})
|
||||
assert topic.approved
|
||||
assert_equal 'lifo', topic.author_name
|
||||
end
|
||||
end
|
||||
|
||||
@@ -1391,6 +1391,7 @@ class ValidatesNumericalityTest < ActiveRecord::TestCase
|
||||
INTEGERS = [0, 10, -10] + INTEGER_STRINGS
|
||||
BIGDECIMAL = BIGDECIMAL_STRINGS.collect! { |bd| BigDecimal.new(bd) }
|
||||
JUNK = ["not a number", "42 not a number", "0xdeadbeef", "00-1", "--3", "+-3", "+3-1", "-+019.0", "12.12.13.12", "123\nnot a number"]
|
||||
INFINITY = [1.0/0.0]
|
||||
|
||||
def setup
|
||||
Topic.instance_variable_set("@validate_callbacks", ActiveSupport::Callbacks::CallbackChain.new)
|
||||
@@ -1402,27 +1403,27 @@ class ValidatesNumericalityTest < ActiveRecord::TestCase
|
||||
Topic.validates_numericality_of :approved
|
||||
|
||||
invalid!(NIL + BLANK + JUNK)
|
||||
valid!(FLOATS + INTEGERS + BIGDECIMAL)
|
||||
valid!(FLOATS + INTEGERS + BIGDECIMAL + INFINITY)
|
||||
end
|
||||
|
||||
def test_validates_numericality_of_with_nil_allowed
|
||||
Topic.validates_numericality_of :approved, :allow_nil => true
|
||||
|
||||
invalid!(BLANK + JUNK)
|
||||
valid!(NIL + FLOATS + INTEGERS + BIGDECIMAL)
|
||||
valid!(NIL + FLOATS + INTEGERS + BIGDECIMAL + INFINITY)
|
||||
end
|
||||
|
||||
def test_validates_numericality_of_with_integer_only
|
||||
Topic.validates_numericality_of :approved, :only_integer => true
|
||||
|
||||
invalid!(NIL + BLANK + JUNK + FLOATS + BIGDECIMAL)
|
||||
invalid!(NIL + BLANK + JUNK + FLOATS + BIGDECIMAL + INFINITY)
|
||||
valid!(INTEGERS)
|
||||
end
|
||||
|
||||
def test_validates_numericality_of_with_integer_only_and_nil_allowed
|
||||
Topic.validates_numericality_of :approved, :only_integer => true, :allow_nil => true
|
||||
|
||||
invalid!(BLANK + JUNK + FLOATS + BIGDECIMAL)
|
||||
invalid!(BLANK + JUNK + FLOATS + BIGDECIMAL + INFINITY)
|
||||
valid!(NIL + INTEGERS)
|
||||
end
|
||||
|
||||
@@ -1443,7 +1444,7 @@ class ValidatesNumericalityTest < ActiveRecord::TestCase
|
||||
def test_validates_numericality_with_equal_to
|
||||
Topic.validates_numericality_of :approved, :equal_to => 10
|
||||
|
||||
invalid!([-10, 11], 'must be equal to 10')
|
||||
invalid!([-10, 11] + INFINITY, 'must be equal to 10')
|
||||
valid!([10])
|
||||
end
|
||||
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
class Author < ActiveRecord::Base
|
||||
has_many :posts
|
||||
has_many :posts, :accessible => true
|
||||
has_many :posts_with_comments, :include => :comments, :class_name => "Post"
|
||||
has_many :posts_with_comments_sorted_by_comment_id, :include => :comments, :class_name => "Post", :order => 'comments.id'
|
||||
has_many :posts_with_categories, :include => :categories, :class_name => "Post"
|
||||
|
||||
@@ -33,6 +33,12 @@ class Post < ActiveRecord::Base
|
||||
has_and_belongs_to_many :categories
|
||||
has_and_belongs_to_many :special_categories, :join_table => "categories_posts", :association_foreign_key => 'category_id'
|
||||
|
||||
belongs_to :creatable_author, :class_name => 'Author', :accessible => true
|
||||
has_one :uncreatable_comment, :class_name => 'Comment', :accessible => false, :order => 'id desc'
|
||||
has_one :creatable_comment, :class_name => 'Comment', :accessible => true, :order => 'id desc'
|
||||
has_many :creatable_comments, :class_name => 'Comment', :accessible => true, :dependent => :destroy
|
||||
has_and_belongs_to_many :creatable_categories, :class_name => 'Category', :accessible => true
|
||||
|
||||
has_many :taggings, :as => :taggable
|
||||
has_many :tags, :through => :taggings do
|
||||
def add_joins_and_select
|
||||
|
||||
@@ -4,6 +4,8 @@ class Topic < ActiveRecord::Base
|
||||
{ :conditions => ['written_on < ?', time] }
|
||||
}
|
||||
named_scope :approved, :conditions => {:approved => true}
|
||||
named_scope :by_lifo, :conditions => {:author_name => 'lifo'}
|
||||
|
||||
named_scope :approved_as_hash_condition, :conditions => {:topics => {:approved => true}}
|
||||
named_scope 'approved_as_string', :conditions => {:approved => true}
|
||||
named_scope :replied, :conditions => ['replies_count > 0']
|
||||
|
||||
@@ -1,5 +1,15 @@
|
||||
*Edge*
|
||||
|
||||
* Add Array#in_groups which splits or iterates over the array in specified number of groups. #579. [Adrian Mugnolo] Example:
|
||||
|
||||
a = (1..10).to_a
|
||||
a.in_groups(3) #=> [[1, 2, 3, 4], [5, 6, 7, nil], [8, 9, 10, nil]]
|
||||
a.in_groups(3, false) #=> [[1, 2, 3, 4], [5, 6, 7], [8, 9, 10]]
|
||||
|
||||
* Fix TimeWithZone unmarshaling: coerce unmarshaled Time instances to utc, because Ruby's marshaling of Time instances doesn't respect the zone [Geoff Buesing]
|
||||
|
||||
* Added Memoizable mixin for caching simple lazy loaded attributes [Josh Peek]
|
||||
|
||||
* Move the test related core_ext stuff out of core_ext so it's only loaded by the test helpers. [Michael Koziarski]
|
||||
|
||||
* Add Inflection rules for String#humanize. #535 [dcmanges]
|
||||
|
||||
@@ -43,6 +43,7 @@ require 'active_support/ordered_hash'
|
||||
require 'active_support/ordered_options'
|
||||
require 'active_support/option_merger'
|
||||
|
||||
require 'active_support/memoizable'
|
||||
require 'active_support/string_inquirer'
|
||||
|
||||
require 'active_support/values/time_zone'
|
||||
|
||||
@@ -4,8 +4,8 @@ module ActiveSupport #:nodoc:
|
||||
module CoreExtensions #:nodoc:
|
||||
module Array #:nodoc:
|
||||
module Grouping
|
||||
# Iterates over the array in groups of size +number+, padding any remaining
|
||||
# slots with +fill_with+ unless it is +false+.
|
||||
# Splits or iterates over the array in groups of size +number+,
|
||||
# padding any remaining slots with +fill_with+ unless it is +false+.
|
||||
#
|
||||
# %w(1 2 3 4 5 6 7).in_groups_of(3) {|g| p g}
|
||||
# ["1", "2", "3"]
|
||||
@@ -39,6 +39,49 @@ module ActiveSupport #:nodoc:
|
||||
end
|
||||
end
|
||||
|
||||
# Splits or iterates over the array in +number+ of groups, padding any
|
||||
# remaining slots with +fill_with+ unless it is +false+.
|
||||
#
|
||||
# %w(1 2 3 4 5 6 7 8 9 10).in_groups(3) {|g| p g}
|
||||
# ["1", "2", "3", "4"]
|
||||
# ["5", "6", "7", nil]
|
||||
# ["8", "9", "10", nil]
|
||||
#
|
||||
# %w(1 2 3 4 5 6 7).in_groups(3, ' ') {|g| p g}
|
||||
# ["1", "2", "3"]
|
||||
# ["4", "5", " "]
|
||||
# ["6", "7", " "]
|
||||
#
|
||||
# %w(1 2 3 4 5 6 7).in_groups(3, false) {|g| p g}
|
||||
# ["1", "2", "3"]
|
||||
# ["4", "5"]
|
||||
# ["6", "7"]
|
||||
def in_groups(number, fill_with = nil)
|
||||
# size / number gives minor group size;
|
||||
# size % number gives how many objects need extra accomodation;
|
||||
# each group hold either division or division + 1 items.
|
||||
division = size / number
|
||||
modulo = size % number
|
||||
|
||||
# create a new array avoiding dup
|
||||
groups = []
|
||||
start = 0
|
||||
|
||||
number.times do |index|
|
||||
length = division + (modulo > 0 && modulo > index ? 1 : 0)
|
||||
padding = fill_with != false &&
|
||||
modulo > 0 && length == division ? 1 : 0
|
||||
groups << slice(start, length).concat([fill_with] * padding)
|
||||
start += length
|
||||
end
|
||||
|
||||
if block_given?
|
||||
groups.each{|g| yield(g) }
|
||||
else
|
||||
groups
|
||||
end
|
||||
end
|
||||
|
||||
# Divides the array into one or more subarrays based on a delimiting +value+
|
||||
# or the result of an optional block.
|
||||
#
|
||||
|
||||
@@ -1,4 +1,14 @@
|
||||
class Module
|
||||
# Returns the name of the module containing this one.
|
||||
#
|
||||
# p M::N.parent_name # => "M"
|
||||
def parent_name
|
||||
unless defined? @parent_name
|
||||
@parent_name = name =~ /::[^:]+\Z/ ? $`.freeze : nil
|
||||
end
|
||||
@parent_name
|
||||
end
|
||||
|
||||
# Returns the module which contains this one according to its name.
|
||||
#
|
||||
# module M
|
||||
@@ -16,8 +26,7 @@ class Module
|
||||
# p Module.new.parent # => Object
|
||||
#
|
||||
def parent
|
||||
parent_name = name.split('::')[0..-2] * '::'
|
||||
parent_name.empty? ? Object : parent_name.constantize
|
||||
parent_name ? parent_name.constantize : Object
|
||||
end
|
||||
|
||||
# Returns all the parents of this module according to its name, ordered from
|
||||
@@ -35,10 +44,12 @@ class Module
|
||||
#
|
||||
def parents
|
||||
parents = []
|
||||
parts = name.split('::')[0..-2]
|
||||
until parts.empty?
|
||||
parents << (parts * '::').constantize
|
||||
parts.pop
|
||||
if parent_name
|
||||
parts = parent_name.split('::')
|
||||
until parts.empty?
|
||||
parents << (parts * '::').constantize
|
||||
parts.pop
|
||||
end
|
||||
end
|
||||
parents << Object unless parents.include? Object
|
||||
parents
|
||||
@@ -70,6 +81,6 @@ class Module
|
||||
# Returns the names of the constants defined locally rather than the
|
||||
# constants themselves. See <tt>local_constants</tt>.
|
||||
def local_constant_names
|
||||
local_constants.map(&:to_s)
|
||||
local_constants.map { |c| c.to_s }
|
||||
end
|
||||
end
|
||||
|
||||
@@ -35,7 +35,7 @@ class Object
|
||||
# C.new(0, 1).instance_variable_names # => ["@y", "@x"]
|
||||
if RUBY_VERSION >= '1.9'
|
||||
def instance_variable_names
|
||||
instance_variables.map(&:to_s)
|
||||
instance_variables.map { |var| var.to_s }
|
||||
end
|
||||
else
|
||||
alias_method :instance_variable_names, :instance_variables
|
||||
|
||||
@@ -387,7 +387,7 @@ module ActiveSupport #:nodoc:
|
||||
ensure
|
||||
# Remove the stack frames that we added.
|
||||
if defined?(watch_frames) && ! watch_frames.blank?
|
||||
frame_ids = watch_frames.collect(&:object_id)
|
||||
frame_ids = watch_frames.collect { |frame| frame.object_id }
|
||||
constant_watch_stack.delete_if do |watch_frame|
|
||||
frame_ids.include? watch_frame.object_id
|
||||
end
|
||||
@@ -437,7 +437,7 @@ module ActiveSupport #:nodoc:
|
||||
protected
|
||||
def log_call(*args)
|
||||
if defined?(RAILS_DEFAULT_LOGGER) && RAILS_DEFAULT_LOGGER && log_activity
|
||||
arg_str = args.collect(&:inspect) * ', '
|
||||
arg_str = args.collect { |arg| arg.inspect } * ', '
|
||||
/in `([a-z_\?\!]+)'/ =~ caller(1).first
|
||||
selector = $1 || '<unknown>'
|
||||
log "called #{selector}(#{arg_str})"
|
||||
|
||||
@@ -8,7 +8,7 @@ class DateTime
|
||||
if ActiveSupport.use_standard_json_time_format
|
||||
xmlschema.inspect
|
||||
else
|
||||
%("#{strftime("%Y/%m/%d %H:%M:%S %z")}")
|
||||
strftime('"%Y/%m/%d %H:%M:%S %z"')
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
35
activesupport/lib/active_support/memoizable.rb
Normal file
35
activesupport/lib/active_support/memoizable.rb
Normal file
@@ -0,0 +1,35 @@
|
||||
module ActiveSupport
|
||||
module Memoizable
|
||||
def self.included(base) #:nodoc:
|
||||
base.extend(ClassMethods)
|
||||
end
|
||||
|
||||
module ClassMethods
|
||||
def memoize(symbol)
|
||||
original_method = "_unmemoized_#{symbol}"
|
||||
memoized_ivar = "@_memoized_#{symbol}"
|
||||
raise "Already memoized #{symbol}" if instance_methods.map(&:to_s).include?(original_method)
|
||||
|
||||
alias_method original_method, symbol
|
||||
class_eval <<-EOS, __FILE__, __LINE__
|
||||
def #{symbol}
|
||||
if defined? #{memoized_ivar}
|
||||
#{memoized_ivar}
|
||||
else
|
||||
#{memoized_ivar} = #{original_method}
|
||||
end
|
||||
end
|
||||
EOS
|
||||
end
|
||||
end
|
||||
|
||||
def freeze
|
||||
methods.each do |method|
|
||||
if m = method.to_s.match(/\A_unmemoized_(.*)/)
|
||||
send(m[1]).freeze
|
||||
end
|
||||
end
|
||||
super
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -263,7 +263,7 @@ module ActiveSupport
|
||||
end
|
||||
|
||||
def marshal_load(variables)
|
||||
initialize(variables[0], ::Time.send!(:get_zone, variables[1]), variables[2])
|
||||
initialize(variables[0].utc, ::Time.send!(:get_zone, variables[1]), variables[2].utc)
|
||||
end
|
||||
|
||||
# Ensure proxy class responds to all methods that underlying time instance responds to.
|
||||
|
||||
@@ -99,7 +99,7 @@ class ArrayExtToSTests < Test::Unit::TestCase
|
||||
end
|
||||
|
||||
class ArrayExtGroupingTests < Test::Unit::TestCase
|
||||
def test_group_by_with_perfect_fit
|
||||
def test_in_groups_of_with_perfect_fit
|
||||
groups = []
|
||||
('a'..'i').to_a.in_groups_of(3) do |group|
|
||||
groups << group
|
||||
@@ -109,7 +109,7 @@ class ArrayExtGroupingTests < Test::Unit::TestCase
|
||||
assert_equal [%w(a b c), %w(d e f), %w(g h i)], ('a'..'i').to_a.in_groups_of(3)
|
||||
end
|
||||
|
||||
def test_group_by_with_padding
|
||||
def test_in_groups_of_with_padding
|
||||
groups = []
|
||||
('a'..'g').to_a.in_groups_of(3) do |group|
|
||||
groups << group
|
||||
@@ -118,7 +118,7 @@ class ArrayExtGroupingTests < Test::Unit::TestCase
|
||||
assert_equal [%w(a b c), %w(d e f), ['g', nil, nil]], groups
|
||||
end
|
||||
|
||||
def test_group_by_pads_with_specified_values
|
||||
def test_in_groups_of_pads_with_specified_values
|
||||
groups = []
|
||||
|
||||
('a'..'g').to_a.in_groups_of(3, 'foo') do |group|
|
||||
@@ -128,7 +128,7 @@ class ArrayExtGroupingTests < Test::Unit::TestCase
|
||||
assert_equal [%w(a b c), %w(d e f), ['g', 'foo', 'foo']], groups
|
||||
end
|
||||
|
||||
def test_group_without_padding
|
||||
def test_in_groups_of_without_padding
|
||||
groups = []
|
||||
|
||||
('a'..'g').to_a.in_groups_of(3, false) do |group|
|
||||
@@ -137,6 +137,48 @@ class ArrayExtGroupingTests < Test::Unit::TestCase
|
||||
|
||||
assert_equal [%w(a b c), %w(d e f), ['g']], groups
|
||||
end
|
||||
|
||||
def test_in_groups_returned_array_size
|
||||
array = (1..7).to_a
|
||||
|
||||
1.upto(array.size + 1) do |number|
|
||||
assert_equal number, array.in_groups(number).size
|
||||
end
|
||||
end
|
||||
|
||||
def test_in_groups_with_empty_array
|
||||
assert_equal [[], [], []], [].in_groups(3)
|
||||
end
|
||||
|
||||
def test_in_groups_with_block
|
||||
array = (1..9).to_a
|
||||
groups = []
|
||||
|
||||
array.in_groups(3) do |group|
|
||||
groups << group
|
||||
end
|
||||
|
||||
assert_equal array.in_groups(3), groups
|
||||
end
|
||||
|
||||
def test_in_groups_with_perfect_fit
|
||||
assert_equal [[1, 2, 3], [4, 5, 6], [7, 8, 9]],
|
||||
(1..9).to_a.in_groups(3)
|
||||
end
|
||||
|
||||
def test_in_groups_with_padding
|
||||
array = (1..7).to_a
|
||||
|
||||
assert_equal [[1, 2, 3], [4, 5, nil], [6, 7, nil]],
|
||||
array.in_groups(3)
|
||||
assert_equal [[1, 2, 3], [4, 5, 'foo'], [6, 7, 'foo']],
|
||||
array.in_groups(3, 'foo')
|
||||
end
|
||||
|
||||
def test_in_groups_without_padding
|
||||
assert_equal [[1, 2, 3], [4, 5], [6, 7]],
|
||||
(1..7).to_a.in_groups(3, false)
|
||||
end
|
||||
end
|
||||
|
||||
class ArraySplitTests < Test::Unit::TestCase
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user