Václav Kubernát | daf4031 | 2020-06-19 11:34:55 +0200 | [diff] [blame] | 1 | #include <boost/algorithm/string/predicate.hpp> |
Jan Kundrát | bd3169c | 2020-02-03 19:31:34 +0100 | [diff] [blame] | 2 | #include <cmath> |
Václav Kubernát | 3c8fe02 | 2020-06-04 01:35:03 +0200 | [diff] [blame] | 3 | #include "datastore_access.hpp" |
| 4 | #include "libyang_utils.hpp" |
| 5 | #include "utils.hpp" |
Václav Kubernát | 02a7115 | 2020-01-21 14:52:51 +0100 | [diff] [blame] | 6 | |
Václav Kubernát | 2e4cafe | 2020-11-05 01:53:21 +0100 | [diff] [blame] | 7 | leaf_data_ leafValueFromNode(libyang::S_Data_Node_Leaf_List node) |
Václav Kubernát | 02a7115 | 2020-01-21 14:52:51 +0100 | [diff] [blame] | 8 | { |
Václav Kubernát | b4e5b18 | 2020-11-16 19:55:09 +0100 | [diff] [blame] | 9 | std::function<leaf_data_(libyang::S_Data_Node_Leaf_List)> impl = [&impl](libyang::S_Data_Node_Leaf_List node) -> leaf_data_ { |
Václav Kubernát | 2e4cafe | 2020-11-05 01:53:21 +0100 | [diff] [blame] | 10 | // value_type() is what's ACTUALLY stored inside `node` |
| 11 | // Leafrefs sometimes don't hold a reference to another, but they have the actual pointed-to value. |
| 12 | switch (node->value_type()) { |
| 13 | case LY_TYPE_ENUM: |
| 14 | return enum_{node->value()->enm()->name()}; |
| 15 | case LY_TYPE_UINT8: |
| 16 | return node->value()->uint8(); |
| 17 | case LY_TYPE_UINT16: |
| 18 | return node->value()->uint16(); |
| 19 | case LY_TYPE_UINT32: |
| 20 | return node->value()->uint32(); |
| 21 | case LY_TYPE_UINT64: |
| 22 | return node->value()->uint64(); |
| 23 | case LY_TYPE_INT8: |
| 24 | return node->value()->int8(); |
| 25 | case LY_TYPE_INT16: |
| 26 | return node->value()->int16(); |
| 27 | case LY_TYPE_INT32: |
| 28 | return node->value()->int32(); |
| 29 | case LY_TYPE_INT64: |
| 30 | return node->value()->int64(); |
Václav Kubernát | b4e5b18 | 2020-11-16 19:55:09 +0100 | [diff] [blame] | 31 | case LY_TYPE_DEC64: { |
Václav Kubernát | 2e4cafe | 2020-11-05 01:53:21 +0100 | [diff] [blame] | 32 | auto v = node->value()->dec64(); |
| 33 | return v.value * std::pow(10, -v.digits); |
| 34 | } |
| 35 | case LY_TYPE_BOOL: |
| 36 | return node->value()->bln(); |
| 37 | case LY_TYPE_STRING: |
| 38 | return std::string{node->value()->string()}; |
| 39 | case LY_TYPE_BINARY: |
| 40 | return binary_{node->value()->binary()}; |
| 41 | case LY_TYPE_IDENT: |
| 42 | return identityRef_{node->value()->ident()->module()->name(), node->value()->ident()->name()}; |
| 43 | case LY_TYPE_EMPTY: |
| 44 | return empty_{}; |
Václav Kubernát | b4e5b18 | 2020-11-16 19:55:09 +0100 | [diff] [blame] | 45 | case LY_TYPE_LEAFREF: { |
Václav Kubernát | 2e4cafe | 2020-11-05 01:53:21 +0100 | [diff] [blame] | 46 | auto refsTo = node->value()->leafref(); |
| 47 | assert(refsTo); |
| 48 | return impl(std::make_shared<libyang::Data_Node_Leaf_List>(node->value()->leafref())); |
| 49 | } |
Václav Kubernát | b4e5b18 | 2020-11-16 19:55:09 +0100 | [diff] [blame] | 50 | case LY_TYPE_BITS: { |
Václav Kubernát | 2e4cafe | 2020-11-05 01:53:21 +0100 | [diff] [blame] | 51 | auto bits = node->value()->bit(); |
| 52 | std::vector<libyang::S_Type_Bit> filterNull; |
Václav Kubernát | b4e5b18 | 2020-11-16 19:55:09 +0100 | [diff] [blame] | 53 | std::copy_if(bits.begin(), bits.end(), std::back_inserter(filterNull), [](auto bit) { return bit; }); |
Václav Kubernát | 2e4cafe | 2020-11-05 01:53:21 +0100 | [diff] [blame] | 54 | bits_ res; |
Václav Kubernát | b4e5b18 | 2020-11-16 19:55:09 +0100 | [diff] [blame] | 55 | std::transform(filterNull.begin(), filterNull.end(), std::inserter(res.m_bits, res.m_bits.end()), [](const auto& bit) { return bit->name(); }); |
Václav Kubernát | 2e4cafe | 2020-11-05 01:53:21 +0100 | [diff] [blame] | 56 | return bits_{res}; |
| 57 | } |
| 58 | default: |
| 59 | return std::string{"(can't print)"}; |
Václav Kubernát | b4e5b18 | 2020-11-16 19:55:09 +0100 | [diff] [blame] | 60 | } |
Václav Kubernát | 2e4cafe | 2020-11-05 01:53:21 +0100 | [diff] [blame] | 61 | }; |
| 62 | return impl(node); |
Václav Kubernát | 02a7115 | 2020-01-21 14:52:51 +0100 | [diff] [blame] | 63 | } |
Václav Kubernát | 3c8fe02 | 2020-06-04 01:35:03 +0200 | [diff] [blame] | 64 | |
Václav Kubernát | daf4031 | 2020-06-19 11:34:55 +0200 | [diff] [blame] | 65 | namespace { |
| 66 | void impl_lyNodesToTree(DatastoreAccess::Tree& res, const std::vector<std::shared_ptr<libyang::Data_Node>> items, std::optional<std::string> ignoredXPathPrefix) |
Václav Kubernát | 3c8fe02 | 2020-06-04 01:35:03 +0200 | [diff] [blame] | 67 | { |
Václav Kubernát | b4e5b18 | 2020-11-16 19:55:09 +0100 | [diff] [blame] | 68 | auto stripXPathPrefix = [&ignoredXPathPrefix](auto path) { |
Václav Kubernát | a878960 | 2020-07-20 15:18:19 +0200 | [diff] [blame] | 69 | return ignoredXPathPrefix && path.find(*ignoredXPathPrefix) != std::string::npos ? path.substr(ignoredXPathPrefix->size()) : path; |
Václav Kubernát | 3c8fe02 | 2020-06-04 01:35:03 +0200 | [diff] [blame] | 70 | }; |
| 71 | |
| 72 | for (const auto& it : items) { |
Václav Kubernát | 3c8fe02 | 2020-06-04 01:35:03 +0200 | [diff] [blame] | 73 | if (it->schema()->nodetype() == LYS_CONTAINER) { |
| 74 | if (libyang::Schema_Node_Container{it->schema()}.presence()) { |
| 75 | // The fact that the container is included in the data tree |
| 76 | // means that it is present and I don't need to check any |
| 77 | // value. |
| 78 | res.emplace_back(stripXPathPrefix(it->path()), special_{SpecialValue::PresenceContainer}); |
| 79 | } |
| 80 | } |
| 81 | if (it->schema()->nodetype() == LYS_LIST) { |
Václav Kubernát | faacd02 | 2020-07-08 16:44:38 +0200 | [diff] [blame] | 82 | res.emplace_back(stripXPathPrefix(it->path()), special_{SpecialValue::List}); |
Václav Kubernát | 3c8fe02 | 2020-06-04 01:35:03 +0200 | [diff] [blame] | 83 | } |
| 84 | if (it->schema()->nodetype() == LYS_LEAF || it->schema()->nodetype() == LYS_LEAFLIST) { |
Václav Kubernát | 2e4cafe | 2020-11-05 01:53:21 +0100 | [diff] [blame] | 85 | auto leaf = std::make_shared<libyang::Data_Node_Leaf_List>(it); |
| 86 | auto value = leafValueFromNode(leaf); |
Václav Kubernát | 3c8fe02 | 2020-06-04 01:35:03 +0200 | [diff] [blame] | 87 | res.emplace_back(stripXPathPrefix(it->path()), value); |
| 88 | } |
| 89 | } |
| 90 | } |
Václav Kubernát | daf4031 | 2020-06-19 11:34:55 +0200 | [diff] [blame] | 91 | } |
| 92 | |
Václav Kubernát | daf4031 | 2020-06-19 11:34:55 +0200 | [diff] [blame] | 93 | void lyNodesToTree(DatastoreAccess::Tree& res, const std::vector<std::shared_ptr<libyang::Data_Node>> items, std::optional<std::string> ignoredXPathPrefix) |
| 94 | { |
| 95 | for (auto it = items.begin(); it < items.end(); it++) { |
| 96 | if ((*it)->schema()->nodetype() == LYS_LEAFLIST) { |
| 97 | auto leafListPath = stripLeafListValueFromPath((*it)->path()); |
| 98 | res.emplace_back(leafListPath, special_{SpecialValue::LeafList}); |
| 99 | while (it != items.end() && boost::starts_with((*it)->path(), leafListPath)) { |
| 100 | impl_lyNodesToTree(res, (*it)->tree_dfs(), ignoredXPathPrefix); |
| 101 | it++; |
| 102 | } |
| 103 | } else { |
| 104 | impl_lyNodesToTree(res, (*it)->tree_dfs(), ignoredXPathPrefix); |
| 105 | } |
| 106 | } |
| 107 | } |
Václav Kubernát | fbab2d4 | 2021-02-05 16:12:34 +0100 | [diff] [blame] | 108 | |
| 109 | DatastoreAccess::Tree rpcOutputToTree(const std::string& rpcPath, libyang::S_Data_Node output) |
| 110 | { |
| 111 | DatastoreAccess::Tree res; |
| 112 | if (output) { |
| 113 | // The output is "some top-level node". If we actually want the output of our RPC/action we need to use |
| 114 | // find_path. Also, our `path` is fully prefixed, but the output paths aren't. So we use outputNode->path() to |
| 115 | // get the unprefixed path. |
| 116 | |
| 117 | auto outputNode = output->find_path(rpcPath.c_str())->data().front(); |
| 118 | lyNodesToTree(res, {outputNode}, joinPaths(outputNode->path(), "/")); |
| 119 | } |
| 120 | return res; |
| 121 | } |
| 122 | |
| 123 | libyang::S_Data_Node treeToRpcInput(libyang::S_Context ctx, const std::string& path, DatastoreAccess::Tree in) |
| 124 | { |
| 125 | auto root = std::make_shared<libyang::Data_Node>(ctx, path.c_str(), nullptr, LYD_ANYDATA_CONSTSTRING, LYD_PATH_OPT_UPDATE); |
| 126 | for (const auto& [k, v] : in) { |
| 127 | root->new_path(ctx, k.c_str(), leafDataToString(v).c_str(), LYD_ANYDATA_CONSTSTRING, LYD_PATH_OPT_UPDATE); |
| 128 | } |
| 129 | |
| 130 | return root; |
| 131 | } |