PySimultan 0.6.0.4__py3-none-any.whl → 0.6.0.8__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (111) hide show
  1. PySimultan2/CHANGELOG.md +6 -0
  2. PySimultan2/__about__.py +1 -1
  3. PySimultan2/data_model.py +35 -1
  4. PySimultan2/default_types.py +9 -0
  5. PySimultan2/files.py +107 -11
  6. PySimultan2/object_mapper.py +1 -0
  7. PySimultan2/simultan_object.py +36 -0
  8. PySimultan2/typings/SIMULTAN/Data/Assets/__init__.pyi +616 -0
  9. PySimultan2/typings/SIMULTAN/Data/Components/__init__.pyi +2538 -0
  10. PySimultan2/typings/SIMULTAN/Data/FlowNetworks/__init__.pyi +499 -0
  11. PySimultan2/typings/SIMULTAN/Data/Geometry/__init__.pyi +2219 -0
  12. PySimultan2/typings/SIMULTAN/Data/MultiValues/__init__.pyi +682 -0
  13. PySimultan2/typings/SIMULTAN/Data/SimMath/__init__.pyi +1654 -0
  14. PySimultan2/typings/SIMULTAN/Data/SimNetworks/__init__.pyi +442 -0
  15. PySimultan2/typings/SIMULTAN/Data/SitePlanner/__init__.pyi +193 -0
  16. PySimultan2/typings/SIMULTAN/Data/Taxonomy/__init__.pyi +367 -0
  17. PySimultan2/typings/SIMULTAN/Data/Users/__init__.pyi +116 -0
  18. PySimultan2/typings/SIMULTAN/Data/ValueMappings/__init__.pyi +212 -0
  19. PySimultan2/typings/SIMULTAN/Data/__init__.pyi +232 -0
  20. PySimultan2/typings/SIMULTAN/DataMapping/__init__.pyi +916 -0
  21. PySimultan2/typings/SIMULTAN/Excel/__init__.pyi +15 -0
  22. PySimultan2/typings/SIMULTAN/Exceptions/__init__.pyi +268 -0
  23. PySimultan2/typings/SIMULTAN/Exchange/SimNetworkConnectors/__init__.pyi +32 -0
  24. PySimultan2/typings/SIMULTAN/Exchange/__init__.pyi +116 -0
  25. PySimultan2/typings/SIMULTAN/Projects/ManagedFiles/__init__.pyi +433 -0
  26. PySimultan2/typings/SIMULTAN/Projects/__init__.pyi +435 -0
  27. PySimultan2/typings/SIMULTAN/Serializer/CODXF/__init__.pyi +103 -0
  28. PySimultan2/typings/SIMULTAN/Serializer/CSV/__init__.pyi +122 -0
  29. PySimultan2/typings/SIMULTAN/Serializer/DXF/__init__.pyi +1335 -0
  30. PySimultan2/typings/SIMULTAN/Serializer/Geometry/__init__.pyi +48 -0
  31. PySimultan2/typings/SIMULTAN/Serializer/JSON/__init__.pyi +562 -0
  32. PySimultan2/typings/SIMULTAN/Serializer/METADXF/__init__.pyi +11 -0
  33. PySimultan2/typings/SIMULTAN/Serializer/PADXF/__init__.pyi +21 -0
  34. PySimultan2/typings/SIMULTAN/Serializer/PPATH/__init__.pyi +11 -0
  35. PySimultan2/typings/SIMULTAN/Serializer/Projects/__init__.pyi +112 -0
  36. PySimultan2/typings/SIMULTAN/Serializer/SIMLINKS/__init__.pyi +5 -0
  37. PySimultan2/typings/SIMULTAN/Serializer/SPDXF/__init__.pyi +13 -0
  38. PySimultan2/typings/SIMULTAN/Serializer/SimGeo/__init__.pyi +54 -0
  39. PySimultan2/typings/SIMULTAN/Serializer/TXDXF/__init__.pyi +46 -0
  40. PySimultan2/typings/SIMULTAN/Serializer/XMI/__init__.pyi +22 -0
  41. PySimultan2/typings/SIMULTAN/Serializer/__init__.pyi +32 -0
  42. PySimultan2/typings/SIMULTAN/Utils/BackgroundWork/__init__.pyi +43 -0
  43. PySimultan2/typings/SIMULTAN/Utils/Collections/__init__.pyi +216 -0
  44. PySimultan2/typings/SIMULTAN/Utils/ElevationProvider/__init__.pyi +66 -0
  45. PySimultan2/typings/SIMULTAN/Utils/Files/__init__.pyi +48 -0
  46. PySimultan2/typings/SIMULTAN/Utils/Randomize/__init__.pyi +11 -0
  47. PySimultan2/typings/SIMULTAN/Utils/Streams/__init__.pyi +59 -0
  48. PySimultan2/typings/SIMULTAN/Utils/UndoRedo/__init__.pyi +133 -0
  49. PySimultan2/typings/SIMULTAN/Utils/__init__.pyi +570 -0
  50. PySimultan2/typings/System/Buffers/Binary/__init__.pyi +248 -0
  51. PySimultan2/typings/System/Buffers/Text/__init__.pyi +91 -0
  52. PySimultan2/typings/System/Buffers/__init__.pyi +192 -0
  53. PySimultan2/typings/System/CodeDom/Compiler/__init__.pyi +137 -0
  54. PySimultan2/typings/System/Collections/Concurrent/__init__.pyi +47 -0
  55. PySimultan2/typings/System/Collections/Generic/__init__.pyi +1293 -0
  56. PySimultan2/typings/System/Collections/ObjectModel/__init__.pyi +166 -0
  57. PySimultan2/typings/System/Collections/Specialized/__init__.pyi +82 -0
  58. PySimultan2/typings/System/Collections/__init__.pyi +403 -0
  59. PySimultan2/typings/System/ComponentModel/__init__.pyi +582 -0
  60. PySimultan2/typings/System/Configuration/Assemblies/__init__.pyi +30 -0
  61. PySimultan2/typings/System/Diagnostics/CodeAnalysis/__init__.pyi +315 -0
  62. PySimultan2/typings/System/Diagnostics/Contracts/__init__.pyi +297 -0
  63. PySimultan2/typings/System/Diagnostics/SymbolStore/__init__.pyi +9 -0
  64. PySimultan2/typings/System/Diagnostics/Tracing/__init__.pyi +641 -0
  65. PySimultan2/typings/System/Diagnostics/__init__.pyi +1101 -0
  66. PySimultan2/typings/System/Globalization/__init__.pyi +1675 -0
  67. PySimultan2/typings/System/IO/Enumeration/__init__.pyi +125 -0
  68. PySimultan2/typings/System/IO/__init__.pyi +2747 -0
  69. PySimultan2/typings/System/Linq/Expressions/__init__.pyi +1815 -0
  70. PySimultan2/typings/System/Net/__init__.pyi +81 -0
  71. PySimultan2/typings/System/Numerics/__init__.pyi +2853 -0
  72. PySimultan2/typings/System/Reflection/Emit/__init__.pyi +1945 -0
  73. PySimultan2/typings/System/Reflection/Metadata/__init__.pyi +24 -0
  74. PySimultan2/typings/System/Reflection/__init__.pyi +2724 -0
  75. PySimultan2/typings/System/Resources/__init__.pyi +205 -0
  76. PySimultan2/typings/System/Runtime/CompilerServices/__init__.pyi +1926 -0
  77. PySimultan2/typings/System/Runtime/ConstrainedExecution/__init__.pyi +49 -0
  78. PySimultan2/typings/System/Runtime/ExceptionServices/__init__.pyi +34 -0
  79. PySimultan2/typings/System/Runtime/InteropServices/ComTypes/__init__.pyi +758 -0
  80. PySimultan2/typings/System/Runtime/InteropServices/Marshalling/__init__.pyi +461 -0
  81. PySimultan2/typings/System/Runtime/InteropServices/ObjectiveC/__init__.pyi +48 -0
  82. PySimultan2/typings/System/Runtime/InteropServices/__init__.pyi +2632 -0
  83. PySimultan2/typings/System/Runtime/Intrinsics/Arm/__init__.pyi +4757 -0
  84. PySimultan2/typings/System/Runtime/Intrinsics/Wasm/__init__.pyi +844 -0
  85. PySimultan2/typings/System/Runtime/Intrinsics/X86/__init__.pyi +5642 -0
  86. PySimultan2/typings/System/Runtime/Intrinsics/__init__.pyi +4504 -0
  87. PySimultan2/typings/System/Runtime/Loader/__init__.pyi +63 -0
  88. PySimultan2/typings/System/Runtime/Remoting/__init__.pyi +7 -0
  89. PySimultan2/typings/System/Runtime/Serialization/__init__.pyi +269 -0
  90. PySimultan2/typings/System/Runtime/Versioning/__init__.pyi +200 -0
  91. PySimultan2/typings/System/Runtime/__init__.pyi +141 -0
  92. PySimultan2/typings/System/Security/Cryptography/__init__.pyi +39 -0
  93. PySimultan2/typings/System/Security/Permissions/__init__.pyi +163 -0
  94. PySimultan2/typings/System/Security/Principal/__init__.pyi +45 -0
  95. PySimultan2/typings/System/Security/__init__.pyi +347 -0
  96. PySimultan2/typings/System/Text/Unicode/__init__.pyi +62 -0
  97. PySimultan2/typings/System/Text/__init__.pyi +1590 -0
  98. PySimultan2/typings/System/Threading/Tasks/Sources/__init__.pyi +76 -0
  99. PySimultan2/typings/System/Threading/Tasks/__init__.pyi +1403 -0
  100. PySimultan2/typings/System/Threading/__init__.pyi +1788 -0
  101. PySimultan2/typings/System/Xml/Schema/__init__.pyi +1255 -0
  102. PySimultan2/typings/System/Xml/Serialization/__init__.pyi +16 -0
  103. PySimultan2/typings/System/Xml/XPath/__init__.pyi +474 -0
  104. PySimultan2/typings/System/Xml/__init__.pyi +2410 -0
  105. PySimultan2/typings/System/__init__.pyi +17821 -0
  106. PySimultan2/utils.py +4 -2
  107. {pysimultan-0.6.0.4.dist-info → pysimultan-0.6.0.8.dist-info}/METADATA +1 -1
  108. pysimultan-0.6.0.8.dist-info/RECORD +174 -0
  109. pysimultan-0.6.0.4.dist-info/RECORD +0 -76
  110. {pysimultan-0.6.0.4.dist-info → pysimultan-0.6.0.8.dist-info}/WHEEL +0 -0
  111. {pysimultan-0.6.0.4.dist-info → pysimultan-0.6.0.8.dist-info}/licenses/LICENSE.txt +0 -0
PySimultan2/CHANGELOG.md CHANGED
@@ -1,3 +1,9 @@
1
+ Version 0.6.0.7 (16.02.2024)
2
+ - Fixed Asset creation and deletion bugs
3
+
4
+ Version 0.6.0.3 (23.01.2024)
5
+ - Added sub_classes and super_classes to SimultanObject
6
+
1
7
  Version 0.6.0.2 (16.01.2024)
2
8
  - Added __delitem__ method to default_types.ComponentList
3
9
 
PySimultan2/__about__.py CHANGED
@@ -1 +1 @@
1
- version = '0.6.0.4'
1
+ version = '0.6.0.8'
PySimultan2/data_model.py CHANGED
@@ -655,6 +655,18 @@ class DataModel:
655
655
  logger.error(f'Error while adding resource {filename} to project {self.project_path}: {e}')
656
656
  raise e
657
657
 
658
+ def get_resource(self, key: Union[str, int, SystemFileInfo, DirectoryInfo]):
659
+
660
+ if isinstance(key, (int, SystemFileInfo, DirectoryInfo)):
661
+ return self.project_data_manager.AssetManager.GetResource(key)
662
+ elif isinstance(key, str):
663
+ if os.path.isfile(key):
664
+ return self.project_data_manager.AssetManager.GetResource(SystemFileInfo(key))
665
+ elif os.path.isdir(key):
666
+ return self.project_data_manager.AssetManager.GetResource(DirectoryInfo(key))
667
+ else:
668
+ return None
669
+
658
670
  def delete_resource(self, resource: Union[ResourceEntry, SystemFileInfo, ContainedResourceFileEntry]):
659
671
  """
660
672
  Delete a resource from the project and the project folder
@@ -678,9 +690,31 @@ class DataModel:
678
690
  parent_directory: DirectoryInfo=None,
679
691
  collision_name_format: str = '{0} ({1})') -> ResourceEntry:
680
692
 
693
+ # check if directory already exists
681
694
  if parent_directory is None:
682
- new_directory = self.project.CreateResourceDirIn(name, None, collision_name_format)
695
+ full_path = os.path.join(self.project.ProjectUnpackFolder.FullPath, name)
683
696
  else:
697
+ full_path = os.path.join(parent_directory.FullPath, name)
698
+
699
+ def get_existing(full_file_path: str):
700
+ new_directory = self.project_data_manager.AssetManager.GetResource(DirectoryInfo(full_path))
701
+ if new_directory is None:
702
+ res = self.project_data_manager.AssetManager.CreateResourceDirIn(os.path.basename(full_path),
703
+ DirectoryInfo(
704
+ os.path.dirname(
705
+ full_path)),
706
+ '0')
707
+
708
+ new_directory = self.project_data_manager.AssetManager.GetResource(res.Item1)
709
+ return new_directory
710
+
711
+ if os.path.exists(full_path):
712
+ new_directory = get_existing(full_path)
713
+ return new_directory
714
+
715
+ if parent_directory is None:
716
+ new_directory = self.project.CreateResourceDirIn(name, None, collision_name_format)
717
+ if parent_directory is not None:
684
718
  new_directory = self.project.CreateResourceDirIn(name, parent_directory, collision_name_format)
685
719
 
686
720
  return new_directory
@@ -5,6 +5,7 @@ import numpy as np
5
5
  import colorlog
6
6
  from typing import Union, List, Type, Set, Tuple, Any, Optional
7
7
 
8
+ from .geometry.geometry_base import classproperty
8
9
  from .utils import (sort_slots, create_simultan_component_for_taxonomy, create_mapped_python_object,
9
10
  set_property_to_sim_component, set_property_to_parameter, set_property_to_file_info,
10
11
  set_property_to_list, set_property_to_value_field, set_property_to_unknown_type,
@@ -32,6 +33,10 @@ class ComponentList(SimultanObject):
32
33
  _create_all = False # if true all properties are evaluated to create python objects when initialized
33
34
  _taxonomy = 'ComponentList'
34
35
 
36
+ @classproperty
37
+ def _original_class(cls) -> Type[ComponentList]:
38
+ return cls
39
+
35
40
  def __init__(self, *args, **kwargs):
36
41
  super().__init__(*args, **kwargs)
37
42
  self.component_policy = kwargs.get('component_policy', 'subcomponent') # component add policy of the content/parameter/property, 'reference' or 'subcomponent'
@@ -405,6 +410,10 @@ class ComponentDictionary(SimultanObject):
405
410
 
406
411
  _taxonomy = 'ComponentDict'
407
412
 
413
+ @classproperty
414
+ def _original_class(cls) -> Type[ComponentList]:
415
+ return cls
416
+
408
417
  def __init__(self, *args, **kwargs):
409
418
  self._dict = {}
410
419
  super().__init__(*args, **kwargs)
PySimultan2/files.py CHANGED
@@ -3,10 +3,12 @@ This module contains classes and functions to handle files and directories.
3
3
  """
4
4
  from __future__ import annotations
5
5
  from datetime import datetime
6
+ import glob
6
7
 
7
8
  import contextlib
8
9
  import os
9
10
  import io
11
+ import re
10
12
  import shutil
11
13
  import tempfile
12
14
  from typing import List, Union, Optional
@@ -124,6 +126,13 @@ def create_asset_from_string(filename: str,
124
126
  :param tag: Tag to be added to the asset.
125
127
  :return: ResourceFileEntry
126
128
  """
129
+
130
+ # check if file already exists
131
+ if target_dir is not None:
132
+ if isinstance(target_dir, DirectoryInfo):
133
+ target_dir = target_dir.full_path
134
+
135
+
127
136
  with tempdir() as dirpath:
128
137
  filepath = os.path.join(dirpath, filename)
129
138
  with open(filepath, 'w') as f:
@@ -290,11 +299,30 @@ class FileInfo(object, metaclass=MetaMock):
290
299
  """
291
300
 
292
301
  data_model = kwargs.get('data_model', config.get_default_data_model())
293
- resource = create_asset_from_string(filename,
294
- content,
295
- target_dir=target_dir,
296
- *args,
297
- **kwargs)
302
+
303
+ if target_dir is not None:
304
+ full_path = os.path.join(target_dir.full_path, filename)
305
+ else:
306
+ full_path = os.path.join(data_model.project.ProjectUnpackFolder, filename)
307
+
308
+
309
+
310
+
311
+ if os.path.isfile(full_path):
312
+ # check if resource entry exists
313
+ resource = data_model.get_resource(full_path)
314
+ if resource is None:
315
+ resource = create_asset_from_string(filename,
316
+ content,
317
+ target_dir=target_dir,
318
+ *args,
319
+ **kwargs)
320
+ else:
321
+ resource = create_asset_from_string(filename,
322
+ content,
323
+ target_dir=target_dir,
324
+ *args,
325
+ **kwargs)
298
326
 
299
327
  file_info = cls(resource_entry=resource,
300
328
  data_model=data_model)
@@ -332,6 +360,7 @@ class FileInfo(object, metaclass=MetaMock):
332
360
  :param kwargs:
333
361
  """
334
362
  # do custom stuff here
363
+ self.deleted = False
335
364
  self._resource_entry: Union[ResourceFileEntry, ContainedResourceFileEntry, None] = None
336
365
 
337
366
  if file_path is not None:
@@ -365,7 +394,7 @@ class FileInfo(object, metaclass=MetaMock):
365
394
 
366
395
  @property
367
396
  def resource_entry(self) -> Union[ResourceFileEntry, ContainedResourceFileEntry, None]:
368
- if self._resource_entry is None:
397
+ if self._resource_entry is None and not self.deleted:
369
398
  if self.data_model is None:
370
399
  logger.warning(f'No data model provided. Using default data model: {config.get_default_data_model().id}.')
371
400
  self.data_model = config.get_default_data_model()
@@ -453,7 +482,10 @@ class FileInfo(object, metaclass=MetaMock):
453
482
  self.file_obj.close()
454
483
 
455
484
  def __repr__(self):
456
- return f'FileInfo({self.file_path})'
485
+ if not self.deleted:
486
+ return f'FileInfo({self.file_path})'
487
+ else:
488
+ return f'FileInfo({self.file_path}) (deleted)'
457
489
 
458
490
  def move(self, new_directory_path: Union[str, DirectoryInfo]) -> FileInfo:
459
491
 
@@ -533,7 +565,11 @@ class FileInfo(object, metaclass=MetaMock):
533
565
  del self._cls_instances[self.resource_entry.Key]
534
566
  self.data_model.delete_resource(self.resource_entry)
535
567
 
536
- os.remove(self.file_path)
568
+ try:
569
+ os.remove(self.file_path)
570
+ except FileNotFoundError:
571
+ pass
572
+ self.deleted = True
537
573
 
538
574
  def to_json(self) -> dict:
539
575
 
@@ -613,6 +649,8 @@ class DirectoryInfo(object, metaclass=DirectoryInfoMetaMock):
613
649
  *args,
614
650
  **kwargs):
615
651
 
652
+ self.deleted = False
653
+
616
654
  self._resource_entry: Optional[ResourceDirectoryEntry] = None
617
655
  self._helper_file: Optional[FileInfo] = None
618
656
  self.data_model: Optional[DataModel] = kwargs.get('data_model', None)
@@ -635,7 +673,7 @@ class DirectoryInfo(object, metaclass=DirectoryInfoMetaMock):
635
673
 
636
674
  @property
637
675
  def helper_file(self) -> Optional[FileInfo]:
638
- if self._helper_file is None or not isinstance(self._helper_file, FileInfo):
676
+ if (self._helper_file is None or not isinstance(self._helper_file, FileInfo)) and not self.deleted:
639
677
  self._helper_file = self.add_file('__dir_helper_file__')
640
678
 
641
679
  return self._helper_file
@@ -646,7 +684,7 @@ class DirectoryInfo(object, metaclass=DirectoryInfoMetaMock):
646
684
 
647
685
  @property
648
686
  def resource_entry(self) -> Optional[ResourceDirectoryEntry]:
649
- if self._resource_entry is None:
687
+ if self._resource_entry is None and not self.deleted:
650
688
  if self.data_model is None:
651
689
  logger.warning(
652
690
  f'No data model provided. Using default data model: {config.get_default_data_model().id}.')
@@ -795,5 +833,63 @@ class DirectoryInfo(object, metaclass=DirectoryInfoMetaMock):
795
833
  def add_tag(self, tag: SimTaxonomyEntry) -> None:
796
834
  add_tag_to_resource(self.resource_entry, tag)
797
835
 
836
+ def delete(self):
837
+
838
+ for file in self.files:
839
+ file.delete()
840
+
841
+ # delete helper file
842
+ if self.helper_file is not None:
843
+ self.helper_file.delete()
844
+
845
+ if self.resource_entry is not None:
846
+ if self.resource_entry.Key in self._cls_instances:
847
+ del self._cls_instances[self.resource_entry.Key]
848
+ self.data_model.delete_resource(self.resource_entry)
849
+
850
+ shutil.rmtree(self.full_path)
851
+ self.deleted = True
852
+
853
+ def delete_files(self,
854
+ pattern: Optional[str] = None,
855
+ recursive: bool = False) -> None:
856
+ """
857
+ Delete files by pattern.
858
+ :param pattern: Pattern to match files. Default is None. if None, all files are deleted.
859
+ :param recursive: If True, delete files recursively.
860
+ :return:
861
+ """
862
+
863
+ if pattern:
864
+ matching = glob.glob(self.full_path + f'/{pattern}', recursive=recursive)
865
+
866
+ for file in self.files:
867
+ if pattern is None or file.file_path in matching:
868
+ logger.debug(f'Deleting file: {file.filename}')
869
+ file.delete()
870
+
871
+ if recursive:
872
+ for sub_dir in self.sub_directories:
873
+ sub_dir.delete_files(pattern=pattern, recursive=recursive)
874
+
875
+ def delete_subdirectories(self,
876
+ pattern: Optional[str] = None,
877
+ recursive: bool = False) -> None:
878
+
879
+ """
880
+ Delete subdirectories by pattern.
881
+ :param pattern: Regex pattern to match subdirectories. Default is None. If None, all subdirectories are deleted.
882
+ :param recursive: If True, delete subdirectories recursively.
883
+ :return:
884
+ """
885
+
886
+ for sub_dir in self.sub_directories:
887
+ if pattern is None or re.match(pattern, sub_dir.relative_path):
888
+ logger.debug(f'Deleting subdirectory: {sub_dir.relative_path}')
889
+ sub_dir.delete()
890
+
798
891
  def __repr__(self):
799
- return f'DirectoryInfo(key:{self.key}, hash: {hash(self)}; {self.full_path})'
892
+ if not self.deleted:
893
+ return f'DirectoryInfo(key:{self.key}, hash: {hash(self)}; {self.full_path})'
894
+ else:
895
+ return f'(Deleted) DirectoryInfo(key:{self.key}, hash: {hash(self)}; {self.full_path}) (deleted)'
@@ -168,6 +168,7 @@ class PythonMapper(object):
168
168
  '_cls_instances': WeakSet(),
169
169
  '_taxonomy_map': self.taxonomy_maps.get(taxonomy, None),
170
170
  '_base': bases,
171
+ '_original_cls': cls,
171
172
  '_object_mapper': self}
172
173
 
173
174
  new_class_dict.update(self.get_properties(taxonomy))
@@ -130,6 +130,42 @@ class SimultanObject(object, metaclass=MetaMock):
130
130
  logger.error(f'Error getting cls_instances: {e}')
131
131
  return []
132
132
 
133
+ @classproperty
134
+ def super_classes(cls):
135
+ superclasses = []
136
+ for key, mcls in cls._object_mapper.mapped_classes.items():
137
+ if mcls is cls:
138
+ continue
139
+ if set(cls._original_cls.__mro__) & set(mcls._original_cls.__mro__) - set([object]) and \
140
+ set(mcls._original_cls.__mro__) - set(cls._original_cls.__mro__) - set((SimultanObject,
141
+ object)):
142
+ superclasses.append(mcls)
143
+ return superclasses
144
+
145
+ @classproperty
146
+ def sub_classes(cls):
147
+ subclasses = []
148
+ for key, mcls in cls._object_mapper.mapped_classes.items():
149
+ if mcls is cls:
150
+ continue
151
+ if set(cls._original_cls.__mro__) & set(mcls._original_cls.__mro__) - set([object]) and \
152
+ set(cls._original_cls.__mro__) - set(mcls._original_cls.__mro__) - set((SimultanObject,
153
+ object)):
154
+ subclasses.append(mcls)
155
+ return subclasses
156
+
157
+ @classproperty
158
+ def super_class_instances(cls) -> set['SimultanObject']:
159
+ instances = set()
160
+ _ = [instances.update(x.cls_instances) for x in cls.super_classes]
161
+ return instances
162
+
163
+ @classproperty
164
+ def sub_class_instances(cls) -> set['SimultanObject']:
165
+ instances = set()
166
+ _ = [instances.update(x.cls_instances) for x in cls.sub_classes]
167
+ return instances
168
+
133
169
  @classmethod
134
170
  def create_simultan_component(cls, *args, **kwargs) -> SimComponent:
135
171
  wrapped_obj = utils.create_simultan_component_for_taxonomy(cls, *args, **kwargs)