1 // -*- mode: cpp; mode: fold -*-
3 // $Id: deblistparser.cc,v 1.29.2.5 2004/01/06 01:43:44 mdz Exp $
4 /* ######################################################################
6 Package Cache Generator - Generator for the cache structure.
8 This builds the cache structure from the abstract package list parser.
10 ##################################################################### */
12 // Include Files /*{{{*/
15 #include <apt-pkg/deblistparser.h>
16 #include <apt-pkg/error.h>
17 #include <apt-pkg/configuration.h>
18 #include <apt-pkg/cachefilter.h>
19 #include <apt-pkg/aptconfiguration.h>
20 #include <apt-pkg/strutl.h>
21 #include <apt-pkg/fileutl.h>
22 #include <apt-pkg/crc-16.h>
23 #include <apt-pkg/md5.h>
24 #include <apt-pkg/mmap.h>
25 #include <apt-pkg/pkgcache.h>
26 #include <apt-pkg/cacheiterators.h>
27 #include <apt-pkg/tagfile.h>
28 #include <apt-pkg/macros.h>
40 static debListParser::WordList PrioList
[] = {
41 {"required",pkgCache::State::Required
},
42 {"important",pkgCache::State::Important
},
43 {"standard",pkgCache::State::Standard
},
44 {"optional",pkgCache::State::Optional
},
45 {"extra",pkgCache::State::Extra
},
48 // ListParser::debListParser - Constructor /*{{{*/
49 // ---------------------------------------------------------------------
50 /* Provide an architecture and only this one and "all" will be accepted
51 in Step(), if no Architecture is given we will accept every arch
52 we would accept in general with checkArchitecture() */
53 debListParser::debListParser(FileFd
*File
, string
const &Arch
) : Tags(File
),
56 this->Arch
= _config
->Find("APT::Architecture");
57 Architectures
= APT::Configuration::getArchitectures();
58 MultiArchEnabled
= Architectures
.size() > 1;
61 // ListParser::Package - Return the package name /*{{{*/
62 // ---------------------------------------------------------------------
63 /* This is to return the name of the package this section describes */
64 string
debListParser::Package() {
65 string
const Result
= Section
.FindS("Package");
66 if(unlikely(Result
.empty() == true))
67 _error
->Error("Encountered a section with no Package: header");
71 // ListParser::Architecture - Return the package arch /*{{{*/
72 // ---------------------------------------------------------------------
73 /* This will return the Architecture of the package this section describes */
74 string
debListParser::Architecture() {
75 return Section
.FindS("Architecture");
78 // ListParser::ArchitectureAll /*{{{*/
79 // ---------------------------------------------------------------------
81 bool debListParser::ArchitectureAll() {
82 return Section
.FindS("Architecture") == "all";
85 // ListParser::Version - Return the version string /*{{{*/
86 // ---------------------------------------------------------------------
87 /* This is to return the string describing the version in debian form,
88 epoch:upstream-release. If this returns the blank string then the
89 entry is assumed to only describe package properties */
90 string
debListParser::Version()
92 return Section
.FindS("Version");
95 unsigned char debListParser::ParseMultiArch(bool const showErrors
) /*{{{*/
98 string
const MultiArch
= Section
.FindS("Multi-Arch");
99 if (MultiArch
.empty() == true || MultiArch
== "no")
100 MA
= pkgCache::Version::None
;
101 else if (MultiArch
== "same") {
102 if (ArchitectureAll() == true)
104 if (showErrors
== true)
105 _error
->Warning("Architecture: all package '%s' can't be Multi-Arch: same",
106 Section
.FindS("Package").c_str());
107 MA
= pkgCache::Version::None
;
110 MA
= pkgCache::Version::Same
;
112 else if (MultiArch
== "foreign")
113 MA
= pkgCache::Version::Foreign
;
114 else if (MultiArch
== "allowed")
115 MA
= pkgCache::Version::Allowed
;
118 if (showErrors
== true)
119 _error
->Warning("Unknown Multi-Arch type '%s' for package '%s'",
120 MultiArch
.c_str(), Section
.FindS("Package").c_str());
121 MA
= pkgCache::Version::None
;
124 if (ArchitectureAll() == true)
125 MA
|= pkgCache::Version::All
;
130 // ListParser::NewVersion - Fill in the version structure /*{{{*/
131 // ---------------------------------------------------------------------
133 bool debListParser::NewVersion(pkgCache::VerIterator
&Ver
)
139 if (Section
.Find("Section",Start
,Stop
) == true)
141 map_stringitem_t
const idx
= StoreString(pkgCacheGenerator::SECTION
, Start
, Stop
- Start
);
144 // Parse the source package name
145 pkgCache::GrpIterator
const G
= Ver
.ParentPkg().Group();
146 Ver
->SourcePkgName
= G
->Name
;
147 Ver
->SourceVerStr
= Ver
->VerStr
;
148 if (Section
.Find("Source",Start
,Stop
) == true)
150 const char * const Space
= (const char * const) memchr(Start
, ' ', Stop
- Start
);
151 pkgCache::VerIterator V
;
156 const char * const Open
= (const char * const) memchr(Space
, '(', Stop
- Space
);
157 if (likely(Open
!= NULL
))
159 const char * const Close
= (const char * const) memchr(Open
, ')', Stop
- Open
);
160 if (likely(Close
!= NULL
))
162 std::string
const version(Open
+ 1, (Close
- Open
) - 1);
163 if (version
!= Ver
.VerStr())
165 map_stringitem_t
const idx
= StoreString(pkgCacheGenerator::VERSIONNUMBER
, version
);
166 Ver
->SourceVerStr
= idx
;
172 std::string
const pkgname(Start
, Stop
- Start
);
173 if (pkgname
!= G
.Name())
175 for (pkgCache::PkgIterator P
= G
.PackageList(); P
.end() == false; P
= G
.NextPkg(P
))
177 for (V
= P
.VersionList(); V
.end() == false; ++V
)
179 if (pkgname
== V
.SourcePkgName())
181 Ver
->SourcePkgName
= V
->SourcePkgName
;
185 if (V
.end() == false)
190 map_stringitem_t
const idx
= StoreString(pkgCacheGenerator::PKGNAME
, pkgname
);
191 Ver
->SourcePkgName
= idx
;
196 Ver
->MultiArch
= ParseMultiArch(true);
198 Ver
->Size
= Section
.FindULL("Size");
199 // Unpacked Size (in K)
200 Ver
->InstalledSize
= Section
.FindULL("Installed-Size");
201 Ver
->InstalledSize
*= 1024;
204 if (Section
.Find("Priority",Start
,Stop
) == true)
206 if (GrabWord(string(Start
,Stop
-Start
),PrioList
,Ver
->Priority
) == false)
207 Ver
->Priority
= pkgCache::State::Extra
;
210 if (ParseDepends(Ver
,"Depends",pkgCache::Dep::Depends
) == false)
212 if (ParseDepends(Ver
,"Pre-Depends",pkgCache::Dep::PreDepends
) == false)
214 if (ParseDepends(Ver
,"Suggests",pkgCache::Dep::Suggests
) == false)
216 if (ParseDepends(Ver
,"Recommends",pkgCache::Dep::Recommends
) == false)
218 if (ParseDepends(Ver
,"Conflicts",pkgCache::Dep::Conflicts
) == false)
220 if (ParseDepends(Ver
,"Breaks",pkgCache::Dep::DpkgBreaks
) == false)
222 if (ParseDepends(Ver
,"Replaces",pkgCache::Dep::Replaces
) == false)
224 if (ParseDepends(Ver
,"Enhances",pkgCache::Dep::Enhances
) == false)
228 if (ParseDepends(Ver
,"Optional",pkgCache::Dep::Suggests
) == false)
231 if (ParseProvides(Ver
) == false)
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 */
242 string
debListParser::Description(std::string
const &lang
)
245 return Section
.FindS("Description");
247 return Section
.FindS(string("Description-").append(lang
).c_str());
250 // ListParser::AvailableDescriptionLanguages /*{{{*/
251 std::vector
<std::string
> debListParser::AvailableDescriptionLanguages()
253 std::vector
<std::string
> const understood
= APT::Configuration::getLanguages();
254 std::vector
<std::string
> avail
;
255 if (Section
.Exists("Description") == true)
257 for (std::vector
<std::string
>::const_iterator lang
= understood
.begin(); lang
!= understood
.end(); ++lang
)
259 std::string
const tagname
= "Description-" + *lang
;
260 if (Section
.Exists(tagname
.c_str()) == true)
261 avail
.push_back(*lang
);
266 // ListParser::Description_md5 - Return the description_md5 MD5SumValue /*{{{*/
267 // ---------------------------------------------------------------------
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
272 MD5SumValue
debListParser::Description_md5()
274 string
const value
= Section
.FindS("Description-md5");
275 if (value
.empty() == true)
277 std::string
const desc
= Description("") + "\n";
279 return MD5SumValue();
282 md5
.Add(desc
.c_str());
285 else if (likely(value
.size() == 32))
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();
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();
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 */
300 bool debListParser::UsePackage(pkgCache::PkgIterator
&Pkg
,
301 pkgCache::VerIterator
&Ver
)
303 string
const static myArch
= _config
->Find("APT::Architecture");
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()))
309 if (Section
.FindFlag("Essential",Pkg
->Flags
,pkgCache::Flag::Essential
) == false)
311 if (Section
.FindFlag("Important",Pkg
->Flags
,pkgCache::Flag::Important
) == false)
314 if (strcmp(Pkg
.Name(),"apt") == 0)
316 if ((essential
== "native" && Pkg
->Arch
!= 0 && myArch
== Pkg
.Arch()) ||
318 Pkg
->Flags
|= pkgCache::Flag::Essential
| pkgCache::Flag::Important
;
320 Pkg
->Flags
|= pkgCache::Flag::Important
;
323 if (ParseStatus(Pkg
,Ver
) == false)
328 // ListParser::VersionHash - Compute a unique hash for this version /*{{{*/
329 // ---------------------------------------------------------------------
331 unsigned short debListParser::VersionHash()
333 const char *Sections
[] ={"Installed-Size",
341 unsigned long Result
= INIT_FCS
;
343 for (const char * const *I
= Sections
; *I
!= 0; ++I
)
347 if (Section
.Find(*I
,Start
,End
) == false || End
- Start
>= (signed)sizeof(S
))
350 /* Strip out any spaces from the text, this undoes dpkgs reformatting
351 of certain fields. dpkg also has the rather interesting notion of
352 reformatting depends operators < -> <= */
354 for (; Start
!= End
; ++Start
)
356 if (isspace(*Start
) != 0)
358 *J
++ = tolower_ascii(*Start
);
360 if ((*Start
== '<' || *Start
== '>') && Start
[1] != *Start
&& Start
[1] != '=')
364 Result
= AddCRC16(Result
,S
,J
- S
);
370 // ListParser::ParseStatus - Parse the status field /*{{{*/
371 // ---------------------------------------------------------------------
372 /* Status lines are of the form,
373 Status: want flag status
374 want = unknown, install, hold, deinstall, purge
376 status = not-installed, config-files, half-installed, unpacked,
377 half-configured, triggers-awaited, triggers-pending, installed
379 bool debListParser::ParseStatus(pkgCache::PkgIterator
&Pkg
,
380 pkgCache::VerIterator
&Ver
)
384 if (Section
.Find("Status",Start
,Stop
) == false)
387 // UsePackage() is responsible for setting the flag in the default case
388 bool const static essential
= _config
->Find("pkgCacheGen::Essential", "") == "installed";
389 if (essential
== true &&
390 Section
.FindFlag("Essential",Pkg
->Flags
,pkgCache::Flag::Essential
) == false)
393 // Isolate the first word
394 const char *I
= Start
;
395 for(; I
< Stop
&& *I
!= ' '; I
++);
396 if (I
>= Stop
|| *I
!= ' ')
397 return _error
->Error("Malformed Status line");
399 // Process the want field
400 WordList WantList
[] = {{"unknown",pkgCache::State::Unknown
},
401 {"install",pkgCache::State::Install
},
402 {"hold",pkgCache::State::Hold
},
403 {"deinstall",pkgCache::State::DeInstall
},
404 {"purge",pkgCache::State::Purge
},
406 if (GrabWord(string(Start
,I
-Start
),WantList
,Pkg
->SelectedState
) == false)
407 return _error
->Error("Malformed 1st word in the Status line");
409 // Isloate the next word
412 for(; I
< Stop
&& *I
!= ' '; I
++);
413 if (I
>= Stop
|| *I
!= ' ')
414 return _error
->Error("Malformed status line, no 2nd word");
416 // Process the flag field
417 WordList FlagList
[] = {{"ok",pkgCache::State::Ok
},
418 {"reinstreq",pkgCache::State::ReInstReq
},
419 {"hold",pkgCache::State::HoldInst
},
420 {"hold-reinstreq",pkgCache::State::HoldReInstReq
},
422 if (GrabWord(string(Start
,I
-Start
),FlagList
,Pkg
->InstState
) == false)
423 return _error
->Error("Malformed 2nd word in the Status line");
425 // Isloate the last word
428 for(; I
< Stop
&& *I
!= ' '; I
++);
430 return _error
->Error("Malformed Status line, no 3rd word");
432 // Process the flag field
433 WordList StatusList
[] = {{"not-installed",pkgCache::State::NotInstalled
},
434 {"config-files",pkgCache::State::ConfigFiles
},
435 {"half-installed",pkgCache::State::HalfInstalled
},
436 {"unpacked",pkgCache::State::UnPacked
},
437 {"half-configured",pkgCache::State::HalfConfigured
},
438 {"triggers-awaited",pkgCache::State::TriggersAwaited
},
439 {"triggers-pending",pkgCache::State::TriggersPending
},
440 {"installed",pkgCache::State::Installed
},
442 if (GrabWord(string(Start
,I
-Start
),StatusList
,Pkg
->CurrentState
) == false)
443 return _error
->Error("Malformed 3rd word in the Status line");
445 /* A Status line marks the package as indicating the current
446 version as well. Only if it is actually installed.. Otherwise
447 the interesting dpkg handling of the status file creates bogus
449 if (!(Pkg
->CurrentState
== pkgCache::State::NotInstalled
||
450 Pkg
->CurrentState
== pkgCache::State::ConfigFiles
))
452 if (Ver
.end() == true)
453 _error
->Warning("Encountered status field in a non-version description");
455 Pkg
->CurrentVer
= Ver
.Index();
461 const char *debListParser::ConvertRelation(const char *I
,unsigned int &Op
)
463 // Determine the operator
471 Op
= pkgCache::Dep::LessEq
;
478 Op
= pkgCache::Dep::Less
;
482 // < is the same as <= and << is really Cs < for some reason
483 Op
= pkgCache::Dep::LessEq
;
491 Op
= pkgCache::Dep::GreaterEq
;
498 Op
= pkgCache::Dep::Greater
;
502 // > is the same as >= and >> is really Cs > for some reason
503 Op
= pkgCache::Dep::GreaterEq
;
507 Op
= pkgCache::Dep::Equals
;
511 // HACK around bad package definitions
513 Op
= pkgCache::Dep::Equals
;
519 // ListParser::ParseDepends - Parse a dependency element /*{{{*/
520 // ---------------------------------------------------------------------
521 /* This parses the dependency elements out of a standard string in place,
523 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
524 std::string
&Package
,std::string
&Ver
,unsigned int &Op
)
525 { return ParseDepends(Start
, Stop
, Package
, Ver
, Op
, false, true, false); }
526 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
527 std::string
&Package
,std::string
&Ver
,unsigned int &Op
,
528 bool const &ParseArchFlags
)
529 { return ParseDepends(Start
, Stop
, Package
, Ver
, Op
, ParseArchFlags
, true, false); }
530 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
531 std::string
&Package
,std::string
&Ver
,unsigned int &Op
,
532 bool const &ParseArchFlags
, bool const &StripMultiArch
)
533 { return ParseDepends(Start
, Stop
, Package
, Ver
, Op
, ParseArchFlags
, StripMultiArch
, false); }
534 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
535 string
&Package
,string
&Ver
,
536 unsigned int &Op
, bool const &ParseArchFlags
,
537 bool const &StripMultiArch
,
538 bool const &ParseRestrictionsList
)
540 // Strip off leading space
541 for (;Start
!= Stop
&& isspace(*Start
) != 0; ++Start
);
543 // Parse off the package name
544 const char *I
= Start
;
545 for (;I
!= Stop
&& isspace(*I
) == 0 && *I
!= '(' && *I
!= ')' &&
546 *I
!= ',' && *I
!= '|' && *I
!= '[' && *I
!= ']' &&
547 *I
!= '<' && *I
!= '>'; ++I
);
550 if (I
!= Stop
&& *I
== ')')
556 // Stash the package name
557 Package
.assign(Start
,I
- Start
);
559 // We don't want to confuse library users which can't handle MultiArch
560 string
const arch
= _config
->Find("APT::Architecture");
561 if (StripMultiArch
== true) {
562 size_t const found
= Package
.rfind(':');
563 if (found
!= string::npos
&&
564 (strcmp(Package
.c_str() + found
, ":any") == 0 ||
565 strcmp(Package
.c_str() + found
, ":native") == 0 ||
566 strcmp(Package
.c_str() + found
+ 1, arch
.c_str()) == 0))
567 Package
= Package
.substr(0,found
);
570 // Skip white space to the '('
571 for (;I
!= Stop
&& isspace(*I
) != 0 ; I
++);
574 if (I
!= Stop
&& *I
== '(')
577 for (I
++; I
!= Stop
&& isspace(*I
) != 0 ; I
++);
580 I
= ConvertRelation(I
,Op
);
583 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
585 I
= (const char*) memchr(I
, ')', Stop
- I
);
586 if (I
== NULL
|| Start
== I
)
589 // Skip trailing whitespace
591 for (; End
> Start
&& isspace(End
[-1]); End
--);
593 Ver
.assign(Start
,End
-Start
);
599 Op
= pkgCache::Dep::NoOp
;
603 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
605 if (ParseArchFlags
== true)
607 APT::CacheFilter::PackageArchitectureMatchesSpecification
matchesArch(arch
, false);
609 // Parse an architecture
610 if (I
!= Stop
&& *I
== '[')
614 if (unlikely(I
== Stop
))
619 bool NegArch
= false;
622 // look for whitespace or ending ']'
623 for (;End
!= Stop
&& !isspace(*End
) && *End
!= ']'; ++End
);
625 if (unlikely(End
== Stop
))
634 std::string
arch(I
, End
);
635 if (arch
.empty() == false && matchesArch(arch
.c_str()) == true)
640 // we found a match, so fast-forward to the end of the wildcards
641 for (; End
!= Stop
&& *End
!= ']'; ++End
);
650 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
657 Package
= ""; /* not for this arch */
661 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
664 if (ParseRestrictionsList
== true)
666 // Parse a restrictions formula which is in disjunctive normal form:
667 // (foo AND bar) OR (blub AND bla)
669 std::vector
<string
> const profiles
= APT::Configuration::getBuildProfiles();
671 // if the next character is a restriction list, then by default the
672 // dependency does not apply and the conditions have to be checked
673 // if the next character is not a restriction list, then by default the
674 // dependency applies
675 bool applies1
= (*I
!= '<');
683 if (unlikely(I
== Stop
))
688 // if of the prior restriction list is already fulfilled, then
689 // we can just skip to the end of the current list
691 for (;End
!= Stop
&& *End
!= '>'; ++End
);
694 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
696 bool applies2
= true;
697 // all the conditions inside a restriction list have to be
698 // met so once we find one that is not met, we can skip to
699 // the end of this list
702 // look for whitespace or ending '>'
703 // End now points to the character after the current term
704 for (;End
!= Stop
&& !isspace(*End
) && *End
!= '>'; ++End
);
706 if (unlikely(End
== Stop
))
709 bool NegRestriction
= false;
712 NegRestriction
= true;
716 std::string
restriction(I
, End
);
718 if (restriction
.empty() == false && profiles
.empty() == false &&
719 std::find(profiles
.begin(), profiles
.end(), restriction
) != profiles
.end())
721 if (NegRestriction
) {
723 // since one of the terms does not apply we don't have to check the others
724 for (; End
!= Stop
&& *End
!= '>'; ++End
);
727 if (!NegRestriction
) {
729 // since one of the terms does not apply we don't have to check the others
730 for (; End
!= Stop
&& *End
!= '>'; ++End
);
737 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
743 for (;I
!= Stop
&& isspace(*I
) != 0; I
++);
751 if (applies1
== false) {
752 Package
= ""; //not for this restriction
756 if (I
!= Stop
&& *I
== '|')
757 Op
|= pkgCache::Dep::Or
;
759 if (I
== Stop
|| *I
== ',' || *I
== '|')
762 for (I
++; I
!= Stop
&& isspace(*I
) != 0; I
++);
769 // ListParser::ParseDepends - Parse a dependency list /*{{{*/
770 // ---------------------------------------------------------------------
771 /* This is the higher level depends parser. It takes a tag and generates
772 a complete depends tree for the given version. */
773 bool debListParser::ParseDepends(pkgCache::VerIterator
&Ver
,
774 const char *Tag
,unsigned int Type
)
778 if (Section
.Find(Tag
,Start
,Stop
) == false)
781 string
const pkgArch
= Ver
.Arch();
789 Start
= ParseDepends(Start
, Stop
, Package
, Version
, Op
, false, false, false);
791 return _error
->Error("Problem parsing dependency %s",Tag
);
792 size_t const found
= Package
.rfind(':');
794 // If negative is unspecific it needs to apply on all architectures
795 if (MultiArchEnabled
== true && found
== string::npos
&&
796 (Type
== pkgCache::Dep::Conflicts
||
797 Type
== pkgCache::Dep::DpkgBreaks
||
798 Type
== pkgCache::Dep::Replaces
))
800 for (std::vector
<std::string
>::const_iterator a
= Architectures
.begin();
801 a
!= Architectures
.end(); ++a
)
802 if (NewDepends(Ver
,Package
,*a
,Version
,Op
,Type
) == false)
804 if (NewDepends(Ver
,Package
,"none",Version
,Op
,Type
) == false)
807 else if (MultiArchEnabled
== true && found
!= string::npos
&&
808 strcmp(Package
.c_str() + found
, ":any") != 0)
810 string Arch
= Package
.substr(found
+1, string::npos
);
811 Package
= Package
.substr(0, found
);
812 // Such dependencies are not supposed to be accepted …
813 // … but this is probably the best thing to do.
814 if (Arch
== "native")
815 Arch
= _config
->Find("APT::Architecture");
816 if (NewDepends(Ver
,Package
,Arch
,Version
,Op
,Type
) == false)
821 if (NewDepends(Ver
,Package
,pkgArch
,Version
,Op
,Type
) == false)
823 if ((Type
== pkgCache::Dep::Conflicts
||
824 Type
== pkgCache::Dep::DpkgBreaks
||
825 Type
== pkgCache::Dep::Replaces
) &&
826 NewDepends(Ver
, Package
,
827 (pkgArch
!= "none") ? "none" : _config
->Find("APT::Architecture"),
828 Version
,Op
,Type
) == false)
837 // ListParser::ParseProvides - Parse the provides list /*{{{*/
838 // ---------------------------------------------------------------------
840 bool debListParser::ParseProvides(pkgCache::VerIterator
&Ver
)
844 if (Section
.Find("Provides",Start
,Stop
) == true)
848 string
const Arch
= Ver
.Arch();
853 Start
= ParseDepends(Start
,Stop
,Package
,Version
,Op
);
855 return _error
->Error("Problem parsing Provides line");
856 if (Op
!= pkgCache::Dep::NoOp
&& Op
!= pkgCache::Dep::Equals
) {
857 _error
->Warning("Ignoring Provides line with non-equal DepCompareOp for package %s", Package
.c_str());
858 } else if ((Ver
->MultiArch
& pkgCache::Version::Foreign
) == pkgCache::Version::Foreign
) {
859 if (NewProvidesAllArch(Ver
, Package
, Version
) == false)
862 if (NewProvides(Ver
, Package
, Arch
, Version
) == false)
871 if ((Ver
->MultiArch
& pkgCache::Version::Allowed
) == pkgCache::Version::Allowed
)
873 string
const Package
= string(Ver
.ParentPkg().Name()).append(":").append("any");
874 return NewProvidesAllArch(Ver
, Package
, Ver
.VerStr());
876 else if ((Ver
->MultiArch
& pkgCache::Version::Foreign
) == pkgCache::Version::Foreign
)
877 return NewProvidesAllArch(Ver
, Ver
.ParentPkg().Name(), Ver
.VerStr());
882 // ListParser::NewProvides - add provides for all architectures /*{{{*/
883 bool debListParser::NewProvidesAllArch(pkgCache::VerIterator
&Ver
, string
const &Package
,
884 string
const &Version
) {
885 for (std::vector
<string
>::const_iterator a
= Architectures
.begin();
886 a
!= Architectures
.end(); ++a
)
888 if (NewProvides(Ver
, Package
, *a
, Version
) == false)
894 // ListParser::GrabWord - Matches a word and returns /*{{{*/
895 // ---------------------------------------------------------------------
896 /* Looks for a word in a list of words - for ParseStatus */
897 bool debListParser::GrabWord(string Word
,WordList
*List
,unsigned char &Out
)
899 for (unsigned int C
= 0; List
[C
].Str
!= 0; C
++)
901 if (strcasecmp(Word
.c_str(),List
[C
].Str
) == 0)
910 // ListParser::Step - Move to the next section in the file /*{{{*/
911 // ---------------------------------------------------------------------
912 /* This has to be careful to only process the correct architecture */
913 bool debListParser::Step()
915 iOffset
= Tags
.Offset();
916 while (Tags
.Step(Section
) == true)
918 /* See if this is the correct Architecture, if it isn't then we
919 drop the whole section. A missing arch tag only happens (in theory)
920 inside the Status file, so that is a positive return */
921 string
const Architecture
= Section
.FindS("Architecture");
923 if (Arch
.empty() == true || Arch
== "any" || MultiArchEnabled
== false)
925 if (APT::Configuration::checkArchitecture(Architecture
) == true)
927 /* parse version stanzas without an architecture only in the status file
928 (and as misfortune bycatch flat-archives) */
929 if ((Arch
.empty() == true || Arch
== "any") && Architecture
.empty() == true)
934 if (Architecture
== Arch
)
937 if (Architecture
== "all" && Arch
== _config
->Find("APT::Architecture"))
941 iOffset
= Tags
.Offset();
946 // ListParser::LoadReleaseInfo - Load the release information /*{{{*/
947 // ---------------------------------------------------------------------
949 bool debListParser::LoadReleaseInfo(pkgCache::PkgFileIterator
&FileI
,
950 FileFd
&File
, string component
)
952 // apt-secure does no longer download individual (per-section) Release
953 // file. to provide Component pinning we use the section name now
954 map_stringitem_t
const storage
= StoreString(pkgCacheGenerator::MIXED
, component
);
955 FileI
->Component
= storage
;
957 pkgTagFile
TagFile(&File
, File
.Size());
958 pkgTagSection Section
;
959 if (_error
->PendingError() == true || TagFile
.Step(Section
) == false)
963 #define APT_INRELEASE(TYPE, TAG, STORE) \
964 data = Section.FindS(TAG); \
965 if (data.empty() == false) \
967 map_stringitem_t const storage = StoreString(pkgCacheGenerator::TYPE, data); \
970 APT_INRELEASE(MIXED
, "Suite", FileI
->Archive
)
971 APT_INRELEASE(MIXED
, "Component", FileI
->Component
)
972 APT_INRELEASE(VERSIONNUMBER
, "Version", FileI
->Version
)
973 APT_INRELEASE(MIXED
, "Origin", FileI
->Origin
)
974 APT_INRELEASE(MIXED
, "Codename", FileI
->Codename
)
975 APT_INRELEASE(MIXED
, "Label", FileI
->Label
)
977 Section
.FindFlag("NotAutomatic", FileI
->Flags
, pkgCache::Flag::NotAutomatic
);
978 Section
.FindFlag("ButAutomaticUpgrades", FileI
->Flags
, pkgCache::Flag::ButAutomaticUpgrades
);
980 return !_error
->PendingError();
983 // ListParser::GetPrio - Convert the priority from a string /*{{{*/
984 // ---------------------------------------------------------------------
986 unsigned char debListParser::GetPrio(string Str
)
989 if (GrabWord(Str
,PrioList
,Out
) == false)
990 Out
= pkgCache::State::Extra
;
995 #if (APT_PKG_MAJOR >= 4 && APT_PKG_MINOR >= 13)
996 bool debListParser::SameVersion(unsigned short const Hash
, /*{{{*/
997 pkgCache::VerIterator
const &Ver
)
999 if (pkgCacheGenerator::ListParser::SameVersion(Hash
, Ver
) == false)
1001 // status file has no (Download)Size, but all others are fair game
1002 // status file is parsed last, so the first version we encounter is
1003 // probably also the version we have downloaded
1004 unsigned long long const Size
= Section
.FindULL("Size");
1005 if (Size
!= 0 && Size
!= Ver
->Size
)
1007 // available everywhere, but easier to check here than to include in VersionHash
1008 unsigned char MultiArch
= ParseMultiArch(false);
1009 if (MultiArch
!= Ver
->MultiArch
)
1011 // for all practical proposes (we can check): same version
1018 debDebFileParser::debDebFileParser(FileFd
*File
, std::string
const &DebFile
)
1019 : debListParser(File
, ""), DebFile(DebFile
)
1023 bool debDebFileParser::UsePackage(pkgCache::PkgIterator
&Pkg
,
1024 pkgCache::VerIterator
&Ver
)
1026 bool res
= debListParser::UsePackage(Pkg
, Ver
);
1027 // we use the full file path as a provides so that the file is found
1029 if(NewProvidesAllArch(Ver
, DebFile
, Ver
.VerStr()) == false)
1034 debListParser::~debListParser() {}