2018-07-07 14:43:29 +02:00
|
|
|
mod sum_of_typos;
|
|
|
|
mod number_of_words;
|
|
|
|
mod words_proximity;
|
|
|
|
mod sum_of_words_attribute;
|
|
|
|
mod sum_of_words_position;
|
|
|
|
mod exact;
|
|
|
|
|
|
|
|
use std::cmp::Ordering;
|
|
|
|
use std::{mem, vec};
|
2018-05-27 15:23:43 +02:00
|
|
|
use fst;
|
2018-08-23 21:32:31 +02:00
|
|
|
use fnv::FnvHashMap;
|
2018-07-06 20:58:06 +02:00
|
|
|
use levenshtein::Levenshtein;
|
2018-07-10 21:29:17 +02:00
|
|
|
use metadata::{DocIndexes, OpWithStateBuilder, UnionWithState};
|
|
|
|
use {Match, DocumentId};
|
2018-07-07 14:43:29 +02:00
|
|
|
use group_by::GroupByMut;
|
2018-05-27 15:23:43 +02:00
|
|
|
|
2018-07-07 14:43:29 +02:00
|
|
|
use self::sum_of_typos::sum_of_typos;
|
|
|
|
use self::number_of_words::number_of_words;
|
|
|
|
use self::words_proximity::words_proximity;
|
|
|
|
use self::sum_of_words_attribute::sum_of_words_attribute;
|
|
|
|
use self::sum_of_words_position::sum_of_words_position;
|
|
|
|
use self::exact::exact;
|
2018-05-27 15:23:43 +02:00
|
|
|
|
2018-06-25 22:26:49 +02:00
|
|
|
#[inline]
|
|
|
|
fn match_query_index(a: &Match, b: &Match) -> bool {
|
|
|
|
a.query_index == b.query_index
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
2018-05-27 15:23:43 +02:00
|
|
|
pub struct Document {
|
2018-07-06 22:05:51 +02:00
|
|
|
pub document_id: DocumentId,
|
|
|
|
pub matches: Vec<Match>,
|
2018-05-27 15:23:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Document {
|
|
|
|
pub fn new(doc: DocumentId, match_: Match) -> Self {
|
|
|
|
Self::from_sorted_matches(doc, vec![match_])
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn from_sorted_matches(doc: DocumentId, matches: Vec<Match>) -> Self {
|
|
|
|
Self {
|
|
|
|
document_id: doc,
|
|
|
|
matches: matches,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-24 21:01:53 +02:00
|
|
|
fn matches_into_iter(matches: FnvHashMap<DocumentId, Vec<Match>>, limit: usize) -> vec::IntoIter<Document> {
|
|
|
|
let mut documents: Vec<_> = matches.into_iter().map(|(id, mut matches)| {
|
|
|
|
matches.sort_unstable();
|
|
|
|
Document::from_sorted_matches(id, matches)
|
|
|
|
}).collect();
|
|
|
|
|
|
|
|
let sorts = &[
|
|
|
|
sum_of_typos,
|
|
|
|
number_of_words,
|
|
|
|
words_proximity,
|
|
|
|
sum_of_words_attribute,
|
|
|
|
sum_of_words_position,
|
|
|
|
exact,
|
|
|
|
];
|
|
|
|
|
2018-08-25 12:35:29 +02:00
|
|
|
let mut groups = vec![documents.as_mut_slice()];
|
|
|
|
|
|
|
|
for sort in sorts {
|
|
|
|
let temp = mem::replace(&mut groups, Vec::new());
|
|
|
|
let mut computed = 0;
|
|
|
|
|
|
|
|
for group in temp {
|
|
|
|
group.sort_unstable_by(sort);
|
|
|
|
for group in GroupByMut::new(group, |a, b| sort(a, b) == Ordering::Equal) {
|
|
|
|
computed += group.len();
|
|
|
|
groups.push(group);
|
|
|
|
if computed >= limit { break }
|
2018-06-25 22:26:49 +02:00
|
|
|
}
|
|
|
|
}
|
2018-05-27 15:23:43 +02:00
|
|
|
}
|
2018-08-24 21:01:53 +02:00
|
|
|
|
|
|
|
documents.truncate(limit);
|
|
|
|
documents.into_iter()
|
2018-05-27 15:23:43 +02:00
|
|
|
}
|
|
|
|
|
2018-08-25 12:35:29 +02:00
|
|
|
pub struct RankedStream<'m, 'v>(RankedStreamInner<'m, 'v>);
|
2018-05-27 15:23:43 +02:00
|
|
|
|
|
|
|
impl<'m, 'v> RankedStream<'m, 'v> {
|
2018-07-10 21:29:17 +02:00
|
|
|
pub fn new(map: &'m fst::Map, indexes: &'v DocIndexes, automatons: Vec<Levenshtein>, limit: usize) -> Self {
|
|
|
|
let mut op = OpWithStateBuilder::new(indexes);
|
2018-05-27 15:23:43 +02:00
|
|
|
|
2018-07-06 20:58:06 +02:00
|
|
|
for automaton in automatons.iter().map(|l| l.dfa.clone()) {
|
2018-07-10 21:29:17 +02:00
|
|
|
let stream = map.search(automaton).with_state();
|
2018-05-27 15:23:43 +02:00
|
|
|
op.push(stream);
|
|
|
|
}
|
|
|
|
|
2018-08-25 12:35:29 +02:00
|
|
|
let inner = RankedStreamInner::Fed {
|
2018-05-27 15:23:43 +02:00
|
|
|
inner: op.union(),
|
|
|
|
automatons: automatons,
|
2018-08-24 21:01:53 +02:00
|
|
|
limit: limit,
|
|
|
|
matches: FnvHashMap::default(),
|
2018-08-25 12:35:29 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
RankedStream(inner)
|
2018-05-27 15:23:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'m, 'v, 'a> fst::Streamer<'a> for RankedStream<'m, 'v> {
|
2018-07-06 22:05:51 +02:00
|
|
|
type Item = Document;
|
2018-05-27 15:23:43 +02:00
|
|
|
|
|
|
|
fn next(&'a mut self) -> Option<Self::Item> {
|
2018-08-25 12:35:29 +02:00
|
|
|
self.0.next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum RankedStreamInner<'m, 'v> {
|
|
|
|
Fed {
|
|
|
|
inner: UnionWithState<'m, 'v, u32>,
|
|
|
|
automatons: Vec<Levenshtein>,
|
|
|
|
limit: usize,
|
|
|
|
matches: FnvHashMap<DocumentId, Vec<Match>>,
|
|
|
|
},
|
|
|
|
Pours {
|
|
|
|
inner: vec::IntoIter<Document>,
|
|
|
|
},
|
|
|
|
}
|
2018-05-27 15:23:43 +02:00
|
|
|
|
2018-08-25 12:35:29 +02:00
|
|
|
impl<'m, 'v, 'a> fst::Streamer<'a> for RankedStreamInner<'m, 'v> {
|
|
|
|
type Item = Document;
|
|
|
|
|
|
|
|
fn next(&'a mut self) -> Option<Self::Item> {
|
|
|
|
loop {
|
2018-05-27 15:23:43 +02:00
|
|
|
match self {
|
2018-08-25 12:35:29 +02:00
|
|
|
RankedStreamInner::Fed { inner, automatons, limit, matches } => {
|
2018-05-27 15:23:43 +02:00
|
|
|
match inner.next() {
|
2018-07-06 20:58:06 +02:00
|
|
|
Some((string, indexed_values)) => {
|
2018-05-27 15:23:43 +02:00
|
|
|
for iv in indexed_values {
|
|
|
|
|
2018-07-06 20:58:06 +02:00
|
|
|
let automaton = &automatons[iv.index];
|
|
|
|
let distance = automaton.dfa.distance(iv.state).to_u8();
|
2018-05-27 15:23:43 +02:00
|
|
|
|
|
|
|
for di in iv.values {
|
|
|
|
let match_ = Match {
|
|
|
|
query_index: iv.index as u32,
|
|
|
|
distance: distance,
|
|
|
|
attribute: di.attribute,
|
|
|
|
attribute_index: di.attribute_index,
|
2018-08-25 12:35:29 +02:00
|
|
|
is_exact: distance == 0 && string.len() == automaton.query_len,
|
2018-05-27 15:23:43 +02:00
|
|
|
};
|
|
|
|
matches.entry(di.document)
|
2018-08-24 21:01:53 +02:00
|
|
|
.or_insert_with(Vec::new)
|
|
|
|
.push(match_);
|
2018-05-27 15:23:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
None => {
|
2018-08-25 12:35:29 +02:00
|
|
|
let matches = mem::replace(matches, FnvHashMap::default());
|
|
|
|
*self = RankedStreamInner::Pours {
|
|
|
|
inner: matches_into_iter(matches, *limit).into_iter()
|
|
|
|
};
|
2018-05-27 15:23:43 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
2018-08-25 12:35:29 +02:00
|
|
|
RankedStreamInner::Pours { inner } => {
|
2018-07-06 22:05:51 +02:00
|
|
|
return inner.next()
|
2018-05-27 15:23:43 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|