Skip to content

Commit f9b977f

Browse files
committed
Merge branch 'master' of https://github.com/cidarlab/pyparchmint
2 parents 94e8778 + 5797979 commit f9b977f

12 files changed

+179
-150
lines changed

parchmint/benchmarking.py

+3-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,9 @@ def characterize_devices(devices: List[Device]) -> npt.NDArray[Any]:
3434

3535
# Save yes if the device has a layer of type CONTROL
3636
file_info[index, 5] = (
37-
"YES" if "CONTROL" in [layer.type for layer in device.layers] else "NO"
37+
"YES"
38+
if "CONTROL" in [layer.layer_type for layer in device.layers]
39+
else "NO"
3840
)
3941

4042
# Save the max connectiveity of the connection in the device

parchmint/component.py

+38-35
Original file line numberDiff line numberDiff line change
@@ -147,9 +147,9 @@ def rotation(self) -> float:
147147
"""
148148
try:
149149
return self.params.get_param("rotation")
150-
except Exception:
151-
print("Could not find rotation for component")
152-
raise KeyError
150+
except Exception as error:
151+
print("Could not find rotation for component", error)
152+
raise Exception("Could not find rotation for component") from error
153153

154154
@rotation.setter
155155
def rotation(self, value):
@@ -258,15 +258,16 @@ def rotate_point(
258258
Returns:
259259
Tuple[float, float]: A tuple containing the rotated coordinates
260260
"""
261+
# pylint: disable=invalid-name, too-many-locals
261262
# Setup the center to be used the translation matrices
262263
center_x = self.xspan / 2
263264
center_y = self.yspan / 2
264265

265266
# Setup all the corner points
266-
old_topLeft = np.array((0, 0, 1)).transpose()
267-
old_topRight = np.array((self.xspan, 0, 1)).transpose()
268-
old_bottomLeft = np.array((0, self.yspan, 1)).transpose()
269-
old_bottomRight = np.array((self.xspan, self.yspan, 1)).transpose()
267+
old_topleft = np.array((0, 0, 1)).transpose()
268+
old_topright = np.array((self.xspan, 0, 1)).transpose()
269+
old_bottomleft = np.array((0, self.yspan, 1)).transpose()
270+
old_bottomright = np.array((self.xspan, self.yspan, 1)).transpose()
270271

271272
pos = np.array(((xpos), (ypos), (1)))
272273

@@ -278,23 +279,23 @@ def rotate_point(
278279
T2 = np.array(((1, 0, center_x), (0, 1, center_y), (0, 0, 1)))
279280

280281
# Rotate the topRight corner and the bottomLeft corner about the center
281-
rotated_topLeft = T2.dot(R.dot(T1.dot(old_bottomLeft)))
282-
rotated_topRight = T2.dot(R.dot(T1.dot(old_topLeft)))
283-
rotated_bottomRight = T2.dot(R.dot(T1.dot(old_topRight)))
284-
rotated_bottomLeft = T2.dot(R.dot(T1.dot(old_bottomRight)))
282+
rotated_topleft = T2.dot(R.dot(T1.dot(old_bottomleft)))
283+
rotated_topright = T2.dot(R.dot(T1.dot(old_topleft)))
284+
rotated_bottomright = T2.dot(R.dot(T1.dot(old_topright)))
285+
rotated_bottomleft = T2.dot(R.dot(T1.dot(old_bottomright)))
285286

286287
# Find the new position of the topleft corner by finding the min of all the corner points
287288
xmin = min(
288-
rotated_topLeft[0],
289-
rotated_topRight[0],
290-
rotated_bottomLeft[0],
291-
rotated_bottomRight[0],
289+
rotated_topleft[0],
290+
rotated_topright[0],
291+
rotated_bottomleft[0],
292+
rotated_bottomright[0],
292293
)
293294
ymin = min(
294-
rotated_topLeft[1],
295-
rotated_topRight[1],
296-
rotated_bottomLeft[1],
297-
rotated_bottomRight[1],
295+
rotated_topleft[1],
296+
rotated_topright[1],
297+
rotated_bottomleft[1],
298+
rotated_bottomright[1],
298299
)
299300

300301
T3 = np.array(((1, 0, -xmin), (0, 1, -ymin), (0, 0, 1)))
@@ -315,6 +316,8 @@ def rotate_point_around_center(
315316
Returns:
316317
Tuple[float, float]: A tuple containing the rotated coordinates
317318
"""
319+
# pylint: disable=invalid-name,too-many-locals
320+
318321
# Setup the center to be used the translation matrices
319322
center_x = self.xpos + self.xspan / 2
320323
center_y = self.ypos + self.yspan / 2
@@ -340,23 +343,23 @@ def get_rotated_component_definition(self, angle: int) -> Component:
340343
Returns:
341344
Component: [description]
342345
"""
343-
new_topLeft = self.rotate_point(0, 0, angle)
344-
new_topRight = self.rotate_point(self.xspan, 0, angle)
345-
new_bottomLeft = self.rotate_point(0, self.yspan, angle)
346-
new_bottomRight = self.rotate_point(self.xspan, self.yspan, angle)
346+
new_topleft = self.rotate_point(0, 0, angle)
347+
new_topright = self.rotate_point(self.xspan, 0, angle)
348+
new_bottomleft = self.rotate_point(0, self.yspan, angle)
349+
new_bottomright = self.rotate_point(self.xspan, self.yspan, angle)
347350

348351
# Find xmin, ymin, xmax, ymax for all the corner points
349352
xmin = min(
350-
new_topLeft[0], new_topRight[0], new_bottomLeft[0], new_bottomRight[0]
353+
new_topleft[0], new_topright[0], new_bottomleft[0], new_bottomright[0]
351354
)
352355
ymin = min(
353-
new_topLeft[1], new_topRight[1], new_bottomLeft[1], new_bottomRight[1]
356+
new_topleft[1], new_topright[1], new_bottomleft[1], new_bottomright[1]
354357
)
355358
xmax = max(
356-
new_topLeft[0], new_topRight[0], new_bottomLeft[0], new_bottomRight[0]
359+
new_topleft[0], new_topright[0], new_bottomleft[0], new_bottomright[0]
357360
)
358361
ymax = max(
359-
new_topLeft[1], new_topRight[1], new_bottomLeft[1], new_bottomRight[1]
362+
new_topleft[1], new_topright[1], new_bottomleft[1], new_bottomright[1]
360363
)
361364

362365
# Find the new xspan and yspan
@@ -408,31 +411,31 @@ def rotate_component(self) -> None:
408411
port.x = new_location[0]
409412
port.y = new_location[1]
410413

411-
new_topLeft = self.rotate_point_around_center(
414+
new_topleft = self.rotate_point_around_center(
412415
self.xpos + 0, self.ypos + 0, self.rotation
413416
)
414-
new_topRight = self.rotate_point_around_center(
417+
new_topright = self.rotate_point_around_center(
415418
self.xpos + self.xspan, self.ypos + 0, self.rotation
416419
)
417-
new_bottomLeft = self.rotate_point_around_center(
420+
new_bottomleft = self.rotate_point_around_center(
418421
self.xpos + 0, self.ypos + self.yspan, self.rotation
419422
)
420-
new_bottomRight = self.rotate_point_around_center(
423+
new_bottomright = self.rotate_point_around_center(
421424
self.xpos + self.xspan, self.ypos + self.yspan, self.rotation
422425
)
423426

424427
# Find xmin, ymin, xmax, ymax for all the corner points
425428
xmin = min(
426-
new_topLeft[0], new_topRight[0], new_bottomLeft[0], new_bottomRight[0]
429+
new_topleft[0], new_topright[0], new_bottomleft[0], new_bottomright[0]
427430
)
428431
ymin = min(
429-
new_topLeft[1], new_topRight[1], new_bottomLeft[1], new_bottomRight[1]
432+
new_topleft[1], new_topright[1], new_bottomleft[1], new_bottomright[1]
430433
)
431434
xmax = max(
432-
new_topLeft[0], new_topRight[0], new_bottomLeft[0], new_bottomRight[0]
435+
new_topleft[0], new_topright[0], new_bottomleft[0], new_bottomright[0]
433436
)
434437
ymax = max(
435-
new_topLeft[1], new_topRight[1], new_bottomLeft[1], new_bottomRight[1]
438+
new_topleft[1], new_topright[1], new_bottomleft[1], new_bottomright[1]
436439
)
437440

438441
# Find the new xspan and yspan

parchmint/device.py

+38-24
Original file line numberDiff line numberDiff line change
@@ -201,17 +201,17 @@ def remove_valve(self, valve_id) -> None:
201201

202202
self.remove_component(valve_id)
203203

204-
def compare(self, device: Device, ignore_parameter_diffs: bool = True) -> bool:
204+
def compare(self, device: Device, compare_params: bool = False) -> bool:
205205
"""compare against the input device. Return true if they are semnatcally feasible.
206206
207207
Args:
208208
device (Device): expected device
209-
ignore_parameter_diffs (bool): ignore parameter differences. Defaults to True.
209+
compare_params (bool): comparision includes parameter differences. Defaults to False.
210210
211211
Returns:
212212
bool: If semntically feasible, return true. Else false.
213213
"""
214-
matcher = SimilarityMatcher(self, device)
214+
matcher = SimilarityMatcher(self, device, compare_params=compare_params)
215215

216216
is_same = matcher.is_isomorphic()
217217
matcher.print_params_diff()
@@ -384,7 +384,7 @@ def remove_layer(self, layer_id: str) -> None:
384384

385385
# Remove all the components and connections associated with the layer
386386
for component in self.components:
387-
if set([layer.ID for layer in component.layers]) == set(layer_to_delete.ID):
387+
if {[layer.ID for layer in component.layers]} == set(layer_to_delete.ID):
388388
self.remove_component(component.ID)
389389
else:
390390
warn(
@@ -429,7 +429,8 @@ def merge_netlist(self, netlist: Device) -> None:
429429
self.add_layer(layer)
430430
layer_mapping[layer] = layer
431431
else:
432-
assert layer.ID is not None
432+
if layer.ID is None:
433+
raise Exception("Layer ID is None, cannot merge the layers")
433434
layer_mapping[layer] = self.get_layer(layer.ID)
434435

435436
for component in netlist.components:
@@ -634,14 +635,16 @@ def to_parchmint_v1_2(self) -> Dict:
634635
ret["version"] = "1.2"
635636

636637
# Add the valvemap information
637-
valve_map = {}
638-
valve_type_map = {}
638+
valve_objects = []
639639
for valve, connection in self._valve_map.items():
640-
valve_map[valve.ID] = connection.ID
641-
ret["valveMap"] = valve_map
642-
for valve, valve_type in self._valve_type_map.items():
643-
valve_type_map[valve.ID] = str(valve_type)
644-
ret["valveTypeMap"] = valve_type_map
640+
valve_object = {
641+
"componentid": valve.ID,
642+
"connectionid": connection.ID,
643+
"type": str(self._valve_type_map[valve]),
644+
}
645+
valve_objects.append(valve_object)
646+
ret["valves"] = valve_objects
647+
645648
return ret
646649

647650
@staticmethod
@@ -727,7 +730,7 @@ def from_parchmint_v1(json_data: Dict) -> Device:
727730
# First always add the layers
728731
if "layers" in json_data.keys():
729732
for layer in json_data["layers"]:
730-
device_ref.add_layer(Layer(layer))
733+
device_ref.add_layer(Layer(json_data=layer))
731734
else:
732735
print("no layers found")
733736

@@ -799,10 +802,16 @@ def from_parchmint_v1_2(json_data: Dict) -> Device:
799802
# First always add the layers
800803
if "layers" in json_data.keys():
801804
for layer in json_data["layers"]:
802-
device_ref.add_layer(Layer(layer))
805+
device_ref.add_layer(Layer(json_data=layer))
803806
else:
804807
print("no layers found")
805808

809+
# Second add all the features
810+
if "features" in json_data.keys():
811+
for feature_json in json_data["features"]:
812+
feature = Feature.from_parchmint_v1_2(feature_json, device_ref)
813+
device_ref.add_feature(feature)
814+
806815
# Loop through the components
807816
if "components" in json_data.keys():
808817
for component_json in json_data["components"]:
@@ -829,23 +838,28 @@ def from_parchmint_v1_2(json_data: Dict) -> Device:
829838
else:
830839
print("no params found")
831840

832-
def get_valve_type(value: str):
833-
if value is ValveType.NORMALLY_OPEN:
841+
def get_valve_type(value: str) -> ValveType:
842+
if value == ValveType.NORMALLY_OPEN:
834843
return ValveType.NORMALLY_OPEN
835-
elif value is ValveType.NORMALLY_CLOSED:
844+
elif value == ValveType.NORMALLY_CLOSED:
836845
return ValveType.NORMALLY_CLOSED
837846
else:
838847
raise Exception("Unknown valve type {}".format(value))
839848

840-
if "valveMap" in json_data.keys():
841-
valve_map = json_data["valveMap"]
842-
valve_type_map = json_data["valveTypeMap"]
849+
if "valves" in json_data.keys():
850+
valve_objects = json_data["valves"]
843851

844-
for key, value in valve_map.items():
852+
for valve_object in valve_objects:
853+
componentid = valve_object["componentid"]
854+
connectionid = valve_object["connectionid"]
855+
# Note - Sets it to default normally open if nothign is specified
856+
valve_type = valve_object["type"] if "type" in valve_object else None
845857
device_ref.map_valve(
846-
device_ref.get_component(key),
847-
device_ref.get_connection(value),
848-
get_valve_type(valve_type_map[key]),
858+
device_ref.get_component(componentid),
859+
device_ref.get_connection(connectionid),
860+
get_valve_type(valve_type)
861+
if valve_type is not None
862+
else ValveType.NORMALLY_OPEN,
849863
)
850864

851865
return device_ref

parchmint/layer.py

+35-9
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
from typing import Optional
1+
from typing import Dict, Optional
22

33
from parchmint.params import Params
44

@@ -9,17 +9,25 @@ class Layer:
99
Used to define a layer object that can be used in the device model.
1010
"""
1111

12-
def __init__(self, json_data=None) -> None:
12+
def __init__(
13+
self,
14+
layer_id: Optional[str] = None,
15+
name: Optional[str] = None,
16+
layer_type: Optional[str] = None,
17+
group: Optional[str] = None,
18+
params: Optional[Params] = None,
19+
json_data: Optional[Dict] = None,
20+
) -> None:
1321
"""Creates a new instance Layer
1422
1523
Args:
1624
json (dict, optional): json dict after json.loads(). Defaults to None.
1725
"""
18-
self._id: str = ""
19-
self.name: str = ""
20-
self.type: str = ""
21-
self.group: str = ""
22-
self.params: Params = Params()
26+
self._id: str = "" if layer_id is None else layer_id
27+
self.name: str = "" if name is None else name
28+
self.layertype: str = "" if layer_type is None else layer_type
29+
self.group: str = "" if group is None else group
30+
self.params: Params = Params() if params is None else params
2331

2432
if json_data:
2533
self.parse_from_json(json_data)
@@ -45,6 +53,24 @@ def ID(self, value: str) -> None:
4553
"""
4654
self._id = value
4755

56+
@property
57+
def layer_type(self) -> str:
58+
"""Returns the layer type
59+
60+
Returns:
61+
str: layer type
62+
"""
63+
return self.layertype
64+
65+
@layer_type.setter
66+
def layer_type(self, value: str) -> None:
67+
"""Sets the layer type
68+
69+
Args:
70+
value (str): layer type
71+
"""
72+
self.layertype = value
73+
4874
def parse_from_json(self, json_data):
4975
"""Loads instance data json dict from json.loads()
5076
@@ -53,7 +79,7 @@ def parse_from_json(self, json_data):
5379
"""
5480
self.name = json_data["name"]
5581
self.ID = json_data["id"]
56-
self.type = json_data["type"]
82+
self.layertype = json_data["type"]
5783
self.group = json_data["group"]
5884
self.params = Params(json_data["params"])
5985

@@ -66,7 +92,7 @@ def to_parchmint_v1(self):
6692
return {
6793
"name": self.name,
6894
"id": self.ID,
69-
"type": self.type,
95+
"type": self.layertype,
7096
"params": self.params.to_parchmint_v1(),
7197
"group": self.group,
7298
}

parchmint/port.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ def __init__(
1010
layer: Optional[str] = None,
1111
x: float = -1,
1212
y: float = -1,
13-
json_data: Dict = {},
13+
json_data: Optional[Dict] = None,
1414
):
1515
"""Creates a ComponentPort which is used to represent the points
1616
where a connection connects on the component

parchmint/target.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ def __init__(
88
self,
99
component_id: Optional[str] = None,
1010
port: Optional[str] = None,
11-
json_data: Dict = {},
11+
json_data: Optional[Dict] = None,
1212
):
1313
"""Creates a Target object that describes where the connection will connect to
1414

0 commit comments

Comments
 (0)