]>
git.saurik.com Git - apt.git/blob - cmdline/apt-cache.cc
1 // -*- mode: cpp; mode: fold -*-
3 // $Id: apt-cache.cc,v 1.25 1999/02/11 00:03:07 jgg 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. The
10 only really usefull function right now is dumpavail which is used
11 by the dselect method. Everything else is ment as a debug aide.
13 Returns 100 on failure, 0 on success.
15 ##################################################################### */
17 // Include Files /*{{{*/
18 #include <apt-pkg/error.h>
19 #include <apt-pkg/pkgcachegen.h>
20 #include <apt-pkg/deblistparser.h>
21 #include <apt-pkg/init.h>
22 #include <apt-pkg/progress.h>
23 #include <apt-pkg/sourcelist.h>
24 #include <apt-pkg/cmndline.h>
25 #include <apt-pkg/strutl.h>
35 // UnMet - Show unmet dependencies /*{{{*/
36 // ---------------------------------------------------------------------
38 bool UnMet(CommandLine
&CmdL
)
40 pkgCache
&Cache
= *GCache
;
41 bool Important
= _config
->FindB("APT::Cache::Important",false);
43 for (pkgCache::PkgIterator P
= Cache
.PkgBegin(); P
.end() == false; P
++)
45 for (pkgCache::VerIterator V
= P
.VersionList(); V
.end() == false; V
++)
48 for (pkgCache::DepIterator D
= V
.DependsList(); D
.end() == false;)
51 pkgCache::DepIterator Start
;
52 pkgCache::DepIterator End
;
55 /* cout << "s: Check " << Start.TargetPkg().Name() << ',' <<
56 End.TargetPkg().Name() << endl;*/
58 // Skip conflicts and replaces
59 if (End
->Type
!= pkgCache::Dep::PreDepends
&&
60 End
->Type
!= pkgCache::Dep::Depends
&&
61 End
->Type
!= pkgCache::Dep::Suggests
&&
62 End
->Type
!= pkgCache::Dep::Recommends
)
65 // Important deps only
66 if (Important
== true)
67 if (End
->Type
!= pkgCache::Dep::PreDepends
&&
68 End
->Type
!= pkgCache::Dep::Depends
)
71 // Verify the or group
73 pkgCache::DepIterator RealStart
= Start
;
76 // See if this dep is Ok
77 pkgCache::Version
**VList
= Start
.AllTargets();
98 cout
<< "Package " << P
.Name() << " version " <<
99 V
.VerStr() << " has an unmet dep:" << endl
;
102 // Print out the dep type
103 cout
<< " " << End
.DepType() << ": ";
109 cout
<< Start
.TargetPkg().Name();
110 if (Start
.TargetVer() != 0)
111 cout
<< " (" << Start
.CompType() << " " << Start
.TargetVer() <<
127 // DumpPackage - Show a dump of a package record /*{{{*/
128 // ---------------------------------------------------------------------
130 bool DumpPackage(CommandLine
&CmdL
)
132 pkgCache
&Cache
= *GCache
;
133 for (const char **I
= CmdL
.FileList
+ 1; *I
!= 0; I
++)
135 pkgCache::PkgIterator Pkg
= Cache
.FindPkg(*I
);
136 if (Pkg
.end() == true)
138 _error
->Warning("Unable to locate package %s",*I
);
142 cout
<< "Package: " << Pkg
.Name() << endl
;
143 cout
<< "Versions: ";
144 for (pkgCache::VerIterator Cur
= Pkg
.VersionList(); Cur
.end() != true; Cur
++)
146 cout
<< Cur
.VerStr();
147 for (pkgCache::VerFileIterator Vf
= Cur
.FileList(); Vf
.end() == false; Vf
++)
148 cout
<< "(" << Vf
.File().FileName() << ")";
154 cout
<< "Reverse Depends: " << endl
;
155 for (pkgCache::DepIterator D
= Pkg
.RevDependsList(); D
.end() != true; D
++)
156 cout
<< " " << D
.ParentPkg().Name() << ',' << D
.TargetPkg().Name() << endl
;
158 cout
<< "Dependencies: " << endl
;
159 for (pkgCache::VerIterator Cur
= Pkg
.VersionList(); Cur
.end() != true; Cur
++)
161 cout
<< Cur
.VerStr() << " - ";
162 for (pkgCache::DepIterator Dep
= Cur
.DependsList(); Dep
.end() != true; Dep
++)
163 cout
<< Dep
.TargetPkg().Name() << " (" << (int)Dep
->CompareOp
<< " " << Dep
.TargetVer() << ") ";
167 cout
<< "Provides: " << endl
;
168 for (pkgCache::VerIterator Cur
= Pkg
.VersionList(); Cur
.end() != true; Cur
++)
170 cout
<< Cur
.VerStr() << " - ";
171 for (pkgCache::PrvIterator Prv
= Cur
.ProvidesList(); Prv
.end() != true; Prv
++)
172 cout
<< Prv
.ParentPkg().Name() << " ";
175 cout
<< "Reverse Provides: " << endl
;
176 for (pkgCache::PrvIterator Prv
= Pkg
.ProvidesList(); Prv
.end() != true; Prv
++)
177 cout
<< Prv
.OwnerPkg().Name() << " " << Prv
.OwnerVer().VerStr();
185 // Stats - Dump some nice statistics /*{{{*/
186 // ---------------------------------------------------------------------
188 bool Stats(CommandLine
&Cmd
)
190 pkgCache
&Cache
= *GCache
;
191 cout
<< "Total Package Names : " << Cache
.Head().PackageCount
<< " (" <<
192 SizeToStr(Cache
.Head().PackageCount
*Cache
.Head().PackageSz
) << ')' << endl
;
193 pkgCache::PkgIterator I
= Cache
.PkgBegin();
200 for (;I
.end() != true; I
++)
202 if (I
->VersionList
!= 0 && I
->ProvidesList
== 0)
208 if (I
->VersionList
!= 0 && I
->ProvidesList
!= 0)
214 if (I
->VersionList
== 0 && I
->ProvidesList
!= 0)
217 if (I
.ProvidesList()->NextProvides
== 0)
225 if (I
->VersionList
== 0 && I
->ProvidesList
== 0)
231 cout
<< " Normal Packages: " << Normal
<< endl
;
232 cout
<< " Pure Virtual Packages: " << Virtual
<< endl
;
233 cout
<< " Single Virtual Packages: " << DVirt
<< endl
;
234 cout
<< " Mixed Virtual Packages: " << NVirt
<< endl
;
235 cout
<< " Missing: " << Missing
<< endl
;
237 cout
<< "Total Distinct Versions: " << Cache
.Head().VersionCount
<< " (" <<
238 SizeToStr(Cache
.Head().VersionCount
*Cache
.Head().VersionSz
) << ')' << endl
;
239 cout
<< "Total Dependencies: " << Cache
.Head().DependsCount
<< " (" <<
240 SizeToStr(Cache
.Head().DependsCount
*Cache
.Head().DependencySz
) << ')' << endl
;
242 cout
<< "Total Ver/File relations: " << Cache
.Head().VerFileCount
<< " (" <<
243 SizeToStr(Cache
.Head().VerFileCount
*Cache
.Head().VerFileSz
) << ')' << endl
;
244 cout
<< "Total Provides Mappings: " << Cache
.Head().ProvidesCount
<< " (" <<
245 SizeToStr(Cache
.Head().ProvidesCount
*Cache
.Head().ProvidesSz
) << ')' << endl
;
248 unsigned long Size
= 0;
249 unsigned long Count
= 0;
250 for (pkgCache::StringItem
*I
= Cache
.StringItemP
+ Cache
.Head().StringList
;
251 I
!= Cache
.StringItemP
; I
= Cache
.StringItemP
+ I
->NextItem
)
254 Size
+= strlen(Cache
.StrP
+ I
->String
);
256 cout
<< "Total Globbed Strings: " << Count
<< " (" << SizeToStr(Size
) << ')' << endl
;
258 unsigned long Slack
= 0;
259 for (int I
= 0; I
!= 7; I
++)
260 Slack
+= Cache
.Head().Pools
[I
].ItemSize
*Cache
.Head().Pools
[I
].Count
;
261 cout
<< "Total Slack space: " << SizeToStr(Slack
) << endl
;
263 unsigned long Total
= 0;
264 Total
= Slack
+ Size
+ Cache
.Head().DependsCount
*Cache
.Head().DependencySz
+
265 Cache
.Head().VersionCount
*Cache
.Head().VersionSz
+
266 Cache
.Head().PackageCount
*Cache
.Head().PackageSz
+
267 Cache
.Head().VerFileCount
*Cache
.Head().VerFileSz
+
268 Cache
.Head().ProvidesCount
*Cache
.Head().ProvidesSz
;
269 cout
<< "Total Space Accounted for: " << SizeToStr(Total
) << endl
;
274 // Check - Check some things about the cache /*{{{*/
275 // ---------------------------------------------------------------------
276 /* Debug aide mostly */
277 bool Check(CommandLine
&Cmd
)
279 pkgCache
&Cache
= *GCache
;
280 pkgCache::PkgIterator Pkg
= Cache
.PkgBegin();
281 for (;Pkg
.end() != true; Pkg
++)
283 if (Pkg
.Section() == 0 && Pkg
->VersionList
!= 0)
284 cout
<< "Bad section " << Pkg
.Name() << endl
;
286 for (pkgCache::VerIterator Cur
= Pkg
.VersionList();
287 Cur
.end() != true; Cur
++)
289 if (Cur
->Priority
< 1 || Cur
->Priority
> 5)
290 cout
<< "Bad prio " << Pkg
.Name() << ',' << Cur
.VerStr() << " == " << (int)Cur
->Priority
<< endl
;
296 // Dump - show everything /*{{{*/
297 // ---------------------------------------------------------------------
299 bool Dump(CommandLine
&Cmd
)
301 pkgCache
&Cache
= *GCache
;
302 for (pkgCache::PkgIterator P
= Cache
.PkgBegin(); P
.end() == false; P
++)
304 cout
<< "Package: " << P
.Name() << endl
;
305 for (pkgCache::VerIterator V
= P
.VersionList(); V
.end() == false; V
++)
307 cout
<< " Version: " << V
.VerStr() << endl
;
308 cout
<< " File: " << V
.FileList().File().FileName() << endl
;
309 for (pkgCache::DepIterator D
= V
.DependsList(); D
.end() == false; D
++)
310 cout
<< " Depends: " << D
.TargetPkg().Name() << ' ' << D
.TargetVer() << endl
;
314 for (pkgCache::PkgFileIterator
F(Cache
); F
.end() == false; F
++)
316 cout
<< "File: " << F
.FileName() << endl
;
317 cout
<< " Size: " << F
->Size
<< endl
;
318 cout
<< " ID: " << F
->ID
<< endl
;
319 cout
<< " Flags: " << F
->Flags
<< endl
;
320 cout
<< " Time: " << TimeRFC1123(F
->mtime
) << endl
;
321 cout
<< " Archive: " << F
.Archive() << endl
;
322 cout
<< " Component: " << F
.Component() << endl
;
323 cout
<< " Version: " << F
.Version() << endl
;
324 cout
<< " Origin: " << F
.Origin() << endl
;
325 cout
<< " Label: " << F
.Label() << endl
;
326 cout
<< " Architecture: " << F
.Architecture() << endl
;
332 // DumpAvail - Print out the available list /*{{{*/
333 // ---------------------------------------------------------------------
334 /* This is needed to make dpkg --merge happy */
335 bool DumpAvail(CommandLine
&Cmd
)
337 pkgCache
&Cache
= *GCache
;
338 unsigned char *Buffer
= new unsigned char[Cache
.HeaderP
->MaxVerFileSize
];
340 for (pkgCache::PkgFileIterator I
= Cache
.FileBegin(); I
.end() == false; I
++)
342 if ((I
->Flags
& pkgCache::Flag::NotSource
) != 0)
345 if (I
.IsOk() == false)
348 return _error
->Error("Package file %s is out of sync.",I
.FileName());
351 FileFd
PkgF(I
.FileName(),FileFd::ReadOnly
);
352 if (_error
->PendingError() == true)
358 /* Write all of the records from this package file, we search the entire
359 structure to find them */
360 for (pkgCache::PkgIterator P
= Cache
.PkgBegin(); P
.end() == false; P
++)
362 for (pkgCache::VerIterator V
= P
.VersionList(); V
.end() == false; V
++)
364 if (V
->FileList
== 0)
366 if (V
.FileList().File() != I
)
369 // Read the record and then write it out again.
370 if (PkgF
.Seek(V
.FileList()->Offset
) == false ||
371 PkgF
.Read(Buffer
,V
.FileList()->Size
) == false ||
372 write(STDOUT_FILENO
,Buffer
,V
.FileList()->Size
) != V
.FileList()->Size
)
384 // DoAdd - Perform an adding operation /*{{{*/
385 // ---------------------------------------------------------------------
387 bool DoAdd(CommandLine
&CmdL
)
389 // Make sure there is at least one argument
390 if (CmdL
.FileSize() <= 1)
391 return _error
->Error("You must give at least one file name");
394 FileFd
CacheF(_config
->FindFile("Dir::Cache::pkgcache"),FileFd::WriteAny
);
395 if (_error
->PendingError() == true)
398 DynamicMMap
Map(CacheF
,MMap::Public
);
399 if (_error
->PendingError() == true)
402 OpTextProgress
Progress(*_config
);
403 pkgCacheGenerator
Gen(Map
,Progress
);
404 if (_error
->PendingError() == true)
407 unsigned long Length
= CmdL
.FileSize() - 1;
408 for (const char **I
= CmdL
.FileList
+ 1; *I
!= 0; I
++)
410 Progress
.OverallProgress(I
- CmdL
.FileList
,Length
,1,"Generating cache");
411 Progress
.SubProgress(Length
);
414 FileFd
TagF(*I
,FileFd::ReadOnly
);
415 debListParser
Parser(TagF
);
416 if (_error
->PendingError() == true)
417 return _error
->Error("Problem opening %s",*I
);
419 if (Gen
.SelectFile(*I
) == false)
420 return _error
->Error("Problem with SelectFile");
422 if (Gen
.MergeList(Parser
) == false)
423 return _error
->Error("Problem with MergeList");
427 GCache
= &Gen
.GetCache();
433 // GenCaches - Call the main cache generator /*{{{*/
434 // ---------------------------------------------------------------------
436 bool GenCaches(CommandLine
&Cmd
)
438 OpTextProgress
Progress(*_config
);
442 return pkgMakeStatusCache(List
,Progress
);
445 // ShowHelp - Show a help screen /*{{{*/
446 // ---------------------------------------------------------------------
448 bool ShowHelp(CommandLine
&Cmd
)
450 cout
<< PACKAGE
<< ' ' << VERSION
<< " for " << ARCHITECTURE
<<
451 " compiled on " << __DATE__
<< " " << __TIME__
<< endl
;
452 if (_config
->FindB("version") == true)
455 cout
<< "Usage: apt-cache [options] command" << endl
;
456 cout
<< " apt-cache [options] add file1 [file1 ...]" << endl
;
457 cout
<< " apt-cache [options] showpkg pkg1 [pkg2 ...]" << endl
;
459 cout
<< "apt-cache is a low-level tool used to manipulate APT's binary" << endl
;
460 cout
<< "cache files stored in " << _config
->FindFile("Dir::Cache") << endl
;
461 cout
<< "It is not ment for ordinary use only as a debug aide." << endl
;
463 cout
<< "Commands:" << endl
;
464 cout
<< " add - Add an package file to the source cache" << endl
;
465 cout
<< " gencaches - Build both the package and source cache" << endl
;
466 cout
<< " showpkg - Show some general information for a single package" << endl
;
467 cout
<< " stats - Show some basic statistics" << endl
;
468 cout
<< " dump - Show the entire file in a terse form" << endl
;
469 cout
<< " dumpavail - Print an available file to stdout" << endl
;
470 cout
<< " unmet - Show unmet dependencies" << endl
;
471 cout
<< " check - Check the cache a bit" << endl
;
473 cout
<< "Options:" << endl
;
474 cout
<< " -h This help text." << endl
;
475 cout
<< " -p=? The package cache. [" << _config
->FindFile("Dir::Cache::pkgcache") << ']' << endl
;
476 cout
<< " -s=? The source cache. [" << _config
->FindFile("Dir::Cache::srcpkgcache") << ']' << endl
;
477 cout
<< " -q Disable progress indicator." << endl
;
478 cout
<< " -i Show only important deps for the unmet command." << endl
;
479 cout
<< " -c=? Read this configuration file" << endl
;
480 cout
<< " -o=? Set an arbitary configuration option, ie -o dir::cache=/tmp" << endl
;
481 cout
<< "See the apt-cache(8) and apt.conf(5) manual pages for more information." << endl
;
485 // CacheInitialize - Initialize things for apt-cache /*{{{*/
486 // ---------------------------------------------------------------------
488 void CacheInitialize()
490 _config
->Set("quiet",0);
491 _config
->Set("help",false);
495 int main(int argc
,const char *argv
[])
497 CommandLine::Args Args
[] = {
498 {'h',"help","help",0},
499 {'v',"version","version",0},
500 {'p',"pkg-cache","Dir::Cache::pkgcache",CommandLine::HasArg
},
501 {'s',"src-cache","Dir::Cache::srcpkgcache",CommandLine::HasArg
},
502 {'q',"quiet","quiet",CommandLine::IntLevel
},
503 {'i',"important","APT::Cache::Important",0},
504 {'c',"config-file",0,CommandLine::ConfigFile
},
505 {'o',"option",0,CommandLine::ArbItem
},
507 CommandLine::Dispatch CmdsA
[] = {{"help",&ShowHelp
},
509 {"gencaches",&GenCaches
},
511 CommandLine::Dispatch CmdsB
[] = {{"showpkg",&DumpPackage
},
514 {"dumpavail",&DumpAvail
},
521 // Parse the command line and initialize the package library
522 CommandLine
CmdL(Args
,_config
);
523 if (pkgInitialize(*_config
) == false ||
524 CmdL
.Parse(argc
,argv
) == false)
526 _error
->DumpErrors();
530 // See if the help should be shown
531 if (_config
->FindB("help") == true ||
532 CmdL
.FileSize() == 0)
533 return ShowHelp(CmdL
);
535 if (CmdL
.DispatchArg(CmdsA
,false) == false && _error
->PendingError() == false)
537 // Open the cache file
538 FileFd
CacheF(_config
->FindFile("Dir::Cache::pkgcache"),FileFd::ReadOnly
);
539 MMap
Map(CacheF
,MMap::Public
| MMap::ReadOnly
);
540 if (_error
->PendingError() == false)
544 if (_error
->PendingError() == false)
545 CmdL
.DispatchArg(CmdsB
);
549 // Print any errors or warnings found during parsing
550 if (_error
->empty() == false)
552 bool Errors
= _error
->PendingError();
553 _error
->DumpErrors();
554 return Errors
== true?100:0;