]>
git.saurik.com Git - apt.git/blob - cmdline/apt-cache.cc
1 // -*- mode: cpp; mode: fold -*-
3 // $Id: apt-cache.cc,v 1.72 2004/04/30 04:34:03 mdz Exp $
4 /* ######################################################################
6 apt-cache - Manages the cache files
8 apt-cache provides some functions fo manipulating the cache files.
9 It uses the command line interface common to all the APT tools.
11 Returns 100 on failure, 0 on success.
13 ##################################################################### */
15 // Include Files /*{{{*/
18 #include <apt-pkg/algorithms.h>
19 #include <apt-pkg/cachefile.h>
20 #include <apt-pkg/cacheset.h>
21 #include <apt-pkg/cmndline.h>
22 #include <apt-pkg/error.h>
23 #include <apt-pkg/fileutl.h>
24 #include <apt-pkg/indexfile.h>
25 #include <apt-pkg/init.h>
26 #include <apt-pkg/metaindex.h>
27 #include <apt-pkg/pkgrecords.h>
28 #include <apt-pkg/pkgsystem.h>
29 #include <apt-pkg/policy.h>
30 #include <apt-pkg/progress.h>
31 #include <apt-pkg/sourcelist.h>
32 #include <apt-pkg/sptr.h>
33 #include <apt-pkg/srcrecords.h>
34 #include <apt-pkg/strutl.h>
35 #include <apt-pkg/tagfile.h>
36 #include <apt-pkg/version.h>
37 #include <apt-pkg/cacheiterators.h>
38 #include <apt-pkg/configuration.h>
39 #include <apt-pkg/depcache.h>
40 #include <apt-pkg/macros.h>
41 #include <apt-pkg/mmap.h>
42 #include <apt-pkg/pkgcache.h>
44 #include <apt-private/private-cacheset.h>
45 #include <apt-private/private-cmndline.h>
67 // LocalitySort - Sort a version list by package file locality /*{{{*/
68 // ---------------------------------------------------------------------
70 static int LocalityCompare(const void *a
, const void *b
)
72 pkgCache::VerFile
*A
= *(pkgCache::VerFile
**)a
;
73 pkgCache::VerFile
*B
= *(pkgCache::VerFile
**)b
;
82 if (A
->File
== B
->File
)
83 return A
->Offset
- B
->Offset
;
84 return A
->File
- B
->File
;
87 static void LocalitySort(pkgCache::VerFile
**begin
,
88 unsigned long Count
,size_t Size
)
90 qsort(begin
,Count
,Size
,LocalityCompare
);
93 static void LocalitySort(pkgCache::DescFile
**begin
,
94 unsigned long Count
,size_t Size
)
96 qsort(begin
,Count
,Size
,LocalityCompare
);
99 // UnMet - Show unmet dependencies /*{{{*/
100 // ---------------------------------------------------------------------
102 static bool ShowUnMet(pkgCache::VerIterator
const &V
, bool const Important
)
105 for (pkgCache::DepIterator D
= V
.DependsList(); D
.end() == false;)
108 pkgCache::DepIterator Start
;
109 pkgCache::DepIterator End
;
112 // Important deps only
113 if (Important
== true)
114 if (End
->Type
!= pkgCache::Dep::PreDepends
&&
115 End
->Type
!= pkgCache::Dep::Depends
)
118 // Skip conflicts and replaces
119 if (End
.IsNegative() == true || End
->Type
== pkgCache::Dep::Replaces
)
122 // Verify the or group
124 pkgCache::DepIterator RealStart
= Start
;
127 // See if this dep is Ok
128 pkgCache::Version
**VList
= Start
.AllTargets();
149 ioprintf(cout
,_("Package %s version %s has an unmet dep:\n"),
150 V
.ParentPkg().FullName(true).c_str(),V
.VerStr());
153 // Print out the dep type
154 cout
<< " " << End
.DepType() << ": ";
160 cout
<< Start
.TargetPkg().FullName(true);
161 if (Start
.TargetVer() != 0)
162 cout
<< " (" << Start
.CompType() << " " << Start
.TargetVer() <<
175 static bool UnMet(CommandLine
&CmdL
)
177 bool const Important
= _config
->FindB("APT::Cache::Important",false);
179 pkgCacheFile CacheFile
;
180 if (unlikely(CacheFile
.GetPkgCache() == NULL
))
183 if (CmdL
.FileSize() <= 1)
185 for (pkgCache::PkgIterator P
= CacheFile
.GetPkgCache()->PkgBegin(); P
.end() == false; ++P
)
186 for (pkgCache::VerIterator V
= P
.VersionList(); V
.end() == false; ++V
)
187 if (ShowUnMet(V
, Important
) == false)
192 CacheSetHelperVirtuals
helper(true, GlobalError::NOTICE
);
193 APT::VersionList verset
= APT::VersionList::FromCommandLine(CacheFile
, CmdL
.FileList
+ 1,
194 APT::CacheSetHelper::CANDIDATE
, helper
);
195 for (APT::VersionList::iterator V
= verset
.begin(); V
!= verset
.end(); ++V
)
196 if (ShowUnMet(V
, Important
) == false)
202 // DumpPackage - Show a dump of a package record /*{{{*/
203 // ---------------------------------------------------------------------
205 static bool DumpPackage(CommandLine
&CmdL
)
207 pkgCacheFile CacheFile
;
208 APT::CacheSetHelper
helper(true, GlobalError::NOTICE
);
209 APT::PackageList pkgset
= APT::PackageList::FromCommandLine(CacheFile
, CmdL
.FileList
+ 1, helper
);
211 for (APT::PackageList::const_iterator Pkg
= pkgset
.begin(); Pkg
!= pkgset
.end(); ++Pkg
)
213 cout
<< "Package: " << Pkg
.FullName(true) << endl
;
214 cout
<< "Versions: " << endl
;
215 for (pkgCache::VerIterator Cur
= Pkg
.VersionList(); Cur
.end() != true; ++Cur
)
217 cout
<< Cur
.VerStr();
218 for (pkgCache::VerFileIterator Vf
= Cur
.FileList(); Vf
.end() == false; ++Vf
)
219 cout
<< " (" << Vf
.File().FileName() << ")";
221 for (pkgCache::DescIterator D
= Cur
.DescriptionList(); D
.end() == false; ++D
)
223 cout
<< " Description Language: " << D
.LanguageCode() << endl
224 << " File: " << D
.FileList().File().FileName() << endl
225 << " MD5: " << D
.md5() << endl
;
232 cout
<< "Reverse Depends: " << endl
;
233 for (pkgCache::DepIterator D
= Pkg
.RevDependsList(); D
.end() != true; ++D
)
235 cout
<< " " << D
.ParentPkg().FullName(true) << ',' << D
.TargetPkg().FullName(true);
237 cout
<< ' ' << DeNull(D
.TargetVer()) << endl
;
242 cout
<< "Dependencies: " << endl
;
243 for (pkgCache::VerIterator Cur
= Pkg
.VersionList(); Cur
.end() != true; ++Cur
)
245 cout
<< Cur
.VerStr() << " - ";
246 for (pkgCache::DepIterator Dep
= Cur
.DependsList(); Dep
.end() != true; ++Dep
)
247 cout
<< Dep
.TargetPkg().FullName(true) << " (" << (int)Dep
->CompareOp
<< " " << DeNull(Dep
.TargetVer()) << ") ";
251 cout
<< "Provides: " << endl
;
252 for (pkgCache::VerIterator Cur
= Pkg
.VersionList(); Cur
.end() != true; ++Cur
)
254 cout
<< Cur
.VerStr() << " - ";
255 for (pkgCache::PrvIterator Prv
= Cur
.ProvidesList(); Prv
.end() != true; ++Prv
)
256 cout
<< Prv
.ParentPkg().FullName(true) << " ";
259 cout
<< "Reverse Provides: " << endl
;
260 for (pkgCache::PrvIterator Prv
= Pkg
.ProvidesList(); Prv
.end() != true; ++Prv
)
261 cout
<< Prv
.OwnerPkg().FullName(true) << " " << Prv
.OwnerVer().VerStr() << endl
;
267 // ShowHashTableStats - Show stats about a hashtable /*{{{*/
268 // ---------------------------------------------------------------------
270 static map_pointer_t
PackageNext(pkgCache::Package
const * const P
) { return P
->NextPackage
; }
271 static map_pointer_t
GroupNext(pkgCache::Group
const * const G
) { return G
->Next
; }
273 static void ShowHashTableStats(std::string Type
,
275 map_pointer_t
*Hashtable
,
277 map_pointer_t(*Next
)(T
const * const))
279 // hashtable stats for the HashTable
280 unsigned long NumBuckets
= Size
;
281 unsigned long UsedBuckets
= 0;
282 unsigned long UnusedBuckets
= 0;
283 unsigned long LongestBucket
= 0;
284 unsigned long ShortestBucket
= NumBuckets
;
285 unsigned long Entries
= 0;
286 for (unsigned int i
=0; i
< NumBuckets
; ++i
)
288 T
*P
= StartP
+ Hashtable
[i
];
289 if(P
== 0 || P
== StartP
)
295 unsigned long ThisBucketSize
= 0;
296 for (; P
!= StartP
; P
= StartP
+ Next(P
))
298 Entries
+= ThisBucketSize
;
299 LongestBucket
= std::max(ThisBucketSize
, LongestBucket
);
300 ShortestBucket
= std::min(ThisBucketSize
, ShortestBucket
);
302 cout
<< "Total buckets in " << Type
<< ": " << NumBuckets
<< std::endl
;
303 cout
<< " Unused: " << UnusedBuckets
<< std::endl
;
304 cout
<< " Used: " << UsedBuckets
<< std::endl
;
305 cout
<< " Average entries: " << Entries
/(double)NumBuckets
<< std::endl
;
306 cout
<< " Longest: " << LongestBucket
<< std::endl
;
307 cout
<< " Shortest: " << ShortestBucket
<< std::endl
;
310 // Stats - Dump some nice statistics /*{{{*/
311 // ---------------------------------------------------------------------
313 static bool Stats(CommandLine
&)
315 pkgCacheFile CacheFile
;
316 pkgCache
*Cache
= CacheFile
.GetPkgCache();
317 if (unlikely(Cache
== NULL
))
320 cout
<< _("Total package names: ") << Cache
->Head().GroupCount
<< " (" <<
321 SizeToStr(Cache
->Head().GroupCount
*Cache
->Head().GroupSz
) << ')' << endl
322 << _("Total package structures: ") << Cache
->Head().PackageCount
<< " (" <<
323 SizeToStr(Cache
->Head().PackageCount
*Cache
->Head().PackageSz
) << ')' << endl
;
330 pkgCache::PkgIterator I
= Cache
->PkgBegin();
331 for (;I
.end() != true; ++I
)
333 if (I
->VersionList
!= 0 && I
->ProvidesList
== 0)
339 if (I
->VersionList
!= 0 && I
->ProvidesList
!= 0)
345 if (I
->VersionList
== 0 && I
->ProvidesList
!= 0)
348 if (I
.ProvidesList()->NextProvides
== 0)
356 if (I
->VersionList
== 0 && I
->ProvidesList
== 0)
362 cout
<< _(" Normal packages: ") << Normal
<< endl
;
363 cout
<< _(" Pure virtual packages: ") << Virtual
<< endl
;
364 cout
<< _(" Single virtual packages: ") << DVirt
<< endl
;
365 cout
<< _(" Mixed virtual packages: ") << NVirt
<< endl
;
366 cout
<< _(" Missing: ") << Missing
<< endl
;
368 cout
<< _("Total distinct versions: ") << Cache
->Head().VersionCount
<< " (" <<
369 SizeToStr(Cache
->Head().VersionCount
*Cache
->Head().VersionSz
) << ')' << endl
;
370 cout
<< _("Total distinct descriptions: ") << Cache
->Head().DescriptionCount
<< " (" <<
371 SizeToStr(Cache
->Head().DescriptionCount
*Cache
->Head().DescriptionSz
) << ')' << endl
;
372 cout
<< _("Total dependencies: ") << Cache
->Head().DependsCount
<< " (" <<
373 SizeToStr(Cache
->Head().DependsCount
*Cache
->Head().DependencySz
) << ')' << endl
;
375 cout
<< _("Total ver/file relations: ") << Cache
->Head().VerFileCount
<< " (" <<
376 SizeToStr(Cache
->Head().VerFileCount
*Cache
->Head().VerFileSz
) << ')' << endl
;
377 cout
<< _("Total Desc/File relations: ") << Cache
->Head().DescFileCount
<< " (" <<
378 SizeToStr(Cache
->Head().DescFileCount
*Cache
->Head().DescFileSz
) << ')' << endl
;
379 cout
<< _("Total Provides mappings: ") << Cache
->Head().ProvidesCount
<< " (" <<
380 SizeToStr(Cache
->Head().ProvidesCount
*Cache
->Head().ProvidesSz
) << ')' << endl
;
383 std::set
<map_stringitem_t
> stritems
;
384 for (pkgCache::GrpIterator G
= Cache
->GrpBegin(); G
.end() == false; ++G
)
385 stritems
.insert(G
->Name
);
386 for (pkgCache::PkgIterator P
= Cache
->PkgBegin(); P
.end() == false; ++P
)
388 stritems
.insert(P
->Arch
);
389 for (pkgCache::VerIterator V
= P
.VersionList(); V
.end() == false; ++V
)
392 stritems
.insert(V
->VerStr
);
394 stritems
.insert(V
->Section
);
395 #if APT_PKG_ABI >= 413
396 stritems
.insert(V
->SourcePkgName
);
397 stritems
.insert(V
->SourceVerStr
);
399 for (pkgCache::DepIterator D
= V
.DependsList(); D
.end() == false; ++D
)
402 stritems
.insert(D
->Version
);
404 for (pkgCache::DescIterator D
= V
.DescriptionList(); D
.end() == false; ++D
)
406 stritems
.insert(D
->md5sum
);
407 stritems
.insert(D
->language_code
);
410 for (pkgCache::PrvIterator Prv
= P
.ProvidesList(); Prv
.end() == false; ++Prv
)
412 if (Prv
->ProvideVersion
!= 0)
413 stritems
.insert(Prv
->ProvideVersion
);
416 for (pkgCache::RlsFileIterator F
= Cache
->RlsFileBegin(); F
!= Cache
->RlsFileEnd(); ++F
)
418 stritems
.insert(F
->FileName
);
419 stritems
.insert(F
->Archive
);
420 stritems
.insert(F
->Codename
);
421 stritems
.insert(F
->Version
);
422 stritems
.insert(F
->Origin
);
423 stritems
.insert(F
->Label
);
424 stritems
.insert(F
->Site
);
426 for (pkgCache::PkgFileIterator F
= Cache
->FileBegin(); F
!= Cache
->FileEnd(); ++F
)
428 stritems
.insert(F
->FileName
);
429 stritems
.insert(F
->Architecture
);
430 stritems
.insert(F
->Component
);
431 stritems
.insert(F
->IndexType
);
433 unsigned long Size
= 0;
434 for (std::set
<map_stringitem_t
>::const_iterator i
= stritems
.begin(); i
!= stritems
.end(); ++i
)
435 Size
+= strlen(Cache
->StrP
+ *i
) + 1;
437 cout
<< _("Total globbed strings: ") << stritems
.size() << " (" << SizeToStr(Size
) << ')' << endl
;
440 unsigned long Slack
= 0;
441 for (int I
= 0; I
!= 7; I
++)
442 Slack
+= Cache
->Head().Pools
[I
].ItemSize
*Cache
->Head().Pools
[I
].Count
;
443 cout
<< _("Total slack space: ") << SizeToStr(Slack
) << endl
;
445 unsigned long Total
= 0;
446 #define APT_CACHESIZE(X,Y) (Cache->Head().X * Cache->Head().Y)
447 Total
= Slack
+ Size
+
448 APT_CACHESIZE(GroupCount
, GroupSz
) +
449 APT_CACHESIZE(PackageCount
, PackageSz
) +
450 APT_CACHESIZE(VersionCount
, VersionSz
) +
451 APT_CACHESIZE(DescriptionCount
, DescriptionSz
) +
452 APT_CACHESIZE(DependsCount
, DependencySz
) +
453 APT_CACHESIZE(ReleaseFileCount
, ReleaseFileSz
) +
454 APT_CACHESIZE(PackageFileCount
, PackageFileSz
) +
455 APT_CACHESIZE(VerFileCount
, VerFileSz
) +
456 APT_CACHESIZE(DescFileCount
, DescFileSz
) +
457 APT_CACHESIZE(ProvidesCount
, ProvidesSz
) +
458 (2 * Cache
->Head().GetHashTableSize() * sizeof(map_id_t
));
459 cout
<< _("Total space accounted for: ") << SizeToStr(Total
) << endl
;
463 ShowHashTableStats
<pkgCache::Package
>("PkgHashTable", Cache
->PkgP
, Cache
->Head().PkgHashTableP(), Cache
->Head().GetHashTableSize(), PackageNext
);
464 ShowHashTableStats
<pkgCache::Group
>("GrpHashTable", Cache
->GrpP
, Cache
->Head().GrpHashTableP(), Cache
->Head().GetHashTableSize(), GroupNext
);
469 // Dump - show everything /*{{{*/
470 // ---------------------------------------------------------------------
471 /* This is worthless except fer debugging things */
472 static bool Dump(CommandLine
&)
474 pkgCacheFile CacheFile
;
475 pkgCache
*Cache
= CacheFile
.GetPkgCache();
476 if (unlikely(Cache
== NULL
))
479 std::cout
<< "Using Versioning System: " << Cache
->VS
->Label
<< std::endl
;
481 for (pkgCache::PkgIterator P
= Cache
->PkgBegin(); P
.end() == false; ++P
)
483 std::cout
<< "Package: " << P
.FullName(true) << std::endl
;
484 for (pkgCache::VerIterator V
= P
.VersionList(); V
.end() == false; ++V
)
486 std::cout
<< " Version: " << V
.VerStr() << std::endl
;
487 std::cout
<< " File: " << V
.FileList().File().FileName() << std::endl
;
488 for (pkgCache::DepIterator D
= V
.DependsList(); D
.end() == false; ++D
)
489 std::cout
<< " Depends: " << D
.TargetPkg().FullName(true) << ' ' <<
490 DeNull(D
.TargetVer()) << std::endl
;
491 for (pkgCache::DescIterator D
= V
.DescriptionList(); D
.end() == false; ++D
)
493 std::cout
<< " Description Language: " << D
.LanguageCode() << std::endl
494 << " File: " << D
.FileList().File().FileName() << std::endl
495 << " MD5: " << D
.md5() << std::endl
;
500 for (pkgCache::PkgFileIterator F
= Cache
->FileBegin(); F
.end() == false; ++F
)
502 std::cout
<< "File: " << F
.FileName() << std::endl
;
503 std::cout
<< " Type: " << F
.IndexType() << std::endl
;
504 std::cout
<< " Size: " << F
->Size
<< std::endl
;
505 std::cout
<< " ID: " << F
->ID
<< std::endl
;
506 std::cout
<< " Flags: " << F
->Flags
<< std::endl
;
507 std::cout
<< " Time: " << TimeRFC1123(F
->mtime
) << std::endl
;
508 std::cout
<< " Archive: " << DeNull(F
.Archive()) << std::endl
;
509 std::cout
<< " Component: " << DeNull(F
.Component()) << std::endl
;
510 std::cout
<< " Version: " << DeNull(F
.Version()) << std::endl
;
511 std::cout
<< " Origin: " << DeNull(F
.Origin()) << std::endl
;
512 std::cout
<< " Site: " << DeNull(F
.Site()) << std::endl
;
513 std::cout
<< " Label: " << DeNull(F
.Label()) << std::endl
;
514 std::cout
<< " Architecture: " << DeNull(F
.Architecture()) << std::endl
;
520 // DumpAvail - Print out the available list /*{{{*/
521 // ---------------------------------------------------------------------
522 /* This is needed to make dpkg --merge happy.. I spent a bit of time to
523 make this run really fast, perhaps I went a little overboard.. */
524 static bool DumpAvail(CommandLine
&)
526 pkgCacheFile CacheFile
;
527 pkgCache
*Cache
= CacheFile
.GetPkgCache();
528 if (unlikely(Cache
== NULL
|| CacheFile
.BuildPolicy() == false))
531 unsigned long Count
= Cache
->HeaderP
->PackageCount
+1;
532 pkgCache::VerFile
**VFList
= new pkgCache::VerFile
*[Count
];
533 memset(VFList
,0,sizeof(*VFList
)*Count
);
535 // Map versions that we want to write out onto the VerList array.
536 for (pkgCache::PkgIterator P
= Cache
->PkgBegin(); P
.end() == false; ++P
)
538 if (P
->VersionList
== 0)
541 /* Find the proper version to use. If the policy says there are no
542 possible selections we return the installed version, if available..
543 This prevents dselect from making it obsolete. */
544 pkgCache::VerIterator V
= CacheFile
.GetPolicy()->GetCandidateVer(P
);
547 if (P
->CurrentVer
== 0)
552 pkgCache::VerFileIterator VF
= V
.FileList();
553 for (; VF
.end() == false ; ++VF
)
554 if ((VF
.File()->Flags
& pkgCache::Flag::NotSource
) == 0)
557 /* Okay, here we have a bit of a problem.. The policy has selected the
558 currently installed package - however it only exists in the
559 status file.. We need to write out something or dselect will mark
560 the package as obsolete! Thus we emit the status file entry, but
561 below we remove the status line to make it valid for the
562 available file. However! We only do this if their do exist *any*
563 non-source versions of the package - that way the dselect obsolete
564 handling works OK. */
565 if (VF
.end() == true)
567 for (pkgCache::VerIterator Cur
= P
.VersionList(); Cur
.end() != true; ++Cur
)
569 for (VF
= Cur
.FileList(); VF
.end() == false; ++VF
)
571 if ((VF
.File()->Flags
& pkgCache::Flag::NotSource
) == 0)
578 if (VF
.end() == false)
586 LocalitySort(VFList
,Count
,sizeof(*VFList
));
588 std::vector
<pkgTagSection::Tag
> RW
;
589 RW
.push_back(pkgTagSection::Tag::Remove("Status"));
590 RW
.push_back(pkgTagSection::Tag::Remove("Config-Version"));
592 stdoutfd
.OpenDescriptor(STDOUT_FILENO
, FileFd::WriteOnly
, false);
594 // Iterate over all the package files and write them out.
595 char *Buffer
= new char[Cache
->HeaderP
->MaxVerFileSize
+10];
596 for (pkgCache::VerFile
**J
= VFList
; *J
!= 0;)
598 pkgCache::PkgFileIterator
File(*Cache
,(*J
)->File
+ Cache
->PkgFileP
);
599 if (File
.IsOk() == false)
601 _error
->Error(_("Package file %s is out of sync."),File
.FileName());
605 FileFd
PkgF(File
.FileName(),FileFd::ReadOnly
, FileFd::Extension
);
606 if (_error
->PendingError() == true)
609 /* Write all of the records from this package file, since we
610 already did locality sorting we can now just seek through the
611 file in read order. We apply 1 more optimization here, since often
612 there will be < 1 byte gaps between records (for the \n) we read that
613 into the next buffer and offset a bit.. */
614 unsigned long Pos
= 0;
617 if ((*J
)->File
+ Cache
->PkgFileP
!= File
)
620 const pkgCache::VerFile
&VF
= **J
;
622 // Read the record and then write it out again.
623 unsigned long Jitter
= VF
.Offset
- Pos
;
626 if (PkgF
.Seek(VF
.Offset
) == false)
631 if (PkgF
.Read(Buffer
,VF
.Size
+ Jitter
) == false)
633 Buffer
[VF
.Size
+ Jitter
] = '\n';
636 if ((File
->Flags
& pkgCache::Flag::NotSource
) == pkgCache::Flag::NotSource
)
639 if (Tags
.Scan(Buffer
+Jitter
,VF
.Size
+1) == false ||
640 Tags
.Write(stdoutfd
, NULL
, RW
) == false ||
641 stdoutfd
.Write("\n", 1) == false)
643 _error
->Error("Internal Error, Unable to parse a package record");
649 if (stdoutfd
.Write(Buffer
+ Jitter
, VF
.Size
+ 1) == false)
653 Pos
= VF
.Offset
+ VF
.Size
;
656 if (_error
->PendingError() == true)
662 return !_error
->PendingError();
665 // ShowDepends - Helper for printing out a dependency tree /*{{{*/
666 static bool ShowDepends(CommandLine
&CmdL
, bool const RevDepends
)
668 pkgCacheFile CacheFile
;
669 pkgCache
*Cache
= CacheFile
.GetPkgCache();
670 if (unlikely(Cache
== NULL
))
673 CacheSetHelperVirtuals
helper(false);
674 APT::VersionList verset
= APT::VersionList::FromCommandLine(CacheFile
, CmdL
.FileList
+ 1, APT::CacheSetHelper::CANDIDATE
, helper
);
675 if (verset
.empty() == true && helper
.virtualPkgs
.empty() == true)
676 return _error
->Error(_("No packages found"));
677 std::vector
<bool> Shown(Cache
->Head().PackageCount
);
679 bool const Recurse
= _config
->FindB("APT::Cache::RecurseDepends", false);
680 bool const Installed
= _config
->FindB("APT::Cache::Installed", false);
681 bool const Important
= _config
->FindB("APT::Cache::Important", false);
682 bool const ShowDepType
= _config
->FindB("APT::Cache::ShowDependencyType", RevDepends
== false);
683 bool const ShowVersion
= _config
->FindB("APT::Cache::ShowVersion", false);
684 bool const ShowPreDepends
= _config
->FindB("APT::Cache::ShowPre-Depends", true);
685 bool const ShowDepends
= _config
->FindB("APT::Cache::ShowDepends", true);
686 bool const ShowRecommends
= _config
->FindB("APT::Cache::ShowRecommends", Important
== false);
687 bool const ShowSuggests
= _config
->FindB("APT::Cache::ShowSuggests", Important
== false);
688 bool const ShowReplaces
= _config
->FindB("APT::Cache::ShowReplaces", Important
== false);
689 bool const ShowConflicts
= _config
->FindB("APT::Cache::ShowConflicts", Important
== false);
690 bool const ShowBreaks
= _config
->FindB("APT::Cache::ShowBreaks", Important
== false);
691 bool const ShowEnhances
= _config
->FindB("APT::Cache::ShowEnhances", Important
== false);
692 bool const ShowOnlyFirstOr
= _config
->FindB("APT::Cache::ShowOnlyFirstOr", false);
694 while (verset
.empty() != true)
696 pkgCache::VerIterator Ver
= *verset
.begin();
697 verset
.erase(verset
.begin());
698 pkgCache::PkgIterator Pkg
= Ver
.ParentPkg();
699 Shown
[Pkg
->ID
] = true;
701 cout
<< Pkg
.FullName(true) << endl
;
703 if (RevDepends
== true)
704 cout
<< "Reverse Depends:" << endl
;
705 for (pkgCache::DepIterator D
= RevDepends
? Pkg
.RevDependsList() : Ver
.DependsList();
706 D
.end() == false; ++D
)
709 case pkgCache::Dep::PreDepends
: if (!ShowPreDepends
) continue; break;
710 case pkgCache::Dep::Depends
: if (!ShowDepends
) continue; break;
711 case pkgCache::Dep::Recommends
: if (!ShowRecommends
) continue; break;
712 case pkgCache::Dep::Suggests
: if (!ShowSuggests
) continue; break;
713 case pkgCache::Dep::Replaces
: if (!ShowReplaces
) continue; break; case pkgCache::Dep::Conflicts
: if (!ShowConflicts
) continue; break;
714 case pkgCache::Dep::DpkgBreaks
: if (!ShowBreaks
) continue; break;
715 case pkgCache::Dep::Enhances
: if (!ShowEnhances
) continue; break;
718 pkgCache::PkgIterator Trg
= RevDepends
? D
.ParentPkg() : D
.TargetPkg();
720 if((Installed
&& Trg
->CurrentVer
!= 0) || !Installed
)
723 if ((D
->CompareOp
& pkgCache::Dep::Or
) == pkgCache::Dep::Or
&& ShowOnlyFirstOr
== false)
729 if (ShowDepType
== true)
730 cout
<< D
.DepType() << ": ";
731 if (Trg
->VersionList
== 0)
732 cout
<< "<" << Trg
.FullName(true) << ">";
734 cout
<< Trg
.FullName(true);
735 if (ShowVersion
== true && D
->Version
!= 0)
736 cout
<< " (" << pkgCache::CompTypeDeb(D
->CompareOp
) << ' ' << D
.TargetVer() << ')';
739 if (Recurse
== true && Shown
[Trg
->ID
] == false)
741 Shown
[Trg
->ID
] = true;
742 verset
.insert(APT::VersionSet::FromPackage(CacheFile
, Trg
, APT::CacheSetHelper::CANDIDATE
, helper
));
747 // Display all solutions
748 SPtrArray
<pkgCache::Version
*> List
= D
.AllTargets();
749 pkgPrioSortList(*Cache
,List
);
750 for (pkgCache::Version
**I
= List
; *I
!= 0; I
++)
752 pkgCache::VerIterator
V(*Cache
,*I
);
753 if (V
!= Cache
->VerP
+ V
.ParentPkg()->VersionList
||
754 V
->ParentPkg
== D
->Package
)
756 cout
<< " " << V
.ParentPkg().FullName(true) << endl
;
758 if (Recurse
== true && Shown
[V
.ParentPkg()->ID
] == false)
760 Shown
[V
.ParentPkg()->ID
] = true;
761 verset
.insert(APT::VersionSet::FromPackage(CacheFile
, V
.ParentPkg(), APT::CacheSetHelper::CANDIDATE
, helper
));
765 if (ShowOnlyFirstOr
== true)
766 while ((D
->CompareOp
& pkgCache::Dep::Or
) == pkgCache::Dep::Or
) ++D
;
770 for (APT::PackageSet::const_iterator Pkg
= helper
.virtualPkgs
.begin();
771 Pkg
!= helper
.virtualPkgs
.end(); ++Pkg
)
772 cout
<< '<' << Pkg
.FullName(true) << '>' << endl
;
777 // Depends - Print out a dependency tree /*{{{*/
778 // ---------------------------------------------------------------------
780 static bool Depends(CommandLine
&CmdL
)
782 return ShowDepends(CmdL
, false);
785 // RDepends - Print out a reverse dependency tree /*{{{*/
786 // ---------------------------------------------------------------------
788 static bool RDepends(CommandLine
&CmdL
)
790 return ShowDepends(CmdL
, true);
793 // xvcg - Generate a graph for xvcg /*{{{*/
794 // ---------------------------------------------------------------------
795 // Code contributed from Junichi Uekawa <dancer@debian.org> on 20 June 2002.
797 static bool XVcg(CommandLine
&CmdL
)
799 pkgCacheFile CacheFile
;
800 pkgCache
*Cache
= CacheFile
.GetPkgCache();
801 if (unlikely(Cache
== NULL
))
804 bool GivenOnly
= _config
->FindB("APT::Cache::GivenOnly",false);
806 /* Normal packages are boxes
807 Pure Provides are triangles
809 rhomb are missing packages*/
810 const char *Shapes
[] = {"ellipse","triangle","box","rhomb"};
812 /* Initialize the list of packages to show.
814 2 = To Show no recurse
815 3 = Emitted no recurse
818 enum States
{None
=0, ToShow
, ToShowNR
, DoneNR
, Done
};
819 enum TheFlags
{ForceNR
=(1<<0)};
820 unsigned char *Show
= new unsigned char[Cache
->Head().PackageCount
];
821 unsigned char *Flags
= new unsigned char[Cache
->Head().PackageCount
];
822 unsigned char *ShapeMap
= new unsigned char[Cache
->Head().PackageCount
];
824 // Show everything if no arguments given
825 if (CmdL
.FileList
[1] == 0)
826 for (unsigned long I
= 0; I
!= Cache
->Head().PackageCount
; I
++)
829 for (unsigned long I
= 0; I
!= Cache
->Head().PackageCount
; I
++)
831 memset(Flags
,0,sizeof(*Flags
)*Cache
->Head().PackageCount
);
834 for (pkgCache::PkgIterator Pkg
= Cache
->PkgBegin(); Pkg
.end() == false; ++Pkg
)
836 if (Pkg
->VersionList
== 0)
839 if (Pkg
->ProvidesList
== 0)
840 ShapeMap
[Pkg
->ID
] = 0;
842 ShapeMap
[Pkg
->ID
] = 1;
847 if (Pkg
->ProvidesList
== 0)
848 ShapeMap
[Pkg
->ID
] = 2;
850 ShapeMap
[Pkg
->ID
] = 3;
854 // Load the list of packages from the command line into the show list
855 APT::CacheSetHelper
helper(true, GlobalError::NOTICE
);
856 std::list
<APT::CacheSetHelper::PkgModifier
> mods
;
857 mods
.push_back(APT::CacheSetHelper::PkgModifier(0, ",", APT::PackageSet::Modifier::POSTFIX
));
858 mods
.push_back(APT::CacheSetHelper::PkgModifier(1, "^", APT::PackageSet::Modifier::POSTFIX
));
859 std::map
<unsigned short, APT::PackageSet
> pkgsets
=
860 APT::PackageSet::GroupedFromCommandLine(CacheFile
, CmdL
.FileList
+ 1, mods
, 0, helper
);
862 for (APT::PackageSet::const_iterator Pkg
= pkgsets
[0].begin();
863 Pkg
!= pkgsets
[0].end(); ++Pkg
)
864 Show
[Pkg
->ID
] = ToShow
;
865 for (APT::PackageSet::const_iterator Pkg
= pkgsets
[1].begin();
866 Pkg
!= pkgsets
[1].end(); ++Pkg
)
868 Show
[Pkg
->ID
] = ToShow
;
869 Flags
[Pkg
->ID
] |= ForceNR
;
873 cout
<< "graph: { title: \"packages\"" << endl
<<
874 "xmax: 700 ymax: 700 x: 30 y: 30" << endl
<<
875 "layout_downfactor: 8" << endl
;
881 for (pkgCache::PkgIterator Pkg
= Cache
->PkgBegin(); Pkg
.end() == false; ++Pkg
)
883 // See we need to show this package
884 if (Show
[Pkg
->ID
] == None
|| Show
[Pkg
->ID
] >= DoneNR
)
887 //printf ("node: { title: \"%s\" label: \"%s\" }\n", Pkg.Name(), Pkg.Name());
890 if (Show
[Pkg
->ID
] == ToShowNR
|| (Flags
[Pkg
->ID
] & ForceNR
) == ForceNR
)
892 // Pure Provides and missing packages have no deps!
893 if (ShapeMap
[Pkg
->ID
] == 0 || ShapeMap
[Pkg
->ID
] == 1)
894 Show
[Pkg
->ID
] = Done
;
896 Show
[Pkg
->ID
] = DoneNR
;
899 Show
[Pkg
->ID
] = Done
;
902 // No deps to map out
903 if (Pkg
->VersionList
== 0 || Show
[Pkg
->ID
] == DoneNR
)
906 pkgCache::VerIterator Ver
= Pkg
.VersionList();
907 for (pkgCache::DepIterator D
= Ver
.DependsList(); D
.end() == false; ++D
)
909 // See if anything can meet this dep
910 // Walk along the actual package providing versions
912 pkgCache::PkgIterator DPkg
= D
.TargetPkg();
913 for (pkgCache::VerIterator I
= DPkg
.VersionList();
914 I
.end() == false && Hit
== false; ++I
)
916 if (Cache
->VS
->CheckDep(I
.VerStr(),D
->CompareOp
,D
.TargetVer()) == true)
920 // Follow all provides
921 for (pkgCache::PrvIterator I
= DPkg
.ProvidesList();
922 I
.end() == false && Hit
== false; ++I
)
924 if (Cache
->VS
->CheckDep(I
.ProvideVersion(),D
->CompareOp
,D
.TargetVer()) == false)
929 // Only graph critical deps
930 if (D
.IsCritical() == true)
932 printf ("edge: { sourcename: \"%s\" targetname: \"%s\" class: 2 ",Pkg
.FullName(true).c_str(), D
.TargetPkg().FullName(true).c_str() );
934 // Colour the node for recursion
935 if (Show
[D
.TargetPkg()->ID
] <= DoneNR
)
937 /* If a conflicts does not meet anything in the database
938 then show the relation but do not recurse */
939 if (Hit
== false && D
.IsNegative() == true)
941 if (Show
[D
.TargetPkg()->ID
] == None
&&
942 Show
[D
.TargetPkg()->ID
] != ToShow
)
943 Show
[D
.TargetPkg()->ID
] = ToShowNR
;
947 if (GivenOnly
== true && Show
[D
.TargetPkg()->ID
] != ToShow
)
948 Show
[D
.TargetPkg()->ID
] = ToShowNR
;
950 Show
[D
.TargetPkg()->ID
] = ToShow
;
957 case pkgCache::Dep::Conflicts
:
958 printf("label: \"conflicts\" color: lightgreen }\n");
960 case pkgCache::Dep::DpkgBreaks
:
961 printf("label: \"breaks\" color: lightgreen }\n");
963 case pkgCache::Dep::Obsoletes
:
964 printf("label: \"obsoletes\" color: lightgreen }\n");
967 case pkgCache::Dep::PreDepends
:
968 printf("label: \"predepends\" color: blue }\n");
980 /* Draw the box colours after the fact since we can not tell what colour
981 they should be until everything is finished drawing */
982 for (pkgCache::PkgIterator Pkg
= Cache
->PkgBegin(); Pkg
.end() == false; ++Pkg
)
984 if (Show
[Pkg
->ID
] < DoneNR
)
987 if (Show
[Pkg
->ID
] == DoneNR
)
988 printf("node: { title: \"%s\" label: \"%s\" color: orange shape: %s }\n", Pkg
.FullName(true).c_str(), Pkg
.FullName(true).c_str(),
989 Shapes
[ShapeMap
[Pkg
->ID
]]);
991 printf("node: { title: \"%s\" label: \"%s\" shape: %s }\n", Pkg
.FullName(true).c_str(), Pkg
.FullName(true).c_str(),
992 Shapes
[ShapeMap
[Pkg
->ID
]]);
1004 // Dotty - Generate a graph for Dotty /*{{{*/
1005 // ---------------------------------------------------------------------
1006 /* Dotty is the graphvis program for generating graphs. It is a fairly
1007 simple queuing algorithm that just writes dependencies and nodes.
1008 http://www.research.att.com/sw/tools/graphviz/ */
1009 static bool Dotty(CommandLine
&CmdL
)
1011 pkgCacheFile CacheFile
;
1012 pkgCache
*Cache
= CacheFile
.GetPkgCache();
1013 if (unlikely(Cache
== NULL
))
1016 bool GivenOnly
= _config
->FindB("APT::Cache::GivenOnly",false);
1018 /* Normal packages are boxes
1019 Pure Provides are triangles
1021 Hexagons are missing packages*/
1022 const char *Shapes
[] = {"hexagon","triangle","box","diamond"};
1024 /* Initialize the list of packages to show.
1026 2 = To Show no recurse
1027 3 = Emitted no recurse
1030 enum States
{None
=0, ToShow
, ToShowNR
, DoneNR
, Done
};
1031 enum TheFlags
{ForceNR
=(1<<0)};
1032 unsigned char *Show
= new unsigned char[Cache
->Head().PackageCount
];
1033 unsigned char *Flags
= new unsigned char[Cache
->Head().PackageCount
];
1034 unsigned char *ShapeMap
= new unsigned char[Cache
->Head().PackageCount
];
1036 // Show everything if no arguments given
1037 if (CmdL
.FileList
[1] == 0)
1038 for (unsigned long I
= 0; I
!= Cache
->Head().PackageCount
; I
++)
1041 for (unsigned long I
= 0; I
!= Cache
->Head().PackageCount
; I
++)
1043 memset(Flags
,0,sizeof(*Flags
)*Cache
->Head().PackageCount
);
1046 for (pkgCache::PkgIterator Pkg
= Cache
->PkgBegin(); Pkg
.end() == false; ++Pkg
)
1048 if (Pkg
->VersionList
== 0)
1051 if (Pkg
->ProvidesList
== 0)
1052 ShapeMap
[Pkg
->ID
] = 0;
1054 ShapeMap
[Pkg
->ID
] = 1;
1059 if (Pkg
->ProvidesList
== 0)
1060 ShapeMap
[Pkg
->ID
] = 2;
1062 ShapeMap
[Pkg
->ID
] = 3;
1066 // Load the list of packages from the command line into the show list
1067 APT::CacheSetHelper
helper(true, GlobalError::NOTICE
);
1068 std::list
<APT::CacheSetHelper::PkgModifier
> mods
;
1069 mods
.push_back(APT::CacheSetHelper::PkgModifier(0, ",", APT::PackageSet::Modifier::POSTFIX
));
1070 mods
.push_back(APT::CacheSetHelper::PkgModifier(1, "^", APT::PackageSet::Modifier::POSTFIX
));
1071 std::map
<unsigned short, APT::PackageSet
> pkgsets
=
1072 APT::PackageSet::GroupedFromCommandLine(CacheFile
, CmdL
.FileList
+ 1, mods
, 0, helper
);
1074 for (APT::PackageSet::const_iterator Pkg
= pkgsets
[0].begin();
1075 Pkg
!= pkgsets
[0].end(); ++Pkg
)
1076 Show
[Pkg
->ID
] = ToShow
;
1077 for (APT::PackageSet::const_iterator Pkg
= pkgsets
[1].begin();
1078 Pkg
!= pkgsets
[1].end(); ++Pkg
)
1080 Show
[Pkg
->ID
] = ToShow
;
1081 Flags
[Pkg
->ID
] |= ForceNR
;
1085 printf("digraph packages {\n");
1086 printf("concentrate=true;\n");
1087 printf("size=\"30,40\";\n");
1093 for (pkgCache::PkgIterator Pkg
= Cache
->PkgBegin(); Pkg
.end() == false; ++Pkg
)
1095 // See we need to show this package
1096 if (Show
[Pkg
->ID
] == None
|| Show
[Pkg
->ID
] >= DoneNR
)
1100 if (Show
[Pkg
->ID
] == ToShowNR
|| (Flags
[Pkg
->ID
] & ForceNR
) == ForceNR
)
1102 // Pure Provides and missing packages have no deps!
1103 if (ShapeMap
[Pkg
->ID
] == 0 || ShapeMap
[Pkg
->ID
] == 1)
1104 Show
[Pkg
->ID
] = Done
;
1106 Show
[Pkg
->ID
] = DoneNR
;
1109 Show
[Pkg
->ID
] = Done
;
1112 // No deps to map out
1113 if (Pkg
->VersionList
== 0 || Show
[Pkg
->ID
] == DoneNR
)
1116 pkgCache::VerIterator Ver
= Pkg
.VersionList();
1117 for (pkgCache::DepIterator D
= Ver
.DependsList(); D
.end() == false; ++D
)
1119 // See if anything can meet this dep
1120 // Walk along the actual package providing versions
1122 pkgCache::PkgIterator DPkg
= D
.TargetPkg();
1123 for (pkgCache::VerIterator I
= DPkg
.VersionList();
1124 I
.end() == false && Hit
== false; ++I
)
1126 if (Cache
->VS
->CheckDep(I
.VerStr(),D
->CompareOp
,D
.TargetVer()) == true)
1130 // Follow all provides
1131 for (pkgCache::PrvIterator I
= DPkg
.ProvidesList();
1132 I
.end() == false && Hit
== false; ++I
)
1134 if (Cache
->VS
->CheckDep(I
.ProvideVersion(),D
->CompareOp
,D
.TargetVer()) == false)
1138 // Only graph critical deps
1139 if (D
.IsCritical() == true)
1141 printf("\"%s\" -> \"%s\"",Pkg
.FullName(true).c_str(),D
.TargetPkg().FullName(true).c_str());
1143 // Colour the node for recursion
1144 if (Show
[D
.TargetPkg()->ID
] <= DoneNR
)
1146 /* If a conflicts does not meet anything in the database
1147 then show the relation but do not recurse */
1148 if (Hit
== false && D
.IsNegative() == true)
1150 if (Show
[D
.TargetPkg()->ID
] == None
&&
1151 Show
[D
.TargetPkg()->ID
] != ToShow
)
1152 Show
[D
.TargetPkg()->ID
] = ToShowNR
;
1156 if (GivenOnly
== true && Show
[D
.TargetPkg()->ID
] != ToShow
)
1157 Show
[D
.TargetPkg()->ID
] = ToShowNR
;
1159 Show
[D
.TargetPkg()->ID
] = ToShow
;
1166 case pkgCache::Dep::Conflicts
:
1167 case pkgCache::Dep::Obsoletes
:
1168 case pkgCache::Dep::DpkgBreaks
:
1169 printf("[color=springgreen];\n");
1172 case pkgCache::Dep::PreDepends
:
1173 printf("[color=blue];\n");
1185 /* Draw the box colours after the fact since we can not tell what colour
1186 they should be until everything is finished drawing */
1187 for (pkgCache::PkgIterator Pkg
= Cache
->PkgBegin(); Pkg
.end() == false; ++Pkg
)
1189 if (Show
[Pkg
->ID
] < DoneNR
)
1192 // Orange box for early recursion stoppage
1193 if (Show
[Pkg
->ID
] == DoneNR
)
1194 printf("\"%s\" [color=orange,shape=%s];\n",Pkg
.FullName(true).c_str(),
1195 Shapes
[ShapeMap
[Pkg
->ID
]]);
1197 printf("\"%s\" [shape=%s];\n",Pkg
.FullName(true).c_str(),
1198 Shapes
[ShapeMap
[Pkg
->ID
]]);
1208 // DisplayRecord - Displays the complete record for the package /*{{{*/
1209 // ---------------------------------------------------------------------
1210 /* This displays the package record from the proper package index file.
1211 It is not used by DumpAvail for performance reasons. */
1213 static APT_PURE
unsigned char const* skipDescriptionFields(unsigned char const * DescP
)
1215 char const * const TagName
= "\nDescription";
1216 size_t const TagLen
= strlen(TagName
);
1217 while ((DescP
= (unsigned char*)strchr((char*)DescP
, '\n')) != NULL
)
1219 if (DescP
[1] == ' ')
1221 else if (strncmp((char*)DescP
, TagName
, TagLen
) == 0)
1230 static bool DisplayRecord(pkgCacheFile
&CacheFile
, pkgCache::VerIterator V
)
1232 pkgCache
*Cache
= CacheFile
.GetPkgCache();
1233 if (unlikely(Cache
== NULL
))
1236 // Find an appropriate file
1237 pkgCache::VerFileIterator Vf
= V
.FileList();
1238 for (; Vf
.end() == false; ++Vf
)
1239 if ((Vf
.File()->Flags
& pkgCache::Flag::NotSource
) == 0)
1241 if (Vf
.end() == true)
1244 // Check and load the package list file
1245 pkgCache::PkgFileIterator I
= Vf
.File();
1246 if (I
.IsOk() == false)
1247 return _error
->Error(_("Package file %s is out of sync."),I
.FileName());
1250 if (PkgF
.Open(I
.FileName(), FileFd::ReadOnly
, FileFd::Extension
) == false)
1253 // Read the record (and ensure that it ends with a newline and NUL)
1254 unsigned char *Buffer
= new unsigned char[Cache
->HeaderP
->MaxVerFileSize
+2];
1255 Buffer
[Vf
->Size
] = '\n';
1256 Buffer
[Vf
->Size
+1] = '\0';
1257 if (PkgF
.Seek(Vf
->Offset
) == false ||
1258 PkgF
.Read(Buffer
,Vf
->Size
) == false)
1264 // Get a pointer to start of Description field
1265 const unsigned char *DescP
= (unsigned char*)strstr((char*)Buffer
, "\nDescription");
1269 DescP
= Buffer
+ Vf
->Size
;
1271 // Write all but Description
1272 size_t const length
= DescP
- Buffer
;
1273 if (length
!= 0 && FileFd::Write(STDOUT_FILENO
, Buffer
, length
) == false)
1279 // Show the right description
1280 pkgRecords
Recs(*Cache
);
1281 pkgCache::DescIterator Desc
= V
.TranslatedDescription();
1282 if (Desc
.end() == false)
1284 pkgRecords::Parser
&P
= Recs
.Lookup(Desc
.FileList());
1285 cout
<< "Description" << ( (strcmp(Desc
.LanguageCode(),"") != 0) ? "-" : "" ) << Desc
.LanguageCode() << ": " << P
.LongDesc();
1286 cout
<< std::endl
<< "Description-md5: " << Desc
.md5() << std::endl
;
1288 // Find the first field after the description (if there is any)
1289 DescP
= skipDescriptionFields(DescP
);
1291 // else we have no translation, so we found a lonely Description-md5 -> don't skip it
1293 // write the rest of the buffer, but skip mixed in Descriptions* fields
1294 while (DescP
!= NULL
)
1296 const unsigned char * const Start
= DescP
;
1297 const unsigned char *End
= (unsigned char*)strstr((char*)DescP
, "\nDescription");
1300 End
= &Buffer
[Vf
->Size
];
1305 ++End
; // get the newline into the output
1306 DescP
= skipDescriptionFields(End
+ strlen("Description"));
1308 size_t const length
= End
- Start
;
1309 if (length
!= 0 && FileFd::Write(STDOUT_FILENO
, Start
, length
) == false)
1316 // write a final newline after the last field
1325 pkgCache::DescFile
*Df
;
1329 // Search - Perform a search /*{{{*/
1330 // ---------------------------------------------------------------------
1331 /* This searches the package names and package descriptions for a pattern */
1332 static bool Search(CommandLine
&CmdL
)
1334 bool const ShowFull
= _config
->FindB("APT::Cache::ShowFull",false);
1335 bool const NamesOnly
= _config
->FindB("APT::Cache::NamesOnly",false);
1336 unsigned int const NumPatterns
= CmdL
.FileSize() -1;
1338 pkgCacheFile CacheFile
;
1339 pkgCache
*Cache
= CacheFile
.GetPkgCache();
1340 pkgDepCache::Policy
*Plcy
= CacheFile
.GetPolicy();
1341 if (unlikely(Cache
== NULL
|| Plcy
== NULL
))
1344 // Make sure there is at least one argument
1345 if (NumPatterns
< 1)
1346 return _error
->Error(_("You must give at least one search pattern"));
1348 // Compile the regex pattern
1349 regex_t
*Patterns
= new regex_t
[NumPatterns
];
1350 memset(Patterns
,0,sizeof(*Patterns
)*NumPatterns
);
1351 for (unsigned I
= 0; I
!= NumPatterns
; I
++)
1353 if (regcomp(&Patterns
[I
],CmdL
.FileList
[I
+1],REG_EXTENDED
| REG_ICASE
|
1357 regfree(&Patterns
[I
]);
1358 return _error
->Error("Regex compilation error");
1362 if (_error
->PendingError() == true)
1364 for (unsigned I
= 0; I
!= NumPatterns
; I
++)
1365 regfree(&Patterns
[I
]);
1369 size_t const descCount
= Cache
->HeaderP
->GroupCount
+ 1;
1370 ExDescFile
*DFList
= new ExDescFile
[descCount
];
1371 memset(DFList
,0,sizeof(*DFList
) * descCount
);
1373 bool *PatternMatch
= new bool[descCount
* NumPatterns
];
1374 memset(PatternMatch
,false,sizeof(*PatternMatch
) * descCount
* NumPatterns
);
1376 // Map versions that we want to write out onto the VerList array.
1377 for (pkgCache::GrpIterator G
= Cache
->GrpBegin(); G
.end() == false; ++G
)
1379 size_t const PatternOffset
= G
->ID
* NumPatterns
;
1380 size_t unmatched
= 0, matched
= 0;
1381 for (unsigned I
= 0; I
< NumPatterns
; ++I
)
1383 if (PatternMatch
[PatternOffset
+ I
] == true)
1385 else if (regexec(&Patterns
[I
],G
.Name(),0,0,0) == 0)
1386 PatternMatch
[PatternOffset
+ I
] = true;
1391 // already dealt with this package?
1392 if (matched
== NumPatterns
)
1395 // Doing names only, drop any that don't match..
1396 if (NamesOnly
== true && unmatched
== NumPatterns
)
1399 // Find the proper version to use
1400 pkgCache::PkgIterator P
= G
.FindPreferredPkg();
1401 if (P
.end() == true)
1403 pkgCache::VerIterator V
= Plcy
->GetCandidateVer(P
);
1404 if (V
.end() == false)
1406 pkgCache::DescIterator
const D
= V
.TranslatedDescription();
1407 //FIXME: packages without a description can't be found
1408 if (D
.end() == true)
1410 DFList
[G
->ID
].Df
= D
.FileList();
1411 DFList
[G
->ID
].ID
= G
->ID
;
1414 if (unmatched
== NumPatterns
)
1417 // Include all the packages that provide matching names too
1418 for (pkgCache::PrvIterator Prv
= P
.ProvidesList() ; Prv
.end() == false; ++Prv
)
1420 pkgCache::VerIterator V
= Plcy
->GetCandidateVer(Prv
.OwnerPkg());
1421 if (V
.end() == true)
1424 unsigned long id
= Prv
.OwnerPkg().Group()->ID
;
1425 pkgCache::DescIterator
const D
= V
.TranslatedDescription();
1426 //FIXME: packages without a description can't be found
1427 if (D
.end() == true)
1429 DFList
[id
].Df
= D
.FileList();
1432 size_t const PrvPatternOffset
= id
* NumPatterns
;
1433 for (unsigned I
= 0; I
< NumPatterns
; ++I
)
1434 PatternMatch
[PrvPatternOffset
+ I
] = PatternMatch
[PatternOffset
+ I
];
1438 LocalitySort(&DFList
->Df
,Cache
->HeaderP
->GroupCount
,sizeof(*DFList
));
1440 // Create the text record parser
1441 pkgRecords
Recs(*Cache
);
1442 // Iterate over all the version records and check them
1443 for (ExDescFile
*J
= DFList
; J
->Df
!= 0; ++J
)
1445 pkgRecords::Parser
&P
= Recs
.Lookup(pkgCache::DescFileIterator(*Cache
,J
->Df
));
1446 size_t const PatternOffset
= J
->ID
* NumPatterns
;
1448 if (NamesOnly
== false)
1450 string
const LongDesc
= P
.LongDesc();
1451 for (unsigned I
= 0; I
< NumPatterns
; ++I
)
1453 if (PatternMatch
[PatternOffset
+ I
] == true)
1455 else if (regexec(&Patterns
[I
],LongDesc
.c_str(),0,0,0) == 0)
1456 PatternMatch
[PatternOffset
+ I
] = true;
1460 bool matchedAll
= true;
1461 for (unsigned I
= 0; I
< NumPatterns
; ++I
)
1462 if (PatternMatch
[PatternOffset
+ I
] == false)
1468 if (matchedAll
== true)
1470 if (ShowFull
== true)
1474 P
.GetRec(Start
,End
);
1475 fwrite(Start
,End
-Start
,1,stdout
);
1479 printf("%s - %s\n",P
.Name().c_str(),P
.ShortDesc().c_str());
1484 delete [] PatternMatch
;
1485 for (unsigned I
= 0; I
!= NumPatterns
; I
++)
1486 regfree(&Patterns
[I
]);
1488 return _error
->Error("Write to stdout failed");
1492 /* ShowAuto - show automatically installed packages (sorted) {{{*/
1493 static bool ShowAuto(CommandLine
&)
1495 pkgCacheFile CacheFile
;
1496 pkgCache
*Cache
= CacheFile
.GetPkgCache();
1497 pkgDepCache
*DepCache
= CacheFile
.GetDepCache();
1498 if (unlikely(Cache
== NULL
|| DepCache
== NULL
))
1501 std::vector
<string
> packages
;
1502 packages
.reserve(Cache
->HeaderP
->PackageCount
/ 3);
1504 for (pkgCache::PkgIterator P
= Cache
->PkgBegin(); P
.end() == false; ++P
)
1505 if ((*DepCache
)[P
].Flags
& pkgCache::Flag::Auto
)
1506 packages
.push_back(P
.Name());
1508 std::sort(packages
.begin(), packages
.end());
1510 for (vector
<string
>::iterator I
= packages
.begin(); I
!= packages
.end(); ++I
)
1513 _error
->Notice(_("This command is deprecated. Please use 'apt-mark showauto' instead."));
1517 // ShowPackage - Dump the package record to the screen /*{{{*/
1518 // ---------------------------------------------------------------------
1520 static bool ShowPackage(CommandLine
&CmdL
)
1522 pkgCacheFile CacheFile
;
1523 CacheSetHelperVirtuals
helper(true, GlobalError::NOTICE
);
1524 APT::CacheSetHelper::VerSelector
const select
= _config
->FindB("APT::Cache::AllVersions", true) ?
1525 APT::CacheSetHelper::ALL
: APT::CacheSetHelper::CANDIDATE
;
1526 APT::VersionList
const verset
= APT::VersionList::FromCommandLine(CacheFile
, CmdL
.FileList
+ 1, select
, helper
);
1527 for (APT::VersionList::const_iterator Ver
= verset
.begin(); Ver
!= verset
.end(); ++Ver
)
1528 if (DisplayRecord(CacheFile
, Ver
) == false)
1531 if (verset
.empty() == true)
1533 if (helper
.virtualPkgs
.empty() == true)
1534 return _error
->Error(_("No packages found"));
1536 _error
->Notice(_("No packages found"));
1541 // ShowPkgNames - Show package names /*{{{*/
1542 // ---------------------------------------------------------------------
1543 /* This does a prefix match on the first argument */
1544 static bool ShowPkgNames(CommandLine
&CmdL
)
1546 pkgCacheFile CacheFile
;
1547 if (unlikely(CacheFile
.BuildCaches(NULL
, false) == false))
1549 pkgCache::GrpIterator I
= CacheFile
.GetPkgCache()->GrpBegin();
1550 bool const All
= _config
->FindB("APT::Cache::AllNames","false");
1552 if (CmdL
.FileList
[1] != 0)
1554 for (;I
.end() != true; ++I
)
1556 if (All
== false && I
->FirstPackage
== 0)
1558 if (I
.FindPkg("any")->VersionList
== 0)
1560 if (strncmp(I
.Name(),CmdL
.FileList
[1],strlen(CmdL
.FileList
[1])) == 0)
1561 cout
<< I
.Name() << endl
;
1568 for (;I
.end() != true; ++I
)
1570 if (All
== false && I
->FirstPackage
== 0)
1572 if (I
.FindPkg("any")->VersionList
== 0)
1574 cout
<< I
.Name() << endl
;
1580 // ShowSrcPackage - Show source package records /*{{{*/
1581 // ---------------------------------------------------------------------
1583 static bool ShowSrcPackage(CommandLine
&CmdL
)
1585 pkgCacheFile CacheFile
;
1586 pkgSourceList
*List
= CacheFile
.GetSourceList();
1587 if (unlikely(List
== NULL
))
1590 // Create the text record parsers
1591 pkgSrcRecords
SrcRecs(*List
);
1592 if (_error
->PendingError() == true)
1596 for (const char **I
= CmdL
.FileList
+ 1; *I
!= 0; I
++)
1600 pkgSrcRecords::Parser
*Parse
;
1601 unsigned found_this
= 0;
1602 while ((Parse
= SrcRecs
.Find(*I
,false)) != 0) {
1603 cout
<< Parse
->AsStr() << endl
;;
1607 if (found_this
== 0) {
1608 _error
->Warning(_("Unable to locate package %s"),*I
);
1613 _error
->Notice(_("No packages found"));
1617 // Policy - Show the results of the preferences file /*{{{*/
1618 // ---------------------------------------------------------------------
1620 static bool Policy(CommandLine
&CmdL
)
1622 pkgCacheFile CacheFile
;
1623 pkgCache
*Cache
= CacheFile
.GetPkgCache();
1624 pkgPolicy
*Plcy
= CacheFile
.GetPolicy();
1625 pkgSourceList
*SrcList
= CacheFile
.GetSourceList();
1626 if (unlikely(Cache
== NULL
|| Plcy
== NULL
|| SrcList
== NULL
))
1629 /* Should the MultiArchKiller be run to see which pseudo packages for an
1630 arch all package are currently installed? Activating it gives a speed
1631 penality for no real gain beside enhanced debugging, so in general no. */
1632 if (_config
->FindB("APT::Cache::Policy::DepCache", false) == true)
1633 CacheFile
.GetDepCache();
1635 // Print out all of the package files
1636 if (CmdL
.FileList
[1] == 0)
1638 cout
<< _("Package files:") << endl
;
1639 for (pkgCache::PkgFileIterator F
= Cache
->FileBegin(); F
.end() == false; ++F
)
1641 // Locate the associated index files so we can derive a description
1643 if (SrcList
->FindIndex(F
,Indx
) == false &&
1644 _system
->FindIndex(F
,Indx
) == false)
1645 return _error
->Error(_("Cache is out of sync, can't x-ref a package file"));
1648 Plcy
->GetPriority(F
),Indx
->Describe(true).c_str());
1650 // Print the reference information for the package
1651 string Str
= F
.RelStr();
1652 if (Str
.empty() == false)
1653 printf(" release %s\n",F
.RelStr().c_str());
1654 if (F
.Site() != 0 && F
.Site()[0] != 0)
1655 printf(" origin %s\n",F
.Site());
1658 // Show any packages have explicit pins
1659 cout
<< _("Pinned packages:") << endl
;
1660 pkgCache::PkgIterator I
= Cache
->PkgBegin();
1661 for (;I
.end() != true; ++I
)
1663 if (Plcy
->GetPriority(I
) == 0)
1666 // Print the package name and the version we are forcing to
1667 cout
<< " " << I
.FullName(true) << " -> ";
1669 pkgCache::VerIterator V
= Plcy
->GetMatch(I
);
1670 if (V
.end() == true)
1671 cout
<< _("(not found)") << endl
;
1673 cout
<< V
.VerStr() << endl
;
1679 char const * const msgInstalled
= _(" Installed: ");
1680 char const * const msgCandidate
= _(" Candidate: ");
1681 short const InstalledLessCandidate
=
1682 mbstowcs(NULL
, msgInstalled
, 0) - mbstowcs(NULL
, msgCandidate
, 0);
1683 short const deepInstalled
=
1684 (InstalledLessCandidate
< 0 ? (InstalledLessCandidate
*-1) : 0) - 1;
1685 short const deepCandidate
=
1686 (InstalledLessCandidate
> 0 ? (InstalledLessCandidate
) : 0) - 1;
1688 // Print out detailed information for each package
1689 APT::CacheSetHelper
helper(true, GlobalError::NOTICE
);
1690 APT::PackageList pkgset
= APT::PackageList::FromCommandLine(CacheFile
, CmdL
.FileList
+ 1, helper
);
1691 for (APT::PackageList::const_iterator Pkg
= pkgset
.begin(); Pkg
!= pkgset
.end(); ++Pkg
)
1693 cout
<< Pkg
.FullName(true) << ":" << endl
;
1695 // Installed version
1696 cout
<< msgInstalled
<< OutputInDepth(deepInstalled
, " ");
1697 if (Pkg
->CurrentVer
== 0)
1698 cout
<< _("(none)") << endl
;
1700 cout
<< Pkg
.CurrentVer().VerStr() << endl
;
1702 // Candidate Version
1703 cout
<< msgCandidate
<< OutputInDepth(deepCandidate
, " ");
1704 pkgCache::VerIterator V
= Plcy
->GetCandidateVer(Pkg
);
1705 if (V
.end() == true)
1706 cout
<< _("(none)") << endl
;
1708 cout
<< V
.VerStr() << endl
;
1711 if (Plcy
->GetPriority(Pkg
) != 0)
1713 cout
<< _(" Package pin: ");
1714 V
= Plcy
->GetMatch(Pkg
);
1715 if (V
.end() == true)
1716 cout
<< _("(not found)") << endl
;
1718 cout
<< V
.VerStr() << endl
;
1721 // Show the priority tables
1722 cout
<< _(" Version table:") << endl
;
1723 for (V
= Pkg
.VersionList(); V
.end() == false; ++V
)
1725 if (Pkg
.CurrentVer() == V
)
1726 cout
<< " *** " << V
.VerStr();
1728 cout
<< " " << V
.VerStr();
1729 cout
<< " " << Plcy
->GetPriority(Pkg
) << endl
;
1730 for (pkgCache::VerFileIterator VF
= V
.FileList(); VF
.end() == false; ++VF
)
1732 // Locate the associated index files so we can derive a description
1734 if (SrcList
->FindIndex(VF
.File(),Indx
) == false &&
1735 _system
->FindIndex(VF
.File(),Indx
) == false)
1736 return _error
->Error(_("Cache is out of sync, can't x-ref a package file"));
1737 printf(" %4i %s\n",Plcy
->GetPriority(VF
.File()),
1738 Indx
->Describe(true).c_str());
1746 // Madison - Look a bit like katie's madison /*{{{*/
1747 // ---------------------------------------------------------------------
1749 static bool Madison(CommandLine
&CmdL
)
1751 pkgCacheFile CacheFile
;
1752 pkgSourceList
*SrcList
= CacheFile
.GetSourceList();
1757 // Create the src text record parsers and ignore errors about missing
1758 // deb-src lines that are generated from pkgSrcRecords::pkgSrcRecords
1759 pkgSrcRecords
SrcRecs(*SrcList
);
1760 if (_error
->PendingError() == true)
1763 APT::CacheSetHelper
helper(true, GlobalError::NOTICE
);
1764 for (const char **I
= CmdL
.FileList
+ 1; *I
!= 0; I
++)
1766 _error
->PushToStack();
1767 APT::PackageList pkgset
= APT::PackageList::FromString(CacheFile
, *I
, helper
);
1768 for (APT::PackageList::const_iterator Pkg
= pkgset
.begin(); Pkg
!= pkgset
.end(); ++Pkg
)
1770 for (pkgCache::VerIterator V
= Pkg
.VersionList(); V
.end() == false; ++V
)
1772 for (pkgCache::VerFileIterator VF
= V
.FileList(); VF
.end() == false; ++VF
)
1774 // This might be nice, but wouldn't uniquely identify the source -mdz
1775 // if (VF.File().Archive() != 0)
1777 // cout << setw(10) << Pkg.Name() << " | " << setw(10) << V.VerStr() << " | "
1778 // << VF.File().Archive() << endl;
1781 // Locate the associated index files so we can derive a description
1782 for (pkgSourceList::const_iterator S
= SrcList
->begin(); S
!= SrcList
->end(); ++S
)
1784 vector
<pkgIndexFile
*> *Indexes
= (*S
)->GetIndexFiles();
1785 for (vector
<pkgIndexFile
*>::const_iterator IF
= Indexes
->begin();
1786 IF
!= Indexes
->end(); ++IF
)
1788 if ((*IF
)->FindInCache(*(VF
.File().Cache())) == VF
.File())
1790 cout
<< setw(10) << Pkg
.FullName(true) << " | " << setw(10) << V
.VerStr() << " | "
1791 << (*IF
)->Describe(true) << endl
;
1800 pkgSrcRecords::Parser
*SrcParser
;
1801 bool foundSomething
= false;
1802 while ((SrcParser
= SrcRecs
.Find(*I
, false)) != 0)
1804 foundSomething
= true;
1805 // Maybe support Release info here too eventually
1806 cout
<< setw(10) << SrcParser
->Package() << " | "
1807 << setw(10) << SrcParser
->Version() << " | "
1808 << SrcParser
->Index().Describe(true) << endl
;
1810 if (foundSomething
== true)
1811 _error
->RevertToStack();
1813 _error
->MergeWithStack();
1819 // GenCaches - Call the main cache generator /*{{{*/
1820 // ---------------------------------------------------------------------
1822 static bool GenCaches(CommandLine
&)
1824 OpTextProgress
Progress(*_config
);
1826 pkgCacheFile CacheFile
;
1827 return CacheFile
.BuildCaches(&Progress
, true);
1830 // ShowHelp - Show a help screen /*{{{*/
1831 // ---------------------------------------------------------------------
1833 static bool ShowHelp(CommandLine
&)
1835 ioprintf(cout
, "%s %s (%s)\n", PACKAGE
, PACKAGE_VERSION
, COMMON_ARCH
);
1837 if (_config
->FindB("version") == true)
1841 _("Usage: apt-cache [options] command\n"
1842 " apt-cache [options] showpkg pkg1 [pkg2 ...]\n"
1843 " apt-cache [options] showsrc pkg1 [pkg2 ...]\n"
1845 "apt-cache is a low-level tool used to query information\n"
1846 "from APT's binary cache files\n"
1849 " gencaches - Build both the package and source cache\n"
1850 " showpkg - Show some general information for a single package\n"
1851 " showsrc - Show source records\n"
1852 " stats - Show some basic statistics\n"
1853 " dump - Show the entire file in a terse form\n"
1854 " dumpavail - Print an available file to stdout\n"
1855 " unmet - Show unmet dependencies\n"
1856 " search - Search the package list for a regex pattern\n"
1857 " show - Show a readable record for the package\n"
1858 " depends - Show raw dependency information for a package\n"
1859 " rdepends - Show reverse dependency information for a package\n"
1860 " pkgnames - List the names of all packages in the system\n"
1861 " dotty - Generate package graphs for GraphViz\n"
1862 " xvcg - Generate package graphs for xvcg\n"
1863 " policy - Show policy settings\n"
1866 " -h This help text.\n"
1867 " -p=? The package cache.\n"
1868 " -s=? The source cache.\n"
1869 " -q Disable progress indicator.\n"
1870 " -i Show only important deps for the unmet command.\n"
1871 " -c=? Read this configuration file\n"
1872 " -o=? Set an arbitrary configuration option, eg -o dir::cache=/tmp\n"
1873 "See the apt-cache(8) and apt.conf(5) manual pages for more information.\n");
1877 int main(int argc
,const char *argv
[]) /*{{{*/
1879 CommandLine::Dispatch Cmds
[] = {{"help",&ShowHelp
},
1880 {"gencaches",&GenCaches
},
1881 {"showsrc",&ShowSrcPackage
},
1882 {"showpkg",&DumpPackage
},
1885 {"dumpavail",&DumpAvail
},
1888 {"depends",&Depends
},
1889 {"rdepends",&RDepends
},
1892 {"show",&ShowPackage
},
1893 {"pkgnames",&ShowPkgNames
},
1894 {"showauto",&ShowAuto
},
1896 {"madison",&Madison
},
1899 std::vector
<CommandLine::Args
> Args
= getCommandArgs("apt-cache", CommandLine::GetCommand(Cmds
, argc
, argv
));
1901 // Set up gettext support
1902 setlocale(LC_ALL
,"");
1903 textdomain(PACKAGE
);
1905 // Parse the command line and initialize the package library
1907 ParseCommandLine(CmdL
, Cmds
, Args
.data(), &_config
, &_system
, argc
, argv
, ShowHelp
);
1911 if (_config
->Exists("APT::Cache::Generate") == true)
1912 _config
->Set("pkgCacheFile::Generate", _config
->FindB("APT::Cache::Generate", true));
1914 // Match the operation
1915 CmdL
.DispatchArg(Cmds
);
1917 // Print any errors or warnings found during parsing
1918 bool const Errors
= _error
->PendingError();
1919 if (_config
->FindI("quiet",0) > 0)
1920 _error
->DumpErrors();
1922 _error
->DumpErrors(GlobalError::DEBUG
);
1923 return Errors
== true ? 100 : 0;