From Fedora Project Wiki
No edit summary
No edit summary
 
(30 intermediate revisions by the same user not shown)
Line 10: Line 10:
Fedora's versioning scheme must adapt to whatever scheme any upstream developer may choose for versioning their software, while at the same time preserving the two goals above.
Fedora's versioning scheme must adapt to whatever scheme any upstream developer may choose for versioning their software, while at the same time preserving the two goals above.


== Definitions ==
The <code>Epoch:</code> tag, which provides the most significant input to RPM's version comparison function, SHOULD ONLY be used or incremented when necessary to avoid ordering issues.  If present, it MUST contain a positive integer.  <code>Epoch:</code> MUST NOT ever be decreased in any way.
 
The tilde (<code>~</code>) notation which alters the way RPM does version comparisons MUST NOT be used.
 
== Some definitions ==


Note that upstreams may each have their own terminology and it is in general impossible to define these terms with complete generality.  For some upstreams, every commit is itself considered a version.  Many upstreams never make releases, instead just letting users take whatever is in the code repository at any given time.
Note that upstreams may each have their own terminology and it is in general impossible to define these terms with complete generality.  For some upstreams, every commit is itself considered a version.  Many upstreams never make releases, instead just letting users take whatever is in the code repository at any given time.
Line 25: Line 29:
;postrelease version
;postrelease version
: Any version which happens after a particular release is technically "post-release", but before upstream begins making prereleases for the next version, any snapshot is considered a postrelease version.
: Any version which happens after a particular release is technically "post-release", but before upstream begins making prereleases for the next version, any snapshot is considered a postrelease version.
;non-sorting version sequence
: A sequence of version strings which is not ordered in the same way that RPM's version comparison function would order it.  RPM has a somewhat complicated version comparison function which it will use to determine if a package is "newer".  If upstream's idea of what constitutes a "newer" version differs from RPM's implementation then simply using upstream's versions directly will result in updates which don't actually update any packages.
== Examples ==
Examples of many possible versioning scenarios are available from [[User:Tibbs/VersioningCleanupExamples]].


== Simple versioning ==
== Simple versioning ==


Most upstream versioning schemes are "simple"; they generate versions like "1.2.03.007a".  They consists of one or more version components, separated by periods.  Each component is a whole number, potentially with leading zeroes.  The rightmost component can also have a letter appended.  The value of a component must *never* be reduced (to a value which sorts lower) without a component somewhere to the left increasing.
Most upstream versioning schemes are "simple"; they generate versions like "1.2.03.007p1".  They consists of one or more version components, separated by periods.  Each component is a whole number, potentially with leading zeroes.  The rightmost component can also include one or more ASCII letters, upper or lower case.  The value of a component must *never* be reduced (to a value which sorts lower) without a component somewhere to the left increasing.  Note that the version sequence ("1.4a", "1.4b", "1.4") does not meet this criterion, as "4" sorts lower than "4b".  The sequence ("1.4", "1.4a", "1.4b") is, however, simple.


This is a very common versioning scheme, and the vast majority of software projects use something which works like this.
This is a very common versioning scheme, and the vast majority of software projects use something which works like this.


To package '''released versions''' of software using this versioning scheme:
To package '''release versions''' of software using this versioning scheme:
* Use the upstream verbatim in the <code>Version:</code> tag.
* Use the upstream verbatim in the <code>Version:</code> tag.  Don't trim leading zeroes.
* Use a <code>Release:</code> tag starting with 1 (never 0).  Append the [[Packaging:DistTag|Dist tag]].  Increment the release by one for each update you make.  Reset to 1 whenever you change <code>Version:</code>.
* Use a <code>Release:</code> tag starting with 1 (never 0).  Append the [[Packaging:DistTag|Dist tag]].  Increment the release (by 1) for each update you make.  Reset to 1 whenever you change <code>Version:</code>.


== More complex versioning ==
== More complex versioning ==
Line 40: Line 51:
There are several ways in which the simple scheme might not work in a particular situation:
There are several ways in which the simple scheme might not work in a particular situation:


* You wish to package a prerelease version.
* Upstream has never chosen a version; only snapshots are available for packaging.
* You wish to package a snapshot
* Upstream simply doesn't use a version scheme which orders properly under RPM's version comparison operation.
* Upstream has never chosen a version; only snapshots are available for packaging
* You wish to package a prerelease version (snapshot or otherwise).
* Upstream simply doesn't use a version scheme which orders properly under RPM's version comparison operation; perhaps it doesn't even
* You wish to package a postrelease snapshot.
* Upstream was thought to be following one scheme but then changed in a way that can't be sorted.
* Upstream was thought to be following one scheme but then changed in a way that can't be sorted.
* You need to apply a small fix to a release branch of Fedora without updating the newer branches.
* More than one of the above may apply (lucky you).  Follow all of the relevant recommendations below together.


More than one of these may apply.
The methods for dealing with most of these issues involves potentially removing some information from the <code>Version:</code> tag while imposing additional structure onto the <code>Release:</code> tag. There are potentially three fields which comprise the structured <code>Release:</code> tag:
 
=== Release tag ===
 
 
 
 
=== Prerelease and Postrelease ===
 
A version is considered prerelease if it
 
 
 
If you are packaging something which upstream intends to release as, say, version X.Y in the future, then y
 
If what you're packaging has progressed beyond an existing
 
If upstream has never chosen a version, use "0".  This is the lowest nonempty version representable under RPM's ordering.
Do not use '0.0" or "0.0.0" or the like as these actually sort higher than "0".  If upstream later chooses "0" as their version then there is no conflict; if they choose anything else then
 
 
then the situation is a bit more complex.  Many upstreams use a scheme where release versions are simple, but prereleases don't order properly with the release version because upstream appends an extra identifier like "rc2" or "beta4" which is then removed from the version when the actual release happens.  And for a shapshot, additional information about when the snapshot was taken must be included
 
=== Version: tag ===
 
If the software being packaged
 
 
== Prelease versions or snapshots ==
 
If upstream does follow a "simple" versioning scheme, but you wish to package either a prerelease version, or some intermediate version taken from upstream's source control system (a snapshot) then the situation is a bit more complex.  Many upstreams use a scheme where release versions are simple, but prereleases don't order properly with the release version because upstream appends an extra identifier like "rc2" or "beta4" which is then removed from the version when the actual release happens.  And for a shapshot, additional information about when the snapshot was taken must be included
 
 
 
=== Version: tag ===
 
=== Release: tag ===
 
 
 
 
== More complicated versioning ==
If the versioning does not fit that pattern, or you wish to package prerelease versions or snapshots, then the situation becomes more complicated. 
 
===
 
If the version is non-numeric (contains tags that are not numbers), you may need to include the additional non-numeric characters in the release field.
 
There are four cases where the version contains non-numeric characters:
 
* Pre-release packages: Packages released as "pre-release" versions, prior to a "final" version. Example tags include "alpha", "beta", "rc", "cvs". Unfortunately, we cannot simply put these letters into the version tag, so we use the Release field for this. Details can be found here: [[#NonNumericRelease|  Non-Numeric Version in Release]]
 
* Post-release packages: Packages released after a "final" version. These packages contain the same numeric version as the "final" version, but have an additional non-numeric identifier. Details can be found here: [[#NonNumericRelease|  Non-Numeric Version in Release]]
 
* Snapshot packages: Packages built from cvs or subversion snapshots. These packages could be either "pre" or "post" release packages. Details can be found here: [[#NonNumericRelease|  Non-Numeric Version in Release]]
 
{{admon/note|Non-use of tilde "~"|rpm-4.10 and above support the Debian-style tilde character which causes a piece of the version to sort before anything else.  At the moment, using this is not permitted for these reasons:
* it is not necessary when the following release tag guidelines are followed properly.
* it introduces another special character to achieve something which can be handled in other ways.}}
 
== Release Tag ==
The release tag of packages which are not pre- or post-release or snapshots must consist of a positive integer release number and a %{?dist} tag: <code>42%{?dist}</code>.  When a minor change (spec file changed, patch added/removed) occurs, or a package is rebuilt to use newer headers or libraries, the release number is incremented. If a major change (new version of the software being packaged) occurs, the version number is changed to reflect the new software version, and the release number is reset to 1.  It is important to follow these guidelines because automated tools will occasionally be used to increment releases and rebuild packages.
 
=== Non-Numeric Version in Release ===
There are three cases in which non-numeric versions occur in the Release field:
 
* Snapshot packages
* Pre-release packages
* Post-release packages
 
==== Snapshot packages ====
 
Snapshot packages contain data about where the snapshot came from as well as
ordering information for rpm.  The information about the snapshot will be
called %{checkout} in this section.
 
%{checkout} consists of the date that the snapshot is made in YYYYMMDD
format, a short (2-5 characters) string identifying the type of revision
control system or that this is a snapshot, and optionally, up to 13
characters (ASCII) alphanumeric characters that could be useful in finding
the revision in the revision control system.
 
For instance, if you create a snapshot from a git repository on January 2,
2011 with git hash 9e88d7e9efb1bcd5b41a408037bb7cfd47220a64, %{checkout}
string could be any of the following:
 
<pre>
20110102snap
20110102git
20110102git9e88d7e
</pre>
 
If the snapshot package is considered a "pre-release package",
follow the guidelines listed in Pre-Release Packages for snapshot packages,
using the %{checkout} that you decide on above.  (For instance, in
kismet-0-0.3.20040204svn, 20040204svn is the %{checkout})
 
If the snapshot is a "post-release package", follow the
guidelines in the Post-Release Packages section.  Where the %{posttag} in
that section is the %{checkout} string you decided on above.
 
Example (post-release cvs):
<pre>
kismet-1.0-1%{?dist} (this is the formal release of kismet 1.0)
kismet-1.0-2%{?dist} (this is a bugfix build to the 1.0 release)
kismet-1.0-3.20050515cvs%{?dist} (move to a post-release cvs checkout)
kismet-1.0-4.20050515cvs%{?dist} (bugfix to the post-release cvs checkout)
kismet-1.0-5.20050517cvs%{?dist} (new cvs checkout, note the increment of %{X})
</pre>
 
==== Pre-Release packages ====
Non-numeric versioned "pre-release" packages can be problematic so they must be treated with care.
These are cases where the upstream "pre-release" version has letters rather than simple numbers in their version.  Often they have tags like alpha, beta, rc, or letters like a and b denoting that it is a version before the "final" number.  Unfortunately, we cannot simply put these letters into the version tag, so we'll use the Release field for this.
 
Release Tag for Pre-Release Packages:
<code>0.%{X}.%{alphatag}%{?dist}</code>
 
Where <code>%{X}</code> is the release number increment, and <code>%{alphatag}</code> is the string that came from the version.
In this case, the period '.' is used as the delimiter between the release number increment, and the non-numeric version string. No other extra characters may appear in the Release field. This is to prevent Release values such as "3jpp_2fc.42-spotwashere%{?dist}".
 
==== Examples ====
{|
|+ '''Example (mozilla pre-release)'''
|-
! Source Archive !! Description
|-
|style=white-space:nowrap| <code>mozilla-1.4a.tar.gz</code> || (this is a pre-release, version 1.4a of mozilla)
|-
| <code>mozilla-1.4.tar.gz</code> || (this is what the 1.4 release will actually look like)
|-
! Release Tag !! Explanation
|-
| <code>mozilla-1.4-0.1.a%{?dist}</code> || (so, this is the acceptable Fedora <code>%{name}-%{version}-%{release}</code>)
|-
| <code>mozilla-1.4-1%{?dist}</code> || (and this is what the 1.4 release Fedora <code>%{name}-%{version}-%{release}</code> should be)
|}
 
 
{|
|+ '''Example (alsa-lib pre-release)'''
|-
! Source Archive !! Description
|-
|style=white-space:nowrap| <code>alsa-lib-0.9.2beta1.tar.gz</code> || (this is a beta release of alsa-lib, version 0.9.2beta1)
|-
! Release Tag !! Explanation
|-
| <code>alsa-lib-0.9.2-0.1.beta1%{?dist}</code> || (this is the correct Fedora <code>%{name}-%{version}-%{release}</code>)
|-
| <code>alsa-lib-0.9.2-0.2.beta1%{?dist}</code> || (this is an incremented Fedora <code>%{name}-%{version}-%{release}</code>. Note that the first 0 is not incremented.)
|}
 


{|
* package release number (<code><pkgrel></code>)
|+ '''Example (kismet pre-release svn checkout)'''
* extra version information (<code><extraver></code>)
|-
* snapshot information (<code><snapinfo></code>)
! Release Tag !! Explanation
* minor release bump (<code><minorbump></code>)
|-
|style=white-space:nowrap| <code>kismet-0-0.1.20040110svn%{?dist}</code> || (this is a pre-release, svn checkout of kismet)
|-
| <code>kismet-0-0.2.20040110svn%{?dist}</code> || (this is a bugfix to the previous package)
|-
| <code>kismet-0-0.3.20040204svn%{?dist}</code> || (this is a new svn checkout, note the increment of <code>%{X}</code>)
|-
| <code>kismet-1.0-1%{?dist}</code> || (this is the formal release of kismet 1.0)
|}


The package release number MUST always be present while the others may or may not be depending on the situation.


{|
Those items which are present are combined (with periods to separate them) to construct the final <code>Release:</code> tag. In the usual notation where square brackets indicate that an item is optional:
|+ '''Upgrade Path Example (mozilla)'''
|-
! Release Tag !! Explanation
|-
|style=white-space:nowrap| <code>mozilla-1.4-0.1.a%{?dist}</code> || (this is the Fedora package for 1.4a, as [[#PreReleaseExampleMozilla|above]])
|-
| <code>mozilla-1.4-0.2.a%{?dist}</code> || (this is the first patch on top of 1.4a)
|-
| <code>mozilla-1.4-0.3.a%{?dist}</code> || (this is another new patch on top of 1.4a)
|-
| <code>mozilla-1.4-0.4.b%{?dist}</code> || (this is the first build after upgrade to 1.4b)
|-
| <code>mozilla-1.4-0.5.b%{?dist}</code> || (this is a new patch on top of 1.4b)
|-
| <code>mozilla-1.4-1%{?dist}</code> || (this is after moving to 1.4 "final", and to a normal version)
|-
| <code>mozilla-1.4-2%{?dist}</code> || (this is a new patch on top of 1.4 "final")
|}


* <code><pkgrel>[.<extraver>][.<snapinfo>]%{?dist}[.<minorbump>]</code>


{|
The actual values to be used for those three fields are situational and are referenced in the sections below. Note that your particular situation might not result in the use of <code><extraver></code> or <code><snapinfo></code>, and in most situations <code><minorbump></code> won't be used at all. Simply do not include those which you don't have.
|+ '''Upgrade Path Example (alsa-lib)'''
|-
! Release Tag !! Explanation
|-
|style=white-space:nowrap| <code>alsa-lib-0.9.2-0.1.beta1%{?dist}</code> || (this is the Fedora package for 0.9.2beta1, as [[#PreReleaseExampleAlsaLib|above]])
|-
| <code>alsa-lib-0.9.2-0.2.beta1%{?dist}</code> || (this is a new patch on top of 0.9.2beta1)
|-
| <code>alsa-lib-0.9.2-0.3.beta2%{?dist}</code> || (this is after upgrading to 0.9.2beta2)
|-
| <code>alsa-lib-0.9.2-0.4.beta3%{?dist}</code> || (this is after upgrading to 0.9.2beta3)
|-
| <code>alsa-lib-0.9.2-0.5.beta3%{?dist}</code> || (this is a new patch on top of 0.9.2beta3)
|-
| <code>alsa-lib-0.9.2-0.6.rc1%{?dist}</code> || (this is after upgrading to 0.9.2rc1)
|-
| <code>alsa-lib-0.9.2-0.7.rc2%{?dist}</code> || (this is after upgrading to 0.9.2rc2)
|-
| <code>alsa-lib-0.9.2-1%{?dist}</code> || (this is after upgrading to 0.9.2 "final", version becomes normal)
|-
| <code>alsa-lib-0.9.2-2%{?dist}</code> || (this is a new patch on top of 0.9.2 "final")
|}


==== Post-Release packages ====
=== Upstream has never chosen a version ===
Like pre-release packages, non-numeric versioned "post-release" packages can be problematic and also must be treated with care. These fall under two generic categories:


* Properly ordered simple versions. These are usually due to quick bugfix releases, such as openssl-0.9.6b or gkrellm-2.1.7a. As new versions come out, the non-numeric tag is properly incremented (e.g. openssl-0.9.6c) or the numeric version is increased and the non-numeric tag is dropped (openssl-0.9.7). In this case, the non-numeric characters are permitted in the Version: field.
When upstream has never chosen a version, you MUST use <code>Version: 0</code>"<code>0</code>" sorts lower than any other possible value that upstream might choose. And if upstream does choose to release "version 0" then you can immediately move to using <code>Release: 1%{?dist}</code> with no ordering issues.
* When upstream uses versions that attempt to have meaning to humans instead of being easy for a computer to order. For example, GA1, CR2, PR3. In this case, the non-numeric string can be put in the Release: field using the following syntax: <code>%{X}.%{posttag}</code>


In this syntax, <code>%{X}</code> is the release number increment, and <code>%{posttag}</code> is the string that came from the version. Here, the period '.' is used as the delimiter between the release number increment and the non-numeric version string. No other extra characters may appear in the Release field.
=== Upstream uses invalid characters in the version ===


Example (complicated post-release):
It's possible that upstream uses characters besides ASCII letters (upper and lower case), digits and periods in its version. They must be removed and potentially replaced with valid characters. Any such alterations MUST be documented in the specfile. It is not possible to cover all potential situations here, so it is left to the packager to alter the upstream versioning scheme consistently.
<pre>
foo-1.1.0-0.1.BETA%{?dist} (this is a prerelease, first beta)
foo-1.1.0-0.2.BETA1%{?dist} (this is a prerelease, second beta)
foo-1.1.0-0.3.BETA2%{?dist} (this is a prerelease, third beta)
foo-1.1.0-0.4.CR1%{?dist} (this is a prerelease, candidate release 1)
foo-1.1.0-0.5.CR2%{?dist} (this is a prerelease, candidate release 2)
foo-1.1.0-1%{?dist} (final release)
foo-1.1.0-2.GA1%{?dist} (post release, GA1)
foo-1.1.0-3.CP1%{?dist} (post release, CP1, after GA1)
foo-1.1.0-4.CP2%{?dist} (post release, CP2, after CP1)
foo-1.1.0-5.SP1%{?dist} (post release, SP1, after CP2)
foo-1.1.0-6.SP1_CP1%{?dist} (post release, SP1_CP1, after SP1)
</pre>


It is important to be careful with the post-release scheme, to ensure that package ordering is correct. It may be necessary to use Epoch to ensure that the current package is considered newer than the previous package. In such cases, the packager should try to convince upstream to be more reasonable with their post-release versioning.
After altering the version to be free of invalid characters, see [[#Unsortable versions]] below if the modifications, when applied to successive releases from upstream, will not order properly.


Also, packagers using the post-release scheme should put a comment in their spec file with a brief description of the upstream conventions for naming/versioning that are being worked around.
=== Unsortable versions ===


=== The %{?dist} Tag ===
When upstream uses a versioning scheme that does not sort properly, first see if there is any portion which can be removed from the right side of the version string such that the remainder is sortable.  This is often possible if upstream uses a sequence like ("1.2pre1", "1.2pre1", "1.2final"). If so, use the removed portion as "extra version information" above, and the remainder as the package version.  If this splitting left a leading or trailing periods from either value, remove them.
Use of the <code>%{?dist}</code> tag in the Release field is mandatory.
Please refer to the [[Packaging:DistTag]] documentation for the details on the appropriate way to do this.


=== Minor release bumps for old branches ===
If this is not possible, use Version: 0 and move the _entire_ version string into "extra version information".
Sometimes, you may find yourself in a situation where an older branch needs a fix, but the newer branches are fine. For example, if foo = 1.0-1%{?dist} in F{{FedoraVersionNumber|previous}} and F{{FedoraVersionNumber|current}}, and only F{{FedoraVersionNumber|previous}} needs a fix. Normally, you would need to bump the release in each of the branches to ensure that F{{FedoraVersionNumber|previous}} < F{{FedoraVersionNumber|current}}, but that is a waste of time and energy for the newer branches which do not need to be touched.


In this case, you can add an extra digit (prefixed by a period) to the very end of the release tag in the F{{FedoraVersionNumber|previous}} branch, instead of bumping it the usual way.
=== Snapshots ===
Example: <BR>
<pre>


Release: 1%{?dist}
All snapshots MUST contain a snapshot information field (<code><snapinfo></code>) in the <code>Release:</code> tag.  That field must at minimum consist of the date in eight-digit "YYYYMMDD" format.  The packager MAY include up to 17 characters of additional information after the date.  The following formats are suggested:
</pre>


<pre>
* <code>YYYYMMDD.<revision></code>
* <code>YYYYMMDD<scm><revision></code>


Release: 1%{?dist}.1
Where <code><scm></code> is a short string identifying the source code control system upstream uses (e.g. "git", "svn", "hg") or the string "snap".  <code><revision></code> is either a short git commit hash, a subversion revision number, or something else useful in identifying the precise revision in upstream's source code control system.  Obviously if CVS is used, no such revision information exists, so it would be omitted, but otherwise it SHOULD be included.
</pre>


This will make a <code>foo-1.0-1.fc{{FedoraVersionNumber|previous}}.1</code> package, which is still less than the <code>foo-1.0-1.fc{{FedoraVersionNumber|current}}</code> package in the F{{FedoraVersionNumber|current}} branch.
=== Prerelease versions ===


As necessary, the last digit (the minor release bump) can be incremented on a per-branch basis as neededHowever, this must never be used in rawhide, because it is always unnecessary thereSimply increment the appropriate portion of Release: in rawhide.
In the <code>Version:</code> tag, use the version that upstream has determined the next release will be.  For the <code><pkgrel></code> field of the <code>Release:</code> tag, use a number of the form "0.N" where N is an integer beginning with 1 and increasing for each revision of the packagePrerelease versions MUST use a <code>Release:</code> tag strictly less than 1, as this is the sole indicator that a prerelease has been packaged.


=== Release and post-release versions ===


'''BE CAREFUL WITH THIS!''' You always want to make sure that packages in branches can be upgraded to packages in more recent branches. Or to put it simply, F{{FedoraVersionNumber|previous}} < FC{{FedoraVersionNumber|current}} < F{{FedoraVersionNumber|next}}. There is a tool in the rpmdevtools package called <code>rpmdev-vercmp</code>. This tool will prompt you for two sets of Epoch, Version, and Release, then tell you which is considered newer by rpm.
For the <code><pkgrel></code> field of the <code>Release:</code> tag, use an integer beginning with 1 and increasing for each revision of the package.  Release and post-release versions MUST use a <code>Release:</code> tag greater than or equal to 1.


== Rawhide is allowed to lag ==
=== Upstream makes unsortable changes ===


XXX Needs workMight not be at the right place in this document.
It is possible that upstream simply adopts a different versioning scheme, fails to follow an expected pattern, or even simply resets their version to some lower value.  If none of the above operations can help with giving a version which sorts properly, or give you a version which simply sorts lower than the packages already in Fedora, then you have little recourse but to increment the <code>Epoch:</code> tag, or to begin using it by adding <code>Epoch: 1</code>At the same time, try to work with upstream to hopefully minimize the need to involve <code>Epoch:</code> in the future.


The one time where we temporarily break the upgrade path is from released Fedora to rawhide.  It is allowable to temporarily have a higher EVR in a released Fedora vs rawhide in cases where we need to get a fix out to users of the Fedora releases quickly (for instance, for a security issue) but the fixed package does not currently build in rawhide (for instance, because of a change in a dependent library that requires porting effort).  The changes should be checked into Fedora's SCM for rawhide although there is not yet a build and getting a rawhide build that restores the upgrade path should be a priority.
=== You need to change an old branch without rebuilding the others ===


<code>rpm</code>'s understanding of what package is newer comes from the values of the <code>Epoch</code>, <code>Version</code>, and <code>Release</code> tags. The next few sections explain what values to use in those tags to ensure that the EVR consistently increases.
Sometimes, you may find yourself in a situation where an older branch needs a fix, but the newer branches are fine. For example, if pkg = 1.0-1%{?dist} in F{{FedoraVersionNumber|previous}} and F{{FedoraVersionNumber|current}}, and only F{{FedoraVersionNumber|previous}} needs a fix. Normally, you would need to bump the release in each of the branches to ensure that F{{FedoraVersionNumber|previous}} < F{{FedoraVersionNumber|current}}, but that is a waste of time and energy for the newer branches which do not need to be touched.


== Old examples from introduction ==
In this case, you MAY set <code><minorbump</code> to an in integer beginning with '1' and increasing by one for each minor bump you need to do.  Remove <code><minorbump></code> once you are able to increase the package release normally without introducing ordering issues.


In Fedora we want to ensure that there is always an upgrade path from Fedora release to Fedora release and from Fedora release to the packages in updates.  To do that we need to make sure the packages in the newer Fedora releases have an equal or higher <code>Epoch:Version-Release</code> (EVR) than the ones in older releases.  For example, Fedora-{{FedoraVersionNumber|previous}} ships with <code>foo-1.0-1.fc{{FedoraVersionNumber|previous}}</code> and Fedora-{{FedoraVersionNumber|current}} ships with <code>foo-1.0-2.fc{{FedoraVersionNumber|current}}</code>.  This example shows that the Fedora-{{FedoraVersionNumber|current}} package will properly upgrade the Fedora-{{FedoraVersionNumber|previous}} package.
== Rawhide is allowed to lag temporarily ==


A more complex example.  Fedora-{{FedoraVersionNumber|previous}} ships with <code>foo-1.0-1.fc{{FedoraVersionNumber|previous}}</code> and Fedora-{{FedoraVersionNumber|current}} ships with <code>foo-1.0-2.fc{{FedoraVersionNumber|current}}</code>.  Fedora-{{FedoraVersionNumber|previous}} gets an updated foo package: <code>foo-1.0.1-1.fc{{FedoraVersionNumber|previous}}</code>.  This update is EVR higher than the package Fedora-{{FedoraVersionNumber|previous}} shipped with so that's good.  However, it is also EVR higher than what Fedora-{{FedoraVersionNumber|current}} shipped with so we need to make a Fedora-{{FedoraVersionNumber|current}} update at the same time, likely: <code>foo-1.0.1-1.fc{{FedoraVersionNumber|current}}</code>making that update means that someone updating from Fedora-{{FedoraVersionNumber|previous}} with updates enabled will get the correct package on Fedora-{{FedoraVersionNumber|current}} with updates enabled.
A package MAY temporarily have a lower EVR in Rawhide when compared to a release branch of Fedora ONLY in the case where the package fails to build in RawhideThis permits important updates to be pushed to existing Fedora releases regardless of the current state of Rawhide.


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

Latest revision as of 16:53, 21 February 2017

In Fedora, the package versioning scheme (which encompasses both the Version: and Release: tags, as well as Epoch:) balances two separate goals:

  • To provide information about the version of the packaged software to users. When a package version does not correspond directly to the upstream version of the software, some useful information is conveyed such as the date a snapshot was taken or whether a pre- or post-release version has been packaged.
  • To ensure proper ordering between updates to a package within a Fedora release, as well as between Fedora releases. Within a release, Fedora does not allow package versions to go backwards, so the versioning scheme must produce successive versions which increase monotonically, even if the upstream versions don't have this property when compared using RPM's version comparison function. The versioning scheme also ensures that for a package built at the same time for two different Fedora releases, the version in the newer release sorts higher than the version in the older release.

Fedora's versioning scheme must adapt to whatever scheme any upstream developer may choose for versioning their software, while at the same time preserving the two goals above.

The Epoch: tag, which provides the most significant input to RPM's version comparison function, SHOULD ONLY be used or incremented when necessary to avoid ordering issues. If present, it MUST contain a positive integer. Epoch: MUST NOT ever be decreased in any way.

The tilde (~) notation which alters the way RPM does version comparisons MUST NOT be used.

Some definitions

Note that upstreams may each have their own terminology and it is in general impossible to define these terms with complete generality. For some upstreams, every commit is itself considered a version. Many upstreams never make releases, instead just letting users take whatever is in the code repository at any given time.

release version
A version of the software which upstream has decided to release. The act of releasing the software can be as simple as adding a git tag. This includes so-called "point releases" or "patchlevels" which some upstreams make, since those are actually assigned versions and released.
snapshot
An archive taken from upstream's source code control system which is not associated with any release version.
prerelease version
Before a release happens, many upstreams will decide which version that will release will have, and then produce "alphas", "betas", "release candidates", or the like which carry that new version but indicate that the release of that version has not yet been made. These we call prerelease versions. Any snapshots made while upstream is preparing for their release are also considered prerelease versions.
postrelease version
Any version which happens after a particular release is technically "post-release", but before upstream begins making prereleases for the next version, any snapshot is considered a postrelease version.
non-sorting version sequence
A sequence of version strings which is not ordered in the same way that RPM's version comparison function would order it. RPM has a somewhat complicated version comparison function which it will use to determine if a package is "newer". If upstream's idea of what constitutes a "newer" version differs from RPM's implementation then simply using upstream's versions directly will result in updates which don't actually update any packages.

Examples

Examples of many possible versioning scenarios are available from User:Tibbs/VersioningCleanupExamples.

Simple versioning

Most upstream versioning schemes are "simple"; they generate versions like "1.2.03.007p1". They consists of one or more version components, separated by periods. Each component is a whole number, potentially with leading zeroes. The rightmost component can also include one or more ASCII letters, upper or lower case. The value of a component must *never* be reduced (to a value which sorts lower) without a component somewhere to the left increasing. Note that the version sequence ("1.4a", "1.4b", "1.4") does not meet this criterion, as "4" sorts lower than "4b". The sequence ("1.4", "1.4a", "1.4b") is, however, simple.

This is a very common versioning scheme, and the vast majority of software projects use something which works like this.

To package release versions of software using this versioning scheme:

  • Use the upstream verbatim in the Version: tag. Don't trim leading zeroes.
  • Use a Release: tag starting with 1 (never 0). Append the Dist tag. Increment the release (by 1) for each update you make. Reset to 1 whenever you change Version:.

More complex versioning

There are several ways in which the simple scheme might not work in a particular situation:

  • Upstream has never chosen a version; only snapshots are available for packaging.
  • Upstream simply doesn't use a version scheme which orders properly under RPM's version comparison operation.
  • You wish to package a prerelease version (snapshot or otherwise).
  • You wish to package a postrelease snapshot.
  • Upstream was thought to be following one scheme but then changed in a way that can't be sorted.
  • You need to apply a small fix to a release branch of Fedora without updating the newer branches.
  • More than one of the above may apply (lucky you). Follow all of the relevant recommendations below together.

The methods for dealing with most of these issues involves potentially removing some information from the Version: tag while imposing additional structure onto the Release: tag. There are potentially three fields which comprise the structured Release: tag:

  • package release number (<pkgrel>)
  • extra version information (<extraver>)
  • snapshot information (<snapinfo>)
  • minor release bump (<minorbump>)

The package release number MUST always be present while the others may or may not be depending on the situation.

Those items which are present are combined (with periods to separate them) to construct the final Release: tag. In the usual notation where square brackets indicate that an item is optional:

  • <pkgrel>[.<extraver>][.<snapinfo>]%{?dist}[.<minorbump>]

The actual values to be used for those three fields are situational and are referenced in the sections below. Note that your particular situation might not result in the use of <extraver> or <snapinfo>, and in most situations <minorbump> won't be used at all. Simply do not include those which you don't have.

Upstream has never chosen a version

When upstream has never chosen a version, you MUST use Version: 0. "0" sorts lower than any other possible value that upstream might choose. And if upstream does choose to release "version 0" then you can immediately move to using Release: 1%{?dist} with no ordering issues.

Upstream uses invalid characters in the version

It's possible that upstream uses characters besides ASCII letters (upper and lower case), digits and periods in its version. They must be removed and potentially replaced with valid characters. Any such alterations MUST be documented in the specfile. It is not possible to cover all potential situations here, so it is left to the packager to alter the upstream versioning scheme consistently.

After altering the version to be free of invalid characters, see #Unsortable versions below if the modifications, when applied to successive releases from upstream, will not order properly.

Unsortable versions

When upstream uses a versioning scheme that does not sort properly, first see if there is any portion which can be removed from the right side of the version string such that the remainder is sortable. This is often possible if upstream uses a sequence like ("1.2pre1", "1.2pre1", "1.2final"). If so, use the removed portion as "extra version information" above, and the remainder as the package version. If this splitting left a leading or trailing periods from either value, remove them.

If this is not possible, use Version: 0 and move the _entire_ version string into "extra version information".

Snapshots

All snapshots MUST contain a snapshot information field (<snapinfo>) in the Release: tag. That field must at minimum consist of the date in eight-digit "YYYYMMDD" format. The packager MAY include up to 17 characters of additional information after the date. The following formats are suggested:

  • YYYYMMDD.<revision>
  • YYYYMMDD<scm><revision>

Where <scm> is a short string identifying the source code control system upstream uses (e.g. "git", "svn", "hg") or the string "snap". <revision> is either a short git commit hash, a subversion revision number, or something else useful in identifying the precise revision in upstream's source code control system. Obviously if CVS is used, no such revision information exists, so it would be omitted, but otherwise it SHOULD be included.

Prerelease versions

In the Version: tag, use the version that upstream has determined the next release will be. For the <pkgrel> field of the Release: tag, use a number of the form "0.N" where N is an integer beginning with 1 and increasing for each revision of the package. Prerelease versions MUST use a Release: tag strictly less than 1, as this is the sole indicator that a prerelease has been packaged.

Release and post-release versions

For the <pkgrel> field of the Release: tag, use an integer beginning with 1 and increasing for each revision of the package. Release and post-release versions MUST use a Release: tag greater than or equal to 1.

Upstream makes unsortable changes

It is possible that upstream simply adopts a different versioning scheme, fails to follow an expected pattern, or even simply resets their version to some lower value. If none of the above operations can help with giving a version which sorts properly, or give you a version which simply sorts lower than the packages already in Fedora, then you have little recourse but to increment the Epoch: tag, or to begin using it by adding Epoch: 1. At the same time, try to work with upstream to hopefully minimize the need to involve Epoch: in the future.

You need to change an old branch without rebuilding the others

Sometimes, you may find yourself in a situation where an older branch needs a fix, but the newer branches are fine. For example, if pkg = 1.0-1%{?dist} in F40 and F41, and only F40 needs a fix. Normally, you would need to bump the release in each of the branches to ensure that F40 < F41, but that is a waste of time and energy for the newer branches which do not need to be touched.

In this case, you MAY set <minorbump to an in integer beginning with '1' and increasing by one for each minor bump you need to do. Remove <minorbump> once you are able to increase the package release normally without introducing ordering issues.

Rawhide is allowed to lag temporarily

A package MAY temporarily have a lower EVR in Rawhide when compared to a release branch of Fedora ONLY in the case where the package fails to build in Rawhide. This permits important updates to be pushed to existing Fedora releases regardless of the current state of Rawhide.