diff options
Diffstat (limited to 'src/cnode/cnode.cpp')
-rw-r--r-- | src/cnode/cnode.cpp | 123 |
1 files changed, 123 insertions, 0 deletions
diff --git a/src/cnode/cnode.cpp b/src/cnode/cnode.cpp new file mode 100644 index 0000000..d1f519e --- /dev/null +++ b/src/cnode/cnode.cpp @@ -0,0 +1,123 @@ +/* + * Copyright (C) 2010 Vyatta, Inc. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see <http://www.gnu.org/licenses/>. + */ + +#include <cstdio> +#include <cstring> +#include <vector> +#include <string> +#include <algorithm> + +#include <cli_cstore.h> +#include <cnode/cnode.hpp> + +using namespace std; +using namespace cnode; + +////// constructors/destructors +CfgNode::CfgNode(Cstore& cstore, vector<string>& path_comps, + bool active, bool recursive) + : _is_tag(false), _is_leaf(false), _is_multi(false), _is_value(false), + _is_default(false), _is_deactivated(false), _is_leaf_typeless(false), + _is_invalid(false), _is_empty(false) +{ + /* first get the def (only if path is not empty). if path is empty, i.e., + * "root", treat it as an intermediate node. + */ + if (path_comps.size() > 0) { + vtw_def def; + if (cstore.validateTmplPath(path_comps, false, def)) { + // got the def + _is_value = def.is_value; + _is_tag = (def.tag && !_is_value); + _is_leaf = (!_is_tag && !_is_value && def.def_type != ERROR_TYPE); + _is_multi = (_is_leaf && def.multi); + _is_default = cstore.cfgPathDefault(path_comps, active); + _is_deactivated = cstore.cfgPathDeactivated(path_comps, active); + cstore.cfgPathGetComment(path_comps, _comment, active); + + if (_is_leaf && _is_value) { + /* recursion should never reach here. if path is specified by user, + * nothing further to do. + */ + return; + } + } else { + // not a valid node + _is_invalid = true; + return; + } + } + + // handle leaf node (note path_comps must be non-empty if this is leaf) + if (_is_leaf) { + _name = path_comps[path_comps.size() - 1]; + if (_is_multi) { + // multi-value node + cstore.cfgPathGetValuesDA(path_comps, _values, active, true); + // ignore return value + } else { + // single-value node + cstore.cfgPathGetValueDA(path_comps, _value, active, true); + // ignore return value + } + return; + } + + // handle intermediate (typeless) or tag + if (_is_value) { + // tag value + _name = path_comps[path_comps.size() - 2]; + _value = path_comps[path_comps.size() - 1]; + } else { + // tag node or typeless node + _name = (path_comps.size() > 0 ? path_comps[path_comps.size() - 1] : ""); + } + + // check child nodes + vector<string> cnodes; + cstore.cfgPathGetChildNodesDA(path_comps, cnodes, active, true); + if (cnodes.size() == 0) { + // empty subtree. done. + vector<string> tcnodes; + cstore.tmplGetChildNodes(path_comps, tcnodes); + if (tcnodes.size() == 0) { + // typeless leaf node + _is_leaf_typeless = true; + } + _is_empty = true; + return; + } + + if (!recursive) { + // nothing further to do + return; + } + + // recurse + for (size_t i = 0; i < cnodes.size(); i++) { + path_comps.push_back(cnodes[i]); + CfgNode *cn = new CfgNode(cstore, path_comps, active, recursive); + if (_is_tag && !_is_value) { + // tag node + _tag_values.push_back(cn); + } else { + // intermediate node or tag value + _child_nodes.push_back(cn); + } + path_comps.pop_back(); + } +} + |