cachestore.py 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. # -*- Mode: Python -*-
  2. # GObject-Introspection - a framework for introspecting GObject libraries
  3. # Copyright (C) 2008-2010 Johan Dahlin
  4. #
  5. # This library is free software; you can redistribute it and/or
  6. # modify it under the terms of the GNU Lesser General Public
  7. # License as published by the Free Software Foundation; either
  8. # version 2 of the License, or (at your option) any later version.
  9. #
  10. # This library 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 GNU
  13. # Lesser General Public License for more details.
  14. #
  15. # You should have received a copy of the GNU Lesser General Public
  16. # License along with this library; if not, write to the
  17. # Free Software Foundation, Inc., 59 Temple Place - Suite 330,
  18. # Boston, MA 02111-1307, USA.
  19. #
  20. from __future__ import absolute_import
  21. from __future__ import division
  22. from __future__ import print_function
  23. from __future__ import unicode_literals
  24. import errno
  25. import glob
  26. import hashlib
  27. import os
  28. import shutil
  29. import sys
  30. import tempfile
  31. try:
  32. import cPickle as pickle
  33. except ImportError:
  34. import pickle
  35. import giscanner
  36. from . import utils
  37. _CACHE_VERSION_FILENAME = '.cache-version'
  38. def _get_versionhash():
  39. toplevel = os.path.dirname(giscanner.__file__)
  40. sources = glob.glob(os.path.join(toplevel, '*.py'))
  41. sources.append(sys.argv[0])
  42. # Using mtimes is a bit (5x) faster than hashing the file contents
  43. mtimes = (str(os.stat(source).st_mtime) for source in sources)
  44. # ASCII encoding is sufficient since we are only dealing with numbers.
  45. return hashlib.sha1(''.join(mtimes).encode('ascii')).hexdigest()
  46. class CacheStore(object):
  47. def __init__(self):
  48. self._directory = self._get_cachedir()
  49. self._check_cache_version()
  50. def _get_cachedir(self):
  51. if 'GI_SCANNER_DISABLE_CACHE' in os.environ:
  52. return None
  53. else:
  54. cachedir = utils.get_user_cache_dir('g-ir-scanner')
  55. return cachedir
  56. def _check_cache_version(self):
  57. if self._directory is None:
  58. return
  59. current_hash = _get_versionhash()
  60. version = os.path.join(self._directory, _CACHE_VERSION_FILENAME)
  61. try:
  62. with open(version, 'r') as version_file:
  63. cache_hash = version_file.read()
  64. except IOError as e:
  65. # File does not exist
  66. if e.errno == errno.ENOENT:
  67. cache_hash = 0
  68. else:
  69. raise
  70. if current_hash == cache_hash:
  71. return
  72. self._clean()
  73. tmp_fd, tmp_filename = tempfile.mkstemp(prefix='g-ir-scanner-cache-version-')
  74. try:
  75. with os.fdopen(tmp_fd, 'w') as tmp_file:
  76. tmp_file.write(current_hash)
  77. # On Unix, this would just be os.rename() but Windows
  78. # doesn't allow that.
  79. shutil.move(tmp_filename, version)
  80. except IOError as e:
  81. # Permission denied
  82. if e.errno == errno.EACCES:
  83. return
  84. else:
  85. raise
  86. def _get_filename(self, filename):
  87. # If we couldn't create the directory we're probably
  88. # on a read only home directory where we just disable
  89. # the cache all together.
  90. if self._directory is None:
  91. return
  92. # Assume UTF-8 encoding for the filenames. This doesn't matter so much
  93. # as long as the results of this method always produce the same hash.
  94. hexdigest = hashlib.sha1(filename.encode('utf-8')).hexdigest()
  95. return os.path.join(self._directory, hexdigest)
  96. def _cache_is_valid(self, store_filename, filename):
  97. return (os.stat(store_filename).st_mtime >=
  98. os.stat(filename).st_mtime)
  99. def _remove_filename(self, filename):
  100. try:
  101. os.unlink(filename)
  102. except IOError as e:
  103. # Permission denied
  104. if e.errno == errno.EACCES:
  105. return
  106. else:
  107. raise
  108. except OSError as e:
  109. # File does not exist
  110. if e.errno == errno.ENOENT:
  111. return
  112. else:
  113. raise
  114. def _clean(self):
  115. for filename in os.listdir(self._directory):
  116. if filename == _CACHE_VERSION_FILENAME:
  117. continue
  118. self._remove_filename(os.path.join(self._directory, filename))
  119. def store(self, filename, data):
  120. store_filename = self._get_filename(filename)
  121. if store_filename is None:
  122. return
  123. if (os.path.exists(store_filename) and self._cache_is_valid(store_filename, filename)):
  124. return None
  125. tmp_fd, tmp_filename = tempfile.mkstemp(prefix='g-ir-scanner-cache-')
  126. try:
  127. with os.fdopen(tmp_fd, 'wb') as tmp_file:
  128. pickle.dump(data, tmp_file)
  129. except IOError as e:
  130. # No space left on device
  131. if e.errno == errno.ENOSPC:
  132. self._remove_filename(tmp_filename)
  133. return
  134. else:
  135. raise
  136. try:
  137. shutil.move(tmp_filename, store_filename)
  138. except IOError as e:
  139. # Permission denied
  140. if e.errno == errno.EACCES:
  141. self._remove_filename(tmp_filename)
  142. else:
  143. raise
  144. def load(self, filename):
  145. store_filename = self._get_filename(filename)
  146. if store_filename is None:
  147. return
  148. try:
  149. fd = open(store_filename, 'rb')
  150. except IOError as e:
  151. if e.errno == errno.ENOENT:
  152. return None
  153. else:
  154. raise
  155. if not self._cache_is_valid(store_filename, filename):
  156. return None
  157. try:
  158. data = pickle.load(fd)
  159. except (AttributeError, EOFError, ValueError, pickle.BadPickleGet):
  160. # Broken cache entry, remove it
  161. self._remove_filename(store_filename)
  162. data = None
  163. return data