Commit ceb03af8 authored by Jason R. Coombs's avatar Jason R. Coombs

Backed out changeset: 98a9f9dcce0e; Fixes #335.

--HG--
branch : distribute
extra : rebase_source : 3f4ff1c880688e6dd72d2fa8fab3c07e7f486a7e
parent f31fb0ff
...@@ -517,7 +517,7 @@ class WorkingSet(object): ...@@ -517,7 +517,7 @@ class WorkingSet(object):
seen[key]=1 seen[key]=1
yield self.by_key[key] yield self.by_key[key]
def add(self, dist, entry=None, insert=True, replace=False): def add(self, dist, entry=None, insert=True):
"""Add `dist` to working set, associated with `entry` """Add `dist` to working set, associated with `entry`
If `entry` is unspecified, it defaults to the ``.location`` of `dist`. If `entry` is unspecified, it defaults to the ``.location`` of `dist`.
...@@ -525,9 +525,8 @@ class WorkingSet(object): ...@@ -525,9 +525,8 @@ class WorkingSet(object):
set's ``.entries`` (if it wasn't already present). set's ``.entries`` (if it wasn't already present).
`dist` is only added to the working set if it's for a project that `dist` is only added to the working set if it's for a project that
doesn't already have a distribution in the set, unless `replace=True`. doesn't already have a distribution in the set. If it's added, any
If it's added, any callbacks registered with the ``subscribe()`` method callbacks registered with the ``subscribe()`` method will be called.
will be called.
""" """
if insert: if insert:
dist.insert_on(self.entries, entry) dist.insert_on(self.entries, entry)
...@@ -536,7 +535,7 @@ class WorkingSet(object): ...@@ -536,7 +535,7 @@ class WorkingSet(object):
entry = dist.location entry = dist.location
keys = self.entry_keys.setdefault(entry,[]) keys = self.entry_keys.setdefault(entry,[])
keys2 = self.entry_keys.setdefault(dist.location,[]) keys2 = self.entry_keys.setdefault(dist.location,[])
if not replace and dist.key in self.by_key: if dist.key in self.by_key:
return # ignore hidden distros return # ignore hidden distros
self.by_key[dist.key] = dist self.by_key[dist.key] = dist
...@@ -546,8 +545,7 @@ class WorkingSet(object): ...@@ -546,8 +545,7 @@ class WorkingSet(object):
keys2.append(dist.key) keys2.append(dist.key)
self._added_new(dist) self._added_new(dist)
def resolve(self, requirements, env=None, installer=None, def resolve(self, requirements, env=None, installer=None, replacement=True):
replacement=True, replace_conflicting=False):
"""List all distributions needed to (recursively) meet `requirements` """List all distributions needed to (recursively) meet `requirements`
`requirements` must be a sequence of ``Requirement`` objects. `env`, `requirements` must be a sequence of ``Requirement`` objects. `env`,
...@@ -557,12 +555,6 @@ class WorkingSet(object): ...@@ -557,12 +555,6 @@ class WorkingSet(object):
will be invoked with each requirement that cannot be met by an will be invoked with each requirement that cannot be met by an
already-installed distribution; it should return a ``Distribution`` or already-installed distribution; it should return a ``Distribution`` or
``None``. ``None``.
Unless `replace_conflicting=True`, raises a VersionConflict exception if
any requirements are found on the path that have the correct name but
the wrong version. Otherwise, if an `installer` is supplied it will be
invoked to obtain the correct version of the requirement and activate
it.
""" """
requirements = list(requirements)[::-1] # set up the stack requirements = list(requirements)[::-1] # set up the stack
...@@ -582,18 +574,10 @@ class WorkingSet(object): ...@@ -582,18 +574,10 @@ class WorkingSet(object):
if dist is None: if dist is None:
# Find the best distribution and add it to the map # Find the best distribution and add it to the map
dist = self.by_key.get(req.key) dist = self.by_key.get(req.key)
if dist is None or (dist not in req and replace_conflicting): if dist is None:
ws = self
if env is None: if env is None:
if dist is None: env = Environment(self.entries)
env = Environment(self.entries) dist = best[req.key] = env.best_match(req, self, installer)
else:
# Use an empty environment and workingset to avoid
# any further conflicts with the conflicting
# distribution
env = Environment([])
ws = WorkingSet([])
dist = best[req.key] = env.best_match(req, ws, installer)
if dist is None: if dist is None:
#msg = ("The '%s' distribution was not found on this " #msg = ("The '%s' distribution was not found on this "
# "system, and is required by this application.") # "system, and is required by this application.")
...@@ -1814,7 +1798,6 @@ def register_namespace_handler(importer_type, namespace_handler): ...@@ -1814,7 +1798,6 @@ def register_namespace_handler(importer_type, namespace_handler):
def _handle_ns(packageName, path_item): def _handle_ns(packageName, path_item):
"""Ensure that named package includes a subpath of path_item (if needed)""" """Ensure that named package includes a subpath of path_item (if needed)"""
importer = get_importer(path_item) importer = get_importer(path_item)
if importer is None: if importer is None:
return None return None
...@@ -1824,19 +1807,14 @@ def _handle_ns(packageName, path_item): ...@@ -1824,19 +1807,14 @@ def _handle_ns(packageName, path_item):
module = sys.modules.get(packageName) module = sys.modules.get(packageName)
if module is None: if module is None:
module = sys.modules[packageName] = types.ModuleType(packageName) module = sys.modules[packageName] = types.ModuleType(packageName)
module.__path__ = [] module.__path__ = []; _set_parent_ns(packageName)
_set_parent_ns(packageName)
elif not hasattr(module,'__path__'): elif not hasattr(module,'__path__'):
raise TypeError("Not a package:", packageName) raise TypeError("Not a package:", packageName)
handler = _find_adapter(_namespace_handlers, importer) handler = _find_adapter(_namespace_handlers, importer)
subpath = handler(importer, path_item, packageName, module) subpath = handler(importer,path_item,packageName,module)
if subpath is not None: if subpath is not None:
path = module.__path__ path = module.__path__; path.append(subpath)
path.append(subpath) loader.load_module(packageName); module.__path__ = path
loader.load_module(packageName)
for path_item in path:
if path_item not in module.__path__:
module.__path__.append(path_item)
return subpath return subpath
def declare_namespace(packageName): def declare_namespace(packageName):
...@@ -2142,7 +2120,7 @@ def _remove_md5_fragment(location): ...@@ -2142,7 +2120,7 @@ def _remove_md5_fragment(location):
class Distribution(object): class Distribution(object):
"""Wrap an actual or potential sys.path entry w/metadata""" """Wrap an actual or potential sys.path entry w/metadata"""
PKG_INFO = 'PKG-INFO' PKG_INFO = 'PKG-INFO'
def __init__(self, def __init__(self,
location=None, metadata=None, project_name=None, version=None, location=None, metadata=None, project_name=None, version=None,
py_version=PY_MAJOR, platform=None, precedence = EGG_DIST py_version=PY_MAJOR, platform=None, precedence = EGG_DIST
...@@ -2481,7 +2459,7 @@ class DistInfoDistribution(Distribution): ...@@ -2481,7 +2459,7 @@ class DistInfoDistribution(Distribution):
from email.parser import Parser from email.parser import Parser
self._pkg_info = Parser().parsestr(self.get_metadata(self.PKG_INFO)) self._pkg_info = Parser().parsestr(self.get_metadata(self.PKG_INFO))
return self._pkg_info return self._pkg_info
@property @property
def _dep_map(self): def _dep_map(self):
try: try:
...@@ -2492,7 +2470,7 @@ class DistInfoDistribution(Distribution): ...@@ -2492,7 +2470,7 @@ class DistInfoDistribution(Distribution):
def _preparse_requirement(self, requires_dist): def _preparse_requirement(self, requires_dist):
"""Convert 'Foobar (1); baz' to ('Foobar ==1', 'baz') """Convert 'Foobar (1); baz' to ('Foobar ==1', 'baz')
Split environment marker, add == prefix to version specifiers as Split environment marker, add == prefix to version specifiers as
necessary, and remove parenthesis. necessary, and remove parenthesis.
""" """
parts = requires_dist.split(';', 1) + [''] parts = requires_dist.split(';', 1) + ['']
...@@ -2501,7 +2479,7 @@ class DistInfoDistribution(Distribution): ...@@ -2501,7 +2479,7 @@ class DistInfoDistribution(Distribution):
distvers = re.sub(self.EQEQ, r"\1==\2\3", distvers) distvers = re.sub(self.EQEQ, r"\1==\2\3", distvers)
distvers = distvers.replace('(', '').replace(')', '') distvers = distvers.replace('(', '').replace(')', '')
return (distvers, mark) return (distvers, mark)
def _compute_dependencies(self): def _compute_dependencies(self):
"""Recompute this distribution's dependencies.""" """Recompute this distribution's dependencies."""
from _markerlib import compile as compile_marker from _markerlib import compile as compile_marker
...@@ -2514,7 +2492,7 @@ class DistInfoDistribution(Distribution): ...@@ -2514,7 +2492,7 @@ class DistInfoDistribution(Distribution):
parsed = parse_requirements(distvers).next() parsed = parse_requirements(distvers).next()
parsed.marker_fn = compile_marker(mark) parsed.marker_fn = compile_marker(mark)
reqs.append(parsed) reqs.append(parsed)
def reqs_for_extra(extra): def reqs_for_extra(extra):
for req in reqs: for req in reqs:
if req.marker_fn(override={'extra':extra}): if req.marker_fn(override={'extra':extra}):
...@@ -2522,13 +2500,13 @@ class DistInfoDistribution(Distribution): ...@@ -2522,13 +2500,13 @@ class DistInfoDistribution(Distribution):
common = frozenset(reqs_for_extra(None)) common = frozenset(reqs_for_extra(None))
dm[None].extend(common) dm[None].extend(common)
for extra in self._parsed_pkg_info.get_all('Provides-Extra') or []: for extra in self._parsed_pkg_info.get_all('Provides-Extra') or []:
extra = safe_extra(extra.strip()) extra = safe_extra(extra.strip())
dm[extra] = list(frozenset(reqs_for_extra(extra)) - common) dm[extra] = list(frozenset(reqs_for_extra(extra)) - common)
return dm return dm
_distributionImpl = {'.egg': Distribution, _distributionImpl = {'.egg': Distribution,
'.egg-info': Distribution, '.egg-info': Distribution,
......
...@@ -242,10 +242,9 @@ class Distribution(_Distribution): ...@@ -242,10 +242,9 @@ class Distribution(_Distribution):
"""Resolve pre-setup requirements""" """Resolve pre-setup requirements"""
from pkg_resources import working_set, parse_requirements from pkg_resources import working_set, parse_requirements
for dist in working_set.resolve( for dist in working_set.resolve(
parse_requirements(requires), installer=self.fetch_build_egg, parse_requirements(requires), installer=self.fetch_build_egg
replace_conflicting=True
): ):
working_set.add(dist, replace=True) working_set.add(dist)
def finalize_options(self): def finalize_options(self):
_Distribution.finalize_options(self) _Distribution.finalize_options(self)
......
...@@ -17,10 +17,8 @@ from setuptools.command.easy_install import easy_install, get_script_args, main ...@@ -17,10 +17,8 @@ from setuptools.command.easy_install import easy_install, get_script_args, main
from setuptools.command.easy_install import PthDistributions from setuptools.command.easy_install import PthDistributions
from setuptools.command import easy_install as easy_install_pkg from setuptools.command import easy_install as easy_install_pkg
from setuptools.dist import Distribution from setuptools.dist import Distribution
from pkg_resources import working_set, VersionConflict
from pkg_resources import Distribution as PRDistribution from pkg_resources import Distribution as PRDistribution
import setuptools.tests.server import setuptools.tests.server
import pkg_resources
try: try:
# import multiprocessing solely for the purpose of testing its existence # import multiprocessing solely for the purpose of testing its existence
...@@ -275,16 +273,48 @@ class TestUserInstallTest(unittest.TestCase): ...@@ -275,16 +273,48 @@ class TestUserInstallTest(unittest.TestCase):
SandboxViolation. SandboxViolation.
""" """
test_pkg = create_setup_requires_package(self.dir) test_setup_attrs = {
'name': 'test_pkg', 'version': '0.0',
'setup_requires': ['foobar'],
'dependency_links': [os.path.abspath(self.dir)]
}
test_pkg = os.path.join(self.dir, 'test_pkg')
test_setup_py = os.path.join(test_pkg, 'setup.py') test_setup_py = os.path.join(test_pkg, 'setup.py')
test_setup_cfg = os.path.join(test_pkg, 'setup.cfg')
os.mkdir(test_pkg)
f = open(test_setup_py, 'w')
f.write(textwrap.dedent("""\
import setuptools
setuptools.setup(**%r)
""" % test_setup_attrs))
f.close()
foobar_path = os.path.join(self.dir, 'foobar-0.1.tar.gz')
make_trivial_sdist(
foobar_path,
textwrap.dedent("""\
import setuptools
setuptools.setup(
name='foobar',
version='0.1'
)
"""))
old_stdout = sys.stdout
old_stderr = sys.stderr
sys.stdout = StringIO.StringIO()
sys.stderr = StringIO.StringIO()
try: try:
quiet_context( reset_setup_stop_context(
lambda: reset_setup_stop_context( lambda: run_setup(test_setup_py, ['install'])
lambda: run_setup(test_setup_py, ['install']) )
))
except SandboxViolation: except SandboxViolation:
self.fail('Installation caused SandboxViolation') self.fail('Installation caused SandboxViolation')
finally:
sys.stdout = old_stdout
sys.stderr = old_stderr
class TestSetupRequires(unittest.TestCase): class TestSetupRequires(unittest.TestCase):
...@@ -330,7 +360,7 @@ class TestSetupRequires(unittest.TestCase): ...@@ -330,7 +360,7 @@ class TestSetupRequires(unittest.TestCase):
tempdir_context(install_at) tempdir_context(install_at)
# create an sdist that has a build-time dependency. # create an sdist that has a build-time dependency.
quiet_context(lambda: self.create_sdist(install)) self.create_sdist(install)
# there should have been two or three requests to the server # there should have been two or three requests to the server
# (three happens on Python 3.3a) # (three happens on Python 3.3a)
...@@ -357,81 +387,6 @@ class TestSetupRequires(unittest.TestCase): ...@@ -357,81 +387,6 @@ class TestSetupRequires(unittest.TestCase):
installer(dist_path) installer(dist_path)
tempdir_context(build_sdist) tempdir_context(build_sdist)
def test_setup_requires_overrides_version_conflict(self):
"""
Regression test for issue #323.
Ensures that a distribution's setup_requires requirements can still be
installed and used locally even if a conflicting version of that
requirement is already on the path.
"""
pr_state = pkg_resources.__getstate__()
fake_dist = PRDistribution('does-not-matter', project_name='foobar',
version='0.0')
working_set.add(fake_dist)
def setup_and_run(temp_dir):
test_pkg = create_setup_requires_package(temp_dir)
test_setup_py = os.path.join(test_pkg, 'setup.py')
try:
stdout, stderr = quiet_context(
lambda: reset_setup_stop_context(
# Don't even need to install the package, just running
# the setup.py at all is sufficient
lambda: run_setup(test_setup_py, ['--name'])
))
except VersionConflict:
self.fail('Installing setup.py requirements caused '
'VersionConflict')
lines = stdout.splitlines()
self.assertTrue(len(lines) > 0)
self.assertTrue(lines[-1].strip(), 'test_pkg')
try:
tempdir_context(setup_and_run)
finally:
pkg_resources.__setstate__(pr_state)
def create_setup_requires_package(path):
"""Creates a source tree under path for a trivial test package that has a
single requirement in setup_requires--a tarball for that requirement is
also created and added to the dependency_links argument.
"""
test_setup_attrs = {
'name': 'test_pkg', 'version': '0.0',
'setup_requires': ['foobar==0.1'],
'dependency_links': [os.path.abspath(path)]
}
test_pkg = os.path.join(path, 'test_pkg')
test_setup_py = os.path.join(test_pkg, 'setup.py')
test_setup_cfg = os.path.join(test_pkg, 'setup.cfg')
os.mkdir(test_pkg)
f = open(test_setup_py, 'w')
f.write(textwrap.dedent("""\
import setuptools
setuptools.setup(**%r)
""" % test_setup_attrs))
f.close()
foobar_path = os.path.join(path, 'foobar-0.1.tar.gz')
make_trivial_sdist(
foobar_path,
textwrap.dedent("""\
import setuptools
setuptools.setup(
name='foobar',
version='0.1'
)
"""))
return test_pkg
def make_trivial_sdist(dist_path, setup_py): def make_trivial_sdist(dist_path, setup_py):
"""Create a simple sdist tarball at dist_path, containing just a """Create a simple sdist tarball at dist_path, containing just a
...@@ -466,7 +421,6 @@ def tempdir_context(f, cd=lambda dir:None): ...@@ -466,7 +421,6 @@ def tempdir_context(f, cd=lambda dir:None):
cd(orig_dir) cd(orig_dir)
shutil.rmtree(temp_dir) shutil.rmtree(temp_dir)
def environment_context(f, **updates): def environment_context(f, **updates):
""" """
Invoke f in the context Invoke f in the context
...@@ -480,7 +434,6 @@ def environment_context(f, **updates): ...@@ -480,7 +434,6 @@ def environment_context(f, **updates):
del os.environ[key] del os.environ[key]
os.environ.update(old_env) os.environ.update(old_env)
def argv_context(f, repl): def argv_context(f, repl):
""" """
Invoke f in the context Invoke f in the context
...@@ -492,7 +445,6 @@ def argv_context(f, repl): ...@@ -492,7 +445,6 @@ def argv_context(f, repl):
finally: finally:
sys.argv[:] = old_argv sys.argv[:] = old_argv
def reset_setup_stop_context(f): def reset_setup_stop_context(f):
""" """
When the distribute tests are run using setup.py test, and then When the distribute tests are run using setup.py test, and then
...@@ -506,21 +458,3 @@ def reset_setup_stop_context(f): ...@@ -506,21 +458,3 @@ def reset_setup_stop_context(f):
f() f()
finally: finally:
distutils.core._setup_stop_after = setup_stop_after distutils.core._setup_stop_after = setup_stop_after
def quiet_context(f):
"""
Redirect stdout/stderr to StringIO objects to prevent console output from
distutils commands.
"""
old_stdout = sys.stdout
old_stderr = sys.stderr
new_stdout = sys.stdout = StringIO.StringIO()
new_stderr = sys.stderr = StringIO.StringIO()
try:
f()
finally:
sys.stdout = old_stdout
sys.stderr = old_stderr
return new_stdout.getvalue(), new_stderr.getvalue()
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment