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