Move update.py into the python package.
A symlink is kept to update.py to keep this working for infra until we update the propose scripts to try the new location and fall back. Until that is done and the symlink removed, update.py cannot refer to other modules (though they can refer to it). Change-Id: If878bc2bb426c0e5478ed031f1144e26f05e85a2
This commit is contained in:
parent
b1e05bef3a
commit
21fa543926
@ -20,7 +20,7 @@ import fixtures
|
||||
import testtools
|
||||
|
||||
from openstack_requirements.tests import common
|
||||
import update
|
||||
from openstack_requirements import update
|
||||
|
||||
|
||||
class UpdateTest(testtools.TestCase):
|
||||
|
@ -21,7 +21,7 @@ from __future__ import print_function
|
||||
import testtools
|
||||
|
||||
from openstack_requirements.tests import common
|
||||
import update
|
||||
from openstack_requirements import update
|
||||
|
||||
|
||||
class UpdateTestPbr(testtools.TestCase):
|
||||
|
@ -17,7 +17,7 @@ from __future__ import print_function
|
||||
import testtools
|
||||
|
||||
from openstack_requirements.tests import common
|
||||
import update
|
||||
from openstack_requirements import update
|
||||
|
||||
|
||||
class UpdateTestWithSuffix(testtools.TestCase):
|
||||
|
271
openstack_requirements/update.py
Normal file
271
openstack_requirements/update.py
Normal file
@ -0,0 +1,271 @@
|
||||
# Copyright 2012 OpenStack Foundation
|
||||
# Copyright 2013 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# 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.
|
||||
|
||||
"""
|
||||
A simple script to update the requirements files from a global set of
|
||||
allowable requirements.
|
||||
|
||||
The script can be called like this:
|
||||
|
||||
$> python update.py ../myproj
|
||||
|
||||
Any requirements listed in the target files will have their versions
|
||||
updated to match the global requirements. Requirements not in the global
|
||||
files will be dropped.
|
||||
"""
|
||||
|
||||
import optparse
|
||||
import os
|
||||
import os.path
|
||||
import sys
|
||||
|
||||
import pkg_resources
|
||||
|
||||
VERBOSE = None
|
||||
|
||||
_setup_py_text = """#!/usr/bin/env python
|
||||
# Copyright (c) 2013 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# 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.
|
||||
|
||||
# THIS FILE IS MANAGED BY THE GLOBAL REQUIREMENTS REPO - DO NOT EDIT
|
||||
import setuptools
|
||||
|
||||
# In python < 2.7.4, a lazy loading of package `pbr` will break
|
||||
# setuptools if some other modules registered functions in `atexit`.
|
||||
# solution from: http://bugs.python.org/issue15881#msg170215
|
||||
try:
|
||||
import multiprocessing # noqa
|
||||
except ImportError:
|
||||
pass
|
||||
|
||||
setuptools.setup(
|
||||
setup_requires=['pbr'],
|
||||
pbr=True)
|
||||
"""
|
||||
|
||||
# A header for the requirements file(s).
|
||||
# TODO(lifeless): Remove this once constraints are in use.
|
||||
_REQS_HEADER = [
|
||||
'# The order of packages is significant, because pip processes '
|
||||
'them in the order\n',
|
||||
'# of appearance. Changing the order has an impact on the overall '
|
||||
'integration\n',
|
||||
'# process, which may cause wedges in the gate later.\n',
|
||||
]
|
||||
|
||||
|
||||
def verbose(msg, stdout):
|
||||
if VERBOSE:
|
||||
stdout.write(msg + "\n")
|
||||
|
||||
|
||||
class Change(object):
|
||||
def __init__(self, name, old, new):
|
||||
self.name = name
|
||||
self.old = old
|
||||
self.new = new
|
||||
|
||||
def __repr__(self):
|
||||
return "%-30.30s -> %s" % (self.old, self.new)
|
||||
|
||||
|
||||
def _package_name(pip_line):
|
||||
"""Return normalized (lower case) package name.
|
||||
|
||||
This is needed for comparing old and new dictionaries of
|
||||
requirements to ensure they match.
|
||||
"""
|
||||
name = pkg_resources.Requirement.parse(pip_line).project_name
|
||||
return name.lower()
|
||||
|
||||
|
||||
def _pass_through(pip):
|
||||
return (not pip or
|
||||
pip.startswith('#') or
|
||||
pip.startswith('http://tarballs.openstack.org/') or
|
||||
pip.startswith('-e') or
|
||||
pip.startswith('-f'))
|
||||
|
||||
|
||||
def _functionally_equal(old_requirement, new_requirement):
|
||||
return old_requirement == new_requirement
|
||||
|
||||
|
||||
def _read(filename):
|
||||
with open(filename, 'r') as f:
|
||||
return f.read()
|
||||
|
||||
|
||||
def _readlines(filename):
|
||||
with open(filename, 'r') as f:
|
||||
return f.readlines()
|
||||
|
||||
|
||||
def _parse_reqs(filename):
|
||||
reqs = dict()
|
||||
pip_requires = _readlines(filename)
|
||||
for pip in pip_requires:
|
||||
pip = pip.strip()
|
||||
if _pass_through(pip):
|
||||
continue
|
||||
reqs[_package_name(pip)] = pip
|
||||
return reqs
|
||||
|
||||
|
||||
def _sync_requirements_file(
|
||||
source_reqs, dest_path, suffix, softupdate, hacking, stdout):
|
||||
dest_reqs = _readlines(dest_path)
|
||||
changes = []
|
||||
# this is specifically for global-requirements gate jobs so we don't
|
||||
# modify the git tree
|
||||
if suffix:
|
||||
dest_path = "%s.%s" % (dest_path, suffix)
|
||||
|
||||
verbose("Syncing %s" % dest_path, stdout)
|
||||
|
||||
with open(dest_path, 'w') as new_reqs:
|
||||
# Check the instructions header
|
||||
if dest_reqs[:len(_REQS_HEADER)] != _REQS_HEADER:
|
||||
new_reqs.writelines(_REQS_HEADER)
|
||||
|
||||
for old_line in dest_reqs:
|
||||
old_require = old_line.strip()
|
||||
|
||||
if _pass_through(old_require):
|
||||
new_reqs.write(old_line)
|
||||
continue
|
||||
|
||||
old_pip = _package_name(old_require)
|
||||
|
||||
# Special cases:
|
||||
# projects need to align hacking version on their own time
|
||||
if "hacking" in old_pip and not hacking:
|
||||
new_reqs.write(old_line)
|
||||
continue
|
||||
|
||||
if old_pip in source_reqs:
|
||||
if _functionally_equal(old_require, source_reqs[old_pip]):
|
||||
new_reqs.write(old_line)
|
||||
else:
|
||||
changes.append(
|
||||
Change(old_pip, old_require, source_reqs[old_pip]))
|
||||
new_reqs.write("%s\n" % source_reqs[old_pip])
|
||||
elif softupdate:
|
||||
# under softupdate we pass through anything we don't
|
||||
# understand, this is intended for ecosystem projects
|
||||
# that want to stay in sync with existing
|
||||
# requirements, but also add their own above and
|
||||
# beyond
|
||||
new_reqs.write(old_line)
|
||||
else:
|
||||
# What do we do if we find something unexpected?
|
||||
#
|
||||
# In the default cause we should die horribly, because
|
||||
# the point of global requirements was a single lever
|
||||
# to control all the pip installs in the gate.
|
||||
#
|
||||
# However, we do have other projects using
|
||||
# devstack jobs that might have legitimate reasons to
|
||||
# override. For those we support NON_STANDARD_REQS=1
|
||||
# environment variable to turn this into a warning only.
|
||||
stdout.write(
|
||||
"'%s' is not in global-requirements.txt\n" % old_pip)
|
||||
if os.getenv('NON_STANDARD_REQS', '0') != '1':
|
||||
raise Exception("nonstandard requirement present.")
|
||||
# always print out what we did if we did a thing
|
||||
if changes:
|
||||
stdout.write(
|
||||
"Version change for: %s\n" % ", ".join([x.name for x in changes]))
|
||||
stdout.write("Updated %s:\n" % dest_path)
|
||||
for change in changes:
|
||||
stdout.write(" %s\n" % change)
|
||||
|
||||
|
||||
def _copy_requires(suffix, softupdate, hacking, dest_dir, stdout, source="."):
|
||||
"""Copy requirements files."""
|
||||
source_reqs = _parse_reqs(os.path.join(source, 'global-requirements.txt'))
|
||||
target_files = [
|
||||
'requirements.txt', 'tools/pip-requires',
|
||||
'test-requirements.txt', 'tools/test-requires',
|
||||
]
|
||||
for py_version in (2, 3):
|
||||
target_files.append('requirements-py%s.txt' % py_version)
|
||||
target_files.append('test-requirements-py%s.txt' % py_version)
|
||||
|
||||
for dest in target_files:
|
||||
dest_path = os.path.join(dest_dir, dest)
|
||||
if os.path.exists(dest_path):
|
||||
_sync_requirements_file(
|
||||
source_reqs, dest_path, suffix, softupdate, hacking, stdout)
|
||||
|
||||
|
||||
def _write_setup_py(dest_path, stdout):
|
||||
target_setup_py = os.path.join(dest_path, 'setup.py')
|
||||
setup_cfg = os.path.join(dest_path, 'setup.cfg')
|
||||
# If it doesn't have a setup.py, then we don't want to update it
|
||||
if not os.path.exists(target_setup_py):
|
||||
return
|
||||
has_pbr = 'pbr' in _read(target_setup_py)
|
||||
if has_pbr:
|
||||
if 'name = pbr' not in _read(setup_cfg):
|
||||
verbose("Syncing setup.py", stdout)
|
||||
# We only want to sync things that are up to date
|
||||
# with pbr mechanics
|
||||
with open(target_setup_py, 'w') as setup_file:
|
||||
setup_file.write(_setup_py_text)
|
||||
|
||||
|
||||
def main(argv=None, stdout=None):
|
||||
parser = optparse.OptionParser()
|
||||
parser.add_option("-o", "--output-suffix", dest="suffix", default="",
|
||||
help="output suffix for updated files (i.e. .global)")
|
||||
parser.add_option("-s", "--soft-update", dest="softupdate",
|
||||
action="store_true",
|
||||
help="Pass through extra requirements without warning.")
|
||||
parser.add_option("-H", "--hacking", dest="hacking",
|
||||
action="store_true",
|
||||
help="Include the hacking project.")
|
||||
parser.add_option("-v", "--verbose", dest="verbose",
|
||||
action="store_true",
|
||||
help="Add further verbosity to output")
|
||||
parser.add_option("--source", dest="source", default=".",
|
||||
help="Dir where global-requirements.txt is located.")
|
||||
options, args = parser.parse_args(argv)
|
||||
if len(args) != 1:
|
||||
print("Must specify directory to update")
|
||||
raise Exception("Must specify one and only one directory to update.")
|
||||
global VERBOSE
|
||||
VERBOSE = options.verbose
|
||||
if stdout is None:
|
||||
stdout = sys.stdout
|
||||
_copy_requires(options.suffix, options.softupdate, options.hacking,
|
||||
args[0], stdout=stdout, source=options.source)
|
||||
_write_setup_py(args[0], stdout=stdout)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
@ -20,3 +20,7 @@ classifier =
|
||||
[global]
|
||||
setup-hooks =
|
||||
pbr.hooks.setup_hook
|
||||
|
||||
[entry_points]
|
||||
console_scripts =
|
||||
update-requirements = openstack_requirements.update:main
|
||||
|
271
update.py
271
update.py
@ -1,271 +0,0 @@
|
||||
# Copyright 2012 OpenStack Foundation
|
||||
# Copyright 2013 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# 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.
|
||||
|
||||
"""
|
||||
A simple script to update the requirements files from a global set of
|
||||
allowable requirements.
|
||||
|
||||
The script can be called like this:
|
||||
|
||||
$> python update.py ../myproj
|
||||
|
||||
Any requirements listed in the target files will have their versions
|
||||
updated to match the global requirements. Requirements not in the global
|
||||
files will be dropped.
|
||||
"""
|
||||
|
||||
import optparse
|
||||
import os
|
||||
import os.path
|
||||
import sys
|
||||
|
||||
import pkg_resources
|
||||
|
||||
VERBOSE = None
|
||||
|
||||
_setup_py_text = """#!/usr/bin/env python
|
||||
# Copyright (c) 2013 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# 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.
|
||||
|
||||
# THIS FILE IS MANAGED BY THE GLOBAL REQUIREMENTS REPO - DO NOT EDIT
|
||||
import setuptools
|
||||
|
||||
# In python < 2.7.4, a lazy loading of package `pbr` will break
|
||||
# setuptools if some other modules registered functions in `atexit`.
|
||||
# solution from: http://bugs.python.org/issue15881#msg170215
|
||||
try:
|
||||
import multiprocessing # noqa
|
||||
except ImportError:
|
||||
pass
|
||||
|
||||
setuptools.setup(
|
||||
setup_requires=['pbr'],
|
||||
pbr=True)
|
||||
"""
|
||||
|
||||
# A header for the requirements file(s).
|
||||
# TODO(lifeless): Remove this once constraints are in use.
|
||||
_REQS_HEADER = [
|
||||
'# The order of packages is significant, because pip processes '
|
||||
'them in the order\n',
|
||||
'# of appearance. Changing the order has an impact on the overall '
|
||||
'integration\n',
|
||||
'# process, which may cause wedges in the gate later.\n',
|
||||
]
|
||||
|
||||
|
||||
def verbose(msg, stdout):
|
||||
if VERBOSE:
|
||||
stdout.write(msg + "\n")
|
||||
|
||||
|
||||
class Change(object):
|
||||
def __init__(self, name, old, new):
|
||||
self.name = name
|
||||
self.old = old
|
||||
self.new = new
|
||||
|
||||
def __repr__(self):
|
||||
return "%-30.30s -> %s" % (self.old, self.new)
|
||||
|
||||
|
||||
def _package_name(pip_line):
|
||||
"""Return normalized (lower case) package name.
|
||||
|
||||
This is needed for comparing old and new dictionaries of
|
||||
requirements to ensure they match.
|
||||
"""
|
||||
name = pkg_resources.Requirement.parse(pip_line).project_name
|
||||
return name.lower()
|
||||
|
||||
|
||||
def _pass_through(pip):
|
||||
return (not pip or
|
||||
pip.startswith('#') or
|
||||
pip.startswith('http://tarballs.openstack.org/') or
|
||||
pip.startswith('-e') or
|
||||
pip.startswith('-f'))
|
||||
|
||||
|
||||
def _functionally_equal(old_requirement, new_requirement):
|
||||
return old_requirement == new_requirement
|
||||
|
||||
|
||||
def _read(filename):
|
||||
with open(filename, 'r') as f:
|
||||
return f.read()
|
||||
|
||||
|
||||
def _readlines(filename):
|
||||
with open(filename, 'r') as f:
|
||||
return f.readlines()
|
||||
|
||||
|
||||
def _parse_reqs(filename):
|
||||
reqs = dict()
|
||||
pip_requires = _readlines(filename)
|
||||
for pip in pip_requires:
|
||||
pip = pip.strip()
|
||||
if _pass_through(pip):
|
||||
continue
|
||||
reqs[_package_name(pip)] = pip
|
||||
return reqs
|
||||
|
||||
|
||||
def _sync_requirements_file(
|
||||
source_reqs, dest_path, suffix, softupdate, hacking, stdout):
|
||||
dest_reqs = _readlines(dest_path)
|
||||
changes = []
|
||||
# this is specifically for global-requirements gate jobs so we don't
|
||||
# modify the git tree
|
||||
if suffix:
|
||||
dest_path = "%s.%s" % (dest_path, suffix)
|
||||
|
||||
verbose("Syncing %s" % dest_path, stdout)
|
||||
|
||||
with open(dest_path, 'w') as new_reqs:
|
||||
# Check the instructions header
|
||||
if dest_reqs[:len(_REQS_HEADER)] != _REQS_HEADER:
|
||||
new_reqs.writelines(_REQS_HEADER)
|
||||
|
||||
for old_line in dest_reqs:
|
||||
old_require = old_line.strip()
|
||||
|
||||
if _pass_through(old_require):
|
||||
new_reqs.write(old_line)
|
||||
continue
|
||||
|
||||
old_pip = _package_name(old_require)
|
||||
|
||||
# Special cases:
|
||||
# projects need to align hacking version on their own time
|
||||
if "hacking" in old_pip and not hacking:
|
||||
new_reqs.write(old_line)
|
||||
continue
|
||||
|
||||
if old_pip in source_reqs:
|
||||
if _functionally_equal(old_require, source_reqs[old_pip]):
|
||||
new_reqs.write(old_line)
|
||||
else:
|
||||
changes.append(
|
||||
Change(old_pip, old_require, source_reqs[old_pip]))
|
||||
new_reqs.write("%s\n" % source_reqs[old_pip])
|
||||
elif softupdate:
|
||||
# under softupdate we pass through anything we don't
|
||||
# understand, this is intended for ecosystem projects
|
||||
# that want to stay in sync with existing
|
||||
# requirements, but also add their own above and
|
||||
# beyond
|
||||
new_reqs.write(old_line)
|
||||
else:
|
||||
# What do we do if we find something unexpected?
|
||||
#
|
||||
# In the default cause we should die horribly, because
|
||||
# the point of global requirements was a single lever
|
||||
# to control all the pip installs in the gate.
|
||||
#
|
||||
# However, we do have other projects using
|
||||
# devstack jobs that might have legitimate reasons to
|
||||
# override. For those we support NON_STANDARD_REQS=1
|
||||
# environment variable to turn this into a warning only.
|
||||
stdout.write(
|
||||
"'%s' is not in global-requirements.txt\n" % old_pip)
|
||||
if os.getenv('NON_STANDARD_REQS', '0') != '1':
|
||||
raise Exception("nonstandard requirement present.")
|
||||
# always print out what we did if we did a thing
|
||||
if changes:
|
||||
stdout.write(
|
||||
"Version change for: %s\n" % ", ".join([x.name for x in changes]))
|
||||
stdout.write("Updated %s:\n" % dest_path)
|
||||
for change in changes:
|
||||
stdout.write(" %s\n" % change)
|
||||
|
||||
|
||||
def _copy_requires(suffix, softupdate, hacking, dest_dir, stdout, source="."):
|
||||
"""Copy requirements files."""
|
||||
source_reqs = _parse_reqs(os.path.join(source, 'global-requirements.txt'))
|
||||
target_files = [
|
||||
'requirements.txt', 'tools/pip-requires',
|
||||
'test-requirements.txt', 'tools/test-requires',
|
||||
]
|
||||
for py_version in (2, 3):
|
||||
target_files.append('requirements-py%s.txt' % py_version)
|
||||
target_files.append('test-requirements-py%s.txt' % py_version)
|
||||
|
||||
for dest in target_files:
|
||||
dest_path = os.path.join(dest_dir, dest)
|
||||
if os.path.exists(dest_path):
|
||||
_sync_requirements_file(
|
||||
source_reqs, dest_path, suffix, softupdate, hacking, stdout)
|
||||
|
||||
|
||||
def _write_setup_py(dest_path, stdout):
|
||||
target_setup_py = os.path.join(dest_path, 'setup.py')
|
||||
setup_cfg = os.path.join(dest_path, 'setup.cfg')
|
||||
# If it doesn't have a setup.py, then we don't want to update it
|
||||
if not os.path.exists(target_setup_py):
|
||||
return
|
||||
has_pbr = 'pbr' in _read(target_setup_py)
|
||||
if has_pbr:
|
||||
if 'name = pbr' not in _read(setup_cfg):
|
||||
verbose("Syncing setup.py", stdout)
|
||||
# We only want to sync things that are up to date
|
||||
# with pbr mechanics
|
||||
with open(target_setup_py, 'w') as setup_file:
|
||||
setup_file.write(_setup_py_text)
|
||||
|
||||
|
||||
def main(argv=None, stdout=None):
|
||||
parser = optparse.OptionParser()
|
||||
parser.add_option("-o", "--output-suffix", dest="suffix", default="",
|
||||
help="output suffix for updated files (i.e. .global)")
|
||||
parser.add_option("-s", "--soft-update", dest="softupdate",
|
||||
action="store_true",
|
||||
help="Pass through extra requirements without warning.")
|
||||
parser.add_option("-H", "--hacking", dest="hacking",
|
||||
action="store_true",
|
||||
help="Include the hacking project.")
|
||||
parser.add_option("-v", "--verbose", dest="verbose",
|
||||
action="store_true",
|
||||
help="Add further verbosity to output")
|
||||
parser.add_option("--source", dest="source", default=".",
|
||||
help="Dir where global-requirements.txt is located.")
|
||||
options, args = parser.parse_args(argv)
|
||||
if len(args) != 1:
|
||||
print("Must specify directory to update")
|
||||
raise Exception("Must specify one and only one directory to update.")
|
||||
global VERBOSE
|
||||
VERBOSE = options.verbose
|
||||
if stdout is None:
|
||||
stdout = sys.stdout
|
||||
_copy_requires(options.suffix, options.softupdate, options.hacking,
|
||||
args[0], stdout=stdout, source=options.source)
|
||||
_write_setup_py(args[0], stdout=stdout)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
Loading…
x
Reference in New Issue
Block a user