RPM versioning with Jenkins
I’m currently deploying Jenkins at work, in order to simplify the build process. At the moment, we have a fairly complicated process that involves a couple dozen steps, which, even though each step is fairly braindead, does require a good amount of focus and planning in order to get everything just right.
Jenkins provides a build number for all the different projects you might configure. Each build number is unique in a given project; this ensures that you can refer to specific builds without too much of a hassle. That’s great, but knowing that I’m trying to achieve fully automated RPM packaging, it doesn’t help me much.
The versioning system in our company (defined by moi, pretty much), is heavily inspired by the usual RPM rules: release numbers are reset to 1 whenever the version number changes. There’s no real way, as far as I know, to use Jenkins’ build number for this, as there is no programmatic way to reset the build number in Jenkins.
Multi-instance exclusive lock in Python
The first issue I tried to tackle was the fact that the Jenkins server will run multiple builds at the same time, possibly for the same project. What this means is that simply having stuff in a file isn’t really sensible due to concurrency issues. I know that portalocker roughly does the same as what I tried to do, but not completely. portalocker provides a lock to a file; my implementation provides a transaction-kinda lock, regardless of whether you want to access files or databases or whatever.
Without further ado, here’s the class:
"""
Manage a system-wide lock used to inform other processes they should wait.
The lockfile argument is the token that should be shared amongst all the
processes wanting to acquire exclusive locks to the same resource.
"""
= 0.1
= %
# I only target linux systems... Sorry.
=
= False
return
"""
Try to acquire a lock. This is a blocking call, until the lock is
acquired. If you want a non-blocking call, try nb_acquire().
"""
""" TODO: Raise """
pass
= True
""" TODO: Raise """
pass
=
return
return False
return True
""" TODO: Raise """
pass
= False
There’s a couple of TODOs, and I might push this stuff to my github account at some point or another, but it gives a general idea on what I tried to do. The jist of it is that I use mkdir as an atomic locking operation. What you lock against is basically just an identifier, which means that whether you want to read/write to files, access a database, fiddle your toes doesn’t realy matter. If the lock acquisition fails, the acquire()
method will block until it is able to acquire the lock. If you don’t want it to block, there’s always nb_acquire()
which will just return a boolean indicating the lock status.
How to use it
Here’s how it could be used:
=
# Primary usage
print
And that’s really just it. The locking magic happens in the with
statement. As soon you enter the with
statement, you are guaranteed to be the sole executor of whatever code you have running. As soon as you leave the scope of the with
statement, the lock is released. Below are a few other scenarios of how you could use the above class:
=
# Another kind of usage
=
print
# If you do not call the following function, you're going to have bugs.
# Yet another kind of usage (non-blocking, this time)
print
print
So, now we can execute a block of code without having to worry about other instances of the script mucking about our stuff. Let’s work on the RPM versioning now.
Efficient RPM release numbers
RPM versioning works like this: 1.2.3-4
where 1.2.3
is the version number, and 4
is the release. For a more in-depth overview of RPM versioning, check out Fedora’s wiki, or something. I usually put the version number of my software in a file somewhere, so getting that isn’t usually a problem. However, the release part of the RPM version is a bit more tricky. The way we use it is that every time we release something to the QA team (or directly to production if it’s a very urgent bugfix), we increment the release number if the version number hasn’t changed. This is the case, for example, if you have to fix something in the packaging process, or if you are applying successive fixes to the release branch (according to nvie’s branching model).
This also means that as soon as the version number changes, the release number should be reset to 1. I wrote the following tool, which uses the Lock
class above, to keep track of all the different releases that have been built so far, based on the package name and version number:
=
=
return
=
pass
=
= 0
+= 1
return
=
=
=
=
= not
=
print
print %
How to use it
The tool is used as follows:
For example:
So, there we have it. A quick tool that keeps track of what release every package is supposed to have, and stores it in an easy-to-maintain location (a JSON file in the user’s home directory). This tool is easily usable in Jenkins scripts, but that’ll be for a different blog entry.