Files
create/src/Mod/BIM/utils/getIfcElements.py
2024-05-16 17:23:10 +02:00

92 lines
3.2 KiB
Python

"""This script converts the computer iterpretable listing ifcXML XSD into a list
of non-abstract children of IfcProduct"""
import xml.sax, json, copy
class IfcElementHandler(xml.sax.ContentHandler):
def __init__(self):
super().__init__()
self.elements = {}
self.current_element_name = None
self.enums = {}
self.current_enum_name = None
self.attribute_stack = []
def startElement(self, name, attrs):
if name == "xs:element" and "substitutionGroup" in attrs:
self.elements[attrs["name"]] = {
"is_abstract": True if "abstract" in attrs else False,
"parent": attrs["substitutionGroup"][len("ifc:") :],
"attributes": [],
}
self.current_element_name = attrs["name"]
elif name == "xs:simpleType" and "name" in attrs and "Enum" in attrs["name"]:
self.current_enum_name = attrs["name"]
self.enums[self.current_enum_name] = []
elif name == "xs:enumeration" and self.current_enum_name:
self.enums[self.current_enum_name].append(attrs["value"].upper())
elif (
name == "xs:attribute"
and self.current_element_name
and "name" in attrs
and "type" in attrs
):
self.elements[self.current_element_name]["attributes"].append(
{
"name": attrs["name"],
"type": attrs["type"].replace("ifc:", ""),
}
)
def endDocument(self):
elements = {}
for name, data in self.elements.items():
for index, attribute in enumerate(data["attributes"]):
data["attributes"][index] = self.resolve_enums(attribute)
for name, data in self.elements.items():
if data["is_abstract"]:
continue
if self.is_an_ifcproduct(data):
self.attribute_stack = []
self.get_parent_attributes(data)
elements[name] = copy.deepcopy(data)
elements[name]["attributes"] = copy.deepcopy(self.attribute_stack)
self.elements = elements
def resolve_enums(self, attribute):
if attribute["type"] in self.enums:
attribute["is_enum"] = True
attribute["enum_values"] = self.enums[attribute["type"]]
return attribute
attribute["is_enum"] = False
attribute["enum_values"] = []
return attribute
def get_parent_attributes(self, data):
self.attribute_stack.extend(data["attributes"])
if (
data["parent"] != "IfcProduct"
): # For now, we treat attributes above IfcProduct in a special way
self.get_parent_attributes(self.elements[data["parent"]])
def is_an_ifcproduct(self, data):
if data["parent"] == "IfcProduct":
return True
else:
for name, parent_data in self.elements.items():
if name == data["parent"]:
return self.is_an_ifcproduct(parent_data)
return False
xsd_path = "IFC4_ADD2.xsd"
handler = IfcElementHandler()
parser = xml.sax.make_parser()
parser.setContentHandler(handler)
parser.parse(xsd_path)
print(json.dumps(handler.elements, indent=4))