emailrelay/src/glib/gstr.cpp
Graeme Walker 2911440f23 v0.9.6
2001-12-18 12:00:00 +00:00

486 lines
10 KiB
C++

//
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
//
// 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 2 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, write to the Free Software
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
//
// ===
//
// gstr.cpp
//
#include "gdef.h"
#include "gstr.h"
#include "gdebug.h"
#include <cmath>
#include <ctype.h>
#include <iomanip>
#include <climits>
#include <string>
#include <sstream>
#include <xlocale>
bool G::Str::replace( std::string &s , const std::string &from ,
const std::string &to , size_t *pos_p )
{
if( from.length() == 0 )
return false ;
size_t pos = pos_p == NULL ? 0 : *pos_p ;
if( pos >= s.length() )
return false ;
pos = s.find( from , pos ) ;
if( pos == std::string::npos )
{
return false ;
}
else
{
s.replace( pos , from.length() , to ) ;
if( pos_p != NULL )
*pos_p = pos + to.length() ;
return true ;
}
}
size_t G::Str::replaceAll( std::string &s , const std::string &from ,
const std::string &to )
{
size_t count = 0U ;
for( size_t pos = 0U ; replace( s , from , to , &pos ) ; count++ )
; // no-op
return count ;
}
void G::Str::trimLeft( std::string & s , const std::string & ws )
{
size_t n = s.find_first_not_of( ws ) ;
if( n == std::string::npos )
s = std::string() ;
else if( n != 0U )
s.erase( 0U , n ) ;
}
void G::Str::trimRight( std::string & s , const std::string & ws )
{
size_t n = s.find_last_not_of( ws ) ;
if( n == std::string::npos )
s = std::string() ;
else if( n != 0U )
s.erase( n+1U , s.length()-n-1U ) ;
}
void G::Str::trim( std::string & s , const std::string & ws )
{
trimLeft(s,ws) ; trimRight(s,ws) ;
}
bool G::Str::isNumeric( const std::string & s , bool allow_minus_sign )
{
const char * p = s.c_str() ;
if( allow_minus_sign && *p == '-' )
p++ ;
for( ; *p ; p++ )
{
if( *p < '0' || *p > '9' )
return false ;
}
return true ;
}
bool G::Str::isPrintableAscii( const std::string & s )
{
for( const char * p = s.c_str() ; *p ; p++ )
{
if( *p < 0x20 || *p >= 0x7f )
return false ;
}
return true ;
}
bool G::Str::isUShort( const std::string & s )
{
try
{
(void) toUShort(s) ;
}
catch( Overflow & )
{
return false ;
}
catch( InvalidFormat & )
{
return false ;
}
return true ;
}
bool G::Str::isUInt( const std::string & s )
{
try
{
(void) toUInt(s) ;
}
catch( Overflow & )
{
return false ;
}
catch( InvalidFormat & )
{
return false ;
}
return true ;
}
bool G::Str::isULong( const std::string & s )
{
try
{
(void) toULong(s) ;
}
catch( Overflow & )
{
return false ;
}
catch( InvalidFormat & )
{
return false ;
}
return true ;
}
unsigned int G::Str::toUInt( const std::string &s , bool limited )
{
unsigned long ulong_val = toULong( s ) ;
unsigned int uint_val = static_cast<unsigned int>( ulong_val ) ;
if( uint_val != ulong_val )
{
if( limited )
uint_val = UINT_MAX ;
else
throw Overflow( s ) ;
}
return uint_val ;
}
unsigned long G::Str::toULong( const std::string &s , bool limited )
{
char * end = NULL ;
unsigned long result = ::strtoul( s.c_str(), &end, 10 ) ;
if( end == 0 || end[0] != '\0' )
throw InvalidFormat( s ) ;
if( result == ULONG_MAX )
{
if( limited )
result = ULONG_MAX ;
else
throw Overflow( s ) ;
}
return result ;
}
unsigned short G::Str::toUShort( const std::string &s , bool limited )
{
unsigned long ulong_val = toULong( s ) ;
unsigned short ushort_val = static_cast<unsigned short>( ulong_val ) ;
if( ushort_val != ulong_val )
{
if( limited )
ushort_val = USHRT_MAX ;
else
throw Overflow( s ) ;
}
return ushort_val ;
}
std::string G::Str::fromUInt( unsigned int n )
{
std::stringstream ss ;
ss << n ;
return ss.str() ;
}
void G::Str::toLower( std::string &s )
{
for( std::string::iterator p = s.begin() ; p != s.end() ; ++p )
{
*p = ::tolower( *p ) ;
}
}
void G::Str::toUpper( std::string &s )
{
for( std::string::iterator p = s.begin() ; p != s.end() ; ++p )
{
*p = ::toupper( *p ) ;
}
}
std::string G::Str::toPrintableAscii( char c , char escape )
{
if( c == escape )
{
return std::string( 2U , c ) ;
}
else if( c >= 0x20 && c < 0x7f )
{
return std::string( 1U , c ) ;
}
std::string result( 1U , escape ) ;
if( c == '\n' )
{
result.append( 1U , 'n' ) ;
}
else if( c == '\t' )
{
result.append( 1U , 't' ) ;
}
else if( c == '\0' )
{
result.append( 1U , '0' ) ;
}
else
{
unsigned int n = c ;
result.append( 1U , char('0'+((n/64U)%8U)) ) ;
result.append( 1U , char('0'+((n/8U)%8U)) ) ;
result.append( 1U , char('0'+(n%8U)) ) ;
}
return result ;
}
std::string G::Str::toPrintableAscii( const std::string & in , char escape )
{
std::string result ;
for( std::string::const_iterator p = in.begin() ; p != in.end() ; ++p )
result.append( toPrintableAscii(*p,escape) ) ;
return result ;
}
std::string G::Str::readLineFrom( std::istream & stream , char ignore )
{
std::string line ;
char c ;
while( stream.get(c) ) // ie. while(stream.good())
{
if( c == '\n' )
break ;
if( ignore != '\0' && c == ignore )
;
else
line.append(1U,c) ;
}
return line ;
}
std::string G::Str::readLineFrom( std::istream & stream , const std::string & eol )
{
std::string result ;
readLineFrom( stream , eol , result ) ;
return result ;
}
void G::Str::readLineFrom( std::istream & stream , const std::string & eol , std::string & line )
{
line.erase() ;
const size_t eol_length = eol.length() ;
char c ;
for( size_t line_length = 1U ; stream.get(c) ; ++line_length )
{
line.append(1U,c) ;
if( line_length >= eol_length )
{
const size_t offset = line_length - eol_length ;
if( line.find(eol,offset) == offset )
{
line.erase(offset) ;
break ;
}
}
}
}
std::string G::Str::wrap( std::string text , const std::string & prefix_1 ,
const std::string & prefix_2 , size_t width )
{
std::string ws( " \t\n" ) ;
std::stringstream ss ;
for( bool first_line = true ; text.length() ; first_line = false )
{
const size_t prefix_length =
first_line ? prefix_1.length() : prefix_2.length() ;
size_t w = (width > prefix_length) ? (width-prefix_length) : width ;
const size_t pos_nl = text.find_first_of("\n") ;
if( pos_nl != std::string::npos && pos_nl != 0U && pos_nl < w )
{
w = pos_nl ;
}
std::string line = text ;
if( text.length() > w )
{
line = text.substr( 0U , w ) ;
if( text.find_first_of(ws,w) != w )
{
const size_t white_space = line.find_last_of( ws ) ;
const size_t black_space = line.find_first_not_of( ws ) ;
if( white_space != std::string::npos &&
black_space != std::string::npos &&
(white_space+1U) != black_space )
{
line = line.substr( 0U , white_space ) ;
}
}
}
if( line.length() != 0U )
{
ss << ( first_line ? prefix_1 : prefix_2 ) << line << std::endl ;
}
text = text.length() == line.length() ?
std::string() : text.substr(line.length()) ;
const size_t black_space = text.find_first_not_of( ws ) ;
if( black_space != 0U && black_space != std::string::npos )
{
size_t newlines = 0U ;
for( size_t pos = 0U ; pos < black_space ; ++pos )
{
if( text.at(pos) == '\n' )
{
newlines++ ;
if( newlines > 1U )
ss << prefix_2 << std::endl ;
}
}
text = text.substr( black_space ) ;
}
}
return ss.str() ;
}
void G::Str::listPushBack( void * out , const std::string & s )
{
reinterpret_cast<Strings*>(out)->push_back( s ) ;
}
void G::Str::arrayPushBack( void * out , const std::string & s )
{
reinterpret_cast<StringArray*>(out)->push_back( s ) ;
}
void G::Str::splitIntoTokens( const std::string &in , Strings &out ,
const std::string & ws )
{
splitIntoTokens( in , reinterpret_cast<void*>(&out) ,
&listPushBack , ws ) ;
}
void G::Str::splitIntoTokens( const std::string &in , StringArray &out ,
const std::string & ws )
{
splitIntoTokens( in , reinterpret_cast<void*>(&out) ,
&arrayPushBack , ws ) ;
}
void G::Str::splitIntoTokens( const std::string & in ,
void * out , void (*fn)(void*,const std::string&) ,
const std::string & ws )
{
for( size_t p = 0U ; p != std::string::npos ; )
{
p = in.find_first_not_of( ws , p ) ;
if( p != std::string::npos )
{
size_t end = in.find_first_of( ws , p ) ;
size_t len = end == std::string::npos ? end : (end-p) ;
(*fn)( out , in.substr( p , len ) ) ;
p = end ;
}
}
}
void G::Str::splitIntoFields( const std::string & in , Strings &out ,
const std::string & ws , char escape , bool discard_bogus )
{
splitIntoFields( in , reinterpret_cast<void*>(&out) ,
&listPushBack , ws , escape , discard_bogus ) ;
}
void G::Str::splitIntoFields( const std::string & in , StringArray &out ,
const std::string & ws , char escape , bool discard_bogus )
{
splitIntoFields( in , reinterpret_cast<void*>(&out) ,
&arrayPushBack , ws , escape , discard_bogus ) ;
}
void G::Str::splitIntoFields( const std::string & in_in , void * out ,
void (*fn)(void*,const std::string&) ,
const std::string & ws , char escape , bool discard_bogus )
{
std::string all( ws ) ;
if( escape != '\0' )
all.append( 1U , escape ) ;
if( in_in.length() )
{
std::string in = in_in ;
size_t start = 0U ;
size_t last_pos = in.length() - 1U ;
size_t pos = 0U ;
for(;;)
{
if( pos >= in.length() ) break ;
pos = in.find_first_of( all , pos ) ;
if( pos == std::string::npos ) break ;
if( in.at(pos) == escape )
{
const bool valid =
pos != last_pos &&
in.find(all,pos+1U) == (pos+1U) ;
if( valid || discard_bogus )
in.erase( pos , 1U ) ;
else
pos++ ;
pos++ ;
}
else
{
(*fn)( out , in.substr(start,pos-start) ) ;
pos++ ;
start = pos ;
}
}
(*fn)( out , in.substr(start,pos-start) ) ;
}
}