-
Notifications
You must be signed in to change notification settings - Fork 0
/
markup.py
102 lines (76 loc) · 2.51 KB
/
markup.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
# -*- coding:utf-8 -*-
"""
Support for different markup languages for the body of a post.
The following markup languages are supported:
- HTML
- Plain text
- ReStructured Text
- Markdown
- Textile
For ReStructuredText and Markdown syntax highlighting of source code is
available.
"""
# TODO: Add summary rendering.
# TODO: Docstrings.
import re
import logging
from cStringIO import StringIO
import config
import utils
# Import markup module from lib/
import addlib
import markdown
import markdown_processor
import rst_directive
import textile
from HTMLEditor import HTMLWordTruncator
from docutils.core import publish_parts
CUT_SEPARATOR_REGEX = r'<!--.*cut.*-->'
def render_rst(content):
warning_stream = StringIO()
parts = publish_parts(content, writer_name='html4css1',
settings_overrides={
'_disable_config': True,
'embed_stylesheet': False,
'warning_stream': warning_stream,
'report_level': 2,
})
rst_warnings = warning_stream.getvalue()
if rst_warnings:
logging.warn(rst_warnings)
return parts['html_body']
def render_markdown(content):
md = markdown.Markdown()
md.textPreprocessors.insert(0, markdown_processor.CodeBlockPreprocessor())
return md.convert(content)
def render_textile(content):
return textile.textile(content.encode('utf-8'))
# Mapping: string ID -> (human readable name, renderer)
MARKUP_MAP = {
'html': ('HTML', lambda c: c),
# 'txt': ('Plain Text', lambda c: html.linebreaks(html.escape(c))),
'markdown': ('Markdown', render_markdown),
'textile': ('Textile', render_textile),
'rst': ('ReStructuredText', render_rst),
}
def get_renderer(post):
"""Returns a render function for this posts body markup."""
return MARKUP_MAP.get(post.body_markup)[1]
def clean_content(content):
"""Clean up the raw body.
Actually this removes the cut separator.
"""
return re.sub(CUT_SEPARATOR_REGEX, '', content)
def render_body(post):
"""Return the post's body rendered to HTML."""
renderer = get_renderer(post)
return renderer(clean_content(post.body))
def render_summary(post):
"""Return the post's summary rendered to HTML."""
renderer = get_renderer(post)
match = re.search(CUT_SEPARATOR_REGEX, post.body)
if match:
return renderer(post.body[:match.start(0)])
else:
truncator = HTMLWordTruncator(config.summary_length)
return truncator.process(renderer(clean_content(post.body)))