291 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			291 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #!/usr/bin/env python
 | |
| 
 | |
| import os, sys, re, string, glob
 | |
| from optparse import OptionParser
 | |
| 
 | |
| # Black list for classes and methods that does not implemented in Java API
 | |
| # Created to exclude referencies to them in @see tag
 | |
| JAVADOC_ENTITY_BLACK_LIST = set(["org.opencv.core.Core#abs", \
 | |
|                                  "org.opencv.core.Core#theRNG", \
 | |
|                                  "org.opencv.core.Core#extractImageCOI", \
 | |
|                                  "org.opencv.core.PCA", \
 | |
|                                  "org.opencv.core.SVD", \
 | |
|                                  "org.opencv.core.RNG", \
 | |
|                                  "org.opencv.imgproc.Imgproc#createMorphologyFilter", \
 | |
|                                  "org.opencv.imgproc.Imgproc#createLinearFilter", \
 | |
|                                  "org.opencv.imgproc.Imgproc#createSeparableLinearFilter", \
 | |
|                                  "org.opencv.imgproc.FilterEngine"])
 | |
| 
 | |
| class JavadocGenerator(object):
 | |
|     def __init__(self, definitions = {}, modules= [], javadoc_marker = "//javadoc:"):
 | |
|         self.definitions = definitions
 | |
|         self.javadoc_marker = javadoc_marker
 | |
|         self.markers_processed = 0
 | |
|         self.markers_documented = 0
 | |
|         self.params_documented = 0
 | |
|         self.params_undocumented = 0
 | |
|         self.known_modules = modules
 | |
|         self.verbose = False
 | |
|         self.show_warnings = True
 | |
|         self.show_errors = True
 | |
| 
 | |
|     def parceJavadocMarker(self, line):
 | |
|         assert line.lstrip().startswith(self.javadoc_marker)
 | |
|         offset = line[:line.find(self.javadoc_marker)]
 | |
|         line = line.strip()[len(self.javadoc_marker):]
 | |
|         args_start = line.rfind("(")
 | |
|         args_end = line.rfind(")")
 | |
|         assert args_start * args_end > 0
 | |
|         if args_start >= 0:
 | |
|             assert args_start < args_end
 | |
|             name = line[:args_start].strip()
 | |
|             if name.startswith("java"):
 | |
|                 name = name[4:]
 | |
|             return (name, offset,  filter(None, list(arg.strip() for arg in line[args_start+1:args_end].split(","))))
 | |
|         name = line.strip()
 | |
|         if name.startswith("java"):
 | |
|             name = name[4:]
 | |
|         return (name, offset, [])
 | |
| 
 | |
|     def document(self, infile, outfile):
 | |
|         inf = open(infile, "rt")
 | |
|         outf = open(outfile, "wt")
 | |
|         module = os.path.splitext(os.path.basename(infile))[0].split("+")[0]
 | |
|         if module not in self.known_modules:
 | |
|             module = "unknown"
 | |
|         try:
 | |
|             for l in inf.readlines():
 | |
|                 org = l
 | |
|                 l = l.replace(" ", "").replace("\t", "")#remove all whitespace
 | |
|                 if l.startswith(self.javadoc_marker):
 | |
|                     marker = self.parceJavadocMarker(l)
 | |
|                     self.markers_processed += 1
 | |
|                     decl = self.definitions.get(marker[0],None)
 | |
|                     if decl:
 | |
|                         javadoc = self.makeJavadoc(decl, marker[2])
 | |
|                         if self.verbose:
 | |
|                             print
 | |
|                             print("Javadoc for \"%s\" File: %s line %s" % (decl["name"], decl["file"], decl["line"]))
 | |
|                             print(javadoc)
 | |
|                         for line in javadoc.split("\n"):
 | |
|                             outf.write(marker[1] + line + "\n")
 | |
|                         self.markers_documented += 1
 | |
|                     elif self.show_errors:
 | |
|                          sys.stderr.write("gen_javadoc error: could not find documentation for %s (module: %s)" % (l.lstrip()[len(self.javadoc_marker):-1].strip(), module))
 | |
|                 else:
 | |
|                     outf.write(org.replace("\t", "    ").rstrip()+"\n")
 | |
|         except:
 | |
|             inf.close()
 | |
|             outf.close()
 | |
|             os.remove(outfile)
 | |
|             raise
 | |
|         else:
 | |
|             inf.close()
 | |
|             outf.close()
 | |
| 
 | |
|     def FinishParagraph(self, text):
 | |
|         return text[:-1] + "</p>\n"
 | |
| 
 | |
|     def ReformatForJavadoc(self, s):
 | |
|         out = ""
 | |
|         in_paragraph = False
 | |
|         in_list = False
 | |
|         for term in s.split("\n"):
 | |
|             in_list_item = False
 | |
|             if term.startswith("*"):
 | |
|                 in_list_item = True
 | |
|                 if in_paragraph:
 | |
|                     out = self.FinishParagraph(out)
 | |
|                     in_paragraph = False
 | |
|                 if not in_list:
 | |
|                     out += " * <ul>\n"
 | |
|                     in_list = True
 | |
|                 term = "  <li>" + term[1:]
 | |
| 
 | |
|             if term.startswith("#."):
 | |
|                 in_list_item = True
 | |
|                 if in_paragraph:
 | |
|                     out = self.FinishParagraph(out)
 | |
|                     in_paragraph = False
 | |
|                 if not in_list:
 | |
|                     out += " * <ul>\n"
 | |
|                     in_list = True
 | |
|                 term = "  <li>" + term[2:]
 | |
| 
 | |
|             if not term:
 | |
|                 if in_paragraph:
 | |
|                     out = self.FinishParagraph(out)
 | |
|                     in_paragraph = False
 | |
|                 out += " *\n"
 | |
|             else:
 | |
|                 if in_list and not in_list_item:
 | |
|                     in_list = False
 | |
|                     if out.endswith(" *\n"):
 | |
|                         out = out[:-3] + " * </ul>\n *\n"
 | |
|                     else:
 | |
|                         out += " * </ul>\n"
 | |
|                 pos_start = 0
 | |
|                 pos_end = min(77, len(term)-1)
 | |
|                 while pos_start < pos_end:
 | |
|                     if pos_end - pos_start == 77:
 | |
|                         while pos_end >= pos_start+60:
 | |
|                             if not term[pos_end].isspace():
 | |
|                                 pos_end -= 1
 | |
|                             else:
 | |
|                                 break
 | |
|                         if pos_end < pos_start+60:
 | |
|                             pos_end = min(pos_start + 77, len(term)-1)
 | |
|                             while pos_end < len(term):
 | |
|                                 if not term[pos_end].isspace():
 | |
|                                     pos_end += 1
 | |
|                                 else:
 | |
|                                     break
 | |
|                     if in_paragraph or term.startswith("@") or in_list_item:
 | |
|                         out += " * "
 | |
|                     else:
 | |
|                         in_paragraph = True
 | |
|                         out += " * <p>"
 | |
|                     out += term[pos_start:pos_end+1].rstrip() + "\n"
 | |
|                     pos_start = pos_end + 1
 | |
|                     pos_end = min(pos_start + 77, len(term)-1)
 | |
| 
 | |
|         if in_paragraph:
 | |
|             out = self.FinishParagraph(out)
 | |
|         if in_list:
 | |
|             out += " * </ul>\n"
 | |
|         return out
 | |
| 
 | |
|     def getJavaName(self, decl, methodSeparator = "."):
 | |
|         name = "org.opencv."
 | |
|         name += decl["module"]
 | |
|         if "class" in decl:
 | |
|             name += "." + decl["class"]
 | |
|         else:
 | |
|             name += "." + decl["module"].capitalize()
 | |
|         if "method" in decl:
 | |
|             name += methodSeparator + decl["method"]
 | |
|         return name
 | |
| 
 | |
|     def getDocURL(self, decl):
 | |
|         url = "http://docs.opencv.org/modules/"
 | |
|         url += decl["module"]
 | |
|         url += "/doc/"
 | |
|         url += os.path.basename(decl["file"]).replace(".rst",".html")
 | |
|         url += "#" + decl["name"].replace("::","-").replace("()","").replace("=","").strip().rstrip("_").replace(" ","-").replace("_","-").lower()
 | |
|         return url
 | |
| 
 | |
|     def makeJavadoc(self, decl, args = None):
 | |
|         doc = ""
 | |
|         prefix = "/**\n"
 | |
| 
 | |
|         if decl.get("isclass", False):
 | |
|             decl_type = "class"
 | |
|         elif decl.get("isstruct", False):
 | |
|             decl_type = "struct"
 | |
|         elif "class" in decl:
 | |
|             decl_type = "method"
 | |
|         else:
 | |
|             decl_type = "function"
 | |
| 
 | |
|         # brief goes first
 | |
|         if "brief" in decl:
 | |
|             doc += prefix + self.ReformatForJavadoc(decl["brief"])
 | |
|             prefix = " *\n"
 | |
|         elif "long" not in decl:
 | |
|             if self.show_warnings:
 | |
|                 print >> sys.stderr, "gen_javadoc warning: no description for " + decl_type + " \"%s\" File: %s (line %s)" % (func["name"], func["file"], func["line"])
 | |
|             doc += prefix + self.ReformatForJavadoc("This " + decl_type + " is undocumented")
 | |
|             prefix = " *\n"
 | |
| 
 | |
|         # long goes after brief
 | |
|         if "long" in decl:
 | |
|             doc += prefix  + self.ReformatForJavadoc(decl["long"])
 | |
|             prefix = " *\n"
 | |
| 
 | |
|         # @param tags
 | |
|         if args and (decl_type == "method" or decl_type == "function"):
 | |
|             documented_params = decl.get("params",{})
 | |
|             for arg in args:
 | |
|                 arg_doc = documented_params.get(arg, None)
 | |
|                 if not arg_doc:
 | |
|                     arg_doc = "a " + arg
 | |
|                     if self.show_warnings:
 | |
|                         sys.stderr.write("gen_javadoc warning: parameter \"%s\" of \"%s\" is undocumented. File: %s (line %s)" % (arg, decl["name"], decl["file"], decl["line"]))
 | |
|                     self.params_undocumented += 1
 | |
|                 else:
 | |
|                     self.params_documented += 1
 | |
|                 doc += prefix + self.ReformatForJavadoc("@param " + arg + " " + arg_doc)
 | |
|                 prefix = ""
 | |
|             prefix = " *\n"
 | |
| 
 | |
|         # @see tags
 | |
|         # always link to documentation
 | |
|         doc += prefix + " * @see <a href=\"" + self.getDocURL(decl) + "\">" + self.getJavaName(decl) + "</a>\n"
 | |
|         prefix = ""
 | |
|         # other links
 | |
|         if "seealso" in decl:
 | |
|             for see in decl["seealso"]:
 | |
|                 seedecl = self.definitions.get(see,None)
 | |
|                 if seedecl:
 | |
|                     javadoc_name = self.getJavaName(seedecl, "#")
 | |
|                     if (javadoc_name not in JAVADOC_ENTITY_BLACK_LIST):
 | |
|                         doc += prefix + " * @see " + javadoc_name + "\n"
 | |
|         prefix = " *\n"
 | |
| 
 | |
|         #doc += prefix + " * File: " + decl["file"] + " (line " + str(decl["line"]) + ")\n"
 | |
| 
 | |
|         return (doc + " */").replace("::",".")
 | |
| 
 | |
|     def printSummary(self):
 | |
|         print("Javadoc Generator Summary:")
 | |
|         print("  Total markers:        %s" % self.markers_processed)
 | |
|         print( "  Undocumented markers: %s" % (self.markers_processed - self.markers_documented))
 | |
|         print( "  Generated comments:   %s" % self.markers_documented)
 | |
| 
 | |
|         print
 | |
|         print("  Documented params:    %s" % self.params_documented)
 | |
|         print("  Undocumented params:  %s" % self.params_undocumented)
 | |
|         print
 | |
| 
 | |
| if __name__ == "__main__":
 | |
| 
 | |
|     selfpath = os.path.dirname(os.path.abspath(sys.argv[0]))
 | |
|     hdr_parser_path = os.path.join(selfpath, "../../python/src2")
 | |
| 
 | |
|     sys.path.append(selfpath)
 | |
|     sys.path.append(hdr_parser_path)
 | |
|     import hdr_parser
 | |
|     import rst_parser
 | |
| 
 | |
|     parser = OptionParser()
 | |
|     parser.add_option("-v", "--verbose", dest="verbose", help="Print verbose log to stdout", action="store_true", default=False)
 | |
|     parser.add_option("", "--no-warnings", dest="warnings", help="Hide warning messages", action="store_false", default=True)
 | |
|     parser.add_option("", "--no-errors", dest="errors", help="Hide error messages", action="store_false", default=True)
 | |
|     parser.add_option("", "--modules", dest="modules", help="comma-separated list of modules to generate comments", metavar="MODS", default=",".join(rst_parser.allmodules))
 | |
| 
 | |
|     (options, args) = parser.parse_args(sys.argv)
 | |
|     options.modules = options.modules.split(",")
 | |
| 
 | |
|     if len(args) < 2 or len(options.modules) < 1:
 | |
|         parser.print_help()
 | |
|         exit(0)
 | |
| 
 | |
|     parser = rst_parser.RstParser(hdr_parser.CppHeaderParser())
 | |
|     for m in options.modules:
 | |
|         parser.parse(m, os.path.join(selfpath, "../../" + m))
 | |
| 
 | |
|     parser.printSummary()
 | |
| 
 | |
|     generator = JavadocGenerator(parser.definitions, options.modules)
 | |
|     generator.verbose = options.verbose
 | |
|     generator.show_warnings = options.warnings
 | |
|     generator.show_errors = options.errors
 | |
| 
 | |
|     for path in args:
 | |
|         folder = os.path.abspath(path)
 | |
|         for jfile in [f for f in glob.glob(os.path.join(folder,"*.java")) if not f.endswith("-jdoc.java")]:
 | |
|             outfile = os.path.abspath(os.path.basename(jfile).replace(".java", "-jdoc.java"))
 | |
|             generator.document(jfile, outfile)
 | |
| 
 | |
|     generator.printSummary()
 | 
