Merge branch 'master' of git://github.com/jimm/mongo-ruby-driver
This commit is contained in:
commit
50d3ff6b03
13
README
13
README
@ -52,6 +52,11 @@ type
|
|||||||
|
|
||||||
= To Do
|
= To Do
|
||||||
|
|
||||||
|
* Study src/main/ed/db/{dbcollection,dbcursor,db}.js in the Babble code.
|
||||||
|
That's what I should be writing to.
|
||||||
|
|
||||||
|
* Capped collection support.
|
||||||
|
|
||||||
* More code comments. More text in this file.
|
* More code comments. More text in this file.
|
||||||
|
|
||||||
* Rake task for rdoc generation.
|
* Rake task for rdoc generation.
|
||||||
@ -69,6 +74,14 @@ type
|
|||||||
* See FIXME in db test.
|
* See FIXME in db test.
|
||||||
|
|
||||||
|
|
||||||
|
= Credits
|
||||||
|
|
||||||
|
Adrian Madrid, aemadrid@gmail.com
|
||||||
|
* examples/benchmarks.rb
|
||||||
|
* examples/irb.rb
|
||||||
|
* modifications to examples/simple.rb
|
||||||
|
|
||||||
|
|
||||||
= License
|
= License
|
||||||
|
|
||||||
== Mongo Ruby Driver
|
== Mongo Ruby Driver
|
||||||
|
@ -69,7 +69,8 @@ module XGen
|
|||||||
end
|
end
|
||||||
|
|
||||||
def drop_indexes
|
def drop_indexes
|
||||||
index_information.each { |info| @db.drop_index(@name, info.name) }
|
# just need to call drop indexes with no args; will drop them all
|
||||||
|
@db.drop_index(@name, '*')
|
||||||
end
|
end
|
||||||
|
|
||||||
def index_information
|
def index_information
|
||||||
|
@ -17,6 +17,7 @@ require 'mongo/objectid'
|
|||||||
require 'mongo/collection'
|
require 'mongo/collection'
|
||||||
require 'mongo/message'
|
require 'mongo/message'
|
||||||
require 'mongo/query'
|
require 'mongo/query'
|
||||||
|
require 'mongo/util/ordered_hash.rb'
|
||||||
|
|
||||||
module XGen
|
module XGen
|
||||||
module Mongo
|
module Mongo
|
||||||
@ -52,8 +53,9 @@ module XGen
|
|||||||
return Collection.new(self, name) if collection_names.include?(name)
|
return Collection.new(self, name) if collection_names.include?(name)
|
||||||
|
|
||||||
# Create new collection
|
# Create new collection
|
||||||
sel = {:create => name}.merge(options)
|
oh = OrderedHash.new
|
||||||
doc = db_command(sel)
|
oh[:create] = name
|
||||||
|
doc = db_command(oh.merge(options))
|
||||||
o = doc['ok']
|
o = doc['ok']
|
||||||
return Collection.new(self, name) if o.kind_of?(Numeric) && (o.to_i == 1 || o.to_i == 0)
|
return Collection.new(self, name) if o.kind_of?(Numeric) && (o.to_i == 1 || o.to_i == 0)
|
||||||
raise "Error creating collection: #{doc.inspect}"
|
raise "Error creating collection: #{doc.inspect}"
|
||||||
@ -89,85 +91,103 @@ module XGen
|
|||||||
send_to_db(MsgMessage.new(msg))
|
send_to_db(MsgMessage.new(msg))
|
||||||
end
|
end
|
||||||
|
|
||||||
def query(collection, query)
|
def query(collection_name, query)
|
||||||
# TODO synchronize
|
# TODO synchronize
|
||||||
send_to_db(QueryMessage.new(@name, collection, query))
|
send_to_db(QueryMessage.new(@name, collection_name, query))
|
||||||
return Cursor.new(self, collection)
|
return Cursor.new(self, collection_name)
|
||||||
end
|
end
|
||||||
|
|
||||||
def remove_from_db(collection, selector)
|
def remove_from_db(collection_name, selector)
|
||||||
# TODO synchronize
|
# TODO synchronize
|
||||||
send_to_db(RemoveMessage.new(@name, collection, selector))
|
send_to_db(RemoveMessage.new(@name, collection_name, selector))
|
||||||
end
|
end
|
||||||
|
|
||||||
def replace_in_db(collection, selector, obj)
|
def replace_in_db(collection_name, selector, obj)
|
||||||
# TODO synchronize
|
# TODO synchronize
|
||||||
send_to_db(UpdateMessage.new(@name, collection, selector, obj, false))
|
send_to_db(UpdateMessage.new(@name, collection_name, selector, obj, false))
|
||||||
end
|
end
|
||||||
alias_method :modify_in_db, :replace_in_db
|
alias_method :modify_in_db, :replace_in_db
|
||||||
|
|
||||||
def repsert_in_db(collection, selector, obj)
|
def repsert_in_db(collection_name, selector, obj)
|
||||||
# TODO if PKInjector, inject
|
# TODO if PKInjector, inject
|
||||||
# TODO synchronize
|
# TODO synchronize
|
||||||
send_to_db(UpdateMessage.new(@name, collection, selector, obj, true))
|
send_to_db(UpdateMessage.new(@name, collection_name, selector, obj, true))
|
||||||
obj
|
obj
|
||||||
end
|
end
|
||||||
|
|
||||||
def count(collection, selector)
|
def count(collection_name, selector)
|
||||||
doc = db_command(:count => collection, :query => selector)
|
oh = OrderedHash.new
|
||||||
|
oh[:count] = collection_name
|
||||||
|
oh[:query] = selector
|
||||||
|
doc = db_command(oh)
|
||||||
o = doc['ok']
|
o = doc['ok']
|
||||||
return doc['n'].to_i if o.to_i == 1
|
return doc['n'].to_i if o.to_i == 1
|
||||||
raise "Error with count command: #{doc.inspect}"
|
raise "Error with count command: #{doc.inspect}"
|
||||||
end
|
end
|
||||||
|
|
||||||
def drop_index(collection, name)
|
def drop_index(collection_name, name)
|
||||||
db_command(:deleteIndexes => collection, :index => name)
|
oh = OrderedHash.new
|
||||||
|
oh[:deleteIndexes] = collection_name
|
||||||
|
oh[:index] = name
|
||||||
|
doc = db_command(oh)
|
||||||
|
o = doc['ok']
|
||||||
|
raise "Error with drop_index command: #{doc.inspect}" unless o.kind_of?(Numeric) && o.to_i == 1
|
||||||
end
|
end
|
||||||
|
|
||||||
def index_information(collection)
|
def index_information(collection_name)
|
||||||
sel = {:ns => full_coll_name(collection)}
|
sel = {:ns => full_coll_name(collection_name)}
|
||||||
# TODO synchronize
|
# TODO synchronize
|
||||||
query(SYSTEM_INDEX_COLLECTION, Query.new(sel)).collect { |row|
|
query(SYSTEM_INDEX_COLLECTION, Query.new(sel)).collect { |row|
|
||||||
h = {:name => row['name']}
|
h = {:name => row['name']}
|
||||||
raise "Name of index on return from db was nil. Coll = #{full_coll_name(collection)}" unless h[:name]
|
raise "Name of index on return from db was nil. Coll = #{full_coll_name(collection_name)}" unless h[:name]
|
||||||
|
|
||||||
h[:keys] = row['keys']
|
h[:keys] = row['key']
|
||||||
raise "Keys for index on return from db was nil. Coll = #{full_coll_name(collection)}" unless h[:keys]
|
raise "Keys for index on return from db was nil. Coll = #{full_coll_name(collection_name)}" unless h[:keys]
|
||||||
|
|
||||||
h[:ns] = row['ns']
|
h[:ns] = row['ns']
|
||||||
raise "Namespace for index on return from db was nil. Coll = #{full_coll_name(collection)}" unless h[:ns]
|
raise "Namespace for index on return from db was nil. Coll = #{full_coll_name(collection_name)}" unless h[:ns]
|
||||||
h[:ns].sub!(/.*\./, '')
|
h[:ns].sub!(/.*\./, '')
|
||||||
raise "Error: ns != collection" unless h[:ns] == collection
|
raise "Error: ns != collection" unless h[:ns] == collection_name
|
||||||
|
|
||||||
h
|
h
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
def create_index(collection, name, fields)
|
def create_index(collection_name, index_name, fields)
|
||||||
sel = {:name => name, :ns => full_coll_name(collection)}
|
sel = {:name => index_name, :ns => full_coll_name(collection_name)}
|
||||||
field_h = {}
|
field_h = {}
|
||||||
fields.each { |f| field_h[f] = 1 }
|
fields.each { |f| field_h[f] = 1 }
|
||||||
sel['key'] = field_h
|
sel[:key] = field_h
|
||||||
# TODO synchronize
|
# TODO synchronize
|
||||||
send_to_db(InsertMessage.new(@name, SYSTEM_INDEX_COLLECTION, sel))
|
send_to_db(InsertMessage.new(@name, SYSTEM_INDEX_COLLECTION, sel))
|
||||||
end
|
end
|
||||||
|
|
||||||
def insert_into_db(collection, objects)
|
def insert_into_db(collection_name, objects)
|
||||||
|
objects.each { |o| o['_id'] ||= ObjectID.new }
|
||||||
# TODO synchronize
|
# TODO synchronize
|
||||||
objects.each { |o| send_to_db(InsertMessage.new(@name, collection, o)) }
|
objects.each { |o| send_to_db(InsertMessage.new(@name, collection_name, o)) }
|
||||||
end
|
end
|
||||||
|
|
||||||
def send_to_db(message)
|
def send_to_db(message)
|
||||||
@socket.print(message.buf.to_s)
|
@socket.print(message.buf.to_s)
|
||||||
end
|
end
|
||||||
|
|
||||||
def full_coll_name(collection)
|
def full_coll_name(collection_name)
|
||||||
"#{@name}.#{collection}"
|
"#{@name}.#{collection_name}"
|
||||||
end
|
end
|
||||||
|
|
||||||
protected
|
protected
|
||||||
|
|
||||||
|
# DB commands need to be ordered, so selector must be an OrderedHash
|
||||||
|
# (or a Hash with only one element). What DB commands really need is
|
||||||
|
# that the "command" key be first.
|
||||||
def db_command(selector)
|
def db_command(selector)
|
||||||
|
if !selector.kind_of?(OrderedHash)
|
||||||
|
if !selector.kind_of?(Hash) || selector.keys.length > 1
|
||||||
|
raise "db_command must be given an OrderedHash when there is more than one key"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
# TODO synchronize
|
# TODO synchronize
|
||||||
q = Query.new(selector)
|
q = Query.new(selector)
|
||||||
q.number_to_return = 1
|
q.number_to_return = 1
|
||||||
|
@ -7,10 +7,10 @@ module XGen
|
|||||||
|
|
||||||
class GetMoreMessage < Message
|
class GetMoreMessage < Message
|
||||||
|
|
||||||
def initialize(name, collection, cursor)
|
def initialize(db_name, collection_name, cursor)
|
||||||
super(OP_GET_MORE)
|
super(OP_GET_MORE)
|
||||||
write_int(0)
|
write_int(0)
|
||||||
write_string("#{name}.#{collection}")
|
write_string("#{db_name}.#{collection_name}")
|
||||||
write_int(0) # num to return; leave it up to the db for now
|
write_int(0) # num to return; leave it up to the db for now
|
||||||
write_long(cursor)
|
write_long(cursor)
|
||||||
end
|
end
|
||||||
|
@ -7,14 +7,13 @@ module XGen
|
|||||||
|
|
||||||
class InsertMessage < Message
|
class InsertMessage < Message
|
||||||
|
|
||||||
def initialize(name, collection, *objs)
|
def initialize(db_name, collection_name, *objs)
|
||||||
super(OP_INSERT)
|
super(OP_INSERT)
|
||||||
write_int(0)
|
write_int(0)
|
||||||
write_string("#{name}.#{collection}")
|
write_string("#{db_name}.#{collection_name}")
|
||||||
objs.each { |o| write_doc(o) }
|
objs.each { |o| write_doc(o) }
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -7,15 +7,19 @@ module XGen
|
|||||||
|
|
||||||
class QueryMessage < Message
|
class QueryMessage < Message
|
||||||
|
|
||||||
def initialize(name, collection, query)
|
def initialize(db_name, collection_name, query)
|
||||||
super(OP_QUERY)
|
super(OP_QUERY)
|
||||||
write_int(0)
|
write_int(0)
|
||||||
write_string("#{name}.#{collection}")
|
write_string("#{db_name}.#{collection_name}")
|
||||||
write_int(query.number_to_skip)
|
write_int(query.number_to_skip)
|
||||||
write_int(query.number_to_return)
|
write_int(query.number_to_return)
|
||||||
write_doc(query.selector)
|
write_doc(query.selector)
|
||||||
write_doc(query.fields) if query.fields
|
write_doc(query.fields) if query.fields
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def first_key(key)
|
||||||
|
@first_key = key
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -7,10 +7,10 @@ module XGen
|
|||||||
|
|
||||||
class RemoveMessage < Message
|
class RemoveMessage < Message
|
||||||
|
|
||||||
def initialize(name, collection, sel)
|
def initialize(db_name, collection_name, sel)
|
||||||
super(OP_DELETE)
|
super(OP_DELETE)
|
||||||
write_int(0)
|
write_int(0)
|
||||||
write_string("#{name}.#{collection}")
|
write_string("#{db_name}.#{collection_name}")
|
||||||
write_int(0) # flags?
|
write_int(0) # flags?
|
||||||
write_doc(sel)
|
write_doc(sel)
|
||||||
end
|
end
|
||||||
|
@ -7,10 +7,10 @@ module XGen
|
|||||||
|
|
||||||
class UpdateMessage < Message
|
class UpdateMessage < Message
|
||||||
|
|
||||||
def initialize(name, collection, sel, obj, repsert)
|
def initialize(db_name, collection_name, sel, obj, repsert)
|
||||||
super(OP_UPDATE)
|
super(OP_UPDATE)
|
||||||
write_int(0)
|
write_int(0)
|
||||||
write_string("#{name}.#{collection}")
|
write_string("#{db_name}.#{collection_name}")
|
||||||
write_int(repsert ? 1 : 0) # 1 if a repsert operation (upsert)
|
write_int(repsert ? 1 : 0) # 1 if a repsert operation (upsert)
|
||||||
write_doc(sel)
|
write_doc(sel)
|
||||||
write_doc(obj)
|
write_doc(obj)
|
||||||
|
@ -22,8 +22,6 @@ module XGen
|
|||||||
|
|
||||||
include Comparable
|
include Comparable
|
||||||
|
|
||||||
UUID_STRING_LENGTH = 24
|
|
||||||
|
|
||||||
@@uuid_generator = UUID.new
|
@@uuid_generator = UUID.new
|
||||||
|
|
||||||
# String UUID
|
# String UUID
|
||||||
|
@ -156,7 +156,10 @@ class BSON
|
|||||||
end
|
end
|
||||||
|
|
||||||
def deserialize_oid_data(buf)
|
def deserialize_oid_data(buf)
|
||||||
XGen::Mongo::Driver::ObjectID.new(buf.get(XGen::Mongo::Driver::ObjectID::UUID_STRING_LENGTH).pack("C*"))
|
high_bytes = buf.get_long
|
||||||
|
low_bytes = buf.get_int
|
||||||
|
hexval = (high_bytes << 32) + low_bytes
|
||||||
|
XGen::Mongo::Driver::ObjectID.new('%012x' % hexval)
|
||||||
end
|
end
|
||||||
|
|
||||||
def serialize_eoo_element(buf)
|
def serialize_eoo_element(buf)
|
||||||
@ -199,7 +202,12 @@ class BSON
|
|||||||
def serialize_oid_element(buf, key, val)
|
def serialize_oid_element(buf, key, val)
|
||||||
buf.put(OID)
|
buf.put(OID)
|
||||||
self.class.serialize_cstr(buf, key)
|
self.class.serialize_cstr(buf, key)
|
||||||
buf.put_array(val.to_s.unpack("C*"))
|
|
||||||
|
hexval = val.to_s.hex
|
||||||
|
high_bytes = hexval >> 32
|
||||||
|
low_bytes = hexval && 0xffffffff
|
||||||
|
buf.put_long(high_bytes)
|
||||||
|
buf.put_int(low_bytes)
|
||||||
end
|
end
|
||||||
|
|
||||||
def serialize_string_element(buf, key, val, type)
|
def serialize_string_element(buf, key, val, type)
|
||||||
|
26
lib/mongo/util/ordered_hash.rb
Normal file
26
lib/mongo/util/ordered_hash.rb
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
class OrderedHash < Hash
|
||||||
|
|
||||||
|
attr_accessor :ordered_keys
|
||||||
|
|
||||||
|
def keys
|
||||||
|
@ordered_keys || []
|
||||||
|
end
|
||||||
|
|
||||||
|
def []=(key, value)
|
||||||
|
@ordered_keys ||= []
|
||||||
|
@ordered_keys << key unless @ordered_keys.include?(key)
|
||||||
|
super(key, value)
|
||||||
|
end
|
||||||
|
|
||||||
|
def each
|
||||||
|
@ordered_keys ||= []
|
||||||
|
@ordered_keys.each { |k| yield k, self[k] }
|
||||||
|
end
|
||||||
|
|
||||||
|
def merge(other)
|
||||||
|
@ordered_keys ||= []
|
||||||
|
@ordered_keys += other.keys # unordered if not an OrderedHash
|
||||||
|
super(other)
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
@ -17,7 +17,7 @@ class DBAPITest < Test::Unit::TestCase
|
|||||||
end
|
end
|
||||||
|
|
||||||
def teardown
|
def teardown
|
||||||
@coll.clear unless @db.socket.closed?
|
@coll.clear unless @coll == nil || @db.socket.closed?
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_clear
|
def test_clear
|
||||||
@ -33,9 +33,9 @@ class DBAPITest < Test::Unit::TestCase
|
|||||||
assert_equal 3, @coll.count
|
assert_equal 3, @coll.count
|
||||||
docs = @coll.find().collect
|
docs = @coll.find().collect
|
||||||
assert_equal 3, docs.length
|
assert_equal 3, docs.length
|
||||||
assert docs.any?{|x| x['a'] == 1}
|
assert docs.detect { |row| row['a'] == 1 }
|
||||||
assert docs.any?{|x| x['a'] == 2}
|
assert docs.detect { |row| row['a'] == 2 }
|
||||||
assert docs.any?{|x| x['b'] == 3}
|
assert docs.detect { |row| row['b'] == 3 }
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_inserted_id
|
def test_inserted_id
|
||||||
@ -73,6 +73,7 @@ class DBAPITest < Test::Unit::TestCase
|
|||||||
def test_drop_collection
|
def test_drop_collection
|
||||||
assert @db.drop_collection(@coll.name), "drop of collection #{@coll.name} failed"
|
assert @db.drop_collection(@coll.name), "drop of collection #{@coll.name} failed"
|
||||||
assert !@db.collection_names.include?(@coll_full_name)
|
assert !@db.collection_names.include?(@coll_full_name)
|
||||||
|
@coll = nil
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_collection_names
|
def test_collection_names
|
||||||
@ -103,17 +104,14 @@ class DBAPITest < Test::Unit::TestCase
|
|||||||
assert_equal @coll_full_name, @db.full_coll_name(@coll.name)
|
assert_equal @coll_full_name, @db.full_coll_name(@coll.name)
|
||||||
end
|
end
|
||||||
|
|
||||||
# FIXME
|
def test_index_information
|
||||||
# def test_index_information
|
@db.create_index(@coll.name, 'index_name', ['a'])
|
||||||
# list = @db.index_information(@coll.name)
|
list = @db.index_information(@coll.name)
|
||||||
# assert_equal 0, list.length
|
assert_equal 1, list.length
|
||||||
|
info = list[0]
|
||||||
# @db.create_index(@coll, 'index_name', {'a' => 1})
|
assert_equal 'index_name', info[:name]
|
||||||
# $stderr.puts @db.create_index(@coll, 'index_name', {'a' => 1}).to_s # DEBUG
|
assert_equal 1, info[:keys]['a']
|
||||||
# list = @db.index_information(@coll.name)
|
end
|
||||||
# $stderr.puts "list = #{list.inspect}" # DEBUG
|
|
||||||
# assert_equal 1, list.length
|
|
||||||
# end
|
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user