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
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
|
// Copyright (C) 2017-2020 Internet Systems Consortium, Inc. ("ISC")
//
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
#include <config.h>
#include <testutils/user_context_utils.h>
using namespace isc::data;
namespace {
/// @brief Encapsulate either a modified copy or a unmodified value
/// @tparam EP ElementPtr or ConstElementPtr (compiler can't infer which one)
template<typename EP>
class Value {
public:
/// @brief Factory for modified copy
static Value mkCopy(EP value) { return (Value(value, false)); }
/// @brief Factory for unmodified original
static Value mkShare(EP value) { return (Value(value, true)); }
/// @brief Get the value
/// @return the value
EP get() const { return (value_); }
/// @brief Get the shared status
/// @return true if original, false if copy
bool isShared() const { return (shared_); }
private:
/// @brief Constructor
/// @param value the modified copy or unmodified value
/// @param shared true if original, false if copy
Value(EP value, bool shared) : value_(value), shared_(shared) { }
/// @brief the value
EP value_;
/// @brief the shared status
bool shared_;
};
/// @brief Recursive helper
///
/// @tparam EP ElementPtr or ConstElementPtr (compiler will infer which one)
/// @param element the element to traverse
/// @return a modified copy where comment entries were moved to user-context
/// or the unmodified original argument encapsulated into a Value
template<typename EP>
Value<EP> moveComments1(EP element) {
bool modified = false;
// On lists recurse on items
if (element->getType() == Element::list) {
ElementPtr result = ElementPtr(new ListElement());
typedef std::vector<ElementPtr> ListType;
const ListType& list = element->listValue();
for (ListType::const_iterator it = list.cbegin();
it != list.cend(); ++it) {
Value<ElementPtr> item = moveComments1(*it);
result->add(item.get());
if (!item.isShared()) {
modified = true;
}
}
if (!modified) {
return (Value<EP>::mkShare(element));
} else {
return (Value<EP>::mkCopy(result));
}
} else if (element->getType() != Element::map) {
return (Value<EP>::mkShare(element));
}
// Process maps: recurse on items
ElementPtr result = ElementPtr(new MapElement());
bool has_comment = false;
typedef std::map<std::string, ConstElementPtr> map_type;
const map_type& map = element->mapValue();
for (map_type::const_iterator it = map.cbegin(); it != map.cend(); ++it) {
if (it->first == "comment") {
// Note there is a comment entry to move
has_comment = true;
} else if (it->first == "user-context") {
// Do not traverse user-context entries
result->set("user-context", it->second);
} else {
// Not comment or user-context
Value<ConstElementPtr> item = moveComments1(it->second);
result->set(it->first, item.get());
if (!item.isShared()) {
modified = true;
}
}
}
// Check if the value should be not modified
if (!has_comment && !modified) {
return (Value<EP>::mkShare(element));
}
if (has_comment) {
// Move the comment entry
ConstElementPtr comment = element->get("comment");
ElementPtr moved = Element::createMap();
moved->set("comment", comment);
ConstElementPtr previous = element->get("user-context");
// If there is already a user context merge it
if (previous) {
merge(moved, previous);
}
result->set("user-context", moved);
}
return (Value<EP>::mkCopy(result));
}
} // anonymous namespace
namespace isc {
namespace test {
ElementPtr moveComments(ElementPtr element) {
Value<ElementPtr> result = moveComments1(element);
return (result.get());
}
ConstElementPtr moveComments(ConstElementPtr element) {
Value<ConstElementPtr> result = moveComments1(element);
return (result.get());
}
} // end of isc::test namespace
} // end of isc namespace
|