1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
|
#include "serialized_enum.h"
#include <library/cpp/testing/unittest/registar.h>
#include <util/generic/deque.h>
#include <util/generic/map.h>
#include <util/generic/typelist.h>
#include <util/generic/vector.h>
Y_UNIT_TEST_SUITE(TestSerializedEnum) {
Y_UNIT_TEST(RepresentationTypes) {
using namespace NEnumSerializationRuntime::NDetail;
static_assert(TIsPromotable<int, int>::value, "int -> int");
static_assert(TIsPromotable<char, int>::value, "char -> int");
static_assert(TIsPromotable<unsigned short, unsigned long>::value, "unsigned short -> unsigned long");
static_assert(TIsPromotable<i64, long long>::value, "i64 -> long long");
static_assert(!TIsPromotable<ui64, ui8>::value, "ui64 -> ui8");
static_assert(!TIsPromotable<i64, short>::value, "i64 -> short");
enum EEmpty {
};
UNIT_ASSERT_C((TTypeList<int, unsigned>::THave<typename TSelectEnumRepresentationType<EEmpty>::TType>::value), "empty enum using signed or unsigned integer underlying type");
using TRepresentationTypeList = TTypeList<int, unsigned, long long, unsigned long long>;
enum class ERegular {
One = 1,
Two = 2,
Five = 5,
};
UNIT_ASSERT(TRepresentationTypeList::THave<typename TSelectEnumRepresentationType<ERegular>::TType>::value);
enum class ESmall: unsigned char {
Six = 6,
};
UNIT_ASSERT(TRepresentationTypeList::THave<typename TSelectEnumRepresentationType<ESmall>::TType>::value);
enum class EHugeUnsigned: ui64 {
Value = 0,
};
UNIT_ASSERT(TRepresentationTypeList::THave<typename TSelectEnumRepresentationType<EHugeUnsigned>::TType>::value);
enum class EHugeSigned: i64 {
Value = -2,
};
UNIT_ASSERT(TRepresentationTypeList::THave<typename TSelectEnumRepresentationType<EHugeSigned>::TType>::value);
}
Y_UNIT_TEST(MappedArrayView) {
enum class ETestEnum: signed char {
Zero = 0,
One = 1,
Two = 2,
Three = 3,
Four = 4,
Eleven = 11,
};
const TVector<int> values = {1, 2, 3, 0, 0, 0, 11, 0, 0, 0, 0, 0, 2};
const auto view = ::NEnumSerializationRuntime::TMappedArrayView<ETestEnum>{values};
UNIT_ASSERT_VALUES_EQUAL(view.size(), values.size());
UNIT_ASSERT_VALUES_EQUAL(view.empty(), false);
UNIT_ASSERT_EQUAL(*view.begin(), ETestEnum::One);
UNIT_ASSERT_EQUAL(view[6], ETestEnum::Eleven);
UNIT_ASSERT_EXCEPTION(view.at(-1), std::out_of_range);
UNIT_ASSERT_VALUES_EQUAL(sizeof(view[4]), sizeof(signed char));
UNIT_ASSERT_VALUES_EQUAL(sizeof(values[4]), sizeof(int));
for (const ETestEnum e : view) {
UNIT_ASSERT_UNEQUAL(e, ETestEnum::Four);
}
const TVector<ETestEnum> typedValues = {ETestEnum::One, ETestEnum::Two, ETestEnum::Three, ETestEnum::Zero, ETestEnum::Zero, ETestEnum::Zero, ETestEnum::Eleven, ETestEnum::Zero, ETestEnum::Zero, ETestEnum::Zero, ETestEnum::Zero, ETestEnum::Zero, ETestEnum::Two};
UNIT_ASSERT_EQUAL(typedValues, view.Materialize());
const TDeque<ETestEnum> typedValuesDeque{typedValues.begin(), typedValues.end()};
UNIT_ASSERT_EQUAL(typedValuesDeque, view.Materialize<TDeque>());
}
Y_UNIT_TEST(MappedDictView) {
enum class ETestEnum: unsigned short {
Zero = 0,
One = 1,
Two = 2,
Three = 3,
Four = 4,
Eleven = 11,
Fake = (unsigned short)(-1),
};
const TMap<unsigned, unsigned> map = {{0, 1}, {1, 2}, {2, 4}, {3, 8}, {4, 16}, {11, 2048}};
const auto view = ::NEnumSerializationRuntime::NDetail::TMappedDictView<ETestEnum, unsigned, unsigned, decltype(map)>{map};
UNIT_ASSERT_VALUES_EQUAL(view.size(), map.size());
UNIT_ASSERT_VALUES_EQUAL(map.empty(), false);
UNIT_ASSERT_EQUAL(view.begin()->first, ETestEnum::Zero);
UNIT_ASSERT_VALUES_EQUAL(view.begin()->second, 1u);
UNIT_ASSERT_VALUES_EQUAL(view.contains(ETestEnum::Fake), false);
UNIT_ASSERT_VALUES_EQUAL(view.contains(ETestEnum::Four), true);
UNIT_ASSERT_EXCEPTION(view.at(ETestEnum::Fake), std::out_of_range);
UNIT_ASSERT_NO_EXCEPTION(view.at(ETestEnum::Eleven));
UNIT_ASSERT_VALUES_EQUAL(view.at(ETestEnum::Three), 8u);
unsigned mask = 0;
unsigned sum = 0;
for (const auto e : view) {
mask |= e.second;
sum += e.second;
}
UNIT_ASSERT_VALUES_EQUAL(mask, 2079);
UNIT_ASSERT_VALUES_EQUAL(sum, 2079);
const TMap<ETestEnum, unsigned> materialized = view.Materialize<TMap>();
UNIT_ASSERT_VALUES_EQUAL(materialized.size(), map.size());
UNIT_ASSERT_VALUES_EQUAL(materialized.at(ETestEnum::Four), 16);
}
}
|