// Licensed to the LF AI & Data foundation 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. #pragma once #include #include #include #include "common/Types.h" #include "exec/expression/EvalCtx.h" #include "exec/expression/VectorFunction.h" #include "exec/expression/Utils.h" #include "exec/QueryContext.h" #include "expr/ITypeExpr.h" #include "query/PlanProto.h" #include "ankerl/unordered_dense.h" namespace milvus { namespace exec { class BaseElement { public: virtual ~BaseElement() = default; }; class SingleElement : public BaseElement { public: using ValueType = std::variant; SingleElement() = default; virtual ~SingleElement() = default; template void SetValue(const proto::plan::GenericValue& value) { value_ = GetValueFromProto(value); } template void SetValue(const T& value) { if constexpr (std::is_same_v || std::is_same_v || std::is_same_v || std::is_same_v || std::is_same_v || std::is_same_v || std::is_same_v || std::is_same_v) { value_ = value; } else { static_assert(sizeof(T) == 0, "Type not supported in SingleElement"); } } template T GetValue() const { try { return std::get(value_); } catch (const std::bad_variant_access& e) { PanicInfo(ErrorCode::UnexpectedError, "SingleElement GetValue() failed: {}", e.what()); } } public: ValueType value_; }; class MultiElement : public BaseElement { public: using ValueType = std::variant; MultiElement() = default; virtual ~MultiElement() = default; virtual bool In(const ValueType& value) const = 0; virtual bool Empty() const = 0; virtual size_t Size() const = 0; }; template class SortVectorElement : public MultiElement { public: explicit SortVectorElement( const std::vector& values) { for (auto& value : values) { values_.push_back(GetValueFromProto(value)); } std::sort(values_.begin(), values_.end()); sorted_ = true; } explicit SortVectorElement(const std::vector& values) { for (const auto& value : values) { values_.push_back(value); } std::sort(values_.begin(), values_.end()); sorted_ = true; } bool Empty() const override { return values_.empty(); } size_t Size() const override { return values_.size(); } bool In(const ValueType& value) const override { AssertInfo(sorted_, "In() should be sorted before"); if (std::holds_alternative(value)) { return std::binary_search( values_.begin(), values_.end(), std::get(value)); } return false; } void Sort() { std::sort(values_.begin(), values_.end()); sorted_ = true; } void AddElement(const T& value) { values_.push_back(value); } public: std::vector values_; bool sorted_{false}; }; template class FlatVectorElement : public MultiElement { public: explicit FlatVectorElement( const std::vector& values) { for (auto& value : values) { values_.push_back(GetValueFromProto(value)); } } explicit FlatVectorElement(const std::vector& values) { for (const auto& value : values) { values_.push_back(value); } } bool Empty() const override { return values_.empty(); } bool In(const ValueType& value) const override { if (std::holds_alternative(value)) { for (const auto& v : values_) { if (v == std::get(value)) return true; } } return false; } size_t Size() const override { return values_.size(); } void AddElement(const T& value) { values_.push_back(value); } public: std::vector values_; }; template class SetElement : public MultiElement { public: explicit SetElement(const std::vector& values) { for (auto& value : values) { values_.insert(GetValueFromProto(value)); } } explicit SetElement(const std::vector& values) { for (const auto& value : values) { values_.insert(value); } } bool Empty() const override { return values_.empty(); } bool In(const ValueType& value) const override { if (std::holds_alternative(value)) { return values_.count(std::get(value)) > 0; } return false; } void AddElement(const T& value) { values_.insert(value); } size_t Size() const override { return values_.size(); } public: ankerl::unordered_dense::set values_; }; template <> class SetElement : public MultiElement { public: explicit SetElement(const std::vector& values) { for (auto& value : values) { bool v = GetValueFromProto(value); if (v) { contains_true = true; } else { contains_false = true; } } } explicit SetElement(const std::vector& values) { for (const auto& value : values) { if (value) { contains_true = true; } else { contains_false = true; } } } bool Empty() const override { return !contains_true && !contains_false; } bool In(const ValueType& value) const override { if (std::holds_alternative(value)) { bool v = std::get(value); return (v && contains_true) || (!v && contains_false); } return false; } void AddElement(const bool& value) { if (value) { contains_true = true; } else { contains_false = true; } } size_t Size() const override { return (contains_true ? 1 : 0) + (contains_false ? 1 : 0); } private: bool contains_true = false; bool contains_false = false; }; } //namespace exec } // namespace milvus