aboutsummaryrefslogtreecommitdiffstats
path: root/activesupport
diff options
context:
space:
mode:
authorJeremy Kemper <jeremy@bitsweat.net>2006-07-05 17:32:16 +0000
committerJeremy Kemper <jeremy@bitsweat.net>2006-07-05 17:32:16 +0000
commit12600d77dd1757365dc86e6aec5cafb48ec49886 (patch)
treee0c2dd9a7ee9e8e8eb3e02be1cd2035d6c7b691b /activesupport
parentc51f9fdc78d1732d6c08a6411ca9d2536aa86d6e (diff)
downloadrails-12600d77dd1757365dc86e6aec5cafb48ec49886.tar.gz
rails-12600d77dd1757365dc86e6aec5cafb48ec49886.tar.bz2
rails-12600d77dd1757365dc86e6aec5cafb48ec49886.zip
HashWithIndifferentAccess shouldn't confuse false and nil. Closes #5601. Nor should it mistreat legitimate nil values.
git-svn-id: http://svn-commit.rubyonrails.org/rails/trunk@4555 5ecf4fe2-1ee6-0310-87b1-e25e094e27de
Diffstat (limited to 'activesupport')
-rw-r--r--activesupport/CHANGELOG2
-rw-r--r--activesupport/lib/active_support/core_ext/hash/indifferent_access.rb17
-rw-r--r--activesupport/test/core_ext/hash_ext_test.rb89
3 files changed, 70 insertions, 38 deletions
diff --git a/activesupport/CHANGELOG b/activesupport/CHANGELOG
index 6d81723705..62450807c2 100644
--- a/activesupport/CHANGELOG
+++ b/activesupport/CHANGELOG
@@ -1,5 +1,7 @@
*SVN*
+* HashWithIndifferentAccess shouldn't confuse false and nil. #5601 [shugo@ruby-lang.org]
+
* Fixed HashWithIndifferentAccess#default #5586 [chris@seagul.co.uk]
* More compatible Hash.create_from_xml. #5523 [nunemaker@gmail.com]
diff --git a/activesupport/lib/active_support/core_ext/hash/indifferent_access.rb b/activesupport/lib/active_support/core_ext/hash/indifferent_access.rb
index 9af626c8da..9a3b02e5d5 100644
--- a/activesupport/lib/active_support/core_ext/hash/indifferent_access.rb
+++ b/activesupport/lib/active_support/core_ext/hash/indifferent_access.rb
@@ -10,15 +10,18 @@ class HashWithIndifferentAccess < Hash
super(constructor)
end
end
-
+
def default(key = nil)
- value = self[key.to_s] if key.is_a?(Symbol)
- value ? value : super
+ if key.is_a?(Symbol) && include?(key = key.to_s)
+ self[key]
+ else
+ super
+ end
end
alias_method :regular_writer, :[]= unless method_defined?(:regular_writer)
alias_method :regular_update, :update unless method_defined?(:regular_update)
-
+
def []=(key, value)
regular_writer(convert_key(key), convert_value(value))
end
@@ -27,7 +30,7 @@ class HashWithIndifferentAccess < Hash
other_hash.each_pair { |key, value| regular_writer(convert_key(key), convert_value(value)) }
self
end
-
+
alias_method :merge!, :update
def key?(key)
@@ -49,7 +52,7 @@ class HashWithIndifferentAccess < Hash
def dup
HashWithIndifferentAccess.new(self)
end
-
+
def merge(hash)
self.dup.update(hash)
end
@@ -60,7 +63,7 @@ class HashWithIndifferentAccess < Hash
def stringify_keys!; self end
def symbolize_keys!; self end
-
+
protected
def convert_key(key)
key.kind_of?(Symbol) ? key.to_s : key
diff --git a/activesupport/test/core_ext/hash_ext_test.rb b/activesupport/test/core_ext/hash_ext_test.rb
index 327d884ecd..28b4304500 100644
--- a/activesupport/test/core_ext/hash_ext_test.rb
+++ b/activesupport/test/core_ext/hash_ext_test.rb
@@ -50,18 +50,18 @@ class HashExtTest < Test::Unit::TestCase
@strings = @strings.with_indifferent_access
@symbols = @symbols.with_indifferent_access
@mixed = @mixed.with_indifferent_access
-
+
assert_equal 'a', @strings.send(:convert_key, :a)
-
+
assert_equal 1, @strings.fetch('a')
assert_equal 1, @strings.fetch(:a.to_s)
assert_equal 1, @strings.fetch(:a)
-
+
hashes = { :@strings => @strings, :@symbols => @symbols, :@mixed => @mixed }
method_map = { :'[]' => 1, :fetch => 1, :values_at => [1],
:has_key? => true, :include? => true, :key? => true,
:member? => true }
-
+
hashes.each do |name, hash|
method_map.sort_by { |m| m.to_s }.each do |meth, expected|
assert_equal(expected, hash.send(meth, 'a'),
@@ -70,7 +70,7 @@ class HashExtTest < Test::Unit::TestCase
"Calling #{name}.#{meth} :a")
end
end
-
+
assert_equal [1, 2], @strings.values_at('a', 'b')
assert_equal [1, 2], @strings.values_at(:a, :b)
assert_equal [1, 2], @symbols.values_at('a', 'b')
@@ -78,13 +78,41 @@ class HashExtTest < Test::Unit::TestCase
assert_equal [1, 2], @mixed.values_at('a', 'b')
assert_equal [1, 2], @mixed.values_at(:a, :b)
end
-
+
+ def test_indifferent_reading
+ hash = HashWithIndifferentAccess.new
+ hash["a"] = 1
+ hash["b"] = true
+ hash["c"] = false
+ hash["d"] = nil
+
+ assert_equal 1, hash[:a]
+ assert_equal true, hash[:b]
+ assert_equal false, hash[:c]
+ assert_equal nil, hash[:d]
+ assert_equal nil, hash[:e]
+ end
+
+ def test_indifferent_reading_with_nonnil_default
+ hash = HashWithIndifferentAccess.new(1)
+ hash["a"] = 1
+ hash["b"] = true
+ hash["c"] = false
+ hash["d"] = nil
+
+ assert_equal 1, hash[:a]
+ assert_equal true, hash[:b]
+ assert_equal false, hash[:c]
+ assert_equal nil, hash[:d]
+ assert_equal 1, hash[:e]
+ end
+
def test_indifferent_writing
hash = HashWithIndifferentAccess.new
hash[:a] = 1
hash['b'] = 2
hash[3] = 3
-
+
assert_equal hash['a'], 1
assert_equal hash['b'], 2
assert_equal hash[:a], 1
@@ -96,7 +124,7 @@ class HashExtTest < Test::Unit::TestCase
hash = HashWithIndifferentAccess.new
hash[:a] = 'a'
hash['b'] = 'b'
-
+
updated_with_strings = hash.update(@strings)
updated_with_symbols = hash.update(@symbols)
updated_with_mixed = hash.update(@mixed)
@@ -119,21 +147,21 @@ class HashExtTest < Test::Unit::TestCase
hash = HashWithIndifferentAccess.new
hash[:a] = 'failure'
hash['b'] = 'failure'
-
+
other = { 'a' => 1, :b => 2 }
-
+
merged = hash.merge(other)
-
+
assert_equal HashWithIndifferentAccess, merged.class
assert_equal 1, merged[:a]
assert_equal 2, merged['b']
-
+
hash.update(other)
-
+
assert_equal 1, hash[:a]
assert_equal 2, hash['b']
end
-
+
def test_indifferent_deleting
get_hash = proc{ { :a => 'foo' }.with_indifferent_access }
hash = get_hash.call
@@ -160,7 +188,7 @@ class HashExtTest < Test::Unit::TestCase
{ :failure => "stuff", :funny => "business" }.assert_valid_keys([ :failure, :funny ])
{ :failure => "stuff", :funny => "business" }.assert_valid_keys(:failure, :funny)
end
-
+
assert_raises(ArgumentError, "Unknown key(s): failore") do
{ :failore => "stuff", :funny => "business" }.assert_valid_keys([ :failure, :funny ])
{ :failore => "stuff", :funny => "business" }.assert_valid_keys(:failure, :funny)
@@ -170,7 +198,7 @@ class HashExtTest < Test::Unit::TestCase
def test_indifferent_subhashes
h = {'user' => {'id' => 5}}.with_indifferent_access
['user', :user].each {|user| [:id, 'id'].each {|id| assert_equal 5, h[user][id], "h[#{user.inspect}][#{id.inspect}] should be 5"}}
-
+
h = {:user => {:id => 5}}.with_indifferent_access
['user', :user].each {|user| [:id, 'id'].each {|id| assert_equal 5, h[user][id], "h[#{user.inspect}][#{id.inspect}] should be 5"}}
end
@@ -265,7 +293,7 @@ class HashToXmlTest < Test::Unit::TestCase
assert xml.include?(%(<address><street>Paulina</street></address>))
assert xml.include?(%(<level_one><second_level>content</second_level></level_one>))
end
-
+
def test_two_levels_with_array
xml = { :name => "David", :addresses => [{ :street => "Paulina" }, { :street => "Evergreen" }] }.to_xml(@xml_options)
assert_equal "<person>", xml.first(8)
@@ -274,7 +302,7 @@ class HashToXmlTest < Test::Unit::TestCase
assert xml.include?(%(<address><street>Evergreen</street></address>))
assert xml.include?(%(<name>David</name>))
end
-
+
def test_three_levels_with_array
xml = { :name => "David", :addresses => [{ :streets => [ { :name => "Paulina" }, { :name => "Paulina" } ] } ] }.to_xml(@xml_options)
assert xml.include?(%(<addresses><address><streets><street><name>))
@@ -295,7 +323,7 @@ class HashToXmlTest < Test::Unit::TestCase
<parent-id></parent-id>
</topic>
EOT
-
+
expected_topic_hash = {
:title => "The First Topic",
:author_name => "David",
@@ -308,7 +336,7 @@ class HashToXmlTest < Test::Unit::TestCase
:author_email_address => "david@loudthinking.com",
:parent_id => nil
}.stringify_keys
-
+
assert_equal expected_topic_hash, Hash.create_from_xml(topic_xml)["topic"]
end
@@ -341,7 +369,7 @@ class HashToXmlTest < Test::Unit::TestCase
</topic>
</topics>
EOT
-
+
expected_topic_hash = {
:title => "The First Topic",
:author_name => "David",
@@ -354,10 +382,10 @@ class HashToXmlTest < Test::Unit::TestCase
:author_email_address => "david@loudthinking.com",
:parent_id => nil
}.stringify_keys
-
+
assert_equal expected_topic_hash, Hash.create_from_xml(topics_xml)["topics"]["topic"].first
end
-
+
def test_single_record_from_xml_with_attributes_other_than_type
topic_xml = <<-EOT
<rsp stat="ok">
@@ -366,7 +394,7 @@ class HashToXmlTest < Test::Unit::TestCase
</photos>
</rsp>
EOT
-
+
expected_topic_hash = {
:id => "175756086",
:owner => "55569174@N00",
@@ -377,29 +405,28 @@ class HashToXmlTest < Test::Unit::TestCase
:isfriend => "0",
:isfamily => "0",
}.stringify_keys
-
+
assert_equal expected_topic_hash, Hash.create_from_xml(topic_xml)["rsp"]["photos"]["photo"]
end
-
+
def test_should_use_default_value_for_unknown_key
hash_wia = HashWithIndifferentAccess.new(3)
assert_equal 3, hash_wia[:new_key]
end
-
+
def test_should_use_default_value_if_no_key_is_supplied
hash_wia = HashWithIndifferentAccess.new(3)
assert_equal 3, hash_wia.default
end
-
+
def test_should_nil_if_no_default_value_is_supplied
hash_wia = HashWithIndifferentAccess.new
assert_nil hash_wia.default
end
-
+
def test_should_copy_the_default_value_when_converting_to_hash_with_indifferent_access
hash = Hash.new(3)
hash_wia = hash.with_indifferent_access
assert_equal 3, hash_wia.default
end
-
-end \ No newline at end of file
+end