]> git.saurik.com Git - apt.git/blame - apt-pkg/deb/deblistparser.cc
Convert package names from Packages files to lower case
[apt.git] / apt-pkg / deb / deblistparser.cc
CommitLineData
f55a958f
AL
1// -*- mode: cpp; mode: fold -*-
2// Description /*{{{*/
b3d44315 3// $Id: deblistparser.cc,v 1.29.2.5 2004/01/06 01:43:44 mdz Exp $
f55a958f
AL
4/* ######################################################################
5
6 Package Cache Generator - Generator for the cache structure.
7
8 This builds the cache structure from the abstract package list parser.
9
10 ##################################################################### */
11 /*}}}*/
12// Include Files /*{{{*/
ea542140
DK
13#include <config.h>
14
094a497d
AL
15#include <apt-pkg/deblistparser.h>
16#include <apt-pkg/error.h>
17#include <apt-pkg/configuration.h>
424ff669 18#include <apt-pkg/cachefilter.h>
45df0ad2 19#include <apt-pkg/aptconfiguration.h>
cdcc6d34 20#include <apt-pkg/strutl.h>
204fbdcc 21#include <apt-pkg/crc-16.h>
a52f938b 22#include <apt-pkg/md5.h>
453b82a3
DK
23#include <apt-pkg/pkgcache.h>
24#include <apt-pkg/cacheiterators.h>
25#include <apt-pkg/tagfile.h>
5c0d3668 26#include <apt-pkg/macros.h>
9c14e3d6 27
453b82a3
DK
28#include <stddef.h>
29#include <string.h>
30#include <algorithm>
31#include <string>
32#include <vector>
e7b470ee 33#include <ctype.h>
f55a958f
AL
34 /*}}}*/
35
8f3ba4e8
DK
36using std::string;
37
f77ea39c 38static const debListParser::WordList PrioList[] = {
84de0cea
MV
39 {"required",pkgCache::State::Required},
40 {"important",pkgCache::State::Important},
41 {"standard",pkgCache::State::Standard},
42 {"optional",pkgCache::State::Optional},
43 {"extra",pkgCache::State::Extra},
67c3067f 44 {NULL, 0}};
b2e465d6 45
f55a958f
AL
46// ListParser::debListParser - Constructor /*{{{*/
47// ---------------------------------------------------------------------
5dd4c8b8
DK
48/* Provide an architecture and only this one and "all" will be accepted
49 in Step(), if no Architecture is given we will accept every arch
50 we would accept in general with checkArchitecture() */
7f8c0eed
DK
51debListParser::debListParser(FileFd *File) :
52 pkgCacheListParser(), d(NULL), Tags(File)
c9443c01 53{
0149949b
AL
54}
55 /*}}}*/
f55a958f
AL
56// ListParser::Package - Return the package name /*{{{*/
57// ---------------------------------------------------------------------
58/* This is to return the name of the package this section describes */
5bf15716 59string debListParser::Package() {
fffa3b57
JAK
60 string Result = Section.FindS("Package");
61
62 // Normalize mixed case package names to lower case, like dpkg does
63 // See Bug#807012 for details
64 std::transform(Result.begin(), Result.end(), Result.begin(), tolower_ascii);
65
33dd02e3
DK
66 if(unlikely(Result.empty() == true))
67 _error->Error("Encountered a section with no Package: header");
68 return Result;
5bf15716
DK
69}
70 /*}}}*/
71// ListParser::Architecture - Return the package arch /*{{{*/
72// ---------------------------------------------------------------------
28166356 73/* This will return the Architecture of the package this section describes */
5bf15716 74string debListParser::Architecture() {
cc480836
DK
75 std::string const Arch = Section.FindS("Architecture");
76 return Arch.empty() ? "none" : Arch;
f55a958f
AL
77}
78 /*}}}*/
857e9c13
DK
79// ListParser::ArchitectureAll /*{{{*/
80// ---------------------------------------------------------------------
81/* */
82bool debListParser::ArchitectureAll() {
33dd02e3 83 return Section.FindS("Architecture") == "all";
f55a958f
AL
84}
85 /*}}}*/
86// ListParser::Version - Return the version string /*{{{*/
87// ---------------------------------------------------------------------
88/* This is to return the string describing the version in debian form,
89 epoch:upstream-release. If this returns the blank string then the
90 entry is assumed to only describe package properties */
91string debListParser::Version()
92{
b0b4efb9 93 return Section.FindS("Version");
f55a958f
AL
94}
95 /*}}}*/
e33dbfe5 96unsigned char debListParser::ParseMultiArch(bool const showErrors) /*{{{*/
0149949b 97{
e33dbfe5 98 unsigned char MA;
28166356 99 string const MultiArch = Section.FindS("Multi-Arch");
19fb04e8 100 if (MultiArch.empty() == true || MultiArch == "no")
22df31be 101 MA = pkgCache::Version::No;
28166356 102 else if (MultiArch == "same") {
ca238ede 103 if (ArchitectureAll() == true)
25396fb0 104 {
e33dbfe5
DK
105 if (showErrors == true)
106 _error->Warning("Architecture: all package '%s' can't be Multi-Arch: same",
107 Section.FindS("Package").c_str());
22df31be 108 MA = pkgCache::Version::No;
25396fb0 109 }
28166356 110 else
e33dbfe5 111 MA = pkgCache::Version::Same;
28166356
DK
112 }
113 else if (MultiArch == "foreign")
e33dbfe5 114 MA = pkgCache::Version::Foreign;
28166356 115 else if (MultiArch == "allowed")
e33dbfe5 116 MA = pkgCache::Version::Allowed;
28166356
DK
117 else
118 {
e33dbfe5
DK
119 if (showErrors == true)
120 _error->Warning("Unknown Multi-Arch type '%s' for package '%s'",
121 MultiArch.c_str(), Section.FindS("Package").c_str());
22df31be 122 MA = pkgCache::Version::No;
25396fb0
DK
123 }
124
ca238ede 125 if (ArchitectureAll() == true)
e33dbfe5 126 MA |= pkgCache::Version::All;
ca238ede 127
e33dbfe5
DK
128 return MA;
129}
130 /*}}}*/
131// ListParser::NewVersion - Fill in the version structure /*{{{*/
132// ---------------------------------------------------------------------
133/* */
134bool debListParser::NewVersion(pkgCache::VerIterator &Ver)
135{
78a5476f
DK
136 const char *Start;
137 const char *Stop;
138
e33dbfe5 139 // Parse the section
78a5476f
DK
140 if (Section.Find("Section",Start,Stop) == true)
141 {
142 map_stringitem_t const idx = StoreString(pkgCacheGenerator::SECTION, Start, Stop - Start);
143 Ver->Section = idx;
144 }
a221efc3
DK
145 // Parse the source package name
146 pkgCache::GrpIterator const G = Ver.ParentPkg().Group();
147 Ver->SourcePkgName = G->Name;
148 Ver->SourceVerStr = Ver->VerStr;
149 if (Section.Find("Source",Start,Stop) == true)
150 {
151 const char * const Space = (const char * const) memchr(Start, ' ', Stop - Start);
152 pkgCache::VerIterator V;
153
154 if (Space != NULL)
155 {
156 Stop = Space;
157 const char * const Open = (const char * const) memchr(Space, '(', Stop - Space);
158 if (likely(Open != NULL))
159 {
160 const char * const Close = (const char * const) memchr(Open, ')', Stop - Open);
161 if (likely(Close != NULL))
162 {
163 std::string const version(Open + 1, (Close - Open) - 1);
164 if (version != Ver.VerStr())
165 {
30b45652 166 map_stringitem_t const idx = StoreString(pkgCacheGenerator::VERSIONNUMBER, version);
a221efc3
DK
167 Ver->SourceVerStr = idx;
168 }
169 }
170 }
171 }
172
173 std::string const pkgname(Start, Stop - Start);
174 if (pkgname != G.Name())
175 {
176 for (pkgCache::PkgIterator P = G.PackageList(); P.end() == false; P = G.NextPkg(P))
177 {
178 for (V = P.VersionList(); V.end() == false; ++V)
179 {
180 if (pkgname == V.SourcePkgName())
181 {
182 Ver->SourcePkgName = V->SourcePkgName;
183 break;
184 }
185 }
186 if (V.end() == false)
187 break;
188 }
189 if (V.end() == true)
190 {
191 map_stringitem_t const idx = StoreString(pkgCacheGenerator::PKGNAME, pkgname);
192 Ver->SourcePkgName = idx;
193 }
194 }
195 }
78a5476f 196
e33dbfe5 197 Ver->MultiArch = ParseMultiArch(true);
0149949b 198 // Archive Size
e2c66de5 199 Ver->Size = Section.FindULL("Size");
0149949b 200 // Unpacked Size (in K)
e2c66de5 201 Ver->InstalledSize = Section.FindULL("Installed-Size");
0149949b
AL
202 Ver->InstalledSize *= 1024;
203
204 // Priority
0149949b 205 if (Section.Find("Priority",Start,Stop) == true)
78a5476f 206 {
b2e465d6 207 if (GrabWord(string(Start,Stop-Start),PrioList,Ver->Priority) == false)
421c8d10 208 Ver->Priority = pkgCache::State::Extra;
0149949b 209 }
dcb79bae 210
8efa2a3b 211 if (ParseDepends(Ver,"Pre-Depends",pkgCache::Dep::PreDepends) == false)
dcb79bae 212 return false;
fd23676e 213 if (ParseDepends(Ver,"Depends",pkgCache::Dep::Depends) == false)
dcb79bae 214 return false;
6c139d6e 215 if (ParseDepends(Ver,"Conflicts",pkgCache::Dep::Conflicts) == false)
dcb79bae 216 return false;
308c7d30
IJ
217 if (ParseDepends(Ver,"Breaks",pkgCache::Dep::DpkgBreaks) == false)
218 return false;
fd23676e
DK
219 if (ParseDepends(Ver,"Recommends",pkgCache::Dep::Recommends) == false)
220 return false;
221 if (ParseDepends(Ver,"Suggests",pkgCache::Dep::Suggests) == false)
222 return false;
f55ece0e 223 if (ParseDepends(Ver,"Replaces",pkgCache::Dep::Replaces) == false)
dcb79bae 224 return false;
f8ae7e8b 225 if (ParseDepends(Ver,"Enhances",pkgCache::Dep::Enhances) == false)
226 return false;
b2e465d6
AL
227 // Obsolete.
228 if (ParseDepends(Ver,"Optional",pkgCache::Dep::Suggests) == false)
229 return false;
230
dcb79bae
AL
231 if (ParseProvides(Ver) == false)
232 return false;
0149949b 233
f55a958f
AL
234 return true;
235}
236 /*}}}*/
a52f938b
OS
237// ListParser::Description - Return the description string /*{{{*/
238// ---------------------------------------------------------------------
239/* This is to return the string describing the package in debian
240 form. If this returns the blank string then the entry is assumed to
241 only describe package properties */
ffe3c68e 242string debListParser::Description(std::string const &lang)
a52f938b 243{
45df0ad2 244 if (lang.empty())
a52f938b
OS
245 return Section.FindS("Description");
246 else
45df0ad2 247 return Section.FindS(string("Description-").append(lang).c_str());
a52f938b 248}
ffe3c68e
DK
249 /*}}}*/
250// ListParser::AvailableDescriptionLanguages /*{{{*/
251std::vector<std::string> debListParser::AvailableDescriptionLanguages()
a52f938b 252{
ffe3c68e
DK
253 std::vector<std::string> const understood = APT::Configuration::getLanguages();
254 std::vector<std::string> avail;
255 if (Section.Exists("Description") == true)
256 avail.push_back("");
257 for (std::vector<std::string>::const_iterator lang = understood.begin(); lang != understood.end(); ++lang)
258 {
259 std::string const tagname = "Description-" + *lang;
260 if (Section.Exists(tagname.c_str()) == true)
261 avail.push_back(*lang);
262 }
263 return avail;
a52f938b 264}
ffe3c68e
DK
265 /*}}}*/
266// ListParser::Description_md5 - Return the description_md5 MD5SumValue /*{{{*/
a52f938b 267// ---------------------------------------------------------------------
770c32ec
MV
268/* This is to return the md5 string to allow the check if it is the right
269 description. If no Description-md5 is found in the section it will be
270 calculated.
271 */
a52f938b
OS
272MD5SumValue debListParser::Description_md5()
273{
fc691496
DK
274 string const value = Section.FindS("Description-md5");
275 if (value.empty() == true)
770c32ec 276 {
ffe3c68e 277 std::string const desc = Description("") + "\n";
99359751
DK
278 if (desc == "\n")
279 return MD5SumValue();
280
a52f938b 281 MD5Summation md5;
99359751 282 md5.Add(desc.c_str());
a52f938b 283 return md5.Result();
fc691496
DK
284 }
285 else if (likely(value.size() == 32))
286 {
287 if (likely(value.find_first_not_of("0123456789abcdefABCDEF") == string::npos))
288 return MD5SumValue(value);
289 _error->Error("Malformed Description-md5 line; includes invalid character '%s'", value.c_str());
290 return MD5SumValue();
291 }
292 _error->Error("Malformed Description-md5 line; doesn't have the required length (32 != %d) '%s'", (int)value.size(), value.c_str());
293 return MD5SumValue();
a52f938b
OS
294}
295 /*}}}*/
f55a958f
AL
296// ListParser::UsePackage - Update a package structure /*{{{*/
297// ---------------------------------------------------------------------
298/* This is called to update the package with any new information
299 that might be found in the section */
32b9a14c
DK
300bool debListParser::UsePackage(pkgCache::PkgIterator &Pkg,
301 pkgCache::VerIterator &Ver)
f55a958f 302{
5dd4c8b8 303 string const static myArch = _config->Find("APT::Architecture");
e5a91f7e
DK
304 // Possible values are: "all", "native", "installed" and "none"
305 // The "installed" mode is handled by ParseStatus(), See #544481 and friends.
306 string const static essential = _config->Find("pkgCacheGen::Essential", "all");
307 if (essential == "all" ||
308 (essential == "native" && Pkg->Arch != 0 && myArch == Pkg.Arch()))
5dd4c8b8
DK
309 if (Section.FindFlag("Essential",Pkg->Flags,pkgCache::Flag::Essential) == false)
310 return false;
138d4b3d 311 if (Section.FindFlag("Important",Pkg->Flags,pkgCache::Flag::Important) == false)
f55a958f 312 return false;
138d4b3d
AL
313
314 if (strcmp(Pkg.Name(),"apt") == 0)
a552f37e
DK
315 {
316 if ((essential == "native" && Pkg->Arch != 0 && myArch == Pkg.Arch()) ||
317 essential == "all")
318 Pkg->Flags |= pkgCache::Flag::Essential | pkgCache::Flag::Important;
319 else
320 Pkg->Flags |= pkgCache::Flag::Important;
321 }
322
f55a958f
AL
323 if (ParseStatus(Pkg,Ver) == false)
324 return false;
325 return true;
326}
327 /*}}}*/
204fbdcc
AL
328// ListParser::VersionHash - Compute a unique hash for this version /*{{{*/
329// ---------------------------------------------------------------------
330/* */
331unsigned short debListParser::VersionHash()
332{
333 const char *Sections[] ={"Installed-Size",
334 "Depends",
335 "Pre-Depends",
f78439bf
AL
336// "Suggests",
337// "Recommends",
204fbdcc 338 "Conflicts",
308c7d30 339 "Breaks",
204fbdcc
AL
340 "Replaces",0};
341 unsigned long Result = INIT_FCS;
418a471f 342 char S[1024];
d9682cf8 343 for (const char * const *I = Sections; *I != 0; ++I)
204fbdcc
AL
344 {
345 const char *Start;
346 const char *End;
347 if (Section.Find(*I,Start,End) == false || End - Start >= (signed)sizeof(S))
348 continue;
349
350 /* Strip out any spaces from the text, this undoes dpkgs reformatting
421c8d10
AL
351 of certain fields. dpkg also has the rather interesting notion of
352 reformatting depends operators < -> <= */
470a5c38 353 char *J = S;
d9682cf8 354 for (; Start != End; ++Start)
421c8d10 355 {
d9682cf8
DK
356 if (isspace(*Start) != 0)
357 continue;
358 *J++ = tolower_ascii(*Start);
359
360 if ((*Start == '<' || *Start == '>') && Start[1] != *Start && Start[1] != '=')
470a5c38 361 *J++ = '=';
421c8d10 362 }
418a471f 363
470a5c38 364 Result = AddCRC16(Result,S,J - S);
204fbdcc
AL
365 }
366
367 return Result;
368}
369 /*}}}*/
1c73b0fc 370// StatusListParser::ParseStatus - Parse the status field /*{{{*/
f55a958f
AL
371// ---------------------------------------------------------------------
372/* Status lines are of the form,
373 Status: want flag status
374 want = unknown, install, hold, deinstall, purge
9227645d
GJ
375 flag = ok, reinstreq
376 status = not-installed, config-files, half-installed, unpacked,
377 half-configured, triggers-awaited, triggers-pending, installed
f55a958f 378 */
b830f576
DK
379bool debListParser::ParseStatus(pkgCache::PkgIterator &,
380 pkgCache::VerIterator &)
1c73b0fc
JAK
381{
382 return true;
383}
384bool debStatusListParser::ParseStatus(pkgCache::PkgIterator &Pkg,
385 pkgCache::VerIterator &Ver)
f55a958f
AL
386{
387 const char *Start;
388 const char *Stop;
389 if (Section.Find("Status",Start,Stop) == false)
390 return true;
6bc703c2
DK
391
392 // UsePackage() is responsible for setting the flag in the default case
393 bool const static essential = _config->Find("pkgCacheGen::Essential", "") == "installed";
394 if (essential == true &&
395 Section.FindFlag("Essential",Pkg->Flags,pkgCache::Flag::Essential) == false)
396 return false;
397
f55a958f
AL
398 // Isolate the first word
399 const char *I = Start;
400 for(; I < Stop && *I != ' '; I++);
401 if (I >= Stop || *I != ' ')
402 return _error->Error("Malformed Status line");
403
404 // Process the want field
6c139d6e
AL
405 WordList WantList[] = {{"unknown",pkgCache::State::Unknown},
406 {"install",pkgCache::State::Install},
407 {"hold",pkgCache::State::Hold},
408 {"deinstall",pkgCache::State::DeInstall},
b2e465d6 409 {"purge",pkgCache::State::Purge},
67c3067f 410 {NULL, 0}};
b2e465d6 411 if (GrabWord(string(Start,I-Start),WantList,Pkg->SelectedState) == false)
f55a958f
AL
412 return _error->Error("Malformed 1st word in the Status line");
413
414 // Isloate the next word
415 I++;
416 Start = I;
417 for(; I < Stop && *I != ' '; I++);
418 if (I >= Stop || *I != ' ')
419 return _error->Error("Malformed status line, no 2nd word");
420
421 // Process the flag field
6c139d6e
AL
422 WordList FlagList[] = {{"ok",pkgCache::State::Ok},
423 {"reinstreq",pkgCache::State::ReInstReq},
424 {"hold",pkgCache::State::HoldInst},
b2e465d6 425 {"hold-reinstreq",pkgCache::State::HoldReInstReq},
67c3067f 426 {NULL, 0}};
b2e465d6 427 if (GrabWord(string(Start,I-Start),FlagList,Pkg->InstState) == false)
f55a958f
AL
428 return _error->Error("Malformed 2nd word in the Status line");
429
430 // Isloate the last word
431 I++;
432 Start = I;
433 for(; I < Stop && *I != ' '; I++);
434 if (I != Stop)
435 return _error->Error("Malformed Status line, no 3rd word");
436
437 // Process the flag field
6c139d6e 438 WordList StatusList[] = {{"not-installed",pkgCache::State::NotInstalled},
9227645d
GJ
439 {"config-files",pkgCache::State::ConfigFiles},
440 {"half-installed",pkgCache::State::HalfInstalled},
6c139d6e
AL
441 {"unpacked",pkgCache::State::UnPacked},
442 {"half-configured",pkgCache::State::HalfConfigured},
9d06bc80
MV
443 {"triggers-awaited",pkgCache::State::TriggersAwaited},
444 {"triggers-pending",pkgCache::State::TriggersPending},
9227645d 445 {"installed",pkgCache::State::Installed},
67c3067f 446 {NULL, 0}};
b2e465d6 447 if (GrabWord(string(Start,I-Start),StatusList,Pkg->CurrentState) == false)
f55a958f
AL
448 return _error->Error("Malformed 3rd word in the Status line");
449
450 /* A Status line marks the package as indicating the current
451 version as well. Only if it is actually installed.. Otherwise
452 the interesting dpkg handling of the status file creates bogus
453 entries. */
6c139d6e
AL
454 if (!(Pkg->CurrentState == pkgCache::State::NotInstalled ||
455 Pkg->CurrentState == pkgCache::State::ConfigFiles))
f55a958f
AL
456 {
457 if (Ver.end() == true)
458 _error->Warning("Encountered status field in a non-version description");
459 else
460 Pkg->CurrentVer = Ver.Index();
461 }
462
dcb79bae
AL
463 return true;
464}
a1826878 465
b2e465d6
AL
466const char *debListParser::ConvertRelation(const char *I,unsigned int &Op)
467{
468 // Determine the operator
469 switch (*I)
470 {
471 case '<':
472 I++;
473 if (*I == '=')
474 {
475 I++;
476 Op = pkgCache::Dep::LessEq;
477 break;
478 }
479
480 if (*I == '<')
481 {
482 I++;
483 Op = pkgCache::Dep::Less;
484 break;
485 }
486
487 // < is the same as <= and << is really Cs < for some reason
488 Op = pkgCache::Dep::LessEq;
489 break;
490
491 case '>':
492 I++;
493 if (*I == '=')
494 {
495 I++;
496 Op = pkgCache::Dep::GreaterEq;
497 break;
498 }
499
500 if (*I == '>')
501 {
502 I++;
503 Op = pkgCache::Dep::Greater;
504 break;
505 }
506
507 // > is the same as >= and >> is really Cs > for some reason
508 Op = pkgCache::Dep::GreaterEq;
509 break;
510
511 case '=':
512 Op = pkgCache::Dep::Equals;
513 I++;
514 break;
515
516 // HACK around bad package definitions
517 default:
518 Op = pkgCache::Dep::Equals;
519 break;
520 }
521 return I;
522}
a1826878
AL
523 /*}}}*/
524// ListParser::ParseDepends - Parse a dependency element /*{{{*/
525// ---------------------------------------------------------------------
526/* This parses the dependency elements out of a standard string in place,
527 bit by bit. */
565ded7b
JS
528const char *debListParser::ParseDepends(const char *Start,const char *Stop,
529 std::string &Package,std::string &Ver,unsigned int &Op)
530 { return ParseDepends(Start, Stop, Package, Ver, Op, false, true, false); }
531const char *debListParser::ParseDepends(const char *Start,const char *Stop,
532 std::string &Package,std::string &Ver,unsigned int &Op,
533 bool const &ParseArchFlags)
534 { return ParseDepends(Start, Stop, Package, Ver, Op, ParseArchFlags, true, false); }
535const char *debListParser::ParseDepends(const char *Start,const char *Stop,
536 std::string &Package,std::string &Ver,unsigned int &Op,
537 bool const &ParseArchFlags, bool const &StripMultiArch)
538 { return ParseDepends(Start, Stop, Package, Ver, Op, ParseArchFlags, StripMultiArch, false); }
dcb79bae
AL
539const char *debListParser::ParseDepends(const char *Start,const char *Stop,
540 string &Package,string &Ver,
41c81fd8 541 unsigned int &Op, bool const &ParseArchFlags,
565ded7b
JS
542 bool const &StripMultiArch,
543 bool const &ParseRestrictionsList)
dcb79bae
AL
544{
545 // Strip off leading space
565ded7b 546 for (;Start != Stop && isspace(*Start) != 0; ++Start);
dcb79bae
AL
547
548 // Parse off the package name
549 const char *I = Start;
550 for (;I != Stop && isspace(*I) == 0 && *I != '(' && *I != ')' &&
565ded7b
JS
551 *I != ',' && *I != '|' && *I != '[' && *I != ']' &&
552 *I != '<' && *I != '>'; ++I);
dcb79bae
AL
553
554 // Malformed, no '('
555 if (I != Stop && *I == ')')
556 return 0;
557
558 if (I == Start)
559 return 0;
560
561 // Stash the package name
562 Package.assign(Start,I - Start);
41c81fd8
DK
563
564 // We don't want to confuse library users which can't handle MultiArch
550f6493 565 string const arch = _config->Find("APT::Architecture");
41c81fd8
DK
566 if (StripMultiArch == true) {
567 size_t const found = Package.rfind(':');
550f6493
DK
568 if (found != string::npos &&
569 (strcmp(Package.c_str() + found, ":any") == 0 ||
570 strcmp(Package.c_str() + found, ":native") == 0 ||
571 strcmp(Package.c_str() + found + 1, arch.c_str()) == 0))
41c81fd8
DK
572 Package = Package.substr(0,found);
573 }
574
dcb79bae
AL
575 // Skip white space to the '('
576 for (;I != Stop && isspace(*I) != 0 ; I++);
577
578 // Parse a version
579 if (I != Stop && *I == '(')
580 {
581 // Skip the '('
582 for (I++; I != Stop && isspace(*I) != 0 ; I++);
583 if (I + 3 >= Stop)
584 return 0;
b2e465d6 585 I = ConvertRelation(I,Op);
dcb79bae
AL
586
587 // Skip whitespace
588 for (;I != Stop && isspace(*I) != 0; I++);
589 Start = I;
404528bd
DK
590 I = (const char*) memchr(I, ')', Stop - I);
591 if (I == NULL || Start == I)
592 return 0;
dcb79bae 593
02f000a9
AL
594 // Skip trailing whitespace
595 const char *End = I;
596 for (; End > Start && isspace(End[-1]); End--);
597
171c75f1 598 Ver.assign(Start,End-Start);
dcb79bae
AL
599 I++;
600 }
601 else
602 {
171c75f1 603 Ver.clear();
6c139d6e 604 Op = pkgCache::Dep::NoOp;
dcb79bae
AL
605 }
606
607 // Skip whitespace
608 for (;I != Stop && isspace(*I) != 0; I++);
b2e465d6
AL
609
610 if (ParseArchFlags == true)
611 {
424ff669 612 APT::CacheFilter::PackageArchitectureMatchesSpecification matchesArch(arch, false);
9e10ad8a 613
b2e465d6
AL
614 // Parse an architecture
615 if (I != Stop && *I == '[')
616 {
cd9694bf 617 ++I;
b2e465d6 618 // malformed
cd9694bf
DK
619 if (unlikely(I == Stop))
620 return 0;
621
622 const char *End = I;
623 bool Found = false;
624 bool NegArch = false;
625 while (I != Stop)
b2e465d6 626 {
cd9694bf
DK
627 // look for whitespace or ending ']'
628 for (;End != Stop && !isspace(*End) && *End != ']'; ++End);
629
630 if (unlikely(End == Stop))
b2e465d6 631 return 0;
9e10ad8a
AL
632
633 if (*I == '!')
cd9694bf 634 {
9e10ad8a 635 NegArch = true;
cd9694bf
DK
636 ++I;
637 }
9e10ad8a 638
424ff669
DK
639 std::string arch(I, End);
640 if (arch.empty() == false && matchesArch(arch.c_str()) == true)
cd9694bf 641 {
424ff669 642 Found = true;
cd9694bf
DK
643 if (I[-1] != '!')
644 NegArch = false;
645 // we found a match, so fast-forward to the end of the wildcards
646 for (; End != Stop && *End != ']'; ++End);
647 }
648
b2e465d6
AL
649 if (*End++ == ']') {
650 I = End;
651 break;
652 }
cd9694bf 653
b2e465d6
AL
654 I = End;
655 for (;I != Stop && isspace(*I) != 0; I++);
cd9694bf 656 }
9e10ad8a 657
cd9694bf 658 if (NegArch == true)
9e10ad8a 659 Found = !Found;
cd9694bf
DK
660
661 if (Found == false)
b2e465d6
AL
662 Package = ""; /* not for this arch */
663 }
cd9694bf 664
b2e465d6
AL
665 // Skip whitespace
666 for (;I != Stop && isspace(*I) != 0; I++);
667 }
668
565ded7b
JS
669 if (ParseRestrictionsList == true)
670 {
ac81c0f9 671 // Parse a restrictions formula which is in disjunctive normal form:
672 // (foo AND bar) OR (blub AND bla)
673
674 std::vector<string> const profiles = APT::Configuration::getBuildProfiles();
675
676 // if the next character is a restriction list, then by default the
677 // dependency does not apply and the conditions have to be checked
678 // if the next character is not a restriction list, then by default the
679 // dependency applies
680 bool applies1 = (*I != '<');
681 while (I != Stop)
565ded7b 682 {
ac81c0f9 683 if (*I != '<')
684 break;
685
565ded7b
JS
686 ++I;
687 // malformed
688 if (unlikely(I == Stop))
689 return 0;
690
565ded7b 691 const char *End = I;
565ded7b 692
ac81c0f9 693 // if of the prior restriction list is already fulfilled, then
694 // we can just skip to the end of the current list
695 if (applies1) {
696 for (;End != Stop && *End != '>'; ++End);
697 I = ++End;
698 // skip whitespace
699 for (;I != Stop && isspace(*I) != 0; I++);
700 } else {
701 bool applies2 = true;
702 // all the conditions inside a restriction list have to be
703 // met so once we find one that is not met, we can skip to
704 // the end of this list
705 while (I != Stop)
565ded7b 706 {
ac81c0f9 707 // look for whitespace or ending '>'
708 // End now points to the character after the current term
709 for (;End != Stop && !isspace(*End) && *End != '>'; ++End);
565ded7b 710
ac81c0f9 711 if (unlikely(End == Stop))
712 return 0;
565ded7b 713
ac81c0f9 714 bool NegRestriction = false;
715 if (*I == '!')
716 {
717 NegRestriction = true;
718 ++I;
719 }
720
721 std::string restriction(I, End);
565ded7b
JS
722
723 if (restriction.empty() == false && profiles.empty() == false &&
ac81c0f9 724 std::find(profiles.begin(), profiles.end(), restriction) != profiles.end())
565ded7b 725 {
ac81c0f9 726 if (NegRestriction) {
727 applies2 = false;
728 // since one of the terms does not apply we don't have to check the others
729 for (; End != Stop && *End != '>'; ++End);
730 }
731 } else {
732 if (!NegRestriction) {
733 applies2 = false;
734 // since one of the terms does not apply we don't have to check the others
735 for (; End != Stop && *End != '>'; ++End);
736 }
737 }
738
739 if (*End++ == '>') {
740 I = End;
741 // skip whitespace
742 for (;I != Stop && isspace(*I) != 0; I++);
743 break;
565ded7b 744 }
565ded7b 745
565ded7b 746 I = End;
ac81c0f9 747 // skip whitespace
748 for (;I != Stop && isspace(*I) != 0; I++);
749 }
750 if (applies2) {
751 applies1 = true;
565ded7b 752 }
565ded7b 753 }
565ded7b
JS
754 }
755
ac81c0f9 756 if (applies1 == false) {
757 Package = ""; //not for this restriction
758 }
565ded7b
JS
759 }
760
dcb79bae 761 if (I != Stop && *I == '|')
6c139d6e 762 Op |= pkgCache::Dep::Or;
dcb79bae
AL
763
764 if (I == Stop || *I == ',' || *I == '|')
765 {
766 if (I != Stop)
767 for (I++; I != Stop && isspace(*I) != 0; I++);
768 return I;
769 }
770
771 return 0;
772}
773 /*}}}*/
774// ListParser::ParseDepends - Parse a dependency list /*{{{*/
775// ---------------------------------------------------------------------
776/* This is the higher level depends parser. It takes a tag and generates
777 a complete depends tree for the given version. */
32b9a14c 778bool debListParser::ParseDepends(pkgCache::VerIterator &Ver,
dcb79bae
AL
779 const char *Tag,unsigned int Type)
780{
781 const char *Start;
782 const char *Stop;
783 if (Section.Find(Tag,Start,Stop) == false)
784 return true;
306eacf6 785
28166356 786 string const pkgArch = Ver.Arch();
dcb79bae 787
8efa2a3b 788 while (1)
dcb79bae 789 {
0f485ee5
TG
790 string Package;
791 string Version;
792 unsigned int Op;
793
565ded7b 794 Start = ParseDepends(Start, Stop, Package, Version, Op, false, false, false);
dcb79bae
AL
795 if (Start == 0)
796 return _error->Error("Problem parsing dependency %s",Tag);
0f485ee5 797 size_t const found = Package.rfind(':');
306eacf6 798
f6ce7ffc 799 if (found == string::npos)
306eacf6 800 {
bb0f6a34 801 if (NewDepends(Ver,Package,pkgArch,Version,Op,Type) == false)
c919ad6e 802 return false;
306eacf6 803 }
f6ce7ffc
DK
804 else if (strcmp(Package.c_str() + found, ":any") == 0)
805 {
806 if (NewDepends(Ver,Package,"any",Version,Op,Type) == false)
807 return false;
808 }
bb0f6a34 809 else
0f485ee5 810 {
0f485ee5 811 // Such dependencies are not supposed to be accepted …
bb0f6a34 812 // … but this is probably the best thing to do anyway
3addaba1
DK
813 if (Package.substr(found + 1) == "native")
814 Package = Package.substr(0, found) + ':' + Ver.Cache()->NativeArch();
815
816 if (NewDepends(Ver, Package, "any", Version, Op | pkgCache::Dep::ArchSpecific, Type) == false)
0f485ee5
TG
817 return false;
818 }
bb0f6a34 819
8efa2a3b
AL
820 if (Start == Stop)
821 break;
dcb79bae
AL
822 }
823 return true;
824}
825 /*}}}*/
826// ListParser::ParseProvides - Parse the provides list /*{{{*/
827// ---------------------------------------------------------------------
828/* */
32b9a14c 829bool debListParser::ParseProvides(pkgCache::VerIterator &Ver)
dcb79bae 830{
3addaba1
DK
831 /* it is unlikely, but while parsing dependencies, we might have already
832 picked up multi-arch implicit provides which we do not want to duplicate here */
833 bool hasProvidesAlready = false;
834 std::string const spzName = Ver.ParentPkg().FullName(false);
835 {
836 for (pkgCache::PrvIterator Prv = Ver.ProvidesList(); Prv.end() == false; ++Prv)
837 {
838 if (Prv.IsMultiArchImplicit() == false || (Prv->Flags & pkgCache::Flag::ArchSpecific) == 0)
839 continue;
840 if (spzName != Prv.OwnerPkg().FullName(false))
841 continue;
842 hasProvidesAlready = true;
843 break;
844 }
845 }
846
384f17b4 847 string const Arch = Ver.Arch();
dcb79bae
AL
848 const char *Start;
849 const char *Stop;
67e0766f 850 if (Section.Find("Provides",Start,Stop) == true)
dcb79bae 851 {
67e0766f
DK
852 string Package;
853 string Version;
67e0766f
DK
854 unsigned int Op;
855
3addaba1 856 do
67e0766f 857 {
3addaba1 858 Start = ParseDepends(Start,Stop,Package,Version,Op, false, false, false);
b209edfa 859 const size_t archfound = Package.rfind(':');
67e0766f
DK
860 if (Start == 0)
861 return _error->Error("Problem parsing Provides line");
40faab46
DK
862 if (Op != pkgCache::Dep::NoOp && Op != pkgCache::Dep::Equals) {
863 _error->Warning("Ignoring Provides line with non-equal DepCompareOp for package %s", Package.c_str());
b209edfa 864 } else if (archfound != string::npos) {
3addaba1
DK
865 std::string spzArch = Package.substr(archfound + 1);
866 if (spzArch != "any")
867 {
868 if (NewProvides(Ver, Package.substr(0, archfound), spzArch, Version, pkgCache::Flag::MultiArchImplicit | pkgCache::Flag::ArchSpecific) == false)
869 return false;
870 }
871 if (NewProvides(Ver, Package, "any", Version, pkgCache::Flag::ArchSpecific) == false)
b209edfa 872 return false;
04340db3 873 } else if ((Ver->MultiArch & pkgCache::Version::Foreign) == pkgCache::Version::Foreign) {
384f17b4
DK
874 if (APT::Configuration::checkArchitecture(Arch))
875 if (NewProvidesAllArch(Ver, Package, Version, 0) == false)
876 return false;
67e0766f 877 } else {
3addaba1
DK
878 if ((Ver->MultiArch & pkgCache::Version::Allowed) == pkgCache::Version::Allowed)
879 {
880 if (NewProvides(Ver, Package + ":any", "any", Version, pkgCache::Flag::MultiArchImplicit) == false)
881 return false;
882 }
8c7af4d4 883 if (NewProvides(Ver, Package, Arch, Version, 0) == false)
67e0766f
DK
884 return false;
885 }
3addaba1
DK
886 if (archfound == std::string::npos)
887 {
888 std::string const spzName = Package + ':' + Ver.ParentPkg().Arch();
889 pkgCache::PkgIterator const spzPkg = Ver.Cache()->FindPkg(spzName, "any");
890 if (spzPkg.end() == false)
891 {
892 if (NewProvides(Ver, spzName, "any", Version, pkgCache::Flag::MultiArchImplicit | pkgCache::Flag::ArchSpecific) == false)
893 return false;
894 }
895 }
896 } while (Start != Stop);
67e0766f 897 }
dcb79bae 898
384f17b4 899 if (APT::Configuration::checkArchitecture(Arch))
4d174dc8 900 {
384f17b4
DK
901 if ((Ver->MultiArch & pkgCache::Version::Allowed) == pkgCache::Version::Allowed)
902 {
903 string const Package = string(Ver.ParentPkg().Name()).append(":").append("any");
3addaba1
DK
904 if (NewProvides(Ver, Package, "any", Ver.VerStr(), pkgCache::Flag::MultiArchImplicit) == false)
905 return false;
384f17b4
DK
906 }
907 else if ((Ver->MultiArch & pkgCache::Version::Foreign) == pkgCache::Version::Foreign)
3addaba1
DK
908 {
909 if (NewProvidesAllArch(Ver, Ver.ParentPkg().Name(), Ver.VerStr(), pkgCache::Flag::MultiArchImplicit) == false)
910 return false;
911 }
dcb79bae 912 }
dcb79bae 913
3addaba1
DK
914 if (hasProvidesAlready == false)
915 {
916 pkgCache::PkgIterator const spzPkg = Ver.Cache()->FindPkg(spzName, "any");
917 if (spzPkg.end() == false)
918 {
919 if (NewProvides(Ver, spzName, "any", Ver.VerStr(), pkgCache::Flag::MultiArchImplicit | pkgCache::Flag::ArchSpecific) == false)
920 return false;
921 }
922 }
60dcec6d
DK
923 return true;
924}
925 /*}}}*/
f55a958f
AL
926// ListParser::GrabWord - Matches a word and returns /*{{{*/
927// ---------------------------------------------------------------------
928/* Looks for a word in a list of words - for ParseStatus */
f77ea39c 929bool debListParser::GrabWord(string Word,const WordList *List,unsigned char &Out)
f55a958f 930{
b2e465d6 931 for (unsigned int C = 0; List[C].Str != 0; C++)
f55a958f
AL
932 {
933 if (strcasecmp(Word.c_str(),List[C].Str) == 0)
934 {
935 Out = List[C].Val;
936 return true;
937 }
938 }
939 return false;
940}
941 /*}}}*/
942// ListParser::Step - Move to the next section in the file /*{{{*/
943// ---------------------------------------------------------------------
1e3f4083 944/* This has to be careful to only process the correct architecture */
f55a958f
AL
945bool debListParser::Step()
946{
dcb79bae 947 iOffset = Tags.Offset();
7f8c0eed 948 return Tags.Step(Section);
f55a958f
AL
949}
950 /*}}}*/
b2e465d6
AL
951// ListParser::GetPrio - Convert the priority from a string /*{{{*/
952// ---------------------------------------------------------------------
953/* */
954unsigned char debListParser::GetPrio(string Str)
955{
956 unsigned char Out;
957 if (GrabWord(Str,PrioList,Out) == false)
958 Out = pkgCache::State::Extra;
959
960 return Out;
961}
962 /*}}}*/
68134bda
DK
963bool debListParser::SameVersion(unsigned short const Hash, /*{{{*/
964 pkgCache::VerIterator const &Ver)
965{
c9443c01 966 if (pkgCacheListParser::SameVersion(Hash, Ver) == false)
68134bda
DK
967 return false;
968 // status file has no (Download)Size, but all others are fair game
969 // status file is parsed last, so the first version we encounter is
970 // probably also the version we have downloaded
971 unsigned long long const Size = Section.FindULL("Size");
972 if (Size != 0 && Size != Ver->Size)
973 return false;
974 // available everywhere, but easier to check here than to include in VersionHash
975 unsigned char MultiArch = ParseMultiArch(false);
976 if (MultiArch != Ver->MultiArch)
977 return false;
978 // for all practical proposes (we can check): same version
979 return true;
980}
981 /*}}}*/
0d29b9d4
MV
982
983debDebFileParser::debDebFileParser(FileFd *File, std::string const &DebFile)
7f8c0eed 984 : debListParser(File), DebFile(DebFile)
0d29b9d4
MV
985{
986}
987
988bool debDebFileParser::UsePackage(pkgCache::PkgIterator &Pkg,
989 pkgCache::VerIterator &Ver)
990{
991 bool res = debListParser::UsePackage(Pkg, Ver);
992 // we use the full file path as a provides so that the file is found
993 // by its name
5465192b 994 if(NewProvides(Ver, DebFile, Pkg.Cache()->NativeArch(), Ver.VerStr(), 0) == false)
0d29b9d4
MV
995 return false;
996 return res;
997}
998
862bafea 999debListParser::~debListParser() {}