Skip to main content

Load and analyze Delsys CSV exports as ``pysampled.Data`` time series.

Project description

delsys

src PyPI - Version Documentation Status GitHub license

Load Delsys CSV exports into Python as pysampled.Data time series.

delsys reads CSV files exported from EMGworks and Trigno Discover, normalizes their many per-format quirks (header layouts, sub-channel orderings, link-device asynchrony), resamples each channel to a configurable per-modality target rate, and groups the result into per-sensor modality bundles (EMG, EKG, IMU, FSR, VO2 Master, HR Strap, Analog) ready for analysis.

Installation

pip install delsys

For local development:

git clone https://github.com/praneethnamburi/delsys
pip install -e "./delsys[dev]"

Quickstart

import delsys

lf = delsys.Log("path/to/Trial.csv", sensor_map="path/to/delsys_channelmap.txt")

# Direct accessors — each returns a single aggregated `pysampled.Data`
# per modality (channels stacked across every sensor that has it), or
# `None` if no sensor does.
lf.emg                            # aggregate EMG
lf.ekg                            # aggregate EKG
lf.acc, lf.gyro                   # aggregate IMU
lf.fsr                            # aggregate FSR
lf.analog                         # aggregate Analog
lf.vo2master                      # VO2 Master link device (8 channels)
lf.hrstrap                        # HR Strap link device

# Side accessors return whole Sensor objects.
lf.left, lf.right, lf.center

# Filtered queries.
lf.find(modality="EMG", side="R")              # right-side EMG bundles
lf.find(location="Forearm")                    # any sensor at "Forearm"
lf.find(sensor_number=5)
lf.find(modality="EMG", as_="signal")          # raw per-channel Signal objects

# A typical EMG envelope pipeline (operate per-sensor).
for emg in lf.emg.split_by_signal_name():
    envelope = emg.process(amp_kind="envelope2")
    rms = emg.rms(envelope_sr=240)             # clean RMS amplitude pipeline

Cleaning ECG and motion artifact from EMG

Log.clean_emg_ekg_artifact() runs a three-stage pipeline (preprocess → ICA-based ECG suppression → ACC-guided motion regression with safety gates) over every EMG channel in the Log, splices the cleaned matrix back into lf.signals, and writes a multi-page PDF report next to the source CSV.

lf = delsys.Log("trial.csv")

# Default: in-place clean + PDF report.
result = lf.clean_emg_ekg_artifact()

# Inspect without mutating.
result = lf.clean_emg_ekg_artifact(in_place=False, generate_report=False)

# Splice only the ECG-cleaned variant back (e.g. when the motion stage
# is over-cleaning on this trial).
lf.clean_emg_ekg_artifact(splice_source="ekgonly")

Interactive helpers:

  • result.review() — cycle through every EMG channel, raw vs each cleaning variant, with arrow-key navigation.
  • result.review_components() — cycle through the ICA components plus their top three input contributors.

See tutorials/cleaning_emg_ekg_artifact.md for the full walkthrough.

See the full API reference at https://delsys.readthedocs.io.

Channelmap files (optional)

When you pass sensor_map="path/to/delsys_channelmap.txt" to Log(), that file labels each sensor number with a sensor type and a body-location tag. This lets you query by side (lf.find(side="R")) and by location (lf.find(location="Forearm")).

The format is one sensor per line, three fields separated by " - ":

Ch 1 - EMG - LBicep
Ch 2 - EMG - RBicep
Ch 11 - EKG - Chest
Ch 12 - Sync - Optitrack Recording Gate
Ch 19 - Quattro - LForearmExtensors (A-Index, B-Middle, C-Ring, D-Little)
Ch 21 - FSR - LFoot (1-Heel, 2-OuterEdge, 3-Ball, 4-Toe)
  • Field 1: any text whose last whitespace-token is the sensor's channel number (Ch 1, Channel 01, 1 all work).
  • Field 2: a type tag (free text — common values: EMG, Quattro, Snap, EKG, FSR, Sync).
  • Field 3: a location label. Its first character is interpreted as the side (L/R/C for left/right/center). Anything else still loads but won't match lf.find(side=...).

Trailing parenthetical notes are informational only — they remain in location but the parser doesn't extract sub-channel labels from them. Blank lines and lines without two " - " separators are silently skipped.

A more comprehensive reference file lives at examples/delsys_channelmap.txt.

Supported export formats

  • EMGworks
  • Trigno Discover 1.4.2
  • Trigno Discover 1.5.0
  • Trigno Discover 1.6.4 (with and without link devices)
  • Trigno Discover 1.7.0

Supported sensors

EMG (Avanti single, Duo, Quattro, Snap-Lead), EKG, ACC, GYRO, FSR, Analog, VO2 Master (link), HR Strap (link).

Scope and contributions

Supported sensor types are limited to those the maintainer has access to. Delsys ships other hardware (e.g. SmO2/Thb appear as stubs in TARGET_SR but are not exercised end-to-end). Contributions adding parsers and tests for additional sensors are very welcome.

If you'd like to contribute, the dev install above pulls pytest, black, and isort. Format with black and isort before opening a PR:

isort src/ tests/ scripts/
black src/ tests/ scripts/
pytest

License

Distributed under the MIT License. See LICENSE for details.

Contact

Praneeth Namburi

Project link: https://github.com/praneethnamburi/delsys

Acknowledgments

This package was developed as part of the ImmersionToolbox initiative at the MIT.nano Immersion Lab. Thanks to NCSOFT for supporting this initiative.

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

delsys-0.4.1.tar.gz (12.2 MB view details)

Uploaded Source

Built Distribution

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

delsys-0.4.1-py3-none-any.whl (68.5 kB view details)

Uploaded Python 3

File details

Details for the file delsys-0.4.1.tar.gz.

File metadata

  • Download URL: delsys-0.4.1.tar.gz
  • Upload date:
  • Size: 12.2 MB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: python-requests/2.31.0

File hashes

Hashes for delsys-0.4.1.tar.gz
Algorithm Hash digest
SHA256 702c7ae06853ad1f61cf64736dcd48d3f8d96e89a5a568ead5cee1f56cdf9ee0
MD5 accc7ffb118c6579bb2ea7b0e3411b42
BLAKE2b-256 b0c98120b830aecba040831bfa92c5f83a8ee18c7387c95aee64a0d3589c0ba1

See more details on using hashes here.

File details

Details for the file delsys-0.4.1-py3-none-any.whl.

File metadata

  • Download URL: delsys-0.4.1-py3-none-any.whl
  • Upload date:
  • Size: 68.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: python-requests/2.31.0

File hashes

Hashes for delsys-0.4.1-py3-none-any.whl
Algorithm Hash digest
SHA256 50afdd87af331bb69cb280fc27120b505071f6b803ed8fa3acf4b66be33bf724
MD5 1a114b8c733f17730c88dc3fe739407c
BLAKE2b-256 f0709d0471374fb287f51fd31d0fccb11720a5447319a26961121289dadb165a

See more details on using hashes here.

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