use crate::{
db::{
data::{DataKey, StorageKey, StructuralSlotReader},
index::{IndexId, IndexKey, IndexPlanReadView, plan::error::IndexPlanError},
},
error::InternalError,
model::{entity::EntityModel, index::IndexModel},
types::EntityTag,
};
use std::ops::Bound;
#[expect(clippy::too_many_arguments)]
pub(super) fn validate_unique_constraint_structural(
entity_path: &'static str,
entity_tag: EntityTag,
model: &'static EntityModel,
read_view: &dyn IndexPlanReadView,
index: &IndexModel,
index_fields: &str,
new_storage_key: Option<StorageKey>,
new_index_key: Option<&IndexKey>,
) -> Result<(), IndexPlanError> {
if !index.is_unique() {
return Ok(());
}
let Some(new_index_key) = new_index_key else {
return Ok(());
};
let Some(new_storage_key) = new_storage_key else {
return Err(InternalError::index_unique_validation_entity_key_required().into());
};
let index_id = IndexId::new(entity_tag, index.ordinal());
if new_index_key.index_id() != &index_id {
return Err(InternalError::index_unique_validation_corruption(
entity_path,
index_fields,
"mismatched unique key index id",
)
.into());
}
let (lower, upper) = new_index_key.raw_bounds_for_all_components();
let lower = Bound::Included(lower);
let upper = Bound::Included(upper);
let unique_probe_limit = 2usize;
let matching_storage_keys = read_view.read_index_keys_in_raw_range(
entity_path,
entity_tag,
index,
(&lower, &upper),
unique_probe_limit,
)?;
if matching_storage_keys.is_empty() {
return Ok(());
}
if matching_storage_keys.len() > 1 {
return Err(InternalError::index_unique_validation_corruption(
entity_path,
index_fields,
format_args!("{} keys", matching_storage_keys.len()),
)
.into());
}
let existing_key = matching_storage_keys[0];
if existing_key == new_storage_key {
return Ok(());
}
let data_key = DataKey::new(entity_tag, existing_key);
let row = read_view
.read_primary_row(&data_key)?
.ok_or_else(|| InternalError::index_unique_validation_row_required(&data_key))?;
let row_fields = decode_unique_row_slots(&data_key, &row, model)?;
let Some(stored_index_key) = build_unique_index_key_from_row_slots(
entity_tag,
entity_path,
&data_key,
existing_key,
&row_fields,
index,
)?
else {
return Err(InternalError::index_unique_validation_corruption(
entity_path,
index_fields,
"stored entity is not indexable for unique key",
)
.into());
};
if !stored_index_key.has_same_components(new_index_key) {
return Err(InternalError::index_unique_validation_corruption(
entity_path,
index_fields,
"index canonical collision",
)
.into());
}
Err(IndexPlanError::unique_violation(
entity_path,
index.fields(),
))
}
fn decode_unique_row_slots<'a>(
data_key: &DataKey,
row: &'a crate::db::data::RawRow,
model: &'static EntityModel,
) -> Result<StructuralSlotReader<'a>, InternalError> {
let row_fields = StructuralSlotReader::from_raw_row(row, model).map_err(|source| {
InternalError::index_unique_validation_row_deserialize_failed(data_key, source)
})?;
row_fields
.validate_storage_key(data_key)
.map_err(|source| {
InternalError::index_unique_validation_primary_key_decode_failed(data_key, source)
})?;
Ok(row_fields)
}
fn build_unique_index_key_from_row_slots(
entity_tag: EntityTag,
entity_path: &'static str,
data_key: &DataKey,
storage_key: StorageKey,
row_fields: &StructuralSlotReader<'_>,
index: &IndexModel,
) -> Result<Option<IndexKey>, InternalError> {
IndexKey::new_from_slots(entity_tag, storage_key, row_fields, index).map_err(|err| {
InternalError::index_unique_validation_key_rebuild_failed(data_key, entity_path, err)
})
}