"""Cache lines from Python source files. This is intended to read lines from modules imported -- hence if a filename is not found, it will look down the module search path for a file by that name. """ __all__ = ["getline", "clearcache", "checkcache", "lazycache"] # The cache. Maps filenames to either a thunk which will provide source code, # or a tuple (size, mtime, lines, fullname) once loaded. cache = {} _interactive_cache = {} def clearcache(): """Clear the cache entirely.""" cache.clear() def getline(filename, lineno, module_globals=None): """Get a line for a Python source file from the cache. Update the cache if it doesn't contain an entry for this file already.""" lines = getlines(filename, module_globals) if 1 <= lineno <= len(lines): return lines[lineno - 1] return '' def getlines(filename, module_globals=None): """Get the lines for a Python source file from the cache. Update the cache if it doesn't contain an entry for this file already.""" entry = cache.get(filename, None) if entry is not None and len(entry) != 1: return entry[2] try: return updatecache(filename, module_globals) except MemoryError: clearcache() return [] def _getline_from_code(filename, lineno): lines = _getlines_from_code(filename) if 1 <= lineno <= len(lines): return lines[lineno - 1] return '' def _make_key(code): return (code.co_filename, code.co_qualname, code.co_firstlineno) def _getlines_from_code(code): code_id = _make_key(code) entry = _interactive_cache.get(code_id, None) if entry is not None and len(entry) != 1: return entry[2] return [] def _source_unavailable(filename): """Return True if the source code is unavailable for such file name.""" return ( not filename or (filename.startswith('<') and filename.endswith('>') and not filename.startswith('')): return None # Try for a __loader__, if available if module_globals and '__name__' in module_globals: spec = module_globals.get('__spec__') name = getattr(spec, 'name', None) or module_globals['__name__'] loader = getattr(spec, 'loader', None) if loader is None: loader = module_globals.get('__loader__') get_source = getattr(loader, 'get_source', None) if name and get_source: def get_lines(name=name, *args, **kwargs): return get_source(name, *args, **kwargs) return (get_lines,) return None def _register_code(code, string, name): entry = (len(string), None, [line + '\n' for line in string.splitlines()], name) stack = [code] while stack: code = stack.pop() for const in code.co_consts: if isinstance(const, type(code)): stack.append(const) key = _make_key(code) _interactive_cache[key] = entry