MeiliSearch/milli/src/search/new/words.rs

114 lines
3.6 KiB
Rust
Raw Normal View History

2023-03-08 09:55:53 +01:00
use std::collections::BTreeSet;
use roaring::RoaringBitmap;
2023-02-22 15:34:37 +01:00
use super::logger::SearchLogger;
2023-03-14 16:37:47 +01:00
use super::query_graph::QueryNodeData;
use super::resolve_query_graph::compute_query_graph_docids;
2023-03-14 16:37:47 +01:00
use super::{QueryGraph, RankingRule, RankingRuleOutput, SearchContext};
use crate::{Result, TermsMatchingStrategy};
pub struct Words {
exhausted: bool, // TODO: remove
query_graph: Option<QueryGraph>,
iterating: bool, // TODO: remove
positions_to_remove: Vec<i8>,
terms_matching_strategy: TermsMatchingStrategy,
}
impl Words {
pub fn new(terms_matching_strategy: TermsMatchingStrategy) -> Self {
Self {
exhausted: true,
query_graph: None,
iterating: false,
positions_to_remove: vec![],
terms_matching_strategy,
}
}
}
2023-03-13 14:03:48 +01:00
impl<'ctx> RankingRule<'ctx, QueryGraph> for Words {
2023-02-22 15:34:37 +01:00
fn id(&self) -> String {
"words".to_owned()
}
fn start_iteration(
&mut self,
2023-03-13 14:03:48 +01:00
_ctx: &mut SearchContext<'ctx>,
2023-02-28 11:49:24 +01:00
_logger: &mut dyn SearchLogger<QueryGraph>,
_parent_candidates: &RoaringBitmap,
parent_query_graph: &QueryGraph,
) -> Result<()> {
self.exhausted = false;
self.query_graph = Some(parent_query_graph.clone());
let positions_to_remove = match self.terms_matching_strategy {
TermsMatchingStrategy::Last => {
let mut all_positions = BTreeSet::new();
2023-03-14 16:37:47 +01:00
for (_, n) in parent_query_graph.nodes.iter() {
match &n.data {
QueryNodeData::Term(term) => {
2023-03-23 09:15:57 +01:00
all_positions.extend(term.positions.clone());
}
2023-03-14 16:37:47 +01:00
QueryNodeData::Deleted | QueryNodeData::Start | QueryNodeData::End => {}
}
}
let mut r: Vec<i8> = all_positions.into_iter().collect();
// don't remove the first term
r.remove(0);
r
}
TermsMatchingStrategy::All => vec![],
};
self.positions_to_remove = positions_to_remove;
self.iterating = true;
Ok(())
}
fn next_bucket(
&mut self,
2023-03-13 14:03:48 +01:00
ctx: &mut SearchContext<'ctx>,
2023-02-22 15:34:37 +01:00
logger: &mut dyn SearchLogger<QueryGraph>,
universe: &RoaringBitmap,
) -> Result<Option<RankingRuleOutput<QueryGraph>>> {
assert!(self.iterating);
assert!(universe.len() > 1);
if self.exhausted {
return Ok(None);
}
let Some(query_graph) = &mut self.query_graph else { panic!() };
2023-02-21 13:57:34 +01:00
logger.log_words_state(query_graph);
let this_bucket = compute_query_graph_docids(ctx, query_graph, universe)?;
2023-02-21 13:57:34 +01:00
let child_query_graph = query_graph.clone();
loop {
if self.positions_to_remove.is_empty() {
self.exhausted = true;
break;
} else {
let position_to_remove = self.positions_to_remove.pop().unwrap();
2023-03-08 13:26:29 +01:00
let did_delete_any_node =
query_graph.remove_words_starting_at_position(position_to_remove);
if did_delete_any_node {
break;
}
}
}
Ok(Some(RankingRuleOutput { query: child_query_graph, candidates: this_bucket }))
}
fn end_iteration(
&mut self,
2023-03-13 14:03:48 +01:00
_ctx: &mut SearchContext<'ctx>,
2023-02-22 15:34:37 +01:00
_logger: &mut dyn SearchLogger<QueryGraph>,
) {
self.iterating = false;
self.exhausted = true;
self.positions_to_remove = vec![];
self.query_graph = None;
}
}