blob: abb241fb2fdd8d0033c3b1e2447baab3015c61d5 [file] [log] [blame] [edit]
/*
* Copyright (C) 2018 The Android Open Source Project
*
* Licensed 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.
*/
#ifndef LIBTEXTCLASSIFIER_UTILS_VARIANT_H_
#define LIBTEXTCLASSIFIER_UTILS_VARIANT_H_
#include <map>
#include <string>
#include <vector>
#include "utils/base/integral_types.h"
#include "utils/base/logging.h"
#include "utils/strings/stringpiece.h"
namespace libtextclassifier3 {
// Represents a type-tagged union of different basic types.
class Variant {
public:
enum Type {
TYPE_EMPTY = 0,
TYPE_INT8_VALUE = 1,
TYPE_UINT8_VALUE = 2,
TYPE_INT_VALUE = 3,
TYPE_UINT_VALUE = 4,
TYPE_INT64_VALUE = 5,
TYPE_UINT64_VALUE = 6,
TYPE_FLOAT_VALUE = 7,
TYPE_DOUBLE_VALUE = 8,
TYPE_BOOL_VALUE = 9,
TYPE_STRING_VALUE = 10,
TYPE_STRING_VECTOR_VALUE = 11,
TYPE_FLOAT_VECTOR_VALUE = 12,
TYPE_INT_VECTOR_VALUE = 13,
TYPE_STRING_VARIANT_MAP_VALUE = 14,
};
Variant() : type_(TYPE_EMPTY) {}
explicit Variant(const int8_t value)
: type_(TYPE_INT8_VALUE), int8_value_(value) {}
explicit Variant(const uint8_t value)
: type_(TYPE_UINT8_VALUE), uint8_value_(value) {}
explicit Variant(const int value)
: type_(TYPE_INT_VALUE), int_value_(value) {}
explicit Variant(const uint value)
: type_(TYPE_UINT_VALUE), uint_value_(value) {}
explicit Variant(const int64 value)
: type_(TYPE_INT64_VALUE), long_value_(value) {}
explicit Variant(const uint64 value)
: type_(TYPE_UINT64_VALUE), ulong_value_(value) {}
explicit Variant(const float value)
: type_(TYPE_FLOAT_VALUE), float_value_(value) {}
explicit Variant(const double value)
: type_(TYPE_DOUBLE_VALUE), double_value_(value) {}
explicit Variant(const StringPiece value)
: type_(TYPE_STRING_VALUE), string_value_(value.ToString()) {}
explicit Variant(const std::string value)
: type_(TYPE_STRING_VALUE), string_value_(value) {}
explicit Variant(const char* value)
: type_(TYPE_STRING_VALUE), string_value_(value) {}
explicit Variant(const bool value)
: type_(TYPE_BOOL_VALUE), bool_value_(value) {}
explicit Variant(const std::vector<std::string>& value)
: type_(TYPE_STRING_VECTOR_VALUE), string_vector_value_(value) {}
explicit Variant(const std::vector<float>& value)
: type_(TYPE_FLOAT_VECTOR_VALUE), float_vector_value_(value) {}
explicit Variant(const std::vector<int>& value)
: type_(TYPE_INT_VECTOR_VALUE), int_vector_value_(value) {}
explicit Variant(const std::map<std::string, Variant>& value)
: type_(TYPE_STRING_VARIANT_MAP_VALUE),
string_variant_map_value_(value) {}
Variant(const Variant&) = default;
Variant& operator=(const Variant&) = default;
template <class T>
struct dependent_false : std::false_type {};
template <typename T>
T Value() const {
static_assert(dependent_false<T>::value, "Not supported.");
}
template <>
int8 Value() const {
TC3_CHECK(Has<int8>());
return int8_value_;
}
template <>
uint8 Value() const {
TC3_CHECK(Has<uint8>());
return uint8_value_;
}
template <>
int Value() const {
TC3_CHECK(Has<int>());
return int_value_;
}
template <>
uint Value() const {
TC3_CHECK(Has<uint>());
return uint_value_;
}
template <>
int64 Value() const {
TC3_CHECK(Has<int64>());
return long_value_;
}
template <>
uint64 Value() const {
TC3_CHECK(Has<uint64>());
return ulong_value_;
}
template <>
float Value() const {
TC3_CHECK(Has<float>());
return float_value_;
}
template <>
double Value() const {
TC3_CHECK(Has<double>());
return double_value_;
}
template <>
bool Value() const {
TC3_CHECK(Has<bool>());
return bool_value_;
}
template <typename T>
const T& ConstRefValue() const;
template <>
const std::string& ConstRefValue() const {
TC3_CHECK(Has<std::string>());
return string_value_;
}
template <>
const std::vector<std::string>& ConstRefValue() const {
TC3_CHECK(Has<std::vector<std::string>>());
return string_vector_value_;
}
template <>
const std::vector<float>& ConstRefValue() const {
TC3_CHECK(Has<std::vector<float>>());
return float_vector_value_;
}
template <>
const std::vector<int>& ConstRefValue() const {
TC3_CHECK(Has<std::vector<int>>());
return int_vector_value_;
}
template <>
const std::map<std::string, Variant>& ConstRefValue() const {
TC3_CHECK((Has<std::map<std::string, Variant>>()));
return string_variant_map_value_;
}
template <typename T>
bool Has() const;
template <>
bool Has<int8>() const {
return type_ == TYPE_INT8_VALUE;
}
template <>
bool Has<uint8>() const {
return type_ == TYPE_UINT8_VALUE;
}
template <>
bool Has<int>() const {
return type_ == TYPE_INT_VALUE;
}
template <>
bool Has<uint>() const {
return type_ == TYPE_UINT_VALUE;
}
template <>
bool Has<int64>() const {
return type_ == TYPE_INT64_VALUE;
}
template <>
bool Has<uint64>() const {
return type_ == TYPE_UINT64_VALUE;
}
template <>
bool Has<float>() const {
return type_ == TYPE_FLOAT_VALUE;
}
template <>
bool Has<double>() const {
return type_ == TYPE_DOUBLE_VALUE;
}
template <>
bool Has<bool>() const {
return type_ == TYPE_BOOL_VALUE;
}
template <>
bool Has<std::string>() const {
return type_ == TYPE_STRING_VALUE;
}
template <>
bool Has<std::vector<std::string>>() const {
return type_ == TYPE_STRING_VECTOR_VALUE;
}
template <>
bool Has<std::vector<float>>() const {
return type_ == TYPE_FLOAT_VECTOR_VALUE;
}
template <>
bool Has<std::vector<int>>() const {
return type_ == TYPE_INT_VECTOR_VALUE;
}
template <>
bool Has<std::map<std::string, Variant>>() const {
return type_ == TYPE_STRING_VARIANT_MAP_VALUE;
}
// Converts the value of this variant to its string representation, regardless
// of the type of the actual value.
std::string ToString() const;
Type GetType() const { return type_; }
bool HasValue() const { return type_ != TYPE_EMPTY; }
private:
Type type_;
union {
int8_t int8_value_;
uint8_t uint8_value_;
int int_value_;
uint uint_value_;
int64 long_value_;
uint64 ulong_value_;
float float_value_;
double double_value_;
bool bool_value_;
};
std::string string_value_;
std::vector<std::string> string_vector_value_;
std::vector<float> float_vector_value_;
std::vector<int> int_vector_value_;
std::map<std::string, Variant> string_variant_map_value_;
};
// Pretty-printing function for Variant.
logging::LoggingStringStream& operator<<(logging::LoggingStringStream& stream,
const Variant& value);
} // namespace libtextclassifier3
#endif // LIBTEXTCLASSIFIER_UTILS_VARIANT_H_