Segmentation models for spinal cord pathologies (tumors, lesions, SC itself) are typically trained and run on full volumes where the spinal cord occupies only a small fraction of the volume. sc-crop solves this by automatically detecting the spinal cord and cropping the volume tightly around it, so your segmentation model only ever sees the relevant region.
This reduces memory usage, speeds up inference, and often improves model accuracy by removing irrelevant background. The recommended workflow is:
- Preprocessing — crop all training images and labels around the detected spinal cord, adding a fixed security margin
- Training — train your segmentation model on the cropped volumes
- Inference — apply the same sc-crop preprocessing to new images, run your model, then optionally restore the segmentation to the original space
Works on both MRI and CT, across contrasts (T1, T2, MP2RAGE, DWI…), field strengths, and pathologies. Based on a YOLO26n model trained on multiple datasets covering cervical and lumbar spine. Available as a CLI tool and as an importable Python package.
Fresh dedicated environment (for testing):
conda create -n sc_crop python=3.13 && conda activate sc_crop
pip install git+https://github.com/ivadomed/sc-crop.git@v0.1.7Or into an existing environment:
pip install git+https://github.com/ivadomed/sc-crop.git@v0.1.7To pin in a requirements.txt: add sc-crop @ git+https://github.com/ivadomed/sc-crop.git@v0.1.7
For GPU inference (adds ultralytics):
pip install "sc-crop[yolo] @ git+https://github.com/ivadomed/sc-crop.git@v0.1.7"The environment where sc-crop was installed must be active for the
sc_cropcommand to be available.
mkdir ~/sc-crop-test && cd ~/sc-crop-test
curl -L https://github.com/ivadomed/sc-crop/releases/download/test-data/t2.nii.gz -o t2.nii.gzsc_crop -i t2.nii.gz -o t2_bbox.txtThe command prints a ready-to-use sct_crop_image command to crop the image using the detected coordinates (if SCT is installed).
Use the bbox file produced by the detect step to crop any volume — image, label, or both with identical boundaries:
sc_crop -i t2.nii.gz --bbox t2_bbox.txt -o t2_crop.nii.gzcurl -L https://github.com/ivadomed/sc-crop/releases/download/test-data/t2_seg.nii.gz -o t2_seg.nii.gz
sc_crop -i t2_seg.nii.gz --bbox t2_bbox.txt -o t2_seg_crop.nii.gzsc_crop -i t2.nii.gz -o t2_crop.nii.gz --detect-crop --pad-sup 50 --pad-inf 80 --pad-left 10 --pad-right 10 --pad-ant 15 --pad-post 15sc_crop -i t2.nii.gz -o t2_crop.nii.gz --detect-crop --pad-si 30 --pad-rl 10 --pad-ap 15Priority: individual (e.g. --pad-sup) > symmetric (e.g. --pad-si) > default.
Run sc_crop --help for all options.
Three functions cover all use cases:
| Function | Description |
|---|---|
detect(img_path) |
Runs the SC detector and returns the bounding box coordinates |
crop(img, bbox) |
Crops any NIfTI volume (image or label) to the bounding box |
uncrop(seg, bbox) |
Restores a segmentation from the cropped space back to the original full image space |
img = nib.load("t2.nii.gz")
bbox = detect(img) # detect the spinal cord, return bounding boxcrop_img = crop(nib.load("t2.nii.gz"), bbox) # also works on t2_seg.nii.gzfull_img = uncrop(crop_img, bbox) # restore to the original spaceThe margin around the detected spinal cord is adjustable per face (in mm).
# Adjust SI only (most common)
bbox = detect(img, pad_superior=50, pad_inferior=80)
# All 3 symmetric
bbox = detect(img, pad_si=30, pad_rl=10, pad_ap=15)
# Symmetric + override one face
bbox = detect(img, pad_si=30, pad_inferior=60)
# Full per-face control
bbox = detect(img, pad_superior=40, pad_inferior=60,
pad_left=10, pad_right=10,
pad_anterior=15, pad_posterior=15)Padding is always clamped to the image boundaries. Priority per face: individual > symmetric > default (sup=40, inf=60, left=right=10, ant=post=15).
sc-crop must be applied with identical padding around the detected spinal cord at both training and inference time. Different padding changes the crop boundaries and the distribution of what the model sees, which degrades performance. Pick your padding values once and keep them fixed throughout.
Crop all images and their labels with the same padding:
from sc_crop import detect, crop
import nibabel as nib
# Use the same padding for every subject
PAD = dict(pad_superior=40, pad_inferior=60, pad_left=10, pad_right=10,
pad_anterior=15, pad_posterior=15)
for subject in subjects:
bbox = detect(subject.image, **PAD)
crop_img = crop(nib.load(subject.image), bbox)
crop_label = crop(nib.load(subject.label), bbox)
nib.save(crop_img, subject.image_crop)
nib.save(crop_label, subject.label_crop)Train normally on *_crop.nii.gz images and labels.
Apply the same padding, run your model on the crop, then restore the segmentation to the original space:
from sc_crop import detect, crop, uncrop
import nibabel as nib
PAD = dict(pad_superior=40, pad_inferior=60, pad_left=10, pad_right=10,
pad_anterior=15, pad_posterior=15) # identical to training
bbox = detect("new_subject.nii.gz", **PAD)
crop_img = crop(nib.load("new_subject.nii.gz"), bbox)
seg_crop = my_model(crop_img) # run your segmentation model
seg_full = uncrop(seg_crop, bbox) # back to original space + affine
nib.save(seg_full, "new_subject_seg.nii.gz")sc-crop is applied before nnUNet's own preprocessing. The cropped images go directly into the nnUNet raw dataset folder; nnUNet never sees the original full volumes.
Step 1 — Populate nnUNet_raw/Dataset{ID}_{Name}/
nnUNet expects images in imagesTr/ named {case}_{0000}.nii.gz and labels in labelsTr/ named {case}.nii.gz. Crop image and label with the same bbox before saving there:
from sc_crop import detect, crop
import nibabel as nib
from pathlib import Path
RAW = Path("nnUNet_raw/Dataset001_MyTask")
PAD = dict(pad_superior=40, pad_inferior=60, pad_left=10, pad_right=10,
pad_anterior=15, pad_posterior=15)
for case_id, img_path, lbl_path in subjects:
bbox = detect(img_path, **PAD) # detect once
nib.save(crop(nib.load(img_path), bbox),
RAW / "imagesTr" / f"{case_id}_0000.nii.gz") # cropped image
nib.save(crop(nib.load(lbl_path), bbox),
RAW / "labelsTr" / f"{case_id}.nii.gz") # same bboxStep 2 — Run the standard nnUNet pipeline
nnUNetv2_plan_and_preprocess -d 001 --verify_dataset_integrity
nnUNetv2_train 001 3d_fullres 0Step 3 — Inference on a new image
The simplest approach uses the built-in segment_onnx / segment_pt functions (or their CLI equivalents) which handle detect → crop → infer → uncrop in one call. Install nnunet-onnx for the inference backend:
pip install "nnunet-onnx @ git+https://github.com/quentinRevillon/nnunet-onnx.git" torch nnunetv2 onnxscript onnxVia CLI:
# PyTorch checkpoint
sc-segment-pt -i new_subject.nii.gz -o seg.nii.gz \
--checkpoint /path/to/fold_0/checkpoint_best.pth
# ONNX model (no nnunetv2 needed at runtime)
sc-segment-onnx -i new_subject.nii.gz -o seg.nii.gz \
--model /path/to/model.onnxVia Python API:
from sc_crop import segment_pt, segment_onnx
import nibabel as nib
seg = segment_pt("new_subject.nii.gz", "/path/to/fold_0/checkpoint_best.pth")
seg = segment_onnx("new_subject.nii.gz", "/path/to/model.onnx")
nib.save(seg, "seg.nii.gz")To convert a checkpoint to ONNX (plans are embedded in the file — no external json needed):
python -m nnunet_onnx.export \
--checkpoint /path/to/fold_0/checkpoint_best.pth \
--output model.onnxThe example scripts are part of the repository — clone it to run them:
git clone https://github.com/ivadomed/sc-crop.git
cd sc-cropexamples/api_examples.py — Python API cookbook covering all usage patterns. If no image is provided, the SCT tutorial T2 is downloaded automatically:
python examples/api_examples.py # auto-download tutorial data
python examples/api_examples.py t2.nii.gz # use your own image
python examples/api_examples.py t2.nii.gz --ex 3 # run example 3 only (padding)| # | Pattern |
|---|---|
| 1 | Basic detect() + crop() |
| 2 | Multi-volume: detect once, crop image + label with the same bbox |
| 3 | Padding variants: default / symmetric / mixed / full individual |
| 4 | detect_and_crop() one-liner |
| 5 | Fake segmentation model + uncrop() round-trip |
| 6 | GPU inference (use_onnx=False) |
examples/infer_with_sc_crop.py — Full inference pipeline template with a built-in fake model (center-cylinder segmentation). If no input is provided, the SCT tutorial T2 is downloaded automatically:
python examples/infer_with_sc_crop.py # auto-download tutorial data
python examples/infer_with_sc_crop.py -i t2.nii.gz -o seg.nii.gzReplace fake_sc_segmentation() with your own model to use in production.
Python ≥ 3.8. Core dependencies installed automatically: nibabel, numpy, pillow, pyyaml, onnxruntime.
ultralytics is optional — required only for GPU/PyTorch inference (use_onnx=False) and the debug panel (debug=True). Install with the [yolo] extra above.
The detection model was trained using ivadomed/model_cropping_sc_contrast-agnostic_yolo. The link between package versions, model versions, and training runs is documented in VERSIONS.md. To publish a new model version, see the release procedure in VERSIONS.md.