agdb 0.12.10

Agnesoft Graph Database
Documentation
use crate::DbElement;
use crate::DbError;
use crate::DbImpl;
use crate::DbValue;
use crate::Query;
use crate::QueryIds;
use crate::QueryResult;
use crate::SearchQuery;
use crate::StorageData;
use crate::query_builder::search::SearchQueryBuilder;

/// Query to select elements with only certain properties of
/// given ids. All ids must exist in the database and all
/// of them must have the requested properties.
///
/// The result will be number of elements and the
/// list of elements with the requested properties.
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
#[cfg_attr(feature = "openapi", derive(utoipa::ToSchema))]
#[cfg_attr(feature = "derive", derive(agdb::DbSerialize))]
#[cfg_attr(feature = "api", derive(agdb::TypeDefImpl))]
#[derive(Clone, Debug, PartialEq)]
pub struct SelectValuesQuery {
    pub keys: Vec<DbValue>,
    pub ids: QueryIds,
}

impl Query for SelectValuesQuery {
    fn process<Store: StorageData>(&self, db: &DbImpl<Store>) -> Result<QueryResult, DbError> {
        let mut result = QueryResult::default();

        let (db_ids, is_search) = match &self.ids {
            QueryIds::Ids(ids) => {
                let mut db_ids = Vec::with_capacity(ids.len());

                for query_id in ids {
                    db_ids.push(db.db_id(query_id)?);
                }

                (db_ids, false)
            }
            QueryIds::Search(search_query) => (search_query.search(db)?, true),
        };

        result.elements.reserve(db_ids.len());
        result.result = db_ids.len() as i64;

        for id in db_ids {
            let values = if self.keys.is_empty() {
                db.values(id)?
            } else {
                db.values_by_keys(id, &self.keys)?
            };

            if !is_search && values.len() != self.keys.len() {
                for key in &self.keys {
                    if !values.iter().any(|x| x.key == *key) {
                        return Err(DbError::from(format!(
                            "Missing key '{}' for id '{}'",
                            key, id.0
                        )));
                    }
                }
            }

            result.elements.push(DbElement {
                id,
                from: db.from_id(id),
                to: db.to_id(id),
                values,
            });
        }

        Ok(result)
    }
}

impl Query for &SelectValuesQuery {
    fn process<Store: StorageData>(&self, db: &DbImpl<Store>) -> Result<QueryResult, DbError> {
        (*self).process(db)
    }
}

impl SearchQueryBuilder for SelectValuesQuery {
    fn search_mut(&mut self) -> &mut SearchQuery {
        if let QueryIds::Search(search) = &mut self.ids {
            search
        } else {
            panic!("Expected search query");
        }
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    #[should_panic]
    fn missing_search() {
        SelectValuesQuery {
            keys: vec![],
            ids: QueryIds::Ids(vec![]),
        }
        .search_mut();
    }
}