blob: 840d4137e7f5446fd22e87f14f4e63ff919d1c8b [file] [log] [blame]
Václav Kubernátdaf40312020-06-19 11:34:55 +02001#include <boost/algorithm/string/predicate.hpp>
Jan Kundrátbd3169c2020-02-03 19:31:34 +01002#include <cmath>
Václav Kubernát3c8fe022020-06-04 01:35:03 +02003#include "datastore_access.hpp"
4#include "libyang_utils.hpp"
5#include "utils.hpp"
Václav Kubernát02a71152020-01-21 14:52:51 +01006
7leaf_data_ leafValueFromValue(const libyang::S_Value& value, LY_DATA_TYPE type)
8{
9 using namespace std::string_literals;
10 switch (type) {
11 case LY_TYPE_INT8:
12 return value->int8();
13 case LY_TYPE_INT16:
14 return value->int16();
15 case LY_TYPE_INT32:
16 return value->int32();
17 case LY_TYPE_INT64:
18 return value->int64();
19 case LY_TYPE_UINT8:
20 return value->uint8();
21 case LY_TYPE_UINT16:
22 return value->uint16();
23 case LY_TYPE_UINT32:
Jan Kundrátcc2538f2020-02-03 11:33:42 +010024 return value->uint32();
Václav Kubernát02a71152020-01-21 14:52:51 +010025 case LY_TYPE_UINT64:
26 return value->uint64();
27 case LY_TYPE_BOOL:
28 return bool(value->bln());
29 case LY_TYPE_STRING:
30 return std::string(value->string());
31 case LY_TYPE_ENUM:
32 return enum_{std::string(value->enm()->name())};
Jan Kundrát0d8abd12020-05-07 02:00:14 +020033 case LY_TYPE_IDENT:
34 return identityRef_{value->ident()->module()->name(), value->ident()->name()};
Jan Kundrát4225b3f2020-01-24 12:58:38 +010035 case LY_TYPE_BINARY:
Jan Kundrát68985442020-05-07 02:15:34 +020036 return binary_{value->binary()};
Jan Kundrát379bb572020-05-07 03:23:13 +020037 case LY_TYPE_EMPTY:
38 return empty_{};
Václav Kubernát19097f32020-10-05 10:08:29 +020039 case LY_TYPE_BITS:
40 {
41 auto bits = value->bit();
42 std::vector<libyang::S_Type_Bit> filterNull;
43 std::copy_if(bits.begin(), bits.end(), std::back_inserter(filterNull), [] (auto bit) { return bit; });
44 bits_ res;
45 std::transform(filterNull.begin(), filterNull.end(), std::inserter(res.m_bits, res.m_bits.end()), [] (const auto& bit) { return bit->name(); });
46 return bits_{res};
47 }
Jan Kundrátbd3169c2020-02-03 19:31:34 +010048 case LY_TYPE_DEC64:
49 {
50 auto v = value->dec64();
51 return v.value * std::pow(10, -v.digits);
52 }
Václav Kubernát7b191ce2020-06-30 16:22:53 +020053 case LY_TYPE_LEAFREF:
54 {
55 libyang::Data_Node_Leaf_List toPrint{value->leafref()};
56 return leafValueFromValue(toPrint.value(), toPrint.value_type());
57 }
Václav Kubernát02a71152020-01-21 14:52:51 +010058 default: // TODO: implement all types
59 return "(can't print)"s;
60 }
61}
Václav Kubernát3c8fe022020-06-04 01:35:03 +020062
Václav Kubernátdaf40312020-06-19 11:34:55 +020063namespace {
64void impl_lyNodesToTree(DatastoreAccess::Tree& res, const std::vector<std::shared_ptr<libyang::Data_Node>> items, std::optional<std::string> ignoredXPathPrefix)
Václav Kubernát3c8fe022020-06-04 01:35:03 +020065{
66 auto stripXPathPrefix = [&ignoredXPathPrefix] (auto path) {
Václav Kubernáta8789602020-07-20 15:18:19 +020067 return ignoredXPathPrefix && path.find(*ignoredXPathPrefix) != std::string::npos ? path.substr(ignoredXPathPrefix->size()) : path;
Václav Kubernát3c8fe022020-06-04 01:35:03 +020068 };
69
70 for (const auto& it : items) {
Václav Kubernát3c8fe022020-06-04 01:35:03 +020071 if (it->schema()->nodetype() == LYS_CONTAINER) {
72 if (libyang::Schema_Node_Container{it->schema()}.presence()) {
73 // The fact that the container is included in the data tree
74 // means that it is present and I don't need to check any
75 // value.
76 res.emplace_back(stripXPathPrefix(it->path()), special_{SpecialValue::PresenceContainer});
77 }
78 }
79 if (it->schema()->nodetype() == LYS_LIST) {
Václav Kubernátfaacd022020-07-08 16:44:38 +020080 res.emplace_back(stripXPathPrefix(it->path()), special_{SpecialValue::List});
Václav Kubernát3c8fe022020-06-04 01:35:03 +020081 }
82 if (it->schema()->nodetype() == LYS_LEAF || it->schema()->nodetype() == LYS_LEAFLIST) {
83 libyang::Data_Node_Leaf_List leaf(it);
Václav Kubernát7b191ce2020-06-30 16:22:53 +020084 auto value = leafValueFromValue(leaf.value(), leaf.value_type());
Václav Kubernát3c8fe022020-06-04 01:35:03 +020085 res.emplace_back(stripXPathPrefix(it->path()), value);
86 }
87 }
88}
Václav Kubernátdaf40312020-06-19 11:34:55 +020089}
90
91// This is very similar to the fillMap lambda in SysrepoAccess, however,
92// Sysrepo returns a weird array-like structure, while libnetconf
93// returns libyang::Data_Node
94void lyNodesToTree(DatastoreAccess::Tree& res, const std::vector<std::shared_ptr<libyang::Data_Node>> items, std::optional<std::string> ignoredXPathPrefix)
95{
96 for (auto it = items.begin(); it < items.end(); it++) {
97 if ((*it)->schema()->nodetype() == LYS_LEAFLIST) {
98 auto leafListPath = stripLeafListValueFromPath((*it)->path());
99 res.emplace_back(leafListPath, special_{SpecialValue::LeafList});
100 while (it != items.end() && boost::starts_with((*it)->path(), leafListPath)) {
101 impl_lyNodesToTree(res, (*it)->tree_dfs(), ignoredXPathPrefix);
102 it++;
103 }
104 } else {
105 impl_lyNodesToTree(res, (*it)->tree_dfs(), ignoredXPathPrefix);
106 }
107 }
108}