mirror of
https://github.com/actix/actix-extras.git
synced 2024-11-30 18:34:36 +01:00
complete redis session backend
This commit is contained in:
parent
fab3c35ba9
commit
8924335338
20
Cargo.toml
20
Cargo.toml
@ -2,12 +2,23 @@
|
|||||||
name = "actix-redis"
|
name = "actix-redis"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
|
authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
|
||||||
|
description = "Redis integration for actix web framework"
|
||||||
|
license = "MIT/Apache-2.0"
|
||||||
|
readme = "README.md"
|
||||||
|
keywords = ["http", "web", "framework", "async", "actix"]
|
||||||
|
homepage = "https://github.com/actix/actix-redis"
|
||||||
|
repository = "https://github.com/actix/actix-redis.git"
|
||||||
|
documentation = "https://docs.rs/actix-redis/"
|
||||||
|
categories = ["network-programming", "asynchronous"]
|
||||||
|
exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"]
|
||||||
|
|
||||||
[lib]
|
[lib]
|
||||||
name = "actix_redis"
|
name = "actix_redis"
|
||||||
path = "src/lib.rs"
|
path = "src/lib.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
rand = "0.3"
|
||||||
|
http = "0.1"
|
||||||
bytes = "0.4"
|
bytes = "0.4"
|
||||||
failure = "^0.1.1"
|
failure = "^0.1.1"
|
||||||
futures = "0.1"
|
futures = "0.1"
|
||||||
@ -16,5 +27,10 @@ serde_json = "1.0"
|
|||||||
tokio-io = "0.1"
|
tokio-io = "0.1"
|
||||||
tokio-core = "0.1"
|
tokio-core = "0.1"
|
||||||
actix = "^0.3.5"
|
actix = "^0.3.5"
|
||||||
actix-web = { path = "../actix-web/" }
|
redis-async = "0.0"
|
||||||
redis-async = { git = "https://github.com/benashford/redis-async-rs.git" }
|
cookie = { version="0.10", features=["percent-encode", "secure"] }
|
||||||
|
|
||||||
|
actix-web = { git = "https://github.com/actix/actix-web.git" }
|
||||||
|
|
||||||
|
[dev-dependencies]
|
||||||
|
env_logger = "0.4"
|
||||||
|
43
README.md
Normal file
43
README.md
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
# Actix redis
|
||||||
|
|
||||||
|
## Redis session backend
|
||||||
|
|
||||||
|
Use redis as session storage.
|
||||||
|
|
||||||
|
You need to pass an address of the redis server and random value to the
|
||||||
|
constructor of `RedisSessionBackend`. This is private key for cookie session,
|
||||||
|
When this value is changed, all session data is lost.
|
||||||
|
|
||||||
|
Note that whatever you write into your session is visible by the user (but not modifiable).
|
||||||
|
|
||||||
|
Constructor panics if key length is less than 32 bytes.
|
||||||
|
|
||||||
|
```rust,ignore
|
||||||
|
# extern crate actix;
|
||||||
|
# extern crate actix_web;
|
||||||
|
# use actix_web::*;
|
||||||
|
use actix_web::middleware::SessionStorage;
|
||||||
|
use actix_redis::RedisSessionBackend;
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
::std::env::set_var("RUST_LOG", "actix_web=info");
|
||||||
|
let _ = env_logger::init();
|
||||||
|
let sys = actix::System::new("basic-example");
|
||||||
|
|
||||||
|
HttpServer::new(
|
||||||
|
|| Application::new()
|
||||||
|
// enable logger
|
||||||
|
.middleware(middleware::Logger::default())
|
||||||
|
// cookie session middleware
|
||||||
|
.middleware(SessionStorage::new(
|
||||||
|
RedisSessionBackend::new("127.0.0.1:6379", &[0; 32])
|
||||||
|
.expect("Can not connect to redis server")
|
||||||
|
))
|
||||||
|
// register simple route, handle all methods
|
||||||
|
.resource("/", |r| r.f(index)))
|
||||||
|
.bind("0.0.0.0:8080").unwrap()
|
||||||
|
.start();
|
||||||
|
|
||||||
|
let _ = sys.run();
|
||||||
|
}
|
||||||
|
```
|
@ -15,11 +15,6 @@ use actix_redis::RedisSessionBackend;
|
|||||||
/// simple handler
|
/// simple handler
|
||||||
fn index(mut req: HttpRequest) -> Result<HttpResponse> {
|
fn index(mut req: HttpRequest) -> Result<HttpResponse> {
|
||||||
println!("{:?}", req);
|
println!("{:?}", req);
|
||||||
if let Ok(ch) = req.payload_mut().readany().poll() {
|
|
||||||
if let futures::Async::Ready(Some(d)) = ch {
|
|
||||||
println!("{}", String::from_utf8_lossy(d.as_ref()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// session
|
// session
|
||||||
if let Some(count) = req.session().get::<i32>("counter")? {
|
if let Some(count) = req.session().get::<i32>("counter")? {
|
||||||
@ -43,14 +38,14 @@ fn main() {
|
|||||||
.middleware(middleware::Logger::default())
|
.middleware(middleware::Logger::default())
|
||||||
// cookie session middleware
|
// cookie session middleware
|
||||||
.middleware(middleware::SessionStorage::new(
|
.middleware(middleware::SessionStorage::new(
|
||||||
RedisSessionBackend::new("127.0.0.1:6379", Duration::from_secs(7200))
|
RedisSessionBackend::new("127.0.0.1:6379", &[0; 32])
|
||||||
.expect("Can not connect to redis server")
|
.expect("Can not connect to redis server")
|
||||||
))
|
))
|
||||||
// register simple route, handle all methods
|
// register simple route, handle all methods
|
||||||
.resource("/", |r| r.f(index)))
|
.resource("/", |r| r.f(index)))
|
||||||
.bind("0.0.0.0:8080").unwrap()
|
.bind("0.0.0.0:8080").unwrap()
|
||||||
|
.threads(1)
|
||||||
.start();
|
.start();
|
||||||
|
|
||||||
println!("Starting http server: 127.0.0.1:8080");
|
|
||||||
let _ = sys.run();
|
let _ = sys.run();
|
||||||
}
|
}
|
@ -1,9 +1,12 @@
|
|||||||
extern crate actix;
|
extern crate actix;
|
||||||
extern crate actix_web;
|
extern crate actix_web;
|
||||||
extern crate bytes;
|
extern crate bytes;
|
||||||
|
extern crate cookie;
|
||||||
extern crate futures;
|
extern crate futures;
|
||||||
extern crate serde;
|
extern crate serde;
|
||||||
extern crate serde_json;
|
extern crate serde_json;
|
||||||
|
extern crate rand;
|
||||||
|
extern crate http;
|
||||||
extern crate tokio_io;
|
extern crate tokio_io;
|
||||||
extern crate tokio_core;
|
extern crate tokio_core;
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
@ -14,4 +17,5 @@ extern crate failure;
|
|||||||
mod redis;
|
mod redis;
|
||||||
mod session;
|
mod session;
|
||||||
|
|
||||||
|
pub use redis::RedisActor;
|
||||||
pub use session::RedisSessionBackend;
|
pub use session::RedisSessionBackend;
|
||||||
|
@ -3,7 +3,6 @@ use std::collections::VecDeque;
|
|||||||
|
|
||||||
use bytes::BytesMut;
|
use bytes::BytesMut;
|
||||||
use futures::Future;
|
use futures::Future;
|
||||||
use futures::future::Either;
|
|
||||||
use futures::unsync::oneshot;
|
use futures::unsync::oneshot;
|
||||||
use tokio_core::net::TcpStream;
|
use tokio_core::net::TcpStream;
|
||||||
use tokio_io::codec::{Decoder, Encoder};
|
use tokio_io::codec::{Decoder, Encoder};
|
||||||
@ -108,7 +107,7 @@ impl Handler<Command> for RedisActor {
|
|||||||
fn handle(&mut self, msg: Command, ctx: &mut Self::Context) -> Response<Self, Command> {
|
fn handle(&mut self, msg: Command, ctx: &mut Self::Context) -> Response<Self, Command> {
|
||||||
let (tx, rx) = oneshot::channel();
|
let (tx, rx) = oneshot::channel();
|
||||||
self.queue.push_back(tx);
|
self.queue.push_back(tx);
|
||||||
ctx.send(Value(msg.0));
|
let _ = ctx.send(Value(msg.0));
|
||||||
|
|
||||||
Self::async_reply(
|
Self::async_reply(
|
||||||
rx.map_err(|_| io::Error::new(io::ErrorKind::Other, "").into())
|
rx.map_err(|_| io::Error::new(io::ErrorKind::Other, "").into())
|
||||||
|
133
src/session.rs
133
src/session.rs
@ -1,12 +1,16 @@
|
|||||||
use std::{io, net};
|
use std::{io, net};
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
use std::time::Duration;
|
use std::iter::FromIterator;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
|
|
||||||
use serde_json;
|
use serde_json;
|
||||||
|
use rand::{self, Rng};
|
||||||
use futures::Future;
|
use futures::Future;
|
||||||
use futures::future::{Either, ok as FutOk, err as FutErr};
|
use futures::future::{Either, ok as FutOk, err as FutErr};
|
||||||
use tokio_core::net::TcpStream;
|
use tokio_core::net::TcpStream;
|
||||||
|
use redis_async::resp::RespValue;
|
||||||
|
use cookie::{CookieJar, Cookie, Key};
|
||||||
|
use http::header::{self, HeaderValue};
|
||||||
use actix::prelude::*;
|
use actix::prelude::*;
|
||||||
use actix_web::{error, Error, HttpRequest, HttpResponse};
|
use actix_web::{error, Error, HttpRequest, HttpResponse};
|
||||||
use actix_web::middleware::{SessionImpl, SessionBackend, Response as MiddlewareResponse};
|
use actix_web::middleware::{SessionImpl, SessionBackend, Response as MiddlewareResponse};
|
||||||
@ -19,6 +23,7 @@ pub struct RedisSession {
|
|||||||
changed: bool,
|
changed: bool,
|
||||||
inner: Rc<Inner>,
|
inner: Rc<Inner>,
|
||||||
state: HashMap<String, String>,
|
state: HashMap<String, String>,
|
||||||
|
value: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SessionImpl for RedisSession {
|
impl SessionImpl for RedisSession {
|
||||||
@ -46,27 +51,44 @@ impl SessionImpl for RedisSession {
|
|||||||
self.state.clear()
|
self.state.clear()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write(&self, mut resp: HttpResponse) -> MiddlewareResponse {
|
fn write(&self, resp: HttpResponse) -> MiddlewareResponse {
|
||||||
if self.changed {
|
if self.changed {
|
||||||
let _ = self.inner.update(&self.state);
|
MiddlewareResponse::Future(self.inner.update(&self.state, resp, self.value.as_ref()))
|
||||||
}
|
} else {
|
||||||
MiddlewareResponse::Done(resp)
|
MiddlewareResponse::Done(resp)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Use redis as session storage.
|
||||||
|
///
|
||||||
|
/// You need to pass an address of the redis server and random value to the
|
||||||
|
/// constructor of `RedisSessionBackend`. This is private key for cookie session,
|
||||||
|
/// When this value is changed, all session data is lost.
|
||||||
|
///
|
||||||
|
/// Note that whatever you write into your session is visible by the user (but not modifiable).
|
||||||
|
///
|
||||||
|
/// Constructor panics if key length is less than 32 bytes.
|
||||||
pub struct RedisSessionBackend(Rc<Inner>);
|
pub struct RedisSessionBackend(Rc<Inner>);
|
||||||
|
|
||||||
impl RedisSessionBackend {
|
impl RedisSessionBackend {
|
||||||
/// Create new redis session backend
|
/// Create new redis session backend
|
||||||
pub fn new<S: net::ToSocketAddrs>(addr: S, ttl: Duration) -> io::Result<RedisSessionBackend> {
|
///
|
||||||
|
/// * `addr` - address of the redis server
|
||||||
|
pub fn new<S: net::ToSocketAddrs>(addr: S, key: &[u8]) -> io::Result<RedisSessionBackend> {
|
||||||
let h = Arbiter::handle();
|
let h = Arbiter::handle();
|
||||||
let mut err = None;
|
let mut err = None;
|
||||||
for addr in addr.to_socket_addrs()? {
|
for addr in addr.to_socket_addrs()? {
|
||||||
match TcpStream::connect(&addr, &h).wait() {
|
match net::TcpStream::connect(&addr) {
|
||||||
Err(e) => err = Some(e),
|
Err(e) => err = Some(e),
|
||||||
Ok(conn) => {
|
Ok(conn) => {
|
||||||
let addr = RedisActor::start(conn);
|
let addr = RedisActor::start(
|
||||||
return Ok(RedisSessionBackend(Rc::new(Inner{ttl: ttl, addr: addr})));
|
TcpStream::from_stream(conn, h).expect("Can not create tcp stream"));
|
||||||
|
return Ok(RedisSessionBackend(
|
||||||
|
Rc::new(Inner{key: Key::from_master(key),
|
||||||
|
ttl: "7200".to_owned(),
|
||||||
|
addr: addr,
|
||||||
|
name: "actix-session".to_owned()})));
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -76,6 +98,17 @@ impl RedisSessionBackend {
|
|||||||
Err(io::Error::new(io::ErrorKind::Other, "Can not connect to redis server."))
|
Err(io::Error::new(io::ErrorKind::Other, "Can not connect to redis server."))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Set time to live in seconds for session value
|
||||||
|
pub fn ttl(mut self, ttl: u16) -> Self {
|
||||||
|
Rc::get_mut(&mut self.0).unwrap().ttl = format!("{}", ttl);
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn cookie_name(mut self, name: &str) -> Self {
|
||||||
|
Rc::get_mut(&mut self.0).unwrap().name = name.to_owned();
|
||||||
|
self
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S> SessionBackend<S> for RedisSessionBackend {
|
impl<S> SessionBackend<S> for RedisSessionBackend {
|
||||||
@ -87,17 +120,19 @@ impl<S> SessionBackend<S> for RedisSessionBackend {
|
|||||||
let inner = Rc::clone(&self.0);
|
let inner = Rc::clone(&self.0);
|
||||||
|
|
||||||
Box::new(self.0.load(req).map(move |state| {
|
Box::new(self.0.load(req).map(move |state| {
|
||||||
if let Some(state) = state {
|
if let Some((state, value)) = state {
|
||||||
RedisSession {
|
RedisSession {
|
||||||
changed: false,
|
changed: false,
|
||||||
inner: inner,
|
inner: inner,
|
||||||
state: state,
|
state: state,
|
||||||
|
value: Some(value),
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
RedisSession {
|
RedisSession {
|
||||||
changed: false,
|
changed: false,
|
||||||
inner: inner,
|
inner: inner,
|
||||||
state: HashMap::new(),
|
state: HashMap::new(),
|
||||||
|
value: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}))
|
}))
|
||||||
@ -105,49 +140,99 @@ impl<S> SessionBackend<S> for RedisSessionBackend {
|
|||||||
}
|
}
|
||||||
|
|
||||||
struct Inner {
|
struct Inner {
|
||||||
ttl: Duration,
|
key: Key,
|
||||||
|
ttl: String,
|
||||||
|
name: String,
|
||||||
addr: Address<RedisActor>,
|
addr: Address<RedisActor>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Inner {
|
impl Inner {
|
||||||
fn load<S>(&self, req: &mut HttpRequest<S>)
|
fn load<S>(&self, req: &mut HttpRequest<S>)
|
||||||
-> Box<Future<Item=Option<HashMap<String, String>>, Error=Error>>
|
-> Box<Future<Item=Option<(HashMap<String, String>, String)>, Error=Error>> {
|
||||||
{
|
|
||||||
if let Ok(cookies) = req.cookies() {
|
if let Ok(cookies) = req.cookies() {
|
||||||
for cookie in cookies {
|
for cookie in cookies {
|
||||||
if cookie.name() == "actix-session" {
|
if cookie.name() == self.name {
|
||||||
|
let mut jar = CookieJar::new();
|
||||||
|
jar.add_original(cookie.clone());
|
||||||
|
if let Some(cookie) = jar.signed(&self.key).get(&self.name) {
|
||||||
|
let value = cookie.value().to_owned();
|
||||||
return Box::new(
|
return Box::new(
|
||||||
self.addr.call_fut(Command(resp_array!["GET", cookie.value()]))
|
self.addr.call_fut(Command(resp_array!["GET", cookie.value()]))
|
||||||
.map_err(Error::from)
|
.map_err(Error::from)
|
||||||
.and_then(|res| {
|
.and_then(move |res| {
|
||||||
match res {
|
match res {
|
||||||
Ok(val) => {
|
Ok(val) => {
|
||||||
println!("VAL {:?}", val);
|
match val {
|
||||||
Ok(Some(HashMap::new()))
|
RespValue::Error(err) =>
|
||||||
|
return Err(
|
||||||
|
error::ErrorInternalServerError(err).into()),
|
||||||
|
RespValue::SimpleString(s) =>
|
||||||
|
if let Ok(val) = serde_json::from_str(&s) {
|
||||||
|
return Ok(Some((val, value)))
|
||||||
},
|
},
|
||||||
Err(err) => Err(
|
RespValue::BulkString(s) => {
|
||||||
io::Error::new(io::ErrorKind::Other, "Error").into())
|
if let Ok(val) = serde_json::from_slice(&s) {
|
||||||
|
return Ok(Some((val, value)))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
_ => (),
|
||||||
|
}
|
||||||
|
Ok(None)
|
||||||
|
},
|
||||||
|
Err(err) => Err(error::ErrorInternalServerError(err).into())
|
||||||
}
|
}
|
||||||
}))
|
}))
|
||||||
|
} else {
|
||||||
|
return Box::new(FutOk(None))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Box::new(FutOk(None))
|
Box::new(FutOk(None))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn update(&self, state: &HashMap<String, String>) -> Box<Future<Item=(), Error=Error>> {
|
fn update(&self, state: &HashMap<String, String>,
|
||||||
|
mut resp: HttpResponse,
|
||||||
|
value: Option<&String>) -> Box<Future<Item=HttpResponse, Error=Error>>
|
||||||
|
{
|
||||||
|
let (value, jar) = if let Some(value) = value {
|
||||||
|
(value.clone(), None)
|
||||||
|
} else {
|
||||||
|
let mut rng = rand::OsRng::new().unwrap();
|
||||||
|
let value = String::from_iter(rng.gen_ascii_chars().take(32));
|
||||||
|
|
||||||
|
let mut cookie = Cookie::new(self.name.clone(), value.clone());
|
||||||
|
cookie.set_path("/");
|
||||||
|
cookie.set_http_only(true);
|
||||||
|
|
||||||
|
// set cookie
|
||||||
|
let mut jar = CookieJar::new();
|
||||||
|
jar.signed(&self.key).add(cookie);
|
||||||
|
|
||||||
|
(value, Some(jar))
|
||||||
|
};
|
||||||
|
|
||||||
Box::new(
|
Box::new(
|
||||||
match serde_json::to_string(state) {
|
match serde_json::to_string(state) {
|
||||||
Err(e) => Either::A(FutErr(e.into())),
|
Err(e) => Either::A(FutErr(e.into())),
|
||||||
Ok(body) => {
|
Ok(body) => {
|
||||||
Either::B(
|
Either::B(
|
||||||
self.addr.call_fut(Command(resp_array!["GET", "test"]))
|
self.addr.call_fut(
|
||||||
|
Command(resp_array!["SET", value, body,"EX", &self.ttl]))
|
||||||
.map_err(Error::from)
|
.map_err(Error::from)
|
||||||
.and_then(|res| {
|
.and_then(move |res| {
|
||||||
match res {
|
match res {
|
||||||
Ok(val) => Ok(()),
|
Ok(_) => {
|
||||||
Err(err) => Err(
|
if let Some(jar) = jar {
|
||||||
error::ErrorInternalServerError(err).into())
|
for cookie in jar.delta() {
|
||||||
|
let val = HeaderValue::from_str(
|
||||||
|
&cookie.to_string())?;
|
||||||
|
resp.headers_mut().append(header::SET_COOKIE, val);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(resp)
|
||||||
|
},
|
||||||
|
Err(err) => Err(error::ErrorInternalServerError(err).into())
|
||||||
}
|
}
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user