OILS / opy / _regtest / src / linecache.py View on Github | oils.pub

139 lines, 91 significant
1"""Cache lines from files.
2
3This is intended to read lines from modules imported -- hence if a filename
4is not found, it will look down the module search path for a file by
5that name.
6"""
7
8import sys
9import os
10
11__all__ = ["getline", "clearcache", "checkcache"]
12
13def getline(filename, lineno, module_globals=None):
14 lines = getlines(filename, module_globals)
15 if 1 <= lineno <= len(lines):
16 return lines[lineno-1]
17 else:
18 return ''
19
20
21# The cache
22
23cache = {} # The cache
24
25
26def clearcache():
27 """Clear the cache entirely."""
28
29 global cache
30 cache = {}
31
32
33def getlines(filename, module_globals=None):
34 """Get the lines for a file from the cache.
35 Update the cache if it doesn't contain an entry for this file already."""
36
37 if filename in cache:
38 return cache[filename][2]
39
40 try:
41 return updatecache(filename, module_globals)
42 except MemoryError:
43 clearcache()
44 return []
45
46
47def checkcache(filename=None):
48 """Discard cache entries that are out of date.
49 (This is not checked upon each call!)"""
50
51 if filename is None:
52 filenames = cache.keys()
53 else:
54 if filename in cache:
55 filenames = [filename]
56 else:
57 return
58
59 for filename in filenames:
60 size, mtime, lines, fullname = cache[filename]
61 if mtime is None:
62 continue # no-op for files loaded via a __loader__
63 try:
64 stat = os.stat(fullname)
65 except os.error:
66 del cache[filename]
67 continue
68 if size != stat.st_size or mtime != stat.st_mtime:
69 del cache[filename]
70
71
72def updatecache(filename, module_globals=None):
73 """Update a cache entry and return its list of lines.
74 If something's wrong, print a message, discard the cache entry,
75 and return an empty list."""
76
77 if filename in cache:
78 del cache[filename]
79 if not filename or (filename.startswith('<') and filename.endswith('>')):
80 return []
81
82 fullname = filename
83 try:
84 stat = os.stat(fullname)
85 except OSError:
86 basename = filename
87
88 # Try for a __loader__, if available
89 if module_globals and '__loader__' in module_globals:
90 name = module_globals.get('__name__')
91 loader = module_globals['__loader__']
92 get_source = getattr(loader, 'get_source', None)
93
94 if name and get_source:
95 try:
96 data = get_source(name)
97 except (ImportError, IOError):
98 pass
99 else:
100 if data is None:
101 # No luck, the PEP302 loader cannot find the source
102 # for this module.
103 return []
104 cache[filename] = (
105 len(data), None,
106 [line+'\n' for line in data.splitlines()], fullname
107 )
108 return cache[filename][2]
109
110 # Try looking through the module search path, which is only useful
111 # when handling a relative filename.
112 if os.path.isabs(filename):
113 return []
114
115 for dirname in sys.path:
116 # When using imputil, sys.path may contain things other than
117 # strings; ignore them when it happens.
118 try:
119 fullname = os.path.join(dirname, basename)
120 except (TypeError, AttributeError):
121 # Not sufficiently string-like to do anything useful with.
122 continue
123 try:
124 stat = os.stat(fullname)
125 break
126 except os.error:
127 pass
128 else:
129 return []
130 try:
131 with open(fullname, 'rU') as fp:
132 lines = fp.readlines()
133 except IOError:
134 return []
135 if lines and not lines[-1].endswith('\n'):
136 lines[-1] += '\n'
137 size, mtime = stat.st_size, stat.st_mtime
138 cache[filename] = size, mtime, lines, fullname
139 return lines