mirror of
https://github.com/mumble-voip/mumble.git
synced 2025-10-26 11:19:16 +00:00
144 lines
5.3 KiB
Python
Executable File
144 lines
5.3 KiB
Python
Executable File
#!/usr/bin/env python
|
|
#
|
|
# Copyright 2005-2020 The Mumble Developers. All rights reserved.
|
|
# Use of this source code is governed by a BSD-style license
|
|
# that can be found in the LICENSE file at the root of the
|
|
# Mumble source tree or at <https://www.mumble.info/LICENSE>.
|
|
#
|
|
# This script returns the Mumble version string for a Mumble Git
|
|
# repository. The script must be run from within a Mumble Git
|
|
# repository.
|
|
# This is a replacement for `git describe` to make snapshots
|
|
# use the future, untagged version number rather than the previous.
|
|
#
|
|
# The version is of form 1.3.0~2020-07-02~g4f336a2~snapshot.
|
|
# It includes the target release version rather than the previous
|
|
# release (as git describe does).
|
|
#
|
|
# Detailed description:
|
|
#
|
|
# Once upon a time, Mumble used the output of `git describe` as
|
|
# its version string.
|
|
#
|
|
# If a commit was tagged, it was a "release", and got a simple
|
|
# string which was the name of the tag.
|
|
#
|
|
# If a commit wasn't tagged, it got '1.2.6-234-gf552ag1', which
|
|
# consists of the number of commits since the latest tag, and
|
|
# the commit hash of the latest commit.
|
|
#
|
|
# However, the output of `git describe` was found to be confusing
|
|
# in practice. This is because the base version of the `git describe`
|
|
# output is the latest tag, which is to say: the *previous*
|
|
# version of Mumble.
|
|
#
|
|
# So, a user running a snapshot that would become 1.3.0 would be
|
|
# running a version like 1.2.6-234-gf552ag1. This is confusing
|
|
# simply by looking at the version numbers, but the way versioning
|
|
# works inside the Mumble client made it worse: the client's version
|
|
# was referred to as "1.3.0" in several places, but the actual version
|
|
# number said 1.2.6-234-gf552ag1.
|
|
#
|
|
# This script is the replacement for `git describe`. It outputs the
|
|
# *actual* base version of the Mumble tree, rather than the latest tag.
|
|
# This means that snapshots for Mumble 1.3.0 now have the base version
|
|
# '1.3.0'.
|
|
#
|
|
# It also changes the version string: Instead of using dashes
|
|
# as a separator in the version string, it now uses tildes. This allows
|
|
# Debian's dpkg version comparer to correctly sort snapshot versions
|
|
# before release versions. The new string also includes 'snapshot' in the
|
|
# version string to denote to users that the given version is a pre-release
|
|
# snapshot.
|
|
# Furthermore the version number does use the latest commit's date rather
|
|
# than therevision number since the last tag.
|
|
# A full new-style version string looks like this:
|
|
# 1.3.0~2020-07-02~g4f336a2~snapshot.
|
|
|
|
from __future__ import (unicode_literals, print_function, division)
|
|
|
|
import argparse
|
|
import os
|
|
import platform
|
|
import subprocess
|
|
import sys
|
|
|
|
def strip(s):
|
|
s = s.replace('\r', '')
|
|
s = s.replace('\n', '')
|
|
return s
|
|
|
|
def cmd(args):
|
|
shell = platform.system() == 'Windows'
|
|
p = subprocess.Popen(args, shell=shell, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
|
|
stdout, stderr = p.communicate()
|
|
if p.returncode != 0:
|
|
raise Exception('cmd(): {0} failed with status {1}: {2}'.format(args, p.returncode, stderr))
|
|
return stdout.decode('utf-8')
|
|
|
|
# Reads the version from CMakeLists.txt
|
|
def readProjectVersion():
|
|
sourceTreeRoot = strip(cmd(['git', 'rev-parse', '--show-toplevel']))
|
|
|
|
version = None
|
|
with open(os.path.join(sourceTreeRoot, 'CMakeLists.txt'), 'r') as f:
|
|
for line in f:
|
|
# The version is specified as e.g. VERSION "1.4.0.${BUILD_NUMBER}"
|
|
if 'VERSION "' in line and '.${BUILD_NUMBER}"' in line:
|
|
line = line.replace('VERSION "', '')
|
|
line = line[0 : line.find('.${BUILD_NUMBER}"')].strip()
|
|
version = line
|
|
break
|
|
|
|
if version is None:
|
|
raise Exception('Unable to read version from CMakeLists.txt')
|
|
|
|
if len(version) == 0 or not '.' in version:
|
|
raise Exception('Bad version: "{0}"'.format(version))
|
|
|
|
return version
|
|
|
|
def main():
|
|
parser = argparse.ArgumentParser()
|
|
parser.add_argument('-n', '--newline', action = "store_true", help = 'Break line after printing version')
|
|
parser.add_argument('-p', '--project', action = "store_true", help = 'Print CMake project version')
|
|
args = parser.parse_args()
|
|
|
|
if args.newline:
|
|
end = None
|
|
else:
|
|
end = ''
|
|
|
|
if args.project:
|
|
projectVersion = readProjectVersion()
|
|
print(projectVersion, end = end)
|
|
return
|
|
|
|
# Get all tags associated with the latest commit
|
|
latestCommitTags = [x for x in cmd(['git', 'tag', '--points-at', 'HEAD']).split("\n") if x]
|
|
|
|
if len(latestCommitTags) > 1:
|
|
raise RuntimeError("Encountered commit with multiple tags: %s" % latestCommitTags)
|
|
|
|
if len(latestCommitTags) == 1:
|
|
# Most recent commit is tagged -> this is a tagged release version
|
|
# Use the tag as the version-string
|
|
version = latestCommitTags[0]
|
|
else:
|
|
# This is a snapshot (i.e. built from a non-tagged commit)
|
|
|
|
# Get the date of the most recent commit
|
|
latestCommitDate = cmd(['git', 'log', '-1', '--format=%cd', '--date=short']).strip()
|
|
|
|
# Get the hash of the most recent commit (shortened)
|
|
latestCommitHash = cmd(['git', 'rev-parse', '--short' , 'HEAD']).strip()
|
|
|
|
projectVersion = readProjectVersion()
|
|
|
|
version = '{0}~{1}~g{2}~snapshot'.format(projectVersion, latestCommitDate, latestCommitHash)
|
|
|
|
print(version, end = end)
|
|
|
|
if __name__ == '__main__':
|
|
main()
|