AI-driven patient-to-clinical-trial matching: hybrid retrieval + LLM eligibility reasoning.
Project description
TrialMatchAI matches patients to the clinical trials they're eligible for. Given a patient — clinical notes, FHIR, Phenopacket, or OMOP — it returns a ranked shortlist of trials, each with a criterion-by-criterion explanation of why the patient does or doesn't qualify. It runs entirely on your own infrastructure on a single GPU server, so patient data never leaves your environment.
Install · Quickstart · How it works · Configuration · CLI
⚕️ For research and informational use only. TrialMatchAI is not medical advice, not a medical device, and must not replace review by qualified healthcare professionals.
Overview
TrialMatchAI runs in two halves: you build the matching system once, then match patients against it repeatably. Both halves are idempotent — they only do work that is not already done, and resume cleanly after an interruption.
- Build is the heavy, one-time setup (GPU). It prepares a searchable corpus of clinical trials — embeddings, biomedical entity annotations, and parsed eligibility constraints — and builds a local search index.
- Match ingests a patient, retrieves candidate trials, and reasons over each trial's eligibility criteria to produce a ranked, explained shortlist — rendered as a self-contained HTML report you can open in any browser.
Trials, models, indexes, and results all live on your own machine.
Requirements
- Python 3.11 (
pyproject.tomlrequires>=3.11,<3.12) piporuv— install from PyPI, or a source checkout for development- An NVIDIA GPU for model-backed matching and fine-tuning
- ~100 GB disk for datasets, model artifacts, search indexes, manifests, and run outputs
- OMOP vocabulary files, only if you want to build the concept-linking table locally
Installation
From PyPI (recommended)
pip install trialmatchai # or: uv pip install trialmatchai
trialmatchai --help
That installs the package and the trialmatchai CLI with its base dependencies.
For model-backed matching, add the runtime extras:
pip install "trialmatchai[llm,gpu,entity]" # full GPU runtime (Linux CUDA host)
pip install "trialmatchai[finetune]" # fine-tuning stack
| Extra | Adds |
|---|---|
entity |
biomedical entity extraction |
llm |
local embedding and language models |
gpu |
GPU inference stack (CUDA / Linux) |
finetune |
training dependencies for trialmatchai finetune |
From source (for development)
git clone https://github.com/cbib/TrialMatchAI.git
cd TrialMatchAI
uv sync # base; add --extra llm --extra gpu --extra entity
uv run trialmatchai --help
Calling the CLI: installed from PyPI, run it directly —
trialmatchai .... From auvsource checkout, prefix withuv run. The examples below use theuv runform; drop the prefix if you installed from PyPI.
Installing the package only gives you the CLI. Matching also needs the trial corpus, model artifacts, and search index — all produced by the build step below.
Quickstart
0. Configure the runtime
cp .env.example .env # optional local overrides
export HF_TOKEN=<token> # required for gated base models (one-time `hf auth login`)
1. Build the system — once
trialmatchai bootstrap-data # download the prepared corpus + model adapters
trialmatchai build # prepare the corpus + build the search index
trialmatchai build --status # see exactly what is built (and what isn't)
build fails fast if a GPU, an extra, or model access is missing, and resumes
from where it left off if interrupted. To use your own trials instead of the
bootstrapped corpus, put normalized JSON in data/trials_jsons/ and build will
prepare them.
To enable entity→concept linking, add --concepts (open vocabularies,
auto-downloaded) — and optionally an OMOP CONCEPT.csv for SNOMED/LOINC/RxNorm on
top:
trialmatchai build --concepts
trialmatchai build --concepts --concepts-csv data/omop/CONCEPT.csv --synonym-csv data/omop/CONCEPT_SYNONYM.csv
What gets fetched, and how
| Resource | How you get it | Automatic? |
|---|---|---|
| Trial corpus (prepared trials + criteria) | trialmatchai bootstrap-data (Zenodo) |
✅ automatic |
| Fine-tuned model adapters | trialmatchai bootstrap-data (Zenodo) |
✅ automatic |
| Fine-tuning datasets (only if you re-train) | trialmatchai bootstrap-data --finetune-data |
✅ automatic (opt-in) |
| Embedding model | downloaded on first use | ✅ automatic |
| Concept-linking vocabularies | trialmatchai build --concepts |
✅ automatic |
| Base language models | downloaded on first use | ⚠️ gated models need a one-time hf auth login + licence acceptance |
| OMOP clinical vocabulary (SNOMED/LOINC/RxNorm) | download CONCEPT.csv from OHDSI Athena |
❌ manual (licensed); linking works without it |
From scratch, that is two commands (bootstrap-data, then build --concepts)
after a one-time hf auth login. Everything else is pulled on demand.
2. Match patients — repeatably
e2e ingests the patient (format auto-detected) and matches end-to-end:
trialmatchai e2e --input data/patients/raw/patient-1.txt
trialmatchai e2e --input data/patients/raw/patient-1.fhir.json
trialmatchai e2e --input data/patients/omop_extract
Results land in results/<patient_id>/ (ranked trials + eligibility
explanations). Re-running skips patients already matched.
Each run also writes a self-contained HTML report you can open in a browser:
results/index.html is a front page over every matched patient, and
results/<patient_id>/report.html is one patient on its own. (Disable with
reporting.emit_html: false.)
Keeping trials current
Validate the installation, then fold new or changed ClinicalTrials.gov studies into the live index (incremental and idempotent — unchanged studies are skipped):
trialmatchai healthcheck # validate config, paths, deps, indexes
trialmatchai update-registry --since 2026-06-01 # one-shot
trialmatchai update-registry --watch --interval 86400 # server: update daily
The updater also drives from cron, a systemd timer, or GitHub Actions — see docs/registry-updater.md.
Manual / advanced control (the steps build and e2e wrap)
trialmatchai index --prepare # prepare + index from trials_jsons (what `build` runs)
trialmatchai import-patient --input patient.txt # stage a profile only
trialmatchai run # match already-staged profiles
trialmatchai trec --tracks "21 22" # benchmark: official TREC Clinical Trials eval
How it works
The flow below is the match path. The one-time build step produces the search index it queries — trial and criterion embeddings, entity annotations, and parsed eligibility constraints.
Patient data (text / FHIR / Phenopacket / OMOP)
│
▼
Interoperable importers → canonical patient profile
│
▼
Biomedical entity extraction + variant recognition → concept linking
│
▼
First-level trial retrieval (lexical + semantic) over the local index
│
▼
Multi-channel query fusion for broad candidate recall
│
▼
Criterion retrieval + reranking
│
▼
Constraint-aware criterion scoring
│
▼
Per-criterion eligibility reasoning
│
▼
Final ranking + explanations in results/
The reranking and eligibility-reasoning stages use language models served locally on the GPU. Every model component — entity extraction, reranker, and eligibility reasoning — is configurable and can be fine-tuned (see Configuration).
Patient inputs
The importer supports:
- free-text notes:
.txtand.md - GA4GH Phenopacket JSON
- HL7 FHIR R4 bundles, individual FHIR resources, NDJSON, and JSONL
- OMOP CDM extract folders with CSV or Parquet tables
Importers preserve provenance and unsupported source elements where possible. The
matching summary is rendered deterministically from the canonical patient profile;
raw patient files are not consumed directly by trialmatchai run. See
docs/interoperability.md for format details.
Data and storage
All storage is local and file-based — no external services. The search index
(data/search, with trial and criterion tables) and the concept-linking database
(data/concepts) are embedded databases. ClinicalTrials.gov records are normalized
to data/trials_jsons/<NCT_ID>.json, then prepared into one trial row and one row
per eligibility criterion (text, embeddings, entity annotations, and parsed
constraints). Tables carry both full-text and vector columns, so retrieval can run
in bm25, vector, or hybrid mode. Imported patients live under
data/patients/.
Configuration
Defaults live in src/trialmatchai/config/config.json; runtime overrides go in
.env or environment variables. Common overrides:
TRIALMATCHAI_SEARCH_MODE=hybrid # bm25 | vector | hybrid
TRIALMATCHAI_OUTPUT_DIR=results
TRIALMATCHAI_TRIALS_JSON_FOLDER=data/trials_jsons
TRIALMATCHAI_FIRST_LEVEL_MAX_TRIALS=1000
TRIALMATCHAI_LOG_JSON=1
The full override list is in .env.example.
To bring your own models, point entity_extraction.model_name,
model.reranker_adapter_path, and model.cot_adapter_path at your checkpoints, or
train them with trialmatchai finetune {cot,reranker,ner} — see the
fine-tuning guide.
CLI reference
There is a single entry point — trialmatchai — and every capability is a
subcommand. Under the hood they are all slices of one idempotent pipeline.
The unified pipeline (run any subset)
| Command | Purpose |
|---|---|
trialmatchai pipeline |
Run the whole pipeline, or any slice: --only / --from / --to / --skip / --force over the stages prepare → concepts → index → ingest → expand → match → eval. Finished work is skipped. See docs. |
The commands below are convenience presets over that pipeline.
Build the system
| Command | Purpose |
|---|---|
trialmatchai build |
Prepare the corpus (embeddings + entities) and build the search index — resumable, with --status |
trialmatchai bootstrap-data |
Download and extract the prepared corpus + model adapters |
trialmatchai build-concepts |
Build the concept table for entity normalization (optional, OMOP) |
trialmatchai update-registry |
Fetch changed ClinicalTrials.gov studies and update the index |
Match patients
| Command | Purpose |
|---|---|
trialmatchai e2e |
Ingest a patient and match end-to-end (idempotent, per-patient resume) |
trialmatchai import-patient |
Import text, FHIR, Phenopacket, or OMOP patient data into a profile |
trialmatchai run |
Match already-staged patient profiles |
trialmatchai report |
Render a self-contained HTML match report — one patient (--patient <id>) or a unified front page over all (--all) |
trialmatchai trec |
Benchmark: end-to-end evaluation on the official TREC Clinical Trials tracks |
Utility
| Command | Purpose |
|---|---|
trialmatchai healthcheck |
Validate config, paths, optional model deps, and search tables |
trialmatchai index |
Lower-level prepare/index of trial and criteria tables |
trialmatchai finetune |
Fine-tune entity extraction, reranker, or eligibility-reasoning models |
Deployment
The supported deployment is a single Python 3.11 GPU server or VM. The search
index and concept-linking database are local files under data/search and
data/concepts. The registry updater is designed for cron, systemd timers, or
GitHub Actions — see docs/registry-updater.md.
Development
uv sync
uv run ruff check .
uv run pytest
uv run pre-commit run --all-files # ruff + gitleaks secret scan + hygiene
uv run pip-audit --progress-spinner off --ignore-vuln CVE-2025-3000
Install the git hooks once so linting and secret scanning run on every commit:
uv run pre-commit install
Security
Never commit real credentials, private keys, datasets, models, local search data, run manifests, or results. Keep runtime values local:
cp .env.example .env
Artifact bootstrap supports optional SHA-256 verification through
TRIALMATCHAI_PROCESSED_TRIALS_SHA256, TRIALMATCHAI_MODELS_SHA256, and
TRIALMATCHAI_CRITERIA_PART_<N>_SHA256.
Dependency auditing currently ignores CVE-2025-3000 because the pinned inference
stack lists no fixed version; revisit when upgrading. See SECURITY.md
for the full policy.
Documentation
Deeper guides live on the documentation site:
- Pipeline & CLI — the stage registry,
--only/--skip/--from/--to/--force, ablation, and presets. - Architecture — first-level retrieval, constraint-aware ranking, and storage layout.
- Patient interoperability — text / FHIR / Phenopacket / OMOP importers.
- Fine-tuning & custom models — swap the entity, reranker, and reasoning models; training-data formats.
- Registry updater — keep trials current from ClinicalTrials.gov.
- API reference — the Python API.
Citation
If you use TrialMatchAI in your research, please cite the Nature Communications paper:
Abdallah, M. et al. TrialMatchAI: an end-to-end AI-powered clinical trial recommendation system to streamline patient-to-trial matching. Nature Communications 17, 4472 (2026). https://doi.org/10.1038/s41467-026-70509-w
@article{abdallah2026trialmatchai,
title = {TrialMatchAI: an end-to-end AI-powered clinical trial recommendation system to streamline patient-to-trial matching},
author = {Abdallah, Majd and Nakken, Sigve and Georges, Mikael and Bierkens, Mariska and Galvis, Johanna and Groppi, Alexis and Karkar, Slim and Meiqari, Lana and Rujano, Maria Alexandra and Canham, Steve and Dienstmann, Rodrigo and Fijneman, Remond and Hovig, Eivind and Meijer, Gerrit and Nikolski, Macha},
journal = {Nature Communications},
volume = {17},
pages = {4472},
year = {2026},
doi = {10.1038/s41467-026-70509-w},
url = {https://doi.org/10.1038/s41467-026-70509-w}
}
Support
- Email: abdallahmajd7@gmail.com
- Software archive (DOI): https://doi.org/10.5281/zenodo.18329084
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 trialmatchai-0.3.1.tar.gz.
File metadata
- Download URL: trialmatchai-0.3.1.tar.gz
- Upload date:
- Size: 318.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.11.26 {"installer":{"name":"uv","version":"0.11.26","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
801006722dedaa04f66e9a8bc32987823e4c55b2833e2d33ce0a37f0a4d76162
|
|
| MD5 |
8c83648fd3901201f91ab7266a464f9a
|
|
| BLAKE2b-256 |
ae458c2c271b1a5f8100a52d25d1825e5f46f7bec22f3b42b01135b8b94ddb37
|
File details
Details for the file trialmatchai-0.3.1-py3-none-any.whl.
File metadata
- Download URL: trialmatchai-0.3.1-py3-none-any.whl
- Upload date:
- Size: 279.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.11.26 {"installer":{"name":"uv","version":"0.11.26","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
fa2d76e8fa50821b345ca83d4e115303ad7215eebc7f9b679b9d78498ce1d0e5
|
|
| MD5 |
fea74b82472a1b6fc514416edd7243b3
|
|
| BLAKE2b-256 |
198f6777e3a1618b199d4642e5305224b999b71a984a9f7c907af1b0d2609589
|