// NeL - MMORPG Framework
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see .
#include "stdmisc.h"
#include "nel/misc/i_xml.h"
#include "nel/misc/sstring.h"
#ifndef NL_DONT_USE_EXTERNAL_CODE
// Include from libxml2
#include
#if defined(NL_OS_WINDOWS) && defined(NL_COMP_VC_VERSION) && NL_COMP_VC_VERSION >= 80
#define USE_LOCALE_ATOF
#include
#endif
using namespace std;
#define NLMISC_READ_BUFFER_SIZE 1024
#ifdef DEBUG_NEW
#define new DEBUG_NEW
#endif
namespace NLMISC
{
// *********************************************************
const char SEPARATOR = ' ';
// ***************************************************************************
#define readnumber(dest,thetype,digits,convfunc) \
string number_as_string; \
serialSeparatedBufferIn( number_as_string ); \
dest = (thetype)convfunc( number_as_string.c_str() );
#ifdef USE_LOCALE_ATOF
#define readnumberlocale(dest,thetype,digits,convfunc) \
string number_as_string; \
serialSeparatedBufferIn( number_as_string ); \
dest = (thetype)convfunc( number_as_string.c_str(), (_locale_t)_Locale );
#define nl_atof _atof_l
#else
#define readnumberlocale(dest,thetype,digits,convfunc) readnumber(dest,thetype,digits,convfunc)
#define nl_atof atof
#endif
// ***************************************************************************
inline void CIXml::flushContentString ()
{
// String size
_ContentString.erase ();
// Reset
_ContentStringIndex = 0;
}
// ***************************************************************************
CIXml::CIXml () : IStream (true /* Input mode */)
{
// Not initialized
_Parser = NULL;
_CurrentElement = NULL;
_CurrentNode = NULL;
_PushBegin = false;
_AttribPresent = false;
_ErrorString = "";
_TryBinaryMode = false;
_BinaryStream = NULL;
#ifdef USE_LOCALE_ATOF
// create C numeric locale
_Locale = _create_locale(LC_NUMERIC, "C");
#else
_Locale = NULL;
#endif
}
// ***************************************************************************
CIXml::CIXml (bool tryBinaryMode) : IStream (true /* Input mode */)
{
// Not initialized
_Parser = NULL;
_CurrentElement = NULL;
_CurrentNode = NULL;
_PushBegin = false;
_AttribPresent = false;
_ErrorString = "";
_TryBinaryMode = tryBinaryMode;
_BinaryStream = NULL;
#ifdef USE_LOCALE_ATOF
// create C numeric locale
_Locale = _create_locale(LC_NUMERIC, "C");
#else
_Locale = NULL;
#endif
}
// ***************************************************************************
CIXml::~CIXml ()
{
// Release
release ();
}
// ***************************************************************************
void CIXml::release ()
{
// Release the parser
if (_Parser)
{
// Free it
xmlClearParserCtxt (_Parser);
xmlFreeParserCtxt (_Parser);
// commented due to the bug #857 xmlCleanupParser ();
_Parser = NULL;
}
// Not initialized
_Parser = NULL;
_CurrentElement = NULL;
_CurrentNode = NULL;
_PushBegin = false;
_AttribPresent = false;
_ErrorString = "";
resetPtrTable();
#ifdef USE_LOCALE_ATOF
if (_Locale) _free_locale((_locale_t)_Locale);
#endif
}
// ***************************************************************************
void xmlGenericErrorFuncRead (void *ctx, const char *msg, ...)
{
// Get the error string
string str;
NLMISC_CONVERT_VARGS (str, msg, NLMISC::MaxCStringSize);
((CIXml*)ctx)->_ErrorString += str;
}
// ***************************************************************************
bool CIXml::init (IStream &stream)
{
// Release
release ();
xmlInitParser();
// Default : XML mode
_BinaryStream = NULL;
// Input stream ?
if (stream.isReading())
{
// Set XML mode
setXMLMode (true);
// Get current position
sint32 pos = stream.getPos ();
// Go to end
bool seekGood = stream.seek (0, end);
nlassert (seekGood);
// Get input stream length
sint32 length = stream.getPos () - pos;
// Go to start
stream.seek (pos, begin);
// The read buffer
char buffer[NLMISC_READ_BUFFER_SIZE];
// Fill the buffer
stream.serialBuffer ((uint8*)buffer, 4);
length -= 4;
// Try binary mode
if (_TryBinaryMode)
{
string header;
header.resize(4);
header[0] = buffer[0];
header[1] = buffer[1];
header[2] = buffer[2];
header[3] = buffer[3];
toLower(header);
// Does it a xml stream ?
if (header != "=NLMISC_READ_BUFFER_SIZE)
{
// Fill the buffer
stream.serialBuffer ((uint8*)buffer, NLMISC_READ_BUFFER_SIZE);
// Read a buffer
int res = xmlParseChunk(_Parser, buffer, NLMISC_READ_BUFFER_SIZE, 0);
// Error code ?
if (res)
{
throw EXmlParsingError (_ErrorString);
}
// Length left
length -= NLMISC_READ_BUFFER_SIZE;
}
// Fill the buffer
stream.serialBuffer ((uint8*)buffer, length);
// Parse the last buffer
int res = xmlParseChunk(_Parser, buffer, length, 1);
// Error code ?
if (res)
{
throw EXmlParsingError (_ErrorString);
}
// Ok
return true;
}
else
{
nlwarning ("XML: The stream is not an input stream.");
}
// Error
return false;
}
// ***************************************************************************
void CIXml::serialSeparatedBufferIn ( string &value, bool checkSeparator )
{
nlassert( isReading() );
// Output stream has been initialized ?
if ( _Parser )
{
// Current node presents ?
if (_CurrentElement)
{
// Write a push attribute ?
if (_PushBegin)
{
// Current attrib is set ?
if (_AttribPresent)
{
// Should have a current element
nlassert (_CurrentElement);
// Get the attribute
xmlChar *attributeValue = xmlGetProp (_CurrentElement, (const xmlChar*)_AttribName.c_str());
// Attribute is here ?
if (attributeValue)
{
// Copy the value
value = (const char*)attributeValue;
// Delete the value
xmlFree ((void*)attributeValue);
}
else
{
// Node name must not be NULL
nlassert (_CurrentElement->name);
// Make an error message
char tmp[512];
smprintf (tmp, 512, "NeL XML Syntax error in block line %d\nAttribute \"%s\" is missing in node \"%s\"",
(int)_CurrentElement->line, _AttribName.c_str(), _CurrentElement->name);
throw EXmlParsingError (tmp);
}
// The attribute has been used
_AttribPresent = false;
}
else
{
// * Error, the stream don't use XML streaming properly
// * You must take care of this in your last serial call:
// * - Between xmlPushBegin() and xmlPushEnd(), before each serial, you must set the attribute name with xmlSetAttrib.
// * - Between xmlPushBegin() and xmlPushEnd(), you must serial only basic objects (numbers and strings).
nlerror ( "Error, the stream don't use XML streaming properly" );
}
}
else
{
// Content length
uint length = (uint)_ContentString.length();
// String empty ?
if (length==0)
{
// Try to open the node
do
{
// If no more node, empty string
if (_CurrentNode == NULL)
{
value = "";
_ContentStringIndex = 0;
_ContentString.erase ();
return;
}
// Node with the good name
if (_CurrentNode->type == XML_TEXT_NODE)
{
// Stop searching
break;
}
else
// Get next
_CurrentNode = _CurrentNode->next;
}
while (_CurrentNode);
// Not found ?
if (_CurrentNode != NULL)
{
// Read the content
const char *content = (const char*)xmlNodeGetContent (_CurrentNode);
if (content)
{
_ContentString = content;
// Delete the value
xmlFree ((void*)content);
}
else
_ContentString.erase ();
// Set the current index
_ContentStringIndex = 0;
// New length
length = (uint)_ContentString.length();
}
}
// Keyword in the buffer ?
if (_ContentStringIndex < length)
{
// First index
uint first = _ContentStringIndex;
// Have to take care of separators ?
if (checkSeparator)
{
// Scan to the end
while (_ContentStringIndex < length)
{
// Not a separator ?
if ( (_ContentString[_ContentStringIndex]==SEPARATOR) || (_ContentString[_ContentStringIndex]=='\n') )
{
_ContentStringIndex++;
break;
}
// Next char
_ContentStringIndex++;
}
}
else
{
// Copy all the string
_ContentStringIndex = length;
}
// Make a string
value.assign (_ContentString, first, _ContentStringIndex-first);
}
else
{
// Should have a name
nlassert (_CurrentElement->name);
// Make an error message
char tmp[512];
smprintf (tmp, 512, "NeL XML Syntax error in block line %d \nMissing keywords in text child node in the node %s",
(int)_CurrentElement->line, _CurrentElement->name);
throw EXmlParsingError (tmp);
}
}
}
else
{
// * Error, no current node present.
// * Check that your serial is initialy made between a xmlPushBegin and xmlPushEnd calls.
nlerror ( "Error, the stream don't use XML streaming properly" );
}
}
else
{
nlerror ( "Output stream has not been initialized" );
}
}
// ***************************************************************************
void CIXml::serial(uint8 &b)
{
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
// Read the number
readnumber( b, uint8, 3, atoi );
}
}
// ***************************************************************************
void CIXml::serial(sint8 &b)
{
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
readnumber( b, sint8, 4, atoi );
}
}
// ***************************************************************************
void CIXml::serial(uint16 &b)
{
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
readnumber( b, uint16, 5, atoi );
}
}
// ***************************************************************************
void CIXml::serial(sint16 &b)
{
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
readnumber( b, sint16, 6, atoi );
}
}
// ***************************************************************************
inline uint32 atoui( const char *ident)
{
return (uint32) strtoul (ident, NULL, 10);
}
void CIXml::serial(uint32 &b)
{
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
readnumber( b, uint32, 10, atoui );
}
}
// ***************************************************************************
void CIXml::serial(sint32 &b)
{
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
readnumber( b, sint32, 11, atoi );
}
}
// ***************************************************************************
void CIXml::serial(uint64 &b)
{
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
readnumber( b, uint64, 20, atoiInt64 );
}
}
// ***************************************************************************
void CIXml::serial(sint64 &b)
{
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
readnumber( b, sint64, 20, atoiInt64 );
}
}
// ***************************************************************************
void CIXml::serial(float &b)
{
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
readnumberlocale( b, float, 128, nl_atof );
}
}
// ***************************************************************************
void CIXml::serial(double &b)
{
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
readnumberlocale( b, double, 128, nl_atof );
}
}
// ***************************************************************************
void CIXml::serial(bool &b)
{
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
serialBit(b);
}
}
// ***************************************************************************
void CIXml::serialBit(bool &bit)
{
if (_BinaryStream)
{
_BinaryStream->serialBit(bit);
}
else
{
uint8 u;
serial (u);
bit = (u!=0);
}
}
// ***************************************************************************
#ifndef NL_OS_CYGWIN
void CIXml::serial(char &b)
{
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
string toto;
serialSeparatedBufferIn ( toto );
// Good value ?
if (toto.length()!=1)
{
// Protect error
if (_Parser)
{
// Should have a name
nlassert (_CurrentElement->name);
// Make an error message
char tmp[512];
smprintf (tmp, 512, "NeL XML Syntax error in block line %d \nValue is not a char in the node named %s",
(int)_CurrentElement->line, _CurrentElement->name);
throw EXmlParsingError (tmp);
}
else
{
nlerror ( "Output stream has not been initialized" );
}
}
else
b=toto[0];
}
}
#endif // NL_OS_CYGWIN
// ***************************************************************************
void CIXml::serial(std::string &b)
{
nlassert( isReading() );
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
// Attibute ?
if (_PushBegin)
{
// Only serial the string
serialSeparatedBufferIn ( b, false );
}
else
{
// Open a string node
xmlPush ("S");
// Serial the string
serialSeparatedBufferIn ( b, false );
// Close the node
xmlPop ();
}
}
}
// ***************************************************************************
void CIXml::serial(ucstring &b)
{
nlassert( isReading() );
if (_BinaryStream)
{
_BinaryStream->serial(b);
}
else
{
// Serial a simple string
string tmp;
// Serial this string
serial (tmp);
// Return a ucstring
b.fromUtf8(tmp);
}
}
// ***************************************************************************
void CIXml::serialBuffer(uint8 *buf, uint len)
{
if (_BinaryStream)
{
_BinaryStream->serialBuffer(buf, len);
}
else
{
// Open a node
xmlPush ("BUFFER");
// Serialize the buffer
for (uint i=0; imyDoc);
// Has a root node ?
if (_CurrentNode)
{
// Node name must not be NULL
nlassert (_CurrentNode->name);
// Node element with the good name ?
if ( (_CurrentNode->type != XML_ELEMENT_NODE) || ( (const char*)_CurrentNode->name != string(nodeName)) )
{
// Make an error message
char tmp[512];
smprintf (tmp, 512, "NeL XML Syntax error : root node has the wrong name : \"%s\" should have \"%s\"",
_CurrentNode->name, nodeName);
throw EXmlParsingError (tmp);
}
}
else
{
// Make an error message
char tmp[512];
smprintf (tmp, 512, "NeL XML Syntax error : no root node found.");
throw EXmlParsingError (tmp);
}
}
// Try to open the node
do
{
// Node name must not be NULL
nlassert (_CurrentNode->name);
// Node with the good name
if ( (_CurrentNode->type == XML_ELEMENT_NODE) && ( (const char*)_CurrentNode->name == string(nodeName)) )
{
// Save current element
_CurrentElement = _CurrentNode;
// Stop searching
break;
}
else
// Get next
_CurrentNode = _CurrentNode->next;
}
while (_CurrentNode);
// Not found ?
if (_CurrentNode == NULL)
{
// Make an error message
char tmp[512];
smprintf (tmp, 512, "NeL XML Syntax error in block line %d \nCan't open the node named %s in node named %s",
(int)_CurrentElement->line, nodeName, _CurrentElement->name);
throw EXmlParsingError (tmp);
}
// Get first child
_CurrentNode = _CurrentNode->children;
// Push begun
_PushBegin = true;
// Flush current string
flushContentString ();
}
else
{
nlerror ( "You must close your xmlPushBegin - xmlPushEnd before calling a new xmlPushBegin.");
return false;
}
}
else
{
nlerror ( "Output stream has not been initialized.");
return false;
}
// Ok
return true;
}
}
// ***************************************************************************
bool CIXml::xmlPushEndInternal ()
{
nlassert( isReading() );
if (_BinaryStream)
{
return true;
}
else
{
// Check _Parser
if ( _Parser )
{
// Can make a xmlPushEnd ?
if ( _PushBegin )
{
// Push begun
_PushBegin = false;
}
else
{
nlerror ( "You must call xmlPushBegin before calling xmlPushEnd.");
return false;
}
}
else
{
nlerror ( "Output stream has not been initialized.");
return false;
}
// Ok
return true;
}
}
// ***************************************************************************
bool CIXml::xmlPopInternal ()
{
nlassert( isReading() );
if (_BinaryStream)
{
return true;
}
else
{
// Check _Parser
if ( _Parser )
{
// Not in the push mode ?
if ( ! _PushBegin )
{
// Some content to write ?
flushContentString ();
// Get parents
_CurrentNode = _CurrentElement;
_CurrentElement = _CurrentElement->parent;
_CurrentNode = _CurrentNode->next;
}
else
{
nlerror ( "You must call xmlPop after xmlPushEnd.");
return false;
}
}
else
{
nlerror ( "Output stream has not been initialized.");
return false;
}
// Ok
return true;
}
}
// ***************************************************************************
bool CIXml::xmlSetAttribInternal (const char *attribName)
{
nlassert( isReading() );
if (_BinaryStream)
{
return true;
}
else
{
// Check _Parser
if ( _Parser )
{
// Can make a xmlPushEnd ?
if ( _PushBegin )
{
// Set attribute name
_AttribName = attribName;
// Attribute name is present
_AttribPresent = true;
}
else
{
nlerror ( "You must call xmlSetAttrib between xmlPushBegin and xmlPushEnd calls.");
return false;
}
}
else
{
nlerror ( "Output stream has not been initialized.");
return false;
}
// Ok
return true;
}
}
// ***************************************************************************
bool CIXml::xmlBreakLineInternal ()
{
// Ok
return true;
}
// ***************************************************************************
bool CIXml::xmlCommentInternal (const char * /* comment */)
{
// Ok
return true;
}
// ***************************************************************************
xmlNodePtr CIXml::getFirstChildNode (xmlNodePtr parent, const char *childName)
{
xmlNodePtr child = parent->children;
while (child)
{
if (strcmp ((const char*)child->name, childName) == 0)
return child;
child = child->next;
}
return NULL;
}
// ***************************************************************************
xmlNodePtr CIXml::getNextChildNode (xmlNodePtr last, const char *childName)
{
last = last->next;
while (last)
{
if (strcmp ((const char*)last->name, childName) == 0)
return last;
last = last->next;
}
return NULL;
}
// ***************************************************************************
xmlNodePtr CIXml::getFirstChildNode (xmlNodePtr parent, xmlElementType type)
{
xmlNodePtr child = parent->children;
while (child)
{
if (child->type == type)
return child;
child = child->next;
}
return NULL;
}
// ***************************************************************************
xmlNodePtr CIXml::getNextChildNode (xmlNodePtr last, xmlElementType type)
{
last = last->next;
while (last)
{
if (last->type == type)
return last;
last = last->next;
}
return NULL;
}
// ***************************************************************************
uint CIXml::countChildren (xmlNodePtr node, const char *childName)
{
uint count=0;
xmlNodePtr child = getFirstChildNode (node, childName);
while (child)
{
count++;
child = getNextChildNode (child, childName);
}
return count;
}
// ***************************************************************************
uint CIXml::countChildren (xmlNodePtr node, xmlElementType type)
{
uint count=0;
xmlNodePtr child = getFirstChildNode (node, type);
while (child)
{
count++;
child = getNextChildNode (child, type);
}
return count;
}
// ***************************************************************************
xmlNodePtr CIXml::getRootNode () const
{
if (_Parser)
if (_Parser->myDoc)
return xmlDocGetRootElement (_Parser->myDoc);
return NULL;
}
// ***************************************************************************
bool CIXml::getPropertyString (std::string &result, xmlNodePtr node, const char *property)
{
// Get the value
const char *value = (const char*)xmlGetProp (node, (xmlChar*)property);
if (value)
{
// Active value
result = value;
// Delete the value
xmlFree ((void*)value);
// Found
return true;
}
return false;
}
// ***************************************************************************
int CIXml::getIntProperty(xmlNodePtr node, const char *property, int defaultValue)
{
CSString s;
bool b;
b=getPropertyString(s,node,property);
if (b==false)
return defaultValue;
s=s.strip();
sint val=s.atoi();
if (val==0 && s!="0")
{
nlwarning("bad integer value: %s",s.c_str());
return defaultValue;
}
return val;
}
// ***************************************************************************
double CIXml::getFloatProperty(xmlNodePtr node, const char *property, float defaultValue)
{
CSString s;
bool b;
b=getPropertyString(s,node,property);
if (b==false)
return defaultValue;
return s.strip().atof();
}
// ***************************************************************************
std::string CIXml::getStringProperty(xmlNodePtr node, const char *property, const std::string& defaultValue)
{
std::string s;
bool b;
b=getPropertyString(s,node,property);
if (b==false)
return defaultValue;
return s;
}
// ***************************************************************************
bool CIXml::getContentString (std::string &result, xmlNodePtr node)
{
const char *valueText = (const char*)xmlNodeGetContent (node);
if (valueText)
{
result = valueText;
// Delete the value
xmlFree ((void*)valueText);
// Found
return true;
}
return false;
}
// ***************************************************************************
} // NLMISC
#endif // NL_DONT_USE_EXTERNAL_CODE