Python library for longitudinal depression trajectories workflows.
Project description
Longitudinal Depression Trajectories Toolkit
About The Project
The Longitudinal Depression Trajectories Toolkit (LDT-Toolkit) initiative is designed for social, medical, and clinical researchers who work with repeated-measure data and require a stepping-stone path from raw cohort files to downstream modelling.
LDT-Toolkit is intended as a general toolkit for longitudinal depression trajectory exploration. It is currently using the Millennium Cohort Study (MCS) from CLS/UCL as a proof-of-concept case study.
The initiative delivers two interconnected components. First, ldt-toolkit (this repository) is the Python engine of tools and reproducible pipelines to accelerate exploration of longitudinal studies toward downstream modelling, while remaining fully usable in Python scripts or notebooks. Second, ldt is a fully interactive Go CLI with a no-code terminal interface for running and orchestrating the toolkit from start to finish.
The toolset supports two broad lines of exploration. Playground methods allow researchers to quickly iterate on their own datasets by running operations across data preparation, data preprocessing, and machine learning phases. With Presets, researchers can run stage-level reproducible pipelines for a given longitudinal study, and community contributions are encouraged so this can grow into a reusable collection of presets for preparing, preprocessing, and modelling longitudinal datasets.
And last, within the data preprocessing stage, we offer a brand-new and novel Trajectories Builder Playground: apply existing trajectory-building algorithms to your longitudinal datasets, or design and submit your own builders to help shape a community consensus around reproducible depression-trajectory construction.
Setup And Launch
[!IMPORTANT] Use both components:
ldt-toolkit(Python toolkit) andldt(Go CLI). The CLI is the primary no-code interface and is highly recommended.
Machine Requirements (R, Python, uv, Go)
Install the following first:
- Python
3.10to3.12AND ORuv - Go (for
ldtCLI) - R +
Rscript(required only when runningLCGA/GMMtools throughlcmm)
Quick checks:
python --version || python3 --version # Not necessary if you use `uv`
uv --version
go version
Rscript --version
Example installs by OS for uv go and R:
# macOS (Homebrew)
brew update
brew install uv go r
# Linux (Ubuntu/Debian)
sudo apt-get update
sudo apt-get install -y golang r-base
curl -LsSf https://astral.sh/uv/install.sh | sh
Install ldt-toolkit (PyPi Index)
uv add ldt-toolkit
# 👆 We recommend using `uv` as a state-of-the-art Python Package Manager
R dependencies for LCGA/GMM (lcmm)
If you plan to run LCGA/GMM, install R-side dependencies:
Rscript --vanilla -e "repos <- 'https://cloud.r-project.org'; required_packages <- c('lcmm'); missing <- setdiff(required_packages, rownames(installed.packages())); if (length(missing)) install.packages(missing, repos = repos) else message('All required R packages are already installed.')"
# 👇 If you have cloned the repo, you can also run:
Rscript --vanilla setup_R/install_requirements.R
Install the ldt GO CLI (No-code Terminal Interface)
Homebrew (macOS/Linux):
brew tap Longitudinal-Depression-Toolkit/homebrew-tap
brew install ldt
Voila! Now, you can run:
ldt
[!WARNING] The full
ldt-toolkitworkflow is not supported by default on Windows yet. As a workaround, run the toolkit inside a Docker Linux environment, or use a macOS-based machine. Please open an issue if you want to contribute Windows support.
Getting Started W/ the Toolset
(1) Generate Synthetic Data (Multi-Technique)
from pathlib import Path
from ldt.data_preparation import EventShockRecovery, MissingDataScenarios, TrendPatterns
out = Path("/path/to/your/project/data")
out.mkdir(parents=True, exist_ok=True)
TrendPatterns(n_samples=400, n_waves=5, random_state=7).prepare().to_csv(
out / "synthetic_trend_patterns.csv", index=False
)
EventShockRecovery().prepare(
n_samples=400,
n_waves=5,
random_state=7,
feature_cols=["depressive_score"],
shock_wave=3,
shock_mean=3.5,
recovery_rate=0.9,
noise_sd=0.8,
).to_csv(out / "synthetic_event_shock.csv", index=False)
MissingDataScenarios().prepare(
n_samples=400,
n_waves=5,
random_state=7,
feature_cols=["depressive_score", "sleep_score"],
mechanism="mixed",
missing_rate=0.20,
dropout_rate=0.10,
mar_strength=1.0,
).to_csv(out / "synthetic_with_missing.csv", index=False)
(2) Build Trajectories (Two Techniques) + ShowTable
from pathlib import Path
from ldt.data_preprocessing import BuildTrajectories, ShowTable
input_long = Path("/path/to/your/project/data/synthetic_trend_patterns.csv")
out = Path("/path/to/your/project/outputs")
out.mkdir(parents=True, exist_ok=True)
BuildTrajectories().fit_preprocess(
mode="from_scratch",
input_path=input_long,
output_path=out / "trajectories_dtw_kmeans.csv",
id_col="subject_id",
time_col="wave",
value_cols=["depressive_score"],
builder="dtw_kmeans",
n_trajectories=4,
)
BuildTrajectories().fit_preprocess(
mode="from_scratch",
input_path=input_long,
output_path=out / "trajectories_clusterMLD.csv",
id_col="subject_id",
time_col="wave",
value_cols=["depressive_score"],
builder="clusterMLD",
n_trajectories=4,
)
ShowTable().fit_preprocess(
input_path=out / "trajectories_clusterMLD.csv",
output_html=out / "trajectories_clusterMLD_report.html",
open_browser=False,
)
(3) End-to-End: Synthetic Data to Standard ML
from pathlib import Path
from ldt.data_preparation import MissingDataScenarios
from ldt.data_preprocessing import (
AggregateLongToCrossSectional,
BuildTrajectories,
CleanDataset,
CombineDatasetWithTrajectories,
MissingImputation,
)
from ldt.machine_learning import StandardMachineLearning
root = Path("/path/to/your/project")
raw_long = root / "data/synthetic_long_with_missing.csv"
clean_long = root / "outputs/long_clean.csv"
imputed_long = root / "outputs/long_imputed.csv"
trajectories = root / "outputs/trajectories.csv"
cross_sectional = root / "outputs/cross_sectional.csv"
model_ready = root / "outputs/model_ready.csv"
long_df = MissingDataScenarios().prepare(
n_samples=1200,
n_waves=6,
random_state=42,
feature_cols=["depressive_score", "sleep_score", "anxiety_score"],
mechanism="mixed",
missing_rate=0.20,
dropout_rate=0.15,
mar_strength=1.10,
)
raw_long.parent.mkdir(parents=True, exist_ok=True)
long_df.to_csv(raw_long, index=False)
CleanDataset().fit_preprocess(input_path=raw_long, output_path=clean_long)
MissingImputation().fit_preprocess(
technique="median_imputation",
input_path=clean_long,
output_path=imputed_long,
)
BuildTrajectories().fit_preprocess(
mode="from_scratch",
input_path=imputed_long,
output_path=trajectories,
id_col="subject_id",
time_col="wave",
value_cols=["depressive_score"],
builder="clusterMLD",
n_trajectories=4,
)
AggregateLongToCrossSectional().fit_preprocess(
input_path=imputed_long,
output_path=cross_sectional,
subject_id_col="subject_id",
numeric_columns=["depressive_score", "sleep_score", "anxiety_score", "age_baseline"],
numeric_agg="mean",
)
CombineDatasetWithTrajectories().fit_preprocess(
input_original_data_path=cross_sectional,
input_trajectories_data_path=trajectories,
output_path=model_ready,
original_id_col="subject_id",
trajectory_id_col="subject_id",
merge_type="left",
trajectory_columns=["trajectory_id", "trajectory_name"],
)
ml_result = StandardMachineLearning().fit_predict(
technique="run_experiment",
input_path=model_ready,
target_column="trajectory_id",
feature_columns="depressive_score,sleep_score,anxiety_score,age_baseline",
estimator_key="random_forest",
metric_keys="accuracy,f1_macro",
cv_folds=5,
validation_split="none",
multiclass_mode="multiclass",
random_seed=42,
output_dir=str(root / "outputs/standard_ml"),
)
print(ml_result["mean_score"], ml_result["report_path"])
But there is more: explore the full documentation for the complete tool and presets catalogue.
Citation
@software{Provost_Longitudinal_Depression_Trajectories_Toolkit,
author = {Provost, Simon and Branco, Bianca and Kwong, Alex},
title = {{Longitudinal Depression Trajectories Toolkit: Machine Learning for Longitudinal Studies}},
version = {0.0.1}
}
Use GitHub's "Cite this repository" for citation metadata updates.
License, Data, Security
- Access to Millennium Cohort Study data is governed by CLS/UCL data access rules.
- Review SECURITY.md for handling expectations.
- Licensed under the MIT License.
Special thanks to @charm.land for their amazing TUI framework!
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 ldt_toolkit-0.0.4.tar.gz.
File metadata
- Download URL: ldt_toolkit-0.0.4.tar.gz
- Upload date:
- Size: 232.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.5.11
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
52ed14b739b65c34bb79d9c7f06185d793323ac0b17d58f4ea0c8c06e9438a50
|
|
| MD5 |
6663676e14c18cbe2a99c2ffd89047be
|
|
| BLAKE2b-256 |
a6d1b27e4fac490ac6e5beef3fc9b79c868d75dddbf1cde3bfd7c5e493d3b5db
|
File details
Details for the file ldt_toolkit-0.0.4-py3-none-any.whl.
File metadata
- Download URL: ldt_toolkit-0.0.4-py3-none-any.whl
- Upload date:
- Size: 363.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.5.11
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
abb5fbfeba7565f6619916c8268d82db583ba2969024ce76ff2bf165e990cf3f
|
|
| MD5 |
186420ce4cb04b14966c82cd412ddf31
|
|
| BLAKE2b-256 |
22e1b970eeed4456b6cfe456bea08e4af2e49066b28372a3290b103273d4eac2
|