Skip to main content

Pytest fixture for recording and replaying serial port traffic.

Project description

pytest-reserial

build PyPI License

Pytest plugin for recording and replaying serial port traffic during tests.

Installation

pip install pytest-reserial

Usage

  1. Write your tests as if they would run with the device connected. While you iterate you can use pytest --disable-reserial to have your code interact with the device without any side effects.

  2. When your tests pass with the device connected, run pytest --record to record serial traffic from the passing tests.

  3. Now you can disconnect the device and run your tests with pytest.

A simple example:

# my_serial_app.py
from serial import Serial

def my_serial_app():
    with Serial(port=/dev/ttyUSB0) as s:
        # When we send '1' to the device, it responds with '2'.
        s.write(b"\x01")
        return s.read()

# Just use the 'reserial' fixture!
def test_my_serial_app(reserial):
    assert my_serial_app() == b"\x02"

Next:

  1. Connect the device.
  2. Run pytest my_serial_app.py --disable-reserial and verify that the test passes with the device connected.
  3. Run pytest --record my_serial_app.py. The test will run again, and the traffic will be recorded.
  4. Disconnect the device.
  5. Run pytest my_serial_app.py. The test will pass!

The logged traffic will be stored as JSON Lines, with one file per test file and one line per test, in the same directory as your test files. The files will have the same names as the test files except with a .jsonl extension instead of .py. For example, if your project layout is:

├── src
│   ├── myproject
│      ├── ...
├── tests
│   ├── test_myproject.py

Then after running pytest --record, the tests/ directory will contain a new file, test_myproject.jsonl, containing the recorded serial traffic from the tests.

Why

Have you ever tried to write tests for a program that talks to an external device over serial (like an Arduino or something)? You probably wrote the tests assuming that the device is question would always be connected when running the tests, right? And later you got bit by one or more of the pitfalls of that approach:

  • You wanted to run the tests when the device wasn't connected. Perhaps you were travelling, or someone had borrowed it. Whatever the reason, you found yourself unable to run the tests, and therefore unable to continue development, until you could connect the device again.

  • You made a change to your program, and one of your tests failed. So far so good, right? That's what tests are for, after all. Only, you can't figure out why the test is failing. You spend several hours trying to fix it, but eventually give up and revert your changes. But the test still fails. So you try another device, and sure enough, now it passes. Turns out, what you thought was a problem with your code was actually a hardware failure.

  • Some of the tests depend on the device being in a certain state, and some of the tests depend on the device being in another state. So you can't run the entire test suite all at once, instead being forced to stop it halfway through and mess with a bunch of wires and buttons before you can run the rest of the tests.

And then you asked yourself, 'How do I write my tests so that the device doesn't need to be connected?' You may have gone down the rabbit hole that is mocking, and then replaced large parts of pyserial with mock interfaces, and ultimately ended up with a test suite that was significantly more complex than the program it was meant to test.

With pytest-reserial, you don't have to worry about any of that. Just write your tests as if the device is always connected. Then, simply use the reserial fixture to record the serial traffic from passing tests, and replay it when the device isn't connected.

Requirements

pytest-reserial depends on pytest and pyserial.

Copyright

MIT License, (C) 2022 Alexander Bessman

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

pytest_reserial-0.6.1.tar.gz (14.4 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

pytest_reserial-0.6.1-py3-none-any.whl (8.7 kB view details)

Uploaded Python 3

File details

Details for the file pytest_reserial-0.6.1.tar.gz.

File metadata

  • Download URL: pytest_reserial-0.6.1.tar.gz
  • Upload date:
  • Size: 14.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pytest_reserial-0.6.1.tar.gz
Algorithm Hash digest
SHA256 c69699ae4b219223bd81c0e3516523d3e8fd91cfc0d825668ace096248830423
MD5 7a0b071dead2f0dc7eb801c94402a05e
BLAKE2b-256 2b5f133606cf5b3b4468fcccdce18c57acd9f8a041179bf4708d85d67ba3f5a2

See more details on using hashes here.

Provenance

The following attestation bundles were made for pytest_reserial-0.6.1.tar.gz:

Publisher: release.yml on bessman/pytest-reserial

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pytest_reserial-0.6.1-py3-none-any.whl.

File metadata

File hashes

Hashes for pytest_reserial-0.6.1-py3-none-any.whl
Algorithm Hash digest
SHA256 da9a5a23a3f1e727674cfd9939d8d471711d6f81245461b54b8fd1cd299f5106
MD5 594cbf7e7a46cc770db322f1db91d518
BLAKE2b-256 7976aa8888c77ede5dbf39bf34be2ab880d61e44a4a06766314fbcddddbab880

See more details on using hashes here.

Provenance

The following attestation bundles were made for pytest_reserial-0.6.1-py3-none-any.whl:

Publisher: release.yml on bessman/pytest-reserial

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page