2020-05-25 22:40:09 +00:00
|
|
|
#!/usr/bin/python3
|
2020-05-23 01:56:39 +00:00
|
|
|
#
|
|
|
|
# Call pandoc to convert markdown to docbook, then expand gtk-doc
|
|
|
|
# abbreviations (|[ ]|, function(), #object, %constant, etc)
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import re
|
|
|
|
import tempfile
|
|
|
|
import os.path
|
|
|
|
import subprocess
|
|
|
|
|
|
|
|
# The following code is taken from gtk-doc
|
|
|
|
|
|
|
|
def ExpandAbbreviations(symbol, text):
|
2020-07-29 11:45:02 +00:00
|
|
|
# Hack!
|
|
|
|
# Strip xlink namespace from hrefs since pandoc insists on
|
|
|
|
# inserting them, and namespace setup doesn't transfer across
|
|
|
|
# xi:include.
|
|
|
|
# Yay for XML!
|
|
|
|
text = re.sub('xlink:href', 'href', text)
|
|
|
|
|
2020-05-23 01:56:39 +00:00
|
|
|
# Convert '@param()'
|
|
|
|
text = re.sub(r'(\A|[^\\])\@(\w+((\.|->)\w+)*)\s*\(\)', r'\1<parameter>\2()</parameter>', text)
|
|
|
|
|
|
|
|
# Convert 'function()' or 'macro()'.
|
|
|
|
# if there is abc_*_def() we don't want to make a link to _def()
|
|
|
|
# FIXME: also handle abc(def(....)) : but that would need to be done recursively :/
|
|
|
|
def f1(m):
|
|
|
|
return m.group(1) + MakeXRef(m.group(2), tagify(m.group(2) + "()", "function"))
|
|
|
|
text = re.sub(r'([^\*.\w])(\w+)\s*\(\)', f1, text)
|
|
|
|
# handle #Object.func()
|
|
|
|
text = re.sub(r'(\A|[^\\])#([\w\-:\.]+[\w]+)\s*\(\)', f1, text)
|
|
|
|
|
|
|
|
# Convert '@param', but not '\@param'.
|
|
|
|
text = re.sub(r'(\A|[^\\])\@(\w+((\.|->)\w+)*)', r'\1<parameter>\2</parameter>', text)
|
|
|
|
text = re.sub(r'/\\\@', r'\@', text)
|
|
|
|
|
|
|
|
# Convert '%constant', but not '\%constant'.
|
|
|
|
# Also allow negative numbers, e.g. %-1.
|
|
|
|
def f2(m):
|
|
|
|
return m.group(1) + MakeXRef(m.group(2), tagify(m.group(2), "literal"))
|
|
|
|
|
|
|
|
text = re.sub(r'(\A|[^\\])\%(-?\w+)', f2, text)
|
|
|
|
text = re.sub(r'\\\%', r'\%', text)
|
|
|
|
|
|
|
|
# Convert '#symbol', but not '\#symbol'.
|
2020-05-24 01:33:25 +00:00
|
|
|
|
|
|
|
# Only convert #foo after a space to avoid interfering with
|
|
|
|
# fragment identifiers in urls
|
2020-05-23 01:56:39 +00:00
|
|
|
def f3(m):
|
|
|
|
return m.group(1) + MakeHashXRef(m.group(2), "type")
|
|
|
|
|
2020-05-24 01:33:25 +00:00
|
|
|
text = re.sub(r'(\A|[ ])#([\w\-:\.]+[\w]+)', f3, text)
|
2020-05-23 01:56:39 +00:00
|
|
|
text = re.sub(r'\\#', '#', text)
|
|
|
|
|
|
|
|
return text
|
|
|
|
|
|
|
|
# Standard C preprocessor directives, which we ignore for '#' abbreviations.
|
|
|
|
PreProcessorDirectives = {
|
|
|
|
'assert', 'define', 'elif', 'else', 'endif', 'error', 'if', 'ifdef', 'ifndef',
|
|
|
|
'include', 'line', 'pragma', 'unassert', 'undef', 'warning'
|
|
|
|
}
|
|
|
|
|
|
|
|
def MakeHashXRef(symbol, tag):
|
|
|
|
text = symbol
|
|
|
|
|
|
|
|
# Check for things like '#include', '#define', and skip them.
|
|
|
|
if symbol in PreProcessorDirectives:
|
|
|
|
return "#%s" % symbol
|
|
|
|
|
|
|
|
# Get rid of special suffixes ('-struct','-enum').
|
|
|
|
text = re.sub(r'-struct$', '', text)
|
|
|
|
text = re.sub(r'-enum$', '', text)
|
|
|
|
|
|
|
|
# If the symbol is in the form "Object::signal", then change the symbol to
|
|
|
|
# "Object-signal" and use "signal" as the text.
|
|
|
|
if '::' in symbol:
|
|
|
|
o, s = symbol.split('::', 1)
|
|
|
|
symbol = '%s-%s' % (o, s)
|
|
|
|
text = u'“' + s + u'”'
|
|
|
|
|
|
|
|
# If the symbol is in the form "Object:property", then change the symbol to
|
|
|
|
# "Object--property" and use "property" as the text.
|
|
|
|
if ':' in symbol:
|
|
|
|
o, p = symbol.split(':', 1)
|
|
|
|
symbol = '%s--%s' % (o, p)
|
|
|
|
text = u'“' + p + u'”'
|
|
|
|
|
|
|
|
if tag != '':
|
|
|
|
text = tagify(text, tag)
|
|
|
|
|
|
|
|
return MakeXRef(symbol, text)
|
|
|
|
|
|
|
|
def MakeXRef(symbol, text=None):
|
|
|
|
"""This returns a cross-reference link to the given symbol.
|
|
|
|
|
|
|
|
Though it doesn't try to do this for a few standard C types that it knows
|
|
|
|
won't be in the documentation.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
symbol (str): the symbol to try to create a XRef to.
|
|
|
|
text (str): text to put inside the XRef, defaults to symbol
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
str: a docbook link
|
|
|
|
"""
|
|
|
|
symbol = symbol.strip()
|
|
|
|
if not text:
|
|
|
|
text = symbol
|
|
|
|
|
|
|
|
# Get rid of special suffixes ('-struct','-enum').
|
|
|
|
text = re.sub(r'-struct$', '', text)
|
|
|
|
text = re.sub(r'-enum$', '', text)
|
|
|
|
|
|
|
|
if ' ' in symbol:
|
|
|
|
return text
|
|
|
|
|
|
|
|
symbol_id = CreateValidSGMLID(symbol)
|
|
|
|
return "<link linkend=\"%s\">%s</link>" % (symbol_id, text)
|
|
|
|
|
|
|
|
def CreateValidSGMLID(xml_id):
|
|
|
|
"""Creates a valid SGML 'id' from the given string.
|
|
|
|
|
|
|
|
According to http://www.w3.org/TR/html4/types.html#type-id "ID and NAME
|
|
|
|
tokens must begin with a letter ([A-Za-z]) and may be followed by any number
|
|
|
|
of letters, digits ([0-9]), hyphens ("-"), underscores ("_"), colons (":"),
|
|
|
|
and periods (".")."
|
|
|
|
|
|
|
|
When creating SGML IDS, we append ":CAPS" to all all-caps identifiers to
|
|
|
|
prevent name clashes (SGML ids are case-insensitive). (It basically never is
|
|
|
|
the case that mixed-case identifiers would collide.)
|
|
|
|
|
|
|
|
Args:
|
|
|
|
id (str): The text to be converted into a valid SGML id.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
str: The converted id.
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Special case, '_' would end up as '' so we use 'gettext-macro' instead.
|
|
|
|
if xml_id == '_':
|
|
|
|
return "gettext-macro"
|
|
|
|
|
|
|
|
xml_id = re.sub(r'[,;]', '', xml_id)
|
|
|
|
xml_id = re.sub(r'[_ ]', '-', xml_id)
|
|
|
|
xml_id = re.sub(r'^-+', '', xml_id)
|
|
|
|
xml_id = xml_id.replace('::', '-')
|
|
|
|
xml_id = xml_id.replace(':', '--')
|
|
|
|
|
|
|
|
# Append ":CAPS" to all all-caps identifiers
|
|
|
|
# FIXME: there are some inconsistencies here, we have index files containing e.g. TRUE--CAPS
|
|
|
|
if xml_id.isupper() and not xml_id.endswith('-CAPS'):
|
|
|
|
xml_id += ':CAPS'
|
|
|
|
|
|
|
|
return xml_id
|
|
|
|
|
|
|
|
def tagify(text, elem):
|
|
|
|
# Adds a tag around some text.
|
|
|
|
# e.g tagify("Text", "literal") => "<literal>Text</literal>".
|
|
|
|
return '<' + elem + '>' + text + '</' + elem + '>'
|
|
|
|
|
|
|
|
# End of gtk-doc excerpts
|
|
|
|
|
|
|
|
MarkdownExtensions = {
|
|
|
|
'-auto_identifiers', # we use explicit identifiers where needed
|
|
|
|
'+header_attributes', # for explicit identifiers
|
|
|
|
'+blank_before_header', # helps with gtk-doc #Object abbreviations
|
|
|
|
'+compact_definition_lists', # to replace <variablelist>
|
|
|
|
'+pipe_tables',
|
|
|
|
'+backtick_code_blocks', # to replace |[ ]|
|
|
|
|
'+fenced_code_attributes', # to add language annotations
|
|
|
|
'-raw_html', # to escape literal tags like <child> in input
|
2020-05-24 18:20:42 +00:00
|
|
|
'+startnum', # to have interrupted lists in the q&a part
|
2020-05-23 01:56:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
def ConvertToDocbook(infile, outfile):
|
|
|
|
basename = os.path.basename(infile)
|
2020-05-24 16:04:15 +00:00
|
|
|
if basename.startswith('section'):
|
|
|
|
division='section'
|
|
|
|
else:
|
|
|
|
division='chapter'
|
2020-05-23 01:56:39 +00:00
|
|
|
input_format = "markdown" + "".join(MarkdownExtensions)
|
|
|
|
output_format = "docbook"
|
|
|
|
subprocess.check_call(["pandoc", infile, "-o", outfile,
|
|
|
|
"--from=" + input_format,
|
|
|
|
"--to=" + output_format,
|
2020-05-24 16:04:15 +00:00
|
|
|
"--top-level-division=" + division])
|
2020-05-23 01:56:39 +00:00
|
|
|
|
|
|
|
def ExpandGtkDocAbbreviations(infile, outfile):
|
|
|
|
contents = open(infile, 'r', encoding='utf-8').read()
|
|
|
|
with open(outfile, 'w', encoding='utf-8') as out:
|
|
|
|
out.write(ExpandAbbreviations("file", contents))
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
tmp = tempfile.mktemp()
|
|
|
|
ConvertToDocbook(sys.argv[1], tmp)
|
|
|
|
ExpandGtkDocAbbreviations(tmp, sys.argv[2])
|
|
|
|
os.remove(tmp)
|