]> git.saurik.com Git - apt.git/blob - cmdline/apt-get.cc
9f9ecd37524cc551f3db788e4a9389e867112302
[apt.git] / cmdline / apt-get.cc
1 // -*- mode: cpp; mode: fold -*-
2 // Description /*{{{*/
3 // $Id: apt-get.cc,v 1.156 2004/08/28 01:05:16 mdz Exp $
4 /* ######################################################################
5
6 apt-get - Cover for dpkg
7
8 This is an allout cover for dpkg implementing a safer front end. It is
9 based largely on libapt-pkg.
10
11 The syntax is different,
12 apt-get [opt] command [things]
13 Where command is:
14 update - Resyncronize the package files from their sources
15 upgrade - Smart-Download the newest versions of all packages
16 dselect-upgrade - Follows dselect's changes to the Status: field
17 and installes new and removes old packages
18 dist-upgrade - Powerfull upgrader designed to handle the issues with
19 a new distribution.
20 install - Download and install a given package (by name, not by .deb)
21 check - Update the package cache and check for broken packages
22 clean - Erase the .debs downloaded to /var/cache/apt/archives and
23 the partial dir too
24
25 ##################################################################### */
26 /*}}}*/
27 // Include Files /*{{{*/
28 #include <apt-pkg/error.h>
29 #include <apt-pkg/cmndline.h>
30 #include <apt-pkg/init.h>
31 #include <apt-pkg/depcache.h>
32 #include <apt-pkg/sourcelist.h>
33 #include <apt-pkg/algorithms.h>
34 #include <apt-pkg/acquire-item.h>
35 #include <apt-pkg/strutl.h>
36 #include <apt-pkg/clean.h>
37 #include <apt-pkg/srcrecords.h>
38 #include <apt-pkg/version.h>
39 #include <apt-pkg/cachefile.h>
40 #include <apt-pkg/sptr.h>
41 #include <apt-pkg/versionmatch.h>
42
43 #include <config.h>
44 #include <apti18n.h>
45
46 #include "acqprogress.h"
47
48 #include <locale.h>
49 #include <langinfo.h>
50 #include <fstream>
51 #include <termios.h>
52 #include <sys/ioctl.h>
53 #include <sys/stat.h>
54 #include <sys/statvfs.h>
55 #include <signal.h>
56 #include <unistd.h>
57 #include <stdio.h>
58 #include <errno.h>
59 #include <regex.h>
60 #include <sys/wait.h>
61 #include <sstream>
62 /*}}}*/
63
64 using namespace std;
65
66 ostream c0out(0);
67 ostream c1out(0);
68 ostream c2out(0);
69 ofstream devnull("/dev/null");
70 unsigned int ScreenWidth = 80;
71
72 // class CacheFile - Cover class for some dependency cache functions /*{{{*/
73 // ---------------------------------------------------------------------
74 /* */
75 class CacheFile : public pkgCacheFile
76 {
77 static pkgCache *SortCache;
78 static int NameComp(const void *a,const void *b);
79
80 public:
81 pkgCache::Package **List;
82
83 void Sort();
84 bool CheckDeps(bool AllowBroken = false);
85 bool BuildCaches(bool WithLock = true)
86 {
87 OpTextProgress Prog(*_config);
88 if (pkgCacheFile::BuildCaches(Prog,WithLock) == false)
89 return false;
90 return true;
91 }
92 bool Open(bool WithLock = true)
93 {
94 OpTextProgress Prog(*_config);
95 if (pkgCacheFile::Open(Prog,WithLock) == false)
96 return false;
97 Sort();
98
99 return true;
100 };
101 bool OpenForInstall()
102 {
103 if (_config->FindB("APT::Get::Print-URIs") == true)
104 return Open(false);
105 else
106 return Open(true);
107 }
108 CacheFile() : List(0) {};
109 };
110 /*}}}*/
111
112 // YnPrompt - Yes No Prompt. /*{{{*/
113 // ---------------------------------------------------------------------
114 /* Returns true on a Yes.*/
115 bool YnPrompt(bool Default=true)
116 {
117 if (_config->FindB("APT::Get::Assume-Yes",false) == true)
118 {
119 c1out << _("Y") << endl;
120 return true;
121 }
122
123 char response[1024] = "";
124 cin.getline(response, sizeof(response));
125
126 if (!cin)
127 return false;
128
129 if (strlen(response) == 0)
130 return Default;
131
132 regex_t Pattern;
133 int Res;
134
135 Res = regcomp(&Pattern, nl_langinfo(YESEXPR),
136 REG_EXTENDED|REG_ICASE|REG_NOSUB);
137
138 if (Res != 0) {
139 char Error[300];
140 regerror(Res,&Pattern,Error,sizeof(Error));
141 return _error->Error(_("Regex compilation error - %s"),Error);
142 }
143
144 Res = regexec(&Pattern, response, 0, NULL, 0);
145 if (Res == 0)
146 return true;
147 return false;
148 }
149 /*}}}*/
150 // AnalPrompt - Annoying Yes No Prompt. /*{{{*/
151 // ---------------------------------------------------------------------
152 /* Returns true on a Yes.*/
153 bool AnalPrompt(const char *Text)
154 {
155 char Buf[1024];
156 cin.getline(Buf,sizeof(Buf));
157 if (strcmp(Buf,Text) == 0)
158 return true;
159 return false;
160 }
161 /*}}}*/
162 // ShowList - Show a list /*{{{*/
163 // ---------------------------------------------------------------------
164 /* This prints out a string of space separated words with a title and
165 a two space indent line wraped to the current screen width. */
166 bool ShowList(ostream &out,string Title,string List,string VersionsList)
167 {
168 if (List.empty() == true)
169 return true;
170 // trim trailing space
171 int NonSpace = List.find_last_not_of(' ');
172 if (NonSpace != -1)
173 {
174 List = List.erase(NonSpace + 1);
175 if (List.empty() == true)
176 return true;
177 }
178
179 // Acount for the leading space
180 int ScreenWidth = ::ScreenWidth - 3;
181
182 out << Title << endl;
183 string::size_type Start = 0;
184 string::size_type VersionsStart = 0;
185 while (Start < List.size())
186 {
187 if(_config->FindB("APT::Get::Show-Versions",false) == true &&
188 VersionsList.size() > 0) {
189 string::size_type End;
190 string::size_type VersionsEnd;
191
192 End = List.find(' ',Start);
193 VersionsEnd = VersionsList.find('\n', VersionsStart);
194
195 out << " " << string(List,Start,End - Start) << " (" <<
196 string(VersionsList,VersionsStart,VersionsEnd - VersionsStart) <<
197 ")" << endl;
198
199 if (End == string::npos || End < Start)
200 End = Start + ScreenWidth;
201
202 Start = End + 1;
203 VersionsStart = VersionsEnd + 1;
204 } else {
205 string::size_type End;
206
207 if (Start + ScreenWidth >= List.size())
208 End = List.size();
209 else
210 End = List.rfind(' ',Start+ScreenWidth);
211
212 if (End == string::npos || End < Start)
213 End = Start + ScreenWidth;
214 out << " " << string(List,Start,End - Start) << endl;
215 Start = End + 1;
216 }
217 }
218
219 return false;
220 }
221 /*}}}*/
222 // ShowBroken - Debugging aide /*{{{*/
223 // ---------------------------------------------------------------------
224 /* This prints out the names of all the packages that are broken along
225 with the name of each each broken dependency and a quite version
226 description.
227
228 The output looks like:
229 The following packages have unmet dependencies:
230 exim: Depends: libc6 (>= 2.1.94) but 2.1.3-10 is to be installed
231 Depends: libldap2 (>= 2.0.2-2) but it is not going to be installed
232 Depends: libsasl7 but it is not going to be installed
233 */
234 void ShowBroken(ostream &out,CacheFile &Cache,bool Now)
235 {
236 out << _("The following packages have unmet dependencies:") << endl;
237 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
238 {
239 pkgCache::PkgIterator I(Cache,Cache.List[J]);
240
241 if (Now == true)
242 {
243 if (Cache[I].NowBroken() == false)
244 continue;
245 }
246 else
247 {
248 if (Cache[I].InstBroken() == false)
249 continue;
250 }
251
252 // Print out each package and the failed dependencies
253 out <<" " << I.Name() << ":";
254 unsigned Indent = strlen(I.Name()) + 3;
255 bool First = true;
256 pkgCache::VerIterator Ver;
257
258 if (Now == true)
259 Ver = I.CurrentVer();
260 else
261 Ver = Cache[I].InstVerIter(Cache);
262
263 if (Ver.end() == true)
264 {
265 out << endl;
266 continue;
267 }
268
269 for (pkgCache::DepIterator D = Ver.DependsList(); D.end() == false;)
270 {
271 // Compute a single dependency element (glob or)
272 pkgCache::DepIterator Start;
273 pkgCache::DepIterator End;
274 D.GlobOr(Start,End); // advances D
275
276 if (Cache->IsImportantDep(End) == false)
277 continue;
278
279 if (Now == true)
280 {
281 if ((Cache[End] & pkgDepCache::DepGNow) == pkgDepCache::DepGNow)
282 continue;
283 }
284 else
285 {
286 if ((Cache[End] & pkgDepCache::DepGInstall) == pkgDepCache::DepGInstall)
287 continue;
288 }
289
290 bool FirstOr = true;
291 while (1)
292 {
293 if (First == false)
294 for (unsigned J = 0; J != Indent; J++)
295 out << ' ';
296 First = false;
297
298 if (FirstOr == false)
299 {
300 for (unsigned J = 0; J != strlen(End.DepType()) + 3; J++)
301 out << ' ';
302 }
303 else
304 out << ' ' << End.DepType() << ": ";
305 FirstOr = false;
306
307 out << Start.TargetPkg().Name();
308
309 // Show a quick summary of the version requirements
310 if (Start.TargetVer() != 0)
311 out << " (" << Start.CompType() << " " << Start.TargetVer() << ")";
312
313 /* Show a summary of the target package if possible. In the case
314 of virtual packages we show nothing */
315 pkgCache::PkgIterator Targ = Start.TargetPkg();
316 if (Targ->ProvidesList == 0)
317 {
318 out << ' ';
319 pkgCache::VerIterator Ver = Cache[Targ].InstVerIter(Cache);
320 if (Now == true)
321 Ver = Targ.CurrentVer();
322
323 if (Ver.end() == false)
324 {
325 if (Now == true)
326 ioprintf(out,_("but %s is installed"),Ver.VerStr());
327 else
328 ioprintf(out,_("but %s is to be installed"),Ver.VerStr());
329 }
330 else
331 {
332 if (Cache[Targ].CandidateVerIter(Cache).end() == true)
333 {
334 if (Targ->ProvidesList == 0)
335 out << _("but it is not installable");
336 else
337 out << _("but it is a virtual package");
338 }
339 else
340 out << (Now?_("but it is not installed"):_("but it is not going to be installed"));
341 }
342 }
343
344 if (Start != End)
345 out << _(" or");
346 out << endl;
347
348 if (Start == End)
349 break;
350 Start++;
351 }
352 }
353 }
354 }
355 /*}}}*/
356 // ShowNew - Show packages to newly install /*{{{*/
357 // ---------------------------------------------------------------------
358 /* */
359 void ShowNew(ostream &out,CacheFile &Cache)
360 {
361 /* Print out a list of packages that are going to be installed extra
362 to what the user asked */
363 string List;
364 string VersionsList;
365 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
366 {
367 pkgCache::PkgIterator I(Cache,Cache.List[J]);
368 if (Cache[I].NewInstall() == true) {
369 List += string(I.Name()) + " ";
370 VersionsList += string(Cache[I].CandVersion) + "\n";
371 }
372 }
373
374 ShowList(out,_("The following NEW packages will be installed:"),List,VersionsList);
375 }
376 /*}}}*/
377 // ShowDel - Show packages to delete /*{{{*/
378 // ---------------------------------------------------------------------
379 /* */
380 void ShowDel(ostream &out,CacheFile &Cache)
381 {
382 /* Print out a list of packages that are going to be removed extra
383 to what the user asked */
384 string List;
385 string VersionsList;
386 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
387 {
388 pkgCache::PkgIterator I(Cache,Cache.List[J]);
389 if (Cache[I].Delete() == true)
390 {
391 if ((Cache[I].iFlags & pkgDepCache::Purge) == pkgDepCache::Purge)
392 List += string(I.Name()) + "* ";
393 else
394 List += string(I.Name()) + " ";
395
396 VersionsList += string(Cache[I].CandVersion)+ "\n";
397 }
398 }
399
400 ShowList(out,_("The following packages will be REMOVED:"),List,VersionsList);
401 }
402 /*}}}*/
403 // ShowKept - Show kept packages /*{{{*/
404 // ---------------------------------------------------------------------
405 /* */
406 void ShowKept(ostream &out,CacheFile &Cache)
407 {
408 string List;
409 string VersionsList;
410 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
411 {
412 pkgCache::PkgIterator I(Cache,Cache.List[J]);
413
414 // Not interesting
415 if (Cache[I].Upgrade() == true || Cache[I].Upgradable() == false ||
416 I->CurrentVer == 0 || Cache[I].Delete() == true)
417 continue;
418
419 List += string(I.Name()) + " ";
420 VersionsList += string(Cache[I].CurVersion) + " => " + Cache[I].CandVersion + "\n";
421 }
422 ShowList(out,_("The following packages have been kept back:"),List,VersionsList);
423 }
424 /*}}}*/
425 // ShowUpgraded - Show upgraded packages /*{{{*/
426 // ---------------------------------------------------------------------
427 /* */
428 void ShowUpgraded(ostream &out,CacheFile &Cache)
429 {
430 string List;
431 string VersionsList;
432 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
433 {
434 pkgCache::PkgIterator I(Cache,Cache.List[J]);
435
436 // Not interesting
437 if (Cache[I].Upgrade() == false || Cache[I].NewInstall() == true)
438 continue;
439
440 List += string(I.Name()) + " ";
441 VersionsList += string(Cache[I].CurVersion) + " => " + Cache[I].CandVersion + "\n";
442 }
443 ShowList(out,_("The following packages will be upgraded:"),List,VersionsList);
444 }
445 /*}}}*/
446 // ShowDowngraded - Show downgraded packages /*{{{*/
447 // ---------------------------------------------------------------------
448 /* */
449 bool ShowDowngraded(ostream &out,CacheFile &Cache)
450 {
451 string List;
452 string VersionsList;
453 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
454 {
455 pkgCache::PkgIterator I(Cache,Cache.List[J]);
456
457 // Not interesting
458 if (Cache[I].Downgrade() == false || Cache[I].NewInstall() == true)
459 continue;
460
461 List += string(I.Name()) + " ";
462 VersionsList += string(Cache[I].CurVersion) + " => " + Cache[I].CandVersion + "\n";
463 }
464 return ShowList(out,_("The following packages will be DOWNGRADED:"),List,VersionsList);
465 }
466 /*}}}*/
467 // ShowHold - Show held but changed packages /*{{{*/
468 // ---------------------------------------------------------------------
469 /* */
470 bool ShowHold(ostream &out,CacheFile &Cache)
471 {
472 string List;
473 string VersionsList;
474 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
475 {
476 pkgCache::PkgIterator I(Cache,Cache.List[J]);
477 if (Cache[I].InstallVer != (pkgCache::Version *)I.CurrentVer() &&
478 I->SelectedState == pkgCache::State::Hold) {
479 List += string(I.Name()) + " ";
480 VersionsList += string(Cache[I].CurVersion) + " => " + Cache[I].CandVersion + "\n";
481 }
482 }
483
484 return ShowList(out,_("The following held packages will be changed:"),List,VersionsList);
485 }
486 /*}}}*/
487 // ShowEssential - Show an essential package warning /*{{{*/
488 // ---------------------------------------------------------------------
489 /* This prints out a warning message that is not to be ignored. It shows
490 all essential packages and their dependents that are to be removed.
491 It is insanely risky to remove the dependents of an essential package! */
492 bool ShowEssential(ostream &out,CacheFile &Cache)
493 {
494 string List;
495 string VersionsList;
496 bool *Added = new bool[Cache->Head().PackageCount];
497 for (unsigned int I = 0; I != Cache->Head().PackageCount; I++)
498 Added[I] = false;
499
500 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
501 {
502 pkgCache::PkgIterator I(Cache,Cache.List[J]);
503 if ((I->Flags & pkgCache::Flag::Essential) != pkgCache::Flag::Essential &&
504 (I->Flags & pkgCache::Flag::Important) != pkgCache::Flag::Important)
505 continue;
506
507 // The essential package is being removed
508 if (Cache[I].Delete() == true)
509 {
510 if (Added[I->ID] == false)
511 {
512 Added[I->ID] = true;
513 List += string(I.Name()) + " ";
514 //VersionsList += string(Cache[I].CurVersion) + "\n"; ???
515 }
516 }
517
518 if (I->CurrentVer == 0)
519 continue;
520
521 // Print out any essential package depenendents that are to be removed
522 for (pkgCache::DepIterator D = I.CurrentVer().DependsList(); D.end() == false; D++)
523 {
524 // Skip everything but depends
525 if (D->Type != pkgCache::Dep::PreDepends &&
526 D->Type != pkgCache::Dep::Depends)
527 continue;
528
529 pkgCache::PkgIterator P = D.SmartTargetPkg();
530 if (Cache[P].Delete() == true)
531 {
532 if (Added[P->ID] == true)
533 continue;
534 Added[P->ID] = true;
535
536 char S[300];
537 snprintf(S,sizeof(S),_("%s (due to %s) "),P.Name(),I.Name());
538 List += S;
539 //VersionsList += "\n"; ???
540 }
541 }
542 }
543
544 delete [] Added;
545 return ShowList(out,_("WARNING: The following essential packages will be removed\n"
546 "This should NOT be done unless you know exactly what you are doing!"),List,VersionsList);
547 }
548
549 /*}}}*/
550 // Stats - Show some statistics /*{{{*/
551 // ---------------------------------------------------------------------
552 /* */
553 void Stats(ostream &out,pkgDepCache &Dep)
554 {
555 unsigned long Upgrade = 0;
556 unsigned long Downgrade = 0;
557 unsigned long Install = 0;
558 unsigned long ReInstall = 0;
559 for (pkgCache::PkgIterator I = Dep.PkgBegin(); I.end() == false; I++)
560 {
561 if (Dep[I].NewInstall() == true)
562 Install++;
563 else
564 {
565 if (Dep[I].Upgrade() == true)
566 Upgrade++;
567 else
568 if (Dep[I].Downgrade() == true)
569 Downgrade++;
570 }
571
572 if (Dep[I].Delete() == false && (Dep[I].iFlags & pkgDepCache::ReInstall) == pkgDepCache::ReInstall)
573 ReInstall++;
574 }
575
576 ioprintf(out,_("%lu upgraded, %lu newly installed, "),
577 Upgrade,Install);
578
579 if (ReInstall != 0)
580 ioprintf(out,_("%lu reinstalled, "),ReInstall);
581 if (Downgrade != 0)
582 ioprintf(out,_("%lu downgraded, "),Downgrade);
583
584 ioprintf(out,_("%lu to remove and %lu not upgraded.\n"),
585 Dep.DelCount(),Dep.KeepCount());
586
587 if (Dep.BadCount() != 0)
588 ioprintf(out,_("%lu not fully installed or removed.\n"),
589 Dep.BadCount());
590 }
591 /*}}}*/
592
593 // CacheFile::NameComp - QSort compare by name /*{{{*/
594 // ---------------------------------------------------------------------
595 /* */
596 pkgCache *CacheFile::SortCache = 0;
597 int CacheFile::NameComp(const void *a,const void *b)
598 {
599 if (*(pkgCache::Package **)a == 0 || *(pkgCache::Package **)b == 0)
600 return *(pkgCache::Package **)a - *(pkgCache::Package **)b;
601
602 const pkgCache::Package &A = **(pkgCache::Package **)a;
603 const pkgCache::Package &B = **(pkgCache::Package **)b;
604
605 return strcmp(SortCache->StrP + A.Name,SortCache->StrP + B.Name);
606 }
607 /*}}}*/
608 // CacheFile::Sort - Sort by name /*{{{*/
609 // ---------------------------------------------------------------------
610 /* */
611 void CacheFile::Sort()
612 {
613 delete [] List;
614 List = new pkgCache::Package *[Cache->Head().PackageCount];
615 memset(List,0,sizeof(*List)*Cache->Head().PackageCount);
616 pkgCache::PkgIterator I = Cache->PkgBegin();
617 for (;I.end() != true; I++)
618 List[I->ID] = I;
619
620 SortCache = *this;
621 qsort(List,Cache->Head().PackageCount,sizeof(*List),NameComp);
622 }
623 /*}}}*/
624 // CacheFile::CheckDeps - Open the cache file /*{{{*/
625 // ---------------------------------------------------------------------
626 /* This routine generates the caches and then opens the dependency cache
627 and verifies that the system is OK. */
628 bool CacheFile::CheckDeps(bool AllowBroken)
629 {
630 if (_error->PendingError() == true)
631 return false;
632
633 // Check that the system is OK
634 if (DCache->DelCount() != 0 || DCache->InstCount() != 0)
635 return _error->Error("Internal error, non-zero counts");
636
637 // Apply corrections for half-installed packages
638 if (pkgApplyStatus(*DCache) == false)
639 return false;
640
641 // Nothing is broken
642 if (DCache->BrokenCount() == 0 || AllowBroken == true)
643 return true;
644
645 // Attempt to fix broken things
646 if (_config->FindB("APT::Get::Fix-Broken",false) == true)
647 {
648 c1out << _("Correcting dependencies...") << flush;
649 if (pkgFixBroken(*DCache) == false || DCache->BrokenCount() != 0)
650 {
651 c1out << _(" failed.") << endl;
652 ShowBroken(c1out,*this,true);
653
654 return _error->Error(_("Unable to correct dependencies"));
655 }
656 if (pkgMinimizeUpgrade(*DCache) == false)
657 return _error->Error(_("Unable to minimize the upgrade set"));
658
659 c1out << _(" Done") << endl;
660 }
661 else
662 {
663 c1out << _("You might want to run `apt-get -f install' to correct these.") << endl;
664 ShowBroken(c1out,*this,true);
665
666 return _error->Error(_("Unmet dependencies. Try using -f."));
667 }
668
669 return true;
670 }
671
672 static bool CheckAuth(pkgAcquire& Fetcher)
673 {
674 string UntrustedList;
675 for (pkgAcquire::ItemIterator I = Fetcher.ItemsBegin(); I < Fetcher.ItemsEnd(); ++I)
676 {
677 if (!(*I)->IsTrusted())
678 {
679 UntrustedList += string((*I)->ShortDesc()) + " ";
680 }
681 }
682
683 if (UntrustedList == "")
684 {
685 return true;
686 }
687
688 ShowList(c2out,_("WARNING: The following packages cannot be authenticated!"),UntrustedList,"");
689
690 if (_config->FindB("APT::Get::AllowUnauthenticated",false) == true)
691 {
692 c2out << "Authentication warning overridden.\n";
693 return true;
694 }
695
696 if (_config->FindI("quiet",0) < 2
697 && _config->FindB("APT::Get::Assume-Yes",false) == false)
698 {
699 c2out << _("Install these packages without verification [y/N]? ") << flush;
700 if (!YnPrompt(false))
701 return _error->Error(_("Some packages could not be authenticated"));
702
703 return true;
704 }
705 else if (_config->FindB("APT::Get::Force-Yes",false) == true)
706 {
707 return true;
708 }
709
710 return _error->Error(_("There are problems and -y was used without --force-yes"));
711 }
712
713
714 /*}}}*/
715
716 // InstallPackages - Actually download and install the packages /*{{{*/
717 // ---------------------------------------------------------------------
718 /* This displays the informative messages describing what is going to
719 happen and then calls the download routines */
720 bool InstallPackages(CacheFile &Cache,bool ShwKept,bool Ask = true,
721 bool Safety = true)
722 {
723 if (_config->FindB("APT::Get::Purge",false) == true)
724 {
725 pkgCache::PkgIterator I = Cache->PkgBegin();
726 for (; I.end() == false; I++)
727 {
728 if (I.Purge() == false && Cache[I].Mode == pkgDepCache::ModeDelete)
729 Cache->MarkDelete(I,true);
730 }
731 }
732
733 bool Fail = false;
734 bool Essential = false;
735
736 // Show all the various warning indicators
737 ShowDel(c1out,Cache);
738 ShowNew(c1out,Cache);
739 if (ShwKept == true)
740 ShowKept(c1out,Cache);
741 Fail |= !ShowHold(c1out,Cache);
742 if (_config->FindB("APT::Get::Show-Upgraded",true) == true)
743 ShowUpgraded(c1out,Cache);
744 Fail |= !ShowDowngraded(c1out,Cache);
745 if (_config->FindB("APT::Get::Download-Only",false) == false)
746 Essential = !ShowEssential(c1out,Cache);
747 Fail |= Essential;
748 Stats(c1out,Cache);
749
750 // Sanity check
751 if (Cache->BrokenCount() != 0)
752 {
753 ShowBroken(c1out,Cache,false);
754 return _error->Error("Internal error, InstallPackages was called with broken packages!");
755 }
756
757 if (Cache->DelCount() == 0 && Cache->InstCount() == 0 &&
758 Cache->BadCount() == 0)
759 return true;
760
761 // No remove flag
762 if (Cache->DelCount() != 0 && _config->FindB("APT::Get::Remove",true) == false)
763 return _error->Error(_("Packages need to be removed but remove is disabled."));
764
765 // Run the simulator ..
766 if (_config->FindB("APT::Get::Simulate") == true)
767 {
768 pkgSimulate PM(Cache);
769 pkgPackageManager::OrderResult Res = PM.DoInstall();
770 if (Res == pkgPackageManager::Failed)
771 return false;
772 if (Res != pkgPackageManager::Completed)
773 return _error->Error("Internal error, Ordering didn't finish");
774 return true;
775 }
776
777 // Create the text record parser
778 pkgRecords Recs(Cache);
779 if (_error->PendingError() == true)
780 return false;
781
782 // Lock the archive directory
783 FileFd Lock;
784 if (_config->FindB("Debug::NoLocking",false) == false &&
785 _config->FindB("APT::Get::Print-URIs") == false)
786 {
787 Lock.Fd(GetLock(_config->FindDir("Dir::Cache::Archives") + "lock"));
788 if (_error->PendingError() == true)
789 return _error->Error(_("Unable to lock the download directory"));
790 }
791
792 // Create the download object
793 AcqTextStatus Stat(ScreenWidth,_config->FindI("quiet",0));
794 pkgAcquire Fetcher(&Stat);
795
796 // Read the source list
797 pkgSourceList List;
798 if (List.ReadMainList() == false)
799 return _error->Error(_("The list of sources could not be read."));
800
801 // Create the package manager and prepare to download
802 SPtr<pkgPackageManager> PM= _system->CreatePM(Cache);
803 if (PM->GetArchives(&Fetcher,&List,&Recs) == false ||
804 _error->PendingError() == true)
805 return false;
806
807 // Display statistics
808 double FetchBytes = Fetcher.FetchNeeded();
809 double FetchPBytes = Fetcher.PartialPresent();
810 double DebBytes = Fetcher.TotalNeeded();
811 if (DebBytes != Cache->DebSize())
812 {
813 c0out << DebBytes << ',' << Cache->DebSize() << endl;
814 c0out << "How odd.. The sizes didn't match, email apt@packages.debian.org" << endl;
815 }
816
817 // Number of bytes
818 if (DebBytes != FetchBytes)
819 ioprintf(c1out,_("Need to get %sB/%sB of archives.\n"),
820 SizeToStr(FetchBytes).c_str(),SizeToStr(DebBytes).c_str());
821 else
822 ioprintf(c1out,_("Need to get %sB of archives.\n"),
823 SizeToStr(DebBytes).c_str());
824
825 // Size delta
826 if (Cache->UsrSize() >= 0)
827 ioprintf(c1out,_("After unpacking %sB of additional disk space will be used.\n"),
828 SizeToStr(Cache->UsrSize()).c_str());
829 else
830 ioprintf(c1out,_("After unpacking %sB disk space will be freed.\n"),
831 SizeToStr(-1*Cache->UsrSize()).c_str());
832
833 if (_error->PendingError() == true)
834 return false;
835
836 /* Check for enough free space, but only if we are actually going to
837 download */
838 if (_config->FindB("APT::Get::Print-URIs") == false &&
839 _config->FindB("APT::Get::Download",true) == true)
840 {
841 struct statvfs Buf;
842 string OutputDir = _config->FindDir("Dir::Cache::Archives");
843 if (statvfs(OutputDir.c_str(),&Buf) != 0)
844 return _error->Errno("statvfs","Couldn't determine free space in %s",
845 OutputDir.c_str());
846 if (unsigned(Buf.f_bfree) < (FetchBytes - FetchPBytes)/Buf.f_bsize)
847 return _error->Error(_("You don't have enough free space in %s."),
848 OutputDir.c_str());
849 }
850
851 // Fail safe check
852 if (_config->FindI("quiet",0) >= 2 ||
853 _config->FindB("APT::Get::Assume-Yes",false) == true)
854 {
855 if (Fail == true && _config->FindB("APT::Get::Force-Yes",false) == false)
856 return _error->Error(_("There are problems and -y was used without --force-yes"));
857 }
858
859 if (Essential == true && Safety == true)
860 {
861 if (_config->FindB("APT::Get::Trivial-Only",false) == true)
862 return _error->Error(_("Trivial Only specified but this is not a trivial operation."));
863
864 const char *Prompt = _("Yes, do as I say!");
865 ioprintf(c2out,
866 _("You are about to do something potentially harmful\n"
867 "To continue type in the phrase '%s'\n"
868 " ?] "),Prompt);
869 c2out << flush;
870 if (AnalPrompt(Prompt) == false)
871 {
872 c2out << _("Abort.") << endl;
873 exit(1);
874 }
875 }
876 else
877 {
878 // Prompt to continue
879 if (Ask == true || Fail == true)
880 {
881 if (_config->FindB("APT::Get::Trivial-Only",false) == true)
882 return _error->Error(_("Trivial Only specified but this is not a trivial operation."));
883
884 if (_config->FindI("quiet",0) < 2 &&
885 _config->FindB("APT::Get::Assume-Yes",false) == false)
886 {
887 c2out << _("Do you want to continue [Y/n]? ") << flush;
888
889 if (YnPrompt() == false)
890 {
891 c2out << _("Abort.") << endl;
892 exit(1);
893 }
894 }
895 }
896 }
897
898 // Just print out the uris an exit if the --print-uris flag was used
899 if (_config->FindB("APT::Get::Print-URIs") == true)
900 {
901 pkgAcquire::UriIterator I = Fetcher.UriBegin();
902 for (; I != Fetcher.UriEnd(); I++)
903 cout << '\'' << I->URI << "' " << flNotDir(I->Owner->DestFile) << ' ' <<
904 I->Owner->FileSize << ' ' << I->Owner->MD5Sum() << endl;
905 return true;
906 }
907
908 if (!CheckAuth(Fetcher))
909 return false;
910
911 /* Unlock the dpkg lock if we are not going to be doing an install
912 after. */
913 if (_config->FindB("APT::Get::Download-Only",false) == true)
914 _system->UnLock();
915
916 // Run it
917 while (1)
918 {
919 bool Transient = false;
920 if (_config->FindB("APT::Get::Download",true) == false)
921 {
922 for (pkgAcquire::ItemIterator I = Fetcher.ItemsBegin(); I < Fetcher.ItemsEnd();)
923 {
924 if ((*I)->Local == true)
925 {
926 I++;
927 continue;
928 }
929
930 // Close the item and check if it was found in cache
931 (*I)->Finished();
932 if ((*I)->Complete == false)
933 Transient = true;
934
935 // Clear it out of the fetch list
936 delete *I;
937 I = Fetcher.ItemsBegin();
938 }
939 }
940
941 if (Fetcher.Run() == pkgAcquire::Failed)
942 return false;
943
944 // Print out errors
945 bool Failed = false;
946 for (pkgAcquire::ItemIterator I = Fetcher.ItemsBegin(); I != Fetcher.ItemsEnd(); I++)
947 {
948 if ((*I)->Status == pkgAcquire::Item::StatDone &&
949 (*I)->Complete == true)
950 continue;
951
952 if ((*I)->Status == pkgAcquire::Item::StatIdle)
953 {
954 Transient = true;
955 // Failed = true;
956 continue;
957 }
958
959 fprintf(stderr,_("Failed to fetch %s %s\n"),(*I)->DescURI().c_str(),
960 (*I)->ErrorText.c_str());
961 Failed = true;
962 }
963
964 /* If we are in no download mode and missing files and there were
965 'failures' then the user must specify -m. Furthermore, there
966 is no such thing as a transient error in no-download mode! */
967 if (Transient == true &&
968 _config->FindB("APT::Get::Download",true) == false)
969 {
970 Transient = false;
971 Failed = true;
972 }
973
974 if (_config->FindB("APT::Get::Download-Only",false) == true)
975 {
976 if (Failed == true && _config->FindB("APT::Get::Fix-Missing",false) == false)
977 return _error->Error(_("Some files failed to download"));
978 c1out << _("Download complete and in download only mode") << endl;
979 return true;
980 }
981
982 if (Failed == true && _config->FindB("APT::Get::Fix-Missing",false) == false)
983 {
984 return _error->Error(_("Unable to fetch some archives, maybe run apt-get update or try with --fix-missing?"));
985 }
986
987 if (Transient == true && Failed == true)
988 return _error->Error(_("--fix-missing and media swapping is not currently supported"));
989
990 // Try to deal with missing package files
991 if (Failed == true && PM->FixMissing() == false)
992 {
993 cerr << _("Unable to correct missing packages.") << endl;
994 return _error->Error(_("Aborting install."));
995 }
996
997 // -- we do it here because there is no libapt::Commit() :/
998 Cache->writeStateFile(NULL);
999
1000 _system->UnLock();
1001 pkgPackageManager::OrderResult Res = PM->DoInstall();
1002 if (Res == pkgPackageManager::Failed || _error->PendingError() == true)
1003 return false;
1004 if (Res == pkgPackageManager::Completed)
1005 return true;
1006
1007 // Reload the fetcher object and loop again for media swapping
1008 Fetcher.Shutdown();
1009 if (PM->GetArchives(&Fetcher,&List,&Recs) == false)
1010 return false;
1011
1012 _system->Lock();
1013 }
1014 }
1015 /*}}}*/
1016 // TryToInstall - Try to install a single package /*{{{*/
1017 // ---------------------------------------------------------------------
1018 /* This used to be inlined in DoInstall, but with the advent of regex package
1019 name matching it was split out.. */
1020 bool TryToInstall(pkgCache::PkgIterator Pkg,pkgDepCache &Cache,
1021 pkgProblemResolver &Fix,bool Remove,bool BrokenFix,
1022 unsigned int &ExpectedInst,bool AllowFail = true)
1023 {
1024 /* This is a pure virtual package and there is a single available
1025 provides */
1026 if (Cache[Pkg].CandidateVer == 0 && Pkg->ProvidesList != 0 &&
1027 Pkg.ProvidesList()->NextProvides == 0)
1028 {
1029 pkgCache::PkgIterator Tmp = Pkg.ProvidesList().OwnerPkg();
1030 ioprintf(c1out,_("Note, selecting %s instead of %s\n"),
1031 Tmp.Name(),Pkg.Name());
1032 Pkg = Tmp;
1033 }
1034
1035 // Handle the no-upgrade case
1036 if (_config->FindB("APT::Get::upgrade",true) == false &&
1037 Pkg->CurrentVer != 0)
1038 {
1039 if (AllowFail == true)
1040 ioprintf(c1out,_("Skipping %s, it is already installed and upgrade is not set.\n"),
1041 Pkg.Name());
1042 return true;
1043 }
1044
1045 // Check if there is something at all to install
1046 pkgDepCache::StateCache &State = Cache[Pkg];
1047 if (Remove == true && Pkg->CurrentVer == 0)
1048 {
1049 Fix.Clear(Pkg);
1050 Fix.Protect(Pkg);
1051 Fix.Remove(Pkg);
1052
1053 /* We want to continue searching for regex hits, so we return false here
1054 otherwise this is not really an error. */
1055 if (AllowFail == false)
1056 return false;
1057
1058 ioprintf(c1out,_("Package %s is not installed, so not removed\n"),Pkg.Name());
1059 return true;
1060 }
1061
1062 if (State.CandidateVer == 0 && Remove == false)
1063 {
1064 if (AllowFail == false)
1065 return false;
1066
1067 if (Pkg->ProvidesList != 0)
1068 {
1069 ioprintf(c1out,_("Package %s is a virtual package provided by:\n"),
1070 Pkg.Name());
1071
1072 pkgCache::PrvIterator I = Pkg.ProvidesList();
1073 for (; I.end() == false; I++)
1074 {
1075 pkgCache::PkgIterator Pkg = I.OwnerPkg();
1076
1077 if (Cache[Pkg].CandidateVerIter(Cache) == I.OwnerVer())
1078 {
1079 if (Cache[Pkg].Install() == true && Cache[Pkg].NewInstall() == false)
1080 c1out << " " << Pkg.Name() << " " << I.OwnerVer().VerStr() <<
1081 _(" [Installed]") << endl;
1082 else
1083 c1out << " " << Pkg.Name() << " " << I.OwnerVer().VerStr() << endl;
1084 }
1085 }
1086 c1out << _("You should explicitly select one to install.") << endl;
1087 }
1088 else
1089 {
1090 ioprintf(c1out,
1091 _("Package %s is not available, but is referred to by another package.\n"
1092 "This may mean that the package is missing, has been obsoleted, or\n"
1093 "is only available from another source\n"),Pkg.Name());
1094
1095 string List;
1096 string VersionsList;
1097 SPtrArray<bool> Seen = new bool[Cache.Head().PackageCount];
1098 memset(Seen,0,Cache.Head().PackageCount*sizeof(*Seen));
1099 pkgCache::DepIterator Dep = Pkg.RevDependsList();
1100 for (; Dep.end() == false; Dep++)
1101 {
1102 if (Dep->Type != pkgCache::Dep::Replaces)
1103 continue;
1104 if (Seen[Dep.ParentPkg()->ID] == true)
1105 continue;
1106 Seen[Dep.ParentPkg()->ID] = true;
1107 List += string(Dep.ParentPkg().Name()) + " ";
1108 //VersionsList += string(Dep.ParentPkg().CurVersion) + "\n"; ???
1109 }
1110 ShowList(c1out,_("However the following packages replace it:"),List,VersionsList);
1111 }
1112
1113 _error->Error(_("Package %s has no installation candidate"),Pkg.Name());
1114 return false;
1115 }
1116
1117 Fix.Clear(Pkg);
1118 Fix.Protect(Pkg);
1119 if (Remove == true)
1120 {
1121 Fix.Remove(Pkg);
1122 Cache.MarkDelete(Pkg,_config->FindB("APT::Get::Purge",false));
1123 return true;
1124 }
1125
1126 // Install it
1127 Cache.MarkInstall(Pkg,false);
1128 if (State.Install() == false)
1129 {
1130 if (_config->FindB("APT::Get::ReInstall",false) == true)
1131 {
1132 if (Pkg->CurrentVer == 0 || Pkg.CurrentVer().Downloadable() == false)
1133 ioprintf(c1out,_("Reinstallation of %s is not possible, it cannot be downloaded.\n"),
1134 Pkg.Name());
1135 else
1136 Cache.SetReInstall(Pkg,true);
1137 }
1138 else
1139 {
1140 if (AllowFail == true)
1141 ioprintf(c1out,_("%s is already the newest version.\n"),
1142 Pkg.Name());
1143 }
1144 }
1145 else
1146 ExpectedInst++;
1147
1148 // Install it with autoinstalling enabled.
1149 if (State.InstBroken() == true && BrokenFix == false)
1150 Cache.MarkInstall(Pkg,true);
1151 return true;
1152 }
1153 /*}}}*/
1154 // TryToChangeVer - Try to change a candidate version /*{{{*/
1155 // ---------------------------------------------------------------------
1156 /* */
1157 bool TryToChangeVer(pkgCache::PkgIterator Pkg,pkgDepCache &Cache,
1158 const char *VerTag,bool IsRel)
1159 {
1160 pkgVersionMatch Match(VerTag,(IsRel == true?pkgVersionMatch::Release :
1161 pkgVersionMatch::Version));
1162
1163 pkgCache::VerIterator Ver = Match.Find(Pkg);
1164
1165 if (Ver.end() == true)
1166 {
1167 if (IsRel == true)
1168 return _error->Error(_("Release '%s' for '%s' was not found"),
1169 VerTag,Pkg.Name());
1170 return _error->Error(_("Version '%s' for '%s' was not found"),
1171 VerTag,Pkg.Name());
1172 }
1173
1174 if (strcmp(VerTag,Ver.VerStr()) != 0)
1175 {
1176 ioprintf(c1out,_("Selected version %s (%s) for %s\n"),
1177 Ver.VerStr(),Ver.RelStr().c_str(),Pkg.Name());
1178 }
1179
1180 Cache.SetCandidateVersion(Ver);
1181 return true;
1182 }
1183 /*}}}*/
1184 // FindSrc - Find a source record /*{{{*/
1185 // ---------------------------------------------------------------------
1186 /* */
1187 pkgSrcRecords::Parser *FindSrc(const char *Name,pkgRecords &Recs,
1188 pkgSrcRecords &SrcRecs,string &Src,
1189 pkgDepCache &Cache)
1190 {
1191 // We want to pull the version off the package specification..
1192 string VerTag;
1193 string TmpSrc = Name;
1194 string::size_type Slash = TmpSrc.rfind('=');
1195 if (Slash != string::npos)
1196 {
1197 VerTag = string(TmpSrc.begin() + Slash + 1,TmpSrc.end());
1198 TmpSrc = string(TmpSrc.begin(),TmpSrc.begin() + Slash);
1199 }
1200
1201 /* Lookup the version of the package we would install if we were to
1202 install a version and determine the source package name, then look
1203 in the archive for a source package of the same name. In theory
1204 we could stash the version string as well and match that too but
1205 today there aren't multi source versions in the archive. */
1206 if (_config->FindB("APT::Get::Only-Source") == false &&
1207 VerTag.empty() == true)
1208 {
1209 pkgCache::PkgIterator Pkg = Cache.FindPkg(TmpSrc);
1210 if (Pkg.end() == false)
1211 {
1212 pkgCache::VerIterator Ver = Cache.GetCandidateVer(Pkg);
1213 if (Ver.end() == false)
1214 {
1215 pkgRecords::Parser &Parse = Recs.Lookup(Ver.FileList());
1216 Src = Parse.SourcePkg();
1217 }
1218 }
1219 }
1220
1221 // No source package name..
1222 if (Src.empty() == true)
1223 Src = TmpSrc;
1224
1225 // The best hit
1226 pkgSrcRecords::Parser *Last = 0;
1227 unsigned long Offset = 0;
1228 string Version;
1229 bool IsMatch = false;
1230
1231 // If we are matching by version then we need exact matches to be happy
1232 if (VerTag.empty() == false)
1233 IsMatch = true;
1234
1235 /* Iterate over all of the hits, which includes the resulting
1236 binary packages in the search */
1237 pkgSrcRecords::Parser *Parse;
1238 SrcRecs.Restart();
1239 while ((Parse = SrcRecs.Find(Src.c_str(),false)) != 0)
1240 {
1241 string Ver = Parse->Version();
1242
1243 // Skip name mismatches
1244 if (IsMatch == true && Parse->Package() != Src)
1245 continue;
1246
1247 if (VerTag.empty() == false)
1248 {
1249 /* Don't want to fall through because we are doing exact version
1250 matching. */
1251 if (Cache.VS().CmpVersion(VerTag,Ver) != 0)
1252 continue;
1253
1254 Last = Parse;
1255 Offset = Parse->Offset();
1256 break;
1257 }
1258
1259 // Newer version or an exact match
1260 if (Last == 0 || Cache.VS().CmpVersion(Version,Ver) < 0 ||
1261 (Parse->Package() == Src && IsMatch == false))
1262 {
1263 IsMatch = Parse->Package() == Src;
1264 Last = Parse;
1265 Offset = Parse->Offset();
1266 Version = Ver;
1267 }
1268 }
1269
1270 if (Last == 0)
1271 return 0;
1272
1273 if (Last->Jump(Offset) == false)
1274 return 0;
1275
1276 return Last;
1277 }
1278 /*}}}*/
1279
1280 // DoUpdate - Update the package lists /*{{{*/
1281 // ---------------------------------------------------------------------
1282 /* */
1283 bool DoUpdate(CommandLine &CmdL)
1284 {
1285 if (CmdL.FileSize() != 1)
1286 return _error->Error(_("The update command takes no arguments"));
1287
1288 // Get the source list
1289 pkgSourceList List;
1290 if (List.ReadMainList() == false)
1291 return false;
1292
1293 // Lock the list directory
1294 FileFd Lock;
1295 if (_config->FindB("Debug::NoLocking",false) == false)
1296 {
1297 Lock.Fd(GetLock(_config->FindDir("Dir::State::Lists") + "lock"));
1298 if (_error->PendingError() == true)
1299 return _error->Error(_("Unable to lock the list directory"));
1300 }
1301
1302 // Create the download object
1303 AcqTextStatus Stat(ScreenWidth,_config->FindI("quiet",0));
1304 pkgAcquire Fetcher(&Stat);
1305
1306
1307 // Just print out the uris an exit if the --print-uris flag was used
1308 if (_config->FindB("APT::Get::Print-URIs") == true)
1309 {
1310 // Populate it with the source selection and get all Indexes
1311 // (GetAll=true)
1312 if (List.GetIndexes(&Fetcher,true) == false)
1313 return false;
1314
1315 pkgAcquire::UriIterator I = Fetcher.UriBegin();
1316 for (; I != Fetcher.UriEnd(); I++)
1317 cout << '\'' << I->URI << "' " << flNotDir(I->Owner->DestFile) << ' ' <<
1318 I->Owner->FileSize << ' ' << I->Owner->MD5Sum() << endl;
1319 return true;
1320 }
1321
1322 // Populate it with the source selection
1323 if (List.GetIndexes(&Fetcher) == false)
1324 return false;
1325
1326 // Run it
1327 if (Fetcher.Run() == pkgAcquire::Failed)
1328 return false;
1329
1330 bool Failed = false;
1331 for (pkgAcquire::ItemIterator I = Fetcher.ItemsBegin(); I != Fetcher.ItemsEnd(); I++)
1332 {
1333 if ((*I)->Status == pkgAcquire::Item::StatDone)
1334 continue;
1335
1336 (*I)->Finished();
1337
1338 fprintf(stderr,_("Failed to fetch %s %s\n"),(*I)->DescURI().c_str(),
1339 (*I)->ErrorText.c_str());
1340 Failed = true;
1341 }
1342
1343 // Clean out any old list files
1344 if (_config->FindB("APT::Get::List-Cleanup",true) == true)
1345 {
1346 if (Fetcher.Clean(_config->FindDir("Dir::State::lists")) == false ||
1347 Fetcher.Clean(_config->FindDir("Dir::State::lists") + "partial/") == false)
1348 return false;
1349 }
1350
1351 // Prepare the cache.
1352 CacheFile Cache;
1353 if (Cache.BuildCaches() == false)
1354 return false;
1355
1356 if (Failed == true)
1357 return _error->Error(_("Some index files failed to download, they have been ignored, or old ones used instead."));
1358
1359 return true;
1360 }
1361 /*}}}*/
1362 // DoUpgrade - Upgrade all packages /*{{{*/
1363 // ---------------------------------------------------------------------
1364 /* Upgrade all packages without installing new packages or erasing old
1365 packages */
1366 bool DoUpgrade(CommandLine &CmdL)
1367 {
1368 CacheFile Cache;
1369 if (Cache.OpenForInstall() == false || Cache.CheckDeps() == false)
1370 return false;
1371
1372 // Do the upgrade
1373 if (pkgAllUpgrade(Cache) == false)
1374 {
1375 ShowBroken(c1out,Cache,false);
1376 return _error->Error(_("Internal error, AllUpgrade broke stuff"));
1377 }
1378
1379 return InstallPackages(Cache,true);
1380 }
1381 /*}}}*/
1382 // RecurseDirty - Mark used packages as dirty /*{{{*/
1383 // ---------------------------------------------------------------------
1384 /* Mark all reachable packages as dirty. */
1385 void RecurseDirty (CacheFile &Cache, pkgCache::PkgIterator Pkg, pkgCache::State::PkgRemoveState DirtLevel)
1386 {
1387 // If it is not installed, and we are in manual mode, ignore it
1388 if ((Pkg->CurrentVer == 0 && Cache[Pkg].Install() == false || Cache[Pkg].Delete() == true) &&
1389 DirtLevel == pkgCache::State::RemoveManual)
1390 {
1391 // fprintf(stdout,"This one is not installed/virtual %s %d %d\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel);
1392 return;
1393 }
1394
1395 // If it is not installed, and it is not virtual, ignore it
1396 if ((Pkg->CurrentVer == 0 && Cache[Pkg].Install() == false || Cache[Pkg].Delete() == true) &&
1397 Pkg->VersionList != 0)
1398 {
1399 // fprintf(stdout,"This one is not installed %s %d %d\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel);
1400 return;
1401 }
1402
1403 // If it is similar or more dirty than we are ;-), because we've been here already, don't mark it
1404 // This is necessary because virtual packages just relay the current level,
1405 // so it may be possible e.g. that this was already seen with ::RemoveSuggested, but
1406 // we are ::RemoveRequired
1407 if (Cache[Pkg].Dirty() >= DirtLevel)
1408 {
1409 //fprintf(stdout,"Seen already %s %d %d\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel);
1410 return;
1411 }
1412
1413 // If it is less important than the current DirtLevel, don't mark it
1414 if (Cache[Pkg].AutomaticRemove != pkgCache::State::RemoveManual &&
1415 Cache[Pkg].AutomaticRemove > DirtLevel)
1416 {
1417 // fprintf(stdout,"We don't need %s %d %d %d\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel, Cache[Pkg].Dirty());
1418 return;
1419 }
1420
1421 // Mark it as used
1422 Cache->SetDirty(Pkg, DirtLevel);
1423
1424 //fprintf(stdout,"We keep %s %d %d\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel);
1425
1426 // We are a virtual package
1427 if (Pkg->VersionList == 0)
1428 {
1429 // fprintf(stdout,"We are virtual %s %d %d\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel);
1430 for (pkgCache::PrvIterator Prv = Pkg.ProvidesList(); ! Prv.end(); ++Prv)
1431 RecurseDirty (Cache, Prv.OwnerPkg(), DirtLevel);
1432 return;
1433 }
1434
1435 // Depending on the type of dependency, follow it
1436 for (pkgCache::DepIterator D = Cache[Pkg].InstVerIter(Cache).DependsList(); ! D.end(); ++D)
1437 {
1438 // fprintf(stdout,"We depend on %s %s\n", D.TargetPkg().Name(), D.DepType());
1439
1440 switch(D->Type)
1441 {
1442 case pkgCache::Dep::Depends:
1443 case pkgCache::Dep::PreDepends:
1444 RecurseDirty (Cache, D.TargetPkg(), pkgCache::State::RemoveRequired);
1445 break;
1446 case pkgCache::Dep::Recommends:
1447 RecurseDirty (Cache, D.TargetPkg(), pkgCache::State::RemoveRecommended);
1448 break;
1449 case pkgCache::Dep::Suggests:
1450 RecurseDirty (Cache, D.TargetPkg(), pkgCache::State::RemoveSuggested);
1451 break;
1452 case pkgCache::Dep::Conflicts:
1453 case pkgCache::Dep::Replaces:
1454 case pkgCache::Dep::Obsoletes:
1455 // We don't handle these here
1456 break;
1457 }
1458 }
1459 // fprintf(stdout,"We keep %s %d %d <END>\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel);
1460 }
1461 /*}}}*/
1462 // DoAutomaticRemove - Remove all automatic unused packages /*{{{*/
1463 // ---------------------------------------------------------------------
1464 /* Remove unused automatic packages */
1465 bool DoAutomaticRemove(CacheFile &Cache)
1466 {
1467 std::cout << "DoAutomaticRemove()" << std::endl;
1468 for (pkgCache::PkgIterator Pkg = Cache->PkgBegin(); ! Pkg.end(); ++Pkg)
1469 if(!Cache[Pkg].Dirty() && Cache[Pkg].AutomaticRemove > 0)
1470 std::cout << "has auto-remove information: " << Pkg.Name()
1471 << " " << (int)Cache[Pkg].AutomaticRemove
1472 << std::endl;
1473
1474
1475 if (_config->FindB("APT::Get::Remove",true) == false)
1476 return _error->Error(_("We are not supposed to delete stuff, can't start AutoRemover"));
1477
1478 for (pkgCache::PkgIterator Pkg = Cache->PkgBegin(); ! Pkg.end(); ++Pkg)
1479 Cache->SetDirty(Pkg, pkgCache::State::RemoveUnknown);
1480
1481 for (pkgCache::PkgIterator Pkg = Cache->PkgBegin(); ! Pkg.end(); ++Pkg)
1482 RecurseDirty (Cache, Pkg, pkgCache::State::RemoveManual);
1483
1484
1485
1486 for (pkgCache::PkgIterator Pkg = Cache->PkgBegin(); ! Pkg.end(); ++Pkg)
1487 {
1488 if (! Cache[Pkg].Dirty() &&
1489 (Pkg->CurrentVer != 0 && Cache[Pkg].Install() == false && Cache[Pkg].Delete() == false))
1490 {
1491 fprintf(stdout,"We could delete %s %d\n", Pkg.Name(), Cache[Pkg].AutomaticRemove);
1492 Cache->MarkDelete(Pkg,_config->FindB("APT::Get::Purge",false));
1493 }
1494 }
1495
1496 // Now see if we destroyed anything
1497 if (Cache->BrokenCount() != 0)
1498 {
1499 c1out << _("Hmm, seems like the AutoRemover destroyed something which really\n"
1500 "shouldn't happen. Please file a bug report against apt.") << endl;
1501 c1out << endl;
1502 c1out << _("The following information may help to resolve the situation:") << endl;
1503 c1out << endl;
1504 ShowBroken(c1out,Cache,false);
1505
1506 return _error->Error(_("Internal Error, AutoRemover broke stuff"));
1507 }
1508 return true;
1509 }
1510 /*}}}*/
1511 // DoInstall - Install packages from the command line /*{{{*/
1512 // ---------------------------------------------------------------------
1513 /* Install named packages */
1514 bool DoInstall(CommandLine &CmdL)
1515 {
1516 CacheFile Cache;
1517 if (Cache.OpenForInstall() == false ||
1518 Cache.CheckDeps(CmdL.FileSize() != 1) == false)
1519 return false;
1520
1521 // Enter the special broken fixing mode if the user specified arguments
1522 bool BrokenFix = false;
1523 if (Cache->BrokenCount() != 0)
1524 BrokenFix = true;
1525
1526 unsigned int ExpectedInst = 0;
1527 unsigned int Packages = 0;
1528 pkgProblemResolver Fix(Cache);
1529
1530 bool DefRemove = false;
1531 if (strcasecmp(CmdL.FileList[0],"remove") == 0)
1532 DefRemove = true;
1533
1534 for (const char **I = CmdL.FileList + 1; *I != 0; I++)
1535 {
1536 // Duplicate the string
1537 unsigned int Length = strlen(*I);
1538 char S[300];
1539 if (Length >= sizeof(S))
1540 continue;
1541 strcpy(S,*I);
1542
1543 // See if we are removing and special indicators..
1544 bool Remove = DefRemove;
1545 char *VerTag = 0;
1546 bool VerIsRel = false;
1547 while (Cache->FindPkg(S).end() == true)
1548 {
1549 // Handle an optional end tag indicating what to do
1550 if (Length >= 1 && S[Length - 1] == '-')
1551 {
1552 Remove = true;
1553 S[--Length] = 0;
1554 continue;
1555 }
1556
1557 if (Length >= 1 && S[Length - 1] == '+')
1558 {
1559 Remove = false;
1560 S[--Length] = 0;
1561 continue;
1562 }
1563
1564 char *Slash = strchr(S,'=');
1565 if (Slash != 0)
1566 {
1567 VerIsRel = false;
1568 *Slash = 0;
1569 VerTag = Slash + 1;
1570 }
1571
1572 Slash = strchr(S,'/');
1573 if (Slash != 0)
1574 {
1575 VerIsRel = true;
1576 *Slash = 0;
1577 VerTag = Slash + 1;
1578 }
1579
1580 break;
1581 }
1582
1583 // Locate the package
1584 pkgCache::PkgIterator Pkg = Cache->FindPkg(S);
1585 Packages++;
1586 if (Pkg.end() == true)
1587 {
1588 // Check if the name is a regex
1589 const char *I;
1590 for (I = S; *I != 0; I++)
1591 if (*I == '?' || *I == '*' || *I == '|' ||
1592 *I == '[' || *I == '^' || *I == '$')
1593 break;
1594 if (*I == 0)
1595 return _error->Error(_("Couldn't find package %s"),S);
1596
1597 // Regexs must always be confirmed
1598 ExpectedInst += 1000;
1599
1600 // Compile the regex pattern
1601 regex_t Pattern;
1602 int Res;
1603 if ((Res = regcomp(&Pattern,S,REG_EXTENDED | REG_ICASE |
1604 REG_NOSUB)) != 0)
1605 {
1606 char Error[300];
1607 regerror(Res,&Pattern,Error,sizeof(Error));
1608 return _error->Error(_("Regex compilation error - %s"),Error);
1609 }
1610
1611 // Run over the matches
1612 bool Hit = false;
1613 for (Pkg = Cache->PkgBegin(); Pkg.end() == false; Pkg++)
1614 {
1615 if (regexec(&Pattern,Pkg.Name(),0,0,0) != 0)
1616 continue;
1617
1618 ioprintf(c1out,_("Note, selecting %s for regex '%s'\n"),
1619 Pkg.Name(),S);
1620
1621 if (VerTag != 0)
1622 if (TryToChangeVer(Pkg,Cache,VerTag,VerIsRel) == false)
1623 return false;
1624
1625 Hit |= TryToInstall(Pkg,Cache,Fix,Remove,BrokenFix,
1626 ExpectedInst,false);
1627 }
1628 regfree(&Pattern);
1629
1630 if (Hit == false)
1631 return _error->Error(_("Couldn't find package %s"),S);
1632 }
1633 else
1634 {
1635 if (VerTag != 0)
1636 if (TryToChangeVer(Pkg,Cache,VerTag,VerIsRel) == false)
1637 return false;
1638 if (TryToInstall(Pkg,Cache,Fix,Remove,BrokenFix,ExpectedInst) == false)
1639 return false;
1640 }
1641 }
1642
1643 /* If we are in the Broken fixing mode we do not attempt to fix the
1644 problems. This is if the user invoked install without -f and gave
1645 packages */
1646 if (BrokenFix == true && Cache->BrokenCount() != 0)
1647 {
1648 c1out << _("You might want to run `apt-get -f install' to correct these:") << endl;
1649 ShowBroken(c1out,Cache,false);
1650
1651 return _error->Error(_("Unmet dependencies. Try 'apt-get -f install' with no packages (or specify a solution)."));
1652 }
1653
1654 // Call the scored problem resolver
1655 Fix.InstallProtect();
1656 if (Fix.Resolve(true) == false)
1657 _error->Discard();
1658
1659 // Now we check the state of the packages,
1660 if (Cache->BrokenCount() != 0)
1661 {
1662 c1out <<
1663 _("Some packages could not be installed. This may mean that you have\n"
1664 "requested an impossible situation or if you are using the unstable\n"
1665 "distribution that some required packages have not yet been created\n"
1666 "or been moved out of Incoming.") << endl;
1667 if (Packages == 1)
1668 {
1669 c1out << endl;
1670 c1out <<
1671 _("Since you only requested a single operation it is extremely likely that\n"
1672 "the package is simply not installable and a bug report against\n"
1673 "that package should be filed.") << endl;
1674 }
1675
1676 c1out << _("The following information may help to resolve the situation:") << endl;
1677 c1out << endl;
1678 ShowBroken(c1out,Cache,false);
1679 return _error->Error(_("Broken packages"));
1680 }
1681
1682 //if (_config->FindB("APT::Get::AutomaticRemove")) {
1683 if (!DoAutomaticRemove(Cache))
1684 return false;
1685 //}
1686
1687 /* Print out a list of packages that are going to be installed extra
1688 to what the user asked */
1689 if (Cache->InstCount() != ExpectedInst)
1690 {
1691 string List;
1692 string VersionsList;
1693 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
1694 {
1695 pkgCache::PkgIterator I(Cache,Cache.List[J]);
1696 if ((*Cache)[I].Install() == false)
1697 continue;
1698
1699 const char **J;
1700 for (J = CmdL.FileList + 1; *J != 0; J++)
1701 if (strcmp(*J,I.Name()) == 0)
1702 break;
1703
1704 if (*J == 0) {
1705 List += string(I.Name()) + " ";
1706 // mark each pkg as auto-installed
1707 Cache[I].AutomaticRemove = pkgCache::State::RemoveRequired;
1708 VersionsList += string(Cache[I].CandVersion) + "\n";
1709 }
1710 }
1711
1712 ShowList(c1out,_("The following extra packages will be installed:"),List,VersionsList);
1713 }
1714
1715 /* Print out a list of suggested and recommended packages */
1716 {
1717 string SuggestsList, RecommendsList, List;
1718 string SuggestsVersions, RecommendsVersions;
1719 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
1720 {
1721 pkgCache::PkgIterator I(Cache,Cache.List[J]);
1722
1723 /* Just look at the ones we want to install */
1724 if ((*Cache)[I].Install() == false)
1725 continue;
1726
1727 for (pkgCache::VerIterator V = I.VersionList(); V.end() == false; V++)
1728 {
1729 for (pkgCache::DepIterator D = V.DependsList(); D.end() == false; )
1730 {
1731 pkgCache::DepIterator Start;
1732 pkgCache::DepIterator End;
1733 D.GlobOr(Start,End); // advances D
1734
1735 /*
1736 * If this is a virtual package, we need to check the list of
1737 * packages that provide it and see if any of those are
1738 * installed
1739 */
1740
1741 bool providedBySomething = false;
1742 for (pkgCache::PrvIterator Prv = Start.TargetPkg().ProvidesList();
1743 Prv.end() != true;
1744 Prv++)
1745 if ((*Cache)[Prv.OwnerPkg()].InstVerIter(*Cache).end() == false)
1746 {
1747 providedBySomething = true;
1748 break;
1749 }
1750
1751 if (providedBySomething) continue;
1752
1753 for(;;)
1754 {
1755 /* Skip if package is installed already, or is about to be */
1756 string target = string(Start.TargetPkg().Name()) + " ";
1757
1758 if ((*Start.TargetPkg()).SelectedState == pkgCache::State::Install
1759 || Cache[Start.TargetPkg()].Install())
1760 break;
1761
1762 /* Skip if we already saw it */
1763 if (int(SuggestsList.find(target)) != -1 || int(RecommendsList.find(target)) != -1)
1764 break;
1765
1766 if (Start->Type == pkgCache::Dep::Suggests) {
1767 SuggestsList += target;
1768 SuggestsVersions += string(Cache[Start.TargetPkg()].CandVersion) + "\n";
1769 }
1770
1771 if (Start->Type == pkgCache::Dep::Recommends) {
1772 RecommendsList += target;
1773 RecommendsVersions += string(Cache[Start.TargetPkg()].CandVersion) + "\n";
1774 }
1775
1776 if (Start >= End)
1777 break;
1778 Start++;
1779 }
1780 }
1781 }
1782 }
1783 ShowList(c1out,_("Suggested packages:"),SuggestsList,SuggestsVersions);
1784 ShowList(c1out,_("Recommended packages:"),RecommendsList,RecommendsVersions);
1785
1786 }
1787
1788 // See if we need to prompt
1789 if (Cache->InstCount() == ExpectedInst && Cache->DelCount() == 0)
1790 return InstallPackages(Cache,false,false);
1791
1792 return InstallPackages(Cache,false);
1793 }
1794 /*}}}*/
1795 // DoDistUpgrade - Automatic smart upgrader /*{{{*/
1796 // ---------------------------------------------------------------------
1797 /* Intelligent upgrader that will install and remove packages at will */
1798 bool DoDistUpgrade(CommandLine &CmdL)
1799 {
1800 CacheFile Cache;
1801 if (Cache.OpenForInstall() == false || Cache.CheckDeps() == false)
1802 return false;
1803
1804 c0out << _("Calculating upgrade... ") << flush;
1805 if (pkgDistUpgrade(*Cache) == false)
1806 {
1807 c0out << _("Failed") << endl;
1808 ShowBroken(c1out,Cache,false);
1809 return false;
1810 }
1811
1812 c0out << _("Done") << endl;
1813
1814 return InstallPackages(Cache,true);
1815 }
1816 /*}}}*/
1817 // DoDSelectUpgrade - Do an upgrade by following dselects selections /*{{{*/
1818 // ---------------------------------------------------------------------
1819 /* Follows dselect's selections */
1820 bool DoDSelectUpgrade(CommandLine &CmdL)
1821 {
1822 CacheFile Cache;
1823 if (Cache.OpenForInstall() == false || Cache.CheckDeps() == false)
1824 return false;
1825
1826 // Install everything with the install flag set
1827 pkgCache::PkgIterator I = Cache->PkgBegin();
1828 for (;I.end() != true; I++)
1829 {
1830 /* Install the package only if it is a new install, the autoupgrader
1831 will deal with the rest */
1832 if (I->SelectedState == pkgCache::State::Install)
1833 Cache->MarkInstall(I,false);
1834 }
1835
1836 /* Now install their deps too, if we do this above then order of
1837 the status file is significant for | groups */
1838 for (I = Cache->PkgBegin();I.end() != true; I++)
1839 {
1840 /* Install the package only if it is a new install, the autoupgrader
1841 will deal with the rest */
1842 if (I->SelectedState == pkgCache::State::Install)
1843 Cache->MarkInstall(I,true);
1844 }
1845
1846 // Apply erasures now, they override everything else.
1847 for (I = Cache->PkgBegin();I.end() != true; I++)
1848 {
1849 // Remove packages
1850 if (I->SelectedState == pkgCache::State::DeInstall ||
1851 I->SelectedState == pkgCache::State::Purge)
1852 Cache->MarkDelete(I,I->SelectedState == pkgCache::State::Purge);
1853 }
1854
1855 /* Resolve any problems that dselect created, allupgrade cannot handle
1856 such things. We do so quite agressively too.. */
1857 if (Cache->BrokenCount() != 0)
1858 {
1859 pkgProblemResolver Fix(Cache);
1860
1861 // Hold back held packages.
1862 if (_config->FindB("APT::Ignore-Hold",false) == false)
1863 {
1864 for (pkgCache::PkgIterator I = Cache->PkgBegin(); I.end() == false; I++)
1865 {
1866 if (I->SelectedState == pkgCache::State::Hold)
1867 {
1868 Fix.Protect(I);
1869 Cache->MarkKeep(I);
1870 }
1871 }
1872 }
1873
1874 if (Fix.Resolve() == false)
1875 {
1876 ShowBroken(c1out,Cache,false);
1877 return _error->Error("Internal error, problem resolver broke stuff");
1878 }
1879 }
1880
1881 // Now upgrade everything
1882 if (pkgAllUpgrade(Cache) == false)
1883 {
1884 ShowBroken(c1out,Cache,false);
1885 return _error->Error("Internal error, problem resolver broke stuff");
1886 }
1887
1888 return InstallPackages(Cache,false);
1889 }
1890 /*}}}*/
1891 // DoClean - Remove download archives /*{{{*/
1892 // ---------------------------------------------------------------------
1893 /* */
1894 bool DoClean(CommandLine &CmdL)
1895 {
1896 if (_config->FindB("APT::Get::Simulate") == true)
1897 {
1898 cout << "Del " << _config->FindDir("Dir::Cache::archives") << "* " <<
1899 _config->FindDir("Dir::Cache::archives") << "partial/*" << endl;
1900 return true;
1901 }
1902
1903 // Lock the archive directory
1904 FileFd Lock;
1905 if (_config->FindB("Debug::NoLocking",false) == false)
1906 {
1907 Lock.Fd(GetLock(_config->FindDir("Dir::Cache::Archives") + "lock"));
1908 if (_error->PendingError() == true)
1909 return _error->Error(_("Unable to lock the download directory"));
1910 }
1911
1912 pkgAcquire Fetcher;
1913 Fetcher.Clean(_config->FindDir("Dir::Cache::archives"));
1914 Fetcher.Clean(_config->FindDir("Dir::Cache::archives") + "partial/");
1915 return true;
1916 }
1917 /*}}}*/
1918 // DoAutoClean - Smartly remove downloaded archives /*{{{*/
1919 // ---------------------------------------------------------------------
1920 /* This is similar to clean but it only purges things that cannot be
1921 downloaded, that is old versions of cached packages. */
1922 class LogCleaner : public pkgArchiveCleaner
1923 {
1924 protected:
1925 virtual void Erase(const char *File,string Pkg,string Ver,struct stat &St)
1926 {
1927 c1out << "Del " << Pkg << " " << Ver << " [" << SizeToStr(St.st_size) << "B]" << endl;
1928
1929 if (_config->FindB("APT::Get::Simulate") == false)
1930 unlink(File);
1931 };
1932 };
1933
1934 bool DoAutoClean(CommandLine &CmdL)
1935 {
1936 // Lock the archive directory
1937 FileFd Lock;
1938 if (_config->FindB("Debug::NoLocking",false) == false)
1939 {
1940 Lock.Fd(GetLock(_config->FindDir("Dir::Cache::Archives") + "lock"));
1941 if (_error->PendingError() == true)
1942 return _error->Error(_("Unable to lock the download directory"));
1943 }
1944
1945 CacheFile Cache;
1946 if (Cache.Open() == false)
1947 return false;
1948
1949 LogCleaner Cleaner;
1950
1951 return Cleaner.Go(_config->FindDir("Dir::Cache::archives"),*Cache) &&
1952 Cleaner.Go(_config->FindDir("Dir::Cache::archives") + "partial/",*Cache);
1953 }
1954 /*}}}*/
1955 // DoCheck - Perform the check operation /*{{{*/
1956 // ---------------------------------------------------------------------
1957 /* Opening automatically checks the system, this command is mostly used
1958 for debugging */
1959 bool DoCheck(CommandLine &CmdL)
1960 {
1961 CacheFile Cache;
1962 Cache.Open();
1963 Cache.CheckDeps();
1964
1965 return true;
1966 }
1967 /*}}}*/
1968 // DoSource - Fetch a source archive /*{{{*/
1969 // ---------------------------------------------------------------------
1970 /* Fetch souce packages */
1971 struct DscFile
1972 {
1973 string Package;
1974 string Version;
1975 string Dsc;
1976 };
1977
1978 bool DoSource(CommandLine &CmdL)
1979 {
1980 CacheFile Cache;
1981 if (Cache.Open(false) == false)
1982 return false;
1983
1984 if (CmdL.FileSize() <= 1)
1985 return _error->Error(_("Must specify at least one package to fetch source for"));
1986
1987 // Read the source list
1988 pkgSourceList List;
1989 if (List.ReadMainList() == false)
1990 return _error->Error(_("The list of sources could not be read."));
1991
1992 // Create the text record parsers
1993 pkgRecords Recs(Cache);
1994 pkgSrcRecords SrcRecs(List);
1995 if (_error->PendingError() == true)
1996 return false;
1997
1998 // Create the download object
1999 AcqTextStatus Stat(ScreenWidth,_config->FindI("quiet",0));
2000 pkgAcquire Fetcher(&Stat);
2001
2002 DscFile *Dsc = new DscFile[CmdL.FileSize()];
2003
2004 // Load the requestd sources into the fetcher
2005 unsigned J = 0;
2006 for (const char **I = CmdL.FileList + 1; *I != 0; I++, J++)
2007 {
2008 string Src;
2009 pkgSrcRecords::Parser *Last = FindSrc(*I,Recs,SrcRecs,Src,*Cache);
2010
2011 if (Last == 0)
2012 return _error->Error(_("Unable to find a source package for %s"),Src.c_str());
2013
2014 // Back track
2015 vector<pkgSrcRecords::File> Lst;
2016 if (Last->Files(Lst) == false)
2017 return false;
2018
2019 // Load them into the fetcher
2020 for (vector<pkgSrcRecords::File>::const_iterator I = Lst.begin();
2021 I != Lst.end(); I++)
2022 {
2023 // Try to guess what sort of file it is we are getting.
2024 if (I->Type == "dsc")
2025 {
2026 Dsc[J].Package = Last->Package();
2027 Dsc[J].Version = Last->Version();
2028 Dsc[J].Dsc = flNotDir(I->Path);
2029 }
2030
2031 // Diff only mode only fetches .diff files
2032 if (_config->FindB("APT::Get::Diff-Only",false) == true &&
2033 I->Type != "diff")
2034 continue;
2035
2036 // Tar only mode only fetches .tar files
2037 if (_config->FindB("APT::Get::Tar-Only",false) == true &&
2038 I->Type != "tar")
2039 continue;
2040
2041 new pkgAcqFile(&Fetcher,Last->Index().ArchiveURI(I->Path),
2042 I->MD5Hash,I->Size,
2043 Last->Index().SourceInfo(*Last,*I),Src);
2044 }
2045 }
2046
2047 // Display statistics
2048 double FetchBytes = Fetcher.FetchNeeded();
2049 double FetchPBytes = Fetcher.PartialPresent();
2050 double DebBytes = Fetcher.TotalNeeded();
2051
2052 // Check for enough free space
2053 struct statvfs Buf;
2054 string OutputDir = ".";
2055 if (statvfs(OutputDir.c_str(),&Buf) != 0)
2056 return _error->Errno("statvfs","Couldn't determine free space in %s",
2057 OutputDir.c_str());
2058 if (unsigned(Buf.f_bfree) < (FetchBytes - FetchPBytes)/Buf.f_bsize)
2059 return _error->Error(_("You don't have enough free space in %s"),
2060 OutputDir.c_str());
2061
2062 // Number of bytes
2063 if (DebBytes != FetchBytes)
2064 ioprintf(c1out,_("Need to get %sB/%sB of source archives.\n"),
2065 SizeToStr(FetchBytes).c_str(),SizeToStr(DebBytes).c_str());
2066 else
2067 ioprintf(c1out,_("Need to get %sB of source archives.\n"),
2068 SizeToStr(DebBytes).c_str());
2069
2070 if (_config->FindB("APT::Get::Simulate",false) == true)
2071 {
2072 for (unsigned I = 0; I != J; I++)
2073 ioprintf(cout,_("Fetch source %s\n"),Dsc[I].Package.c_str());
2074 return true;
2075 }
2076
2077 // Just print out the uris an exit if the --print-uris flag was used
2078 if (_config->FindB("APT::Get::Print-URIs") == true)
2079 {
2080 pkgAcquire::UriIterator I = Fetcher.UriBegin();
2081 for (; I != Fetcher.UriEnd(); I++)
2082 cout << '\'' << I->URI << "' " << flNotDir(I->Owner->DestFile) << ' ' <<
2083 I->Owner->FileSize << ' ' << I->Owner->MD5Sum() << endl;
2084 return true;
2085 }
2086
2087 // Run it
2088 if (Fetcher.Run() == pkgAcquire::Failed)
2089 return false;
2090
2091 // Print error messages
2092 bool Failed = false;
2093 for (pkgAcquire::ItemIterator I = Fetcher.ItemsBegin(); I != Fetcher.ItemsEnd(); I++)
2094 {
2095 if ((*I)->Status == pkgAcquire::Item::StatDone &&
2096 (*I)->Complete == true)
2097 continue;
2098
2099 fprintf(stderr,_("Failed to fetch %s %s\n"),(*I)->DescURI().c_str(),
2100 (*I)->ErrorText.c_str());
2101 Failed = true;
2102 }
2103 if (Failed == true)
2104 return _error->Error(_("Failed to fetch some archives."));
2105
2106 if (_config->FindB("APT::Get::Download-only",false) == true)
2107 {
2108 c1out << _("Download complete and in download only mode") << endl;
2109 return true;
2110 }
2111
2112 // Unpack the sources
2113 pid_t Process = ExecFork();
2114
2115 if (Process == 0)
2116 {
2117 for (unsigned I = 0; I != J; I++)
2118 {
2119 string Dir = Dsc[I].Package + '-' + Cache->VS().UpstreamVersion(Dsc[I].Version.c_str());
2120
2121 // Diff only mode only fetches .diff files
2122 if (_config->FindB("APT::Get::Diff-Only",false) == true ||
2123 _config->FindB("APT::Get::Tar-Only",false) == true ||
2124 Dsc[I].Dsc.empty() == true)
2125 continue;
2126
2127 // See if the package is already unpacked
2128 struct stat Stat;
2129 if (stat(Dir.c_str(),&Stat) == 0 &&
2130 S_ISDIR(Stat.st_mode) != 0)
2131 {
2132 ioprintf(c0out ,_("Skipping unpack of already unpacked source in %s\n"),
2133 Dir.c_str());
2134 }
2135 else
2136 {
2137 // Call dpkg-source
2138 char S[500];
2139 snprintf(S,sizeof(S),"%s -x %s",
2140 _config->Find("Dir::Bin::dpkg-source","dpkg-source").c_str(),
2141 Dsc[I].Dsc.c_str());
2142 if (system(S) != 0)
2143 {
2144 fprintf(stderr,_("Unpack command '%s' failed.\n"),S);
2145 _exit(1);
2146 }
2147 }
2148
2149 // Try to compile it with dpkg-buildpackage
2150 if (_config->FindB("APT::Get::Compile",false) == true)
2151 {
2152 // Call dpkg-buildpackage
2153 char S[500];
2154 snprintf(S,sizeof(S),"cd %s && %s %s",
2155 Dir.c_str(),
2156 _config->Find("Dir::Bin::dpkg-buildpackage","dpkg-buildpackage").c_str(),
2157 _config->Find("DPkg::Build-Options","-b -uc").c_str());
2158
2159 if (system(S) != 0)
2160 {
2161 fprintf(stderr,_("Build command '%s' failed.\n"),S);
2162 _exit(1);
2163 }
2164 }
2165 }
2166
2167 _exit(0);
2168 }
2169
2170 // Wait for the subprocess
2171 int Status = 0;
2172 while (waitpid(Process,&Status,0) != Process)
2173 {
2174 if (errno == EINTR)
2175 continue;
2176 return _error->Errno("waitpid","Couldn't wait for subprocess");
2177 }
2178
2179 if (WIFEXITED(Status) == 0 || WEXITSTATUS(Status) != 0)
2180 return _error->Error(_("Child process failed"));
2181
2182 return true;
2183 }
2184 /*}}}*/
2185 // DoBuildDep - Install/removes packages to satisfy build dependencies /*{{{*/
2186 // ---------------------------------------------------------------------
2187 /* This function will look at the build depends list of the given source
2188 package and install the necessary packages to make it true, or fail. */
2189 bool DoBuildDep(CommandLine &CmdL)
2190 {
2191 CacheFile Cache;
2192 if (Cache.Open(true) == false)
2193 return false;
2194
2195 if (CmdL.FileSize() <= 1)
2196 return _error->Error(_("Must specify at least one package to check builddeps for"));
2197
2198 // Read the source list
2199 pkgSourceList List;
2200 if (List.ReadMainList() == false)
2201 return _error->Error(_("The list of sources could not be read."));
2202
2203 // Create the text record parsers
2204 pkgRecords Recs(Cache);
2205 pkgSrcRecords SrcRecs(List);
2206 if (_error->PendingError() == true)
2207 return false;
2208
2209 // Create the download object
2210 AcqTextStatus Stat(ScreenWidth,_config->FindI("quiet",0));
2211 pkgAcquire Fetcher(&Stat);
2212
2213 unsigned J = 0;
2214 for (const char **I = CmdL.FileList + 1; *I != 0; I++, J++)
2215 {
2216 string Src;
2217 pkgSrcRecords::Parser *Last = FindSrc(*I,Recs,SrcRecs,Src,*Cache);
2218 if (Last == 0)
2219 return _error->Error(_("Unable to find a source package for %s"),Src.c_str());
2220
2221 // Process the build-dependencies
2222 vector<pkgSrcRecords::Parser::BuildDepRec> BuildDeps;
2223 if (Last->BuildDepends(BuildDeps, _config->FindB("APT::Get::Arch-Only",false)) == false)
2224 return _error->Error(_("Unable to get build-dependency information for %s"),Src.c_str());
2225
2226 // Also ensure that build-essential packages are present
2227 Configuration::Item const *Opts = _config->Tree("APT::Build-Essential");
2228 if (Opts)
2229 Opts = Opts->Child;
2230 for (; Opts; Opts = Opts->Next)
2231 {
2232 if (Opts->Value.empty() == true)
2233 continue;
2234
2235 pkgSrcRecords::Parser::BuildDepRec rec;
2236 rec.Package = Opts->Value;
2237 rec.Type = pkgSrcRecords::Parser::BuildDependIndep;
2238 rec.Op = 0;
2239 BuildDeps.push_back(rec);
2240 }
2241
2242 if (BuildDeps.size() == 0)
2243 {
2244 ioprintf(c1out,_("%s has no build depends.\n"),Src.c_str());
2245 continue;
2246 }
2247
2248 // Install the requested packages
2249 unsigned int ExpectedInst = 0;
2250 vector <pkgSrcRecords::Parser::BuildDepRec>::iterator D;
2251 pkgProblemResolver Fix(Cache);
2252 bool skipAlternatives = false; // skip remaining alternatives in an or group
2253 for (D = BuildDeps.begin(); D != BuildDeps.end(); D++)
2254 {
2255 bool hasAlternatives = (((*D).Op & pkgCache::Dep::Or) == pkgCache::Dep::Or);
2256
2257 if (skipAlternatives == true)
2258 {
2259 if (!hasAlternatives)
2260 skipAlternatives = false; // end of or group
2261 continue;
2262 }
2263
2264 if ((*D).Type == pkgSrcRecords::Parser::BuildConflict ||
2265 (*D).Type == pkgSrcRecords::Parser::BuildConflictIndep)
2266 {
2267 pkgCache::PkgIterator Pkg = Cache->FindPkg((*D).Package);
2268 // Build-conflicts on unknown packages are silently ignored
2269 if (Pkg.end() == true)
2270 continue;
2271
2272 pkgCache::VerIterator IV = (*Cache)[Pkg].InstVerIter(*Cache);
2273
2274 /*
2275 * Remove if we have an installed version that satisfies the
2276 * version criteria
2277 */
2278 if (IV.end() == false &&
2279 Cache->VS().CheckDep(IV.VerStr(),(*D).Op,(*D).Version.c_str()) == true)
2280 TryToInstall(Pkg,Cache,Fix,true,false,ExpectedInst);
2281 }
2282 else // BuildDep || BuildDepIndep
2283 {
2284 pkgCache::PkgIterator Pkg = Cache->FindPkg((*D).Package);
2285 if (_config->FindB("Debug::BuildDeps",false) == true)
2286 cout << "Looking for " << (*D).Package << "...\n";
2287
2288 if (Pkg.end() == true)
2289 {
2290 if (_config->FindB("Debug::BuildDeps",false) == true)
2291 cout << " (not found)" << (*D).Package << endl;
2292
2293 if (hasAlternatives)
2294 continue;
2295
2296 return _error->Error(_("%s dependency for %s cannot be satisfied "
2297 "because the package %s cannot be found"),
2298 Last->BuildDepType((*D).Type),Src.c_str(),
2299 (*D).Package.c_str());
2300 }
2301
2302 /*
2303 * if there are alternatives, we've already picked one, so skip
2304 * the rest
2305 *
2306 * TODO: this means that if there's a build-dep on A|B and B is
2307 * installed, we'll still try to install A; more importantly,
2308 * if A is currently broken, we cannot go back and try B. To fix
2309 * this would require we do a Resolve cycle for each package we
2310 * add to the install list. Ugh
2311 */
2312
2313 /*
2314 * If this is a virtual package, we need to check the list of
2315 * packages that provide it and see if any of those are
2316 * installed
2317 */
2318 pkgCache::PrvIterator Prv = Pkg.ProvidesList();
2319 for (; Prv.end() != true; Prv++)
2320 {
2321 if (_config->FindB("Debug::BuildDeps",false) == true)
2322 cout << " Checking provider " << Prv.OwnerPkg().Name() << endl;
2323
2324 if ((*Cache)[Prv.OwnerPkg()].InstVerIter(*Cache).end() == false)
2325 break;
2326 }
2327
2328 // Get installed version and version we are going to install
2329 pkgCache::VerIterator IV = (*Cache)[Pkg].InstVerIter(*Cache);
2330
2331 if ((*D).Version[0] != '\0') {
2332 // Versioned dependency
2333
2334 pkgCache::VerIterator CV = (*Cache)[Pkg].CandidateVerIter(*Cache);
2335
2336 for (; CV.end() != true; CV++)
2337 {
2338 if (Cache->VS().CheckDep(CV.VerStr(),(*D).Op,(*D).Version.c_str()) == true)
2339 break;
2340 }
2341 if (CV.end() == true)
2342 if (hasAlternatives)
2343 {
2344 continue;
2345 }
2346 else
2347 {
2348 return _error->Error(_("%s dependency for %s cannot be satisfied "
2349 "because no available versions of package %s "
2350 "can satisfy version requirements"),
2351 Last->BuildDepType((*D).Type),Src.c_str(),
2352 (*D).Package.c_str());
2353 }
2354 }
2355 else
2356 {
2357 // Only consider virtual packages if there is no versioned dependency
2358 if (Prv.end() == false)
2359 {
2360 if (_config->FindB("Debug::BuildDeps",false) == true)
2361 cout << " Is provided by installed package " << Prv.OwnerPkg().Name() << endl;
2362 skipAlternatives = hasAlternatives;
2363 continue;
2364 }
2365 }
2366
2367 if (IV.end() == false)
2368 {
2369 if (_config->FindB("Debug::BuildDeps",false) == true)
2370 cout << " Is installed\n";
2371
2372 if (Cache->VS().CheckDep(IV.VerStr(),(*D).Op,(*D).Version.c_str()) == true)
2373 {
2374 skipAlternatives = hasAlternatives;
2375 continue;
2376 }
2377
2378 if (_config->FindB("Debug::BuildDeps",false) == true)
2379 cout << " ...but the installed version doesn't meet the version requirement\n";
2380
2381 if (((*D).Op & pkgCache::Dep::LessEq) == pkgCache::Dep::LessEq)
2382 {
2383 return _error->Error(_("Failed to satisfy %s dependency for %s: Installed package %s is too new"),
2384 Last->BuildDepType((*D).Type),
2385 Src.c_str(),
2386 Pkg.Name());
2387 }
2388 }
2389
2390
2391 if (_config->FindB("Debug::BuildDeps",false) == true)
2392 cout << " Trying to install " << (*D).Package << endl;
2393
2394 if (TryToInstall(Pkg,Cache,Fix,false,false,ExpectedInst) == true)
2395 {
2396 // We successfully installed something; skip remaining alternatives
2397 skipAlternatives = hasAlternatives;
2398 continue;
2399 }
2400 else if (hasAlternatives)
2401 {
2402 if (_config->FindB("Debug::BuildDeps",false) == true)
2403 cout << " Unsatisfiable, trying alternatives\n";
2404 continue;
2405 }
2406 else
2407 {
2408 return _error->Error(_("Failed to satisfy %s dependency for %s: %s"),
2409 Last->BuildDepType((*D).Type),
2410 Src.c_str(),
2411 (*D).Package.c_str());
2412 }
2413 }
2414 }
2415
2416 Fix.InstallProtect();
2417 if (Fix.Resolve(true) == false)
2418 _error->Discard();
2419
2420 // Now we check the state of the packages,
2421 if (Cache->BrokenCount() != 0)
2422 return _error->Error(_("Build-dependencies for %s could not be satisfied."),*I);
2423 }
2424
2425 if (InstallPackages(Cache, false, true) == false)
2426 return _error->Error(_("Failed to process build dependencies"));
2427 return true;
2428 }
2429 /*}}}*/
2430
2431 // DoMoo - Never Ask, Never Tell /*{{{*/
2432 // ---------------------------------------------------------------------
2433 /* */
2434 bool DoMoo(CommandLine &CmdL)
2435 {
2436 cout <<
2437 " (__) \n"
2438 " (oo) \n"
2439 " /------\\/ \n"
2440 " / | || \n"
2441 " * /\\---/\\ \n"
2442 " ~~ ~~ \n"
2443 "....\"Have you mooed today?\"...\n";
2444
2445 return true;
2446 }
2447 /*}}}*/
2448 // ShowHelp - Show a help screen /*{{{*/
2449 // ---------------------------------------------------------------------
2450 /* */
2451 bool ShowHelp(CommandLine &CmdL)
2452 {
2453 ioprintf(cout,_("%s %s for %s %s compiled on %s %s\n"),PACKAGE,VERSION,
2454 COMMON_OS,COMMON_CPU,__DATE__,__TIME__);
2455
2456 if (_config->FindB("version") == true)
2457 {
2458 cout << _("Supported modules:") << endl;
2459
2460 for (unsigned I = 0; I != pkgVersioningSystem::GlobalListLen; I++)
2461 {
2462 pkgVersioningSystem *VS = pkgVersioningSystem::GlobalList[I];
2463 if (_system != 0 && _system->VS == VS)
2464 cout << '*';
2465 else
2466 cout << ' ';
2467 cout << "Ver: " << VS->Label << endl;
2468
2469 /* Print out all the packaging systems that will work with
2470 this VS */
2471 for (unsigned J = 0; J != pkgSystem::GlobalListLen; J++)
2472 {
2473 pkgSystem *Sys = pkgSystem::GlobalList[J];
2474 if (_system == Sys)
2475 cout << '*';
2476 else
2477 cout << ' ';
2478 if (Sys->VS->TestCompatibility(*VS) == true)
2479 cout << "Pkg: " << Sys->Label << " (Priority " << Sys->Score(*_config) << ")" << endl;
2480 }
2481 }
2482
2483 for (unsigned I = 0; I != pkgSourceList::Type::GlobalListLen; I++)
2484 {
2485 pkgSourceList::Type *Type = pkgSourceList::Type::GlobalList[I];
2486 cout << " S.L: '" << Type->Name << "' " << Type->Label << endl;
2487 }
2488
2489 for (unsigned I = 0; I != pkgIndexFile::Type::GlobalListLen; I++)
2490 {
2491 pkgIndexFile::Type *Type = pkgIndexFile::Type::GlobalList[I];
2492 cout << " Idx: " << Type->Label << endl;
2493 }
2494
2495 return true;
2496 }
2497
2498 cout <<
2499 _("Usage: apt-get [options] command\n"
2500 " apt-get [options] install|remove pkg1 [pkg2 ...]\n"
2501 " apt-get [options] source pkg1 [pkg2 ...]\n"
2502 "\n"
2503 "apt-get is a simple command line interface for downloading and\n"
2504 "installing packages. The most frequently used commands are update\n"
2505 "and install.\n"
2506 "\n"
2507 "Commands:\n"
2508 " update - Retrieve new lists of packages\n"
2509 " upgrade - Perform an upgrade\n"
2510 " install - Install new packages (pkg is libc6 not libc6.deb)\n"
2511 " remove - Remove packages\n"
2512 " source - Download source archives\n"
2513 " build-dep - Configure build-dependencies for source packages\n"
2514 " dist-upgrade - Distribution upgrade, see apt-get(8)\n"
2515 " dselect-upgrade - Follow dselect selections\n"
2516 " clean - Erase downloaded archive files\n"
2517 " autoclean - Erase old downloaded archive files\n"
2518 " check - Verify that there are no broken dependencies\n"
2519 "\n"
2520 "Options:\n"
2521 " -h This help text.\n"
2522 " -q Loggable output - no progress indicator\n"
2523 " -qq No output except for errors\n"
2524 " -d Download only - do NOT install or unpack archives\n"
2525 " -s No-act. Perform ordering simulation\n"
2526 " -y Assume Yes to all queries and do not prompt\n"
2527 " -f Attempt to continue if the integrity check fails\n"
2528 " -m Attempt to continue if archives are unlocatable\n"
2529 " -u Show a list of upgraded packages as well\n"
2530 " -b Build the source package after fetching it\n"
2531 " -V Show verbose version numbers\n"
2532 " -c=? Read this configuration file\n"
2533 " -o=? Set an arbitrary configuration option, eg -o dir::cache=/tmp\n"
2534 "See the apt-get(8), sources.list(5) and apt.conf(5) manual\n"
2535 "pages for more information and options.\n"
2536 " This APT has Super Cow Powers.\n");
2537 return true;
2538 }
2539 /*}}}*/
2540 // GetInitialize - Initialize things for apt-get /*{{{*/
2541 // ---------------------------------------------------------------------
2542 /* */
2543 void GetInitialize()
2544 {
2545 _config->Set("quiet",0);
2546 _config->Set("help",false);
2547 _config->Set("APT::Get::Download-Only",false);
2548 _config->Set("APT::Get::Simulate",false);
2549 _config->Set("APT::Get::Assume-Yes",false);
2550 _config->Set("APT::Get::Fix-Broken",false);
2551 _config->Set("APT::Get::Force-Yes",false);
2552 _config->Set("APT::Get::List-Cleanup",true);
2553 _config->Set("APT::Get::AutomaticRemove",false);
2554 }
2555 /*}}}*/
2556 // SigWinch - Window size change signal handler /*{{{*/
2557 // ---------------------------------------------------------------------
2558 /* */
2559 void SigWinch(int)
2560 {
2561 // Riped from GNU ls
2562 #ifdef TIOCGWINSZ
2563 struct winsize ws;
2564
2565 if (ioctl(1, TIOCGWINSZ, &ws) != -1 && ws.ws_col >= 5)
2566 ScreenWidth = ws.ws_col - 1;
2567 #endif
2568 }
2569 /*}}}*/
2570
2571 int main(int argc,const char *argv[])
2572 {
2573 CommandLine::Args Args[] = {
2574 {'h',"help","help",0},
2575 {'v',"version","version",0},
2576 {'V',"verbose-versions","APT::Get::Show-Versions",0},
2577 {'q',"quiet","quiet",CommandLine::IntLevel},
2578 {'q',"silent","quiet",CommandLine::IntLevel},
2579 {'d',"download-only","APT::Get::Download-Only",0},
2580 {'b',"compile","APT::Get::Compile",0},
2581 {'b',"build","APT::Get::Compile",0},
2582 {'s',"simulate","APT::Get::Simulate",0},
2583 {'s',"just-print","APT::Get::Simulate",0},
2584 {'s',"recon","APT::Get::Simulate",0},
2585 {'s',"dry-run","APT::Get::Simulate",0},
2586 {'s',"no-act","APT::Get::Simulate",0},
2587 {'y',"yes","APT::Get::Assume-Yes",0},
2588 {'y',"assume-yes","APT::Get::Assume-Yes",0},
2589 {'f',"fix-broken","APT::Get::Fix-Broken",0},
2590 {'u',"show-upgraded","APT::Get::Show-Upgraded",0},
2591 {'m',"ignore-missing","APT::Get::Fix-Missing",0},
2592 {'t',"target-release","APT::Default-Release",CommandLine::HasArg},
2593 {'t',"default-release","APT::Default-Release",CommandLine::HasArg},
2594 {0,"download","APT::Get::Download",0},
2595 {0,"fix-missing","APT::Get::Fix-Missing",0},
2596 {0,"ignore-hold","APT::Ignore-Hold",0},
2597 {0,"upgrade","APT::Get::upgrade",0},
2598 {0,"force-yes","APT::Get::force-yes",0},
2599 {0,"print-uris","APT::Get::Print-URIs",0},
2600 {0,"diff-only","APT::Get::Diff-Only",0},
2601 {0,"tar-only","APT::Get::tar-Only",0},
2602 {0,"purge","APT::Get::Purge",0},
2603 {0,"list-cleanup","APT::Get::List-Cleanup",0},
2604 {0,"reinstall","APT::Get::ReInstall",0},
2605 {0,"trivial-only","APT::Get::Trivial-Only",0},
2606 {0,"remove","APT::Get::Remove",0},
2607 {0,"only-source","APT::Get::Only-Source",0},
2608 {0,"arch-only","APT::Get::Arch-Only",0},
2609 {0,"experimental-automatic-remove","APT::Get::AutomaticRemove",0},
2610 {0,"allow-unauthenticated","APT::Get::AllowUnauthenticated",0},
2611 {'c',"config-file",0,CommandLine::ConfigFile},
2612 {'o',"option",0,CommandLine::ArbItem},
2613 {0,0,0,0}};
2614 CommandLine::Dispatch Cmds[] = {{"update",&DoUpdate},
2615 {"upgrade",&DoUpgrade},
2616 {"install",&DoInstall},
2617 {"remove",&DoInstall},
2618 {"dist-upgrade",&DoDistUpgrade},
2619 {"dselect-upgrade",&DoDSelectUpgrade},
2620 {"build-dep",&DoBuildDep},
2621 {"clean",&DoClean},
2622 {"autoclean",&DoAutoClean},
2623 {"check",&DoCheck},
2624 {"source",&DoSource},
2625 {"moo",&DoMoo},
2626 {"help",&ShowHelp},
2627 {0,0}};
2628
2629 // Set up gettext support
2630 setlocale(LC_ALL,"");
2631 textdomain(PACKAGE);
2632
2633 // Parse the command line and initialize the package library
2634 CommandLine CmdL(Args,_config);
2635 if (pkgInitConfig(*_config) == false ||
2636 CmdL.Parse(argc,argv) == false ||
2637 pkgInitSystem(*_config,_system) == false)
2638 {
2639 if (_config->FindB("version") == true)
2640 ShowHelp(CmdL);
2641
2642 _error->DumpErrors();
2643 return 100;
2644 }
2645
2646 // See if the help should be shown
2647 if (_config->FindB("help") == true ||
2648 _config->FindB("version") == true ||
2649 CmdL.FileSize() == 0)
2650 {
2651 ShowHelp(CmdL);
2652 return 0;
2653 }
2654
2655 // Deal with stdout not being a tty
2656 if (!isatty(STDOUT_FILENO) && _config->FindI("quiet",0) < 1)
2657 _config->Set("quiet","1");
2658
2659 // Setup the output streams
2660 c0out.rdbuf(cout.rdbuf());
2661 c1out.rdbuf(cout.rdbuf());
2662 c2out.rdbuf(cout.rdbuf());
2663 if (_config->FindI("quiet",0) > 0)
2664 c0out.rdbuf(devnull.rdbuf());
2665 if (_config->FindI("quiet",0) > 1)
2666 c1out.rdbuf(devnull.rdbuf());
2667
2668 // Setup the signals
2669 signal(SIGPIPE,SIG_IGN);
2670 signal(SIGWINCH,SigWinch);
2671 SigWinch(0);
2672
2673 // Match the operation
2674 CmdL.DispatchArg(Cmds);
2675
2676 // Print any errors or warnings found during parsing
2677 if (_error->empty() == false)
2678 {
2679 bool Errors = _error->PendingError();
2680 _error->DumpErrors();
2681 return Errors == true?100:0;
2682 }
2683
2684 return 0;
2685 }