PySimultan 0.4.22__py3-none-any.whl → 0.5.2.5__py3-none-any.whl
Sign up to get free protection for your applications and to get access to all the features.
- PySimultan2/CHANGELOG.md +4 -0
- PySimultan2/__about__.py +1 -1
- PySimultan2/__init__.py +1 -0
- PySimultan2/data_model.py +185 -80
- PySimultan2/default_types.py +125 -23
- PySimultan2/files.py +55 -5
- PySimultan2/object_mapper.py +146 -9
- PySimultan2/resources/ComponentBuilder.dll +0 -0
- PySimultan2/resources/ComponentBuilder.runtimeconfig.json +2 -1
- PySimultan2/resources/ComponentBuilder.xml +47 -1
- PySimultan2/resources/SIMULTAN.AutoUpdate.Client.dll +0 -0
- PySimultan2/resources/SIMULTAN.AutoUpdate.DataTransferLibrary.dll +0 -0
- PySimultan2/resources/SIMULTAN.Lang.dll +0 -0
- PySimultan2/resources/SIMULTAN.Lang.xml +94 -7
- PySimultan2/resources/SIMULTAN.Plugins.dll +0 -0
- PySimultan2/resources/SIMULTAN.UI.dll +0 -0
- PySimultan2/resources/SIMULTAN.UI.xml +136 -43
- PySimultan2/resources/SIMULTAN.dll +0 -0
- PySimultan2/resources/SIMULTAN.xml +364 -97
- PySimultan2/resources/System.Collections.Immutable.dll +0 -0
- PySimultan2/resources/System.Reflection.Metadata.dll +0 -0
- PySimultan2/resources/System.Reflection.MetadataLoadContext.dll +0 -0
- PySimultan2/resources/componentmanager.user +0 -0
- PySimultan2/simultan_object.py +64 -15
- PySimultan2/taxonomy_maps.py +20 -12
- PySimultan2/type_setter_lookup.py +46 -33
- PySimultan2/utils.py +79 -14
- {pysimultan-0.4.22.dist-info → pysimultan-0.5.2.5.dist-info}/METADATA +2 -3
- {pysimultan-0.4.22.dist-info → pysimultan-0.5.2.5.dist-info}/RECORD +31 -32
- {pysimultan-0.4.22.dist-info → pysimultan-0.5.2.5.dist-info}/WHEEL +1 -1
- PySimultan2/resources/GeometryViewer.dll +0 -0
- PySimultan2/resources/GeometryViewer.dll.config +0 -42
- PySimultan2/resources/GeometryViewer.xml +0 -6425
- PySimultan2/resources/SitePlanner.dll +0 -0
- PySimultan2/resources/SitePlanner.dll.config +0 -11
- PySimultan2/resources/SitePlanner.xml +0 -2736
- {pysimultan-0.4.22.dist-info → pysimultan-0.5.2.5.dist-info}/licenses/LICENSE.txt +0 -0
PySimultan2/CHANGELOG.md
ADDED
PySimultan2/__about__.py
CHANGED
@@ -1 +1 @@
|
|
1
|
-
version = '0.
|
1
|
+
version = '0.5.2.5'
|
PySimultan2/__init__.py
CHANGED
PySimultan2/data_model.py
CHANGED
@@ -2,6 +2,8 @@ import atexit
|
|
2
2
|
import os
|
3
3
|
import shutil
|
4
4
|
from uuid import uuid4
|
5
|
+
from functools import lru_cache
|
6
|
+
from time import sleep
|
5
7
|
from colorlog import getLogger
|
6
8
|
from . import config
|
7
9
|
from .utils import *
|
@@ -11,7 +13,7 @@ from typing import Union, Tuple, TYPE_CHECKING
|
|
11
13
|
|
12
14
|
# from SIMULTAN import Projects
|
13
15
|
# noinspection PyUnresolvedReferences
|
14
|
-
from SIMULTAN.Projects import ExtendedProjectData
|
16
|
+
from SIMULTAN.Projects import ExtendedProjectData, CompactProject
|
15
17
|
# noinspection PyUnresolvedReferences
|
16
18
|
from SIMULTAN import Utils
|
17
19
|
# noinspection PyUnresolvedReferences
|
@@ -19,7 +21,7 @@ from SIMULTAN.Data import Users as SimultanUsers
|
|
19
21
|
# from SIMULTAN.Serializer import Projects
|
20
22
|
from SIMULTAN.Serializer.SimGeo import *
|
21
23
|
from SIMULTAN.Serializer.Projects import *
|
22
|
-
from SIMULTAN.Data.Components import SimComponent
|
24
|
+
from SIMULTAN.Data.Components import SimComponent, SimComponentCollection
|
23
25
|
from SIMULTAN.Data.MultiValues import SimMultiValueBigTable, SimMultiValueField3D
|
24
26
|
from SIMULTAN.Data.Assets import ResourceEntry
|
25
27
|
from SIMULTAN.Data.Geometry import OffsetAlgorithm
|
@@ -33,13 +35,14 @@ from SIMULTAN.Data.Geometry import GeometryModel as NetGeometryModel
|
|
33
35
|
from SIMULTAN.Data.Geometry import Layer, Vertex, Edge, PEdge, Face, Volume, EdgeLoop
|
34
36
|
|
35
37
|
from System.Security import SecureString
|
38
|
+
from SIMULTAN.Data import SimId
|
36
39
|
from System import Guid
|
37
40
|
from System.IO import *
|
38
41
|
from System.Security import *
|
39
42
|
from System.Security.Cryptography import *
|
40
43
|
from System.Text import *
|
41
44
|
|
42
|
-
from .files import add_tag_to_resource
|
45
|
+
from .files import add_tag_to_resource, FileInfo as PythonFileInfo
|
43
46
|
|
44
47
|
|
45
48
|
if TYPE_CHECKING:
|
@@ -72,19 +75,6 @@ class IAuthenticationServiceNew(SimultanUsers.IAuthenticationService):
|
|
72
75
|
return user.Item1
|
73
76
|
|
74
77
|
|
75
|
-
# data_models = WeakSet()
|
76
|
-
# data_models_dict = WeakValueDictionary()
|
77
|
-
|
78
|
-
|
79
|
-
# def get_default_data_model():
|
80
|
-
# return data_models[0]
|
81
|
-
#
|
82
|
-
#
|
83
|
-
# def add_data_model(data_model):
|
84
|
-
# data_models.add(data_model)
|
85
|
-
|
86
|
-
|
87
|
-
# noinspection PyUnresolvedReferences
|
88
78
|
class DataModel:
|
89
79
|
|
90
80
|
@classmethod
|
@@ -143,17 +133,17 @@ class DataModel:
|
|
143
133
|
atexit.register(self.cleanup)
|
144
134
|
|
145
135
|
self.id = uuid4()
|
146
|
-
self.data = None
|
147
|
-
self._project_data_manager = None
|
136
|
+
self.data: Optional[SimComponentCollection] = None
|
137
|
+
self._project_data_manager: Optional[ExtendedProjectData] = None
|
148
138
|
self._user = None
|
149
|
-
self._project = None
|
139
|
+
self._project: Optional[CompactProject] = None
|
150
140
|
self._zip_loader = None
|
151
141
|
|
152
|
-
self.project_data_manager = kwargs.get('project_data_manager', None)
|
142
|
+
self.project_data_manager: Optional[ExtendedProjectData] = kwargs.get('project_data_manager', None)
|
153
143
|
|
154
|
-
self.project_path = kwargs.get('project_path', None)
|
144
|
+
self.project_path: Optional[str] = kwargs.get('project_path', None)
|
155
145
|
|
156
|
-
self.service_provider = Utils.ServicesProvider()
|
146
|
+
self.service_provider: Utils.ServicesProvider = Utils.ServicesProvider()
|
157
147
|
|
158
148
|
self.i_aut_service = IAuthenticationServiceNew
|
159
149
|
self.i_aut_service.user_name = self.user_name
|
@@ -168,6 +158,8 @@ class DataModel:
|
|
168
158
|
|
169
159
|
self.__mongo_instance = None
|
170
160
|
|
161
|
+
self.component_dict: dict[SimId, SimComponent] = {}
|
162
|
+
|
171
163
|
@property
|
172
164
|
def assets(self):
|
173
165
|
return self.project_data_manager.AssetManager.Resources
|
@@ -282,6 +274,10 @@ class DataModel:
|
|
282
274
|
|
283
275
|
def import_data_model(self):
|
284
276
|
self.data = self.project_data_manager.Components
|
277
|
+
|
278
|
+
self.create_component_dict.cache_clear()
|
279
|
+
self.get_component_by_id.cache_clear()
|
280
|
+
|
285
281
|
return self.data
|
286
282
|
|
287
283
|
def add_field(self, field: SimMultiValueField3D):
|
@@ -313,6 +309,7 @@ class DataModel:
|
|
313
309
|
return
|
314
310
|
|
315
311
|
self.data.Add(component)
|
312
|
+
self.create_component_dict.cache_clear()
|
316
313
|
# logger.info(
|
317
314
|
# f'Added component {component.Id} {component.Name} {type(component)} to project {self.project_path}')
|
318
315
|
|
@@ -351,6 +348,32 @@ class DataModel:
|
|
351
348
|
elif index is not None:
|
352
349
|
self.data.RemoveItem(index)
|
353
350
|
|
351
|
+
self.create_component_dict.cache_clear()
|
352
|
+
self.get_component_by_id.cache_clear()
|
353
|
+
|
354
|
+
def remove_component(self,
|
355
|
+
component: Union[SimComponent, SimultanObject] = None,
|
356
|
+
index: int = None):
|
357
|
+
"""
|
358
|
+
Remove a component from the project
|
359
|
+
:param component:
|
360
|
+
:param index:
|
361
|
+
:return:
|
362
|
+
"""
|
363
|
+
if hasattr(component, '_wrapped_obj'):
|
364
|
+
component = component._wrapped_obj
|
365
|
+
|
366
|
+
if component.Parent is not None:
|
367
|
+
scce = next((x for x in component.Parent.Components if component.Id.Equals(x.Component.Id)), None)
|
368
|
+
if scce is not None:
|
369
|
+
component.Parent.Components.Remove(scce)
|
370
|
+
else:
|
371
|
+
index = self.data.Items.IndexOf(component)
|
372
|
+
self.data.RemoveItem(index)
|
373
|
+
|
374
|
+
self.create_component_dict.cache_clear()
|
375
|
+
self.get_component_by_id.cache_clear()
|
376
|
+
|
354
377
|
def save(self):
|
355
378
|
"""
|
356
379
|
Save the project
|
@@ -423,6 +446,7 @@ class DataModel:
|
|
423
446
|
slot = SimSlot(slot_name, str(slot_extension))
|
424
447
|
ComponentManagement.AddReferencedComponentSlot(comp, slot, self.user)
|
425
448
|
ComponentManagement.AddReferencedComponent(comp, slot, ref_comp, self.user)
|
449
|
+
self.create_component_dict.cache_clear()
|
426
450
|
|
427
451
|
def remove_referenced_component(self, comp: SimComponent, index: int):
|
428
452
|
if index is not None:
|
@@ -431,6 +455,9 @@ class DataModel:
|
|
431
455
|
index = self.data.Items.IndexOf(comp)
|
432
456
|
self.data.RemoveItem(index)
|
433
457
|
|
458
|
+
self.create_component_dict.cache_clear()
|
459
|
+
self.get_component_by_id.cache_clear()
|
460
|
+
|
434
461
|
def add_new_geometry_model(self, file_name: str, model_name: str = None, return_resource=False):
|
435
462
|
"""
|
436
463
|
Create and add a new fc_geometry model
|
@@ -439,6 +466,7 @@ class DataModel:
|
|
439
466
|
:param return_resource: return the resource
|
440
467
|
:return: GeometryViewer.Model.GeometryModel, geo_resource
|
441
468
|
"""
|
469
|
+
self.get_file_infos.cache_clear()
|
442
470
|
geo_resource = self.add_geometry_resource(file_name)
|
443
471
|
file_info = FileInfo(geo_resource.CurrentFullPath)
|
444
472
|
try:
|
@@ -469,6 +497,7 @@ class DataModel:
|
|
469
497
|
model_name,
|
470
498
|
self.service_provider)
|
471
499
|
|
500
|
+
self.get_file_infos.cache_clear()
|
472
501
|
return new_resource
|
473
502
|
|
474
503
|
def add_empty_resource(self, filename: str):
|
@@ -479,6 +508,7 @@ class DataModel:
|
|
479
508
|
"""
|
480
509
|
# return self.project.AddResourceFile(FileInfo(str(filename)))
|
481
510
|
|
511
|
+
self.get_file_infos.cache_clear()
|
482
512
|
return self.project.AddEmptyResource(FileInfo(str(filename)))
|
483
513
|
|
484
514
|
def add_resource(self,
|
@@ -490,43 +520,50 @@ class DataModel:
|
|
490
520
|
:param tag: tag to add to the resource
|
491
521
|
:return:
|
492
522
|
"""
|
493
|
-
|
494
|
-
del_copy = False
|
495
|
-
|
496
|
-
existing_files = [x.Name for x in self.project_data_manager.AssetManager.Resources]
|
497
523
|
try:
|
498
|
-
|
499
|
-
|
500
|
-
|
501
|
-
|
502
|
-
|
503
|
-
|
504
|
-
|
505
|
-
|
506
|
-
|
507
|
-
|
508
|
-
|
509
|
-
counter
|
510
|
-
|
511
|
-
|
512
|
-
|
513
|
-
|
514
|
-
|
515
|
-
|
516
|
-
|
517
|
-
|
518
|
-
|
519
|
-
|
520
|
-
|
521
|
-
|
522
|
-
|
523
|
-
|
524
|
-
|
525
|
-
|
526
|
-
|
527
|
-
|
528
|
-
|
529
|
-
|
524
|
+
|
525
|
+
del_copy = False
|
526
|
+
|
527
|
+
existing_files = [x.Name for x in self.project_data_manager.AssetManager.Resources]
|
528
|
+
try:
|
529
|
+
act_filename = filename.replace('\\', os.sep)
|
530
|
+
except TypeError:
|
531
|
+
act_filename = filename
|
532
|
+
|
533
|
+
if os.path.basename(act_filename) in existing_files:
|
534
|
+
# create copy with running counter in temp dir and use this file:
|
535
|
+
counter = 1
|
536
|
+
while True:
|
537
|
+
new_filename = os.path.basename(filename) + f'({str(counter)})'
|
538
|
+
if new_filename not in existing_files and not os.path.exists(new_filename):
|
539
|
+
break
|
540
|
+
counter += 1
|
541
|
+
shutil.copy(filename, os.path.join(os.path.dirname(filename), new_filename))
|
542
|
+
filename = os.path.join(os.path.dirname(filename), new_filename)
|
543
|
+
del_copy = True
|
544
|
+
|
545
|
+
if isinstance(filename, (str, PosixPath, WindowsPath)):
|
546
|
+
filename = FileInfo(str(filename))
|
547
|
+
|
548
|
+
resource = self.project.CopyResourceAsContainedFileEntry(filename,
|
549
|
+
self.project.ProjectUnpackFolder,
|
550
|
+
'1')
|
551
|
+
|
552
|
+
if del_copy:
|
553
|
+
os.remove(str(filename))
|
554
|
+
|
555
|
+
# file_id = self.project_data_manager.AssetManager.AddResourceEntry(FileInfo(filename))
|
556
|
+
# return self.project_data_manager.AssetManager.Resources[file_id]
|
557
|
+
if tag is not None:
|
558
|
+
add_tag_to_resource(resource, tag)
|
559
|
+
|
560
|
+
# sleep(0.2) # this is necessary to avoid race conditions System.ArgumentException: An item with the same
|
561
|
+
# key has already been added. Key: at System.Collections.Generic.Dictionary`2.TryInsert(TKey key,
|
562
|
+
# TValue value, InsertionBehavior behavior)
|
563
|
+
return resource
|
564
|
+
except Exception as e:
|
565
|
+
logger.error(f'Error while adding resource {filename} to project {self.project_path}: {e}')
|
566
|
+
raise e
|
530
567
|
|
531
568
|
def delete_resource(self, resource: Union[ResourceEntry, FileInfo, ContainedResourceFileEntry]):
|
532
569
|
"""
|
@@ -543,6 +580,7 @@ class DataModel:
|
|
543
580
|
logger.info(f'Deleted resource {resource.Name} from project {self.project_path}')
|
544
581
|
else:
|
545
582
|
logger.error(f'Could not delete resource {resource.Name} from project {self.project_path}')
|
583
|
+
self.get_file_infos.cache_clear()
|
546
584
|
return success
|
547
585
|
|
548
586
|
def add_table(self, table: SimMultiValueBigTable):
|
@@ -584,6 +622,9 @@ class DataModel:
|
|
584
622
|
# else:
|
585
623
|
# return get_component_geometry(self, geometry_model, component)
|
586
624
|
|
625
|
+
def get_taxonomy_by_key(self, key: str):
|
626
|
+
return next((x for x in self.taxonomies if x.Key == key), None)
|
627
|
+
|
587
628
|
def create_taxonomy(self, name: str, key: str, description: str = ''):
|
588
629
|
"""
|
589
630
|
Create a new taxonomy and add it to the project
|
@@ -594,21 +635,16 @@ class DataModel:
|
|
594
635
|
"""
|
595
636
|
return create_taxonomy(name, key, description, data_model=self)
|
596
637
|
|
597
|
-
def get_taxonomy_entry(self,
|
598
|
-
|
599
|
-
|
600
|
-
return entry
|
638
|
+
def get_taxonomy_entry(self,
|
639
|
+
key,
|
640
|
+
taxonomy: Union[SimTaxonomy, str] = None) -> SimTaxonomyEntry:
|
601
641
|
|
602
|
-
|
603
|
-
|
604
|
-
from .redis_utils.base_classes import save_in_redis
|
642
|
+
taxonomy = self.get_taxonomy_by_key(taxonomy) if isinstance(taxonomy, str) else taxonomy
|
643
|
+
taxonomies = [taxonomy] if taxonomy is not None else self.taxonomies
|
605
644
|
|
606
|
-
|
607
|
-
|
608
|
-
|
609
|
-
logger.info(f'Created new mongo instance for {self.project_data_manager.Project.Name}')
|
610
|
-
else:
|
611
|
-
logger.info(f'Found existing mongo instance for {self.project_data_manager.Project.Name}')
|
645
|
+
for entry in taxonomies:
|
646
|
+
if entry.Key == key:
|
647
|
+
return entry
|
612
648
|
|
613
649
|
def save_in_mongodb(self, db):
|
614
650
|
"""
|
@@ -618,14 +654,8 @@ class DataModel:
|
|
618
654
|
"""
|
619
655
|
self._mongo_instance.save(db)
|
620
656
|
|
621
|
-
def
|
622
|
-
|
623
|
-
save the object in the redis database
|
624
|
-
:return: None
|
625
|
-
"""
|
626
|
-
save_in_redis(self)
|
627
|
-
|
628
|
-
def get_taxonomy_entries(self):
|
657
|
+
def get_taxonomy_entries(self,
|
658
|
+
taxonomy: Union[SimTaxonomy, str] = None) -> dict[str, SimTaxonomyEntry]:
|
629
659
|
|
630
660
|
def add_sub_entries(tax_entry, tax_entries_dict):
|
631
661
|
for sub_entry in tax_entry.Children:
|
@@ -634,7 +664,9 @@ class DataModel:
|
|
634
664
|
return tax_entries_dict
|
635
665
|
|
636
666
|
taxonomy_entries = {}
|
637
|
-
|
667
|
+
taxonomies = [taxonomy] if isinstance(taxonomy, SimTaxonomy) else self.taxonomies
|
668
|
+
|
669
|
+
for taxonomy in taxonomies:
|
638
670
|
taxonomy_entries[taxonomy.Key] = taxonomy
|
639
671
|
for entry in list(taxonomy.Entries):
|
640
672
|
taxonomy_entries[entry.Key] = entry
|
@@ -677,6 +709,79 @@ class DataModel:
|
|
677
709
|
mapper.current_data_model = self
|
678
710
|
return mapper.get_typed_data(self, component_list=self.data.Items, create_all=False)
|
679
711
|
|
712
|
+
@lru_cache()
|
713
|
+
def create_component_dict(self):
|
714
|
+
new_component_list = set()
|
715
|
+
|
716
|
+
def get_subcomponents(sim_component: SimComponent):
|
717
|
+
new_subcomponents = set()
|
718
|
+
if isinstance(sim_component, SimultanObject):
|
719
|
+
sim_component = sim_component._wrapped_obj
|
720
|
+
|
721
|
+
if sim_component in new_component_list:
|
722
|
+
return
|
723
|
+
else:
|
724
|
+
new_component_list.add(sim_component)
|
725
|
+
|
726
|
+
if sim_component is None:
|
727
|
+
return []
|
728
|
+
|
729
|
+
for sub_component in sim_component.Components.Items:
|
730
|
+
if sub_component is None:
|
731
|
+
continue
|
732
|
+
new_subcomponents.add(sub_component.Component)
|
733
|
+
for ref_component in sim_component.ReferencedComponents.Items:
|
734
|
+
if ref_component is None:
|
735
|
+
continue
|
736
|
+
new_subcomponents.add(ref_component.Target)
|
737
|
+
|
738
|
+
for new_subcomponent in new_subcomponents:
|
739
|
+
get_subcomponents(new_subcomponent)
|
740
|
+
|
741
|
+
new_component_list.update(new_subcomponents)
|
742
|
+
|
743
|
+
for component in self.data.Items:
|
744
|
+
if component is None:
|
745
|
+
continue
|
746
|
+
get_subcomponents(component)
|
747
|
+
component_list = list(new_component_list)
|
748
|
+
|
749
|
+
self.component_dict = {x.Id: x for x in component_list}
|
750
|
+
return self.component_dict
|
751
|
+
|
752
|
+
@lru_cache()
|
753
|
+
def get_component_by_id(self,
|
754
|
+
item_id: SimId,
|
755
|
+
search_subcomponents=False) -> Union[SimComponent, None]:
|
756
|
+
|
757
|
+
# print(item_id.GlobalId, item_id.LocalId)
|
758
|
+
# _ = [print((x.Id.GlobalId, x.Id.LocalId)) for x in self.data.Items]
|
759
|
+
|
760
|
+
component = next((x for x in self.data.Items if x.Id.Equals(item_id)), None)
|
761
|
+
|
762
|
+
if component is None and search_subcomponents:
|
763
|
+
component = self.component_dict.get(item_id, None)
|
764
|
+
if component is None:
|
765
|
+
self.create_component_dict()
|
766
|
+
component = self.component_dict.get(item_id, None)
|
767
|
+
|
768
|
+
return component
|
769
|
+
|
770
|
+
def get_component_by_name(self, name: str) -> list[SimComponent]:
|
771
|
+
return [x for x in self.create_component_dict.values() if x.Name == name]
|
772
|
+
|
773
|
+
@lru_cache()
|
774
|
+
def get_file_infos(self) -> list[PythonFileInfo]:
|
775
|
+
return [PythonFileInfo(resource_entry=asset) for asset in self.assets]
|
776
|
+
|
777
|
+
def get_file_info_by_key(self,
|
778
|
+
key: int) -> Optional[PythonFileInfo]:
|
779
|
+
|
780
|
+
if isinstance(key, str):
|
781
|
+
key = int(key)
|
782
|
+
|
783
|
+
return next((PythonFileInfo(resource_entry=asset) for asset in self.assets if asset.Key == key), None)
|
784
|
+
|
680
785
|
def __del__(self):
|
681
786
|
self.cleanup()
|
682
787
|
|