1144 lines
26 KiB
1144 lines
26 KiB
/*==LICENSE==* |
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools |
|
Copyright (C) 2011 Cyan Worlds, Inc. |
|
|
|
This program is free software: you can redistribute it and/or modify |
|
it under the terms of the GNU 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 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/>. |
|
|
|
Additional permissions under GNU GPL version 3 section 7 |
|
|
|
If you modify this Program, or any covered work, by linking or |
|
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK, |
|
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent |
|
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK |
|
(or a modified version of those libraries), |
|
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA, |
|
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG |
|
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the |
|
licensors of this Program grant you additional |
|
permission to convey the resulting work. Corresponding Source for a |
|
non-source form of such a combination shall include the source code for |
|
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered |
|
work. |
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com |
|
or by snail mail at: |
|
Cyan Worlds, Inc. |
|
14617 N Newport Hwy |
|
Mead, WA 99021 |
|
|
|
*==LICENSE==*/ |
|
#include "plConfigInfo.h" |
|
|
|
#include "hsTypes.h" |
|
#include "hsStlUtils.h" |
|
#include <fstream> |
|
#include <algorithm> |
|
#include <stdarg.h> |
|
#include <sstream> |
|
|
|
const std::string& plConfigInfo::GlobalSection() |
|
{ |
|
static std::string section("global"); |
|
return section; |
|
} |
|
|
|
plConfigInfo::plConfigInfo() |
|
{ |
|
} |
|
|
|
plConfigInfo::plConfigInfo(const plConfigInfo & src) |
|
: fSections(src.fSections) |
|
{ |
|
} |
|
|
|
plConfigInfo & plConfigInfo::operator =(const plConfigInfo & src) |
|
{ |
|
fSections = src.fSections; |
|
return *this; |
|
} |
|
|
|
void plConfigInfo::Clear() |
|
{ |
|
fSections.clear(); |
|
} |
|
|
|
void plConfigInfo::RemoveSection(const std::string & section) |
|
{ |
|
fSections.erase(section.c_str()); |
|
} |
|
|
|
void plConfigInfo::RemoveKey(const std::string & section, const std::string & key) |
|
{ |
|
Sections::iterator si = fSections.find(section.c_str()); |
|
if (si != fSections.end()) |
|
fSections[section.c_str()].RemoveKey(key); |
|
} |
|
|
|
bool plConfigInfo::HasSection(const std::string & section) const |
|
{ |
|
return fSections.find(section.c_str())!=fSections.end(); |
|
} |
|
|
|
bool plConfigInfo::HasKey(const std::string & section, const std::string & key) |
|
{ |
|
Sections::iterator si = fSections.find(section.c_str()); |
|
if (si == fSections.end()) |
|
return false; |
|
return (si->second.HasKey(key)); |
|
} |
|
|
|
bool plConfigInfo::HasKeyAny(const std::string & key) |
|
{ |
|
for (Sections::iterator si=fSections.begin(); si!=fSections.end(); ++si) |
|
{ |
|
if (si->second.HasKey(key)) |
|
return true; |
|
} |
|
return false; |
|
} |
|
|
|
bool plConfigInfo::HasKeyIn(const std::string & key, const char * section1, ...) |
|
{ |
|
const char * section = section1; |
|
va_list va; |
|
va_start(va,section1); |
|
std::vector<std::string> sections; |
|
while (section) |
|
{ |
|
sections.push_back( section ); |
|
section = va_arg(va,const char *); |
|
} |
|
va_end(va); |
|
return HasKeyIn( key, sections ); |
|
} |
|
|
|
bool plConfigInfo::HasKeyIn(const std::string & key, const std::vector<std::string> & sections ) |
|
{ |
|
for ( int i=0; i<sections.size(); i++ ) |
|
{ |
|
const char * section = sections[i].c_str(); |
|
if (HasSection(section)) |
|
{ |
|
if (fSections[section].HasKey(key)) |
|
return true; |
|
} |
|
} |
|
return false; |
|
} |
|
|
|
bool plConfigInfo::KeyHasValue(const std::string & section, const std::string & key, const std::string & value) |
|
{ |
|
Sections::iterator si = fSections.find(section.c_str()); |
|
if (si == fSections.end()) |
|
return false; |
|
return si->second.KeyHasValue(key,value); |
|
} |
|
|
|
bool plConfigInfo::KeyHasValue(const std::string & section, const std::string & key, int value) |
|
{ |
|
Sections::iterator si = fSections.find(section.c_str()); |
|
if (si == fSections.end()) |
|
return false; |
|
return si->second.KeyHasValue(key,value); |
|
} |
|
|
|
bool plConfigInfo::KeyHasValue(const std::string & section, const std::string & key, double value) |
|
{ |
|
Sections::iterator si = fSections.find(section.c_str()); |
|
if (si == fSections.end()) |
|
return false; |
|
return si->second.KeyHasValue(key,value); |
|
} |
|
|
|
bool plConfigInfo::AddValue(const std::string & section, const std::string & key, const std::string & value, KAddValueMode mode) |
|
{ |
|
fSections[section.c_str()].AddValue(key,value,mode); |
|
return true; |
|
} |
|
|
|
bool plConfigInfo::AddValue(const std::string & section, const std::string & key, int value, KAddValueMode mode) |
|
{ |
|
char buf[20]; |
|
sprintf(buf, "%d", value); |
|
std::string v(buf); |
|
return AddValue(section,key,v,mode); |
|
} |
|
|
|
bool plConfigInfo::AddValue(const std::string & section, const std::string & key, double value, KAddValueMode mode) |
|
{ |
|
char buf[30]; |
|
sprintf(buf, "%f", value); |
|
std::string v(buf); |
|
return AddValue(section,key,v,mode); |
|
} |
|
|
|
bool plConfigInfo::AddValues(const std::string & section, const std::string & key, const std::vector<std::string> & values, KAddValueMode mode) |
|
{ |
|
return fSections[section.c_str()].AddValues(key,values); |
|
} |
|
|
|
plKeysAndValues plConfigInfo::GetSection(const std::string & section, bool & found) |
|
{ |
|
found = HasSection(section); |
|
if (found) |
|
return fSections[section.c_str()]; |
|
else |
|
return plKeysAndValues(); // empty |
|
} |
|
|
|
std::vector<std::string> plConfigInfo::GetSectionNames() |
|
{ |
|
std::vector<std::string> results; |
|
for (Sections::const_iterator ii=fSections.begin(); ii!=fSections.end(); ++ii) |
|
results.push_back(ii->first.c_str()); |
|
return results; |
|
} |
|
|
|
std::string plConfigInfo::GetValue(const std::string & section, const std::string & key, const std::string & defval, bool * outFound) const |
|
{ |
|
return fSections[section.c_str()].GetValue(key,defval,outFound); |
|
} |
|
|
|
int plConfigInfo::GetValue(const std::string & section, const std::string & key, int defval, bool * outFound) const |
|
{ |
|
return fSections[section.c_str()].GetValue(key,defval,outFound); |
|
} |
|
|
|
double plConfigInfo::GetValue(const std::string & section, const std::string & key, double defval, bool * outFound) const |
|
{ |
|
return fSections[section.c_str()].GetValue(key,defval,outFound); |
|
} |
|
|
|
std::vector<std::string> plConfigInfo::GetAllValues(const std::string & section, const std::string & key) const |
|
{ |
|
Sections::iterator si = fSections.find(section.c_str()); |
|
if (si != fSections.end()) |
|
return si->second.GetAllValues(key); |
|
std::vector<std::string> empty; |
|
return empty; |
|
} |
|
|
|
|
|
std::string plConfigInfo::GetValueAny(const std::string & key, const std::string & defval, bool * outFound) const |
|
{ |
|
if (outFound) *outFound=false; |
|
for (Sections::iterator si=fSections.begin(); si!=fSections.end(); ++si) |
|
if (si->second.HasKey(key)) |
|
return si->second.GetValue(key,defval,outFound); |
|
return defval; |
|
} |
|
|
|
int plConfigInfo::GetValueAny(const std::string & key, int defval, bool * outFound) const |
|
{ |
|
if (outFound) *outFound=false; |
|
for (Sections::iterator si=fSections.begin(); si!=fSections.end(); ++si) |
|
if (si->second.HasKey(key)) |
|
return si->second.GetValue(key,defval,outFound); |
|
return defval; |
|
} |
|
|
|
double plConfigInfo::GetValueAny(const std::string & key, double defval, bool * outFound) const |
|
{ |
|
if (outFound) *outFound=false; |
|
for (Sections::iterator si=fSections.begin(); si!=fSections.end(); ++si) |
|
if (si->second.HasKey(key)) |
|
return si->second.GetValue(key,defval,outFound); |
|
return defval; |
|
} |
|
|
|
std::vector<std::string> plConfigInfo::GetAllValuesAny(const std::string & key) const |
|
{ |
|
for (Sections::iterator si=fSections.begin(); si!=fSections.end(); ++si) |
|
if (si->second.HasKey(key)) |
|
return si->second.GetAllValues(key); |
|
std::vector<std::string> empty; |
|
return empty; |
|
} |
|
|
|
|
|
std::string plConfigInfo::GetValueIn(const std::string & key, const std::string & defval, bool * outFound, const char * section1, ...) const |
|
{ |
|
if (outFound) *outFound=false; |
|
const char * section = section1; |
|
va_list sections; |
|
va_start(sections,section1); |
|
while (section) |
|
{ |
|
if (HasSection(section)) |
|
{ |
|
plKeysAndValues & kv = fSections[section]; |
|
if (kv.HasKey(key)) |
|
return kv.GetValue(key,defval,outFound); |
|
} |
|
section = va_arg(sections,const char *); |
|
} |
|
va_end(sections); |
|
return defval; |
|
} |
|
|
|
std::string plConfigInfo::GetValueIn(const std::string & key, const std::string & defval, bool * outFound, const std::vector<std::string> & sections ) const |
|
{ |
|
if (outFound) *outFound=false; |
|
for ( int i=0; i<sections.size(); i++ ) |
|
{ |
|
if (HasSection(sections[i])) |
|
{ |
|
plKeysAndValues & kv = fSections[sections[i].c_str()]; |
|
if (kv.HasKey(key)) |
|
return kv.GetValue(key,defval,outFound); |
|
} |
|
} |
|
return defval; |
|
} |
|
|
|
int plConfigInfo::GetValueIn(const std::string & key, int defval, bool * outFound, const char * section1, ...) const |
|
{ |
|
if (outFound) *outFound=false; |
|
const char * section = section1; |
|
va_list sections; |
|
va_start(sections,section1); |
|
while (section) |
|
{ |
|
if (HasSection(section)) |
|
{ |
|
plKeysAndValues & kv = fSections[section]; |
|
if (kv.HasKey(key)) |
|
return kv.GetValue(key,defval,outFound); |
|
} |
|
section = va_arg(sections,const char *); |
|
} |
|
va_end(sections); |
|
return defval; |
|
} |
|
|
|
int plConfigInfo::GetValueIn(const std::string & key, int defval, bool * outFound, const std::vector<std::string> & sections ) const |
|
{ |
|
if (outFound) *outFound=false; |
|
for ( int i=0; i<sections.size(); i++ ) |
|
{ |
|
if (HasSection(sections[i])) |
|
{ |
|
plKeysAndValues & kv = fSections[sections[i].c_str()]; |
|
if (kv.HasKey(key)) |
|
return kv.GetValue(key,defval,outFound); |
|
} |
|
} |
|
return defval; |
|
} |
|
|
|
double plConfigInfo::GetValueIn(const std::string & key, double defval, bool * outFound, const char * section1, ...) const |
|
{ |
|
if (outFound) *outFound=false; |
|
const char * section = section1; |
|
va_list sections; |
|
va_start(sections,section1); |
|
while (section) |
|
{ |
|
if (HasSection(section)) |
|
{ |
|
plKeysAndValues & kv = fSections[section]; |
|
if (kv.HasKey(key)) |
|
return kv.GetValue(key,defval,outFound); |
|
} |
|
section = va_arg(sections,const char *); |
|
} |
|
va_end(sections); |
|
return defval; |
|
} |
|
|
|
double plConfigInfo::GetValueIn(const std::string & key, double defval, bool * outFound, const std::vector<std::string> & sections ) const |
|
{ |
|
if (outFound) *outFound=false; |
|
for ( int i=0; i<sections.size(); i++ ) |
|
{ |
|
if (HasSection(sections[i])) |
|
{ |
|
plKeysAndValues & kv = fSections[sections[i].c_str()]; |
|
if (kv.HasKey(key)) |
|
return kv.GetValue(key,defval,outFound); |
|
} |
|
} |
|
return defval; |
|
} |
|
|
|
std::vector<std::string> plConfigInfo::GetAllValuesIn(const std::string & key, const char * section1, ...) |
|
{ |
|
const char * section = section1; |
|
va_list sections; |
|
va_start(sections,section1); |
|
std::vector<std::string> result; |
|
while (section) |
|
{ |
|
if (HasSection(section)) |
|
{ |
|
plKeysAndValues & kv = fSections[section]; |
|
if (kv.HasKey(key)) |
|
{ |
|
std::vector<std::string> values = kv.GetAllValues(key); |
|
result.insert(result.end(),values.begin(),values.end()); |
|
} |
|
} |
|
section = va_arg(sections,const char *); |
|
} |
|
va_end(sections); |
|
return result; |
|
} |
|
|
|
bool plConfigInfo::GetSectionIterators(Sections::const_iterator & iter, Sections::const_iterator & end) const |
|
{ |
|
iter = fSections.begin(); |
|
end = fSections.end(); |
|
return true; |
|
} |
|
|
|
bool plConfigInfo::GetKeyIterators(const xtl::istring & section, Keys::const_iterator & iter, Keys::const_iterator & end) const |
|
{ |
|
Sections::const_iterator si = fSections.find(section); |
|
if (si==fSections.end()) |
|
return false; |
|
return fSections[section].GetKeyIterators(iter, end); |
|
} |
|
|
|
bool plConfigInfo::GetValueIterators(const xtl::istring & section, const xtl::istring & key, Values::const_iterator & iter, Values::const_iterator & end) const |
|
{ |
|
Sections::const_iterator si = fSections.find(section); |
|
if (si==fSections.end()) |
|
return false; |
|
return fSections[section].GetValueIterators(key, iter, end); |
|
} |
|
|
|
bool plConfigInfo::ReadFrom(plConfigSource * src, KAddValueMode mode) |
|
{ |
|
return src->ReadInto(*this,mode); |
|
} |
|
|
|
bool plConfigInfo::WriteTo(plConfigSource * src) |
|
{ |
|
return src->WriteOutOf(*this); |
|
} |
|
|
|
|
|
//////////////////////////////////////////////// |
|
|
|
void plConfigSource::SplitAt(std::string & key, std::string & value, char splitter, std::string & in) |
|
{ |
|
if(in.length() == 0) |
|
return; |
|
|
|
int t = in.find(splitter); |
|
if(t == std::string::npos) |
|
{ |
|
key = in; |
|
return; |
|
} |
|
|
|
key.assign(in.substr(0,t)); |
|
value.assign(in.substr(t+1,in.size()-t-1)); |
|
} |
|
|
|
|
|
bool plConfigSource::ReadString(const std::string & in) |
|
{ |
|
std::string work = in; |
|
xtl::trim(work); |
|
|
|
// comment |
|
if (work[0] == '#') |
|
return true; |
|
|
|
// comment |
|
if (work[0] == ';') |
|
return true; |
|
|
|
// section |
|
if (work[0] == '[') |
|
{ |
|
int close = work.find_first_of("]"); |
|
if(close == std::string::npos) |
|
return false; |
|
fCurrSection = work.substr(1,close-1); |
|
fEffectiveSection = fCurrSection; |
|
return true; |
|
} |
|
|
|
// key=value |
|
std::string key, value; |
|
SplitAt(key, value, '=', work); |
|
|
|
// dot notation makes section change for this key=value only. |
|
int t = key.find('.'); |
|
if (t>0 && t<key.size()-1) |
|
{ |
|
fEffectiveSection.assign(key.substr(0,t)); |
|
key.assign(key.substr(t+1)); |
|
} |
|
|
|
bool ret=ReadPair(key, value); |
|
fEffectiveSection = fCurrSection; |
|
|
|
if(ret && strcmp("LoadIni",key.c_str()) == 0) |
|
{ |
|
ret = ReadSubSource( value.c_str() ); |
|
} |
|
|
|
return ret; |
|
} |
|
|
|
bool plConfigSource::ReadPair(std::string & key, std::string & value) |
|
{ |
|
hsAssert(fConfigInfo, "plConfigSource::ProcessPair: fConfigInfo not set."); |
|
|
|
xtl::trim(key); |
|
xtl::trim(value); |
|
xtl::trim(value,"\"'"); |
|
|
|
if (key.size() == 0) |
|
return true; |
|
|
|
return fConfigInfo->AddValue(fEffectiveSection, key, value, fAddMode); |
|
} |
|
|
|
|
|
bool plConfigSource::ReadList(char ** l) |
|
{ |
|
while(*l != NULL) |
|
{ |
|
ReadString(*l); |
|
l++; |
|
} |
|
return true; |
|
} |
|
|
|
|
|
bool plConfigSource::ReadInto(plConfigInfo & configInfo, KAddValueMode mode) |
|
{ |
|
fConfigInfo = &configInfo; |
|
fAddMode = mode; |
|
return true; |
|
} |
|
|
|
|
|
bool plConfigSource::WriteOutOf(plConfigInfo & configInfo) |
|
{ |
|
fConfigInfo = &configInfo; |
|
return true; |
|
} |
|
|
|
|
|
///////////////////////////////////////////////// |
|
|
|
plCmdLineConfigSource::plCmdLineConfigSource(int argc, char ** argv, const char * mySection) |
|
: fArgc(argc) |
|
, fArgv(argv) |
|
, fMySection(mySection?mySection:"") |
|
{} |
|
|
|
|
|
bool plCmdLineConfigSource::ReadInto(plConfigInfo & configInfo, KAddValueMode mode) |
|
{ |
|
int argc = fArgc; |
|
char ** argv = fArgv; |
|
|
|
if (!plConfigSource::ReadInto(configInfo, mode)) |
|
return false; |
|
|
|
fCurrSection = fMySection; |
|
fEffectiveSection = fCurrSection; |
|
|
|
if(argc < 1) |
|
return true; |
|
|
|
fConfigInfo->AddValue(fEffectiveSection.c_str(), "ARGV0", *argv, fAddMode); |
|
argc--; |
|
argv++; |
|
|
|
while(argc > 0) |
|
{ |
|
if(ReadString(*argv) != true) |
|
{ |
|
// TODO: log error here |
|
return false; |
|
} |
|
argv++; |
|
argc--; |
|
} |
|
|
|
return true; |
|
} |
|
|
|
///////////////////////////////////////////////// |
|
|
|
plEnvConfigSource::plEnvConfigSource(char ** envp, const char * mySection) |
|
: fEnvp(envp) |
|
, fMySection(mySection?mySection:"") |
|
{} |
|
|
|
|
|
bool plEnvConfigSource::ReadInto(plConfigInfo & configInfo, KAddValueMode mode) |
|
{ |
|
if (!plConfigSource::ReadInto(configInfo, mode)) |
|
return false; |
|
|
|
if (fEnvp != NULL) |
|
{ |
|
fCurrSection = fMySection; |
|
fEffectiveSection = fCurrSection; |
|
return ReadList(fEnvp); |
|
} |
|
|
|
return true; |
|
} |
|
|
|
|
|
///////////////////////////////////////////////// |
|
|
|
plIniConfigSource::plIniConfigSource(const char * iniFileName) |
|
: fFileName(iniFileName) |
|
{} |
|
|
|
|
|
bool plIniConfigSource::ReadInto(plConfigInfo & configInfo, KAddValueMode mode) |
|
{ |
|
if (!plConfigSource::ReadInto(configInfo, mode)) |
|
return false; |
|
|
|
fCurrSection = plConfigInfo::GlobalSection(); |
|
fEffectiveSection = fCurrSection; |
|
|
|
if(fFileName.size() < 2) |
|
return false; |
|
|
|
|
|
std::ifstream file; |
|
file.open(fFileName.c_str()); |
|
|
|
if(!file.is_open()) |
|
{ |
|
// TODO log error here |
|
return false; |
|
} |
|
|
|
char buf[4096]; |
|
|
|
while (!file.eof()) |
|
{ |
|
file.getline(buf, 4096); |
|
|
|
if(!ReadString(buf)) |
|
{ |
|
// TODO log warning here |
|
} |
|
} |
|
file.close(); |
|
|
|
return true; |
|
} |
|
|
|
bool plIniConfigSource::WriteOutOf(plConfigInfo & configInfo) |
|
{ |
|
if (!plConfigSource::WriteOutOf(configInfo)) |
|
return false; |
|
|
|
std::ofstream file; |
|
file.open(fFileName.c_str()); |
|
|
|
if(!file.is_open()) |
|
{ |
|
// TODO log error here |
|
return false; |
|
} |
|
|
|
file |
|
<< "# This is an auto-generated file." << std::endl |
|
<< std::endl |
|
; |
|
|
|
plConfigInfo::Sections::const_iterator si, se; |
|
plConfigInfo::Keys::const_iterator ki, ke; |
|
plConfigInfo::Values::const_iterator vi, ve; |
|
|
|
fConfigInfo->GetSectionIterators(si,se); |
|
for (si; si!=se; ++si) |
|
{ |
|
file << std::endl << "[" << si->first.c_str() << "]"<< std::endl; |
|
if (fConfigInfo->GetKeyIterators(si->first, ki, ke)) |
|
for (ki; ki!=ke; ++ki) |
|
{ |
|
if (fConfigInfo->GetValueIterators(si->first, ki->first, vi, ve)) |
|
for (vi; vi!=ve; ++vi) |
|
{ |
|
file << ki->first.c_str() << "=" << vi->c_str() << std::endl; |
|
} |
|
} |
|
} |
|
|
|
file.close(); |
|
|
|
return true; |
|
} |
|
|
|
|
|
///////////////////////////////////////////////// |
|
|
|
plIniStreamConfigSource::plIniStreamConfigSource(hsStream * stream) |
|
: fStream(stream) |
|
{} |
|
|
|
|
|
bool plIniStreamConfigSource::ReadInto(plConfigInfo & configInfo, KAddValueMode mode) |
|
{ |
|
if (!plConfigSource::ReadInto(configInfo, mode)) |
|
return false; |
|
|
|
fCurrSection = "global"; |
|
fEffectiveSection = fCurrSection; |
|
|
|
if ( !fStream ) |
|
return false; |
|
|
|
char buf[4096]; |
|
|
|
while (!fStream->AtEnd()) |
|
{ |
|
fStream->ReadLn( buf, sizeof(buf) ); |
|
|
|
if(!ReadString(buf)) |
|
{ |
|
// TODO log warning here |
|
} |
|
} |
|
|
|
return true; |
|
} |
|
|
|
bool plIniStreamConfigSource::WriteOutOf(plConfigInfo & configInfo) |
|
{ |
|
if (!plConfigSource::WriteOutOf(configInfo)) |
|
return false; |
|
|
|
if ( !fStream ) |
|
return false; |
|
|
|
std::stringstream ss; |
|
|
|
plConfigInfo::Sections::const_iterator si, se; |
|
plConfigInfo::Keys::const_iterator ki, ke; |
|
plConfigInfo::Values::const_iterator vi, ve; |
|
|
|
fConfigInfo->GetSectionIterators(si,se); |
|
for (si; si!=se; ++si) |
|
{ |
|
ss << std::endl << "[" << si->first.c_str() << "]"<< std::endl; |
|
if (fConfigInfo->GetKeyIterators(si->first, ki, ke)) |
|
for (ki; ki!=ke; ++ki) |
|
{ |
|
if (fConfigInfo->GetValueIterators(si->first, ki->first, vi, ve)) |
|
for (vi; vi!=ve; ++vi) |
|
{ |
|
ss << ki->first.c_str() << "=" << vi->c_str() << std::endl; |
|
} |
|
} |
|
} |
|
|
|
fStream->WriteString( ss.str().c_str() ); |
|
|
|
return true; |
|
} |
|
|
|
|
|
///////////////////////////////////////////////// |
|
|
|
plIniSectionConfigSource::plIniSectionConfigSource(const char * iniFileName, std::vector<std::string> & sections) |
|
: plIniConfigSource(iniFileName) |
|
{ |
|
for (int i=0; i<sections.size(); i++) |
|
fSections.push_back(sections[i].c_str()); |
|
} |
|
|
|
|
|
bool plIniSectionConfigSource::ReadPair(std::string & key, std::string & value) |
|
{ |
|
hsAssert(fConfigInfo, "plConfigSource::ProcessPair: fConfigInfo not set."); |
|
|
|
// the current section must be in list of sections. |
|
std::vector<xtl::istring>::iterator ii = std::find(fSections.begin(), fSections.end(), fCurrSection.c_str()); |
|
|
|
if (ii==fSections.end()) |
|
return true; |
|
|
|
xtl::trim(key); |
|
xtl::trim(value); |
|
xtl::trim(value,"\"'"); |
|
|
|
if (key.size() == 0) |
|
return true; |
|
|
|
if (key == "section") |
|
fSections.push_back(value.c_str()); |
|
|
|
return fConfigInfo->AddValue(fEffectiveSection, key, value, fAddMode); |
|
} |
|
|
|
|
|
bool plIniSectionConfigSource::ReadSubSource( const char * name ) |
|
{ |
|
std::vector<std::string> sections; |
|
for ( int i=0; i<fSections.size(); i++ ) |
|
sections.push_back( fSections[i].c_str() ); |
|
return fConfigInfo->ReadFrom(&plIniSectionConfigSource( name, sections )); |
|
} |
|
|
|
///////////////////////////////////////////////// |
|
|
|
plIniNoSectionsConfigSource::plIniNoSectionsConfigSource(const char * filename) |
|
: fFileName(filename) |
|
{ |
|
fEffectiveSection = fCurrSection = ""; |
|
} |
|
|
|
bool plIniNoSectionsConfigSource::ReadString(const std::string & in) |
|
{ |
|
std::string work = in; |
|
xtl::trim(work); |
|
|
|
// ignore comments |
|
if (work[0]=='#' || work[0]==';') |
|
return true; |
|
|
|
// ignore sections |
|
if (work[0] == '[') |
|
return true; |
|
|
|
// parse key value |
|
std::string key, value; |
|
SplitAt(key, value, '=', work); |
|
|
|
return ReadPair(key, value); |
|
} |
|
|
|
bool plIniNoSectionsConfigSource::ReadInto(plConfigInfo & configInfo, KAddValueMode mode) |
|
{ |
|
if (!plConfigSource::ReadInto(configInfo, mode)) |
|
return false; |
|
|
|
if(fFileName.size() < 2) |
|
return false; |
|
|
|
std::ifstream file; |
|
file.open(fFileName.c_str()); |
|
|
|
if(!file.is_open()) |
|
{ |
|
// TODO log error here |
|
return false; |
|
} |
|
|
|
char buf[4096]; |
|
|
|
while (!file.eof()) |
|
{ |
|
file.getline(buf, 4096); |
|
|
|
if(!ReadString(buf)) |
|
{ |
|
// TODO log warning here |
|
} |
|
} |
|
file.close(); |
|
|
|
return true; |
|
} |
|
|
|
|
|
bool plIniNoSectionsConfigSource::WriteOutOf(plConfigInfo & configInfo) |
|
{ |
|
if (!plConfigSource::WriteOutOf(configInfo)) |
|
return false; |
|
|
|
std::ofstream file; |
|
file.open(fFileName.c_str()); |
|
|
|
if(!file.is_open()) |
|
{ |
|
// TODO log error here |
|
return false; |
|
} |
|
|
|
file |
|
<< "# This is an auto-generated file." << std::endl |
|
<< std::endl |
|
; |
|
|
|
plConfigInfo::Sections::const_iterator si, se; |
|
plConfigInfo::Keys::const_iterator ki, ke; |
|
plConfigInfo::Values::const_iterator vi, ve; |
|
|
|
fConfigInfo->GetSectionIterators(si,se); |
|
for (si; si!=se; ++si) |
|
{ |
|
if (fConfigInfo->GetKeyIterators(si->first, ki, ke)) |
|
for (ki; ki!=ke; ++ki) |
|
{ |
|
if (fConfigInfo->GetValueIterators(si->first, ki->first, vi, ve)) |
|
for (vi; vi!=ve; ++vi) |
|
{ |
|
file << ki->first.c_str() << "=" << vi->c_str() << std::endl; |
|
} |
|
} |
|
} |
|
|
|
file.close(); |
|
|
|
return true; |
|
} |
|
|
|
|
|
///////////////////////////////////////////////// |
|
|
|
bool plDebugConfigSource::WriteOutOf(plConfigInfo & configInfo) |
|
{ |
|
if (!plConfigSource::WriteOutOf(configInfo)) |
|
return false; |
|
|
|
plConfigInfo::Sections::const_iterator si, se; |
|
plConfigInfo::Keys::const_iterator ki, ke; |
|
plConfigInfo::Values::const_iterator vi, ve; |
|
|
|
char buf[1024]; |
|
fConfigInfo->GetSectionIterators(si,se); |
|
for (si; si!=se; ++si) |
|
{ |
|
sprintf(buf,"\n[%s]\n",si->first.c_str()); |
|
hsStatusMessage(buf); |
|
if (fConfigInfo->GetKeyIterators(si->first, ki, ke)) |
|
for (ki; ki!=ke; ++ki) |
|
{ |
|
if (fConfigInfo->GetValueIterators(si->first, ki->first, vi, ve)) |
|
for (vi; vi!=ve; ++vi) |
|
{ |
|
sprintf(buf,"%s=%s\n",ki->first.c_str(),vi->c_str()); |
|
hsStatusMessage(buf); |
|
} |
|
} |
|
} |
|
|
|
return true; |
|
} |
|
|
|
//////////////////////////////////////////////////////////////////// |
|
|
|
void plConfigValueBase::ConfigRead(plConfigInfo * opts) |
|
{ |
|
if (fReadEvaluate()) |
|
{ |
|
std::string value; |
|
bool found; |
|
value = opts->GetValue(GetConfigGroup(),GetConfigName(),"",&found); |
|
if (found) |
|
SetValue(fReadModify(value).c_str()); |
|
} |
|
} |
|
|
|
void plConfigValueBase::ConfigWrite(plConfigInfo * opts) |
|
{ |
|
if (fWriteEvaluate()) |
|
{ |
|
opts->AddValue(GetConfigGroup(),GetConfigName(),fWriteModify(GetValue()),kAlwaysAdd); |
|
} |
|
} |
|
|
|
void plConfigValueBase::SetValue(const char * value) |
|
{ |
|
ISetValue(fSetModify(value).c_str()); |
|
} |
|
|
|
std::string plConfigValueBase::GetValue() const |
|
{ |
|
return fGetModify(IGetValue()); |
|
} |
|
|
|
void plConfigValueBase::SetReadEvaluate(plClass * targetObj, TEvaluate evalFunc) |
|
{ |
|
fReadEvaluate = plEvaluate(targetObj,evalFunc); |
|
} |
|
|
|
void plConfigValueBase::SetWriteEvaluate(plClass * targetObj, TEvaluate evalFunc) |
|
{ |
|
fWriteEvaluate = plEvaluate(targetObj,evalFunc); |
|
} |
|
|
|
void plConfigValueBase::SetWriteEvaluate(plClass * targetObj, TEvaluateConst evalFunc) |
|
{ |
|
fWriteEvaluate = plEvaluate(targetObj,evalFunc); |
|
} |
|
|
|
void plConfigValueBase::SetReadModify(plClass * targetObj, TModify modifyFunc) |
|
{ |
|
fReadModify = plModify(targetObj,modifyFunc); |
|
} |
|
|
|
void plConfigValueBase::SetWriteModify(plClass * targetObj, TModify modifyFunc) |
|
{ |
|
fWriteModify = plModify(targetObj,modifyFunc); |
|
} |
|
|
|
void plConfigValueBase::SetGetModify(plClass * targetObj, TModify modifyFunc) |
|
{ |
|
fGetModify = plModify(targetObj,modifyFunc); |
|
} |
|
|
|
void plConfigValueBase::SetSetModify(plClass * targetObj, TModify modifyFunc) |
|
{ |
|
fSetModify = plModify(targetObj,modifyFunc); |
|
} |
|
|
|
|
|
//////////////////////////////////////////////////////////////////// |
|
|
|
plConfigGroup::plConfigGroup(const char * groupName) |
|
: fGroupName(groupName) |
|
{} |
|
|
|
bool plConfigGroup::Read(plConfigSource * src) |
|
{ |
|
if (!fOpts.ReadFrom(src)) |
|
return false; |
|
for (int i=0; i<fItems.size(); i++) |
|
fItems[i]->ConfigRead(&fOpts); |
|
return true; |
|
} |
|
|
|
bool plConfigGroup::Write(plConfigSource * src) |
|
{ |
|
for (int i=0; i<fItems.size(); i++) |
|
fItems[i]->ConfigWrite(&fOpts); |
|
return fOpts.WriteTo(src); |
|
} |
|
|
|
void plConfigGroup::AddItem(plConfigValueBase * item, const char * name) |
|
{ |
|
item->SetConfigGroup(fGroupName.c_str()); |
|
if (name) |
|
item->SetConfigName(name); |
|
fItems.push_back(item); |
|
} |
|
|
|
//////////////////////////////////////////////////////////////////// |
|
|
|
plConfigAggregateValue::plConfigAggregateValue( |
|
const char * name, |
|
plConfigValueBase * item1, |
|
plConfigValueBase * item2, |
|
plConfigValueBase * item3, |
|
plConfigValueBase * item4, |
|
plConfigValueBase * item5, |
|
plConfigValueBase * item6, |
|
plConfigValueBase * item7) |
|
{ |
|
SetConfigName(name); |
|
AddItems(item1,item2,item3,item4,item5,item6,item7); |
|
} |
|
|
|
void plConfigAggregateValue::AddItems( |
|
plConfigValueBase * item1, |
|
plConfigValueBase * item2, |
|
plConfigValueBase * item3, |
|
plConfigValueBase * item4, |
|
plConfigValueBase * item5, |
|
plConfigValueBase * item6, |
|
plConfigValueBase * item7) |
|
{ |
|
fItems.clear(); |
|
if (item1) AddItem(item1); |
|
if (item2) AddItem(item2); |
|
if (item3) AddItem(item3); |
|
if (item4) AddItem(item4); |
|
if (item5) AddItem(item5); |
|
if (item6) AddItem(item6); |
|
if (item7) AddItem(item7); |
|
} |
|
|
|
void plConfigAggregateValue::ISetValue(const char * value) |
|
{ |
|
std::string work = value; |
|
int p=0,i=0; |
|
do |
|
{ |
|
xtl::trim(work); |
|
p = work.find(" "); |
|
fItems[i]->SetValue(work.substr(0,p).c_str()); |
|
work.erase(0,p); |
|
i++; |
|
} while (i<fItems.size() && p!=std::string::npos); |
|
} |
|
|
|
std::string plConfigAggregateValue::IGetValue() const |
|
{ |
|
std::string value; |
|
for (int i=0; i<fItems.size(); i++) |
|
{ |
|
value.append(fItems[i]->GetValue()); |
|
value.append(" "); |
|
} |
|
return xtl::trim(value); |
|
} |
|
|
|
void plConfigAggregateValue::AddItem(plConfigValueBase * item) |
|
{ |
|
fItems.push_back(item); |
|
} |
|
|
|
//////////////////////////////////////////////////////////////////// |
|
|
|
plWWWAuthenticateConfigSource::plWWWAuthenticateConfigSource(const std::string& auth) |
|
: fAuth(auth) |
|
{ |
|
fEffectiveSection = fCurrSection = ""; |
|
} |
|
|
|
bool plWWWAuthenticateConfigSource::ReadInto(plConfigInfo & configInfo, KAddValueMode mode) |
|
{ |
|
if (!plConfigSource::ReadInto(configInfo, mode)) |
|
return false; |
|
|
|
fCurrSection = "global"; |
|
fEffectiveSection = fCurrSection; |
|
|
|
unsigned int i = 0; |
|
|
|
while (i < fAuth.size()) |
|
{ |
|
bool inQuote = false; |
|
unsigned int begin = i,end; |
|
while (i < fAuth.size() |
|
&& ((fAuth[i] != ',' && !inQuote) || inQuote)) |
|
{ |
|
if (fAuth[i] == '"') |
|
inQuote = ! inQuote; |
|
i++; |
|
} |
|
end = i; |
|
|
|
std::string buf; |
|
buf.assign(fAuth,begin,end-begin); |
|
if(!ReadString(buf.c_str())) |
|
{ |
|
// TODO log warning here |
|
} |
|
i++; |
|
} |
|
|
|
return true; |
|
}
|
|
|