yate/engine/NamedList.cpp

376 lines
10 KiB
C++

/**
* NamedList.cpp
* This file is part of the YATE Project http://YATE.null.ro
*
* Yet Another Telephony Engine - a fully featured software PBX and IVR
* Copyright (C) 2004-2014 Null Team
*
* This software is distributed under multiple licenses;
* see the COPYING file in the main directory for licensing
* information for this specific distribution.
*
* This use of this software may be subject to additional restrictions.
* See the LEGAL file in the main directory for details.
*
* 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.
*/
#include "yateclass.h"
using namespace TelEngine;
static const NamedList s_empty("");
const NamedList& NamedList::empty()
{
return s_empty;
}
NamedList::NamedList(const char* name)
: String(name)
{
}
NamedList::NamedList(const NamedList& original)
: String(original)
{
ObjList* dest = &m_params;
for (const ObjList* l = original.m_params.skipNull(); l; l = l->skipNext()) {
const NamedString* p = static_cast<const NamedString*>(l->get());
dest = dest->append(new NamedString(p->name(),*p));
}
}
NamedList::NamedList(const char* name, const NamedList& original, const String& prefix)
: String(name)
{
copySubParams(original,prefix);
}
NamedList& NamedList::operator=(const NamedList& value)
{
String::operator=(value);
clearParams();
return copyParams(value);
}
void* NamedList::getObject(const String& name) const
{
if (name == YATOM("NamedList"))
return const_cast<NamedList*>(this);
return String::getObject(name);
}
NamedList& NamedList::addParam(NamedString* param)
{
XDebug(DebugInfo,"NamedList::addParam(%p) [\"%s\",\"%s\"]",
param,(param ? param->name().c_str() : ""),TelEngine::c_safe(param));
if (param)
m_params.append(param);
return *this;
}
NamedList& NamedList::addParam(const char* name, const char* value, bool emptyOK)
{
XDebug(DebugInfo,"NamedList::addParam(\"%s\",\"%s\",%s)",name,value,String::boolText(emptyOK));
if (emptyOK || !TelEngine::null(value))
m_params.append(new NamedString(name, value));
return *this;
}
NamedList& NamedList::setParam(const String& name, const char* value)
{
XDebug(DebugInfo,"NamedList::setParam(\"%s\",\"%s\")",name.c_str(),value);
ObjList *p = m_params.skipNull();
while (p) {
NamedString *s = static_cast<NamedString*>(p->get());
if (s->name() == name) {
*s = value;
return *this;
}
ObjList* next = p->skipNext();
if (next)
p = next;
else
break;
}
if (p)
p->append(new NamedString(name,value));
else
m_params.append(new NamedString(name,value));
return *this;
}
NamedList& NamedList::clearParam(const String& name, char childSep)
{
XDebug(DebugInfo,"NamedList::clearParam(\"%s\",'%.1s')",
name.c_str(),&childSep);
String tmp;
if (childSep)
tmp << name << childSep;
ObjList *p = &m_params;
while (p) {
NamedString *s = static_cast<NamedString *>(p->get());
if (s && ((s->name() == name) || s->name().startsWith(tmp)))
p->remove();
else
p = p->next();
}
return *this;
}
// Remove a specific parameter
NamedList& NamedList::clearParam(NamedString* param, bool delParam)
{
if (!param)
return *this;
ObjList* o = m_params.find(param);
if (o)
o->remove(delParam);
XDebug(DebugInfo,"NamedList::clearParam(%p) found=%p",param,o);
return *this;
}
NamedList& NamedList::copyParam(const NamedList& original, const String& name, char childSep)
{
XDebug(DebugInfo,"NamedList::copyParam(%p,\"%s\",'%.1s')",
&original,name.c_str(),&childSep);
if (!childSep) {
// faster and simpler - used in most cases
const NamedString* s = original.getParam(name);
return s ? setParam(name,*s) : clearParam(name);
}
clearParam(name,childSep);
String tmp;
tmp << name << childSep;
ObjList* dest = &m_params;
for (const ObjList* l = original.m_params.skipNull(); l; l = l->skipNext()) {
const NamedString* s = static_cast<const NamedString*>(l->get());
if ((s->name() == name) || s->name().startsWith(tmp))
dest = dest->append(new NamedString(s->name(),*s));
}
return *this;
}
NamedList& NamedList::copyParams(const NamedList& original)
{
XDebug(DebugInfo,"NamedList::copyParams(%p) [%p]",&original,this);
for (const ObjList* l = original.m_params.skipNull(); l; l = l->skipNext()) {
const NamedString* p = static_cast<const NamedString*>(l->get());
setParam(p->name(),*p);
}
return *this;
}
NamedList& NamedList::copyParams(const NamedList& original, ObjList* list, char childSep)
{
XDebug(DebugInfo,"NamedList::copyParams(%p,%p,'%.1s') [%p]",
&original,list,&childSep,this);
for (; list; list = list->next()) {
GenObject* obj = list->get();
if (!obj)
continue;
String name = obj->toString();
name.trimBlanks();
if (name)
copyParam(original,name,childSep);
}
return *this;
}
NamedList& NamedList::copyParams(const NamedList& original, const String& list, char childSep)
{
XDebug(DebugInfo,"NamedList::copyParams(%p,\"%s\",'%.1s') [%p]",
&original,list.c_str(),&childSep,this);
ObjList* l = list.split(',',false);
if (l) {
copyParams(original,l,childSep);
l->destruct();
}
return *this;
}
NamedList& NamedList::copySubParams(const NamedList& original, const String& prefix,
bool skipPrefix, bool replace)
{
XDebug(DebugInfo,"NamedList::copySubParams(%p,\"%s\",%s,%s) [%p]",
&original,prefix.c_str(),String::boolText(skipPrefix),
String::boolText(replace),this);
if (prefix) {
unsigned int offs = skipPrefix ? prefix.length() : 0;
ObjList* dest = &m_params;
for (const ObjList* l = original.m_params.skipNull(); l; l = l->skipNext()) {
const NamedString* s = static_cast<const NamedString*>(l->get());
if (s->name().startsWith(prefix)) {
const char* name = s->name().c_str() + offs;
if (!*name)
continue;
if (!replace)
dest = dest->append(new NamedString(name,*s));
else if (offs)
setParam(name,*s);
else
setParam(s->name(),*s);
}
}
}
return *this;
}
bool NamedList::hasSubParams(const char* prefix) const
{
XDebug(DebugInfo,"NamedList::hasSubParams(\"%s\") [%p]",prefix,this);
if (!TelEngine::null(prefix)) {
for (const ObjList* l = m_params.skipNull(); l; l = l->skipNext()) {
const NamedString* s = static_cast<const NamedString*>(l->get());
if (s->name().startsWith(prefix))
return true;
}
}
return false;
}
void NamedList::dump(String& str, const char* separator, char quote, bool force) const
{
if (force && str.null())
str << separator;
str << quote << *this << quote;
const ObjList *p = m_params.skipNull();
for (; p; p = p->skipNext()) {
const NamedString* s = static_cast<const NamedString *>(p->get());
String tmp;
tmp << quote << s->name() << quote << "=" << quote << *s << quote;
str.append(tmp,separator);
}
}
int NamedList::getIndex(const NamedString* param) const
{
if (!param)
return -1;
const ObjList *p = &m_params;
for (int i=0; p; p=p->next(),i++) {
if (static_cast<const NamedString *>(p->get()) == param)
return i;
}
return -1;
}
int NamedList::getIndex(const String& name) const
{
const ObjList *p = &m_params;
for (int i=0; p; p=p->next(),i++) {
NamedString *s = static_cast<NamedString *>(p->get());
if (s && (s->name() == name))
return i;
}
return -1;
}
NamedString* NamedList::getParam(const String& name) const
{
XDebug(DebugInfo,"NamedList::getParam(\"%s\")",name.c_str());
const ObjList *p = m_params.skipNull();
for (; p; p=p->skipNext()) {
NamedString *s = static_cast<NamedString *>(p->get());
if (s->name() == name)
return s;
}
return 0;
}
NamedString* NamedList::getParam(unsigned int index) const
{
XDebug(DebugInfo,"NamedList::getParam(%u)",index);
return static_cast<NamedString *>(m_params[index]);
}
const String& NamedList::operator[](const String& name) const
{
const String* s = getParam(name);
return s ? *s : String::empty();
}
const char* NamedList::getValue(const String& name, const char* defvalue) const
{
XDebug(DebugInfo,"NamedList::getValue(\"%s\",\"%s\")",name.c_str(),defvalue);
const NamedString *s = getParam(name);
return s ? s->c_str() : defvalue;
}
int NamedList::getIntValue(const String& name, int defvalue, int minvalue, int maxvalue,
bool clamp) const
{
const NamedString *s = getParam(name);
return s ? s->toInteger(defvalue,0,minvalue,maxvalue,clamp) : defvalue;
}
int NamedList::getIntValue(const String& name, const TokenDict* tokens, int defvalue) const
{
const NamedString *s = getParam(name);
return s ? s->toInteger(tokens,defvalue) : defvalue;
}
double NamedList::getDoubleValue(const String& name, double defvalue) const
{
const NamedString *s = getParam(name);
return s ? s->toDouble(defvalue) : defvalue;
}
bool NamedList::getBoolValue(const String& name, bool defvalue) const
{
const NamedString *s = getParam(name);
return s ? s->toBoolean(defvalue) : defvalue;
}
int NamedList::replaceParams(String& str, bool sqlEsc, char extraEsc) const
{
int p1 = 0;
int cnt = 0;
while ((p1 = str.find("${",p1)) >= 0) {
int p2 = str.find('}',p1+2);
if (p2 > 0) {
String def;
String tmp = str.substr(p1+2,p2-p1-2);
tmp.trimBlanks();
int pq = tmp.find('$');
if (pq >= 0) {
// param is in ${<name>$<default>} format
def = tmp.substr(pq+1).trimBlanks();
tmp = tmp.substr(0,pq).trimBlanks();
}
DDebug(DebugAll,"NamedList replacing parameter '%s' [%p]",tmp.c_str(),this);
const String* ns = getParam(tmp);
if (ns) {
if (sqlEsc) {
const DataBlock* data = 0;
if (ns->null()) {
NamedPointer* np = YOBJECT(NamedPointer,ns);
if (np)
data = YOBJECT(DataBlock,np->userData());
}
if (data)
tmp = data->sqlEscape(extraEsc);
else
tmp = ns->sqlEscape(extraEsc);
}
else
tmp = *ns;
}
else
tmp = def;
str = str.substr(0,p1) + tmp + str.substr(p2+1);
// advance search offset past the string we just replaced
p1 += tmp.length();
cnt++;
}
else
return -1;
}
return cnt;
}
/* vi: set ts=8 sw=4 sts=4 noet: */