##// END OF EJS Templates
dispatch: write shell alias output to ui out descriptor
dispatch: write shell alias output to ui out descriptor

File last commit:

r14433:7658221d default
r14640:406b6d7b default
Show More
minirst.py
482 lines | 16.6 KiB | text/x-python | PythonLexer
Martin Geisler
minimal reStructuredText parser
r9156 # minirst.py - minimal reStructuredText parser
#
Martin Geisler
minirst: support containers...
r10443 # Copyright 2009, 2010 Matt Mackall <mpm@selenic.com> and others
Martin Geisler
minimal reStructuredText parser
r9156 #
# This software may be used and distributed according to the terms of the
Matt Mackall
Update license to GPLv2+
r10263 # GNU General Public License version 2 or any later version.
Martin Geisler
minimal reStructuredText parser
r9156
"""simplified reStructuredText parser.
This parser knows just enough about reStructuredText to parse the
Mercurial docstrings.
It cheats in a major way: nested blocks are not really nested. They
are just indented blocks that look like they are nested. This relies
on the user to keep the right indentation for the blocks.
Martin Geisler
minirst: link to HelpStyleGuide in docstring
r12958 Remember to update http://mercurial.selenic.com/wiki/HelpStyleGuide
when adding support for new constructs.
Martin Geisler
minimal reStructuredText parser
r9156 """
FUJIWARA Katsunori
replace Python standard textwrap by MBCS sensitive one for i18n text...
r11297 import re, sys
FUJIWARA Katsunori
minirst: use unicode string as intermediate form for replacement...
r11464 import util, encoding
Erik Zielke
minirst: Support for admonitions...
r12388 from i18n import _
FUJIWARA Katsunori
minirst: use unicode string as intermediate form for replacement...
r11464
def replace(text, substs):
utext = text.decode(encoding.encoding)
for f, t in substs:
utext = utext.replace(f, t)
return utext.encode(encoding.encoding)
Martin Geisler
minimal reStructuredText parser
r9156
Martin Geisler
minirst: refactor/simplify findblocks
r12651
_blockre = re.compile(r"\n(?:\s*\n)+")
Martin Geisler
minimal reStructuredText parser
r9156 def findblocks(text):
"""Find continuous blocks of lines in text.
Returns a list of dictionaries representing the blocks. Each block
has an 'indent' field and a 'lines' field.
"""
Martin Geisler
minirst: refactor/simplify findblocks
r12651 blocks = []
for b in _blockre.split(text.strip()):
lines = b.splitlines()
indent = min((len(l) - len(l.lstrip())) for l in lines)
lines = [l[indent:] for l in lines]
blocks.append(dict(indent=indent, lines=lines))
Martin Geisler
minimal reStructuredText parser
r9156 return blocks
def findliteralblocks(blocks):
"""Finds literal blocks and adds a 'type' field to the blocks.
Literal blocks are given the type 'literal', all other blocks are
given type the 'paragraph'.
"""
i = 0
while i < len(blocks):
# Searching for a block that looks like this:
#
# +------------------------------+
# | paragraph |
# | (ends with "::") |
# +------------------------------+
# +---------------------------+
# | indented literal block |
# +---------------------------+
blocks[i]['type'] = 'paragraph'
Matt Mackall
many, many trivial check-code fixups
r10282 if blocks[i]['lines'][-1].endswith('::') and i + 1 < len(blocks):
Martin Geisler
minimal reStructuredText parser
r9156 indent = blocks[i]['indent']
Matt Mackall
many, many trivial check-code fixups
r10282 adjustment = blocks[i + 1]['indent'] - indent
Martin Geisler
minimal reStructuredText parser
r9156
if blocks[i]['lines'] == ['::']:
# Expanded form: remove block
del blocks[i]
i -= 1
elif blocks[i]['lines'][-1].endswith(' ::'):
# Partially minimized form: remove space and both
# colons.
blocks[i]['lines'][-1] = blocks[i]['lines'][-1][:-3]
else:
# Fully minimized form: remove just one colon.
blocks[i]['lines'][-1] = blocks[i]['lines'][-1][:-1]
# List items are formatted with a hanging indent. We must
# correct for this here while we still have the original
# information on the indentation of the subsequent literal
# blocks available.
Martin Geisler
minirst: prepare for general types of bullet lists...
r9738 m = _bulletre.match(blocks[i]['lines'][0])
if m:
indent += m.end()
adjustment -= m.end()
Martin Geisler
minimal reStructuredText parser
r9156
# Mark the following indented blocks.
Matt Mackall
many, many trivial check-code fixups
r10282 while i + 1 < len(blocks) and blocks[i + 1]['indent'] > indent:
blocks[i + 1]['type'] = 'literal'
blocks[i + 1]['indent'] -= adjustment
Martin Geisler
minimal reStructuredText parser
r9156 i += 1
i += 1
return blocks
Martin Geisler
minirst: support line blocks
r10447 _bulletre = re.compile(r'(-|[0-9A-Za-z]+\.|\(?[0-9A-Za-z]+\)|\|) ')
Erik Zielke
minirst: improved support for option lists....
r13011 _optionre = re.compile(r'^(-([a-zA-Z0-9]), )?(--[a-z0-9-]+)'
r'((.*) +)(.*)$')
Martin Geisler
minirst: improve layout of field lists...
r10065 _fieldre = re.compile(r':(?![: ])([^:]*)(?<! ):[ ]+(.*)')
Martin Geisler
minirst: combine list parsing in one function...
r9737 _definitionre = re.compile(r'[^ ]')
def splitparagraphs(blocks):
"""Split paragraphs into lists."""
# Tuples with (list type, item regexp, single line items?). Order
# matters: definition lists has the least specific regexp and must
# come last.
listtypes = [('bullet', _bulletre, True),
('option', _optionre, True),
('field', _fieldre, True),
('definition', _definitionre, False)]
def match(lines, i, itemre, singleline):
"""Does itemre match an item at line i?
A list item can be followed by an idented line or another list
item (but only if singleline is True).
"""
line1 = lines[i]
Matt Mackall
many, many trivial check-code fixups
r10282 line2 = i + 1 < len(lines) and lines[i + 1] or ''
Martin Geisler
minirst: combine list parsing in one function...
r9737 if not itemre.match(line1):
return False
if singleline:
return line2 == '' or line2[0] == ' ' or itemre.match(line2)
else:
return line2.startswith(' ')
i = 0
while i < len(blocks):
if blocks[i]['type'] == 'paragraph':
lines = blocks[i]['lines']
for type, itemre, singleline in listtypes:
if match(lines, 0, itemre, singleline):
items = []
for j, line in enumerate(lines):
if match(lines, j, itemre, singleline):
items.append(dict(type=type, lines=[],
indent=blocks[i]['indent']))
items[-1]['lines'].append(line)
Matt Mackall
many, many trivial check-code fixups
r10282 blocks[i:i + 1] = items
Martin Geisler
minirst: combine list parsing in one function...
r9737 break
i += 1
return blocks
Martin Geisler
minimal reStructuredText parser
r9156
Martin Geisler
minirst: improve layout of field lists...
r10065 _fieldwidth = 12
def updatefieldlists(blocks):
"""Find key and maximum key width for field lists."""
i = 0
while i < len(blocks):
if blocks[i]['type'] != 'field':
i += 1
continue
keywidth = 0
j = i
while j < len(blocks) and blocks[j]['type'] == 'field':
m = _fieldre.match(blocks[j]['lines'][0])
key, rest = m.groups()
blocks[j]['lines'][0] = rest
blocks[j]['key'] = key
keywidth = max(keywidth, len(key))
j += 1
for block in blocks[i:j]:
block['keywidth'] = keywidth
i = j + 1
return blocks
Erik Zielke
minirst: improved support for option lists....
r13011 def updateoptionlists(blocks):
i = 0
while i < len(blocks):
if blocks[i]['type'] != 'option':
i += 1
continue
optstrwidth = 0
j = i
while j < len(blocks) and blocks[j]['type'] == 'option':
m = _optionre.match(blocks[j]['lines'][0])
shortoption = m.group(2)
group3 = m.group(3)
longoption = group3[2:].strip()
desc = m.group(6).strip()
longoptionarg = m.group(5).strip()
blocks[j]['lines'][0] = desc
noshortop = ''
if not shortoption:
noshortop = ' '
opt = "%s%s" % (shortoption and "-%s " % shortoption or '',
("%s--%s %s") % (noshortop, longoption,
longoptionarg))
opt = opt.rstrip()
blocks[j]['optstr'] = opt
optstrwidth = max(optstrwidth, encoding.colwidth(opt))
j += 1
for block in blocks[i:j]:
block['optstrwidth'] = optstrwidth
i = j + 1
return blocks
Martin Geisler
minirst: support containers...
r10443 def prunecontainers(blocks, keep):
"""Prune unwanted containers.
The blocks must have a 'type' field, i.e., they should have been
run through findliteralblocks first.
"""
Martin Geisler
minirst: report pruned container types
r10444 pruned = []
Martin Geisler
minirst: support containers...
r10443 i = 0
while i + 1 < len(blocks):
# Searching for a block that looks like this:
#
# +-------+---------------------------+
# | ".. container ::" type |
# +---+ |
# | blocks |
# +-------------------------------+
if (blocks[i]['type'] == 'paragraph' and
blocks[i]['lines'][0].startswith('.. container::')):
indent = blocks[i]['indent']
adjustment = blocks[i + 1]['indent'] - indent
containertype = blocks[i]['lines'][0][15:]
prune = containertype not in keep
Martin Geisler
minirst: report pruned container types
r10444 if prune:
pruned.append(containertype)
Martin Geisler
minirst: support containers...
r10443
# Always delete "..container:: type" block
del blocks[i]
j = i
while j < len(blocks) and blocks[j]['indent'] > indent:
if prune:
del blocks[j]
i -= 1 # adjust outer index
else:
blocks[j]['indent'] -= adjustment
j += 1
i += 1
Martin Geisler
minirst: report pruned container types
r10444 return blocks, pruned
Martin Geisler
minirst: support containers...
r10443
Martin Geisler
minirst: support all recommended title adornments
r10984 _sectionre = re.compile(r"""^([-=`:.'"~^_*+#])\1+$""")
Martin Geisler
minimal reStructuredText parser
r9156 def findsections(blocks):
"""Finds sections.
The blocks must have a 'type' field, i.e., they should have been
run through findliteralblocks first.
"""
for block in blocks:
# Searching for a block that looks like this:
#
# +------------------------------+
# | Section title |
# | ------------- |
# +------------------------------+
if (block['type'] == 'paragraph' and
len(block['lines']) == 2 and
Matt Mackall
minirst: use colwidth to match title lengths (issue2455)
r12867 encoding.colwidth(block['lines'][0]) == len(block['lines'][1]) and
Martin Geisler
minirst: support all recommended title adornments
r10984 _sectionre.match(block['lines'][1])):
Martin Geisler
minirst: correctly format sections containing inline markup...
r10983 block['underline'] = block['lines'][1][0]
Martin Geisler
minimal reStructuredText parser
r9156 block['type'] = 'section'
Martin Geisler
minirst: correctly format sections containing inline markup...
r10983 del block['lines'][1]
Martin Geisler
minimal reStructuredText parser
r9156 return blocks
Martin Geisler
minirst: convert ``foo`` into "foo" upon display...
r9623 def inlineliterals(blocks):
FUJIWARA Katsunori
minirst: use unicode string as intermediate form for replacement...
r11464 substs = [('``', '"')]
Martin Geisler
minirst: convert ``foo`` into "foo" upon display...
r9623 for b in blocks:
Martin Geisler
minirst: correctly format sections containing inline markup...
r10983 if b['type'] in ('paragraph', 'section'):
FUJIWARA Katsunori
minirst: use unicode string as intermediate form for replacement...
r11464 b['lines'] = [replace(l, substs) for l in b['lines']]
Martin Geisler
minirst: convert ``foo`` into "foo" upon display...
r9623 return blocks
Martin Geisler
doc, minirst: support hg interpreted text role
r10972 def hgrole(blocks):
FUJIWARA Katsunori
minirst: use unicode string as intermediate form for replacement...
r11464 substs = [(':hg:`', '"hg '), ('`', '"')]
Martin Geisler
doc, minirst: support hg interpreted text role
r10972 for b in blocks:
Martin Geisler
minirst: correctly format sections containing inline markup...
r10983 if b['type'] in ('paragraph', 'section'):
Martin Geisler
minirst: handle line breaks in hg role
r11192 # Turn :hg:`command` into "hg command". This also works
# when there is a line break in the command and relies on
# the fact that we have no stray back-quotes in the input
# (run the blocks through inlineliterals first).
FUJIWARA Katsunori
minirst: use unicode string as intermediate form for replacement...
r11464 b['lines'] = [replace(l, substs) for l in b['lines']]
Martin Geisler
doc, minirst: support hg interpreted text role
r10972 return blocks
Martin Geisler
minimal reStructuredText parser
r9156 def addmargins(blocks):
"""Adds empty blocks for vertical spacing.
This groups bullets, options, and definitions together with no vertical
space between them, and adds an empty block between all other blocks.
"""
i = 1
while i < len(blocks):
Matt Mackall
many, many trivial check-code fixups
r10282 if (blocks[i]['type'] == blocks[i - 1]['type'] and
Martin Geisler
minirst: add margin around definition items...
r10936 blocks[i]['type'] in ('bullet', 'option', 'field')):
Martin Geisler
minimal reStructuredText parser
r9156 i += 1
else:
blocks.insert(i, dict(lines=[''], indent=0, type='margin'))
i += 2
return blocks
Martin Geisler
minirst: ignore comments
r12819 def prunecomments(blocks):
"""Remove comments."""
i = 0
while i < len(blocks):
b = blocks[i]
Erik Zielke
minirst: modified minirst to also recognize empty comments....
r13009 if b['type'] == 'paragraph' and (b['lines'][0].startswith('.. ') or
b['lines'] == ['..']):
Martin Geisler
minirst: ignore comments
r12819 del blocks[i]
Martin Geisler
minirst: better interaction between comments and margins...
r13003 if i < len(blocks) and blocks[i]['type'] == 'margin':
del blocks[i]
Martin Geisler
minirst: ignore comments
r12819 else:
i += 1
return blocks
Martin Geisler
minirst: small code cleanup
r12620 _admonitionre = re.compile(r"\.\. (admonition|attention|caution|danger|"
r"error|hint|important|note|tip|warning)::",
flags=re.IGNORECASE)
Erik Zielke
minirst: Support for admonitions...
r12388 def findadmonitions(blocks):
"""
Makes the type of the block an admonition block if
the first line is an admonition directive
"""
i = 0
while i < len(blocks):
Martin Geisler
minirst: small code cleanup
r12620 m = _admonitionre.match(blocks[i]['lines'][0])
Erik Zielke
minirst: Support for admonitions...
r12388 if m:
blocks[i]['type'] = 'admonition'
admonitiontitle = blocks[i]['lines'][0][3:m.end() - 2].lower()
firstline = blocks[i]['lines'][0][m.end() + 1:]
Martin Geisler
minirst: small code cleanup
r12620 if firstline:
blocks[i]['lines'].insert(1, ' ' + firstline)
Erik Zielke
minirst: Support for admonitions...
r12388
blocks[i]['admonitiontitle'] = admonitiontitle
del blocks[i]['lines'][0]
i = i + 1
return blocks
Martin Geisler
minimal reStructuredText parser
r9156
Martin Geisler
minirst: pull admonition titles out formatblock function
r12652 _admonitiontitles = {'attention': _('Attention:'),
'caution': _('Caution:'),
'danger': _('!Danger!') ,
'error': _('Error:'),
'hint': _('Hint:'),
'important': _('Important:'),
'note': _('Note:'),
'tip': _('Tip:'),
'warning': _('Warning!')}
Erik Zielke
minirst: improved support for option lists....
r13011 def formatoption(block, width):
desc = ' '.join(map(str.strip, block['lines']))
colwidth = encoding.colwidth(block['optstr'])
usablewidth = width - 1
hanging = block['optstrwidth']
initindent = '%s%s ' % (block['optstr'], ' ' * ((hanging - colwidth)))
hangindent = ' ' * (encoding.colwidth(initindent) + 1)
return ' %s' % (util.wrap(desc, usablewidth,
initindent=initindent,
hangindent=hangindent))
Martin Geisler
minimal reStructuredText parser
r9156 def formatblock(block, width):
"""Format a block according to width."""
Martin Geisler
util, minirst: do not crash with COLUMNS=0
r9417 if width <= 0:
width = 78
Martin Geisler
minimal reStructuredText parser
r9156 indent = ' ' * block['indent']
Erik Zielke
minirst: Support for admonitions...
r12388 if block['type'] == 'admonition':
Martin Geisler
minirst: pull admonition titles out formatblock function
r12652 admonition = _admonitiontitles[block['admonitiontitle']]
Erik Zielke
minirst: Support for admonitions...
r12388 hang = len(block['lines'][-1]) - len(block['lines'][-1].lstrip())
defindent = indent + hang * ' '
text = ' '.join(map(str.strip, block['lines']))
return '%s\n%s' % (indent + admonition, util.wrap(text, width=width,
initindent=defindent,
hangindent=defindent))
Martin Geisler
minimal reStructuredText parser
r9156 if block['type'] == 'margin':
return ''
Martin Geisler
minirst: remove unnecessary "elif:" statements
r9735 if block['type'] == 'literal':
Martin Geisler
minirst: indent literal blocks with two spaces...
r9291 indent += ' '
return indent + ('\n' + indent).join(block['lines'])
Martin Geisler
minirst: remove unnecessary "elif:" statements
r9735 if block['type'] == 'section':
Matt Mackall
minirst: use colwidth to match title lengths (issue2455)
r12867 underline = encoding.colwidth(block['lines'][0]) * block['underline']
Martin Geisler
minirst: correctly format sections containing inline markup...
r10983 return "%s%s\n%s%s" % (indent, block['lines'][0],indent, underline)
Martin Geisler
minirst: remove unnecessary "elif:" statements
r9735 if block['type'] == 'definition':
Martin Geisler
minimal reStructuredText parser
r9156 term = indent + block['lines'][0]
Martin Geisler
minirst: combine list parsing in one function...
r9737 hang = len(block['lines'][-1]) - len(block['lines'][-1].lstrip())
defindent = indent + hang * ' '
Martin Geisler
minimal reStructuredText parser
r9156 text = ' '.join(map(str.strip, block['lines'][1:]))
FUJIWARA Katsunori
replace Python standard textwrap by MBCS sensitive one for i18n text...
r11297 return '%s\n%s' % (term, util.wrap(text, width=width,
initindent=defindent,
hangindent=defindent))
Martin Geisler
minirst: removed unnecessary initindent variable
r10937 subindent = indent
Martin Geisler
minirst: remove unnecessary "elif:" statements
r9735 if block['type'] == 'bullet':
Martin Geisler
minirst: support line blocks
r10447 if block['lines'][0].startswith('| '):
# Remove bullet for line blocks and add no extra
# indention.
block['lines'][0] = block['lines'][0][2:]
else:
m = _bulletre.match(block['lines'][0])
subindent = indent + m.end() * ' '
Martin Geisler
minirst: combine list parsing in one function...
r9737 elif block['type'] == 'field':
Martin Geisler
minirst: improve layout of field lists...
r10065 keywidth = block['keywidth']
key = block['key']
subindent = indent + _fieldwidth * ' '
if len(key) + 2 > _fieldwidth:
# key too large, use full line width
key = key.ljust(width)
elif keywidth + 2 < _fieldwidth:
# all keys are small, add only two spaces
key = key.ljust(keywidth + 2)
subindent = indent + (keywidth + 2) * ' '
else:
# mixed sizes, use fieldwidth for this one
key = key.ljust(_fieldwidth)
block['lines'][0] = key + block['lines'][0]
Martin Geisler
minirst: combine list parsing in one function...
r9737 elif block['type'] == 'option':
Erik Zielke
minirst: improved support for option lists....
r13011 return formatoption(block, width)
Martin Geisler
minimal reStructuredText parser
r9156
Martin Geisler
minirst: combine list parsing in one function...
r9737 text = ' '.join(map(str.strip, block['lines']))
FUJIWARA Katsunori
replace Python standard textwrap by MBCS sensitive one for i18n text...
r11297 return util.wrap(text, width=width,
initindent=indent,
hangindent=subindent)
Martin Geisler
minimal reStructuredText parser
r9156
Martin Geisler
minirst: report pruned container types
r10444 def format(text, width, indent=0, keep=None):
Martin Geisler
minimal reStructuredText parser
r9156 """Parse and format the text according to width."""
blocks = findblocks(text)
Martin Geisler
help: un-indent help topics...
r9540 for b in blocks:
b['indent'] += indent
Martin Geisler
minimal reStructuredText parser
r9156 blocks = findliteralblocks(blocks)
Martin Geisler
minirst: report pruned container types
r10444 blocks, pruned = prunecontainers(blocks, keep or [])
Martin Geisler
minirst: correctly format sections containing inline markup...
r10983 blocks = findsections(blocks)
Martin Geisler
minirst: convert ``foo`` into "foo" upon display...
r9623 blocks = inlineliterals(blocks)
Martin Geisler
doc, minirst: support hg interpreted text role
r10972 blocks = hgrole(blocks)
Martin Geisler
minirst: combine list parsing in one function...
r9737 blocks = splitparagraphs(blocks)
Martin Geisler
minirst: improve layout of field lists...
r10065 blocks = updatefieldlists(blocks)
Erik Zielke
minirst: improved support for option lists....
r13011 blocks = updateoptionlists(blocks)
Martin Geisler
minirst: better interaction between comments and margins...
r13003 blocks = addmargins(blocks)
Martin Geisler
minirst: ignore comments
r12819 blocks = prunecomments(blocks)
Erik Zielke
minirst: Support for admonitions...
r12388 blocks = findadmonitions(blocks)
Martin Geisler
minirst: report pruned container types
r10444 text = '\n'.join(formatblock(b, width) for b in blocks)
if keep is None:
return text
else:
return text, pruned
Martin Geisler
minimal reStructuredText parser
r9156
if __name__ == "__main__":
from pprint import pprint
Martin Geisler
minirst: support containers...
r10443 def debug(func, *args):
blocks = func(*args)
Martin Geisler
minimal reStructuredText parser
r9156 print "*** after %s:" % func.__name__
pprint(blocks)
print
return blocks
Martin Geisler
minirst: read test input from stdin
r14433 text = sys.stdin.read()
Martin Geisler
minimal reStructuredText parser
r9156 blocks = debug(findblocks, text)
blocks = debug(findliteralblocks, blocks)
Martin Geisler
minirst: read test input from stdin
r14433 blocks, pruned = debug(prunecontainers, blocks, sys.argv[1:])
Martin Geisler
minirst: run inlineliterals too in debug mode
r10063 blocks = debug(inlineliterals, blocks)
Martin Geisler
minirst: combine list parsing in one function...
r9737 blocks = debug(splitparagraphs, blocks)
Martin Geisler
minirst: improve layout of field lists...
r10065 blocks = debug(updatefieldlists, blocks)
Erik Zielke
minirst: improved support for option lists....
r13011 blocks = debug(updateoptionlists, blocks)
Martin Geisler
minimal reStructuredText parser
r9156 blocks = debug(findsections, blocks)
Martin Geisler
minirst: better interaction between comments and margins...
r13003 blocks = debug(addmargins, blocks)
Martin Geisler
minirst: ignore comments
r12819 blocks = debug(prunecomments, blocks)
Erik Zielke
minirst: Support for admonitions...
r12388 blocks = debug(findadmonitions, blocks)
Martin Geisler
minimal reStructuredText parser
r9156 print '\n'.join(formatblock(b, 30) for b in blocks)