]>
git.saurik.com Git - apt.git/blob - apt-pkg/aptconfiguration.cc
   1 // -*- mode: cpp; mode: fold -*- 
   3 /* ###################################################################### 
   5    Provide access methods to various configuration settings, 
   6    setup defaults and returns validate settings. 
   8    ##################################################################### */ 
  10 // Include Files                                                        /*{{{*/ 
  11 #include <apt-pkg/aptconfiguration.h> 
  12 #include <apt-pkg/configuration.h> 
  13 #include <apt-pkg/error.h> 
  14 #include <apt-pkg/fileutl.h> 
  15 #include <apt-pkg/macros.h> 
  16 #include <apt-pkg/strutl.h> 
  18 #include <sys/types.h> 
  27 // getCompressionTypes - Return Vector of usbale compressiontypes       /*{{{*/ 
  28 // --------------------------------------------------------------------- 
  29 /* return a vector of compression types in the prefered order. */ 
  30 std::vector
<std::string
> 
  31 const Configuration::getCompressionTypes(bool const &Cached
) { 
  32         static std::vector
<std::string
> types
; 
  33         if (types
.empty() == false) { 
  40         // setup the defaults for the compressiontypes => method mapping 
  41         _config
->CndSet("Acquire::CompressionTypes::bz2","bzip2"); 
  42         _config
->CndSet("Acquire::CompressionTypes::xz","xz"); 
  43         _config
->CndSet("Acquire::CompressionTypes::lzma","lzma"); 
  44         _config
->CndSet("Acquire::CompressionTypes::gz","gzip"); 
  46         setDefaultConfigurationForCompressors(); 
  48         // accept non-list order as override setting for config settings on commandline 
  49         std::string 
const overrideOrder 
= _config
->Find("Acquire::CompressionTypes::Order",""); 
  50         if (overrideOrder
.empty() == false) 
  51                 types
.push_back(overrideOrder
); 
  53         // load the order setting into our vector 
  54         std::vector
<std::string
> const order 
= _config
->FindVector("Acquire::CompressionTypes::Order"); 
  55         for (std::vector
<std::string
>::const_iterator o 
= order
.begin(); 
  56              o 
!= order
.end(); ++o
) { 
  57                 if ((*o
).empty() == true) 
  59                 // ignore types we have no method ready to use 
  60                 if (_config
->Exists(string("Acquire::CompressionTypes::").append(*o
)) == false) 
  62                 // ignore types we have no app ready to use 
  63                 string 
const appsetting 
= string("Dir::Bin::").append(*o
); 
  64                 if (_config
->Exists(appsetting
) == true) { 
  65                         std::string 
const app 
= _config
->FindFile(appsetting
.c_str(), ""); 
  66                         if (app
.empty() == false && FileExists(app
) == false) 
  72         // move again over the option tree to add all missing compression types 
  73         ::Configuration::Item 
const *Types 
= _config
->Tree("Acquire::CompressionTypes"); 
  77         for (; Types 
!= 0; Types 
= Types
->Next
) { 
  78                 if (Types
->Tag 
== "Order" || Types
->Tag
.empty() == true) 
  80                 // ignore types we already have in the vector 
  81                 if (std::find(types
.begin(),types
.end(),Types
->Tag
) != types
.end()) 
  83                 // ignore types we have no app ready to use 
  84                 string 
const appsetting 
= string("Dir::Bin::").append(Types
->Value
); 
  85                 if (appsetting
.empty() == false && _config
->Exists(appsetting
) == true) { 
  86                         std::string 
const app 
= _config
->FindFile(appsetting
.c_str(), ""); 
  87                         if (app
.empty() == false && FileExists(app
) == false) 
  90                 types
.push_back(Types
->Tag
); 
  93         // add the special "uncompressed" type 
  94         if (std::find(types
.begin(), types
.end(), "uncompressed") == types
.end()) 
  96                 string 
const uncompr 
= _config
->FindFile("Dir::Bin::uncompressed", ""); 
  97                 if (uncompr
.empty() == true || FileExists(uncompr
) == true) 
  98                         types
.push_back("uncompressed"); 
 104 // GetLanguages - Return Vector of Language Codes                       /*{{{*/ 
 105 // --------------------------------------------------------------------- 
 106 /* return a vector of language codes in the prefered order. 
 107    the special word "environment" will be replaced with the long and the short 
 108    code of the local settings and it will be insured that this will not add 
 109    duplicates. So in an german local the setting "environment, de_DE, en, de" 
 110    will result in "de_DE, de, en". 
 111    The special word "none" is the stopcode for the not-All code vector */ 
 112 std::vector
<std::string
> const Configuration::getLanguages(bool const &All
, 
 113                                 bool const &Cached
, char const ** const Locale
) { 
 116         // The detection is boring and has a lot of cornercases, 
 117         // so we cache the results to calculated it only once. 
 118         std::vector
<string
> static allCodes
; 
 119         std::vector
<string
> static codes
; 
 121         // we have something in the cache 
 122         if (codes
.empty() == false || allCodes
.empty() == false) { 
 123                 if (Cached 
== true) { 
 124                         if(All 
== true && allCodes
.empty() == false) 
 134         // Include all Language codes we have a Translation file for in /var/lib/apt/lists 
 135         // so they will be all included in the Cache. 
 136         std::vector
<string
> builtin
; 
 137         DIR *D 
= opendir(_config
->FindDir("Dir::State::lists").c_str()); 
 139                 builtin
.push_back("none"); 
 140                 for (struct dirent 
*Ent 
= readdir(D
); Ent 
!= 0; Ent 
= readdir(D
)) { 
 141                         string 
const name 
= Ent
->d_name
; 
 142                         size_t const foundDash 
= name
.rfind("-"); 
 143                         size_t const foundUnderscore 
= name
.rfind("_"); 
 144                         if (foundDash 
== string::npos 
|| foundUnderscore 
== string::npos 
|| 
 145                             foundDash 
<= foundUnderscore 
|| 
 146                             name
.substr(foundUnderscore
+1, foundDash
-(foundUnderscore
+1)) != "Translation") 
 148                         string 
const c 
= name
.substr(foundDash
+1); 
 149                         if (unlikely(c
.empty() == true) || c 
== "en") 
 151                         // Skip unusual files, like backups or that alike 
 152                         string::const_iterator s 
= c
.begin(); 
 153                         for (;s 
!= c
.end(); ++s
) { 
 154                                 if (isalpha(*s
) == 0) 
 159                         if (std::find(builtin
.begin(), builtin
.end(), c
) != builtin
.end()) 
 161                         builtin
.push_back(c
); 
 166         // FIXME: Remove support for the old APT::Acquire::Translation 
 167         // it was undocumented and so it should be not very widthly used 
 168         string 
const oldAcquire 
= _config
->Find("APT::Acquire::Translation",""); 
 169         if (oldAcquire
.empty() == false && oldAcquire 
!= "environment") { 
 170                 // TRANSLATORS: the two %s are APT configuration options 
 171                 _error
->Notice("Option '%s' is deprecated. Please use '%s' instead, see 'man 5 apt.conf' for details.", 
 172                                 "APT::Acquire::Translation", "Acquire::Languages"); 
 173                 if (oldAcquire 
!= "none") 
 174                         codes
.push_back(oldAcquire
); 
 175                 codes
.push_back("en"); 
 177                 for (std::vector
<string
>::const_iterator b 
= builtin
.begin(); 
 178                      b 
!= builtin
.end(); ++b
) 
 179                         if (std::find(allCodes
.begin(), allCodes
.end(), *b
) == allCodes
.end()) 
 180                                 allCodes
.push_back(*b
); 
 187         // get the environment language codes: LC_MESSAGES (and later LANGUAGE) 
 188         // we extract both, a long and a short code and then we will 
 189         // check if we actually need both (rare) or if the short is enough 
 190         string 
const envMsg 
= string(Locale 
== 0 ? std::setlocale(LC_MESSAGES
, NULL
) : *Locale
); 
 191         size_t const lenShort 
= (envMsg
.find('_') != string::npos
) ? envMsg
.find('_') : 2; 
 192         size_t const lenLong 
= (envMsg
.find_first_of(".@") != string::npos
) ? envMsg
.find_first_of(".@") : (lenShort 
+ 3); 
 194         string 
const envLong 
= envMsg
.substr(0,lenLong
); 
 195         string 
const envShort 
= envLong
.substr(0,lenShort
); 
 197         // It is very likely we will need the environment codes later, 
 198         // so let us generate them now from LC_MESSAGES and LANGUAGE 
 199         std::vector
<string
> environment
; 
 200         if (envShort 
!= "C") { 
 201                 // take care of LC_MESSAGES 
 202                 if (envLong 
!= envShort
) 
 203                         environment
.push_back(envLong
); 
 204                 environment
.push_back(envShort
); 
 205                 // take care of LANGUAGE 
 206                 const char *language_env 
= getenv("LANGUAGE") == 0 ? "" : getenv("LANGUAGE"); 
 207                 string envLang 
= Locale 
== 0 ? language_env 
: *(Locale
+1); 
 208                 if (envLang
.empty() == false) { 
 209                         std::vector
<string
> env 
= VectorizeString(envLang
,':'); 
 210                         short addedLangs 
= 0; // add a maximum of 3 fallbacks from the environment 
 211                         for (std::vector
<string
>::const_iterator e 
= env
.begin(); 
 212                              e 
!= env
.end() && addedLangs 
< 3; ++e
) { 
 213                                 if (unlikely(e
->empty() == true) || *e 
== "en") 
 215                                 if (*e 
== envLong 
|| *e 
== envShort
) 
 217                                 if (std::find(environment
.begin(), environment
.end(), *e
) != environment
.end()) 
 220                                 environment
.push_back(*e
); 
 224                 environment
.push_back("en"); 
 227         // Support settings like Acquire::Languages=none on the command line to 
 228         // override the configuration settings vector of languages. 
 229         string 
const forceLang 
= _config
->Find("Acquire::Languages",""); 
 230         if (forceLang
.empty() == false) { 
 231                 if (forceLang 
== "environment") { 
 233                 } else if (forceLang 
!= "none") 
 234                         codes
.push_back(forceLang
); 
 236                 for (std::vector
<string
>::const_iterator b 
= builtin
.begin(); 
 237                      b 
!= builtin
.end(); ++b
) 
 238                         if (std::find(allCodes
.begin(), allCodes
.end(), *b
) == allCodes
.end()) 
 239                                 allCodes
.push_back(*b
); 
 246         // cornercase: LANG=C, so we use only "en" Translation 
 247         if (envShort 
== "C") { 
 248                 allCodes 
= codes 
= environment
; 
 249                 allCodes
.insert(allCodes
.end(), builtin
.begin(), builtin
.end()); 
 256         std::vector
<string
> const lang 
= _config
->FindVector("Acquire::Languages"); 
 257         // the default setting -> "environment, en" 
 258         if (lang
.empty() == true) { 
 260                 if (envShort 
!= "en") 
 261                         codes
.push_back("en"); 
 263                 for (std::vector
<string
>::const_iterator b 
= builtin
.begin(); 
 264                      b 
!= builtin
.end(); ++b
) 
 265                         if (std::find(allCodes
.begin(), allCodes
.end(), *b
) == allCodes
.end()) 
 266                                 allCodes
.push_back(*b
); 
 273         // the configs define the order, so add the environment 
 274         // then needed and ensure the codes are not listed twice. 
 275         bool noneSeen 
= false; 
 276         for (std::vector
<string
>::const_iterator l 
= lang
.begin(); 
 277              l 
!= lang
.end(); ++l
) { 
 278                 if (*l 
== "environment") { 
 279                         for (std::vector
<string
>::const_iterator e 
= environment
.begin(); 
 280                              e 
!= environment
.end(); ++e
) { 
 281                                 if (std::find(allCodes
.begin(), allCodes
.end(), *e
) != allCodes
.end()) 
 283                                 if (noneSeen 
== false) 
 285                                 allCodes
.push_back(*e
); 
 288                 } else if (*l 
== "none") { 
 291                 } else if (std::find(allCodes
.begin(), allCodes
.end(), *l
) != allCodes
.end()) 
 294                 if (noneSeen 
== false) 
 296                 allCodes
.push_back(*l
); 
 299         for (std::vector
<string
>::const_iterator b 
= builtin
.begin(); 
 300              b 
!= builtin
.end(); ++b
) 
 301                 if (std::find(allCodes
.begin(), allCodes
.end(), *b
) == allCodes
.end()) 
 302                         allCodes
.push_back(*b
); 
 310 // getArchitectures - Return Vector of prefered Architectures           /*{{{*/ 
 311 std::vector
<std::string
> const Configuration::getArchitectures(bool const &Cached
) { 
 314         std::vector
<string
> static archs
; 
 315         if (likely(Cached 
== true) && archs
.empty() == false) 
 318         string 
const arch 
= _config
->Find("APT::Architecture"); 
 319         archs 
= _config
->FindVector("APT::Architectures"); 
 321         if (unlikely(arch
.empty() == true)) 
 324         // FIXME: It is a bit unclean to have debian specific code here⦠
 325         if (archs
.empty() == true) { 
 326                 archs
.push_back(arch
); 
 327                 string dpkgcall 
= _config
->Find("Dir::Bin::dpkg", "dpkg"); 
 328                 std::vector
<string
> const dpkgoptions 
= _config
->FindVector("DPkg::options"); 
 329                 for (std::vector
<string
>::const_iterator o 
= dpkgoptions
.begin(); 
 330                      o 
!= dpkgoptions
.end(); ++o
) 
 331                         dpkgcall
.append(" ").append(*o
); 
 332                 dpkgcall
.append(" --print-foreign-architectures 2> /dev/null"); 
 333                 FILE *dpkg 
= popen(dpkgcall
.c_str(), "r"); 
 336                         if (fgets(buf
, sizeof(buf
), dpkg
) != NULL
) { 
 337                                 char* arch 
= strtok(buf
, " "); 
 338                                 while (arch 
!= NULL
) { 
 339                                         for (; isspace(*arch
) != 0; ++arch
); 
 340                                         if (arch
[0] != '\0') { 
 341                                                 char const* archend 
= arch
; 
 342                                                 for (; isspace(*archend
) == 0 && *archend 
!= '\0'; ++archend
); 
 343                                                 archs
.push_back(string(arch
, (archend 
- arch
))); 
 345                                         arch 
= strtok(NULL
, " "); 
 353         if (archs
.empty() == true || 
 354             std::find(archs
.begin(), archs
.end(), arch
) == archs
.end()) 
 355                 archs
.insert(archs
.begin(), arch
); 
 357         // erase duplicates and empty strings 
 358         for (std::vector
<string
>::reverse_iterator a 
= archs
.rbegin(); 
 359              a 
!= archs
.rend(); ++a
) { 
 360                 if (a
->empty() == true || std::find(a 
+ 1, archs
.rend(), *a
) != archs
.rend()) 
 361                         archs
.erase(a
.base()-1); 
 362                 if (a 
== archs
.rend()) 
 369 // checkArchitecture - are we interested in the given Architecture?     /*{{{*/ 
 370 bool const Configuration::checkArchitecture(std::string 
const &Arch
) { 
 373         std::vector
<std::string
> const archs 
= getArchitectures(true); 
 374         return (std::find(archs
.begin(), archs
.end(), Arch
) != archs
.end()); 
 377 // setDefaultConfigurationForCompressors                                /*{{{*/ 
 378 void Configuration::setDefaultConfigurationForCompressors() { 
 379         // Set default application paths to check for optional compression types 
 380         _config
->CndSet("Dir::Bin::lzma", "/usr/bin/lzma"); 
 381         _config
->CndSet("Dir::Bin::xz", "/usr/bin/xz"); 
 382         _config
->CndSet("Dir::Bin::bzip2", "/bin/bzip2"); 
 385 // getCompressors - Return Vector of usbale compressors                 /*{{{*/ 
 386 // --------------------------------------------------------------------- 
 387 /* return a vector of compressors used by apt-ftparchive in the 
 388    multicompress functionality or to detect data.tar files */ 
 389 std::vector
<APT::Configuration::Compressor
> 
 390 const Configuration::getCompressors(bool const Cached
) { 
 391         static std::vector
<APT::Configuration::Compressor
> compressors
; 
 392         if (compressors
.empty() == false) { 
 399         setDefaultConfigurationForCompressors(); 
 401         compressors
.push_back(Compressor(".", "", "", "", "", 1)); 
 402         if (_config
->Exists("Dir::Bin::gzip") == false || FileExists(_config
->FindFile("Dir::Bin::gzip")) == true) 
 403                 compressors
.push_back(Compressor("gzip",".gz","gzip","-9n","-d",2)); 
 404         if (_config
->Exists("Dir::Bin::bzip2") == false || FileExists(_config
->FindFile("Dir::Bin::bzip2")) == true) 
 405                 compressors
.push_back(Compressor("bzip2",".bz2","bzip2","-9","-d",3)); 
 406         if (_config
->Exists("Dir::Bin::lzma") == false || FileExists(_config
->FindFile("Dir::Bin::lzma")) == true) 
 407                 compressors
.push_back(Compressor("lzma",".lzma","lzma","-9","-d",4)); 
 408         if (_config
->Exists("Dir::Bin::xz") == false || FileExists(_config
->FindFile("Dir::Bin::xz")) == true) 
 409                 compressors
.push_back(Compressor("xz",".xz","xz","-6","-d",5)); 
 411         std::vector
<std::string
> const comp 
= _config
->FindVector("APT::Compressor"); 
 412         for (std::vector
<std::string
>::const_iterator c 
= comp
.begin(); 
 413              c 
!= comp
.end(); ++c
) { 
 414                 if (*c 
== "." || *c 
== "gzip" || *c 
== "bzip2" || *c 
== "lzma" || *c 
== "xz") 
 416                 compressors
.push_back(Compressor(c
->c_str(), std::string(".").append(*c
).c_str(), c
->c_str(), "-9", "-d", 100)); 
 422 // getCompressorExtensions - supported data.tar extensions              /*{{{*/ 
 423 // --------------------------------------------------------------------- 
 425 std::vector
<std::string
> const Configuration::getCompressorExtensions() { 
 426         std::vector
<APT::Configuration::Compressor
> const compressors 
= getCompressors(); 
 427         std::vector
<std::string
> ext
; 
 428         for (std::vector
<APT::Configuration::Compressor
>::const_iterator c 
= compressors
.begin(); 
 429              c 
!= compressors
.end(); ++c
) 
 430                 if (c
->Extension
.empty() == false && c
->Extension 
!= ".") 
 431                         ext
.push_back(c
->Extension
); 
 435 // Compressor constructor                                               /*{{{*/ 
 436 // --------------------------------------------------------------------- 
 438 Configuration::Compressor::Compressor(char const *name
, char const *extension
, 
 440                                       char const *compressArg
, char const *uncompressArg
, 
 441                                       unsigned short const cost
) { 
 442         std::string 
const config 
= string("APT:Compressor::").append(name
).append("::"); 
 443         Name 
= _config
->Find(std::string(config
).append("Name"), name
); 
 444         Extension 
= _config
->Find(std::string(config
).append("Extension"), extension
); 
 445         Binary 
= _config
->Find(std::string(config
).append("Binary"), binary
); 
 446         Cost 
= _config
->FindI(std::string(config
).append("Cost"), cost
); 
 447         std::string 
const compConf 
= std::string(config
).append("CompressArg"); 
 448         if (_config
->Exists(compConf
) == true) 
 449                 CompressArgs 
= _config
->FindVector(compConf
); 
 450         else if (compressArg 
!= NULL
) 
 451                 CompressArgs
.push_back(compressArg
); 
 452         std::string 
const uncompConf 
= std::string(config
).append("UncompressArg"); 
 453         if (_config
->Exists(uncompConf
) == true) 
 454                 UncompressArgs 
= _config
->FindVector(uncompConf
); 
 455         else if (uncompressArg 
!= NULL
) 
 456                 UncompressArgs
.push_back(uncompressArg
);