| # 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 |
| |
| 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 |