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