highlight.py
151 lines
| 4.2 KiB
| text/x-python
|
PythonLexer
/ hgext / highlight.py
Adam Hupp
|
r5532 | """ | ||
This is Mercurial extension for syntax highlighting in the file | ||||
revision view of hgweb. | ||||
It depends on the pygments syntax highlighting library: | ||||
http://pygments.org/ | ||||
To enable the extension add this to hgrc: | ||||
[extensions] | ||||
hgext.highlight = | ||||
There is a single configuration option: | ||||
[web] | ||||
pygments_style = <style> | ||||
The default is 'colorful'. If this is changed the corresponding CSS | ||||
file should be re-generated by running | ||||
# pygmentize -f html -S <newstyle> | ||||
-- Adam Hupp <adam@hupp.org> | ||||
""" | ||||
from mercurial import demandimport | ||||
demandimport.ignore.extend(['pkgutil', | ||||
'pkg_resources', | ||||
'__main__',]) | ||||
import mimetypes | ||||
from mercurial.hgweb import hgweb_mod | ||||
from mercurial.hgweb.hgweb_mod import hgweb | ||||
from mercurial import util | ||||
from mercurial.hgweb.common import paritygen | ||||
from mercurial.node import hex | ||||
from pygments import highlight | ||||
from pygments.util import ClassNotFound | ||||
from pygments.lexers import guess_lexer_for_filename, TextLexer | ||||
from pygments.formatters import HtmlFormatter | ||||
Bryan O'Sullivan
|
r5533 | SYNTAX_CSS = ('\n<link rel="stylesheet" href="#staticurl#highlight.css" ' | ||
'type="text/css" />') | ||||
Adam Hupp
|
r5532 | |||
class StripedHtmlFormatter(HtmlFormatter): | ||||
def __init__(self, stripecount, *args, **kwargs): | ||||
super(StripedHtmlFormatter, self).__init__(*args, **kwargs) | ||||
self.stripecount = stripecount | ||||
Bryan O'Sullivan
|
r5533 | |||
Adam Hupp
|
r5532 | def wrap(self, source, outfile): | ||
yield 0, "<div class='highlight'>" | ||||
yield 0, "<pre>" | ||||
parity = paritygen(self.stripecount) | ||||
for n, i in source: | ||||
if n == 1: | ||||
Bryan O'Sullivan
|
r5533 | i = "<div class='parity%s'>%s</div>" % (parity.next(), i) | ||
Adam Hupp
|
r5532 | yield n, i | ||
yield 0, "</pre>" | ||||
yield 0, "</div>" | ||||
Christian Ebert
|
r5748 | def pygments_format(filename, text, forcetext, stripecount, style): | ||
Adam Hupp
|
r5532 | if not forcetext: | ||
try: | ||||
Christian Ebert
|
r5748 | lexer = guess_lexer_for_filename(filename, text, | ||
Christian Ebert
|
r5655 | encoding=util._encoding) | ||
Adam Hupp
|
r5532 | except ClassNotFound: | ||
Christian Ebert
|
r5655 | lexer = TextLexer(encoding=util._encoding) | ||
Adam Hupp
|
r5532 | else: | ||
Christian Ebert
|
r5655 | lexer = TextLexer(encoding=util._encoding) | ||
Bryan O'Sullivan
|
r5533 | |||
formatter = StripedHtmlFormatter(stripecount, style=style, | ||||
Christian Ebert
|
r5748 | linenos='inline', encoding=util._encoding) | ||
Adam Hupp
|
r5532 | |||
Christian Ebert
|
r5748 | return highlight(text, lexer, formatter) | ||
Adam Hupp
|
r5532 | |||
Christian Ebert
|
r5616 | def filerevision_pygments(self, tmpl, fctx): | ||
Bryan O'Sullivan
|
r5533 | """Reimplement hgweb.filerevision to use syntax highlighting""" | ||
Christian Ebert
|
r5616 | f = fctx.path() | ||
Adam Hupp
|
r5532 | |||
rawtext = fctx.data() | ||||
text = rawtext | ||||
Christian Ebert
|
r5616 | fl = fctx.filelog() | ||
n = fctx.filenode() | ||||
mt = mimetypes.guess_type(f)[0] | ||||
Adam Hupp
|
r5532 | |||
Christian Ebert
|
r5748 | # we always want hgweb.encoding | ||
util._encoding = self.encoding | ||||
Adam Hupp
|
r5532 | if util.binary(text): | ||
mt = mt or 'application/octet-stream' | ||||
text = "(binary:%s)" % mt | ||||
# don't parse (binary:...) as anything | ||||
forcetext = True | ||||
else: | ||||
mt = mt or 'text/plain' | ||||
Christian Ebert
|
r5748 | |||
# encode to hgweb.encoding for lexers and formatter | ||||
text = util.tolocal(text) | ||||
Adam Hupp
|
r5532 | forcetext = False | ||
def lines(text): | ||||
for line in text.splitlines(True): | ||||
yield {"line": line} | ||||
style = self.config("web", "pygments_style", "colorful") | ||||
Christian Ebert
|
r5748 | text_formatted = lines(pygments_format(f, text, forcetext, | ||
Christian Ebert
|
r5654 | self.stripecount, style)) | ||
Adam Hupp
|
r5532 | |||
Bryan O'Sullivan
|
r5533 | # override per-line template | ||
Christian Ebert
|
r5616 | tmpl.cache['fileline'] = '#line#' | ||
Adam Hupp
|
r5532 | |||
# append a <link ...> to the syntax highlighting css | ||||
Christian Ebert
|
r5616 | old_header = ''.join(tmpl('header')) | ||
Adam Hupp
|
r5532 | if SYNTAX_CSS not in old_header: | ||
new_header = old_header + SYNTAX_CSS | ||||
Christian Ebert
|
r5616 | tmpl.cache['header'] = new_header | ||
Adam Hupp
|
r5532 | |||
Christian Ebert
|
r5616 | yield tmpl("filerevision", | ||
file=f, | ||||
path=hgweb_mod._up(f), # fixme: make public | ||||
text=text_formatted, | ||||
raw=rawtext, | ||||
mimetype=mt, | ||||
rev=fctx.rev(), | ||||
node=hex(fctx.node()), | ||||
author=fctx.user(), | ||||
date=fctx.date(), | ||||
desc=fctx.description(), | ||||
parent=self.siblings(fctx.parents()), | ||||
child=self.siblings(fctx.children()), | ||||
rename=self.renamelink(fl, n), | ||||
permissions=fctx.manifest().flags(f)) | ||||
Adam Hupp
|
r5532 | |||
Bryan O'Sullivan
|
r5533 | |||
Adam Hupp
|
r5532 | # monkeypatch in the new version | ||
# should be safer than overriding the method in a derived class | ||||
# and then patching the class | ||||
hgweb.filerevision = filerevision_pygments | ||||