1
0
mirror of https://github.com/actix/actix-extras.git synced 2025-01-23 23:34:35 +01:00

341 lines
9.9 KiB
Rust
Raw Normal View History

2019-03-05 10:12:49 -08:00
//! User sessions.
//!
2019-03-05 18:47:18 -08:00
//! Actix provides a general solution for session management. Session
//! middlewares could provide different implementations which could
//! be accessed via general session api.
2019-03-05 10:12:49 -08:00
//!
//! By default, only cookie session backend is implemented. Other
//! backend implementations can be added.
//!
2019-03-05 18:47:18 -08:00
//! In general, you insert a *session* middleware and initialize it
//! , such as a `CookieSessionBackend`. To access session data,
//! [*Session*](struct.Session.html) extractor must be used. Session
//! extractor allows us to get or set session data.
2019-03-05 10:12:49 -08:00
//!
2019-12-08 12:31:16 +06:00
//! ```rust,no_run
//! use actix_web::{web, App, HttpServer, HttpResponse, Error};
2019-03-05 18:47:18 -08:00
//! use actix_session::{Session, CookieSession};
2019-03-05 10:12:49 -08:00
//!
2019-03-05 18:47:18 -08:00
//! fn index(session: Session) -> Result<&'static str, Error> {
2019-03-05 10:12:49 -08:00
//! // access session data
2019-03-05 18:47:18 -08:00
//! if let Some(count) = session.get::<i32>("counter")? {
2019-03-05 10:12:49 -08:00
//! println!("SESSION value: {}", count);
2019-03-05 18:47:18 -08:00
//! session.set("counter", count+1)?;
2019-03-05 10:12:49 -08:00
//! } else {
2019-03-05 18:47:18 -08:00
//! session.set("counter", 1)?;
2019-03-05 10:12:49 -08:00
//! }
//!
//! Ok("Welcome!")
//! }
//!
2019-12-08 12:31:16 +06:00
//! #[actix_rt::main]
//! async fn main() -> std::io::Result<()> {
2019-03-05 18:47:18 -08:00
//! HttpServer::new(
2019-03-25 13:02:10 -07:00
//! || App::new().wrap(
2019-03-05 18:47:18 -08:00
//! CookieSession::signed(&[0; 32]) // <- create cookie based session middleware
2019-03-05 10:12:49 -08:00
//! .secure(false)
2019-03-05 18:47:18 -08:00
//! )
//! .service(web::resource("/").to(|| HttpResponse::Ok())))
2019-03-05 18:47:18 -08:00
//! .bind("127.0.0.1:59880")?
//! .run()
2019-12-08 12:31:16 +06:00
//! .await
2019-03-05 10:12:49 -08:00
//! }
//! ```
use std::cell::RefCell;
2019-12-04 18:32:18 +06:00
use std::collections::HashMap;
2019-03-05 10:12:49 -08:00
use std::rc::Rc;
use actix_web::dev::{
Extensions, Payload, RequestHead, ServiceRequest, ServiceResponse,
};
2019-04-07 14:43:07 -07:00
use actix_web::{Error, FromRequest, HttpMessage, HttpRequest};
2020-03-15 06:58:47 +09:00
use futures_util::future::{ok, Ready};
2019-03-05 10:12:49 -08:00
use serde::de::DeserializeOwned;
use serde::Serialize;
#[cfg(feature = "cookie-session")]
2019-03-05 18:47:18 -08:00
mod cookie;
#[cfg(feature = "cookie-session")]
2019-03-05 18:47:18 -08:00
pub use crate::cookie::CookieSession;
2019-03-05 10:12:49 -08:00
/// The high-level interface you use to modify session data.
///
/// Session object could be obtained with
/// [`UserSession::get_session`](trait.UserSession.html#tymethod.get_session)
/// method. The `UserSession` trait is implemented for `HttpRequest`, `ServiceRequest`, and
/// `RequestHead`.
2019-03-05 10:12:49 -08:00
///
/// ```rust
2019-03-05 18:47:18 -08:00
/// use actix_session::Session;
2019-03-05 10:12:49 -08:00
/// use actix_web::*;
///
2019-03-05 18:47:18 -08:00
/// fn index(session: Session) -> Result<&'static str> {
2019-03-05 10:12:49 -08:00
/// // access session data
2019-03-05 18:47:18 -08:00
/// if let Some(count) = session.get::<i32>("counter")? {
/// session.set("counter", count + 1)?;
2019-03-05 10:12:49 -08:00
/// } else {
2019-03-05 18:47:18 -08:00
/// session.set("counter", 1)?;
2019-03-05 10:12:49 -08:00
/// }
///
/// Ok("Welcome!")
/// }
/// # fn main() {}
/// ```
2019-03-05 18:47:18 -08:00
pub struct Session(Rc<RefCell<SessionInner>>);
2019-03-05 10:12:49 -08:00
/// Helper trait that allows to get session
pub trait UserSession {
fn get_session(&self) -> Session;
}
impl UserSession for HttpRequest {
fn get_session(&self) -> Session {
Session::get_session(&mut *self.extensions_mut())
}
}
impl UserSession for ServiceRequest {
fn get_session(&self) -> Session {
Session::get_session(&mut *self.extensions_mut())
}
}
impl UserSession for RequestHead {
fn get_session(&self) -> Session {
Session::get_session(&mut *self.extensions_mut())
}
}
#[derive(PartialEq, Clone, Debug)]
pub enum SessionStatus {
Changed,
Purged,
Renewed,
2019-07-08 23:25:51 +06:00
Unchanged,
}
impl Default for SessionStatus {
fn default() -> SessionStatus {
SessionStatus::Unchanged
}
}
2019-03-05 18:47:18 -08:00
#[derive(Default)]
struct SessionInner {
state: HashMap<String, String>,
pub status: SessionStatus,
2019-03-05 10:12:49 -08:00
}
impl Session {
/// Get a `value` from the session.
2019-03-05 18:47:18 -08:00
pub fn get<T: DeserializeOwned>(&self, key: &str) -> Result<Option<T>, Error> {
if let Some(s) = self.0.borrow().state.get(key) {
Ok(Some(serde_json::from_str(s)?))
} else {
Ok(None)
2019-03-05 10:12:49 -08:00
}
}
/// Set a `value` from the session.
2019-03-05 18:47:18 -08:00
pub fn set<T: Serialize>(&self, key: &str, value: T) -> Result<(), Error> {
let mut inner = self.0.borrow_mut();
if inner.status != SessionStatus::Purged {
inner.status = SessionStatus::Changed;
inner
.state
.insert(key.to_owned(), serde_json::to_string(&value)?);
}
2019-03-05 18:47:18 -08:00
Ok(())
2019-03-05 10:12:49 -08:00
}
/// Remove value from the session.
pub fn remove(&self, key: &str) {
2019-03-05 18:47:18 -08:00
let mut inner = self.0.borrow_mut();
if inner.status != SessionStatus::Purged {
inner.status = SessionStatus::Changed;
inner.state.remove(key);
}
2019-03-05 10:12:49 -08:00
}
/// Clear the session.
pub fn clear(&self) {
2019-03-05 18:47:18 -08:00
let mut inner = self.0.borrow_mut();
if inner.status != SessionStatus::Purged {
inner.status = SessionStatus::Changed;
inner.state.clear()
}
2019-03-05 18:47:18 -08:00
}
/// Removes session, both client and server side.
pub fn purge(&self) {
let mut inner = self.0.borrow_mut();
inner.status = SessionStatus::Purged;
inner.state.clear();
}
/// Renews the session key, assigning existing session state to new key.
pub fn renew(&self) {
let mut inner = self.0.borrow_mut();
if inner.status != SessionStatus::Purged {
inner.status = SessionStatus::Renewed;
}
}
/// Adds the given key-value pairs to the session on the request.
///
/// Values that match keys already existing on the session will be overwritten. Values should
/// already be JSON serialized.
///
/// # Example
///
/// ```
/// # use actix_session::Session;
/// # use actix_web::test;
/// #
/// let mut req = test::TestRequest::default().to_srv_request();
///
/// Session::set_session(
/// vec![("counter".to_string(), serde_json::to_string(&0).unwrap())].into_iter(),
/// &mut req,
/// );
/// ```
pub fn set_session(
2019-03-05 18:47:18 -08:00
data: impl Iterator<Item = (String, String)>,
req: &mut ServiceRequest,
2019-03-05 18:47:18 -08:00
) {
2019-04-07 14:43:07 -07:00
let session = Session::get_session(&mut *req.extensions_mut());
2019-03-05 18:47:18 -08:00
let mut inner = session.0.borrow_mut();
inner.state.extend(data);
}
pub fn get_changes<B>(
res: &mut ServiceResponse<B>,
2019-07-08 23:25:51 +06:00
) -> (
SessionStatus,
Option<impl Iterator<Item = (String, String)>>,
) {
2019-03-05 18:47:18 -08:00
if let Some(s_impl) = res
.request()
.extensions()
.get::<Rc<RefCell<SessionInner>>>()
{
let state =
std::mem::replace(&mut s_impl.borrow_mut().state, HashMap::new());
(s_impl.borrow().status.clone(), Some(state.into_iter()))
2019-03-05 18:47:18 -08:00
} else {
(SessionStatus::Unchanged, None)
2019-03-05 10:12:49 -08:00
}
}
2019-03-05 18:47:18 -08:00
fn get_session(extensions: &mut Extensions) -> Session {
2019-04-07 14:43:07 -07:00
if let Some(s_impl) = extensions.get::<Rc<RefCell<SessionInner>>>() {
2019-03-05 18:47:18 -08:00
return Session(Rc::clone(&s_impl));
}
let inner = Rc::new(RefCell::new(SessionInner::default()));
2019-04-07 14:43:07 -07:00
extensions.insert(inner.clone());
2019-03-05 18:47:18 -08:00
Session(inner)
}
2019-03-05 10:12:49 -08:00
}
/// Extractor implementation for Session type.
///
/// ```rust
/// # use actix_web::*;
2019-03-05 18:47:18 -08:00
/// use actix_session::Session;
2019-03-05 10:12:49 -08:00
///
/// fn index(session: Session) -> Result<&'static str> {
/// // access session data
/// if let Some(count) = session.get::<i32>("counter")? {
/// session.set("counter", count + 1)?;
/// } else {
/// session.set("counter", 1)?;
/// }
///
/// Ok("Welcome!")
/// }
/// # fn main() {}
/// ```
impl FromRequest for Session {
2019-03-05 18:47:18 -08:00
type Error = Error;
2019-11-21 13:08:22 +06:00
type Future = Ready<Result<Session, Error>>;
2019-04-13 16:35:25 -07:00
type Config = ();
2019-03-05 10:12:49 -08:00
#[inline]
fn from_request(req: &HttpRequest, _: &mut Payload) -> Self::Future {
2019-11-21 13:08:22 +06:00
ok(Session::get_session(&mut *req.extensions_mut()))
2019-03-05 10:12:49 -08:00
}
}
2019-03-24 20:21:20 -07:00
#[cfg(test)]
mod tests {
use actix_web::{test, HttpResponse};
use super::*;
#[test]
fn session() {
2019-04-02 15:00:10 -07:00
let mut req = test::TestRequest::default().to_srv_request();
2019-03-24 20:21:20 -07:00
Session::set_session(
vec![("key".to_string(), serde_json::to_string("value").unwrap())].into_iter(),
2019-03-24 20:21:20 -07:00
&mut req,
);
2019-04-07 14:43:07 -07:00
let session = Session::get_session(&mut *req.extensions_mut());
2019-03-24 20:21:20 -07:00
let res = session.get::<String>("key").unwrap();
assert_eq!(res, Some("value".to_string()));
session.set("key2", "value2".to_string()).unwrap();
session.remove("key");
let mut res = req.into_response(HttpResponse::Ok().finish());
let (_status, state) = Session::get_changes(&mut res);
let changes: Vec<_> = state.unwrap().collect();
2019-03-24 20:21:20 -07:00
assert_eq!(changes, [("key2".to_string(), "\"value2\"".to_string())]);
}
#[test]
fn get_session() {
let mut req = test::TestRequest::default().to_srv_request();
Session::set_session(
vec![("key".to_string(), serde_json::to_string(&true).unwrap())].into_iter(),
&mut req,
);
let session = req.get_session();
let res = session.get("key").unwrap();
assert_eq!(res, Some(true));
}
#[test]
fn get_session_from_request_head() {
let mut req = test::TestRequest::default().to_srv_request();
Session::set_session(
vec![("key".to_string(), serde_json::to_string(&10).unwrap())].into_iter(),
&mut req,
);
let session = req.head_mut().get_session();
let res = session.get::<u32>("key").unwrap();
assert_eq!(res, Some(10));
}
#[test]
fn purge_session() {
2019-07-17 15:48:37 +06:00
let req = test::TestRequest::default().to_srv_request();
let session = Session::get_session(&mut *req.extensions_mut());
assert_eq!(session.0.borrow().status, SessionStatus::Unchanged);
session.purge();
assert_eq!(session.0.borrow().status, SessionStatus::Purged);
}
#[test]
fn renew_session() {
2019-07-17 15:48:37 +06:00
let req = test::TestRequest::default().to_srv_request();
let session = Session::get_session(&mut *req.extensions_mut());
assert_eq!(session.0.borrow().status, SessionStatus::Unchanged);
session.renew();
assert_eq!(session.0.borrow().status, SessionStatus::Renewed);
}
2019-03-24 20:21:20 -07:00
}