Copyright © 2024 Pat Deegan, psychogenic.com
This library lets you manipulate kicad schematic (and other) source files with Python, simply.
At a minimum it presents a usable interface to the s-expression structure to add or modify data, but it also does a lot behind the scenes to make both scripting and exploration using the REPL shell efficient and enjoyable.
-
Explore, modify and create elements of a schematic using Python
-
Access to every bit of the source file using a common API
-
Kicad-specific enhancements to allow for
- searching for elements by location, by connection (things that are attached to other things via wires), by type...
- search for symbols by name or regex, on reference or value
- easily access the location of any symbol pin, and more
-
Lots of REPL-friendly features, so you can see at a glance what an element is, access it by name (e.g. schematic.symbol.R14.dnp), use TAB-completion
This array of LEDs
was created, wired up and labelled with a short script, available in an example.
I did a quick walk-through demo live on the maker cast, and it starts exactly here, in MakerCast episode 56
Kicad schematic, layout and other files are stored as trees of s-expressions, like
(kicad_sch (version 20230121) (generator eeschema)
(uuid 20adca1d-43a1-4784-9682-8b7dd1c7d330)
(title_block (title "My Demo Board") (date "2024-01-31") (rev "1.0.5")
(company "Psychogenic Technologies") (comment 1 "(C) 2023, 2024 Pat Deegan")
)
(lib_symbols
(symbol "74xx:74CBTLV3257" (in_bom yes) (on_board yes)
(property "Reference" "U" (at 17.78 1.27 0) (effects (font (size 1.27 1.27))))
(property "Value" "74CBTLV3257" (at 15.24 -1.27 0)(effects (font (size 1.27 1.27))))
...
(symbol (lib_id "SomeLib:Partname") (at 317.5 45.72 0) (unit 1)
(in_bom yes) (on_board yes) (dnp no)
(uuid 342c76f3-b2b8-40b2-a0b0-d83e480188cc)
(property "Reference" "J4" (at 311.15 29.21 0)(effects (font (size 1.27 1.27))))
(property "Value" "TT04_BREAKOUT_REVB" (at 317.5 76.2 0) (effects (font (size 1.27 1.27))))
(pin "A10" (uuid bf1f9b27-0b93-4778-ac69-684e16bea09c))
...
which are great for machine parsing, but not so much for a quick scripted manipulation.
Though you can use the lib to quickly explore and modify the contents of any s-expression file, skip is optimized for use with kicad schematics (eeschema kicad_sch files).
Effort has been made to make exploring the contents easy. This means:
Where possible, collections have named attributes: do use TAB-completion, schem. etc
Examples
>>> schem.symbol.U TAB TAB
outputs (something like, depending on schematic):
schem.symbol.U1 schem.symbol.U3 schem.symbol.U4_B schem.symbol.U4_D
schem.symbol.U2 schem.symbol.U4_A schem.symbol.U4_C schem.symbol.U4_E
and
>>> schem.symbol.U1.property. TAB TAB
outputs
schem.symbol.U1.property.Characteristics
schem.symbol.U1.property.Datasheet
schem.symbol.U1.property.DigikeyPN
schem.symbol.U1.property.Footprint
schem.symbol.U1.property.MPN
schem.symbol.U1.property.Reference
schem.symbol.U1.property.Value
__repr__
and __str__
overrides so you have an idea what you're looking at. Just enter the variable in the REPL console and it should spit something sane out.
>>> schem
<Schematic 'samp/my_schematic.kicad_sch'>
>>> schem.symbol[23]
<symbol C28>
>>> schem.symbol[23].property
<Collection [<PropertyString Reference = 'C28'>,
<PropertyString Value = '100nF'>,
<PropertyString Footprint = 'Capacitor_SMD:C_0402_1005Metric'>,
<PropertyString Datasheet = '~'>, <PropertyString JLC = ''>,
<PropertyString Part_number = ''>, <PropertyString MPN = 'CL05A104KA5NNNC'>,
<PropertyString Characteristics = '100n 10% 25V XR 0402'>,
<PropertyString MPN_ALT = 'GRM155R71E104KE14J'>]>
>>> schem.wire[22]
<Wire [314.96, 152.4] - [317.5, 152.4]>
>>> schem.text[4]
<text 'Options fo...'>
# and much more, e.g. for library symbols, junctions, labels, whatever's in there
Here's some sample interaction with the library. The basic functions allow you to view and edit attributes. More involved helpers let you search and crawl the schematic to find things.
# load a schematic
schem = skip.Schematic('samp/my_schem.kicad_sch')
# loop over all components, treat collection as array
for component in schem.symbol:
component # do something
# search through the symbols by reference or value, using regex or starts_with
>>> schem.symbol.reference_matches(r'(C|R)2[158]')
[<symbol C25>, <symbol C28>, <symbol R25>, <symbol C21>,
<symbol R21>, <symbol R28>]
>>> sorted(schem.symbol.value_startswith('10k'))
[<symbol R12>, <symbol R30>, <symbol R31>, <symbol R32>, <symbol R33>,
<symbol R42>, <symbol R43>, <symbol R44>, <symbol R45>, <symbol R46>,
<symbol R47>, <symbol R48>, <symbol R8>, <symbol R9>]
# or refer to components by name
conn = schem.symbol.J15
# symbols have attributes
if not conn.in_bom:
conn.dnp.value = True
# and properties (things that can be named by user)
for p in conn.property:
print(f'{p.name} = {p.value}')
# will output "Reference = J15", "MPN = USB4500-03-0-A" etc
# and change properties, of course
>>> conn.property.MPN.value = 'ABC123'
>>> conn.property.MPN
<PropertyString MPN = 'ABC123'>
# clone pretty much anything and modify it
# this is the easiest way to do complex elements
>>> mpn_alt = conn.property.MPN.clone()
>>> mpn_alt.name = 'MPN_ALT'
>>> mpn_alt.value = 'ABC456'
# or create wholly new elements (for some types at the moment)
>>> a_wire = schem.wire.new()
>>> a_wire.start.value = [schem.symbol.D1.pin.K.location.x, 20]
>>> a_wire.end.value = [schem.symbol.D1.pin.K.location.x,
schem.symbol.D1.pin.K.location.y]
>>> a_wire
<Wire [2.54, 20] - [2.54, 25.4]>
>>> title = schem.text.new()
>>> title.value = 'Super Schem'
>>> title.move(50, 10)
>>> title
<text Super Sch...>
# save the result
>>> schem.write('/tmp/newfile.kicad_sch')
# let's verify the change
>>> schem.read('/tmp/newfile.kicad_sch')
>>> conn = schem.symbol.J15
>>> for p in conn.property:
p
<PropertyString Reference = 'J15'>
<PropertyString Value = 'USB4500-03-0-A_REVA'>
<PropertyString MPN = 'ABC123'>
<PropertyString MPN_ALT = 'ABC456'>
Collections, and the source file object, have helpers to locate relevant elements.
Where applicable, such as for symbols (components), directly attached elements (via wires) may be listed using attached_*
>>> conn = sch.symbol.J15
>>> conn.attached_ TAB TAB
conn.attached_all
conn.attached_global_labels
conn.attached_labels
conn.attached_symbols
conn.attached_wires
>>> conn.attached_symbols
[<symbol C3>, <symbol R16>, <symbol C47>,
<symbol R20>, <symbol C46>, <symbol R21>,
<symbol F1>]
>>> conn.attached_labels
[<label CC2>, <label CC1>]
# or list everything attached
>>> conn.attached_all
[<symbol C3>, <symbol R16>, <symbol C47>,
<symbol R20>, <symbol C46>, <symbol R21>,
<symbol F1>, <global_label usb_d->,
<global_label usb_d+>, <label CC2>,
<label CC1>]
Symbols may be located by reference or value
-
schem.symbol.reference_matches(REGEX)
-
schem.symbol.reference_startswith(STR)
In addition, containers with 'positionable' elements have
-
within_circle(X, Y, RADIUS)
-
within_rectangle(X1, Y1, X2, Y2)
-
within_reach_of(ELEMENT, RADIUS) # circle around ELEMENT's position
-
between_elements(ELEMENT1, ELEMENT2) # within rectangle formed by two elements
A collection will only return results of it's own type (e.g. global_labels.within_circle() will only return global labels).
To search the entire schematic, the same within* and between() methods exist on the source file object (the schem, here). This will return any label, global label or symbol with the constrained bounds.
>>> schem.global_label.between_elements(sch.symbol.C49, sch.symbol.R16)
[<global_label usb_d->, <global_label usb_d+>,
<global_label usb_d+>, <global_label usb_d->]
>>>
>>> schem.between_elements(sch.symbol.C49, sch.symbol.R16)
[<symbol C44>, <symbol #PWR0121>, <symbol J15>,
<symbol C47>, <symbol #PWR0122>, <symbol D5>, <symbol C48>,
<symbol C45>, <symbol #PWR021>, <symbol C49>, <symbol R20>,
<symbol C46>, <symbol #FLG02>, <symbol F1>, <symbol C3>,
<symbol R16>, <symbol R21>, <symbol D6>, <label CC2>,
<label CC1>, <label vfused>, <global_label usb_d->,
<global_label usb_d+>, <global_label usb_d+>,
<global_label usb_d->]
Further documentation to come. For now, use the above, load a schematic in a console, and explore what's available using TAB-/code-completion.
The top-level objects are Schematic
for... schematics, and PCB
(layouts), derived from SourceFile
. Most of the work to date has gone into the schematics, because that's where functionality was most needed.
All SourceFile
objects have:
-
a constructor, e.g.
Schematic(FILEPATH)
, that takes the file to ingest as a parameter; -
read(FILEPATH)
, to read in a file (discarding anything present in the object, thus far; -
write(FILEPATH)
, to output the current state of the tree to a file; -
reload()
, to read the last file read -
write()
, to overwrite the last file read (no warnings, be smert)
Derivatives may have additional functionality. Schematic, for instance, has methods that can list all symbols (components), labels and global labels present within a given area, using
-
within_rectangle(X1, Y1, X2, Y2)
, between (X1,Y1) and (X2,Y2) rectangle -
within_circle(X, Y, RADIUS)
, within RADIUS of (X,Y) -
within_reach_of(ELEMENT, DISTANCE)
, within a given distance of ELEMENT's location; and -
between_elements(EL1, EL2)
, bounded between the coordinates of those two elements
A Schematic will, depending on the contents of the original source, have elements and collections named
-
title_block
, which has atitle
,comment
,company
and other items within; -
wire
, all the wires in the schem; -
symbol
, all the symbols (components) in the schem -
global_label
, all the global labels; -
label
, all the labels (i.e. net names); -
text
, all the text blocks
and others (image
, polyline
, rectangle
, sheet
, lib_symbols
etc etc -- go explore).
For entities that may be constructed with new()
, a collection will be present regardless of whether the schematic actually contains any of these (e.g. for labels, global labels, text and wires).
Many of these have interesting additions, see below.
Elements can be clone()
ed (called on the item to copy) or, for some types at least, constructed with a new()
call on the parent collection (see below).
Pretty much everything other than the top-level source file object is either some value parsed from the s-expressions or a collection thereof.
All the parsed values have a few methods and attributes at their core, some of them are further wrapped (transparently) to provide additional function.
Finally, the containers all behave as lists, but may have named attributes and other functions as well.
Any element will have, at a minimum the following attributes and methods
-
entity_type
: a name, from the source file itself, for this type of element; -
value
: a value associated with this element; -
clone()
: a means of making a (deep) copy of the element; and -
delete()
: removal of element (and all it's children) from it's parent tree
Cloned elements will be at copied at into the level they were created at, if applicable. Meaning if you want to create a property for symbol X1, then clone a property from symbol X1 (anyone), and change it's name, value, whatever.
Leaf elements that are of a boolean nature look that way when cast, e.g.
if schem.symbol.C24.dnp:
# do something
The most important thing when setting values is to remember to set them on the .value
schem.symbol.R11.dnp.value = True # yes,
# because doing
schem.symbol.R11.dnp = True # BAD !
# would overwrite the 'dnp' attrib on R11 with a plain
# boolean, which isn't what you want
Elements that are leaves will not have any children or additional attributes.
Many elements do have sub-elements. These can be iterated over through a children
attribute, but mostly it is worthwhile using the automatically generated attributes directly.
Some common examples that will be found are:
-
at
: location of this positioned element -
effects
: present for things like text and labels, these have their own sub-attributes, likefont
orjustify
;
Elements with at
may be re-positioned, using either
-
move(x, y, [rotation])
to set the location; or -
translation(deltax, deltay)
to shift the location
You could, in fact, set the at
value directly:
schem.symbol.R4.at.value = [10, 20, 0]
The risk here is that R4 has a bunch of children (like the reference, etc) that have now not moved, whereas move() and translation() handle that for you.
Wires are a bit annoying to deal with, so functions have been added to make life easier.
The have start
and end
points, which you can manipulate if you wish.
The also support:
-
start_at(COORDS_OR_POSITIONABLE)
-
end_at(COORDS_OR_POSITIONABLE)
-
delta_x
get/set attribute, calculated as end - start -
delta_y
get/set attribute, calculated as end - start
Thus you can do things like forcing it horizontal ("delta_y = 0") or stretching it out ("delta_x = 25.4") or start it on a pin without thinking of coordinates at all
join_wire = sch.wire.new()
join_wire.start_at(sch.symbol.D2.pin.A)
See the charlieplex example for that, and junctions and labels and more.
Any time there is more than one of some entity type, say 'wire' or 'symbol' it winds up as part of a collection of the same name, in whichever parent it is resident.
Any collection can be treated as a list. So you can loop over them, or access them by index.
# try
>>> for w in sch.wire:
... w
...
<Wire [149.86, 273.05] - [154.94, 273.05]>
<Wire [63.5, 264.16] - [63.5, 265.43]>
<Wire [140.97, 45.72] - [146.05, 45.72]>
<Wire [285.75, 50.8] - [289.56, 50.8]>
<Wire [152.4, 73.66] - [152.4, 74.93]>
# ...
# or
>>> sch.wire[0].start
<xy [149.86, 273.05]>
>>> sch.wire[0].end
<xy [154.94, 273.05]>
>>> sch.wire[0].length
5.08
Same applies to all of them.
>>> sch.label # acts like a list
<Collection [<label CC2>, <label CC1>, <label ~{CRVRST}>, <label vfused>]>
>>>
So label
, global_label
, symbol
, text
, junction
, image
etc depending on what's in there... TAB TAB to find out!
Some collections contain elements that have an identifier that it would be reasonable to believe is unique, such as symbol
. In such cases, named attributes are available as well.
This isn't of great use for general purpose scripts, but for navigating in a REPL it's a huge time saver.
These are all dynamically generated based on the contents of the source.
This allows for exploration using tab-completion, which is sweet:
>>> sch.symbol.U
sch.symbol.U1
sch.symbol.U2
sch.symbol.U3
sch.symbol.U4_A
sch.symbol.U4_B
sch.symbol.U4_C
sch.symbol.U4_D
sch.symbol.U4_E
sch.symbol.U7
For some entities held in collections, it is possible to go beyond cloning and create elements using new()
.
At this time:
wire
label
global_labels
andtext
collections allow you to call new on the collection itself. It will be constructed and returned to you immediately, available in the collection thereafter.
glabel = schem.global_label.new()
glabel.value
'GLABEL'
glabel.value = 'going_places'
glabel.shape.value = 'output'
glabel.move(100, 200)
These are based on the templates in skip.element_template
, and have a suitable default structure. The uuid
may be ignored, it is replaced by a uuid library generated value on clone()
or new()
in any case.
Symbol collection
The symbol collection is so central it has a few additional methods for getting a hold of elements:
-
reference_startswith(STR)
, lists all symbols with reference starting with this string, e.g. 'R' for resistors; -
reference_matches(REGEX)
, all symbols with refs matching this regex; and the same for values -
value_startswith(STR)
; and -
value_matches(REGEX)
Some elements have attributes which are themselves collections, such as symbol properties
, pins
, etc
>>> for p in sch.symbol.U2.property:
... p
...
<PropertyString Reference = 'U2'>
<PropertyString Value = 'TLV1117LV33'>
<PropertyString Footprint = 'Package_TO_SOT_SMD:SOT-223-3_TabPin2'>
<PropertyString Datasheet = 'https://www.ti.com/lit/ds/symlink/tlv1117lv.pdf'>
<PropertyString MPN = 'TLV1117LV33DCYR'>
<PropertyString MPN_ALT = 'AZ1117CH-3.3TRG1'>
<PropertyString Characteristics = 'VREG 3.3V SOT-223-4'>
>>> sch.symbol.U2.property.MPN.value
'TLV1117LV33DCYR'
>>> sch.lib_symbols.Regulator_Linear_AP2112K_1_8.pin
<Collection [<Pin 1 "VIN">, <Pin 2 "GND">, <Pin 3 "EN">,
<Pin 4 "NC">, <Pin 5 "VOUT">]>
These attribute names must, however, be Python-y... so starting with a digit or some weird character won't work out.
For many this is fine
>>> sch.symbol.U2.pin.VO.number
'2'
For others, say when the pins have no name set and only a number available, a prefix n is used. In more complex cases, a set of cleanups are executed
>>> sch.symbol.U4_B.pin
<Collection [<SymbolPin 5 "~">, <SymbolPin 6 "~">, <SymbolPin 7 "~">]>
>>> sch.symbol.U4_B.pin.n5.name
'~'
# here's a tough one, the pin is named "mio[36]/~{ctrl_sel_rst}". This becomes
>>> sch.symbol.J4_C.pin.mio36_nctrl_sel_rst.name
'mio[36]/~{ctrl_sel_rst}'
In the example above, the not (~) prefix becomes n and invalid python chars discarded.
Worst case is that you can treat the collection as an array, but when doing a quick fix from a terminal or poking around, the names come in very handy.
Just like the source file itself, the collections which contained "positioned" elements (namely symbols, labels and such) have the same search functions, but in these cases they will only return elements contained within the collection itself.
-
within_rectangle(X1, Y1, X2, Y2)
, between (X1,Y1) and (X2,Y2) rectangle -
within_circle(X, Y, RADIUS)
, within RADIUS of (X,Y) -
within_reach_of(ELEMENT, DISTANCE)
, within a given distance of ELEMENT's location; and -
between_elements(EL1, EL2)
, bounded between the coordinates of those two elements
So schem.symbol.within_circle(100, 100, 50)
will only return matching symbols, nothing else.
Some elements in here are more involved and important that others, namely the symbols (components).
In addition to their bare, source-based, attributes common to all elements, symbols also have
-
a
property
collection, with the Reference, Value, anything you've added to the edit dialog like MPN etc -
a
pin
collection, which actually uses magic to combine with the lib_symbol this is based on and figure out pin locations -
and various means to find other connected things
For that last point, dymanic properties exist that allow you
Methods to crawl schematic The schematic isn't a netlist, so the 'connected' things mentioned above are found the hard way, basically crawling along wires.
You can do these on individual pins, or on a symbol as a whole:
-
attached_wires
, a list of any wire directly connected to this pin (or entire symbol); -
attached_symbols
, a list of symbols connected, directly on indirectly (through wires), to this pin (or any pin of entire symbol); -
attached_labels
, a list of labels atop wires connected, directly on indirectly, to this pin (or any pin of entire symbol); -
attached_global_labels
, global labels connected to wires that are connected to pins
>>> sch.symbol.SW4.attached_all
[<symbol R11>, <symbol R10>, <symbol R6>, <symbol R5>, <symbol R4>,
<symbol R3>, <symbol R2>, <symbol R1>, <global_label in6>,
<global_label in5>, <global_label in4>, <global_label in3>,
<global_label in2>, <global_label in1>, <global_label in0>,
<global_label in7>]
# the pin version returns everything connected to it,
# hence the parent symbol as well
>>> sch.symbol.SW4.pin.n10.attached_all
[<symbol SW4>, <global_label in6>]
That's it for now. Explore a schematic you know in the console, let me know how it goes and have fun.
2024-04-04 Pat Deegan