XML to Open Project Wiki implemented
This commit is contained in:
parent
f95f099eb2
commit
7a008afc67
1 changed files with 229 additions and 35 deletions
|
@ -9,16 +9,47 @@
|
||||||
# Contributor: Jorge Araya Navarro <elcorreo@deshackra.com>
|
# Contributor: Jorge Araya Navarro <elcorreo@deshackra.com>
|
||||||
#
|
#
|
||||||
|
|
||||||
|
# IMPORTANT NOTICE:
|
||||||
|
# If you are going to modify anything from this file, please be sure to follow
|
||||||
|
# the Style Guide for Python Code or often called "PEP8". To do this
|
||||||
|
# automagically just install autopep8:
|
||||||
|
#
|
||||||
|
# $ sudo pip3 install autopep8
|
||||||
|
#
|
||||||
|
# and run:
|
||||||
|
#
|
||||||
|
# $ autopep8 makedocs.py
|
||||||
|
#
|
||||||
|
# Before committing your changes. Also be sure to delete any trailing
|
||||||
|
# whitespace you may left.
|
||||||
|
#
|
||||||
|
# TODO:
|
||||||
|
# * Refactor code. Refactor strings.
|
||||||
|
# * Adapt this script for generating content in other markup formats like
|
||||||
|
# DokuWiki, Markdown, etc.
|
||||||
|
# * Because people will translate class_list.xml, we should implement
|
||||||
|
# internalization.
|
||||||
|
#
|
||||||
|
# Also check other TODO entries in this script for more information on what is
|
||||||
|
# left to do.
|
||||||
|
|
||||||
import re
|
import re
|
||||||
import argparse
|
import argparse
|
||||||
import logging
|
import logging
|
||||||
from os import path
|
from os import path
|
||||||
from itertools import izip_longest
|
from itertools import zip_longest
|
||||||
from xml.etree import ElementTree
|
from xml.etree import ElementTree
|
||||||
|
|
||||||
|
# add an option to change the verbosity
|
||||||
logging.basicConfig(level=logging.INFO)
|
logging.basicConfig(level=logging.INFO)
|
||||||
|
|
||||||
|
|
||||||
|
def tb(string):
|
||||||
|
""" Return a byte representation of a string
|
||||||
|
"""
|
||||||
|
return bytes(string, "UTF-8")
|
||||||
|
|
||||||
|
|
||||||
def getxmlfloc():
|
def getxmlfloc():
|
||||||
""" Returns the supposed location of the XML file
|
""" Returns the supposed location of the XML file
|
||||||
"""
|
"""
|
||||||
|
@ -30,7 +61,7 @@ def sortkey(c):
|
||||||
""" Symbols are first, letters second
|
""" Symbols are first, letters second
|
||||||
"""
|
"""
|
||||||
if "_" == c.attrib["name"][0]:
|
if "_" == c.attrib["name"][0]:
|
||||||
return True
|
return "A"
|
||||||
else:
|
else:
|
||||||
return c.attrib["name"]
|
return c.attrib["name"]
|
||||||
|
|
||||||
|
@ -50,7 +81,8 @@ def toOP(text):
|
||||||
text = text.replace(group.group(0), "\n\n", 1)
|
text = text.replace(group.group(0), "\n\n", 1)
|
||||||
elif gd["command"] == "div":
|
elif gd["command"] == "div":
|
||||||
if gd["value"] == '"center"':
|
if gd["value"] == '"center"':
|
||||||
alignstr = "{display:block; margin-left:auto; margin-right:auto;}"
|
alignstr = ("{display:block; margin-left:auto;"
|
||||||
|
" margin-right:auto;}")
|
||||||
elif gd["value"] == '"left"':
|
elif gd["value"] == '"left"':
|
||||||
alignstr = "<"
|
alignstr = "<"
|
||||||
elif gd["value"] == '"right"':
|
elif gd["value"] == '"right"':
|
||||||
|
@ -68,9 +100,115 @@ def toOP(text):
|
||||||
text = text.replace(group.group(0), "_", 1)
|
text = text.replace(group.group(0), "_", 1)
|
||||||
elif gd["command"] == "u" or gd["command"] == "/u":
|
elif gd["command"] == "u" or gd["command"] == "/u":
|
||||||
text = text.replace(group.group(0), "+", 1)
|
text = text.replace(group.group(0), "+", 1)
|
||||||
# TODO: Process other non-html commands
|
# Process other non-html commands
|
||||||
|
groups = re.finditer((r'\[method ((?P<class>[aA0-zZ9_]+)(?:\.))'
|
||||||
|
r'?(?P<method>[aA0-zZ9_]+)\]'), text)
|
||||||
|
for group in groups:
|
||||||
|
gd = group.groupdict()
|
||||||
|
if gd["class"]:
|
||||||
|
replacewith = ("\"<code>{gclass}.{method}</code>(Go "
|
||||||
|
"to page {gclass}, section {method})\""
|
||||||
|
":/class_{lkclass}#{lkmethod}".
|
||||||
|
format(gclass=gd["class"],
|
||||||
|
method=gd["method"],
|
||||||
|
lkclass=gd["class"].lower(),
|
||||||
|
lkmethod=gd["method"].lower()))
|
||||||
|
else:
|
||||||
|
# The method is located in the same wiki page
|
||||||
|
replacewith = ("\"<code>{method}</code>(Jump to method"
|
||||||
|
" {method})\":#{lkmethod}".
|
||||||
|
format(method=gd["method"],
|
||||||
|
lkmethod=gd["method"].lower()))
|
||||||
|
|
||||||
|
text = text.replace(group.group(0), replacewith, 1)
|
||||||
|
# Finally, [Classes] are around brackets, make them direct links
|
||||||
|
groups = re.finditer(r'\[(?P<class>[az0-AZ0_]+)\]', text)
|
||||||
|
for group in groups:
|
||||||
|
gd = group.groupdict()
|
||||||
|
replacewith = ("\"<code>{gclass}</code>(Go to page of class"
|
||||||
|
" {gclass})\":/class_{lkclass}".
|
||||||
|
format(gclass=gd["class"],
|
||||||
|
lkclass=gd["class"].lower()))
|
||||||
|
text = text.replace(group.group(0), replacewith, 1)
|
||||||
|
|
||||||
return text + "\n\n"
|
return text + "\n\n"
|
||||||
|
|
||||||
|
|
||||||
|
def mkfn(node, is_signal=False):
|
||||||
|
""" Return a string containing a unsorted item for a function
|
||||||
|
"""
|
||||||
|
finalstr = ""
|
||||||
|
name = node.attrib["name"]
|
||||||
|
rtype = node.find("return")
|
||||||
|
if rtype:
|
||||||
|
rtype = rtype.attrib["type"]
|
||||||
|
else:
|
||||||
|
rtype = "void"
|
||||||
|
# write the return type and the function name first
|
||||||
|
finalstr += "* "
|
||||||
|
# return type
|
||||||
|
if not is_signal:
|
||||||
|
if rtype != "void":
|
||||||
|
finalstr += " \"{rtype}({title})\":/class_{link} ".format(
|
||||||
|
rtype=rtype,
|
||||||
|
title="Go to page of class " + rtype,
|
||||||
|
link=rtype.lower())
|
||||||
|
else:
|
||||||
|
finalstr += " void "
|
||||||
|
|
||||||
|
# function name
|
||||||
|
if not is_signal:
|
||||||
|
finalstr += "\"*{funcname}*({title})\":#{link} <b>(</b> ".format(
|
||||||
|
funcname=name,
|
||||||
|
title="Jump to description for node " + name,
|
||||||
|
link=name.lower())
|
||||||
|
else:
|
||||||
|
# Signals have no description
|
||||||
|
finalstr += "*{funcname}* <b>(</b>".format(funcname=name)
|
||||||
|
# loop for the arguments of the function, if any
|
||||||
|
args = []
|
||||||
|
for arg in sorted(
|
||||||
|
node.iter(tag="argument"),
|
||||||
|
key=lambda a: int(a.attrib["index"])):
|
||||||
|
|
||||||
|
twd = "{type} {name}={default}"
|
||||||
|
tnd = "{type} {name}"
|
||||||
|
tlk = ("\"*{cls}*(Go to page of class {cls})"
|
||||||
|
"\":/class_{lcls}")
|
||||||
|
ntype = arg.attrib["type"]
|
||||||
|
nname = arg.attrib["name"]
|
||||||
|
|
||||||
|
if "default" in arg.attrib:
|
||||||
|
args.insert(
|
||||||
|
-1,
|
||||||
|
twd.format(
|
||||||
|
type=tlk.format(
|
||||||
|
cls=ntype,
|
||||||
|
lcls=ntype.lower()),
|
||||||
|
name=nname,
|
||||||
|
default=arg.attrib["default"]))
|
||||||
|
else:
|
||||||
|
# No default value present
|
||||||
|
args.insert(
|
||||||
|
-1,
|
||||||
|
tnd.format(
|
||||||
|
type=tlk.format(
|
||||||
|
cls=ntype,
|
||||||
|
lcls=ntype.lower()),
|
||||||
|
name=nname))
|
||||||
|
# join the arguments together
|
||||||
|
finalstr += ", ".join(args)
|
||||||
|
# and, close the function with a )
|
||||||
|
finalstr += " <b>)</b>"
|
||||||
|
# write the qualifier, if any
|
||||||
|
if "qualifiers" in node.attrib:
|
||||||
|
qualifier = node.attrib["qualifiers"]
|
||||||
|
finalstr += " " + qualifier
|
||||||
|
|
||||||
|
finalstr += "\n"
|
||||||
|
|
||||||
|
return finalstr
|
||||||
|
|
||||||
desc = "Generates documentation from a XML file to different markup languages"
|
desc = "Generates documentation from a XML file to different markup languages"
|
||||||
|
|
||||||
parser = argparse.ArgumentParser(description=desc)
|
parser = argparse.ArgumentParser(description=desc)
|
||||||
|
@ -100,18 +238,20 @@ if "version" not in root.attrib:
|
||||||
|
|
||||||
version = root.attrib["version"]
|
version = root.attrib["version"]
|
||||||
classes = sorted(root, key=sortkey)
|
classes = sorted(root, key=sortkey)
|
||||||
|
logging.debug("Number of classes: {}".format(len(classes)))
|
||||||
|
logging.debug("len(classes) / 2 + 1: {}".format(int(len(classes) / 2 + 1)))
|
||||||
# first column is always longer, second column of classes should be shorter
|
# first column is always longer, second column of classes should be shorter
|
||||||
zclasses = izip_longest(classes[:len(classes) / 2 + 1],
|
zclasses = zip_longest(classes[:int(len(classes) / 2 + 1)],
|
||||||
classes[len(classes) / 2 + 1:],
|
classes[int(len(classes) / 2 + 1):],
|
||||||
fillvalue="")
|
fillvalue="")
|
||||||
|
|
||||||
# We write the class_list file and also each class file at once
|
# We write the class_list file and also each class file at once
|
||||||
with open(path.join(args.outputdir, "class_list.txt"), "wb") as fcl:
|
with open(path.join(args.outputdir, "class_list.txt"), "wb") as fcl:
|
||||||
# Write header of table
|
# Write header of table
|
||||||
fcl.write("|^.\n")
|
fcl.write(tb("|^.\n"))
|
||||||
fcl.write("|_. Index symbol |_. Class name "
|
fcl.write(tb("|_. Index symbol |_. Class name "
|
||||||
"|_. Index symbol |_. Class name |\n")
|
"|_. Index symbol |_. Class name |\n"))
|
||||||
fcl.write("|-.\n")
|
fcl.write(tb("|-.\n"))
|
||||||
|
|
||||||
indexletterl = ""
|
indexletterl = ""
|
||||||
indexletterr = ""
|
indexletterr = ""
|
||||||
|
@ -120,65 +260,119 @@ with open(path.join(args.outputdir, "class_list.txt"), "wb") as fcl:
|
||||||
# write the index symbol column, left
|
# write the index symbol column, left
|
||||||
if indexletterl != gdclassl.attrib["name"][0]:
|
if indexletterl != gdclassl.attrib["name"][0]:
|
||||||
indexletterl = gdclassl.attrib["name"][0]
|
indexletterl = gdclassl.attrib["name"][0]
|
||||||
fcl.write("| *{}* |".format(indexletterl.upper()))
|
fcl.write(tb("| *{}* |".format(indexletterl.upper())))
|
||||||
else:
|
else:
|
||||||
# empty cell
|
# empty cell
|
||||||
fcl.write("| |")
|
fcl.write(tb("| |"))
|
||||||
# write the class name column, left
|
# write the class name column, left
|
||||||
fcl.write("\"{name}({title})\":/class_{link}".format(
|
fcl.write(tb("\"{name}({title})\":/class_{link}".format(
|
||||||
name=gdclassl.attrib["name"],
|
name=gdclassl.attrib["name"],
|
||||||
title="Go to page of class " + gdclassl.attrib["name"],
|
title="Go to page of class " + gdclassl.attrib["name"],
|
||||||
link="class_" + gdclassl.attrib["name"].lower()))
|
link="class_" + gdclassl.attrib["name"].lower())))
|
||||||
|
|
||||||
# write the index symbol column, right
|
# write the index symbol column, right
|
||||||
if isinstance(gdclassr, ElementTree.Element):
|
if isinstance(gdclassr, ElementTree.Element):
|
||||||
if indexletterr != gdclassr.attrib["name"][0]:
|
if indexletterr != gdclassr.attrib["name"][0]:
|
||||||
indexletterr = gdclassr.attrib["name"][0]
|
indexletterr = gdclassr.attrib["name"][0]
|
||||||
fcl.write("| *{}* |".format(indexletterr.upper()))
|
fcl.write(tb("| *{}* |".format(indexletterr.upper())))
|
||||||
else:
|
else:
|
||||||
# empty cell
|
# empty cell
|
||||||
fcl.write("| |")
|
fcl.write(tb("| |"))
|
||||||
# We are dealing with an empty string
|
# We are dealing with an empty string
|
||||||
else:
|
else:
|
||||||
# two empty cell
|
# two empty cell
|
||||||
fcl.write("| | |\n")
|
fcl.write(tb("| | |\n"))
|
||||||
# We won't get the name of the class since there is no ElementTree
|
# We won't get the name of the class since there is no ElementTree
|
||||||
# object for the right side of the tuple, so we iterate the next
|
# object for the right side of the tuple, so we iterate the next
|
||||||
# tuple instead
|
# tuple instead
|
||||||
continue
|
continue
|
||||||
|
|
||||||
# write the class name column (if any), right
|
# write the class name column (if any), right
|
||||||
fcl.write("\"{name}({title})\":/{link} |\n".format(
|
fcl.write(tb("\"{name}({title})\":/{link} |\n".format(
|
||||||
name=gdclassr.attrib["name"],
|
name=gdclassr.attrib["name"],
|
||||||
title="Go to page of class " + gdclassr.attrib["name"],
|
title="Go to page of class " + gdclassr.attrib["name"],
|
||||||
link=gdclassr.attrib["name"].lower()))
|
link=gdclassr.attrib["name"].lower())))
|
||||||
|
|
||||||
# row written #
|
# row written #
|
||||||
# now, let's write each class page for each class
|
# now, let's write each class page for each class
|
||||||
for gdclass in [gdclassl, gdclassr]:
|
for gdclass in [gdclassl, gdclassr]:
|
||||||
if not isinstance(ElementTree.Element, gdclass):
|
if not isinstance(gdclass, ElementTree.Element):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
classname = gdclass.attrib["name"]
|
classname = gdclass.attrib["name"]
|
||||||
with open(path.join(args.outputdir, "{}.txt".format(
|
with open(path.join(args.outputdir, "{}.txt".format(
|
||||||
classname.lower())), "wb") as clsf:
|
classname.lower())), "wb") as clsf:
|
||||||
# First level header with the name of the class
|
# First level header with the name of the class
|
||||||
clsf.write("h1. {}\n".format(classname))
|
clsf.write(tb("h1. {}\n\n".format(classname)))
|
||||||
# lay the attributes
|
# lay the attributes
|
||||||
if "inherits" in gdclass.attrib:
|
if "inherits" in gdclass.attrib:
|
||||||
inh = gdclass.attrib["inherits"].strip()
|
inh = gdclass.attrib["inherits"].strip()
|
||||||
clsf.write(
|
clsf.write(tb(("h4. Inherits: \"<code>{name}"
|
||||||
"*Inherits:* \"{name}({title})\":/class_{link}\n".
|
"</code>({title})\":/class_{link}\n\n").
|
||||||
format(
|
format(
|
||||||
name=classname,
|
name=inh,
|
||||||
title="Go to page of class " + classname,
|
title="Go to page of class " + inh,
|
||||||
link=classname.lower()))
|
link=classname.lower())))
|
||||||
if "category" in gdclass.attrib:
|
if "category" in gdclass.attrib:
|
||||||
clsf.write("*Category:* {}".
|
clsf.write(tb("h4. Category: {}\n\n".
|
||||||
format(gdclass.attrib["category"].strip()))
|
format(gdclass.attrib["category"].strip())))
|
||||||
# lay child nodes
|
# lay child nodes
|
||||||
for gdchild in gdclass.iter():
|
briefd = gdclass.find("brief_description")
|
||||||
if gdchild.tag == "brief_description":
|
logging.debug(
|
||||||
clsf.write("h2. Brief Description\n")
|
"Brief description was found?: {}".format(briefd))
|
||||||
clsf.write(toOP(gdchild.text.strip()))
|
if briefd.text.strip():
|
||||||
# convert commands in text
|
logging.debug(
|
||||||
|
"Text: {}".format(briefd.text.strip()))
|
||||||
|
clsf.write(tb("h2. Brief Description\n\n"))
|
||||||
|
clsf.write(tb(toOP(briefd.text.strip()) +
|
||||||
|
"\"read more\":#more\n\n"))
|
||||||
|
|
||||||
|
# Write the list of member functions of this class
|
||||||
|
methods = gdclass.find("methods")
|
||||||
|
if methods and len(methods) > 0:
|
||||||
|
clsf.write(tb("\nh3. Member Functions\n\n"))
|
||||||
|
for method in methods.iter(tag='method'):
|
||||||
|
clsf.write(tb(mkfn(method)))
|
||||||
|
|
||||||
|
signals = gdclass.find("signals")
|
||||||
|
if signals and len(signals) > 0:
|
||||||
|
clsf.write(tb("\nh3. Signals\n\n"))
|
||||||
|
for signal in signals.iter(tag='signal'):
|
||||||
|
clsf.write(tb(mkfn(signal, True)))
|
||||||
|
# TODO: <members> tag is necessary to process? it does not
|
||||||
|
# exists in class_list.xml file.
|
||||||
|
|
||||||
|
consts = gdclass.find("constants")
|
||||||
|
if consts and len(consts) > 0:
|
||||||
|
clsf.write(tb("\nh3. Numeric Constants\n\n"))
|
||||||
|
for const in sorted(consts, key=lambda k:
|
||||||
|
k.attrib["name"]):
|
||||||
|
if const.text.strip():
|
||||||
|
clsf.write(tb("* *{name}* = *{value}* - {desc}\n".
|
||||||
|
format(
|
||||||
|
name=const.attrib["name"],
|
||||||
|
value=const.attrib["value"],
|
||||||
|
desc=const.text.strip())))
|
||||||
|
else:
|
||||||
|
# Constant have no description
|
||||||
|
clsf.write(tb("* *{name}* = *{value}*\n".
|
||||||
|
format(
|
||||||
|
name=const.attrib["name"],
|
||||||
|
value=const.attrib["value"])))
|
||||||
|
descrip = gdclass.find("description")
|
||||||
|
clsf.write(tb("\nh3(#more). Description\n\n"))
|
||||||
|
if descrip.text:
|
||||||
|
clsf.write(tb(descrip.text.strip() + "\n"))
|
||||||
|
else:
|
||||||
|
clsf.write(tb("_Nothing here, yet..._\n"))
|
||||||
|
|
||||||
|
# and finally, the description for each method
|
||||||
|
if methods and len(methods) > 0:
|
||||||
|
clsf.write(tb("\nh3. Member Function Description\n\n"))
|
||||||
|
for method in methods.iter(tag='method'):
|
||||||
|
clsf.write(tb("h4(#{n}). {name}\n\n".format(
|
||||||
|
n=method.attrib["name"].lower(),
|
||||||
|
name=method.attrib["name"])))
|
||||||
|
clsf.write(tb(mkfn(method) + "\n"))
|
||||||
|
clsf.write(tb(toOP(method.find(
|
||||||
|
"description").text.strip())))
|
||||||
|
|
Loading…
Reference in a new issue