bff2dc6e3379683e585b235d6c8451c5fb3815a8
1 // -*- mode: cpp; mode: fold -*-
3 // $Id: configuration.cc,v 1.4 1998/09/22 05:30:26 jgg Exp $
4 /* ######################################################################
8 This class provides a configuration file and command line parser
9 for a tree-oriented configuration environment. All runtime configuration
12 ##################################################################### */
14 // Include files /*{{{*/
16 #pragma implementation "apt-pkg/configuration.h"
18 #include <apt-pkg/configuration.h>
19 #include <apt-pkg/error.h>
26 Configuration
*_config
= new Configuration
;
28 // Configuration::Configuration - Constructor /*{{{*/
29 // ---------------------------------------------------------------------
31 Configuration::Configuration()
36 // Configuration::Lookup - Lookup a single item /*{{{*/
37 // ---------------------------------------------------------------------
38 /* This will lookup a single item by name below another item. It is a
39 helper function for the main lookup function */
40 Configuration::Item
*Configuration::Lookup(Item
*Head
,const char *S
,
41 unsigned long Len
,bool Create
)
44 Item
*I
= Head
->Child
;
45 Item
**Last
= &Head
->Child
;
46 for (; I
!= 0; Last
= &I
->Next
, I
= I
->Next
)
47 if ((Res
= stringcasecmp(I
->Tag
.begin(),I
->Tag
.end(),S
,S
+ Len
)) == 0)
56 I
->Tag
= string(S
,Len
);
63 // Configuration::Lookup - Lookup a fully scoped item /*{{{*/
64 // ---------------------------------------------------------------------
65 /* This performs a fully scoped lookup of a given name, possibly creating
67 Configuration::Item
*Configuration::Lookup(const char *Name
,bool Create
)
69 const char *Start
= Name
;
70 const char *End
= Start
+ strlen(Name
);
71 const char *TagEnd
= Name
;
73 for (; End
- TagEnd
> 2; TagEnd
++)
75 if (TagEnd
[0] == ':' && TagEnd
[1] == ':')
77 Itm
= Lookup(Itm
,Start
,TagEnd
- Start
,Create
);
80 TagEnd
= Start
= TagEnd
+ 2;
84 Itm
= Lookup(Itm
,Start
,End
- Start
,Create
);
88 // Configuration::Find - Find a value /*{{{*/
89 // ---------------------------------------------------------------------
91 string
Configuration::Find(const char *Name
,const char *Default
)
93 Item
*Itm
= Lookup(Name
,false);
94 if (Itm
== 0 || Itm
->Value
.empty() == true)
105 // Configuration::FindDir - Find a directory /*{{{*/
106 // ---------------------------------------------------------------------
107 /* Directories are stored as the base dir in the Parent node and the
109 string
Configuration::FindDir(const char *Name
,const char *Default
= 0)
111 Item
*Itm
= Lookup(Name
,false);
112 if (Itm
== 0 || Itm
->Value
.empty() == true)
120 if (Itm
->Value
[0] == '/' || Itm
->Parent
== 0)
122 if (Itm
->Parent
->Value
.end()[-1] == '/')
123 return Itm
->Parent
->Value
+ Itm
->Value
;
125 return Itm
->Parent
->Value
+ '/' + Itm
->Value
;
128 // Configuration::FindI - Find an integer value /*{{{*/
129 // ---------------------------------------------------------------------
131 int Configuration::FindI(const char *Name
,int Default
)
133 Item
*Itm
= Lookup(Name
,false);
134 if (Itm
== 0 || Itm
->Value
.empty() == true)
138 int Res
= strtol(Itm
->Value
.c_str(),&End
,0);
139 if (End
== Itm
->Value
.c_str())
145 // Configuration::FindB - Find a boolean type /*{{{*/
146 // ---------------------------------------------------------------------
148 bool Configuration::FindB(const char *Name
,bool Default
)
150 Item
*Itm
= Lookup(Name
,false);
151 if (Itm
== 0 || Itm
->Value
.empty() == true)
155 int Res
= strtol(Itm
->Value
.c_str(),&End
,0);
156 if (End
== Itm
->Value
.c_str() || Res
< 0 || Res
> 1)
158 // Check for positives
159 if (strcasecmp(Itm
->Value
,"no") == 0 ||
160 strcasecmp(Itm
->Value
,"false") == 0 ||
161 strcasecmp(Itm
->Value
,"without") == 0 ||
162 strcasecmp(Itm
->Value
,"disable") == 0)
165 // Check for negatives
166 if (strcasecmp(Itm
->Value
,"no") == 0 ||
167 strcasecmp(Itm
->Value
,"false") == 0 ||
168 strcasecmp(Itm
->Value
,"without") == 0 ||
169 strcasecmp(Itm
->Value
,"disable") == 0)
178 // Configuration::Set - Set a value /*{{{*/
179 // ---------------------------------------------------------------------
181 void Configuration::Set(const char *Name
,string Value
)
183 Item
*Itm
= Lookup(Name
,true);
189 // Configuration::Set - Set an integer value /*{{{*/
190 // ---------------------------------------------------------------------
192 void Configuration::Set(const char *Name
,int Value
)
194 Item
*Itm
= Lookup(Name
,true);
198 snprintf(S
,sizeof(S
),"%i",Value
);
202 // Configuration::Exists - Returns true if the Name exists /*{{{*/
203 // ---------------------------------------------------------------------
205 bool Configuration::Exists(const char *Name
)
207 Item
*Itm
= Lookup(Name
,false);
214 // ReadConfigFile - Read a configuration file /*{{{*/
215 // ---------------------------------------------------------------------
216 /* The configuration format is very much like the named.conf format
217 used in bind8, in fact this routine can parse most named.conf files. */
218 bool ReadConfigFile(Configuration
&Conf
,string FName
)
220 // Open the stream for reading
221 ifstream
F(FName
.c_str(),ios::in
| ios::nocreate
);
223 return _error
->Errno("ifstream::ifstream","Opening configuration file %s",FName
.c_str());
232 bool InComment
= false;
233 while (F
.eof() == false)
235 F
.getline(Buffer
,sizeof(Buffer
));
237 _strtabexpand(Buffer
,sizeof(Buffer
));
240 // Multi line comment
241 if (InComment
== true)
243 for (const char *I
= Buffer
; *I
!= 0; I
++)
245 if (*I
== '*' && I
[1] == '/')
247 memmove(Buffer
,I
+2,strlen(I
+2) + 1);
252 if (InComment
== true)
256 // Discard single line comments
257 for (char *I
= Buffer
; *I
!= 0; I
++)
259 if (*I
== '/' && I
[1] == '/')
266 // Look for multi line comments
267 for (char *I
= Buffer
; *I
!= 0; I
++)
269 if (*I
== '/' && I
[1] == '*')
272 for (char *J
= Buffer
; *J
!= 0; J
++)
274 if (*J
== '*' && J
[1] == '/')
276 memmove(I
,J
+2,strlen(J
+2) + 1);
282 if (InComment
== true)
294 // We now have a valid line fragment
295 for (char *I
= Buffer
; *I
!= 0;)
297 if (*I
== '{' || *I
== ';' || *I
== '}')
299 // Put the last fragement into the buffer
300 char *Start
= Buffer
;
302 for (; Start
!= I
&& isspace(*Start
) != 0; Start
++);
303 for (; Stop
!= Start
&& isspace(Stop
[-1]) != 0; Stop
--);
304 if (LineBuffer
.empty() == false && Stop
- Start
!= 0)
306 LineBuffer
+= string(Start
,Stop
- Start
);
308 // Remove the fragment
310 memmove(Buffer
,I
+ 1,strlen(I
+ 1) + 1);
316 string::size_type Pos
= ParentTag
.rfind("::");
317 if (Pos
== string::npos
)
318 ParentTag
= string();
320 ParentTag
= string(ParentTag
,0,Pos
);
324 if (TermChar
== '{' && LineBuffer
.empty() == true)
325 return _error
->Error("Syntax error %s:%u: Block starts with no name.",FName
.c_str(),CurLine
);
327 if (LineBuffer
.empty() == true)
331 string::size_type Pos
= LineBuffer
.find(' ');
332 if (Pos
== string::npos
)
335 Pos
= LineBuffer
.length();
337 return _error
->Error("Syntax error %s:%u: Tag with no value",FName
.c_str(),CurLine
);
340 string Tag
= string(LineBuffer
,0,Pos
);
345 if (ParentTag
.empty() == true)
348 ParentTag
+= string("::") + Tag
;
352 // We dont have a value to set
353 if (Pos
== LineBuffer
.length())
355 LineBuffer
= string();
359 // Parse off the word
361 if (ParseCWord(LineBuffer
.c_str()+Pos
,Word
) == false)
362 return _error
->Error("Syntax error %s:%u: Malformed value",FName
.c_str(),CurLine
);
364 // Generate the item name
366 if (ParentTag
.empty() == true)
370 if (Tag
.empty() == true)
373 Item
= ParentTag
+ "::" + Tag
;
376 // Set the item in the configuration class
380 LineBuffer
= string();
386 // Store the fragment
387 const char *Stripd
= _strstrip(Buffer
);
388 if (*Stripd
!= 0 && LineBuffer
.empty() == false)
390 LineBuffer
+= Stripd
;