2015-09-17 00:58:34 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
"""
|
|
|
|
Query the github API for the git tags of a project, and return a list of
|
|
|
|
version tags for recent releases, or the default release.
|
|
|
|
|
|
|
|
The default release is the most recent non-RC version.
|
|
|
|
|
2017-01-19 17:52:13 +01:00
|
|
|
Recent is a list of unique major.minor versions, where each is the most
|
2015-09-17 00:58:34 +02:00
|
|
|
recent version in the series.
|
|
|
|
|
|
|
|
For example, if the list of versions is:
|
|
|
|
|
|
|
|
1.8.0-rc2
|
|
|
|
1.8.0-rc1
|
|
|
|
1.7.1
|
|
|
|
1.7.0
|
|
|
|
1.7.0-rc1
|
|
|
|
1.6.2
|
|
|
|
1.6.1
|
|
|
|
|
|
|
|
`default` would return `1.7.1` and
|
|
|
|
`recent -n 3` would return `1.8.0-rc2 1.7.1 1.6.2`
|
|
|
|
"""
|
2015-10-30 21:22:51 +01:00
|
|
|
from __future__ import absolute_import
|
2015-09-17 00:58:34 +02:00
|
|
|
from __future__ import print_function
|
2015-10-30 21:22:51 +01:00
|
|
|
from __future__ import unicode_literals
|
2015-09-17 00:58:34 +02:00
|
|
|
|
|
|
|
import argparse
|
|
|
|
import itertools
|
|
|
|
import operator
|
2016-05-12 20:41:40 +02:00
|
|
|
import sys
|
2015-09-17 00:58:34 +02:00
|
|
|
from collections import namedtuple
|
|
|
|
|
|
|
|
import requests
|
|
|
|
|
|
|
|
|
|
|
|
GITHUB_API = 'https://api.github.com/repos'
|
|
|
|
|
2018-10-04 10:40:39 +02:00
|
|
|
STAGES = ['tp', 'beta', 'rc']
|
|
|
|
|
2015-09-17 00:58:34 +02:00
|
|
|
|
2018-09-01 01:58:30 +02:00
|
|
|
class Version(namedtuple('_Version', 'major minor patch stage edition')):
|
2015-09-17 00:58:34 +02:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def parse(cls, version):
|
2017-06-06 04:26:37 +02:00
|
|
|
edition = None
|
2015-09-17 00:58:34 +02:00
|
|
|
version = version.lstrip('v')
|
2018-09-01 01:58:30 +02:00
|
|
|
version, _, stage = version.partition('-')
|
|
|
|
if stage:
|
2018-10-04 10:40:39 +02:00
|
|
|
if not any(marker in stage for marker in STAGES):
|
2018-09-01 01:58:30 +02:00
|
|
|
edition = stage
|
|
|
|
stage = None
|
|
|
|
elif '-' in stage:
|
|
|
|
edition, stage = stage.split('-')
|
2015-09-17 00:58:34 +02:00
|
|
|
major, minor, patch = version.split('.', 3)
|
2018-09-01 01:58:30 +02:00
|
|
|
return cls(major, minor, patch, stage, edition)
|
2015-09-17 00:58:34 +02:00
|
|
|
|
|
|
|
@property
|
|
|
|
def major_minor(self):
|
|
|
|
return self.major, self.minor
|
|
|
|
|
|
|
|
@property
|
|
|
|
def order(self):
|
|
|
|
"""Return a representation that allows this object to be sorted
|
|
|
|
correctly with the default comparator.
|
|
|
|
"""
|
2018-10-04 10:40:39 +02:00
|
|
|
# non-GA releases should appear before GA releases
|
|
|
|
# Order: tp -> beta -> rc -> GA
|
|
|
|
if self.stage:
|
|
|
|
for st in STAGES:
|
|
|
|
if st in self.stage:
|
|
|
|
stage = (STAGES.index(st), self.stage)
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
stage = (len(STAGES),)
|
|
|
|
|
2018-09-01 01:58:30 +02:00
|
|
|
return (int(self.major), int(self.minor), int(self.patch)) + stage
|
2015-09-17 00:58:34 +02:00
|
|
|
|
|
|
|
def __str__(self):
|
2018-09-01 01:58:30 +02:00
|
|
|
stage = '-{}'.format(self.stage) if self.stage else ''
|
2017-06-06 04:26:37 +02:00
|
|
|
edition = '-{}'.format(self.edition) if self.edition else ''
|
2018-09-01 01:58:30 +02:00
|
|
|
return '.'.join(map(str, self[:3])) + edition + stage
|
2015-09-17 00:58:34 +02:00
|
|
|
|
|
|
|
|
2018-03-07 20:39:24 +01:00
|
|
|
BLACKLIST = [ # List of versions known to be broken and should not be used
|
|
|
|
Version.parse('18.03.0-ce-rc2'),
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2015-09-17 00:58:34 +02:00
|
|
|
def group_versions(versions):
|
|
|
|
"""Group versions by `major.minor` releases.
|
|
|
|
|
|
|
|
Example:
|
|
|
|
|
|
|
|
>>> group_versions([
|
|
|
|
Version(1, 0, 0),
|
|
|
|
Version(2, 0, 0, 'rc1'),
|
|
|
|
Version(2, 0, 0),
|
|
|
|
Version(2, 1, 0),
|
|
|
|
])
|
|
|
|
|
|
|
|
[
|
|
|
|
[Version(1, 0, 0)],
|
|
|
|
[Version(2, 0, 0), Version(2, 0, 0, 'rc1')],
|
|
|
|
[Version(2, 1, 0)],
|
|
|
|
]
|
|
|
|
"""
|
|
|
|
return list(
|
|
|
|
list(releases)
|
|
|
|
for _, releases
|
|
|
|
in itertools.groupby(versions, operator.attrgetter('major_minor'))
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def get_latest_versions(versions, num=1):
|
|
|
|
"""Return a list of the most recent versions for each major.minor version
|
|
|
|
group.
|
|
|
|
"""
|
|
|
|
versions = group_versions(versions)
|
2017-06-06 04:26:37 +02:00
|
|
|
num = min(len(versions), num)
|
2015-09-17 00:58:34 +02:00
|
|
|
return [versions[index][0] for index in range(num)]
|
|
|
|
|
|
|
|
|
|
|
|
def get_default(versions):
|
2018-09-01 01:58:30 +02:00
|
|
|
"""Return a :class:`Version` for the latest GA version."""
|
2015-09-17 00:58:34 +02:00
|
|
|
for version in versions:
|
2018-09-01 01:58:30 +02:00
|
|
|
if not version.stage:
|
2015-09-17 00:58:34 +02:00
|
|
|
return version
|
|
|
|
|
|
|
|
|
2016-05-12 20:41:40 +02:00
|
|
|
def get_versions(tags):
|
|
|
|
for tag in tags:
|
|
|
|
try:
|
2018-03-07 20:39:24 +01:00
|
|
|
v = Version.parse(tag['name'])
|
2018-09-01 01:58:30 +02:00
|
|
|
if v in BLACKLIST:
|
|
|
|
continue
|
|
|
|
yield v
|
2016-05-12 20:41:40 +02:00
|
|
|
except ValueError:
|
|
|
|
print("Skipping invalid tag: {name}".format(**tag), file=sys.stderr)
|
|
|
|
|
|
|
|
|
2017-06-06 04:26:37 +02:00
|
|
|
def get_github_releases(projects):
|
2015-09-17 00:58:34 +02:00
|
|
|
"""Query the Github API for a list of version tags and return them in
|
|
|
|
sorted order.
|
|
|
|
|
|
|
|
See https://developer.github.com/v3/repos/#list-tags
|
|
|
|
"""
|
2017-06-06 04:26:37 +02:00
|
|
|
versions = []
|
|
|
|
for project in projects:
|
|
|
|
url = '{}/{}/tags'.format(GITHUB_API, project)
|
|
|
|
response = requests.get(url)
|
|
|
|
response.raise_for_status()
|
|
|
|
versions.extend(get_versions(response.json()))
|
2015-09-17 00:58:34 +02:00
|
|
|
return sorted(versions, reverse=True, key=operator.attrgetter('order'))
|
|
|
|
|
|
|
|
|
|
|
|
def parse_args(argv):
|
|
|
|
parser = argparse.ArgumentParser(description=__doc__)
|
|
|
|
parser.add_argument('project', help="Github project name (ex: docker/docker)")
|
|
|
|
parser.add_argument('command', choices=['recent', 'default'])
|
|
|
|
parser.add_argument('-n', '--num', type=int, default=2,
|
|
|
|
help="Number of versions to return from `recent`")
|
|
|
|
return parser.parse_args(argv)
|
|
|
|
|
|
|
|
|
|
|
|
def main(argv=None):
|
|
|
|
args = parse_args(argv)
|
2017-06-06 04:26:37 +02:00
|
|
|
versions = get_github_releases(args.project.split(','))
|
2015-09-17 00:58:34 +02:00
|
|
|
|
|
|
|
if args.command == 'recent':
|
|
|
|
print(' '.join(map(str, get_latest_versions(versions, args.num))))
|
|
|
|
elif args.command == 'default':
|
|
|
|
print(get_default(versions))
|
|
|
|
else:
|
|
|
|
raise ValueError("Unknown command {}".format(args.command))
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|