2013-04-23 18:51:17 +02:00
|
|
|
from __future__ import absolute_import
|
|
|
|
|
2013-03-28 18:48:37 +01:00
|
|
|
import re
|
2013-07-12 22:01:31 +02:00
|
|
|
import os.path
|
|
|
|
import sourcemap
|
2013-03-28 18:48:37 +01:00
|
|
|
|
|
|
|
|
|
|
|
class SourceMap(object):
|
2013-07-12 22:01:31 +02:00
|
|
|
'''Map (line, column) pairs from generated to source file.'''
|
2013-03-28 18:48:37 +01:00
|
|
|
|
2013-07-12 22:01:31 +02:00
|
|
|
def __init__(self, sourcemap_dir):
|
|
|
|
self._dir = sourcemap_dir
|
|
|
|
self._indices = {}
|
2013-03-28 18:48:37 +01:00
|
|
|
|
2013-07-12 22:01:31 +02:00
|
|
|
def _index_for(self, minified_src):
|
|
|
|
'''Return the source map index for minified_src, loading it if not
|
|
|
|
already loaded.'''
|
|
|
|
if minified_src not in self._indices:
|
|
|
|
with open(os.path.join(self._dir, minified_src + '.map')) as fp:
|
|
|
|
self._indices[minified_src] = sourcemap.load(fp)
|
2013-03-28 18:48:37 +01:00
|
|
|
|
2013-07-12 22:01:31 +02:00
|
|
|
return self._indices[minified_src]
|
2013-03-28 18:48:37 +01:00
|
|
|
|
|
|
|
def annotate_stacktrace(self, stacktrace):
|
|
|
|
out = ''
|
|
|
|
for ln in stacktrace.splitlines():
|
|
|
|
out += ln + '\n'
|
2013-07-12 22:01:31 +02:00
|
|
|
match = re.search(r'/static/min/(.+)(\.[0-9a-f]+)\.js:(\d+):(\d+)', ln)
|
2013-03-28 18:48:37 +01:00
|
|
|
if match:
|
2013-07-12 22:01:31 +02:00
|
|
|
# Get the appropriate source map for the minified file.
|
|
|
|
minified_src = match.groups()[0] + '.js'
|
|
|
|
index = self._index_for(minified_src)
|
|
|
|
|
|
|
|
gen_line, gen_col = map(int, match.groups()[2:4])
|
|
|
|
# The sourcemap lib is 0-based, so subtract 1 from line and col.
|
|
|
|
try:
|
|
|
|
result = index.lookup(line=gen_line-1, column=gen_col-1)
|
2013-03-28 18:48:37 +01:00
|
|
|
out += (' = %s line %d column %d\n' %
|
2013-07-12 22:01:31 +02:00
|
|
|
(result.src, result.src_line+1, result.src_col+1))
|
|
|
|
except IndexError:
|
2013-07-30 23:56:20 +02:00
|
|
|
out += ' [Unable to look up in source map]\n'
|
2013-03-28 18:48:37 +01:00
|
|
|
|
|
|
|
if ln.startswith(' at'):
|
|
|
|
out += '\n'
|
|
|
|
return out
|