Reputation: 893
I want to make lazy connection to a Redis database. I have a Db
struct, which holds the Redis Client
. By default it is None
. Here is example code in Python:
import redis
class Db:
def __init__(self):
self.client = None
def get_client(self):
if self.client is None:
self.client = redis.StrictRedis(host='127.0.0.1')
return self.client
I tried this
extern crate redis;
use redis::Client;
struct Db {
client: Option<Client>,
}
impl Db {
fn new() -> Db {
Db { client: None }
}
fn get_client(&mut self) -> Result<&Client, &'static str> {
if let Some(ref client) = self.client {
Ok(client)
} else {
let connection_string = "redis://127.0.0.1";
match Client::open(connection_string) {
Ok(client) => {
self.client = Some(client);
Ok(&self.client.unwrap())
}
Err(err) => Err("Error!"),
}
}
}
}
fn main() {
let mut db = Db::new();
db.get_client();
}
And I have compile errors. I almost understand what compiler says, but I don't know how to solve the problem.
error: borrowed value does not live long enough
--> src/main.rs:28:29
|
28 | Ok(&self.client.unwrap())
| ^^^^^^^^^^^^^^^^^^^^ does not live long enough
29 | },
| - temporary value only lives until here
|
note: borrowed value must be valid for the anonymous lifetime #1 defined on the body at 19:66...
--> src/main.rs:19:67
|
19 | fn get_client(&mut self) -> Result<&Client, &'static str> {
| ^
error[E0507]: cannot move out of borrowed content
--> src/main.rs:28:29
|
28 | Ok(&self.client.unwrap())
| ^^^^ cannot move out of borrowed content
Upvotes: 3
Views: 1427
Reputation: 893
So, there is solution, to which I came
struct Db {
connection: Option<Connection>,
host: String,
port: u16,
db: u8,
}
impl Db {
fn new(host: String, port: u16, db: u8) -> Db {
Db {
host: host,
port: port,
db: db,
connection: None,
}
}
fn get_connection(&mut self) -> RedisResult<&Connection> {
if let Some(ref connection) = self.connection {
Ok(connection)
}
else {
let connection_string = format!("redis://{0}:{1}/{2}", self.host, self.port, self.db);
self.connection = Some(
Client::open(connection_string.as_ref())?.get_connection()?);
Ok(self.connection.as_ref().unwrap())
}
}
fn keys(&mut self) -> RedisResult<Vec<String>> {
let key_iter: redis::Iter<String> = self.get_connection()?.scan()?;
Ok(key_iter.collect())
}
}
Upvotes: 2
Reputation: 1803
If you call unwrap()
you move T
out of the Option
. Since you only borrowed self
this leads to the cannot move out of borrowed content
error.
If you want to borrow the value inside an Option<T>
you can use the as_ref
method:
extern crate redis;
use redis::Client;
struct Db {
client: Option<Client>,
}
impl Db {
fn new() -> Db {
Db { client: None }
}
fn get_client(&mut self) -> Result<&Client, &'static str> {
if let Some(ref client) = self.client {
Ok(client)
} else {
let connection_string = "redis://127.0.0.1";
match Client::open(connection_string) {
Ok(client) => {
self.client = Some(client);
Ok(self.client.as_ref().unwrap())
}
Err(_) => Err("Error!"),
}
}
}
}
fn main() {
let mut db = Db::new();
db.get_client().expect("get_client failed");
}
Upvotes: 3