|
|
50e8b0 |
From b97ef609100fbdd5895dab48cdab578dfeba396c Mon Sep 17 00:00:00 2001
|
|
|
50e8b0 |
From: Lumir Balhar <lbalhar@redhat.com>
|
|
|
50e8b0 |
Date: Fri, 10 Sep 2021 13:38:40 +0200
|
|
|
50e8b0 |
Subject: [PATCH 1/2] Implement handling of yanked_reason from the HTML anchor
|
|
|
50e8b0 |
|
|
|
50e8b0 |
---
|
|
|
50e8b0 |
pip/index.py | 10 ++++++++--
|
|
|
50e8b0 |
1 file changed, 8 insertions(+), 2 deletions(-)
|
|
|
50e8b0 |
|
|
|
50e8b0 |
diff --git a/pip/index.py b/pip/index.py
|
|
|
50e8b0 |
index f653f6e6a..ced52ce5a 100644
|
|
|
50e8b0 |
--- a/pip/index.py
|
|
|
50e8b0 |
+++ b/pip/index.py
|
|
|
50e8b0 |
@@ -865,7 +865,11 @@ class HTMLPage(object):
|
|
|
50e8b0 |
)
|
|
|
50e8b0 |
pyrequire = anchor.get('data-requires-python')
|
|
|
50e8b0 |
pyrequire = unescape(pyrequire) if pyrequire else None
|
|
|
50e8b0 |
- yield Link(url, self, requires_python=pyrequire)
|
|
|
50e8b0 |
+ yanked_reason = anchor.get('data-yanked', default=None)
|
|
|
50e8b0 |
+ # Empty or valueless attribute are both parsed as empty string
|
|
|
50e8b0 |
+ if yanked_reason is not None:
|
|
|
50e8b0 |
+ yanked_reason = unescape(yanked_reason)
|
|
|
50e8b0 |
+ yield Link(url, self, requires_python=pyrequire, yanked_reason=yanked_reason)
|
|
|
50e8b0 |
|
|
|
50e8b0 |
_clean_re = re.compile(r'[^a-z0-9$&+,/:;=?@.#%_\\|-]', re.I)
|
|
|
50e8b0 |
|
|
|
50e8b0 |
@@ -879,7 +883,7 @@ class HTMLPage(object):
|
|
|
50e8b0 |
|
|
|
50e8b0 |
class Link(object):
|
|
|
50e8b0 |
|
|
|
50e8b0 |
- def __init__(self, url, comes_from=None, requires_python=None):
|
|
|
50e8b0 |
+ def __init__(self, url, comes_from=None, requires_python=None, yanked_reason=None):
|
|
|
50e8b0 |
"""
|
|
|
50e8b0 |
Object representing a parsed link from https://pypi.python.org/simple/*
|
|
|
50e8b0 |
|
|
|
50e8b0 |
@@ -900,6 +904,8 @@ class Link(object):
|
|
|
50e8b0 |
self.url = url
|
|
|
50e8b0 |
self.comes_from = comes_from
|
|
|
50e8b0 |
self.requires_python = requires_python if requires_python else None
|
|
|
50e8b0 |
+ self.yanked_reason = yanked_reason
|
|
|
50e8b0 |
+ self.yanked = yanked_reason is not None
|
|
|
50e8b0 |
|
|
|
50e8b0 |
def __str__(self):
|
|
|
50e8b0 |
if self.requires_python:
|
|
|
50e8b0 |
--
|
|
|
50e8b0 |
2.31.1
|
|
|
50e8b0 |
|
|
|
50e8b0 |
From d8dc6ee5d6809736dce43dc1e57d497f9ff91f26 Mon Sep 17 00:00:00 2001
|
|
|
50e8b0 |
From: Lumir Balhar <lbalhar@redhat.com>
|
|
|
50e8b0 |
Date: Fri, 10 Sep 2021 13:43:22 +0200
|
|
|
50e8b0 |
Subject: [PATCH 2/2] Skip all yanked candidates if possible
|
|
|
50e8b0 |
|
|
|
50e8b0 |
---
|
|
|
50e8b0 |
pip/index.py | 21 +++++++++++++++++++++
|
|
|
50e8b0 |
1 file changed, 21 insertions(+)
|
|
|
50e8b0 |
|
|
|
50e8b0 |
diff --git a/pip/index.py b/pip/index.py
|
|
|
50e8b0 |
index ced52ce5a..823bbaf7d 100644
|
|
|
50e8b0 |
--- a/pip/index.py
|
|
|
50e8b0 |
+++ b/pip/index.py
|
|
|
50e8b0 |
@@ -489,6 +489,27 @@ class PackageFinder(object):
|
|
|
50e8b0 |
if applicable_candidates:
|
|
|
50e8b0 |
best_candidate = max(applicable_candidates,
|
|
|
50e8b0 |
key=self._candidate_sort_key)
|
|
|
50e8b0 |
+ # If we cannot find a non-yanked candidate,
|
|
|
50e8b0 |
+ # use the best one and print a warning about it.
|
|
|
50e8b0 |
+ # Otherwise, try to find another best candidate, ignoring
|
|
|
50e8b0 |
+ # all the yanked releases.
|
|
|
50e8b0 |
+ if getattr(best_candidate.location, "yanked", False):
|
|
|
50e8b0 |
+ nonyanked_candidates = [
|
|
|
50e8b0 |
+ c for c in applicable_candidates
|
|
|
50e8b0 |
+ if not getattr(c.location, "yanked", False)
|
|
|
50e8b0 |
+ ]
|
|
|
50e8b0 |
+
|
|
|
50e8b0 |
+ if set(nonyanked_candidates):
|
|
|
50e8b0 |
+ best_candidate = max(nonyanked_candidates,
|
|
|
50e8b0 |
+ key=self._candidate_sort_key)
|
|
|
50e8b0 |
+ else:
|
|
|
50e8b0 |
+ warning_message = (
|
|
|
50e8b0 |
+ "WARNING: The candidate selected for download or install "
|
|
|
50e8b0 |
+ "is a yanked version: '{}' candidate (version {} at {})"
|
|
|
50e8b0 |
+ ).format(best_candidate.project, best_candidate.version, best_candidate.location)
|
|
|
50e8b0 |
+ if best_candidate.location.yanked_reason:
|
|
|
50e8b0 |
+ warning_message += "\nReason for being yanked: {}".format(best_candidate.location.yanked_reason)
|
|
|
50e8b0 |
+ logger.warning(warning_message)
|
|
|
50e8b0 |
else:
|
|
|
50e8b0 |
best_candidate = None
|
|
|
50e8b0 |
|
|
|
50e8b0 |
--
|
|
|
50e8b0 |
2.31.1
|
|
|
50e8b0 |
|