use crate::{
db::{
access::{AccessPath, AccessPlan, ExecutableAccessPlan},
index::{
IndexBoundsSpec, IndexId, IndexRangeBoundEncodeError, RawIndexKey, build_index_bounds,
},
},
error::InternalError,
model::index::IndexModel,
types::EntityTag,
value::Value,
};
use std::ops::Bound;
pub(in crate::db) type LoweredKey = RawIndexKey;
#[derive(Clone, Debug, Eq, PartialEq)]
pub(in crate::db) struct LoweredAccess<'a, K> {
executable: ExecutableAccessPlan<'a, K>,
index_prefix_specs: Vec<LoweredIndexPrefixSpec>,
index_range_specs: Vec<LoweredIndexRangeSpec>,
}
impl<'a, K> LoweredAccess<'a, K> {
#[must_use]
pub(in crate::db) const fn executable(&self) -> &ExecutableAccessPlan<'a, K> {
&self.executable
}
#[must_use]
pub(in crate::db) const fn index_prefix_specs(&self) -> &[LoweredIndexPrefixSpec] {
self.index_prefix_specs.as_slice()
}
#[must_use]
pub(in crate::db) const fn index_range_specs(&self) -> &[LoweredIndexRangeSpec] {
self.index_range_specs.as_slice()
}
#[must_use]
pub(in crate::db) fn into_parts(
self,
) -> (
ExecutableAccessPlan<'a, K>,
Vec<LoweredIndexPrefixSpec>,
Vec<LoweredIndexRangeSpec>,
) {
(
self.executable,
self.index_prefix_specs,
self.index_range_specs,
)
}
}
#[derive(Debug)]
pub(in crate::db) enum LoweredAccessError {
IndexPrefix(InternalError),
IndexRange(InternalError),
}
impl LoweredAccessError {
#[must_use]
pub(in crate::db) fn into_internal_error(self) -> InternalError {
match self {
Self::IndexPrefix(err) | Self::IndexRange(err) => err,
}
}
}
pub(in crate::db) fn lower_access<K>(
entity_tag: EntityTag,
access: &AccessPlan<K>,
) -> Result<LoweredAccess<'_, K>, LoweredAccessError> {
let mut index_prefix_specs = Vec::new();
let mut index_range_specs = Vec::new();
let executable = lower_access_node(
entity_tag,
access,
&mut index_prefix_specs,
&mut index_range_specs,
)?;
Ok(LoweredAccess {
executable,
index_prefix_specs,
index_range_specs,
})
}
#[derive(Clone, Debug, Eq, PartialEq)]
pub(in crate::db) struct LoweredIndexPrefixSpec {
index: IndexModel,
lower: Bound<LoweredKey>,
upper: Bound<LoweredKey>,
}
impl LoweredIndexPrefixSpec {
const INVALID_REASON: &str = "validated index-prefix plan could not be lowered to raw bounds";
#[must_use]
pub(in crate::db) const fn new(
index: IndexModel,
lower: Bound<LoweredKey>,
upper: Bound<LoweredKey>,
) -> Self {
Self {
index,
lower,
upper,
}
}
#[must_use]
pub(in crate::db) const fn index(&self) -> &IndexModel {
&self.index
}
#[must_use]
pub(in crate::db) const fn lower(&self) -> &Bound<LoweredKey> {
&self.lower
}
#[must_use]
pub(in crate::db) const fn upper(&self) -> &Bound<LoweredKey> {
&self.upper
}
#[must_use]
pub(in crate::db) const fn invalid_reason() -> &'static str {
Self::INVALID_REASON
}
}
#[derive(Clone, Debug, Eq, PartialEq)]
pub(in crate::db) struct LoweredIndexRangeSpec {
index: IndexModel,
lower: Bound<LoweredKey>,
upper: Bound<LoweredKey>,
}
impl LoweredIndexRangeSpec {
const INVALID_REASON: &str = "validated index-range plan could not be lowered to raw bounds";
#[must_use]
pub(in crate::db) const fn new(
index: IndexModel,
lower: Bound<LoweredKey>,
upper: Bound<LoweredKey>,
) -> Self {
Self {
index,
lower,
upper,
}
}
#[must_use]
pub(in crate::db) const fn index(&self) -> &IndexModel {
&self.index
}
#[must_use]
pub(in crate::db) const fn lower(&self) -> &Bound<LoweredKey> {
&self.lower
}
#[must_use]
pub(in crate::db) const fn upper(&self) -> &Bound<LoweredKey> {
&self.upper
}
#[must_use]
pub(in crate::db) const fn invalid_reason() -> &'static str {
Self::INVALID_REASON
}
fn validated_spec_not_indexable(err: IndexRangeBoundEncodeError) -> InternalError {
InternalError::query_executor_invariant(err.validated_spec_not_indexable_reason())
}
}
fn lower_index_range_bounds_for_scope(
entity_tag: EntityTag,
index: &IndexModel,
prefix: &[Value],
lower: &Bound<Value>,
upper: &Bound<Value>,
) -> Result<(Bound<LoweredKey>, Bound<LoweredKey>), InternalError> {
let index_id = IndexId::new(entity_tag, index.ordinal());
build_index_bounds(
&index_id,
index,
IndexBoundsSpec::component_range(prefix, lower, upper),
)
.map_err(LoweredIndexRangeSpec::validated_spec_not_indexable)
}
fn lower_access_node<'a, K>(
entity_tag: EntityTag,
access: &'a AccessPlan<K>,
index_prefix_specs: &mut Vec<LoweredIndexPrefixSpec>,
index_range_specs: &mut Vec<LoweredIndexRangeSpec>,
) -> Result<ExecutableAccessPlan<'a, K>, LoweredAccessError> {
match access {
AccessPlan::Path(path) => {
let path = path.as_ref();
lower_index_specs_for_path(entity_tag, path, index_prefix_specs, index_range_specs)?;
Ok(ExecutableAccessPlan::from_access_path(path))
}
AccessPlan::Union(children) => {
let mut lowered_children = Vec::with_capacity(children.len());
for child in children {
lowered_children.push(lower_access_node(
entity_tag,
child,
index_prefix_specs,
index_range_specs,
)?);
}
Ok(ExecutableAccessPlan::union(lowered_children))
}
AccessPlan::Intersection(children) => {
let mut lowered_children = Vec::with_capacity(children.len());
for child in children {
lowered_children.push(lower_access_node(
entity_tag,
child,
index_prefix_specs,
index_range_specs,
)?);
}
Ok(ExecutableAccessPlan::intersection(lowered_children))
}
}
}
fn lower_index_specs_for_path<K>(
entity_tag: EntityTag,
path: &AccessPath<K>,
index_prefix_specs: &mut Vec<LoweredIndexPrefixSpec>,
index_range_specs: &mut Vec<LoweredIndexRangeSpec>,
) -> Result<(), LoweredAccessError> {
match path {
AccessPath::IndexPrefix { index, values } => {
lower_index_prefix_values_for_specs(entity_tag, *index, values, index_prefix_specs)
.map_err(LoweredAccessError::IndexPrefix)?;
}
AccessPath::IndexMultiLookup { index, values } => {
for value in values {
lower_index_prefix_values_for_specs(
entity_tag,
*index,
std::slice::from_ref(value),
index_prefix_specs,
)
.map_err(LoweredAccessError::IndexPrefix)?;
}
}
AccessPath::IndexRange { spec } => {
debug_assert_eq!(
spec.field_slots().len(),
spec.prefix_values().len().saturating_add(1),
"semantic range field-slot arity must remain prefix_len + range slot",
);
let (lower, upper) = lower_index_range_bounds_for_scope(
entity_tag,
spec.index(),
spec.prefix_values(),
spec.lower(),
spec.upper(),
)
.map_err(LoweredAccessError::IndexRange)?;
index_range_specs.push(LoweredIndexRangeSpec::new(*spec.index(), lower, upper));
}
AccessPath::ByKey(_)
| AccessPath::ByKeys(_)
| AccessPath::KeyRange { .. }
| AccessPath::FullScan => {}
}
Ok(())
}
fn lower_index_prefix_values_for_specs(
entity_tag: EntityTag,
index: IndexModel,
values: &[Value],
specs: &mut Vec<LoweredIndexPrefixSpec>,
) -> Result<(), InternalError> {
let index_id = IndexId::new(entity_tag, index.ordinal());
let (lower, upper) = build_index_bounds(&index_id, &index, IndexBoundsSpec::Prefix { values })
.map_err(|_| {
InternalError::query_executor_invariant("validated index-prefix value is not indexable")
})?;
specs.push(LoweredIndexPrefixSpec::new(index, lower, upper));
Ok(())
}