mirror of https://github.com/apache/lucene.git
LUCENE-5118: multiplier to spatial makeDistanceValueSource
git-svn-id: https://svn.apache.org/repos/asf/lucene/dev/trunk@1506632 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
f3856fa67c
commit
47bf70812b
|
@ -64,6 +64,9 @@ New features
|
||||||
(default is false). If true then edits are measured in Unicode code
|
(default is false). If true then edits are measured in Unicode code
|
||||||
points instead of UTF8 bytes. (Artem Lukanin via Mike McCandless)
|
points instead of UTF8 bytes. (Artem Lukanin via Mike McCandless)
|
||||||
|
|
||||||
|
* LUCENE-5118: SpatialStrategy.makeDistanceValueSource() now has an optional
|
||||||
|
multiplier for scaling degrees to another unit. (David Smiley)
|
||||||
|
|
||||||
Bug Fixes
|
Bug Fixes
|
||||||
|
|
||||||
* LUCENE-5116: IndexWriter.addIndexes(IndexReader...) should drop empty (or all
|
* LUCENE-5116: IndexWriter.addIndexes(IndexReader...) should drop empty (or all
|
||||||
|
|
|
@ -101,12 +101,21 @@ public abstract class SpatialStrategy {
|
||||||
*/
|
*/
|
||||||
public abstract Field[] createIndexableFields(Shape shape);
|
public abstract Field[] createIndexableFields(Shape shape);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* See {@link #makeDistanceValueSource(com.spatial4j.core.shape.Point, double)} called with
|
||||||
|
* a multiplier of 1.0 (i.e. units of degrees).
|
||||||
|
*/
|
||||||
|
public ValueSource makeDistanceValueSource(Point queryPoint) {
|
||||||
|
return makeDistanceValueSource(queryPoint, 1.0);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Make a ValueSource returning the distance between the center of the
|
* Make a ValueSource returning the distance between the center of the
|
||||||
* indexed shape and {@code queryPoint}. If there are multiple indexed shapes
|
* indexed shape and {@code queryPoint}. If there are multiple indexed shapes
|
||||||
* then the closest one is chosen.
|
* then the closest one is chosen. The result is multiplied by {@code multiplier}, which
|
||||||
|
* conveniently is used to get the desired units.
|
||||||
*/
|
*/
|
||||||
public abstract ValueSource makeDistanceValueSource(Point queryPoint);
|
public abstract ValueSource makeDistanceValueSource(Point queryPoint, double multiplier);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Make a Query based principally on {@link org.apache.lucene.spatial.query.SpatialOperation}
|
* Make a Query based principally on {@link org.apache.lucene.spatial.query.SpatialOperation}
|
||||||
|
@ -139,7 +148,7 @@ public abstract class SpatialStrategy {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a ValueSource with values ranging from 1 to 0, depending inversely
|
* Returns a ValueSource with values ranging from 1 to 0, depending inversely
|
||||||
* on the distance from {@link #makeDistanceValueSource(com.spatial4j.core.shape.Point)}.
|
* on the distance from {@link #makeDistanceValueSource(com.spatial4j.core.shape.Point,double)}.
|
||||||
* The formula is {@code c/(d + c)} where 'd' is the distance and 'c' is
|
* The formula is {@code c/(d + c)} where 'd' is the distance and 'c' is
|
||||||
* one tenth the distance to the farthest edge from the center. Thus the
|
* one tenth the distance to the farthest edge from the center. Thus the
|
||||||
* scores will be 1 for indexed points at the center of the query shape and as
|
* scores will be 1 for indexed points at the center of the query shape and as
|
||||||
|
@ -151,7 +160,7 @@ public abstract class SpatialStrategy {
|
||||||
ctx.makePoint(bbox.getMinX(), bbox.getMinY()), bbox.getMaxX(), bbox.getMaxY());
|
ctx.makePoint(bbox.getMinX(), bbox.getMinY()), bbox.getMaxX(), bbox.getMaxY());
|
||||||
double distToEdge = diagonalDist * 0.5;
|
double distToEdge = diagonalDist * 0.5;
|
||||||
float c = (float)distToEdge * 0.1f;//one tenth
|
float c = (float)distToEdge * 0.1f;//one tenth
|
||||||
return new ReciprocalFloatFunction(makeDistanceValueSource(queryShape.getCenter()), 1f, c, c);
|
return new ReciprocalFloatFunction(makeDistanceValueSource(queryShape.getCenter(), 1.0), 1f, c, c);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -135,9 +135,9 @@ public class BBoxStrategy extends SpatialStrategy {
|
||||||
//---------------------------------
|
//---------------------------------
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ValueSource makeDistanceValueSource(Point queryPoint) {
|
public ValueSource makeDistanceValueSource(Point queryPoint, double multiplier) {
|
||||||
return new BBoxSimilarityValueSource(
|
return new BBoxSimilarityValueSource(
|
||||||
this, new DistanceSimilarity(this.getSpatialContext(), queryPoint));
|
this, new DistanceSimilarity(this.getSpatialContext(), queryPoint, multiplier));
|
||||||
}
|
}
|
||||||
|
|
||||||
public ValueSource makeBBoxAreaSimilarityValueSource(Rectangle queryBox) {
|
public ValueSource makeBBoxAreaSimilarityValueSource(Rectangle queryBox) {
|
||||||
|
|
|
@ -30,13 +30,15 @@ import org.apache.lucene.search.Explanation;
|
||||||
*/
|
*/
|
||||||
public class DistanceSimilarity implements BBoxSimilarity {
|
public class DistanceSimilarity implements BBoxSimilarity {
|
||||||
private final Point queryPoint;
|
private final Point queryPoint;
|
||||||
|
private final double multiplier;
|
||||||
private final DistanceCalculator distCalc;
|
private final DistanceCalculator distCalc;
|
||||||
private final double nullValue;
|
private final double nullValue;
|
||||||
|
|
||||||
public DistanceSimilarity(SpatialContext ctx, Point queryPoint) {
|
public DistanceSimilarity(SpatialContext ctx, Point queryPoint, double multiplier) {
|
||||||
this.queryPoint = queryPoint;
|
this.queryPoint = queryPoint;
|
||||||
|
this.multiplier = multiplier;
|
||||||
this.distCalc = ctx.getDistCalc();
|
this.distCalc = ctx.getDistCalc();
|
||||||
this.nullValue = (ctx.isGeo() ? 180 : Double.MAX_VALUE);
|
this.nullValue = (ctx.isGeo() ? 180 * multiplier : Double.MAX_VALUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -45,14 +47,43 @@ public class DistanceSimilarity implements BBoxSimilarity {
|
||||||
if (indexRect == null) {
|
if (indexRect == null) {
|
||||||
score = nullValue;
|
score = nullValue;
|
||||||
} else {
|
} else {
|
||||||
score = distCalc.distance(queryPoint, indexRect.getCenter());
|
score = distCalc.distance(queryPoint, indexRect.getCenter()) * multiplier;
|
||||||
}
|
}
|
||||||
if (exp != null) {
|
if (exp != null) {
|
||||||
exp.setValue((float)score);
|
exp.setValue((float)score);
|
||||||
exp.setDescription(this.getClass().getSimpleName());
|
exp.setDescription(this.getClass().getSimpleName());
|
||||||
exp.addDetail(new Explanation(-1f,""+queryPoint));
|
exp.addDetail(new Explanation(-1f, "" + queryPoint));
|
||||||
exp.addDetail(new Explanation(-1f,""+indexRect));
|
exp.addDetail(new Explanation(-1f,""+indexRect));
|
||||||
|
exp.addDetail(new Explanation((float)multiplier,"multiplier"));
|
||||||
}
|
}
|
||||||
return score;
|
return score;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object o) {
|
||||||
|
if (this == o) return true;
|
||||||
|
if (o == null || getClass() != o.getClass()) return false;
|
||||||
|
|
||||||
|
DistanceSimilarity that = (DistanceSimilarity) o;
|
||||||
|
|
||||||
|
if (Double.compare(that.multiplier, multiplier) != 0) return false;
|
||||||
|
if (Double.compare(that.nullValue, nullValue) != 0) return false;
|
||||||
|
if (!distCalc.equals(that.distCalc)) return false;
|
||||||
|
if (!queryPoint.equals(that.queryPoint)) return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
int result;
|
||||||
|
long temp;
|
||||||
|
result = queryPoint.hashCode();
|
||||||
|
temp = Double.doubleToLongBits(multiplier);
|
||||||
|
result = 31 * result + (int) (temp ^ (temp >>> 32));
|
||||||
|
result = 31 * result + distCalc.hashCode();
|
||||||
|
temp = Double.doubleToLongBits(nullValue);
|
||||||
|
result = 31 * result + (int) (temp ^ (temp >>> 32));
|
||||||
|
return result;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -56,7 +56,7 @@ import java.util.concurrent.ConcurrentHashMap;
|
||||||
* <li>Only {@link org.apache.lucene.spatial.query.SpatialOperation#Intersects}
|
* <li>Only {@link org.apache.lucene.spatial.query.SpatialOperation#Intersects}
|
||||||
* is supported. If only points are indexed then this is effectively equivalent
|
* is supported. If only points are indexed then this is effectively equivalent
|
||||||
* to IsWithin.</li>
|
* to IsWithin.</li>
|
||||||
* <li>The strategy supports {@link #makeDistanceValueSource(com.spatial4j.core.shape.Point)}
|
* <li>The strategy supports {@link #makeDistanceValueSource(com.spatial4j.core.shape.Point,double)}
|
||||||
* even for multi-valued data, so long as the indexed data is all points; the
|
* even for multi-valued data, so long as the indexed data is all points; the
|
||||||
* behavior is undefined otherwise. However, <em>it will likely be removed in
|
* behavior is undefined otherwise. However, <em>it will likely be removed in
|
||||||
* the future</em> in lieu of using another strategy with a more scalable
|
* the future</em> in lieu of using another strategy with a more scalable
|
||||||
|
@ -182,7 +182,7 @@ public abstract class PrefixTreeStrategy extends SpatialStrategy {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ValueSource makeDistanceValueSource(Point queryPoint) {
|
public ValueSource makeDistanceValueSource(Point queryPoint, double multiplier) {
|
||||||
PointPrefixTreeFieldCacheProvider p = provider.get( getFieldName() );
|
PointPrefixTreeFieldCacheProvider p = provider.get( getFieldName() );
|
||||||
if( p == null ) {
|
if( p == null ) {
|
||||||
synchronized (this) {//double checked locking idiom is okay since provider is threadsafe
|
synchronized (this) {//double checked locking idiom is okay since provider is threadsafe
|
||||||
|
@ -194,7 +194,7 @@ public abstract class PrefixTreeStrategy extends SpatialStrategy {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return new ShapeFieldCacheDistanceValueSource(ctx, p, queryPoint);
|
return new ShapeFieldCacheDistanceValueSource(ctx, p, queryPoint, multiplier);
|
||||||
}
|
}
|
||||||
|
|
||||||
public SpatialPrefixTree getGrid() {
|
public SpatialPrefixTree getGrid() {
|
||||||
|
|
|
@ -38,14 +38,17 @@ import java.util.Map;
|
||||||
*/
|
*/
|
||||||
public class ShapeFieldCacheDistanceValueSource extends ValueSource {
|
public class ShapeFieldCacheDistanceValueSource extends ValueSource {
|
||||||
|
|
||||||
private final ShapeFieldCacheProvider<Point> provider;
|
|
||||||
private final SpatialContext ctx;
|
private final SpatialContext ctx;
|
||||||
private final Point from;
|
private final Point from;
|
||||||
|
private final ShapeFieldCacheProvider<Point> provider;
|
||||||
|
private final double multiplier;
|
||||||
|
|
||||||
public ShapeFieldCacheDistanceValueSource(SpatialContext ctx, ShapeFieldCacheProvider<Point> provider, Point from) {
|
public ShapeFieldCacheDistanceValueSource(SpatialContext ctx,
|
||||||
|
ShapeFieldCacheProvider<Point> provider, Point from, double multiplier) {
|
||||||
this.ctx = ctx;
|
this.ctx = ctx;
|
||||||
this.from = from;
|
this.from = from;
|
||||||
this.provider = provider;
|
this.provider = provider;
|
||||||
|
this.multiplier = multiplier;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -60,7 +63,7 @@ public class ShapeFieldCacheDistanceValueSource extends ValueSource {
|
||||||
provider.getCache(readerContext.reader());
|
provider.getCache(readerContext.reader());
|
||||||
private final Point from = ShapeFieldCacheDistanceValueSource.this.from;
|
private final Point from = ShapeFieldCacheDistanceValueSource.this.from;
|
||||||
private final DistanceCalculator calculator = ctx.getDistCalc();
|
private final DistanceCalculator calculator = ctx.getDistCalc();
|
||||||
private final double nullValue = (ctx.isGeo() ? 180 : Double.MAX_VALUE);
|
private final double nullValue = (ctx.isGeo() ? 180 * multiplier : Double.MAX_VALUE);
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public float floatVal(int doc) {
|
public float floatVal(int doc) {
|
||||||
|
@ -69,13 +72,14 @@ public class ShapeFieldCacheDistanceValueSource extends ValueSource {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public double doubleVal(int doc) {
|
public double doubleVal(int doc) {
|
||||||
|
|
||||||
List<Point> vals = cache.getShapes( doc );
|
List<Point> vals = cache.getShapes( doc );
|
||||||
if( vals != null ) {
|
if( vals != null ) {
|
||||||
double v = calculator.distance(from, vals.get(0));
|
double v = calculator.distance(from, vals.get(0));
|
||||||
for( int i=1; i<vals.size(); i++ ) {
|
for( int i=1; i<vals.size(); i++ ) {
|
||||||
v = Math.min(v, calculator.distance(from, vals.get(i)));
|
v = Math.min(v, calculator.distance(from, vals.get(i)));
|
||||||
}
|
}
|
||||||
return v;
|
return v * multiplier;
|
||||||
}
|
}
|
||||||
return nullValue;
|
return nullValue;
|
||||||
}
|
}
|
||||||
|
@ -97,6 +101,7 @@ public class ShapeFieldCacheDistanceValueSource extends ValueSource {
|
||||||
if (!ctx.equals(that.ctx)) return false;
|
if (!ctx.equals(that.ctx)) return false;
|
||||||
if (!from.equals(that.from)) return false;
|
if (!from.equals(that.from)) return false;
|
||||||
if (!provider.equals(that.provider)) return false;
|
if (!provider.equals(that.provider)) return false;
|
||||||
|
if (multiplier != that.multiplier) return false;
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,13 +39,15 @@ public class DistanceValueSource extends ValueSource {
|
||||||
|
|
||||||
private PointVectorStrategy strategy;
|
private PointVectorStrategy strategy;
|
||||||
private final Point from;
|
private final Point from;
|
||||||
|
private final double multiplier;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor.
|
* Constructor.
|
||||||
*/
|
*/
|
||||||
public DistanceValueSource(PointVectorStrategy strategy, Point from) {
|
public DistanceValueSource(PointVectorStrategy strategy, Point from, double multiplier) {
|
||||||
this.strategy = strategy;
|
this.strategy = strategy;
|
||||||
this.from = from;
|
this.from = from;
|
||||||
|
this.multiplier = multiplier;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -72,7 +74,8 @@ public class DistanceValueSource extends ValueSource {
|
||||||
|
|
||||||
private final Point from = DistanceValueSource.this.from;
|
private final Point from = DistanceValueSource.this.from;
|
||||||
private final DistanceCalculator calculator = strategy.getSpatialContext().getDistCalc();
|
private final DistanceCalculator calculator = strategy.getSpatialContext().getDistCalc();
|
||||||
private final double nullValue = (strategy.getSpatialContext().isGeo() ? 180 : Double.MAX_VALUE);
|
private final double nullValue =
|
||||||
|
(strategy.getSpatialContext().isGeo() ? 180 * multiplier : Double.MAX_VALUE);
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public float floatVal(int doc) {
|
public float floatVal(int doc) {
|
||||||
|
@ -84,7 +87,7 @@ public class DistanceValueSource extends ValueSource {
|
||||||
// make sure it has minX and area
|
// make sure it has minX and area
|
||||||
if (validX.get(doc)) {
|
if (validX.get(doc)) {
|
||||||
assert validY.get(doc);
|
assert validY.get(doc);
|
||||||
return calculator.distance(from, ptX.get(doc), ptY.get(doc));
|
return calculator.distance(from, ptX.get(doc), ptY.get(doc)) * multiplier;
|
||||||
}
|
}
|
||||||
return nullValue;
|
return nullValue;
|
||||||
}
|
}
|
||||||
|
@ -105,6 +108,7 @@ public class DistanceValueSource extends ValueSource {
|
||||||
|
|
||||||
if (!from.equals(that.from)) return false;
|
if (!from.equals(that.from)) return false;
|
||||||
if (!strategy.equals(that.strategy)) return false;
|
if (!strategy.equals(that.strategy)) return false;
|
||||||
|
if (multiplier != that.multiplier) return false;
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -120,8 +120,8 @@ public class PointVectorStrategy extends SpatialStrategy {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ValueSource makeDistanceValueSource(Point queryPoint) {
|
public ValueSource makeDistanceValueSource(Point queryPoint, double multiplier) {
|
||||||
return new DistanceValueSource(this, queryPoint);
|
return new DistanceValueSource(this, queryPoint, multiplier);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -34,6 +34,7 @@ import org.junit.Test;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.Arrays;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
public class DistanceStrategyTest extends StrategyTestCase {
|
public class DistanceStrategyTest extends StrategyTestCase {
|
||||||
|
@ -121,6 +122,11 @@ public class DistanceStrategyTest extends StrategyTestCase {
|
||||||
|
|
||||||
void checkDistValueSource(String ptStr, float... distances) throws IOException {
|
void checkDistValueSource(String ptStr, float... distances) throws IOException {
|
||||||
Point pt = (Point) ctx.readShape(ptStr);
|
Point pt = (Point) ctx.readShape(ptStr);
|
||||||
checkValueSource(strategy.makeDistanceValueSource(pt), distances, 1.0e-4f);
|
float multiplier = random().nextFloat() * 100f;
|
||||||
|
float[] dists2 = Arrays.copyOf(distances, distances.length);
|
||||||
|
for (int i = 0; i < dists2.length; i++) {
|
||||||
|
dists2[i] *= multiplier;
|
||||||
|
}
|
||||||
|
checkValueSource(strategy.makeDistanceValueSource(pt, multiplier), dists2, 1.0e-3f);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -162,7 +162,8 @@ public class SpatialExample extends LuceneTestCase {
|
||||||
//--Match all, order by distance ascending
|
//--Match all, order by distance ascending
|
||||||
{
|
{
|
||||||
Point pt = ctx.makePoint(60, -50);
|
Point pt = ctx.makePoint(60, -50);
|
||||||
ValueSource valueSource = strategy.makeDistanceValueSource(pt);//the distance (in degrees)
|
double degToKm = DistanceUtils.degrees2Dist(1, DistanceUtils.EARTH_MEAN_RADIUS_KM);
|
||||||
|
ValueSource valueSource = strategy.makeDistanceValueSource(pt, degToKm);//the distance (in km)
|
||||||
Sort distSort = new Sort(valueSource.getSortField(false)).rewrite(indexSearcher);//false=asc dist
|
Sort distSort = new Sort(valueSource.getSortField(false)).rewrite(indexSearcher);//false=asc dist
|
||||||
TopDocs docs = indexSearcher.search(new MatchAllDocsQuery(), 10, distSort);
|
TopDocs docs = indexSearcher.search(new MatchAllDocsQuery(), 10, distSort);
|
||||||
assertDocMatchedIds(indexSearcher, docs, 4, 20, 2);
|
assertDocMatchedIds(indexSearcher, docs, 4, 20, 2);
|
||||||
|
|
|
@ -240,12 +240,14 @@ public abstract class AbstractSpatialFieldType<T extends SpatialStrategy> extend
|
||||||
|
|
||||||
//We get the valueSource for the score then the filter and combine them.
|
//We get the valueSource for the score then the filter and combine them.
|
||||||
ValueSource valueSource;
|
ValueSource valueSource;
|
||||||
if ("distance".equals(score))
|
if ("distance".equals(score)) {
|
||||||
valueSource = strategy.makeDistanceValueSource(spatialArgs.getShape().getCenter());
|
double multiplier = 1.0;//TODO support units=kilometers
|
||||||
else if ("recipDistance".equals(score))
|
valueSource = strategy.makeDistanceValueSource(spatialArgs.getShape().getCenter(), multiplier);
|
||||||
|
} else if ("recipDistance".equals(score)) {
|
||||||
valueSource = strategy.makeRecipDistanceValueSource(spatialArgs.getShape());
|
valueSource = strategy.makeRecipDistanceValueSource(spatialArgs.getShape());
|
||||||
else
|
} else {
|
||||||
throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, "'score' local-param must be one of 'none', 'distance', or 'recipDistance'");
|
throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, "'score' local-param must be one of 'none', 'distance', or 'recipDistance'");
|
||||||
|
}
|
||||||
FunctionQuery functionQuery = new FunctionQuery(valueSource);
|
FunctionQuery functionQuery = new FunctionQuery(valueSource);
|
||||||
|
|
||||||
if (localParams != null && !localParams.getBool(FILTER_PARAM, true))
|
if (localParams != null && !localParams.getBool(FILTER_PARAM, true))
|
||||||
|
|
Loading…
Reference in New Issue