use crate::{bson::rawdoc, Collection};
use crate::{
bson::Document,
bson_compat::{cstr, CStr},
bson_util::to_raw_bson_array_ser,
cmap::{Command, RawCommandResponse, StreamDescription},
error::{ErrorKind, Result},
index::IndexModel,
operation::{append_options_to_raw_document, OperationWithDefaults},
options::{CreateIndexOptions, WriteConcern},
results::CreateIndexesResult,
};
use super::{ExecutionContext, WriteConcernOnlyBody};
#[derive(Debug)]
pub(crate) struct CreateIndexes {
target: Collection<Document>,
indexes: Vec<IndexModel>,
options: Option<CreateIndexOptions>,
}
impl CreateIndexes {
pub(crate) fn new(
target: Collection<Document>,
indexes: Vec<IndexModel>,
options: Option<CreateIndexOptions>,
) -> Self {
Self {
target,
indexes,
options,
}
}
}
impl OperationWithDefaults for CreateIndexes {
type O = CreateIndexesResult;
const NAME: &'static CStr = cstr!("createIndexes");
fn build(&mut self, description: &StreamDescription) -> Result<Command> {
if description.max_wire_version.unwrap_or(0) < 9
&& self
.options
.as_ref()
.is_some_and(|options| options.commit_quorum.is_some())
{
return Err(ErrorKind::InvalidArgument {
message: "Specifying a commit quorum to create_index(es) is not supported on \
server versions < 4.4"
.to_string(),
}
.into());
}
self.indexes.iter_mut().for_each(|i| i.update_name()); let indexes = to_raw_bson_array_ser(&self.indexes)?;
let mut body = rawdoc! {
Self::NAME: self.target.name(),
"indexes": indexes,
};
append_options_to_raw_document(&mut body, self.options.as_ref())?;
Ok(Command::from_operation(self, body))
}
fn handle_response<'a>(
&'a self,
response: &'a RawCommandResponse,
_context: ExecutionContext<'a>,
) -> Result<Self::O> {
let response: WriteConcernOnlyBody = response.body()?;
response.validate()?;
let index_names = self.indexes.iter().filter_map(|i| i.get_name()).collect();
Ok(CreateIndexesResult { index_names })
}
fn is_backpressure_retryable(&self, options: &crate::options::ClientOptions) -> bool {
options.retry_writes != Some(false)
}
fn write_concern(&self) -> super::Feature<&WriteConcern> {
self.options
.as_ref()
.and_then(|o| o.write_concern.as_ref())
.into()
}
fn target(&self) -> super::OperationTarget {
(&self.target).into()
}
#[cfg(feature = "opentelemetry")]
type Otel = crate::otel::Witness<Self>;
}
#[cfg(feature = "opentelemetry")]
impl crate::otel::OtelInfoDefaults for CreateIndexes {}