LUCENE-7189: improve geo debugging technology

This commit is contained in:
Mike McCandless 2016-04-12 06:10:08 -04:00
parent a3ee984bc8
commit e034b04b06
2 changed files with 108 additions and 14 deletions

View File

@ -17,8 +17,6 @@ package org.apache.lucene.geo;
* limitations under the License. * limitations under the License.
*/ */
import org.apache.lucene.geo.Polygon;
import org.apache.lucene.geo.Rectangle;
import org.apache.lucene.util.SloppyMath; import org.apache.lucene.util.SloppyMath;
/** Draws shapes on the earth surface and renders using the very cool http://www.webglearth.org. /** Draws shapes on the earth surface and renders using the very cool http://www.webglearth.org.
@ -40,6 +38,16 @@ public class EarthDebugger {
b.append(" var earth = new WE.map('earth_div');\n"); b.append(" var earth = new WE.map('earth_div');\n");
} }
public EarthDebugger(double centerLat, double centerLon, double altitudeMeters) {
b.append("<!DOCTYPE HTML>\n");
b.append("<html>\n");
b.append(" <head>\n");
b.append(" <script src=\"http://www.webglearth.com/v2/api.js\"></script>\n");
b.append(" <script>\n");
b.append(" function initialize() {\n");
b.append(" var earth = new WE.map('earth_div', {center: [" + centerLat + ", " + centerLon + "], altitude: " + altitudeMeters + "});\n");
}
public void addPolygon(Polygon poly) { public void addPolygon(Polygon poly) {
addPolygon(poly, "#00ff00"); addPolygon(poly, "#00ff00");
} }
@ -54,29 +62,36 @@ public class EarthDebugger {
for(int i=0;i<polyLats.length;i++) { for(int i=0;i<polyLats.length;i++) {
b.append(" [" + polyLats[i] + ", " + polyLons[i] + "],\n"); b.append(" [" + polyLats[i] + ", " + polyLons[i] + "],\n");
} }
b.append(" ], {color: '" + color + "'});\n"); b.append(" ], {color: '" + color + "', fillColor: \"#000000\", fillOpacity: 0.0001});\n");
b.append(" " + name + ".addTo(earth);\n"); b.append(" " + name + ".addTo(earth);\n");
for (Polygon hole : poly.getHoles()) { for (Polygon hole : poly.getHoles()) {
addPolygon(poly, "#ffffff"); addPolygon(hole, "#ffffff");
} }
} }
private static double MAX_LAT_LON_PER_STEP = 5.0; private static double MAX_KM_PER_STEP = 100.0;
// Web GL earth connects dots by tunneling under the earth, so we approximate a great circle by sampling it, to minimize how deep in the
// earth each segment tunnels:
private int getStepCount(double minLat, double maxLat, double minLon, double maxLon) {
double distanceMeters = SloppyMath.haversinMeters(minLat, minLon, maxLat, maxLon);
return Math.max(1, (int) Math.round((distanceMeters / 1000.0) / MAX_KM_PER_STEP));
}
// first point is inclusive, last point is exclusive! // first point is inclusive, last point is exclusive!
private void drawSegment(double minLat, double maxLat, double minLon, double maxLon) { private void drawSegment(double minLat, double maxLat, double minLon, double maxLon) {
int steps = (int) Math.round(Math.max(Math.abs(maxLat-minLat)/MAX_LAT_LON_PER_STEP, Math.abs(maxLon-minLon)/MAX_LAT_LON_PER_STEP)); int steps = getStepCount(minLat, maxLat, minLon, maxLon);
if (steps < 1) {
steps = 1;
}
for(int i=0;i<steps;i++) { for(int i=0;i<steps;i++) {
b.append(" [" + (minLat + (maxLat - minLat) * i / steps) + ", " + (minLon + (maxLon - minLon) * i / steps) + "],\n"); b.append(" [" + (minLat + (maxLat - minLat) * i / steps) + ", " + (minLon + (maxLon - minLon) * i / steps) + "],\n");
} }
} }
public void addRect(double minLat, double maxLat, double minLon, double maxLon) { public void addRect(double minLat, double maxLat, double minLon, double maxLon) {
int steps = 20; addRect(minLat, maxLat, minLon, maxLon, "#ff0000");
}
public void addRect(double minLat, double maxLat, double minLon, double maxLon, String color) {
String name = "rect" + nextShape; String name = "rect" + nextShape;
nextShape++; nextShape++;
@ -97,7 +112,7 @@ public class EarthDebugger {
b.append(" // min lat, min lon\n"); b.append(" // min lat, min lon\n");
b.append(" [" + minLat + ", " + minLon + "]\n"); b.append(" [" + minLat + ", " + minLon + "]\n");
b.append(" ], {color: \"#ff0000\", fillColor: \"#ff0000\"});\n"); b.append(" ], {color: \"" + color + "\", fillColor: \"" + color + "\"});\n");
b.append(" " + name + ".addTo(earth);\n"); b.append(" " + name + ".addTo(earth);\n");
} }
@ -108,7 +123,7 @@ public class EarthDebugger {
b.append(" var " + name + " = WE.polygon([\n"); b.append(" var " + name + " = WE.polygon([\n");
double lon; double lon;
int steps = 36; int steps = getStepCount(lat, minLon, lat, maxLon);
for(lon = minLon;lon<=maxLon;lon += (maxLon-minLon)/steps) { for(lon = minLon;lon<=maxLon;lon += (maxLon-minLon)/steps) {
b.append(" [" + lat + ", " + lon + "],\n"); b.append(" [" + lat + ", " + lon + "],\n");
} }
@ -128,7 +143,7 @@ public class EarthDebugger {
b.append(" var " + name + " = WE.polygon([\n"); b.append(" var " + name + " = WE.polygon([\n");
double lat; double lat;
int steps = 36; int steps = getStepCount(minLat, lon, maxLat, lon);
for(lat = minLat;lat<=maxLat;lat += (maxLat-minLat)/steps) { for(lat = minLat;lat<=maxLat;lat += (maxLat-minLat)/steps) {
b.append(" [" + lat + ", " + lon + "],\n"); b.append(" [" + lat + ", " + lon + "],\n");
} }
@ -151,7 +166,7 @@ public class EarthDebugger {
nextShape++; nextShape++;
b.append(" var " + name + " = WE.polygon([\n"); b.append(" var " + name + " = WE.polygon([\n");
inverseHaversin(b, centerLat, centerLon, radiusMeters); inverseHaversin(b, centerLat, centerLon, radiusMeters);
b.append(" ], {color: '#00ff00'});\n"); b.append(" ], {color: '#00ff00', fillColor: \"#000000\", fillOpacity: 0.0001 });\n");
b.append(" " + name + ".addTo(earth);\n"); b.append(" " + name + ".addTo(earth);\n");
if (alsoAddBBox) { if (alsoAddBBox) {

View File

@ -0,0 +1,79 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.lucene.index;
import java.util.ArrayList;
import java.util.List;
import org.apache.lucene.index.PointValues.IntersectVisitor;
import org.apache.lucene.util.StringHelper;
/** Simple utility class to track the current BKD stack based solely on calls to {@link IntersectVisitor#compare}. */
public class PointsStackTracker {
private final int numDims;
private final int bytesPerDim;
public final List<Cell> stack = new ArrayList<>();
public class Cell {
public final byte[] minPackedValue;
public final byte[] maxPackedValue;
public Cell(byte[] minPackedValue, byte[] maxPackedValue) {
this.minPackedValue = minPackedValue.clone();
this.maxPackedValue = maxPackedValue.clone();
}
public boolean contains(Cell other) {
for(int dim=0;dim<numDims;dim++) {
int offset = dim * bytesPerDim;
// other.min < min?
if (StringHelper.compare(bytesPerDim, other.minPackedValue, offset, minPackedValue, offset) < 0) {
return false;
}
// other.max > max?
if (StringHelper.compare(bytesPerDim, other.maxPackedValue, offset, maxPackedValue, offset) > 0) {
return false;
}
}
return true;
}
}
public PointsStackTracker(int numDims, int bytesPerDim) {
this.numDims = numDims;
this.bytesPerDim = bytesPerDim;
}
public void onCompare(byte[] minPackedValue, byte[] maxPackedValue) {
Cell cell = new Cell(minPackedValue, maxPackedValue);
// Pop stack:
while (stack.size() > 0 && stack.get(stack.size()-1).contains(cell) == false) {
stack.remove(stack.size()-1);
//System.out.println(" pop");
}
// Push stack:
stack.add(cell);
}
// TODO: expose other details about the stack...
}