1 // -*- mode: cpp; mode: fold -*-
3 // $Id: strutl.cc,v 1.45 2002/04/26 05:52:27 jgg Exp $
4 /* ######################################################################
6 String Util - Some useful string functions.
8 These have been collected from here and there to do all sorts of useful
9 things to strings. They are useful in file parsers, URI handlers and
10 especially in APT methods.
12 This source is placed in the Public Domain, do with it what you will
13 It was originally written by Jason Gunthorpe <jgg@gpu.srv.ualberta.ca>
15 ##################################################################### */
19 #pragma implementation "apt-pkg/strutl.h"
22 #include <apt-pkg/strutl.h>
23 #include <apt-pkg/fileutl.h>
24 #include <apt-pkg/error.h>
39 // strstrip - Remove white space from the front and back of a string /*{{{*/
40 // ---------------------------------------------------------------------
41 /* This is handy to use when parsing a file. It also removes \n's left
42 over from fgets and company */
43 char *_strstrip(char *String
)
45 for (;*String
!= 0 && (*String
== ' ' || *String
== '\t'); String
++);
50 char *End
= String
+ strlen(String
) - 1;
51 for (;End
!= String
- 1 && (*End
== ' ' || *End
== '\t' || *End
== '\n' ||
52 *End
== '\r'); End
--);
58 // strtabexpand - Converts tabs into 8 spaces /*{{{*/
59 // ---------------------------------------------------------------------
61 char *_strtabexpand(char *String
,size_t Len
)
63 for (char *I
= String
; I
!= I
+ Len
&& *I
!= 0; I
++)
67 if (I
+ 8 > String
+ Len
)
73 /* Assume the start of the string is 0 and find the next 8 char
79 Len
= 8 - ((String
- I
) % 8);
87 memmove(I
+ Len
,I
+ 1,strlen(I
) + 1);
88 for (char *J
= I
; J
+ Len
!= I
; *I
= ' ', I
++);
93 // ParseQuoteWord - Parse a single word out of a string /*{{{*/
94 // ---------------------------------------------------------------------
95 /* This grabs a single word, converts any % escaped characters to their
96 proper values and advances the pointer. Double quotes are understood
97 and striped out as well. This is for URI/URL parsing. It also can
98 understand [] brackets.*/
99 bool ParseQuoteWord(const char *&String
,string
&Res
)
101 // Skip leading whitespace
102 const char *C
= String
;
103 for (;*C
!= 0 && *C
== ' '; C
++);
107 // Jump to the next word
108 for (;*C
!= 0 && isspace(*C
) == 0; C
++)
112 for (C
++; *C
!= 0 && *C
!= '"'; C
++);
118 for (C
++; *C
!= 0 && *C
!= ']'; C
++);
124 // Now de-quote characters
127 const char *Start
= String
;
129 for (I
= Buffer
; I
< Buffer
+ sizeof(Buffer
) && Start
!= C
; I
++)
131 if (*Start
== '%' && Start
+ 2 < C
)
136 *I
= (char)strtol(Tmp
,0,16);
149 // Skip ending white space
150 for (;*C
!= 0 && isspace(*C
) != 0; C
++);
155 // ParseCWord - Parses a string like a C "" expression /*{{{*/
156 // ---------------------------------------------------------------------
157 /* This expects a series of space separated strings enclosed in ""'s.
158 It concatenates the ""'s into a single string. */
159 bool ParseCWord(const char *&String
,string
&Res
)
161 // Skip leading whitespace
162 const char *C
= String
;
163 for (;*C
!= 0 && *C
== ' '; C
++);
169 if (strlen(String
) >= sizeof(Buffer
))
176 for (C
++; *C
!= 0 && *C
!= '"'; C
++)
185 if (C
!= String
&& isspace(*C
) != 0 && isspace(C
[-1]) != 0)
187 if (isspace(*C
) == 0)
197 // QuoteString - Convert a string into quoted from /*{{{*/
198 // ---------------------------------------------------------------------
200 string
QuoteString(string Str
,const char *Bad
)
203 for (string::iterator I
= Str
.begin(); I
!= Str
.end(); I
++)
205 if (strchr(Bad
,*I
) != 0 || isprint(*I
) == 0 ||
206 *I
<= 0x20 || *I
>= 0x7F)
209 sprintf(Buf
,"%%%02x",(int)*I
);
218 // DeQuoteString - Convert a string from quoted from /*{{{*/
219 // ---------------------------------------------------------------------
220 /* This undoes QuoteString */
221 string
DeQuoteString(string Str
)
224 for (string::const_iterator I
= Str
.begin(); I
!= Str
.end(); I
++)
226 if (*I
== '%' && I
+ 2 < Str
.end())
232 Res
+= (char)strtol(Tmp
,0,16);
243 // SizeToStr - Convert a long into a human readable size /*{{{*/
244 // ---------------------------------------------------------------------
245 /* A max of 4 digits are shown before conversion to the next highest unit.
246 The max length of the string will be 5 chars unless the size is > 10
248 string
SizeToStr(double Size
)
257 /* bytes, KiloBytes, MegaBytes, GigaBytes, TeraBytes, PetaBytes,
258 ExaBytes, ZettaBytes, YottaBytes */
259 char Ext
[] = {'\0','k','M','G','T','P','E','Z','Y'};
263 if (ASize
< 100 && I
!= 0)
265 sprintf(S
,"%.1f%c",ASize
,Ext
[I
]);
271 sprintf(S
,"%.0f%c",ASize
,Ext
[I
]);
281 // TimeToStr - Convert the time into a string /*{{{*/
282 // ---------------------------------------------------------------------
283 /* Converts a number of seconds to a hms format */
284 string
TimeToStr(unsigned long Sec
)
292 sprintf(S
,"%lid %lih%lim%lis",Sec
/60/60/24,(Sec
/60/60) % 24,(Sec
/60) % 60,Sec
% 60);
298 sprintf(S
,"%lih%lim%lis",Sec
/60/60,(Sec
/60) % 60,Sec
% 60);
304 sprintf(S
,"%lim%lis",Sec
/60,Sec
% 60);
308 sprintf(S
,"%lis",Sec
);
315 // SubstVar - Substitute a string for another string /*{{{*/
316 // ---------------------------------------------------------------------
317 /* This replaces all occurances of Subst with Contents in Str. */
318 string
SubstVar(string Str
,string Subst
,string Contents
)
320 string::size_type Pos
= 0;
321 string::size_type OldPos
= 0;
324 while (OldPos
< Str
.length() &&
325 (Pos
= Str
.find(Subst
,OldPos
)) != string::npos
)
327 Temp
+= string(Str
,OldPos
,Pos
) + Contents
;
328 OldPos
= Pos
+ Subst
.length();
334 return Temp
+ string(Str
,OldPos
);
337 string
SubstVar(string Str
,const struct SubstVar
*Vars
)
339 for (; Vars
->Subst
!= 0; Vars
++)
340 Str
= SubstVar(Str
,Vars
->Subst
,*Vars
->Contents
);
344 // URItoFileName - Convert the uri into a unique file name /*{{{*/
345 // ---------------------------------------------------------------------
346 /* This converts a URI into a safe filename. It quotes all unsafe characters
347 and converts / to _ and removes the scheme identifier. The resulting
348 file name should be unique and never occur again for a different file */
349 string
URItoFileName(string URI
)
351 // Nuke 'sensitive' items
354 U
.Password
= string();
357 // "\x00-\x20{}|\\\\^\\[\\]<>\"\x7F-\xFF";
358 URI
= QuoteString(U
,"\\|{}[]<>\"^~_=!@#$%^&*");
359 string::iterator J
= URI
.begin();
360 for (; J
!= URI
.end(); J
++)
366 // Base64Encode - Base64 Encoding routine for short strings /*{{{*/
367 // ---------------------------------------------------------------------
368 /* This routine performs a base64 transformation on a string. It was ripped
369 from wget and then patched and bug fixed.
371 This spec can be found in rfc2045 */
372 string
Base64Encode(string S
)
375 static char tbl
[64] = {'A','B','C','D','E','F','G','H',
376 'I','J','K','L','M','N','O','P',
377 'Q','R','S','T','U','V','W','X',
378 'Y','Z','a','b','c','d','e','f',
379 'g','h','i','j','k','l','m','n',
380 'o','p','q','r','s','t','u','v',
381 'w','x','y','z','0','1','2','3',
382 '4','5','6','7','8','9','+','/'};
384 // Pre-allocate some space
386 Final
.reserve((4*S
.length() + 2)/3 + 2);
388 /* Transform the 3x8 bits to 4x6 bits, as required by
390 for (string::const_iterator I
= S
.begin(); I
< S
.end(); I
+= 3)
392 char Bits
[3] = {0,0,0};
399 Final
+= tbl
[Bits
[0] >> 2];
400 Final
+= tbl
[((Bits
[0] & 3) << 4) + (Bits
[1] >> 4)];
402 if (I
+ 1 >= S
.end())
405 Final
+= tbl
[((Bits
[1] & 0xf) << 2) + (Bits
[2] >> 6)];
407 if (I
+ 2 >= S
.end())
410 Final
+= tbl
[Bits
[2] & 0x3f];
413 /* Apply the padding elements, this tells how many bytes the remote
414 end should discard */
415 if (S
.length() % 3 == 2)
417 if (S
.length() % 3 == 1)
423 // stringcmp - Arbitary string compare /*{{{*/
424 // ---------------------------------------------------------------------
425 /* This safely compares two non-null terminated strings of arbitary
427 int stringcmp(const char *A
,const char *AEnd
,const char *B
,const char *BEnd
)
429 for (; A
!= AEnd
&& B
!= BEnd
; A
++, B
++)
433 if (A
== AEnd
&& B
== BEnd
)
445 int stringcmp(string::const_iterator A
,string::const_iterator AEnd
,
446 const char *B
,const char *BEnd
)
448 for (; A
!= AEnd
&& B
!= BEnd
; A
++, B
++)
452 if (A
== AEnd
&& B
== BEnd
)
462 int stringcmp(string::const_iterator A
,string::const_iterator AEnd
,
463 string::const_iterator B
,string::const_iterator BEnd
)
465 for (; A
!= AEnd
&& B
!= BEnd
; A
++, B
++)
469 if (A
== AEnd
&& B
== BEnd
)
481 // stringcasecmp - Arbitary case insensitive string compare /*{{{*/
482 // ---------------------------------------------------------------------
484 int stringcasecmp(const char *A
,const char *AEnd
,const char *B
,const char *BEnd
)
486 for (; A
!= AEnd
&& B
!= BEnd
; A
++, B
++)
487 if (toupper(*A
) != toupper(*B
))
490 if (A
== AEnd
&& B
== BEnd
)
496 if (toupper(*A
) < toupper(*B
))
501 int stringcasecmp(string::const_iterator A
,string::const_iterator AEnd
,
502 const char *B
,const char *BEnd
)
504 for (; A
!= AEnd
&& B
!= BEnd
; A
++, B
++)
505 if (toupper(*A
) != toupper(*B
))
508 if (A
== AEnd
&& B
== BEnd
)
514 if (toupper(*A
) < toupper(*B
))
518 int stringcasecmp(string::const_iterator A
,string::const_iterator AEnd
,
519 string::const_iterator B
,string::const_iterator BEnd
)
521 for (; A
!= AEnd
&& B
!= BEnd
; A
++, B
++)
522 if (toupper(*A
) != toupper(*B
))
525 if (A
== AEnd
&& B
== BEnd
)
531 if (toupper(*A
) < toupper(*B
))
537 // LookupTag - Lookup the value of a tag in a taged string /*{{{*/
538 // ---------------------------------------------------------------------
539 /* The format is like those used in package files and the method
540 communication system */
541 string
LookupTag(string Message
,const char *Tag
,const char *Default
)
543 // Look for a matching tag.
544 int Length
= strlen(Tag
);
545 for (string::iterator I
= Message
.begin(); I
+ Length
< Message
.end(); I
++)
548 if (I
[Length
] == ':' && stringcasecmp(I
,I
+Length
,Tag
) == 0)
550 // Find the end of line and strip the leading/trailing spaces
553 for (; isspace(*I
) != 0 && I
< Message
.end(); I
++);
554 for (J
= I
; *J
!= '\n' && J
< Message
.end(); J
++);
555 for (; J
> I
&& isspace(J
[-1]) != 0; J
--);
560 for (; *I
!= '\n' && I
< Message
.end(); I
++);
563 // Failed to find a match
569 // StringToBool - Converts a string into a boolean /*{{{*/
570 // ---------------------------------------------------------------------
571 /* This inspects the string to see if it is true or if it is false and
572 then returns the result. Several varients on true/false are checked. */
573 int StringToBool(string Text
,int Default
)
576 int Res
= strtol(Text
.c_str(),&End
,0);
577 if (End
!= Text
.c_str() && Res
>= 0 && Res
<= 1)
580 // Check for positives
581 if (strcasecmp(Text
.c_str(),"no") == 0 ||
582 strcasecmp(Text
.c_str(),"false") == 0 ||
583 strcasecmp(Text
.c_str(),"without") == 0 ||
584 strcasecmp(Text
.c_str(),"off") == 0 ||
585 strcasecmp(Text
.c_str(),"disable") == 0)
588 // Check for negatives
589 if (strcasecmp(Text
.c_str(),"yes") == 0 ||
590 strcasecmp(Text
.c_str(),"true") == 0 ||
591 strcasecmp(Text
.c_str(),"with") == 0 ||
592 strcasecmp(Text
.c_str(),"on") == 0 ||
593 strcasecmp(Text
.c_str(),"enable") == 0)
599 // TimeRFC1123 - Convert a time_t into RFC1123 format /*{{{*/
600 // ---------------------------------------------------------------------
601 /* This converts a time_t into a string time representation that is
602 year 2000 complient and timezone neutral */
603 string
TimeRFC1123(time_t Date
)
605 struct tm Conv
= *gmtime(&Date
);
608 const char *Day
[] = {"Sun","Mon","Tue","Wed","Thu","Fri","Sat"};
609 const char *Month
[] = {"Jan","Feb","Mar","Apr","May","Jun","Jul",
610 "Aug","Sep","Oct","Nov","Dec"};
612 sprintf(Buf
,"%s, %02i %s %i %02i:%02i:%02i GMT",Day
[Conv
.tm_wday
],
613 Conv
.tm_mday
,Month
[Conv
.tm_mon
],Conv
.tm_year
+1900,Conv
.tm_hour
,
614 Conv
.tm_min
,Conv
.tm_sec
);
618 // ReadMessages - Read messages from the FD /*{{{*/
619 // ---------------------------------------------------------------------
620 /* This pulls full messages from the input FD into the message buffer.
621 It assumes that messages will not pause during transit so no
622 fancy buffering is used. */
623 bool ReadMessages(int Fd
, vector
<string
> &List
)
630 int Res
= read(Fd
,End
,sizeof(Buffer
) - (End
-Buffer
));
631 if (Res
< 0 && errno
== EINTR
)
634 // Process is dead, this is kind of bad..
639 if (Res
< 0 && errno
== EAGAIN
)
646 // Look for the end of the message
647 for (char *I
= Buffer
; I
+ 1 < End
; I
++)
649 if (I
[0] != '\n' || I
[1] != '\n')
652 // Pull the message out
653 string
Message(Buffer
,0,I
-Buffer
);
656 for (; I
< End
&& *I
== '\n'; I
++);
658 memmove(Buffer
,I
,End
-Buffer
);
661 List
.push_back(Message
);
666 if (WaitFd(Fd
) == false)
671 // MonthConv - Converts a month string into a number /*{{{*/
672 // ---------------------------------------------------------------------
673 /* This was lifted from the boa webserver which lifted it from 'wn-v1.07'
674 Made it a bit more robust with a few touppers though. */
675 static int MonthConv(char *Month
)
677 switch (toupper(*Month
))
680 return toupper(Month
[1]) == 'P'?3:7;
686 if (toupper(Month
[1]) == 'A')
688 return toupper(Month
[2]) == 'N'?5:6;
690 return toupper(Month
[2]) == 'R'?2:4;
698 // Pretend it is January..
704 // timegm - Internal timegm function if gnu is not available /*{{{*/
705 // ---------------------------------------------------------------------
706 /* Ripped this evil little function from wget - I prefer the use of
707 GNU timegm if possible as this technique will have interesting problems
708 with leap seconds, timezones and other.
710 Converts struct tm to time_t, assuming the data in tm is UTC rather
711 than local timezone (mktime assumes the latter).
713 Contributed by Roger Beeman <beeman@cisco.com>, with the help of
714 Mark Baushke <mdb@cisco.com> and the rest of the Gurus at CISCO. */
715 #ifndef __USE_MISC // glib sets this
716 static time_t timegm(struct tm
*t
)
723 tb
= mktime (gmtime (&tl
));
724 return (tl
<= tb
? (tl
+ (tl
- tb
)) : (tl
- (tb
- tl
)));
728 // StrToTime - Converts a string into a time_t /*{{{*/
729 // ---------------------------------------------------------------------
730 /* This handles all 3 populare time formats including RFC 1123, RFC 1036
731 and the C library asctime format. It requires the GNU library function
732 'timegm' to convert a struct tm in UTC to a time_t. For some bizzar
733 reason the C library does not provide any such function :< This also
734 handles the weird, but unambiguous FTP time format*/
735 bool StrToTime(string Val
,time_t &Result
)
739 const char *I
= Val
.c_str();
741 // Skip the day of the week
742 for (;*I
!= 0 && *I
!= ' '; I
++);
744 // Handle RFC 1123 time
746 if (sscanf(I
," %d %3s %d %d:%d:%d GMT",&Tm
.tm_mday
,Month
,&Tm
.tm_year
,
747 &Tm
.tm_hour
,&Tm
.tm_min
,&Tm
.tm_sec
) != 6)
749 // Handle RFC 1036 time
750 if (sscanf(I
," %d-%3s-%d %d:%d:%d GMT",&Tm
.tm_mday
,Month
,
751 &Tm
.tm_year
,&Tm
.tm_hour
,&Tm
.tm_min
,&Tm
.tm_sec
) == 6)
756 if (sscanf(I
," %3s %d %d:%d:%d %d",Month
,&Tm
.tm_mday
,
757 &Tm
.tm_hour
,&Tm
.tm_min
,&Tm
.tm_sec
,&Tm
.tm_year
) != 6)
760 if (sscanf(Val
.c_str(),"%4d%2d%2d%2d%2d%2d",&Tm
.tm_year
,&Tm
.tm_mon
,
761 &Tm
.tm_mday
,&Tm
.tm_hour
,&Tm
.tm_min
,&Tm
.tm_sec
) != 6)
770 Tm
.tm_mon
= MonthConv(Month
);
773 // Convert to local time and then to GMT
774 Result
= timegm(&Tm
);
778 // StrToNum - Convert a fixed length string to a number /*{{{*/
779 // ---------------------------------------------------------------------
780 /* This is used in decoding the crazy fixed length string headers in
782 bool StrToNum(const char *Str
,unsigned long &Res
,unsigned Len
,unsigned Base
)
785 if (Len
>= sizeof(S
))
790 // All spaces is a zero
793 for (I
= 0; S
[I
] == ' '; I
++);
798 Res
= strtoul(S
,&End
,Base
);
805 // HexDigit - Convert a hex character into an integer /*{{{*/
806 // ---------------------------------------------------------------------
807 /* Helper for Hex2Num */
808 static int HexDigit(int c
)
810 if (c
>= '0' && c
<= '9')
812 if (c
>= 'a' && c
<= 'f')
814 if (c
>= 'A' && c
<= 'F')
819 // Hex2Num - Convert a long hex number into a buffer /*{{{*/
820 // ---------------------------------------------------------------------
821 /* The length of the buffer must be exactly 1/2 the length of the string. */
822 bool Hex2Num(string Str
,unsigned char *Num
,unsigned int Length
)
824 if (Str
.length() != Length
*2)
827 // Convert each digit. We store it in the same order as the string
829 for (string::const_iterator I
= Str
.begin(); I
!= Str
.end();J
++, I
+= 2)
831 if (isxdigit(*I
) == 0 || isxdigit(I
[1]) == 0)
834 Num
[J
] = HexDigit(I
[0]) << 4;
835 Num
[J
] += HexDigit(I
[1]);
841 // TokSplitString - Split a string up by a given token /*{{{*/
842 // ---------------------------------------------------------------------
843 /* This is intended to be a faster splitter, it does not use dynamic
844 memories. Input is changed to insert nulls at each token location. */
845 bool TokSplitString(char Tok
,char *Input
,char **List
,
846 unsigned long ListMax
)
848 // Strip any leading spaces
850 char *Stop
= Start
+ strlen(Start
);
851 for (; *Start
!= 0 && isspace(*Start
) != 0; Start
++);
853 unsigned long Count
= 0;
857 // Skip to the next Token
858 for (; Pos
!= Stop
&& *Pos
!= Tok
; Pos
++);
860 // Back remove spaces
862 for (; End
> Start
&& (End
[-1] == Tok
|| isspace(End
[-1]) != 0); End
--);
865 List
[Count
++] = Start
;
866 if (Count
>= ListMax
)
873 for (; Pos
!= Stop
&& (*Pos
== Tok
|| isspace(*Pos
) != 0 || *Pos
== 0); Pos
++);
881 // RegexChoice - Simple regex list/list matcher /*{{{*/
882 // ---------------------------------------------------------------------
884 unsigned long RegexChoice(RxChoiceList
*Rxs
,const char **ListBegin
,
885 const char **ListEnd
)
887 for (RxChoiceList
*R
= Rxs
; R
->Str
!= 0; R
++)
890 unsigned long Hits
= 0;
891 for (; ListBegin
!= ListEnd
; ListBegin
++)
893 // Check if the name is a regex
896 for (I
= *ListBegin
; *I
!= 0; I
++)
897 if (*I
== '.' || *I
== '?' || *I
== '*' || *I
== '|')
902 // Compile the regex pattern
905 if (regcomp(&Pattern
,*ListBegin
,REG_EXTENDED
| REG_ICASE
|
911 for (RxChoiceList
*R
= Rxs
; R
->Str
!= 0; R
++)
916 if (strcasecmp(R
->Str
,*ListBegin
) != 0)
920 if (regexec(&Pattern
,R
->Str
,0,0,0) != 0)
935 _error
->Warning(_("Selection %s not found"),*ListBegin
);
941 // ioprintf - C format string outputter to C++ iostreams /*{{{*/
942 // ---------------------------------------------------------------------
943 /* This is used to make the internationalization strinc easier to translate
944 and to allow reordering of parameters */
945 void ioprintf(ostream
&out
,const char *format
,...)
948 va_start(args
,format
);
950 // sprintf the description
952 vsnprintf(S
,sizeof(S
),format
,args
);
957 // CheckDomainList - See if Host is in a , seperate list /*{{{*/
958 // ---------------------------------------------------------------------
959 /* The domain list is a comma seperate list of domains that are suffix
960 matched against the argument */
961 bool CheckDomainList(string Host
,string List
)
963 string::const_iterator Start
= List
.begin();
964 for (string::const_iterator Cur
= List
.begin(); Cur
<= List
.end(); Cur
++)
966 if (Cur
< List
.end() && *Cur
!= ',')
969 // Match the end of the string..
970 if ((Host
.size() >= (unsigned)(Cur
- Start
)) &&
972 stringcasecmp(Host
.end() - (Cur
- Start
),Host
.end(),Start
,Cur
) == 0)
981 // URI::CopyFrom - Copy from an object /*{{{*/
982 // ---------------------------------------------------------------------
983 /* This parses the URI into all of its components */
984 void URI::CopyFrom(string U
)
986 string::const_iterator I
= U
.begin();
988 // Locate the first colon, this separates the scheme
989 for (; I
< U
.end() && *I
!= ':' ; I
++);
990 string::const_iterator FirstColon
= I
;
992 /* Determine if this is a host type URI with a leading double //
993 and then search for the first single / */
994 string::const_iterator SingleSlash
= I
;
995 if (I
+ 3 < U
.end() && I
[1] == '/' && I
[2] == '/')
998 /* Find the / indicating the end of the hostname, ignoring /'s in the
1000 bool InBracket
= false;
1001 for (; SingleSlash
< U
.end() && (*SingleSlash
!= '/' || InBracket
== true); SingleSlash
++)
1003 if (*SingleSlash
== '[')
1005 if (InBracket
== true && *SingleSlash
== ']')
1009 if (SingleSlash
> U
.end())
1010 SingleSlash
= U
.end();
1012 // We can now write the access and path specifiers
1013 Access
= string(U
,0,FirstColon
- U
.begin());
1014 if (SingleSlash
!= U
.end())
1015 Path
= string(U
,SingleSlash
- U
.begin());
1016 if (Path
.empty() == true)
1019 // Now we attempt to locate a user:pass@host fragment
1020 if (FirstColon
[1] == '/' && FirstColon
[2] == '/')
1024 if (FirstColon
>= U
.end())
1027 if (FirstColon
> SingleSlash
)
1028 FirstColon
= SingleSlash
;
1030 // Find the colon...
1032 if (I
> SingleSlash
)
1034 for (; I
< SingleSlash
&& *I
!= ':'; I
++);
1035 string::const_iterator SecondColon
= I
;
1037 // Search for the @ after the colon
1038 for (; I
< SingleSlash
&& *I
!= '@'; I
++);
1039 string::const_iterator At
= I
;
1041 // Now write the host and user/pass
1042 if (At
== SingleSlash
)
1044 if (FirstColon
< SingleSlash
)
1045 Host
= string(U
,FirstColon
- U
.begin(),SingleSlash
- FirstColon
);
1049 Host
= string(U
,At
- U
.begin() + 1,SingleSlash
- At
- 1);
1050 User
= string(U
,FirstColon
- U
.begin(),SecondColon
- FirstColon
);
1051 if (SecondColon
< At
)
1052 Password
= string(U
,SecondColon
- U
.begin() + 1,At
- SecondColon
- 1);
1055 // Now we parse the RFC 2732 [] hostnames.
1056 unsigned long PortEnd
= 0;
1058 for (unsigned I
= 0; I
!= Host
.length();)
1067 if (InBracket
== true && Host
[I
] == ']')
1078 if (InBracket
== true)
1084 // Now we parse off a port number from the hostname
1086 string::size_type Pos
= Host
.rfind(':');
1087 if (Pos
== string::npos
|| Pos
< PortEnd
)
1090 Port
= atoi(string(Host
,Pos
+1).c_str());
1091 Host
= string(Host
,0,Pos
);
1094 // URI::operator string - Convert the URI to a string /*{{{*/
1095 // ---------------------------------------------------------------------
1097 URI::operator string()
1101 if (Access
.empty() == false)
1104 if (Host
.empty() == false)
1106 if (Access
.empty() == false)
1109 if (User
.empty() == false)
1112 if (Password
.empty() == false)
1113 Res
+= ":" + Password
;
1117 // Add RFC 2732 escaping characters
1118 if (Access
.empty() == false &&
1119 (Host
.find('/') != string::npos
|| Host
.find(':') != string::npos
))
1120 Res
+= '[' + Host
+ ']';
1127 sprintf(S
,":%u",Port
);
1132 if (Path
.empty() == false)
1143 // URI::SiteOnly - Return the schema and site for the URI /*{{{*/
1144 // ---------------------------------------------------------------------
1146 string
URI::SiteOnly(string URI
)
1150 U
.Password
= string();