Skip to content

Instantly share code, notes, and snippets.

@mofosyne
Last active September 22, 2024 15:49
Show Gist options
  • Save mofosyne/93ad8d64fb40ef80f6af3b09dc755414 to your computer and use it in GitHub Desktop.
Save mofosyne/93ad8d64fb40ef80f6af3b09dc755414 to your computer and use it in GitHub Desktop.
KiCADv8 Style Prettyfy S-Expression Formatter (sexp formatter)
#!/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)
@mofosyne
Copy link
Author

mofosyne commented Sep 20, 2024

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

@mofosyne
Copy link
Author

Posted to https://gitlab.com/kicad/code/kicad/-/issues/15232 to note it's potential usefulness

@mofosyne
Copy link
Author

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.

https://github.com/KiCad/kicad-source-mirror/blob/eb58d7e44c50d2338949cacc3c96f35a54c8f9d1/pcbnew/plugins/kicad/pcb_plugin.cpp#L473

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).

@mofosyne
Copy link
Author

mofosyne commented Sep 20, 2024

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 ')'"

@mofosyne
Copy link
Author

mofosyne commented Sep 22, 2024

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)
	)
)

@mofosyne
Copy link
Author

This is now moved to https://github.com/mofosyne/sexp_formatter so others can contribute in the future

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment