2013-12-22 18:55:48 +01:00
|
|
|
#!/usr/bin/python
|
|
|
|
import sys
|
|
|
|
import os
|
|
|
|
import inspect
|
|
|
|
import fnmatch
|
|
|
|
import monkDebug as debug
|
|
|
|
import monkTools as tools
|
|
|
|
import monkNode as Node
|
|
|
|
import monkParse as Parse
|
|
|
|
import monkHtml
|
2013-12-25 23:56:09 +01:00
|
|
|
import re
|
2016-03-25 22:52:03 +01:00
|
|
|
import json
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
class Module:
|
|
|
|
##
|
|
|
|
## @brief Module class represent all system needed for a specific
|
|
|
|
## module like
|
|
|
|
## - type (bin/lib ...)
|
|
|
|
## - dependency
|
|
|
|
## - flags
|
|
|
|
## - files
|
|
|
|
## - ...
|
|
|
|
##
|
|
|
|
def __init__(self, file, moduleName, moduleType):
|
|
|
|
## Remove all variable to prevent error of multiple deffinition of the module ...
|
2016-03-25 22:52:03 +01:00
|
|
|
self.origin_file=''
|
|
|
|
self.origin_folder=''
|
2013-12-22 18:55:48 +01:00
|
|
|
# type of the module:
|
|
|
|
self.type='LIBRARY'
|
|
|
|
# Name of the module
|
|
|
|
self.name=moduleName
|
2016-03-23 21:27:40 +01:00
|
|
|
self.list_doc_file = []
|
|
|
|
self.list_tutorial_file = []
|
2016-03-25 22:52:03 +01:00
|
|
|
self.web_site = ""
|
|
|
|
self.web_source = ""
|
|
|
|
self.path_parsing = ""
|
|
|
|
self.path_global_doc = ""
|
|
|
|
self.external_link = []
|
2013-12-22 18:55:48 +01:00
|
|
|
self.title = moduleName + " Library"
|
2016-03-25 22:52:03 +01:00
|
|
|
self.style_html = ""
|
2013-12-22 18:55:48 +01:00
|
|
|
## end of basic INIT ...
|
2013-12-26 21:17:20 +01:00
|
|
|
if moduleType.upper() == 'APPLICATION':
|
|
|
|
self.type = 'application'
|
|
|
|
elif moduleType.upper() == 'LIBRARY':
|
|
|
|
self.type = "library"
|
2013-12-22 18:55:48 +01:00
|
|
|
else :
|
|
|
|
debug.error('for module "%s"' %moduleName)
|
|
|
|
debug.error(' ==> error : "%s" ' %moduleType)
|
|
|
|
raise 'Input value error'
|
2016-03-25 22:52:03 +01:00
|
|
|
self.structure_lib = Node.MainNode(self.type, moduleName)
|
|
|
|
self.origin_file = file;
|
|
|
|
self.origin_folder = tools.get_current_path(self.origin_file)
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
##
|
2016-03-25 22:52:03 +01:00
|
|
|
## @brief Set the module web_site (activate only when compile in release mode, else "../moduleName/)
|
|
|
|
## @param[in] url New web_site url
|
2013-12-22 18:55:48 +01:00
|
|
|
##
|
|
|
|
def set_website(self, url):
|
2016-03-25 22:52:03 +01:00
|
|
|
self.web_site = url
|
2013-12-22 18:55:48 +01:00
|
|
|
|
2013-12-26 07:49:52 +01:00
|
|
|
def get_website(self):
|
2016-03-25 22:52:03 +01:00
|
|
|
return self.web_site
|
2013-12-26 07:49:52 +01:00
|
|
|
|
2013-12-28 16:07:12 +01:00
|
|
|
def set_website_sources(self, url):
|
2016-03-25 22:52:03 +01:00
|
|
|
self.web_source = url
|
2013-12-28 16:07:12 +01:00
|
|
|
|
|
|
|
def get_website_sources(self):
|
2016-03-25 22:52:03 +01:00
|
|
|
return self.web_source
|
2013-12-28 16:07:12 +01:00
|
|
|
|
|
|
|
|
2013-12-22 18:55:48 +01:00
|
|
|
##
|
|
|
|
## @brief set the parsing folder
|
|
|
|
## @param[in] path New path to parse
|
|
|
|
##
|
|
|
|
def set_path(self, path):
|
2016-03-25 22:52:03 +01:00
|
|
|
self.path_parsing = path
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
##
|
|
|
|
## @brief set the glabal documentation parsing folder
|
|
|
|
## @param[in] path New path to parse
|
|
|
|
##
|
|
|
|
def set_path_general_doc(self, path):
|
2016-03-25 22:52:03 +01:00
|
|
|
self.path_global_doc = path
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
##
|
|
|
|
## @brief List of validate external library link (disable otherwise)
|
|
|
|
## @param[in] availlable List of all module link availlable
|
|
|
|
##
|
|
|
|
def set_external_link(self, availlable):
|
2016-03-25 22:52:03 +01:00
|
|
|
self.external_link = availlable
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
##
|
|
|
|
## @brief Set the library title
|
|
|
|
## @param[in] title New title to set.
|
|
|
|
##
|
|
|
|
def set_title(self, title):
|
|
|
|
self.title = title
|
|
|
|
|
|
|
|
##
|
|
|
|
## @brief new html basic css file
|
|
|
|
## @param[in] file File of the css style sheet
|
|
|
|
##
|
|
|
|
def set_html_css(self, cssFile):
|
2016-03-25 22:52:03 +01:00
|
|
|
self.style_html = cssFile
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
##
|
|
|
|
## @brief Create the module documentation:
|
|
|
|
##
|
|
|
|
def parse_code(self):
|
|
|
|
debug.info('Parse documantation code : ' + self.name)
|
2016-03-25 22:52:03 +01:00
|
|
|
if self.path_parsing != "":
|
|
|
|
for root, dirnames, filenames in os.walk(self.path_parsing):
|
2017-10-03 15:19:07 +02:00
|
|
|
tmpList = fnmatch.filter(filenames, "*.hpp")
|
2013-12-22 18:55:48 +01:00
|
|
|
# Import the module :
|
|
|
|
for filename in tmpList:
|
|
|
|
fileCompleteName = os.path.join(root, filename)
|
|
|
|
debug.debug(" Find a file : '" + fileCompleteName + "'")
|
|
|
|
self.add_file(fileCompleteName)
|
2013-12-26 07:49:52 +01:00
|
|
|
# all file is parset ==> now we create the namespacing of all elements:
|
2016-03-25 22:52:03 +01:00
|
|
|
self.structure_lib.set_namespace()
|
|
|
|
self.structure_lib.set_module_link(self)
|
|
|
|
#self.structure_lib.complete_display()
|
2013-12-25 23:56:09 +01:00
|
|
|
|
2013-12-22 18:55:48 +01:00
|
|
|
# display the hierarchie of all the class and namespace ...
|
2016-03-25 22:52:03 +01:00
|
|
|
#self.structure_lib.debug_display()
|
|
|
|
if self.path_global_doc != "":
|
|
|
|
for root, dirnames, filenames in os.walk(self.path_global_doc):
|
2017-10-03 15:19:07 +02:00
|
|
|
tmpList = fnmatch.filter(filenames, "*.md")
|
2013-12-22 18:55:48 +01:00
|
|
|
# Import the module :
|
|
|
|
for filename in tmpList:
|
|
|
|
fileCompleteName = os.path.join(root, filename)
|
2016-03-25 22:52:03 +01:00
|
|
|
tutorialPath = os.path.join(self.path_global_doc, "tutorial/")
|
|
|
|
pathBase = fileCompleteName[len(self.path_global_doc):len(fileCompleteName)-3]
|
2017-10-03 15:19:07 +02:00
|
|
|
while len(pathBase) > 0 \
|
|
|
|
and pathBase[0] == '/':
|
|
|
|
pathBase = pathBase[1:]
|
2014-09-03 21:04:06 +02:00
|
|
|
debug.verbose(" Find a doc file : fileCompleteName='" + fileCompleteName + "'")
|
2013-12-22 18:55:48 +01:00
|
|
|
if fileCompleteName[:len(tutorialPath)] == tutorialPath:
|
2017-10-03 15:19:07 +02:00
|
|
|
debug.warning("add_tutorial_doc : '" + fileCompleteName + "' ==> '" + pathBase + "'")
|
2013-12-22 18:55:48 +01:00
|
|
|
self.add_tutorial_doc(fileCompleteName, pathBase)
|
2013-12-27 21:35:30 +01:00
|
|
|
else:
|
2017-10-03 15:19:07 +02:00
|
|
|
debug.warning("add_file_doc : '" + fileCompleteName + "' ==> '" + pathBase + "'")
|
2013-12-27 21:35:30 +01:00
|
|
|
self.add_file_doc(fileCompleteName, pathBase)
|
2013-12-22 18:55:48 +01:00
|
|
|
|
2016-03-23 21:27:40 +01:00
|
|
|
##
|
|
|
|
## @brief Sort a list of n element containing a list of element (order with the first)
|
|
|
|
## @param[in] value List to order
|
|
|
|
## @return ordered list
|
|
|
|
##
|
|
|
|
def sort_list_first_elem(self, value):
|
|
|
|
# order the list:
|
|
|
|
order_elem = []
|
|
|
|
for elem in value:
|
|
|
|
order_elem.append(elem[0])
|
|
|
|
order_elem.sort()
|
|
|
|
out = []
|
|
|
|
for elem in order_elem:
|
|
|
|
for old_val in value:
|
|
|
|
if elem == old_val[0]:
|
|
|
|
out.append(old_val)
|
|
|
|
break;
|
|
|
|
return out
|
|
|
|
|
2013-12-22 18:55:48 +01:00
|
|
|
##
|
|
|
|
## @brief Add a documentation file at the parsing system
|
|
|
|
## @param[in] filename File To add at the parsing element system.
|
|
|
|
## @param[in] outPath output system file.
|
|
|
|
## @return True if no error occured, False otherwise
|
|
|
|
##
|
|
|
|
def add_file_doc(self, filename, outPath):
|
|
|
|
debug.debug("adding file in documantation : '" + filename + "'");
|
2014-09-03 21:04:06 +02:00
|
|
|
done = False
|
2016-03-23 21:27:40 +01:00
|
|
|
for iii in range(0,len(self.list_doc_file)):
|
|
|
|
if self.list_doc_file[iii][0] > filename:
|
|
|
|
self.list_doc_file.insert(iii, [filename, outPath])
|
2014-09-03 21:04:06 +02:00
|
|
|
done = True
|
|
|
|
break
|
|
|
|
if done == False:
|
2016-03-23 21:27:40 +01:00
|
|
|
self.list_doc_file.append([filename, outPath])
|
|
|
|
self.list_doc_file = self.sort_list_first_elem(self.list_doc_file)
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
##
|
|
|
|
## @brief Add a documentation file at the parsing system
|
|
|
|
## @param[in] filename File To add at the parsing element system.
|
|
|
|
## @param[in] outPath output system file.
|
|
|
|
## @return True if no error occured, False otherwise
|
|
|
|
##
|
|
|
|
def add_tutorial_doc(self, filename, outPath):
|
2014-09-03 21:04:06 +02:00
|
|
|
count = int(filename.split('/')[-1].split('_')[0])
|
2013-12-22 18:55:48 +01:00
|
|
|
debug.debug("adding file in documantation : '" + filename + "'");
|
2014-09-03 21:04:06 +02:00
|
|
|
done = False
|
2016-03-23 21:27:40 +01:00
|
|
|
for iii in range(0,len(self.list_tutorial_file)):
|
|
|
|
if self.list_tutorial_file[iii][0] > filename:
|
|
|
|
self.list_tutorial_file.insert(iii, [filename, outPath])
|
2014-09-03 21:04:06 +02:00
|
|
|
done = True
|
|
|
|
break
|
|
|
|
if done == False:
|
2016-03-23 21:27:40 +01:00
|
|
|
self.list_tutorial_file.append([filename, outPath])
|
|
|
|
self.list_tutorial_file = self.sort_list_first_elem(self.list_tutorial_file)
|
|
|
|
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
##
|
|
|
|
## @brief Add a file at the parsing system
|
|
|
|
## @param[in] filename File To add at the parsing element system.
|
|
|
|
## @return True if no error occured, False otherwise
|
|
|
|
##
|
|
|
|
def add_file(self, filename):
|
|
|
|
debug.debug("adding file in documantation : '" + filename + "'");
|
|
|
|
|
|
|
|
#parsedFile = Parse.parse_file("Widget.h")
|
|
|
|
#debug.error("plop")
|
|
|
|
parsedFile = Parse.parse_file(filename)
|
2016-03-25 22:52:03 +01:00
|
|
|
self.structure_lib = parsedFile.fusion(self.structure_lib)
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
##
|
|
|
|
## @brief Generate Documentation at the folder ...
|
|
|
|
## @param[in] destFolder Destination folder.
|
|
|
|
## @param[in] mode (optinnal) generation output mode {html, markdown ...}
|
|
|
|
##
|
|
|
|
def generate(self):
|
|
|
|
debug.info('Generate documantation code : ' + self.name)
|
2016-03-25 22:52:03 +01:00
|
|
|
#json_data = json.dumps(self, sort_keys=True, indent=4)
|
|
|
|
#tools.file_write_data(os.path.join("out", "doc", self.name + ".json"), json_data)
|
|
|
|
destFolder = os.path.join("out", "doc", self.name)
|
2013-12-26 21:17:20 +01:00
|
|
|
tools.remove_folder_and_sub_folder(destFolder);
|
2016-03-25 22:52:03 +01:00
|
|
|
if monkHtml.generate(self, destFolder + '/') == False:
|
2013-12-22 18:55:48 +01:00
|
|
|
debug.warning("Generation Documentation ==> return an error for " + self.name)
|
|
|
|
|
|
|
|
|
|
|
|
def get_base_doc_node(self):
|
2016-03-25 22:52:03 +01:00
|
|
|
return self.structure_lib
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
##
|
|
|
|
## @brief Get the heritage list (parent) of one element.
|
|
|
|
## @param[in] element Element name.
|
|
|
|
## @return List of all element herited
|
|
|
|
##
|
|
|
|
def get_heritage_list(self, element):
|
|
|
|
list = []
|
|
|
|
# get element class :
|
2016-03-25 22:52:03 +01:00
|
|
|
if element in self.list_class.keys():
|
|
|
|
localClass = self.list_class[element]
|
2013-12-22 18:55:48 +01:00
|
|
|
if len(localClass['inherits']) != 0:
|
|
|
|
# TODO : Support multiple heritage ...
|
|
|
|
isFirst = True
|
|
|
|
for heritedClass in localClass['inherits']:
|
|
|
|
if isFirst == True:
|
|
|
|
list = self.get_heritage_list(heritedClass['class'])
|
|
|
|
break;
|
|
|
|
debug.verbose("find parent : " + element)
|
|
|
|
list.append(element);
|
|
|
|
return list
|
|
|
|
|
|
|
|
##
|
|
|
|
## @brief Get the heritage list (child) of this element.
|
|
|
|
## @param[in] curentClassName Element name.
|
|
|
|
## @return List of all childs
|
|
|
|
##
|
|
|
|
def get_down_heritage_list(self, curentClassName):
|
|
|
|
list = []
|
|
|
|
# get element class :
|
2016-03-25 22:52:03 +01:00
|
|
|
for element in self.list_class:
|
|
|
|
localClass = self.list_class[element]
|
2013-12-22 18:55:48 +01:00
|
|
|
if len(localClass['inherits']) != 0:
|
|
|
|
for heritedClass in localClass['inherits']:
|
|
|
|
if curentClassName == heritedClass['class']:
|
|
|
|
list.append(element)
|
|
|
|
break;
|
|
|
|
debug.verbose("find childs : " + str(list))
|
|
|
|
return list
|
|
|
|
|
2013-12-26 21:17:20 +01:00
|
|
|
def get_whith_specific_parrent(self, name, appName=None):
|
2016-03-25 22:52:03 +01:00
|
|
|
if self.structure_lib.get_node_type() == "library":
|
|
|
|
return self.structure_lib.get_whith_specific_parrent(name)
|
|
|
|
if appName != self.structure_lib.get_name():
|
2013-12-26 21:17:20 +01:00
|
|
|
return []
|
2016-03-25 22:52:03 +01:00
|
|
|
return self.structure_lib.get_whith_specific_parrent(name)
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
moduleList=[]
|
|
|
|
__startModuleName="monk_"
|
|
|
|
|
|
|
|
def import_path(path):
|
|
|
|
global moduleList
|
|
|
|
matches = []
|
|
|
|
debug.debug('Start find sub File : "%s"' %path)
|
|
|
|
for root, dirnames, filenames in os.walk(path):
|
|
|
|
tmpList = fnmatch.filter(filenames, __startModuleName + "*.py")
|
|
|
|
# Import the module :
|
|
|
|
for filename in tmpList:
|
|
|
|
debug.debug(' Find a file : "%s"' %os.path.join(root, filename))
|
|
|
|
#matches.append(os.path.join(root, filename))
|
|
|
|
sys.path.append(os.path.dirname(os.path.join(root, filename)) )
|
|
|
|
moduleName = filename.replace('.py', '')
|
|
|
|
moduleName = moduleName.replace(__startModuleName, '')
|
|
|
|
debug.debug("integrate module: '" + moduleName + "' from '" + os.path.join(root, filename) + "'")
|
|
|
|
theModule = __import__(__startModuleName + moduleName)
|
|
|
|
tmpElement = theModule.create()
|
2013-12-23 22:04:35 +01:00
|
|
|
try:
|
|
|
|
tmpdesc = theModule.get_desc()
|
|
|
|
except:
|
|
|
|
tmpdesc = ""
|
2013-12-22 18:55:48 +01:00
|
|
|
if (tmpElement == None) :
|
|
|
|
debug.warning("Request load module '" + name + "' not define for this platform")
|
|
|
|
moduleList.append({"name":moduleName, "path":os.path.join(root, filename), "node":tmpElement, "desc":tmpdesc})
|
|
|
|
|
|
|
|
def get_module(name):
|
|
|
|
global moduleList
|
|
|
|
for mod in moduleList:
|
|
|
|
if mod["name"] == name:
|
|
|
|
return mod["node"]
|
|
|
|
return None
|
|
|
|
|
|
|
|
def get_all_module():
|
|
|
|
global moduleList
|
|
|
|
AllList = []
|
|
|
|
for mod in moduleList:
|
|
|
|
AllList.append(mod["node"])
|
|
|
|
return AllList
|
|
|
|
|
|
|
|
def list_all_module_with_desc():
|
|
|
|
global moduleList
|
|
|
|
tmpList = []
|
|
|
|
for mod in moduleList:
|
|
|
|
tmpList.append([mod["name"], mod["desc"]])
|
|
|
|
return tmpList
|
|
|
|
|
2013-12-23 22:04:35 +01:00
|
|
|
def get_link_type(type):
|
|
|
|
return ""
|
2013-12-22 18:55:48 +01:00
|
|
|
|
2013-12-25 23:56:09 +01:00
|
|
|
def get_element_with_name(type):
|
|
|
|
global moduleList
|
2013-12-26 07:49:52 +01:00
|
|
|
debug.verbose("try find : " + str(type) + " ")
|
2013-12-25 23:56:09 +01:00
|
|
|
ret = re.sub(r'::', ':', type)
|
|
|
|
ret = ret.split(":")
|
|
|
|
for mod in moduleList:
|
|
|
|
element = mod['node'].get_base_doc_node().find(ret)
|
|
|
|
if element != None:
|
2013-12-26 07:49:52 +01:00
|
|
|
debug.debug("we find : " + type + " = " + str(ret) + " " + str(element))
|
2013-12-25 23:56:09 +01:00
|
|
|
return element
|
2013-12-26 21:17:20 +01:00
|
|
|
debug.debug("we not find : " + type + " = " + str(ret))
|
|
|
|
return None
|
|
|
|
|
|
|
|
def get_whith_specific_parrent(name, appName=None):
|
|
|
|
global moduleList
|
|
|
|
ret = []
|
|
|
|
for mod in moduleList:
|
|
|
|
tmpRet = mod['node'].get_whith_specific_parrent(name, appName)
|
|
|
|
if len(tmpRet) != 0:
|
|
|
|
for tmp in tmpRet:
|
|
|
|
ret.append(tmp)
|
|
|
|
return ret
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def display_color(val):
|
|
|
|
# storage keyword :
|
|
|
|
val = re.sub(r'(inline|const|class|virtual|private|public|protected|friend|const|extern|auto|register|static|volatile|typedef|struct|union|enum)',
|
|
|
|
r'<span class="code-storage-keyword">\1</span>',
|
|
|
|
val)
|
|
|
|
# type :
|
|
|
|
val = re.sub(r'(bool|BOOL|char(16_t|32_t)?|double|float|u?int(8|16|32|64|128)?(_t)?|long|short|signed|size_t|unsigned|void|(I|U)(8|16|32|64|128))',
|
|
|
|
r'<span class="code-type">\1</span>',
|
|
|
|
val)
|
|
|
|
return val
|
|
|
|
|