Files
tantivy/src/query/score_combiner.rs

117 lines
2.8 KiB
Rust

use crate::query::Scorer;
use crate::Score;
/// The `ScoreCombiner` trait defines how to compute
/// an overall score given a list of scores.
pub trait ScoreCombiner: Default + Clone + Send + Copy + 'static {
/// Aggregates the score combiner with the given scorer.
///
/// The `ScoreCombiner` may decide to call `.scorer.score()`
/// or not.
fn update<TScorer: Scorer>(&mut self, scorer: &mut TScorer);
/// Clears the score combiner state back to its initial state.
fn clear(&mut self);
/// Returns the aggregate score.
fn score(&self) -> Score;
}
/// Just ignores scores. The `DoNothingCombiner` does not
/// even call the scorers `.score()` function.
///
/// It is useful to optimize the case when scoring is disabled.
#[derive(Default, Clone, Copy)] //< these should not be too much work :)
pub struct DoNothingCombiner;
impl ScoreCombiner for DoNothingCombiner {
fn update<TScorer: Scorer>(&mut self, _scorer: &mut TScorer) {}
fn clear(&mut self) {}
fn score(&self) -> Score {
1.0
}
}
/// Sums the score of different scorers.
#[derive(Default, Clone, Copy)]
pub struct SumCombiner {
score: Score,
}
impl ScoreCombiner for SumCombiner {
fn update<TScorer: Scorer>(&mut self, scorer: &mut TScorer) {
self.score += scorer.score();
}
fn clear(&mut self) {
self.score = 0.0;
}
fn score(&self) -> Score {
self.score
}
}
/// Sums the score of different scorers and keeps the count
/// of scorers which matched.
#[derive(Default, Clone, Copy)]
pub struct SumWithCoordsCombiner {
num_fields: usize,
score: Score,
}
impl ScoreCombiner for SumWithCoordsCombiner {
fn update<TScorer: Scorer>(&mut self, scorer: &mut TScorer) {
self.score += scorer.score();
self.num_fields += 1;
}
fn clear(&mut self) {
self.score = 0.0;
self.num_fields = 0;
}
fn score(&self) -> Score {
self.score
}
}
/// Take max score of different scorers
/// and optionally sum it with other matches multiplied by `tie_breaker`
#[derive(Default, Clone, Copy)]
pub struct DisjunctionMaxCombiner {
max: Score,
sum: Score,
tie_breaker: Score,
}
impl DisjunctionMaxCombiner {
/// Creates `DisjunctionMaxCombiner` with tie breaker
pub fn with_tie_breaker(tie_breaker: Score) -> DisjunctionMaxCombiner {
DisjunctionMaxCombiner {
max: 0.0,
sum: 0.0,
tie_breaker,
}
}
}
impl ScoreCombiner for DisjunctionMaxCombiner {
fn update<TScorer: Scorer>(&mut self, scorer: &mut TScorer) {
let score = scorer.score();
self.max = Score::max(score, self.max);
self.sum += score;
}
fn clear(&mut self) {
self.max = 0.0;
self.sum = 0.0;
}
fn score(&self) -> Score {
self.max + (self.sum - self.max) * self.tie_breaker
}
}