You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Original title: std::marker::Send is not implemented for Rc<RefCell<actix_session::session::SessionInner>>
Expected Behavior
I am trying to use session object with Redis as the storage in a distributed system with async-graphql to set and delete session for userid but having issues with that because actix-session does not implement Send and cannot be moved across threads. It has type: Rc<RefCell<actix_session::SessionInner>>. async-graphql requires that it has a Send trait.
Current Behavior
Full compiler error message:
`Rc<RefCell<actix_session::session::SessionInner>>` cannot be sent between threads safely
within `actix_session::Session`, the trait `std::marker::Send` is not implementedfor `Rc<RefCell<actix_session::session::SessionInner>>`rustc[E0277](https://doc.rust-lang.org/error-index.html#E0277)
graphql.rs(74,23): required by a bound introduced by this call
lib.rs(1,1): required because it appears within the type `actix_session::Session`
request.rs(91,26): required by a bound in `Request::data`
`Rc<RefCell<actix_session::session::SessionInner>>` cannot be shared between threads safely
within `actix_session::Session`, the trait `Sync` is not implemented for `Rc<RefCell<actix_session::session::SessionInner>>`rustc[E0277](https://doc.rust-lang.org/error-index.html#E0277)
graphql.rs(74,23): required by a bound introduced by this call
lib.rs(1,1): required because it appears within the type `actix_session::Session`
Possible Solution
Make Session type an Arc/Mutex rather than a RefCell
Present workaround
use std::ops::Deref;use actix_session::Session;use send_wrapper::SendWrapper;use uuid::Uuid;use wither::bson::oid::ObjectId;#[derive(Clone,Debug)]structShared<T>(pubOption<SendWrapper<T>>);impl<T>Shared<T>{pubfnnew(v:T) -> Self{Self(Some(SendWrapper::new(v)))}}impl<T>DerefforShared<T>{typeTarget = T;fnderef(&self) -> &Self::Target{&*self.0.as_deref().unwrap()}}typeSessionShared = Shared<actix_session::Session>;#[derive(Debug, thiserror::Error)]pubenumTypedSessionError{#[error("Failed to parse data")]ParsingFailure(#[from] serde_json::Error),#[error(transparent)]Unknown(#[from] anyhow::Error),// source and Display delegate to anyhow::Error}typeTypedSessionResult<T> = Result<T,TypedSessionError>;/*This is somewhat like a hack: https://github.com/async-graphql/async-graphql/issues/426Session is Session(Rc<RefCell<SessionInner>>) and probably okay to useSendWrapper for now but having it implement Send would allowusing Arc/Mutex*/pubstructTypedSession(SessionShared);implTypedSession{constUSER_ID_KEY:&'static str = "user_id";pubfnnew(session:Session) -> Self{Self(Shared::new(session))}pubfnrenew(&self){self.0.renew();}pubfninsert_user_uuid(&self,user_id:Uuid) -> TypedSessionResult<()>{self.0.insert(Self::USER_ID_KEY, user_id)?;Ok(())}pubfnget_user_uuid(&self) -> TypedSessionResult<Option<Uuid>>{Ok(self.0.get::<Uuid>(Self::USER_ID_KEY)?)}pubfnclear(&self){self.0.clear()}}
## Steps to Reproduce(for bugs)Try with async-graphql `GraphqlRequest` as shown below
```rs
#[post("/graphql")]
pub asyncfn index(
schema: web::Data<MyGraphQLSchema>,
req:HttpRequest,
gql_request:GraphQLRequest,) -> GraphQLResponse{
let mut request = gql_request.into_inner();let session = req.get_session();
request = request.data(session);// ---> This will fail because it expects it to have a Send Marker
schema.execute(request).await.into()}
The text was updated successfully, but these errors were encountered:
robjtede
changed the title
std::marker::Send is not implemented for Rc<RefCell<actix_session::session::SessionInner>>std::marker::Send is not implemented for Rc<RefCell<actix_session::session::SessionInner>>May 30, 2022
robjtede
changed the title
std::marker::Send is not implemented for Rc<RefCell<actix_session::session::SessionInner>>
option for Send-able SessionJul 19, 2022
The Rc<RefCell<T>> wrapper is an intentional choice to take advantage of Actix Web's ability to support !Send handler but I understand other parts of the ecosystem have trouble with !Send types such as this.
What I'm thinking for the solution is an opt-in crate feature that compiles Session as Arc<RwLock<actix_session::SessionInner>>:
actix-session = { version = "...", features = ["send"] }
I'd be somewhat wary of making that kind of change via a cargo feature. What happens if you end up with two different subsets of your dependency tree needing the opposite behaviour?
Original title:
std::marker::Send
is not implemented forRc<RefCell<actix_session::session::SessionInner>>
Expected Behavior
I am trying to use session object with
Redis
as the storage in a distributed system withasync-graphql
to set and delete session foruserid
but having issues with that becauseactix-session
does not implementSend
and cannot be moved across threads. It has type:Rc<RefCell<actix_session::SessionInner>>
.async-graphql
requires that it has aSend
trait.Current Behavior
Full compiler error message:
Possible Solution
Make Session type an
Arc/Mutex
rather than aRefCell
Present workaround
Your Environment
Mac OS monterey macbook m1 max 14 inch 2021
rustc -V
):Rust 1.61
actix-session: 0.6.2
Sidenote:
Issues are also mentioned here:
The text was updated successfully, but these errors were encountered: