]>
git.saurik.com Git - apt.git/blob - apt-pkg/aptconfiguration.cc
e97ebfed7e2e55c3299abbea8fbf7b64f0d0396b
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>
26 // getCompressionTypes - Return Vector of usbale compressiontypes /*{{{*/
27 // ---------------------------------------------------------------------
28 /* return a vector of compression types in the prefered order. */
29 std::vector
<std::string
>
30 const Configuration::getCompressionTypes(bool const &Cached
) {
31 static std::vector
<std::string
> types
;
32 if (types
.empty() == false) {
39 // setup the defaults for the compressiontypes => method mapping
40 _config
->CndSet("Acquire::CompressionTypes::bz2","bzip2");
41 _config
->CndSet("Acquire::CompressionTypes::lzma","lzma");
42 _config
->CndSet("Acquire::CompressionTypes::gz","gzip");
44 // Set default application paths to check for optional compression types
45 _config
->CndSet("Dir::Bin::lzma", "/usr/bin/lzma");
46 _config
->CndSet("Dir::Bin::bzip2", "/bin/bzip2");
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 // get the environment language codes: LC_MESSAGES (and later LANGUAGE)
167 // we extract both, a long and a short code and then we will
168 // check if we actually need both (rare) or if the short is enough
169 string
const envMsg
= string(Locale
== 0 ? std::setlocale(LC_MESSAGES
, NULL
) : *Locale
);
170 size_t const lenShort
= (envMsg
.find('_') != string::npos
) ? envMsg
.find('_') : 2;
171 size_t const lenLong
= (envMsg
.find_first_of(".@") != string::npos
) ? envMsg
.find_first_of(".@") : (lenShort
+ 3);
173 string envLong
= envMsg
.substr(0,lenLong
);
174 string
const envShort
= envLong
.substr(0,lenShort
);
175 bool envLongIncluded
= true;
177 // to save the servers from unneeded queries, we only try also long codes
178 // for languages it is realistic to have a long code translation fileā¦
179 // TODO: Improve translation acquire system to drop them dynamic
180 char const *needLong
[] = { "cs", "en", "pt", "sv", "zh", NULL
};
181 if (envLong
!= envShort
) {
182 for (char const **l
= needLong
; *l
!= NULL
; l
++)
183 if (envShort
.compare(*l
) == 0) {
184 envLongIncluded
= false;
189 // we don't add the long code, but we allow the user to do so
190 if (envLongIncluded
== true)
193 // FIXME: Remove support for the old APT::Acquire::Translation
194 // it was undocumented and so it should be not very widthly used
195 string
const oldAcquire
= _config
->Find("APT::Acquire::Translation","");
196 if (oldAcquire
.empty() == false && oldAcquire
!= "environment") {
197 // TRANSLATORS: the two %s are APT configuration options
198 _error
->Notice("Option '%s' is deprecated. Please use '%s' instead, see 'man 5 apt.conf' for details.",
199 "APT::Acquire::Translation", "Acquire::Languages");
200 if (oldAcquire
!= "none")
201 codes
.push_back(oldAcquire
);
202 codes
.push_back("en");
204 for (std::vector
<string
>::const_iterator b
= builtin
.begin();
205 b
!= builtin
.end(); ++b
)
206 if (std::find(allCodes
.begin(), allCodes
.end(), *b
) == allCodes
.end())
207 allCodes
.push_back(*b
);
214 // It is very likely we will need to environment codes later,
215 // so let us generate them now from LC_MESSAGES and LANGUAGE
216 std::vector
<string
> environment
;
217 if (envShort
!= "C") {
218 // take care of LC_MESSAGES
219 if (envLongIncluded
== false)
220 environment
.push_back(envLong
);
221 environment
.push_back(envShort
);
222 // take care of LANGUAGE
223 const char *language_env
= getenv("LANGUAGE") == 0 ? "" : getenv("LANGUAGE");
224 string envLang
= Locale
== 0 ? language_env
: *(Locale
+1);
225 if (envLang
.empty() == false) {
226 std::vector
<string
> env
= VectorizeString(envLang
,':');
227 short addedLangs
= 0; // add a maximum of 3 fallbacks from the environment
228 for (std::vector
<string
>::const_iterator e
= env
.begin();
229 e
!= env
.end() && addedLangs
< 3; ++e
) {
230 if (unlikely(e
->empty() == true) || *e
== "en")
232 if (*e
== envLong
|| *e
== envShort
)
234 if (std::find(environment
.begin(), environment
.end(), *e
) != environment
.end())
236 if (e
->find('_') != string::npos
) {
237 // Drop LongCodes here - ShortCodes are also included
238 string
const shorty
= e
->substr(0, e
->find('_'));
239 char const **n
= needLong
;
240 for (; *n
!= NULL
; ++n
)
247 environment
.push_back(*e
);
251 environment
.push_back("en");
254 // Support settings like Acquire::Translation=none on the command line to
255 // override the configuration settings vector of languages.
256 string
const forceLang
= _config
->Find("Acquire::Languages","");
257 if (forceLang
.empty() == false) {
258 if (forceLang
== "environment") {
260 } else if (forceLang
!= "none")
261 codes
.push_back(forceLang
);
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 // cornercase: LANG=C, so we use only "en" Translation
274 if (envShort
== "C") {
275 allCodes
= codes
= environment
;
276 allCodes
.insert(allCodes
.end(), builtin
.begin(), builtin
.end());
283 std::vector
<string
> const lang
= _config
->FindVector("Acquire::Languages");
284 // the default setting -> "environment, en"
285 if (lang
.empty() == true) {
287 if (envShort
!= "en")
288 codes
.push_back("en");
290 for (std::vector
<string
>::const_iterator b
= builtin
.begin();
291 b
!= builtin
.end(); ++b
)
292 if (std::find(allCodes
.begin(), allCodes
.end(), *b
) == allCodes
.end())
293 allCodes
.push_back(*b
);
300 // the configs define the order, so add the environment
301 // then needed and ensure the codes are not listed twice.
302 bool noneSeen
= false;
303 for (std::vector
<string
>::const_iterator l
= lang
.begin();
304 l
!= lang
.end(); l
++) {
305 if (*l
== "environment") {
306 for (std::vector
<string
>::const_iterator e
= environment
.begin();
307 e
!= environment
.end(); ++e
) {
308 if (std::find(allCodes
.begin(), allCodes
.end(), *e
) != allCodes
.end())
310 if (noneSeen
== false)
312 allCodes
.push_back(*e
);
315 } else if (*l
== "none") {
318 } else if (std::find(allCodes
.begin(), allCodes
.end(), *l
) != allCodes
.end())
321 if (noneSeen
== false)
323 allCodes
.push_back(*l
);
326 for (std::vector
<string
>::const_iterator b
= builtin
.begin();
327 b
!= builtin
.end(); ++b
)
328 if (std::find(allCodes
.begin(), allCodes
.end(), *b
) == allCodes
.end())
329 allCodes
.push_back(*b
);
337 // getArchitectures - Return Vector of prefered Architectures /*{{{*/
338 std::vector
<std::string
> const Configuration::getArchitectures(bool const &Cached
) {
341 std::vector
<string
> static archs
;
342 if (likely(Cached
== true) && archs
.empty() == false)
345 archs
= _config
->FindVector("APT::Architectures");
346 string
const arch
= _config
->Find("APT::Architecture");
347 if (unlikely(arch
.empty() == true))
350 if (archs
.empty() == true ||
351 std::find(archs
.begin(), archs
.end(), arch
) == archs
.end())
352 archs
.push_back(arch
);
354 // erase duplicates and empty strings
355 for (std::vector
<string
>::reverse_iterator a
= archs
.rbegin();
356 a
!= archs
.rend(); ++a
) {
357 if (a
->empty() == true || std::find(a
+ 1, archs
.rend(), *a
) != archs
.rend())
358 archs
.erase(a
.base()-1);
359 if (a
== archs
.rend())
366 // checkArchitecture - are we interested in the given Architecture? /*{{{*/
367 bool const Configuration::checkArchitecture(std::string
const &Arch
) {
370 std::vector
<std::string
> const archs
= getArchitectures(true);
371 return (std::find(archs
.begin(), archs
.end(), Arch
) != archs
.end());