Repository URL to install this package:
|
Version:
1.0.2-20151008 ▾
|
# -*- Mode: Python -*-
# GObject-Introspection - a framework for introspecting GObject libraries
# Copyright (C) 2008-2010 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 glob
import hashlib
import os
import shutil
import sys
import tempfile
import giscanner
_CACHE_VERSION_FILENAME = '.cache-version'
def _get_versionhash():
toplevel = os.path.dirname(giscanner.__file__)
# Use pyc instead of py to avoid extra IO
sources = glob.glob(os.path.join(toplevel, '*.pyc'))
sources.append(sys.argv[0])
# Using mtimes is a bit (5x) faster than hashing the file contents
mtimes = (str(os.stat(source).st_mtime) for source in sources)
return hashlib.sha1(''.join(mtimes)).hexdigest()
def _get_cachedir():
if 'GI_SCANNER_DISABLE_CACHE' in os.environ:
return None
homedir = os.path.expanduser('~')
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):
try:
os.mkdir(cachedir, 0o755)
except OSError:
return None
scannerdir = os.path.join(cachedir, 'g-ir-scanner')
if not os.path.exists(scannerdir):
try:
os.mkdir(scannerdir, 0o755)
except OSError:
return None
# 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 as e:
if e.errno != errno.EPERM:
raise
self._directory = None
self._check_cache_version()
def _check_cache_version(self):
if self._directory is None:
return
current_hash = _get_versionhash()
version = os.path.join(self._directory, _CACHE_VERSION_FILENAME)
try:
cache_hash = open(version).read()
except IOError as e:
# File does not exist
if e.errno == errno.ENOENT:
cache_hash = 0
else:
raise
if current_hash == cache_hash:
return
versiontmp = version + '.tmp'
self._clean()
try:
fp = open(versiontmp, 'w')
except IOError as e:
# Permission denied
if e.errno == errno.EACCES:
return
else:
raise
fp.write(current_hash)
fp.close()
# On Unix, this would just be os.rename() but Windows
# doesn't allow that.
shutil.move(versiontmp, version)
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 as e:
# Permission denied
if e.errno == errno.EACCES:
return
else:
raise
except OSError as e:
# File does not exist
if e.errno == errno.ENOENT:
return
else:
raise
def _clean(self):
for filename in os.listdir(self._directory):
if filename == _CACHE_VERSION_FILENAME:
continue
self._remove_filename(os.path.join(self._directory, filename))
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 as 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 as 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 as 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