From 3353c6c8b8fcd666b469ba266edc3dcb5d93ddf3 Mon Sep 17 00:00:00 2001 From: Asheem Mamoowala Date: Fri, 29 Jun 2018 12:56:44 -0700 Subject: Keep original input for filters using legacy syntax. --- include/mbgl/style/filter.hpp | 20 +++++++++++-- src/mbgl/style/conversion/filter.cpp | 21 +++++++++++++- test/style/filter.test.cpp | 55 ++++++++++++++++++++++++++++++++++++ 3 files changed, 92 insertions(+), 4 deletions(-) diff --git a/include/mbgl/style/filter.hpp b/include/mbgl/style/filter.hpp index ce4015bb69..c9dc9fb1ec 100644 --- a/include/mbgl/style/filter.hpp +++ b/include/mbgl/style/filter.hpp @@ -11,14 +11,18 @@ namespace mbgl { namespace style { - + class Filter { public: optional> expression; - +private: + optional legacyFilter; +public: Filter() : expression() {} - Filter(expression::ParseResult _expression) : expression(std::move(*_expression)) { + Filter(expression::ParseResult _expression, optional _filter = {}) + : expression(std::move(*_expression)), + legacyFilter(std::move(_filter)){ assert(!expression || *expression != nullptr); } @@ -35,6 +39,16 @@ public: friend bool operator!=(const Filter& lhs, const Filter& rhs) { return !(lhs == rhs); } + + mbgl::Value serialize() const { + if (legacyFilter) { + return *legacyFilter; + } + else if (expression) { + return (**expression).serialize(); + } + return NullValue(); + } }; } // namespace style diff --git a/src/mbgl/style/conversion/filter.cpp b/src/mbgl/style/conversion/filter.cpp index cd588345ef..b690c96388 100644 --- a/src/mbgl/style/conversion/filter.cpp +++ b/src/mbgl/style/conversion/filter.cpp @@ -14,6 +14,7 @@ using namespace mbgl::style::expression; static bool isExpression(const Convertible& filter); ParseResult convertLegacyFilter(const Convertible& values, Error& error); +optional serializeLegacyFilter(const Convertible& values); optional Converter::operator()(const Convertible& value, Error& error) const { if (isExpression(value)) { @@ -31,7 +32,7 @@ optional Converter::operator()(const Convertible& value, Error& assert(error.message.size() > 0); return {}; } - return Filter(optional>(std::move(*expression))); + return Filter(optional>(std::move(*expression)), serializeLegacyFilter(value)); } } @@ -218,6 +219,24 @@ ParseResult convertLegacyFilter(const Convertible& values, Error& error) { } } +optional serializeLegacyFilter(const Convertible& values) { + if (isUndefined(values)) { + return {}; + } else if (isArray(values)) { + std::vector result; + for (std::size_t i = 0; i < arrayLength(values); i++) { + auto arrayValue = serializeLegacyFilter(arrayMember(values, i)); + if (arrayValue) { + result.push_back(*arrayValue); + } else { + result.push_back(NullValue()); + } + } + return (mbgl::Value)result; + } + return toValue(values); +} + } // namespace conversion } // namespace style } // namespace mbgl diff --git a/test/style/filter.test.cpp b/test/style/filter.test.cpp index bac3801223..e04a569203 100644 --- a/test/style/filter.test.cpp +++ b/test/style/filter.test.cpp @@ -11,6 +11,10 @@ #include #include #include +#include + +#include +#include using namespace mbgl; using namespace mbgl::style; @@ -39,6 +43,42 @@ void invalidFilter(const char * json) { EXPECT_NE(error.message, ""); } +void writeJSON(rapidjson::Writer& writer, const Value& value) { + value.match( + [&] (const NullValue&) { writer.Null(); }, + [&] (bool b) { writer.Bool(b); }, + [&] (uint64_t t) { writer.Uint64(t); }, + [&] (int64_t t) { writer.Int64(t); }, + [&] (double f) { + // make sure integer values are stringified without trailing ".0". + f == std::floor(f) ? writer.Int(f) : writer.Double(f); + }, + [&] (const std::string& s) { writer.String(s); }, + [&] (const std::vector& arr) { + writer.StartArray(); + for(const auto& item : arr) { + writeJSON(writer, item); + } + writer.EndArray(); + }, + [](const auto&) { + }); +} + +std::string stringifyFilter(const char * json) { + conversion::Error error; + optional filter = conversion::convertJSON(json, error); + EXPECT_TRUE(bool(filter)); + EXPECT_EQ(error.message, ""); + + auto value = filter->serialize(); + + rapidjson::StringBuffer s; + rapidjson::Writer writer(s); + writeJSON(writer, value); + return s.GetString(); +} + TEST(Filter, EqualsNull) { auto f = R"(["==", "foo", null])"; ASSERT_TRUE(filter(f, {{ "foo", mapbox::geometry::null_value }})); @@ -181,6 +221,21 @@ TEST(Filter, LegacyProperty) { ASSERT_FALSE(filter("[\"==\", \"two\", 4]", {{"two", std::string("2")}})); } +TEST(Filter, Serialize) { + std::string json = R"(["any",["==","foo",0],["==","foo",1]])"; + EXPECT_EQ(stringifyFilter(json.c_str()), std::string(json)); + + json = R"(["<=","two",2])"; + EXPECT_EQ(stringifyFilter(json.c_str()), std::string(json)); + + //Constant folding for Expression filters + json = R"(["==",["+",1,1],2])"; + EXPECT_EQ(stringifyFilter(json.c_str()), std::string("true")); + + json = R"(["all",["==",["get","foo"],0],["==",["get","foo"],1]])"; + EXPECT_EQ(stringifyFilter(json.c_str()), std::string(json)); +} + TEST(Filter, ExpressionLegacyMix) { conversion::Error error; optional filter = conversion::convertJSON(R"(["any", ["all", ["==", ["geometry-type"], "LineString"]], ["==", "x", 1]])", error); -- cgit v1.2.1