123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159 |
- # Copyright 2016 Julien Danjou
- # Copyright 2016 Joshua Harlow
- # Copyright 2013-2014 Ray Holder
- #
- # Licensed under the Apache License, Version 2.0 (the "License");
- # you may not use this file except in compliance with the License.
- # You may obtain a copy of the License at
- #
- # http://www.apache.org/licenses/LICENSE-2.0
- #
- # Unless required by applicable law or agreed to in writing, software
- # distributed under the License is distributed on an "AS IS" BASIS,
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- # See the License for the specific language governing permissions and
- # limitations under the License.
- import inspect
- import sys
- import time
- from functools import update_wrapper
- from pip._vendor import six
- # sys.maxint / 2, since Python 3.2 doesn't have a sys.maxint...
- try:
- MAX_WAIT = sys.maxint / 2
- except AttributeError:
- MAX_WAIT = 1073741823
- if six.PY2:
- from functools import WRAPPER_ASSIGNMENTS, WRAPPER_UPDATES
- def wraps(fn):
- """Do the same as six.wraps but only copy attributes that exist.
- For example, object instances don't have __name__ attribute, so
- six.wraps fails. This is fixed in Python 3
- (https://bugs.python.org/issue3445), but didn't get backported to six.
- Also, see https://github.com/benjaminp/six/issues/250.
- """
- def filter_hasattr(obj, attrs):
- return tuple(a for a in attrs if hasattr(obj, a))
- return six.wraps(
- fn,
- assigned=filter_hasattr(fn, WRAPPER_ASSIGNMENTS),
- updated=filter_hasattr(fn, WRAPPER_UPDATES),
- )
- def capture(fut, tb):
- # TODO(harlowja): delete this in future, since its
- # has to repeatedly calculate this crap.
- fut.set_exception_info(tb[1], tb[2])
- def getargspec(func):
- # This was deprecated in Python 3.
- return inspect.getargspec(func)
- else:
- from functools import wraps # noqa
- def capture(fut, tb):
- fut.set_exception(tb[1])
- def getargspec(func):
- return inspect.getfullargspec(func)
- def visible_attrs(obj, attrs=None):
- if attrs is None:
- attrs = {}
- for attr_name, attr in inspect.getmembers(obj):
- if attr_name.startswith("_"):
- continue
- attrs[attr_name] = attr
- return attrs
- def find_ordinal(pos_num):
- # See: https://en.wikipedia.org/wiki/English_numerals#Ordinal_numbers
- if pos_num == 0:
- return "th"
- elif pos_num == 1:
- return "st"
- elif pos_num == 2:
- return "nd"
- elif pos_num == 3:
- return "rd"
- elif pos_num >= 4 and pos_num <= 20:
- return "th"
- else:
- return find_ordinal(pos_num % 10)
- def to_ordinal(pos_num):
- return "%i%s" % (pos_num, find_ordinal(pos_num))
- def get_callback_name(cb):
- """Get a callback fully-qualified name.
- If no name can be produced ``repr(cb)`` is called and returned.
- """
- segments = []
- try:
- segments.append(cb.__qualname__)
- except AttributeError:
- try:
- segments.append(cb.__name__)
- if inspect.ismethod(cb):
- try:
- # This attribute doesn't exist on py3.x or newer, so
- # we optionally ignore it... (on those versions of
- # python `__qualname__` should have been found anyway).
- segments.insert(0, cb.im_class.__name__)
- except AttributeError:
- pass
- except AttributeError:
- pass
- if not segments:
- return repr(cb)
- else:
- try:
- # When running under sphinx it appears this can be none?
- if cb.__module__:
- segments.insert(0, cb.__module__)
- except AttributeError:
- pass
- return ".".join(segments)
- try:
- now = time.monotonic # noqa
- except AttributeError:
- from monotonic import monotonic as now # noqa
- class cached_property(object):
- """A property that is computed once per instance.
- Upon being computed it replaces itself with an ordinary attribute. Deleting
- the attribute resets the property.
- Source: https://github.com/bottlepy/bottle/blob/1de24157e74a6971d136550afe1b63eec5b0df2b/bottle.py#L234-L246
- """ # noqa: E501
- def __init__(self, func):
- update_wrapper(self, func)
- self.func = func
- def __get__(self, obj, cls):
- if obj is None:
- return self
- value = obj.__dict__[self.func.__name__] = self.func(obj)
- return value
|