use crate::{
db::{
access::validate_access_structure_model as validate_access_structure_model_shared,
query::plan::{
AccessPlannedQuery, LogicalPlan, OrderSpec, ScalarPlan,
access_satisfies_deterministic_secondary_order_contract,
expr::supported_order_expr_requires_index_satisfied_access,
validate::{
GroupPlanError, PlanError, PolicyPlanError,
grouped::{
validate_group_cursor_constraints, validate_group_policy,
validate_group_structure, validate_projection_expr_types,
},
order::{
validate_no_duplicate_non_pk_order_fields, validate_order,
validate_primary_key_tie_break,
},
validate_plan_shape,
},
},
query::predicate::validate_predicate,
schema::SchemaInfo,
},
model::entity::EntityModel,
};
fn validate_access_structure_for_plan(
schema: &SchemaInfo,
model: &EntityModel,
plan: &AccessPlannedQuery,
) -> Result<(), PlanError> {
validate_access_structure_model_shared(schema, model, &plan.access).map_err(PlanError::from)
}
pub(in crate::db::query) fn validate_query_semantics(
schema: &SchemaInfo,
model: &EntityModel,
plan: &AccessPlannedQuery,
) -> Result<(), PlanError> {
let logical = plan.scalar_plan();
let projection = plan.projection_spec(model);
validate_plan_core(
schema,
model,
logical,
plan,
validate_order,
validate_access_structure_for_plan,
true,
)?;
validate_projection_expr_types(schema, &projection)?;
Ok(())
}
pub(in crate::db::query) fn validate_group_query_semantics(
schema: &SchemaInfo,
model: &EntityModel,
plan: &AccessPlannedQuery,
) -> Result<(), PlanError> {
let (logical, group, having_expr) = match &plan.logical {
LogicalPlan::Grouped(grouped) => (
&grouped.scalar,
&grouped.group,
grouped.having_expr.as_ref(),
),
LogicalPlan::Scalar(_) => {
return Err(PlanError::from(
GroupPlanError::grouped_logical_plan_required(),
));
}
};
let projection = plan.projection_spec(model);
validate_plan_core(
schema,
model,
logical,
plan,
validate_order,
validate_access_structure_for_plan,
false,
)?;
validate_group_structure(schema, model, group, &projection, having_expr)?;
validate_group_policy(schema, logical, group, having_expr)?;
validate_group_cursor_constraints(logical, group)?;
validate_projection_expr_types(schema, &projection)?;
Ok(())
}
fn validate_plan_core<FOrder, FAccess>(
schema: &SchemaInfo,
model: &EntityModel,
logical: &ScalarPlan,
plan: &AccessPlannedQuery,
validate_order_fn: FOrder,
validate_access_fn: FAccess,
require_primary_key_tie_break: bool,
) -> Result<(), PlanError>
where
FOrder: Fn(&SchemaInfo, &OrderSpec) -> Result<(), PlanError>,
FAccess: Fn(&SchemaInfo, &EntityModel, &AccessPlannedQuery) -> Result<(), PlanError>,
{
if let Some(predicate) = &logical.predicate {
validate_predicate(schema, predicate)?;
}
if let Some(order) = &logical.order {
validate_order_fn(schema, order)?;
validate_no_duplicate_non_pk_order_fields(model, order)?;
if require_primary_key_tie_break {
validate_primary_key_tie_break(model, order)?;
}
validate_expression_order_support(model, plan, order)?;
}
validate_access_fn(schema, model, plan)?;
validate_plan_shape(&plan.logical)?;
Ok(())
}
fn validate_expression_order_support(
_model: &EntityModel,
plan: &AccessPlannedQuery,
order: &OrderSpec,
) -> Result<(), PlanError> {
let expressions_requiring_index_support = order
.fields
.iter()
.map(crate::db::query::plan::OrderTerm::expr)
.any(supported_order_expr_requires_index_satisfied_access);
if !expressions_requiring_index_support {
return Ok(());
}
if plan.access.is_singleton_or_empty_primary_key_access() || plan.access.is_explicit_empty() {
return Ok(());
}
let access_capabilities = plan.access_capabilities();
let planner_route_profile = plan.planner_route_profile();
let logical_pushdown_eligibility = planner_route_profile.logical_pushdown_eligibility();
let secondary_contract_active = logical_pushdown_eligibility.secondary_order_allowed()
&& !logical_pushdown_eligibility.requires_full_materialization();
let secondary_pushdown_eligible = planner_route_profile
.secondary_order_contract()
.is_some_and(|order_contract| {
access_satisfies_deterministic_secondary_order_contract(
&access_capabilities,
order_contract,
)
});
if secondary_contract_active && secondary_pushdown_eligible {
return Ok(());
}
Err(PlanError::from(
PolicyPlanError::expression_order_requires_index_satisfied_access(),
))
}