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.0.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.0-py3-none-any.whl (68.3 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: delsys-0.4.0.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.0.tar.gz
Algorithm Hash digest
SHA256 cce5044cb5a9b619e8345a2d304e30091fc2236824978358ac7782903c21a067
MD5 6d3d87957ca6b882dbbaccf3396c561a
BLAKE2b-256 5bb5623f2e9f303df77fc615d1f0854cda8a48f959e1f8d908675ff440deb398

See more details on using hashes here.

File details

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

File metadata

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

File hashes

Hashes for delsys-0.4.0-py3-none-any.whl
Algorithm Hash digest
SHA256 7092bb0ec2a169f444b4183dc66ed8676fc2a36a5bd0fa00b60c59599525e74d
MD5 560b9852e8d09e0737f8c7473137825f
BLAKE2b-256 83abbad8e3a4f5fee4aeb4f4e127d9cf4309be77619c77b65a3b892b04228f50

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