How to create a part#
This tutorial demonstrates how to create a part.
What is a part?#
A part is either a volume or a face type bodies that are defined by a number of mesh triangles.
Then a material optical property can be then applied to a part (like bodies, faces).
[1]:
from pathlib import Path
from ansys.speos.core import Body, Face, Part, Project, Speos
# If using docker container
assets_data_path = Path("/app") / "assets"
# If using local server
# assets_data_path = Path().resolve().parent.parent / "tests" / "assets"
# If using a different path
# assets_data_path = Path("path/to/downloaded/example/assets")
Create connection with speos rpc server#
[2]:
speos = Speos(host="localhost", port=50098)
New Project#
The only way to create an optical property, is to create it from a project.
[3]:
p = Project(speos=speos)
print(p)
{
"name": "",
"description": "",
"metadata": {},
"part_guid": "",
"sources": [],
"sensors": [],
"simulations": [],
"materials": [],
"scenes": []
}
Create#
Before creating a body, a Root part needs to be created and committed.
[4]:
root_part = p.create_root_part().commit()
print(root_part)
{
"name": "RootPart",
"description": "",
"metadata": {},
"body_guids": [],
"parts": []
}
Create bodies in root part.#
A body can either a volume or face type. Both use the method named “create_body”.
[5]:
body_b1 = root_part.create_body(name="TheBodyB1").commit()
body_b2 = root_part.create_body(name="TheBodyB2").commit()
print(root_part)
{
"name": "RootPart",
"body_guids": [
"99e0a1f1-99dd-4304-81fb-f24c9240a5a9",
"24a16163-beed-43db-b129-93d94b6d1c28"
],
"description": "",
"metadata": {},
"parts": [],
"bodys": [
{
"name": "TheBodyB1",
"description": "",
"metadata": {},
"face_guids": []
},
{
"name": "TheBodyB2",
"description": "",
"metadata": {},
"face_guids": []
}
]
}
Create faces inside a body.#
A body can have one (example, surface/open-volume type of body) or multiple faces (close-volume type of body).
Each face is then defined by a number of triangles/facets.
Each triangle/facet is defined by vertices and vertice normals.
[6]:
face_b1_f1 = (
body_b1.create_face(name="TheFaceF1")
.set_vertices([0, 0, 0, 1, 0, 0, 0, 1, 0])
.set_facets([0, 1, 2])
.set_normals([0, 0, 1, 0, 0, 1, 0, 0, 1])
.commit()
)
print(root_part)
{
"name": "RootPart",
"body_guids": [
"99e0a1f1-99dd-4304-81fb-f24c9240a5a9",
"24a16163-beed-43db-b129-93d94b6d1c28"
],
"description": "",
"metadata": {},
"parts": [],
"bodys": [
{
"name": "TheBodyB1",
"face_guids": [
"8fe07862-b57d-4923-a154-a6651c7f7f51"
],
"description": "",
"metadata": {},
"faces": [
{
"name": "TheFaceF1",
"vertices": [
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
]
},
{
"name": "TheBodyB2",
"description": "",
"metadata": {},
"face_guids": []
}
]
}
Create bodies in sub part.#
Part can also be created under a sub-part.
The location sub-part can be defined using set_axis_system method.
[7]:
sub_part1 = (
root_part.create_sub_part(name="TheSubPartSP1")
.set_axis_system(axis_system=[5, 5, 5, 1, 0, 0, 0, 1, 0, 0, 0, 1])
.commit()
)
print(root_part)
{
"name": "RootPart",
"body_guids": [
"99e0a1f1-99dd-4304-81fb-f24c9240a5a9",
"24a16163-beed-43db-b129-93d94b6d1c28"
],
"parts": [
{
"name": "TheSubPartSP1",
"description": "UniqueId_9b448726-8923-48f9-8249-4dbc2e93acdb",
"part_guid": "3b197b63-433a-41ce-a85a-c5d7d02f6fbc",
"axis_system": [
5.0,
5.0,
5.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0
],
"part": {
"name": "TheSubPartSP1",
"description": "",
"metadata": {},
"body_guids": [],
"parts": []
}
}
],
"description": "",
"metadata": {},
"bodys": [
{
"name": "TheBodyB1",
"face_guids": [
"8fe07862-b57d-4923-a154-a6651c7f7f51"
],
"description": "",
"metadata": {},
"faces": [
{
"name": "TheFaceF1",
"vertices": [
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
]
},
{
"name": "TheBodyB2",
"description": "",
"metadata": {},
"face_guids": []
}
]
}
Create body and faces in sub part body#
[8]:
body_sp1_b1 = sub_part1.create_body(name="TheBodySP1_B1").commit()
print(root_part)
face_sp1_b1_f1 = (
body_sp1_b1.create_face(name="TheFaceSP1_B1_F1")
.set_vertices([0, 1, 0, 0, 2, 0, 1, 2, 0])
.set_facets([0, 1, 2])
.set_normals([0, 0, 1, 0, 0, 1, 0, 0, 1])
.commit()
)
print(root_part)
# -
{
"name": "RootPart",
"body_guids": [
"99e0a1f1-99dd-4304-81fb-f24c9240a5a9",
"24a16163-beed-43db-b129-93d94b6d1c28"
],
"parts": [
{
"name": "TheSubPartSP1",
"description": "UniqueId_9b448726-8923-48f9-8249-4dbc2e93acdb",
"part_guid": "3b197b63-433a-41ce-a85a-c5d7d02f6fbc",
"axis_system": [
5.0,
5.0,
5.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0
],
"part": {
"name": "TheSubPartSP1",
"body_guids": [
"c5f30eb4-ea2e-4011-8b9d-43671a886b56"
],
"description": "",
"metadata": {},
"parts": [],
"bodys": [
{
"name": "TheBodySP1_B1",
"description": "",
"metadata": {},
"face_guids": []
}
]
}
}
],
"description": "",
"metadata": {},
"bodys": [
{
"name": "TheBodyB1",
"face_guids": [
"8fe07862-b57d-4923-a154-a6651c7f7f51"
],
"description": "",
"metadata": {},
"faces": [
{
"name": "TheFaceF1",
"vertices": [
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
]
},
{
"name": "TheBodyB2",
"description": "",
"metadata": {},
"face_guids": []
}
]
}
{
"name": "RootPart",
"body_guids": [
"99e0a1f1-99dd-4304-81fb-f24c9240a5a9",
"24a16163-beed-43db-b129-93d94b6d1c28"
],
"parts": [
{
"name": "TheSubPartSP1",
"description": "UniqueId_9b448726-8923-48f9-8249-4dbc2e93acdb",
"part_guid": "3b197b63-433a-41ce-a85a-c5d7d02f6fbc",
"axis_system": [
5.0,
5.0,
5.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0
],
"part": {
"name": "TheSubPartSP1",
"body_guids": [
"c5f30eb4-ea2e-4011-8b9d-43671a886b56"
],
"description": "",
"metadata": {},
"parts": [],
"bodys": [
{
"name": "TheBodySP1_B1",
"face_guids": [
"4d5778ba-230a-4659-a1cb-e9e0ef1515d7"
],
"description": "",
"metadata": {},
"faces": [
{
"name": "TheFaceSP1_B1_F1",
"vertices": [
0.0,
1.0,
0.0,
0.0,
2.0,
0.0,
1.0,
2.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
]
}
]
}
}
],
"description": "",
"metadata": {},
"bodys": [
{
"name": "TheBodyB1",
"face_guids": [
"8fe07862-b57d-4923-a154-a6651c7f7f51"
],
"description": "",
"metadata": {},
"faces": [
{
"name": "TheFaceF1",
"vertices": [
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
]
},
{
"name": "TheBodyB2",
"description": "",
"metadata": {},
"face_guids": []
}
]
}
Create sub parts in sub part#
[9]:
sub_part11 = (
sub_part1.create_sub_part(name="TheSubPartSP11")
.set_axis_system([1, 1, 1, 1, 0, 0, 0, 1, 0, 0, 0, 1])
.commit()
)
print(root_part)
{
"name": "RootPart",
"body_guids": [
"99e0a1f1-99dd-4304-81fb-f24c9240a5a9",
"24a16163-beed-43db-b129-93d94b6d1c28"
],
"parts": [
{
"name": "TheSubPartSP1",
"description": "UniqueId_9b448726-8923-48f9-8249-4dbc2e93acdb",
"part_guid": "3b197b63-433a-41ce-a85a-c5d7d02f6fbc",
"axis_system": [
5.0,
5.0,
5.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0
],
"part": {
"name": "TheSubPartSP1",
"body_guids": [
"c5f30eb4-ea2e-4011-8b9d-43671a886b56"
],
"parts": [
{
"name": "TheSubPartSP11",
"description": "UniqueId_3e37831b-e7e2-4c35-9319-bf5ef7f7d307",
"part_guid": "928971bf-a119-4af5-a76e-d60fa5b88cdd",
"axis_system": [
1.0,
1.0,
1.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0
],
"part": {
"name": "TheSubPartSP11",
"description": "",
"metadata": {},
"body_guids": [],
"parts": []
}
}
],
"description": "",
"metadata": {},
"bodys": [
{
"name": "TheBodySP1_B1",
"face_guids": [
"4d5778ba-230a-4659-a1cb-e9e0ef1515d7"
],
"description": "",
"metadata": {},
"faces": [
{
"name": "TheFaceSP1_B1_F1",
"vertices": [
0.0,
1.0,
0.0,
0.0,
2.0,
0.0,
1.0,
2.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
]
}
]
}
}
],
"description": "",
"metadata": {},
"bodys": [
{
"name": "TheBodyB1",
"face_guids": [
"8fe07862-b57d-4923-a154-a6651c7f7f51"
],
"description": "",
"metadata": {},
"faces": [
{
"name": "TheFaceF1",
"vertices": [
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
]
},
{
"name": "TheBodyB2",
"description": "",
"metadata": {},
"face_guids": []
}
]
}
Read#
Find with exact name#
Find the root part
[10]:
features = p.find(name="", feature_type=Part)
print(features[0])
{
"name": "RootPart",
"body_guids": [
"99e0a1f1-99dd-4304-81fb-f24c9240a5a9",
"24a16163-beed-43db-b129-93d94b6d1c28"
],
"parts": [
{
"name": "TheSubPartSP1",
"description": "UniqueId_9b448726-8923-48f9-8249-4dbc2e93acdb",
"part_guid": "3b197b63-433a-41ce-a85a-c5d7d02f6fbc",
"axis_system": [
5.0,
5.0,
5.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0
],
"part": {
"name": "TheSubPartSP1",
"body_guids": [
"c5f30eb4-ea2e-4011-8b9d-43671a886b56"
],
"parts": [
{
"name": "TheSubPartSP11",
"description": "UniqueId_3e37831b-e7e2-4c35-9319-bf5ef7f7d307",
"part_guid": "928971bf-a119-4af5-a76e-d60fa5b88cdd",
"axis_system": [
1.0,
1.0,
1.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0
],
"part": {
"name": "TheSubPartSP11",
"description": "",
"metadata": {},
"body_guids": [],
"parts": []
}
}
],
"description": "",
"metadata": {},
"bodys": [
{
"name": "TheBodySP1_B1",
"face_guids": [
"4d5778ba-230a-4659-a1cb-e9e0ef1515d7"
],
"description": "",
"metadata": {},
"faces": [
{
"name": "TheFaceSP1_B1_F1",
"vertices": [
0.0,
1.0,
0.0,
0.0,
2.0,
0.0,
1.0,
2.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
]
}
]
}
}
],
"description": "",
"metadata": {},
"bodys": [
{
"name": "TheBodyB1",
"face_guids": [
"8fe07862-b57d-4923-a154-a6651c7f7f51"
],
"description": "",
"metadata": {},
"faces": [
{
"name": "TheFaceF1",
"vertices": [
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
]
},
{
"name": "TheBodyB2",
"description": "",
"metadata": {},
"face_guids": []
}
]
}
Find a specific body in root part
[11]:
features = p.find(name="TheBodyB1", feature_type=Body)
print(features[0])
{
"name": "TheBodyB1",
"face_guids": [
"8fe07862-b57d-4923-a154-a6651c7f7f51"
],
"description": "",
"metadata": {},
"faces": [
{
"name": "TheFaceF1",
"vertices": [
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
]
}
Find a specific face of a body in root part
[12]:
features = p.find(name="TheBodyB1/TheFaceF1", feature_type=Face)
print(features[0])
{
"name": "TheFaceF1",
"vertices": [
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
Find a sub part
[13]:
features = p.find(name="TheSubPartSP1", feature_type=Part.SubPart)
print(features[0])
{
"name": "TheSubPartSP1",
"description": "UniqueId_9b448726-8923-48f9-8249-4dbc2e93acdb",
"part_guid": "3b197b63-433a-41ce-a85a-c5d7d02f6fbc",
"axis_system": [
5.0,
5.0,
5.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0
],
"part": {
"name": "TheSubPartSP1",
"body_guids": [
"c5f30eb4-ea2e-4011-8b9d-43671a886b56"
],
"parts": [
{
"name": "TheSubPartSP11",
"description": "UniqueId_3e37831b-e7e2-4c35-9319-bf5ef7f7d307",
"part_guid": "928971bf-a119-4af5-a76e-d60fa5b88cdd",
"axis_system": [
1.0,
1.0,
1.0,
1.0,
0.0,
0.0,
0.0,
1.0,
0.0,
0.0,
0.0,
1.0
],
"part": {
"name": "TheSubPartSP11",
"description": "",
"metadata": {},
"body_guids": [],
"parts": []
}
}
],
"description": "",
"metadata": {},
"bodys": [
{
"name": "TheBodySP1_B1",
"face_guids": [
"4d5778ba-230a-4659-a1cb-e9e0ef1515d7"
],
"description": "",
"metadata": {},
"faces": [
{
"name": "TheFaceSP1_B1_F1",
"vertices": [
0.0,
1.0,
0.0,
0.0,
2.0,
0.0,
1.0,
2.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
]
}
]
}
}
Find a specific body in sub part
[14]:
features = p.find(name="TheSubPartSP1/TheBodySP1_B1", feature_type=Body)
print(features[0])
{
"name": "TheBodySP1_B1",
"face_guids": [
"4d5778ba-230a-4659-a1cb-e9e0ef1515d7"
],
"description": "",
"metadata": {},
"faces": [
{
"name": "TheFaceSP1_B1_F1",
"vertices": [
0.0,
1.0,
0.0,
0.0,
2.0,
0.0,
1.0,
2.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
]
}
Find a specific face of a body in sub part
[15]:
features = p.find(name="TheSubPartSP1/TheBodySP1_B1/TheFaceSP1_B1_F1", feature_type=Face)
print(features[0])
{
"name": "TheFaceSP1_B1_F1",
"vertices": [
0.0,
1.0,
0.0,
0.0,
2.0,
0.0,
1.0,
2.0,
0.0
],
"facets": [
0,
1,
2
],
"normals": [
0.0,
0.0,
1.0,
0.0,
0.0,
1.0,
0.0,
0.0,
1.0
],
"description": "",
"metadata": {},
"texture_coordinates_channels": []
}
Find with approximation name#
Find all bodies in root part
[16]:
features = p.find(name=".*", name_regex=True, feature_type=Body)
for feat in features:
print(feat._name)
TheBodyB1
TheBodyB2
Find all faces inside body called “TheBodyB1”
[17]:
features = p.find(name="TheBodyB1/.*", name_regex=True, feature_type=Face)
for feat in features:
print(feat._name)
TheFaceF1
If you want to retrieve several kind of geometry features at a certain level, give feature_type=Part
all the geometry features at root part level:
[18]:
features = p.find(name=".*", name_regex=True, feature_type=Part)
for feat in features:
print(str(type(feat)) + " : name=" + feat._name)
<class 'ansys.speos.core.body.Body'> : name=TheBodyB1
<class 'ansys.speos.core.body.Body'> : name=TheBodyB2
<class 'ansys.speos.core.part.Part.SubPart'> : name=TheSubPartSP1
all the geometry features at second level: e.g.:
TheBodyB1’s all faces
TheSubPartSP1’s all bodies
TheSubPartSP1’s all sub part
[19]:
features = p.find(name=".*/.*", name_regex=True, feature_type=Part)
for feat in features:
print(str(type(feat)) + " : name=" + feat._name)
<class 'ansys.speos.core.face.Face'> : name=TheFaceF1
<class 'ansys.speos.core.body.Body'> : name=TheBodySP1_B1
<class 'ansys.speos.core.part.Part.SubPart'> : name=TheSubPartSP11
[20]:
# all the geometry features at the third level:
# e.g. TheSubPartSP1's all bodies' faces
features = p.find(name=".*/.*/.*", name_regex=True, feature_type=Part)
for feat in features:
print(str(type(feat)) + " : name=" + feat._name)
# -
<class 'ansys.speos.core.face.Face'> : name=TheFaceSP1_B1_F1
Delete#
[21]:
root_part.delete()
print(root_part)
local: {
"name": "RootPart",
"description": "",
"metadata": {},
"body_guids": [],
"parts": []
}