summaryrefslogtreecommitdiff
path: root/library
Unidiff
Diffstat (limited to 'library') (more/less context) (ignore whitespace changes)
-rw-r--r--library/config.cpp4
1 files changed, 3 insertions, 1 deletions
diff --git a/library/config.cpp b/library/config.cpp
index 8b60f60..0bfb476 100644
--- a/library/config.cpp
+++ b/library/config.cpp
@@ -143,436 +143,438 @@ bool Config::hasKey( const QString &key ) const
143*/ 143*/
144void Config::setGroup( const QString &gname ) 144void Config::setGroup( const QString &gname )
145{ 145{
146 QMap< QString, ConfigGroup>::Iterator it = groups.find( gname ); 146 QMap< QString, ConfigGroup>::Iterator it = groups.find( gname );
147 if ( it == groups.end() ) { 147 if ( it == groups.end() ) {
148 git = groups.insert( gname, ConfigGroup() ); 148 git = groups.insert( gname, ConfigGroup() );
149 changed = TRUE; 149 changed = TRUE;
150 return; 150 return;
151 } 151 }
152 git = it; 152 git = it;
153} 153}
154 154
155/*! 155/*!
156 Writes a (\a key, \a value) entry to the current group. 156 Writes a (\a key, \a value) entry to the current group.
157 157
158 \sa readEntry() 158 \sa readEntry()
159*/ 159*/
160void Config::writeEntry( const QString &key, const char* value ) 160void Config::writeEntry( const QString &key, const char* value )
161{ 161{
162 writeEntry(key,QString(value)); 162 writeEntry(key,QString(value));
163} 163}
164 164
165/*! 165/*!
166 Writes a (\a key, \a value) entry to the current group. 166 Writes a (\a key, \a value) entry to the current group.
167 167
168 \sa readEntry() 168 \sa readEntry()
169*/ 169*/
170void Config::writeEntry( const QString &key, const QString &value ) 170void Config::writeEntry( const QString &key, const QString &value )
171{ 171{
172 if ( git == groups.end() ) { 172 if ( git == groups.end() ) {
173 qWarning( "no group set" ); 173 qWarning( "no group set" );
174 return; 174 return;
175 } 175 }
176 if ( (*git)[key] != value ) { 176 if ( (*git)[key] != value ) {
177 ( *git ).insert( key, value ); 177 ( *git ).insert( key, value );
178 changed = TRUE; 178 changed = TRUE;
179 } 179 }
180} 180}
181 181
182/* 182/*
183 Note that the degree of protection offered by the encryption here is 183 Note that the degree of protection offered by the encryption here is
184 only sufficient to avoid the most casual observation of the configuration 184 only sufficient to avoid the most casual observation of the configuration
185 files. People with access to the files can write down the contents and 185 files. People with access to the files can write down the contents and
186 decrypt it using this source code. 186 decrypt it using this source code.
187 187
188 Conceivably, and at some burden to the user, this encryption could 188 Conceivably, and at some burden to the user, this encryption could
189 be improved. 189 be improved.
190*/ 190*/
191static QString encipher(const QString& plain) 191static QString encipher(const QString& plain)
192{ 192{
193 // mainly, we make it long 193 // mainly, we make it long
194 QString cipher; 194 QString cipher;
195 int mix=28730492; 195 int mix=28730492;
196 for (int i=0; i<(int)plain.length(); i++) { 196 for (int i=0; i<(int)plain.length(); i++) {
197 int u = plain[i].unicode(); 197 int u = plain[i].unicode();
198 int c = u ^ mix; 198 int c = u ^ mix;
199 QString x = QString::number(c,36); 199 QString x = QString::number(c,36);
200 cipher.append(QChar('a'+x.length())); 200 cipher.append(QChar('a'+x.length()));
201 cipher.append(x); 201 cipher.append(x);
202 mix *= u; 202 mix *= u;
203 } 203 }
204 return cipher; 204 return cipher;
205} 205}
206 206
207static QString decipher(const QString& cipher) 207static QString decipher(const QString& cipher)
208{ 208{
209 QString plain; 209 QString plain;
210 int mix=28730492; 210 int mix=28730492;
211 for (int i=0; i<(int)cipher.length();) { 211 for (int i=0; i<(int)cipher.length();) {
212 int l = cipher[i].unicode()-'a'; 212 int l = cipher[i].unicode()-'a';
213 QString x = cipher.mid(i+1,l); i+=l+1; 213 QString x = cipher.mid(i+1,l); i+=l+1;
214 int u = x.toInt(0,36) ^ mix; 214 int u = x.toInt(0,36) ^ mix;
215 plain.append(QChar(u)); 215 plain.append(QChar(u));
216 mix *= u; 216 mix *= u;
217 } 217 }
218 return plain; 218 return plain;
219} 219}
220 220
221/*! 221/*!
222 Writes an encrypted (\a key, \a value) entry to the current group. 222 Writes an encrypted (\a key, \a value) entry to the current group.
223 223
224 Note that the degree of protection offered by the encryption is 224 Note that the degree of protection offered by the encryption is
225 only sufficient to avoid the most casual observation of the configuration 225 only sufficient to avoid the most casual observation of the configuration
226 files. 226 files.
227 227
228 \sa readEntry() 228 \sa readEntry()
229*/ 229*/
230void Config::writeEntryCrypt( const QString &key, const QString &value ) 230void Config::writeEntryCrypt( const QString &key, const QString &value )
231{ 231{
232 if ( git == groups.end() ) { 232 if ( git == groups.end() ) {
233 qWarning( "no group set" ); 233 qWarning( "no group set" );
234 return; 234 return;
235 } 235 }
236 QString evalue = encipher(value); 236 QString evalue = encipher(value);
237 if ( (*git)[key] != evalue ) { 237 if ( (*git)[key] != evalue ) {
238 ( *git ).insert( key, evalue ); 238 ( *git ).insert( key, evalue );
239 changed = TRUE; 239 changed = TRUE;
240 } 240 }
241} 241}
242 242
243/*! 243/*!
244 Writes a (\a key, \a num) entry to the current group. 244 Writes a (\a key, \a num) entry to the current group.
245 245
246 \sa readNumEntry() 246 \sa readNumEntry()
247*/ 247*/
248void Config::writeEntry( const QString &key, int num ) 248void Config::writeEntry( const QString &key, int num )
249{ 249{
250 QString s; 250 QString s;
251 s.setNum( num ); 251 s.setNum( num );
252 writeEntry( key, s ); 252 writeEntry( key, s );
253} 253}
254 254
255#ifdef Q_HAS_BOOL_TYPE 255#ifdef Q_HAS_BOOL_TYPE
256/*! 256/*!
257 Writes a (\a key, \a b) entry to the current group. This is equivalent 257 Writes a (\a key, \a b) entry to the current group. This is equivalent
258 to writing a 0 or 1 as an integer entry. 258 to writing a 0 or 1 as an integer entry.
259 259
260 \sa readBoolEntry() 260 \sa readBoolEntry()
261*/ 261*/
262void Config::writeEntry( const QString &key, bool b ) 262void Config::writeEntry( const QString &key, bool b )
263{ 263{
264 QString s; 264 QString s;
265 s.setNum( ( int )b ); 265 s.setNum( ( int )b );
266 writeEntry( key, s ); 266 writeEntry( key, s );
267} 267}
268#endif 268#endif
269 269
270/*! 270/*!
271 Writes a (\a key, \a lst) entry to the current group. The list 271 Writes a (\a key, \a lst) entry to the current group. The list
272 is separated by \a sep, so the strings must not contain that character. 272 is separated by \a sep, so the strings must not contain that character.
273 273
274 \sa readListEntry() 274 \sa readListEntry()
275*/ 275*/
276void Config::writeEntry( const QString &key, const QStringList &lst, const QChar &sep ) 276void Config::writeEntry( const QString &key, const QStringList &lst, const QChar &sep )
277{ 277{
278 QString s; 278 QString s;
279 QStringList::ConstIterator it = lst.begin(); 279 QStringList::ConstIterator it = lst.begin();
280 for ( ; it != lst.end(); ++it ) 280 for ( ; it != lst.end(); ++it )
281 s += *it + sep; 281 s += *it + sep;
282 writeEntry( key, s ); 282 writeEntry( key, s );
283} 283}
284 284
285/*! 285/*!
286 Removes the \a key entry from the current group. Does nothing if 286 Removes the \a key entry from the current group. Does nothing if
287 there is no such entry. 287 there is no such entry.
288*/ 288*/
289 289
290void Config::removeEntry( const QString &key ) 290void Config::removeEntry( const QString &key )
291{ 291{
292 if ( git == groups.end() ) { 292 if ( git == groups.end() ) {
293 qWarning( "no group set" ); 293 qWarning( "no group set" );
294 return; 294 return;
295 } 295 }
296 ( *git ).remove( key ); 296 ( *git ).remove( key );
297 changed = TRUE; 297 changed = TRUE;
298} 298}
299 299
300/*! 300/*!
301 \fn bool Config::operator == ( const Config & other ) const 301 \fn bool Config::operator == ( const Config & other ) const
302 302
303 Tests for equality with \a other. Config objects are equal if they refer to the same filename. 303 Tests for equality with \a other. Config objects are equal if they refer to the same filename.
304*/ 304*/
305 305
306/*! 306/*!
307 \fn bool Config::operator != ( const Config & other ) const 307 \fn bool Config::operator != ( const Config & other ) const
308 308
309 Tests for inequality with \a other. Config objects are equal if they refer to the same filename. 309 Tests for inequality with \a other. Config objects are equal if they refer to the same filename.
310*/ 310*/
311 311
312/*! 312/*!
313 \fn QString Config::readEntry( const QString &key, const QString &deflt ) const 313 \fn QString Config::readEntry( const QString &key, const QString &deflt ) const
314 314
315 Reads a string entry stored with \a key, defaulting to \a deflt if there is no entry. 315 Reads a string entry stored with \a key, defaulting to \a deflt if there is no entry.
316*/ 316*/
317 317
318/*! 318/*!
319 \internal 319 \internal
320 For compatibility, non-const version. 320 For compatibility, non-const version.
321*/ 321*/
322QString Config::readEntry( const QString &key, const QString &deflt ) 322QString Config::readEntry( const QString &key, const QString &deflt )
323{ 323{
324 QString res = readEntryDirect( key+"["+lang+"]" ); 324 QString res = readEntryDirect( key+"["+lang+"]" );
325 if ( !res.isNull() ) 325 if ( !res.isNull() )
326 return res; 326 return res;
327 if ( !glang.isEmpty() ) { 327 if ( !glang.isEmpty() ) {
328 res = readEntryDirect( key+"["+glang+"]" ); 328 res = readEntryDirect( key+"["+glang+"]" );
329 if ( !res.isNull() ) 329 if ( !res.isNull() )
330 return res; 330 return res;
331 } 331 }
332 return readEntryDirect( key, deflt ); 332 return readEntryDirect( key, deflt );
333} 333}
334 334
335/*! 335/*!
336 \fn QString Config::readEntryCrypt( const QString &key, const QString &deflt ) const 336 \fn QString Config::readEntryCrypt( const QString &key, const QString &deflt ) const
337 337
338 Reads an encrypted string entry stored with \a key, defaulting to \a deflt if there is no entry. 338 Reads an encrypted string entry stored with \a key, defaulting to \a deflt if there is no entry.
339*/ 339*/
340 340
341/*! 341/*!
342 \internal 342 \internal
343 For compatibility, non-const version. 343 For compatibility, non-const version.
344*/ 344*/
345QString Config::readEntryCrypt( const QString &key, const QString &deflt ) 345QString Config::readEntryCrypt( const QString &key, const QString &deflt )
346{ 346{
347 QString res = readEntryDirect( key+"["+lang+"]" ); 347 QString res = readEntryDirect( key+"["+lang+"]" );
348 if ( res.isNull() && glang.isEmpty() ) 348 if ( res.isNull() && glang.isEmpty() )
349 res = readEntryDirect( key+"["+glang+"]" ); 349 res = readEntryDirect( key+"["+glang+"]" );
350 if ( res.isNull() ) 350 if ( res.isNull() )
351 res = readEntryDirect( key, QString::null ); 351 res = readEntryDirect( key, QString::null );
352 if ( res.isNull() ) 352 if ( res.isNull() )
353 return deflt; 353 return deflt;
354 return decipher(res); 354 return decipher(res);
355} 355}
356 356
357/*! 357/*!
358 \fn QString Config::readEntryDirect( const QString &key, const QString &deflt ) const 358 \fn QString Config::readEntryDirect( const QString &key, const QString &deflt ) const
359 \internal 359 \internal
360*/ 360*/
361 361
362/*! 362/*!
363 \internal 363 \internal
364 For compatibility, non-const version. 364 For compatibility, non-const version.
365*/ 365*/
366QString Config::readEntryDirect( const QString &key, const QString &deflt ) 366QString Config::readEntryDirect( const QString &key, const QString &deflt )
367{ 367{
368 if ( git == groups.end() ) { 368 if ( git == groups.end() ) {
369 //qWarning( "no group set" ); 369 //qWarning( "no group set" );
370 return deflt; 370 return deflt;
371 } 371 }
372 ConfigGroup::ConstIterator it = ( *git ).find( key ); 372 ConfigGroup::ConstIterator it = ( *git ).find( key );
373 if ( it != ( *git ).end() ) 373 if ( it != ( *git ).end() )
374 return *it; 374 return *it;
375 else 375 else
376 return deflt; 376 return deflt;
377} 377}
378 378
379/*! 379/*!
380 \fn int Config::readNumEntry( const QString &key, int deflt ) const 380 \fn int Config::readNumEntry( const QString &key, int deflt ) const
381 Reads a numeric entry stored with \a key, defaulting to \a deflt if there is no entry. 381 Reads a numeric entry stored with \a key, defaulting to \a deflt if there is no entry.
382*/ 382*/
383 383
384/*! 384/*!
385 \internal 385 \internal
386 For compatibility, non-const version. 386 For compatibility, non-const version.
387*/ 387*/
388int Config::readNumEntry( const QString &key, int deflt ) 388int Config::readNumEntry( const QString &key, int deflt )
389{ 389{
390 QString s = readEntry( key ); 390 QString s = readEntry( key );
391 if ( s.isEmpty() ) 391 if ( s.isEmpty() )
392 return deflt; 392 return deflt;
393 else 393 else
394 return s.toInt(); 394 return s.toInt();
395} 395}
396 396
397/*! 397/*!
398 \fn bool Config::readBoolEntry( const QString &key, bool deflt ) const 398 \fn bool Config::readBoolEntry( const QString &key, bool deflt ) const
399 Reads a bool entry stored with \a key, defaulting to \a deflt if there is no entry. 399 Reads a bool entry stored with \a key, defaulting to \a deflt if there is no entry.
400*/ 400*/
401 401
402/*! 402/*!
403 \internal 403 \internal
404 For compatibility, non-const version. 404 For compatibility, non-const version.
405*/ 405*/
406bool Config::readBoolEntry( const QString &key, bool deflt ) 406bool Config::readBoolEntry( const QString &key, bool deflt )
407{ 407{
408 QString s = readEntry( key ); 408 QString s = readEntry( key );
409 if ( s.isEmpty() ) 409 if ( s.isEmpty() )
410 return deflt; 410 return deflt;
411 else 411 else
412 return (bool)s.toInt(); 412 return (bool)s.toInt();
413} 413}
414 414
415/*! 415/*!
416 \fn QStringList Config::readListEntry( const QString &key, const QChar &sep ) const 416 \fn QStringList Config::readListEntry( const QString &key, const QChar &sep ) const
417 Reads a string list entry stored with \a key, and with \a sep as the separator. 417 Reads a string list entry stored with \a key, and with \a sep as the separator.
418*/ 418*/
419 419
420/*! 420/*!
421 \internal 421 \internal
422 For compatibility, non-const version. 422 For compatibility, non-const version.
423*/ 423*/
424QStringList Config::readListEntry( const QString &key, const QChar &sep ) 424QStringList Config::readListEntry( const QString &key, const QChar &sep )
425{ 425{
426 QString s = readEntry( key ); 426 QString s = readEntry( key );
427 if ( s.isEmpty() ) 427 if ( s.isEmpty() )
428 return QStringList(); 428 return QStringList();
429 else 429 else
430 return QStringList::split( sep, s ); 430 return QStringList::split( sep, s );
431} 431}
432 432
433/*! 433/*!
434 Removes all entries from the current group. 434 Removes all entries from the current group.
435*/ 435*/
436void Config::clearGroup() 436void Config::clearGroup()
437{ 437{
438 if ( git == groups.end() ) { 438 if ( git == groups.end() ) {
439 qWarning( "no group set" ); 439 qWarning( "no group set" );
440 return; 440 return;
441 } 441 }
442 if ( !(*git).isEmpty() ) { 442 if ( !(*git).isEmpty() ) {
443 ( *git ).clear(); 443 ( *git ).clear();
444 changed = TRUE; 444 changed = TRUE;
445 } 445 }
446} 446}
447 447
448/*! 448/*!
449 \internal 449 \internal
450*/ 450*/
451void Config::write( const QString &fn ) 451void Config::write( const QString &fn )
452{ 452{
453 QString strNewFile; 453 QString strNewFile;
454 if ( !fn.isEmpty() ) 454 if ( !fn.isEmpty() )
455 filename = fn; 455 filename = fn;
456 strNewFile = filename + ".new"; 456 strNewFile = filename + ".new";
457 457
458 QFile f( strNewFile ); 458 QFile f( strNewFile );
459 if ( !f.open( IO_WriteOnly|IO_Raw ) ) { 459 if ( !f.open( IO_WriteOnly|IO_Raw ) ) {
460 qWarning( "could not open for writing `%s'", strNewFile.latin1() ); 460 qWarning( "could not open for writing `%s'", strNewFile.latin1() );
461 git = groups.end(); 461 git = groups.end();
462 return; 462 return;
463 } 463 }
464 464
465 QString str; 465 QString str;
466 QCString cstr; 466 QCString cstr;
467 QMap< QString, ConfigGroup >::Iterator g_it = groups.begin(); 467 QMap< QString, ConfigGroup >::Iterator g_it = groups.begin();
468 468
469 for ( ; g_it != groups.end(); ++g_it ) { 469 for ( ; g_it != groups.end(); ++g_it ) {
470 str += "[" + g_it.key() + "]\n"; 470 str += "[" + g_it.key() + "]\n";
471 ConfigGroup::Iterator e_it = ( *g_it ).begin(); 471 ConfigGroup::Iterator e_it = ( *g_it ).begin();
472 for ( ; e_it != ( *g_it ).end(); ++e_it ) 472 for ( ; e_it != ( *g_it ).end(); ++e_it )
473 str += e_it.key() + " = " + *e_it + "\n"; 473 str += e_it.key() + " = " + *e_it + "\n";
474 } 474 }
475 cstr = str.utf8(); 475 cstr = str.utf8();
476 476
477 int total_length; 477 int total_length;
478 total_length = f.writeBlock( cstr.data(), cstr.length() ); 478 total_length = f.writeBlock( cstr.data(), cstr.length() );
479 if ( total_length != int(cstr.length()) ) { 479 if ( total_length != int(cstr.length()) ) {
480 QMessageBox::critical( 0, QObject::tr("Out of Space"), 480 QMessageBox::critical( 0, QObject::tr("Out of Space"),
481 QObject::tr("There was a problem creating\nConfiguration Information \nfor this program.\n\nPlease free up some space and\ntry again.") ); 481 QObject::tr("There was a problem creating\nConfiguration Information \nfor this program.\n\nPlease free up some space and\ntry again.") );
482 f.close(); 482 f.close();
483 QFile::remove( strNewFile ); 483 QFile::remove( strNewFile );
484 return; 484 return;
485 } 485 }
486 486
487 f.close(); 487 f.close();
488 // now rename the file... 488 // now rename the file...
489 if ( rename( strNewFile, filename ) < 0 ) { 489 if ( rename( strNewFile, filename ) < 0 ) {
490 qWarning( "problem renaming the file %s to %s", strNewFile.latin1(), 490 qWarning( "problem renaming the file %s to %s", strNewFile.latin1(),
491 filename.latin1() ); 491 filename.latin1() );
492 QFile::remove( strNewFile ); 492 QFile::remove( strNewFile );
493 } 493 }
494} 494}
495 495
496/*! 496/*!
497 Returns whether the Config is in a valid state. 497 Returns whether the Config is in a valid state.
498*/ 498*/
499bool Config::isValid() const 499bool Config::isValid() const
500{ 500{
501 return groups.end() != git; 501 return groups.end() != git;
502} 502}
503 503
504/*! 504/*!
505 \internal 505 \internal
506*/ 506*/
507void Config::read() 507void Config::read()
508{ 508{
509 changed = FALSE; 509 changed = FALSE;
510 510
511 if ( !QFileInfo( filename ).exists() ) { 511 if ( !QFileInfo( filename ).exists() ) {
512 git = groups.end(); 512 git = groups.end();
513 return; 513 return;
514 } 514 }
515 515
516 QFile f( filename ); 516 QFile f( filename );
517 if ( !f.open( IO_ReadOnly ) ) { 517 if ( !f.open( IO_ReadOnly ) ) {
518 git = groups.end(); 518 git = groups.end();
519 return; 519 return;
520 } 520 }
521 521
522 522
523 // hack to avoid problems if big files are passed to test 523 // hack to avoid problems if big files are passed to test
524 // if they are valid configs ( like passing a mp3 ... ) 524 // if they are valid configs ( like passing a mp3 ... )
525 // I just hope that there are no conf files > 100000 byte 525 // I just hope that there are no conf files > 100000 byte
526 // not the best solution, find something else later 526 // not the best solution, find something else later
527 if ( f.size() > 100000 ) { 527 if ( f.getch()!='[' ||f.size() > 100000 ) {
528 git = groups.end();
528 return; 529 return;
529 } 530 }
531 f.ungetch('[');
530 532
531 533
532 QTextStream s( &f ); 534 QTextStream s( &f );
533#if QT_VERSION <= 230 && defined(QT_NO_CODECS) 535#if QT_VERSION <= 230 && defined(QT_NO_CODECS)
534 // The below should work, but doesn't in Qt 2.3.0 536 // The below should work, but doesn't in Qt 2.3.0
535 s.setCodec( QTextCodec::codecForMib( 106 ) ); 537 s.setCodec( QTextCodec::codecForMib( 106 ) );
536#else 538#else
537 s.setEncoding( QTextStream::UnicodeUTF8 ); 539 s.setEncoding( QTextStream::UnicodeUTF8 );
538#endif 540#endif
539 541
540 QStringList list = QStringList::split('\n', s.read() ); 542 QStringList list = QStringList::split('\n', s.read() );
541 f.close(); 543 f.close();
542 544
543 for ( QStringList::Iterator it = list.begin(); it != list.end(); ++it ) { 545 for ( QStringList::Iterator it = list.begin(); it != list.end(); ++it ) {
544 if ( !parse( *it ) ) { 546 if ( !parse( *it ) ) {
545 git = groups.end(); 547 git = groups.end();
546 return; 548 return;
547 } 549 }
548 } 550 }
549} 551}
550 552
551/*! 553/*!
552 \internal 554 \internal
553*/ 555*/
554bool Config::parse( const QString &l ) 556bool Config::parse( const QString &l )
555{ 557{
556 QString line = l.stripWhiteSpace(); 558 QString line = l.stripWhiteSpace();
557 559
558 if ( line [0] == QChar ( '#' )) 560 if ( line [0] == QChar ( '#' ))
559 return true; // ignore comments 561 return true; // ignore comments
560 562
561 if ( line[ 0 ] == QChar( '[' ) ) { 563 if ( line[ 0 ] == QChar( '[' ) ) {
562 QString gname = line; 564 QString gname = line;
563 gname = gname.remove( 0, 1 ); 565 gname = gname.remove( 0, 1 );
564 if ( gname[ (int)gname.length() - 1 ] == QChar( ']' ) ) 566 if ( gname[ (int)gname.length() - 1 ] == QChar( ']' ) )
565 gname = gname.remove( gname.length() - 1, 1 ); 567 gname = gname.remove( gname.length() - 1, 1 );
566 git = groups.insert( gname, ConfigGroup() ); 568 git = groups.insert( gname, ConfigGroup() );
567 } else if ( !line.isEmpty() ) { 569 } else if ( !line.isEmpty() ) {
568 if ( git == groups.end() ) 570 if ( git == groups.end() )
569 return FALSE; 571 return FALSE;
570 int eq = line.find( '=' ); 572 int eq = line.find( '=' );
571 if ( eq == -1 ) 573 if ( eq == -1 )
572 return FALSE; 574 return FALSE;
573 QString key = line.left(eq).stripWhiteSpace(); 575 QString key = line.left(eq).stripWhiteSpace();
574 QString value = line.mid(eq+1).stripWhiteSpace(); 576 QString value = line.mid(eq+1).stripWhiteSpace();
575 ( *git ).insert( key, value ); 577 ( *git ).insert( key, value );
576 } 578 }
577 return TRUE; 579 return TRUE;
578} 580}