Qt-based framework for controlling scientific instruments over serial ports
Project description
QInstrument
A Qt-based framework for controlling scientific instruments over serial ports.
Instruments are represented as Qt objects with a uniform property system,
automatic UI binding, and JSON-based configuration persistence.
Any Qt binding (PyQt5, PyQt6, PySide2, PySide6) is supported via qtpy.
Instruments
IPG Photonics
- YLR Series: Ytterbium fiber laser
Laser Quantum
- Opus: Continuous-wave laser
PiezoDrive
- PDUS210: Piezo transducer driver
Prior Scientific
- Proscan II/III: Motorized microscope stage controller
Stanford Research Systems
- DS345: 30 MHz Synthesized Function Generator
- SR830: 100 kHz Digital Lock-in Amplifier
- SR844: 200 MHz RF Lock-in Amplifier
Tektronix
- TDS1000: Digital oscilloscope
Installation
pip install QInstrument
pip install PyQt6 # or PyQt5, PySide2, PySide6
Installing from PyPI also places a qinstrument command on your PATH.
To install from source:
git clone https://github.com/davidgrier/QInstrument
cd QInstrument
python -m venv .qi
source .qi/bin/activate
pip install -e ".[dev]"
Quick start
Rack application
Launch the rack to control multiple instruments at once:
qinstrument DS345 SR830
On subsequent runs, qinstrument with no arguments restores the
last-used instrument list automatically. Use the Add instrument…
button to add instruments at runtime, or right-click any instrument
to remove it.
Single instrument widget
Each instrument widget also has a built-in example() entry point:
python -m QInstrument.instruments.DS345.widget
This finds a connected DS345 and opens its control panel. If no instrument is detected it falls back automatically to a simulated (fake) device so the UI is always usable.
Embedding a widget in your application
from qtpy.QtWidgets import QApplication
from QInstrument.instruments.DS345 import QDS345Widget
app = QApplication([])
widget = QDS345Widget()
widget.show()
app.exec()
Using a simulated instrument
When hardware is not available, pass a fake device directly:
from QInstrument.instruments.DS345 import QDS345Widget, QFakeDS345
app = QApplication([])
widget = QDS345Widget(device=QFakeDS345())
widget.show()
app.exec()
Architecture
QtCore.QObject
└── QAbstractInstrument # property/method registry, thread-safe get/set
└── QSerialInstrument # holds QSerialInterface; adds open/find/identify
└── QXxxInstrument # concrete instrument
QtSerialPort.QSerialPort
└── QSerialInterface # raw serial I/O (owned by QSerialInstrument)
QWidget
├── QInstrumentWidget # loads .ui file; auto-binds widgets to properties;
│ # saves/restores device state via Configure
└── QInstrumentRack # holds multiple QInstrumentWidgets; runtime
# add/remove; saves/restores instrument list
Each instrument lives in instruments/<Name>/ with three files:
| File | Purpose |
|---|---|
instrument.py |
Serial communication and property registration |
fake.py |
Simulated instrument for UI development without hardware |
widget.py |
Qt widget, .ui file binding, example() entry point |
Development
Run the test suite:
source .qi/bin/activate
pytest tests/
Tests run automatically before every git push via a pre-push hook.
To install the hook in a fresh clone:
cp hooks/pre-push .git/hooks/pre-push # if tracked, else set up manually
chmod +x .git/hooks/pre-push
Acknowledgements
Work on this project at New York University is supported by the National Science Foundation of the United States under award number DMR-2438983.
Project details
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 qinstrument-2.1.0.tar.gz.
File metadata
- Download URL: qinstrument-2.1.0.tar.gz
- Upload date:
- Size: 90.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8dcab4733da3637287f5935b4cb4725b7033c7a86872cd9432e4af6b3c38657c
|
|
| MD5 |
89db3016652d472ed3ffcbbd9e5ed103
|
|
| BLAKE2b-256 |
66da753eb690a306a94ddf50020e6824a674f63645239eb50b0b63bace3c8580
|
Provenance
The following attestation bundles were made for qinstrument-2.1.0.tar.gz:
Publisher:
publish.yml on davidgrier/QInstrument
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
qinstrument-2.1.0.tar.gz -
Subject digest:
8dcab4733da3637287f5935b4cb4725b7033c7a86872cd9432e4af6b3c38657c - Sigstore transparency entry: 1280338229
- Sigstore integration time:
-
Permalink:
davidgrier/QInstrument@9e10a373a9f629e789ba2e46ca3a981bc9862ec4 -
Branch / Tag:
refs/tags/v2.1.0 - Owner: https://github.com/davidgrier
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@9e10a373a9f629e789ba2e46ca3a981bc9862ec4 -
Trigger Event:
push
-
Statement type:
File details
Details for the file qinstrument-2.1.0-py3-none-any.whl.
File metadata
- Download URL: qinstrument-2.1.0-py3-none-any.whl
- Upload date:
- Size: 108.3 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 |
38a05d2a5e135a01b0ed77fee6643d3add6c729271cd668f2e957eecc2e77f50
|
|
| MD5 |
51c9950bd46b5a73b216a2066ab35e16
|
|
| BLAKE2b-256 |
09f5d72fd5ad681008ab4e6c85379a8a00566f555e941c4827513a57a51c79a1
|
Provenance
The following attestation bundles were made for qinstrument-2.1.0-py3-none-any.whl:
Publisher:
publish.yml on davidgrier/QInstrument
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
qinstrument-2.1.0-py3-none-any.whl -
Subject digest:
38a05d2a5e135a01b0ed77fee6643d3add6c729271cd668f2e957eecc2e77f50 - Sigstore transparency entry: 1280338233
- Sigstore integration time:
-
Permalink:
davidgrier/QInstrument@9e10a373a9f629e789ba2e46ca3a981bc9862ec4 -
Branch / Tag:
refs/tags/v2.1.0 - Owner: https://github.com/davidgrier
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@9e10a373a9f629e789ba2e46ca3a981bc9862ec4 -
Trigger Event:
push
-
Statement type: