From 0f72d6e131fe3ae12b0f4532ecff31506e9e2c0d Mon Sep 17 00:00:00 2001 From: thibault Date: Mon, 20 May 2024 04:05:25 -0400 Subject: [PATCH] Fix bug raised by #32: Match scope when selecting object in dependency --- .readthedocs.yaml | 15 ++ doc/source/conf.py | 6 +- doc/source/org-doc.rst | 263 ++++++++++++++++++++++--------- hpp2plantuml.org | 254 ++++++++++++++++++++--------- src/hpp2plantuml/__init__.py | 2 +- src/hpp2plantuml/hpp2plantuml.py | 176 +++++++++++++-------- tests/simple_classes.puml | 14 ++ tests/simple_classes_1_2.hpp | 8 + tests/simple_classes_3.hpp | 1 + tests/simple_classes_nodep.puml | 14 ++ tests/test_hpp2plantuml.py | 19 +-- 11 files changed, 545 insertions(+), 227 deletions(-) create mode 100644 .readthedocs.yaml diff --git a/.readthedocs.yaml b/.readthedocs.yaml new file mode 100644 index 0000000..753d403 --- /dev/null +++ b/.readthedocs.yaml @@ -0,0 +1,15 @@ +version: 2 + +build: + os: ubuntu-22.04 + tools: + python: "3.11" + +sphinx: + configuration: doc/source/conf.py + +python: + install: + - method: pip + path: . + - requirements: doc/source/readthedocs-requirements.txt diff --git a/doc/source/conf.py b/doc/source/conf.py index 36c55f0..7bc0b54 100644 --- a/doc/source/conf.py +++ b/doc/source/conf.py @@ -79,9 +79,9 @@ # built documents. # # The short X.Y version. -version = u'v' + u'0.8.4' +version = u'v' + u'0.8.5' # The full version, including alpha/beta/rc tags. -release = u'v' + u'0.8.4' +release = u'v' + u'0.8.5' # The language for content autogenerated by Sphinx. Refer to documentation # for a list of supported languages. @@ -155,7 +155,7 @@ # The name for this set of Sphinx documents. # " v documentation" by default. # -# html_title = u'hpp2plantuml ' + u'v' + u'0.8.4' +# html_title = u'hpp2plantuml ' + u'v' + u'0.8.5' # A shorter title for the navigation bar. Default is the same as html_title. # diff --git a/doc/source/org-doc.rst b/doc/source/org-doc.rst index 80566b9..2b30318 100644 --- a/doc/source/org-doc.rst +++ b/doc/source/org-doc.rst @@ -13,7 +13,7 @@ The current version of the code is: :: :name: hpp2plantuml-version - 0.8.4 + 0.8.5 The source code can be found on GitHub: @@ -453,7 +453,8 @@ which is used to determine aggregation relationships between classes. element is the class name and the second element is a CppClass object) """ - super().__init__(header_class[1]['declaration_method'], header_class[0]) + super().__init__(header_class[1]['declaration_method'], + header_class[0]) self._abstract = header_class[1]['abstract'] self._template_type = None if 'template' in header_class[1]: @@ -1148,6 +1149,28 @@ it corresponds to the ``+..`` link type (`http://plantuml.com/class-diagram 0 else None + obj_ns_c = obj_ns_list[:ns_list_trunc] + for c, c_n, c_ns in zip(class_list_obj, class_list, + class_list_ns): + obj_other_ns = c_ns.split('::')[:-1] + if obj_ns_c == obj_other_ns and (f_cmp(parent, c_n) or + f_cmp(parent, c_ns)): + return c['obj'] + pi += 1 + return None + def build_inheritance_list(self): """Build list of inheritance between objects @@ -1508,20 +1570,15 @@ be used to avoid this sorting step. # Build list of classes in diagram class_list_obj, class_list, class_list_ns = self._get_class_list() - # Create relationships - # Inheritance for obj in self._objects: obj_name = obj.name if isinstance(obj, Class): - for parent in obj.build_inheritance_list(): - parent_obj = None - if parent in class_list: - parent_obj = class_list_obj[ - class_list.index(parent)]['obj'] - elif parent in class_list_ns: - parent_obj = class_list_obj[ - class_list_ns.index(parent)]['obj'] + ns_list = [s for s in obj._namespace.split('::') if s != ''] + cl_plist = [s for s in obj.name.split('::') if s != ''] + obj_ns_list_base = ns_list + cl_plist[:-1] + for parent_in in obj.build_inheritance_list(): + parent_obj = self.find_parent(parent_in, obj_ns_list_base) if parent_obj is not None: self._inheritance_list.append( ClassInheritanceRelationship( @@ -1543,32 +1600,36 @@ be used to avoid this sorting step. # Build list of classes in diagram class_list_obj, class_list, class_list_ns = self._get_class_list() - # Build member type list - variable_type_list = {} - for obj in self._objects: - obj_name = obj.name - if isinstance(obj, Class): - variable_type_list[obj_name] = obj.build_variable_type_list() # Create aggregation links aggregation_counts = {} - for child_class in class_list: - if child_class in variable_type_list.keys(): - var_types = variable_type_list[child_class] - for var_type in var_types: - for parent in class_list or parent in class_list_ns: - if re.search(r'\b' + parent + r'\b', var_type): + for obj in self._objects: + obj_name = obj.name + if isinstance(obj, Class): + ns_list = [s for s in obj._namespace.split('::') if s != ''] + cl_plist = [s for s in obj.name.split('::') if s != ''] + obj_ns_list_base = ns_list + cl_plist[:-1] + for var_in in obj.build_variable_type_list(): + var_obj = self.find_parent( + var_in, obj_ns_list_base, + f_cmp=lambda x, y: re.search(r'\b' + y + r'\b', x)) + if var_obj: + p_ns_list = [s for s in var_obj._namespace.split('::') + if s != ''] + p_cl_plist = [s for s in var_obj.name.split('::') + if s != ''] + if '{}*'.format(var_obj.name) in var_in: + rel_type = 'aggregation' + else: rel_type = 'composition' - if '{}*'.format(parent) in var_type: - rel_type = 'aggregation' - self._augment_comp(aggregation_counts, parent, - child_class, rel_type=rel_type) + self._augment_comp(aggregation_counts, + '::'.join(p_ns_list + p_cl_plist), + '::'.join(ns_list + cl_plist), + rel_type=rel_type) + for obj_class, obj_comp_list in aggregation_counts.items(): for comp_parent, rel_type, comp_count in obj_comp_list: - if obj_class in class_list: - obj_class_idx = class_list.index(obj_class) - comp_parent_idx = class_list.index(comp_parent) - elif obj_class in class_list_ns: + if obj_class in class_list_ns: obj_class_idx = class_list_ns.index(obj_class) comp_parent_idx = class_list_ns.index(comp_parent) obj_class_obj = class_list_obj[obj_class_idx]['obj'] @@ -1592,31 +1653,21 @@ be used to avoid this sorting step. self._dependency_list = [] class_list_obj, class_list, class_list_ns = self._get_class_list() - # Create relationships - - # Add all objects name to list - objects_name = [] - for obj in self._objects: - objects_name.append(obj.name) - # Dependency for obj in self._objects: if isinstance(obj, Class): + ns_list = [s for s in obj._namespace.split('::') if s != ''] + cl_plist = [s for s in obj.name.split('::') if s != ''] + obj_ns_list_base = ns_list + cl_plist[:-1] for member in obj._member_list: # Check if the member is a method if isinstance(member, ClassMethod): - for method in member._param_list: - index = ValueError - try: - # Check if the method param type is a Class - # type - index = [re.search(o, method[0]) is not None - for o in objects_name].index(True) - except ValueError: - pass - if index != ValueError and method[0] != obj.name: - depend_obj = self._objects[index] - + for param in member._param_list: + depend_obj = self.find_parent( + param[0], obj_ns_list_base, + f_cmp=lambda x, y: re.search( + r'\b' + y + r'\b', x)) + if depend_obj is not None and param[0] != obj.name: self._dependency_list.append( ClassDependencyRelationship( depend_obj, obj)) @@ -1632,17 +1683,16 @@ be used to avoid this sorting step. for obj in self._objects: obj_name = obj.name if isinstance(obj, (Class, Enum)): - parent = obj._parent - parent_obj = None - if parent and parent in class_list: - parent_obj = class_list_obj[ - class_list.index(parent)]['obj'] - elif parent and parent in class_list_ns: - parent_obj = class_list_obj[ - class_list_ns.index(parent)]['obj'] - if parent_obj is not None: - self._nesting_list.append(ClassNestingRelationship( - parent_obj, obj)) + ns_list = [s for s in obj._namespace.split('::') if s != ''] + cl_plist = [s for s in obj.name.split('::') if s != ''] + obj_ns_list_base = ns_list + cl_plist[:-1] + if obj._parent: + parent_obj = self.find_parent( + obj._parent, obj_ns_list_base, + f_cmp=lambda x, y: re.search(r'\b' + y + r'\b', x)) + if parent_obj is not None: + self._nesting_list.append(ClassNestingRelationship( + parent_obj, obj)) def _augment_comp(self, c_dict, c_parent, c_child, rel_type='aggregation'): """Increment the aggregation reference count @@ -1767,7 +1817,7 @@ variable types by eliminating spaces around ``\*`` characters. str The type string after cleanup """ - return re.sub('\s*([<>])\s*', r'\1', + return re.sub(r'\s*([<>])\s*', r'\1', re.sub(r'[ ]+([*&])', r'\1', re.sub(r'(\s)+', r'\1', type_str))) @@ -2059,7 +2109,7 @@ to parse input arguments. The function passes the command line arguments to the required=False, default=None, metavar='JINJA-FILE', help='path to jinja2 template file') parser.add_argument('--version', action='version', - version='%(prog)s ' + '0.8.4') + version='%(prog)s ' + '0.8.5') args = parser.parse_args() if len(args.input_files) > 0: CreatePlantUMLFile(args.input_files, args.output_file, @@ -2433,7 +2483,7 @@ TableĀ `tbl-unittest-enum`_. +-------------------------------------+-----------------------------------------+ | "enum Test\n{\n A = 0, B = 12\n };" | "enum Test {\n\tA\n\tB\n}\n" | +-------------------------------------+-----------------------------------------+ - | "enum { A, B };" | "enum empty {\n\tA\n\tB\n}\n"" | + | "enum { A, B };" | "enum empty {\n\tA\n\tB\n}\n" | +-------------------------------------+-----------------------------------------+ @@ -2556,6 +2606,14 @@ The following can be extended to improve testing, as long as the corresponding bool _AbstractMethod(int param) override; }; + // Duplicate class names in different scopes + class Class04 { + int b; + }; + class Class04_derived { + int c; + }; + .. code:: c++ :name: hpp-simple-classes-3 @@ -2603,6 +2661,7 @@ The following can be extended to improve testing, as long as the corresponding protected: Struct _s; Enum _e; + Class04* _p; }; }; }; @@ -2688,6 +2747,16 @@ The comparison takes into account the white space, indentation, etc. } + class Class04 { + -b : int + } + + + class Class04_derived { + -c : int + } + + enum Enum01 { VALUE_0 VALUE_1 @@ -2754,6 +2823,7 @@ The comparison takes into account the white space, indentation, etc. namespace NestedNamespace { class Class04_ns { + #_p : Class04* #_e : Enum #_s : Struct } @@ -2798,6 +2868,9 @@ The comparison takes into account the white space, indentation, etc. Interface.Class04 *-- Enum01 + Interface.NestedNamespace.Class04_ns o-- Interface.Class04 + + Interface.NestedNamespace.Class04_ns *-- Interface.Enum @@ -2881,6 +2954,16 @@ The comparison takes into account the white space, indentation, etc. } + class Class04 { + -b : int + } + + + class Class04_derived { + -c : int + } + + enum Enum01 { VALUE_0 VALUE_1 @@ -2947,6 +3030,7 @@ The comparison takes into account the white space, indentation, etc. namespace NestedNamespace { class Class04_ns { + #_p : Class04* #_e : Enum #_s : Struct } @@ -2991,6 +3075,9 @@ The comparison takes into account the white space, indentation, etc. Interface.Class04 *-- Enum01 + Interface.NestedNamespace.Class04_ns o-- Interface.Class04 + + Interface.NestedNamespace.Class04_ns *-- Interface.Enum @@ -3128,14 +3215,15 @@ The system test validates the following: diag_str_list_add.sort_elements() assert diag_render_ref == diag_str_list_add.render() - # Create from string - diag_str = hpp2plantuml.Diagram(flag_dep=flag_dep) - diag_str.create_from_string('\n'.join(input_str_list)) - assert diag_render_ref == diag_str.render() - # Reset and parse - diag_str.clear() - diag_str.create_from_string('\n'.join(input_str_list)) - assert diag_render_ref == diag_str.render() + # # Create from string + # # Disable test to allow duplicate class name (in different files) + # diag_str = hpp2plantuml.Diagram(flag_dep=flag_dep) + # diag_str.create_from_string('\n'.join(input_str_list)) + # assert diag_render_ref == diag_str.render() + # # Reset and parse + # diag_str.clear() + # diag_str.create_from_string('\n'.join(input_str_list)) + # assert diag_render_ref == diag_str.render() # Manually build object diag_manual_add = hpp2plantuml.Diagram(flag_dep=flag_dep) @@ -3150,7 +3238,7 @@ The system test validates the following: assert diag_render_ref == diag_manual_add.render() def test_main_function(self): - #self._test_main_function_helper(False) + self._test_main_function_helper(False) self._test_main_function_helper(True) def _test_main_function_helper(self, flag_dep=False): @@ -3342,7 +3430,7 @@ obtained using the source block described `sec-org-el-version`_. __title__ = "hpp2plantuml" __description__ = "Convert C++ header files to PlantUML" - __version__ = '0.8.4' + __version__ = '0.8.5' __uri__ = "https://github.com/thibaultmarin/hpp2plantuml" __doc__ = __description__ + " <" + __uri__ + ">" __author__ = "Thibault Marin" @@ -3853,9 +3941,9 @@ content of the file is mostly following the defaults, with a few exceptions: # built documents. # # The short X.Y version. - version = u'v' + u'0.8.4' + version = u'v' + u'0.8.5' # The full version, including alpha/beta/rc tags. - release = u'v' + u'0.8.4' + release = u'v' + u'0.8.5' # The language for content autogenerated by Sphinx. Refer to documentation # for a list of supported languages. @@ -3929,7 +4017,7 @@ content of the file is mostly following the defaults, with a few exceptions: # The name for this set of Sphinx documents. # " v documentation" by default. # - # html_title = u'hpp2plantuml ' + u'v' + u'0.8.4' + # html_title = u'hpp2plantuml ' + u'v' + u'0.8.5' # A shorter title for the navigation bar. Default is the same as html_title. # @@ -4295,6 +4383,25 @@ file is added for the ``numpydoc`` package. numpydoc +``readthedocs`` requires a configuration file in the project folder: + +:: + + + version: 2 + + build: + os: ubuntu-22.04 + tools: + python: "3.11" + + sphinx: + configuration: doc/source/conf.py + + python: + install: + - requirements: doc/source/readthedocs-requirements.txt + Org-mode setup -------------- diff --git a/hpp2plantuml.org b/hpp2plantuml.org index 4e63210..1b3b4cd 100644 --- a/hpp2plantuml.org +++ b/hpp2plantuml.org @@ -16,7 +16,7 @@ this single org-file. The current version of the code is: #+NAME: hpp2plantuml-version -: 0.8.4 +: 0.8.5 The source code can be found on GitHub: https://github.com/thibaultmarin/hpp2plantuml. @@ -436,7 +436,8 @@ class Class(Container): element is the class name and the second element is a CppClass object) """ - super().__init__(header_class[1]['declaration_method'], header_class[0]) + super().__init__(header_class[1]['declaration_method'], + header_class[0]) self._abstract = header_class[1]['abstract'] self._template_type = None if 'template' in header_class[1]: @@ -1134,6 +1135,28 @@ class ClassNestingRelationship(ClassRelationship): Additional parameters passed to parent class """ super().__init__('nesting', c_parent, c_child, **kwargs) + + def _render_name(self, class_name, class_namespace): + """Specialize object name rendering for nesting relationship + + Use ``.`` separator if objects are in a namespace, ``::`` otherwise. + + Parameters + ---------- + class_name : str + Name of the class + class_namespace : str + Namespace or None if the class is defined in the default namespace + + Returns + ------- + str + Class name with appropriate prefix for use with link rendering + """ + if class_namespace: + return super()._render_name(class_name.replace('::', '.'), + class_namespace) + return super()._render_name(class_name, class_namespace) #+END_SRC @@ -1470,6 +1493,45 @@ class Diagram(object): class_list = [c['name'] for c in class_list_obj] return class_list_obj, class_list, class_list_ns + def find_parent(self, parent_in, obj_ns_list_base, f_cmp=None): + """Find object matching name and scope of input + + Parameters + ---------- + parent_in : str + Name of object to locate + obj_ns_list_base : list(str) + List of namespaces and nested classes for current object + f_cmp : callable + Comparison function with two arguments object name to search and + regular expression (string equality by default) + + Returns + ------- + Class or None + Matching :class:`Class` object if found + """ + if f_cmp is None: + f_cmp = lambda x, y: x == y + class_list_obj, class_list, class_list_ns = self._get_class_list() + parent_s = parent_in.split('::') + parent = parent_s[-1] + obj_ns_list = obj_ns_list_base + parent_s[:-1] + parent_obj = None + found = False + pi = 0 + while not found and pi <= len(obj_ns_list): + ns_list_trunc = -pi if pi > 0 else None + obj_ns_c = obj_ns_list[:ns_list_trunc] + for c, c_n, c_ns in zip(class_list_obj, class_list, + class_list_ns): + obj_other_ns = c_ns.split('::')[:-1] + if obj_ns_c == obj_other_ns and (f_cmp(parent, c_n) or + f_cmp(parent, c_ns)): + return c['obj'] + pi += 1 + return None + def build_inheritance_list(self): """Build list of inheritance between objects @@ -1485,20 +1547,15 @@ class Diagram(object): # Build list of classes in diagram class_list_obj, class_list, class_list_ns = self._get_class_list() - # Create relationships - # Inheritance for obj in self._objects: obj_name = obj.name if isinstance(obj, Class): - for parent in obj.build_inheritance_list(): - parent_obj = None - if parent in class_list: - parent_obj = class_list_obj[ - class_list.index(parent)]['obj'] - elif parent in class_list_ns: - parent_obj = class_list_obj[ - class_list_ns.index(parent)]['obj'] + ns_list = [s for s in obj._namespace.split('::') if s != ''] + cl_plist = [s for s in obj.name.split('::') if s != ''] + obj_ns_list_base = ns_list + cl_plist[:-1] + for parent_in in obj.build_inheritance_list(): + parent_obj = self.find_parent(parent_in, obj_ns_list_base) if parent_obj is not None: self._inheritance_list.append( ClassInheritanceRelationship( @@ -1520,32 +1577,36 @@ class Diagram(object): # Build list of classes in diagram class_list_obj, class_list, class_list_ns = self._get_class_list() - # Build member type list - variable_type_list = {} - for obj in self._objects: - obj_name = obj.name - if isinstance(obj, Class): - variable_type_list[obj_name] = obj.build_variable_type_list() # Create aggregation links aggregation_counts = {} - for child_class in class_list: - if child_class in variable_type_list.keys(): - var_types = variable_type_list[child_class] - for var_type in var_types: - for parent in class_list or parent in class_list_ns: - if re.search(r'\b' + parent + r'\b', var_type): + for obj in self._objects: + obj_name = obj.name + if isinstance(obj, Class): + ns_list = [s for s in obj._namespace.split('::') if s != ''] + cl_plist = [s for s in obj.name.split('::') if s != ''] + obj_ns_list_base = ns_list + cl_plist[:-1] + for var_in in obj.build_variable_type_list(): + var_obj = self.find_parent( + var_in, obj_ns_list_base, + f_cmp=lambda x, y: re.search(r'\b' + y + r'\b', x)) + if var_obj: + p_ns_list = [s for s in var_obj._namespace.split('::') + if s != ''] + p_cl_plist = [s for s in var_obj.name.split('::') + if s != ''] + if '{}*'.format(var_obj.name) in var_in: + rel_type = 'aggregation' + else: rel_type = 'composition' - if '{}*'.format(parent) in var_type: - rel_type = 'aggregation' - self._augment_comp(aggregation_counts, parent, - child_class, rel_type=rel_type) + self._augment_comp(aggregation_counts, + '::'.join(p_ns_list + p_cl_plist), + '::'.join(ns_list + cl_plist), + rel_type=rel_type) + for obj_class, obj_comp_list in aggregation_counts.items(): for comp_parent, rel_type, comp_count in obj_comp_list: - if obj_class in class_list: - obj_class_idx = class_list.index(obj_class) - comp_parent_idx = class_list.index(comp_parent) - elif obj_class in class_list_ns: + if obj_class in class_list_ns: obj_class_idx = class_list_ns.index(obj_class) comp_parent_idx = class_list_ns.index(comp_parent) obj_class_obj = class_list_obj[obj_class_idx]['obj'] @@ -1569,31 +1630,21 @@ class Diagram(object): self._dependency_list = [] class_list_obj, class_list, class_list_ns = self._get_class_list() - # Create relationships - - # Add all objects name to list - objects_name = [] - for obj in self._objects: - objects_name.append(obj.name) - # Dependency for obj in self._objects: if isinstance(obj, Class): + ns_list = [s for s in obj._namespace.split('::') if s != ''] + cl_plist = [s for s in obj.name.split('::') if s != ''] + obj_ns_list_base = ns_list + cl_plist[:-1] for member in obj._member_list: # Check if the member is a method if isinstance(member, ClassMethod): - for method in member._param_list: - index = ValueError - try: - # Check if the method param type is a Class - # type - index = [re.search(o, method[0]) is not None - for o in objects_name].index(True) - except ValueError: - pass - if index != ValueError and method[0] != obj.name: - depend_obj = self._objects[index] - + for param in member._param_list: + depend_obj = self.find_parent( + param[0], obj_ns_list_base, + f_cmp=lambda x, y: re.search( + r'\b' + y + r'\b', x)) + if depend_obj is not None and param[0] != obj.name: self._dependency_list.append( ClassDependencyRelationship( depend_obj, obj)) @@ -1609,17 +1660,16 @@ class Diagram(object): for obj in self._objects: obj_name = obj.name if isinstance(obj, (Class, Enum)): - parent = obj._parent - parent_obj = None - if parent and parent in class_list: - parent_obj = class_list_obj[ - class_list.index(parent)]['obj'] - elif parent and parent in class_list_ns: - parent_obj = class_list_obj[ - class_list_ns.index(parent)]['obj'] - if parent_obj is not None: - self._nesting_list.append(ClassNestingRelationship( - parent_obj, obj)) + ns_list = [s for s in obj._namespace.split('::') if s != ''] + cl_plist = [s for s in obj.name.split('::') if s != ''] + obj_ns_list_base = ns_list + cl_plist[:-1] + if obj._parent: + parent_obj = self.find_parent( + obj._parent, obj_ns_list_base, + f_cmp=lambda x, y: re.search(r'\b' + y + r'\b', x)) + if parent_obj is not None: + self._nesting_list.append(ClassNestingRelationship( + parent_obj, obj)) def _augment_comp(self, c_dict, c_parent, c_child, rel_type='aggregation'): """Increment the aggregation reference count @@ -1744,7 +1794,7 @@ def _cleanup_type(type_str): str The type string after cleanup """ - return re.sub('\s*([<>])\s*', r'\1', + return re.sub(r'\s*([<>])\s*', r'\1', re.sub(r'[ ]+([*&])', r'\1', re.sub(r'(\s)+', r'\1', type_str))) @@ -2379,7 +2429,7 @@ Table{{{tt}}}[[tbl-unittest-enum]]. |-------------------------------------+-----------------------------------------| | "enum Test { A, B, CD, E };" | "enum Test {\n\tA\n\tB\n\tCD\n\tE\n}\n" | | "enum Test\n{\n A = 0, B = 12\n };" | "enum Test {\n\tA\n\tB\n}\n" | -| "enum { A, B };" | "enum empty {\n\tA\n\tB\n}\n"" | +| "enum { A, B };" | "enum empty {\n\tA\n\tB\n}\n" | #+NAME: test-unit-enum @@ -2490,6 +2540,14 @@ private: static bool _StaticPrivateMethod(bool param); bool _AbstractMethod(int param) override; }; + +// Duplicate class names in different scopes +class Class04 { + int b; +}; +class Class04_derived { + int c; +}; #+END_SRC #+NAME: hpp-simple-classes-3 @@ -2538,6 +2596,7 @@ namespace Interface { protected: Struct _s; Enum _e; + Class04* _p; }; }; }; @@ -2623,6 +2682,16 @@ class Class03 > { } +class Class04 { + -b : int +} + + +class Class04_derived { + -c : int +} + + enum Enum01 { VALUE_0 VALUE_1 @@ -2689,6 +2758,7 @@ namespace Interface { namespace NestedNamespace { class Class04_ns { + #_p : Class04* #_e : Enum #_s : Struct } @@ -2733,6 +2803,9 @@ Interface.Class04 o-- Class01 Interface.Class04 *-- Enum01 +Interface.NestedNamespace.Class04_ns o-- Interface.Class04 + + Interface.NestedNamespace.Class04_ns *-- Interface.Enum @@ -2815,6 +2888,16 @@ class Class03 > { } +class Class04 { + -b : int +} + + +class Class04_derived { + -c : int +} + + enum Enum01 { VALUE_0 VALUE_1 @@ -2881,6 +2964,7 @@ namespace Interface { namespace NestedNamespace { class Class04_ns { + #_p : Class04* #_e : Enum #_s : Struct } @@ -2925,6 +3009,9 @@ Interface.Class04 o-- Class01 Interface.Class04 *-- Enum01 +Interface.NestedNamespace.Class04_ns o-- Interface.Class04 + + Interface.NestedNamespace.Class04_ns *-- Interface.Enum @@ -3057,14 +3144,15 @@ class TestFullDiagram(): diag_str_list_add.sort_elements() assert diag_render_ref == diag_str_list_add.render() - # Create from string - diag_str = hpp2plantuml.Diagram(flag_dep=flag_dep) - diag_str.create_from_string('\n'.join(input_str_list)) - assert diag_render_ref == diag_str.render() - # Reset and parse - diag_str.clear() - diag_str.create_from_string('\n'.join(input_str_list)) - assert diag_render_ref == diag_str.render() + # # Create from string + # # Disable test to allow duplicate class name (in different files) + # diag_str = hpp2plantuml.Diagram(flag_dep=flag_dep) + # diag_str.create_from_string('\n'.join(input_str_list)) + # assert diag_render_ref == diag_str.render() + # # Reset and parse + # diag_str.clear() + # diag_str.create_from_string('\n'.join(input_str_list)) + # assert diag_render_ref == diag_str.render() # Manually build object diag_manual_add = hpp2plantuml.Diagram(flag_dep=flag_dep) @@ -3079,7 +3167,7 @@ class TestFullDiagram(): assert diag_render_ref == diag_manual_add.render() def test_main_function(self): - #self._test_main_function_helper(False) + self._test_main_function_helper(False) self._test_main_function_helper(True) def _test_main_function_helper(self, flag_dep=False): @@ -3875,6 +3963,26 @@ file is added for the =numpydoc= package. numpydoc #+END_SRC +=readthedocs= requires a configuration file in the project folder: + +#+begin_src conf :tangle .readthedocs.yaml +version: 2 + +build: + os: ubuntu-22.04 + tools: + python: "3.11" + +sphinx: + configuration: doc/source/conf.py + +python: + install: + - method: pip + path: . + - requirements: doc/source/readthedocs-requirements.txt +#+end_src + * DONE Org-mode setup diff --git a/src/hpp2plantuml/__init__.py b/src/hpp2plantuml/__init__.py index 7738b0f..26b28b4 100644 --- a/src/hpp2plantuml/__init__.py +++ b/src/hpp2plantuml/__init__.py @@ -106,7 +106,7 @@ __title__ = "hpp2plantuml" __description__ = "Convert C++ header files to PlantUML" -__version__ = '0.8.4' +__version__ = '0.8.5' __uri__ = "https://github.com/thibaultmarin/hpp2plantuml" __doc__ = __description__ + " <" + __uri__ + ">" __author__ = "Thibault Marin" diff --git a/src/hpp2plantuml/hpp2plantuml.py b/src/hpp2plantuml/hpp2plantuml.py index 847f8f5..442b298 100644 --- a/src/hpp2plantuml/hpp2plantuml.py +++ b/src/hpp2plantuml/hpp2plantuml.py @@ -236,7 +236,8 @@ def __init__(self, header_class): element is the class name and the second element is a CppClass object) """ - super().__init__(header_class[1]['declaration_method'], header_class[0]) + super().__init__(header_class[1]['declaration_method'], + header_class[0]) self._abstract = header_class[1]['abstract'] self._template_type = None if 'template' in header_class[1]: @@ -803,6 +804,28 @@ def __init__(self, c_parent, c_child, **kwargs): """ super().__init__('nesting', c_parent, c_child, **kwargs) + def _render_name(self, class_name, class_namespace): + """Specialize object name rendering for nesting relationship + + Use ``.`` separator if objects are in a namespace, ``::`` otherwise. + + Parameters + ---------- + class_name : str + Name of the class + class_namespace : str + Namespace or None if the class is defined in the default namespace + + Returns + ------- + str + Class name with appropriate prefix for use with link rendering + """ + if class_namespace: + return super()._render_name(class_name.replace('::', '.'), + class_namespace) + return super()._render_name(class_name, class_namespace) + # %% Diagram class @@ -1070,6 +1093,45 @@ def _get_class_list(self): class_list = [c['name'] for c in class_list_obj] return class_list_obj, class_list, class_list_ns + def find_parent(self, parent_in, obj_ns_list_base, f_cmp=None): + """Find object matching name and scope of input + + Parameters + ---------- + parent_in : str + Name of object to locate + obj_ns_list_base : list(str) + List of namespaces and nested classes for current object + f_cmp : callable + Comparison function with two arguments object name to search and + regular expression (string equality by default) + + Returns + ------- + Class or None + Matching :class:`Class` object if found + """ + if f_cmp is None: + f_cmp = lambda x, y: x == y + class_list_obj, class_list, class_list_ns = self._get_class_list() + parent_s = parent_in.split('::') + parent = parent_s[-1] + obj_ns_list = obj_ns_list_base + parent_s[:-1] + parent_obj = None + found = False + pi = 0 + while not found and pi <= len(obj_ns_list): + ns_list_trunc = -pi if pi > 0 else None + obj_ns_c = obj_ns_list[:ns_list_trunc] + for c, c_n, c_ns in zip(class_list_obj, class_list, + class_list_ns): + obj_other_ns = c_ns.split('::')[:-1] + if obj_ns_c == obj_other_ns and (f_cmp(parent, c_n) or + f_cmp(parent, c_ns)): + return c['obj'] + pi += 1 + return None + def build_inheritance_list(self): """Build list of inheritance between objects @@ -1085,20 +1147,15 @@ def build_inheritance_list(self): # Build list of classes in diagram class_list_obj, class_list, class_list_ns = self._get_class_list() - # Create relationships - # Inheritance for obj in self._objects: obj_name = obj.name if isinstance(obj, Class): - for parent in obj.build_inheritance_list(): - parent_obj = None - if parent in class_list: - parent_obj = class_list_obj[ - class_list.index(parent)]['obj'] - elif parent in class_list_ns: - parent_obj = class_list_obj[ - class_list_ns.index(parent)]['obj'] + ns_list = [s for s in obj._namespace.split('::') if s != ''] + cl_plist = [s for s in obj.name.split('::') if s != ''] + obj_ns_list_base = ns_list + cl_plist[:-1] + for parent_in in obj.build_inheritance_list(): + parent_obj = self.find_parent(parent_in, obj_ns_list_base) if parent_obj is not None: self._inheritance_list.append( ClassInheritanceRelationship( @@ -1120,32 +1177,36 @@ def build_aggregation_list(self): # Build list of classes in diagram class_list_obj, class_list, class_list_ns = self._get_class_list() - # Build member type list - variable_type_list = {} - for obj in self._objects: - obj_name = obj.name - if isinstance(obj, Class): - variable_type_list[obj_name] = obj.build_variable_type_list() # Create aggregation links aggregation_counts = {} - for child_class in class_list: - if child_class in variable_type_list.keys(): - var_types = variable_type_list[child_class] - for var_type in var_types: - for parent in class_list or parent in class_list_ns: - if re.search(r'\b' + parent + r'\b', var_type): + for obj in self._objects: + obj_name = obj.name + if isinstance(obj, Class): + ns_list = [s for s in obj._namespace.split('::') if s != ''] + cl_plist = [s for s in obj.name.split('::') if s != ''] + obj_ns_list_base = ns_list + cl_plist[:-1] + for var_in in obj.build_variable_type_list(): + var_obj = self.find_parent( + var_in, obj_ns_list_base, + f_cmp=lambda x, y: re.search(r'\b' + y + r'\b', x)) + if var_obj: + p_ns_list = [s for s in var_obj._namespace.split('::') + if s != ''] + p_cl_plist = [s for s in var_obj.name.split('::') + if s != ''] + if '{}*'.format(var_obj.name) in var_in: + rel_type = 'aggregation' + else: rel_type = 'composition' - if '{}*'.format(parent) in var_type: - rel_type = 'aggregation' - self._augment_comp(aggregation_counts, parent, - child_class, rel_type=rel_type) + self._augment_comp(aggregation_counts, + '::'.join(p_ns_list + p_cl_plist), + '::'.join(ns_list + cl_plist), + rel_type=rel_type) + for obj_class, obj_comp_list in aggregation_counts.items(): for comp_parent, rel_type, comp_count in obj_comp_list: - if obj_class in class_list: - obj_class_idx = class_list.index(obj_class) - comp_parent_idx = class_list.index(comp_parent) - elif obj_class in class_list_ns: + if obj_class in class_list_ns: obj_class_idx = class_list_ns.index(obj_class) comp_parent_idx = class_list_ns.index(comp_parent) obj_class_obj = class_list_obj[obj_class_idx]['obj'] @@ -1169,31 +1230,21 @@ def build_dependency_list(self): self._dependency_list = [] class_list_obj, class_list, class_list_ns = self._get_class_list() - # Create relationships - - # Add all objects name to list - objects_name = [] - for obj in self._objects: - objects_name.append(obj.name) - # Dependency for obj in self._objects: if isinstance(obj, Class): + ns_list = [s for s in obj._namespace.split('::') if s != ''] + cl_plist = [s for s in obj.name.split('::') if s != ''] + obj_ns_list_base = ns_list + cl_plist[:-1] for member in obj._member_list: # Check if the member is a method if isinstance(member, ClassMethod): - for method in member._param_list: - index = ValueError - try: - # Check if the method param type is a Class - # type - index = [re.search(o, method[0]) is not None - for o in objects_name].index(True) - except ValueError: - pass - if index != ValueError and method[0] != obj.name: - depend_obj = self._objects[index] - + for param in member._param_list: + depend_obj = self.find_parent( + param[0], obj_ns_list_base, + f_cmp=lambda x, y: re.search( + r'\b' + y + r'\b', x)) + if depend_obj is not None and param[0] != obj.name: self._dependency_list.append( ClassDependencyRelationship( depend_obj, obj)) @@ -1209,17 +1260,16 @@ def build_nesting_list(self): for obj in self._objects: obj_name = obj.name if isinstance(obj, (Class, Enum)): - parent = obj._parent - parent_obj = None - if parent and parent in class_list: - parent_obj = class_list_obj[ - class_list.index(parent)]['obj'] - elif parent and parent in class_list_ns: - parent_obj = class_list_obj[ - class_list_ns.index(parent)]['obj'] - if parent_obj is not None: - self._nesting_list.append(ClassNestingRelationship( - parent_obj, obj)) + ns_list = [s for s in obj._namespace.split('::') if s != ''] + cl_plist = [s for s in obj.name.split('::') if s != ''] + obj_ns_list_base = ns_list + cl_plist[:-1] + if obj._parent: + parent_obj = self.find_parent( + obj._parent, obj_ns_list_base, + f_cmp=lambda x, y: re.search(r'\b' + y + r'\b', x)) + if parent_obj is not None: + self._nesting_list.append(ClassNestingRelationship( + parent_obj, obj)) def _augment_comp(self, c_dict, c_parent, c_child, rel_type='aggregation'): """Increment the aggregation reference count @@ -1330,7 +1380,7 @@ def _cleanup_type(type_str): str The type string after cleanup """ - return re.sub('\s*([<>])\s*', r'\1', + return re.sub(r'\s*([<>])\s*', r'\1', re.sub(r'[ ]+([*&])', r'\1', re.sub(r'(\s)+', r'\1', type_str))) @@ -1498,7 +1548,7 @@ def main(): required=False, default=None, metavar='JINJA-FILE', help='path to jinja2 template file') parser.add_argument('--version', action='version', - version='%(prog)s ' + '0.8.4') + version='%(prog)s ' + '0.8.5') args = parser.parse_args() if len(args.input_files) > 0: CreatePlantUMLFile(args.input_files, args.output_file, diff --git a/tests/simple_classes.puml b/tests/simple_classes.puml index ce47180..de1f2a9 100644 --- a/tests/simple_classes.puml +++ b/tests/simple_classes.puml @@ -43,6 +43,16 @@ class Class03 > { } +class Class04 { + -b : int +} + + +class Class04_derived { + -c : int +} + + enum Enum01 { VALUE_0 VALUE_1 @@ -109,6 +119,7 @@ namespace Interface { namespace NestedNamespace { class Class04_ns { + #_p : Class04* #_e : Enum #_s : Struct } @@ -153,6 +164,9 @@ Interface.Class04 o-- Class01 Interface.Class04 *-- Enum01 +Interface.NestedNamespace.Class04_ns o-- Interface.Class04 + + Interface.NestedNamespace.Class04_ns *-- Interface.Enum diff --git a/tests/simple_classes_1_2.hpp b/tests/simple_classes_1_2.hpp index 705fa43..7775a3a 100644 --- a/tests/simple_classes_1_2.hpp +++ b/tests/simple_classes_1_2.hpp @@ -27,3 +27,11 @@ class Class02 : public Class01 { static bool _StaticPrivateMethod(bool param); bool _AbstractMethod(int param) override; }; + +// Duplicate class names in different scopes +class Class04 { + int b; +}; +class Class04_derived { + int c; +}; diff --git a/tests/simple_classes_3.hpp b/tests/simple_classes_3.hpp index 37d8a4e..3a322a2 100644 --- a/tests/simple_classes_3.hpp +++ b/tests/simple_classes_3.hpp @@ -42,6 +42,7 @@ namespace Interface { protected: Struct _s; Enum _e; + Class04* _p; }; }; }; diff --git a/tests/simple_classes_nodep.puml b/tests/simple_classes_nodep.puml index 181052b..d43a6c5 100644 --- a/tests/simple_classes_nodep.puml +++ b/tests/simple_classes_nodep.puml @@ -43,6 +43,16 @@ class Class03 > { } +class Class04 { + -b : int +} + + +class Class04_derived { + -c : int +} + + enum Enum01 { VALUE_0 VALUE_1 @@ -109,6 +119,7 @@ namespace Interface { namespace NestedNamespace { class Class04_ns { + #_p : Class04* #_e : Enum #_s : Struct } @@ -153,6 +164,9 @@ Interface.Class04 o-- Class01 Interface.Class04 *-- Enum01 +Interface.NestedNamespace.Class04_ns o-- Interface.Class04 + + Interface.NestedNamespace.Class04_ns *-- Interface.Enum diff --git a/tests/test_hpp2plantuml.py b/tests/test_hpp2plantuml.py index 513efb5..c4ebd13 100644 --- a/tests/test_hpp2plantuml.py +++ b/tests/test_hpp2plantuml.py @@ -297,14 +297,15 @@ def _test_full_files_helper(self, flag_dep=False): diag_str_list_add.sort_elements() assert diag_render_ref == diag_str_list_add.render() - # Create from string - diag_str = hpp2plantuml.Diagram(flag_dep=flag_dep) - diag_str.create_from_string('\n'.join(input_str_list)) - assert diag_render_ref == diag_str.render() - # Reset and parse - diag_str.clear() - diag_str.create_from_string('\n'.join(input_str_list)) - assert diag_render_ref == diag_str.render() + # # Create from string + # # Disable test to allow duplicate class name (in different files) + # diag_str = hpp2plantuml.Diagram(flag_dep=flag_dep) + # diag_str.create_from_string('\n'.join(input_str_list)) + # assert diag_render_ref == diag_str.render() + # # Reset and parse + # diag_str.clear() + # diag_str.create_from_string('\n'.join(input_str_list)) + # assert diag_render_ref == diag_str.render() # Manually build object diag_manual_add = hpp2plantuml.Diagram(flag_dep=flag_dep) @@ -319,7 +320,7 @@ def _test_full_files_helper(self, flag_dep=False): assert diag_render_ref == diag_manual_add.render() def test_main_function(self): - #self._test_main_function_helper(False) + self._test_main_function_helper(False) self._test_main_function_helper(True) def _test_main_function_helper(self, flag_dep=False):