// 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 "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::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; i0) 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 parts; NLMISC::explode(name, string("."), parts); return _getParam(parts.begin(), parts.end()); } void TParsedCommandLine::setParam(const std::string &name, const std::string &value) { vector 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::iterator it, std::vector::iterator end) const { return const_cast(*this)._getParam(it, end); } TParsedCommandLine *TParsedCommandLine::_getParam(std::vector::iterator it, std::vector::iterator end) { if (it == end) { // end of recursion, we found the searched param return this; } // look for sub param for (uint i=0; iParamName == *it) return SubParams[i]->_getParam(++it, end); } // parameter not found return NULL; } } // namespace NLMISC