From Fedora Project Wiki
(Change the alternatives language to strict usage of MUST and MAY.)
(move to drafts category)
 
(4 intermediate revisions by 2 users not shown)
Line 7: Line 7:


== Usage within Fedora ==
== Usage within Fedora ==
Alternatives **MAY** be used to allow parallel installation of software when:
Alternatives '''MAY''' be used to allow parallel installation of software when:
* the software can be used as a drop-in replacement and functions with sufficient similarity that users and other programs would, within reason, not need to know which variant is currently installed
* the software can be used as a drop-in replacement and functions with sufficient similarity that users and other programs would, within reason, not need to know which variant is currently installed
'''AND'''
* the selection of the software is only performed system-wide by the system administrator and end users do not have a need to switch between the variants.
* the selection of the software is only performed system-wide by the system administrator and end users do not have a need to switch between the variants.


Inversely, alternatives **MUST NOT** be used when:
Inversely, alternatives '''MUST NOT''' be used when:
* The software is not a drop-in replacement.  For instance, if common command line arguments are different between the two variants, alternatives **MUST NOT** be used.
* The software is not a drop-in replacement.  For instance, if common command line arguments are different between the two variants, alternatives '''MUST NOT''' be used.
* End users will care which variant they are using.  If a non-root user would gain value by switching between the variants then alternatives **MUST NOT** be used.
'''OR'''
* End users will care which variant they are using.  If a non-root user would gain value by switching between the variants then alternatives '''MUST NOT''' be used.


A good example of using alternatives are the various MTAs which all provide <code>/usr/sbin/sendmail</code> with similar command line arguments.
A good example of using alternatives are the various MTAs which all provide <code>/usr/sbin/sendmail</code> with similar command line arguments.
Line 21: Line 23:
* choice of editor when the user invokes "vi" where the user will care about feature availability, compatibility with plugins, etc
* choice of editor when the user invokes "vi" where the user will care about feature availability, compatibility with plugins, etc


Cases where parallel installation is desirable but alternatives is unsuitable may be scenarios where environment-modules is appropriate.
Cases where parallel installation is desirable but alternatives is unsuitable may be scenarios where [[Packaging:EnvironmentModules|environment-modules]] is appropriate.  MPI and python-sphinx (from Fedora 26 on) are example packages using environment-modules for this purpose.


== How to use alternatives ==
== How to use alternatives ==
Line 137: Line 139:
</pre>
</pre>


[[Category:Packaging guidelines]]
[[Category:Packaging guidelines drafts]]

Latest revision as of 17:58, 13 February 2017

Alternatives

What are alternatives

Alternatives provide means for parallel installation of packages which provide the same functionality by maintaining sets of symlinks (one per package) pointing to alternativized files like this: /path/original-file -> /etc/alternatives/packagename-original-file -> /path/original-file.suffix For more information, see update-alternatives(8) manpage.

Usage within Fedora

Alternatives MAY be used to allow parallel installation of software when:

  • the software can be used as a drop-in replacement and functions with sufficient similarity that users and other programs would, within reason, not need to know which variant is currently installed

AND

  • the selection of the software is only performed system-wide by the system administrator and end users do not have a need to switch between the variants.

Inversely, alternatives MUST NOT be used when:

  • The software is not a drop-in replacement. For instance, if common command line arguments are different between the two variants, alternatives MUST NOT be used.

OR

  • End users will care which variant they are using. If a non-root user would gain value by switching between the variants then alternatives MUST NOT be used.

A good example of using alternatives are the various MTAs which all provide /usr/sbin/sendmail with similar command line arguments.

Bad examples of using alternatives include:

  • the various MPI environments where users care both about which MPI environment they compile against and which one they run against
  • choice of editor when the user invokes "vi" where the user will care about feature availability, compatibility with plugins, etc

Cases where parallel installation is desirable but alternatives is unsuitable may be scenarios where environment-modules is appropriate. MPI and python-sphinx (from Fedora 26 on) are example packages using environment-modules for this purpose.

How to use alternatives

If a package is using alternatives, the files which would otherwise conflict must be installed with an appropriate suffix (for example: %{_sbindir}/sendmail.postfix instead of %{_sbindir}/sendmail), the original locations must be touched (for example: touch %{_sbindir}/sendmail), the links set up by alternatives must be listed as %ghost in the file list and proper Requires: must be added, like in the examples below.

Putting the alternativized files in the file list ensures that they are owned by respective packages, which means that commands like:

  • rpm -qf /usr/bin/foo
  • yum install /usr/bin/foo
  • repoquery --whatprovides /usr/bin/foo

all work properly. Using %ghost for this purpose allows using globs and generated file lists.

Examples

Example from antlr.spec:

Requires(post): %{_sbindir}/update-alternatives
Requires(postun): %{_sbindir}/update-alternatives
...
%install
...
touch %{buildroot}%{_bindir}/antlr

%post
%{_sbindir}/update-alternatives --install %{_bindir}/antlr \
  %{name} %{_bindir}/antlr-java 10

%postun
if [ $1 -eq 0 ] ; then
  %{_sbindir}/update-alternatives --remove %{name} %{_bindir}/antlr-java
fi
...
%files
...
%ghost %{_bindir}/antlr
%{_bindir}/antlr-java

And a more complex example of alternatives invocation from sendmail.spec, slightly edited:

Requires(post): %{_sbindir}/update-alternatives
Requires(postun): %{_sbindir}/update-alternatives
Requires(preun): %{_sbindir}/update-alternatives
...
%install
...
# rename files for alternative usage
mv %{buildroot}%{_sbindir}/sendmail %{buildroot}%{_sbindir}/sendmail.sendmail
touch %{buildroot}%{_sbindir}/sendmail
for i in mailq newaliases rmail; do
	mv %{buildroot}%{_bindir}/$i %{buildroot}%{_bindir}/$i.sendmail
	touch %{buildroot}%{_bindir}/$i
done
mv %{buildroot}%{_mandir}/man1/mailq.1 %{buildroot}%{_mandir}/man1/mailq.sendmail.1
touch %{buildroot}%{_mandir}/man1/mailq.1
mv %{buildroot}%{_mandir}/man1/newaliases.1 %{buildroot}%{_mandir}/man1/newaliases.sendmail.1
touch %{buildroot}%{_mandir}/man1/newaliases.1
mv %{buildroot}%{_mandir}/man5/aliases.5 %{buildroot}%{_mandir}/man5/aliases.sendmail.5
touch %{buildroot}%{_mandir}/man5/aliases.5
mv %{buildroot}%{_mandir}/man8/sendmail.8 %{buildroot}%{_mandir}/man8/sendmail.sendmail.8
touch %{buildroot}%{_mandir}/man8/sendmail.8

%postun
if [ "$1" -ge "1" ]; then
	if [ "`readlink %{_sysconfdir}/alternatives/mta`" == "%{_sbindir}/sendmail.sendmail" ]; then
		%{_sbindir}/alternatives --set mta %{_sbindir}/sendmail.sendmail
	fi
fi

%post
# Set up the alternatives files for MTAs.
%{_sbindir}/update-alternatives --install %{_sbindir}/sendmail mta %{_sbindir}/sendmail.sendmail 90 \
	--slave %{_bindir}/mailq mta-mailq %{_bindir}/mailq.sendmail \
	--slave %{_bindir}/newaliases mta-newaliases %{_bindir}/newaliases.sendmail \
	--slave %{_bindir}/rmail mta-rmail %{_bindir}/rmail.sendmail \
	--slave /usr/lib/sendmail mta-sendmail /usr/lib/sendmail.sendmail \
	--slave %{_sysconfdir}/pam.d/smtp mta-pam %{_sysconfdir}/pam.d/smtp.sendmail \
	--slave %{_mandir}/man8/sendmail.8.gz mta-sendmailman %{_mandir}/man8/sendmail.sendmail.8.gz \
	--slave %{_mandir}/man1/mailq.1.gz mta-mailqman %{_mandir}/man1/mailq.sendmail.1.gz \
	--slave %{_mandir}/man1/newaliases.1.gz mta-newaliasesman %{_mandir}/man1/newaliases.sendmail.1.gz \
	--slave %{_mandir}/man5/aliases.5.gz mta-aliasesman %{_mandir}/man5/aliases.sendmail.5.gz \
	--initscript sendmail
...

%preun
if [ $1 = 0 ]; then
	%{_sbindir}/update-alternatives --remove mta %{_sbindir}/sendmail.sendmail
fi
...

%files
...
%ghost %{_sbindir}/sendmail
%ghost %{_bindir}/mailq
%ghost %{_bindir}/newaliases
%ghost %{_bindir}/rmail
%ghost /usr/lib/sendmail
%ghost %{_sysconfdir}/pam.d/smtp
%ghost %{_mandir}/man8/sendmail.8.gz
%ghost %{_mandir}/man1/mailq.1.gz
%ghost %{_mandir}/man1/newaliases.1.gz
%ghost %{_mandir}/man5/aliases.5.gz

%{_sbindir}/sendmail.sendmail
%{_bindir}/mailq.sendmail
%{_bindir}/newaliases.sendmail
%{_bindir}/rmail.sendmail
/usr/lib/sendmail.sendmail
%config(noreplace) %{_sysconfdir}/pam.d/smtp.sendmail
%{_mandir}/man8/sendmail.sendmail.8.gz
%{_mandir}/man1/mailq.sendmail.1.gz
%{_mandir}/man1/newaliases.sendmail.1.gz
%{_mandir}/man5/aliases.sendmail.5.gz

%attr(0755,root,root) %{_initrddir}/sendmail