]>
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/fileutl.h>
14 #include <apt-pkg/macros.h>
15 #include <apt-pkg/strutl.h>
17 #include <sys/types.h>
25 // getCompressionTypes - Return Vector of usbale compressiontypes /*{{{*/
26 // ---------------------------------------------------------------------
27 /* return a vector of compression types in the prefered order. */
28 std::vector
<std::string
>
29 const Configuration::getCompressionTypes(bool const &Cached
) {
30 static std::vector
<std::string
> types
;
31 if (types
.empty() == false) {
38 // setup the defaults for the compressiontypes => method mapping
39 _config
->CndSet("Acquire::CompressionTypes::bz2","bzip2");
40 _config
->CndSet("Acquire::CompressionTypes::lzma","lzma");
41 _config
->CndSet("Acquire::CompressionTypes::gz","gzip");
43 // Set default application paths to check for optional compression types
44 _config
->CndSet("Dir::Bin::lzma", "/usr/bin/lzma");
45 _config
->CndSet("Dir::Bin::bzip2", "/bin/bzip2");
47 // accept non-list order as override setting for config settings on commandline
48 std::string
const overrideOrder
= _config
->Find("Acquire::CompressionTypes::Order","");
49 if (overrideOrder
.empty() == false)
50 types
.push_back(overrideOrder
);
52 // load the order setting into our vector
53 std::vector
<std::string
> const order
= _config
->FindVector("Acquire::CompressionTypes::Order");
54 for (std::vector
<std::string
>::const_iterator o
= order
.begin();
55 o
!= order
.end(); o
++) {
56 if ((*o
).empty() == true)
58 // ignore types we have no method ready to use
59 if (_config
->Exists(string("Acquire::CompressionTypes::").append(*o
)) == false)
61 // ignore types we have no app ready to use
62 string
const appsetting
= string("Dir::Bin::").append(*o
);
63 if (_config
->Exists(appsetting
) == true) {
64 std::string
const app
= _config
->FindFile(appsetting
.c_str(), "");
65 if (app
.empty() == false && FileExists(app
) == false)
71 // move again over the option tree to add all missing compression types
72 ::Configuration::Item
const *Types
= _config
->Tree("Acquire::CompressionTypes");
76 for (; Types
!= 0; Types
= Types
->Next
) {
77 if (Types
->Tag
== "Order" || Types
->Tag
.empty() == true)
79 // ignore types we already have in the vector
80 if (std::find(types
.begin(),types
.end(),Types
->Tag
) != types
.end())
82 // ignore types we have no app ready to use
83 string
const appsetting
= string("Dir::Bin::").append(Types
->Value
);
84 if (appsetting
.empty() == false && _config
->Exists(appsetting
) == true) {
85 std::string
const app
= _config
->FindFile(appsetting
.c_str(), "");
86 if (app
.empty() == false && FileExists(app
) == false)
89 types
.push_back(Types
->Tag
);
95 // GetLanguages - Return Vector of Language Codes /*{{{*/
96 // ---------------------------------------------------------------------
97 /* return a vector of language codes in the prefered order.
98 the special word "environment" will be replaced with the long and the short
99 code of the local settings and it will be insured that this will not add
100 duplicates. So in an german local the setting "environment, de_DE, en, de"
101 will result in "de_DE, de, en".
102 The special word "none" is the stopcode for the not-All code vector */
103 std::vector
<std::string
> const Configuration::getLanguages(bool const &All
,
104 bool const &Cached
, char const ** const Locale
) {
107 // The detection is boring and has a lot of cornercases,
108 // so we cache the results to calculated it only once.
109 std::vector
<string
> static allCodes
;
110 std::vector
<string
> static codes
;
112 // we have something in the cache
113 if (codes
.empty() == false || allCodes
.empty() == false) {
114 if (Cached
== true) {
115 if(All
== true && allCodes
.empty() == false)
125 // Include all Language codes we have a Translation file for in /var/lib/apt/lists
126 // so they will be all included in the Cache.
127 std::vector
<string
> builtin
;
128 DIR *D
= opendir(_config
->FindDir("Dir::State::lists").c_str());
130 builtin
.push_back("none");
131 for (struct dirent
*Ent
= readdir(D
); Ent
!= 0; Ent
= readdir(D
)) {
132 string
const name
= Ent
->d_name
;
133 size_t const foundDash
= name
.rfind("-");
134 size_t const foundUnderscore
= name
.rfind("_");
135 if (foundDash
== string::npos
|| foundUnderscore
== string::npos
||
136 foundDash
<= foundUnderscore
||
137 name
.substr(foundUnderscore
+1, foundDash
-(foundUnderscore
+1)) != "Translation")
139 string
const c
= name
.substr(foundDash
+1);
140 if (unlikely(c
.empty() == true) || c
== "en")
142 // Skip unusual files, like backups or that alike
143 string::const_iterator s
= c
.begin();
144 for (;s
!= c
.end(); ++s
) {
145 if (isalpha(*s
) == 0)
150 if (std::find(builtin
.begin(), builtin
.end(), c
) != builtin
.end())
152 builtin
.push_back(c
);
156 // get the environment language codes: LC_MESSAGES (and later LANGUAGE)
157 // we extract both, a long and a short code and then we will
158 // check if we actually need both (rare) or if the short is enough
159 string
const envMsg
= string(Locale
== 0 ? std::setlocale(LC_MESSAGES
, NULL
) : *Locale
);
160 size_t const lenShort
= (envMsg
.find('_') != string::npos
) ? envMsg
.find('_') : 2;
161 size_t const lenLong
= (envMsg
.find_first_of(".@") != string::npos
) ? envMsg
.find_first_of(".@") : (lenShort
+ 3);
163 string envLong
= envMsg
.substr(0,lenLong
);
164 string
const envShort
= envLong
.substr(0,lenShort
);
165 bool envLongIncluded
= true;
167 // first cornercase: LANG=C, so we use only "en" Translation
168 if (envLong
== "C") {
169 codes
.push_back("en");
171 allCodes
.insert(allCodes
.end(), builtin
.begin(), builtin
.end());
178 // to save the servers from unneeded queries, we only try also long codes
179 // for languages it is realistic to have a long code translation fileā¦
180 // TODO: Improve translation acquire system to drop them dynamic
181 char const *needLong
[] = { "cs", "en", "pt", "sv", "zh", NULL
};
182 if (envLong
!= envShort
) {
183 for (char const **l
= needLong
; *l
!= NULL
; l
++)
184 if (envShort
.compare(*l
) == 0) {
185 envLongIncluded
= false;
190 // we don't add the long code, but we allow the user to do so
191 if (envLongIncluded
== true)
194 // FIXME: Remove support for the old APT::Acquire::Translation
195 // it was undocumented and so it should be not very widthly used
196 string
const oldAcquire
= _config
->Find("APT::Acquire::Translation","");
197 if (oldAcquire
.empty() == false && oldAcquire
!= "environment") {
198 if (oldAcquire
!= "none")
199 codes
.push_back(oldAcquire
);
200 codes
.push_back("en");
202 for (std::vector
<string
>::const_iterator b
= builtin
.begin();
203 b
!= builtin
.end(); ++b
)
204 if (std::find(allCodes
.begin(), allCodes
.end(), *b
) == allCodes
.end())
205 allCodes
.push_back(*b
);
212 // It is very likely we will need to environment codes later,
213 // so let us generate them now from LC_MESSAGES and LANGUAGE
214 std::vector
<string
> environment
;
215 // take care of LC_MESSAGES
216 if (envLongIncluded
== false)
217 environment
.push_back(envLong
);
218 environment
.push_back(envShort
);
219 // take care of LANGUAGE
220 string envLang
= Locale
== 0 ? getenv("LANGUAGE") : *(Locale
+1);
221 if (envLang
.empty() == false) {
222 std::vector
<string
> env
= ExplodeString(envLang
,':');
223 short addedLangs
= 0; // add a maximum of 3 fallbacks from the environment
224 for (std::vector
<string
>::const_iterator e
= env
.begin();
225 e
!= env
.end() && addedLangs
< 3; ++e
) {
226 if (unlikely(e
->empty() == true) || *e
== "en")
228 if (*e
== envLong
|| *e
== envShort
)
230 if (std::find(environment
.begin(), environment
.end(), *e
) != environment
.end())
232 if (e
->find('_') != string::npos
) {
233 // Drop LongCodes here - ShortCodes are also included
234 string
const shorty
= e
->substr(0, e
->find('_'));
235 char const **n
= needLong
;
236 for (; *n
!= NULL
; ++n
)
243 environment
.push_back(*e
);
247 // Support settings like Acquire::Translation=none on the command line to
248 // override the configuration settings vector of languages.
249 string
const forceLang
= _config
->Find("Acquire::Languages","");
250 if (forceLang
.empty() == false) {
251 if (forceLang
== "environment") {
253 } else if (forceLang
!= "none")
254 codes
.push_back(forceLang
);
256 for (std::vector
<string
>::const_iterator b
= builtin
.begin();
257 b
!= builtin
.end(); ++b
)
258 if (std::find(allCodes
.begin(), allCodes
.end(), *b
) == allCodes
.end())
259 allCodes
.push_back(*b
);
266 std::vector
<string
> const lang
= _config
->FindVector("Acquire::Languages");
267 // the default setting -> "environment, en"
268 if (lang
.empty() == true) {
270 if (envShort
!= "en")
271 codes
.push_back("en");
273 for (std::vector
<string
>::const_iterator b
= builtin
.begin();
274 b
!= builtin
.end(); ++b
)
275 if (std::find(allCodes
.begin(), allCodes
.end(), *b
) == allCodes
.end())
276 allCodes
.push_back(*b
);
283 // the configs define the order, so add the environment
284 // then needed and ensure the codes are not listed twice.
285 bool noneSeen
= false;
286 for (std::vector
<string
>::const_iterator l
= lang
.begin();
287 l
!= lang
.end(); l
++) {
288 if (*l
== "environment") {
289 for (std::vector
<string
>::const_iterator e
= environment
.begin();
290 e
!= environment
.end(); ++e
) {
291 if (std::find(allCodes
.begin(), allCodes
.end(), *e
) != allCodes
.end())
293 if (noneSeen
== false)
295 allCodes
.push_back(*e
);
298 } else if (*l
== "none") {
301 } else if (std::find(allCodes
.begin(), allCodes
.end(), *l
) != allCodes
.end())
304 if (noneSeen
== false)
306 allCodes
.push_back(*l
);
309 for (std::vector
<string
>::const_iterator b
= builtin
.begin();
310 b
!= builtin
.end(); ++b
)
311 if (std::find(allCodes
.begin(), allCodes
.end(), *b
) == allCodes
.end())
312 allCodes
.push_back(*b
);
320 // getArchitectures - Return Vector of prefered Architectures /*{{{*/
321 std::vector
<std::string
> const Configuration::getArchitectures(bool const &Cached
) {
324 std::vector
<string
> static archs
;
325 if (likely(Cached
== true) && archs
.empty() == false)
328 string
const arch
= _config
->Find("APT::Architecture");
329 archs
= _config
->FindVector("APT::Architectures");
330 if (archs
.empty() == true ||
331 std::find(archs
.begin(), archs
.end(), arch
) == archs
.end())
332 archs
.push_back(arch
);
336 // checkArchitecture - are we interested in the given Architecture? /*{{{*/
337 bool const Configuration::checkArchitecture(std::string
const &Arch
) {
340 std::vector
<std::string
> const archs
= getArchitectures(true);
341 return (std::find(archs
.begin(), archs
.end(), Arch
) != archs
.end());