summaryrefslogtreecommitdiff
path: root/gerbonara/tests/test_kicad_footprints.py
diff options
context:
space:
mode:
authorjaseg <git@jaseg.de>2023-04-15 17:09:20 +0200
committerjaseg <git@jaseg.de>2023-04-15 17:09:20 +0200
commit2400ff8e5fea41c1f8c6251d37a02209ec253f93 (patch)
tree395968d05156c094709fda605a9fe572aed32b1d /gerbonara/tests/test_kicad_footprints.py
parentb43e4e2eec99b92a1e87f6388703db1ca33518d1 (diff)
downloadgerbonara-2400ff8e5fea41c1f8c6251d37a02209ec253f93.tar.gz
gerbonara-2400ff8e5fea41c1f8c6251d37a02209ec253f93.tar.bz2
gerbonara-2400ff8e5fea41c1f8c6251d37a02209ec253f93.zip
cad: Add KiCad symbol/footprint parser
Diffstat (limited to 'gerbonara/tests/test_kicad_footprints.py')
-rw-r--r--gerbonara/tests/test_kicad_footprints.py57
1 files changed, 57 insertions, 0 deletions
diff --git a/gerbonara/tests/test_kicad_footprints.py b/gerbonara/tests/test_kicad_footprints.py
new file mode 100644
index 0000000..a238e1c
--- /dev/null
+++ b/gerbonara/tests/test_kicad_footprints.py
@@ -0,0 +1,57 @@
+
+from itertools import zip_longest
+import re
+
+from ..cad.kicad.sexp import build_sexp
+from ..cad.kicad.sexp_mapper import sexp
+from ..cad.kicad.footprints import Footprint
+
+def test_parse(kicad_mod_file):
+ Footprint.open(kicad_mod_file)
+
+def test_round_trip(kicad_mod_file):
+ print('========== Stage 1 load ==========')
+ orig_fp = Footprint.open(kicad_mod_file)
+ print('========== Stage 1 save ==========')
+ stage1_sexp = build_sexp(orig_fp.sexp())
+ with open('/tmp/foo.sexp', 'w') as f:
+ f.write(stage1_sexp)
+
+ print('========== Stage 2 load ==========')
+ reparsed_fp = Footprint.parse(stage1_sexp)
+ print('========== Stage 2 save ==========')
+ stage2_sexp = build_sexp(reparsed_fp.sexp())
+ print('========== Checks ==========')
+
+ for stage1, stage2 in zip_longest(stage1_sexp.splitlines(), stage2_sexp.splitlines()):
+ assert stage1 == stage2
+
+ return
+
+ original = re.sub(r'\(', '\n(', re.sub(r'\s+', ' ', kicad_mod_file.read_text()))
+ original = re.sub(r'\) \)', '))', original)
+ original = re.sub(r'\) \)', '))', original)
+ original = re.sub(r'\) \)', '))', original)
+ original = re.sub(r'\) \)', '))', original)
+ stage1 = re.sub(r'\(', '\n(', re.sub(r'\s+', ' ', stage1_sexp))
+ for original, stage1 in zip_longest(original.splitlines(), stage1.splitlines()):
+ if original.startswith('(version'):
+ continue
+
+ original, stage1 = original.strip(), stage1.strip()
+ if original != stage1:
+ if any(original.startswith(f'({foo}') for foo in ['arc', 'circle', 'rectangle', 'polyline', 'text']):
+ # These files have symbols with graphic primitives in non-standard order
+ return
+
+ if original.startswith('(symbol') and stage1.startswith('(symbol'):
+ # Re-export can change symbol order. This is ok.
+ return
+
+ if original.startswith('(at') and stage1.startswith('(at'):
+ # There is some disagreement as to whether rotation angles are ints or floats, and the spec doesn't say.
+ return
+
+ assert original == stage1
+
+