]>
git.saurik.com Git - apt.git/blob - apt-pkg/contrib/cmndline.cc
   1 // -*- mode: cpp; mode: fold -*- 
   3 // $Id: cmndline.cc,v 1.15 2003/02/10 01:40:58 doogie Exp $ 
   4 /* ###################################################################### 
   6    Command Line Class - Sophisticated command line parser 
   8    This source is placed in the Public Domain, do with it what you will 
   9    It was originally written by Jason Gunthorpe <jgg@debian.org>. 
  11    ##################################################################### */ 
  13 // Include files                                                        /*{{{*/ 
  14 #include <apt-pkg/cmndline.h> 
  15 #include <apt-pkg/error.h> 
  16 #include <apt-pkg/strutl.h> 
  22 // CommandLine::CommandLine - Constructor                               /*{{{*/ 
  23 // --------------------------------------------------------------------- 
  25 CommandLine::CommandLine(Args 
*AList
,Configuration 
*Conf
) : ArgList(AList
),  
  26                                  Conf(Conf
), FileList(0) 
  30 // CommandLine::~CommandLine - Destructor                               /*{{{*/ 
  31 // --------------------------------------------------------------------- 
  33 CommandLine::~CommandLine() 
  38 // CommandLine::Parse - Main action member                              /*{{{*/ 
  39 // --------------------------------------------------------------------- 
  41 bool CommandLine::Parse(int argc
,const char **argv
) 
  44    FileList 
= new const char *[argc
]; 
  45    const char **Files 
= FileList
; 
  47    for (I 
= 1; I 
!= argc
; I
++) 
  49       const char *Opt 
= argv
[I
]; 
  51       // It is not an option 
  60       // Double dash signifies the end of option processing 
  61       if (*Opt 
== '-' && Opt
[1] == 0) 
  67       // Single dash is a short option 
  70          // Iterate over each letter 
  73             // Search for the option 
  75             for (A 
= ArgList
; A
->end() == false && A
->ShortOpt 
!= *Opt
; A
++); 
  77                return _error
->Error(_("Command line option '%c' [from %s] is not known."),*Opt
,argv
[I
]); 
  79             if (HandleOpt(I
,argc
,argv
,Opt
,A
) == false) 
  89       // Match up to a = against the list 
  90       const char *OptEnd 
= Opt
; 
  92       for (; *OptEnd 
!= 0 && *OptEnd 
!= '='; OptEnd
++); 
  93       for (A 
= ArgList
; A
->end() == false &&  
  94            stringcasecmp(Opt
,OptEnd
,A
->LongOpt
) != 0; A
++); 
  96       // Failed, look for a word after the first - (no-foo) 
  97       bool PreceedMatch 
= false; 
 100          for (; Opt 
!= OptEnd 
&& *Opt 
!= '-'; Opt
++); 
 103             return _error
->Error(_("Command line option %s is not understood"),argv
[I
]); 
 106          for (A 
= ArgList
; A
->end() == false && 
 107               stringcasecmp(Opt
,OptEnd
,A
->LongOpt
) != 0; A
++); 
 110          if (A
->end() == true && OptEnd 
- Opt 
!= 1) 
 111             return _error
->Error(_("Command line option %s is not understood"),argv
[I
]); 
 113          // The option could be a single letter option prefixed by a no-.. 
 114          if (A
->end() == true) 
 116             for (A 
= ArgList
; A
->end() == false && A
->ShortOpt 
!= *Opt
; A
++); 
 118             if (A
->end() == true) 
 119                return _error
->Error(_("Command line option %s is not understood"),argv
[I
]); 
 122          // The option is not boolean 
 123          if (A
->IsBoolean() == false) 
 124             return _error
->Error(_("Command line option %s is not boolean"),argv
[I
]); 
 130       if (HandleOpt(I
,argc
,argv
,OptEnd
,A
,PreceedMatch
) == false) 
 134    // Copy any remaining file names over 
 135    for (; I 
!= argc
; I
++) 
 139    SaveInConfig(argc
, argv
); 
 144 // CommandLine::HandleOpt - Handle a single option including all flags  /*{{{*/ 
 145 // --------------------------------------------------------------------- 
 146 /* This is a helper function for parser, it looks at a given argument 
 147    and looks for specific patterns in the string, it gets tokanized 
 148    -ruffly- like -*[yes|true|enable]-(o|longopt)[=][ ][argument] */ 
 149 bool CommandLine::HandleOpt(int &I
,int argc
,const char *argv
[], 
 150                             const char *&Opt
,Args 
*A
,bool PreceedMatch
) 
 152    const char *Argument 
= 0; 
 153    bool CertainArg 
= false; 
 156    /* Determine the possible location of an option or 0 if their is 
 158    if (Opt
[1] == 0 || (Opt
[1] == '=' && Opt
[2] == 0)) 
 160       if (I 
+ 1 < argc 
&& argv
[I
+1][0] != '-') 
 161          Argument 
= argv
[I
+1]; 
 163       // Equals was specified but we fell off the end! 
 164       if (Opt
[1] == '=' && Argument 
== 0) 
 165          return _error
->Error(_("Option %s requires an argument."),argv
[I
]); 
 182    // Option is an argument set 
 183    if ((A
->Flags 
& HasArg
) == HasArg
) 
 186          return _error
->Error(_("Option %s requires an argument."),argv
[I
]); 
 190       // Parse a configuration file 
 191       if ((A
->Flags 
& ConfigFile
) == ConfigFile
) 
 192          return ReadConfigFile(*Conf
,Argument
); 
 194       // Arbitrary item specification 
 195       if ((A
->Flags 
& ArbItem
) == ArbItem
) 
 198          for (J 
= Argument
; *J 
!= 0 && *J 
!= '='; J
++); 
 200             return _error
->Error(_("Option %s: Configuration item specification must have an =<val>."),argv
[I
]); 
 206                return _error
->Error(_("Option %s: Configuration item specification must have an =<val>."),argv
[I
]); 
 207             Conf
->Set(string(Argument
,J
-Argument
),string(argv
[I
++ +1])); 
 210             Conf
->Set(string(Argument
,J
-Argument
),string(J
+1)); 
 215       const char *I 
= A
->ConfName
; 
 216       for (; *I 
!= 0 && *I 
!= ' '; I
++); 
 218          Conf
->Set(string(A
->ConfName
,0,I
-A
->ConfName
),string(I
+1) + Argument
); 
 220          Conf
->Set(A
->ConfName
,string(I
) + Argument
); 
 225    // Option is an integer level 
 226    if ((A
->Flags 
& IntLevel
) == IntLevel
) 
 228       // There might be an argument 
 232          unsigned long Value 
= strtol(Argument
,&EndPtr
,10); 
 234          // Conversion failed and the argument was specified with an =s 
 235          if (EndPtr 
== Argument 
&& CertainArg 
== true) 
 236             return _error
->Error(_("Option %s requires an integer argument, not '%s'"),argv
[I
],Argument
); 
 238          // Conversion was ok, set the value and return 
 239          if (EndPtr 
!= 0 && EndPtr 
!= Argument 
&& *EndPtr 
== 0) 
 241             Conf
->Set(A
->ConfName
,Value
); 
 248       // Increase the level 
 249       Conf
->Set(A
->ConfName
,Conf
->FindI(A
->ConfName
)+1); 
 253    // Option is a boolean 
 254    int Sense 
= -1;  // -1 is unspecified, 0 is yes 1 is no 
 256    // Look for an argument. 
 259       // Look at preceeding text 
 263          if (PreceedMatch 
== false) 
 266          if (strlen(argv
[I
]) >= sizeof(Buffer
)) 
 267             return _error
->Error(_("Option '%s' is too long"),argv
[I
]); 
 269          // Skip the leading dash 
 270          const char *J 
= argv
[I
]; 
 271          for (; *J 
!= 0 && *J 
== '-'; J
++); 
 273          const char *JEnd 
= J
; 
 274          for (; *JEnd 
!= 0 && *JEnd 
!= '-'; JEnd
++); 
 277             strncpy(Buffer
,J
,JEnd 
- J
); 
 278             Buffer
[JEnd 
- J
] = 0; 
 287       Sense 
= StringToBool(Argument
); 
 291          if (Argument 
!= Buffer
) 
 299       if (CertainArg 
== true) 
 300          return _error
->Error(_("Sense %s is not understood, try true or false."),Argument
); 
 305    // Indeterminate sense depends on the flag 
 308       if ((A
->Flags 
& InvBoolean
) == InvBoolean
) 
 314    Conf
->Set(A
->ConfName
,Sense
); 
 318 // CommandLine::FileSize - Count the number of filenames                /*{{{*/ 
 319 // --------------------------------------------------------------------- 
 321 unsigned int CommandLine::FileSize() const 
 323    unsigned int Count 
= 0; 
 324    for (const char **I 
= FileList
; I 
!= 0 && *I 
!= 0; I
++) 
 329 // CommandLine::DispatchArg - Do something with the first arg           /*{{{*/ 
 330 // --------------------------------------------------------------------- 
 332 bool CommandLine::DispatchArg(Dispatch 
*Map
,bool NoMatch
) 
 335    for (I 
= 0; Map
[I
].Match 
!= 0; I
++) 
 337       if (strcmp(FileList
[0],Map
[I
].Match
) == 0) 
 339          bool Res 
= Map
[I
].Handler(*this); 
 340          if (Res 
== false && _error
->PendingError() == false) 
 341             _error
->Error("Handler silently failed"); 
 347    if (Map
[I
].Match 
== 0) 
 350          _error
->Error(_("Invalid operation %s"),FileList
[0]); 
 356 // CommandLine::SaveInConfig - for output later in a logfile or so      /*{{{*/ 
 357 // --------------------------------------------------------------------- 
 358 /* We save the commandline here to have it around later for e.g. logging. 
 359    It feels a bit like a hack here and isn't bulletproof, but it is better 
 360    than nothing after all. */ 
 361 void CommandLine::SaveInConfig(unsigned int const &argc
, char const * const * const argv
) 
 363    char cmdline
[100 + argc 
* 50]; 
 364    unsigned int length 
= 0; 
 365    bool lastWasOption 
= false; 
 366    bool closeQuote 
= false; 
 367    for (unsigned int i 
= 0; i 
< argc 
&& length 
< sizeof(cmdline
); ++i
, ++length
) 
 369       for (unsigned int j 
= 0; argv
[i
][j
] != '\0' && length 
< sizeof(cmdline
)-1; ++j
, ++length
) 
 371          cmdline
[length
] = argv
[i
][j
]; 
 372          if (lastWasOption 
== true && argv
[i
][j
] == '=') 
 374             // That is possibly an option: Quote it if it includes spaces, 
 375             // the benefit is that this will eliminate also most false positives 
 376             const char* c 
= &argv
[i
][j
+1]; 
 377             for (; *c 
!= '\0' && *c 
!= ' '; ++c
); 
 378             if (*c 
== '\0') continue; 
 379             cmdline
[++length
] = '"'; 
 383       if (closeQuote 
== true) 
 384          cmdline
[length
++] = '"'; 
 385       // Problem: detects also --hello 
 386       if (cmdline
[length
-1] == 'o') 
 387          lastWasOption 
= true; 
 388       cmdline
[length
] = ' '; 
 390    cmdline
[--length
] = '\0'; 
 391    _config
->Set("CommandLine::AsString", cmdline
);