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