2015-11-22 17:39:51 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
|
|
|
import collections
|
2016-08-08 21:00:31 +00:00
|
|
|
import logging
|
2016-03-20 23:28:26 +00:00
|
|
|
import os
|
2015-11-22 17:39:51 +00:00
|
|
|
import re
|
|
|
|
import sys
|
|
|
|
|
2016-09-12 08:15:21 +00:00
|
|
|
import bs4
|
2015-11-22 17:39:51 +00:00
|
|
|
from oslo_config import cfg
|
|
|
|
import pkg_resources
|
2016-09-12 08:15:21 +00:00
|
|
|
import prettytable
|
2015-11-22 17:39:51 +00:00
|
|
|
import requests
|
|
|
|
|
2016-09-02 07:56:56 +00:00
|
|
|
# NOTE(SamYaple): Update the search path to prefer PROJECT_ROOT as the source
|
|
|
|
# of packages to import if we are using local tools instead of
|
|
|
|
# pip installed kolla tools
|
2016-03-20 23:28:26 +00:00
|
|
|
PROJECT_ROOT = os.path.abspath(os.path.join(
|
|
|
|
os.path.dirname(os.path.realpath(__file__)), '..'))
|
|
|
|
if PROJECT_ROOT not in sys.path:
|
|
|
|
sys.path.insert(0, PROJECT_ROOT)
|
|
|
|
|
2015-11-22 17:39:51 +00:00
|
|
|
from kolla.common import config as common_config
|
|
|
|
|
2016-08-08 21:00:31 +00:00
|
|
|
logging.basicConfig(format="%(message)s")
|
|
|
|
LOG = logging.getLogger('version-check')
|
|
|
|
|
2016-03-21 00:31:30 +00:00
|
|
|
# Filter list for non-projects
|
|
|
|
NOT_PROJECTS = [
|
|
|
|
'nova-novncproxy',
|
|
|
|
'nova-spicehtml5proxy',
|
|
|
|
'openstack-base',
|
|
|
|
'profiles'
|
|
|
|
]
|
2015-11-22 17:39:51 +00:00
|
|
|
TARBALLS_BASE_URL = 'http://tarballs.openstack.org'
|
2016-03-21 00:31:30 +00:00
|
|
|
VERSIONS = {'local': dict()}
|
2015-11-22 17:39:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
def retrieve_upstream_versions():
|
|
|
|
upstream_versions = dict()
|
2016-03-21 00:31:30 +00:00
|
|
|
for project in VERSIONS['local']:
|
2015-11-22 17:39:51 +00:00
|
|
|
winner = None
|
2016-03-21 00:31:30 +00:00
|
|
|
series = VERSIONS['local'][project].split('.')[0]
|
2015-11-22 17:39:51 +00:00
|
|
|
base = '{}/{}'.format(TARBALLS_BASE_URL, project)
|
2016-08-08 21:00:31 +00:00
|
|
|
LOG.debug("Getting latest version for project %s from %s",
|
|
|
|
project, base)
|
2015-11-22 17:39:51 +00:00
|
|
|
r = requests.get(base)
|
2016-09-12 08:15:21 +00:00
|
|
|
s = bs4.BeautifulSoup(r.text, 'html.parser')
|
2015-11-22 17:39:51 +00:00
|
|
|
|
|
|
|
for link in s.find_all('a'):
|
|
|
|
version = link.get('href')
|
|
|
|
if (version.endswith('.tar.gz') and
|
|
|
|
version.startswith('{}-{}'.format(project, series))):
|
|
|
|
split = '{}-|.tar.gz'.format(project)
|
|
|
|
candidate = re.split(split, version)[1]
|
2016-03-21 00:31:30 +00:00
|
|
|
# Ignore 2014, 2015 versions as they are older
|
|
|
|
if candidate.startswith('201'):
|
|
|
|
continue
|
2015-11-22 17:39:51 +00:00
|
|
|
if not winner or more_recent(candidate, winner):
|
|
|
|
winner = candidate
|
|
|
|
|
|
|
|
if not winner:
|
2016-08-08 21:00:31 +00:00
|
|
|
LOG.warning("Could not find a version for %s", project)
|
2015-11-22 17:39:51 +00:00
|
|
|
continue
|
|
|
|
|
2016-03-21 00:31:30 +00:00
|
|
|
if '-' in winner:
|
|
|
|
winner = winner.split('-')[1]
|
2015-11-22 17:39:51 +00:00
|
|
|
upstream_versions[project] = winner
|
2016-08-08 21:00:31 +00:00
|
|
|
LOG.debug("Found latest version %s for project %s", winner, project)
|
2015-11-22 17:39:51 +00:00
|
|
|
|
2016-03-21 00:31:30 +00:00
|
|
|
VERSIONS['upstream'] = collections.OrderedDict(
|
|
|
|
sorted(upstream_versions.items()))
|
2015-11-22 17:39:51 +00:00
|
|
|
|
|
|
|
|
2016-08-08 21:00:31 +00:00
|
|
|
def retrieve_local_versions(conf):
|
2016-03-21 00:31:30 +00:00
|
|
|
for section in common_config.SOURCES:
|
2016-08-08 21:00:31 +00:00
|
|
|
if section in NOT_PROJECTS:
|
|
|
|
continue
|
|
|
|
|
|
|
|
project = section.split('-')[0]
|
|
|
|
|
|
|
|
if section not in conf.list_all_sections():
|
|
|
|
LOG.debug("Project %s not found in configuration file, using "
|
|
|
|
"default from kolla.common.config", project)
|
|
|
|
raw_version = common_config.SOURCES[section]['location']
|
|
|
|
else:
|
|
|
|
raw_version = getattr(conf, section).location
|
|
|
|
|
|
|
|
version = raw_version.split('/')[-1].split('.tar.gz')[0]
|
|
|
|
if '-' in version:
|
|
|
|
version = version.split('-')[1]
|
|
|
|
|
|
|
|
LOG.debug("Use local version %s for project %s", version, project)
|
|
|
|
VERSIONS['local'][project] = version
|
2015-11-22 17:39:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
def more_recent(candidate, reference):
|
|
|
|
return pkg_resources.parse_version(candidate) > \
|
|
|
|
pkg_resources.parse_version(reference)
|
|
|
|
|
|
|
|
|
|
|
|
def diff_link(project, old_ref, new_ref):
|
|
|
|
return "https://github.com/openstack/{}/compare/{}...{}".format(
|
|
|
|
project, old_ref, new_ref)
|
|
|
|
|
|
|
|
|
|
|
|
def compare_versions():
|
|
|
|
up_to_date = True
|
2016-09-12 08:15:21 +00:00
|
|
|
result = prettytable.PrettyTable(["Project", "Current version",
|
|
|
|
"Latest version", "Comparing changes"])
|
2016-08-08 21:00:31 +00:00
|
|
|
result.align = "l"
|
|
|
|
|
2015-11-22 17:39:51 +00:00
|
|
|
for project in VERSIONS['upstream']:
|
2016-08-08 21:00:31 +00:00
|
|
|
if project not in VERSIONS['local']:
|
|
|
|
continue
|
|
|
|
|
|
|
|
upstream_version = VERSIONS['upstream'][project]
|
|
|
|
local_version = VERSIONS['local'][project]
|
|
|
|
|
|
|
|
if more_recent(upstream_version, local_version):
|
|
|
|
result.add_row([
|
|
|
|
project,
|
|
|
|
VERSIONS['local'][project],
|
|
|
|
VERSIONS['upstream'][project],
|
|
|
|
diff_link(project, local_version, upstream_version)
|
|
|
|
])
|
|
|
|
up_to_date = False
|
|
|
|
|
2015-11-22 17:39:51 +00:00
|
|
|
if up_to_date:
|
2016-08-08 21:00:31 +00:00
|
|
|
result = "Everything is up to date"
|
|
|
|
|
|
|
|
print(result)
|
2015-11-22 17:39:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
conf = cfg.ConfigOpts()
|
2016-08-08 21:00:31 +00:00
|
|
|
common_config.parse(conf, sys.argv[1:], prog='version-check')
|
|
|
|
|
|
|
|
if conf.debug:
|
|
|
|
LOG.setLevel(logging.DEBUG)
|
2015-11-22 17:39:51 +00:00
|
|
|
|
2016-08-08 21:00:31 +00:00
|
|
|
retrieve_local_versions(conf)
|
2015-11-22 17:39:51 +00:00
|
|
|
retrieve_upstream_versions()
|
|
|
|
|
|
|
|
compare_versions()
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|