Napari plugin for the Pentachrome histology pipeline: VSI extraction, nnUNet inference, statistics.
Project description
pentachrome-plugin
Napari plugin for the Pentachrome histology pipeline. The public plugin exposes a single clinician-facing widget, Guided Analysis, that walks through the whole workflow in one pane:
- Extract — pull tissue-region TIFFs out of Olympus
.vsifiles. - Detect — run the trained nnUNet Epithelium / MultiStructure models on those images and load colorized masks back into the viewer.
- Measure — per-region statistics (thickness, composition, cell densities), with CSV export.
Each phase also exists as a separate advanced widget (VsiExtractorWidget,
NnUnetInferenceWidget, AnalysisWidget) used during development. These are not
registered in the public menu, see From source (development).
Source and issues: https://github.com/dtsilis7/Pentrachrome-Pipeline
Requirements: Windows, Python 3.10, napari >= 0.4.18, and the nnUNet model
weights. VSI reading uses bioio-bioformats (a plugin dependency) which
auto-downloads a JDK on first use — no Java install, no javabridge. Only the
model weights are installed separately (see below).
Installation (Windows, PowerShell)
A working install has three parts, in this order:
- A conda environment (Python 3.10) with NumPy<2
- The plugin itself (pulls in the JDK-free
bioio-bioformatsVSI reader) - The nnUNet model weights (downloaded separately)
⚠️
pip install pentachrome-plugingives you the UI and VSI extraction, but inference still needs the model weights (Step 3) in the same env. Installing through napari's plugin manager only does the pip part.
Step 1 — environment
Create a Python 3.10 env pinned to numpy<2 (the bioio-bioformats VSI reader is
pinned <2, and nnU-Net / the imaging stack share the 1.x ABI).
conda create -n pentachrome python=3.10 "numpy<2" -y
conda activate pentachrome
No JDK or javabridge to install. VSI reading goes through bioio-bioformats
(pulled in by Step 2), which provisions the Bio-Formats JVM via scyjava + cjdk and
auto-downloads a JDK the first time you extract (cached afterward).
Step 2 — the plugin
pip install pentachrome-plugin # also installs the bioio-bioformats VSI reader
pip install nnunetv2 # required for the Detect step
You can also install the plugin through napari's Plugins -> Install/Uninstall Plugins dialog (search "pentachrome"), but that only covers this step, you still need Step 1 and Step 3 in the same environment.
Step 3 — model weights
Download the nnUNet weights and point the widget at them — see Model weights below.
Verify
python -m napari
In napari, open Plugins -> Guided Analysis — the widget should load and show its Extract / Analyze / Statistics steps.
From source (development)
For working on the plugin itself, do Step 1 above, then install editable from a
checkout instead of from PyPI (cd into the plugin directory first, or pass the
absolute path):
conda activate pentachrome
cd "...\pentachrome_plugin"
pip install -e .
The three phase widgets are de-registered from the public menu. To open them
standalone during development, run the repo's dev_widgets.py — it docks the
Extractor / Inference / Statistics widgets as tabs:
python dev_widgets.py
Launch
conda activate pentachrome # or whichever env you installed into
python -m napari
In napari: Plugins -> Guided Analysis.
The per-phase sections below (nnUNet Inference, Mask Statistics, etc.) describe the underlying widgets, which the Guided Analysis pane drives end-to-end. Their Plugins -> ... menu references apply only to the dev widgets opened via
dev_widgets.py; end users reach the same functionality through Guided Analysis.
Model weights
The nnUNet weights (~900 MB) aren't bundled in the PyPI package. Download
[nnunet_results.zip]
from the releases page,
unzip it, and point the inference widget's nnUNet results field at the
extracted folder (the one containing Dataset001_Epithelium and
Dataset002_MultiStructure).
nnUNet Inference (Phase 2)
Requires nnunetv2 installed in the same environment (the nnUNetv2_predict CLI must be on PATH) — this is covered by Step 2 of Installation above.
Workflow:
- Load TIFFs into napari (e.g. via Phase 1's auto-load checkbox, or drag-and-drop).
- Open Plugins -> nnUNet Inference.
- Select one or more image layers in the list.
- Tick Epithelium, MultiStructure, or both.
- Set Output folder (where raw + colorized masks go) and nnUNet results (folder containing
Dataset001_EpitheliumandDataset002_MultiStructure). The results path auto-fills ifnnUNet_Training/nnUNet_results/resultsis found. - Pick Device (
cpuorcuda) and click Analyze.
Speed vs quality (important on CPU)
nnUNet inference on a laptop CPU is slow because every image goes through folds × mirror augmentations × sliding-window patches forward passes. With defaults that can be 20+ passes per image. The widget exposes three knobs in the Speed / quality group:
| Knob | Default | What it does |
|---|---|---|
| Epithelium folds | Fold 0 only |
Use 1 of the 5 trained folds for Dataset001. All 5 ensembled is best quality but ~5x slower. Dataset002 only has fold 0 trained, so it's always 1 fold. |
| Disable test-time mirroring | on | Passes --disable_tta. Skips the 4 mirror augmentations the model normally averages over. ~4x faster, small accuracy hit. |
| Sliding-window step | 0.5 |
Passes -step_size. Larger = fewer overlapping patches = faster but rougher tile borders. Try 0.7 for a middle ground. |
With all three defaults on a CPU laptop, one ROI tile should take a few minutes instead of 30+. Switch to All 5 folds + TTA on once you've moved to a GPU box.
Continuing from the extractor
The two widgets are linked through two small bridges, so you can run Extract -> Analyze in a single napari session without re-picking files:
- When the extractor auto-loads a TIFF as a viewer layer, it stashes the on-disk path on
layer.metadata['source_tiff']. The inference widget reads that during staging and copies the original file into_staging_input/rather than re-saving the in-memory array, important for 15k x 15k tiles. - When an extraction completes, the inference widget's "Use last extractor output" button pre-fills the output folder to
<extractor_output_root>/_inference, so masks land next to the per-VSI subfolders the extractor created.
Both bridges are in-process only (see _session.py); they reset when napari closes.
Outputs land in:
<output_folder>/
_staging_input/ # nnUNet-named (_0000.tif) copies of the selected layers
epithelium_raw/ # binary masks from Dataset001
epithelium_colored/ # RGB colorized masks (red epithelium)
multistructure_raw/ # 6-class masks from Dataset002
multistructure_colored/ # RGB colorized masks (Elastin/Collagen/Nuclei/Mucins/Membrane/Goblets)
Colorized masks are added to the viewer as RGB image layers when the run finishes.
nnUNet inference architecture
Same subprocess pattern as Phase 1. The widget never imports torch or nnUNetv2 directly; it spawns _inference_worker.py which:
- sets
nnUNet_resultsto the configured results dir, - calls
nnUNetv2_predictonce per enabled model (folds 0-4 for Epithelium, fold 0 for MultiStructure, matchingrun_inference.py), - colorizes the resulting integer masks with the palettes from
colorize_masks.py/compare_grid.py, - streams JSON-line events on stdout for the widget's progress bar and log.
How it works
- The widget itself never touches the JVM. When you click Extract ROIs, it spawns
_vsi_worker_bioio.pyas a separate Python process. - That worker starts the Bio-Formats JVM through
bioio-bioformats(scyjava + cjdk — no user JDK), loops over the VSI files usingTileMaskStitcher(vsi_handler/tile_mask_stitcher_bioio.py, which drivesloci.formats.ImageReaderdirectly viavsi_handler/_bioio_reader.py), writes numbered TIFFs into<output_root>/<vsi_basename>/, and emits JSON-line progress events on stdout. - The widget streams those events on a background thread and updates the progress bar / log without blocking the UI.
- When the worker exits, the JVM dies with it. The next extraction batch starts a fresh JVM in a fresh process - this avoids the "JVM cannot be restarted" pitfall during a long napari session.
Defaults
The parameter defaults mirror Processing_VSI_Files.py:
| Parameter | Default |
|---|---|
| Series | 6 |
| Tile width / height | 15000 |
| Threshold | 50 |
| Min ROI area | 150000 |
| Merge margin | 1000 |
| Extra crop margin | 100 |
Layout
pentachrome_plugin/
pyproject.toml
README.md
src/pentachrome_plugin/
__init__.py
napari.yaml # napari manifest
_session.py # in-process cross-widget state (extractor -> inference -> analysis)
_widget.py # VsiExtractorWidget (Phase 1)
_vsi_worker_bioio.py # VSI subprocess entrypoint (bioio-bioformats)
vsi_handler/ # bioio VSI reader (_bioio_reader.py) + tile/mask stitcher
_inference_widget.py # NnUnetInferenceWidget (Phase 2)
_inference_worker.py # nnUNet subprocess entrypoint
_analysis_widget.py # AnalysisWidget (Phase 3, in-process)
Phase 3 (Mask Statistics) lives alongside these and registers through napari.yaml.
Mask Statistics (Phase 3)
Pure in-process; no subprocess needed (no JVM, no torch). Reuses
EpithelialAnalysis/Analyzers/ (Descriptors.py, Thickness.py), so the
same metrics that fed the original region_summary.csv show up in the
widget.
Workflow:
- Run Phase 2 first so
epithelium_raw/andmultistructure_raw/exist. - Open Plugins -> Mask Statistics.
- Select one or more image layers in the list (their names must match the
mask filenames in
epithelium_raw//multistructure_raw/; if the inference widget staged them, that's already true). - Click Use last inference output (or browse).
- Tweak Pixel size, Region dilation, Min epithelium area if
needed (defaults match
Main.py). - Click Analyze.
For each detected epithelial region the widget reports:
| Column | What it is |
|---|---|
| Area (mm^2) | Region area after the 50 um dilation |
| Thickness mean/std (um) | Medial-axis thickness of (membrane within eroded region) U goblets U nuclei |
| Elastin / Collagen / Other % | Fraction of stained structure pixels, same definition as compute_structure_percentages |
| Mucin % | Mucin pixels as a fraction of the epithelium area (not of total structure pixels) |
| Nuclei / mm^2 and Goblets / mm^2 | Density per mm^2 of epithelium, goblet hyperplasia is a classic COPD readout |
| Nuclei (n), Goblets (n) | Raw counts inside the region |
A bold (all regions) row appended per image gives area-weighted means of the percentages / thickness and totals for the counts. Export CSV... saves the whole table (per-region rows + aggregate rows).
The elastin organization score (ElastinAnalyzer.determine_organized_region)
from Main.py is intentionally not yet exposed, it's much heavier (skan +
shapely + ROI polygons) and will land as a separate toggle.
Class isolation
A "Class isolation" group at the top of the widget lets you view a single class (or a combination) without rerunning anything:
- Pick a source layer (the original TIFF, not a colorized mask).
- Tick one or more of Elastin, Collagen, Nuclei, Mucins, Cell Membrane, Goblets, Epithelium.
- Click one of:
- Show as mask — adds a new layer that's white everywhere except the ticked classes, colored with the same palette as the inference widget.
- Show on original — adds a copy of the original image with all pixels outside the ticked classes turned white. Useful for sanity- checking the segmentation against the stain.
- Clear isolated layers removes everything this panel added in one go.
Masks are read on demand from the inference output folder; the original layer's pixels are taken from the viewer.
License
This project is licensed under the MIT License, see the LICENSE file for details.
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 pentachrome_plugin-0.5.0.tar.gz.
File metadata
- Download URL: pentachrome_plugin-0.5.0.tar.gz
- Upload date:
- Size: 80.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.10.20
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
1300118f4f240e483a4e1da3c42d5efb8918c8302323242dd33e7a9a87a34663
|
|
| MD5 |
7e9edb5aa6b11fe265d1e009efbede53
|
|
| BLAKE2b-256 |
1a72e8ca9f8ef4cf034302eb02679b0e9e459c3a20c3d8e2523d316f4f8f08f1
|
File details
Details for the file pentachrome_plugin-0.5.0-py3-none-any.whl.
File metadata
- Download URL: pentachrome_plugin-0.5.0-py3-none-any.whl
- Upload date:
- Size: 82.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.10.20
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7eef568c8065d02922651148621e829e05b23c1fcddd6d99a44ffcd329b52edb
|
|
| MD5 |
ca39ad9528da11b001dd7c03340d551d
|
|
| BLAKE2b-256 |
6296843a57c6fd70f31705607a4d4fd5b7fd32d75d8b22bf481265c92ccd3d08
|