2013-04-11 16:34:04 +02:00
#!/usr/bin/env python
2012-11-23 19:57:22 +01:00
2012-07-05 07:11:15 +02:00
import os , sys , re , string , fnmatch
2013-08-20 15:34:32 +02:00
allmodules = [ " core " , " flann " , " imgproc " , " ml " , " highgui " , " video " , " features2d " , " calib3d " , " objdetect " , " legacy " , " contrib " , " cuda " , " androidcamera " , " java " , " python " , " stitching " , " ts " , " photo " , " nonfree " , " videostab " , " ocl " , " softcascade " , " superres " ]
2011-07-11 15:33:05 +02:00
verbose = False
show_warnings = True
show_errors = True
2012-05-28 09:34:49 +02:00
show_critical_errors = True
2011-07-05 19:29:54 +02:00
2012-03-30 14:07:45 +02:00
params_blacklist = {
" fromarray " : ( " object " , " allowND " ) , # python only function
" reprojectImageTo3D " : ( " ddepth " ) , # python only argument
" composeRT " : ( " d*d* " ) , # wildchards in parameter names are not supported by this parser
2012-11-12 11:42:28 +01:00
" CvSVM::train_auto " : ( " \\ *Grid " ) , # wildchards in parameter names are not supported by this parser
2012-03-30 14:07:45 +02:00
" error " : " args " , # parameter of supporting macro
" getConvertElem " : ( " from " , " cn " , " to " , " beta " , " alpha " ) , # arguments of returned functions
" gpu::swapChannels " : ( " dstOrder " ) # parameter is not parsed correctly by the hdr_parser
}
2012-05-29 12:36:19 +02:00
ERROR_001_SECTIONFAILURE = 1
2012-05-29 12:57:25 +02:00
WARNING_002_HDRWHITESPACE = 2
2012-05-29 12:36:19 +02:00
ERROR_003_PARENTHESES = 3
WARNING_004_TABS = 4
ERROR_005_REDEFENITIONPARAM = 5
ERROR_006_REDEFENITIONFUNC = 6
WARNING_007_UNDOCUMENTEDPARAM = 7
WARNING_008_MISSINGPARAM = 8
WARNING_009_HDRMISMATCH = 9
ERROR_010_NOMODULE = 10
2012-05-29 12:57:25 +02:00
ERROR_011_EOLEXPECTED = 11
2012-05-29 12:36:19 +02:00
2012-03-30 14:07:45 +02:00
params_mapping = {
" composeRT " : {
" dr3dr1 " : " d*d* " ,
" dr3dr2 " : " d*d* " ,
" dr3dt1 " : " d*d* " ,
" dr3dt2 " : " d*d* " ,
" dt3dr1 " : " d*d* " ,
" dt3dr2 " : " d*d* " ,
" dt3dt1 " : " d*d* " ,
" dt3dt2 " : " d*d* "
} ,
" CvSVM::train_auto " : {
" coeffGrid " : " \\ *Grid " ,
" degreeGrid " : " \\ *Grid " ,
" gammaGrid " : " \\ *Grid " ,
" nuGrid " : " \\ *Grid " ,
" pGrid " : " \\ *Grid "
}
}
2012-07-05 09:30:45 +02:00
known_text_sections_names = [ " Appendix " , " Results " , " Prerequisites " , " Introduction " , " Description " ]
2011-07-05 19:29:54 +02:00
class DeclarationParser ( object ) :
def __init__ ( self , line = None ) :
if line is None :
self . fdecl = " "
self . lang = " "
self . balance = 0
return
self . lang = self . getLang ( line )
assert self . lang is not None
self . fdecl = line [ line . find ( " :: " ) + 2 : ] . strip ( )
self . balance = self . fdecl . count ( " ( " ) - self . fdecl . count ( " ) " )
assert self . balance > = 0
def append ( self , line ) :
self . fdecl + = line
self . balance = self . fdecl . count ( " ( " ) - self . fdecl . count ( " ) " )
def isready ( self ) :
return self . balance == 0
2012-11-12 11:42:28 +01:00
@classmethod
def getLang ( cls , line ) :
2011-07-05 19:29:54 +02:00
if line . startswith ( " .. ocv:function:: " ) :
return " C++ "
if line . startswith ( " .. ocv:cfunction:: " ) :
return " C "
if line . startswith ( " .. ocv:pyfunction:: " ) :
return " Python2 "
if line . startswith ( " .. ocv:jfunction:: " ) :
return " Java "
return None
2012-05-28 09:34:49 +02:00
2011-07-05 19:29:54 +02:00
def hasDeclaration ( self , line ) :
return self . getLang ( line ) is not None
class ParamParser ( object ) :
def __init__ ( self , line = None ) :
if line is None :
self . prefix = " "
self . name = " "
self . comment = " "
self . active = False
return
offset = line . find ( " :param " )
assert offset > 0
self . prefix = line [ : offset ]
2012-11-12 11:42:28 +01:00
assert self . prefix == " " * len ( self . prefix ) , " :param definition should be prefixed with spaces "
2011-07-05 19:29:54 +02:00
line = line [ offset + 6 : ] . lstrip ( )
name_end = line . find ( " : " )
assert name_end > 0
self . name = line [ : name_end ]
self . comment = line [ name_end + 1 : ] . lstrip ( )
self . active = True
def append ( self , line ) :
assert self . active
if ( self . hasDeclaration ( line ) ) :
self . active = False
elif line . startswith ( self . prefix ) or not line :
self . comment + = " \n " + line . lstrip ( )
else :
self . active = False
2012-05-28 09:34:49 +02:00
2012-11-12 11:42:28 +01:00
@classmethod
def hasDeclaration ( cls , line ) :
2011-07-05 19:29:54 +02:00
return line . lstrip ( ) . startswith ( " :param " )
class RstParser ( object ) :
def __init__ ( self , cpp_parser ) :
self . cpp_parser = cpp_parser
self . definitions = { }
2011-07-07 13:23:51 +02:00
self . sections_parsed = 0
self . sections_total = 0
self . sections_skipped = 0
2011-07-05 19:29:54 +02:00
2011-07-07 13:23:51 +02:00
def parse ( self , module_name , module_path = None ) :
if module_path is None :
module_path = " ../ " + module_name
2012-07-05 07:11:15 +02:00
doclist = [ ]
for root , dirs , files in os . walk ( os . path . join ( module_path , " doc " ) ) :
for filename in fnmatch . filter ( files , " *.rst " ) :
doclist . append ( os . path . join ( root , filename ) )
2012-07-05 09:30:45 +02:00
2011-07-05 19:29:54 +02:00
for doc in doclist :
2011-07-07 13:23:51 +02:00
self . parse_rst_file ( module_name , doc )
2012-05-28 09:34:49 +02:00
2011-07-07 13:23:51 +02:00
def parse_section_safe ( self , module_name , section_name , file_name , lineno , lines ) :
try :
self . parse_section ( module_name , section_name , file_name , lineno , lines )
except AssertionError , args :
2011-07-11 15:33:05 +02:00
if show_errors :
2012-05-29 12:36:19 +02:00
print >> sys . stderr , " RST parser error E %03d : assertion in \" %s \" at %s : %s " % ( ERROR_001_SECTIONFAILURE , section_name , file_name , lineno )
2011-07-11 15:33:05 +02:00
print >> sys . stderr , " Details: %s " % args
2011-07-07 13:23:51 +02:00
def parse_section ( self , module_name , section_name , file_name , lineno , lines ) :
self . sections_total + = 1
# skip sections having whitespace in name
2012-05-29 12:36:19 +02:00
#if section_name.find(" ") >= 0 and section_name.find("::operator") < 0:
2012-07-05 09:30:45 +02:00
if ( section_name . find ( " " ) > = 0 and not bool ( re . match ( r " ( \ w+::)*operator \ s*( \ w+|>>|<<| \ ( \ )|->| \ + \ +|--|=|==| \ +=|-=) " , section_name ) ) ) or section_name . endswith ( " : " ) :
2011-07-11 15:33:05 +02:00
if show_errors :
2012-05-29 19:13:24 +02:00
print >> sys . stderr , " RST parser warning W %03d : SKIPPED: \" %s \" File: %s : %s " % ( WARNING_002_HDRWHITESPACE , section_name , file_name , lineno )
2011-07-07 13:23:51 +02:00
self . sections_skipped + = 1
return
2011-07-05 19:29:54 +02:00
func = { }
func [ " name " ] = section_name
func [ " file " ] = file_name
func [ " line " ] = lineno
2011-07-07 13:23:51 +02:00
func [ " module " ] = module_name
2011-07-05 19:29:54 +02:00
# parse section name
2011-07-07 13:23:51 +02:00
section_name = self . parse_namespace ( func , section_name )
class_separator_idx = section_name . find ( " :: " )
2011-07-05 19:29:54 +02:00
if class_separator_idx > 0 :
2011-07-07 13:23:51 +02:00
func [ " class " ] = section_name [ : class_separator_idx ]
func [ " method " ] = section_name [ class_separator_idx + 2 : ]
2011-07-05 19:29:54 +02:00
else :
2011-07-07 13:23:51 +02:00
func [ " method " ] = section_name
2011-07-05 19:29:54 +02:00
2011-07-07 18:33:02 +02:00
capturing_seealso = False
2011-07-05 19:29:54 +02:00
skip_code_lines = False
expected_brief = True
2012-10-11 17:20:20 +02:00
was_code_line = False
2011-07-05 19:29:54 +02:00
fdecl = DeclarationParser ( )
pdecl = ParamParser ( )
2012-11-12 11:42:28 +01:00
ll = None
2011-07-05 19:29:54 +02:00
for l in lines :
# read tail of function/method declaration if needed
if not fdecl . isready ( ) :
fdecl . append ( ll )
if fdecl . isready ( ) :
self . add_new_fdecl ( func , fdecl )
continue
2011-07-07 18:33:02 +02:00
# continue capture seealso
if capturing_seealso :
2011-07-08 06:31:13 +02:00
if not l or l . startswith ( " " ) :
2012-11-12 11:42:28 +01:00
seealso = func . get ( " seealso " , [ ] )
2011-07-07 18:33:02 +02:00
seealso . extend ( l . split ( " , " ) )
func [ " seealso " ] = seealso
2011-07-05 19:29:54 +02:00
continue
else :
2011-07-07 18:33:02 +02:00
capturing_seealso = False
2011-07-05 19:29:54 +02:00
2011-07-07 18:33:02 +02:00
ll = l . strip ( )
if ll == " .. " :
expected_brief = False
skip_code_lines = False
continue
# skip lines if line-skipping mode is activated
if skip_code_lines :
2012-10-17 16:20:46 +02:00
if not l :
continue
2012-11-12 11:42:28 +01:00
if not l . startswith ( " " ) :
2011-07-07 18:33:02 +02:00
skip_code_lines = False
2012-05-28 09:34:49 +02:00
2011-07-11 15:33:05 +02:00
if ll . startswith ( " .. code-block:: " ) or ll . startswith ( " .. image:: " ) :
2011-07-07 18:33:02 +02:00
skip_code_lines = True
2012-10-17 01:18:30 +02:00
2011-07-07 18:33:02 +02:00
continue
2012-05-28 09:34:49 +02:00
2011-07-05 19:29:54 +02:00
# todo: parse structure members; skip them for now
if ll . startswith ( " .. ocv:member:: " ) :
2012-10-17 01:18:30 +02:00
#print ll
2011-07-05 19:29:54 +02:00
skip_code_lines = True
continue
2012-05-28 09:34:49 +02:00
2011-07-11 15:33:05 +02:00
#ignore references (todo: collect them)
if l . startswith ( " .. [ " ) :
continue
2011-07-05 19:29:54 +02:00
2011-07-08 06:31:13 +02:00
if ll . startswith ( " .. " ) :
expected_brief = False
elif ll . endswith ( " :: " ) :
# turn on line-skipping mode for code fragments
2012-10-11 17:20:20 +02:00
#print ll
2011-07-08 06:31:13 +02:00
skip_code_lines = True
ll = ll [ : len ( ll ) - 2 ]
2012-05-28 09:34:49 +02:00
2011-07-11 15:33:05 +02:00
# continue param parsing (process params after processing .. at the beginning of the line and :: at the end)
if pdecl . active :
pdecl . append ( l )
if pdecl . active :
continue
else :
self . add_new_pdecl ( func , pdecl )
# do not continue - current line can contain next parameter definition
2011-07-08 06:31:13 +02:00
2011-07-07 18:33:02 +02:00
# parse ".. seealso::" blocks
if ll . startswith ( " .. seealso:: " ) :
if ll . endswith ( " .. seealso:: " ) :
capturing_seealso = True
else :
2012-11-12 11:42:28 +01:00
seealso = func . get ( " seealso " , [ ] )
2011-07-07 18:33:02 +02:00
seealso . extend ( ll [ ll . find ( " :: " ) + 2 : ] . split ( " , " ) )
func [ " seealso " ] = seealso
continue
# skip ".. index::"
if ll . startswith ( " .. index:: " ) :
continue
2011-07-07 16:49:24 +02:00
2011-07-05 19:29:54 +02:00
# parse class & struct definitions
if ll . startswith ( " .. ocv:class:: " ) :
func [ " class " ] = ll [ ll . find ( " :: " ) + 2 : ] . strip ( )
if " method " in func :
del func [ " method " ]
func [ " isclass " ] = True
expected_brief = True
continue
if ll . startswith ( " .. ocv:struct:: " ) :
func [ " class " ] = ll [ ll . find ( " :: " ) + 2 : ] . strip ( )
if " method " in func :
del func [ " method " ]
func [ " isstruct " ] = True
expected_brief = True
continue
# parse function/method definitions
if fdecl . hasDeclaration ( ll ) :
fdecl = DeclarationParser ( ll )
if fdecl . isready ( ) :
self . add_new_fdecl ( func , fdecl )
continue
# parse parameters
if pdecl . hasDeclaration ( l ) :
pdecl = ParamParser ( l )
continue
# record brief description
if expected_brief :
func [ " brief " ] = func . get ( " brief " , " " ) + " \n " + ll
if skip_code_lines :
2011-07-07 18:33:02 +02:00
expected_brief = False # force end brief if code block begins
2011-07-05 19:29:54 +02:00
continue
# record other lines as long description
2012-10-11 17:20:20 +02:00
if ( skip_code_lines ) :
2012-10-17 16:20:46 +02:00
ll = ll . replace ( " /* " , " / * " )
ll = ll . replace ( " */ " , " * / " )
if ( was_code_line ) :
func [ " long " ] = func . get ( " long " , " " ) + " \n " + ll + " \n "
else :
2012-11-12 11:42:28 +01:00
was_code_line = True
2012-10-17 16:20:46 +02:00
func [ " long " ] = func . get ( " long " , " " ) + ll + " \n <code> \n \n // C++ code: \n \n "
else :
if ( was_code_line ) :
2012-11-12 11:42:28 +01:00
func [ " long " ] = func . get ( " long " , " " ) + " \n " + ll + " \n </code> \n "
was_code_line = False
2012-10-17 16:20:46 +02:00
else :
func [ " long " ] = func . get ( " long " , " " ) + " \n " + ll
2011-07-05 19:29:54 +02:00
# endfor l in lines
2012-05-28 09:34:49 +02:00
2011-07-07 13:23:51 +02:00
if fdecl . balance != 0 :
2012-05-28 09:34:49 +02:00
if show_critical_errors :
2012-05-29 12:36:19 +02:00
print >> sys . stderr , " RST parser error E %03d : invalid parentheses balance in \" %s \" at %s : %s " % ( ERROR_003_PARENTHESES , section_name , file_name , lineno )
2011-07-11 15:33:05 +02:00
return
2011-07-05 19:29:54 +02:00
# save last parameter if needed
if pdecl . active :
self . add_new_pdecl ( func , pdecl )
# add definition to list
func = self . normalize ( func )
if self . validate ( func ) :
self . definitions [ func [ " name " ] ] = func
2011-07-07 13:23:51 +02:00
self . sections_parsed + = 1
2011-07-11 15:33:05 +02:00
if verbose :
self . print_info ( func )
2011-07-05 19:29:54 +02:00
elif func :
2012-07-05 09:30:45 +02:00
if func [ " name " ] in known_text_sections_names :
if show_errors :
print >> sys . stderr , " RST parser warning W %03d : SKIPPED: \" %s \" File: %s : %s " % ( WARNING_002_HDRWHITESPACE , section_name , file_name , lineno )
self . sections_skipped + = 1
elif show_errors :
2011-07-11 15:33:05 +02:00
self . print_info ( func , True , sys . stderr )
2011-07-05 19:29:54 +02:00
2011-07-07 13:23:51 +02:00
def parse_rst_file ( self , module_name , doc ) :
2011-07-05 19:29:54 +02:00
doc = os . path . abspath ( doc )
lineno = 0
2011-07-07 13:23:51 +02:00
whitespace_warnings = 0
max_whitespace_warnings = 10
2012-05-28 09:34:49 +02:00
2011-07-05 19:29:54 +02:00
lines = [ ]
flineno = 0
fname = " "
prev_line = None
df = open ( doc , " rt " )
for l in df . readlines ( ) :
lineno + = 1
2011-07-07 13:23:51 +02:00
# handle tabs
if l . find ( " \t " ) > = 0 :
whitespace_warnings + = 1
2011-07-11 15:33:05 +02:00
if whitespace_warnings < = max_whitespace_warnings and show_warnings :
2012-05-29 12:36:19 +02:00
print >> sys . stderr , " RST parser warning W %03d : tab symbol instead of space is used at %s : %s " % ( WARNING_004_TABS , doc , lineno )
2011-07-07 13:23:51 +02:00
l = l . replace ( " \t " , " " )
2012-05-28 09:34:49 +02:00
2011-07-07 13:23:51 +02:00
# handle first line
2011-07-05 19:29:54 +02:00
if prev_line == None :
prev_line = l . rstrip ( )
continue
2011-07-07 13:23:51 +02:00
2011-07-05 19:29:54 +02:00
ll = l . rstrip ( )
2012-09-07 22:40:06 +02:00
if len ( prev_line ) > 0 and len ( ll ) > = len ( prev_line ) and ( ll == " - " * len ( ll ) or ll == " + " * len ( ll ) or ll == " = " * len ( ll ) ) :
2011-07-07 13:23:51 +02:00
# new function candidate
2011-07-05 19:29:54 +02:00
if len ( lines ) > 1 :
2011-07-07 13:23:51 +02:00
self . parse_section_safe ( module_name , fname , doc , flineno , lines [ : len ( lines ) - 1 ] )
2011-07-05 19:29:54 +02:00
lines = [ ]
flineno = lineno - 1
fname = prev_line . strip ( )
elif flineno > 0 :
2012-05-28 09:34:49 +02:00
lines . append ( ll )
2011-07-05 19:29:54 +02:00
prev_line = ll
df . close ( )
2011-07-07 13:23:51 +02:00
# don't forget about the last function section in file!!!
2011-07-05 19:29:54 +02:00
if len ( lines ) > 1 :
2011-07-07 16:49:24 +02:00
self . parse_section_safe ( module_name , fname , doc , flineno , lines )
2011-07-07 13:23:51 +02:00
2012-11-12 11:42:28 +01:00
@classmethod
def parse_namespace ( cls , func , section_name ) :
2013-02-25 11:33:00 +01:00
known_namespaces = [ " cv " , " gpu " , " flann " , " superres " ]
2011-07-07 13:23:51 +02:00
l = section_name . strip ( )
for namespace in known_namespaces :
if l . startswith ( namespace + " :: " ) :
func [ " namespace " ] = namespace
return l [ len ( namespace ) + 2 : ]
return section_name
2011-07-05 19:29:54 +02:00
def add_new_fdecl ( self , func , decl ) :
2012-05-29 12:57:25 +02:00
if decl . fdecl . endswith ( " ; " ) :
print >> sys . stderr , " RST parser error E %03d : unexpected semicolon at the end of declaration in \" %s \" at %s : %s " \
% ( ERROR_011_EOLEXPECTED , func [ " name " ] , func [ " file " ] , func [ " line " ] )
2012-11-12 11:42:28 +01:00
decls = func . get ( " decls " , [ ] )
2011-07-05 19:29:54 +02:00
if ( decl . lang == " C++ " or decl . lang == " C " ) :
rst_decl = self . cpp_parser . parse_func_decl_no_wrap ( decl . fdecl )
2012-05-28 09:34:49 +02:00
decls . append ( [ decl . lang , decl . fdecl , rst_decl ] )
2011-07-05 19:29:54 +02:00
else :
2012-05-28 09:34:49 +02:00
decls . append ( [ decl . lang , decl . fdecl ] )
2011-07-05 19:29:54 +02:00
func [ " decls " ] = decls
2012-11-12 11:42:28 +01:00
@classmethod
def add_new_pdecl ( cls , func , decl ) :
params = func . get ( " params " , { } )
2011-07-05 19:29:54 +02:00
if decl . name in params :
2011-07-11 15:33:05 +02:00
if show_errors :
2012-03-30 14:07:45 +02:00
#check black_list
if decl . name not in params_blacklist . get ( func [ " name " ] , [ ] ) :
2012-05-29 12:36:19 +02:00
print >> sys . stderr , " RST parser error E %03d : redefinition of parameter \" %s \" in \" %s \" at %s : %s " \
% ( ERROR_005_REDEFENITIONPARAM , decl . name , func [ " name " ] , func [ " file " ] , func [ " line " ] )
2011-07-05 19:29:54 +02:00
else :
params [ decl . name ] = decl . comment
func [ " params " ] = params
2011-07-11 15:33:05 +02:00
def print_info ( self , func , skipped = False , out = sys . stdout ) :
print >> out
2011-07-05 19:29:54 +02:00
if skipped :
2011-07-11 15:33:05 +02:00
print >> out , " SKIPPED DEFINITION: "
print >> out , " name: %s " % ( func . get ( " name " , " ~empty~ " ) )
2012-05-29 12:36:19 +02:00
print >> out , " file: %s : %s " % ( func . get ( " file " , " ~empty~ " ) , func . get ( " line " , " ~empty~ " ) )
2012-11-12 11:42:28 +01:00
print >> out , " is class: %s " % func . get ( " isclass " , False )
print >> out , " is struct: %s " % func . get ( " isstruct " , False )
2011-07-11 15:33:05 +02:00
print >> out , " module: %s " % func . get ( " module " , " ~unknown~ " )
print >> out , " namespace: %s " % func . get ( " namespace " , " ~empty~ " )
print >> out , " class: %s " % ( func . get ( " class " , " ~empty~ " ) )
print >> out , " method: %s " % ( func . get ( " method " , " ~empty~ " ) )
print >> out , " brief: %s " % ( func . get ( " brief " , " ~empty~ " ) )
2011-07-05 19:29:54 +02:00
if " decls " in func :
2011-07-11 15:33:05 +02:00
print >> out , " declarations: "
2011-07-05 19:29:54 +02:00
for d in func [ " decls " ] :
2012-11-12 11:42:28 +01:00
print >> out , " %7s : %s " % ( d [ 0 ] , re . sub ( r " [ ]+ " , " " , d [ 1 ] ) )
2011-07-07 18:33:02 +02:00
if " seealso " in func :
2011-07-11 15:33:05 +02:00
print >> out , " seealso: " , func [ " seealso " ]
2011-07-05 19:29:54 +02:00
if " params " in func :
2011-07-11 15:33:05 +02:00
print >> out , " parameters: "
2011-07-05 19:29:54 +02:00
for name , comment in func [ " params " ] . items ( ) :
2011-07-11 15:33:05 +02:00
print >> out , " %23s : %s " % ( name , comment )
print >> out , " long: %s " % ( func . get ( " long " , " ~empty~ " ) )
print >> out
2011-07-05 19:29:54 +02:00
def validate ( self , func ) :
2012-11-12 11:42:28 +01:00
if func . get ( " decls " , None ) is None :
if not func . get ( " isclass " , False ) and not func . get ( " isstruct " , False ) :
2011-07-11 15:33:05 +02:00
return False
2011-07-05 19:29:54 +02:00
if func [ " name " ] in self . definitions :
2011-07-11 15:33:05 +02:00
if show_errors :
2012-05-29 12:36:19 +02:00
print >> sys . stderr , " RST parser error E %03d : \" %s \" from: %s : %s is already documented at %s : %s " \
% ( ERROR_006_REDEFENITIONFUNC , func [ " name " ] , func [ " file " ] , func [ " line " ] , self . definitions [ func [ " name " ] ] [ " file " ] , self . definitions [ func [ " name " ] ] [ " line " ] )
2011-07-11 15:33:05 +02:00
return False
2011-07-07 16:49:24 +02:00
return self . validateParams ( func )
def validateParams ( self , func ) :
2012-11-12 11:42:28 +01:00
documentedParams = func . get ( " params " , { } ) . keys ( )
2011-07-07 16:49:24 +02:00
params = [ ]
2012-05-28 09:34:49 +02:00
2011-07-07 16:49:24 +02:00
for decl in func . get ( " decls " , [ ] ) :
if len ( decl ) > 2 :
args = decl [ 2 ] [ 3 ] # decl[2] -> [ funcname, return_ctype, [modifiers], [args] ]
for arg in args :
# arg -> [ ctype, name, def val, [mod], argno ]
if arg [ 0 ] != " ... " :
params . append ( arg [ 1 ] )
params = list ( set ( params ) ) #unique
# 1. all params are documented
for p in params :
2011-07-11 15:33:05 +02:00
if p not in documentedParams and show_warnings :
2012-05-29 12:36:19 +02:00
print >> sys . stderr , " RST parser warning W %03d : parameter \" %s \" of \" %s \" is undocumented. %s : %s " % ( WARNING_007_UNDOCUMENTEDPARAM , p , func [ " name " ] , func [ " file " ] , func [ " line " ] )
2011-07-07 16:49:24 +02:00
# 2. only real params are documented
for p in documentedParams :
2011-07-11 15:33:05 +02:00
if p not in params and show_warnings :
2012-03-30 14:07:45 +02:00
if p not in params_blacklist . get ( func [ " name " ] , [ ] ) :
2012-05-29 12:36:19 +02:00
print >> sys . stderr , " RST parser warning W %03d : unexisting parameter \" %s \" of \" %s \" is documented at %s : %s " % ( WARNING_008_MISSINGPARAM , p , func [ " name " ] , func [ " file " ] , func [ " line " ] )
2011-07-05 19:29:54 +02:00
return True
def normalize ( self , func ) :
if not func :
return func
2012-03-30 14:07:45 +02:00
fnname = func [ " name " ]
fnname = self . normalizeText ( fnname )
fnname = re . sub ( r ' _ \ ?D$ ' , " _nD " , fnname ) # tailing _?D can be mapped to _nD
fnname = re . sub ( r ' \ ?D$ ' , " ND " , fnname ) # tailing ?D can be mapped to ND
fnname = re . sub ( r ' \ (s \ )$ ' , " s " , fnname ) # tailing (s) can be mapped to s
func [ " name " ] = fnname
2011-07-05 19:29:54 +02:00
if " method " in func :
func [ " method " ] = self . normalizeText ( func [ " method " ] )
if " class " in func :
func [ " class " ] = self . normalizeText ( func [ " class " ] )
if " brief " in func :
2012-11-12 11:42:28 +01:00
func [ " brief " ] = self . normalizeText ( func . get ( " brief " , None ) )
2011-07-05 19:29:54 +02:00
if not func [ " brief " ] :
del func [ " brief " ]
if " long " in func :
2012-11-12 11:42:28 +01:00
func [ " long " ] = self . normalizeText ( func . get ( " long " , None ) )
2011-07-05 19:29:54 +02:00
if not func [ " long " ] :
del func [ " long " ]
if " decls " in func :
func [ " decls " ] . sort ( )
if " params " in func :
params = { }
for name , comment in func [ " params " ] . items ( ) :
cmt = self . normalizeText ( comment )
if cmt :
params [ name ] = cmt
2012-03-30 14:07:45 +02:00
# expand some wellknown params
pmap = params_mapping . get ( fnname )
if pmap :
for name , alias in pmap . items ( ) :
params [ name ] = params [ alias ]
2011-07-05 19:29:54 +02:00
func [ " params " ] = params
2011-07-07 18:33:02 +02:00
if " seealso " in func :
seealso = [ ]
for see in func [ " seealso " ] :
item = self . normalizeText ( see . rstrip ( " . " ) ) . strip ( " \" " )
2011-07-08 06:31:13 +02:00
if item and ( item . find ( " " ) < 0 or item . find ( " ::operator " ) > 0 ) :
2011-07-07 18:33:02 +02:00
seealso . append ( item )
func [ " seealso " ] = list ( set ( seealso ) )
2011-07-08 06:31:13 +02:00
if not func [ " seealso " ] :
del func [ " seealso " ]
2011-07-07 13:23:51 +02:00
# special case for old C functions - section name should omit "cv" prefix
2012-11-12 11:42:28 +01:00
if not func . get ( " isclass " , False ) and not func . get ( " isstruct " , False ) :
2011-07-07 13:23:51 +02:00
self . fixOldCFunctionName ( func )
2011-07-05 19:29:54 +02:00
return func
2011-07-07 13:23:51 +02:00
def fixOldCFunctionName ( self , func ) :
2012-05-28 09:34:49 +02:00
if not " decls " in func :
2011-07-07 13:23:51 +02:00
return
fname = None
for decl in func [ " decls " ] :
if decl [ 0 ] != " C " and decl [ 0 ] != " Python1 " :
return
if decl [ 0 ] == " C " :
fname = decl [ 2 ] [ 0 ]
if fname is None :
return
fname = fname . replace ( " . " , " :: " )
2011-07-07 13:32:22 +02:00
if fname . startswith ( " cv::cv " ) :
2012-05-29 12:36:19 +02:00
if fname [ 6 : ] == func . get ( " name " , " " ) . replace ( " * " , " _n " ) :
2011-07-07 13:32:22 +02:00
func [ " name " ] = fname [ 4 : ]
func [ " method " ] = fname [ 4 : ]
2011-07-11 15:33:05 +02:00
elif show_warnings :
2012-05-29 12:36:19 +02:00
print >> sys . stderr , " RST parser warning W %03d : \" %s \" - section name is \" %s \" instead of \" %s \" at %s : %s " % ( WARNING_009_HDRMISMATCH , fname , func [ " name " ] , fname [ 6 : ] , func [ " file " ] , func [ " line " ] )
2011-07-07 13:32:22 +02:00
#self.print_info(func)
2012-05-28 09:34:49 +02:00
2011-07-05 19:29:54 +02:00
def normalizeText ( self , s ) :
if s is None :
return s
2012-04-30 16:33:52 +02:00
s = re . sub ( r " \ . \ . math::[ \ r]* \ n+((.| \ n)*?)( \ n[ \ r]* \ n|$) " , mathReplace2 , s )
2011-07-11 15:33:05 +02:00
s = re . sub ( r " :math:`([^`]+?)` " , mathReplace , s )
s = re . sub ( r " *:sup: " , " ^ " , s )
2012-05-28 09:34:49 +02:00
2011-07-11 15:33:05 +02:00
s = s . replace ( " :ocv:class: " , " " )
s = s . replace ( " :ocv:struct: " , " " )
s = s . replace ( " :ocv:func: " , " " )
s = s . replace ( " :ocv:cfunc: " , " " )
s = s . replace ( " :c:type: " , " " )
s = s . replace ( " :c:func: " , " " )
s = s . replace ( " :ref: " , " " )
s = s . replace ( " :math: " , " " )
s = s . replace ( " :func: " , " " )
s = s . replace ( " ]_ " , " ] " )
s = s . replace ( " .. note:: " , " Note: " )
s = s . replace ( " .. table:: " , " " )
s = s . replace ( " .. ocv:function:: " , " " )
s = s . replace ( " .. ocv:cfunction:: " , " " )
# remove ".. identifier:" lines
s = re . sub ( r " (^| \ n) \ . \ . [a-zA-Z_0-9]+(::[a-zA-Z_0-9]+)?:( \ n|$) " , " \n " , s )
# unwrap urls
s = re . sub ( r " `([^`<]+ )<(https?://[^>]+)>`_ " , " \\ 1( \\ 2) " , s )
# remove tailing ::
s = re . sub ( r " ::( \ n|$) " , " \\ 1 " , s )
2012-05-28 09:34:49 +02:00
2011-07-05 19:29:54 +02:00
# normalize line endings
s = re . sub ( r " \ r \ n " , " \n " , s )
# remove extra line breaks before/after _ or ,
2011-07-08 06:31:13 +02:00
s = re . sub ( r " \ n[ ]*([_,]) \ n " , r " \ 1 " , s )
2011-07-05 19:29:54 +02:00
# remove extra line breaks after `
#s = re.sub(r"`\n", "` ", s)
2011-07-11 15:33:05 +02:00
# remove extra space after ( and before .,)
2011-07-08 06:31:13 +02:00
s = re . sub ( r " \ ([ \ n ]+ " , " ( " , s )
2011-07-11 15:33:05 +02:00
s = re . sub ( r " [ \ n ]+( \ .|,| \ )) " , " \\ 1 " , s )
2011-07-07 18:33:02 +02:00
# remove extra line breaks after ".. note::"
s = re . sub ( r " \ . \ . note:: \ n+ " , " .. note:: " , s )
2011-07-05 19:29:54 +02:00
# remove extra line breaks before *
2011-07-11 15:33:05 +02:00
s = re . sub ( r " \ n+ \ * " , " \n * " , s )
2011-07-07 18:33:02 +02:00
# remove extra line breaks after *
s = re . sub ( r " \ n \ * \ n+ " , " \n * " , s )
2011-07-05 19:29:54 +02:00
# remove extra line breaks before #.
2011-07-11 15:33:05 +02:00
s = re . sub ( r " \ n+# \ . " , " \n #. " , s )
2011-07-05 19:29:54 +02:00
# remove extra line breaks after #.
2011-07-07 18:33:02 +02:00
s = re . sub ( r " \ n# \ . \ n+ " , " \n #. " , s )
2011-07-05 19:29:54 +02:00
# remove extra line breaks before `
2011-07-08 17:00:11 +02:00
#s = re.sub(r"\n[ ]*`", " `", s)
2011-07-05 19:29:54 +02:00
# remove trailing whitespaces
2011-07-07 13:23:51 +02:00
s = re . sub ( r " [ ]+$ " , " " , s )
2011-07-05 19:29:54 +02:00
# remove .. for references
2011-07-11 15:33:05 +02:00
#s = re.sub(r"\.\. \[", "[", s)
2011-07-05 19:29:54 +02:00
# unescape
s = re . sub ( r " \\ (.) " , " \\ 1 " , s )
2012-05-28 09:34:49 +02:00
2011-07-08 06:31:13 +02:00
# remove whitespace before .
s = re . sub ( r " [ ]+ \ . " , " . " , s )
# remove tailing whitespace
s = re . sub ( r " +( \ n|$) " , " \\ 1 " , s )
# remove leading whitespace
s = re . sub ( r " (^| \ n) + " , " \\ 1 " , s )
# compress line breaks
s = re . sub ( r " \ n \ n+ " , " \n \n " , s )
# remove other newlines
2011-07-11 15:33:05 +02:00
s = re . sub ( r " ([^. \ n \\ =]) \ n([^*# \ n]| \ *[^ ]) " , " \\ 1 \\ 2 " , s )
2011-07-08 06:31:13 +02:00
# compress whitespace
s = re . sub ( r " + " , " " , s )
2011-07-11 15:33:05 +02:00
# restore math
2012-11-12 11:42:28 +01:00
s = re . sub ( r " *<BR> * " , " \n " , s )
2011-07-11 15:33:05 +02:00
2011-07-08 06:31:13 +02:00
# remove extra space before .
s = re . sub ( r " [ \ n ]+ \ . " , " . " , s )
s = s . replace ( " ** " , " " )
2012-04-30 16:33:52 +02:00
s = re . sub ( r " ``([^ \ n]+?)`` " , " <code> \\ 1</code> " , s )
2011-07-08 06:31:13 +02:00
s = s . replace ( " `` " , " \" " )
s = s . replace ( " ` " , " \" " )
s = s . replace ( " \" \" " , " \" " )
2011-07-05 19:29:54 +02:00
s = s . strip ( )
return s
2012-05-28 09:34:49 +02:00
2011-07-11 15:33:05 +02:00
def printSummary ( self ) :
print " RST Parser Summary: "
print " Total sections: %s " % self . sections_total
print " Skipped sections: %s " % self . sections_skipped
print " Parsed sections: %s " % self . sections_parsed
print " Invalid sections: %s " % ( self . sections_total - self . sections_parsed - self . sections_skipped )
# statistic by language
stat = { }
classes = 0
structs = 0
for name , d in self . definitions . items ( ) :
2012-11-12 11:42:28 +01:00
if d . get ( " isclass " , False ) :
classes + = 1
elif d . get ( " isstruct " , False ) :
structs + = 1
else :
for decl in d . get ( " decls " , [ ] ) :
stat [ decl [ 0 ] ] = stat . get ( decl [ 0 ] , 0 ) + 1
2011-07-11 15:33:05 +02:00
print
print " classes documented: %s " % classes
print " structs documented: %s " % structs
for lang in sorted ( stat . items ( ) ) :
print " %7s functions documented: %s " % lang
2011-07-11 17:03:42 +02:00
print
2011-11-24 13:20:45 +01:00
2011-07-11 15:33:05 +02:00
def mathReplace2 ( match ) :
m = mathReplace ( match )
#print "%s ===> %s" % (match.group(0), m)
return " \n \n " + m + " <BR><BR> "
def hdotsforReplace ( match ) :
return ' ... ' * int ( match . group ( 1 ) )
def matrixReplace ( match ) :
m = match . group ( 2 )
m = re . sub ( r " *& * " , " " , m )
return m
2012-05-28 09:34:49 +02:00
2011-07-11 15:33:05 +02:00
def mathReplace ( match ) :
m = match . group ( 1 )
m = m . replace ( " \n " , " <BR> " )
2012-10-10 11:28:26 +02:00
m = m . replace ( " < " , " < " )
m = m . replace ( " > " , " > " )
2011-07-11 15:33:05 +02:00
m = re . sub ( r " \\ text(tt|rm)? { (.*?)} " , " \\ 2 " , m )
m = re . sub ( r " \\ mbox { (.*?)} " , " \\ 1 " , m )
m = re . sub ( r " \\ mathrm { (.*?)} " , " \\ 1 " , m )
m = re . sub ( r " \\ vecthree { (.*?)} { (.*?)} { (.*?)} " , " [ \\ 1 \\ 2 \\ 3] " , m )
m = re . sub ( r " \\ bar { (.*?)} " , " \\ 1` " , m )
m = re . sub ( r " \\ sqrt \ [( \ d)* \ ] { (.*?)} " , " sqrt \\ 1( \\ 2) " , m )
m = re . sub ( r " \\ sqrt { (.*?)} " , " sqrt( \\ 1) " , m )
m = re . sub ( r " \\ frac { (.*?)} { (.*?)} " , " ( \\ 1)/( \\ 2) " , m )
m = re . sub ( r " \\ fork { (.*?)} { (.*?)} { (.*?)} { (.*?)} " , " \\ 1 \\ 2; \\ 3 \\ 4 " , m )
m = re . sub ( r " \\ forkthree { (.*?)} { (.*?)} { (.*?)} { (.*?)} { (.*?)} { (.*?)} " , " \\ 1 \\ 2; \\ 3 \\ 4; \\ 5 \\ 6 " , m )
m = re . sub ( r " \\ stackrel { (.*?)} { (.*?)} " , " \\ 1 \\ 2 " , m )
m = re . sub ( r " \\ sum _ { (.*?)} " , " sum { by: \\ 1} " , m )
m = re . sub ( r " + " , " " , m )
m = re . sub ( r " \\ begin { (?P<gtype>array|bmatrix)}(?: { [ \ |lcr \ . ]+})? *(.*?) \\ end { (?P=gtype)} " , matrixReplace , m )
m = re . sub ( r " \\ hdotsfor { ( \ d+)} " , hdotsforReplace , m )
m = re . sub ( r " \\ vecthreethree { (.*?)} { (.*?)} { (.*?)} { (.*?)} { (.*?)} { (.*?)} { (.*?)} { (.*?)} { (.*?)} " , " <BR>| \\ 1 \\ 2 \\ 3|<BR>| \\ 4 \\ 5 \\ 6|<BR>| \\ 7 \\ 8 \\ 9|<BR> " , m )
2012-05-28 09:34:49 +02:00
2011-07-11 15:33:05 +02:00
m = re . sub ( r " \\ left[ ]* \\ lfloor[ ]* " , " [ " , m )
m = re . sub ( r " [ ]* \\ right[ ]* \\ rfloor " , " ] " , m )
m = re . sub ( r " \\ left[ ]* \ ([ ]* " , " ( " , m )
m = re . sub ( r " [ ]* \\ right[ ]* \ ) " , " ) " , m )
m = re . sub ( r " ([^ \\ ]) \ $ " , " \\ 1 " , m )
m = m . replace ( " \\ times " , " x " )
m = m . replace ( " \\ pm " , " +- " )
m = m . replace ( " \\ cdot " , " * " )
m = m . replace ( " \\ sim " , " ~ " )
m = m . replace ( " \\ leftarrow " , " <- " )
m = m . replace ( " \\ rightarrow " , " -> " )
m = m . replace ( " \\ leftrightarrow " , " <-> " )
m = re . sub ( r " * \\ neg * " , " ! " , m )
m = re . sub ( r " * \\ neq? * " , " != " , m )
m = re . sub ( r " * \\ geq? * " , " >= " , m )
m = re . sub ( r " * \\ leq? * " , " <= " , m )
m = re . sub ( r " * \\ vee * " , " V " , m )
m = re . sub ( r " * \\ oplus * " , " (+) " , m )
m = re . sub ( r " * \\ mod * " , " mod " , m )
m = re . sub ( r " ( *) \\ partial * " , " \\ 1d " , m )
m = re . sub ( r " ( *) \\ quad * " , " \\ 1 " , m )
m = m . replace ( " \\ , " , " " )
m = m . replace ( " \\ : " , " " )
m = m . replace ( " \\ ; " , " " )
m = m . replace ( " \\ ! " , " " )
m = m . replace ( " \\ \\ " , " <BR> " )
m = m . replace ( " \\ wedge " , " / \\ \\ " )
m = re . sub ( r " \\ (.) " , " \\ 1 " , m )
m = re . sub ( r " \ ([ ]+ " , " ( " , m )
m = re . sub ( r " [ ]+( \ .|,| \ ))(<BR>| |$) " , " \\ 1 \\ 2 " , m )
m = re . sub ( r " + \ |[ ]+([a-zA-Z0-9_(]) " , " | \\ 1 " , m )
m = re . sub ( r " ([a-zA-Z0-9_)}])[ ]+( \ (| \ |) " , " \\ 1 \\ 2 " , m )
m = re . sub ( r " { \ ((-?[a-zA-Z0-9_]+) \ )} " , " \\ 1 " , m )
m = re . sub ( r " { (-?[a-zA-Z0-9_]+)} " , " ( \\ 1) " , m )
m = re . sub ( r " \ (([0-9]+) \ ) " , " \\ 1 " , m )
m = m . replace ( " { " , " ( " )
m = m . replace ( " } " , " ) " )
#print "%s ===> %s" % (match.group(0), m)
2012-04-30 16:33:52 +02:00
return " <em> " + m + " </em> "
2011-07-05 19:29:54 +02:00
if __name__ == " __main__ " :
2011-07-08 17:00:11 +02:00
if len ( sys . argv ) < 2 :
2011-07-05 19:29:54 +02:00
print " Usage: \n " , os . path . basename ( sys . argv [ 0 ] ) , " <module path> "
exit ( 0 )
2012-05-28 09:34:49 +02:00
2011-07-11 15:33:05 +02:00
if len ( sys . argv ) > = 3 :
if sys . argv [ 2 ] . lower ( ) == " verbose " :
verbose = True
2011-07-05 19:29:54 +02:00
rst_parser_dir = os . path . dirname ( os . path . abspath ( sys . argv [ 0 ] ) )
2012-06-21 16:50:05 +02:00
hdr_parser_path = os . path . join ( rst_parser_dir , " ../../python/src2 " )
2011-07-05 19:29:54 +02:00
sys . path . append ( hdr_parser_path )
import hdr_parser
2011-11-24 13:20:45 +01:00
module = sys . argv [ 1 ]
2011-07-05 19:29:54 +02:00
2012-06-21 16:50:05 +02:00
if module != " all " and not os . path . isdir ( os . path . join ( rst_parser_dir , " ../../ " + module ) ) :
2012-05-29 12:36:19 +02:00
print " RST parser error E %03d : module \" %s \" could not be found. " % ( ERROR_010_NOMODULE , module )
2011-07-05 19:29:54 +02:00
exit ( 1 )
parser = RstParser ( hdr_parser . CppHeaderParser ( ) )
2012-05-28 09:34:49 +02:00
2011-07-07 13:23:51 +02:00
if module == " all " :
2011-07-11 17:03:42 +02:00
for m in allmodules :
2012-06-21 16:50:05 +02:00
parser . parse ( m , os . path . join ( rst_parser_dir , " ../../ " + m ) )
2011-07-07 13:23:51 +02:00
else :
2012-06-21 16:50:05 +02:00
parser . parse ( module , os . path . join ( rst_parser_dir , " ../../ " + module ) )
2011-07-07 13:23:51 +02:00
# summary
2011-07-11 15:33:05 +02:00
parser . printSummary ( )