#[cfg(feature = "sql")]
use crate::db::query::plan::expr::ProjectionSelection;
use crate::db::{
predicate::Predicate,
query::{
intent::{IntentError, KeyAccess, KeyAccessKind, KeyAccessState, state::QueryIntent},
plan::{
FieldSlot, GroupAggregateSpec, GroupHavingClause, GroupHavingSpec,
GroupedExecutionConfig, OrderDirection, OrderSpec,
},
},
};
impl<K> QueryIntent<K> {
pub(in crate::db::query::intent) fn append_predicate(&mut self, predicate: Predicate) {
let scalar = self.scalar_mut();
scalar.predicate = match scalar.predicate.take() {
Some(existing) => Some(Predicate::And(vec![existing, predicate])),
None => Some(predicate),
};
}
pub(in crate::db::query::intent) fn push_order_ascending(&mut self, field: &str) {
self.push_order_field(field, OrderDirection::Asc);
}
pub(in crate::db::query::intent) fn push_order_descending(&mut self, field: &str) {
self.push_order_field(field, OrderDirection::Desc);
}
pub(in crate::db::query::intent) fn set_order_spec(&mut self, order: OrderSpec) {
self.scalar_mut().order = Some(order);
}
pub(in crate::db::query::intent) const fn set_distinct(&mut self) {
self.scalar_mut().distinct = true;
}
#[cfg(feature = "sql")]
pub(in crate::db::query::intent) fn set_projection_selection(
&mut self,
projection_selection: ProjectionSelection,
) {
self.scalar_mut().projection_selection = projection_selection;
}
pub(in crate::db::query::intent) fn set_by_id(&mut self, id: K) {
self.set_key_access(KeyAccessKind::Single, KeyAccess::Single(id));
}
pub(in crate::db::query::intent) fn set_by_ids<I>(&mut self, ids: I)
where
I: IntoIterator<Item = K>,
{
self.set_key_access(
KeyAccessKind::Many,
KeyAccess::Many(ids.into_iter().collect()),
);
}
pub(in crate::db::query::intent) fn set_only(&mut self, id: K) {
self.set_key_access(KeyAccessKind::Only, KeyAccess::Single(id));
}
pub(in crate::db::query::intent) fn push_group_field_slot(&mut self, field_slot: FieldSlot) {
if matches!(self, Self::Delete(_)) {
self.mark_delete_grouping_requested();
return;
}
let group = &mut self.ensure_grouped_mut().group;
if !group
.group_fields
.iter()
.any(|existing| existing.index() == field_slot.index())
{
group.group_fields.push(field_slot);
}
}
pub(in crate::db::query::intent) fn push_group_aggregate(
&mut self,
aggregate: GroupAggregateSpec,
) {
if matches!(self, Self::Delete(_)) {
self.mark_delete_grouping_requested();
return;
}
self.ensure_grouped_mut().group.aggregates.push(aggregate);
}
pub(in crate::db::query::intent) fn set_grouped_limits(
&mut self,
max_groups: u64,
max_group_bytes: u64,
) {
if matches!(self, Self::Delete(_)) {
self.mark_delete_grouping_requested();
return;
}
self.ensure_grouped_mut().group.execution =
GroupedExecutionConfig::with_hard_limits(max_groups, max_group_bytes);
}
pub(in crate::db::query::intent) fn push_having_clause(
&mut self,
clause: GroupHavingClause,
) -> Result<(), IntentError> {
if matches!(self, Self::Delete(_)) {
if self.is_grouped() {
self.mark_delete_grouping_requested();
return Ok(());
}
return Err(IntentError::having_requires_group_by());
}
let Some(grouped) = self.grouped_mut() else {
return Err(IntentError::having_requires_group_by());
};
let having = grouped.having.get_or_insert(GroupHavingSpec {
clauses: Vec::new(),
});
having.clauses.push(clause);
Ok(())
}
fn set_key_access(&mut self, kind: KeyAccessKind, access: KeyAccess<K>) {
let scalar = self.scalar_mut();
if let Some(existing) = &scalar.key_access
&& existing.kind != kind
{
scalar.key_access_conflict = true;
}
scalar.key_access = Some(KeyAccessState { kind, access });
}
fn push_order_field(&mut self, field: &str, direction: OrderDirection) {
let scalar = self.scalar_mut();
scalar.order = Some(match scalar.order.take() {
Some(mut spec) => {
spec.fields.push((field.to_string(), direction));
spec
}
None => OrderSpec {
fields: vec![(field.to_string(), direction)],
},
});
}
}