# -*- Mode: Python -*- # GObject-Introspection - a framework for introspecting GObject libraries # Copyright (C) 2008 Johan Dahlin # # This library is free software; you can redistribute it and/or # modify it under the terms of the GNU Lesser General Public # License as published by the Free Software Foundation; either # version 2 of the License, or (at your option) any later version. # # This library is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU # Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public # License along with this library; if not, write to the # Free Software Foundation, Inc., 59 Temple Place - Suite 330, # Boston, MA 02111-1307, USA. # import errno import cPickle import hashlib import os import shutil import tempfile def _get_cachedir(): homedir = os.environ.get('HOME') if homedir is None: return None if not os.path.exists(homedir): return None cachedir = os.path.join(homedir, '.cache') if not os.path.exists(cachedir): os.mkdir(cachedir, 0755) scannerdir = os.path.join(cachedir, 'g-ir-scanner') if not os.path.exists(scannerdir): os.mkdir(scannerdir, 0755) # If it exists and is a file, don't cache at all elif not os.path.isdir(scannerdir): return None return scannerdir class CacheStore(object): def __init__(self): try: self._directory = _get_cachedir() except OSError, e: if e.errno != errno.EPERM: raise self._directory = None def _get_filename(self, filename): # If we couldn't create the directory we're probably # on a read only home directory where we just disable # the cache all together. if self._directory is None: return hexdigest = hashlib.sha1(filename).hexdigest() return os.path.join(self._directory, hexdigest) def _cache_is_valid(self, store_filename, filename): return (os.stat(store_filename).st_mtime >= os.stat(filename).st_mtime) def _remove_filename(self, filename): try: os.unlink(filename) except IOError, e: # Permission denied if e.errno == errno.EACCES: return else: raise except OSError, e: # File does not exist if e.errno == errno.ENOENT: return else: raise def store(self, filename, data): store_filename = self._get_filename(filename) if store_filename is None: return if (os.path.exists(store_filename) and self._cache_is_valid(store_filename, filename)): return None tmp_fd, tmp_filename = tempfile.mkstemp(prefix='g-ir-scanner-cache-') try: cPickle.dump(data, os.fdopen(tmp_fd, 'w')) except IOError, e: # No space left on device if e.errno == errno.ENOSPC: self._remove_filename(tmp_filename) return else: raise try: shutil.move(tmp_filename, store_filename) except IOError, e: # Permission denied if e.errno == errno.EACCES: self._remove_filename(tmp_filename) else: raise def load(self, filename): store_filename = self._get_filename(filename) if store_filename is None: return try: fd = open(store_filename) except IOError, e: if e.errno == errno.ENOENT: return None else: raise if not self._cache_is_valid(store_filename, filename): return None try: data = cPickle.load(fd) except (AttributeError, EOFError, ValueError, cPickle.BadPickleGet): # Broken cache entry, remove it self._remove_filename(store_filename) data = None return data