Commit 758dd001 authored by scoder's avatar scoder

Merge pull request #257 from Carreau/clean-annotate

Clean html-annotate generation
parents ca0f1e18 85b455b9
...@@ -10,13 +10,9 @@ import Version ...@@ -10,13 +10,9 @@ import Version
from Code import CCodeWriter from Code import CCodeWriter
from Cython import Utils from Cython import Utils
# need one-characters subsitutions (for now) so offsets aren't off from contextlib import closing
special_chars = [
(u'&', u'\xF2', u'&'),
(u'<', u'\xF0', u'&lt;'),
(u'>', u'\xF1', u'&gt;'),
]
# need one-characters subsitutions (for now) so offsets aren't off
class AnnotationCCodeWriter(CCodeWriter): class AnnotationCCodeWriter(CCodeWriter):
...@@ -54,97 +50,123 @@ class AnnotationCCodeWriter(CCodeWriter): ...@@ -54,97 +50,123 @@ class AnnotationCCodeWriter(CCodeWriter):
def annotate(self, pos, item): def annotate(self, pos, item):
self.annotations.append((pos, item)) self.annotations.append((pos, item))
def save_annotation(self, source_filename, target_filename): def _css(self):
self.mark_pos(None) """css template will later allow to choose a colormap"""
f = Utils.open_source_file(source_filename) css = self._css_template
lines = f.readlines() for i in range(255):
for k, line in enumerate(lines): color = u"FFFF%02x" % int(255/(1+i/10.0))
for c, cc, html in special_chars: css = css+'\n.cython.score-%s {background-color: #%s;}' % (str(i),color)
line = line.replace(c, cc)
lines[k] = line return css
f.close()
all = [] _js = """
if False: function toggleDiv(id) {
for pos, item in self.annotations: theDiv = id.nextElementSibling
if pos[0].filename == source_filename: if (theDiv.style.display != 'block') theDiv.style.display = 'block';
start = item.start() else theDiv.style.display = 'none';
size, end = item.end() }
if size: """
all.append((pos, start))
all.append(((source_filename, pos[1], pos[2]+size), end))
else: _css_template = """
all.append((pos, start+end))
body { font-family: courier; font-size: 12; }
all.sort(reverse=True)
for pos, item in all: .cython.code { font-size: 9; color: #444444; display: none; margin-left: 20px; }
_, line_no, col = pos .cython.py_c_api { color: red; }
line_no -= 1 .cython.py_macro_api { color: #FF7000; }
col += 1 .cython.pyx_c_api { color: #FF3000; }
line = lines[line_no] .cython.pyx_macro_api { color: #FF7000; }
lines[line_no] = line[:col] + item + line[col:] .cython.refnanny { color: #FFA000; }
.cython.error_goto { color: #FFA000; }
.cython.tag { }
.cython.coerce { color: #008000; border: 1px dotted #008000 }
.cython.py_attr { color: #FF0000; font-weight: bold; }
.cython.c_attr { color: #0000FF; }
.cython.py_call { color: #FF0000; font-weight: bold; }
.cython.c_call { color: #0000FF; }
.cython.line { margin: 0em }
"""
def save_annotation(self, source_filename, target_filename):
with closing(Utils.open_source_file(source_filename)) as f:
lines = f.readlines()
code_source_file = self.code.get(source_filename, {})
c_file = Utils.decode_filename(os.path.basename(target_filename))
html_filename = os.path.splitext(target_filename)[0] + ".html" html_filename = os.path.splitext(target_filename)[0] + ".html"
f = codecs.open(html_filename, "w", encoding="UTF-8") with codecs.open(html_filename, "w", encoding="UTF-8") as out_buffer:
f.write(u'<!DOCTYPE html>\n') out_buffer.write(self._save_annotation(lines, code_source_file , c_file))
f.write(u'<!-- Generated by Cython %s -->\n' % Version.watermark)
f.write(u'<html>\n') def _save_annotation_header(self, c_file):
f.write(u""" outlist = []
outlist.append(u'<!DOCTYPE html>\n')
outlist.append(u'<!-- Generated by Cython %s -->\n' % Version.watermark)
outlist.append(u'<html>\n')
outlist.append(u"""
<head> <head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<style type="text/css"> <style type="text/css">
{css}
</style>
<script>
{js}
</script>
</head>
""".format(css=self._css(), js=self._js))
outlist.append(u'<body>\n')
outlist.append(u'<p>Generated by Cython %s</p>\n' % Version.watermark)
if c_file :
outlist.append(u'<p>Raw output: <a href="%s">%s</a></p>\n' % (c_file, c_file))
return outlist
body { font-family: courier; font-size: 12; } def _save_annotation_footer(self):
return (u'</body></html>\n',)
.code { font-size: 9; color: #444444; display: none; margin-left: 20px; }
.py_c_api { color: red; }
.py_macro_api { color: #FF7000; }
.pyx_c_api { color: #FF3000; }
.pyx_macro_api { color: #FF7000; }
.refnanny { color: #FFA000; }
.error_goto { color: #FFA000; }
.tag { }
.coerce { color: #008000; border: 1px dotted #008000 } def _save_annotation(self, lines, code_source_file , c_file=None):
"""
lines : original cython source code split by lines
code_source_file : generated c code keyed by line number in original file
target filename : name of the file in which to store the generated html
c_file : filename in which the c_code has been written
.py_attr { color: #FF0000; font-weight: bold; } """
.c_attr { color: #0000FF; }
.py_call { color: #FF0000; font-weight: bold; } outlist = []
.c_call { color: #0000FF; } outlist.extend(self._save_annotation_header(c_file))
.line { margin: 0em } outlist.extend(self._save_annotation_body(lines, code_source_file , c_file=None))
outlist.extend(self._save_annotation_footer())
return ''.join(outlist)
</style>
<script>
function toggleDiv(id) {
theDiv = document.getElementById(id);
if (theDiv.style.display != 'block') theDiv.style.display = 'block';
else theDiv.style.display = 'none';
}
</script>
</head>
""")
f.write(u'<body>\n')
f.write(u'<p>Generated by Cython %s\n' % Version.watermark)
c_file = Utils.decode_filename(os.path.basename(target_filename))
f.write(u'<p>Raw output: <a href="%s">%s</a>\n' % (c_file, c_file))
def _save_annotation_body(self, lines, code_source_file , c_file=None):
outlist=[]
pos_comment_marker = u'/* \N{HORIZONTAL ELLIPSIS} */\n'
zero_calls = dict((name, 0) for name in zero_calls = dict((name, 0) for name in
'refnanny py_macro_api py_c_api pyx_macro_api pyx_c_api error_goto'.split()) 'refnanny py_macro_api py_c_api pyx_macro_api pyx_c_api error_goto'.split())
self.mark_pos(None)
def annotate(match): def annotate(match):
group_name = match.lastgroup group_name = match.lastgroup
calls[group_name] += 1 calls[group_name] += 1
return ur"<span class='%s'>%s</span>" % ( return ur"<span class='cython %s'>%s</span>" % (
group_name, match.group(group_name)) group_name, match.group(group_name))
pos_comment_marker = u'/* \N{HORIZONTAL ELLIPSIS} */\n'
k = 0 for k, line in enumerate(lines):
code_source_file = self.code.get(source_filename, {}) line = html_escape(line)
for line in lines:
k += 1
try: try:
code = code_source_file[k] code = code_source_file[k]
except KeyError: except KeyError:
...@@ -159,19 +181,15 @@ function toggleDiv(id) { ...@@ -159,19 +181,15 @@ function toggleDiv(id) {
code = _parse_code(annotate, code) code = _parse_code(annotate, code)
score = (5 * calls['py_c_api'] + 2 * calls['pyx_c_api'] + score = (5 * calls['py_c_api'] + 2 * calls['pyx_c_api'] +
calls['py_macro_api'] + calls['pyx_macro_api']) calls['py_macro_api'] + calls['pyx_macro_api'])
color = u"FFFF%02x" % int(255/(1+score/10.0))
f.write(u"<pre class='line' style='background-color: #%s' onclick='toggleDiv(\"line%s\")'>" % (color, k))
f.write(u" %d: " % k) outlist.append(u"<pre class='cython line score-%s' onclick='toggleDiv(this)'>" % (score))
for c, cc, html in special_chars:
line = line.replace(cc, html)
f.write(line.rstrip())
f.write(u'</pre>\n') outlist.append(u" %d: " % k)
f.write(u"<pre id='line%s' class='code' style='background-color: #%s'>%s</pre>" % (k, color, code)) outlist.append(line.rstrip())
f.write(u'</body></html>\n')
f.close()
outlist.append(u'</pre>\n')
outlist.append(u"<pre class='cython code score-%s' >%s</pre>" % (score, code))
return outlist
_parse_code = re.compile( _parse_code = re.compile(
ur'(?P<refnanny>__Pyx_X?(?:GOT|GIVE)REF|__Pyx_RefNanny[A-Za-z]+)|' ur'(?P<refnanny>__Pyx_X?(?:GOT|GIVE)REF|__Pyx_RefNanny[A-Za-z]+)|'
...@@ -201,7 +219,7 @@ class AnnotationItem(object): ...@@ -201,7 +219,7 @@ class AnnotationItem(object):
self.size = size self.size = size
def start(self): def start(self):
return u"<span class='tag %s' title='%s'>%s" % (self.style, self.text, self.tag) return u"<span class='cython tag %s' title='%s'>%s" % (self.style, self.text, self.tag)
def end(self): def end(self):
return self.size, u"</span>" return self.size, u"</span>"
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment