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.
- PySimultan2/CHANGELOG.md +6 -0
- PySimultan2/__about__.py +1 -1
- PySimultan2/data_model.py +35 -1
- PySimultan2/default_types.py +9 -0
- PySimultan2/files.py +107 -11
- PySimultan2/object_mapper.py +1 -0
- PySimultan2/simultan_object.py +36 -0
- PySimultan2/typings/SIMULTAN/Data/Assets/__init__.pyi +616 -0
- PySimultan2/typings/SIMULTAN/Data/Components/__init__.pyi +2538 -0
- PySimultan2/typings/SIMULTAN/Data/FlowNetworks/__init__.pyi +499 -0
- PySimultan2/typings/SIMULTAN/Data/Geometry/__init__.pyi +2219 -0
- PySimultan2/typings/SIMULTAN/Data/MultiValues/__init__.pyi +682 -0
- PySimultan2/typings/SIMULTAN/Data/SimMath/__init__.pyi +1654 -0
- PySimultan2/typings/SIMULTAN/Data/SimNetworks/__init__.pyi +442 -0
- PySimultan2/typings/SIMULTAN/Data/SitePlanner/__init__.pyi +193 -0
- PySimultan2/typings/SIMULTAN/Data/Taxonomy/__init__.pyi +367 -0
- PySimultan2/typings/SIMULTAN/Data/Users/__init__.pyi +116 -0
- PySimultan2/typings/SIMULTAN/Data/ValueMappings/__init__.pyi +212 -0
- PySimultan2/typings/SIMULTAN/Data/__init__.pyi +232 -0
- PySimultan2/typings/SIMULTAN/DataMapping/__init__.pyi +916 -0
- PySimultan2/typings/SIMULTAN/Excel/__init__.pyi +15 -0
- PySimultan2/typings/SIMULTAN/Exceptions/__init__.pyi +268 -0
- PySimultan2/typings/SIMULTAN/Exchange/SimNetworkConnectors/__init__.pyi +32 -0
- PySimultan2/typings/SIMULTAN/Exchange/__init__.pyi +116 -0
- PySimultan2/typings/SIMULTAN/Projects/ManagedFiles/__init__.pyi +433 -0
- PySimultan2/typings/SIMULTAN/Projects/__init__.pyi +435 -0
- PySimultan2/typings/SIMULTAN/Serializer/CODXF/__init__.pyi +103 -0
- PySimultan2/typings/SIMULTAN/Serializer/CSV/__init__.pyi +122 -0
- PySimultan2/typings/SIMULTAN/Serializer/DXF/__init__.pyi +1335 -0
- PySimultan2/typings/SIMULTAN/Serializer/Geometry/__init__.pyi +48 -0
- PySimultan2/typings/SIMULTAN/Serializer/JSON/__init__.pyi +562 -0
- PySimultan2/typings/SIMULTAN/Serializer/METADXF/__init__.pyi +11 -0
- PySimultan2/typings/SIMULTAN/Serializer/PADXF/__init__.pyi +21 -0
- PySimultan2/typings/SIMULTAN/Serializer/PPATH/__init__.pyi +11 -0
- PySimultan2/typings/SIMULTAN/Serializer/Projects/__init__.pyi +112 -0
- PySimultan2/typings/SIMULTAN/Serializer/SIMLINKS/__init__.pyi +5 -0
- PySimultan2/typings/SIMULTAN/Serializer/SPDXF/__init__.pyi +13 -0
- PySimultan2/typings/SIMULTAN/Serializer/SimGeo/__init__.pyi +54 -0
- PySimultan2/typings/SIMULTAN/Serializer/TXDXF/__init__.pyi +46 -0
- PySimultan2/typings/SIMULTAN/Serializer/XMI/__init__.pyi +22 -0
- PySimultan2/typings/SIMULTAN/Serializer/__init__.pyi +32 -0
- PySimultan2/typings/SIMULTAN/Utils/BackgroundWork/__init__.pyi +43 -0
- PySimultan2/typings/SIMULTAN/Utils/Collections/__init__.pyi +216 -0
- PySimultan2/typings/SIMULTAN/Utils/ElevationProvider/__init__.pyi +66 -0
- PySimultan2/typings/SIMULTAN/Utils/Files/__init__.pyi +48 -0
- PySimultan2/typings/SIMULTAN/Utils/Randomize/__init__.pyi +11 -0
- PySimultan2/typings/SIMULTAN/Utils/Streams/__init__.pyi +59 -0
- PySimultan2/typings/SIMULTAN/Utils/UndoRedo/__init__.pyi +133 -0
- PySimultan2/typings/SIMULTAN/Utils/__init__.pyi +570 -0
- PySimultan2/typings/System/Buffers/Binary/__init__.pyi +248 -0
- PySimultan2/typings/System/Buffers/Text/__init__.pyi +91 -0
- PySimultan2/typings/System/Buffers/__init__.pyi +192 -0
- PySimultan2/typings/System/CodeDom/Compiler/__init__.pyi +137 -0
- PySimultan2/typings/System/Collections/Concurrent/__init__.pyi +47 -0
- PySimultan2/typings/System/Collections/Generic/__init__.pyi +1293 -0
- PySimultan2/typings/System/Collections/ObjectModel/__init__.pyi +166 -0
- PySimultan2/typings/System/Collections/Specialized/__init__.pyi +82 -0
- PySimultan2/typings/System/Collections/__init__.pyi +403 -0
- PySimultan2/typings/System/ComponentModel/__init__.pyi +582 -0
- PySimultan2/typings/System/Configuration/Assemblies/__init__.pyi +30 -0
- PySimultan2/typings/System/Diagnostics/CodeAnalysis/__init__.pyi +315 -0
- PySimultan2/typings/System/Diagnostics/Contracts/__init__.pyi +297 -0
- PySimultan2/typings/System/Diagnostics/SymbolStore/__init__.pyi +9 -0
- PySimultan2/typings/System/Diagnostics/Tracing/__init__.pyi +641 -0
- PySimultan2/typings/System/Diagnostics/__init__.pyi +1101 -0
- PySimultan2/typings/System/Globalization/__init__.pyi +1675 -0
- PySimultan2/typings/System/IO/Enumeration/__init__.pyi +125 -0
- PySimultan2/typings/System/IO/__init__.pyi +2747 -0
- PySimultan2/typings/System/Linq/Expressions/__init__.pyi +1815 -0
- PySimultan2/typings/System/Net/__init__.pyi +81 -0
- PySimultan2/typings/System/Numerics/__init__.pyi +2853 -0
- PySimultan2/typings/System/Reflection/Emit/__init__.pyi +1945 -0
- PySimultan2/typings/System/Reflection/Metadata/__init__.pyi +24 -0
- PySimultan2/typings/System/Reflection/__init__.pyi +2724 -0
- PySimultan2/typings/System/Resources/__init__.pyi +205 -0
- PySimultan2/typings/System/Runtime/CompilerServices/__init__.pyi +1926 -0
- PySimultan2/typings/System/Runtime/ConstrainedExecution/__init__.pyi +49 -0
- PySimultan2/typings/System/Runtime/ExceptionServices/__init__.pyi +34 -0
- PySimultan2/typings/System/Runtime/InteropServices/ComTypes/__init__.pyi +758 -0
- PySimultan2/typings/System/Runtime/InteropServices/Marshalling/__init__.pyi +461 -0
- PySimultan2/typings/System/Runtime/InteropServices/ObjectiveC/__init__.pyi +48 -0
- PySimultan2/typings/System/Runtime/InteropServices/__init__.pyi +2632 -0
- PySimultan2/typings/System/Runtime/Intrinsics/Arm/__init__.pyi +4757 -0
- PySimultan2/typings/System/Runtime/Intrinsics/Wasm/__init__.pyi +844 -0
- PySimultan2/typings/System/Runtime/Intrinsics/X86/__init__.pyi +5642 -0
- PySimultan2/typings/System/Runtime/Intrinsics/__init__.pyi +4504 -0
- PySimultan2/typings/System/Runtime/Loader/__init__.pyi +63 -0
- PySimultan2/typings/System/Runtime/Remoting/__init__.pyi +7 -0
- PySimultan2/typings/System/Runtime/Serialization/__init__.pyi +269 -0
- PySimultan2/typings/System/Runtime/Versioning/__init__.pyi +200 -0
- PySimultan2/typings/System/Runtime/__init__.pyi +141 -0
- PySimultan2/typings/System/Security/Cryptography/__init__.pyi +39 -0
- PySimultan2/typings/System/Security/Permissions/__init__.pyi +163 -0
- PySimultan2/typings/System/Security/Principal/__init__.pyi +45 -0
- PySimultan2/typings/System/Security/__init__.pyi +347 -0
- PySimultan2/typings/System/Text/Unicode/__init__.pyi +62 -0
- PySimultan2/typings/System/Text/__init__.pyi +1590 -0
- PySimultan2/typings/System/Threading/Tasks/Sources/__init__.pyi +76 -0
- PySimultan2/typings/System/Threading/Tasks/__init__.pyi +1403 -0
- PySimultan2/typings/System/Threading/__init__.pyi +1788 -0
- PySimultan2/typings/System/Xml/Schema/__init__.pyi +1255 -0
- PySimultan2/typings/System/Xml/Serialization/__init__.pyi +16 -0
- PySimultan2/typings/System/Xml/XPath/__init__.pyi +474 -0
- PySimultan2/typings/System/Xml/__init__.pyi +2410 -0
- PySimultan2/typings/System/__init__.pyi +17821 -0
- PySimultan2/utils.py +4 -2
- {pysimultan-0.6.0.4.dist-info → pysimultan-0.6.0.8.dist-info}/METADATA +1 -1
- pysimultan-0.6.0.8.dist-info/RECORD +174 -0
- pysimultan-0.6.0.4.dist-info/RECORD +0 -76
- {pysimultan-0.6.0.4.dist-info → pysimultan-0.6.0.8.dist-info}/WHEEL +0 -0
- {pysimultan-0.6.0.4.dist-info → pysimultan-0.6.0.8.dist-info}/licenses/LICENSE.txt +0 -0
PySimultan2/CHANGELOG.md
CHANGED
PySimultan2/__about__.py
CHANGED
@@ -1 +1 @@
|
|
1
|
-
version = '0.6.0.
|
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
|
-
|
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
|
PySimultan2/default_types.py
CHANGED
@@ -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
|
-
|
294
|
-
|
295
|
-
|
296
|
-
|
297
|
-
|
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
|
-
|
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
|
-
|
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
|
-
|
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)'
|
PySimultan2/object_mapper.py
CHANGED
@@ -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))
|
PySimultan2/simultan_object.py
CHANGED
@@ -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)
|