diff --git a/scylla-cql/src/frame/response/result.rs b/scylla-cql/src/frame/response/result.rs index 87bbaaeaa6..b335b11b9f 100644 --- a/scylla-cql/src/frame/response/result.rs +++ b/scylla-cql/src/frame/response/result.rs @@ -812,7 +812,11 @@ fn deser_rows( let server_metadata = deser_result_metadata(buf)?; let metadata = match cached_metadata { - Some(metadata) => metadata.clone(), + Some(cached) => ResultMetadata { + col_count: cached.col_count, + paging_state: server_metadata.paging_state, + col_specs: cached.col_specs.clone(), + }, None => { // No cached_metadata provided. Server is supposed to provide the result metadata. if server_metadata.col_count != server_metadata.col_specs.len() { diff --git a/scylla/tests/integration/skip_metadata_optimization.rs b/scylla/tests/integration/skip_metadata_optimization.rs index 66c431d9de..6ee9ea2319 100644 --- a/scylla/tests/integration/skip_metadata_optimization.rs +++ b/scylla/tests/integration/skip_metadata_optimization.rs @@ -14,6 +14,7 @@ use std::sync::Arc; #[cfg(not(scylla_cloud_tests))] async fn test_skip_result_metadata() { setup_tracing(); + use bytes::Bytes; use scylla_proxy::{ResponseOpcode, ResponseRule}; const NO_METADATA_FLAG: i32 = 0x0004; @@ -79,6 +80,70 @@ async fn test_skip_result_metadata() { prepared.set_use_cached_result_metadata(true); test_with_flags_predicate(&session, &prepared, &mut rx, |flags| flags & NO_METADATA_FLAG != 0).await; + // Verify that the optimisation does not break paging + { + let ks = unique_keyspace_name(); + + session.query(format!("CREATE KEYSPACE IF NOT EXISTS {} WITH REPLICATION = {{'class' : 'NetworkTopologyStrategy', 'replication_factor' : 1}}", ks), &[]).await.unwrap(); + session.use_keyspace(ks, true).await.unwrap(); + + type RowT = (i32, i32, String); + session + .query( + "CREATE TABLE IF NOT EXISTS t2 (a int, b int, c text, primary key (a, b))", + &[], + ) + .await + .unwrap(); + + let insert_stmt = session + .prepare("INSERT INTO t2 (a, b, c) VALUES (?, ?, ?)") + .await + .unwrap(); + + for idx in 0..10 { + session + .execute(&insert_stmt, (idx, idx + 1, "Some text")) + .await + .unwrap(); + } + + { + let select_query = "SELECT a, b, c FROM t2"; + + let rs = session + .query(select_query, ()) + .await + .unwrap() + .rows_typed::() + .unwrap() + .collect::, _>>() + .unwrap(); + + let mut results_from_manual_paging: Vec = vec![]; + let mut prepared_paged = session.prepare(select_query).await.unwrap(); + prepared_paged.set_use_cached_result_metadata(true); + prepared_paged.set_page_size(1); + let mut paging_state: Option = None; + let mut watchdog = 0; + loop { + let mut rs_manual = session + .execute_paged(&prepared_paged, &[], paging_state) + .await + .unwrap(); + eprintln!("Paging state: {:?}", rs_manual.paging_state); + paging_state = rs_manual.paging_state.take(); + results_from_manual_paging + .extend(rs_manual.rows_typed::().unwrap().map(Result::unwrap)); + if watchdog > 30 || paging_state.is_none() { + break; + } + watchdog += 1; + } + assert_eq!(results_from_manual_paging, rs); + } + } + running_proxy }).await;