Skip to content

Commit

Permalink
Add a LegacySpecifierSet; resolves #74
Browse files Browse the repository at this point in the history
  • Loading branch information
Jim Porter committed Nov 1, 2017
1 parent d2ed39a commit 50c0aa3
Show file tree
Hide file tree
Showing 3 changed files with 304 additions and 55 deletions.
46 changes: 45 additions & 1 deletion docs/specifiers.rst
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,7 @@ Reference
This class abstracts handling specifying the dependencies of a project. It
can be passed a single specifier (``>=3.0``), a comma-separated list of
specifiers (``>=3.0,!=3.1``), or no specifier at all. Each individual
specifier be attempted to be parsed as a PEP 440 specifier
specifier will be attempted to be parsed as a PEP 440 specifier
(:class:`Specifier`) or as a legacy, setuptools style specifier
(:class:`LegacySpecifier`). You may combine :class:`SpecifierSet` instances
using the ``&`` operator (``SpecifierSet(">2") & SpecifierSet("<4")``).
Expand Down Expand Up @@ -128,6 +128,50 @@ Reference
all prerelease versions from being included.


.. class:: LegacySpecifierSet(specifier)

This class abstracts the handling of a set of legacy, setuptools style
specifiers. It can be passed a single specifier (``>=3.0``), a
comma-separated list of specifiers (``>=3.0,!=3.1``), or no specifier at
all. Each individual specifier will be parsed as a
(:class:`LegacySpecifier`). You may combine :class:`LegacySpecifierSet`
instances using the ``&`` operator
(``LegacySpecifierSet(">2") & LegacySpecifierSet("<4")``).

Both the membership tests and the combination support using raw strings
in place of already instantiated objects.

:param str specifiers: The string representation of a specifier or a
comma-separated list of specifiers which will
be parsed and normalized before use.
:raises InvalidSpecifier: If the given ``specifiers`` are not parseable
than this exception will be raised.

.. attribute:: prereleases

See :attr:`SpecifierSet.prereleases`.

.. method:: __contains__(version)

See :meth:`SpecifierSet.__contains__()`.

.. method:: contains(version, prereleases=None)

See :meth:`SpecifierSet.contains()`.

.. method:: __len__()

See :meth:`SpecifierSet.__len__()`.

.. method:: __iter__()

See :meth:`SpecifierSet.__iter__()`.

.. method:: filter(iterable, prereleases=None)

See :meth:`SpecifierSet.filter()`.


.. class:: Specifier(specifier, prereleases=None)

This class abstracts the handling of a single `PEP 440`_ compatible
Expand Down
160 changes: 107 additions & 53 deletions packaging/specifiers.py
Original file line number Diff line number Diff line change
Expand Up @@ -586,24 +586,11 @@ def _pad_version(left, right):
)


class SpecifierSet(BaseSpecifier):

def __init__(self, specifiers="", prereleases=None):
# Split on , to break each indidivual specifier into it's own item, and
# strip each item to remove leading/trailing whitespace.
specifiers = [s.strip() for s in specifiers.split(",") if s.strip()]

# Parsed each individual specifier, attempting first to make it a
# Specifier and falling back to a LegacySpecifier.
parsed = set()
for specifier in specifiers:
try:
parsed.add(Specifier(specifier))
except InvalidSpecifier:
parsed.add(LegacySpecifier(specifier))
class _BaseSpecifierSet(BaseSpecifier):

def __init__(self, parsed_specifiers, prereleases):
# Turn our parsed specifiers into a frozen set and save them for later.
self._specs = frozenset(parsed)
self._specs = frozenset(parsed_specifiers)

# Store our prereleases value so we can use it later to determine if
# we accept prereleases or not.
Expand All @@ -616,7 +603,8 @@ def __repr__(self):
else ""
)

return "<SpecifierSet({0!r}{1})>".format(str(self), pre)
return "<{0}({1!r}{2})>".format(self.__class__.__name__, str(self),
pre)

def __str__(self):
return ",".join(sorted(str(s) for s in self._specs))
Expand All @@ -626,11 +614,13 @@ def __hash__(self):

def __and__(self, other):
if isinstance(other, string_types):
other = SpecifierSet(other)
elif not isinstance(other, SpecifierSet):
other = self.__class__(other)
elif not isinstance(other, self.__class__):
# Currently, SpecifierSets and LegacySpecifierSets can't be
# combined.
return NotImplemented

specifier = SpecifierSet()
specifier = self.__class__()
specifier._specs = frozenset(self._specs | other._specs)

if self._prereleases is None and other._prereleases is not None:
Expand All @@ -641,28 +631,28 @@ def __and__(self, other):
specifier._prereleases = self._prereleases
else:
raise ValueError(
"Cannot combine SpecifierSets with True and False prerelease "
"overrides."
"Cannot combine {}s with True and False prerelease "
"overrides.".format(self.__class__.__name__)
)

return specifier

def __eq__(self, other):
if isinstance(other, string_types):
other = SpecifierSet(other)
other = self.__class__(other)
elif isinstance(other, _IndividualSpecifier):
other = SpecifierSet(str(other))
elif not isinstance(other, SpecifierSet):
other = self.__class__(str(other))
elif not isinstance(other, _BaseSpecifierSet):
return NotImplemented

return self._specs == other._specs

def __ne__(self, other):
if isinstance(other, string_types):
other = SpecifierSet(other)
other = self.__class__(other)
elif isinstance(other, _IndividualSpecifier):
other = SpecifierSet(str(other))
elif not isinstance(other, SpecifierSet):
other = self.__class__(str(other))
elif not isinstance(other, _BaseSpecifierSet):
return NotImplemented

return self._specs != other._specs
Expand Down Expand Up @@ -699,8 +689,7 @@ def __contains__(self, item):

def contains(self, item, prereleases=None):
# Ensure that our item is a Version or LegacyVersion instance.
if not isinstance(item, (LegacyVersion, Version)):
item = parse(item)
item = self._coerce_version(item)

# Determine if we're forcing a prerelease or not, if we're not forcing
# one for this particular filter call, then we'll use whatever the
Expand Down Expand Up @@ -744,31 +733,96 @@ def filter(self, iterable, prereleases=None):
# which will filter out any pre-releases, unless there are no final
# releases, and which will filter out LegacyVersion in general.
else:
filtered = []
found_prereleases = []
return self._filter_prereleases(iterable, prereleases)

for item in iterable:
# Ensure that we some kind of Version class for this item.
if not isinstance(item, (LegacyVersion, Version)):
parsed_version = parse(item)
else:
parsed_version = item

# Filter out any item which is parsed as a LegacyVersion
if isinstance(parsed_version, LegacyVersion):
continue
class LegacySpecifierSet(_BaseSpecifierSet):

# Store any item which is a pre-release for later unless we've
# already found a final version or we are accepting prereleases
if parsed_version.is_prerelease and not prereleases:
if not filtered:
found_prereleases.append(item)
else:
filtered.append(item)
def __init__(self, specifiers=""):
# Split on , to break each individual specifier into its own item, and
# strip each item to remove leading/trailing whitespace.
specifiers = [s.strip() for s in specifiers.split(",") if s.strip()]

# Parsed each individual specifier as a LegacySpecifier.
parsed = set(LegacySpecifier(specifier) for specifier in specifiers)

super(LegacySpecifierSet, self).__init__(parsed, None)

def _coerce_version(self, version):
if not isinstance(version, (LegacyVersion, Version)):
version = LegacyVersion(version)
return version

def _filter_prereleases(self, iterable, prereleases):
filtered = []
found_prereleases = []

for item in iterable:
# Ensure that we some kind of Version class for this item.
parsed_version = self._coerce_version(item)

# Store any item which is a pre-release for later unless we've
# already found a final version or we are accepting prereleases
if parsed_version.is_prerelease and not prereleases:
if not filtered:
found_prereleases.append(item)
else:
filtered.append(item)

# If we've found no items except for pre-releases, then we'll go
# ahead and use the pre-releases
if not filtered and found_prereleases and prereleases is None:
return found_prereleases

return filtered


class SpecifierSet(_BaseSpecifierSet):

def __init__(self, specifiers="", prereleases=None):
# Split on , to break each individual specifier into its own item, and
# strip each item to remove leading/trailing whitespace.
specifiers = [s.strip() for s in specifiers.split(",") if s.strip()]

# Parsed each individual specifier, attempting first to make it a
# Specifier and falling back to a LegacySpecifier.
parsed = set()
for specifier in specifiers:
try:
parsed.add(Specifier(specifier))
except InvalidSpecifier:
parsed.add(LegacySpecifier(specifier))

super(SpecifierSet, self).__init__(parsed, prereleases)

def _coerce_version(self, version):
if not isinstance(version, (LegacyVersion, Version)):
version = parse(version)
return version

def _filter_prereleases(self, iterable, prereleases):
filtered = []
found_prereleases = []

for item in iterable:
# Ensure that we some kind of Version class for this item.
parsed_version = self._coerce_version(item)

# Filter out any item which is parsed as a LegacyVersion
if isinstance(parsed_version, LegacyVersion):
continue

# Store any item which is a pre-release for later unless we've
# already found a final version or we are accepting prereleases
if parsed_version.is_prerelease and not prereleases:
if not filtered:
found_prereleases.append(item)
else:
filtered.append(item)

# If we've found no items except for pre-releases, then we'll go
# ahead and use the pre-releases
if not filtered and found_prereleases and prereleases is None:
return found_prereleases
# If we've found no items except for pre-releases, then we'll go
# ahead and use the pre-releases
if not filtered and found_prereleases and prereleases is None:
return found_prereleases

return filtered
return filtered
Loading

0 comments on commit 50c0aa3

Please sign in to comment.