You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
ryzom-core/code/nel/src/net/module_common.cpp

253 lines
6.1 KiB
C++

// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
// 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 <http://www.gnu.org/licenses/>.
#include "stdnet.h"
#include "nel/misc/sstring.h"
#include "nel/net/module_common.h"
using namespace std;
using namespace NLMISC;
namespace NLNET
{
TParsedCommandLine::TParsedCommandLine(const TParsedCommandLine& copy)
:ParamName(copy.ParamName), ParamValue(copy.ParamValue)
{
uint first = 0, last = (uint)copy.SubParams.size();
SubParams.resize( last );
for (; first != last; ++first)
{
// calls recursively copy constructor
SubParams[first] = new TParsedCommandLine(*copy.SubParams[first]);
}
}
TParsedCommandLine::~TParsedCommandLine()
{
clear();
};
void TParsedCommandLine::clear()
{
for (std::vector<TParsedCommandLine*>::iterator it=SubParams.begin(); it!=SubParams.end(); ++it)
{
delete (*it);
}
SubParams.clear();
ParamName.clear();
ParamValue.clear();
}
bool TParsedCommandLine::parseParamList(const std::string &rawParamString)
{
// Cleanup the struct
clear();
return _parseParamList(rawParamString);
}
std::string TParsedCommandLine::toString() const
{
string ret;
ret = ParamName;
if (!ParamValue.empty())
{
ret += " = "+ParamValue;
}
if (!SubParams.empty())
{
ret += " ( ";
for (uint i=0; i<SubParams.size(); ++i)
{
if (i >0)
ret += " ";
ret += SubParams[i]->toString();
}
ret += " ) ";
}
return ret;
}
bool TParsedCommandLine::_parseParamList(const std::string &rawParamString)
{
CSString parsedString(rawParamString);
for (CSString part = parsedString.strtok(" \t", true, false);
!part.empty();
part = parsedString.strtok(" \t", true, false))
{
if (part[0] == '(')
{
// this is a sub parameter list
if (SubParams.empty() || SubParams.back()->ParamName.empty())
{
nlwarning("While parsing param string '%s', missing param header", rawParamString.c_str());
return false;
}
if (!SubParams.back()->ParamValue.empty())
{
nlwarning("While parsing param string '%s', Invalid sub param header '%s' for sub part '%s', must not define value",
rawParamString.c_str(),
SubParams.back()->ParamName.c_str(),
part.c_str());
return false;
}
if (part[part.size()-1] != ')')
{
nlwarning("While parsing param string '%s', Invalid sub param value '%s' missing closing ')'",
rawParamString.c_str(),
part.c_str());
return false;
}
part = part.stripBlockDelimiters();
if (!SubParams.back()->_parseParamList(part))
{
nlwarning("Error parsing sub param list for header '%s' in '%s'",
SubParams.back()->ParamName.c_str(),
rawParamString.c_str());
return false;
}
}
else if (part[part.size()-1] == ')')
{
nlwarning("While parsing param string '%s', Invalid param value '%s' : missing openning '('",
rawParamString.c_str(),
part.c_str());
return false;
}
else if (part[0] == '\"')
{
// this is a quoted parameter value
if (SubParams.empty() || !SubParams.back()->ParamValue.empty())
{
nlwarning("While parsing param string '%s', param '%s' already have the value '%s'",
rawParamString.c_str(),
SubParams.back()->ParamName.c_str(),
SubParams.back()->ParamValue.c_str());
return false;
}
SubParams.back()->ParamValue = part.unquote();
}
else
{
// this is a simple param
CSString name = part.splitTo('=', true, true);
if (name.empty())
{
nlwarning("Can't find param name for value '%s' in the param string '%s'",
part.c_str(),
rawParamString.c_str());
return false;
}
CSString value = part.strtok("=");
SubParams.push_back( new TParsedCommandLine() );
SubParams.back()->ParamName = name;
SubParams.back()->ParamValue = value;
}
}
return true;
}
const TParsedCommandLine *TParsedCommandLine::getParam(const std::string &name) const
{
vector<string> parts;
NLMISC::explode(name, string("."), parts);
return _getParam(parts.begin(), parts.end());
}
void TParsedCommandLine::setParam(const std::string &name, const std::string &value)
{
vector<string> parts;
NLMISC::explode(name, string("."), parts);
if (!name.empty())
{
// at least one part in the name
// check if sub ojbcct exist
TParsedCommandLine *sub = _getParam(parts.begin(), (parts.begin()+1));
if (sub == NULL)
{
TParsedCommandLine * newElem = new TParsedCommandLine();
newElem->ParamName = parts[0];
SubParams.push_back(newElem);
sub = SubParams.back();
}
if (!name.empty())
{
// name is more deep, need to resurse
parts.erase(parts.begin());
std::string subName;
join(parts, ".", subName);
sub->setParam(subName, value);
}
else
{
// last level, set the value
sub->ParamValue = value;
}
}
}
const TParsedCommandLine *TParsedCommandLine::_getParam(std::vector<std::string>::iterator it, std::vector<std::string>::iterator end) const
{
return const_cast<TParsedCommandLine&>(*this)._getParam(it, end);
}
TParsedCommandLine *TParsedCommandLine::_getParam(std::vector<std::string>::iterator it, std::vector<std::string>::iterator end)
{
if (it == end)
{
// end of recursion, we found the searched param
return this;
}
// look for sub param
for (uint i=0; i<SubParams.size(); ++i)
{
if (SubParams[i]->ParamName == *it)
return SubParams[i]->_getParam(++it, end);
}
// parameter not found
return NULL;
}
} // namespace NLMISC