Skip to content

Commit

Permalink
session: add methods for creating compatible session with different API
Browse files Browse the repository at this point in the history
In order to make migration from the old API easier and allow doing it
gradually, some components of the client programs would probably like to
use the old API while the new components will use the new API. However,
in the current design, Session and LegacySession are two different types
and it's not possible to "cast" one to another - even though they have
nearly the same fields and implementations.

The previous commit made Cluster cloneable, based on the observation
that it's perfectly fine to clone Cluster's fields, construct a new one
and treat it as a shared facade, handle to the same "semantic" cluster.
The same applies to Session, actually - cloning a session would have
similar effect (though we encourage users to keep Session in an Arc so
that cloning is cheaper).

Instead of making GenericSession cloneable, we introduce methods which,
in reality, perform a clone but change the kind of session's API. This
allows to have two session objects which share the same resources but
have different APIs. This should be very useful when migrating large
projects to the new API - components that need to use the new API can
just "convert" the session to the new interface and use that.
  • Loading branch information
piodul committed Mar 23, 2023
1 parent 34ba917 commit b3d824f
Show file tree
Hide file tree
Showing 2 changed files with 99 additions and 2 deletions.
51 changes: 49 additions & 2 deletions scylla/src/transport/session.rs
Original file line number Diff line number Diff line change
Expand Up @@ -145,7 +145,7 @@ where
metrics: Arc<Metrics>,
auto_await_schema_agreement_timeout: Option<Duration>,
refresh_metadata_on_auto_schema_agreement: bool,
keyspace_name: ArcSwapOption<String>,
keyspace_name: Arc<ArcSwapOption<String>>,
_phantom_deser_api: PhantomData<DeserializationApi>,
}

Expand Down Expand Up @@ -672,6 +672,28 @@ impl GenericSession<CurrentDeserializationApi> {
) -> Result<QueryResult, QueryError> {
self.do_batch(batch, values).await
}

/// Creates a new Session instance that shared resources with
/// the current Session but supports the legacy API.
///
/// This method is provided in order to make migration to the new
/// deserialization API easier. For example, if your program in general uses
/// the new API but you still have some modules left that use the old one,
/// you can use this method to create an instance that supports the old API
/// and pass it to the module that you intend to migrate later.
pub fn make_shared_session_with_legacy_api(&self) -> LegacySession {
LegacySession {
cluster: self.cluster.clone(),
auto_await_schema_agreement_timeout: self.auto_await_schema_agreement_timeout,
default_execution_profile_handle: self.default_execution_profile_handle.clone(),
metrics: self.metrics.clone(),
refresh_metadata_on_auto_schema_agreement: self
.refresh_metadata_on_auto_schema_agreement,
schema_agreement_interval: self.schema_agreement_interval,
keyspace_name: self.keyspace_name.clone(),
_phantom_deser_api: PhantomData,
}
}
}

impl GenericSession<LegacyDeserializationApi> {
Expand Down Expand Up @@ -748,6 +770,31 @@ impl GenericSession<LegacyDeserializationApi> {
) -> Result<LegacyQueryResult, QueryError> {
Ok(self.do_batch(batch, values).await?.into_legacy_result()?)
}

/// Creates a new Session instance that shares resources with
/// the current Session but supports the new API.
///
/// This method is provided in order to make migration to the new
/// deserialization API easier. For example, if your program in general uses
/// the old API but you want to migrate some modules to the new one, you
/// can use this method to create an instance that supports the new API
/// and pass it to the module that you intend to migrate.
///
/// The new session object will use the same connections and cluster
/// metadata.
pub fn make_shared_session_with_new_api(&self) -> Session {
Session {
cluster: self.cluster.clone(),
auto_await_schema_agreement_timeout: self.auto_await_schema_agreement_timeout,
default_execution_profile_handle: self.default_execution_profile_handle.clone(),
metrics: self.metrics.clone(),
refresh_metadata_on_auto_schema_agreement: self
.refresh_metadata_on_auto_schema_agreement,
schema_agreement_interval: self.schema_agreement_interval,
keyspace_name: self.keyspace_name.clone(),
_phantom_deser_api: PhantomData,
}
}
}

/// Represents a CQL session, which can be used to communicate
Expand Down Expand Up @@ -875,7 +922,7 @@ where
auto_await_schema_agreement_timeout: config.auto_await_schema_agreement_timeout,
refresh_metadata_on_auto_schema_agreement: config
.refresh_metadata_on_auto_schema_agreement,
keyspace_name: ArcSwapOption::default(), // will be set by use_keyspace
keyspace_name: Arc::new(ArcSwapOption::default()), // will be set by use_keyspace
_phantom_deser_api: PhantomData,
};

Expand Down
50 changes: 50 additions & 0 deletions scylla/src/transport/session_test.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2721,3 +2721,53 @@ async fn test_get_keyspace_name() {
.unwrap();
assert_eq!(*session.get_keyspace().unwrap(), ks);
}

#[tokio::test]
async fn test_api_migration_session_sharing() {
let uri = std::env::var("SCYLLA_URI").unwrap_or_else(|_| "127.0.0.1:9042".to_string());
{
let session = SessionBuilder::new()
.known_node(&uri)
.build()
.await
.unwrap();
let session_shared = session.make_shared_session_with_legacy_api();

// If we are unlucky then we will race with metadata fetch/cluster update
// and both invocations will return different cluster data. This should be
// SUPER rare, but in order to reduce the chance of flakiness to a minimum
// we will try it three times in a row. Cluster data is updated once per
// minute, so this should be good enough.
let mut matched = false;
for _ in 0..3 {
let cd1 = session.get_cluster_data();
let cd2 = session_shared.get_cluster_data();

if Arc::ptr_eq(&cd1, &cd2) {
matched = true;
break;
}
}
assert!(matched);
}
{
let session = SessionBuilder::new()
.known_node(&uri)
.build_legacy()
.await
.unwrap();
let session_shared = session.make_shared_session_with_new_api();

let mut matched = false;
for _ in 0..3 {
let cd1 = session.get_cluster_data();
let cd2 = session_shared.get_cluster_data();

if Arc::ptr_eq(&cd1, &cd2) {
matched = true;
break;
}
}
assert!(matched);
}
}

0 comments on commit b3d824f

Please sign in to comment.