/*
* Copyright (c) 2011-2015, Intel Corporation
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without modification,
* are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation and/or
* other materials provided with the distribution.
*
* 3. Neither the name of the copyright holder nor the names of its contributors
* may be used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "Element.h"
#include "XmlElementSerializingContext.h"
#include "ElementLibrary.h"
#include "ErrorContext.hpp"
#include "PfError.hpp"
#include <algorithm>
#include <assert.h>
#include <stdio.h>
#include <stdarg.h>
#include <stdlib.h>
using std::string;
const std::string CElement::gDescriptionPropertyName = "Description";
CElement::CElement(const string &strName) : _strName(strName)
{
}
CElement::~CElement()
{
removeChildren();
}
void CElement::setDescription(const string &strDescription)
{
_strDescription = strDescription;
}
const string &CElement::getDescription() const
{
return _strDescription;
}
bool CElement::childrenAreDynamic() const
{
// By default, children are searched and not created during xml parsing
return false;
}
bool CElement::init(string &strError)
{
for (CElement *child : _childArray) {
if (!child->init(strError)) {
return false;
}
}
return true;
}
string CElement::dumpContent(utility::ErrorContext &errorContext, const size_t depth) const
{
string output;
string strIndent;
// Level
size_t indents = depth;
while (indents--) {
strIndent += " ";
}
// Type
output += strIndent + "- " + getKind();
// Name
if (!_strName.empty()) {
output += ": " + getName();
}
// Value
string strValue = logValue(errorContext);
if (!strValue.empty()) {
output += " = " + strValue;
}
output += "\n";
for (CElement *pChild : _childArray) {
output += pChild->dumpContent(errorContext, depth + 1);
}
return output;
}
// Element properties
void CElement::showProperties(string &strResult) const
{
strResult += "Kind: " + getKind() + "\n";
showDescriptionProperty(strResult);
}
void CElement::showDescriptionProperty(std::string &strResult) const
{
if (!getDescription().empty()) {
strResult += gDescriptionPropertyName + ": " + getDescription() + "\n";
}
}
// Content dumping
string CElement::logValue(utility::ErrorContext & /*ctx*/) const
{
return "";
}
// From IXmlSink
bool CElement::fromXml(const CXmlElement &xmlElement,
CXmlSerializingContext &serializingContext) try {
xmlElement.getAttribute(gDescriptionPropertyName, _strDescription);
// Propagate through children
CXmlElement::CChildIterator childIterator(xmlElement);
CXmlElement childElement;
while (childIterator.next(childElement)) {
CElement *pChild;
if (!childrenAreDynamic()) {
pChild = findChildOfKind(childElement.getType());
if (!pChild) {
serializingContext.setError("Unable to handle XML element: " +
childElement.getPath());
return false;
}
} else {
// Child needs creation
pChild = createChild(childElement, serializingContext);
if (!pChild) {
return false;
}
}
// Dig
if (!pChild->fromXml(childElement, serializingContext)) {
return false;
}
}
return true;
} catch (const PfError &e) {
serializingContext.appendLineToError(e.what());
return false;
}
void CElement::childrenToXml(CXmlElement &xmlElement,
CXmlSerializingContext &serializingContext) const
{
// Browse children and propagate
for (CElement *pChild : _childArray) {
// Create corresponding child element
CXmlElement xmlChildElement;
xmlElement.createChild(xmlChildElement, pChild->getXmlElementName());
// Propagate
pChild->toXml(xmlChildElement, serializingContext);
}
}
void CElement::toXml(CXmlElement &xmlElement, CXmlSerializingContext &serializingContext) const
{
setXmlNameAttribute(xmlElement);
setXmlDescriptionAttribute(xmlElement);
childrenToXml(xmlElement, serializingContext);
}
void CElement::setXmlDescriptionAttribute(CXmlElement &xmlElement) const
{
const string &description = getDescription();
if (!description.empty()) {
xmlElement.setAttribute(gDescriptionPropertyName, description);
}
}
void CElement::setXmlNameAttribute(CXmlElement &xmlElement) const
{
// By default, set Name attribute if any
string strName = getName();
if (!strName.empty()) {
xmlElement.setNameAttribute(strName);
}
}
// Name
void CElement::setName(const string &strName)
{
_strName = strName;
}
const string &CElement::getName() const
{
return _strName;
}
bool CElement::rename(const string &strName, string &strError)
{
// Check for conflict with brotherhood if relevant
if (_pParent && _pParent->childrenAreDynamic()) {
for (CElement *pParentChild : _pParent->_childArray) {
if (pParentChild != this && pParentChild->getName() == strName) {
// Conflict
strError = "Name conflicts with brother element";
return false;
}
}
}
// Change name
setName(strName);
return true;
}
string CElement::getPathName() const
{
if (!_strName.empty()) {
return _strName;
} else {
return getKind();
}
}
// Hierarchy
void CElement::addChild(CElement *pChild)
{
_childArray.push_back(pChild);
pChild->_pParent = this;
}
CElement *CElement::getChild(size_t index)
{
assert(index <= _childArray.size());
return _childArray[index];
}
const CElement *CElement::getChild(size_t index) const
{
assert(index <= _childArray.size());
return _childArray[index];
}
CElement *CElement::createChild(const CXmlElement &childElement,
CXmlSerializingContext &serializingContext)
{
// Context
CXmlElementSerializingContext &elementSerializingContext =
static_cast<CXmlElementSerializingContext &>(serializingContext);
// Child needs creation
CElement *pChild = elementSerializingContext.getElementLibrary()->createElement(childElement);
if (!pChild) {
elementSerializingContext.setError("Unable to create XML element " +
childElement.getPath());
return nullptr;
}
// Store created child!
addChild(pChild);
return pChild;
}
bool CElement::removeChild(CElement *pChild)
{
auto childIt = find(begin(_childArray), end(_childArray), pChild);
if (childIt != end(_childArray)) {
_childArray.erase(childIt);
return true;
}
return false;
}
void CElement::listChildren(string &strChildList) const
{
// Get list of children names
for (CElement *pChild : _childArray) {
strChildList += pChild->getName() + "\n";
}
}
string CElement::listQualifiedPaths(bool bDive, size_t level) const
{
string strResult;
// Dive Will cause only leaf nodes to be printed
if (!bDive || !getNbChildren()) {
strResult = getQualifiedPath() + "\n";
}
if (bDive || !level) {
// Get list of children paths
for (CElement *pChild : _childArray) {
strResult += pChild->listQualifiedPaths(bDive, level + 1);
}
}
return strResult;
}
void CElement::listChildrenPaths(string &strChildList) const
{
// Get list of children paths
for (CElement *pChild : _childArray) {
strChildList += pChild->getPath() + "\n";
}
}
size_t CElement::getNbChildren() const
{
return _childArray.size();
}
const CElement *CElement::getParent() const
{
return _pParent;
}
CElement *CElement::getParent()
{
return _pParent;
}
void CElement::clean()
{
if (childrenAreDynamic()) {
removeChildren();
} else {
// Just propagate
for (CElement *pChild : _childArray) {
pChild->clean();
}
}
}
void CElement::removeChildren()
{
// Delete in reverse order
ChildArrayReverseIterator it;
for (it = _childArray.rbegin(); it != _childArray.rend(); ++it) {
delete *it;
}
_childArray.clear();
}
const CElement *CElement::findDescendant(CPathNavigator &pathNavigator) const
{
string *pStrChildName = pathNavigator.next();
if (!pStrChildName) {
return this;
}
const CElement *pChild = findChild(*pStrChildName);
if (!pChild) {
return nullptr;
}
return pChild->findDescendant(pathNavigator);
}
CElement *CElement::findDescendant(CPathNavigator &pathNavigator)
{
string *pStrChildName = pathNavigator.next();
if (!pStrChildName) {
return this;
}
CElement *pChild = findChild(*pStrChildName);
if (!pChild) {
return nullptr;
}
return pChild->findDescendant(pathNavigator);
}
bool CElement::isDescendantOf(const CElement *pCandidateAscendant) const
{
if (!_pParent) {
return false;
}
if (_pParent == pCandidateAscendant) {
return true;
}
return _pParent->isDescendantOf(pCandidateAscendant);
}
CElement *CElement::findChild(const string &strName)
{
for (CElement *pChild : _childArray) {
if (pChild->getPathName() == strName) {
return pChild;
}
}
return nullptr;
}
const CElement *CElement::findChild(const string &strName) const
{
for (CElement *pChild : _childArray) {
if (pChild->getPathName() == strName) {
return pChild;
}
}
return nullptr;
}
CElement *CElement::findChildOfKind(const string &strKind)
{
for (CElement *pChild : _childArray) {
if (pChild->getKind() == strKind) {
return pChild;
}
}
return nullptr;
}
const CElement *CElement::findChildOfKind(const string &strKind) const
{
for (CElement *pChild : _childArray) {
if (pChild->getKind() == strKind) {
return pChild;
}
}
return nullptr;
}
string CElement::getPath() const
{
// Take out root element from the path
if (_pParent && _pParent->_pParent) {
return _pParent->getPath() + "/" + getPathName();
}
return "/" + getPathName();
}
string CElement::getQualifiedPath() const
{
return getPath() + " [" + getKind() + "]";
}
string CElement::getXmlElementName() const
{
// Default to element kind
return getKind();
}