]> git.saurik.com Git - apt.git/blob - cmdline/apt-get.cc
bc8cd1ae56605cfc584ab254088425e366c1752e
[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 VersionsList += string(Cache[I].CandVersion) + "\n";
1707 }
1708 }
1709
1710 ShowList(c1out,_("The following extra packages will be installed:"),List,VersionsList);
1711 }
1712
1713 /* Print out a list of suggested and recommended packages */
1714 {
1715 string SuggestsList, RecommendsList, List;
1716 string SuggestsVersions, RecommendsVersions;
1717 for (unsigned J = 0; J < Cache->Head().PackageCount; J++)
1718 {
1719 pkgCache::PkgIterator I(Cache,Cache.List[J]);
1720
1721 /* Just look at the ones we want to install */
1722 if ((*Cache)[I].Install() == false)
1723 continue;
1724
1725 for (pkgCache::VerIterator V = I.VersionList(); V.end() == false; V++)
1726 {
1727 for (pkgCache::DepIterator D = V.DependsList(); D.end() == false; )
1728 {
1729 pkgCache::DepIterator Start;
1730 pkgCache::DepIterator End;
1731 D.GlobOr(Start,End); // advances D
1732
1733 /*
1734 * If this is a virtual package, we need to check the list of
1735 * packages that provide it and see if any of those are
1736 * installed
1737 */
1738
1739 bool providedBySomething = false;
1740 for (pkgCache::PrvIterator Prv = Start.TargetPkg().ProvidesList();
1741 Prv.end() != true;
1742 Prv++)
1743 if ((*Cache)[Prv.OwnerPkg()].InstVerIter(*Cache).end() == false)
1744 {
1745 providedBySomething = true;
1746 break;
1747 }
1748
1749 if (providedBySomething) continue;
1750
1751 for(;;)
1752 {
1753 /* Skip if package is installed already, or is about to be */
1754 string target = string(Start.TargetPkg().Name()) + " ";
1755
1756 if ((*Start.TargetPkg()).SelectedState == pkgCache::State::Install
1757 || Cache[Start.TargetPkg()].Install())
1758 break;
1759
1760 /* Skip if we already saw it */
1761 if (int(SuggestsList.find(target)) != -1 || int(RecommendsList.find(target)) != -1)
1762 break;
1763
1764 if (Start->Type == pkgCache::Dep::Suggests) {
1765 SuggestsList += target;
1766 SuggestsVersions += string(Cache[Start.TargetPkg()].CandVersion) + "\n";
1767 }
1768
1769 if (Start->Type == pkgCache::Dep::Recommends) {
1770 RecommendsList += target;
1771 RecommendsVersions += string(Cache[Start.TargetPkg()].CandVersion) + "\n";
1772 }
1773
1774 if (Start >= End)
1775 break;
1776 Start++;
1777 }
1778 }
1779 }
1780 }
1781 ShowList(c1out,_("Suggested packages:"),SuggestsList,SuggestsVersions);
1782 ShowList(c1out,_("Recommended packages:"),RecommendsList,RecommendsVersions);
1783
1784 }
1785
1786 // See if we need to prompt
1787 if (Cache->InstCount() == ExpectedInst && Cache->DelCount() == 0)
1788 return InstallPackages(Cache,false,false);
1789
1790 return InstallPackages(Cache,false);
1791 }
1792 /*}}}*/
1793 // DoDistUpgrade - Automatic smart upgrader /*{{{*/
1794 // ---------------------------------------------------------------------
1795 /* Intelligent upgrader that will install and remove packages at will */
1796 bool DoDistUpgrade(CommandLine &CmdL)
1797 {
1798 CacheFile Cache;
1799 if (Cache.OpenForInstall() == false || Cache.CheckDeps() == false)
1800 return false;
1801
1802 c0out << _("Calculating upgrade... ") << flush;
1803 if (pkgDistUpgrade(*Cache) == false)
1804 {
1805 c0out << _("Failed") << endl;
1806 ShowBroken(c1out,Cache,false);
1807 return false;
1808 }
1809
1810 c0out << _("Done") << endl;
1811
1812 return InstallPackages(Cache,true);
1813 }
1814 /*}}}*/
1815 // DoDSelectUpgrade - Do an upgrade by following dselects selections /*{{{*/
1816 // ---------------------------------------------------------------------
1817 /* Follows dselect's selections */
1818 bool DoDSelectUpgrade(CommandLine &CmdL)
1819 {
1820 CacheFile Cache;
1821 if (Cache.OpenForInstall() == false || Cache.CheckDeps() == false)
1822 return false;
1823
1824 // Install everything with the install flag set
1825 pkgCache::PkgIterator I = Cache->PkgBegin();
1826 for (;I.end() != true; I++)
1827 {
1828 /* Install the package only if it is a new install, the autoupgrader
1829 will deal with the rest */
1830 if (I->SelectedState == pkgCache::State::Install)
1831 Cache->MarkInstall(I,false);
1832 }
1833
1834 /* Now install their deps too, if we do this above then order of
1835 the status file is significant for | groups */
1836 for (I = Cache->PkgBegin();I.end() != true; I++)
1837 {
1838 /* Install the package only if it is a new install, the autoupgrader
1839 will deal with the rest */
1840 if (I->SelectedState == pkgCache::State::Install)
1841 Cache->MarkInstall(I,true);
1842 }
1843
1844 // Apply erasures now, they override everything else.
1845 for (I = Cache->PkgBegin();I.end() != true; I++)
1846 {
1847 // Remove packages
1848 if (I->SelectedState == pkgCache::State::DeInstall ||
1849 I->SelectedState == pkgCache::State::Purge)
1850 Cache->MarkDelete(I,I->SelectedState == pkgCache::State::Purge);
1851 }
1852
1853 /* Resolve any problems that dselect created, allupgrade cannot handle
1854 such things. We do so quite agressively too.. */
1855 if (Cache->BrokenCount() != 0)
1856 {
1857 pkgProblemResolver Fix(Cache);
1858
1859 // Hold back held packages.
1860 if (_config->FindB("APT::Ignore-Hold",false) == false)
1861 {
1862 for (pkgCache::PkgIterator I = Cache->PkgBegin(); I.end() == false; I++)
1863 {
1864 if (I->SelectedState == pkgCache::State::Hold)
1865 {
1866 Fix.Protect(I);
1867 Cache->MarkKeep(I);
1868 }
1869 }
1870 }
1871
1872 if (Fix.Resolve() == false)
1873 {
1874 ShowBroken(c1out,Cache,false);
1875 return _error->Error("Internal error, problem resolver broke stuff");
1876 }
1877 }
1878
1879 // Now upgrade everything
1880 if (pkgAllUpgrade(Cache) == false)
1881 {
1882 ShowBroken(c1out,Cache,false);
1883 return _error->Error("Internal error, problem resolver broke stuff");
1884 }
1885
1886 return InstallPackages(Cache,false);
1887 }
1888 /*}}}*/
1889 // DoClean - Remove download archives /*{{{*/
1890 // ---------------------------------------------------------------------
1891 /* */
1892 bool DoClean(CommandLine &CmdL)
1893 {
1894 if (_config->FindB("APT::Get::Simulate") == true)
1895 {
1896 cout << "Del " << _config->FindDir("Dir::Cache::archives") << "* " <<
1897 _config->FindDir("Dir::Cache::archives") << "partial/*" << endl;
1898 return true;
1899 }
1900
1901 // Lock the archive directory
1902 FileFd Lock;
1903 if (_config->FindB("Debug::NoLocking",false) == false)
1904 {
1905 Lock.Fd(GetLock(_config->FindDir("Dir::Cache::Archives") + "lock"));
1906 if (_error->PendingError() == true)
1907 return _error->Error(_("Unable to lock the download directory"));
1908 }
1909
1910 pkgAcquire Fetcher;
1911 Fetcher.Clean(_config->FindDir("Dir::Cache::archives"));
1912 Fetcher.Clean(_config->FindDir("Dir::Cache::archives") + "partial/");
1913 return true;
1914 }
1915 /*}}}*/
1916 // DoAutoClean - Smartly remove downloaded archives /*{{{*/
1917 // ---------------------------------------------------------------------
1918 /* This is similar to clean but it only purges things that cannot be
1919 downloaded, that is old versions of cached packages. */
1920 class LogCleaner : public pkgArchiveCleaner
1921 {
1922 protected:
1923 virtual void Erase(const char *File,string Pkg,string Ver,struct stat &St)
1924 {
1925 c1out << "Del " << Pkg << " " << Ver << " [" << SizeToStr(St.st_size) << "B]" << endl;
1926
1927 if (_config->FindB("APT::Get::Simulate") == false)
1928 unlink(File);
1929 };
1930 };
1931
1932 bool DoAutoClean(CommandLine &CmdL)
1933 {
1934 // Lock the archive directory
1935 FileFd Lock;
1936 if (_config->FindB("Debug::NoLocking",false) == false)
1937 {
1938 Lock.Fd(GetLock(_config->FindDir("Dir::Cache::Archives") + "lock"));
1939 if (_error->PendingError() == true)
1940 return _error->Error(_("Unable to lock the download directory"));
1941 }
1942
1943 CacheFile Cache;
1944 if (Cache.Open() == false)
1945 return false;
1946
1947 LogCleaner Cleaner;
1948
1949 return Cleaner.Go(_config->FindDir("Dir::Cache::archives"),*Cache) &&
1950 Cleaner.Go(_config->FindDir("Dir::Cache::archives") + "partial/",*Cache);
1951 }
1952 /*}}}*/
1953 // DoCheck - Perform the check operation /*{{{*/
1954 // ---------------------------------------------------------------------
1955 /* Opening automatically checks the system, this command is mostly used
1956 for debugging */
1957 bool DoCheck(CommandLine &CmdL)
1958 {
1959 CacheFile Cache;
1960 Cache.Open();
1961 Cache.CheckDeps();
1962
1963 return true;
1964 }
1965 /*}}}*/
1966 // DoSource - Fetch a source archive /*{{{*/
1967 // ---------------------------------------------------------------------
1968 /* Fetch souce packages */
1969 struct DscFile
1970 {
1971 string Package;
1972 string Version;
1973 string Dsc;
1974 };
1975
1976 bool DoSource(CommandLine &CmdL)
1977 {
1978 CacheFile Cache;
1979 if (Cache.Open(false) == false)
1980 return false;
1981
1982 if (CmdL.FileSize() <= 1)
1983 return _error->Error(_("Must specify at least one package to fetch source for"));
1984
1985 // Read the source list
1986 pkgSourceList List;
1987 if (List.ReadMainList() == false)
1988 return _error->Error(_("The list of sources could not be read."));
1989
1990 // Create the text record parsers
1991 pkgRecords Recs(Cache);
1992 pkgSrcRecords SrcRecs(List);
1993 if (_error->PendingError() == true)
1994 return false;
1995
1996 // Create the download object
1997 AcqTextStatus Stat(ScreenWidth,_config->FindI("quiet",0));
1998 pkgAcquire Fetcher(&Stat);
1999
2000 DscFile *Dsc = new DscFile[CmdL.FileSize()];
2001
2002 // Load the requestd sources into the fetcher
2003 unsigned J = 0;
2004 for (const char **I = CmdL.FileList + 1; *I != 0; I++, J++)
2005 {
2006 string Src;
2007 pkgSrcRecords::Parser *Last = FindSrc(*I,Recs,SrcRecs,Src,*Cache);
2008
2009 if (Last == 0)
2010 return _error->Error(_("Unable to find a source package for %s"),Src.c_str());
2011
2012 // Back track
2013 vector<pkgSrcRecords::File> Lst;
2014 if (Last->Files(Lst) == false)
2015 return false;
2016
2017 // Load them into the fetcher
2018 for (vector<pkgSrcRecords::File>::const_iterator I = Lst.begin();
2019 I != Lst.end(); I++)
2020 {
2021 // Try to guess what sort of file it is we are getting.
2022 if (I->Type == "dsc")
2023 {
2024 Dsc[J].Package = Last->Package();
2025 Dsc[J].Version = Last->Version();
2026 Dsc[J].Dsc = flNotDir(I->Path);
2027 }
2028
2029 // Diff only mode only fetches .diff files
2030 if (_config->FindB("APT::Get::Diff-Only",false) == true &&
2031 I->Type != "diff")
2032 continue;
2033
2034 // Tar only mode only fetches .tar files
2035 if (_config->FindB("APT::Get::Tar-Only",false) == true &&
2036 I->Type != "tar")
2037 continue;
2038
2039 new pkgAcqFile(&Fetcher,Last->Index().ArchiveURI(I->Path),
2040 I->MD5Hash,I->Size,
2041 Last->Index().SourceInfo(*Last,*I),Src);
2042 }
2043 }
2044
2045 // Display statistics
2046 double FetchBytes = Fetcher.FetchNeeded();
2047 double FetchPBytes = Fetcher.PartialPresent();
2048 double DebBytes = Fetcher.TotalNeeded();
2049
2050 // Check for enough free space
2051 struct statvfs Buf;
2052 string OutputDir = ".";
2053 if (statvfs(OutputDir.c_str(),&Buf) != 0)
2054 return _error->Errno("statvfs","Couldn't determine free space in %s",
2055 OutputDir.c_str());
2056 if (unsigned(Buf.f_bfree) < (FetchBytes - FetchPBytes)/Buf.f_bsize)
2057 return _error->Error(_("You don't have enough free space in %s"),
2058 OutputDir.c_str());
2059
2060 // Number of bytes
2061 if (DebBytes != FetchBytes)
2062 ioprintf(c1out,_("Need to get %sB/%sB of source archives.\n"),
2063 SizeToStr(FetchBytes).c_str(),SizeToStr(DebBytes).c_str());
2064 else
2065 ioprintf(c1out,_("Need to get %sB of source archives.\n"),
2066 SizeToStr(DebBytes).c_str());
2067
2068 if (_config->FindB("APT::Get::Simulate",false) == true)
2069 {
2070 for (unsigned I = 0; I != J; I++)
2071 ioprintf(cout,_("Fetch source %s\n"),Dsc[I].Package.c_str());
2072 return true;
2073 }
2074
2075 // Just print out the uris an exit if the --print-uris flag was used
2076 if (_config->FindB("APT::Get::Print-URIs") == true)
2077 {
2078 pkgAcquire::UriIterator I = Fetcher.UriBegin();
2079 for (; I != Fetcher.UriEnd(); I++)
2080 cout << '\'' << I->URI << "' " << flNotDir(I->Owner->DestFile) << ' ' <<
2081 I->Owner->FileSize << ' ' << I->Owner->MD5Sum() << endl;
2082 return true;
2083 }
2084
2085 // Run it
2086 if (Fetcher.Run() == pkgAcquire::Failed)
2087 return false;
2088
2089 // Print error messages
2090 bool Failed = false;
2091 for (pkgAcquire::ItemIterator I = Fetcher.ItemsBegin(); I != Fetcher.ItemsEnd(); I++)
2092 {
2093 if ((*I)->Status == pkgAcquire::Item::StatDone &&
2094 (*I)->Complete == true)
2095 continue;
2096
2097 fprintf(stderr,_("Failed to fetch %s %s\n"),(*I)->DescURI().c_str(),
2098 (*I)->ErrorText.c_str());
2099 Failed = true;
2100 }
2101 if (Failed == true)
2102 return _error->Error(_("Failed to fetch some archives."));
2103
2104 if (_config->FindB("APT::Get::Download-only",false) == true)
2105 {
2106 c1out << _("Download complete and in download only mode") << endl;
2107 return true;
2108 }
2109
2110 // Unpack the sources
2111 pid_t Process = ExecFork();
2112
2113 if (Process == 0)
2114 {
2115 for (unsigned I = 0; I != J; I++)
2116 {
2117 string Dir = Dsc[I].Package + '-' + Cache->VS().UpstreamVersion(Dsc[I].Version.c_str());
2118
2119 // Diff only mode only fetches .diff files
2120 if (_config->FindB("APT::Get::Diff-Only",false) == true ||
2121 _config->FindB("APT::Get::Tar-Only",false) == true ||
2122 Dsc[I].Dsc.empty() == true)
2123 continue;
2124
2125 // See if the package is already unpacked
2126 struct stat Stat;
2127 if (stat(Dir.c_str(),&Stat) == 0 &&
2128 S_ISDIR(Stat.st_mode) != 0)
2129 {
2130 ioprintf(c0out ,_("Skipping unpack of already unpacked source in %s\n"),
2131 Dir.c_str());
2132 }
2133 else
2134 {
2135 // Call dpkg-source
2136 char S[500];
2137 snprintf(S,sizeof(S),"%s -x %s",
2138 _config->Find("Dir::Bin::dpkg-source","dpkg-source").c_str(),
2139 Dsc[I].Dsc.c_str());
2140 if (system(S) != 0)
2141 {
2142 fprintf(stderr,_("Unpack command '%s' failed.\n"),S);
2143 _exit(1);
2144 }
2145 }
2146
2147 // Try to compile it with dpkg-buildpackage
2148 if (_config->FindB("APT::Get::Compile",false) == true)
2149 {
2150 // Call dpkg-buildpackage
2151 char S[500];
2152 snprintf(S,sizeof(S),"cd %s && %s %s",
2153 Dir.c_str(),
2154 _config->Find("Dir::Bin::dpkg-buildpackage","dpkg-buildpackage").c_str(),
2155 _config->Find("DPkg::Build-Options","-b -uc").c_str());
2156
2157 if (system(S) != 0)
2158 {
2159 fprintf(stderr,_("Build command '%s' failed.\n"),S);
2160 _exit(1);
2161 }
2162 }
2163 }
2164
2165 _exit(0);
2166 }
2167
2168 // Wait for the subprocess
2169 int Status = 0;
2170 while (waitpid(Process,&Status,0) != Process)
2171 {
2172 if (errno == EINTR)
2173 continue;
2174 return _error->Errno("waitpid","Couldn't wait for subprocess");
2175 }
2176
2177 if (WIFEXITED(Status) == 0 || WEXITSTATUS(Status) != 0)
2178 return _error->Error(_("Child process failed"));
2179
2180 return true;
2181 }
2182 /*}}}*/
2183 // DoBuildDep - Install/removes packages to satisfy build dependencies /*{{{*/
2184 // ---------------------------------------------------------------------
2185 /* This function will look at the build depends list of the given source
2186 package and install the necessary packages to make it true, or fail. */
2187 bool DoBuildDep(CommandLine &CmdL)
2188 {
2189 CacheFile Cache;
2190 if (Cache.Open(true) == false)
2191 return false;
2192
2193 if (CmdL.FileSize() <= 1)
2194 return _error->Error(_("Must specify at least one package to check builddeps for"));
2195
2196 // Read the source list
2197 pkgSourceList List;
2198 if (List.ReadMainList() == false)
2199 return _error->Error(_("The list of sources could not be read."));
2200
2201 // Create the text record parsers
2202 pkgRecords Recs(Cache);
2203 pkgSrcRecords SrcRecs(List);
2204 if (_error->PendingError() == true)
2205 return false;
2206
2207 // Create the download object
2208 AcqTextStatus Stat(ScreenWidth,_config->FindI("quiet",0));
2209 pkgAcquire Fetcher(&Stat);
2210
2211 unsigned J = 0;
2212 for (const char **I = CmdL.FileList + 1; *I != 0; I++, J++)
2213 {
2214 string Src;
2215 pkgSrcRecords::Parser *Last = FindSrc(*I,Recs,SrcRecs,Src,*Cache);
2216 if (Last == 0)
2217 return _error->Error(_("Unable to find a source package for %s"),Src.c_str());
2218
2219 // Process the build-dependencies
2220 vector<pkgSrcRecords::Parser::BuildDepRec> BuildDeps;
2221 if (Last->BuildDepends(BuildDeps, _config->FindB("APT::Get::Arch-Only",false)) == false)
2222 return _error->Error(_("Unable to get build-dependency information for %s"),Src.c_str());
2223
2224 // Also ensure that build-essential packages are present
2225 Configuration::Item const *Opts = _config->Tree("APT::Build-Essential");
2226 if (Opts)
2227 Opts = Opts->Child;
2228 for (; Opts; Opts = Opts->Next)
2229 {
2230 if (Opts->Value.empty() == true)
2231 continue;
2232
2233 pkgSrcRecords::Parser::BuildDepRec rec;
2234 rec.Package = Opts->Value;
2235 rec.Type = pkgSrcRecords::Parser::BuildDependIndep;
2236 rec.Op = 0;
2237 BuildDeps.push_back(rec);
2238 }
2239
2240 if (BuildDeps.size() == 0)
2241 {
2242 ioprintf(c1out,_("%s has no build depends.\n"),Src.c_str());
2243 continue;
2244 }
2245
2246 // Install the requested packages
2247 unsigned int ExpectedInst = 0;
2248 vector <pkgSrcRecords::Parser::BuildDepRec>::iterator D;
2249 pkgProblemResolver Fix(Cache);
2250 bool skipAlternatives = false; // skip remaining alternatives in an or group
2251 for (D = BuildDeps.begin(); D != BuildDeps.end(); D++)
2252 {
2253 bool hasAlternatives = (((*D).Op & pkgCache::Dep::Or) == pkgCache::Dep::Or);
2254
2255 if (skipAlternatives == true)
2256 {
2257 if (!hasAlternatives)
2258 skipAlternatives = false; // end of or group
2259 continue;
2260 }
2261
2262 if ((*D).Type == pkgSrcRecords::Parser::BuildConflict ||
2263 (*D).Type == pkgSrcRecords::Parser::BuildConflictIndep)
2264 {
2265 pkgCache::PkgIterator Pkg = Cache->FindPkg((*D).Package);
2266 // Build-conflicts on unknown packages are silently ignored
2267 if (Pkg.end() == true)
2268 continue;
2269
2270 pkgCache::VerIterator IV = (*Cache)[Pkg].InstVerIter(*Cache);
2271
2272 /*
2273 * Remove if we have an installed version that satisfies the
2274 * version criteria
2275 */
2276 if (IV.end() == false &&
2277 Cache->VS().CheckDep(IV.VerStr(),(*D).Op,(*D).Version.c_str()) == true)
2278 TryToInstall(Pkg,Cache,Fix,true,false,ExpectedInst);
2279 }
2280 else // BuildDep || BuildDepIndep
2281 {
2282 pkgCache::PkgIterator Pkg = Cache->FindPkg((*D).Package);
2283 if (_config->FindB("Debug::BuildDeps",false) == true)
2284 cout << "Looking for " << (*D).Package << "...\n";
2285
2286 if (Pkg.end() == true)
2287 {
2288 if (_config->FindB("Debug::BuildDeps",false) == true)
2289 cout << " (not found)" << (*D).Package << endl;
2290
2291 if (hasAlternatives)
2292 continue;
2293
2294 return _error->Error(_("%s dependency for %s cannot be satisfied "
2295 "because the package %s cannot be found"),
2296 Last->BuildDepType((*D).Type),Src.c_str(),
2297 (*D).Package.c_str());
2298 }
2299
2300 /*
2301 * if there are alternatives, we've already picked one, so skip
2302 * the rest
2303 *
2304 * TODO: this means that if there's a build-dep on A|B and B is
2305 * installed, we'll still try to install A; more importantly,
2306 * if A is currently broken, we cannot go back and try B. To fix
2307 * this would require we do a Resolve cycle for each package we
2308 * add to the install list. Ugh
2309 */
2310
2311 /*
2312 * If this is a virtual package, we need to check the list of
2313 * packages that provide it and see if any of those are
2314 * installed
2315 */
2316 pkgCache::PrvIterator Prv = Pkg.ProvidesList();
2317 for (; Prv.end() != true; Prv++)
2318 {
2319 if (_config->FindB("Debug::BuildDeps",false) == true)
2320 cout << " Checking provider " << Prv.OwnerPkg().Name() << endl;
2321
2322 if ((*Cache)[Prv.OwnerPkg()].InstVerIter(*Cache).end() == false)
2323 break;
2324 }
2325
2326 // Get installed version and version we are going to install
2327 pkgCache::VerIterator IV = (*Cache)[Pkg].InstVerIter(*Cache);
2328
2329 if ((*D).Version[0] != '\0') {
2330 // Versioned dependency
2331
2332 pkgCache::VerIterator CV = (*Cache)[Pkg].CandidateVerIter(*Cache);
2333
2334 for (; CV.end() != true; CV++)
2335 {
2336 if (Cache->VS().CheckDep(CV.VerStr(),(*D).Op,(*D).Version.c_str()) == true)
2337 break;
2338 }
2339 if (CV.end() == true)
2340 if (hasAlternatives)
2341 {
2342 continue;
2343 }
2344 else
2345 {
2346 return _error->Error(_("%s dependency for %s cannot be satisfied "
2347 "because no available versions of package %s "
2348 "can satisfy version requirements"),
2349 Last->BuildDepType((*D).Type),Src.c_str(),
2350 (*D).Package.c_str());
2351 }
2352 }
2353 else
2354 {
2355 // Only consider virtual packages if there is no versioned dependency
2356 if (Prv.end() == false)
2357 {
2358 if (_config->FindB("Debug::BuildDeps",false) == true)
2359 cout << " Is provided by installed package " << Prv.OwnerPkg().Name() << endl;
2360 skipAlternatives = hasAlternatives;
2361 continue;
2362 }
2363 }
2364
2365 if (IV.end() == false)
2366 {
2367 if (_config->FindB("Debug::BuildDeps",false) == true)
2368 cout << " Is installed\n";
2369
2370 if (Cache->VS().CheckDep(IV.VerStr(),(*D).Op,(*D).Version.c_str()) == true)
2371 {
2372 skipAlternatives = hasAlternatives;
2373 continue;
2374 }
2375
2376 if (_config->FindB("Debug::BuildDeps",false) == true)
2377 cout << " ...but the installed version doesn't meet the version requirement\n";
2378
2379 if (((*D).Op & pkgCache::Dep::LessEq) == pkgCache::Dep::LessEq)
2380 {
2381 return _error->Error(_("Failed to satisfy %s dependency for %s: Installed package %s is too new"),
2382 Last->BuildDepType((*D).Type),
2383 Src.c_str(),
2384 Pkg.Name());
2385 }
2386 }
2387
2388
2389 if (_config->FindB("Debug::BuildDeps",false) == true)
2390 cout << " Trying to install " << (*D).Package << endl;
2391
2392 if (TryToInstall(Pkg,Cache,Fix,false,false,ExpectedInst) == true)
2393 {
2394 // We successfully installed something; skip remaining alternatives
2395 skipAlternatives = hasAlternatives;
2396 continue;
2397 }
2398 else if (hasAlternatives)
2399 {
2400 if (_config->FindB("Debug::BuildDeps",false) == true)
2401 cout << " Unsatisfiable, trying alternatives\n";
2402 continue;
2403 }
2404 else
2405 {
2406 return _error->Error(_("Failed to satisfy %s dependency for %s: %s"),
2407 Last->BuildDepType((*D).Type),
2408 Src.c_str(),
2409 (*D).Package.c_str());
2410 }
2411 }
2412 }
2413
2414 Fix.InstallProtect();
2415 if (Fix.Resolve(true) == false)
2416 _error->Discard();
2417
2418 // Now we check the state of the packages,
2419 if (Cache->BrokenCount() != 0)
2420 return _error->Error(_("Build-dependencies for %s could not be satisfied."),*I);
2421 }
2422
2423 if (InstallPackages(Cache, false, true) == false)
2424 return _error->Error(_("Failed to process build dependencies"));
2425 return true;
2426 }
2427 /*}}}*/
2428
2429 // DoMoo - Never Ask, Never Tell /*{{{*/
2430 // ---------------------------------------------------------------------
2431 /* */
2432 bool DoMoo(CommandLine &CmdL)
2433 {
2434 cout <<
2435 " (__) \n"
2436 " (oo) \n"
2437 " /------\\/ \n"
2438 " / | || \n"
2439 " * /\\---/\\ \n"
2440 " ~~ ~~ \n"
2441 "....\"Have you mooed today?\"...\n";
2442
2443 return true;
2444 }
2445 /*}}}*/
2446 // ShowHelp - Show a help screen /*{{{*/
2447 // ---------------------------------------------------------------------
2448 /* */
2449 bool ShowHelp(CommandLine &CmdL)
2450 {
2451 ioprintf(cout,_("%s %s for %s %s compiled on %s %s\n"),PACKAGE,VERSION,
2452 COMMON_OS,COMMON_CPU,__DATE__,__TIME__);
2453
2454 if (_config->FindB("version") == true)
2455 {
2456 cout << _("Supported modules:") << endl;
2457
2458 for (unsigned I = 0; I != pkgVersioningSystem::GlobalListLen; I++)
2459 {
2460 pkgVersioningSystem *VS = pkgVersioningSystem::GlobalList[I];
2461 if (_system != 0 && _system->VS == VS)
2462 cout << '*';
2463 else
2464 cout << ' ';
2465 cout << "Ver: " << VS->Label << endl;
2466
2467 /* Print out all the packaging systems that will work with
2468 this VS */
2469 for (unsigned J = 0; J != pkgSystem::GlobalListLen; J++)
2470 {
2471 pkgSystem *Sys = pkgSystem::GlobalList[J];
2472 if (_system == Sys)
2473 cout << '*';
2474 else
2475 cout << ' ';
2476 if (Sys->VS->TestCompatibility(*VS) == true)
2477 cout << "Pkg: " << Sys->Label << " (Priority " << Sys->Score(*_config) << ")" << endl;
2478 }
2479 }
2480
2481 for (unsigned I = 0; I != pkgSourceList::Type::GlobalListLen; I++)
2482 {
2483 pkgSourceList::Type *Type = pkgSourceList::Type::GlobalList[I];
2484 cout << " S.L: '" << Type->Name << "' " << Type->Label << endl;
2485 }
2486
2487 for (unsigned I = 0; I != pkgIndexFile::Type::GlobalListLen; I++)
2488 {
2489 pkgIndexFile::Type *Type = pkgIndexFile::Type::GlobalList[I];
2490 cout << " Idx: " << Type->Label << endl;
2491 }
2492
2493 return true;
2494 }
2495
2496 cout <<
2497 _("Usage: apt-get [options] command\n"
2498 " apt-get [options] install|remove pkg1 [pkg2 ...]\n"
2499 " apt-get [options] source pkg1 [pkg2 ...]\n"
2500 "\n"
2501 "apt-get is a simple command line interface for downloading and\n"
2502 "installing packages. The most frequently used commands are update\n"
2503 "and install.\n"
2504 "\n"
2505 "Commands:\n"
2506 " update - Retrieve new lists of packages\n"
2507 " upgrade - Perform an upgrade\n"
2508 " install - Install new packages (pkg is libc6 not libc6.deb)\n"
2509 " remove - Remove packages\n"
2510 " source - Download source archives\n"
2511 " build-dep - Configure build-dependencies for source packages\n"
2512 " dist-upgrade - Distribution upgrade, see apt-get(8)\n"
2513 " dselect-upgrade - Follow dselect selections\n"
2514 " clean - Erase downloaded archive files\n"
2515 " autoclean - Erase old downloaded archive files\n"
2516 " check - Verify that there are no broken dependencies\n"
2517 "\n"
2518 "Options:\n"
2519 " -h This help text.\n"
2520 " -q Loggable output - no progress indicator\n"
2521 " -qq No output except for errors\n"
2522 " -d Download only - do NOT install or unpack archives\n"
2523 " -s No-act. Perform ordering simulation\n"
2524 " -y Assume Yes to all queries and do not prompt\n"
2525 " -f Attempt to continue if the integrity check fails\n"
2526 " -m Attempt to continue if archives are unlocatable\n"
2527 " -u Show a list of upgraded packages as well\n"
2528 " -b Build the source package after fetching it\n"
2529 " -V Show verbose version numbers\n"
2530 " -c=? Read this configuration file\n"
2531 " -o=? Set an arbitrary configuration option, eg -o dir::cache=/tmp\n"
2532 "See the apt-get(8), sources.list(5) and apt.conf(5) manual\n"
2533 "pages for more information and options.\n"
2534 " This APT has Super Cow Powers.\n");
2535 return true;
2536 }
2537 /*}}}*/
2538 // GetInitialize - Initialize things for apt-get /*{{{*/
2539 // ---------------------------------------------------------------------
2540 /* */
2541 void GetInitialize()
2542 {
2543 _config->Set("quiet",0);
2544 _config->Set("help",false);
2545 _config->Set("APT::Get::Download-Only",false);
2546 _config->Set("APT::Get::Simulate",false);
2547 _config->Set("APT::Get::Assume-Yes",false);
2548 _config->Set("APT::Get::Fix-Broken",false);
2549 _config->Set("APT::Get::Force-Yes",false);
2550 _config->Set("APT::Get::List-Cleanup",true);
2551 _config->Set("APT::Get::AutomaticRemove",false);
2552 }
2553 /*}}}*/
2554 // SigWinch - Window size change signal handler /*{{{*/
2555 // ---------------------------------------------------------------------
2556 /* */
2557 void SigWinch(int)
2558 {
2559 // Riped from GNU ls
2560 #ifdef TIOCGWINSZ
2561 struct winsize ws;
2562
2563 if (ioctl(1, TIOCGWINSZ, &ws) != -1 && ws.ws_col >= 5)
2564 ScreenWidth = ws.ws_col - 1;
2565 #endif
2566 }
2567 /*}}}*/
2568
2569 int main(int argc,const char *argv[])
2570 {
2571 CommandLine::Args Args[] = {
2572 {'h',"help","help",0},
2573 {'v',"version","version",0},
2574 {'V',"verbose-versions","APT::Get::Show-Versions",0},
2575 {'q',"quiet","quiet",CommandLine::IntLevel},
2576 {'q',"silent","quiet",CommandLine::IntLevel},
2577 {'d',"download-only","APT::Get::Download-Only",0},
2578 {'b',"compile","APT::Get::Compile",0},
2579 {'b',"build","APT::Get::Compile",0},
2580 {'s',"simulate","APT::Get::Simulate",0},
2581 {'s',"just-print","APT::Get::Simulate",0},
2582 {'s',"recon","APT::Get::Simulate",0},
2583 {'s',"dry-run","APT::Get::Simulate",0},
2584 {'s',"no-act","APT::Get::Simulate",0},
2585 {'y',"yes","APT::Get::Assume-Yes",0},
2586 {'y',"assume-yes","APT::Get::Assume-Yes",0},
2587 {'f',"fix-broken","APT::Get::Fix-Broken",0},
2588 {'u',"show-upgraded","APT::Get::Show-Upgraded",0},
2589 {'m',"ignore-missing","APT::Get::Fix-Missing",0},
2590 {'t',"target-release","APT::Default-Release",CommandLine::HasArg},
2591 {'t',"default-release","APT::Default-Release",CommandLine::HasArg},
2592 {0,"download","APT::Get::Download",0},
2593 {0,"fix-missing","APT::Get::Fix-Missing",0},
2594 {0,"ignore-hold","APT::Ignore-Hold",0},
2595 {0,"upgrade","APT::Get::upgrade",0},
2596 {0,"force-yes","APT::Get::force-yes",0},
2597 {0,"print-uris","APT::Get::Print-URIs",0},
2598 {0,"diff-only","APT::Get::Diff-Only",0},
2599 {0,"tar-only","APT::Get::tar-Only",0},
2600 {0,"purge","APT::Get::Purge",0},
2601 {0,"list-cleanup","APT::Get::List-Cleanup",0},
2602 {0,"reinstall","APT::Get::ReInstall",0},
2603 {0,"trivial-only","APT::Get::Trivial-Only",0},
2604 {0,"remove","APT::Get::Remove",0},
2605 {0,"only-source","APT::Get::Only-Source",0},
2606 {0,"arch-only","APT::Get::Arch-Only",0},
2607 {0,"experimental-automatic-remove","APT::Get::AutomaticRemove",0},
2608 {0,"allow-unauthenticated","APT::Get::AllowUnauthenticated",0},
2609 {'c',"config-file",0,CommandLine::ConfigFile},
2610 {'o',"option",0,CommandLine::ArbItem},
2611 {0,0,0,0}};
2612 CommandLine::Dispatch Cmds[] = {{"update",&DoUpdate},
2613 {"upgrade",&DoUpgrade},
2614 {"install",&DoInstall},
2615 {"remove",&DoInstall},
2616 {"dist-upgrade",&DoDistUpgrade},
2617 {"dselect-upgrade",&DoDSelectUpgrade},
2618 {"build-dep",&DoBuildDep},
2619 {"clean",&DoClean},
2620 {"autoclean",&DoAutoClean},
2621 {"check",&DoCheck},
2622 {"source",&DoSource},
2623 {"moo",&DoMoo},
2624 {"help",&ShowHelp},
2625 {0,0}};
2626
2627 // Set up gettext support
2628 setlocale(LC_ALL,"");
2629 textdomain(PACKAGE);
2630
2631 // Parse the command line and initialize the package library
2632 CommandLine CmdL(Args,_config);
2633 if (pkgInitConfig(*_config) == false ||
2634 CmdL.Parse(argc,argv) == false ||
2635 pkgInitSystem(*_config,_system) == false)
2636 {
2637 if (_config->FindB("version") == true)
2638 ShowHelp(CmdL);
2639
2640 _error->DumpErrors();
2641 return 100;
2642 }
2643
2644 // See if the help should be shown
2645 if (_config->FindB("help") == true ||
2646 _config->FindB("version") == true ||
2647 CmdL.FileSize() == 0)
2648 {
2649 ShowHelp(CmdL);
2650 return 0;
2651 }
2652
2653 // Deal with stdout not being a tty
2654 if (!isatty(STDOUT_FILENO) && _config->FindI("quiet",0) < 1)
2655 _config->Set("quiet","1");
2656
2657 // Setup the output streams
2658 c0out.rdbuf(cout.rdbuf());
2659 c1out.rdbuf(cout.rdbuf());
2660 c2out.rdbuf(cout.rdbuf());
2661 if (_config->FindI("quiet",0) > 0)
2662 c0out.rdbuf(devnull.rdbuf());
2663 if (_config->FindI("quiet",0) > 1)
2664 c1out.rdbuf(devnull.rdbuf());
2665
2666 // Setup the signals
2667 signal(SIGPIPE,SIG_IGN);
2668 signal(SIGWINCH,SigWinch);
2669 SigWinch(0);
2670
2671 // Match the operation
2672 CmdL.DispatchArg(Cmds);
2673
2674 // Print any errors or warnings found during parsing
2675 if (_error->empty() == false)
2676 {
2677 bool Errors = _error->PendingError();
2678 _error->DumpErrors();
2679 return Errors == true?100:0;
2680 }
2681
2682 return 0;
2683 }