#include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace DB { namespace ErrorCodes { extern const int BAD_ARGUMENTS; } namespace { template class FunctionPolygonConvexHull : public IFunction { public: static const char * name; explicit FunctionPolygonConvexHull() = default; static FunctionPtr create(ContextPtr) { return std::make_shared(); } String getName() const override { return name; } bool isVariadic() const override { return false; } bool isSuitableForShortCircuitArgumentsExecution(const DataTypesWithConstInfo & /*arguments*/) const override { return true; } size_t getNumberOfArguments() const override { return 1; } DataTypePtr getReturnTypeImpl(const DataTypes &) const override { return DataTypeFactory::instance().get("Polygon"); } ColumnPtr executeImpl(const ColumnsWithTypeAndName & arguments, const DataTypePtr & /*result_type*/, size_t input_rows_count) const override { PolygonSerializer serializer; callOnGeometryDataType(arguments[0].type, [&] (const auto & type) { using TypeConverter = std::decay_t; using Converter = typename TypeConverter::Type; if constexpr (std::is_same_v>) throw Exception(ErrorCodes::BAD_ARGUMENTS, "The argument of function {} must not be a Point", getName()); else { auto geometries = Converter::convert(arguments[0].column->convertToFullColumnIfConst()); for (size_t i = 0; i < input_rows_count; ++i) { Polygon convex_hull{}; boost::geometry::convex_hull(geometries[i], convex_hull); serializer.add(convex_hull); } } } ); return serializer.finalize(); } bool useDefaultImplementationForConstants() const override { return true; } }; template <> const char * FunctionPolygonConvexHull::name = "polygonConvexHullCartesian"; } REGISTER_FUNCTION(PolygonConvexHull) { factory.registerFunction>(); } }