]> git.saurik.com Git - apt.git/blob - cmdline/apt-get.cc
f1496c9e27b16b0ee9ef6bef06751efef5d3deb0
[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 _system->UnLock();
998 pkgPackageManager::OrderResult Res = PM->DoInstall();
999 if (Res == pkgPackageManager::Failed || _error->PendingError() == true)
1000 return false;
1001 if (Res == pkgPackageManager::Completed)
1002 return true;
1003
1004 // Reload the fetcher object and loop again for media swapping
1005 Fetcher.Shutdown();
1006 if (PM->GetArchives(&Fetcher,&List,&Recs) == false)
1007 return false;
1008
1009 _system->Lock();
1010 }
1011 }
1012 /*}}}*/
1013 // TryToInstall - Try to install a single package /*{{{*/
1014 // ---------------------------------------------------------------------
1015 /* This used to be inlined in DoInstall, but with the advent of regex package
1016 name matching it was split out.. */
1017 bool TryToInstall(pkgCache::PkgIterator Pkg,pkgDepCache &Cache,
1018 pkgProblemResolver &Fix,bool Remove,bool BrokenFix,
1019 unsigned int &ExpectedInst,bool AllowFail = true)
1020 {
1021 /* This is a pure virtual package and there is a single available
1022 provides */
1023 if (Cache[Pkg].CandidateVer == 0 && Pkg->ProvidesList != 0 &&
1024 Pkg.ProvidesList()->NextProvides == 0)
1025 {
1026 pkgCache::PkgIterator Tmp = Pkg.ProvidesList().OwnerPkg();
1027 ioprintf(c1out,_("Note, selecting %s instead of %s\n"),
1028 Tmp.Name(),Pkg.Name());
1029 Pkg = Tmp;
1030 }
1031
1032 // Handle the no-upgrade case
1033 if (_config->FindB("APT::Get::upgrade",true) == false &&
1034 Pkg->CurrentVer != 0)
1035 {
1036 if (AllowFail == true)
1037 ioprintf(c1out,_("Skipping %s, it is already installed and upgrade is not set.\n"),
1038 Pkg.Name());
1039 return true;
1040 }
1041
1042 // Check if there is something at all to install
1043 pkgDepCache::StateCache &State = Cache[Pkg];
1044 if (Remove == true && Pkg->CurrentVer == 0)
1045 {
1046 Fix.Clear(Pkg);
1047 Fix.Protect(Pkg);
1048 Fix.Remove(Pkg);
1049
1050 /* We want to continue searching for regex hits, so we return false here
1051 otherwise this is not really an error. */
1052 if (AllowFail == false)
1053 return false;
1054
1055 ioprintf(c1out,_("Package %s is not installed, so not removed\n"),Pkg.Name());
1056 return true;
1057 }
1058
1059 if (State.CandidateVer == 0 && Remove == false)
1060 {
1061 if (AllowFail == false)
1062 return false;
1063
1064 if (Pkg->ProvidesList != 0)
1065 {
1066 ioprintf(c1out,_("Package %s is a virtual package provided by:\n"),
1067 Pkg.Name());
1068
1069 pkgCache::PrvIterator I = Pkg.ProvidesList();
1070 for (; I.end() == false; I++)
1071 {
1072 pkgCache::PkgIterator Pkg = I.OwnerPkg();
1073
1074 if (Cache[Pkg].CandidateVerIter(Cache) == I.OwnerVer())
1075 {
1076 if (Cache[Pkg].Install() == true && Cache[Pkg].NewInstall() == false)
1077 c1out << " " << Pkg.Name() << " " << I.OwnerVer().VerStr() <<
1078 _(" [Installed]") << endl;
1079 else
1080 c1out << " " << Pkg.Name() << " " << I.OwnerVer().VerStr() << endl;
1081 }
1082 }
1083 c1out << _("You should explicitly select one to install.") << endl;
1084 }
1085 else
1086 {
1087 ioprintf(c1out,
1088 _("Package %s is not available, but is referred to by another package.\n"
1089 "This may mean that the package is missing, has been obsoleted, or\n"
1090 "is only available from another source\n"),Pkg.Name());
1091
1092 string List;
1093 string VersionsList;
1094 SPtrArray<bool> Seen = new bool[Cache.Head().PackageCount];
1095 memset(Seen,0,Cache.Head().PackageCount*sizeof(*Seen));
1096 pkgCache::DepIterator Dep = Pkg.RevDependsList();
1097 for (; Dep.end() == false; Dep++)
1098 {
1099 if (Dep->Type != pkgCache::Dep::Replaces)
1100 continue;
1101 if (Seen[Dep.ParentPkg()->ID] == true)
1102 continue;
1103 Seen[Dep.ParentPkg()->ID] = true;
1104 List += string(Dep.ParentPkg().Name()) + " ";
1105 //VersionsList += string(Dep.ParentPkg().CurVersion) + "\n"; ???
1106 }
1107 ShowList(c1out,_("However the following packages replace it:"),List,VersionsList);
1108 }
1109
1110 _error->Error(_("Package %s has no installation candidate"),Pkg.Name());
1111 return false;
1112 }
1113
1114 Fix.Clear(Pkg);
1115 Fix.Protect(Pkg);
1116 if (Remove == true)
1117 {
1118 Fix.Remove(Pkg);
1119 Cache.MarkDelete(Pkg,_config->FindB("APT::Get::Purge",false));
1120 return true;
1121 }
1122
1123 // Install it
1124 Cache.MarkInstall(Pkg,false);
1125 if (State.Install() == false)
1126 {
1127 if (_config->FindB("APT::Get::ReInstall",false) == true)
1128 {
1129 if (Pkg->CurrentVer == 0 || Pkg.CurrentVer().Downloadable() == false)
1130 ioprintf(c1out,_("Reinstallation of %s is not possible, it cannot be downloaded.\n"),
1131 Pkg.Name());
1132 else
1133 Cache.SetReInstall(Pkg,true);
1134 }
1135 else
1136 {
1137 if (AllowFail == true)
1138 ioprintf(c1out,_("%s is already the newest version.\n"),
1139 Pkg.Name());
1140 }
1141 }
1142 else
1143 ExpectedInst++;
1144
1145 // Install it with autoinstalling enabled.
1146 if (State.InstBroken() == true && BrokenFix == false)
1147 Cache.MarkInstall(Pkg,true);
1148 return true;
1149 }
1150 /*}}}*/
1151 // TryToChangeVer - Try to change a candidate version /*{{{*/
1152 // ---------------------------------------------------------------------
1153 /* */
1154 bool TryToChangeVer(pkgCache::PkgIterator Pkg,pkgDepCache &Cache,
1155 const char *VerTag,bool IsRel)
1156 {
1157 pkgVersionMatch Match(VerTag,(IsRel == true?pkgVersionMatch::Release :
1158 pkgVersionMatch::Version));
1159
1160 pkgCache::VerIterator Ver = Match.Find(Pkg);
1161
1162 if (Ver.end() == true)
1163 {
1164 if (IsRel == true)
1165 return _error->Error(_("Release '%s' for '%s' was not found"),
1166 VerTag,Pkg.Name());
1167 return _error->Error(_("Version '%s' for '%s' was not found"),
1168 VerTag,Pkg.Name());
1169 }
1170
1171 if (strcmp(VerTag,Ver.VerStr()) != 0)
1172 {
1173 ioprintf(c1out,_("Selected version %s (%s) for %s\n"),
1174 Ver.VerStr(),Ver.RelStr().c_str(),Pkg.Name());
1175 }
1176
1177 Cache.SetCandidateVersion(Ver);
1178 return true;
1179 }
1180 /*}}}*/
1181 // FindSrc - Find a source record /*{{{*/
1182 // ---------------------------------------------------------------------
1183 /* */
1184 pkgSrcRecords::Parser *FindSrc(const char *Name,pkgRecords &Recs,
1185 pkgSrcRecords &SrcRecs,string &Src,
1186 pkgDepCache &Cache)
1187 {
1188 // We want to pull the version off the package specification..
1189 string VerTag;
1190 string TmpSrc = Name;
1191 string::size_type Slash = TmpSrc.rfind('=');
1192 if (Slash != string::npos)
1193 {
1194 VerTag = string(TmpSrc.begin() + Slash + 1,TmpSrc.end());
1195 TmpSrc = string(TmpSrc.begin(),TmpSrc.begin() + Slash);
1196 }
1197
1198 /* Lookup the version of the package we would install if we were to
1199 install a version and determine the source package name, then look
1200 in the archive for a source package of the same name. In theory
1201 we could stash the version string as well and match that too but
1202 today there aren't multi source versions in the archive. */
1203 if (_config->FindB("APT::Get::Only-Source") == false &&
1204 VerTag.empty() == true)
1205 {
1206 pkgCache::PkgIterator Pkg = Cache.FindPkg(TmpSrc);
1207 if (Pkg.end() == false)
1208 {
1209 pkgCache::VerIterator Ver = Cache.GetCandidateVer(Pkg);
1210 if (Ver.end() == false)
1211 {
1212 pkgRecords::Parser &Parse = Recs.Lookup(Ver.FileList());
1213 Src = Parse.SourcePkg();
1214 }
1215 }
1216 }
1217
1218 // No source package name..
1219 if (Src.empty() == true)
1220 Src = TmpSrc;
1221
1222 // The best hit
1223 pkgSrcRecords::Parser *Last = 0;
1224 unsigned long Offset = 0;
1225 string Version;
1226 bool IsMatch = false;
1227
1228 // If we are matching by version then we need exact matches to be happy
1229 if (VerTag.empty() == false)
1230 IsMatch = true;
1231
1232 /* Iterate over all of the hits, which includes the resulting
1233 binary packages in the search */
1234 pkgSrcRecords::Parser *Parse;
1235 SrcRecs.Restart();
1236 while ((Parse = SrcRecs.Find(Src.c_str(),false)) != 0)
1237 {
1238 string Ver = Parse->Version();
1239
1240 // Skip name mismatches
1241 if (IsMatch == true && Parse->Package() != Src)
1242 continue;
1243
1244 if (VerTag.empty() == false)
1245 {
1246 /* Don't want to fall through because we are doing exact version
1247 matching. */
1248 if (Cache.VS().CmpVersion(VerTag,Ver) != 0)
1249 continue;
1250
1251 Last = Parse;
1252 Offset = Parse->Offset();
1253 break;
1254 }
1255
1256 // Newer version or an exact match
1257 if (Last == 0 || Cache.VS().CmpVersion(Version,Ver) < 0 ||
1258 (Parse->Package() == Src && IsMatch == false))
1259 {
1260 IsMatch = Parse->Package() == Src;
1261 Last = Parse;
1262 Offset = Parse->Offset();
1263 Version = Ver;
1264 }
1265 }
1266
1267 if (Last == 0)
1268 return 0;
1269
1270 if (Last->Jump(Offset) == false)
1271 return 0;
1272
1273 return Last;
1274 }
1275 /*}}}*/
1276
1277 // DoUpdate - Update the package lists /*{{{*/
1278 // ---------------------------------------------------------------------
1279 /* */
1280 bool DoUpdate(CommandLine &CmdL)
1281 {
1282 if (CmdL.FileSize() != 1)
1283 return _error->Error(_("The update command takes no arguments"));
1284
1285 // Get the source list
1286 pkgSourceList List;
1287 if (List.ReadMainList() == false)
1288 return false;
1289
1290 // Lock the list directory
1291 FileFd Lock;
1292 if (_config->FindB("Debug::NoLocking",false) == false)
1293 {
1294 Lock.Fd(GetLock(_config->FindDir("Dir::State::Lists") + "lock"));
1295 if (_error->PendingError() == true)
1296 return _error->Error(_("Unable to lock the list directory"));
1297 }
1298
1299 // Create the download object
1300 AcqTextStatus Stat(ScreenWidth,_config->FindI("quiet",0));
1301 pkgAcquire Fetcher(&Stat);
1302
1303
1304 // Just print out the uris an exit if the --print-uris flag was used
1305 if (_config->FindB("APT::Get::Print-URIs") == true)
1306 {
1307 // Populate it with the source selection and get all Indexes
1308 // (GetAll=true)
1309 if (List.GetIndexes(&Fetcher,true) == false)
1310 return false;
1311
1312 pkgAcquire::UriIterator I = Fetcher.UriBegin();
1313 for (; I != Fetcher.UriEnd(); I++)
1314 cout << '\'' << I->URI << "' " << flNotDir(I->Owner->DestFile) << ' ' <<
1315 I->Owner->FileSize << ' ' << I->Owner->MD5Sum() << endl;
1316 return true;
1317 }
1318
1319 // Populate it with the source selection
1320 if (List.GetIndexes(&Fetcher) == false)
1321 return false;
1322
1323 // Run it
1324 if (Fetcher.Run() == pkgAcquire::Failed)
1325 return false;
1326
1327 bool Failed = false;
1328 for (pkgAcquire::ItemIterator I = Fetcher.ItemsBegin(); I != Fetcher.ItemsEnd(); I++)
1329 {
1330 if ((*I)->Status == pkgAcquire::Item::StatDone)
1331 continue;
1332
1333 (*I)->Finished();
1334
1335 fprintf(stderr,_("Failed to fetch %s %s\n"),(*I)->DescURI().c_str(),
1336 (*I)->ErrorText.c_str());
1337 Failed = true;
1338 }
1339
1340 // Clean out any old list files
1341 if (_config->FindB("APT::Get::List-Cleanup",true) == true)
1342 {
1343 if (Fetcher.Clean(_config->FindDir("Dir::State::lists")) == false ||
1344 Fetcher.Clean(_config->FindDir("Dir::State::lists") + "partial/") == false)
1345 return false;
1346 }
1347
1348 // Prepare the cache.
1349 CacheFile Cache;
1350 if (Cache.BuildCaches() == false)
1351 return false;
1352
1353 if (Failed == true)
1354 return _error->Error(_("Some index files failed to download, they have been ignored, or old ones used instead."));
1355
1356 return true;
1357 }
1358 /*}}}*/
1359 // DoUpgrade - Upgrade all packages /*{{{*/
1360 // ---------------------------------------------------------------------
1361 /* Upgrade all packages without installing new packages or erasing old
1362 packages */
1363 bool DoUpgrade(CommandLine &CmdL)
1364 {
1365 CacheFile Cache;
1366 if (Cache.OpenForInstall() == false || Cache.CheckDeps() == false)
1367 return false;
1368
1369 // Do the upgrade
1370 if (pkgAllUpgrade(Cache) == false)
1371 {
1372 ShowBroken(c1out,Cache,false);
1373 return _error->Error(_("Internal error, AllUpgrade broke stuff"));
1374 }
1375
1376 return InstallPackages(Cache,true);
1377 }
1378 /*}}}*/
1379 // RecurseDirty - Mark used packages as dirty /*{{{*/
1380 // ---------------------------------------------------------------------
1381 /* Mark all reachable packages as dirty. */
1382 void RecurseDirty (CacheFile &Cache, pkgCache::PkgIterator Pkg, pkgCache::State::PkgRemoveState DirtLevel)
1383 {
1384 // If it is not installed, and we are in manual mode, ignore it
1385 if ((Pkg->CurrentVer == 0 && Cache[Pkg].Install() == false || Cache[Pkg].Delete() == true) &&
1386 DirtLevel == pkgCache::State::RemoveManual)
1387 {
1388 // fprintf(stdout,"This one is not installed/virtual %s %d %d\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel);
1389 return;
1390 }
1391
1392 // If it is not installed, and it is not virtual, ignore it
1393 if ((Pkg->CurrentVer == 0 && Cache[Pkg].Install() == false || Cache[Pkg].Delete() == true) &&
1394 Pkg->VersionList != 0)
1395 {
1396 // fprintf(stdout,"This one is not installed %s %d %d\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel);
1397 return;
1398 }
1399
1400 // If it is similar or more dirty than we are ;-), because we've been here already, don't mark it
1401 // This is necessary because virtual packages just relay the current level,
1402 // so it may be possible e.g. that this was already seen with ::RemoveSuggested, but
1403 // we are ::RemoveRequired
1404 if (Cache[Pkg].Dirty() >= DirtLevel)
1405 {
1406 //fprintf(stdout,"Seen already %s %d %d\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel);
1407 return;
1408 }
1409
1410 // If it is less important than the current DirtLevel, don't mark it
1411 if (Cache[Pkg].AutomaticRemove != pkgCache::State::RemoveManual &&
1412 Cache[Pkg].AutomaticRemove > DirtLevel)
1413 {
1414 // fprintf(stdout,"We don't need %s %d %d %d\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel, Cache[Pkg].Dirty());
1415 return;
1416 }
1417
1418 // Mark it as used
1419 Cache->SetDirty(Pkg, DirtLevel);
1420
1421 //fprintf(stdout,"We keep %s %d %d\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel);
1422
1423 // We are a virtual package
1424 if (Pkg->VersionList == 0)
1425 {
1426 // fprintf(stdout,"We are virtual %s %d %d\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel);
1427 for (pkgCache::PrvIterator Prv = Pkg.ProvidesList(); ! Prv.end(); ++Prv)
1428 RecurseDirty (Cache, Prv.OwnerPkg(), DirtLevel);
1429 return;
1430 }
1431
1432 // Depending on the type of dependency, follow it
1433 for (pkgCache::DepIterator D = Cache[Pkg].InstVerIter(Cache).DependsList(); ! D.end(); ++D)
1434 {
1435 // fprintf(stdout,"We depend on %s %s\n", D.TargetPkg().Name(), D.DepType());
1436
1437 switch(D->Type)
1438 {
1439 case pkgCache::Dep::Depends:
1440 case pkgCache::Dep::PreDepends:
1441 RecurseDirty (Cache, D.TargetPkg(), pkgCache::State::RemoveRequired);
1442 break;
1443 case pkgCache::Dep::Recommends:
1444 RecurseDirty (Cache, D.TargetPkg(), pkgCache::State::RemoveRecommended);
1445 break;
1446 case pkgCache::Dep::Suggests:
1447 RecurseDirty (Cache, D.TargetPkg(), pkgCache::State::RemoveSuggested);
1448 break;
1449 case pkgCache::Dep::Conflicts:
1450 case pkgCache::Dep::Replaces:
1451 case pkgCache::Dep::Obsoletes:
1452 // We don't handle these here
1453 break;
1454 }
1455 }
1456 // fprintf(stdout,"We keep %s %d %d <END>\n", Pkg.Name(), Pkg->AutomaticRemove, DirtLevel);
1457 }
1458 /*}}}*/
1459 // DoAutomaticRemove - Remove all automatic unused packages /*{{{*/
1460 // ---------------------------------------------------------------------
1461 /* Remove unused automatic packages */
1462 bool DoAutomaticRemove(CacheFile &Cache)
1463 {
1464 std::cout << "DoAutomaticRemove()" << std::endl;
1465 for (pkgCache::PkgIterator Pkg = Cache->PkgBegin(); ! Pkg.end(); ++Pkg)
1466 if(!Cache[Pkg].Dirty() && Cache[Pkg].AutomaticRemove > 0)
1467 std::cout << "has auto-remove information: " << Pkg.Name()
1468 << " " << (int)Cache[Pkg].AutomaticRemove
1469 << std::endl;
1470
1471
1472 if (_config->FindB("APT::Get::Remove",true) == false)
1473 return _error->Error(_("We are not supposed to delete stuff, can't start AutoRemover"));
1474
1475 for (pkgCache::PkgIterator Pkg = Cache->PkgBegin(); ! Pkg.end(); ++Pkg)
1476 Cache->SetDirty(Pkg, pkgCache::State::RemoveUnknown);
1477
1478 for (pkgCache::PkgIterator Pkg = Cache->PkgBegin(); ! Pkg.end(); ++Pkg)
1479 RecurseDirty (Cache, Pkg, pkgCache::State::RemoveManual);
1480
1481
1482
1483 for (pkgCache::PkgIterator Pkg = Cache->PkgBegin(); ! Pkg.end(); ++Pkg)
1484 {
1485 if (! Cache[Pkg].Dirty() &&
1486 (Pkg->CurrentVer != 0 && Cache[Pkg].Install() == false && Cache[Pkg].Delete() == false))
1487 {
1488 fprintf(stdout,"We could delete %s %d\n", Pkg.Name(), Cache[Pkg].AutomaticRemove);
1489 Cache->MarkDelete(Pkg,_config->FindB("APT::Get::Purge",false));
1490 }
1491 }
1492
1493 // Now see if we destroyed anything
1494 if (Cache->BrokenCount() != 0)
1495 {
1496 c1out << _("Hmm, seems like the AutoRemover destroyed something which really\n"
1497 "shouldn't happen. Please file a bug report against apt.") << endl;
1498 c1out << endl;
1499 c1out << _("The following information may help to resolve the situation:") << endl;
1500 c1out << endl;
1501 ShowBroken(c1out,Cache,false);
1502
1503 return _error->Error(_("Internal Error, AutoRemover broke stuff"));
1504 }
1505 return true;
1506 }
1507 /*}}}*/
1508 // DoInstall - Install packages from the command line /*{{{*/
1509 // ---------------------------------------------------------------------
1510 /* Install named packages */
1511 bool DoInstall(CommandLine &CmdL)
1512 {
1513 CacheFile Cache;
1514 if (Cache.OpenForInstall() == false ||
1515 Cache.CheckDeps(CmdL.FileSize() != 1) == false)
1516 return false;
1517
1518 // Enter the special broken fixing mode if the user specified arguments
1519 bool BrokenFix = false;
1520 if (Cache->BrokenCount() != 0)
1521 BrokenFix = true;
1522
1523 unsigned int ExpectedInst = 0;
1524 unsigned int Packages = 0;
1525 pkgProblemResolver Fix(Cache);
1526
1527 bool DefRemove = false;
1528 if (strcasecmp(CmdL.FileList[0],"remove") == 0)
1529 DefRemove = true;
1530
1531 for (const char **I = CmdL.FileList + 1; *I != 0; I++)
1532 {
1533 // Duplicate the string
1534 unsigned int Length = strlen(*I);
1535 char S[300];
1536 if (Length >= sizeof(S))
1537 continue;
1538 strcpy(S,*I);
1539
1540 // See if we are removing and special indicators..
1541 bool Remove = DefRemove;
1542 char *VerTag = 0;
1543 bool VerIsRel = false;
1544 while (Cache->FindPkg(S).end() == true)
1545 {
1546 // Handle an optional end tag indicating what to do
1547 if (Length >= 1 && S[Length - 1] == '-')
1548 {
1549 Remove = true;
1550 S[--Length] = 0;
1551 continue;
1552 }
1553
1554 if (Length >= 1 && S[Length - 1] == '+')
1555 {
1556 Remove = false;
1557 S[--Length] = 0;
1558 continue;
1559 }
1560
1561 char *Slash = strchr(S,'=');
1562 if (Slash != 0)
1563 {
1564 VerIsRel = false;
1565 *Slash = 0;
1566 VerTag = Slash + 1;
1567 }
1568
1569 Slash = strchr(S,'/');
1570 if (Slash != 0)
1571 {
1572 VerIsRel = true;
1573 *Slash = 0;
1574 VerTag = Slash + 1;
1575 }
1576
1577 break;
1578 }
1579
1580 // Locate the package
1581 pkgCache::PkgIterator Pkg = Cache->FindPkg(S);
1582 Packages++;
1583 if (Pkg.end() == true)
1584 {
1585 // Check if the name is a regex
1586 const char *I;
1587 for (I = S; *I != 0; I++)
1588 if (*I == '?' || *I == '*' || *I == '|' ||
1589 *I == '[' || *I == '^' || *I == '$')
1590 break;
1591 if (*I == 0)
1592 return _error->Error(_("Couldn't find package %s"),S);
1593
1594 // Regexs must always be confirmed
1595 ExpectedInst += 1000;
1596
1597 // Compile the regex pattern
1598 regex_t Pattern;
1599 int Res;
1600 if ((Res = regcomp(&Pattern,S,REG_EXTENDED | REG_ICASE |
1601 REG_NOSUB)) != 0)
1602 {
1603 char Error[300];
1604 regerror(Res,&Pattern,Error,sizeof(Error));
1605 return _error->Error(_("Regex compilation error - %s"),Error);
1606 }
1607
1608 // Run over the matches
1609 bool Hit = false;
1610 for (Pkg = Cache->PkgBegin(); Pkg.end() == false; Pkg++)
1611 {
1612 if (regexec(&Pattern,Pkg.Name(),0,0,0) != 0)
1613 continue;
1614
1615 ioprintf(c1out,_("Note, selecting %s for regex '%s'\n"),
1616 Pkg.Name(),S);
1617
1618 if (VerTag != 0)
1619 if (TryToChangeVer(Pkg,Cache,VerTag,VerIsRel) == false)
1620 return false;
1621
1622 Hit |= TryToInstall(Pkg,Cache,Fix,Remove,BrokenFix,
1623 ExpectedInst,false);
1624 }
1625 regfree(&Pattern);
1626
1627 if (Hit == false)
1628 return _error->Error(_("Couldn't find package %s"),S);
1629 }
1630 else
1631 {
1632 if (VerTag != 0)
1633 if (TryToChangeVer(Pkg,Cache,VerTag,VerIsRel) == false)
1634 return false;
1635 if (TryToInstall(Pkg,Cache,Fix,Remove,BrokenFix,ExpectedInst) == false)
1636 return false;
1637 }
1638 }
1639
1640 /* If we are in the Broken fixing mode we do not attempt to fix the
1641 problems. This is if the user invoked install without -f and gave
1642 packages */
1643 if (BrokenFix == true && Cache->BrokenCount() != 0)
1644 {
1645 c1out << _("You might want to run `apt-get -f install' to correct these:") << endl;
1646 ShowBroken(c1out,Cache,false);
1647
1648 return _error->Error(_("Unmet dependencies. Try 'apt-get -f install' with no packages (or specify a solution)."));
1649 }
1650
1651 // Call the scored problem resolver
1652 Fix.InstallProtect();
1653 if (Fix.Resolve(true) == false)
1654 _error->Discard();
1655
1656 // Now we check the state of the packages,
1657 if (Cache->BrokenCount() != 0)
1658 {
1659 c1out <<
1660 _("Some packages could not be installed. This may mean that you have\n"
1661 "requested an impossible situation or if you are using the unstable\n"
1662 "distribution that some required packages have not yet been created\n"
1663 "or been moved out of Incoming.") << endl;
1664 if (Packages == 1)
1665 {
1666 c1out << endl;
1667 c1out <<
1668 _("Since you only requested a single operation it is extremely likely that\n"
1669 "the package is simply not installable and a bug report against\n"
1670 "that package should be filed.") << endl;
1671 }
1672
1673 c1out << _("The following information may help to resolve the situation:") << endl;
1674 c1out << endl;
1675 ShowBroken(c1out,Cache,false);
1676 return _error->Error(_("Broken packages"));
1677 }
1678
1679 //if (_config->FindB("APT::Get::AutomaticRemove")) {
1680 if (!DoAutomaticRemove(Cache))
1681 return false;
1682 //}
1683
1684 /* Print out a list of packages that are going to be installed extra
1685 to what the user asked */
1686 if (Cache->InstCount() != ExpectedInst)
1687 {
1688 string List;
1689 string VersionsList;
1690 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
1691 {
1692 pkgCache::PkgIterator I(Cache,Cache.List[J]);
1693 if ((*Cache)[I].Install() == false)
1694 continue;
1695
1696 const char **J;
1697 for (J = CmdL.FileList + 1; *J != 0; J++)
1698 if (strcmp(*J,I.Name()) == 0)
1699 break;
1700
1701 if (*J == 0) {
1702 List += string(I.Name()) + " ";
1703 VersionsList += string(Cache[I].CandVersion) + "\n";
1704 }
1705 }
1706
1707 ShowList(c1out,_("The following extra packages will be installed:"),List,VersionsList);
1708 }
1709
1710 /* Print out a list of suggested and recommended packages */
1711 {
1712 string SuggestsList, RecommendsList, List;
1713 string SuggestsVersions, RecommendsVersions;
1714 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
1715 {
1716 pkgCache::PkgIterator I(Cache,Cache.List[J]);
1717
1718 /* Just look at the ones we want to install */
1719 if ((*Cache)[I].Install() == false)
1720 continue;
1721
1722 for (pkgCache::VerIterator V = I.VersionList(); V.end() == false; V++)
1723 {
1724 for (pkgCache::DepIterator D = V.DependsList(); D.end() == false; )
1725 {
1726 pkgCache::DepIterator Start;
1727 pkgCache::DepIterator End;
1728 D.GlobOr(Start,End); // advances D
1729
1730 /*
1731 * If this is a virtual package, we need to check the list of
1732 * packages that provide it and see if any of those are
1733 * installed
1734 */
1735
1736 bool providedBySomething = false;
1737 for (pkgCache::PrvIterator Prv = Start.TargetPkg().ProvidesList();
1738 Prv.end() != true;
1739 Prv++)
1740 if ((*Cache)[Prv.OwnerPkg()].InstVerIter(*Cache).end() == false)
1741 {
1742 providedBySomething = true;
1743 break;
1744 }
1745
1746 if (providedBySomething) continue;
1747
1748 for(;;)
1749 {
1750 /* Skip if package is installed already, or is about to be */
1751 string target = string(Start.TargetPkg().Name()) + " ";
1752
1753 if ((*Start.TargetPkg()).SelectedState == pkgCache::State::Install
1754 || Cache[Start.TargetPkg()].Install())
1755 break;
1756
1757 /* Skip if we already saw it */
1758 if (int(SuggestsList.find(target)) != -1 || int(RecommendsList.find(target)) != -1)
1759 break;
1760
1761 if (Start->Type == pkgCache::Dep::Suggests) {
1762 SuggestsList += target;
1763 SuggestsVersions += string(Cache[Start.TargetPkg()].CandVersion) + "\n";
1764 }
1765
1766 if (Start->Type == pkgCache::Dep::Recommends) {
1767 RecommendsList += target;
1768 RecommendsVersions += string(Cache[Start.TargetPkg()].CandVersion) + "\n";
1769 }
1770
1771 if (Start >= End)
1772 break;
1773 Start++;
1774 }
1775 }
1776 }
1777 }
1778 ShowList(c1out,_("Suggested packages:"),SuggestsList,SuggestsVersions);
1779 ShowList(c1out,_("Recommended packages:"),RecommendsList,RecommendsVersions);
1780
1781 }
1782
1783 // See if we need to prompt
1784 if (Cache->InstCount() == ExpectedInst && Cache->DelCount() == 0)
1785 return InstallPackages(Cache,false,false);
1786
1787 return InstallPackages(Cache,false);
1788 }
1789 /*}}}*/
1790 // DoDistUpgrade - Automatic smart upgrader /*{{{*/
1791 // ---------------------------------------------------------------------
1792 /* Intelligent upgrader that will install and remove packages at will */
1793 bool DoDistUpgrade(CommandLine &CmdL)
1794 {
1795 CacheFile Cache;
1796 if (Cache.OpenForInstall() == false || Cache.CheckDeps() == false)
1797 return false;
1798
1799 c0out << _("Calculating upgrade... ") << flush;
1800 if (pkgDistUpgrade(*Cache) == false)
1801 {
1802 c0out << _("Failed") << endl;
1803 ShowBroken(c1out,Cache,false);
1804 return false;
1805 }
1806
1807 c0out << _("Done") << endl;
1808
1809 return InstallPackages(Cache,true);
1810 }
1811 /*}}}*/
1812 // DoDSelectUpgrade - Do an upgrade by following dselects selections /*{{{*/
1813 // ---------------------------------------------------------------------
1814 /* Follows dselect's selections */
1815 bool DoDSelectUpgrade(CommandLine &CmdL)
1816 {
1817 CacheFile Cache;
1818 if (Cache.OpenForInstall() == false || Cache.CheckDeps() == false)
1819 return false;
1820
1821 // Install everything with the install flag set
1822 pkgCache::PkgIterator I = Cache->PkgBegin();
1823 for (;I.end() != true; I++)
1824 {
1825 /* Install the package only if it is a new install, the autoupgrader
1826 will deal with the rest */
1827 if (I->SelectedState == pkgCache::State::Install)
1828 Cache->MarkInstall(I,false);
1829 }
1830
1831 /* Now install their deps too, if we do this above then order of
1832 the status file is significant for | groups */
1833 for (I = Cache->PkgBegin();I.end() != true; I++)
1834 {
1835 /* Install the package only if it is a new install, the autoupgrader
1836 will deal with the rest */
1837 if (I->SelectedState == pkgCache::State::Install)
1838 Cache->MarkInstall(I,true);
1839 }
1840
1841 // Apply erasures now, they override everything else.
1842 for (I = Cache->PkgBegin();I.end() != true; I++)
1843 {
1844 // Remove packages
1845 if (I->SelectedState == pkgCache::State::DeInstall ||
1846 I->SelectedState == pkgCache::State::Purge)
1847 Cache->MarkDelete(I,I->SelectedState == pkgCache::State::Purge);
1848 }
1849
1850 /* Resolve any problems that dselect created, allupgrade cannot handle
1851 such things. We do so quite agressively too.. */
1852 if (Cache->BrokenCount() != 0)
1853 {
1854 pkgProblemResolver Fix(Cache);
1855
1856 // Hold back held packages.
1857 if (_config->FindB("APT::Ignore-Hold",false) == false)
1858 {
1859 for (pkgCache::PkgIterator I = Cache->PkgBegin(); I.end() == false; I++)
1860 {
1861 if (I->SelectedState == pkgCache::State::Hold)
1862 {
1863 Fix.Protect(I);
1864 Cache->MarkKeep(I);
1865 }
1866 }
1867 }
1868
1869 if (Fix.Resolve() == false)
1870 {
1871 ShowBroken(c1out,Cache,false);
1872 return _error->Error("Internal error, problem resolver broke stuff");
1873 }
1874 }
1875
1876 // Now upgrade everything
1877 if (pkgAllUpgrade(Cache) == false)
1878 {
1879 ShowBroken(c1out,Cache,false);
1880 return _error->Error("Internal error, problem resolver broke stuff");
1881 }
1882
1883 return InstallPackages(Cache,false);
1884 }
1885 /*}}}*/
1886 // DoClean - Remove download archives /*{{{*/
1887 // ---------------------------------------------------------------------
1888 /* */
1889 bool DoClean(CommandLine &CmdL)
1890 {
1891 if (_config->FindB("APT::Get::Simulate") == true)
1892 {
1893 cout << "Del " << _config->FindDir("Dir::Cache::archives") << "* " <<
1894 _config->FindDir("Dir::Cache::archives") << "partial/*" << endl;
1895 return true;
1896 }
1897
1898 // Lock the archive directory
1899 FileFd Lock;
1900 if (_config->FindB("Debug::NoLocking",false) == false)
1901 {
1902 Lock.Fd(GetLock(_config->FindDir("Dir::Cache::Archives") + "lock"));
1903 if (_error->PendingError() == true)
1904 return _error->Error(_("Unable to lock the download directory"));
1905 }
1906
1907 pkgAcquire Fetcher;
1908 Fetcher.Clean(_config->FindDir("Dir::Cache::archives"));
1909 Fetcher.Clean(_config->FindDir("Dir::Cache::archives") + "partial/");
1910 return true;
1911 }
1912 /*}}}*/
1913 // DoAutoClean - Smartly remove downloaded archives /*{{{*/
1914 // ---------------------------------------------------------------------
1915 /* This is similar to clean but it only purges things that cannot be
1916 downloaded, that is old versions of cached packages. */
1917 class LogCleaner : public pkgArchiveCleaner
1918 {
1919 protected:
1920 virtual void Erase(const char *File,string Pkg,string Ver,struct stat &St)
1921 {
1922 c1out << "Del " << Pkg << " " << Ver << " [" << SizeToStr(St.st_size) << "B]" << endl;
1923
1924 if (_config->FindB("APT::Get::Simulate") == false)
1925 unlink(File);
1926 };
1927 };
1928
1929 bool DoAutoClean(CommandLine &CmdL)
1930 {
1931 // Lock the archive directory
1932 FileFd Lock;
1933 if (_config->FindB("Debug::NoLocking",false) == false)
1934 {
1935 Lock.Fd(GetLock(_config->FindDir("Dir::Cache::Archives") + "lock"));
1936 if (_error->PendingError() == true)
1937 return _error->Error(_("Unable to lock the download directory"));
1938 }
1939
1940 CacheFile Cache;
1941 if (Cache.Open() == false)
1942 return false;
1943
1944 LogCleaner Cleaner;
1945
1946 return Cleaner.Go(_config->FindDir("Dir::Cache::archives"),*Cache) &&
1947 Cleaner.Go(_config->FindDir("Dir::Cache::archives") + "partial/",*Cache);
1948 }
1949 /*}}}*/
1950 // DoCheck - Perform the check operation /*{{{*/
1951 // ---------------------------------------------------------------------
1952 /* Opening automatically checks the system, this command is mostly used
1953 for debugging */
1954 bool DoCheck(CommandLine &CmdL)
1955 {
1956 CacheFile Cache;
1957 Cache.Open();
1958 Cache.CheckDeps();
1959
1960 return true;
1961 }
1962 /*}}}*/
1963 // DoSource - Fetch a source archive /*{{{*/
1964 // ---------------------------------------------------------------------
1965 /* Fetch souce packages */
1966 struct DscFile
1967 {
1968 string Package;
1969 string Version;
1970 string Dsc;
1971 };
1972
1973 bool DoSource(CommandLine &CmdL)
1974 {
1975 CacheFile Cache;
1976 if (Cache.Open(false) == false)
1977 return false;
1978
1979 if (CmdL.FileSize() <= 1)
1980 return _error->Error(_("Must specify at least one package to fetch source for"));
1981
1982 // Read the source list
1983 pkgSourceList List;
1984 if (List.ReadMainList() == false)
1985 return _error->Error(_("The list of sources could not be read."));
1986
1987 // Create the text record parsers
1988 pkgRecords Recs(Cache);
1989 pkgSrcRecords SrcRecs(List);
1990 if (_error->PendingError() == true)
1991 return false;
1992
1993 // Create the download object
1994 AcqTextStatus Stat(ScreenWidth,_config->FindI("quiet",0));
1995 pkgAcquire Fetcher(&Stat);
1996
1997 DscFile *Dsc = new DscFile[CmdL.FileSize()];
1998
1999 // Load the requestd sources into the fetcher
2000 unsigned J = 0;
2001 for (const char **I = CmdL.FileList + 1; *I != 0; I++, J++)
2002 {
2003 string Src;
2004 pkgSrcRecords::Parser *Last = FindSrc(*I,Recs,SrcRecs,Src,*Cache);
2005
2006 if (Last == 0)
2007 return _error->Error(_("Unable to find a source package for %s"),Src.c_str());
2008
2009 // Back track
2010 vector<pkgSrcRecords::File> Lst;
2011 if (Last->Files(Lst) == false)
2012 return false;
2013
2014 // Load them into the fetcher
2015 for (vector<pkgSrcRecords::File>::const_iterator I = Lst.begin();
2016 I != Lst.end(); I++)
2017 {
2018 // Try to guess what sort of file it is we are getting.
2019 if (I->Type == "dsc")
2020 {
2021 Dsc[J].Package = Last->Package();
2022 Dsc[J].Version = Last->Version();
2023 Dsc[J].Dsc = flNotDir(I->Path);
2024 }
2025
2026 // Diff only mode only fetches .diff files
2027 if (_config->FindB("APT::Get::Diff-Only",false) == true &&
2028 I->Type != "diff")
2029 continue;
2030
2031 // Tar only mode only fetches .tar files
2032 if (_config->FindB("APT::Get::Tar-Only",false) == true &&
2033 I->Type != "tar")
2034 continue;
2035
2036 new pkgAcqFile(&Fetcher,Last->Index().ArchiveURI(I->Path),
2037 I->MD5Hash,I->Size,
2038 Last->Index().SourceInfo(*Last,*I),Src);
2039 }
2040 }
2041
2042 // Display statistics
2043 double FetchBytes = Fetcher.FetchNeeded();
2044 double FetchPBytes = Fetcher.PartialPresent();
2045 double DebBytes = Fetcher.TotalNeeded();
2046
2047 // Check for enough free space
2048 struct statvfs Buf;
2049 string OutputDir = ".";
2050 if (statvfs(OutputDir.c_str(),&Buf) != 0)
2051 return _error->Errno("statvfs","Couldn't determine free space in %s",
2052 OutputDir.c_str());
2053 if (unsigned(Buf.f_bfree) < (FetchBytes - FetchPBytes)/Buf.f_bsize)
2054 return _error->Error(_("You don't have enough free space in %s"),
2055 OutputDir.c_str());
2056
2057 // Number of bytes
2058 if (DebBytes != FetchBytes)
2059 ioprintf(c1out,_("Need to get %sB/%sB of source archives.\n"),
2060 SizeToStr(FetchBytes).c_str(),SizeToStr(DebBytes).c_str());
2061 else
2062 ioprintf(c1out,_("Need to get %sB of source archives.\n"),
2063 SizeToStr(DebBytes).c_str());
2064
2065 if (_config->FindB("APT::Get::Simulate",false) == true)
2066 {
2067 for (unsigned I = 0; I != J; I++)
2068 ioprintf(cout,_("Fetch source %s\n"),Dsc[I].Package.c_str());
2069 return true;
2070 }
2071
2072 // Just print out the uris an exit if the --print-uris flag was used
2073 if (_config->FindB("APT::Get::Print-URIs") == true)
2074 {
2075 pkgAcquire::UriIterator I = Fetcher.UriBegin();
2076 for (; I != Fetcher.UriEnd(); I++)
2077 cout << '\'' << I->URI << "' " << flNotDir(I->Owner->DestFile) << ' ' <<
2078 I->Owner->FileSize << ' ' << I->Owner->MD5Sum() << endl;
2079 return true;
2080 }
2081
2082 // Run it
2083 if (Fetcher.Run() == pkgAcquire::Failed)
2084 return false;
2085
2086 // Print error messages
2087 bool Failed = false;
2088 for (pkgAcquire::ItemIterator I = Fetcher.ItemsBegin(); I != Fetcher.ItemsEnd(); I++)
2089 {
2090 if ((*I)->Status == pkgAcquire::Item::StatDone &&
2091 (*I)->Complete == true)
2092 continue;
2093
2094 fprintf(stderr,_("Failed to fetch %s %s\n"),(*I)->DescURI().c_str(),
2095 (*I)->ErrorText.c_str());
2096 Failed = true;
2097 }
2098 if (Failed == true)
2099 return _error->Error(_("Failed to fetch some archives."));
2100
2101 if (_config->FindB("APT::Get::Download-only",false) == true)
2102 {
2103 c1out << _("Download complete and in download only mode") << endl;
2104 return true;
2105 }
2106
2107 // Unpack the sources
2108 pid_t Process = ExecFork();
2109
2110 if (Process == 0)
2111 {
2112 for (unsigned I = 0; I != J; I++)
2113 {
2114 string Dir = Dsc[I].Package + '-' + Cache->VS().UpstreamVersion(Dsc[I].Version.c_str());
2115
2116 // Diff only mode only fetches .diff files
2117 if (_config->FindB("APT::Get::Diff-Only",false) == true ||
2118 _config->FindB("APT::Get::Tar-Only",false) == true ||
2119 Dsc[I].Dsc.empty() == true)
2120 continue;
2121
2122 // See if the package is already unpacked
2123 struct stat Stat;
2124 if (stat(Dir.c_str(),&Stat) == 0 &&
2125 S_ISDIR(Stat.st_mode) != 0)
2126 {
2127 ioprintf(c0out ,_("Skipping unpack of already unpacked source in %s\n"),
2128 Dir.c_str());
2129 }
2130 else
2131 {
2132 // Call dpkg-source
2133 char S[500];
2134 snprintf(S,sizeof(S),"%s -x %s",
2135 _config->Find("Dir::Bin::dpkg-source","dpkg-source").c_str(),
2136 Dsc[I].Dsc.c_str());
2137 if (system(S) != 0)
2138 {
2139 fprintf(stderr,_("Unpack command '%s' failed.\n"),S);
2140 _exit(1);
2141 }
2142 }
2143
2144 // Try to compile it with dpkg-buildpackage
2145 if (_config->FindB("APT::Get::Compile",false) == true)
2146 {
2147 // Call dpkg-buildpackage
2148 char S[500];
2149 snprintf(S,sizeof(S),"cd %s && %s %s",
2150 Dir.c_str(),
2151 _config->Find("Dir::Bin::dpkg-buildpackage","dpkg-buildpackage").c_str(),
2152 _config->Find("DPkg::Build-Options","-b -uc").c_str());
2153
2154 if (system(S) != 0)
2155 {
2156 fprintf(stderr,_("Build command '%s' failed.\n"),S);
2157 _exit(1);
2158 }
2159 }
2160 }
2161
2162 _exit(0);
2163 }
2164
2165 // Wait for the subprocess
2166 int Status = 0;
2167 while (waitpid(Process,&Status,0) != Process)
2168 {
2169 if (errno == EINTR)
2170 continue;
2171 return _error->Errno("waitpid","Couldn't wait for subprocess");
2172 }
2173
2174 if (WIFEXITED(Status) == 0 || WEXITSTATUS(Status) != 0)
2175 return _error->Error(_("Child process failed"));
2176
2177 return true;
2178 }
2179 /*}}}*/
2180 // DoBuildDep - Install/removes packages to satisfy build dependencies /*{{{*/
2181 // ---------------------------------------------------------------------
2182 /* This function will look at the build depends list of the given source
2183 package and install the necessary packages to make it true, or fail. */
2184 bool DoBuildDep(CommandLine &CmdL)
2185 {
2186 CacheFile Cache;
2187 if (Cache.Open(true) == false)
2188 return false;
2189
2190 if (CmdL.FileSize() <= 1)
2191 return _error->Error(_("Must specify at least one package to check builddeps for"));
2192
2193 // Read the source list
2194 pkgSourceList List;
2195 if (List.ReadMainList() == false)
2196 return _error->Error(_("The list of sources could not be read."));
2197
2198 // Create the text record parsers
2199 pkgRecords Recs(Cache);
2200 pkgSrcRecords SrcRecs(List);
2201 if (_error->PendingError() == true)
2202 return false;
2203
2204 // Create the download object
2205 AcqTextStatus Stat(ScreenWidth,_config->FindI("quiet",0));
2206 pkgAcquire Fetcher(&Stat);
2207
2208 unsigned J = 0;
2209 for (const char **I = CmdL.FileList + 1; *I != 0; I++, J++)
2210 {
2211 string Src;
2212 pkgSrcRecords::Parser *Last = FindSrc(*I,Recs,SrcRecs,Src,*Cache);
2213 if (Last == 0)
2214 return _error->Error(_("Unable to find a source package for %s"),Src.c_str());
2215
2216 // Process the build-dependencies
2217 vector<pkgSrcRecords::Parser::BuildDepRec> BuildDeps;
2218 if (Last->BuildDepends(BuildDeps, _config->FindB("APT::Get::Arch-Only",false)) == false)
2219 return _error->Error(_("Unable to get build-dependency information for %s"),Src.c_str());
2220
2221 // Also ensure that build-essential packages are present
2222 Configuration::Item const *Opts = _config->Tree("APT::Build-Essential");
2223 if (Opts)
2224 Opts = Opts->Child;
2225 for (; Opts; Opts = Opts->Next)
2226 {
2227 if (Opts->Value.empty() == true)
2228 continue;
2229
2230 pkgSrcRecords::Parser::BuildDepRec rec;
2231 rec.Package = Opts->Value;
2232 rec.Type = pkgSrcRecords::Parser::BuildDependIndep;
2233 rec.Op = 0;
2234 BuildDeps.push_back(rec);
2235 }
2236
2237 if (BuildDeps.size() == 0)
2238 {
2239 ioprintf(c1out,_("%s has no build depends.\n"),Src.c_str());
2240 continue;
2241 }
2242
2243 // Install the requested packages
2244 unsigned int ExpectedInst = 0;
2245 vector <pkgSrcRecords::Parser::BuildDepRec>::iterator D;
2246 pkgProblemResolver Fix(Cache);
2247 bool skipAlternatives = false; // skip remaining alternatives in an or group
2248 for (D = BuildDeps.begin(); D != BuildDeps.end(); D++)
2249 {
2250 bool hasAlternatives = (((*D).Op & pkgCache::Dep::Or) == pkgCache::Dep::Or);
2251
2252 if (skipAlternatives == true)
2253 {
2254 if (!hasAlternatives)
2255 skipAlternatives = false; // end of or group
2256 continue;
2257 }
2258
2259 if ((*D).Type == pkgSrcRecords::Parser::BuildConflict ||
2260 (*D).Type == pkgSrcRecords::Parser::BuildConflictIndep)
2261 {
2262 pkgCache::PkgIterator Pkg = Cache->FindPkg((*D).Package);
2263 // Build-conflicts on unknown packages are silently ignored
2264 if (Pkg.end() == true)
2265 continue;
2266
2267 pkgCache::VerIterator IV = (*Cache)[Pkg].InstVerIter(*Cache);
2268
2269 /*
2270 * Remove if we have an installed version that satisfies the
2271 * version criteria
2272 */
2273 if (IV.end() == false &&
2274 Cache->VS().CheckDep(IV.VerStr(),(*D).Op,(*D).Version.c_str()) == true)
2275 TryToInstall(Pkg,Cache,Fix,true,false,ExpectedInst);
2276 }
2277 else // BuildDep || BuildDepIndep
2278 {
2279 pkgCache::PkgIterator Pkg = Cache->FindPkg((*D).Package);
2280 if (_config->FindB("Debug::BuildDeps",false) == true)
2281 cout << "Looking for " << (*D).Package << "...\n";
2282
2283 if (Pkg.end() == true)
2284 {
2285 if (_config->FindB("Debug::BuildDeps",false) == true)
2286 cout << " (not found)" << (*D).Package << endl;
2287
2288 if (hasAlternatives)
2289 continue;
2290
2291 return _error->Error(_("%s dependency for %s cannot be satisfied "
2292 "because the package %s cannot be found"),
2293 Last->BuildDepType((*D).Type),Src.c_str(),
2294 (*D).Package.c_str());
2295 }
2296
2297 /*
2298 * if there are alternatives, we've already picked one, so skip
2299 * the rest
2300 *
2301 * TODO: this means that if there's a build-dep on A|B and B is
2302 * installed, we'll still try to install A; more importantly,
2303 * if A is currently broken, we cannot go back and try B. To fix
2304 * this would require we do a Resolve cycle for each package we
2305 * add to the install list. Ugh
2306 */
2307
2308 /*
2309 * If this is a virtual package, we need to check the list of
2310 * packages that provide it and see if any of those are
2311 * installed
2312 */
2313 pkgCache::PrvIterator Prv = Pkg.ProvidesList();
2314 for (; Prv.end() != true; Prv++)
2315 {
2316 if (_config->FindB("Debug::BuildDeps",false) == true)
2317 cout << " Checking provider " << Prv.OwnerPkg().Name() << endl;
2318
2319 if ((*Cache)[Prv.OwnerPkg()].InstVerIter(*Cache).end() == false)
2320 break;
2321 }
2322
2323 // Get installed version and version we are going to install
2324 pkgCache::VerIterator IV = (*Cache)[Pkg].InstVerIter(*Cache);
2325
2326 if ((*D).Version[0] != '\0') {
2327 // Versioned dependency
2328
2329 pkgCache::VerIterator CV = (*Cache)[Pkg].CandidateVerIter(*Cache);
2330
2331 for (; CV.end() != true; CV++)
2332 {
2333 if (Cache->VS().CheckDep(CV.VerStr(),(*D).Op,(*D).Version.c_str()) == true)
2334 break;
2335 }
2336 if (CV.end() == true)
2337 if (hasAlternatives)
2338 {
2339 continue;
2340 }
2341 else
2342 {
2343 return _error->Error(_("%s dependency for %s cannot be satisfied "
2344 "because no available versions of package %s "
2345 "can satisfy version requirements"),
2346 Last->BuildDepType((*D).Type),Src.c_str(),
2347 (*D).Package.c_str());
2348 }
2349 }
2350 else
2351 {
2352 // Only consider virtual packages if there is no versioned dependency
2353 if (Prv.end() == false)
2354 {
2355 if (_config->FindB("Debug::BuildDeps",false) == true)
2356 cout << " Is provided by installed package " << Prv.OwnerPkg().Name() << endl;
2357 skipAlternatives = hasAlternatives;
2358 continue;
2359 }
2360 }
2361
2362 if (IV.end() == false)
2363 {
2364 if (_config->FindB("Debug::BuildDeps",false) == true)
2365 cout << " Is installed\n";
2366
2367 if (Cache->VS().CheckDep(IV.VerStr(),(*D).Op,(*D).Version.c_str()) == true)
2368 {
2369 skipAlternatives = hasAlternatives;
2370 continue;
2371 }
2372
2373 if (_config->FindB("Debug::BuildDeps",false) == true)
2374 cout << " ...but the installed version doesn't meet the version requirement\n";
2375
2376 if (((*D).Op & pkgCache::Dep::LessEq) == pkgCache::Dep::LessEq)
2377 {
2378 return _error->Error(_("Failed to satisfy %s dependency for %s: Installed package %s is too new"),
2379 Last->BuildDepType((*D).Type),
2380 Src.c_str(),
2381 Pkg.Name());
2382 }
2383 }
2384
2385
2386 if (_config->FindB("Debug::BuildDeps",false) == true)
2387 cout << " Trying to install " << (*D).Package << endl;
2388
2389 if (TryToInstall(Pkg,Cache,Fix,false,false,ExpectedInst) == true)
2390 {
2391 // We successfully installed something; skip remaining alternatives
2392 skipAlternatives = hasAlternatives;
2393 continue;
2394 }
2395 else if (hasAlternatives)
2396 {
2397 if (_config->FindB("Debug::BuildDeps",false) == true)
2398 cout << " Unsatisfiable, trying alternatives\n";
2399 continue;
2400 }
2401 else
2402 {
2403 return _error->Error(_("Failed to satisfy %s dependency for %s: %s"),
2404 Last->BuildDepType((*D).Type),
2405 Src.c_str(),
2406 (*D).Package.c_str());
2407 }
2408 }
2409 }
2410
2411 Fix.InstallProtect();
2412 if (Fix.Resolve(true) == false)
2413 _error->Discard();
2414
2415 // Now we check the state of the packages,
2416 if (Cache->BrokenCount() != 0)
2417 return _error->Error(_("Build-dependencies for %s could not be satisfied."),*I);
2418 }
2419
2420 if (InstallPackages(Cache, false, true) == false)
2421 return _error->Error(_("Failed to process build dependencies"));
2422 return true;
2423 }
2424 /*}}}*/
2425
2426 // DoMoo - Never Ask, Never Tell /*{{{*/
2427 // ---------------------------------------------------------------------
2428 /* */
2429 bool DoMoo(CommandLine &CmdL)
2430 {
2431 cout <<
2432 " (__) \n"
2433 " (oo) \n"
2434 " /------\\/ \n"
2435 " / | || \n"
2436 " * /\\---/\\ \n"
2437 " ~~ ~~ \n"
2438 "....\"Have you mooed today?\"...\n";
2439
2440 return true;
2441 }
2442 /*}}}*/
2443 // ShowHelp - Show a help screen /*{{{*/
2444 // ---------------------------------------------------------------------
2445 /* */
2446 bool ShowHelp(CommandLine &CmdL)
2447 {
2448 ioprintf(cout,_("%s %s for %s %s compiled on %s %s\n"),PACKAGE,VERSION,
2449 COMMON_OS,COMMON_CPU,__DATE__,__TIME__);
2450
2451 if (_config->FindB("version") == true)
2452 {
2453 cout << _("Supported modules:") << endl;
2454
2455 for (unsigned I = 0; I != pkgVersioningSystem::GlobalListLen; I++)
2456 {
2457 pkgVersioningSystem *VS = pkgVersioningSystem::GlobalList[I];
2458 if (_system != 0 && _system->VS == VS)
2459 cout << '*';
2460 else
2461 cout << ' ';
2462 cout << "Ver: " << VS->Label << endl;
2463
2464 /* Print out all the packaging systems that will work with
2465 this VS */
2466 for (unsigned J = 0; J != pkgSystem::GlobalListLen; J++)
2467 {
2468 pkgSystem *Sys = pkgSystem::GlobalList[J];
2469 if (_system == Sys)
2470 cout << '*';
2471 else
2472 cout << ' ';
2473 if (Sys->VS->TestCompatibility(*VS) == true)
2474 cout << "Pkg: " << Sys->Label << " (Priority " << Sys->Score(*_config) << ")" << endl;
2475 }
2476 }
2477
2478 for (unsigned I = 0; I != pkgSourceList::Type::GlobalListLen; I++)
2479 {
2480 pkgSourceList::Type *Type = pkgSourceList::Type::GlobalList[I];
2481 cout << " S.L: '" << Type->Name << "' " << Type->Label << endl;
2482 }
2483
2484 for (unsigned I = 0; I != pkgIndexFile::Type::GlobalListLen; I++)
2485 {
2486 pkgIndexFile::Type *Type = pkgIndexFile::Type::GlobalList[I];
2487 cout << " Idx: " << Type->Label << endl;
2488 }
2489
2490 return true;
2491 }
2492
2493 cout <<
2494 _("Usage: apt-get [options] command\n"
2495 " apt-get [options] install|remove pkg1 [pkg2 ...]\n"
2496 " apt-get [options] source pkg1 [pkg2 ...]\n"
2497 "\n"
2498 "apt-get is a simple command line interface for downloading and\n"
2499 "installing packages. The most frequently used commands are update\n"
2500 "and install.\n"
2501 "\n"
2502 "Commands:\n"
2503 " update - Retrieve new lists of packages\n"
2504 " upgrade - Perform an upgrade\n"
2505 " install - Install new packages (pkg is libc6 not libc6.deb)\n"
2506 " remove - Remove packages\n"
2507 " source - Download source archives\n"
2508 " build-dep - Configure build-dependencies for source packages\n"
2509 " dist-upgrade - Distribution upgrade, see apt-get(8)\n"
2510 " dselect-upgrade - Follow dselect selections\n"
2511 " clean - Erase downloaded archive files\n"
2512 " autoclean - Erase old downloaded archive files\n"
2513 " check - Verify that there are no broken dependencies\n"
2514 "\n"
2515 "Options:\n"
2516 " -h This help text.\n"
2517 " -q Loggable output - no progress indicator\n"
2518 " -qq No output except for errors\n"
2519 " -d Download only - do NOT install or unpack archives\n"
2520 " -s No-act. Perform ordering simulation\n"
2521 " -y Assume Yes to all queries and do not prompt\n"
2522 " -f Attempt to continue if the integrity check fails\n"
2523 " -m Attempt to continue if archives are unlocatable\n"
2524 " -u Show a list of upgraded packages as well\n"
2525 " -b Build the source package after fetching it\n"
2526 " -V Show verbose version numbers\n"
2527 " -c=? Read this configuration file\n"
2528 " -o=? Set an arbitrary configuration option, eg -o dir::cache=/tmp\n"
2529 "See the apt-get(8), sources.list(5) and apt.conf(5) manual\n"
2530 "pages for more information and options.\n"
2531 " This APT has Super Cow Powers.\n");
2532 return true;
2533 }
2534 /*}}}*/
2535 // GetInitialize - Initialize things for apt-get /*{{{*/
2536 // ---------------------------------------------------------------------
2537 /* */
2538 void GetInitialize()
2539 {
2540 _config->Set("quiet",0);
2541 _config->Set("help",false);
2542 _config->Set("APT::Get::Download-Only",false);
2543 _config->Set("APT::Get::Simulate",false);
2544 _config->Set("APT::Get::Assume-Yes",false);
2545 _config->Set("APT::Get::Fix-Broken",false);
2546 _config->Set("APT::Get::Force-Yes",false);
2547 _config->Set("APT::Get::List-Cleanup",true);
2548 _config->Set("APT::Get::AutomaticRemove",false);
2549 }
2550 /*}}}*/
2551 // SigWinch - Window size change signal handler /*{{{*/
2552 // ---------------------------------------------------------------------
2553 /* */
2554 void SigWinch(int)
2555 {
2556 // Riped from GNU ls
2557 #ifdef TIOCGWINSZ
2558 struct winsize ws;
2559
2560 if (ioctl(1, TIOCGWINSZ, &ws) != -1 && ws.ws_col >= 5)
2561 ScreenWidth = ws.ws_col - 1;
2562 #endif
2563 }
2564 /*}}}*/
2565
2566 int main(int argc,const char *argv[])
2567 {
2568 CommandLine::Args Args[] = {
2569 {'h',"help","help",0},
2570 {'v',"version","version",0},
2571 {'V',"verbose-versions","APT::Get::Show-Versions",0},
2572 {'q',"quiet","quiet",CommandLine::IntLevel},
2573 {'q',"silent","quiet",CommandLine::IntLevel},
2574 {'d',"download-only","APT::Get::Download-Only",0},
2575 {'b',"compile","APT::Get::Compile",0},
2576 {'b',"build","APT::Get::Compile",0},
2577 {'s',"simulate","APT::Get::Simulate",0},
2578 {'s',"just-print","APT::Get::Simulate",0},
2579 {'s',"recon","APT::Get::Simulate",0},
2580 {'s',"dry-run","APT::Get::Simulate",0},
2581 {'s',"no-act","APT::Get::Simulate",0},
2582 {'y',"yes","APT::Get::Assume-Yes",0},
2583 {'y',"assume-yes","APT::Get::Assume-Yes",0},
2584 {'f',"fix-broken","APT::Get::Fix-Broken",0},
2585 {'u',"show-upgraded","APT::Get::Show-Upgraded",0},
2586 {'m',"ignore-missing","APT::Get::Fix-Missing",0},
2587 {'t',"target-release","APT::Default-Release",CommandLine::HasArg},
2588 {'t',"default-release","APT::Default-Release",CommandLine::HasArg},
2589 {0,"download","APT::Get::Download",0},
2590 {0,"fix-missing","APT::Get::Fix-Missing",0},
2591 {0,"ignore-hold","APT::Ignore-Hold",0},
2592 {0,"upgrade","APT::Get::upgrade",0},
2593 {0,"force-yes","APT::Get::force-yes",0},
2594 {0,"print-uris","APT::Get::Print-URIs",0},
2595 {0,"diff-only","APT::Get::Diff-Only",0},
2596 {0,"tar-only","APT::Get::tar-Only",0},
2597 {0,"purge","APT::Get::Purge",0},
2598 {0,"list-cleanup","APT::Get::List-Cleanup",0},
2599 {0,"reinstall","APT::Get::ReInstall",0},
2600 {0,"trivial-only","APT::Get::Trivial-Only",0},
2601 {0,"remove","APT::Get::Remove",0},
2602 {0,"only-source","APT::Get::Only-Source",0},
2603 {0,"arch-only","APT::Get::Arch-Only",0},
2604 {0,"experimental-automatic-remove","APT::Get::AutomaticRemove",0},
2605 {0,"allow-unauthenticated","APT::Get::AllowUnauthenticated",0},
2606 {'c',"config-file",0,CommandLine::ConfigFile},
2607 {'o',"option",0,CommandLine::ArbItem},
2608 {0,0,0,0}};
2609 CommandLine::Dispatch Cmds[] = {{"update",&DoUpdate},
2610 {"upgrade",&DoUpgrade},
2611 {"install",&DoInstall},
2612 {"remove",&DoInstall},
2613 {"dist-upgrade",&DoDistUpgrade},
2614 {"dselect-upgrade",&DoDSelectUpgrade},
2615 {"build-dep",&DoBuildDep},
2616 {"clean",&DoClean},
2617 {"autoclean",&DoAutoClean},
2618 {"check",&DoCheck},
2619 {"source",&DoSource},
2620 {"moo",&DoMoo},
2621 {"help",&ShowHelp},
2622 {0,0}};
2623
2624 // Set up gettext support
2625 setlocale(LC_ALL,"");
2626 textdomain(PACKAGE);
2627
2628 // Parse the command line and initialize the package library
2629 CommandLine CmdL(Args,_config);
2630 if (pkgInitConfig(*_config) == false ||
2631 CmdL.Parse(argc,argv) == false ||
2632 pkgInitSystem(*_config,_system) == false)
2633 {
2634 if (_config->FindB("version") == true)
2635 ShowHelp(CmdL);
2636
2637 _error->DumpErrors();
2638 return 100;
2639 }
2640
2641 // See if the help should be shown
2642 if (_config->FindB("help") == true ||
2643 _config->FindB("version") == true ||
2644 CmdL.FileSize() == 0)
2645 {
2646 ShowHelp(CmdL);
2647 return 0;
2648 }
2649
2650 // Deal with stdout not being a tty
2651 if (!isatty(STDOUT_FILENO) && _config->FindI("quiet",0) < 1)
2652 _config->Set("quiet","1");
2653
2654 // Setup the output streams
2655 c0out.rdbuf(cout.rdbuf());
2656 c1out.rdbuf(cout.rdbuf());
2657 c2out.rdbuf(cout.rdbuf());
2658 if (_config->FindI("quiet",0) > 0)
2659 c0out.rdbuf(devnull.rdbuf());
2660 if (_config->FindI("quiet",0) > 1)
2661 c1out.rdbuf(devnull.rdbuf());
2662
2663 // Setup the signals
2664 signal(SIGPIPE,SIG_IGN);
2665 signal(SIGWINCH,SigWinch);
2666 SigWinch(0);
2667
2668 // Match the operation
2669 CmdL.DispatchArg(Cmds);
2670
2671 // Print any errors or warnings found during parsing
2672 if (_error->empty() == false)
2673 {
2674 bool Errors = _error->PendingError();
2675 _error->DumpErrors();
2676 return Errors == true?100:0;
2677 }
2678
2679 return 0;
2680 }