You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

59 lines
2.0 KiB

  1. ###################################################################################
  2. #
  3. # Copyright (C) 2018 MuK IT GmbH
  4. #
  5. # This program is free software: you can redistribute it and/or modify
  6. # it under the terms of the GNU Affero General Public License as
  7. # published by the Free Software Foundation, either version 3 of the
  8. # License, or (at your option) any later version.
  9. #
  10. # This program is distributed in the hope that it will be useful,
  11. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. # GNU Affero General Public License for more details.
  14. #
  15. # You should have received a copy of the GNU Affero General Public License
  16. # along with this program. If not, see <http://www.gnu.org/licenses/>.
  17. #
  18. ###################################################################################
  19. import time
  20. import logging
  21. import datetime
  22. import functools
  23. _logger = logging.getLogger(__name__)
  24. #----------------------------------------------------------
  25. # Properties
  26. #----------------------------------------------------------
  27. class cached_property(object):
  28. def __init__(self, timeout=None):
  29. self.timeout = timeout
  30. def __call__(self, func):
  31. return functools.update_wrapper(self, func)
  32. def __get__(self, obj, cls):
  33. if obj is None:
  34. return self
  35. try:
  36. value, last_updated = obj.__dict__[self.__name__]
  37. except KeyError:
  38. pass
  39. else:
  40. if self.timeout is None:
  41. return value
  42. elif self.timeout >= time.time() - last_updated:
  43. return value
  44. value = self.__wrapped__(obj)
  45. obj.__dict__[self.__name__] = (value, time.time())
  46. return value
  47. def __delete__(self, obj):
  48. obj.__dict__.pop(self.__name__, None)
  49. def __set__(self, obj, value):
  50. obj.__dict__[self.__name__] = (value, time())