Skip to content

Commit

Permalink
fix geo predicate rule and geography isvalid (#3180)
Browse files Browse the repository at this point in the history
  • Loading branch information
jievince authored Oct 22, 2021
1 parent 17c04dd commit db6da98
Show file tree
Hide file tree
Showing 9 changed files with 67 additions and 10 deletions.
6 changes: 6 additions & 0 deletions src/common/datatypes/Geography.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,9 @@ bool LineString::isValid() const {
if (coordList.size() < 2) {
return false;
}
for (const auto& coord : coordList) {
if (!coord.isValid()) return false;
}
auto s2Region = geo::GeoUtils::s2RegionFromGeography(*this);
CHECK_NOTNULL(s2Region);
return static_cast<S2Polyline*>(s2Region.get())->IsValid();
Expand All @@ -71,6 +74,9 @@ bool Polygon::isValid() const {
if (coordList.front() != coordList.back()) {
return false;
}
for (const auto& coord : coordList) {
if (!coord.isValid()) return false;
}
}
auto s2Region = geo::GeoUtils::s2RegionFromGeography(*this);
CHECK_NOTNULL(s2Region);
Expand Down
16 changes: 11 additions & 5 deletions src/common/function/FunctionManager.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -368,11 +368,16 @@ std::unordered_map<std::string, std::vector<TypeSignature>> FunctionManager::typ
{
TypeSignature({Value::Type::GEOGRAPHY, Value::Type::GEOGRAPHY, Value::Type::FLOAT},
Value::Type::BOOL),
TypeSignature({Value::Type::GEOGRAPHY, Value::Type::GEOGRAPHY, Value::Type::INT},
Value::Type::BOOL),
TypeSignature({Value::Type::GEOGRAPHY,
Value::Type::GEOGRAPHY,
Value::Type::FLOAT,
Value::Type::BOOL},
Value::Type::BOOL),
TypeSignature(
{Value::Type::GEOGRAPHY, Value::Type::GEOGRAPHY, Value::Type::INT, Value::Type::BOOL},
Value::Type::BOOL),
}},
// geo measures
{"st_distance",
Expand Down Expand Up @@ -2497,7 +2502,7 @@ FunctionManager::FunctionManager() {
attr.isPure_ = true;
attr.body_ = [](const auto &args) -> Value {
if (!args[0].get().isGeography() || !args[1].get().isGeography() ||
!args[2].get().isFloat()) {
!args[2].get().isNumeric()) {
return Value::kNullBadType;
}
bool exclusive = false;
Expand All @@ -2507,10 +2512,11 @@ FunctionManager::FunctionManager() {
}
exclusive = args[3].get().getBool();
}
return geo::GeoFunction::dWithin(args[0].get().getGeography(),
args[1].get().getGeography(),
args[2].get().getFloat(),
exclusive);
return geo::GeoFunction::dWithin(
args[0].get().getGeography(),
args[1].get().getGeography(),
args[2].get().isFloat() ? args[2].get().getFloat() : args[2].get().getInt(),
exclusive);
};
}
// geo measures
Expand Down
4 changes: 4 additions & 0 deletions src/common/geo/GeoFunction.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -436,6 +436,10 @@ std::vector<uint64_t> GeoFunction::s2CoveringCellIds(
opts.set_max_cells(maxCells);

if (bufferInMeters == 0.0) {
if (a.shape() == GeoShape::POINT) {
const S2Point& gPoint = static_cast<const S2PointRegion*>(aRegion.get())->point();
return {S2CellId(gPoint).id()};
}
return coveringCellIds(*aRegion, opts);
}

Expand Down
2 changes: 1 addition & 1 deletion src/common/geo/io/wkb/test/WKBTest.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -82,7 +82,7 @@ TEST_F(WKBTest, TestWKB) {
LineString v(std::vector<Coordinate>{Coordinate(26.4, 78.9), Coordinate(138.725, 91.0)});
auto result = read(v);
ASSERT_TRUE(result.ok()) << result.status();
EXPECT_EQ(true, v.isValid());
EXPECT_EQ(false, v.isValid());
}
{
LineString v(std::vector<Coordinate>{Coordinate(0, 1), Coordinate(2, 3), Coordinate(0, 1)});
Expand Down
21 changes: 21 additions & 0 deletions src/common/geo/test/GeoFunctionTest.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1096,6 +1096,16 @@ TEST(isValid, lineString) {
bool b = line.isValid();
EXPECT_EQ(false, b);
}
{
auto line = Geography::fromWKT("LINESTRING(1.0 1.0, 181.0 2.0)").value();
bool b = line.isValid();
EXPECT_EQ(false, b);
}
{
auto line = Geography::fromWKT("LINESTRING(1.0 1.0, 1.0 90.001)").value();
bool b = line.isValid();
EXPECT_EQ(false, b);
}
}

TEST(isValid, polygon) {
Expand Down Expand Up @@ -1153,6 +1163,17 @@ TEST(isValid, polygon) {
// bool b = polygon.isValid();
// EXPECT_EQ(false, b); // Expect false, got true
// }
{
auto polygon =
Geography::fromWKT("POLYGON((1.0 1.0, -180.0001 2.0, 0.0 2.0, 1.0 1.0))").value();
bool b = polygon.isValid();
EXPECT_EQ(false, b);
}
{
auto polygon = Geography::fromWKT("POLYGON((1.0 1.0, 2.0 2.0, 0.0 -90.001, 1.0 1.0))").value();
bool b = polygon.isValid();
EXPECT_EQ(false, b);
}
}

} // namespace geo
Expand Down
2 changes: 1 addition & 1 deletion src/common/utils/IndexKeyUtils.h
Original file line number Diff line number Diff line change
Expand Up @@ -437,7 +437,7 @@ class IndexKeyUtils final {
break;
}
case Value::Type::GEOGRAPHY: {
// LOG(FATAL) << "unable to get geography value from index key"
// NOTE: The data read from index key is S2CellId which type is uint64, not wkb
len = sizeof(uint64_t);
break;
}
Expand Down
6 changes: 3 additions & 3 deletions src/graph/optimizer/rule/GeoPredicateIndexScanBaseRule.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -81,7 +81,7 @@ StatusOr<TransformResult> GeoPredicateIndexScanBaseRule::transform(
first->kind() == Expression::Kind::kEdgeProperty);
DCHECK(second->kind() == Expression::Kind::kConstant);
const auto& secondVal = static_cast<const ConstantExpression*>(second)->value();
DCHECK(secondVal.type() == Value::Type::GEOGRAPHY);
DCHECK(secondVal.isGeography());
const auto& geog = secondVal.getGeography();

// TODO(jie): Get index params from meta to construct RegionCoverParams
Expand All @@ -100,8 +100,8 @@ StatusOr<TransformResult> GeoPredicateIndexScanBaseRule::transform(
auto* third = geoPredicate->args()->args()[2];
DCHECK_EQ(third->kind(), Expression::Kind::kConstant);
const auto& thirdVal = static_cast<const ConstantExpression*>(third)->value();
DCHECK_EQ(thirdVal.type(), Value::Type::FLOAT);
double distanceInMeters = thirdVal.getFloat();
DCHECK(thirdVal.isNumeric());
double distanceInMeters = thirdVal.isFloat() ? thirdVal.getFloat() : thirdVal.getInt();
scanRanges = geoIndex.dWithin(geog, distanceInMeters);
}
std::vector<IndexQueryContext> idxCtxs;
Expand Down
2 changes: 2 additions & 0 deletions src/graph/validator/LookupValidator.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -224,6 +224,8 @@ Status LookupValidator::validateFilter() {
auto ret = checkFilter(filter);
NG_RETURN_IF_ERROR(ret);
lookupCtx_->filter = std::move(ret).value();
// Make sure the type of the rewritted filter expr is right
NG_RETURN_IF_ERROR(deduceExprType(lookupCtx_->filter));
}
NG_RETURN_IF_ERROR(deduceProps(lookupCtx_->filter, exprProps_));
return Status::OK();
Expand Down
18 changes: 18 additions & 0 deletions tests/tck/features/geo/GeoBase.feature
Original file line number Diff line number Diff line change
Expand Up @@ -420,6 +420,16 @@ Feature: Geo base
LOOKUP ON any_shape WHERE ST_Intersects(any_shape.geo, any_shape.geo) YIELD ST_ASText(any_shape.geo);
"""
Then a SemanticError should be raised at runtime: Expression ST_Intersects(any_shape.geo,any_shape.geo) not supported yet
When executing query:
"""
LOOKUP ON any_shape WHERE ST_Distance(any_shape.geo, ST_Point(3, 8.4)) < true YIELD ST_ASText(any_shape.geo);
"""
Then a SemanticError should be raised at runtime:
When executing query:
"""
LOOKUP ON any_shape WHERE ST_DWithin(any_shape.geo, ST_Point(3, 8.4), true) YIELD ST_ASText(any_shape.geo);
"""
Then a SemanticError should be raised at runtime:
# Match with geo predicate
When executing query:
"""
Expand Down Expand Up @@ -523,6 +533,14 @@ Feature: Geo base
| VertexID | ST_ASText(any_shape.geo) |
| "101" | "POINT(3 8)" |
| "102" | "LINESTRING(3 8, 4.7 73.23)" |
When executing query:
"""
LOOKUP ON any_shape WHERE ST_DWithin(any_shape.geo, ST_Point(3, 8), 100) YIELD ST_ASText(any_shape.geo);
"""
Then the result should be, in any order:
| VertexID | ST_ASText(any_shape.geo) |
| "101" | "POINT(3 8)" |
| "102" | "LINESTRING(3 8, 4.7 73.23)" |
# ST_Covers
When executing query:
"""
Expand Down

0 comments on commit db6da98

Please sign in to comment.