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/crc-16.h>
22 #include <apt-pkg/md5.h>
23 #include <apt-pkg/pkgcache.h>
24 #include <apt-pkg/cacheiterators.h>
25 #include <apt-pkg/tagfile.h>
26 #include <apt-pkg/macros.h>
37 using APT::StringView
;
39 static const debListParser::WordList PrioList
[] = {
40 {"required",pkgCache::State::Required
},
41 {"important",pkgCache::State::Important
},
42 {"standard",pkgCache::State::Standard
},
43 {"optional",pkgCache::State::Optional
},
44 {"extra",pkgCache::State::Extra
},
47 // ListParser::debListParser - Constructor /*{{{*/
48 // ---------------------------------------------------------------------
49 /* Provide an architecture and only this one and "all" will be accepted
50 in Step(), if no Architecture is given we will accept every arch
51 we would accept in general with checkArchitecture() */
52 debListParser::debListParser(FileFd
*File
) :
53 pkgCacheListParser(), d(NULL
), Tags(File
)
57 // ListParser::Package - Return the package name /*{{{*/
58 // ---------------------------------------------------------------------
59 /* This is to return the name of the package this section describes */
60 string
debListParser::Package() {
61 string Result
= Section
.Find("Package").to_string();
63 // Normalize mixed case package names to lower case, like dpkg does
64 // See Bug#807012 for details
65 std::transform(Result
.begin(), Result
.end(), Result
.begin(), tolower_ascii
);
67 if(unlikely(Result
.empty() == true))
68 _error
->Error("Encountered a section with no Package: header");
72 // ListParser::Architecture - Return the package arch /*{{{*/
73 // ---------------------------------------------------------------------
74 /* This will return the Architecture of the package this section describes */
75 APT::StringView
debListParser::Architecture() {
76 auto const Arch
= Section
.Find("Architecture");
77 return Arch
.empty() ? "none" : Arch
;
80 // ListParser::ArchitectureAll /*{{{*/
81 // ---------------------------------------------------------------------
83 bool debListParser::ArchitectureAll() {
84 return Section
.Find("Architecture") == "all";
87 // ListParser::Version - Return the version string /*{{{*/
88 // ---------------------------------------------------------------------
89 /* This is to return the string describing the version in debian form,
90 epoch:upstream-release. If this returns the blank string then the
91 entry is assumed to only describe package properties */
92 APT::StringView
debListParser::Version()
94 return Section
.Find("Version");
97 unsigned char debListParser::ParseMultiArch(bool const showErrors
) /*{{{*/
100 auto const MultiArch
= Section
.Find("Multi-Arch");
101 if (MultiArch
.empty() == true || MultiArch
== "no")
102 MA
= pkgCache::Version::No
;
103 else if (MultiArch
== "same") {
104 if (ArchitectureAll() == true)
106 if (showErrors
== true)
107 _error
->Warning("Architecture: all package '%s' can't be Multi-Arch: same",
108 Section
.FindS("Package").c_str());
109 MA
= pkgCache::Version::No
;
112 MA
= pkgCache::Version::Same
;
114 else if (MultiArch
== "foreign")
115 MA
= pkgCache::Version::Foreign
;
116 else if (MultiArch
== "allowed")
117 MA
= pkgCache::Version::Allowed
;
120 if (showErrors
== true)
121 _error
->Warning("Unknown Multi-Arch type '%s' for package '%s'",
122 MultiArch
.to_string().c_str(), Section
.FindS("Package").c_str());
123 MA
= pkgCache::Version::No
;
126 if (ArchitectureAll() == true)
127 MA
|= pkgCache::Version::All
;
132 // ListParser::NewVersion - Fill in the version structure /*{{{*/
133 // ---------------------------------------------------------------------
135 bool debListParser::NewVersion(pkgCache::VerIterator
&Ver
)
141 if (Section
.Find("Section",Start
,Stop
) == true)
143 map_stringitem_t
const idx
= StoreString(pkgCacheGenerator::SECTION
, Start
, Stop
- Start
);
146 // Parse the source package name
147 pkgCache::GrpIterator G
= Ver
.ParentPkg().Group();
148 Ver
->SourcePkgName
= G
->Name
;
149 Ver
->SourceVerStr
= Ver
->VerStr
;
150 if (Section
.Find("Source",Start
,Stop
) == true)
152 const char * const Space
= (const char * const) memchr(Start
, ' ', Stop
- Start
);
153 pkgCache::VerIterator V
;
157 const char * const Open
= (const char * const) memchr(Space
, '(', Stop
- Space
);
158 if (likely(Open
!= NULL
))
160 const char * const Close
= (const char * const) memchr(Open
, ')', Stop
- Open
);
161 if (likely(Close
!= NULL
))
163 APT::StringView
const version(Open
+ 1, (Close
- Open
) - 1);
164 if (version
!= Ver
.VerStr())
166 map_stringitem_t
const idx
= StoreString(pkgCacheGenerator::VERSIONNUMBER
, version
);
167 G
= Ver
.ParentPkg().Group();
168 Ver
->SourceVerStr
= idx
;
175 APT::StringView
const pkgname(Start
, Stop
- Start
);
176 if (pkgname
!= G
.Name())
178 for (pkgCache::PkgIterator P
= G
.PackageList(); P
.end() == false; P
= G
.NextPkg(P
))
180 for (V
= P
.VersionList(); V
.end() == false; ++V
)
182 if (pkgname
== V
.SourcePkgName())
184 Ver
->SourcePkgName
= V
->SourcePkgName
;
188 if (V
.end() == false)
193 map_stringitem_t
const idx
= StoreString(pkgCacheGenerator::PKGNAME
, pkgname
);
194 G
= Ver
.ParentPkg().Group();
195 Ver
->SourcePkgName
= idx
;
200 Ver
->MultiArch
= ParseMultiArch(true);
202 Ver
->Size
= Section
.FindULL("Size");
203 // Unpacked Size (in K)
204 Ver
->InstalledSize
= Section
.FindULL("Installed-Size");
205 Ver
->InstalledSize
*= 1024;
208 if (Section
.Find("Priority",Start
,Stop
) == true)
210 if (GrabWord(StringView(Start
,Stop
-Start
),PrioList
,Ver
->Priority
) == false)
211 Ver
->Priority
= pkgCache::State::Extra
;
214 if (ParseDepends(Ver
,"Pre-Depends",pkgCache::Dep::PreDepends
) == false)
216 if (ParseDepends(Ver
,"Depends",pkgCache::Dep::Depends
) == false)
218 if (ParseDepends(Ver
,"Conflicts",pkgCache::Dep::Conflicts
) == false)
220 if (ParseDepends(Ver
,"Breaks",pkgCache::Dep::DpkgBreaks
) == false)
222 if (ParseDepends(Ver
,"Recommends",pkgCache::Dep::Recommends
) == false)
224 if (ParseDepends(Ver
,"Suggests",pkgCache::Dep::Suggests
) == false)
226 if (ParseDepends(Ver
,"Replaces",pkgCache::Dep::Replaces
) == false)
228 if (ParseDepends(Ver
,"Enhances",pkgCache::Dep::Enhances
) == false)
231 if (ParseDepends(Ver
,"Optional",pkgCache::Dep::Suggests
) == false)
234 if (ParseProvides(Ver
) == false)
240 // ListParser::AvailableDescriptionLanguages /*{{{*/
241 std::vector
<std::string
> debListParser::AvailableDescriptionLanguages()
243 std::vector
<std::string
> const understood
= APT::Configuration::getLanguages();
244 std::vector
<std::string
> avail
;
245 static constexpr int prefixLen
= 12;
246 static constexpr int avgLanguageLen
= 5;
249 tagname
.reserve(prefixLen
+ avgLanguageLen
);
250 tagname
.assign("Description-");
251 if (Section
.Exists("Description") == true)
253 for (std::vector
<std::string
>::const_iterator lang
= understood
.begin(); lang
!= understood
.end(); ++lang
)
255 tagname
.resize(prefixLen
);
256 tagname
.append(*lang
);
257 if (Section
.Exists(tagname
) == true)
258 avail
.push_back(*lang
);
263 // ListParser::Description_md5 - Return the description_md5 MD5SumValue /*{{{*/
264 // ---------------------------------------------------------------------
265 /* This is to return the md5 string to allow the check if it is the right
266 description. If no Description-md5 is found in the section it will be
269 MD5SumValue
debListParser::Description_md5()
271 StringView
const value
= Section
.Find("Description-md5");
272 if (value
.empty() == true)
274 StringView
const desc
= Section
.Find("Description");
276 return MD5SumValue();
279 md5
.Add(desc
.data(), desc
.size());
283 else if (likely(value
.size() == 32))
285 MD5SumValue sumvalue
;
286 if (sumvalue
.Set(value
))
289 _error
->Error("Malformed Description-md5 line; includes invalid character '%.*s'", (int)value
.length(), value
.data());
290 return MD5SumValue();
292 _error
->Error("Malformed Description-md5 line; doesn't have the required length (32 != %d) '%.*s'", (int)value
.size(), (int)value
.length(), value
.data());
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 static const StringView Sections
[] ={"Installed-Size",
341 unsigned long Result
= INIT_FCS
;
343 for (StringView I
: Sections
)
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_ascii(*Start
) != 0)
358 *J
++ = tolower_ascii(*Start
);
360 /* Normalize <= to < and >= to >. This is the wrong way around, but
361 * more efficient that the right way. And since we're only hashing
362 * it does not matter which way we normalize. */
363 if ((*Start
== '<' || *Start
== '>') && Start
[1] == '=') {
368 Result
= AddCRC16(Result
,S
,J
- S
);
374 // StatusListParser::ParseStatus - Parse the status field /*{{{*/
375 // ---------------------------------------------------------------------
376 /* Status lines are of the form,
377 Status: want flag status
378 want = unknown, install, hold, deinstall, purge
380 status = not-installed, config-files, half-installed, unpacked,
381 half-configured, triggers-awaited, triggers-pending, installed
383 bool debListParser::ParseStatus(pkgCache::PkgIterator
&,
384 pkgCache::VerIterator
&)
388 bool debStatusListParser::ParseStatus(pkgCache::PkgIterator
&Pkg
,
389 pkgCache::VerIterator
&Ver
)
393 if (Section
.Find("Status",Start
,Stop
) == false)
396 // UsePackage() is responsible for setting the flag in the default case
397 bool const static essential
= _config
->Find("pkgCacheGen::Essential", "") == "installed";
398 if (essential
== true &&
399 Section
.FindFlag("Essential",Pkg
->Flags
,pkgCache::Flag::Essential
) == false)
402 // Isolate the first word
403 const char *I
= Start
;
404 for(; I
< Stop
&& *I
!= ' '; I
++);
405 if (I
>= Stop
|| *I
!= ' ')
406 return _error
->Error("Malformed Status line");
408 // Process the want field
409 WordList WantList
[] = {{"unknown",pkgCache::State::Unknown
},
410 {"install",pkgCache::State::Install
},
411 {"hold",pkgCache::State::Hold
},
412 {"deinstall",pkgCache::State::DeInstall
},
413 {"purge",pkgCache::State::Purge
},
415 if (GrabWord(StringView(Start
,I
-Start
),WantList
,Pkg
->SelectedState
) == false)
416 return _error
->Error("Malformed 1st word in the Status line");
418 // Isloate the next word
421 for(; I
< Stop
&& *I
!= ' '; I
++);
422 if (I
>= Stop
|| *I
!= ' ')
423 return _error
->Error("Malformed status line, no 2nd word");
425 // Process the flag field
426 WordList FlagList
[] = {{"ok",pkgCache::State::Ok
},
427 {"reinstreq",pkgCache::State::ReInstReq
},
428 {"hold",pkgCache::State::HoldInst
},
429 {"hold-reinstreq",pkgCache::State::HoldReInstReq
},
431 if (GrabWord(StringView(Start
,I
-Start
),FlagList
,Pkg
->InstState
) == false)
432 return _error
->Error("Malformed 2nd word in the Status line");
434 // Isloate the last word
437 for(; I
< Stop
&& *I
!= ' '; I
++);
439 return _error
->Error("Malformed Status line, no 3rd word");
441 // Process the flag field
442 WordList StatusList
[] = {{"not-installed",pkgCache::State::NotInstalled
},
443 {"config-files",pkgCache::State::ConfigFiles
},
444 {"half-installed",pkgCache::State::HalfInstalled
},
445 {"unpacked",pkgCache::State::UnPacked
},
446 {"half-configured",pkgCache::State::HalfConfigured
},
447 {"triggers-awaited",pkgCache::State::TriggersAwaited
},
448 {"triggers-pending",pkgCache::State::TriggersPending
},
449 {"installed",pkgCache::State::Installed
},
451 if (GrabWord(StringView(Start
,I
-Start
),StatusList
,Pkg
->CurrentState
) == false)
452 return _error
->Error("Malformed 3rd word in the Status line");
454 /* A Status line marks the package as indicating the current
455 version as well. Only if it is actually installed.. Otherwise
456 the interesting dpkg handling of the status file creates bogus
458 if (!(Pkg
->CurrentState
== pkgCache::State::NotInstalled
||
459 Pkg
->CurrentState
== pkgCache::State::ConfigFiles
))
461 if (Ver
.end() == true)
462 _error
->Warning("Encountered status field in a non-version description");
464 Pkg
->CurrentVer
= Ver
.Index();
470 const char *debListParser::ConvertRelation(const char *I
,unsigned int &Op
)
472 // Determine the operator
480 Op
= pkgCache::Dep::LessEq
;
487 Op
= pkgCache::Dep::Less
;
491 // < is the same as <= and << is really Cs < for some reason
492 Op
= pkgCache::Dep::LessEq
;
500 Op
= pkgCache::Dep::GreaterEq
;
507 Op
= pkgCache::Dep::Greater
;
511 // > is the same as >= and >> is really Cs > for some reason
512 Op
= pkgCache::Dep::GreaterEq
;
516 Op
= pkgCache::Dep::Equals
;
520 // HACK around bad package definitions
522 Op
= pkgCache::Dep::Equals
;
528 // ListParser::ParseDepends - Parse a dependency element /*{{{*/
529 // ---------------------------------------------------------------------
530 /* This parses the dependency elements out of a standard string in place,
532 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
533 std::string
&Package
,std::string
&Ver
,unsigned int &Op
)
534 { return ParseDepends(Start
, Stop
, Package
, Ver
, Op
, false, true, false); }
535 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
536 std::string
&Package
,std::string
&Ver
,unsigned int &Op
,
537 bool const &ParseArchFlags
)
538 { return ParseDepends(Start
, Stop
, Package
, Ver
, Op
, ParseArchFlags
, true, false); }
539 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
540 std::string
&Package
,std::string
&Ver
,unsigned int &Op
,
541 bool const &ParseArchFlags
, bool const &StripMultiArch
)
542 { return ParseDepends(Start
, Stop
, Package
, Ver
, Op
, ParseArchFlags
, StripMultiArch
, false); }
543 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
544 string
&Package
,string
&Ver
,
545 unsigned int &Op
, bool const &ParseArchFlags
,
546 bool const &StripMultiArch
,
547 bool const &ParseRestrictionsList
)
549 StringView PackageView
;
552 auto res
= ParseDepends(Start
, Stop
, PackageView
, VerView
, Op
, (bool)ParseArchFlags
,
553 (bool) StripMultiArch
, (bool) ParseRestrictionsList
);
554 Package
= PackageView
.to_string();
555 Ver
= VerView
.to_string();
559 const char *debListParser::ParseDepends(const char *Start
,const char *Stop
,
560 StringView
&Package
,StringView
&Ver
,
561 unsigned int &Op
, bool ParseArchFlags
,
563 bool ParseRestrictionsList
)
565 // Strip off leading space
566 for (;Start
!= Stop
&& isspace_ascii(*Start
) != 0; ++Start
);
568 // Parse off the package name
569 const char *I
= Start
;
570 for (;I
!= Stop
&& isspace_ascii(*I
) == 0 && *I
!= '(' && *I
!= ')' &&
571 *I
!= ',' && *I
!= '|' && *I
!= '[' && *I
!= ']' &&
572 *I
!= '<' && *I
!= '>'; ++I
);
575 if (I
!= Stop
&& *I
== ')')
581 // Stash the package name
582 Package
= StringView(Start
, I
- Start
);
584 // We don't want to confuse library users which can't handle MultiArch
585 if (StripMultiArch
== true) {
586 string
const arch
= _config
->Find("APT::Architecture");
587 size_t const found
= Package
.rfind(':');
588 if (found
!= StringView::npos
&&
589 (Package
.substr(found
) == ":any" ||
590 Package
.substr(found
) == ":native" ||
591 Package
.substr(found
+1) == arch
))
592 Package
= Package
.substr(0,found
);
595 // Skip white space to the '('
596 for (;I
!= Stop
&& isspace_ascii(*I
) != 0 ; I
++);
599 if (I
!= Stop
&& *I
== '(')
602 for (I
++; I
!= Stop
&& isspace_ascii(*I
) != 0 ; I
++);
605 I
= ConvertRelation(I
,Op
);
608 for (;I
!= Stop
&& isspace_ascii(*I
) != 0; I
++);
610 I
= (const char*) memchr(I
, ')', Stop
- I
);
611 if (I
== NULL
|| Start
== I
)
614 // Skip trailing whitespace
616 for (; End
> Start
&& isspace_ascii(End
[-1]); End
--);
618 Ver
= StringView(Start
,End
-Start
);
624 Op
= pkgCache::Dep::NoOp
;
628 for (;I
!= Stop
&& isspace_ascii(*I
) != 0; I
++);
630 if (unlikely(ParseArchFlags
== true))
632 string
const arch
= _config
->Find("APT::Architecture");
633 APT::CacheFilter::PackageArchitectureMatchesSpecification
matchesArch(arch
, false);
635 // Parse an architecture
636 if (I
!= Stop
&& *I
== '[')
640 if (unlikely(I
== Stop
))
645 bool NegArch
= false;
648 // look for whitespace or ending ']'
649 for (;End
!= Stop
&& !isspace_ascii(*End
) && *End
!= ']'; ++End
);
651 if (unlikely(End
== Stop
))
660 std::string
const arch(I
, End
);
661 if (arch
.empty() == false && matchesArch(arch
.c_str()) == true)
666 // we found a match, so fast-forward to the end of the wildcards
667 for (; End
!= Stop
&& *End
!= ']'; ++End
);
676 for (;I
!= Stop
&& isspace_ascii(*I
) != 0; I
++);
683 Package
= ""; /* not for this arch */
687 for (;I
!= Stop
&& isspace_ascii(*I
) != 0; I
++);
690 if (unlikely(ParseRestrictionsList
== true))
692 // Parse a restrictions formula which is in disjunctive normal form:
693 // (foo AND bar) OR (blub AND bla)
695 std::vector
<string
> const profiles
= APT::Configuration::getBuildProfiles();
697 // if the next character is a restriction list, then by default the
698 // dependency does not apply and the conditions have to be checked
699 // if the next character is not a restriction list, then by default the
700 // dependency applies
701 bool applies1
= (*I
!= '<');
709 if (unlikely(I
== Stop
))
714 // if of the prior restriction list is already fulfilled, then
715 // we can just skip to the end of the current list
717 for (;End
!= Stop
&& *End
!= '>'; ++End
);
720 for (;I
!= Stop
&& isspace_ascii(*I
) != 0; I
++);
722 bool applies2
= true;
723 // all the conditions inside a restriction list have to be
724 // met so once we find one that is not met, we can skip to
725 // the end of this list
728 // look for whitespace or ending '>'
729 // End now points to the character after the current term
730 for (;End
!= Stop
&& !isspace_ascii(*End
) && *End
!= '>'; ++End
);
732 if (unlikely(End
== Stop
))
735 bool NegRestriction
= false;
738 NegRestriction
= true;
742 std::string
const restriction(I
, End
);
743 if (restriction
.empty() == false && profiles
.empty() == false &&
744 std::find(profiles
.begin(), profiles
.end(), restriction
) != profiles
.end())
746 if (NegRestriction
) {
748 // since one of the terms does not apply we don't have to check the others
749 for (; End
!= Stop
&& *End
!= '>'; ++End
);
752 if (!NegRestriction
) {
754 // since one of the terms does not apply we don't have to check the others
755 for (; End
!= Stop
&& *End
!= '>'; ++End
);
762 for (;I
!= Stop
&& isspace_ascii(*I
) != 0; I
++);
768 for (;I
!= Stop
&& isspace_ascii(*I
) != 0; I
++);
776 if (applies1
== false) {
777 Package
= ""; //not for this restriction
781 if (I
!= Stop
&& *I
== '|')
782 Op
|= pkgCache::Dep::Or
;
784 if (I
== Stop
|| *I
== ',' || *I
== '|')
787 for (I
++; I
!= Stop
&& isspace_ascii(*I
) != 0; I
++);
794 // ListParser::ParseDepends - Parse a dependency list /*{{{*/
795 // ---------------------------------------------------------------------
796 /* This is the higher level depends parser. It takes a tag and generates
797 a complete depends tree for the given version. */
798 bool debListParser::ParseDepends(pkgCache::VerIterator
&Ver
,
799 StringView Tag
,unsigned int Type
)
803 if (Section
.Find(Tag
,Start
,Stop
) == false || Start
== Stop
)
806 string
const pkgArch
= Ver
.Arch();
814 Start
= ParseDepends(Start
, Stop
, Package
, Version
, Op
, false, false, false);
816 return _error
->Error("Problem parsing dependency %.*s",(int)Tag
.length(), Tag
.data());
817 size_t const found
= Package
.rfind(':');
819 if (found
== string::npos
)
821 if (NewDepends(Ver
,Package
,pkgArch
,Version
,Op
,Type
) == false)
824 else if (Package
.substr(found
) == ":any")
826 if (NewDepends(Ver
,Package
,"any",Version
,Op
,Type
) == false)
831 // Such dependencies are not supposed to be accepted …
832 // … but this is probably the best thing to do anyway
833 if (Package
.substr(found
+ 1) == "native")
835 std::string
const Pkg
= Package
.substr(0, found
).to_string() + ':' + Ver
.Cache()->NativeArch();
836 if (NewDepends(Ver
, Pkg
, "any", Version
, Op
| pkgCache::Dep::ArchSpecific
, Type
) == false)
839 else if (NewDepends(Ver
, Package
, "any", Version
, Op
| pkgCache::Dep::ArchSpecific
, Type
) == false)
849 // ListParser::ParseProvides - Parse the provides list /*{{{*/
850 // ---------------------------------------------------------------------
852 bool debListParser::ParseProvides(pkgCache::VerIterator
&Ver
)
854 /* it is unlikely, but while parsing dependencies, we might have already
855 picked up multi-arch implicit provides which we do not want to duplicate here */
856 bool hasProvidesAlready
= false;
857 std::string
const spzName
= Ver
.ParentPkg().FullName(false);
859 for (pkgCache::PrvIterator Prv
= Ver
.ProvidesList(); Prv
.end() == false; ++Prv
)
861 if (Prv
.IsMultiArchImplicit() == false || (Prv
->Flags
& pkgCache::Flag::ArchSpecific
) == 0)
863 if (spzName
!= Prv
.OwnerPkg().FullName(false))
865 hasProvidesAlready
= true;
870 string
const Arch
= Ver
.Arch();
873 if (Section
.Find("Provides",Start
,Stop
) == true)
881 Start
= ParseDepends(Start
,Stop
,Package
,Version
,Op
, false, false, false);
882 const size_t archfound
= Package
.rfind(':');
884 return _error
->Error("Problem parsing Provides line");
885 if (unlikely(Op
!= pkgCache::Dep::NoOp
&& Op
!= pkgCache::Dep::Equals
)) {
886 _error
->Warning("Ignoring Provides line with non-equal DepCompareOp for package %s", Package
.to_string().c_str());
887 } else if (archfound
!= string::npos
) {
888 StringView spzArch
= Package
.substr(archfound
+ 1);
889 if (spzArch
!= "any")
891 if (NewProvides(Ver
, Package
.substr(0, archfound
), spzArch
, Version
, pkgCache::Flag::MultiArchImplicit
| pkgCache::Flag::ArchSpecific
) == false)
894 if (NewProvides(Ver
, Package
, "any", Version
, pkgCache::Flag::ArchSpecific
) == false)
896 } else if ((Ver
->MultiArch
& pkgCache::Version::Foreign
) == pkgCache::Version::Foreign
) {
897 if (APT::Configuration::checkArchitecture(Arch
))
899 if (NewProvidesAllArch(Ver
, Package
, Version
, 0) == false)
902 else if (NewProvides(Ver
, Package
, Arch
, Version
, 0) == false)
905 if ((Ver
->MultiArch
& pkgCache::Version::Allowed
) == pkgCache::Version::Allowed
)
907 if (NewProvides(Ver
, Package
.to_string().append(":any"), "any", Version
, pkgCache::Flag::MultiArchImplicit
) == false)
910 if (NewProvides(Ver
, Package
, Arch
, Version
, 0) == false)
913 if (archfound
== std::string::npos
)
915 string spzName
= Package
.to_string();
916 spzName
.push_back(':');
917 spzName
.append(Ver
.ParentPkg().Arch());
918 pkgCache::PkgIterator
const spzPkg
= Ver
.Cache()->FindPkg(spzName
, "any");
919 if (spzPkg
.end() == false)
921 if (NewProvides(Ver
, spzName
, "any", Version
, pkgCache::Flag::MultiArchImplicit
| pkgCache::Flag::ArchSpecific
) == false)
925 } while (Start
!= Stop
);
928 if (APT::Configuration::checkArchitecture(Arch
))
930 if ((Ver
->MultiArch
& pkgCache::Version::Allowed
) == pkgCache::Version::Allowed
)
932 string
const Package
= string(Ver
.ParentPkg().Name()).append(":").append("any");
933 if (NewProvides(Ver
, Package
, "any", Ver
.VerStr(), pkgCache::Flag::MultiArchImplicit
) == false)
936 else if ((Ver
->MultiArch
& pkgCache::Version::Foreign
) == pkgCache::Version::Foreign
)
938 if (NewProvidesAllArch(Ver
, Ver
.ParentPkg().Name(), Ver
.VerStr(), pkgCache::Flag::MultiArchImplicit
) == false)
943 if (hasProvidesAlready
== false)
945 pkgCache::PkgIterator
const spzPkg
= Ver
.Cache()->FindPkg(spzName
, "any");
946 if (spzPkg
.end() == false)
948 if (NewProvides(Ver
, spzName
, "any", Ver
.VerStr(), pkgCache::Flag::MultiArchImplicit
| pkgCache::Flag::ArchSpecific
) == false)
955 // ListParser::GrabWord - Matches a word and returns /*{{{*/
956 // ---------------------------------------------------------------------
957 /* Looks for a word in a list of words - for ParseStatus */
958 bool debListParser::GrabWord(StringView Word
, WordList
const *List
, unsigned char &Out
)
960 for (unsigned int C
= 0; List
[C
].Str
.empty() == false; C
++)
962 if (Word
.length() == List
[C
].Str
.length() &&
963 strncasecmp(Word
.data(), List
[C
].Str
.data(), Word
.length()) == 0)
972 // ListParser::Step - Move to the next section in the file /*{{{*/
973 // ---------------------------------------------------------------------
974 /* This has to be careful to only process the correct architecture */
975 bool debListParser::Step()
977 iOffset
= Tags
.Offset();
978 return Tags
.Step(Section
);
981 // ListParser::GetPrio - Convert the priority from a string /*{{{*/
982 // ---------------------------------------------------------------------
984 unsigned char debListParser::GetPrio(string Str
)
987 if (GrabWord(Str
,PrioList
,Out
) == false)
988 Out
= pkgCache::State::Extra
;
993 bool debListParser::SameVersion(unsigned short const Hash
, /*{{{*/
994 pkgCache::VerIterator
const &Ver
)
996 if (pkgCacheListParser::SameVersion(Hash
, Ver
) == false)
998 // status file has no (Download)Size, but all others are fair game
999 // status file is parsed last, so the first version we encounter is
1000 // probably also the version we have downloaded
1001 unsigned long long const Size
= Section
.FindULL("Size");
1002 if (Size
!= 0 && Ver
->Size
!= 0 && Size
!= Ver
->Size
)
1004 // available everywhere, but easier to check here than to include in VersionHash
1005 unsigned char MultiArch
= ParseMultiArch(false);
1006 if (MultiArch
!= Ver
->MultiArch
)
1008 // for all practical proposes (we can check): same version
1013 debDebFileParser::debDebFileParser(FileFd
*File
, std::string
const &DebFile
)
1014 : debListParser(File
), DebFile(DebFile
)
1018 bool debDebFileParser::UsePackage(pkgCache::PkgIterator
&Pkg
,
1019 pkgCache::VerIterator
&Ver
)
1021 bool res
= debListParser::UsePackage(Pkg
, Ver
);
1022 // we use the full file path as a provides so that the file is found
1024 if(NewProvides(Ver
, DebFile
, Pkg
.Cache()->NativeArch(), Ver
.VerStr(), 0) == false)
1029 debListParser::~debListParser() {}