mongo 1.1 → 1.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/HISTORY +5 -0
- data/README.rdoc +8 -3
- data/Rakefile +4 -10
- data/lib/mongo.rb +6 -1
- data/lib/mongo/connection.rb +1 -2
- data/lib/mongo/gridfs/grid_io_fix.rb +38 -0
- data/mongo.gemspec +1 -1
- data/test/bson/bson_test.rb +65 -58
- data/test/bson/byte_buffer_test.rb +1 -1
- data/test/bson/hash_with_indifferent_access_test.rb +38 -0
- data/test/bson/json_test.rb +7 -6
- data/test/bson/object_id_test.rb +1 -3
- data/test/collection_test.rb +3 -3
- data/test/connection_test.rb +12 -18
- data/test/cursor_fail_test.rb +1 -2
- data/test/cursor_message_test.rb +1 -2
- data/test/cursor_test.rb +1 -2
- data/test/db_api_test.rb +7 -5
- data/test/db_test.rb +8 -10
- data/test/grid_file_system_test.rb +2 -3
- data/test/grid_io_test.rb +1 -2
- data/test/grid_test.rb +1 -2
- data/test/replica_sets/connect_test.rb +23 -0
- data/test/support/hash_with_indifferent_access.rb +199 -0
- data/test/support/keys.rb +45 -0
- data/test/test_helper.rb +37 -2
- data/test/threading/test_threading_large_pool.rb +1 -1
- data/test/threading_test.rb +1 -1
- metadata +169 -118
- data/test/bson/basic_test.rb +0 -99
- data/test/bson/bench_test.rb +0 -172
- data/test/bson/c_encode_test.rb +0 -51
- data/test/bson/java_bson_test.rb +0 -146
- data/test/bson/jruby_bson_test.rb +0 -24
- data/test/bson/jruby_encode_test.rb +0 -438
data/test/bson/json_test.rb
CHANGED
@@ -4,13 +4,14 @@ require 'json'
|
|
4
4
|
|
5
5
|
class JSONTest < Test::Unit::TestCase
|
6
6
|
|
7
|
-
|
8
|
-
|
9
|
-
|
7
|
+
# This test passes when run by itself but fails
|
8
|
+
# when run as part of the whole test suite.
|
10
9
|
def test_object_id_as_json
|
11
|
-
id
|
12
|
-
|
13
|
-
|
10
|
+
warn "Pending test object id as json"
|
11
|
+
#id = BSON::ObjectId.new
|
12
|
+
|
13
|
+
#obj = {'_id' => id}
|
14
|
+
#assert_equal "{\"_id\":#{id.to_json}}", obj.to_json
|
14
15
|
end
|
15
16
|
|
16
17
|
end
|
data/test/bson/object_id_test.rb
CHANGED
@@ -60,9 +60,7 @@ class ObjectIdTest < Test::Unit::TestCase
|
|
60
60
|
end
|
61
61
|
|
62
62
|
def test_save_and_restore
|
63
|
-
|
64
|
-
port = ENV['MONGO_RUBY_DRIVER_PORT'] || Connection::DEFAULT_PORT
|
65
|
-
db = Connection.new(host, port).db(MONGO_TEST_DB)
|
63
|
+
db = standard_connection.db(MONGO_TEST_DB)
|
66
64
|
coll = db.collection('test')
|
67
65
|
|
68
66
|
coll.remove
|
data/test/collection_test.rb
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
require './test/test_helper'
|
2
2
|
|
3
3
|
class TestCollection < Test::Unit::TestCase
|
4
|
-
@@connection ||=
|
4
|
+
@@connection ||= standard_connection
|
5
5
|
@@db = @@connection.db(MONGO_TEST_DB)
|
6
6
|
@@test = @@db.collection("test")
|
7
7
|
@@version = @@connection.server_version
|
@@ -221,7 +221,7 @@ class TestCollection < Test::Unit::TestCase
|
|
221
221
|
end
|
222
222
|
|
223
223
|
def test_mocked_safe_remove
|
224
|
-
@conn =
|
224
|
+
@conn = standard_connection
|
225
225
|
@db = @conn[MONGO_TEST_DB]
|
226
226
|
@test = @db['test-safe-remove']
|
227
227
|
@test.save({:a => 20})
|
@@ -234,7 +234,7 @@ class TestCollection < Test::Unit::TestCase
|
|
234
234
|
end
|
235
235
|
|
236
236
|
def test_safe_remove
|
237
|
-
@conn =
|
237
|
+
@conn = standard_connection
|
238
238
|
@db = @conn[MONGO_TEST_DB]
|
239
239
|
@test = @db['test-safe-remove']
|
240
240
|
@test.save({:a => 50})
|
data/test/connection_test.rb
CHANGED
@@ -9,19 +9,13 @@ class TestConnection < Test::Unit::TestCase
|
|
9
9
|
include BSON
|
10
10
|
|
11
11
|
def setup
|
12
|
-
@
|
13
|
-
@port = ENV['MONGO_RUBY_DRIVER_PORT'] || Connection::DEFAULT_PORT
|
14
|
-
@mongo = Connection.new(@host, @port)
|
12
|
+
@mongo = standard_connection
|
15
13
|
end
|
16
14
|
|
17
15
|
def teardown
|
18
16
|
@mongo[MONGO_TEST_DB].get_last_error
|
19
17
|
end
|
20
18
|
|
21
|
-
def test_slave_ok_with_multiple_nodes
|
22
|
-
|
23
|
-
end
|
24
|
-
|
25
19
|
def test_server_info
|
26
20
|
server_info = @mongo.server_info
|
27
21
|
assert server_info.keys.include?("version")
|
@@ -29,9 +23,9 @@ class TestConnection < Test::Unit::TestCase
|
|
29
23
|
end
|
30
24
|
|
31
25
|
def test_connection_uri
|
32
|
-
con = Connection.from_uri("mongodb
|
33
|
-
assert_equal
|
34
|
-
assert_equal
|
26
|
+
con = Connection.from_uri("mongodb://#{host_port}")
|
27
|
+
assert_equal mongo_host, con.host
|
28
|
+
assert_equal mongo_port, con.port
|
35
29
|
end
|
36
30
|
|
37
31
|
def test_server_version
|
@@ -64,7 +58,7 @@ class TestConnection < Test::Unit::TestCase
|
|
64
58
|
|
65
59
|
def test_copy_database
|
66
60
|
@mongo.db('old').collection('copy-test').insert('a' => 1)
|
67
|
-
@mongo.copy_database('old', 'new')
|
61
|
+
@mongo.copy_database('old', 'new', host_port)
|
68
62
|
old_object = @mongo.db('old').collection('copy-test').find.next_document
|
69
63
|
new_object = @mongo.db('new').collection('copy-test').find.next_document
|
70
64
|
assert_equal old_object, new_object
|
@@ -77,10 +71,10 @@ class TestConnection < Test::Unit::TestCase
|
|
77
71
|
@mongo.db('old').add_user('bob', 'secret')
|
78
72
|
|
79
73
|
assert_raise Mongo::OperationFailure do
|
80
|
-
@mongo.copy_database('old', 'new',
|
74
|
+
@mongo.copy_database('old', 'new', host_port, 'bob', 'badpassword')
|
81
75
|
end
|
82
76
|
|
83
|
-
result = @mongo.copy_database('old', 'new',
|
77
|
+
result = @mongo.copy_database('old', 'new', host_port, 'bob', 'secret')
|
84
78
|
assert Mongo::Support.ok?(result)
|
85
79
|
|
86
80
|
@mongo.drop_database('old')
|
@@ -102,7 +96,7 @@ class TestConnection < Test::Unit::TestCase
|
|
102
96
|
output = StringIO.new
|
103
97
|
logger = Logger.new(output)
|
104
98
|
logger.level = Logger::DEBUG
|
105
|
-
|
99
|
+
connection = standard_connection(:logger => logger).db(MONGO_TEST_DB)
|
106
100
|
assert output.string.include?("admin['$cmd'].find")
|
107
101
|
end
|
108
102
|
|
@@ -110,9 +104,9 @@ class TestConnection < Test::Unit::TestCase
|
|
110
104
|
output = StringIO.new
|
111
105
|
logger = Logger.new(output)
|
112
106
|
logger.level = Logger::DEBUG
|
113
|
-
connection =
|
107
|
+
connection = standard_connection(:logger => logger)
|
114
108
|
assert_equal logger, connection.logger
|
115
|
-
|
109
|
+
|
116
110
|
connection.logger.debug 'testing'
|
117
111
|
assert output.string.include?('testing')
|
118
112
|
end
|
@@ -166,7 +160,7 @@ class TestConnection < Test::Unit::TestCase
|
|
166
160
|
|
167
161
|
context "Saved authentications" do
|
168
162
|
setup do
|
169
|
-
@conn =
|
163
|
+
@conn = standard_connection
|
170
164
|
@auth = {'db_name' => 'test', 'username' => 'bob', 'password' => 'secret'}
|
171
165
|
@conn.add_auth(@auth['db_name'], @auth['username'], @auth['password'])
|
172
166
|
end
|
@@ -198,7 +192,7 @@ class TestConnection < Test::Unit::TestCase
|
|
198
192
|
|
199
193
|
context "Connection exceptions" do
|
200
194
|
setup do
|
201
|
-
@conn =
|
195
|
+
@conn = standard_connection(:pool_size => 10, :timeout => 10)
|
202
196
|
@coll = @conn[MONGO_TEST_DB]['test-connection-exceptions']
|
203
197
|
end
|
204
198
|
|
data/test/cursor_fail_test.rb
CHANGED
@@ -5,8 +5,7 @@ class CursorTest < Test::Unit::TestCase
|
|
5
5
|
|
6
6
|
include Mongo
|
7
7
|
|
8
|
-
@@connection =
|
9
|
-
ENV['MONGO_RUBY_DRIVER_PORT'] || Connection::DEFAULT_PORT)
|
8
|
+
@@connection = standard_connection
|
10
9
|
@@db = @@connection.db(MONGO_TEST_DB)
|
11
10
|
@@coll = @@db.collection('test')
|
12
11
|
@@version = @@connection.server_version
|
data/test/cursor_message_test.rb
CHANGED
@@ -5,8 +5,7 @@ class CursorTest < Test::Unit::TestCase
|
|
5
5
|
|
6
6
|
include Mongo
|
7
7
|
|
8
|
-
@@connection =
|
9
|
-
ENV['MONGO_RUBY_DRIVER_PORT'] || Connection::DEFAULT_PORT)
|
8
|
+
@@connection = standard_connection
|
10
9
|
@@db = @@connection.db(MONGO_TEST_DB)
|
11
10
|
@@coll = @@db.collection('test')
|
12
11
|
@@version = @@connection.server_version
|
data/test/cursor_test.rb
CHANGED
@@ -5,8 +5,7 @@ class CursorTest < Test::Unit::TestCase
|
|
5
5
|
|
6
6
|
include Mongo
|
7
7
|
|
8
|
-
@@connection =
|
9
|
-
ENV['MONGO_RUBY_DRIVER_PORT'] || Connection::DEFAULT_PORT)
|
8
|
+
@@connection = standard_connection
|
10
9
|
@@db = @@connection.db(MONGO_TEST_DB)
|
11
10
|
@@coll = @@db.collection('test')
|
12
11
|
@@version = @@connection.server_version
|
data/test/db_api_test.rb
CHANGED
@@ -4,8 +4,7 @@ class DBAPITest < Test::Unit::TestCase
|
|
4
4
|
include Mongo
|
5
5
|
include BSON
|
6
6
|
|
7
|
-
@@conn =
|
8
|
-
ENV['MONGO_RUBY_DRIVER_PORT'] || Connection::DEFAULT_PORT)
|
7
|
+
@@conn = standard_connection
|
9
8
|
@@db = @@conn.db(MONGO_TEST_DB)
|
10
9
|
@@coll = @@db.collection('test')
|
11
10
|
@@version = @@conn.server_version
|
@@ -621,7 +620,6 @@ class DBAPITest < Test::Unit::TestCase
|
|
621
620
|
assert_equal("mike", @@coll.find_one()["hello"])
|
622
621
|
end
|
623
622
|
|
624
|
-
if !RUBY_PLATFORM =~ /java/
|
625
623
|
def test_invalid_key_names
|
626
624
|
@@coll.remove
|
627
625
|
|
@@ -658,7 +656,6 @@ class DBAPITest < Test::Unit::TestCase
|
|
658
656
|
@@coll.insert({"hello" => {"hel.lo" => "world"}})
|
659
657
|
end
|
660
658
|
end
|
661
|
-
end
|
662
659
|
|
663
660
|
def test_collection_names
|
664
661
|
assert_raise TypeError do
|
@@ -742,7 +739,12 @@ class DBAPITest < Test::Unit::TestCase
|
|
742
739
|
utf8 = "hello world".encode("UTF-8")
|
743
740
|
iso8859 = "hello world".encode("ISO-8859-1")
|
744
741
|
|
745
|
-
|
742
|
+
if RUBY_PLATFORM =~ /jruby/
|
743
|
+
assert_equal "ASCII-8BIT", ascii.encoding.name
|
744
|
+
else
|
745
|
+
assert_equal "US-ASCII", ascii.encoding.name
|
746
|
+
end
|
747
|
+
|
746
748
|
assert_equal "UTF-8", utf8.encoding.name
|
747
749
|
assert_equal "ISO-8859-1", iso8859.encoding.name
|
748
750
|
|
data/test/db_test.rb
CHANGED
@@ -14,9 +14,7 @@ class DBTest < Test::Unit::TestCase
|
|
14
14
|
|
15
15
|
include Mongo
|
16
16
|
|
17
|
-
@@
|
18
|
-
@@port = ENV['MONGO_RUBY_DRIVER_PORT'] || Connection::DEFAULT_PORT
|
19
|
-
@@conn = Connection.new(@@host, @@port)
|
17
|
+
@@conn = standard_connection
|
20
18
|
@@db = @@conn.db(MONGO_TEST_DB)
|
21
19
|
@@users = @@db.collection('system.users')
|
22
20
|
@@version = @@conn.server_version
|
@@ -30,7 +28,7 @@ class DBTest < Test::Unit::TestCase
|
|
30
28
|
rescue => ex
|
31
29
|
assert_match /NilClass/, ex.to_s
|
32
30
|
ensure
|
33
|
-
@@db =
|
31
|
+
@@db = standard_connection.db(MONGO_TEST_DB)
|
34
32
|
@@users = @@db.collection('system.users')
|
35
33
|
end
|
36
34
|
end
|
@@ -39,7 +37,7 @@ class DBTest < Test::Unit::TestCase
|
|
39
37
|
output = StringIO.new
|
40
38
|
logger = Logger.new(output)
|
41
39
|
logger.level = Logger::DEBUG
|
42
|
-
conn =
|
40
|
+
conn = standard_connection(:logger => logger)
|
43
41
|
assert_equal logger, conn.logger
|
44
42
|
|
45
43
|
conn.logger.debug 'testing'
|
@@ -76,7 +74,7 @@ class DBTest < Test::Unit::TestCase
|
|
76
74
|
end
|
77
75
|
|
78
76
|
def test_pk_factory
|
79
|
-
db =
|
77
|
+
db = standard_connection.db(MONGO_TEST_DB, :pk => TestPKFactory.new)
|
80
78
|
coll = db.collection('test')
|
81
79
|
coll.remove
|
82
80
|
|
@@ -99,7 +97,7 @@ class DBTest < Test::Unit::TestCase
|
|
99
97
|
end
|
100
98
|
|
101
99
|
def test_pk_factory_reset
|
102
|
-
conn =
|
100
|
+
conn = standard_connection
|
103
101
|
db = conn.db(MONGO_TEST_DB)
|
104
102
|
db.pk_factory = Object.new # first time
|
105
103
|
begin
|
@@ -127,10 +125,10 @@ class DBTest < Test::Unit::TestCase
|
|
127
125
|
|
128
126
|
def test_authenticate_with_connection_uri
|
129
127
|
@@db.add_user('spongebob', 'squarepants')
|
130
|
-
assert Mongo::Connection.from_uri("mongodb://spongebob:squarepants
|
128
|
+
assert Mongo::Connection.from_uri("mongodb://spongebob:squarepants@#{host_port}/#{@@db.name}")
|
131
129
|
|
132
130
|
assert_raise Mongo::AuthenticationError do
|
133
|
-
Mongo::Connection.from_uri("mongodb://wrong:info
|
131
|
+
Mongo::Connection.from_uri("mongodb://wrong:info@#{host_port}/#{@@db.name}")
|
134
132
|
end
|
135
133
|
end
|
136
134
|
|
@@ -196,7 +194,7 @@ class DBTest < Test::Unit::TestCase
|
|
196
194
|
end
|
197
195
|
|
198
196
|
def test_text_port_number_raises_no_errors
|
199
|
-
conn =
|
197
|
+
conn = standard_connection
|
200
198
|
db = conn[MONGO_TEST_DB]
|
201
199
|
db.collection('users').remove
|
202
200
|
end
|
@@ -4,9 +4,8 @@ include Mongo
|
|
4
4
|
class GridFileSystemTest < Test::Unit::TestCase
|
5
5
|
context "GridFileSystem:" do
|
6
6
|
setup do
|
7
|
-
@con =
|
8
|
-
|
9
|
-
@db = @con.db(MONGO_TEST_DB)
|
7
|
+
@con = standard_connection
|
8
|
+
@db = @con.db(MONGO_TEST_DB)
|
10
9
|
end
|
11
10
|
|
12
11
|
teardown do
|
data/test/grid_io_test.rb
CHANGED
@@ -5,8 +5,7 @@ class GridIOTest < Test::Unit::TestCase
|
|
5
5
|
|
6
6
|
context "GridIO" do
|
7
7
|
setup do
|
8
|
-
@db
|
9
|
-
ENV['MONGO_RUBY_DRIVER_PORT'] || Connection::DEFAULT_PORT).db(MONGO_TEST_DB)
|
8
|
+
@db = standard_connection.db(MONGO_TEST_DB)
|
10
9
|
@files = @db.collection('fs.files')
|
11
10
|
@chunks = @db.collection('fs.chunks')
|
12
11
|
@chunks.create_index([['files_id', Mongo::ASCENDING], ['n', Mongo::ASCENDING]])
|
data/test/grid_test.rb
CHANGED
@@ -4,8 +4,7 @@ include Mongo
|
|
4
4
|
class GridTest < Test::Unit::TestCase
|
5
5
|
context "Tests:" do
|
6
6
|
setup do
|
7
|
-
@db
|
8
|
-
ENV['MONGO_RUBY_DRIVER_PORT'] || Connection::DEFAULT_PORT).db(MONGO_TEST_DB)
|
7
|
+
@db = standard_connection.db(MONGO_TEST_DB)
|
9
8
|
@files = @db.collection('test-fs.files')
|
10
9
|
@chunks = @db.collection('test-fs.chunks')
|
11
10
|
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
$:.unshift(File.join(File.dirname(__FILE__), '..', 'lib'))
|
2
|
+
require 'mongo'
|
3
|
+
require 'test/unit'
|
4
|
+
require './test/test_helper'
|
5
|
+
|
6
|
+
# NOTE: This test expects a replica set of three nodes to be running on local host.
|
7
|
+
class ConnectTest < Test::Unit::TestCase
|
8
|
+
include Mongo
|
9
|
+
|
10
|
+
def test_connect
|
11
|
+
@conn = Mongo::Connection.multi([['localhost', 27017], ['localhost', 27018], ['localhost', 27019]])
|
12
|
+
assert @conn.connected?
|
13
|
+
end
|
14
|
+
|
15
|
+
def test_connect_with_first_node_down
|
16
|
+
puts "Please kill the node at 27017."
|
17
|
+
gets
|
18
|
+
|
19
|
+
@conn = Mongo::Connection.multi([['localhost', 27017], ['localhost', 27018], ['localhost', 27019]])
|
20
|
+
assert @conn.connected?
|
21
|
+
end
|
22
|
+
|
23
|
+
end
|
@@ -0,0 +1,199 @@
|
|
1
|
+
# Note: HashWithIndifferentAccess is so commonly used
|
2
|
+
# that we always need to make sure that the driver works
|
3
|
+
# with it.
|
4
|
+
#require File.join(File.dirname(__FILE__), 'keys.rb')
|
5
|
+
|
6
|
+
# This class has dubious semantics and we only have it so that
|
7
|
+
# people can write params[:key] instead of params['key']
|
8
|
+
# and they get the same value for both keys.
|
9
|
+
|
10
|
+
class Hash
|
11
|
+
# Return a new hash with all keys converted to strings.
|
12
|
+
def stringify_keys
|
13
|
+
dup.stringify_keys!
|
14
|
+
end
|
15
|
+
|
16
|
+
# Destructively convert all keys to strings.
|
17
|
+
def stringify_keys!
|
18
|
+
keys.each do |key|
|
19
|
+
self[key.to_s] = delete(key)
|
20
|
+
end
|
21
|
+
self
|
22
|
+
end
|
23
|
+
|
24
|
+
# Return a new hash with all keys converted to symbols, as long as
|
25
|
+
# they respond to +to_sym+.
|
26
|
+
def symbolize_keys
|
27
|
+
dup.symbolize_keys!
|
28
|
+
end
|
29
|
+
|
30
|
+
# Destructively convert all keys to symbols, as long as they respond
|
31
|
+
# to +to_sym+.
|
32
|
+
def symbolize_keys!
|
33
|
+
keys.each do |key|
|
34
|
+
self[(key.to_sym rescue key) || key] = delete(key)
|
35
|
+
end
|
36
|
+
self
|
37
|
+
end
|
38
|
+
|
39
|
+
alias_method :to_options, :symbolize_keys
|
40
|
+
#alias_method :to_options!, :symbolize_keys!
|
41
|
+
|
42
|
+
# Validate all keys in a hash match *valid keys, raising ArgumentError on a mismatch.
|
43
|
+
# Note that keys are NOT treated indifferently, meaning if you use strings for keys but assert symbols
|
44
|
+
# as keys, this will fail.
|
45
|
+
#
|
46
|
+
# ==== Examples
|
47
|
+
# { :name => "Rob", :years => "28" }.assert_valid_keys(:name, :age) # => raises "ArgumentError: Unknown key(s): years"
|
48
|
+
# { :name => "Rob", :age => "28" }.assert_valid_keys("name", "age") # => raises "ArgumentError: Unknown key(s): name, age"
|
49
|
+
# { :name => "Rob", :age => "28" }.assert_valid_keys(:name, :age) # => passes, raises nothing
|
50
|
+
def assert_valid_keys(*valid_keys)
|
51
|
+
unknown_keys = keys - [valid_keys].flatten
|
52
|
+
raise(ArgumentError, "Unknown key(s): #{unknown_keys.join(", ")}") unless unknown_keys.empty?
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
56
|
+
module ActiveSupport
|
57
|
+
class HashWithIndifferentAccess < Hash
|
58
|
+
def extractable_options?
|
59
|
+
true
|
60
|
+
end
|
61
|
+
|
62
|
+
def initialize(constructor = {})
|
63
|
+
if constructor.is_a?(Hash)
|
64
|
+
super()
|
65
|
+
update(constructor)
|
66
|
+
else
|
67
|
+
super(constructor)
|
68
|
+
end
|
69
|
+
end
|
70
|
+
|
71
|
+
def default(key = nil)
|
72
|
+
if key.is_a?(Symbol) && include?(key = key.to_s)
|
73
|
+
self[key]
|
74
|
+
else
|
75
|
+
super
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
def self.new_from_hash_copying_default(hash)
|
80
|
+
ActiveSupport::HashWithIndifferentAccess.new(hash).tap do |new_hash|
|
81
|
+
new_hash.default = hash.default
|
82
|
+
end
|
83
|
+
end
|
84
|
+
|
85
|
+
alias_method :regular_writer, :[]= unless method_defined?(:regular_writer)
|
86
|
+
alias_method :regular_update, :update unless method_defined?(:regular_update)
|
87
|
+
|
88
|
+
# Assigns a new value to the hash:
|
89
|
+
#
|
90
|
+
# hash = HashWithIndifferentAccess.new
|
91
|
+
# hash[:key] = "value"
|
92
|
+
#
|
93
|
+
def []=(key, value)
|
94
|
+
regular_writer(convert_key(key), convert_value(value))
|
95
|
+
end
|
96
|
+
|
97
|
+
# Updates the instantized hash with values from the second:
|
98
|
+
#
|
99
|
+
# hash_1 = HashWithIndifferentAccess.new
|
100
|
+
# hash_1[:key] = "value"
|
101
|
+
#
|
102
|
+
# hash_2 = HashWithIndifferentAccess.new
|
103
|
+
# hash_2[:key] = "New Value!"
|
104
|
+
#
|
105
|
+
# hash_1.update(hash_2) # => {"key"=>"New Value!"}
|
106
|
+
#
|
107
|
+
def update(other_hash)
|
108
|
+
other_hash.each_pair { |key, value| regular_writer(convert_key(key), convert_value(value)) }
|
109
|
+
self
|
110
|
+
end
|
111
|
+
|
112
|
+
alias_method :merge!, :update
|
113
|
+
|
114
|
+
# Checks the hash for a key matching the argument passed in:
|
115
|
+
#
|
116
|
+
# hash = HashWithIndifferentAccess.new
|
117
|
+
# hash["key"] = "value"
|
118
|
+
# hash.key? :key # => true
|
119
|
+
# hash.key? "key" # => true
|
120
|
+
#
|
121
|
+
def key?(key)
|
122
|
+
super(convert_key(key))
|
123
|
+
end
|
124
|
+
|
125
|
+
alias_method :include?, :key?
|
126
|
+
alias_method :has_key?, :key?
|
127
|
+
alias_method :member?, :key?
|
128
|
+
|
129
|
+
# Fetches the value for the specified key, same as doing hash[key]
|
130
|
+
def fetch(key, *extras)
|
131
|
+
super(convert_key(key), *extras)
|
132
|
+
end
|
133
|
+
|
134
|
+
# Returns an array of the values at the specified indices:
|
135
|
+
#
|
136
|
+
# hash = HashWithIndifferentAccess.new
|
137
|
+
# hash[:a] = "x"
|
138
|
+
# hash[:b] = "y"
|
139
|
+
# hash.values_at("a", "b") # => ["x", "y"]
|
140
|
+
#
|
141
|
+
def values_at(*indices)
|
142
|
+
indices.collect {|key| self[convert_key(key)]}
|
143
|
+
end
|
144
|
+
|
145
|
+
# Returns an exact copy of the hash.
|
146
|
+
def dup
|
147
|
+
HashWithIndifferentAccess.new(self)
|
148
|
+
end
|
149
|
+
|
150
|
+
# Merges the instantized and the specified hashes together, giving precedence to the values from the second hash
|
151
|
+
# Does not overwrite the existing hash.
|
152
|
+
def merge(hash)
|
153
|
+
self.dup.update(hash)
|
154
|
+
end
|
155
|
+
|
156
|
+
# Performs the opposite of merge, with the keys and values from the first hash taking precedence over the second.
|
157
|
+
# This overloaded definition prevents returning a regular hash, if reverse_merge is called on a HashWithDifferentAccess.
|
158
|
+
def reverse_merge(other_hash)
|
159
|
+
super self.class.new_from_hash_copying_default(other_hash)
|
160
|
+
end
|
161
|
+
|
162
|
+
def reverse_merge!(other_hash)
|
163
|
+
replace(reverse_merge( other_hash ))
|
164
|
+
end
|
165
|
+
|
166
|
+
# Removes a specified key from the hash.
|
167
|
+
def delete(key)
|
168
|
+
super(convert_key(key))
|
169
|
+
end
|
170
|
+
|
171
|
+
def stringify_keys!; self end
|
172
|
+
def stringify_keys; dup end
|
173
|
+
def symbolize_keys; to_hash.symbolize_keys end
|
174
|
+
def to_options!; self end
|
175
|
+
|
176
|
+
# Convert to a Hash with String keys.
|
177
|
+
def to_hash
|
178
|
+
Hash.new(default).merge!(self)
|
179
|
+
end
|
180
|
+
|
181
|
+
protected
|
182
|
+
def convert_key(key)
|
183
|
+
key.kind_of?(Symbol) ? key.to_s : key
|
184
|
+
end
|
185
|
+
|
186
|
+
def convert_value(value)
|
187
|
+
case value
|
188
|
+
when Hash
|
189
|
+
self.class.new_from_hash_copying_default(value)
|
190
|
+
when Array
|
191
|
+
value.collect { |e| e.is_a?(Hash) ? self.class.new_from_hash_copying_default(e) : e }
|
192
|
+
else
|
193
|
+
value
|
194
|
+
end
|
195
|
+
end
|
196
|
+
end
|
197
|
+
end
|
198
|
+
|
199
|
+
HashWithIndifferentAccess = ActiveSupport::HashWithIndifferentAccess
|