Use an Arc to share ClientPool between threads

pull/4/head
Thijs Cadier 9 years ago
parent 8b3a94a6e8
commit 5aa1b62faf

@ -20,14 +20,18 @@ use super::database::Database;
use super::uri::Uri; use super::uri::Uri;
use super::read_prefs::ReadPrefs; use super::read_prefs::ReadPrefs;
// TODO: We're using a sort of poor man's Arc here /// Client pool to a MongoDB cluster.
// with this root bool, there must be a better way. ///
/// This client pool cannot be cloned, but it can be shared between threads by using an `Arc`.
/// Use the pool to pop a client and do operations. The client will be automatically added
/// back to the pool when it goes out of scope.
///
/// See: http://api.mongodb.org/c/current/mongoc_client_pool_t.html
pub struct ClientPool { pub struct ClientPool {
root_instance: bool, // Uri and SslOptions need to be present for the lifetime of this pool otherwise the C driver
// Uri and SslOptions need to be present for the // loses access to resources it needs.
// lifetime of this pool.
uri: Uri, uri: Uri,
ssl_options: Option<SslOptions>, _ssl_options: Option<SslOptions>,
inner: *mut bindings::mongoc_client_pool_t inner: *mut bindings::mongoc_client_pool_t
} }
@ -55,9 +59,8 @@ impl ClientPool {
None => () None => ()
}; };
ClientPool { ClientPool {
root_instance: true,
uri: uri, uri: uri,
ssl_options: ssl_options, _ssl_options: ssl_options,
inner: pool inner: pool
} }
} }
@ -82,6 +85,7 @@ impl ClientPool {
/// See: http://api.mongodb.org/c/current/mongoc_client_pool_push.html /// See: http://api.mongodb.org/c/current/mongoc_client_pool_push.html
unsafe fn push(&self, mongo_client: *mut bindings::mongoc_client_t) { unsafe fn push(&self, mongo_client: *mut bindings::mongoc_client_t) {
assert!(!self.inner.is_null()); assert!(!self.inner.is_null());
assert!(!mongo_client.is_null());
bindings::mongoc_client_pool_push( bindings::mongoc_client_pool_push(
self.inner, self.inner,
mongo_client mongo_client
@ -98,27 +102,13 @@ impl fmt::Debug for ClientPool {
} }
} }
impl Clone for ClientPool {
fn clone(&self) -> ClientPool {
assert!(!self.inner.is_null());
ClientPool {
root_instance: false,
uri: self.uri.clone(),
ssl_options: self.ssl_options.clone(),
inner: self.inner.clone()
}
}
}
impl Drop for ClientPool { impl Drop for ClientPool {
fn drop(&mut self) { fn drop(&mut self) {
if self.root_instance {
assert!(!self.inner.is_null()); assert!(!self.inner.is_null());
unsafe { unsafe {
bindings::mongoc_client_pool_destroy(self.inner); bindings::mongoc_client_pool_destroy(self.inner);
} }
} }
}
} }
pub struct SslOptions { pub struct SslOptions {

@ -1,5 +1,6 @@
use std::env; use std::env;
use std::path::PathBuf; use std::path::PathBuf;
use std::sync::Arc;
use std::thread; use std::thread;
use mongo_driver::uri::Uri; use mongo_driver::uri::Uri;
@ -51,7 +52,7 @@ fn test_new_pool_pop_client_and_take_database_and_collection() {
#[test] #[test]
fn test_new_pool_and_pop_client_in_threads() { fn test_new_pool_and_pop_client_in_threads() {
let uri = Uri::new("mongodb://localhost:27017/").unwrap(); let uri = Uri::new("mongodb://localhost:27017/").unwrap();
let pool = ClientPool::new(uri, None); let pool = Arc::new(ClientPool::new(uri, None));
let pool1 = pool.clone(); let pool1 = pool.clone();
let guard1 = thread::spawn(move || { let guard1 = thread::spawn(move || {

@ -1,3 +1,4 @@
use std::sync::Arc;
use std::thread; use std::thread;
use bson; use bson;
@ -34,7 +35,7 @@ fn test_tailing_cursor() {
// See: http://api.mongodb.org/c/1.1.8/cursors.html#tailable // See: http://api.mongodb.org/c/1.1.8/cursors.html#tailable
let uri = Uri::new("mongodb://localhost:27017/").unwrap(); let uri = Uri::new("mongodb://localhost:27017/").unwrap();
let pool = ClientPool::new(uri, None); let pool = Arc::new(ClientPool::new(uri, None));
let client = pool.pop(); let client = pool.pop();
let database = client.get_database("rust_test"); let database = client.get_database("rust_test");
database.get_collection("capped").drop().unwrap_or(()); database.get_collection("capped").drop().unwrap_or(());

Loading…
Cancel
Save