-
-
Save mofosyne/93ad8d64fb40ef80f6af3b09dc755414 to your computer and use it in GitHub Desktop.
#!/usr/bin/env python3 | |
# KiCADv8 Style Prettify S-Expression Formatter (sexp formatter) | |
# By Brian Khuu, 2024 | |
# This script reformats KiCad-like S-expressions to match a specific formatting style. | |
# Note: This script modifies formatting only; it does not perform linting or validation. | |
# Context: Compact element settings are added to support KiCAD-specific handling for readability, e.g., PCB_PLUGIN::formatPolyPts. | |
import os | |
import argparse | |
from pathlib import Path | |
def prettify_sexpr(sexpr_str, compact_element_settings): | |
""" | |
Prettifies KiCad-like S-expressions according to a KiCADv8-style formatting. | |
Args: | |
sexpr_str (str): The input S-expression string to be formatted. | |
compact_element_settings (list of dict): A list of dictionaries containing element-specific settings. | |
Each dictionary should contain: | |
- "prefix" (str): The prefix of elements that should be handled specially. | |
- "elements per line" (int): The number of elements per line in compact mode (optional). | |
Returns: | |
str: The prettified S-expression string. | |
Example: | |
# Input S-expression string | |
sexpr = "(module (fp_text \"example\"))" | |
# Settings for compact element handling | |
compact_settings = [{"prefix": "pts", "elements per line": 4}] | |
# Prettify the S-expression | |
formatted_sexpr = prettify_sexpr(sexpr, compact_settings) | |
print(formatted_sexpr) | |
""" | |
indent = 0 | |
result = [] | |
in_quote = False | |
escape_next_char = False | |
singular_element = False | |
in_prefix_scan = False | |
prefix_keyword_buffer = "" | |
prefix_stack = [] | |
element_count_stack = [0] | |
# Iterate through the s-expression and apply formatting | |
for char in sexpr_str: | |
if char == '"' or in_quote: | |
# Handle quoted strings, preserving their internal formatting | |
result.append(char) | |
if escape_next_char: | |
escape_next_char = False | |
elif char == '\\': | |
escape_next_char = True | |
elif char == '"': | |
in_quote = not in_quote | |
elif char == '(': | |
# Check for compact element handling | |
in_compact_mode = False | |
elements_per_line = 0 | |
if compact_element_settings: | |
parent_prefix = prefix_stack[-1] if (len(prefix_stack) > 0) else None | |
for setting in compact_element_settings: | |
if setting.get("prefix") in prefix_stack: | |
in_compact_mode = True | |
if setting.get("prefix") == parent_prefix: | |
elements_per_line = setting.get("elements per line", 0) | |
if in_compact_mode: | |
if elements_per_line > 0: | |
parent_element_count = element_count_stack[-1] | |
if parent_element_count != 0 and ((parent_element_count % elements_per_line) == 0): | |
result.append('\n' + '\t' * indent) | |
result.append('(') | |
else: | |
# New line and add an opening parenthesis with the current indentation | |
result.append('\n' + '\t' * indent + '(') | |
# Start Prefix Keyword Scanning | |
in_prefix_scan = True | |
prefix_keyword_buffer = "" | |
# Start tracking if element is singular | |
singular_element = True | |
# Element Count Tracking | |
element_count_stack[-1] += 1 | |
element_count_stack.append(0) | |
indent += 1 | |
elif char == ')': | |
# Handle closing elements | |
indent -= 1 | |
element_count_stack.pop() | |
if singular_element: | |
result.append(')') | |
singular_element = False | |
else: | |
result.append('\n' + '\t' * indent + ')') | |
if in_prefix_scan: | |
prefix_stack.append(prefix_keyword_buffer) | |
in_prefix_scan = False | |
prefix_stack.pop() | |
elif char.isspace(): | |
# Handling spaces | |
if result and not result[-1].isspace() and result[-1] != '(': | |
result.append(' ') | |
if in_prefix_scan: | |
# Capture Prefix Keyword | |
prefix_stack.append(prefix_keyword_buffer) | |
# Handle special compact elements | |
if compact_element_settings: | |
for setting in compact_element_settings: | |
if setting.get("prefix") == prefix_keyword_buffer: | |
result.append('\n' + '\t' * indent) | |
break | |
in_prefix_scan = False | |
else: | |
# Handle any other characters | |
result.append(char) | |
# Capture Prefix Keyword | |
if in_prefix_scan: | |
prefix_keyword_buffer += char | |
# Dev Note: In my opinion, this shouldn't be here... but is here so that we can match KiCADv8's behavior when a ')' is following a non ')' | |
singular_element = True | |
# Join results list and strip out any spaces in the beginning and end of the document | |
formatted_sexp = ''.join(result).strip() | |
# Strip out any extra space on the right hand side of each line | |
formatted_sexp = '\n'.join(line.rstrip() for line in formatted_sexp.split('\n')) + '\n' | |
# Formatting of s-expression completed | |
return formatted_sexp | |
# Argument Parsing | |
parser = argparse.ArgumentParser(description="Prettify S-expression files") | |
parser.add_argument("src", type=Path, help="Source file path") | |
parser.add_argument("--dst", type=Path, help="Destination file path") | |
args = parser.parse_args() | |
src_file = args.src.resolve() | |
dst_file = args.dst.resolve() if args.dst else None | |
# Open source file | |
with open(src_file, "r") as file: | |
sexp_data = file.read() | |
# Compact element settings for special handling | |
compact_element_settings = [] | |
src_basename = os.path.basename(src_file) | |
if src_basename.endswith(".kicad_sym"): | |
compact_element_settings.append({"prefix":"pts", "elements per line": 6}) | |
elif src_basename.endswith(".kicad_mod"): | |
pass | |
elif src_basename.endswith(".kicad_sch"): | |
compact_element_settings.append({"prefix":"pts", "elements per line": 6}) | |
elif src_basename.endswith(".kicad_pcb"): | |
compact_element_settings.append({"prefix":"pts", "elements per line": 4}) | |
# Format the S-expression | |
pretty_sexpr = prettify_sexpr(sexp_data, compact_element_settings) | |
# Save or output the result | |
if dst_file: | |
with open(dst_file, "w") as file: | |
file.write(pretty_sexpr) | |
else: | |
print(pretty_sexpr) |
Posted to https://gitlab.com/kicad/code/kicad/-/issues/15232 to note it's potential usefulness
According to craftjon, this specific handling of point sub-elements is intentionally as it would "make the files easier to scan through in a text editor and when shown in a diff view"
As shown in this commit, which introduces void PCB_PLUGIN::formatPolyPts
inside pcbnew/plugins/kicad/pcb_plugin.cpp
(Note that this file was renamed to pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.cpp
) which showed that pts
has it's own print out handling.
This handling applies to both xy, arc and likely any other elements under pts
.
I think it may be worth adding this change in (but fork out the simpler rule to it's own gist, for those who don't want to support kicad specific handling).
Corrected to match KiCADv8 style exactly as far as i know.
I did also noticed however that the behavior of pin_names formatting is inconsistent as well, but at least it's marked down in github kicad issue ticket #15232 'DRAFT: S-Expression normalization' as being under the Misc todo.
ergo was expecting:
(pin_names
(offset 0.762) hide
)
but got
(pin_names
(offset 0.762) hide)
For now, I've added a hack and marked it as a dev note "In my opinion, this shouldn't be here... but is here so that we can match KiCADv8's behavior when a ')' is following a non ')'"
Updated to account for certain elements being presented in a compact form where you have multiple elements in one line.
Also note that these compact form also has limited number of elements per line so that the compact element doesn't span multiple lines. This is set via [{"prefix":"pts", "elements per line": 4}]
for compact_element_settings
(filled_polygon
(layer "In2.Cu")
(pts
(xy 96.314254 106.66524) (xy 96.345329 106.697099) (xy 96.351073 106.709027) (xy 96.351332 106.708907)
(xy 96.44196 106.903259) (xy 96.446227 106.910649) (xy 96.566712 107.082722) (xy 96.572195 107.089256)
(xy 96.720743 107.237804) (xy 96.727272 107.243282) (xy 96.899355 107.363776) (xy 96.906739 107.368039)
(xy 96.972707 107.398801) (xy 97.007845 107.426114) (xy 97.027653 107.465967) (xy 97.028208 107.510467)
(xy 97.009399 107.550802) (xy 96.97662 107.578133) (xy 96.807974 107.665923) (xy 96.801186 107.670198)
(xy 96.629186 107.799339) (xy 96.623185 107.804668) (xy 96.474577 107.960177) (xy 96.469533 107.966406)
(xy 96.348317 108.144102) (xy 96.344355 108.151077) (xy 96.291377 108.26521) (xy 96.264127 108.300396)
(xy 96.224309 108.320274) (xy 96.179809 108.320906) (xy 96.139442 108.302168) (xy 96.111202 108.267772)
(xy 96.110042 108.265369) (xy 96.078039 108.19674) (xy 96.073772 108.18935) (xy 95.953287 108.017277)
(xy 95.947804 108.010743) (xy 95.799256 107.862195) (xy 95.792727 107.856717) (xy 95.620644 107.736223)
(xy 95.61326 107.73196) (xy 95.418907 107.641332) (xy 95.419381 107.640313) (xy 95.387657 107.619824)
(xy 95.36445 107.581849) (xy 95.360019 107.537566) (xy 95.37524 107.495746) (xy 95.407099 107.464671)
(xy 95.419027 107.458926) (xy 95.418907 107.458668) (xy 95.61326 107.368039) (xy 95.620644 107.363776)
(xy 95.792727 107.243282) (xy 95.799256 107.237804) (xy 95.947804 107.089256) (xy 95.953287 107.082722)
(xy 96.073772 106.910649) (xy 96.078039 106.903259) (xy 96.168668 106.708907) (xy 96.169686 106.709381)
(xy 96.190176 106.677657) (xy 96.228151 106.65445) (xy 96.272434 106.650019)
)
)
This is now moved to https://github.com/mofosyne/sexp_formatter so others can contribute in the future
This was created to solve a problem with debugging an issue in mvnmgrx/kiutils#120 . Suspect there may be further application in assisting with testing/debugging for format regression in KiCADv8 via mvnmgrx/kiutils#121