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