mirror of
https://github.com/Tautulli/Tautulli.git
synced 2025-08-22 06:13:25 -07:00
Merge branch 'nightly' into dependabot/pip/nightly/tokenize-rt-6.0.0
This commit is contained in:
commit
a1ee20a7c5
23 changed files with 705 additions and 202 deletions
|
@ -38,7 +38,7 @@ CL_BLANK = "
|
||||||
URI_SCHEME = "cloudinary"
|
URI_SCHEME = "cloudinary"
|
||||||
API_VERSION = "v1_1"
|
API_VERSION = "v1_1"
|
||||||
|
|
||||||
VERSION = "1.40.0"
|
VERSION = "1.41.0"
|
||||||
|
|
||||||
_USER_PLATFORM_DETAILS = "; ".join((platform(), "Python {}".format(python_version())))
|
_USER_PLATFORM_DETAILS = "; ".join((platform(), "Python {}".format(python_version())))
|
||||||
|
|
||||||
|
|
|
@ -543,10 +543,6 @@ def create_upload_preset(**options):
|
||||||
return call_api("post", uri, params, **options)
|
return call_api("post", uri, params, **options)
|
||||||
|
|
||||||
|
|
||||||
def create_folder(path, **options):
|
|
||||||
return call_api("post", ["folders", path], {}, **options)
|
|
||||||
|
|
||||||
|
|
||||||
def root_folders(**options):
|
def root_folders(**options):
|
||||||
return call_api("get", ["folders"], only(options, "next_cursor", "max_results"), **options)
|
return call_api("get", ["folders"], only(options, "next_cursor", "max_results"), **options)
|
||||||
|
|
||||||
|
@ -555,6 +551,24 @@ def subfolders(of_folder_path, **options):
|
||||||
return call_api("get", ["folders", of_folder_path], only(options, "next_cursor", "max_results"), **options)
|
return call_api("get", ["folders", of_folder_path], only(options, "next_cursor", "max_results"), **options)
|
||||||
|
|
||||||
|
|
||||||
|
def create_folder(path, **options):
|
||||||
|
return call_api("post", ["folders", path], {}, **options)
|
||||||
|
|
||||||
|
|
||||||
|
def rename_folder(from_path, to_path, **options):
|
||||||
|
"""
|
||||||
|
Renames folder
|
||||||
|
|
||||||
|
:param from_path: The full path of an existing asset folder.
|
||||||
|
:param to_path: The full path of the new asset folder.
|
||||||
|
:param options: Additional options
|
||||||
|
|
||||||
|
:rtype: Response
|
||||||
|
"""
|
||||||
|
params = {"to_folder": to_path}
|
||||||
|
return call_api("put", ["folders", from_path], params, **options)
|
||||||
|
|
||||||
|
|
||||||
def delete_folder(path, **options):
|
def delete_folder(path, **options):
|
||||||
"""Deletes folder
|
"""Deletes folder
|
||||||
|
|
||||||
|
@ -727,7 +741,7 @@ def update_metadata_field(field_external_id, field, **options):
|
||||||
|
|
||||||
def __metadata_field_params(field):
|
def __metadata_field_params(field):
|
||||||
return only(field, "type", "external_id", "label", "mandatory", "restrictions",
|
return only(field, "type", "external_id", "label", "mandatory", "restrictions",
|
||||||
"default_value", "validation", "datasource")
|
"default_value", "default_disabled", "validation", "datasource")
|
||||||
|
|
||||||
|
|
||||||
def delete_metadata_field(field_external_id, **options):
|
def delete_metadata_field(field_external_id, **options):
|
||||||
|
|
|
@ -11,7 +11,7 @@ AUTH_TOKEN_UNSAFE_RE = r'([ "#%&\'\/:;<=>?@\[\\\]^`{\|}~]+)'
|
||||||
|
|
||||||
|
|
||||||
def generate(url=None, acl=None, start_time=None, duration=None,
|
def generate(url=None, acl=None, start_time=None, duration=None,
|
||||||
expiration=None, ip=None, key=None, token_name=AUTH_TOKEN_NAME):
|
expiration=None, ip=None, key=None, token_name=AUTH_TOKEN_NAME, **_):
|
||||||
|
|
||||||
if expiration is None:
|
if expiration is None:
|
||||||
if duration is not None:
|
if duration is not None:
|
||||||
|
|
|
@ -820,7 +820,7 @@ def cloudinary_url(source, **options):
|
||||||
transformation = re.sub(r'([^:])/+', r'\1/', transformation)
|
transformation = re.sub(r'([^:])/+', r'\1/', transformation)
|
||||||
|
|
||||||
signature = None
|
signature = None
|
||||||
if sign_url and not auth_token:
|
if sign_url and (not auth_token or auth_token.pop('set_url_signature', False)):
|
||||||
to_sign = "/".join(__compact([transformation, source_to_sign]))
|
to_sign = "/".join(__compact([transformation, source_to_sign]))
|
||||||
if long_url_signature:
|
if long_url_signature:
|
||||||
# Long signature forces SHA256
|
# Long signature forces SHA256
|
||||||
|
|
|
@ -25,7 +25,7 @@ from ._compat import (
|
||||||
install,
|
install,
|
||||||
)
|
)
|
||||||
from ._functools import method_cache, pass_none
|
from ._functools import method_cache, pass_none
|
||||||
from ._itertools import always_iterable, unique_everseen
|
from ._itertools import always_iterable, bucket, unique_everseen
|
||||||
from ._meta import PackageMetadata, SimplePath
|
from ._meta import PackageMetadata, SimplePath
|
||||||
|
|
||||||
from contextlib import suppress
|
from contextlib import suppress
|
||||||
|
@ -39,6 +39,7 @@ __all__ = [
|
||||||
'DistributionFinder',
|
'DistributionFinder',
|
||||||
'PackageMetadata',
|
'PackageMetadata',
|
||||||
'PackageNotFoundError',
|
'PackageNotFoundError',
|
||||||
|
'SimplePath',
|
||||||
'distribution',
|
'distribution',
|
||||||
'distributions',
|
'distributions',
|
||||||
'entry_points',
|
'entry_points',
|
||||||
|
@ -388,7 +389,7 @@ class Distribution(metaclass=abc.ABCMeta):
|
||||||
if not name:
|
if not name:
|
||||||
raise ValueError("A distribution name is required.")
|
raise ValueError("A distribution name is required.")
|
||||||
try:
|
try:
|
||||||
return next(iter(cls.discover(name=name)))
|
return next(iter(cls._prefer_valid(cls.discover(name=name))))
|
||||||
except StopIteration:
|
except StopIteration:
|
||||||
raise PackageNotFoundError(name)
|
raise PackageNotFoundError(name)
|
||||||
|
|
||||||
|
@ -412,6 +413,16 @@ class Distribution(metaclass=abc.ABCMeta):
|
||||||
resolver(context) for resolver in cls._discover_resolvers()
|
resolver(context) for resolver in cls._discover_resolvers()
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _prefer_valid(dists: Iterable[Distribution]) -> Iterable[Distribution]:
|
||||||
|
"""
|
||||||
|
Prefer (move to the front) distributions that have metadata.
|
||||||
|
|
||||||
|
Ref python/importlib_resources#489.
|
||||||
|
"""
|
||||||
|
buckets = bucket(dists, lambda dist: bool(dist.metadata))
|
||||||
|
return itertools.chain(buckets[True], buckets[False])
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def at(path: str | os.PathLike[str]) -> Distribution:
|
def at(path: str | os.PathLike[str]) -> Distribution:
|
||||||
"""Return a Distribution for the indicated metadata path.
|
"""Return a Distribution for the indicated metadata path.
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
from collections import defaultdict, deque
|
||||||
from itertools import filterfalse
|
from itertools import filterfalse
|
||||||
|
|
||||||
|
|
||||||
|
@ -71,3 +72,100 @@ def always_iterable(obj, base_type=(str, bytes)):
|
||||||
return iter(obj)
|
return iter(obj)
|
||||||
except TypeError:
|
except TypeError:
|
||||||
return iter((obj,))
|
return iter((obj,))
|
||||||
|
|
||||||
|
|
||||||
|
# Copied from more_itertools 10.3
|
||||||
|
class bucket:
|
||||||
|
"""Wrap *iterable* and return an object that buckets the iterable into
|
||||||
|
child iterables based on a *key* function.
|
||||||
|
|
||||||
|
>>> iterable = ['a1', 'b1', 'c1', 'a2', 'b2', 'c2', 'b3']
|
||||||
|
>>> s = bucket(iterable, key=lambda x: x[0]) # Bucket by 1st character
|
||||||
|
>>> sorted(list(s)) # Get the keys
|
||||||
|
['a', 'b', 'c']
|
||||||
|
>>> a_iterable = s['a']
|
||||||
|
>>> next(a_iterable)
|
||||||
|
'a1'
|
||||||
|
>>> next(a_iterable)
|
||||||
|
'a2'
|
||||||
|
>>> list(s['b'])
|
||||||
|
['b1', 'b2', 'b3']
|
||||||
|
|
||||||
|
The original iterable will be advanced and its items will be cached until
|
||||||
|
they are used by the child iterables. This may require significant storage.
|
||||||
|
|
||||||
|
By default, attempting to select a bucket to which no items belong will
|
||||||
|
exhaust the iterable and cache all values.
|
||||||
|
If you specify a *validator* function, selected buckets will instead be
|
||||||
|
checked against it.
|
||||||
|
|
||||||
|
>>> from itertools import count
|
||||||
|
>>> it = count(1, 2) # Infinite sequence of odd numbers
|
||||||
|
>>> key = lambda x: x % 10 # Bucket by last digit
|
||||||
|
>>> validator = lambda x: x in {1, 3, 5, 7, 9} # Odd digits only
|
||||||
|
>>> s = bucket(it, key=key, validator=validator)
|
||||||
|
>>> 2 in s
|
||||||
|
False
|
||||||
|
>>> list(s[2])
|
||||||
|
[]
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, iterable, key, validator=None):
|
||||||
|
self._it = iter(iterable)
|
||||||
|
self._key = key
|
||||||
|
self._cache = defaultdict(deque)
|
||||||
|
self._validator = validator or (lambda x: True)
|
||||||
|
|
||||||
|
def __contains__(self, value):
|
||||||
|
if not self._validator(value):
|
||||||
|
return False
|
||||||
|
|
||||||
|
try:
|
||||||
|
item = next(self[value])
|
||||||
|
except StopIteration:
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
self._cache[value].appendleft(item)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
def _get_values(self, value):
|
||||||
|
"""
|
||||||
|
Helper to yield items from the parent iterator that match *value*.
|
||||||
|
Items that don't match are stored in the local cache as they
|
||||||
|
are encountered.
|
||||||
|
"""
|
||||||
|
while True:
|
||||||
|
# If we've cached some items that match the target value, emit
|
||||||
|
# the first one and evict it from the cache.
|
||||||
|
if self._cache[value]:
|
||||||
|
yield self._cache[value].popleft()
|
||||||
|
# Otherwise we need to advance the parent iterator to search for
|
||||||
|
# a matching item, caching the rest.
|
||||||
|
else:
|
||||||
|
while True:
|
||||||
|
try:
|
||||||
|
item = next(self._it)
|
||||||
|
except StopIteration:
|
||||||
|
return
|
||||||
|
item_value = self._key(item)
|
||||||
|
if item_value == value:
|
||||||
|
yield item
|
||||||
|
break
|
||||||
|
elif self._validator(item_value):
|
||||||
|
self._cache[item_value].append(item)
|
||||||
|
|
||||||
|
def __iter__(self):
|
||||||
|
for item in self._it:
|
||||||
|
item_value = self._key(item)
|
||||||
|
if self._validator(item_value):
|
||||||
|
self._cache[item_value].append(item)
|
||||||
|
|
||||||
|
yield from self._cache.keys()
|
||||||
|
|
||||||
|
def __getitem__(self, value):
|
||||||
|
if not self._validator(value):
|
||||||
|
return iter(())
|
||||||
|
|
||||||
|
return self._get_values(value)
|
||||||
|
|
|
@ -14,6 +14,14 @@ def compose(*funcs):
|
||||||
"""
|
"""
|
||||||
Compose any number of unary functions into a single unary function.
|
Compose any number of unary functions into a single unary function.
|
||||||
|
|
||||||
|
Comparable to
|
||||||
|
`function composition <https://en.wikipedia.org/wiki/Function_composition>`_
|
||||||
|
in mathematics:
|
||||||
|
|
||||||
|
``h = g ∘ f`` implies ``h(x) = g(f(x))``.
|
||||||
|
|
||||||
|
In Python, ``h = compose(g, f)``.
|
||||||
|
|
||||||
>>> import textwrap
|
>>> import textwrap
|
||||||
>>> expected = str.strip(textwrap.dedent(compose.__doc__))
|
>>> expected = str.strip(textwrap.dedent(compose.__doc__))
|
||||||
>>> strip_and_dedent = compose(str.strip, textwrap.dedent)
|
>>> strip_and_dedent = compose(str.strip, textwrap.dedent)
|
||||||
|
|
|
@ -27,7 +27,7 @@ from .exceptions import (
|
||||||
)
|
)
|
||||||
from .jwks_client import PyJWKClient
|
from .jwks_client import PyJWKClient
|
||||||
|
|
||||||
__version__ = "2.8.0"
|
__version__ = "2.9.0"
|
||||||
|
|
||||||
__title__ = "PyJWT"
|
__title__ = "PyJWT"
|
||||||
__description__ = "JSON Web Token implementation in Python"
|
__description__ = "JSON Web Token implementation in Python"
|
||||||
|
|
|
@ -3,9 +3,8 @@ from __future__ import annotations
|
||||||
import hashlib
|
import hashlib
|
||||||
import hmac
|
import hmac
|
||||||
import json
|
import json
|
||||||
import sys
|
|
||||||
from abc import ABC, abstractmethod
|
from abc import ABC, abstractmethod
|
||||||
from typing import TYPE_CHECKING, Any, ClassVar, NoReturn, Union, cast, overload
|
from typing import TYPE_CHECKING, Any, ClassVar, Literal, NoReturn, cast, overload
|
||||||
|
|
||||||
from .exceptions import InvalidKeyError
|
from .exceptions import InvalidKeyError
|
||||||
from .types import HashlibHash, JWKDict
|
from .types import HashlibHash, JWKDict
|
||||||
|
@ -21,14 +20,8 @@ from .utils import (
|
||||||
to_base64url_uint,
|
to_base64url_uint,
|
||||||
)
|
)
|
||||||
|
|
||||||
if sys.version_info >= (3, 8):
|
|
||||||
from typing import Literal
|
|
||||||
else:
|
|
||||||
from typing_extensions import Literal
|
|
||||||
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from cryptography.exceptions import InvalidSignature
|
from cryptography.exceptions import InvalidSignature, UnsupportedAlgorithm
|
||||||
from cryptography.hazmat.backends import default_backend
|
from cryptography.hazmat.backends import default_backend
|
||||||
from cryptography.hazmat.primitives import hashes
|
from cryptography.hazmat.primitives import hashes
|
||||||
from cryptography.hazmat.primitives.asymmetric import padding
|
from cryptography.hazmat.primitives.asymmetric import padding
|
||||||
|
@ -194,18 +187,16 @@ class Algorithm(ABC):
|
||||||
@overload
|
@overload
|
||||||
@staticmethod
|
@staticmethod
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def to_jwk(key_obj, as_dict: Literal[True]) -> JWKDict:
|
def to_jwk(key_obj, as_dict: Literal[True]) -> JWKDict: ... # pragma: no cover
|
||||||
... # pragma: no cover
|
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
@staticmethod
|
@staticmethod
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def to_jwk(key_obj, as_dict: Literal[False] = False) -> str:
|
def to_jwk(key_obj, as_dict: Literal[False] = False) -> str: ... # pragma: no cover
|
||||||
... # pragma: no cover
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def to_jwk(key_obj, as_dict: bool = False) -> Union[JWKDict, str]:
|
def to_jwk(key_obj, as_dict: bool = False) -> JWKDict | str:
|
||||||
"""
|
"""
|
||||||
Serializes a given key into a JWK
|
Serializes a given key into a JWK
|
||||||
"""
|
"""
|
||||||
|
@ -274,16 +265,18 @@ class HMACAlgorithm(Algorithm):
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def to_jwk(key_obj: str | bytes, as_dict: Literal[True]) -> JWKDict:
|
def to_jwk(
|
||||||
... # pragma: no cover
|
key_obj: str | bytes, as_dict: Literal[True]
|
||||||
|
) -> JWKDict: ... # pragma: no cover
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def to_jwk(key_obj: str | bytes, as_dict: Literal[False] = False) -> str:
|
def to_jwk(
|
||||||
... # pragma: no cover
|
key_obj: str | bytes, as_dict: Literal[False] = False
|
||||||
|
) -> str: ... # pragma: no cover
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def to_jwk(key_obj: str | bytes, as_dict: bool = False) -> Union[JWKDict, str]:
|
def to_jwk(key_obj: str | bytes, as_dict: bool = False) -> JWKDict | str:
|
||||||
jwk = {
|
jwk = {
|
||||||
"k": base64url_encode(force_bytes(key_obj)).decode(),
|
"k": base64url_encode(force_bytes(key_obj)).decode(),
|
||||||
"kty": "oct",
|
"kty": "oct",
|
||||||
|
@ -350,22 +343,25 @@ if has_crypto:
|
||||||
RSAPrivateKey, load_pem_private_key(key_bytes, password=None)
|
RSAPrivateKey, load_pem_private_key(key_bytes, password=None)
|
||||||
)
|
)
|
||||||
except ValueError:
|
except ValueError:
|
||||||
|
try:
|
||||||
return cast(RSAPublicKey, load_pem_public_key(key_bytes))
|
return cast(RSAPublicKey, load_pem_public_key(key_bytes))
|
||||||
|
except (ValueError, UnsupportedAlgorithm):
|
||||||
|
raise InvalidKeyError("Could not parse the provided public key.")
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
@staticmethod
|
|
||||||
def to_jwk(key_obj: AllowedRSAKeys, as_dict: Literal[True]) -> JWKDict:
|
|
||||||
... # pragma: no cover
|
|
||||||
|
|
||||||
@overload
|
|
||||||
@staticmethod
|
|
||||||
def to_jwk(key_obj: AllowedRSAKeys, as_dict: Literal[False] = False) -> str:
|
|
||||||
... # pragma: no cover
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def to_jwk(
|
def to_jwk(
|
||||||
key_obj: AllowedRSAKeys, as_dict: bool = False
|
key_obj: AllowedRSAKeys, as_dict: Literal[True]
|
||||||
) -> Union[JWKDict, str]:
|
) -> JWKDict: ... # pragma: no cover
|
||||||
|
|
||||||
|
@overload
|
||||||
|
@staticmethod
|
||||||
|
def to_jwk(
|
||||||
|
key_obj: AllowedRSAKeys, as_dict: Literal[False] = False
|
||||||
|
) -> str: ... # pragma: no cover
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def to_jwk(key_obj: AllowedRSAKeys, as_dict: bool = False) -> JWKDict | str:
|
||||||
obj: dict[str, Any] | None = None
|
obj: dict[str, Any] | None = None
|
||||||
|
|
||||||
if hasattr(key_obj, "private_numbers"):
|
if hasattr(key_obj, "private_numbers"):
|
||||||
|
@ -533,7 +529,7 @@ if has_crypto:
|
||||||
|
|
||||||
return der_to_raw_signature(der_sig, key.curve)
|
return der_to_raw_signature(der_sig, key.curve)
|
||||||
|
|
||||||
def verify(self, msg: bytes, key: "AllowedECKeys", sig: bytes) -> bool:
|
def verify(self, msg: bytes, key: AllowedECKeys, sig: bytes) -> bool:
|
||||||
try:
|
try:
|
||||||
der_sig = raw_to_der_signature(sig, key.curve)
|
der_sig = raw_to_der_signature(sig, key.curve)
|
||||||
except ValueError:
|
except ValueError:
|
||||||
|
@ -552,18 +548,18 @@ if has_crypto:
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def to_jwk(key_obj: AllowedECKeys, as_dict: Literal[True]) -> JWKDict:
|
def to_jwk(
|
||||||
... # pragma: no cover
|
key_obj: AllowedECKeys, as_dict: Literal[True]
|
||||||
|
) -> JWKDict: ... # pragma: no cover
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def to_jwk(key_obj: AllowedECKeys, as_dict: Literal[False] = False) -> str:
|
def to_jwk(
|
||||||
... # pragma: no cover
|
key_obj: AllowedECKeys, as_dict: Literal[False] = False
|
||||||
|
) -> str: ... # pragma: no cover
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def to_jwk(
|
def to_jwk(key_obj: AllowedECKeys, as_dict: bool = False) -> JWKDict | str:
|
||||||
key_obj: AllowedECKeys, as_dict: bool = False
|
|
||||||
) -> Union[JWKDict, str]:
|
|
||||||
if isinstance(key_obj, EllipticCurvePrivateKey):
|
if isinstance(key_obj, EllipticCurvePrivateKey):
|
||||||
public_numbers = key_obj.public_key().public_numbers()
|
public_numbers = key_obj.public_key().public_numbers()
|
||||||
elif isinstance(key_obj, EllipticCurvePublicKey):
|
elif isinstance(key_obj, EllipticCurvePublicKey):
|
||||||
|
@ -771,16 +767,18 @@ if has_crypto:
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def to_jwk(key: AllowedOKPKeys, as_dict: Literal[True]) -> JWKDict:
|
def to_jwk(
|
||||||
... # pragma: no cover
|
key: AllowedOKPKeys, as_dict: Literal[True]
|
||||||
|
) -> JWKDict: ... # pragma: no cover
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def to_jwk(key: AllowedOKPKeys, as_dict: Literal[False] = False) -> str:
|
def to_jwk(
|
||||||
... # pragma: no cover
|
key: AllowedOKPKeys, as_dict: Literal[False] = False
|
||||||
|
) -> str: ... # pragma: no cover
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def to_jwk(key: AllowedOKPKeys, as_dict: bool = False) -> Union[JWKDict, str]:
|
def to_jwk(key: AllowedOKPKeys, as_dict: bool = False) -> JWKDict | str:
|
||||||
if isinstance(key, (Ed25519PublicKey, Ed448PublicKey)):
|
if isinstance(key, (Ed25519PublicKey, Ed448PublicKey)):
|
||||||
x = key.public_bytes(
|
x = key.public_bytes(
|
||||||
encoding=Encoding.Raw,
|
encoding=Encoding.Raw,
|
||||||
|
|
|
@ -5,7 +5,13 @@ import time
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
from .algorithms import get_default_algorithms, has_crypto, requires_cryptography
|
from .algorithms import get_default_algorithms, has_crypto, requires_cryptography
|
||||||
from .exceptions import InvalidKeyError, PyJWKError, PyJWKSetError, PyJWTError
|
from .exceptions import (
|
||||||
|
InvalidKeyError,
|
||||||
|
MissingCryptographyError,
|
||||||
|
PyJWKError,
|
||||||
|
PyJWKSetError,
|
||||||
|
PyJWTError,
|
||||||
|
)
|
||||||
from .types import JWKDict
|
from .types import JWKDict
|
||||||
|
|
||||||
|
|
||||||
|
@ -50,21 +56,25 @@ class PyJWK:
|
||||||
raise InvalidKeyError(f"Unsupported kty: {kty}")
|
raise InvalidKeyError(f"Unsupported kty: {kty}")
|
||||||
|
|
||||||
if not has_crypto and algorithm in requires_cryptography:
|
if not has_crypto and algorithm in requires_cryptography:
|
||||||
raise PyJWKError(f"{algorithm} requires 'cryptography' to be installed.")
|
raise MissingCryptographyError(
|
||||||
|
f"{algorithm} requires 'cryptography' to be installed."
|
||||||
|
)
|
||||||
|
|
||||||
self.Algorithm = self._algorithms.get(algorithm)
|
self.algorithm_name = algorithm
|
||||||
|
|
||||||
if not self.Algorithm:
|
if algorithm in self._algorithms:
|
||||||
|
self.Algorithm = self._algorithms[algorithm]
|
||||||
|
else:
|
||||||
raise PyJWKError(f"Unable to find an algorithm for key: {self._jwk_data}")
|
raise PyJWKError(f"Unable to find an algorithm for key: {self._jwk_data}")
|
||||||
|
|
||||||
self.key = self.Algorithm.from_jwk(self._jwk_data)
|
self.key = self.Algorithm.from_jwk(self._jwk_data)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def from_dict(obj: JWKDict, algorithm: str | None = None) -> "PyJWK":
|
def from_dict(obj: JWKDict, algorithm: str | None = None) -> PyJWK:
|
||||||
return PyJWK(obj, algorithm)
|
return PyJWK(obj, algorithm)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def from_json(data: str, algorithm: None = None) -> "PyJWK":
|
def from_json(data: str, algorithm: None = None) -> PyJWK:
|
||||||
obj = json.loads(data)
|
obj = json.loads(data)
|
||||||
return PyJWK.from_dict(obj, algorithm)
|
return PyJWK.from_dict(obj, algorithm)
|
||||||
|
|
||||||
|
@ -94,7 +104,9 @@ class PyJWKSet:
|
||||||
for key in keys:
|
for key in keys:
|
||||||
try:
|
try:
|
||||||
self.keys.append(PyJWK(key))
|
self.keys.append(PyJWK(key))
|
||||||
except PyJWTError:
|
except PyJWTError as error:
|
||||||
|
if isinstance(error, MissingCryptographyError):
|
||||||
|
raise error
|
||||||
# skip unusable keys
|
# skip unusable keys
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
@ -104,16 +116,16 @@ class PyJWKSet:
|
||||||
)
|
)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def from_dict(obj: dict[str, Any]) -> "PyJWKSet":
|
def from_dict(obj: dict[str, Any]) -> PyJWKSet:
|
||||||
keys = obj.get("keys", [])
|
keys = obj.get("keys", [])
|
||||||
return PyJWKSet(keys)
|
return PyJWKSet(keys)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def from_json(data: str) -> "PyJWKSet":
|
def from_json(data: str) -> PyJWKSet:
|
||||||
obj = json.loads(data)
|
obj = json.loads(data)
|
||||||
return PyJWKSet.from_dict(obj)
|
return PyJWKSet.from_dict(obj)
|
||||||
|
|
||||||
def __getitem__(self, kid: str) -> "PyJWK":
|
def __getitem__(self, kid: str) -> PyJWK:
|
||||||
for key in self.keys:
|
for key in self.keys:
|
||||||
if key.key_id == kid:
|
if key.key_id == kid:
|
||||||
return key
|
return key
|
||||||
|
|
|
@ -11,6 +11,7 @@ from .algorithms import (
|
||||||
has_crypto,
|
has_crypto,
|
||||||
requires_cryptography,
|
requires_cryptography,
|
||||||
)
|
)
|
||||||
|
from .api_jwk import PyJWK
|
||||||
from .exceptions import (
|
from .exceptions import (
|
||||||
DecodeError,
|
DecodeError,
|
||||||
InvalidAlgorithmError,
|
InvalidAlgorithmError,
|
||||||
|
@ -172,7 +173,7 @@ class PyJWS:
|
||||||
def decode_complete(
|
def decode_complete(
|
||||||
self,
|
self,
|
||||||
jwt: str | bytes,
|
jwt: str | bytes,
|
||||||
key: AllowedPublicKeys | str | bytes = "",
|
key: AllowedPublicKeys | PyJWK | str | bytes = "",
|
||||||
algorithms: list[str] | None = None,
|
algorithms: list[str] | None = None,
|
||||||
options: dict[str, Any] | None = None,
|
options: dict[str, Any] | None = None,
|
||||||
detached_payload: bytes | None = None,
|
detached_payload: bytes | None = None,
|
||||||
|
@ -190,7 +191,7 @@ class PyJWS:
|
||||||
merged_options = {**self.options, **options}
|
merged_options = {**self.options, **options}
|
||||||
verify_signature = merged_options["verify_signature"]
|
verify_signature = merged_options["verify_signature"]
|
||||||
|
|
||||||
if verify_signature and not algorithms:
|
if verify_signature and not algorithms and not isinstance(key, PyJWK):
|
||||||
raise DecodeError(
|
raise DecodeError(
|
||||||
'It is required that you pass in a value for the "algorithms" argument when calling decode().'
|
'It is required that you pass in a value for the "algorithms" argument when calling decode().'
|
||||||
)
|
)
|
||||||
|
@ -217,7 +218,7 @@ class PyJWS:
|
||||||
def decode(
|
def decode(
|
||||||
self,
|
self,
|
||||||
jwt: str | bytes,
|
jwt: str | bytes,
|
||||||
key: AllowedPublicKeys | str | bytes = "",
|
key: AllowedPublicKeys | PyJWK | str | bytes = "",
|
||||||
algorithms: list[str] | None = None,
|
algorithms: list[str] | None = None,
|
||||||
options: dict[str, Any] | None = None,
|
options: dict[str, Any] | None = None,
|
||||||
detached_payload: bytes | None = None,
|
detached_payload: bytes | None = None,
|
||||||
|
@ -289,9 +290,11 @@ class PyJWS:
|
||||||
signing_input: bytes,
|
signing_input: bytes,
|
||||||
header: dict[str, Any],
|
header: dict[str, Any],
|
||||||
signature: bytes,
|
signature: bytes,
|
||||||
key: AllowedPublicKeys | str | bytes = "",
|
key: AllowedPublicKeys | PyJWK | str | bytes = "",
|
||||||
algorithms: list[str] | None = None,
|
algorithms: list[str] | None = None,
|
||||||
) -> None:
|
) -> None:
|
||||||
|
if algorithms is None and isinstance(key, PyJWK):
|
||||||
|
algorithms = [key.algorithm_name]
|
||||||
try:
|
try:
|
||||||
alg = header["alg"]
|
alg = header["alg"]
|
||||||
except KeyError:
|
except KeyError:
|
||||||
|
@ -300,6 +303,10 @@ class PyJWS:
|
||||||
if not alg or (algorithms is not None and alg not in algorithms):
|
if not alg or (algorithms is not None and alg not in algorithms):
|
||||||
raise InvalidAlgorithmError("The specified alg value is not allowed")
|
raise InvalidAlgorithmError("The specified alg value is not allowed")
|
||||||
|
|
||||||
|
if isinstance(key, PyJWK):
|
||||||
|
alg_obj = key.Algorithm
|
||||||
|
prepared_key = key.key
|
||||||
|
else:
|
||||||
try:
|
try:
|
||||||
alg_obj = self.get_algorithm_by_name(alg)
|
alg_obj = self.get_algorithm_by_name(alg)
|
||||||
except NotImplementedError as e:
|
except NotImplementedError as e:
|
||||||
|
|
|
@ -5,7 +5,7 @@ import warnings
|
||||||
from calendar import timegm
|
from calendar import timegm
|
||||||
from collections.abc import Iterable
|
from collections.abc import Iterable
|
||||||
from datetime import datetime, timedelta, timezone
|
from datetime import datetime, timedelta, timezone
|
||||||
from typing import TYPE_CHECKING, Any
|
from typing import TYPE_CHECKING, Any, List
|
||||||
|
|
||||||
from . import api_jws
|
from . import api_jws
|
||||||
from .exceptions import (
|
from .exceptions import (
|
||||||
|
@ -21,6 +21,7 @@ from .warnings import RemovedInPyjwt3Warning
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
from .algorithms import AllowedPrivateKeys, AllowedPublicKeys
|
from .algorithms import AllowedPrivateKeys, AllowedPublicKeys
|
||||||
|
from .api_jwk import PyJWK
|
||||||
|
|
||||||
|
|
||||||
class PyJWT:
|
class PyJWT:
|
||||||
|
@ -100,7 +101,7 @@ class PyJWT:
|
||||||
def decode_complete(
|
def decode_complete(
|
||||||
self,
|
self,
|
||||||
jwt: str | bytes,
|
jwt: str | bytes,
|
||||||
key: AllowedPublicKeys | str | bytes = "",
|
key: AllowedPublicKeys | PyJWK | str | bytes = "",
|
||||||
algorithms: list[str] | None = None,
|
algorithms: list[str] | None = None,
|
||||||
options: dict[str, Any] | None = None,
|
options: dict[str, Any] | None = None,
|
||||||
# deprecated arg, remove in pyjwt3
|
# deprecated arg, remove in pyjwt3
|
||||||
|
@ -110,7 +111,7 @@ class PyJWT:
|
||||||
# passthrough arguments to _validate_claims
|
# passthrough arguments to _validate_claims
|
||||||
# consider putting in options
|
# consider putting in options
|
||||||
audience: str | Iterable[str] | None = None,
|
audience: str | Iterable[str] | None = None,
|
||||||
issuer: str | None = None,
|
issuer: str | List[str] | None = None,
|
||||||
leeway: float | timedelta = 0,
|
leeway: float | timedelta = 0,
|
||||||
# kwargs
|
# kwargs
|
||||||
**kwargs: Any,
|
**kwargs: Any,
|
||||||
|
@ -185,7 +186,7 @@ class PyJWT:
|
||||||
def decode(
|
def decode(
|
||||||
self,
|
self,
|
||||||
jwt: str | bytes,
|
jwt: str | bytes,
|
||||||
key: AllowedPublicKeys | str | bytes = "",
|
key: AllowedPublicKeys | PyJWK | str | bytes = "",
|
||||||
algorithms: list[str] | None = None,
|
algorithms: list[str] | None = None,
|
||||||
options: dict[str, Any] | None = None,
|
options: dict[str, Any] | None = None,
|
||||||
# deprecated arg, remove in pyjwt3
|
# deprecated arg, remove in pyjwt3
|
||||||
|
@ -195,7 +196,7 @@ class PyJWT:
|
||||||
# passthrough arguments to _validate_claims
|
# passthrough arguments to _validate_claims
|
||||||
# consider putting in options
|
# consider putting in options
|
||||||
audience: str | Iterable[str] | None = None,
|
audience: str | Iterable[str] | None = None,
|
||||||
issuer: str | None = None,
|
issuer: str | List[str] | None = None,
|
||||||
leeway: float | timedelta = 0,
|
leeway: float | timedelta = 0,
|
||||||
# kwargs
|
# kwargs
|
||||||
**kwargs: Any,
|
**kwargs: Any,
|
||||||
|
@ -300,7 +301,7 @@ class PyJWT:
|
||||||
try:
|
try:
|
||||||
exp = int(payload["exp"])
|
exp = int(payload["exp"])
|
||||||
except ValueError:
|
except ValueError:
|
||||||
raise DecodeError("Expiration Time claim (exp) must be an" " integer.")
|
raise DecodeError("Expiration Time claim (exp) must be an integer.")
|
||||||
|
|
||||||
if exp <= (now - leeway):
|
if exp <= (now - leeway):
|
||||||
raise ExpiredSignatureError("Signature has expired")
|
raise ExpiredSignatureError("Signature has expired")
|
||||||
|
@ -362,6 +363,10 @@ class PyJWT:
|
||||||
if "iss" not in payload:
|
if "iss" not in payload:
|
||||||
raise MissingRequiredClaimError("iss")
|
raise MissingRequiredClaimError("iss")
|
||||||
|
|
||||||
|
if isinstance(issuer, list):
|
||||||
|
if payload["iss"] not in issuer:
|
||||||
|
raise InvalidIssuerError("Invalid issuer")
|
||||||
|
else:
|
||||||
if payload["iss"] != issuer:
|
if payload["iss"] != issuer:
|
||||||
raise InvalidIssuerError("Invalid issuer")
|
raise InvalidIssuerError("Invalid issuer")
|
||||||
|
|
||||||
|
|
|
@ -58,6 +58,10 @@ class PyJWKError(PyJWTError):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class MissingCryptographyError(PyJWKError):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
class PyJWKSetError(PyJWTError):
|
class PyJWKSetError(PyJWTError):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
|
@ -131,26 +131,15 @@ def is_pem_format(key: bytes) -> bool:
|
||||||
|
|
||||||
|
|
||||||
# Based on https://github.com/pyca/cryptography/blob/bcb70852d577b3f490f015378c75cba74986297b/src/cryptography/hazmat/primitives/serialization/ssh.py#L40-L46
|
# Based on https://github.com/pyca/cryptography/blob/bcb70852d577b3f490f015378c75cba74986297b/src/cryptography/hazmat/primitives/serialization/ssh.py#L40-L46
|
||||||
_CERT_SUFFIX = b"-cert-v01@openssh.com"
|
_SSH_KEY_FORMATS = (
|
||||||
_SSH_PUBKEY_RC = re.compile(rb"\A(\S+)[ \t]+(\S+)")
|
|
||||||
_SSH_KEY_FORMATS = [
|
|
||||||
b"ssh-ed25519",
|
b"ssh-ed25519",
|
||||||
b"ssh-rsa",
|
b"ssh-rsa",
|
||||||
b"ssh-dss",
|
b"ssh-dss",
|
||||||
b"ecdsa-sha2-nistp256",
|
b"ecdsa-sha2-nistp256",
|
||||||
b"ecdsa-sha2-nistp384",
|
b"ecdsa-sha2-nistp384",
|
||||||
b"ecdsa-sha2-nistp521",
|
b"ecdsa-sha2-nistp521",
|
||||||
]
|
)
|
||||||
|
|
||||||
|
|
||||||
def is_ssh_key(key: bytes) -> bool:
|
def is_ssh_key(key: bytes) -> bool:
|
||||||
if any(string_value in key for string_value in _SSH_KEY_FORMATS):
|
return key.startswith(_SSH_KEY_FORMATS)
|
||||||
return True
|
|
||||||
|
|
||||||
ssh_pubkey_match = _SSH_PUBKEY_RC.match(key)
|
|
||||||
if ssh_pubkey_match:
|
|
||||||
key_type = ssh_pubkey_match.group(1)
|
|
||||||
if _CERT_SUFFIX == key_type[-len(_CERT_SUFFIX) :]:
|
|
||||||
return True
|
|
||||||
|
|
||||||
return False
|
|
||||||
|
|
|
@ -3,4 +3,4 @@
|
||||||
from .more import * # noqa
|
from .more import * # noqa
|
||||||
from .recipes import * # noqa
|
from .recipes import * # noqa
|
||||||
|
|
||||||
__version__ = '10.3.0'
|
__version__ = '10.4.0'
|
||||||
|
|
|
@ -3,8 +3,9 @@ import warnings
|
||||||
|
|
||||||
from collections import Counter, defaultdict, deque, abc
|
from collections import Counter, defaultdict, deque, abc
|
||||||
from collections.abc import Sequence
|
from collections.abc import Sequence
|
||||||
|
from contextlib import suppress
|
||||||
from functools import cached_property, partial, reduce, wraps
|
from functools import cached_property, partial, reduce, wraps
|
||||||
from heapq import heapify, heapreplace, heappop
|
from heapq import heapify, heapreplace
|
||||||
from itertools import (
|
from itertools import (
|
||||||
chain,
|
chain,
|
||||||
combinations,
|
combinations,
|
||||||
|
@ -21,10 +22,10 @@ from itertools import (
|
||||||
zip_longest,
|
zip_longest,
|
||||||
product,
|
product,
|
||||||
)
|
)
|
||||||
from math import comb, e, exp, factorial, floor, fsum, log, perm, tau
|
from math import comb, e, exp, factorial, floor, fsum, log, log1p, perm, tau
|
||||||
from queue import Empty, Queue
|
from queue import Empty, Queue
|
||||||
from random import random, randrange, uniform
|
from random import random, randrange, shuffle, uniform
|
||||||
from operator import itemgetter, mul, sub, gt, lt, ge, le
|
from operator import itemgetter, mul, sub, gt, lt, le
|
||||||
from sys import hexversion, maxsize
|
from sys import hexversion, maxsize
|
||||||
from time import monotonic
|
from time import monotonic
|
||||||
|
|
||||||
|
@ -34,7 +35,6 @@ from .recipes import (
|
||||||
UnequalIterablesError,
|
UnequalIterablesError,
|
||||||
consume,
|
consume,
|
||||||
flatten,
|
flatten,
|
||||||
pairwise,
|
|
||||||
powerset,
|
powerset,
|
||||||
take,
|
take,
|
||||||
unique_everseen,
|
unique_everseen,
|
||||||
|
@ -473,12 +473,10 @@ def ilen(iterable):
|
||||||
This consumes the iterable, so handle with care.
|
This consumes the iterable, so handle with care.
|
||||||
|
|
||||||
"""
|
"""
|
||||||
# This approach was selected because benchmarks showed it's likely the
|
# This is the "most beautiful of the fast variants" of this function.
|
||||||
# fastest of the known implementations at the time of writing.
|
# If you think you can improve on it, please ensure that your version
|
||||||
# See GitHub tracker: #236, #230.
|
# is both 10x faster and 10x more beautiful.
|
||||||
counter = count()
|
return sum(compress(repeat(1), zip(iterable)))
|
||||||
deque(zip(iterable, counter), maxlen=0)
|
|
||||||
return next(counter)
|
|
||||||
|
|
||||||
|
|
||||||
def iterate(func, start):
|
def iterate(func, start):
|
||||||
|
@ -666,9 +664,9 @@ def distinct_permutations(iterable, r=None):
|
||||||
>>> sorted(distinct_permutations([1, 0, 1]))
|
>>> sorted(distinct_permutations([1, 0, 1]))
|
||||||
[(0, 1, 1), (1, 0, 1), (1, 1, 0)]
|
[(0, 1, 1), (1, 0, 1), (1, 1, 0)]
|
||||||
|
|
||||||
Equivalent to ``set(permutations(iterable))``, except duplicates are not
|
Equivalent to yielding from ``set(permutations(iterable))``, except
|
||||||
generated and thrown away. For larger input sequences this is much more
|
duplicates are not generated and thrown away. For larger input sequences
|
||||||
efficient.
|
this is much more efficient.
|
||||||
|
|
||||||
Duplicate permutations arise when there are duplicated elements in the
|
Duplicate permutations arise when there are duplicated elements in the
|
||||||
input iterable. The number of items returned is
|
input iterable. The number of items returned is
|
||||||
|
@ -683,6 +681,25 @@ def distinct_permutations(iterable, r=None):
|
||||||
>>> sorted(distinct_permutations(range(3), r=2))
|
>>> sorted(distinct_permutations(range(3), r=2))
|
||||||
[(0, 1), (0, 2), (1, 0), (1, 2), (2, 0), (2, 1)]
|
[(0, 1), (0, 2), (1, 0), (1, 2), (2, 0), (2, 1)]
|
||||||
|
|
||||||
|
*iterable* need not be sortable, but note that using equal (``x == y``)
|
||||||
|
but non-identical (``id(x) != id(y)``) elements may produce surprising
|
||||||
|
behavior. For example, ``1`` and ``True`` are equal but non-identical:
|
||||||
|
|
||||||
|
>>> list(distinct_permutations([1, True, '3'])) # doctest: +SKIP
|
||||||
|
[
|
||||||
|
(1, True, '3'),
|
||||||
|
(1, '3', True),
|
||||||
|
('3', 1, True)
|
||||||
|
]
|
||||||
|
>>> list(distinct_permutations([1, 2, '3'])) # doctest: +SKIP
|
||||||
|
[
|
||||||
|
(1, 2, '3'),
|
||||||
|
(1, '3', 2),
|
||||||
|
(2, 1, '3'),
|
||||||
|
(2, '3', 1),
|
||||||
|
('3', 1, 2),
|
||||||
|
('3', 2, 1)
|
||||||
|
]
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Algorithm: https://w.wiki/Qai
|
# Algorithm: https://w.wiki/Qai
|
||||||
|
@ -749,14 +766,44 @@ def distinct_permutations(iterable, r=None):
|
||||||
i += 1
|
i += 1
|
||||||
head[i:], tail[:] = tail[: r - i], tail[r - i :]
|
head[i:], tail[:] = tail[: r - i], tail[r - i :]
|
||||||
|
|
||||||
items = sorted(iterable)
|
items = list(iterable)
|
||||||
|
|
||||||
|
try:
|
||||||
|
items.sort()
|
||||||
|
sortable = True
|
||||||
|
except TypeError:
|
||||||
|
sortable = False
|
||||||
|
|
||||||
|
indices_dict = defaultdict(list)
|
||||||
|
|
||||||
|
for item in items:
|
||||||
|
indices_dict[items.index(item)].append(item)
|
||||||
|
|
||||||
|
indices = [items.index(item) for item in items]
|
||||||
|
indices.sort()
|
||||||
|
|
||||||
|
equivalent_items = {k: cycle(v) for k, v in indices_dict.items()}
|
||||||
|
|
||||||
|
def permuted_items(permuted_indices):
|
||||||
|
return tuple(
|
||||||
|
next(equivalent_items[index]) for index in permuted_indices
|
||||||
|
)
|
||||||
|
|
||||||
size = len(items)
|
size = len(items)
|
||||||
if r is None:
|
if r is None:
|
||||||
r = size
|
r = size
|
||||||
|
|
||||||
|
# functools.partial(_partial, ... )
|
||||||
|
algorithm = _full if (r == size) else partial(_partial, r=r)
|
||||||
|
|
||||||
if 0 < r <= size:
|
if 0 < r <= size:
|
||||||
return _full(items) if (r == size) else _partial(items, r)
|
if sortable:
|
||||||
|
return algorithm(items)
|
||||||
|
else:
|
||||||
|
return (
|
||||||
|
permuted_items(permuted_indices)
|
||||||
|
for permuted_indices in algorithm(indices)
|
||||||
|
)
|
||||||
|
|
||||||
return iter(() if r else ((),))
|
return iter(() if r else ((),))
|
||||||
|
|
||||||
|
@ -1743,7 +1790,9 @@ def zip_offset(*iterables, offsets, longest=False, fillvalue=None):
|
||||||
return zip(*staggered)
|
return zip(*staggered)
|
||||||
|
|
||||||
|
|
||||||
def sort_together(iterables, key_list=(0,), key=None, reverse=False):
|
def sort_together(
|
||||||
|
iterables, key_list=(0,), key=None, reverse=False, strict=False
|
||||||
|
):
|
||||||
"""Return the input iterables sorted together, with *key_list* as the
|
"""Return the input iterables sorted together, with *key_list* as the
|
||||||
priority for sorting. All iterables are trimmed to the length of the
|
priority for sorting. All iterables are trimmed to the length of the
|
||||||
shortest one.
|
shortest one.
|
||||||
|
@ -1782,6 +1831,10 @@ def sort_together(iterables, key_list=(0,), key=None, reverse=False):
|
||||||
>>> sort_together([(1, 2, 3), ('c', 'b', 'a')], reverse=True)
|
>>> sort_together([(1, 2, 3), ('c', 'b', 'a')], reverse=True)
|
||||||
[(3, 2, 1), ('a', 'b', 'c')]
|
[(3, 2, 1), ('a', 'b', 'c')]
|
||||||
|
|
||||||
|
If the *strict* keyword argument is ``True``, then
|
||||||
|
``UnequalIterablesError`` will be raised if any of the iterables have
|
||||||
|
different lengths.
|
||||||
|
|
||||||
"""
|
"""
|
||||||
if key is None:
|
if key is None:
|
||||||
# if there is no key function, the key argument to sorted is an
|
# if there is no key function, the key argument to sorted is an
|
||||||
|
@ -1804,8 +1857,9 @@ def sort_together(iterables, key_list=(0,), key=None, reverse=False):
|
||||||
*get_key_items(zipped_items)
|
*get_key_items(zipped_items)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
zipper = zip_equal if strict else zip
|
||||||
return list(
|
return list(
|
||||||
zip(*sorted(zip(*iterables), key=key_argument, reverse=reverse))
|
zipper(*sorted(zipper(*iterables), key=key_argument, reverse=reverse))
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ -2747,8 +2801,6 @@ class seekable:
|
||||||
>>> it.seek(0)
|
>>> it.seek(0)
|
||||||
>>> next(it), next(it), next(it)
|
>>> next(it), next(it), next(it)
|
||||||
('0', '1', '2')
|
('0', '1', '2')
|
||||||
>>> next(it)
|
|
||||||
'3'
|
|
||||||
|
|
||||||
You can also seek forward:
|
You can also seek forward:
|
||||||
|
|
||||||
|
@ -2756,15 +2808,29 @@ class seekable:
|
||||||
>>> it.seek(10)
|
>>> it.seek(10)
|
||||||
>>> next(it)
|
>>> next(it)
|
||||||
'10'
|
'10'
|
||||||
>>> it.relative_seek(-2) # Seeking relative to the current position
|
|
||||||
>>> next(it)
|
|
||||||
'9'
|
|
||||||
>>> it.seek(20) # Seeking past the end of the source isn't a problem
|
>>> it.seek(20) # Seeking past the end of the source isn't a problem
|
||||||
>>> list(it)
|
>>> list(it)
|
||||||
[]
|
[]
|
||||||
>>> it.seek(0) # Resetting works even after hitting the end
|
>>> it.seek(0) # Resetting works even after hitting the end
|
||||||
|
>>> next(it)
|
||||||
|
'0'
|
||||||
|
|
||||||
|
Call :meth:`relative_seek` to seek relative to the source iterator's
|
||||||
|
current position.
|
||||||
|
|
||||||
|
>>> it = seekable((str(n) for n in range(20)))
|
||||||
>>> next(it), next(it), next(it)
|
>>> next(it), next(it), next(it)
|
||||||
('0', '1', '2')
|
('0', '1', '2')
|
||||||
|
>>> it.relative_seek(2)
|
||||||
|
>>> next(it)
|
||||||
|
'5'
|
||||||
|
>>> it.relative_seek(-3) # Source is at '6', we move back to '3'
|
||||||
|
>>> next(it)
|
||||||
|
'3'
|
||||||
|
>>> it.relative_seek(-3) # Source is at '4', we move back to '1'
|
||||||
|
>>> next(it)
|
||||||
|
'1'
|
||||||
|
|
||||||
|
|
||||||
Call :meth:`peek` to look ahead one item without advancing the iterator:
|
Call :meth:`peek` to look ahead one item without advancing the iterator:
|
||||||
|
|
||||||
|
@ -2873,8 +2939,10 @@ class seekable:
|
||||||
consume(self, remainder)
|
consume(self, remainder)
|
||||||
|
|
||||||
def relative_seek(self, count):
|
def relative_seek(self, count):
|
||||||
index = len(self._cache)
|
if self._index is None:
|
||||||
self.seek(max(index + count, 0))
|
self._index = len(self._cache)
|
||||||
|
|
||||||
|
self.seek(max(self._index + count, 0))
|
||||||
|
|
||||||
|
|
||||||
class run_length:
|
class run_length:
|
||||||
|
@ -2903,7 +2971,7 @@ class run_length:
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def decode(iterable):
|
def decode(iterable):
|
||||||
return chain.from_iterable(repeat(k, n) for k, n in iterable)
|
return chain.from_iterable(starmap(repeat, iterable))
|
||||||
|
|
||||||
|
|
||||||
def exactly_n(iterable, n, predicate=bool):
|
def exactly_n(iterable, n, predicate=bool):
|
||||||
|
@ -2924,14 +2992,34 @@ def exactly_n(iterable, n, predicate=bool):
|
||||||
return len(take(n + 1, filter(predicate, iterable))) == n
|
return len(take(n + 1, filter(predicate, iterable))) == n
|
||||||
|
|
||||||
|
|
||||||
def circular_shifts(iterable):
|
def circular_shifts(iterable, steps=1):
|
||||||
"""Return a list of circular shifts of *iterable*.
|
"""Yield the circular shifts of *iterable*.
|
||||||
|
|
||||||
>>> circular_shifts(range(4))
|
>>> list(circular_shifts(range(4)))
|
||||||
[(0, 1, 2, 3), (1, 2, 3, 0), (2, 3, 0, 1), (3, 0, 1, 2)]
|
[(0, 1, 2, 3), (1, 2, 3, 0), (2, 3, 0, 1), (3, 0, 1, 2)]
|
||||||
|
|
||||||
|
Set *steps* to the number of places to rotate to the left
|
||||||
|
(or to the right if negative). Defaults to 1.
|
||||||
|
|
||||||
|
>>> list(circular_shifts(range(4), 2))
|
||||||
|
[(0, 1, 2, 3), (2, 3, 0, 1)]
|
||||||
|
|
||||||
|
>>> list(circular_shifts(range(4), -1))
|
||||||
|
[(0, 1, 2, 3), (3, 0, 1, 2), (2, 3, 0, 1), (1, 2, 3, 0)]
|
||||||
|
|
||||||
"""
|
"""
|
||||||
lst = list(iterable)
|
buffer = deque(iterable)
|
||||||
return take(len(lst), windowed(cycle(lst), len(lst)))
|
if steps == 0:
|
||||||
|
raise ValueError('Steps should be a non-zero integer')
|
||||||
|
|
||||||
|
buffer.rotate(steps)
|
||||||
|
steps = -steps
|
||||||
|
n = len(buffer)
|
||||||
|
n //= math.gcd(n, steps)
|
||||||
|
|
||||||
|
for __ in repeat(None, n):
|
||||||
|
buffer.rotate(steps)
|
||||||
|
yield tuple(buffer)
|
||||||
|
|
||||||
|
|
||||||
def make_decorator(wrapping_func, result_index=0):
|
def make_decorator(wrapping_func, result_index=0):
|
||||||
|
@ -3191,7 +3279,7 @@ def partitions(iterable):
|
||||||
yield [sequence[i:j] for i, j in zip((0,) + i, i + (n,))]
|
yield [sequence[i:j] for i, j in zip((0,) + i, i + (n,))]
|
||||||
|
|
||||||
|
|
||||||
def set_partitions(iterable, k=None):
|
def set_partitions(iterable, k=None, min_size=None, max_size=None):
|
||||||
"""
|
"""
|
||||||
Yield the set partitions of *iterable* into *k* parts. Set partitions are
|
Yield the set partitions of *iterable* into *k* parts. Set partitions are
|
||||||
not order-preserving.
|
not order-preserving.
|
||||||
|
@ -3215,6 +3303,20 @@ def set_partitions(iterable, k=None):
|
||||||
['b', 'ac']
|
['b', 'ac']
|
||||||
['a', 'b', 'c']
|
['a', 'b', 'c']
|
||||||
|
|
||||||
|
if *min_size* and/or *max_size* are given, the minimum and/or maximum size
|
||||||
|
per block in partition is set.
|
||||||
|
|
||||||
|
>>> iterable = 'abc'
|
||||||
|
>>> for part in set_partitions(iterable, min_size=2):
|
||||||
|
... print([''.join(p) for p in part])
|
||||||
|
['abc']
|
||||||
|
>>> for part in set_partitions(iterable, max_size=2):
|
||||||
|
... print([''.join(p) for p in part])
|
||||||
|
['a', 'bc']
|
||||||
|
['ab', 'c']
|
||||||
|
['b', 'ac']
|
||||||
|
['a', 'b', 'c']
|
||||||
|
|
||||||
"""
|
"""
|
||||||
L = list(iterable)
|
L = list(iterable)
|
||||||
n = len(L)
|
n = len(L)
|
||||||
|
@ -3226,6 +3328,11 @@ def set_partitions(iterable, k=None):
|
||||||
elif k > n:
|
elif k > n:
|
||||||
return
|
return
|
||||||
|
|
||||||
|
min_size = min_size if min_size is not None else 0
|
||||||
|
max_size = max_size if max_size is not None else n
|
||||||
|
if min_size > max_size:
|
||||||
|
return
|
||||||
|
|
||||||
def set_partitions_helper(L, k):
|
def set_partitions_helper(L, k):
|
||||||
n = len(L)
|
n = len(L)
|
||||||
if k == 1:
|
if k == 1:
|
||||||
|
@ -3242,9 +3349,15 @@ def set_partitions(iterable, k=None):
|
||||||
|
|
||||||
if k is None:
|
if k is None:
|
||||||
for k in range(1, n + 1):
|
for k in range(1, n + 1):
|
||||||
yield from set_partitions_helper(L, k)
|
yield from filter(
|
||||||
|
lambda z: all(min_size <= len(bk) <= max_size for bk in z),
|
||||||
|
set_partitions_helper(L, k),
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
yield from set_partitions_helper(L, k)
|
yield from filter(
|
||||||
|
lambda z: all(min_size <= len(bk) <= max_size for bk in z),
|
||||||
|
set_partitions_helper(L, k),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class time_limited:
|
class time_limited:
|
||||||
|
@ -3535,32 +3648,27 @@ def map_if(iterable, pred, func, func_else=lambda x: x):
|
||||||
yield func(item) if pred(item) else func_else(item)
|
yield func(item) if pred(item) else func_else(item)
|
||||||
|
|
||||||
|
|
||||||
def _sample_unweighted(iterable, k):
|
def _sample_unweighted(iterator, k, strict):
|
||||||
# Implementation of "Algorithm L" from the 1994 paper by Kim-Hung Li:
|
# Algorithm L in the 1994 paper by Kim-Hung Li:
|
||||||
# "Reservoir-Sampling Algorithms of Time Complexity O(n(1+log(N/n)))".
|
# "Reservoir-Sampling Algorithms of Time Complexity O(n(1+log(N/n)))".
|
||||||
|
|
||||||
# Fill up the reservoir (collection of samples) with the first `k` samples
|
reservoir = list(islice(iterator, k))
|
||||||
reservoir = take(k, iterable)
|
if strict and len(reservoir) < k:
|
||||||
|
raise ValueError('Sample larger than population')
|
||||||
|
W = 1.0
|
||||||
|
|
||||||
# Generate random number that's the largest in a sample of k U(0,1) numbers
|
with suppress(StopIteration):
|
||||||
# Largest order statistic: https://en.wikipedia.org/wiki/Order_statistic
|
while True:
|
||||||
W = exp(log(random()) / k)
|
|
||||||
|
|
||||||
# The number of elements to skip before changing the reservoir is a random
|
|
||||||
# number with a geometric distribution. Sample it using random() and logs.
|
|
||||||
next_index = k + floor(log(random()) / log(1 - W))
|
|
||||||
|
|
||||||
for index, element in enumerate(iterable, k):
|
|
||||||
if index == next_index:
|
|
||||||
reservoir[randrange(k)] = element
|
|
||||||
# The new W is the largest in a sample of k U(0, `old_W`) numbers
|
|
||||||
W *= exp(log(random()) / k)
|
W *= exp(log(random()) / k)
|
||||||
next_index += floor(log(random()) / log(1 - W)) + 1
|
skip = floor(log(random()) / log1p(-W))
|
||||||
|
element = next(islice(iterator, skip, None))
|
||||||
|
reservoir[randrange(k)] = element
|
||||||
|
|
||||||
|
shuffle(reservoir)
|
||||||
return reservoir
|
return reservoir
|
||||||
|
|
||||||
|
|
||||||
def _sample_weighted(iterable, k, weights):
|
def _sample_weighted(iterator, k, weights, strict):
|
||||||
# Implementation of "A-ExpJ" from the 2006 paper by Efraimidis et al. :
|
# Implementation of "A-ExpJ" from the 2006 paper by Efraimidis et al. :
|
||||||
# "Weighted random sampling with a reservoir".
|
# "Weighted random sampling with a reservoir".
|
||||||
|
|
||||||
|
@ -3569,7 +3677,10 @@ def _sample_weighted(iterable, k, weights):
|
||||||
|
|
||||||
# Fill up the reservoir (collection of samples) with the first `k`
|
# Fill up the reservoir (collection of samples) with the first `k`
|
||||||
# weight-keys and elements, then heapify the list.
|
# weight-keys and elements, then heapify the list.
|
||||||
reservoir = take(k, zip(weight_keys, iterable))
|
reservoir = take(k, zip(weight_keys, iterator))
|
||||||
|
if strict and len(reservoir) < k:
|
||||||
|
raise ValueError('Sample larger than population')
|
||||||
|
|
||||||
heapify(reservoir)
|
heapify(reservoir)
|
||||||
|
|
||||||
# The number of jumps before changing the reservoir is a random variable
|
# The number of jumps before changing the reservoir is a random variable
|
||||||
|
@ -3577,7 +3688,7 @@ def _sample_weighted(iterable, k, weights):
|
||||||
smallest_weight_key, _ = reservoir[0]
|
smallest_weight_key, _ = reservoir[0]
|
||||||
weights_to_skip = log(random()) / smallest_weight_key
|
weights_to_skip = log(random()) / smallest_weight_key
|
||||||
|
|
||||||
for weight, element in zip(weights, iterable):
|
for weight, element in zip(weights, iterator):
|
||||||
if weight >= weights_to_skip:
|
if weight >= weights_to_skip:
|
||||||
# The notation here is consistent with the paper, but we store
|
# The notation here is consistent with the paper, but we store
|
||||||
# the weight-keys in log-space for better numerical stability.
|
# the weight-keys in log-space for better numerical stability.
|
||||||
|
@ -3591,44 +3702,103 @@ def _sample_weighted(iterable, k, weights):
|
||||||
else:
|
else:
|
||||||
weights_to_skip -= weight
|
weights_to_skip -= weight
|
||||||
|
|
||||||
# Equivalent to [element for weight_key, element in sorted(reservoir)]
|
ret = [element for weight_key, element in reservoir]
|
||||||
return [heappop(reservoir)[1] for _ in range(k)]
|
shuffle(ret)
|
||||||
|
return ret
|
||||||
|
|
||||||
|
|
||||||
def sample(iterable, k, weights=None):
|
def _sample_counted(population, k, counts, strict):
|
||||||
|
element = None
|
||||||
|
remaining = 0
|
||||||
|
|
||||||
|
def feed(i):
|
||||||
|
# Advance *i* steps ahead and consume an element
|
||||||
|
nonlocal element, remaining
|
||||||
|
|
||||||
|
while i + 1 > remaining:
|
||||||
|
i = i - remaining
|
||||||
|
element = next(population)
|
||||||
|
remaining = next(counts)
|
||||||
|
remaining -= i + 1
|
||||||
|
return element
|
||||||
|
|
||||||
|
with suppress(StopIteration):
|
||||||
|
reservoir = []
|
||||||
|
for _ in range(k):
|
||||||
|
reservoir.append(feed(0))
|
||||||
|
if strict and len(reservoir) < k:
|
||||||
|
raise ValueError('Sample larger than population')
|
||||||
|
|
||||||
|
W = 1.0
|
||||||
|
while True:
|
||||||
|
W *= exp(log(random()) / k)
|
||||||
|
skip = floor(log(random()) / log1p(-W))
|
||||||
|
element = feed(skip)
|
||||||
|
reservoir[randrange(k)] = element
|
||||||
|
|
||||||
|
shuffle(reservoir)
|
||||||
|
return reservoir
|
||||||
|
|
||||||
|
|
||||||
|
def sample(iterable, k, weights=None, *, counts=None, strict=False):
|
||||||
"""Return a *k*-length list of elements chosen (without replacement)
|
"""Return a *k*-length list of elements chosen (without replacement)
|
||||||
from the *iterable*. Like :func:`random.sample`, but works on iterables
|
from the *iterable*. Similar to :func:`random.sample`, but works on
|
||||||
of unknown length.
|
iterables of unknown length.
|
||||||
|
|
||||||
>>> iterable = range(100)
|
>>> iterable = range(100)
|
||||||
>>> sample(iterable, 5) # doctest: +SKIP
|
>>> sample(iterable, 5) # doctest: +SKIP
|
||||||
[81, 60, 96, 16, 4]
|
[81, 60, 96, 16, 4]
|
||||||
|
|
||||||
An iterable with *weights* may also be given:
|
For iterables with repeated elements, you may supply *counts* to
|
||||||
|
indicate the repeats.
|
||||||
|
|
||||||
|
>>> iterable = ['a', 'b']
|
||||||
|
>>> counts = [3, 4] # Equivalent to 'a', 'a', 'a', 'b', 'b', 'b', 'b'
|
||||||
|
>>> sample(iterable, k=3, counts=counts) # doctest: +SKIP
|
||||||
|
['a', 'a', 'b']
|
||||||
|
|
||||||
|
An iterable with *weights* may be given:
|
||||||
|
|
||||||
>>> iterable = range(100)
|
>>> iterable = range(100)
|
||||||
>>> weights = (i * i + 1 for i in range(100))
|
>>> weights = (i * i + 1 for i in range(100))
|
||||||
>>> sampled = sample(iterable, 5, weights=weights) # doctest: +SKIP
|
>>> sampled = sample(iterable, 5, weights=weights) # doctest: +SKIP
|
||||||
[79, 67, 74, 66, 78]
|
[79, 67, 74, 66, 78]
|
||||||
|
|
||||||
The algorithm can also be used to generate weighted random permutations.
|
Weighted selections are made without replacement.
|
||||||
The relative weight of each item determines the probability that it
|
After an element is selected, it is removed from the pool and the
|
||||||
appears late in the permutation.
|
relative weights of the other elements increase (this
|
||||||
|
does not match the behavior of :func:`random.sample`'s *counts*
|
||||||
|
parameter). Note that *weights* may not be used with *counts*.
|
||||||
|
|
||||||
>>> data = "abcdefgh"
|
If the length of *iterable* is less than *k*,
|
||||||
>>> weights = range(1, len(data) + 1)
|
``ValueError`` is raised if *strict* is ``True`` and
|
||||||
>>> sample(data, k=len(data), weights=weights) # doctest: +SKIP
|
all elements are returned (in shuffled order) if *strict* is ``False``.
|
||||||
['c', 'a', 'b', 'e', 'g', 'd', 'h', 'f']
|
|
||||||
|
By default, the `Algorithm L <https://w.wiki/ANrM>`__ reservoir sampling
|
||||||
|
technique is used. When *weights* are provided,
|
||||||
|
`Algorithm A-ExpJ <https://w.wiki/ANrS>`__ is used.
|
||||||
"""
|
"""
|
||||||
|
iterator = iter(iterable)
|
||||||
|
|
||||||
|
if k < 0:
|
||||||
|
raise ValueError('k must be non-negative')
|
||||||
|
|
||||||
if k == 0:
|
if k == 0:
|
||||||
return []
|
return []
|
||||||
|
|
||||||
iterable = iter(iterable)
|
if weights is not None and counts is not None:
|
||||||
if weights is None:
|
raise TypeError('weights and counts are mutally exclusive')
|
||||||
return _sample_unweighted(iterable, k)
|
|
||||||
else:
|
elif weights is not None:
|
||||||
weights = iter(weights)
|
weights = iter(weights)
|
||||||
return _sample_weighted(iterable, k, weights)
|
return _sample_weighted(iterator, k, weights, strict)
|
||||||
|
|
||||||
|
elif counts is not None:
|
||||||
|
counts = iter(counts)
|
||||||
|
return _sample_counted(iterator, k, counts, strict)
|
||||||
|
|
||||||
|
else:
|
||||||
|
return _sample_unweighted(iterator, k, strict)
|
||||||
|
|
||||||
|
|
||||||
def is_sorted(iterable, key=None, reverse=False, strict=False):
|
def is_sorted(iterable, key=None, reverse=False, strict=False):
|
||||||
|
@ -3650,12 +3820,16 @@ def is_sorted(iterable, key=None, reverse=False, strict=False):
|
||||||
False
|
False
|
||||||
|
|
||||||
The function returns ``False`` after encountering the first out-of-order
|
The function returns ``False`` after encountering the first out-of-order
|
||||||
item. If there are no out-of-order items, the iterable is exhausted.
|
item, which means it may produce results that differ from the built-in
|
||||||
|
:func:`sorted` function for objects with unusual comparison dynamics.
|
||||||
|
If there are no out-of-order items, the iterable is exhausted.
|
||||||
"""
|
"""
|
||||||
|
compare = le if strict else lt
|
||||||
|
it = iterable if (key is None) else map(key, iterable)
|
||||||
|
it_1, it_2 = tee(it)
|
||||||
|
next(it_2 if reverse else it_1, None)
|
||||||
|
|
||||||
compare = (le if reverse else ge) if strict else (lt if reverse else gt)
|
return not any(map(compare, it_1, it_2))
|
||||||
it = iterable if key is None else map(key, iterable)
|
|
||||||
return not any(starmap(compare, pairwise(it)))
|
|
||||||
|
|
||||||
|
|
||||||
class AbortThread(BaseException):
|
class AbortThread(BaseException):
|
||||||
|
|
|
@ -2,6 +2,8 @@
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import sys
|
||||||
|
|
||||||
from types import TracebackType
|
from types import TracebackType
|
||||||
from typing import (
|
from typing import (
|
||||||
Any,
|
Any,
|
||||||
|
@ -28,6 +30,9 @@ from typing_extensions import Protocol
|
||||||
_T = TypeVar('_T')
|
_T = TypeVar('_T')
|
||||||
_T1 = TypeVar('_T1')
|
_T1 = TypeVar('_T1')
|
||||||
_T2 = TypeVar('_T2')
|
_T2 = TypeVar('_T2')
|
||||||
|
_T3 = TypeVar('_T3')
|
||||||
|
_T4 = TypeVar('_T4')
|
||||||
|
_T5 = TypeVar('_T5')
|
||||||
_U = TypeVar('_U')
|
_U = TypeVar('_U')
|
||||||
_V = TypeVar('_V')
|
_V = TypeVar('_V')
|
||||||
_W = TypeVar('_W')
|
_W = TypeVar('_W')
|
||||||
|
@ -35,6 +40,12 @@ _T_co = TypeVar('_T_co', covariant=True)
|
||||||
_GenFn = TypeVar('_GenFn', bound=Callable[..., Iterator[Any]])
|
_GenFn = TypeVar('_GenFn', bound=Callable[..., Iterator[Any]])
|
||||||
_Raisable = BaseException | Type[BaseException]
|
_Raisable = BaseException | Type[BaseException]
|
||||||
|
|
||||||
|
# The type of isinstance's second argument (from typeshed builtins)
|
||||||
|
if sys.version_info >= (3, 10):
|
||||||
|
_ClassInfo = type | UnionType | tuple[_ClassInfo, ...]
|
||||||
|
else:
|
||||||
|
_ClassInfo = type | tuple[_ClassInfo, ...]
|
||||||
|
|
||||||
@type_check_only
|
@type_check_only
|
||||||
class _SizedIterable(Protocol[_T_co], Sized, Iterable[_T_co]): ...
|
class _SizedIterable(Protocol[_T_co], Sized, Iterable[_T_co]): ...
|
||||||
|
|
||||||
|
@ -135,7 +146,7 @@ def interleave_evenly(
|
||||||
) -> Iterator[_T]: ...
|
) -> Iterator[_T]: ...
|
||||||
def collapse(
|
def collapse(
|
||||||
iterable: Iterable[Any],
|
iterable: Iterable[Any],
|
||||||
base_type: type | None = ...,
|
base_type: _ClassInfo | None = ...,
|
||||||
levels: int | None = ...,
|
levels: int | None = ...,
|
||||||
) -> Iterator[Any]: ...
|
) -> Iterator[Any]: ...
|
||||||
@overload
|
@overload
|
||||||
|
@ -213,6 +224,7 @@ def stagger(
|
||||||
class UnequalIterablesError(ValueError):
|
class UnequalIterablesError(ValueError):
|
||||||
def __init__(self, details: tuple[int, int, int] | None = ...) -> None: ...
|
def __init__(self, details: tuple[int, int, int] | None = ...) -> None: ...
|
||||||
|
|
||||||
|
# zip_equal
|
||||||
@overload
|
@overload
|
||||||
def zip_equal(__iter1: Iterable[_T1]) -> Iterator[tuple[_T1]]: ...
|
def zip_equal(__iter1: Iterable[_T1]) -> Iterator[tuple[_T1]]: ...
|
||||||
@overload
|
@overload
|
||||||
|
@ -221,11 +233,35 @@ def zip_equal(
|
||||||
) -> Iterator[tuple[_T1, _T2]]: ...
|
) -> Iterator[tuple[_T1, _T2]]: ...
|
||||||
@overload
|
@overload
|
||||||
def zip_equal(
|
def zip_equal(
|
||||||
__iter1: Iterable[_T],
|
__iter1: Iterable[_T1], __iter2: Iterable[_T2], __iter3: Iterable[_T3]
|
||||||
__iter2: Iterable[_T],
|
) -> Iterator[tuple[_T1, _T2, _T3]]: ...
|
||||||
__iter3: Iterable[_T],
|
@overload
|
||||||
*iterables: Iterable[_T],
|
def zip_equal(
|
||||||
) -> Iterator[tuple[_T, ...]]: ...
|
__iter1: Iterable[_T1],
|
||||||
|
__iter2: Iterable[_T2],
|
||||||
|
__iter3: Iterable[_T3],
|
||||||
|
__iter4: Iterable[_T4],
|
||||||
|
) -> Iterator[tuple[_T1, _T2, _T3, _T4]]: ...
|
||||||
|
@overload
|
||||||
|
def zip_equal(
|
||||||
|
__iter1: Iterable[_T1],
|
||||||
|
__iter2: Iterable[_T2],
|
||||||
|
__iter3: Iterable[_T3],
|
||||||
|
__iter4: Iterable[_T4],
|
||||||
|
__iter5: Iterable[_T5],
|
||||||
|
) -> Iterator[tuple[_T1, _T2, _T3, _T4, _T5]]: ...
|
||||||
|
@overload
|
||||||
|
def zip_equal(
|
||||||
|
__iter1: Iterable[Any],
|
||||||
|
__iter2: Iterable[Any],
|
||||||
|
__iter3: Iterable[Any],
|
||||||
|
__iter4: Iterable[Any],
|
||||||
|
__iter5: Iterable[Any],
|
||||||
|
__iter6: Iterable[Any],
|
||||||
|
*iterables: Iterable[Any],
|
||||||
|
) -> Iterator[tuple[Any, ...]]: ...
|
||||||
|
|
||||||
|
# zip_offset
|
||||||
@overload
|
@overload
|
||||||
def zip_offset(
|
def zip_offset(
|
||||||
__iter1: Iterable[_T1],
|
__iter1: Iterable[_T1],
|
||||||
|
@ -285,12 +321,13 @@ def sort_together(
|
||||||
key_list: Iterable[int] = ...,
|
key_list: Iterable[int] = ...,
|
||||||
key: Callable[..., Any] | None = ...,
|
key: Callable[..., Any] | None = ...,
|
||||||
reverse: bool = ...,
|
reverse: bool = ...,
|
||||||
|
strict: bool = ...,
|
||||||
) -> list[tuple[_T, ...]]: ...
|
) -> list[tuple[_T, ...]]: ...
|
||||||
def unzip(iterable: Iterable[Sequence[_T]]) -> tuple[Iterator[_T], ...]: ...
|
def unzip(iterable: Iterable[Sequence[_T]]) -> tuple[Iterator[_T], ...]: ...
|
||||||
def divide(n: int, iterable: Iterable[_T]) -> list[Iterator[_T]]: ...
|
def divide(n: int, iterable: Iterable[_T]) -> list[Iterator[_T]]: ...
|
||||||
def always_iterable(
|
def always_iterable(
|
||||||
obj: object,
|
obj: object,
|
||||||
base_type: type | tuple[type | tuple[Any, ...], ...] | None = ...,
|
base_type: _ClassInfo | None = ...,
|
||||||
) -> Iterator[Any]: ...
|
) -> Iterator[Any]: ...
|
||||||
def adjacent(
|
def adjacent(
|
||||||
predicate: Callable[[_T], bool],
|
predicate: Callable[[_T], bool],
|
||||||
|
@ -454,7 +491,9 @@ class run_length:
|
||||||
def exactly_n(
|
def exactly_n(
|
||||||
iterable: Iterable[_T], n: int, predicate: Callable[[_T], object] = ...
|
iterable: Iterable[_T], n: int, predicate: Callable[[_T], object] = ...
|
||||||
) -> bool: ...
|
) -> bool: ...
|
||||||
def circular_shifts(iterable: Iterable[_T]) -> list[tuple[_T, ...]]: ...
|
def circular_shifts(
|
||||||
|
iterable: Iterable[_T], steps: int = 1
|
||||||
|
) -> list[tuple[_T, ...]]: ...
|
||||||
def make_decorator(
|
def make_decorator(
|
||||||
wrapping_func: Callable[..., _U], result_index: int = ...
|
wrapping_func: Callable[..., _U], result_index: int = ...
|
||||||
) -> Callable[..., Callable[[Callable[..., Any]], Callable[..., _U]]]: ...
|
) -> Callable[..., Callable[[Callable[..., Any]], Callable[..., _U]]]: ...
|
||||||
|
@ -500,7 +539,10 @@ def replace(
|
||||||
) -> Iterator[_T | _U]: ...
|
) -> Iterator[_T | _U]: ...
|
||||||
def partitions(iterable: Iterable[_T]) -> Iterator[list[list[_T]]]: ...
|
def partitions(iterable: Iterable[_T]) -> Iterator[list[list[_T]]]: ...
|
||||||
def set_partitions(
|
def set_partitions(
|
||||||
iterable: Iterable[_T], k: int | None = ...
|
iterable: Iterable[_T],
|
||||||
|
k: int | None = ...,
|
||||||
|
min_size: int | None = ...,
|
||||||
|
max_size: int | None = ...,
|
||||||
) -> Iterator[list[list[_T]]]: ...
|
) -> Iterator[list[list[_T]]]: ...
|
||||||
|
|
||||||
class time_limited(Generic[_T], Iterator[_T]):
|
class time_limited(Generic[_T], Iterator[_T]):
|
||||||
|
@ -538,10 +580,22 @@ def map_if(
|
||||||
func: Callable[[Any], Any],
|
func: Callable[[Any], Any],
|
||||||
func_else: Callable[[Any], Any] | None = ...,
|
func_else: Callable[[Any], Any] | None = ...,
|
||||||
) -> Iterator[Any]: ...
|
) -> Iterator[Any]: ...
|
||||||
|
def _sample_unweighted(
|
||||||
|
iterator: Iterator[_T], k: int, strict: bool
|
||||||
|
) -> list[_T]: ...
|
||||||
|
def _sample_counted(
|
||||||
|
population: Iterator[_T], k: int, counts: Iterable[int], strict: bool
|
||||||
|
) -> list[_T]: ...
|
||||||
|
def _sample_weighted(
|
||||||
|
iterator: Iterator[_T], k: int, weights, strict
|
||||||
|
) -> list[_T]: ...
|
||||||
def sample(
|
def sample(
|
||||||
iterable: Iterable[_T],
|
iterable: Iterable[_T],
|
||||||
k: int,
|
k: int,
|
||||||
weights: Iterable[float] | None = ...,
|
weights: Iterable[float] | None = ...,
|
||||||
|
*,
|
||||||
|
counts: Iterable[int] | None = ...,
|
||||||
|
strict: bool = False,
|
||||||
) -> list[_T]: ...
|
) -> list[_T]: ...
|
||||||
def is_sorted(
|
def is_sorted(
|
||||||
iterable: Iterable[_T],
|
iterable: Iterable[_T],
|
||||||
|
@ -577,7 +631,7 @@ class callback_iter(Generic[_T], Iterator[_T]):
|
||||||
|
|
||||||
def windowed_complete(
|
def windowed_complete(
|
||||||
iterable: Iterable[_T], n: int
|
iterable: Iterable[_T], n: int
|
||||||
) -> Iterator[tuple[_T, ...]]: ...
|
) -> Iterator[tuple[tuple[_T, ...], tuple[_T, ...], tuple[_T, ...]]]: ...
|
||||||
def all_unique(
|
def all_unique(
|
||||||
iterable: Iterable[_T], key: Callable[[_T], _U] | None = ...
|
iterable: Iterable[_T], key: Callable[[_T], _U] | None = ...
|
||||||
) -> bool: ...
|
) -> bool: ...
|
||||||
|
@ -608,9 +662,61 @@ class countable(Generic[_T], Iterator[_T]):
|
||||||
items_seen: int
|
items_seen: int
|
||||||
|
|
||||||
def chunked_even(iterable: Iterable[_T], n: int) -> Iterator[list[_T]]: ...
|
def chunked_even(iterable: Iterable[_T], n: int) -> Iterator[list[_T]]: ...
|
||||||
|
@overload
|
||||||
def zip_broadcast(
|
def zip_broadcast(
|
||||||
|
__obj1: _T | Iterable[_T],
|
||||||
|
*,
|
||||||
|
scalar_types: _ClassInfo | None = ...,
|
||||||
|
strict: bool = ...,
|
||||||
|
) -> Iterable[tuple[_T, ...]]: ...
|
||||||
|
@overload
|
||||||
|
def zip_broadcast(
|
||||||
|
__obj1: _T | Iterable[_T],
|
||||||
|
__obj2: _T | Iterable[_T],
|
||||||
|
*,
|
||||||
|
scalar_types: _ClassInfo | None = ...,
|
||||||
|
strict: bool = ...,
|
||||||
|
) -> Iterable[tuple[_T, ...]]: ...
|
||||||
|
@overload
|
||||||
|
def zip_broadcast(
|
||||||
|
__obj1: _T | Iterable[_T],
|
||||||
|
__obj2: _T | Iterable[_T],
|
||||||
|
__obj3: _T | Iterable[_T],
|
||||||
|
*,
|
||||||
|
scalar_types: _ClassInfo | None = ...,
|
||||||
|
strict: bool = ...,
|
||||||
|
) -> Iterable[tuple[_T, ...]]: ...
|
||||||
|
@overload
|
||||||
|
def zip_broadcast(
|
||||||
|
__obj1: _T | Iterable[_T],
|
||||||
|
__obj2: _T | Iterable[_T],
|
||||||
|
__obj3: _T | Iterable[_T],
|
||||||
|
__obj4: _T | Iterable[_T],
|
||||||
|
*,
|
||||||
|
scalar_types: _ClassInfo | None = ...,
|
||||||
|
strict: bool = ...,
|
||||||
|
) -> Iterable[tuple[_T, ...]]: ...
|
||||||
|
@overload
|
||||||
|
def zip_broadcast(
|
||||||
|
__obj1: _T | Iterable[_T],
|
||||||
|
__obj2: _T | Iterable[_T],
|
||||||
|
__obj3: _T | Iterable[_T],
|
||||||
|
__obj4: _T | Iterable[_T],
|
||||||
|
__obj5: _T | Iterable[_T],
|
||||||
|
*,
|
||||||
|
scalar_types: _ClassInfo | None = ...,
|
||||||
|
strict: bool = ...,
|
||||||
|
) -> Iterable[tuple[_T, ...]]: ...
|
||||||
|
@overload
|
||||||
|
def zip_broadcast(
|
||||||
|
__obj1: _T | Iterable[_T],
|
||||||
|
__obj2: _T | Iterable[_T],
|
||||||
|
__obj3: _T | Iterable[_T],
|
||||||
|
__obj4: _T | Iterable[_T],
|
||||||
|
__obj5: _T | Iterable[_T],
|
||||||
|
__obj6: _T | Iterable[_T],
|
||||||
*objects: _T | Iterable[_T],
|
*objects: _T | Iterable[_T],
|
||||||
scalar_types: type | tuple[type | tuple[Any, ...], ...] | None = ...,
|
scalar_types: _ClassInfo | None = ...,
|
||||||
strict: bool = ...,
|
strict: bool = ...,
|
||||||
) -> Iterable[tuple[_T, ...]]: ...
|
) -> Iterable[tuple[_T, ...]]: ...
|
||||||
def unique_in_window(
|
def unique_in_window(
|
||||||
|
|
|
@ -795,8 +795,30 @@ def triplewise(iterable):
|
||||||
[('A', 'B', 'C'), ('B', 'C', 'D'), ('C', 'D', 'E')]
|
[('A', 'B', 'C'), ('B', 'C', 'D'), ('C', 'D', 'E')]
|
||||||
|
|
||||||
"""
|
"""
|
||||||
for (a, _), (b, c) in pairwise(pairwise(iterable)):
|
# This deviates from the itertools documentation reciple - see
|
||||||
yield a, b, c
|
# https://github.com/more-itertools/more-itertools/issues/889
|
||||||
|
t1, t2, t3 = tee(iterable, 3)
|
||||||
|
next(t3, None)
|
||||||
|
next(t3, None)
|
||||||
|
next(t2, None)
|
||||||
|
return zip(t1, t2, t3)
|
||||||
|
|
||||||
|
|
||||||
|
def _sliding_window_islice(iterable, n):
|
||||||
|
# Fast path for small, non-zero values of n.
|
||||||
|
iterators = tee(iterable, n)
|
||||||
|
for i, iterator in enumerate(iterators):
|
||||||
|
next(islice(iterator, i, i), None)
|
||||||
|
return zip(*iterators)
|
||||||
|
|
||||||
|
|
||||||
|
def _sliding_window_deque(iterable, n):
|
||||||
|
# Normal path for other values of n.
|
||||||
|
it = iter(iterable)
|
||||||
|
window = deque(islice(it, n - 1), maxlen=n)
|
||||||
|
for x in it:
|
||||||
|
window.append(x)
|
||||||
|
yield tuple(window)
|
||||||
|
|
||||||
|
|
||||||
def sliding_window(iterable, n):
|
def sliding_window(iterable, n):
|
||||||
|
@ -812,11 +834,16 @@ def sliding_window(iterable, n):
|
||||||
|
|
||||||
For a variant with more features, see :func:`windowed`.
|
For a variant with more features, see :func:`windowed`.
|
||||||
"""
|
"""
|
||||||
it = iter(iterable)
|
if n > 20:
|
||||||
window = deque(islice(it, n - 1), maxlen=n)
|
return _sliding_window_deque(iterable, n)
|
||||||
for x in it:
|
elif n > 2:
|
||||||
window.append(x)
|
return _sliding_window_islice(iterable, n)
|
||||||
yield tuple(window)
|
elif n == 2:
|
||||||
|
return pairwise(iterable)
|
||||||
|
elif n == 1:
|
||||||
|
return zip(iterable)
|
||||||
|
else:
|
||||||
|
raise ValueError(f'n should be at least one, not {n}')
|
||||||
|
|
||||||
|
|
||||||
def subslices(iterable):
|
def subslices(iterable):
|
||||||
|
@ -1038,9 +1065,6 @@ def totient(n):
|
||||||
>>> totient(12)
|
>>> totient(12)
|
||||||
4
|
4
|
||||||
"""
|
"""
|
||||||
# The itertools docs use unique_justseen instead of set; see
|
for prime in set(factor(n)):
|
||||||
# https://github.com/more-itertools/more-itertools/issues/823
|
n -= n // prime
|
||||||
for p in set(factor(n)):
|
|
||||||
n = n // p * (p - 1)
|
|
||||||
|
|
||||||
return n
|
return n
|
||||||
|
|
|
@ -10,6 +10,9 @@ from numbers import Number
|
||||||
from typing import Union, Tuple, Iterable
|
from typing import Union, Tuple, Iterable
|
||||||
from typing import cast
|
from typing import cast
|
||||||
|
|
||||||
|
import dateutil.parser
|
||||||
|
import dateutil.tz
|
||||||
|
|
||||||
|
|
||||||
# some useful constants
|
# some useful constants
|
||||||
osc_per_year = 290_091_329_207_984_000
|
osc_per_year = 290_091_329_207_984_000
|
||||||
|
@ -611,3 +614,40 @@ def date_range(start=None, stop=None, step=None):
|
||||||
while start < stop:
|
while start < stop:
|
||||||
yield start
|
yield start
|
||||||
start += step
|
start += step
|
||||||
|
|
||||||
|
|
||||||
|
tzinfos = dict(
|
||||||
|
AEST=dateutil.tz.gettz("Australia/Sydney"),
|
||||||
|
AEDT=dateutil.tz.gettz("Australia/Sydney"),
|
||||||
|
ACST=dateutil.tz.gettz("Australia/Darwin"),
|
||||||
|
ACDT=dateutil.tz.gettz("Australia/Adelaide"),
|
||||||
|
AWST=dateutil.tz.gettz("Australia/Perth"),
|
||||||
|
EST=dateutil.tz.gettz("America/New_York"),
|
||||||
|
EDT=dateutil.tz.gettz("America/New_York"),
|
||||||
|
CST=dateutil.tz.gettz("America/Chicago"),
|
||||||
|
CDT=dateutil.tz.gettz("America/Chicago"),
|
||||||
|
MST=dateutil.tz.gettz("America/Denver"),
|
||||||
|
MDT=dateutil.tz.gettz("America/Denver"),
|
||||||
|
PST=dateutil.tz.gettz("America/Los_Angeles"),
|
||||||
|
PDT=dateutil.tz.gettz("America/Los_Angeles"),
|
||||||
|
GMT=dateutil.tz.gettz("Etc/GMT"),
|
||||||
|
UTC=dateutil.tz.gettz("UTC"),
|
||||||
|
CET=dateutil.tz.gettz("Europe/Berlin"),
|
||||||
|
CEST=dateutil.tz.gettz("Europe/Berlin"),
|
||||||
|
IST=dateutil.tz.gettz("Asia/Kolkata"),
|
||||||
|
BST=dateutil.tz.gettz("Europe/London"),
|
||||||
|
MSK=dateutil.tz.gettz("Europe/Moscow"),
|
||||||
|
EET=dateutil.tz.gettz("Europe/Helsinki"),
|
||||||
|
EEST=dateutil.tz.gettz("Europe/Helsinki"),
|
||||||
|
# Add more mappings as needed
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def parse(*args, **kwargs):
|
||||||
|
"""
|
||||||
|
Parse the input using dateutil.parser.parse with friendly tz support.
|
||||||
|
|
||||||
|
>>> parse('2024-07-26 12:59:00 EDT')
|
||||||
|
datetime.datetime(...America/New_York...)
|
||||||
|
"""
|
||||||
|
return dateutil.parser.parse(*args, tzinfos=tzinfos, **kwargs)
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
apscheduler==3.10.1
|
apscheduler==3.10.1
|
||||||
importlib-metadata==8.0.0
|
importlib-metadata==8.2.0
|
||||||
importlib-resources==6.4.0
|
importlib-resources==6.4.0
|
||||||
pyinstaller==6.8.0
|
pyinstaller==6.8.0
|
||||||
pyopenssl==24.1.0
|
pyopenssl==24.1.0
|
||||||
|
|
|
@ -1504,6 +1504,18 @@ def dbcheck():
|
||||||
except sqlite3.OperationalError:
|
except sqlite3.OperationalError:
|
||||||
logger.warn("Unable to capitalize Windows platform values in session_history table.")
|
logger.warn("Unable to capitalize Windows platform values in session_history table.")
|
||||||
|
|
||||||
|
# Upgrade session_history table from earlier versions
|
||||||
|
try:
|
||||||
|
result = c_db.execute("SELECT platform FROM session_history "
|
||||||
|
"WHERE platform = 'macos'").fetchall()
|
||||||
|
if len(result) > 0:
|
||||||
|
logger.debug("Altering database. Capitalizing macOS platform values in session_history table.")
|
||||||
|
c_db.execute(
|
||||||
|
"UPDATE session_history SET platform = 'macOS' WHERE platform = 'macos' "
|
||||||
|
)
|
||||||
|
except sqlite3.OperationalError:
|
||||||
|
logger.warn("Unable to capitalize macOS platform values in session_history table.")
|
||||||
|
|
||||||
# Upgrade session_history_metadata table from earlier versions
|
# Upgrade session_history_metadata table from earlier versions
|
||||||
try:
|
try:
|
||||||
c_db.execute("SELECT full_title FROM session_history_metadata")
|
c_db.execute("SELECT full_title FROM session_history_metadata")
|
||||||
|
|
|
@ -102,7 +102,8 @@ PLATFORM_NAME_OVERRIDES = {
|
||||||
'Mystery 5': 'Xbox 360',
|
'Mystery 5': 'Xbox 360',
|
||||||
'WebMAF': 'Playstation 4',
|
'WebMAF': 'Playstation 4',
|
||||||
'windows': 'Windows',
|
'windows': 'Windows',
|
||||||
'osx': 'macOS'
|
'osx': 'macOS',
|
||||||
|
'macos': 'macOS',
|
||||||
}
|
}
|
||||||
|
|
||||||
PMS_PLATFORM_NAME_OVERRIDES = {
|
PMS_PLATFORM_NAME_OVERRIDES = {
|
||||||
|
|
|
@ -6,7 +6,7 @@ bleach==6.1.0
|
||||||
certifi==2024.7.4
|
certifi==2024.7.4
|
||||||
cheroot==10.0.1
|
cheroot==10.0.1
|
||||||
cherrypy==18.10.0
|
cherrypy==18.10.0
|
||||||
cloudinary==1.40.0
|
cloudinary==1.41.0
|
||||||
distro==1.9.0
|
distro==1.9.0
|
||||||
dnspython==2.6.1
|
dnspython==2.6.1
|
||||||
facebook-sdk==3.1.0
|
facebook-sdk==3.1.0
|
||||||
|
@ -16,7 +16,7 @@ gntp==1.0.3
|
||||||
html5lib==1.1
|
html5lib==1.1
|
||||||
httpagentparser==1.9.5
|
httpagentparser==1.9.5
|
||||||
idna==3.7
|
idna==3.7
|
||||||
importlib-metadata==8.0.0
|
importlib-metadata==8.2.0
|
||||||
importlib-resources==6.4.0
|
importlib-resources==6.4.0
|
||||||
git+https://github.com/Tautulli/ipwhois.git@master#egg=ipwhois
|
git+https://github.com/Tautulli/ipwhois.git@master#egg=ipwhois
|
||||||
IPy==1.01
|
IPy==1.01
|
||||||
|
@ -29,7 +29,7 @@ platformdirs==4.2.2
|
||||||
plexapi==4.15.15
|
plexapi==4.15.15
|
||||||
portend==3.2.0
|
portend==3.2.0
|
||||||
profilehooks==1.12.0
|
profilehooks==1.12.0
|
||||||
PyJWT==2.8.0
|
PyJWT==2.9.0
|
||||||
pyparsing==3.1.2
|
pyparsing==3.1.2
|
||||||
python-dateutil==2.9.0.post0
|
python-dateutil==2.9.0.post0
|
||||||
python-twitter==3.5
|
python-twitter==3.5
|
||||||
|
@ -39,7 +39,7 @@ requests-oauthlib==2.0.0
|
||||||
rumps==0.4.0; platform_system == "Darwin"
|
rumps==0.4.0; platform_system == "Darwin"
|
||||||
simplejson==3.19.2
|
simplejson==3.19.2
|
||||||
six==1.16.0
|
six==1.16.0
|
||||||
tempora==5.6.0
|
tempora==5.7.0
|
||||||
tokenize-rt==6.0.0
|
tokenize-rt==6.0.0
|
||||||
tzdata==2024.1
|
tzdata==2024.1
|
||||||
tzlocal==5.0.1
|
tzlocal==5.0.1
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue