/usr/share/pyshared/instant/cache.py is in python-instant 1.0.0-1.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 | """This module contains helper functions for working with the module cache.
Example operations:
- modulename = modulename_from_checksum(checksum)
- modulename = modulename_from_checksum(compute_checksum(signature))
- module = import_module_directly(path, modulename)
- module = import_module(modulename)
- module = import_module(checksum)
- module = import_module(compute_checksum(signature))
- modules = cached_modules()
- modules = cached_modules(cache_dir)
"""
import os, sys, re
from output import instant_warning, instant_assert, instant_debug
from paths import get_default_cache_dir, validate_cache_dir
from signatures import compute_checksum
from locking import get_lock, release_lock
# TODO: We could make this an argument, but it's used indirectly several places so take care.
_modulename_prefix = "instant_module_"
def modulename_from_checksum(checksum):
"Construct a module name from a checksum for use in cache."
return _modulename_prefix + checksum
def checksum_from_modulename(modulename):
"Construct a module name from a checksum for use in cache."
return modulename.remove(_modulename_prefix)
def import_module_directly(path, modulename):
"Import a module with the given module name that resides in the given path."
sys.path.insert(0, path)
e = None
try:
module = __import__(modulename)
except Exception, e:
instant_warning("In instant.import_module_directly: Failed to import module '%s' from '%s';\n%s:%s;" % (modulename, path, type(e).__name__, e))
module = None
finally:
sys.path.pop(0)
return module, e
_memory_cache = {}
def memory_cached_module(moduleid):
"Returns the cached module if found."
import sys
module = _memory_cache.get(moduleid, None)
instant_debug("Found '%s' in memory cache with key '%r'." % (module, moduleid))
return module
def place_module_in_memory_cache(moduleid, module):
"Place a compiled module in cache with given id."
_memory_cache[moduleid] = module
instant_debug("Added module '%s' to cache with key '%r'." % (module, moduleid))
def is_valid_module_name(name):
NAMELENGTHLIMIT = 100
return len(name) < NAMELENGTHLIMIT and bool(re.search(r"^[a-zA-Z_][\w]*$", name))
def import_and_cache_module(path, modulename, moduleids):
module, e = import_module_directly(path, modulename)
instant_assert(module is not None, "Failed to import module found in cache. Modulename: '%s';\nPath: '%s';\n%s:%s;" % (modulename, path, type(e).__name__, e))
for moduleid in moduleids:
place_module_in_memory_cache(moduleid, module)
return module
def check_memory_cache(moduleid):
# Check memory cache first with the given moduleid
moduleids = [moduleid]
module = memory_cached_module(moduleid)
if module: return module, moduleids
# Get signature from moduleid if it isn't a string,
# and check memory cache again
if hasattr(moduleid, "signature"):
moduleid = moduleid.signature()
instant_debug("In instant.check_memory_cache: Got signature "\
"'%s' from moduleid.signature()." % moduleid)
module = memory_cached_module(moduleid)
if module:
# FIXME (GNW): I haved commented this out since it can continually
# insert indentical objects into the cache. This
# function checks the cache, so why should it also
# insert?
#for moduleid in moduleids:
# place_module_in_memory_cache(moduleid, module)
return module, moduleids
moduleids.append(moduleid)
# Construct a filename from the checksum of moduleid if it
# isn't already a valid name, and check memory cache again
if not is_valid_module_name(moduleid):
moduleid = modulename_from_checksum(compute_checksum(moduleid))
instant_debug("In instant.check_memory_cache: Constructed module name "\
"'%s' from moduleid '%s'." % (moduleid, moduleids[-1]))
module = memory_cached_module(moduleid)
if module: return module, moduleids
moduleids.append(moduleid)
instant_debug("In instant.check_memory_cache: Failed to find module.")
return None, moduleids
def check_disk_cache(modulename, cache_dir, moduleids):
# Get file lock to avoid race conditions in cache
lock = get_lock(cache_dir, modulename)
# Ensure a valid cache_dir
cache_dir = validate_cache_dir(cache_dir)
# Check on disk, in current directory and cache directory
for path in (os.getcwd(), cache_dir):
if os.path.isdir(os.path.join(path, modulename)):
# Found existing directory, try to import and place in memory cache
module = import_and_cache_module(path, modulename, moduleids)
if module:
instant_debug("In instant.check_disk_cache: Imported module "\
"'%s' from '%s'." % (modulename, path))
release_lock(lock)
return module
else:
instant_debug("In instant.check_disk_cache: Failed to imported "\
"module '%s' from '%s'." % (modulename, path))
# All attempts failed
instant_debug("In instant.check_disk_cache: Can't import module with modulename "\
"%r using cache directory %r." % (modulename, cache_dir))
release_lock(lock)
return None
def import_module(moduleid, cache_dir=None):
"""Import module from cache given its moduleid and an optional cache directory.
The moduleid can be either
- the module name
- a signature string, of which a checksum is taken to look up in the cache
- a checksum string, which is used directly to look up in the cache
- a hashable non-string object with a function moduleid.signature() which is used to get a signature string
The hashable object is used to look up in the memory cache before signature() is called.
If the module is found on disk, it is placed in the memory cache.
"""
# Look for module in memory cache
module, moduleids = check_memory_cache(moduleid)
if module: return module
# Look for module in disk cache
modulename = moduleids[-1]
return check_disk_cache(modulename, cache_dir, moduleids)
def cached_modules(cache_dir=None):
"Return a list with the names of all cached modules."
cache_dir = validate_cache_dir(cache_dir)
return os.listdir(cache_dir)
|