Skip to main content

A general-purpose respiratory tracking toolbox for interoception research

Project description

respyra logo

respyra

A general-purpose respiratory tracking toolbox for interoception research

Tests PyPI Docs License: MIT Python 3.10


respyra is a Python toolbox that integrates a Vernier Go Direct Respiration Belt (GDX-RB) with PsychoPy to enable real-time respiratory tracking experiments. Participants follow a sinusoidal target dot with their breathing while receiving continuous visual biofeedback. The toolbox supports configurable experimental conditions including multi-frequency target waveforms and visuomotor perturbations (visual gain manipulation).

Full documentation | PyPI | Paper

Task Schematic

Task schematic

Task Screenshots

Range Calibration Baseline
Range Calibration Baseline
Countdown Tracking (good) Tracking (poor)
Countdown Tracking veridical Tracking bad

Installation

From PyPI (recommended)

pip install respyra

For post-session visualization (adds pandas and matplotlib):

pip install "respyra[vis]"

Development install

git clone https://github.com/embodied-computation-group/respyra.git
cd respyra

Create a virtual environment with Python 3.10:

# Windows (with Python Launcher)
py -3.10 -m venv .venv
.venv\Scripts\activate

# macOS / Linux
python3.10 -m venv .venv
source .venv/bin/activate

Install in editable mode with dev extras:

pip install -e ".[dev,vis]"

Running tests

pytest tests/ -v

PsychoPy and godirect are mocked at the sys.modules level so the test suite runs without hardware or heavy dependencies. See CONTRIBUTING.md for details on the mock strategy and adding new tests.

Requirements

  • Python 3.10 -- PsychoPy does not yet support 3.11+
  • Vernier Go Direct Respiration Belt (GDX-RB) -- required for hardware experiments; display demos run without a belt

Quick start

Run a no-hardware display demo to verify PsychoPy is working:

python -m respyra.demos.demo_display

With a belt connected, run the full experiment:

respyra-task

See the full documentation for detailed installation, quickstart, and user guide.

Running the experiment

python -m respyra.scripts.breath_tracking_task
# or, after pip install:
respyra-task

Session flow

  1. Belt connection -- BLE with automatic USB fallback (connects before PsychoPy to avoid Windows COM conflicts)
  2. Participant info dialog -- enter participant ID and session number
  3. Range calibration (15 s) -- comfortable deep breaths to establish breathing range, with percentile-based outlier rejection and sensor saturation detection
  4. Trial loop (per condition x N reps):
    • Baseline (10 s) -- breathe naturally
    • Countdown (3 s) -- target dot blends from current position into the target waveform
    • Tracking (30 s) -- follow the sinusoidal target dot with breathing
    • Feedback -- mean absolute tracking error for the trial
  5. Data saved to data/ as CSV (one row per sample, flushed incrementally)

Experimental conditions

Conditions are defined in respyra/configs/breath_tracking.py using composable frequency segments:

Condition Pattern Feedback gain
slow_steady 3 cycles at 0.1 Hz (30 s) 1.0 (veridical)
mixed_rhythm 3 cycles at 0.1 Hz + 1 cycle at 0.3 Hz 1.0 (veridical)
perturbed_slow 3 cycles at 0.1 Hz (30 s) 1.5 (amplified trace)

The feedback gain perturbation multiplies the displayed breathing trace around the participant's center, similar to cursor rotation in visuomotor reaching studies. The target dot, tracking error, and color feedback remain based on the true (unperturbed) signal -- only the visual trace is distorted.

Visual feedback

The target dot changes color based on real-time tracking error:

  • Graded mode (default) -- continuous green (good) to yellow to red (poor) using HSV interpolation
  • Binary mode -- yellow/red threshold
  • Trinary mode -- yellow/orange/red with two thresholds

Post-session visualization

python -m respyra.utils.vis.plot_session data/sub-01_ses-001_2026-02-24.csv
# or, after pip install:
respyra-plot data/sub-01_ses-001_2026-02-24.csv

Generates a 6-panel summary figure saved as {csv_stem}_summary.png:

  1. Full session force trace with target overlay
  2. Signed tracking error per trial
  3. Per-trial mean absolute error (bar chart)
  4. Error distribution by condition (box plot)
  5. Baseline calibration stability across trials
  6. Summary statistics (MAE, RMSE, per-condition breakdown)

Project structure

respyra/
  core/             Reusable modules
    breath_belt.py    Non-blocking belt I/O (threaded reader + queue)
    display.py        PsychoPy window, SignalTrace waveform renderer
    data_logger.py    Incremental CSV logging with crash resilience
    events.py         Keyboard input helpers
    target_generator.py  Sinusoidal target waveform from segment definitions
    gdx/              Vernier gdx wrapper (from godirect-examples, not on PyPI)
  configs/          Experiment parameters (no magic numbers in scripts)
  scripts/          Runnable experiment sessions
  demos/            Standalone single-feature test scripts
  utils/vis/        Post-session visualization
docs/               Sphinx documentation source
media/              Stimulus assets and icons
data/               Session output (gitignored)

Demos

python -m respyra.demos.demo_belt_connection   # Test belt connectivity (terminal only)
python -m respyra.demos.demo_display           # PsychoPy display with synthetic data
python -m respyra.demos.demo_threaded_belt     # Threaded belt queue-draining pattern

Documentation

Full documentation is available at embodied-computation-group.github.io/respyra, including:

Why the Vernier belt?

The Vernier Go Direct Respiration Belt (GDX-RB) was chosen because it is inexpensive, well-documented, and available worldwide through educational science suppliers. It provides wireless (BLE) respiratory force measurement out of the box, making it accessible to researchers and teaching labs without specialized biomedical equipment budgets. While not a research-grade instrument, it offers a reliable and practical solution for respiratory tracking and interoception research at low cost. For applications requiring more sophisticated monitoring (e.g., dual-band respiratory inductance plethysmography or spirometry), respyra's modular sensor interface is designed to accommodate alternative hardware.

Platform notes

Windows BLE: The Vernier belt's BLE scanner (Bleak) requires COM in MTA mode on the main thread. PsychoPy sets COM to STA on import. The framework handles this by connecting the belt before importing PsychoPy.

Linux: Requires udev rules for USB access. See the installation guide.

macOS: Works with both BLE and USB out of the box.

Contributing

Contributions are welcome! See CONTRIBUTING.md for development setup, testing, linting, and pull request guidelines.

License

MIT

The respyra/core/gdx/ module is derived from VernierST/godirect-examples and is licensed under the BSD 3-Clause License.

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

respyra-0.3.0.tar.gz (33.8 MB view details)

Uploaded Source

Built Distribution

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

respyra-0.3.0-py3-none-any.whl (61.7 kB view details)

Uploaded Python 3

File details

Details for the file respyra-0.3.0.tar.gz.

File metadata

  • Download URL: respyra-0.3.0.tar.gz
  • Upload date:
  • Size: 33.8 MB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for respyra-0.3.0.tar.gz
Algorithm Hash digest
SHA256 fe2497c6a0b2b8a38e679d75942e5aa3b195273e7f182b91c93b6f6ddc306d0f
MD5 627245cbf7abb6c5c27f2ab83bc47205
BLAKE2b-256 17fc6a44d24d1ebaf660dfb9644abdca8e1d2c26819e2683630f0c82f81d52aa

See more details on using hashes here.

Provenance

The following attestation bundles were made for respyra-0.3.0.tar.gz:

Publisher: publish.yml on embodied-computation-group/respyra

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

File details

Details for the file respyra-0.3.0-py3-none-any.whl.

File metadata

  • Download URL: respyra-0.3.0-py3-none-any.whl
  • Upload date:
  • Size: 61.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for respyra-0.3.0-py3-none-any.whl
Algorithm Hash digest
SHA256 e27b98e64f2ba4ab417130c6c0e498717d60e7abe71813eb1f87ae09a207c360
MD5 72d3278402ea3a0a110d6f975d170640
BLAKE2b-256 042835788948d78a8fbcb0812b6c840fecdb4f8036b945ea86a534256bab9c62

See more details on using hashes here.

Provenance

The following attestation bundles were made for respyra-0.3.0-py3-none-any.whl:

Publisher: publish.yml on embodied-computation-group/respyra

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