Metadata-Version: 2.4
Name: pyimgano
Version: 0.6.38
Summary: Enterprise-Grade Visual Anomaly Detection Toolkit with 120+ registry models and 80+ operations
Author: PyImgAno Contributors
Maintainer: PyImgAno Contributors
License-Expression: MIT
Project-URL: Homepage, https://github.com/skygazer42/pyimgano
Project-URL: Documentation, https://skygazer42.github.io/pyimgano/
Project-URL: Repository, https://github.com/skygazer42/pyimgano
Project-URL: Bug Tracker, https://github.com/skygazer42/pyimgano/issues
Project-URL: Changelog, https://github.com/skygazer42/pyimgano/blob/main/CHANGELOG.md
Keywords: anomaly-detection,computer-vision,deep-learning,machine-learning,image-processing,industrial-inspection,defect-detection
Classifier: Development Status :: 4 - Beta
Classifier: Intended Audience :: Developers
Classifier: Intended Audience :: Science/Research
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Topic :: Scientific/Engineering :: Artificial Intelligence
Classifier: Topic :: Scientific/Engineering :: Image Recognition
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Requires-Python: >=3.9
Description-Content-Type: text/markdown
License-File: LICENSE
Requires-Dist: numpy>=1.19
Requires-Dist: scipy>=1.5.1
Requires-Dist: scikit-learn>=0.22.0
Requires-Dist: Pillow>=8.0.0
Requires-Dist: opencv-python>=4.5.0
Requires-Dist: defusedxml>=0.7.1
Requires-Dist: joblib>=1.0.0
Requires-Dist: tqdm>=4.0.0
Provides-Extra: torch
Requires-Dist: torch>=1.9.0; extra == "torch"
Requires-Dist: torchvision>=0.10.0; extra == "torch"
Provides-Extra: onnx
Requires-Dist: onnx>=1.12.0; extra == "onnx"
Requires-Dist: onnxruntime>=1.14.0; extra == "onnx"
Requires-Dist: onnxscript>=0.1.0; extra == "onnx"
Provides-Extra: openvino
Requires-Dist: openvino>=2023.0.0; extra == "openvino"
Provides-Extra: skimage
Requires-Dist: scikit-image>=0.18.0; extra == "skimage"
Provides-Extra: numba
Requires-Dist: numba>=0.51; extra == "numba"
Provides-Extra: diffusion
Requires-Dist: pyimgano[torch]; extra == "diffusion"
Requires-Dist: diffusers>=0.21.0; extra == "diffusion"
Requires-Dist: transformers>=4.30.0; extra == "diffusion"
Requires-Dist: accelerate>=0.20.0; extra == "diffusion"
Provides-Extra: viz
Requires-Dist: matplotlib>=3.3.0; extra == "viz"
Requires-Dist: seaborn>=0.11.0; extra == "viz"
Provides-Extra: anomalib
Requires-Dist: pyimgano[torch]; extra == "anomalib"
Requires-Dist: anomalib>=0.10.0; extra == "anomalib"
Provides-Extra: patchcore-inspection
Provides-Extra: faiss
Requires-Dist: faiss-cpu>=1.7.4; extra == "faiss"
Provides-Extra: clip
Requires-Dist: pyimgano[torch]; extra == "clip"
Requires-Dist: open_clip_torch>=2.0.0; extra == "clip"
Provides-Extra: mamba
Requires-Dist: pyimgano[torch]; extra == "mamba"
Requires-Dist: mamba-ssm>=2.0.0; extra == "mamba"
Provides-Extra: dev
Requires-Dist: pytest>=7.0.0; extra == "dev"
Requires-Dist: pytest-cov>=4.0.0; extra == "dev"
Requires-Dist: pytest-xdist>=3.0.0; extra == "dev"
Requires-Dist: black>=23.0.0; extra == "dev"
Requires-Dist: isort>=5.12.0; extra == "dev"
Requires-Dist: ruff>=0.1.0; extra == "dev"
Requires-Dist: tox>=4.0.0; extra == "dev"
Requires-Dist: build>=1.2.0; extra == "dev"
Requires-Dist: twine>=6.0.0; extra == "dev"
Requires-Dist: flake8>=6.0.0; extra == "dev"
Requires-Dist: mypy>=1.0.0; extra == "dev"
Requires-Dist: types-Pillow; extra == "dev"
Requires-Dist: types-setuptools; extra == "dev"
Requires-Dist: bandit[toml]>=1.7.0; extra == "dev"
Requires-Dist: safety>=3.0.0; extra == "dev"
Requires-Dist: pre-commit>=3.0.0; extra == "dev"
Provides-Extra: docs
Requires-Dist: sphinx>=5.0.0; extra == "docs"
Requires-Dist: sphinx-rtd-theme>=1.2.0; extra == "docs"
Requires-Dist: sphinxcontrib-napoleon>=0.7; extra == "docs"
Provides-Extra: all
Requires-Dist: pyimgano[backends,dev,diffusion,docs,numba,onnx,openvino,skimage,torch,viz]; extra == "all"
Provides-Extra: backends
Requires-Dist: pyimgano[anomalib,clip,faiss]; extra == "backends"
Dynamic: license-file

<p align="center">
  <img src="assets/banner.svg" alt="pyimgano banner" width="100%"/>
</p>

<p align="center">
  <strong>Production-oriented visual anomaly detection for industrial inspection.</strong><br/>
  <sub>Image-level + Pixel-level · 120+ Models · Train → Deploy in one pipeline</sub>
</p>

<p align="center">
  <a href="https://pypi.org/project/pyimgano/"><img src="https://img.shields.io/pypi/v/pyimgano.svg?style=flat-square&logo=pypi&logoColor=white&label=PyPI" alt="PyPI"/></a>
  <a href="https://pypi.org/project/pyimgano/"><img src="https://img.shields.io/pypi/pyversions/pyimgano.svg?style=flat-square&logo=python&logoColor=white" alt="Python"/></a>
  <a href="https://github.com/skygazer42/pyimgano/actions/workflows/ci.yml"><img src="https://img.shields.io/github/actions/workflow/status/skygazer42/pyimgano/ci.yml?style=flat-square&logo=githubactions&logoColor=white&label=CI" alt="CI"/></a>
  <a href="LICENSE"><img src="https://img.shields.io/badge/License-MIT-yellow.svg?style=flat-square" alt="License: MIT"/></a>
  <a href="https://github.com/psf/black"><img src="https://img.shields.io/badge/code%20style-black-000000.svg?style=flat-square" alt="Code style: black"/></a>
  <a href="https://pepy.tech/project/pyimgano"><img src="https://img.shields.io/pepy/dt/pyimgano?style=flat-square&logo=python&logoColor=white&label=Downloads" alt="Downloads"/></a>
</p>

<p align="center">
  <a href="#-installation">Installation</a> •
  <a href="#-quickstart">Quickstart</a> •
  <a href="#-model-zoo">Model Zoo</a> •
  <a href="#-api-cheatsheet">API</a> •
  <a href="#-benchmarking">Benchmarking</a> •
  <a href="#-documentation">Docs</a> •
  <a href="#-citation">Citation</a>
</p>

<p align="center">
  <b>Translations:</b> <a href="README_cn.md">中文</a> · <a href="README_ja.md">日本語</a> · <a href="README_ko.md">한국어</a>
</p>

---

## Why pyimgano?

Most anomaly detection libraries target either **research** (maximizing paper metrics) or **tabular data** (PyOD-style). `pyimgano` bridges the gap for teams that need to go from algorithm selection to **production deployment** on real industrial images:

| | Research libs | Tabular AD (PyOD…) | **pyimgano** |
|---|---|---|---|
| Image-level scoring | ✓ | ✓ | ✓ |
| Pixel-level anomaly maps | ✓ | — | ✓ |
| Industrial IO (numpy, tiling, formats) | — | — | ✓ |
| Deploy bundles (ONNX / OpenVINO) | partial | — | ✓ |
| Reproducible CLI + audit trail | — | — | ✓ |
| 120+ unified model registry | — | ✓ | ✓ |
| Synthetic anomaly generation | — | — | ✓ |

---

## ✨ Key Features

<p align="center">
  <img src="assets/features.svg" alt="Key Features" width="100%"/>
</p>

<table>
<tr>
<td width="50%">

**🧠 Unified Model Registry**
- 120+ registered entry points in a single `create_model()` call
- Classical (ECOD, KNN, IF, PCA…) + Deep (PatchCore, PaDiM, STFPM…) + VLM (WinCLIP, AnomalyDINO…)
- Lazy-loading registry — no heavy imports at startup

</td>
<td width="50%">

**🔍 Image + Pixel Anomaly Detection**
- Image-level anomaly scores
- Pixel-level anomaly maps with defect masks
- Connected-component regions with BBox and area
- ROI gating & binary morphology

</td>
</tr>
<tr>
<td>

**⚡ Production CLI Pipeline**
- `pyimgano-train` → `pyimgano-infer` → JSONL
- Reproducible runs with `report.json` + `per_image.jsonl`
- Deploy bundles for containers/servers
- Run comparison & leaderboard tables

</td>
<td>

**🚀 Deploy-ready Inference**
- Export to ONNX / OpenVINO / TorchScript
- Auditable `infer_config.json` + `calibration_card.json`
- High-resolution tiling for large images
- Explicit `ImageFormat` (RGB/BGR, U8/U16/F32, HWC/CHW)

</td>
</tr>
</table>

---

## 🏗️ Architecture

<p align="center">
  <img src="assets/architecture.svg" alt="Architecture Overview" width="100%"/>
</p>

<details>
<summary><b>Pipeline flow (Mermaid)</b></summary>

```mermaid
graph LR
    A[📁 Images / NumPy] --> B[Preprocessing]
    B --> C[Feature Extraction]
    C --> D[Model Zoo<br/>120+ Detectors]
    D --> E[Calibration]
    E --> F{Output}
    F --> G[🎯 Anomaly Score]
    F --> H[🗺️ Pixel Map]
    F --> I[🔲 Defect Regions]
    F --> J[📄 JSONL Report]

    style A fill:#1e293b,stroke:#38bdf8,color:#e2e8f0
    style B fill:#1e293b,stroke:#38bdf8,color:#e2e8f0
    style C fill:#1e293b,stroke:#818cf8,color:#e2e8f0
    style D fill:#1e293b,stroke:#f472b6,color:#e2e8f0
    style E fill:#1e293b,stroke:#fb923c,color:#e2e8f0
    style F fill:#0f172a,stroke:#34d399,color:#e2e8f0
    style G fill:#1e293b,stroke:#34d399,color:#e2e8f0
    style H fill:#1e293b,stroke:#34d399,color:#e2e8f0
    style I fill:#1e293b,stroke:#34d399,color:#e2e8f0
    style J fill:#1e293b,stroke:#34d399,color:#e2e8f0
```

</details>

---

## 📦 Installation

```bash
pip install pyimgano
```

<details>
<summary><b>Install from source (latest dev)</b></summary>

```bash
git clone https://github.com/skygazer42/pyimgano.git
cd pyimgano
pip install -e ".[dev]"
```

</details>

<details>
<summary><b>Optional extras</b></summary>

```bash
pip install "pyimgano[torch]"       # PyTorch deep backends + TorchScript export
pip install "pyimgano[onnx]"        # ONNX runtime
pip install "pyimgano[openvino]"    # OpenVINO runtime
pip install "pyimgano[skimage]"     # scikit-image baselines (SSIM/HOG/LBP/Gabor…)
pip install "pyimgano[numba]"       # Numba-accelerated baselines
pip install "pyimgano[viz]"         # matplotlib / seaborn plots
pip install "pyimgano[diffusion]"   # Diffusion-based methods
pip install "pyimgano[clip]"        # OpenCLIP backends
pip install "pyimgano[faiss]"       # Faster kNN for memory-bank methods
pip install "pyimgano[anomalib]"    # Anomalib checkpoint wrappers
pip install "pyimgano[backends]"    # clip + faiss + anomalib
pip install "pyimgano[all]"         # Everything
```

See `docs/OPTIONAL_DEPENDENCIES.md` for the full extras map.

</details>

---

## 🚀 Quickstart

### Python API — 5 lines to detect

```python
from pyimgano.models import create_model

detector = create_model("vision_patchcore", device="cuda")
detector.fit(train_paths)                           # normal/reference images
scores = detector.decision_function(test_paths)     # anomaly scores
```

<details>
<summary><b>Classical baseline (CPU, no pixel maps)</b></summary>

```python
from pyimgano.models import create_model
from pyimgano.utils import ImagePreprocessor

extractor = ImagePreprocessor(resize=(224, 224), output_tensor=False)
detector = create_model("vision_ecod", feature_extractor=extractor, contamination=0.1, n_jobs=-1)

detector.fit(train_paths)
scores = detector.decision_function(test_paths)
labels, confidence = detector.predict(test_paths, return_confidence=True)
```

</details>

<details>
<summary><b>NumPy-first industrial inference</b></summary>

```python
import numpy as np
from pyimgano.inference import calibrate_threshold, infer
from pyimgano.inputs import ImageFormat

train_frames = [np.zeros((64, 64, 3), dtype=np.uint8) for _ in range(8)]
detector.fit(train_frames)
calibrate_threshold(detector, train_frames, input_format=ImageFormat.RGB_U8_HWC, quantile=0.995)

results = infer(detector, test_frames, input_format=ImageFormat.RGB_U8_HWC, include_maps=True)
print(results[0].score, results[0].label)
```

</details>

### CLI — End-to-end pipeline

```bash
# 1. Quick demo (creates dataset + runs suite + exports tables)
pyimgano-demo

# 2. Environment check
pyimgano-doctor --suite industrial-v4

# 3. Train → export infer config
pyimgano-train \
  --config examples/configs/industrial_adapt_defects_fp40.json \
  --export-infer-config

# 4. Inference → JSONL + defect masks
pyimgano-infer \
  --infer-config /path/to/run_dir/artifacts/infer_config.json \
  --input /path/to/images \
  --defects \
  --save-masks /tmp/pyimgano_masks \
  --save-jsonl /tmp/pyimgano_results.jsonl
```

The JSONL output carries stable deployment metadata for downstream systems, including
`decision_summary` on each success record; Python best-effort batch integrations also
receive a `triage_summary` aggregate from `run_continue_on_error_inference(...)`.

<details>
<summary><b>One-off inference (no workbench)</b></summary>

```bash
pyimgano-infer \
  --model vision_patchcore \
  --preset industrial-balanced \
  --device cuda \
  --train-dir /path/to/normal/train_images \
  --calibration-quantile 0.995 \
  --input /path/to/images \
  --include-maps
```

Pass extra kwargs via `--model-kwargs '{"backbone":"wide_resnet50","coreset_sampling_ratio":0.1}'`.

</details>

<details>
<summary><b>Model discovery</b></summary>

```bash
pyimgano --list models
pyimgano list models
pyimgano -- list models --json
pyimgano --list models --family patchcore
pyimgano --list preprocessing --deployable-only
pyim --list models
pyim --list models --family patchcore
pyim --list models --year 2021 --type deep-vision
pyim --list models --type flow-based
pyim --list preprocessing --deployable-only
```

`pyimgano` is now the umbrella CLI. `pyimgano list ...` and `pyimgano -- list ...`
map to the same discovery flow as `pyimgano --list ...`. `pyim` remains the
shorter discovery alias.

</details>

### Shortest audited operator path

If you want the shortest “train -> validate -> gate” path for production-style
handoff, the root CLI now exposes it directly:

```bash
pyimgano --help
pyimgano -- list preprocessing --deployable-only
pyimgano train --config examples/configs/industrial_adapt_audited.json --export-infer-config --export-deploy-bundle
pyimgano validate-infer-config runs/<run_dir>/deploy_bundle/infer_config.json
pyimgano runs quality runs/<run_dir> --require-status audited --json
pyimgano runs acceptance runs/<run_dir> --require-status audited --check-bundle-hashes --json
pyimgano weights audit-bundle runs/<run_dir>/deploy_bundle --check-hashes --json
```

That flow uses the checked-in
[`industrial_adapt_audited.json`](examples/configs/industrial_adapt_audited.json)
example and is described in more detail in
[`docs/INDUSTRIAL_FASTPATH.md`](docs/INDUSTRIAL_FASTPATH.md).

For reproducible benchmark reporting, pair that operator loop with the built-in
official preset discovery and publication gate:

```bash
pyimgano benchmark --list-official-configs
pyimgano benchmark --official-config-info official_mvtec_industrial_v4_cpu_offline.json --json
pyimgano runs acceptance /path/to/suite_export --json
pyimgano runs publication /path/to/suite_export --json
```

That keeps the shortest industrial fast-path and the publication path visible
from the same umbrella CLI entrypoint.

If your deploy bundle carries `model_card.json` and `weights_manifest.json`,
`pyimgano weights audit-bundle ...` gives you a single delivery gate for both.
If you want one aggregated release check instead of calling the low-level
validators individually, use `pyimgano runs acceptance ...`; it now auto-routes
run directories and suite publication exports through the matching gate.
For suite exports, the publication gate now also enforces benchmark provenance
(`benchmark_config.source` + `sha256`), `evaluation_contract`, and `citation`
instead of trusting a manually stamped `publication_ready=true`.
It also expects audit refs back to `report.json`, `config.json`, and
`environment.json`, plus matching sha256 digests for those files, so exported
leaderboards remain traceable to the saved run. The exported leaderboard tables
themselves now carry recorded sha256 digests too, so `leaderboard.csv`,
`best_by_baseline.csv`, `skipped.csv` (and markdown variants when exported)
are verified before a suite export is treated as publication-ready.

---

## 🧠 Model Zoo

`pyimgano` ships **120+ registered model entry points** spanning classical statistics to modern vision-language models.

### Algorithm Capability Matrix

<p align="center">
  <img src="assets/algo_matrix.svg" alt="Algorithm Capability Matrix" width="100%"/>
</p>

### Recommended Baselines

| Goal | Model | Tags | Notes |
|:-----|:------|:-----|:------|
| 🎯 Strong pixel localization | `vision_patchcore` | `numpy,pixel_map` | Best default for MVTec/VisA-style data |
| 🛡️ Robust to noisy normals | `vision_softpatch` | `numpy,pixel_map` | Filters outlier patches in memory bank |
| 🪶 Lightweight pixel baseline | `vision_padim` / `vision_spade` | `numpy,pixel_map` | Simpler, easier to tune |
| 📸 Few-shot / small normal set | `vision_anomalydino` | `numpy,pixel_map` | DINOv2-based, downloads weights on first run |
| 💻 CPU-only / precomputed | `vision_ecod` / `vision_copod` | `classical` | Fast, parameter-free, score-only |
| 🔌 Anomalib integration | `vision_*_anomalib` | `deep` | Requires `pyimgano[anomalib]` |

<details>
<summary><b>Full algorithm taxonomy</b></summary>

| Category | Type | Algorithms | Count |
|:---------|:-----|:-----------|------:|
| **Statistical** | Density / Distribution | ECOD, COPOD, HBOS, KDE, GMM, MCD, QMCD | 24 |
| **Proximity** | Neighbor-based | KNN, LOF, COF, LDOF, ODIN, INNE, LoOP | 18 |
| **Subspace** | Projection | PCA, KPCA, SOD, ROD, LODA | 8 |
| **Tree / Graph** | Isolation | Isolation Forest, RRCF, HST, MST, R-Graph | 10 |
| **Ensemble** | Multi-detector | Feature Bagging, LSCP, SUOD, Score Ensemble | 6 |
| **Memory Bank** | Patch-based deep | PatchCore, PaDiM, SPADE, SoftPatch, MemSeg | 12 |
| **Student-Teacher** | Knowledge distill | STFPM, Reverse Distillation, EfficientAD, AST | 8 |
| **Flow-based** | Normalizing flows | FastFlow, CFlow, CS-Flow | 6 |
| **Reconstruction** | AE / VAE | AE, VAE, VQ-VAE, DRAEM, DFM | 10 |
| **VLM / Zero-shot** | Vision-Language | WinCLIP, AnomalyDINO, PromptAD | 6 |
| **Template** | Pixel statistics | SSIM, NCC, Phase correlation, Pixel Gaussian | 8 |
| **Industrial** | Pipeline wrappers | ResNet18, ONNX, TorchScript pipelines | 14+ |

</details>

> 📖 See `docs/ALGORITHM_SELECTION_GUIDE.md` for a detailed decision flowchart.

---

## 📋 API Cheatsheet

```python
from pyimgano.models import create_model, list_models

# ╭──────────────────────────────────────────────╮
# │  Discovery                                   │
# ╰──────────────────────────────────────────────╯
list_models()                              # all registered names
list_models(tags=["vision", "pixel_map"])  # filter by tags

# ╭──────────────────────────────────────────────╮
# │  Core detector API                           │
# ╰──────────────────────────────────────────────╯
detector = create_model("vision_patchcore", device="cuda", contamination=0.1)

detector.fit(X_train)                      # train on normal images
scores = detector.decision_function(X)     # → np.ndarray of anomaly scores
labels = detector.predict(X)               # → 0 (normal) / 1 (anomaly)
labels, conf = detector.predict(X, return_confidence=True)

# ╭──────────────────────────────────────────────╮
# │  Pixel-level (models with pixel_map tag)     │
# ╰──────────────────────────────────────────────╯
anomaly_map = detector.predict_anomaly_map(image)  # → H×W float array

# ╭──────────────────────────────────────────────╮
# │  Industrial inference                        │
# ╰──────────────────────────────────────────────╯
from pyimgano.inference import infer, calibrate_threshold
from pyimgano.inputs import ImageFormat

calibrate_threshold(detector, X_train, input_format=ImageFormat.RGB_U8_HWC, quantile=0.995)
results = infer(detector, X_test, input_format=ImageFormat.RGB_U8_HWC, include_maps=True)
# results[i].score, results[i].label, results[i].anomaly_map

# ╭──────────────────────────────────────────────╮
# │  Evaluation                                  │
# ╰──────────────────────────────────────────────╯
from pyimgano import evaluate_detector, compute_auroc

metrics = evaluate_detector(detector, X_test, y_test)  # full report
auroc = compute_auroc(y_true, scores)                   # single metric
```

---

## 📊 Benchmarking

Run curated **baseline suites** with a single command and get aggregated leaderboard tables:

```bash
# Discover suites
pyimgano-benchmark --list-suites
pyimgano-benchmark --suite-info industrial-v4 --json

# Run a suite
pyimgano-benchmark \
  --dataset mvtec --root /path/to/mvtec_ad --category bottle \
  --suite industrial-v4 --device cpu --no-pretrained \
  --suite-export both --output-dir /tmp/suite_run

# Optional: small grid search
pyimgano-benchmark \
  --dataset mvtec --root /path/to/mvtec_ad --category bottle \
  --suite industrial-v4 --suite-sweep industrial-template-small \
  --suite-sweep-max-variants 1 --suite-export csv \
  --output-dir /tmp/suite_sweep
```

<details>
<summary><b>Run comparison</b></summary>

```bash
pyimgano-runs list --root runs
pyimgano-runs list --root runs --kind robustness --same-robustness-protocol-as runs/run_a --json
pyimgano-runs compare runs/run_a runs/run_b --json
pyimgano-runs compare runs/run_a runs/run_b --baseline runs/run_a --require-same-split --json
pyimgano-runs compare runs/run_a runs/run_b --baseline runs/run_a --require-same-robustness-protocol --json
pyimgano-runs acceptance /path/to/suite_export --json
pyimgano-runs publication /path/to/suite_export --json
```

</details>

> Some suite entries are **optional** and are skipped if extras are missing (with hints like `pip install 'pyimgano[torch]'`).
> For reproducible publications, use built-in official presets via `--config official_mvtec_industrial_v4_cpu_offline.json`.

---

## 🎨 Synthetic Anomaly Generation

When real defects are scarce, generate controlled synthetic anomalies for smoke tests, robustness checks, and pixel-mask pipelines:

```bash
pyimgano-synthesize \
  --in-dir /path/to/normal_images \
  --out-root ./out_synth_dataset \
  --category synthetic_demo \
  --presets scratch stain tape edge_wear \
  --roi-mask /path/to/roi_mask.png \
  --blend alpha --alpha 0.9 \
  --n-train 50 --n-test-normal 20 --n-test-anomaly 20 \
  --seed 0
```

Supported presets: `scratch` · `stain` · `tape` · `edge_wear` · Perlin noise · CutPaste · custom

> 📖 Guide: `docs/SYNTHETIC_ANOMALY_GENERATION.md`

---

## 🏭 Industrial Outputs (Defects)

When `--defects` is enabled, `pyimgano-infer` derives structured defect information from the anomaly map:

```mermaid
graph LR
    A[Anomaly Map] --> B[Threshold]
    B --> C[Binary Mask]
    C --> D[Morphology<br/>open / close / fill]
    D --> E[Connected Components]
    E --> F[Defect Regions]

    F --> G["area · bbox · score"]
    F --> H[ROI Gating]

    style A fill:#1e293b,stroke:#f472b6,color:#e2e8f0
    style C fill:#1e293b,stroke:#818cf8,color:#e2e8f0
    style F fill:#1e293b,stroke:#34d399,color:#e2e8f0
```

- **Binary defect mask** — optional artifact output
- **Connected-component regions** — area, bbox, per-region score
- **ROI gating** — only flag defects inside the region of interest
- **Morphology** — open/close/fill holes to reduce noise
- **Stable triage metadata** — per-record `decision_summary` for review routing and low-confidence rejection handling

> 📖 Guides: `docs/INDUSTRIAL_INFERENCE.md` · `docs/FALSE_POSITIVE_DEBUGGING.md`

---

## 🔧 CLI Reference

| Command | Purpose |
|:--------|:--------|
| `pyimgano` | Top-level umbrella CLI (`pyimgano --help`, `pyimgano list models`) |
| `pyimgano-demo` | Quick end-to-end demo |
| `pyimgano-doctor` | Environment & dependency check |
| `pyimgano-train` | Train + export `infer_config.json` |
| `pyimgano-infer` | Inference → JSONL + masks |
| `pyimgano-benchmark` | Suite / sweep benchmarking |
| `pyimgano-synthesize` | Synthetic anomaly generation |
| `pyimgano-runs` | Run listing & comparison |
| `pyimgano-weights` | Weight manifest & model cards |
| `pyimgano-features` | Feature extraction utilities |
| `pyimgano-datasets` | Dataset management |
| `pyimgano-defects` | Defect post-processing |
| `pyim` | Unified discovery (`pyim --list models`) |

> 📖 Full reference: `docs/CLI_REFERENCE.md`

`pyimgano-runs quality`, `pyimgano-runs acceptance`, and `pyimgano-runs publication` now expose structured
trust metadata as well:

- `trust_summary` for saved run artifacts
- `trust_signals` for suite publication bundles

This makes it easier to gate CI/release automation on auditable machine-readable
signals instead of parsing free-form warnings.

---

## ⚖️ Weights & Cache Policy

- `pyimgano` does **not** ship model weights inside the wheel.
- Weights are cached by upstream libraries (torchvision / OpenCLIP / HuggingFace).
- Set cache env vars on servers: `TORCH_HOME`, `HF_HOME`, `XDG_CACHE_HOME`

```bash
# Auditable weight management
pyimgano-weights template manifest > ./weights_manifest.json
pyimgano-weights validate ./weights_manifest.json --check-files --json
```

---

## 📚 Documentation

<table>
<tr>
<td width="33%">

**Getting Started**
- [`QUICKSTART.md`](docs/QUICKSTART.md)
- [`WORKBENCH.md`](docs/WORKBENCH.md)
- [`CLI_REFERENCE.md`](docs/CLI_REFERENCE.md)
- [`STABILITY.md`](docs/STABILITY.md)

</td>
<td width="33%">

**Industrial / Production**
- [`INDUSTRIAL_INFERENCE.md`](docs/INDUSTRIAL_INFERENCE.md)
- [`INDUSTRIAL_FASTPATH.md`](docs/INDUSTRIAL_FASTPATH.md)
- Example config: [`industrial_adapt_audited.json`](examples/configs/industrial_adapt_audited.json)
- [`ROBUSTNESS_BENCHMARK.md`](docs/ROBUSTNESS_BENCHMARK.md)
- [`FALSE_POSITIVE_DEBUGGING.md`](docs/FALSE_POSITIVE_DEBUGGING.md)

</td>
<td width="33%">

**Reference**
- [`MODEL_INDEX.md`](docs/MODEL_INDEX.md)
- [`ALGORITHM_SELECTION_GUIDE.md`](docs/ALGORITHM_SELECTION_GUIDE.md)
- [`EVALUATION_AND_BENCHMARK.md`](docs/EVALUATION_AND_BENCHMARK.md)
- [`PUBLISHING.md`](docs/PUBLISHING.md)

</td>
</tr>
</table>

---

## 🤝 Contributing

We welcome contributions! Please see:

- [`CONTRIBUTING.md`](CONTRIBUTING.md) — development workflow
- [`CODE_OF_CONDUCT.md`](CODE_OF_CONDUCT.md)
- [`SECURITY.md`](SECURITY.md)

---

## 📄 License

[MIT](LICENSE)

---

## 📝 Citation

GitHub citation metadata is provided via [`CITATION.cff`](CITATION.cff).

```bibtex
@software{pyimgano2026,
  author    = {PyImgAno Contributors},
  title     = {pyimgano: Production-oriented Visual Anomaly Detection},
  year      = {2026},
  url       = {https://github.com/skygazer42/pyimgano}
}
```

---

<p align="center">
  <sub>Made with care for industrial inspection teams.</sub><br/>
  <a href="https://github.com/skygazer42/pyimgano">⭐ Star us on GitHub</a>
</p>
