1
0
mirror of https://github.com/meilisearch/MeiliSearch synced 2025-01-13 23:07:28 +01:00

Add crop algorithm with naive match algorithm

This commit is contained in:
ManyTheFish 2022-03-28 15:57:05 +02:00
parent d96e72e5dc
commit 3be1790803

@ -92,13 +92,15 @@ impl MatcherBuilder {
// }
// }
pub struct Match<'t> {
token: &'t Token<'t>,
#[derive(Clone)]
pub struct Match {
match_len: usize,
// id of the query word that matches.
id: usize,
// position of the word in the whole text.
position: usize,
word_position: usize,
// position of the token in the whole text.
token_position: usize,
}
pub struct MatchBounds {
@ -106,12 +108,6 @@ pub struct MatchBounds {
length: usize,
}
impl<'t> From<&Match<'t>> for MatchBounds {
fn from(m: &Match) -> Self {
MatchBounds { start: m.token.byte_start, length: m.match_len }
}
}
pub struct Matcher<'t, 'm> {
text: &'t str,
tokens: &'t [Token<'t>],
@ -120,26 +116,22 @@ pub struct Matcher<'t, 'm> {
crop_marker: &'m str,
highlight_prefix: &'m str,
highlight_suffix: &'m str,
matches: Option<Vec<Match<'t>>>,
matches: Option<Vec<Match>>,
}
impl<'t> Matcher<'t, '_> {
fn compute_matches(&mut self) -> &mut Self {
let mut matches = Vec::new();
let mut position = 0;
let mut word_position = 0;
let mut token_position = 0;
for token in self.tokens {
match token.is_separator() {
Some(SeparatorKind::Hard) => position += 7,
None => {
if let Some((match_len, id)) =
self.matching_words.matching_bytes_with_id(&token)
{
matches.push(Match { token, match_len, id, position });
}
position += 1;
if token.is_separator().is_none() {
if let Some((match_len, id)) = self.matching_words.matching_bytes_with_id(&token) {
matches.push(Match { match_len, id, word_position, token_position });
}
_otherwise => {}
word_position += 1;
}
token_position += 1;
}
self.matches = Some(matches);
@ -149,21 +141,104 @@ impl<'t> Matcher<'t, '_> {
pub fn matches(&mut self) -> Vec<MatchBounds> {
match &self.matches {
None => self.compute_matches().matches(),
Some(matches) => matches.iter().map(MatchBounds::from).collect(),
Some(matches) => matches
.iter()
.map(|m| MatchBounds {
start: self.tokens[m.token_position].byte_start,
length: m.match_len,
})
.collect(),
}
}
fn crop_bounds(&self, matches: &[Match<'t>]) -> (usize, usize) {
let byte_end = self
.tokens
.iter()
.filter(|t| t.is_separator().is_none())
.enumerate()
.take_while(|(i, _)| *i < self.crop_size)
.last()
.map_or(self.text.len(), |(_, t)| t.byte_end);
fn crop_around(&self, matches: &[Match]) -> (usize, usize) {
let first_match_word_position = matches.first().map(|m| m.word_position).unwrap_or(0);
let first_match_token_position = matches.first().map(|m| m.token_position).unwrap_or(0);
let last_match_word_position = matches.last().map(|m| m.word_position).unwrap_or(0);
let last_match_token_position = matches.last().map(|m| m.token_position).unwrap_or(0);
(0, byte_end)
// TODO: buggy if no match and fisrt token is a sepparator
let mut remaining_words =
self.crop_size + first_match_word_position - last_match_word_position - 1;
let mut first_token_position = first_match_token_position;
let mut last_token_position = last_match_token_position;
while remaining_words > 0 {
match (
first_token_position.checked_sub(1).and_then(|i| self.tokens.get(i)),
last_token_position.checked_add(1).and_then(|i| self.tokens.get(i)),
) {
(Some(ft), Some(lt)) => {
match (ft.is_separator(), lt.is_separator()) {
// if they are both separators and are the same kind then advance both
(Some(f_kind), Some(s_kind)) => {
if f_kind == s_kind {
first_token_position -= 1;
last_token_position += 1;
} else if f_kind == SeparatorKind::Hard {
last_token_position += 1;
} else {
first_token_position -= 1;
}
}
// left is a word, advance left
(None, Some(_)) => {
first_token_position -= 1;
remaining_words -= 1;
}
// right is a word, advance right
(Some(_), None) => {
last_token_position += 1;
remaining_words -= 1;
}
// both are words, advance left then right if remaining_word > 0
(None, None) => {
first_token_position -= 1;
remaining_words -= 1;
if remaining_words > 0 {
last_token_position += 1;
remaining_words -= 1;
}
}
}
}
(Some(ft), None) => {
first_token_position -= 1;
if ft.is_separator().is_none() {
remaining_words -= 1;
}
}
(None, Some(lt)) => {
last_token_position += 1;
if lt.is_separator().is_none() {
remaining_words -= 1;
}
}
(None, None) => break,
}
}
// if tokens after the end of the window are separators,
// then add them to the window in order to keep context in cropped text.
while let Some(_separator_kind) = last_token_position
.checked_add(1)
.and_then(|i| self.tokens.get(i))
.and_then(|t| t.is_separator())
{
last_token_position += 1;
}
(self.tokens[first_token_position].byte_start, self.tokens[last_token_position].byte_end)
}
fn crop_bounds(&self, matches: &[Match]) -> (usize, usize) {
match matches {
// at least 2 matches
[first, last, ..] => self.crop_around(&[first.clone()][..]),
// less than 2 matches
_ => self.crop_around(matches),
}
}
pub fn format(&mut self, highlight: bool, crop: bool) -> Cow<'t, str> {
@ -187,20 +262,23 @@ impl<'t> Matcher<'t, '_> {
if highlight {
// insert highlight markers around matches.
let tokens = self.tokens;
for m in matches
.iter()
.skip_while(|m| m.token.byte_start < byte_start)
.take_while(|m| m.token.byte_start < byte_end)
.skip_while(|m| tokens[m.token_position].byte_start < byte_start)
.take_while(|m| tokens[m.token_position].byte_start < byte_end)
{
if byte_index < m.token.byte_start {
formatted.push(&self.text[byte_index..m.token.byte_start]);
let token = &tokens[m.token_position];
if byte_index < token.byte_start {
formatted.push(&self.text[byte_index..token.byte_start]);
}
formatted.push(self.highlight_prefix);
formatted.push(&self.text[m.token.byte_start..m.token.byte_end]);
formatted.push(&self.text[token.byte_start..token.byte_end]);
formatted.push(self.highlight_suffix);
byte_index = m.token.byte_end;
byte_index = token.byte_end;
}
}
@ -271,7 +349,7 @@ mod tests {
assert_eq!(&matcher.format(highlight, crop), &text);
// Text containing all matches.
let text = "Natalie risk her future to build a world with the boy she loves. Emily Henry: The Love That Split The World";
let text = "Natalie risk her future to build a world with the boy she loves. Emily Henry: The Love That Split The World.";
let analyzed = analyzer.analyze(&text);
let tokens: Vec<_> = analyzed.tokens().collect();
let mut matcher = builder.build(&tokens[..], text);
@ -306,12 +384,12 @@ mod tests {
assert_eq!(&matcher.format(highlight, crop), &text);
// Text containing all matches.
let text = "Natalie risk her future to build a world with the boy she loves. Emily Henry: The Love That Split The World";
let text = "Natalie risk her future to build a world with the boy she loves. Emily Henry: The Love That Split The World.";
let analyzed = analyzer.analyze(&text);
let tokens: Vec<_> = analyzed.tokens().collect();
let mut matcher = builder.build(&tokens[..], text);
// no crop should return complete text with highlighted matches.
assert_eq!(&matcher.format(highlight, crop), "Natalie risk her future to build a <em>world</em> with <em>the</em> boy she loves. Emily Henry: <em>The</em> Love That <em>Split</em> <em>The</em> <em>World</em>");
assert_eq!(&matcher.format(highlight, crop), "Natalie risk her future to build a <em>world</em> with <em>the</em> boy she loves. Emily Henry: <em>The</em> Love That <em>Split</em> <em>The</em> <em>World</em>.");
// Text containing some matches.
let text = "Natalie risk her future to build a world with the boy she loves.";
@ -343,18 +421,18 @@ mod tests {
// no highlight should return 10 first words with a marker at the end.
assert_eq!(
&matcher.format(highlight, crop),
"A quick brown fox can not jump 32 feet, right"
"A quick brown fox can not jump 32 feet, right? "
);
// Text containing all matches.
let text = "Natalie risk her future to build a world with the boy she loves. Emily Henry: The Love That Split The World";
// Test phrase propagation
let text = "Natalie risk her future. Split The World is a book written by Emily Henry. I never read it.";
let analyzed = analyzer.analyze(&text);
let tokens: Vec<_> = analyzed.tokens().collect();
let mut matcher = builder.build(&tokens[..], text);
// no highlight should return 10 last words with a marker at the start.
// should crop the phrase instead of croping around the match.
assert_eq!(
&matcher.format(highlight, crop),
"she loves. Emily Henry: The Love That Split The World"
"Split The World is a book written by Emily Henry. …"
);
// Text containing some matches.
@ -368,6 +446,17 @@ mod tests {
"…future to build a world with the boy she loves."
);
// Text containing all matches.
let text = "Natalie risk her future to build a world with the boy she loves. Emily Henry: The Love That Split The World.";
let analyzed = analyzer.analyze(&text);
let tokens: Vec<_> = analyzed.tokens().collect();
let mut matcher = builder.build(&tokens[..], text);
// no highlight should return 10 last words with a marker at the start.
assert_eq!(
&matcher.format(highlight, crop),
"…she loves. Emily Henry: The Love That Split The World."
);
// Text containing a match unordered and a match ordered.
let text = "The world split void void void void void void void void void split the world void void";
let analyzed = analyzer.analyze(&text);
@ -398,17 +487,9 @@ mod tests {
// both should return 10 first words with a marker at the end.
assert_eq!(
&matcher.format(highlight, crop),
"A quick brown fox can not jump 32 feet, right"
"A quick brown fox can not jump 32 feet, right? "
);
// Text containing all matches.
let text = "Natalie risk her future to build a world with the boy she loves. Emily Henry: The Love That Split The World";
let analyzed = analyzer.analyze(&text);
let tokens: Vec<_> = analyzed.tokens().collect();
let mut matcher = builder.build(&tokens[..], text);
// both should return 10 last words with a marker at the start and highlighted matches.
assert_eq!(&matcher.format(highlight, crop), "…she loves. Emily Henry: <em>The</em> Love That <em>Split</em> <em>The</em> <em>World</em>");
// Text containing some matches.
let text = "Natalie risk her future to build a world with the boy she loves.";
let analyzed = analyzer.analyze(&text);
@ -420,6 +501,14 @@ mod tests {
"…future to build a <em>world</em> with <em>the</em> boy she loves."
);
// Text containing all matches.
let text = "Natalie risk her future to build a world with the boy she loves. Emily Henry: The Love That Split The World.";
let analyzed = analyzer.analyze(&text);
let tokens: Vec<_> = analyzed.tokens().collect();
let mut matcher = builder.build(&tokens[..], text);
// both should return 10 last words with a marker at the start and highlighted matches.
assert_eq!(&matcher.format(highlight, crop), "…she loves. Emily Henry: <em>The</em> Love That <em>Split</em> <em>The</em> <em>World</em>.");
// Text containing a match unordered and a match ordered.
let text = "The world split void void void void void void void void void split the world void void";
let analyzed = analyzer.analyze(&text);