Write ladder logic in Python. Simulate it. Test it. Deploy it -- Currently targets AutomationDirect CLICK PLC and CircuitPython ProductivityOpen P1AM-200.
Project description
pyrung
Ladder logic in Python that reads like ladder, scans like a PLC, and deploys to real hardware.
pyrung turns Python's with block into a ladder rung — condition on the rail, instructions in the body.
from pyrung import Bool, PLC, Program, Rung, out
Button = Bool("Button")
Light = Bool("Light")
with Program() as logic:
with Rung(Button):
out(Light)
with PLC(logic) as plc:
Button.value = True
plc.step()
assert Light.value is True
- Documentation: https://ssweber.github.io/pyrung/
- LLM docs index: https://ssweber.github.io/pyrung/llms.txt
- New to ladder logic? Know Python? Learn Ladder Logic.
Why?
Ladder is still the dominant programming language in North American manufacturing, but ladder programmers have no git, no pytest, no CI. They draw logic in a vendor GUI, download to hardware, and hope. pyrung lets you write and test logic in Python first, then deploy it — same source, three paths:
pyrung_to_ladder()encodes your rungs for ClickNick, a companion editor for Click Programming Software- Generate a self-contained CircuitPython scan loop for a P1AM-200
- Run as an emulated Click over Modbus — spin up two pyrung programs and test them talking to each other, no hardware required
The code reads like a ladder diagram. with Rung(Or(Start, Motor), ~Stop): out(Motor) is a seal-in circuit, and it looks like one. There's no if/else. Power flows through the rung or it doesn't. The scan cycle is deterministic, timers accumulate the same way, rung order matters the same way. What you learn in pyrung transfers directly to a real ladder editor.
Who's this for?
Controls engineers who want to test Click PLC logic without hardware. The Click dialect handles address mapping, memory bank validation, and nickname and ladder file export. Have an existing project? ClickNick imports it.
Python developers entering industrial automation. pyrung teaches you ladder logic in the language and tools you already have. The VS Code debugger lets you step through scans and watch power flow rung by rung. Start with the learning guide.
Makers and P1AM-200 users who want a real scan loop without writing the plumbing. The CircuitPython dialect generates a complete program from the same logic you write and test on your laptop.
Quick start
# Requires Python 3.11+
uv add pyrung
Download the starter project (pyrung-starter-VERSION.zip) from the GitHub releases page for ready-to-run examples with Click CSV round-trip. For the VS Code debugger, grab pyrung-debug-VERSION.vsix from the same page and install it:
code --install-extension pyrung-debug-VERSION.vsix
A motor with start/stop logic
from pyrung import Bool, Program, Rung, latch, reset
Start = Bool("Start")
Stop = Bool("Stop")
Running = Bool("Running")
with Program() as logic:
with Rung(Start):
latch(Running)
with Rung(Stop):
reset(Running)
Test it
from pyrung import PLC
with PLC(logic) as plc:
Start.value = True
plc.step()
assert Running.value is True
# Release start — motor stays latched
Start.value = False
plc.step()
assert Running.value is True
Stop.value = True
plc.step()
assert Running.value is False
Map to Click hardware when you're ready
from pyrung.click import TagMap, x, y
mapping = TagMap({
Start: x[1], # Physical input → X001
Stop: x[2], # Physical input → X002
Running: y[1], # Physical output → Y001
})
mapping.validate(logic) # Checks against Click constraints
mapping.to_nickname_file("motor.csv") # For Click programming software
What's included
Core engine
Coils, latches, timers, counters, branching, subroutines, structured tags, and edge detection. Every scan is a pure function — same inputs, same outputs — so you can fork, rewind, and diff any state in history.
Click PLC dialect
Hardware address mapping, memory bank validation, Modbus instructions, and nickname and ladder file export.
CircuitPython dialect
Generate a self-contained scan loop for the P1AM-200 with 35 supported I/O modules, SD-backed retentive storage, watchdog, and Modbus TCP.
Analysis and verification
plc.cause() and plc.effect() trace causal chains through scan history — why did this tag change, and what would it take to recover? prove() exhaustively checks a property over all reachable states, with counterexample traces when it fails. Automated fault coverage proves every device has an alarm path. Lock files catch behavioral regressions in PRs.
VS Code debugger
Step through scans rung by rung, set breakpoints, force tags, diff states, and time-travel through scan history. Data View for live tag watching, Graph View for dependency visualization, and session capture for mining invariants into pytest tests.
Disclaimers
- Simulation is best-effort. pyrung models Click PLC behavior as closely as practical, but it is not a certified simulator. You are responsible for verifying your program on real hardware before production use.
- Modbus is unauthenticated. The emulated Click Modbus interface and CircuitPython Modbus TCP server listen on the network with no encryption or access control — standard for Modbus, but keep them off untrusted networks.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file pyrung-0.8.1.tar.gz.
File metadata
- Download URL: pyrung-0.8.1.tar.gz
- Upload date:
- Size: 2.3 MB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e27182c86eb63d7965d75faabaeef5cf37970c98aa1a181e85cfcfa90bff9d94
|
|
| MD5 |
d53a2bb8aea9e01726a932d03424d3b2
|
|
| BLAKE2b-256 |
1ba432098512d9c9061ec52dfa00f1621668d0ed86fd8a1355be554ae26a0493
|
Provenance
The following attestation bundles were made for pyrung-0.8.1.tar.gz:
Publisher:
publish.yml on ssweber/pyrung
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pyrung-0.8.1.tar.gz -
Subject digest:
e27182c86eb63d7965d75faabaeef5cf37970c98aa1a181e85cfcfa90bff9d94 - Sigstore transparency entry: 1438092852
- Sigstore integration time:
-
Permalink:
ssweber/pyrung@62be3325ee84359fadc2f2ce6db31ff15fb8fd41 -
Branch / Tag:
refs/tags/v0.8.2 - Owner: https://github.com/ssweber
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@62be3325ee84359fadc2f2ce6db31ff15fb8fd41 -
Trigger Event:
release
-
Statement type:
File details
Details for the file pyrung-0.8.1-py3-none-any.whl.
File metadata
- Download URL: pyrung-0.8.1-py3-none-any.whl
- Upload date:
- Size: 600.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a3a8f337171da90649265c9138bee05b0f8466623e3a7fb8c0127caa8e112300
|
|
| MD5 |
59b934ef4d83879145b84ac47e0fb962
|
|
| BLAKE2b-256 |
c6830e3fcc887fdcf762c28b3740d219c2e4807f2670abd35630650dfa56c0e6
|
Provenance
The following attestation bundles were made for pyrung-0.8.1-py3-none-any.whl:
Publisher:
publish.yml on ssweber/pyrung
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pyrung-0.8.1-py3-none-any.whl -
Subject digest:
a3a8f337171da90649265c9138bee05b0f8466623e3a7fb8c0127caa8e112300 - Sigstore transparency entry: 1438092863
- Sigstore integration time:
-
Permalink:
ssweber/pyrung@62be3325ee84359fadc2f2ce6db31ff15fb8fd41 -
Branch / Tag:
refs/tags/v0.8.2 - Owner: https://github.com/ssweber
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@62be3325ee84359fadc2f2ce6db31ff15fb8fd41 -
Trigger Event:
release
-
Statement type: