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