blob: 318aadadea0afe864b75cf93e86910d75f7a7249 [file] [log] [blame]
Václav Kubernát24df80e2018-06-06 15:18:03 +02001/*
2 * Copyright (C) 2018 CESNET, https://photonics.cesnet.cz/
3 * Copyright (C) 2018 FIT CVUT, https://fit.cvut.cz/
4 *
5 * Written by Václav Kubernát <kubervac@fit.cvut.cz>
6 *
7*/
8
9#include <experimental/iterator>
Václav Kubernát509ce652019-05-29 19:46:44 +020010#include <sstream>
Václav Kubernát24df80e2018-06-06 15:18:03 +020011#include "ast_path.hpp"
12#include "utils.hpp"
13
14container_::container_(const std::string& name)
15 : m_name(name)
16{
17}
18
19bool container_::operator==(const container_& b) const
20{
21 return this->m_name == b.m_name;
22}
23
24leaf_::leaf_(const std::string& name)
25 : m_name(name)
26{
27}
28
Václav Kubernát5b8a8f32020-05-20 00:57:22 +020029bool leafListElement_::operator==(const leafListElement_& b) const
30{
31 return this->m_name == b.m_name && this->m_value == b.m_value;
32}
33
Václav Kubernát5b8a8f32020-05-20 00:57:22 +020034leafList_::leafList_(const std::string& name)
35 : m_name(name)
36{
37}
38
39bool leafList_::operator==(const leafList_& b) const
40{
41 return this->m_name == b.m_name;
42}
43
Václav Kubernát744f57f2018-06-29 22:46:26 +020044bool module_::operator==(const module_& b) const
45{
46 return this->m_name == b.m_name;
47}
48
Václav Kubernát2eaceb82018-10-08 19:56:30 +020049dataNode_::dataNode_() = default;
Václav Kubernát744f57f2018-06-29 22:46:26 +020050
Václav Kubernát2eaceb82018-10-08 19:56:30 +020051dataNode_::dataNode_(decltype(m_suffix) node)
Václav Kubernát744f57f2018-06-29 22:46:26 +020052 : m_suffix(node)
53{
54}
55
Václav Kubernát2eaceb82018-10-08 19:56:30 +020056dataNode_::dataNode_(module_ module, decltype(m_suffix) node)
Václav Kubernát744f57f2018-06-29 22:46:26 +020057 : m_prefix(module)
58 , m_suffix(node)
59{
60}
61
Václav Kubernát2db124c2020-05-28 21:58:36 +020062dataNode_::dataNode_(boost::optional<module_> module, decltype(m_suffix) node)
63 : m_prefix(module)
64 , m_suffix(node)
65{
66
67}
68
Václav Kubernát5c75b252018-10-10 18:33:47 +020069schemaNode_::schemaNode_(decltype(m_suffix) node)
70 : m_suffix(node)
71{
72}
73
Václav Kubernát2eaceb82018-10-08 19:56:30 +020074schemaNode_::schemaNode_(module_ module, decltype(m_suffix) node)
75 : m_prefix(module)
76 , m_suffix(node)
77{
78}
Václav Kubernát744f57f2018-06-29 22:46:26 +020079
Václav Kubernát2eaceb82018-10-08 19:56:30 +020080schemaNode_::schemaNode_() = default;
81
82bool schemaNode_::operator==(const schemaNode_& b) const
83{
84 return this->m_suffix == b.m_suffix && this->m_prefix == b.m_prefix;
85}
86
87bool dataNode_::operator==(const dataNode_& b) const
Václav Kubernát744f57f2018-06-29 22:46:26 +020088{
89 return this->m_suffix == b.m_suffix && this->m_prefix == b.m_prefix;
90}
91
Václav Kubernát24df80e2018-06-06 15:18:03 +020092bool leaf_::operator==(const leaf_& b) const
93{
94 return this->m_name == b.m_name;
95}
96
Václav Kubernátc15fe822020-06-04 11:28:39 +020097listElement_::listElement_(const std::string& listName, const ListInstance& keys)
Václav Kubernát24df80e2018-06-06 15:18:03 +020098 : m_name(listName)
99 , m_keys(keys)
100{
101}
102
103bool listElement_::operator==(const listElement_& b) const
104{
105 return (this->m_name == b.m_name && this->m_keys == b.m_keys);
106}
107
Václav Kubernát2eaceb82018-10-08 19:56:30 +0200108bool list_::operator==(const list_& b) const
109{
110 return (this->m_name == b.m_name);
111}
112
113list_::list_(const std::string& listName)
114 : m_name(listName)
115{
116}
117
118bool schemaPath_::operator==(const schemaPath_& b) const
119{
120 if (this->m_nodes.size() != b.m_nodes.size())
121 return false;
122 return this->m_nodes == b.m_nodes;
123}
124
125bool dataPath_::operator==(const dataPath_& b) const
Václav Kubernát24df80e2018-06-06 15:18:03 +0200126{
127 if (this->m_nodes.size() != b.m_nodes.size())
128 return false;
129 return this->m_nodes == b.m_nodes;
130}
131
Václav Kubernátb5ca1542020-05-27 01:03:54 +0200132struct nodeToSchemaStringVisitor {
Václav Kubernát24df80e2018-06-06 15:18:03 +0200133 std::string operator()(const nodeup_&) const
134 {
135 return "..";
136 }
137 template <class T>
138 std::string operator()(const T& node) const
139 {
140 return node.m_name;
141 }
142};
Jan Kundrát2a8f4332018-09-14 17:05:31 +0200143
144std::string escapeListKeyString(const std::string& what)
145{
146 // If we have both single and double quote, then we're screwed, but that "shouldn't happen"
147 // in <= YANG 1.1 due to limitations in XPath 1.0.
148 if (what.find('\'') != std::string::npos) {
149 return '\"' + what + '\"';
150 } else {
151 return '\'' + what + '\'';
152 }
153}
154
Václav Kubernátb5ca1542020-05-27 01:03:54 +0200155struct nodeToDataStringVisitor {
Václav Kubernát24df80e2018-06-06 15:18:03 +0200156 std::string operator()(const listElement_& node) const
157 {
158 std::ostringstream res;
159 res << node.m_name + "[";
160 std::transform(node.m_keys.begin(), node.m_keys.end(),
Václav Kubernát5395e712019-12-03 18:24:33 +0100161 std::experimental::make_ostream_joiner(res, "]["),
Václav Kubernát7707cae2020-01-16 12:04:53 +0100162 [] (const auto& it) { return it.first + "=" + escapeListKeyString(leafDataToString(it.second)); });
Václav Kubernát24df80e2018-06-06 15:18:03 +0200163 res << "]";
Václav Kubernátebca2552018-06-08 19:06:02 +0200164 return res.str();
Václav Kubernát24df80e2018-06-06 15:18:03 +0200165 }
Václav Kubernát5b8a8f32020-05-20 00:57:22 +0200166 std::string operator()(const leafListElement_& node) const
167 {
168 return node.m_name + "[.=" + escapeListKeyString(leafDataToString(node.m_value)) + "]";
169 }
Václav Kubernát24df80e2018-06-06 15:18:03 +0200170 std::string operator()(const nodeup_&) const
171 {
172 return "..";
173 }
174 template <class T>
175 std::string operator()(const T& node) const
176 {
177 return node.m_name;
178 }
179};
180
Václav Kubernát2eaceb82018-10-08 19:56:30 +0200181std::string nodeToSchemaString(decltype(dataPath_::m_nodes)::value_type node)
Václav Kubernát744f57f2018-06-29 22:46:26 +0200182{
Václav Kubernátb5ca1542020-05-27 01:03:54 +0200183 return std::visit(nodeToSchemaStringVisitor(), node.m_suffix);
Václav Kubernát744f57f2018-06-29 22:46:26 +0200184}
185
Václav Kubernátefcac932020-01-10 15:26:32 +0100186std::string pathToDataString(const dataPath_& path, Prefixes prefixes)
Václav Kubernát24df80e2018-06-06 15:18:03 +0200187{
188 std::string res;
Václav Kubernát5395e712019-12-03 18:24:33 +0100189 if (path.m_scope == Scope::Absolute) {
190 res = "/";
191 }
Václav Kubernátefcac932020-01-10 15:26:32 +0100192
Václav Kubernátef085742020-04-21 09:28:44 +0200193 for (const auto& it : path.m_nodes) {
Václav Kubernát744f57f2018-06-29 22:46:26 +0200194 if (it.m_prefix)
Václav Kubernátb5ca1542020-05-27 01:03:54 +0200195 res = joinPaths(res, it.m_prefix.value().m_name + ":" + std::visit(nodeToDataStringVisitor(), it.m_suffix));
Václav Kubernát744f57f2018-06-29 22:46:26 +0200196 else
Václav Kubernátb5ca1542020-05-27 01:03:54 +0200197 res = joinPaths(res, (prefixes == Prefixes::Always ? path.m_nodes.at(0).m_prefix.value().m_name + ":" : "") + std::visit(nodeToDataStringVisitor(), it.m_suffix));
Václav Kubernátefcac932020-01-10 15:26:32 +0100198 }
Václav Kubernát744f57f2018-06-29 22:46:26 +0200199
200 return res;
201}
202
Václav Kubernátefcac932020-01-10 15:26:32 +0100203std::string pathToSchemaString(const schemaPath_& path, Prefixes prefixes)
Václav Kubernát744f57f2018-06-29 22:46:26 +0200204{
205 std::string res;
Václav Kubernátefcac932020-01-10 15:26:32 +0100206 if (path.m_scope == Scope::Absolute) {
207 res = "/";
Václav Kubernát744f57f2018-06-29 22:46:26 +0200208 }
209
Václav Kubernátef085742020-04-21 09:28:44 +0200210 for (const auto& it : path.m_nodes) {
Václav Kubernát744f57f2018-06-29 22:46:26 +0200211 if (it.m_prefix)
Václav Kubernátb5ca1542020-05-27 01:03:54 +0200212 res = joinPaths(res, it.m_prefix.value().m_name + ":" + std::visit(nodeToSchemaStringVisitor(), it.m_suffix));
Václav Kubernát744f57f2018-06-29 22:46:26 +0200213 else
Václav Kubernátb5ca1542020-05-27 01:03:54 +0200214 res = joinPaths(res, (prefixes == Prefixes::Always ? path.m_nodes.at(0).m_prefix.value().m_name + ":" : "") + std::visit(nodeToSchemaStringVisitor(), it.m_suffix));
Václav Kubernát744f57f2018-06-29 22:46:26 +0200215 }
Václav Kubernát24df80e2018-06-06 15:18:03 +0200216 return res;
217}
218
Václav Kubernátefcac932020-01-10 15:26:32 +0100219std::string pathToSchemaString(const dataPath_& path, Prefixes prefixes)
Václav Kubernát24df80e2018-06-06 15:18:03 +0200220{
Václav Kubernátefcac932020-01-10 15:26:32 +0100221 return pathToSchemaString(dataPathToSchemaPath(path), prefixes);
Václav Kubernát5c75b252018-10-10 18:33:47 +0200222}
223
Václav Kubernátb5ca1542020-05-27 01:03:54 +0200224struct dataSuffixToSchemaSuffix {
225 using ReturnType = decltype(schemaNode_::m_suffix);
226 ReturnType operator()(const listElement_& listElement) const
Václav Kubernát2eaceb82018-10-08 19:56:30 +0200227 {
228 return list_{listElement.m_name};
229 }
230
Václav Kubernátb5ca1542020-05-27 01:03:54 +0200231 ReturnType operator()(const leafListElement_& leafListElement) const
Václav Kubernát5b8a8f32020-05-20 00:57:22 +0200232 {
233 return leafList_{leafListElement.m_name};
234 }
235
Václav Kubernát2eaceb82018-10-08 19:56:30 +0200236 template <typename T>
Václav Kubernátb5ca1542020-05-27 01:03:54 +0200237 ReturnType operator()(const T& suffix) const
Václav Kubernát2eaceb82018-10-08 19:56:30 +0200238 {
239 return suffix;
240 }
241};
242
243schemaNode_ dataNodeToSchemaNode(const dataNode_& node)
244{
245 schemaNode_ res;
246 res.m_prefix = node.m_prefix;
Václav Kubernátb5ca1542020-05-27 01:03:54 +0200247 res.m_suffix = std::visit(dataSuffixToSchemaSuffix(), node.m_suffix);
Václav Kubernát2eaceb82018-10-08 19:56:30 +0200248 return res;
249}
250
251schemaPath_ dataPathToSchemaPath(const dataPath_& path)
252{
Václav Kubernátbf083ec2019-02-19 13:58:09 +0100253 schemaPath_ res{path.m_scope, {}};
Václav Kubernát2eaceb82018-10-08 19:56:30 +0200254
Václav Kubernátbf083ec2019-02-19 13:58:09 +0100255 std::transform(path.m_nodes.begin(), path.m_nodes.end(),
256 std::back_inserter(res.m_nodes),
257 [](const dataNode_& node) { return dataNodeToSchemaNode(node); });
Václav Kubernát2eaceb82018-10-08 19:56:30 +0200258
Václav Kubernátbf083ec2019-02-19 13:58:09 +0100259 return res;
Václav Kubernát2eaceb82018-10-08 19:56:30 +0200260}
Václav Kubernáte781b902020-06-15 14:35:11 +0200261
262namespace {
263template <typename NodeType>
264void impl_pushFragment(std::vector<NodeType>& where, const NodeType& what)
265{
266 if (std::holds_alternative<nodeup_>(what.m_suffix)) {
267 if (!where.empty()) { // Allow going up, when already at root
268 where.pop_back();
269 }
270 } else {
271 where.push_back(what);
272 }
273}
274}
275
276void schemaPath_::pushFragment(const schemaNode_& fragment)
277{
278 impl_pushFragment(m_nodes, fragment);
279}
280
281void dataPath_::pushFragment(const dataNode_& fragment)
282{
283 impl_pushFragment(m_nodes, fragment);
284}