selene-db-graph 1.2.0

In-memory property-graph storage core (ArcSwap + imbl CoW, label/typed indexes, write funnel) for selene-db.
Documentation
//! Vector-index mutation methods for the transaction mutator.

use selene_core::{Change, DbString, HnswIndexConfig, SchemaChange, SchemaVectorIndexKind};

use crate::graph::VectorIndexEntry;
use crate::{GraphError, GraphResult, Mutator, VectorIndexConfig, VectorIndexKind};

impl<'tx, 'g> Mutator<'tx, 'g> {
    /// Register a durable node vector index in the active write transaction.
    ///
    /// # Errors
    ///
    /// Returns [`GraphError::VectorIndexAlreadyExists`] if the pair already
    /// exists, [`GraphError::VectorIndexInvalidDimension`] when `dimension` is
    /// zero, or [`GraphError::VectorIndexValueRejected`] if an existing non-null
    /// value for `(label, property)` is not a vector with `dimension`.
    pub fn create_vector_index(
        &mut self,
        label: DbString,
        property: DbString,
        kind: VectorIndexKind,
        dimension: u32,
    ) -> GraphResult<()> {
        self.create_vector_index_named(label, property, kind, dimension, None)
    }

    /// Register a durable node vector index with optional catalog name.
    pub fn create_vector_index_named(
        &mut self,
        label: DbString,
        property: DbString,
        kind: VectorIndexKind,
        dimension: u32,
        name: Option<DbString>,
    ) -> GraphResult<()> {
        self.create_vector_index_named_with_config(label, property, kind, dimension, name, None)
    }

    /// Register a durable node vector index with optional HNSW construction config.
    pub fn create_vector_index_named_with_config(
        &mut self,
        label: DbString,
        property: DbString,
        kind: VectorIndexKind,
        dimension: u32,
        name: Option<DbString>,
        hnsw_config: Option<HnswIndexConfig>,
    ) -> GraphResult<()> {
        self.create_vector_index_named_with_configs(
            label,
            property,
            kind,
            dimension,
            name,
            VectorIndexConfig::new(hnsw_config, None),
        )
    }

    /// Register a durable node vector index with optional ANN construction config.
    pub fn create_vector_index_named_with_configs(
        &mut self,
        label: DbString,
        property: DbString,
        kind: VectorIndexKind,
        dimension: u32,
        name: Option<DbString>,
        config: VectorIndexConfig,
    ) -> GraphResult<()> {
        if self
            .txn
            .read()
            .vector_index
            .contains_key(&(label.clone(), property.clone()))
        {
            return Err(GraphError::VectorIndexAlreadyExists { label, property });
        }
        let index = crate::vector_index::build_vector_index_with_configs(
            self.txn.read(),
            label.clone(),
            property.clone(),
            kind,
            dimension,
            config,
        )?;
        let hnsw_config = index.hnsw_config();
        let ivf_config = index.ivf_config();
        let graph_id = self.txn.read().graph_id();
        self.txn.guard_mut().vector_index.insert(
            (label.clone(), property.clone()),
            VectorIndexEntry::new(index, name.clone()),
        );
        self.txn.changes.push(Change::SchemaChanged {
            graph: graph_id,
            change: SchemaChange::VectorIndexCreated {
                label,
                property,
                kind: schema_kind_from(kind),
                dimension,
                name,
                hnsw_config,
                ivf_config,
            },
        });
        Ok(())
    }

    /// Drop a durable node vector index from the active write transaction.
    ///
    /// The operation is idempotent. Dropping an absent index succeeds and emits
    /// no WAL change.
    pub fn drop_vector_index(&mut self, label: DbString, property: DbString) -> GraphResult<()> {
        if !self
            .txn
            .read()
            .vector_index
            .contains_key(&(label.clone(), property.clone()))
        {
            return Ok(());
        }
        let graph_id = self.txn.read().graph_id();
        self.txn
            .guard_mut()
            .vector_index
            .remove(&(label.clone(), property.clone()));
        self.txn.changes.push(Change::SchemaChanged {
            graph: graph_id,
            change: SchemaChange::VectorIndexDropped { label, property },
        });
        Ok(())
    }
}

const fn schema_kind_from(kind: VectorIndexKind) -> SchemaVectorIndexKind {
    match kind {
        VectorIndexKind::Flat => SchemaVectorIndexKind::Flat,
        VectorIndexKind::HnswSquaredEuclidean => SchemaVectorIndexKind::HnswSquaredEuclidean,
        VectorIndexKind::HnswCosine => SchemaVectorIndexKind::HnswCosine,
        VectorIndexKind::HnswNegativeInnerProduct => {
            SchemaVectorIndexKind::HnswNegativeInnerProduct
        }
        VectorIndexKind::IvfSquaredEuclidean => SchemaVectorIndexKind::IvfSquaredEuclidean,
        VectorIndexKind::IvfCosine => SchemaVectorIndexKind::IvfCosine,
        VectorIndexKind::IvfNegativeInnerProduct => SchemaVectorIndexKind::IvfNegativeInnerProduct,
        VectorIndexKind::TurboQuantCosine => SchemaVectorIndexKind::TurboQuantCosine,
    }
}

#[cfg(test)]
#[path = "vector_index/tests.rs"]
mod tests;