Skip to content

Commit

Permalink
connection: quietly prepare unprepared queries in batch before sending
Browse files Browse the repository at this point in the history
Before sending an unprepared statement in batch, quietly prepare it on one connection to obtain the information about the
bind markers. If the list of values to be bound is empty, we
just send it as an unprepared query.
  • Loading branch information
sylwiaszunejko committed Sep 18, 2023
1 parent e9af9a1 commit c4bf01a
Showing 1 changed file with 42 additions and 3 deletions.
45 changes: 42 additions & 3 deletions scylla/src/transport/connection.rs
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ pub(crate) use ssl_config::SslConfig;

use crate::authentication::AuthenticatorProvider;
use scylla_cql::frame::response::authenticate::Authenticate;
use std::collections::{BTreeSet, HashMap};
use std::collections::{BTreeSet, HashMap, HashSet};
use std::convert::TryFrom;
use std::io::ErrorKind;
use std::net::{IpAddr, SocketAddr};
Expand All @@ -52,7 +52,7 @@ use crate::frame::{
request::{self, batch, execute, query, register, SerializableRequest},
response::{event::Event, result, NonErrorResponse, Response, ResponseOpcode},
server_event_type::EventType,
value::{BatchValues, ValueList},
value::{BatchValues, BatchValuesIterator, ValueList},
FrameParams, SerializedRequest,
};
use crate::query::Query;
Expand Down Expand Up @@ -772,11 +772,50 @@ impl Connection {

pub(crate) async fn batch_with_consistency(
&self,
batch: &Batch,
init_batch: &Batch,
values: impl BatchValues,
consistency: Consistency,
serial_consistency: Option<SerialConsistency>,
) -> Result<QueryResult, QueryError> {
let batch = {
let mut batch: Batch = Default::default();
batch.config = init_batch.config.clone();

let mut to_prepare = HashSet::<usize>::new();

{
let mut value_iter = values.batch_values_iter();
for (id, stmt) in init_batch.statements.iter().enumerate() {
let value = value_iter.next_serialized().transpose()?;
if let BatchStatement::Query(_) = stmt {
if let Some(v) = value {
if v.len() > 0 {
to_prepare.insert(id);
}
}
}
}
}

for (id, stmt) in init_batch.statements.iter().enumerate() {
match stmt {
BatchStatement::Query(query) => {
if to_prepare.contains(&id) {
let prepared = self.prepare(query).await?;
batch.append_statement(prepared.clone());
} else {
batch.append_statement(query.clone());
}
}
BatchStatement::PreparedStatement(prepared) => {
batch.append_statement(prepared.clone());
}
}
}

batch
};

let batch_frame = batch::Batch {
statements: Cow::Borrowed(&batch.statements),
values,
Expand Down

0 comments on commit c4bf01a

Please sign in to comment.