aboutsummaryrefslogtreecommitdiff
path: root/src/ee/part
diff options
context:
space:
mode:
authorTrygve Laugstøl <trygvis@inamo.no>2019-05-11 14:48:15 +0200
committerTrygve Laugstøl <trygvis@inamo.no>2019-05-11 14:48:15 +0200
commit0149fcfa2bd9ac8c9f6b05851f7264f005aa2305 (patch)
tree3757a7148377bf18e32234d375444ffccbc5a8f0 /src/ee/part
parenteffb3470dd1be3a0dd1eaa83991cef45b5e08dab (diff)
downloadee-python-0149fcfa2bd9ac8c9f6b05851f7264f005aa2305.tar.gz
ee-python-0149fcfa2bd9ac8c9f6b05851f7264f005aa2305.tar.bz2
ee-python-0149fcfa2bd9ac8c9f6b05851f7264f005aa2305.tar.xz
ee-python-0149fcfa2bd9ac8c9f6b05851f7264f005aa2305.zip
drawio-to-parts: new tool.
ee.xsd: new type: Assembly and AssemblyPart. Should probably be its own file type. ee.part: Better DSL instead of using the raw xml types.
Diffstat (limited to 'src/ee/part')
-rw-r--r--src/ee/part/__init__.py202
-rw-r--r--src/ee/part/excel.py9
2 files changed, 201 insertions, 10 deletions
diff --git a/src/ee/part/__init__.py b/src/ee/part/__init__.py
index 4354771..996eeff 100644
--- a/src/ee/part/__init__.py
+++ b/src/ee/part/__init__.py
@@ -13,11 +13,152 @@ __all__ = [
]
+class Reference(object):
+ pass
+
+ def to_xml(self):
+ return None
+
+
+class PartReference(Reference):
+ def __init__(self, uri: str):
+ self.uri = uri
+
+ def to_xml(self):
+ return types.PartReference(part_uri=self.uri)
+
+
+class SchematicReference(Reference):
+ def __init__(self, reference: str):
+ self.reference = reference
+
+ def to_xml(self):
+ return types.SchematicReference(reference=self.reference)
+
+
+class PartNumber(Reference):
+ def __init__(self, value: str):
+ self.value = value
+
+ def to_xml(self):
+ return types.PartNumber(value=self.value)
+
+
+class SupplierPartNumber(Reference):
+ def __init__(self, value: str):
+ self.value = value
+
+ def to_xml(self):
+ return types.SupplierPartNumber(value=self.value)
+
+
+class ReferenceList(object):
+ def __init__(self, part_uri):
+ self.part_uri_ = part_uri
+ self.part_references: List[PartReference] = []
+ self.schematic_references: List[SchematicReference] = []
+ self.mpns: List[PartNumber] = []
+ self.spns: List[SupplierPartNumber] = []
+ self.description_references: List[str] = []
+
+ def to_xml(self):
+ part_references = [r.to_xml() for r in self.part_references if isinstance(r, PartReference)]
+ schematic_references = [r.to_xml() for r in self.schematic_references if isinstance(r, SchematicReference)]
+ mpns = [r.to_xml() for r in self.mpns if isinstance(r, PartNumber)]
+ spns = [r.to_xml() for r in self.spns if isinstance(r, SupplierPartNumber)]
+ description_references = self.description_references
+
+ if len(part_references) or len(schematic_references) or len(mpns) or len(spns) or \
+ len(description_references):
+ return types.ReferenceList(part_reference=part_references,
+ schematic_reference=schematic_references,
+ part_number=mpns,
+ supplier_part_number=spns,
+ description=description_references)
+
+ # Part Reference
+
+ def add_part_reference(self, uri):
+ self.part_references.append(PartReference(uri))
+
+ def get_exactly_one_part_reference(self) -> PartReference:
+ refs = self.part_references
+ if len(refs) == 0:
+ raise EeException("This part does not contain any part references{}".
+ format(", uri=" + self.part_uri_ if self.part_uri_ else ""))
+ if len(refs) != 1:
+ raise EeException("This part does not contain exactly one part reference: {}".
+ format(", ".join([ref.uri for ref in refs])))
+
+ return refs[0]
+
+ # Schematic references
+
+ def add_schematic_reference(self, ref):
+ self.schematic_references.append(SchematicReference(reference=ref))
+
+ def get_only_schematic_reference(self) -> Optional[SchematicReference]:
+ return next(iter(self.schematic_references), None)
+
+ def get_exactly_one_schematic_reference(self) -> SchematicReference:
+ refs = self.schematic_references
+ if len(refs) == 0:
+ raise EeException("This part does not contain any schematic references{}".
+ format(", uri=" + self.part_uri_ if self.part_uri_ else ""))
+ if len(refs) != 1:
+ raise EeException("This part does not contain exactly one schematic reference: {}".
+ format(", ".join([ref.reference for ref in refs])))
+
+ return refs[0]
+
+ # MPNs
+
+ def add_mpn(self, mpn: str):
+ self.mpns.append(PartNumber(value=mpn))
+
+ def get_only_mpn(self) -> Optional[PartNumber]:
+ return next(iter(self.mpns), None)
+
+ def get_exactly_one_mpn(self) -> PartNumber:
+ mpns = self.mpns
+ if len(mpns) == 0:
+ raise EeException("This part does not contain any manufacturer part numbers{}".
+ format(", uri=" + self.part_uri_ if self.part_uri_ else ""))
+ if len(mpns) != 1:
+ raise EeException("This part does not contain exactly one mpn: {}".
+ format(", ".join([mpn.value for mpn in mpns])))
+
+ return mpns[0]
+
+ # SPNs
+
+ def add_spn(self, mpn: str):
+ self.spns.append(SupplierPartNumber(value=mpn))
+
+ def get_only_spn(self) -> Optional[SupplierPartNumber]:
+ return next(iter(self.spns), None)
+
+ def get_exactly_one_spn(self) -> SupplierPartNumber:
+ spns = self.spns
+ if len(spns) == 0:
+ raise EeException("This part does not contain any supplier part numbers{}".
+ format(", uri=" + self.part_uri_ if self.part_uri_ else ""))
+ if len(spns) != 1:
+ raise EeException("This part does not contain exactly one spn: {}".
+ format(", ".join([spn.value for spn in spns])))
+
+ return spns[0]
+
+ def add_description_reference(self, description: str):
+ self.description_references.append(description)
+
+
+# TODO: Replace self.xml.referencesProp with ReferenceList
class Part(object):
def __init__(self, xml: types.Part):
assert type(xml) == types.Part
self.xml = xml
- xml.referencesProp = xml.referencesProp if xml.referencesProp is not None else types.ReferencesList()
+ xml.referencesProp = xml.referencesProp if xml.referencesProp is not None else types.ReferenceList()
xml.price_breaksProp = xml.price_breaksProp if xml.price_breaksProp is not None else types.PriceBreakList()
xml.linksProp = xml.linksProp if xml.linksProp is not None else types.LinkList()
xml.factsProp = xml.factsProp if xml.factsProp is not None else types.FactList()
@@ -172,11 +313,27 @@ class Entry(object):
self.pn = next((p.valueProp for p in Part(part).get_mpns()), None)
+class AssemblyPart(object):
+ def __init__(self, uri: Optional[str]):
+ self.count = 0
+ self.sub_parts: List[AssemblyPart] = []
+ self.references = ReferenceList(uri)
+
+ def add_sub_part(self, ap: "AssemblyPart"):
+ self.sub_parts.append(ap)
+
+
+class Assembly(object):
+ def __init__(self):
+ self.parts: List[AssemblyPart] = []
+
+
class PartDb(object):
def __init__(self):
self.parts: List[Entry] = []
self.pn_index: MutableMapping[str, Entry] = {}
self.new_entries = 0
+ self._assembly: Optional[Assembly] = None
def add_entry(self, part: Union[Part, types.Part], new: bool):
if isinstance(part, Part):
@@ -202,6 +359,16 @@ class PartDb(object):
entry = self.pn_index.get(pn, None)
return entry.part if entry else None
+ @property
+ def has_assembly(self):
+ return self._assembly is not None
+
+ @property
+ def assembly(self):
+ if self._assembly is None:
+ self._assembly = Assembly()
+ return self._assembly
+
def load_db(path: Path) -> PartDb:
db = PartDb()
@@ -223,12 +390,35 @@ def find_root_tag(root):
def save_db(path: Path, db: PartDb, sort=False):
part_db = types.PartDb()
- parts = part_db.parts = types.PartList()
- for part in db.iterparts(sort=sort):
- p = Part(part)
- p.clean_xml()
- parts.partProp.append(p.underlying)
+ if db.size() > 0:
+ part_db.parts = types.PartList()
+
+ for part in db.iterparts(sort=sort):
+ p = Part(part)
+ p.clean_xml()
+ part_db.parts.partProp.append(p.underlying)
+
+ if db.has_assembly:
+
+ def to_xml(ap: AssemblyPart):
+ xml = types.AssemblyPart()
+ if ap.count != 0:
+ xml.countProp = ap.count
+ if ap.sub_parts:
+ xml.sub_partsProp = types.AssemblyPartList([to_xml(ap_) for ap_ in ap.sub_parts])
+
+ xml.set_references(ap.references.to_xml())
+
+ return xml
+
+ assembly = db.assembly
+ part_list = types.AssemblyPartList()
+
+ for ap in assembly.parts:
+ part_list.add_assembly_part(to_xml(ap))
+
+ part_db.assemblyProp = types.Assembly(assembly_parts=part_list)
with path.open("w") as f:
part_db.export(outfile=f, level=0, name_=find_root_tag(part_db))
diff --git a/src/ee/part/excel.py b/src/ee/part/excel.py
index 12b583d..b0c4a2b 100644
--- a/src/ee/part/excel.py
+++ b/src/ee/part/excel.py
@@ -1,3 +1,4 @@
+import uuid
from pathlib import Path
from typing import Optional, Mapping
@@ -38,12 +39,12 @@ def from_excel(path: Path, sheet_name: Optional[str]) -> PartDb:
print("Bad part, line #{}. MPN or description is required".format(idx))
continue
- part = Part(types.Part())
+ uri = "urn:uuid:{}".format(uuid.uuid5(uuid.NAMESPACE_URL, url))
+ part = Part(types.Part(uri=uri))
- if not mpn:
- mpn = desc.replace(" ", "-").lower()
+ if mpn:
+ part.add_mpn(mpn)
- part.add_mpn(mpn)
part.xml.descriptionProp = desc
if price: