2017-11-29 00:45:18 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
"""Project version handling."""
|
|
|
|
from __future__ import (
|
|
|
|
absolute_import, division, print_function, unicode_literals)
|
2017-06-08 19:43:58 +00:00
|
|
|
|
2017-03-28 00:44:58 +00:00
|
|
|
import unicodedata
|
2017-11-29 00:45:18 +00:00
|
|
|
from builtins import object, range
|
2013-08-08 21:12:56 +00:00
|
|
|
from collections import defaultdict
|
2017-06-08 19:43:58 +00:00
|
|
|
|
|
|
|
import six
|
2017-11-29 00:45:18 +00:00
|
|
|
from packaging.version import InvalidVersion, Version
|
2015-06-29 10:32:24 +00:00
|
|
|
|
2017-11-29 00:45:18 +00:00
|
|
|
from readthedocs.builds.constants import (
|
2017-11-29 15:33:16 +00:00
|
|
|
LATEST_VERBOSE_NAME, STABLE_VERBOSE_NAME, TAG)
|
2013-08-08 21:12:56 +00:00
|
|
|
|
2015-06-18 10:09:43 +00:00
|
|
|
|
2015-06-29 09:17:15 +00:00
|
|
|
def get_major(version):
|
2017-11-29 00:45:18 +00:00
|
|
|
"""
|
|
|
|
Return the major version.
|
|
|
|
|
|
|
|
:param version: version to get the major
|
|
|
|
:type version: packaging.version.Version
|
|
|
|
"""
|
2015-08-21 20:55:02 +00:00
|
|
|
# pylint: disable=protected-access
|
2015-06-29 09:17:15 +00:00
|
|
|
return version._version.release[0]
|
2015-06-18 10:09:43 +00:00
|
|
|
|
2013-08-08 21:12:56 +00:00
|
|
|
|
2015-06-29 09:17:15 +00:00
|
|
|
def get_minor(version):
|
2017-11-29 00:45:18 +00:00
|
|
|
"""
|
|
|
|
Return the minor version.
|
|
|
|
|
|
|
|
:param version: version to get the minor
|
|
|
|
:type version: packaging.version.Version
|
|
|
|
"""
|
2015-08-21 20:55:02 +00:00
|
|
|
# pylint: disable=protected-access
|
2015-06-29 09:17:15 +00:00
|
|
|
try:
|
|
|
|
return version._version.release[1]
|
|
|
|
except IndexError:
|
|
|
|
return 0
|
2013-08-08 21:12:56 +00:00
|
|
|
|
|
|
|
|
2015-06-18 10:09:43 +00:00
|
|
|
class VersionManager(object):
|
2015-08-21 20:55:02 +00:00
|
|
|
|
2017-11-29 00:45:18 +00:00
|
|
|
"""Prune list of versions based on version windows."""
|
2015-08-21 20:55:02 +00:00
|
|
|
|
2014-02-15 02:34:53 +00:00
|
|
|
def __init__(self):
|
|
|
|
self._state = defaultdict(lambda: defaultdict(list))
|
2013-08-08 21:12:56 +00:00
|
|
|
|
|
|
|
def add(self, version):
|
2015-06-29 09:17:15 +00:00
|
|
|
self._state[get_major(version)][get_minor(version)].append(version)
|
2013-08-08 21:12:56 +00:00
|
|
|
|
|
|
|
def prune_major(self, num_latest):
|
|
|
|
all_keys = sorted(set(self._state.keys()))
|
|
|
|
major_keep = []
|
2015-08-21 20:55:02 +00:00
|
|
|
for __ in range(num_latest):
|
2017-06-14 16:38:03 +00:00
|
|
|
if all_keys:
|
2013-08-08 21:12:56 +00:00
|
|
|
major_keep.append(all_keys.pop(-1))
|
|
|
|
for to_remove in all_keys:
|
|
|
|
del self._state[to_remove]
|
|
|
|
|
|
|
|
def prune_minor(self, num_latest):
|
2017-05-30 22:09:59 +00:00
|
|
|
for major, minors in list(self._state.items()):
|
2013-08-08 21:12:56 +00:00
|
|
|
all_keys = sorted(set(minors.keys()))
|
|
|
|
minor_keep = []
|
2015-08-21 20:55:02 +00:00
|
|
|
for __ in range(num_latest):
|
2017-06-14 16:38:03 +00:00
|
|
|
if all_keys:
|
2013-08-08 21:12:56 +00:00
|
|
|
minor_keep.append(all_keys.pop(-1))
|
|
|
|
for to_remove in all_keys:
|
|
|
|
del self._state[major][to_remove]
|
2015-06-10 08:56:52 +00:00
|
|
|
|
2013-08-08 21:12:56 +00:00
|
|
|
def prune_point(self, num_latest):
|
2017-05-30 22:09:59 +00:00
|
|
|
for major, minors in list(self._state.items()):
|
|
|
|
for minor in list(minors.keys()):
|
2013-08-08 21:12:56 +00:00
|
|
|
try:
|
2015-08-21 20:55:02 +00:00
|
|
|
self._state[major][minor] = sorted(
|
|
|
|
set(self._state[major][minor]))[-num_latest:]
|
|
|
|
except TypeError:
|
2013-08-08 21:12:56 +00:00
|
|
|
# Raise these for now.
|
|
|
|
raise
|
|
|
|
|
2015-06-29 09:17:15 +00:00
|
|
|
def get_version_list(self):
|
|
|
|
versions = []
|
2017-05-30 22:09:59 +00:00
|
|
|
for major_val in list(self._state.values()):
|
|
|
|
for version_list in list(major_val.values()):
|
2015-06-29 09:17:15 +00:00
|
|
|
versions.extend(version_list)
|
|
|
|
versions = sorted(versions)
|
2015-07-02 12:51:15 +00:00
|
|
|
return [
|
2017-11-29 00:45:18 +00:00
|
|
|
version.public for version in versions if not version.is_prerelease
|
|
|
|
]
|
2015-06-29 09:17:15 +00:00
|
|
|
|
2015-06-18 10:09:43 +00:00
|
|
|
|
2015-06-29 08:02:42 +00:00
|
|
|
def version_windows(versions, major=1, minor=1, point=1):
|
2017-11-29 00:45:18 +00:00
|
|
|
"""
|
|
|
|
Return list of versions that have been pruned to version windows.
|
2015-08-21 20:55:02 +00:00
|
|
|
|
2016-07-15 16:23:01 +00:00
|
|
|
Uses :py:class:`VersionManager` to prune the list of versions
|
2015-08-21 20:55:02 +00:00
|
|
|
|
|
|
|
:param versions: List of version strings
|
|
|
|
:param major: Major version window
|
|
|
|
:param minor: Minor version window
|
|
|
|
:param point: Point version window
|
|
|
|
"""
|
2015-06-18 10:09:43 +00:00
|
|
|
# TODO: This needs some documentation on how VersionManager etc works and
|
|
|
|
# some examples what the expected outcome is.
|
|
|
|
|
2015-06-29 08:40:52 +00:00
|
|
|
version_identifiers = []
|
|
|
|
for version_string in versions:
|
2015-06-29 10:32:24 +00:00
|
|
|
try:
|
|
|
|
version_identifiers.append(Version(version_string))
|
2016-03-11 04:09:47 +00:00
|
|
|
except (InvalidVersion, UnicodeEncodeError):
|
2015-06-29 10:32:24 +00:00
|
|
|
pass
|
2015-06-29 08:40:52 +00:00
|
|
|
|
2013-08-08 21:12:56 +00:00
|
|
|
major_version_window = major
|
|
|
|
minor_version_window = minor
|
|
|
|
point_version_window = point
|
2015-06-10 08:56:52 +00:00
|
|
|
|
2013-08-08 21:12:56 +00:00
|
|
|
manager = VersionManager()
|
2015-06-29 08:40:52 +00:00
|
|
|
for v in version_identifiers:
|
2015-06-18 10:09:43 +00:00
|
|
|
manager.add(v)
|
2013-08-08 21:12:56 +00:00
|
|
|
manager.prune_major(major_version_window)
|
|
|
|
manager.prune_minor(minor_version_window)
|
|
|
|
manager.prune_point(point_version_window)
|
2015-06-29 09:17:15 +00:00
|
|
|
return manager.get_version_list()
|
2015-06-29 10:30:59 +00:00
|
|
|
|
|
|
|
|
|
|
|
def parse_version_failsafe(version_string):
|
2017-11-29 00:45:18 +00:00
|
|
|
"""
|
|
|
|
Parse a version in string form and return Version object.
|
|
|
|
|
|
|
|
If there is an error parsing the string, ``None`` is returned.
|
|
|
|
|
|
|
|
:param version_string: version as string object (e.g. '3.10.1')
|
|
|
|
:type version_string: str or unicode
|
|
|
|
|
|
|
|
:returns: version object created from a string object
|
|
|
|
|
|
|
|
:rtype: packaging.version.Version
|
|
|
|
"""
|
2017-06-09 16:36:25 +00:00
|
|
|
if not isinstance(version_string, six.text_type):
|
|
|
|
uni_version = version_string.decode('utf-8')
|
|
|
|
else:
|
|
|
|
uni_version = version_string
|
|
|
|
|
2015-06-29 10:30:59 +00:00
|
|
|
try:
|
2017-06-09 16:36:25 +00:00
|
|
|
normalized_version = unicodedata.normalize('NFKD', uni_version)
|
|
|
|
ascii_version = normalized_version.encode('ascii', 'ignore')
|
|
|
|
final_form = ascii_version.decode('ascii')
|
|
|
|
return Version(final_form)
|
2017-03-28 00:44:58 +00:00
|
|
|
except (UnicodeError, InvalidVersion):
|
2015-06-29 10:30:59 +00:00
|
|
|
return None
|
2015-06-29 10:34:36 +00:00
|
|
|
|
|
|
|
|
|
|
|
def comparable_version(version_string):
|
2017-11-29 00:45:18 +00:00
|
|
|
"""
|
|
|
|
Can be used as ``key`` argument to ``sorted``.
|
2015-06-29 10:34:36 +00:00
|
|
|
|
2016-06-02 19:18:14 +00:00
|
|
|
The ``LATEST`` version shall always beat other versions in comparison.
|
2015-06-29 10:34:36 +00:00
|
|
|
``STABLE`` should be listed second. If we cannot figure out the version
|
2016-06-08 23:46:20 +00:00
|
|
|
number then we sort it to the bottom of the list.
|
2017-11-29 00:45:18 +00:00
|
|
|
|
|
|
|
:param version_string: version as string object (e.g. '3.10.1' or 'latest')
|
|
|
|
:type version_string: str or unicode
|
|
|
|
|
|
|
|
:returns: a comparable version object (e.g. 'latest' -> Version('99999.0'))
|
|
|
|
|
|
|
|
:rtype: packaging.version.Version
|
2015-08-21 20:55:02 +00:00
|
|
|
"""
|
2015-06-29 10:34:36 +00:00
|
|
|
comparable = parse_version_failsafe(version_string)
|
|
|
|
if not comparable:
|
|
|
|
if version_string == LATEST_VERBOSE_NAME:
|
|
|
|
comparable = Version('99999.0')
|
|
|
|
elif version_string == STABLE_VERBOSE_NAME:
|
|
|
|
comparable = Version('9999.0')
|
|
|
|
else:
|
2016-06-08 23:46:20 +00:00
|
|
|
comparable = Version('0.01')
|
2015-06-29 10:34:36 +00:00
|
|
|
return comparable
|
2015-06-29 10:38:59 +00:00
|
|
|
|
|
|
|
|
2015-07-02 14:52:28 +00:00
|
|
|
def sort_versions(version_list):
|
2017-11-29 00:45:18 +00:00
|
|
|
"""
|
|
|
|
Take a list of Version models and return a sorted list.
|
2015-07-02 14:52:28 +00:00
|
|
|
|
2017-11-29 00:45:18 +00:00
|
|
|
:param version_list: list of Version models
|
|
|
|
:type version_list: list(readthedocs.builds.models.Version)
|
|
|
|
|
|
|
|
:returns: sorted list in descending order (latest version first) of versions
|
|
|
|
|
|
|
|
:rtype: list(tupe(readthedocs.builds.models.Version,
|
|
|
|
packaging.version.Version))
|
2015-07-02 14:52:28 +00:00
|
|
|
"""
|
2015-06-29 10:38:59 +00:00
|
|
|
versions = []
|
|
|
|
for version_obj in version_list:
|
|
|
|
version_slug = version_obj.verbose_name
|
|
|
|
comparable_version = parse_version_failsafe(version_slug)
|
|
|
|
if comparable_version:
|
|
|
|
versions.append((version_obj, comparable_version))
|
|
|
|
|
2017-11-29 00:45:18 +00:00
|
|
|
return list(
|
|
|
|
sorted(
|
|
|
|
versions,
|
|
|
|
key=lambda version_info: version_info[1],
|
|
|
|
reverse=True,
|
|
|
|
))
|
2015-07-02 14:52:28 +00:00
|
|
|
|
|
|
|
|
2015-08-21 20:55:02 +00:00
|
|
|
def highest_version(version_list):
|
2017-11-29 00:45:18 +00:00
|
|
|
"""
|
|
|
|
Return the highest version for a given ``version_list``.
|
|
|
|
|
|
|
|
:rtype: tupe(readthedocs.builds.models.Version, packaging.version.Version)
|
|
|
|
"""
|
2015-07-02 14:52:28 +00:00
|
|
|
versions = sort_versions(version_list)
|
2015-06-29 10:38:59 +00:00
|
|
|
if versions:
|
|
|
|
return versions[0]
|
2017-06-14 16:38:03 +00:00
|
|
|
return (None, None)
|
2015-07-02 14:52:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
def determine_stable_version(version_list):
|
2017-11-29 00:45:18 +00:00
|
|
|
"""
|
|
|
|
Determine a stable version for version list.
|
2015-08-21 20:55:02 +00:00
|
|
|
|
2017-11-29 00:45:18 +00:00
|
|
|
:param version_list: list of versions
|
|
|
|
:type version_list: list(readthedocs.builds.models.Version)
|
|
|
|
|
|
|
|
:returns: version considered the most recent stable one or ``None`` if there
|
|
|
|
is no stable version in the list
|
|
|
|
|
|
|
|
:rtype: readthedocs.builds.models.Version
|
2015-07-02 14:52:28 +00:00
|
|
|
"""
|
|
|
|
versions = sort_versions(version_list)
|
2017-11-29 00:45:18 +00:00
|
|
|
versions = [(version_obj, comparable)
|
|
|
|
for version_obj, comparable in versions
|
|
|
|
if not comparable.is_prerelease]
|
|
|
|
|
2015-07-02 14:52:28 +00:00
|
|
|
if versions:
|
2017-11-29 15:33:16 +00:00
|
|
|
# We take preference for tags over branches. If we don't find any tag,
|
|
|
|
# we just return the first branch found.
|
|
|
|
for version_obj, comparable in versions:
|
|
|
|
if version_obj.type == TAG:
|
|
|
|
return version_obj
|
|
|
|
|
2015-07-02 14:52:28 +00:00
|
|
|
version_obj, comparable = versions[0]
|
|
|
|
return version_obj
|
2017-06-14 16:38:03 +00:00
|
|
|
return None
|