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