$16 GRAYBYTE WORDPRESS FILE MANAGER $67

SERVER : premium201.web-hosting.com #1 SMP Wed Mar 26 12:08:09 UTC 2025
SERVER IP : 104.21.43.35 | ADMIN IP 216.73.216.180
OPTIONS : CRL = ON | WGT = ON | SDO = OFF | PKEX = OFF
DEACTIVATED : mail

/opt/imunify360/venv/lib/python3.11/site-packages/pip/_internal/utils/

HOME
Current File : /opt/imunify360/venv/lib/python3.11/site-packages/pip/_internal/utils//hashes.py
from __future__ import annotations

import hashlib
from collections.abc import Iterable
from typing import TYPE_CHECKING, BinaryIO, NoReturn

from pip._internal.exceptions import HashMismatch, HashMissing, InstallationError
from pip._internal.utils.misc import read_chunks

if TYPE_CHECKING:
    from hashlib import _Hash


# The recommended hash algo of the moment. Change this whenever the state of
# the art changes; it won't hurt backward compatibility.
FAVORITE_HASH = "sha256"


# Names of hashlib algorithms allowed by the --hash option and ``pip hash``
# Currently, those are the ones at least as collision-resistant as sha256.
STRONG_HASHES = ["sha256", "sha384", "sha512"]


class Hashes:
    """A wrapper that builds multiple hashes at once and checks them against
    known-good values

    """

    def __init__(self, hashes: dict[str, list[str]] | None = None) -> None:
        """
        :param hashes: A dict of algorithm names pointing to lists of allowed
            hex digests
        """
        allowed = {}
        if hashes is not None:
            for alg, keys in hashes.items():
                # Make sure values are always sorted (to ease equality checks)
                allowed[alg] = [k.lower() for k in sorted(keys)]
        self._allowed = allowed

    def __and__(self, other: Hashes) -> Hashes:
        if not isinstance(other, Hashes):
            return NotImplemented

        # If either of the Hashes object is entirely empty (i.e. no hash
        # specified at all), all hashes from the other object are allowed.
        if not other:
            return self
        if not self:
            return other

        # Otherwise only hashes that present in both objects are allowed.
        new = {}
        for alg, values in other._allowed.items():
            if alg not in self._allowed:
                continue
            new[alg] = [v for v in values if v in self._allowed[alg]]
        return Hashes(new)

    @property
    def digest_count(self) -> int:
        return sum(len(digests) for digests in self._allowed.values())

    def is_hash_allowed(self, hash_name: str, hex_digest: str) -> bool:
        """Return whether the given hex digest is allowed."""
        return hex_digest in self._allowed.get(hash_name, [])

    def check_against_chunks(self, chunks: Iterable[bytes]) -> None:
        """Check good hashes against ones built from iterable of chunks of
        data.

        Raise HashMismatch if none match.

        """
        gots = {}
        for hash_name in self._allowed.keys():
            try:
                gots[hash_name] = hashlib.new(hash_name)
            except (ValueError, TypeError):
                raise InstallationError(f"Unknown hash name: {hash_name}")

        for chunk in chunks:
            for hash in gots.values():
                hash.update(chunk)

        for hash_name, got in gots.items():
            if got.hexdigest() in self._allowed[hash_name]:
                return
        self._raise(gots)

    def _raise(self, gots: dict[str, _Hash]) -> NoReturn:
        raise HashMismatch(self._allowed, gots)

    def check_against_file(self, file: BinaryIO) -> None:
        """Check good hashes against a file-like object

        Raise HashMismatch if none match.

        """
        return self.check_against_chunks(read_chunks(file))

    def check_against_path(self, path: str) -> None:
        with open(path, "rb") as file:
            return self.check_against_file(file)

    def has_one_of(self, hashes: dict[str, str]) -> bool:
        """Return whether any of the given hashes are allowed."""
        for hash_name, hex_digest in hashes.items():
            if self.is_hash_allowed(hash_name, hex_digest):
                return True
        return False

    def __bool__(self) -> bool:
        """Return whether I know any known-good hashes."""
        return bool(self._allowed)

    def __eq__(self, other: object) -> bool:
        if not isinstance(other, Hashes):
            return NotImplemented
        return self._allowed == other._allowed

    def __hash__(self) -> int:
        return hash(
            ",".join(
                sorted(
                    ":".join((alg, digest))
                    for alg, digest_list in self._allowed.items()
                    for digest in digest_list
                )
            )
        )


class MissingHashes(Hashes):
    """A workalike for Hashes used when we're missing a hash for a requirement

    It computes the actual hash of the requirement and raises a HashMissing
    exception showing it to the user.

    """

    def __init__(self) -> None:
        """Don't offer the ``hashes`` kwarg."""
        # Pass our favorite hash in to generate a "gotten hash". With the
        # empty list, it will never match, so an error will always raise.
        super().__init__(hashes={FAVORITE_HASH: []})

    def _raise(self, gots: dict[str, _Hash]) -> NoReturn:
        raise HashMissing(gots[FAVORITE_HASH].hexdigest())


Current_dir [ NOT WRITEABLE ] Document_root [ NOT WRITEABLE ]


[ Back ]
NAME
SIZE
LAST TOUCH
USER
CAN-I?
FUNCTIONS
..
--
3 Mar 2026 8.59 AM
root / root
0755
__pycache__
--
3 Mar 2026 8.59 AM
root / root
0755
__init__.py
0 KB
13 Feb 2026 12.40 PM
root / root
0644
_jaraco_text.py
3.271 KB
13 Feb 2026 12.40 PM
root / root
0644
_log.py
0.991 KB
13 Feb 2026 12.40 PM
root / root
0644
appdirs.py
1.642 KB
13 Feb 2026 12.40 PM
root / root
0644
compat.py
2.455 KB
13 Feb 2026 12.40 PM
root / root
0644
compatibility_tags.py
6.475 KB
13 Feb 2026 12.40 PM
root / root
0644
datetime.py
0.848 KB
13 Feb 2026 12.40 PM
root / root
0644
deprecation.py
3.609 KB
13 Feb 2026 12.40 PM
root / root
0644
direct_url_helpers.py
3.125 KB
13 Feb 2026 12.40 PM
root / root
0644
egg_link.py
2.401 KB
13 Feb 2026 12.40 PM
root / root
0644
entrypoints.py
3.246 KB
13 Feb 2026 12.40 PM
root / root
0644
filesystem.py
6.73 KB
13 Feb 2026 12.40 PM
root / root
0644
filetypes.py
0.673 KB
13 Feb 2026 12.40 PM
root / root
0644
glibc.py
3.639 KB
13 Feb 2026 12.40 PM
root / root
0644
hashes.py
4.881 KB
13 Feb 2026 12.40 PM
root / root
0644
logging.py
13.1 KB
13 Feb 2026 12.40 PM
root / root
0644
misc.py
23.166 KB
13 Feb 2026 12.40 PM
root / root
0644
packaging.py
1.563 KB
13 Feb 2026 12.40 PM
root / root
0644
pylock.py
3.728 KB
13 Feb 2026 12.40 PM
root / root
0644
retry.py
1.427 KB
13 Feb 2026 12.40 PM
root / root
0644
subprocess.py
8.772 KB
13 Feb 2026 12.40 PM
root / root
0644
temp_dir.py
9.089 KB
13 Feb 2026 12.40 PM
root / root
0644
unpacking.py
12.668 KB
13 Feb 2026 12.40 PM
root / root
0644
urls.py
1.563 KB
13 Feb 2026 12.40 PM
root / root
0644
virtualenv.py
3.374 KB
13 Feb 2026 12.40 PM
root / root
0644
wheel.py
4.363 KB
13 Feb 2026 12.40 PM
root / root
0644

GRAYBYTE WORDPRESS FILE MANAGER @ 2025 CONTACT ME
Static GIF