2017-12-25 09:46:52 +00:00
import argparse
2020-04-24 22:04:31 +00:00
import math
2017-12-25 09:46:52 +00:00
import json
import requests
import sys
2022-10-25 13:44:20 +00:00
from enum import Enum
2021-03-20 23:57:24 +00:00
from libversion import Version
2022-10-25 13:44:20 +00:00
from typing import (
Callable ,
Iterable ,
List ,
NamedTuple ,
Optional ,
Tuple ,
TypeVar ,
Type ,
cast ,
)
2017-12-25 09:46:52 +00:00
2020-04-24 22:26:49 +00:00
2022-10-25 13:44:20 +00:00
EnumValue = TypeVar ( " EnumValue " , bound = Enum )
def enum_to_arg ( enum : Enum ) - > str :
return enum . name . lower ( ) . replace ( " _ " , " - " )
def arg_to_enum ( enum_meta : Type [ EnumValue ] , name : str ) - > EnumValue :
return enum_meta [ name . upper ( ) . replace ( " - " , " _ " ) ]
def enum_to_arg_choices ( enum_meta : Type [ EnumValue ] ) - > Tuple [ str , . . . ] :
return tuple ( enum_to_arg ( v ) for v in cast ( Iterable [ EnumValue ] , enum_meta ) )
class Stability ( Enum ) :
STABLE = " stable "
UNSTABLE = " unstable "
VersionPolicy = Callable [ [ Version ] , bool ]
VersionPredicate = Callable [ [ Version , Stability ] , bool ]
class VersionPredicateHolder ( NamedTuple ) :
function : VersionPredicate
def version_to_list ( version : str ) - > List [ int ] :
2020-04-24 22:26:49 +00:00
return list ( map ( int , version . split ( " . " ) ) )
2017-12-25 09:46:52 +00:00
2022-10-25 13:44:20 +00:00
def odd_unstable ( version : Version , selected : Stability ) - > bool :
2021-03-20 23:57:24 +00:00
try :
2022-10-25 13:44:20 +00:00
version_parts = version_to_list ( version . value )
2021-03-20 23:57:24 +00:00
except :
# Failing to parse as a list of numbers likely means the version contains a string tag like “beta”, therefore it is not a stable release.
2022-10-25 13:44:20 +00:00
return selected != Stability . STABLE
2021-03-20 23:57:24 +00:00
2022-10-25 13:44:20 +00:00
if len ( version_parts ) < 2 :
2017-12-25 09:46:52 +00:00
return True
2022-10-25 13:44:20 +00:00
even = version_parts [ 1 ] % 2 == 0
prerelease = ( version_parts [ 1 ] > = 90 and version_parts [ 1 ] < 100 ) or ( version_parts [ 1 ] > = 900 and version_parts [ 1 ] < 1000 )
2018-03-03 05:02:38 +00:00
stable = even and not prerelease
2022-10-25 13:44:20 +00:00
if selected == Stability . STABLE :
2018-03-03 05:02:38 +00:00
return stable
2017-12-25 09:46:52 +00:00
else :
2018-03-03 04:49:47 +00:00
return True
2017-12-25 09:46:52 +00:00
2020-04-24 22:26:49 +00:00
2022-10-25 13:44:20 +00:00
def tagged ( version : Version , selected : Stability ) - > bool :
if selected == Stability . STABLE :
2021-03-20 23:57:24 +00:00
return not ( " alpha " in version . value or " beta " in version . value or " rc " in version . value )
else :
return True
2022-10-25 13:44:20 +00:00
def no_policy ( version : Version , selected : Stability ) - > bool :
2017-12-25 09:46:52 +00:00
return True
2020-04-24 22:26:49 +00:00
2022-10-25 13:44:20 +00:00
class VersionPolicyKind ( Enum ) :
# HACK: Using function as values directly would make Enum
# think they are methods and skip them.
ODD_UNSTABLE = VersionPredicateHolder ( odd_unstable )
TAGGED = VersionPredicateHolder ( tagged )
NONE = VersionPredicateHolder ( no_policy )
2017-12-25 09:46:52 +00:00
2020-04-24 22:26:49 +00:00
2022-10-25 13:44:20 +00:00
def make_version_policy (
version_policy_kind : VersionPolicyKind ,
selected : Stability ,
upper_bound : Optional [ Version ] ,
) - > VersionPolicy :
version_predicate = version_policy_kind . value . function
2020-04-24 22:04:31 +00:00
if not upper_bound :
2021-03-20 23:57:24 +00:00
return lambda version : version_predicate ( version , selected )
2020-04-24 22:04:31 +00:00
else :
2021-03-20 23:57:24 +00:00
return lambda version : version_predicate ( version , selected ) and version < upper_bound
2017-12-25 09:46:52 +00:00
2020-04-24 22:26:49 +00:00
2022-10-25 13:44:20 +00:00
def find_versions ( package_name : str , version_policy : VersionPolicy ) - > List [ Version ] :
# The structure of cache.json: https://gitlab.gnome.org/Infrastructure/sysadmin-bin/blob/master/ftpadmin#L762
cache = json . loads ( requests . get ( f " https://ftp.gnome.org/pub/GNOME/sources/ { package_name } /cache.json " ) . text )
if type ( cache ) != list or cache [ 0 ] != 4 :
raise Exception ( " Unknown format of cache.json file. " )
versions : Iterable [ Version ] = map ( Version , cache [ 2 ] [ package_name ] )
versions = sorted ( filter ( version_policy , versions ) )
return versions
parser = argparse . ArgumentParser (
description = " Find latest version for a GNOME package by crawling their release server. " ,
)
parser . add_argument (
" package-name " ,
help = " Name of the directory in https://ftp.gnome.org/pub/GNOME/sources/ containing the package. " ,
)
parser . add_argument (
" version-policy " ,
help = " Policy determining which versions are considered stable. GNOME packages usually denote stability by alpha/beta/rc tag in the version. For older packages, odd minor versions are unstable but there are exceptions. " ,
choices = enum_to_arg_choices ( VersionPolicyKind ) ,
nargs = " ? " ,
default = enum_to_arg ( VersionPolicyKind . TAGGED ) ,
)
parser . add_argument (
" requested-release " ,
help = " Most of the time, we will want to update to stable version but sometimes it is useful to test. " ,
choices = enum_to_arg_choices ( Stability ) ,
nargs = " ? " ,
default = enum_to_arg ( Stability . STABLE ) ,
)
parser . add_argument (
" --upper-bound " ,
dest = " upper-bound " ,
help = " Only look for versions older than this one (useful for pinning dependencies). " ,
)
2017-12-25 09:46:52 +00:00
2020-04-24 22:26:49 +00:00
if __name__ == " __main__ " :
2017-12-25 09:46:52 +00:00
args = parser . parse_args ( )
2020-04-24 22:26:49 +00:00
package_name = getattr ( args , " package-name " )
2022-10-25 13:44:20 +00:00
requested_release = arg_to_enum ( Stability , getattr ( args , " requested-release " ) )
2020-04-24 22:26:49 +00:00
upper_bound = getattr ( args , " upper-bound " )
2022-10-25 13:44:20 +00:00
if upper_bound is not None :
2021-03-20 23:57:24 +00:00
upper_bound = Version ( upper_bound )
2022-10-25 13:44:20 +00:00
version_policy_kind = arg_to_enum ( VersionPolicyKind , getattr ( args , " version-policy " ) )
version_policy = make_version_policy ( version_policy_kind , requested_release , upper_bound )
2017-12-25 09:46:52 +00:00
2022-10-25 13:44:20 +00:00
try :
versions = find_versions ( package_name , version_policy )
except Exception as error :
print ( error , file = sys . stderr )
2017-12-25 09:46:52 +00:00
sys . exit ( 1 )
if len ( versions ) == 0 :
2020-04-24 22:26:49 +00:00
print ( " No versions matched. " , file = sys . stderr )
2017-12-25 09:46:52 +00:00
sys . exit ( 1 )
2021-03-20 23:57:24 +00:00
print ( versions [ - 1 ] . value )