update-alternatives.pl 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768
  1. #!/usr/bin/perl --
  2. use strict;
  3. use warnings;
  4. my $admindir = "/var/lib/dpkg"; # This line modified by Makefile
  5. my $dpkglibdir = "../utils"; # This line modified by Makefile
  6. my $version = '0.93.80'; # This line modified by Makefile
  7. push (@INC, $dpkglibdir);
  8. require 'dpkg-gettext.pl';
  9. textdomain("dpkg");
  10. ($0) = $0 =~ m:.*/(.+):;
  11. # Global variables:
  12. my $altdir = '/etc/alternatives';
  13. # FIXME: this should not override the previous assignment.
  14. $admindir = $admindir . '/alternatives';
  15. my $testmode = 0;
  16. my $verbosemode = 0;
  17. my $action = ''; # Action to perform (display / install / remove / display / auto / config)
  18. my $mode = 'auto'; # Update mode for alternative (manual / auto)
  19. my $state; # State of alternative:
  20. # expected: alternative with highest priority is the active alternative
  21. # expected-inprogress: busy selecting alternative with highest priority
  22. # unexpected: alternative another alternative is active / error during readlink
  23. # nonexistent: alternative-symlink does not exist
  24. my %versionnum; # Map from currently available versions into @versions and @priorities
  25. my @versions; # List of available versions for alternative
  26. my @priorities; # Map from @version-index to priority
  27. my $best;
  28. my $bestpri;
  29. my $bestnum;
  30. my $link; # Link we are working with
  31. my $linkname;
  32. my $alink; # Alternative we are managing (ie the symlink we're making/removing) (install only)
  33. my $name; # Name of the alternative (the symlink) we are processing
  34. my $apath; # Path of alternative we are offering
  35. my $apriority; # Priority of link (only when we are installing an alternative)
  36. my %aslavelink;
  37. my %aslavepath;
  38. my %aslavelinkcount;
  39. my $slink;
  40. my $sname;
  41. my $spath;
  42. my @slavenames; # List with names of slavelinks
  43. my %slavenum; # Map from name of slavelink to slave-index (into @slavelinks)
  44. my @slavelinks; # List of slavelinks (indexed by slave-index)
  45. my %slavepath; # Map from (@version-index,slavename) to slave-path
  46. my %slavelinkcount;
  47. my $enoent = `$dpkglibdir/enoent` || die sprintf(_g("Cannot get ENOENT value from %s: %s"), "$dpkglibdir/enoent", $!);
  48. sub ENOENT { $enoent; }
  49. sub version {
  50. printf _g("Debian %s version %s.\n"), $0, $version;
  51. printf _g("
  52. Copyright (C) 1995 Ian Jackson.
  53. Copyright (C) 2000-2002 Wichert Akkerman.");
  54. printf _g("
  55. This is free software; see the GNU General Public Licence version 2 or
  56. later for copying conditions. There is NO warranty.
  57. ");
  58. }
  59. sub usage {
  60. printf _g(
  61. "Usage: %s [<option> ...] <command>
  62. Commands:
  63. --install <link> <name> <path> <priority>
  64. [--slave <link> <name> <path>] ...
  65. add a group of alternatives to the system.
  66. --remove <name> <path> remove <path> from the <name> group alternative.
  67. --remove-all <name> remove <name> group from the alternatives system.
  68. --auto <name> switch the master link <name> to automatic mode.
  69. --display <name> display information about the <name> group.
  70. --list <name> display all targets of the <name> group.
  71. --config <name> show alternatives for the <name> group and ask the
  72. user to select which one to use.
  73. --set <name> <path> set <path> as alternative for <name>.
  74. --all call --config on all alternatives.
  75. <link> is the symlink pointing to %s/<name>.
  76. (e.g. /usr/bin/pager)
  77. <name> is the master name for this link group.
  78. (e.g. pager)
  79. <path> is the location of one of the alternative target files.
  80. (e.g. /usr/bin/less)
  81. <priority> is an integer; options with higher numbers have higher priority in
  82. automatic mode.
  83. Options:
  84. --altdir <directory> change the alternatives directory.
  85. --admindir <directory> change the administrative directory.
  86. --test don't do anything, just demonstrate.
  87. --verbose verbose operation, more output.
  88. --quiet quiet operation, minimal output.
  89. --help show this help message.
  90. --version show the version.
  91. "), $0, $altdir;
  92. }
  93. sub quit
  94. {
  95. printf STDERR "%s: %s\n", $0, "@_";
  96. exit(2);
  97. }
  98. sub badusage
  99. {
  100. printf STDERR "%s: %s\n\n", $0, "@_";
  101. &usage;
  102. exit(2);
  103. }
  104. sub read_link_group
  105. {
  106. if (open(AF, "$admindir/$name")) {
  107. $mode = gl("update_mode");
  108. $mode eq 'auto' || $mode eq 'manual' || badfmt(_g("invalid update mode"));
  109. $link = gl("link");
  110. while (($sname = gl("sname")) ne '') {
  111. push(@slavenames, $sname);
  112. defined($slavenum{$sname}) && badfmt(sprintf(_g("duplicate slave %s"), $sname));
  113. $slavenum{$sname} = $#slavenames;
  114. $slink = gl("slink");
  115. $slink eq $link && badfmt(sprintf(_g("slave link same as main link %s"), $link));
  116. $slavelinkcount{$slink}++ && badfmt(sprintf(_g("duplicate slave link %s"), $slink));
  117. push(@slavelinks, $slink);
  118. }
  119. while (($version = gl("version")) ne '') {
  120. defined($versionnum{$version}) && badfmt(sprintf(_g("duplicate path %s"), $version));
  121. if (-r $version) {
  122. push(@versions, $version);
  123. my $i;
  124. $versionnum{$version} = $i = $#versions;
  125. my $priority = gl("priority");
  126. $priority =~ m/^[-+]?\d+$/ || badfmt(sprintf(_g("priority %s %s"), $version, $priority));
  127. $priorities[$i] = $priority;
  128. for (my $j = 0; $j <= $#slavenames; $j++) {
  129. $slavepath{$i,$j} = gl("spath");
  130. }
  131. } else {
  132. # File not found - remove
  133. pr(sprintf(_g("Alternative for %s points to %s - which wasn't found. Removing from list of alternatives."), $name, $version))
  134. if $verbosemode > 0;
  135. gl("priority");
  136. for (my $j = 0; $j <= $#slavenames; $j++) {
  137. gl("spath");
  138. }
  139. }
  140. }
  141. close(AF);
  142. return 0;
  143. } elsif ($! != ENOENT) {
  144. quit(sprintf(_g("unable to open %s: %s"), "$admindir/$name", $!));
  145. } elsif ($! == ENOENT) {
  146. return 1;
  147. }
  148. }
  149. sub fill_missing_slavepaths()
  150. {
  151. for (my $j = 0; $j <= $#slavenames; $j++) {
  152. for (my $i = 0; $i <= $#versions; $i++) {
  153. $slavepath{$i,$j} = '' if !defined $slavepath{$i,$j};
  154. }
  155. }
  156. }
  157. sub find_best_version
  158. {
  159. $best = '';
  160. for (my $i = 0; $i <= $#versions; $i++) {
  161. if ($best eq '' || $priorities[$i] > $bestpri) {
  162. $best = $versions[$i];
  163. $bestpri = $priorities[$i];
  164. $bestnum = $i;
  165. }
  166. }
  167. }
  168. sub display_link_group
  169. {
  170. pr(sprintf(_g("%s - status is %s."), $name, $mode));
  171. $linkname = readlink("$altdir/$name");
  172. if (defined($linkname)) {
  173. pr(sprintf(_g(" link currently points to %s"), $linkname));
  174. } elsif ($! == ENOENT) {
  175. pr(_g(" link currently absent"));
  176. } else {
  177. pr(sprintf(_g(" link unreadable - %s"), $!));
  178. }
  179. for (my $i = 0; $i <= $#versions; $i++) {
  180. pr(sprintf(_g("%s - priority %s"), $versions[$i], $priorities[$i]));
  181. for (my $j = 0; $j <= $#slavenames; $j++) {
  182. my $tspath = $slavepath{$i, $j};
  183. next unless length($tspath);
  184. pr(sprintf(_g(" slave %s: %s"), $slavenames[$j], $tspath));
  185. }
  186. }
  187. if ($best eq '') {
  188. pr(_g("No versions available."));
  189. } else {
  190. pr(sprintf(_g("Current \`best' version is %s."), $best));
  191. }
  192. }
  193. sub list_link_group
  194. {
  195. for (my $i = 0; $i <= $#versions; $i++) {
  196. pr("$versions[$i]");
  197. }
  198. }
  199. sub check_many_actions()
  200. {
  201. return unless $action;
  202. badusage(sprintf(_g("two commands specified: %s and --%s"), $_, $action));
  203. }
  204. #
  205. # Main program
  206. #
  207. $| = 1;
  208. while (@ARGV) {
  209. $_= shift(@ARGV);
  210. last if m/^--$/;
  211. if (!m/^--/) {
  212. &quit(sprintf(_g("unknown argument \`%s'"), $_));
  213. } elsif (m/^--help$/) {
  214. &usage; exit(0);
  215. } elsif (m/^--version$/) {
  216. &version; exit(0);
  217. } elsif (m/^--test$/) {
  218. $testmode= 1;
  219. } elsif (m/^--verbose$/) {
  220. $verbosemode= +1;
  221. } elsif (m/^--quiet$/) {
  222. $verbosemode= -1;
  223. } elsif (m/^--install$/) {
  224. check_many_actions();
  225. @ARGV >= 4 || &badusage(_g("--install needs <link> <name> <path> <priority>"));
  226. ($alink,$name,$apath,$apriority,@ARGV) = @ARGV;
  227. $apriority =~ m/^[-+]?\d+/ || &badusage(_g("priority must be an integer"));
  228. $action = 'install';
  229. } elsif (m/^--(remove|set)$/) {
  230. check_many_actions();
  231. @ARGV >= 2 || &badusage(sprintf(_g("--%s needs <name> <path>"), $1));
  232. ($name,$apath,@ARGV) = @ARGV;
  233. $action = $1;
  234. } elsif (m/^--(display|auto|config|list|remove-all)$/) {
  235. check_many_actions();
  236. @ARGV || &badusage(sprintf(_g("--%s needs <name>"), $1));
  237. $action = $1;
  238. $name= shift(@ARGV);
  239. } elsif (m/^--slave$/) {
  240. @ARGV >= 3 || &badusage(_g("--slave needs <link> <name> <path>"));
  241. ($slink,$sname,$spath,@ARGV) = @ARGV;
  242. defined($aslavelink{$sname}) && &badusage(sprintf(_g("slave name %s duplicated"), $sname));
  243. $aslavelinkcount{$slink}++ && &badusage(sprintf(_g("slave link %s duplicated"), $slink));
  244. $aslavelink{$sname}= $slink;
  245. $aslavepath{$sname}= $spath;
  246. } elsif (m/^--altdir$/) {
  247. @ARGV || &badusage(sprintf(_g("--%s needs a <directory> argument"), "altdir"));
  248. $altdir= shift(@ARGV);
  249. } elsif (m/^--admindir$/) {
  250. @ARGV || &badusage(sprintf(_g("--%s needs a <directory> argument"), "admindir"));
  251. $admindir= shift(@ARGV);
  252. } elsif (m/^--all$/) {
  253. $action = 'all';
  254. } else {
  255. &badusage(sprintf(_g("unknown option \`%s'"), $_));
  256. }
  257. }
  258. defined($name) && defined($aslavelink{$name}) &&
  259. badusage(sprintf(_g("name %s is both primary and slave"), $name));
  260. defined($alink) && $aslavelinkcount{$alink} &&
  261. badusage(sprintf(_g("link %s is both primary and slave"), $alink));
  262. $action ||
  263. badusage(_g("need --display, --config, --set, --install, --remove, --all, --remove-all or --auto"));
  264. $action eq 'install' || !%aslavelink ||
  265. badusage(_g("--slave only allowed with --install"));
  266. if ($action eq 'all') {
  267. &config_all();
  268. exit 0;
  269. }
  270. if (read_link_group()) {
  271. if ($action eq 'remove') {
  272. # FIXME: Be consistent for now with the case when we try to remove a
  273. # non-existing path from an existing link group file.
  274. exit 0;
  275. } elsif ($action ne 'install') {
  276. pr(sprintf(_g("No alternatives for %s."), $name));
  277. exit 1;
  278. }
  279. }
  280. if ($action eq 'display') {
  281. find_best_version();
  282. display_link_group();
  283. exit 0;
  284. }
  285. if ($action eq 'list') {
  286. list_link_group();
  287. exit 0;
  288. }
  289. find_best_version();
  290. if ($action eq 'config') {
  291. config_alternatives($name);
  292. }
  293. if ($action eq 'set') {
  294. set_alternatives($name);
  295. }
  296. if (defined($linkname= readlink("$altdir/$name"))) {
  297. if ($linkname eq $best) {
  298. $state= 'expected';
  299. } elsif (defined(readlink("$altdir/$name.dpkg-tmp"))) {
  300. $state= 'expected-inprogress';
  301. } else {
  302. $state= 'unexpected';
  303. }
  304. } elsif ($! == &ENOENT) {
  305. $state= 'nonexistent';
  306. } else {
  307. $state= 'unexpected';
  308. }
  309. # Possible values for:
  310. # $mode manual, auto
  311. # $state expected, expected-inprogress, unexpected, nonexistent
  312. # $action auto, install, remove, remove-all
  313. # all independent
  314. if ($action eq 'auto') {
  315. &pr(sprintf(_g("Setting up automatic selection of %s."), $name))
  316. if $verbosemode > 0;
  317. unlink("$altdir/$name.dpkg-tmp") || $! == &ENOENT ||
  318. &quit(sprintf(_g("unable to remove %s: %s"), "$altdir/$name.dpkg-tmp", $!));
  319. unlink("$altdir/$name") || $! == &ENOENT ||
  320. &quit(sprintf(_g("unable to remove %s: %s"), "$altdir/$name", $!));
  321. $state= 'nonexistent';
  322. $mode = 'auto';
  323. }
  324. # $mode manual, auto
  325. # $state expected, expected-inprogress, unexpected, nonexistent
  326. # $action auto, install, remove
  327. # action=auto <=> state=nonexistent
  328. if ($state eq 'unexpected' && $mode eq 'auto') {
  329. &pr(sprintf(_g("%s has been changed (manually or by a script).\n".
  330. "Switching to manual updates only."), "$altdir/$name"))
  331. if $verbosemode > 0;
  332. $mode = 'manual';
  333. }
  334. # $mode manual, auto
  335. # $state expected, expected-inprogress, unexpected, nonexistent
  336. # $action auto, install, remove
  337. # action=auto <=> state=nonexistent
  338. # state=unexpected => mode=manual
  339. &pr(sprintf(_g("Checking available versions of %s, updating links in %s ...\n".
  340. "(You may modify the symlinks there yourself if desired - see \`man ln'.)"), $name, $altdir))
  341. if $verbosemode > 0;
  342. if ($action eq 'install') {
  343. if (defined($link) && $link ne $alink) {
  344. &pr(sprintf(_g("Renaming %s link from %s to %s."), $name, $link, $alink))
  345. if $verbosemode > 0;
  346. rename_mv($link,$alink) || $! == &ENOENT ||
  347. &quit(sprintf(_g("unable to rename %s to %s: %s"), $link, $alink, $!));
  348. }
  349. $link= $alink;
  350. my $i;
  351. if (!defined($i= $versionnum{$apath})) {
  352. push(@versions,$apath);
  353. $versionnum{$apath}= $i= $#versions;
  354. }
  355. $priorities[$i]= $apriority;
  356. for $sname (keys %aslavelink) {
  357. my $j;
  358. if (!defined($j= $slavenum{$sname})) {
  359. push(@slavenames,$sname);
  360. $slavenum{$sname}= $j= $#slavenames;
  361. }
  362. my $oldslavelink = $slavelinks[$j];
  363. my $newslavelink = $aslavelink{$sname};
  364. $slavelinkcount{$oldslavelink}-- if defined($oldslavelink);
  365. $slavelinkcount{$newslavelink}++ &&
  366. &quit(sprintf(_g("slave link name %s duplicated"), $newslavelink));
  367. if (defined($oldslavelink) && $newslavelink ne $oldslavelink) {
  368. &pr(sprintf(_g("Renaming %s slave link from %s to %s."), $sname, $oldslavelink, $newslavelink))
  369. if $verbosemode > 0;
  370. rename_mv($oldslavelink,$newslavelink) || $! == &ENOENT ||
  371. &quit(sprintf(_g("unable to rename %s to %s: %s"), $oldslavelink, $newslavelink, $!));
  372. }
  373. $slavelinks[$j]= $newslavelink;
  374. }
  375. for (my $j = 0; $j <= $#slavenames; $j++) {
  376. $slavepath{$i,$j}= $aslavepath{$slavenames[$j]};
  377. }
  378. fill_missing_slavepaths();
  379. }
  380. if ($action eq 'remove') {
  381. my $hits = 0;
  382. if ($mode eq "manual" and $state ne "expected" and (map { $hits += $apath eq $_ } @versions) and $hits and $linkname eq $apath) {
  383. &pr(_g("Removing manually selected alternative - switching to auto mode"));
  384. $mode = "auto";
  385. }
  386. if (defined(my $i = $versionnum{$apath})) {
  387. my $k = $#versions;
  388. $versionnum{$versions[$k]}= $i;
  389. delete $versionnum{$versions[$i]};
  390. $versions[$i]= $versions[$k]; $#versions--;
  391. $priorities[$i]= $priorities[$k]; $#priorities--;
  392. for (my $j = 0; $j <= $#slavenames; $j++) {
  393. $slavepath{$i,$j}= $slavepath{$k,$j};
  394. delete $slavepath{$k,$j};
  395. }
  396. } else {
  397. &pr(sprintf(_g("Alternative %s for %s not registered, not removing."), $apath, $name))
  398. if $verbosemode > 0;
  399. }
  400. }
  401. if ($action eq 'remove-all') {
  402. $mode = "auto";
  403. my $k = $#versions;
  404. for (my $i = 0; $i <= $#versions; $i++) {
  405. $k--;
  406. delete $versionnum{$versions[$i]};
  407. $#priorities--;
  408. for (my $j = 0; $j <= $#slavenames; $j++) {
  409. $slavepath{$i,$j}= $slavepath{$k,$j};
  410. delete $slavepath{$k,$j};
  411. }
  412. }
  413. $#versions=$k;
  414. }
  415. for (my $j = 0; $j <= $#slavenames; $j++) {
  416. my $i;
  417. for ($i = 0; $i <= $#versions; $i++) {
  418. last if $slavepath{$i,$j} ne '';
  419. }
  420. if ($i > $#versions) {
  421. &pr(sprintf(_g("Discarding obsolete slave link %s (%s)."), $slavenames[$j], $slavelinks[$j]))
  422. if $verbosemode > 0;
  423. unlink("$altdir/$slavenames[$j]") || $! == &ENOENT ||
  424. &quit(sprintf(_g("unable to remove %s: %s"), "$altdir/$slavenames[$j]", $!));
  425. unlink($slavelinks[$j]) || $! == &ENOENT ||
  426. &quit(sprintf(_g("unable to remove %s: %s"), $slavelinks[$j], $!));
  427. my $k = $#slavenames;
  428. $slavenum{$slavenames[$k]}= $j;
  429. delete $slavenum{$slavenames[$j]};
  430. $slavelinkcount{$slavelinks[$j]}--;
  431. $slavenames[$j]= $slavenames[$k]; $#slavenames--;
  432. $slavelinks[$j]= $slavelinks[$k]; $#slavelinks--;
  433. for (my $i = 0; $i <= $#versions; $i++) {
  434. $slavepath{$i,$j}= $slavepath{$i,$k};
  435. delete $slavepath{$i,$k};
  436. }
  437. $j--;
  438. }
  439. }
  440. if ($mode eq 'manual') {
  441. &pr(sprintf(_g("Automatic updates of %s are disabled, leaving it alone."), "$altdir/$name"))
  442. if $verbosemode > 0;
  443. &pr(sprintf(_g("To return to automatic updates use \`update-alternatives --auto %s'."), $name))
  444. if $verbosemode > 0;
  445. } else {
  446. if ($state eq 'expected-inprogress') {
  447. &pr(sprintf(_g("Recovering from previous failed update of %s ..."), $name));
  448. rename_mv("$altdir/$name.dpkg-tmp","$altdir/$name") ||
  449. &quit(sprintf(_g("unable to rename %s to %s: %s"), "$altdir/$name.dpkg-tmp", "$altdir/$name", $!));
  450. $state= 'expected';
  451. }
  452. }
  453. # $mode manual, auto
  454. # $state expected, expected-inprogress, unexpected, nonexistent
  455. # $action auto, install, remove
  456. # action=auto <=> state=nonexistent
  457. # state=unexpected => mode=manual
  458. # mode=auto => state!=expected-inprogress && state!=unexpected
  459. open(AF,">$admindir/$name.dpkg-new") ||
  460. &quit(sprintf(_g("unable to open %s for write: %s"), "$admindir/$name.dpkg-new", $!));
  461. paf($mode);
  462. &paf($link);
  463. for (my $j = 0; $j <= $#slavenames; $j++) {
  464. &paf($slavenames[$j]);
  465. &paf($slavelinks[$j]);
  466. }
  467. find_best_version();
  468. &paf('');
  469. for (my $i = 0; $i <= $#versions; $i++) {
  470. &paf($versions[$i]);
  471. &paf($priorities[$i]);
  472. for (my $j = 0; $j <= $#slavenames; $j++) {
  473. &paf($slavepath{$i,$j});
  474. }
  475. }
  476. &paf('');
  477. close(AF) || &quit(sprintf(_g("unable to close %s: %s"), "$admindir/$name.dpkg-new", $!));
  478. if ($mode eq 'auto') {
  479. if ($best eq '') {
  480. &pr(sprintf(_g("Last package providing %s (%s) removed, deleting it."), $name, $link))
  481. if $verbosemode > 0;
  482. unlink("$altdir/$name") || $! == &ENOENT ||
  483. &quit(sprintf(_g("unable to remove %s: %s"), "$altdir/$name", $!));
  484. unlink("$link") || $! == &ENOENT ||
  485. &quit(sprintf(_g("unable to remove %s: %s"), "$link", $!));
  486. unlink("$admindir/$name.dpkg-new") ||
  487. &quit(sprintf(_g("unable to remove %s: %s"), "$admindir/$name.dpkg-new", $!));
  488. unlink("$admindir/$name") || $! == &ENOENT ||
  489. &quit(sprintf(_g("unable to remove %s: %s"), "$admindir/$name", $!));
  490. exit(0);
  491. } else {
  492. $linkname = readlink($link);
  493. if (!defined($linkname) && $! != ENOENT) {
  494. &pr(sprintf(_g("warning: %s is supposed to be a symlink to %s\n".
  495. " (or nonexistent); however, readlink failed: %s"), $link, "$altdir/$name", $!))
  496. if $verbosemode > 0;
  497. } elsif (!defined($linkname) ||
  498. (defined($linkname) && $linkname ne "$altdir/$name")) {
  499. unlink("$link.dpkg-tmp") || $! == &ENOENT ||
  500. &quit(sprintf(_g("unable to ensure %s nonexistent: %s"), "$link.dpkg-tmp", $!));
  501. symlink("$altdir/$name","$link.dpkg-tmp") ||
  502. &quit(sprintf(_g("unable to make %s a symlink to %s: %s"), "$link.dpkg-tmp", "$altdir/$name", $!));
  503. rename_mv("$link.dpkg-tmp",$link) ||
  504. &quit(sprintf(_g("unable to install %s as %s: %s"), "$link.dpkg-tmp", $link, $!));
  505. }
  506. if (defined($linkname= readlink("$altdir/$name")) && $linkname eq $best) {
  507. &pr(sprintf(_g("Leaving %s (%s) pointing to %s."), $name, $link, $best))
  508. if $verbosemode > 0;
  509. } else {
  510. &pr(sprintf(_g("Updating %s (%s) to point to %s."), $name, $link, $best))
  511. if $verbosemode > 0;
  512. }
  513. unlink("$altdir/$name.dpkg-tmp") || $! == &ENOENT ||
  514. &quit(sprintf(_g("unable to ensure %s nonexistent: %s"), "$altdir/$name.dpkg-tmp", $!));
  515. symlink($best,"$altdir/$name.dpkg-tmp");
  516. }
  517. }
  518. rename_mv("$admindir/$name.dpkg-new","$admindir/$name") ||
  519. &quit(sprintf(_g("unable to rename %s to %s: %s"), "$admindir/$name.dpkg-new", "$admindir/$name", $!));
  520. if ($mode eq 'auto') {
  521. rename_mv("$altdir/$name.dpkg-tmp","$altdir/$name") ||
  522. &quit(sprintf(_g("unable to install %s as %s: %s"), "$altdir/$name.dpkg-tmp", "$altdir/$name", $!));
  523. for (my $j = 0; $j <= $#slavenames; $j++) {
  524. $sname= $slavenames[$j];
  525. $slink= $slavelinks[$j];
  526. $linkname = readlink($slink);
  527. if (!defined($linkname) && $! != ENOENT) {
  528. &pr(sprintf(_g("warning: %s is supposed to be a slave symlink to\n".
  529. " %s, or nonexistent; however, readlink failed: %s"), $slink, "$altdir/$sname", $!))
  530. if $verbosemode > 0;
  531. } elsif (!defined($linkname) ||
  532. (defined($linkname) && $linkname ne "$altdir/$sname")) {
  533. unlink("$slink.dpkg-tmp") || $! == &ENOENT ||
  534. &quit(sprintf(_g("unable to ensure %s nonexistent: %s"), "$slink.dpkg-tmp", $!));
  535. symlink("$altdir/$sname","$slink.dpkg-tmp") ||
  536. &quit(sprintf(_g("unable to make %s a symlink to %s: %s"), "$slink.dpkg-tmp", "$altdir/$sname", $!));
  537. rename_mv("$slink.dpkg-tmp",$slink) ||
  538. &quit(sprintf(_g("unable to install %s as %s: %s"), "$slink.dpkg-tmp", $slink, $!));
  539. }
  540. $spath= $slavepath{$bestnum,$j};
  541. unlink("$altdir/$sname.dpkg-tmp") || $! == &ENOENT ||
  542. &quit(sprintf(_g("unable to ensure %s nonexistent: %s"), "$altdir/$sname.dpkg-tmp", $!));
  543. if ($spath eq '') {
  544. &pr(sprintf(_g("Removing %s (%s), not appropriate with %s."), $sname, $slink, $best))
  545. if $verbosemode > 0;
  546. unlink("$altdir/$sname") || $! == &ENOENT ||
  547. &quit(sprintf(_g("unable to remove %s: %s"), "$altdir/$sname", $!));
  548. unlink("$slink") || $! == &ENOENT ||
  549. &quit(sprintf(_g("unable to remove %s: %s"), $slink, $!));
  550. } else {
  551. if (defined($linkname= readlink("$altdir/$sname")) && $linkname eq $spath) {
  552. &pr(sprintf(_g("Leaving %s (%s) pointing to %s."), $sname, $slink, $spath))
  553. if $verbosemode > 0;
  554. } else {
  555. &pr(sprintf(_g("Updating %s (%s) to point to %s."), $sname, $slink, $spath))
  556. if $verbosemode > 0;
  557. }
  558. symlink("$spath","$altdir/$sname.dpkg-tmp") ||
  559. &quit(sprintf(_g("unable to make %s a symlink to %s: %s"), "$altdir/$sname.dpkg-tmp", $spath, $!));
  560. rename_mv("$altdir/$sname.dpkg-tmp","$altdir/$sname") ||
  561. &quit(sprintf(_g("unable to install %s as %s: %s"), "$altdir/$sname.dpkg-tmp", "$altdir/$sname", $!));
  562. }
  563. }
  564. }
  565. sub config_message {
  566. if ($#versions < 0) {
  567. print "\n";
  568. printf _g("There is no program which provides %s.\n".
  569. "Nothing to configure.\n"), $name;
  570. return -1;
  571. }
  572. if ($#versions == 0) {
  573. print "\n";
  574. printf _g("There is only 1 program which provides %s\n".
  575. "(%s). Nothing to configure.\n"), $name, $versions[0];
  576. return -1;
  577. }
  578. print STDOUT "\n";
  579. printf(STDOUT _g("There are %s alternatives which provide \`%s'.\n\n".
  580. " Selection Alternative\n".
  581. "-----------------------------------------------\n"),
  582. $#versions+1, $name);
  583. for (my $i = 0; $i <= $#versions; $i++) {
  584. printf(STDOUT "%s%s %8s %s\n",
  585. (readlink("$altdir/$name") eq $versions[$i]) ? '*' : ' ',
  586. ($best eq $versions[$i]) ? '+' : ' ',
  587. $i+1, $versions[$i]);
  588. }
  589. printf(STDOUT "\n"._g("Press enter to keep the default[*], or type selection number: "));
  590. return 0;
  591. }
  592. sub config_alternatives {
  593. my $preferred;
  594. do {
  595. return if config_message() < 0;
  596. $preferred=<STDIN>;
  597. chop($preferred);
  598. } until $preferred eq '' || $preferred>=1 && $preferred<=$#versions+1 &&
  599. ($preferred =~ m/[0-9]*/);
  600. if ($preferred ne '') {
  601. $mode = "manual";
  602. $preferred--;
  603. printf STDOUT _g("Using \`%s' to provide \`%s'.")."\n", $versions[$preferred], $name;
  604. my $spath = $versions[$preferred];
  605. symlink("$spath","$altdir/$name.dpkg-tmp") ||
  606. &quit(sprintf(_g("unable to make %s a symlink to %s: %s"), "$altdir/$name.dpkg-tmp", $spath, $!));
  607. rename_mv("$altdir/$name.dpkg-tmp","$altdir/$name") ||
  608. &quit(sprintf(_g("unable to install %s as %s: %s"), "$altdir/$name.dpkg-tmp", "$altdir/$name", $!));
  609. # Link slaves...
  610. for( my $slnum = 0; $slnum < @slavenames; $slnum++ ) {
  611. my $slave = $slavenames[$slnum];
  612. if ($slavepath{$preferred,$slnum} ne '') {
  613. checked_symlink($slavepath{$preferred,$slnum},
  614. "$altdir/$slave.dpkg-tmp");
  615. checked_mv("$altdir/$slave.dpkg-tmp", "$altdir/$slave");
  616. } else {
  617. &pr(sprintf(_g("Removing %s (%s), not appropriate with %s."), $slave, $slavelinks[$slnum], $versions[$preferred]))
  618. if $verbosemode > 0;
  619. unlink("$altdir/$slave") || $! == &ENOENT ||
  620. &quit(sprintf(_g("unable to remove %s: %s"), "$altdir/$slave", $!));
  621. }
  622. }
  623. }
  624. }
  625. sub set_alternatives {
  626. $mode = "manual";
  627. # Get prefered number
  628. my $preferred = -1;
  629. for (my $i = 0; $i <= $#versions; $i++) {
  630. if($versions[$i] eq $apath) {
  631. $preferred = $i;
  632. last;
  633. }
  634. }
  635. if($preferred == -1){
  636. &quit(sprintf(_g("Cannot find alternative `%s'."), $apath)."\n")
  637. }
  638. printf STDOUT _g("Using \`%s' to provide \`%s'.")."\n", $apath, $name;
  639. symlink("$apath","$altdir/$name.dpkg-tmp") ||
  640. &quit(sprintf(_g("unable to make %s a symlink to %s: %s"), "$altdir/$name.dpkg-tmp", $apath, $!));
  641. rename_mv("$altdir/$name.dpkg-tmp","$altdir/$name") ||
  642. &quit(sprintf(_g("unable to install %s as %s: %s"), "$altdir/$name.dpkg-tmp", "$altdir/$name", $!));
  643. # Link slaves...
  644. for (my $slnum = 0; $slnum < @slavenames; $slnum++ ) {
  645. my $slave = $slavenames[$slnum];
  646. if ($slavepath{$preferred,$slnum} ne '') {
  647. checked_symlink($slavepath{$preferred,$slnum},
  648. "$altdir/$slave.dpkg-tmp");
  649. checked_mv("$altdir/$slave.dpkg-tmp", "$altdir/$slave");
  650. } else {
  651. &pr(sprintf(_g("Removing %s (%s), not appropriate with %s."), $slave, $slavelinks[$slnum], $versions[$preferred]))
  652. if $verbosemode > 0;
  653. unlink("$altdir/$slave") || $! == &ENOENT ||
  654. &quit(sprintf(_g("unable to remove %s: %s"), "$altdir/$slave", $!));
  655. }
  656. }
  657. }
  658. sub pr { print(STDOUT "@_\n") || &quit(sprintf(_g("error writing stdout: %s"), $!)); }
  659. sub paf {
  660. $_[0] =~ m/\n/ && &quit(sprintf(_g("newlines prohibited in update-alternatives files (%s)"), $_[0]));
  661. print(AF "$_[0]\n") || &quit(sprintf(_g("error writing stdout: %s"), $!));
  662. }
  663. sub gl {
  664. $!=0; $_= <AF>;
  665. length($_) || &quit(sprintf(_g("error or eof reading %s for %s (%s)"), "$admindir/$name", $_[0], $!));
  666. s/\n$// || &badfmt(sprintf(_g("missing newline after %s"), $_[0]));
  667. $_;
  668. }
  669. sub badfmt {
  670. &quit(sprintf(_g("internal error: %s corrupt: %s"), "$admindir/$name", $_[0]));
  671. }
  672. sub rename_mv {
  673. return (rename($_[0], $_[1]) || (system(("mv", $_[0], $_[1])) == 0));
  674. }
  675. sub checked_symlink {
  676. my ($filename, $linkname) = @_;
  677. symlink($filename, $linkname) ||
  678. &quit(sprintf(_g("unable to make %s a symlink to %s: %s"), $linkname, $filename, $!));
  679. }
  680. sub checked_mv {
  681. my ($source, $dest) = @_;
  682. rename_mv($source, $dest) ||
  683. &quit(sprintf(_g("unable to install %s as %s: %s"), $source, $dest, $!));
  684. }
  685. sub config_all {
  686. opendir ADMINDIR, $admindir or die sprintf(_g("Serious problem: %s"), $!);
  687. my @filenames = grep !/^\.\.?$/, readdir ADMINDIR;
  688. close ADMINDIR;
  689. foreach my $name (@filenames) {
  690. system "$0 --config $name";
  691. exit $? if $?;
  692. }
  693. }
  694. exit(0);
  695. # vim: nowrap ts=8 sw=4