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