mirror of
https://github.com/meilisearch/MeiliSearch
synced 2025-01-25 12:47:28 +01:00
Fix the geo sort when lat and lng are strings
This commit is contained in:
parent
f9d94c5845
commit
d383afc82b
@ -100,7 +100,7 @@ fn facet_number_values<'a>(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Return an iterator over each string value in the given field of the given document.
|
/// Return an iterator over each string value in the given field of the given document.
|
||||||
fn facet_string_values<'a>(
|
pub fn facet_string_values<'a>(
|
||||||
docid: u32,
|
docid: u32,
|
||||||
field_id: u16,
|
field_id: u16,
|
||||||
index: &Index,
|
index: &Index,
|
||||||
|
@ -6,6 +6,7 @@ use heed::{RoPrefix, RoTxn};
|
|||||||
use roaring::RoaringBitmap;
|
use roaring::RoaringBitmap;
|
||||||
use rstar::RTree;
|
use rstar::RTree;
|
||||||
|
|
||||||
|
use super::facet_string_values;
|
||||||
use super::ranking_rules::{RankingRule, RankingRuleOutput, RankingRuleQueryTrait};
|
use super::ranking_rules::{RankingRule, RankingRuleOutput, RankingRuleQueryTrait};
|
||||||
use crate::heed_codec::facet::{FieldDocIdFacetCodec, OrderedF64Codec};
|
use crate::heed_codec::facet::{FieldDocIdFacetCodec, OrderedF64Codec};
|
||||||
use crate::score_details::{self, ScoreDetails};
|
use crate::score_details::{self, ScoreDetails};
|
||||||
@ -157,23 +158,7 @@ impl<Q: RankingRuleQueryTrait> GeoSort<Q> {
|
|||||||
let mut documents = self
|
let mut documents = self
|
||||||
.geo_candidates
|
.geo_candidates
|
||||||
.iter()
|
.iter()
|
||||||
.map(|id| -> Result<_> {
|
.map(|id| -> Result<_> { Ok((id, geo_value(id, lat, lng, ctx.index, ctx.txn)?)) })
|
||||||
Ok((
|
|
||||||
id,
|
|
||||||
[
|
|
||||||
facet_number_values(id, lat, ctx.index, ctx.txn)?
|
|
||||||
.next()
|
|
||||||
.expect("A geo faceted document doesn't contain any lat")?
|
|
||||||
.0
|
|
||||||
.2,
|
|
||||||
facet_number_values(id, lng, ctx.index, ctx.txn)?
|
|
||||||
.next()
|
|
||||||
.expect("A geo faceted document doesn't contain any lng")?
|
|
||||||
.0
|
|
||||||
.2,
|
|
||||||
],
|
|
||||||
))
|
|
||||||
})
|
|
||||||
.collect::<Result<Vec<(u32, [f64; 2])>>>()?;
|
.collect::<Result<Vec<(u32, [f64; 2])>>>()?;
|
||||||
// computing the distance between two points is expensive thus we cache the result
|
// computing the distance between two points is expensive thus we cache the result
|
||||||
documents
|
documents
|
||||||
@ -185,6 +170,37 @@ impl<Q: RankingRuleQueryTrait> GeoSort<Q> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Extracts the lat and long values from a single document.
|
||||||
|
///
|
||||||
|
/// If it is not able to find it in the facet number index it will extract it
|
||||||
|
/// from the facet string index and parse it as f64 (as the geo extraction behaves).
|
||||||
|
fn geo_value(
|
||||||
|
docid: u32,
|
||||||
|
field_lat: u16,
|
||||||
|
field_lng: u16,
|
||||||
|
index: &Index,
|
||||||
|
rtxn: &RoTxn,
|
||||||
|
) -> Result<[f64; 2]> {
|
||||||
|
let extract_geo = |geo_field: u16| -> Result<f64> {
|
||||||
|
match facet_number_values(docid, geo_field, index, rtxn)?.next() {
|
||||||
|
Some(Ok(((_, _, geo), ()))) => Ok(geo),
|
||||||
|
Some(Err(e)) => Err(e.into()),
|
||||||
|
None => match facet_string_values(docid, geo_field, index, rtxn)?.next() {
|
||||||
|
Some(Ok((_, geo))) => {
|
||||||
|
Ok(geo.parse::<f64>().expect("cannot parse geo field as f64"))
|
||||||
|
}
|
||||||
|
Some(Err(e)) => Err(e.into()),
|
||||||
|
None => panic!("A geo faceted document doesn't contain any lat or lng"),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let lat = extract_geo(field_lat)?;
|
||||||
|
let lng = extract_geo(field_lng)?;
|
||||||
|
|
||||||
|
Ok([lat, lng])
|
||||||
|
}
|
||||||
|
|
||||||
impl<'ctx, Q: RankingRuleQueryTrait> RankingRule<'ctx, Q> for GeoSort<Q> {
|
impl<'ctx, Q: RankingRuleQueryTrait> RankingRule<'ctx, Q> for GeoSort<Q> {
|
||||||
fn id(&self) -> String {
|
fn id(&self) -> String {
|
||||||
"geo_sort".to_owned()
|
"geo_sort".to_owned()
|
||||||
|
@ -42,6 +42,7 @@ use roaring::RoaringBitmap;
|
|||||||
use sort::Sort;
|
use sort::Sort;
|
||||||
use space::Neighbor;
|
use space::Neighbor;
|
||||||
|
|
||||||
|
use self::distinct::facet_string_values;
|
||||||
use self::geo_sort::GeoSort;
|
use self::geo_sort::GeoSort;
|
||||||
pub use self::geo_sort::Strategy as GeoSortStrategy;
|
pub use self::geo_sort::Strategy as GeoSortStrategy;
|
||||||
use self::graph_based_ranking_rule::Words;
|
use self::graph_based_ranking_rule::Words;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user