<!--
SPDX-FileCopyrightText: Copyright (c) 2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
SPDX-License-Identifier: Apache-2.0
-->
Clinical ASR Flywheel — Stage 3 (Eval)
⚠ Agent: read the Critical Workflow Rules section below before answering. This SKILL.md is self-contained —
,
, and
are pointers, not load-bearing. Answer methodology questions from this file directly; only invoke tools when the user explicitly asks to execute against a real manifest.
You are the
score-and-route stage. The user arrives with a NeMo-format
(either from
/digital-health-clinical-asr-build
or carried in from elsewhere). You transcribe it via the chosen ASR NIM, score four metrics, produce a five-section leaderboard, and read the decision tree to decide whether the user should advance to
/digital-health-clinical-asr-finetune
, loop back to
/digital-health-clinical-asr-build
, or stop and harden the eval.
This skill does not generate audio. If the manifest is missing or empty, send the user back to
/digital-health-clinical-asr-build
.
Audio leaves your environment — disclose this to the user before any clip is sent
This stage transmits each manifest row's WAV file plus its reference text to an external NVIDIA service. Surface this before invoking the first ASR call:
| Service | What gets sent | When |
|---|
| NVIDIA NVCF Parakeet/Nemotron ASR () | Every audio clip referenced by the manifest (raw PCM bytes), plus the reference transcript and the clinical-extension metadata for scoring | Step 3b, one call per manifest row |
The clips should be
synthetic audio generated by Stage 2 (Magpie TTS over a user-curated term list) — not real patient audio.
Do not pass real ASR recordings, real patient encounters, or any PHI through this skill. Scoring then runs locally (pure-Python WER/CER/KER/SER, or
if installed). The scoring step itself does not transmit anything; only the ASR step does.
Critical workflow rules (apply on every activation)
For methodology questions (leaderboard structure, KER definition, decision tree), answer from this file. Don't invoke tools, call other skills, or run scripts unless the user explicitly asks to execute against a real manifest. Surface these facts in any response:
- Off-ramp first. If the user is asking about something outside scoring, route and stop without running any workflow:
- ASR model-catalog selection / comparison / alternative NIMs →
- ASR auth (API keys, bearer tokens, function IDs) →
- ASR gRPC protocol, streaming, batching, chunking, retries →
- NIM deploy / / →
- NGC / Docker / NVIDIA Container Toolkit →
- No manifest yet →
/digital-health-clinical-asr-build
- Wants to fine-tune now with a known KER →
/digital-health-clinical-asr-finetune
- Default ASR NIM is
nvidia/parakeet-tdt-0.6b-v2
(NVCF function-id d3fe9151-442b-4204-a70d-5fcc597fd610
, offline gRPC). Env-var overrides: (leaderboard display name), (swap to a different hosted NIM — e.g. Whisper Large v3 while the Parakeet backend is faulting, or a fine-tuned NIM), (self-hosted gRPC; takes precedence). Echo the chosen NIM and the resolved function-id back before spending API credits.
- ASR transcription is inlined in Step 3b (NVCF gRPC +
riva.client.ASRService.offline_recognize
, same auth pattern as Stage 1). For deeper protocol/auth questions, alternative NIM catalogs, or self-hosted Riva NIM configuration, defer to .
- KER is the headline. Per-row check: the flagged words must appear in order, contiguous, adjacent in the normalized hypothesis. is a miss. Aggregate WER hides clinically dangerous failures; both are reported, KER is the gate.
- The by- split is the most informative single number in the leaderboard. The vs delta proves the SSML override pipeline is doing real work. Read it aloud to the user.
- Special-case routing. rows good, rows bad → pronunciation-coverage gap, not a model gap. Route back to
/digital-health-clinical-asr-build
Step 2d. Do NOT recommend /digital-health-clinical-asr-finetune
as a first response.
- Five-section leaderboard order. Headline (WER/CER/KER/SER) → KER by → KER by → KER by → Per-term KER worst-first. The by- section is mandatory; it is the proof the SSML pipeline works.
Purpose
Score a clinical-ASR manifest, produce a five-section KER leaderboard, and route the user via the post-eval decision tree. Methodology details (metric definitions, normalization, leaderboard order, special-case routing) live in Critical Workflow Rules above and Instructions below.
When to use this skill
Activate on user phrases like:
- "Score my ASR manifest"
- "What's the KER on Parakeet TDT v2?"
- "Run the eval on cycle-N"
- "Compare two ASR models on the clinical benchmark"
- "Generate the leaderboard"
- "I have a manifest.jsonl, how do I score it?"
- "Why is KER 0.4 when WER is 0.07?"
- "Should we fine-tune?" (this is the eval-side question — the post-eval decision tree lives in this skill)
Literal-keyword non-activation check — if the user's message contains any of
,
,
,
,
,
,
,
,
,
,
,
,
,
,
, or asks "which ASR model is best" / "compare models" / "vendor differences" —
do NOT activate the scoring workflow. Apply Critical Workflow Rule #1 above to route to the right sibling skill and stop. This applies even if the user mentions "KER" or "eval" alongside the keyword.
Prerequisites
- A NeMo-format manifest with the clinical extension fields (, , , , , ). The schema is documented in the build skill's
references/manifest-schema.md
.
- exported (Stage 1 prerequisite still applies).
- + installed (Stage 1 prerequisite). For self-hosted Riva NIM details, see Option B.
- Audio files actually present on disk — run the audio-existence pre-flight from the manifest-schema reference before spending API credits.
Instructions
3a. Pick the ASR NIM
Default:
nvidia/parakeet-tdt-0.6b-v2
via NVCF gRPC (offline), function-id
d3fe9151-442b-4204-a70d-5fcc597fd610
. NVIDIA's current English ASR recommendation — fastest/cheapest in the catalog, and supported in NeMo's stock SFT recipe so the Stage 3 baseline and a Stage 4 fine-tune ride the same model family.
Three runtime env-var override knobs (
for leaderboard display,
to swap to a different hosted NIM,
for self-hosted gRPC) plus the full alternate-NIM catalog (Parakeet TDT 1.1B, Parakeet CTC 1.1B, Whisper Large v3, Nemotron streaming) with function IDs and call-shape notes:
references/offline-asr-recipe.md
.
Echo the chosen NIM, the resolved function-id, and any env-var overrides to the user before spending API credits. A 200-row manifest on hosted Parakeet TDT v2 is cheap; an accidental run against the wrong model on a 1,000-row manifest is not.
3b. Transcribe
For each row in
, transcribe
and write
(one JSON object per row, JSONL or a JSON array — caller's choice):
json
{
"audio_filepath": "...",
"ref": "<row.text>",
"hyp": "<asr output>",
"term": "<row.term>",
"entity_category": "<row.entity_category>",
"ipa_source": "<row.ipa_source>",
"voice_id": "<row.voice_id>",
"noise_level": "<row.noise_level>",
"context_type": "<row.context_type>"
}
Recipe (full Python in
references/offline-asr-recipe.md
):
transcribe_manifest(api_key, manifest_path, out_path, language_code="en-US")
opens an offline gRPC stream to NVCF (or to
if set for self-hosted Riva), calls
riva.client.ASRService.offline_recognize
per row — sentences in a clinical manifest are ≤ 30 s so no streaming/batching needed — and writes the JSONL above. Same
shape as the Stage 1 setup smoke test. The agent harness passes
explicitly; the recipe reads the three env-var overrides (
,
,
) at the top so auditors see the knobs in one place.
Whisper fallback (when Parakeet's NVCF backend faults with
CUDA illegal-memory-access
from Triton) and
self-hosted Riva NIM (
ASR_ENDPOINT=localhost:50051
) env-var patterns: see
references/offline-asr-recipe.md
(§Whisper fallback, §Self-hosted Riva NIM).
Resilience knobs deferred to the user. If NVCF returns
mid-batch, the loop raises on that row; re-run from the failing row. Streaming/batching/retry-with-backoff are out of scope — see
.
3c. Score four metrics
For every row, compute:
| Metric | What it measures | Why we keep it |
|---|
| WER | Word error rate (Levenshtein on tokens, after normalization) | Industry standard; blunt instrument for clinical |
| CER | Character error rate | Catches near-misses on long compound names |
| KER ★ | Keyword error rate — did the flagged appear in the hypothesis (normalized, contiguous match)? | Headline clinical signal |
| SER | Sentence error rate (1 if any wrong, 0 if perfect) | Sanity bound; what the doctor experiences |
Normalization (apply to both and before all four metrics):
- Lowercase.
- NFKD-normalize (smart quotes → ASCII, etc.).
- Strip punctuation except hyphen.
- Collapse whitespace runs to a single space.
Inline scoring recipes —
/
/
/
/
/
(pure-Python, no
dependency): see
references/scoring-recipes.md
. Aggregate across rows by taking
for each metric.
Strict KER — term words must appear
in order, adjacent in the normalized hypothesis. This is conservative:
counts as a miss. That's the right call clinically — a downstream pharmacy lookup will fail on the misspelled token.
KER does not punish surrounding errors. A row where the term is correct and the rest of the sentence is garbage still scores KER=0; the WER on that row will surface the broader problem separately.
3d. Breakdowns + leaderboard
Write a five-section markdown leaderboard, in this order:
- Headline — overall WER, CER, KER, SER for the chosen model.
- KER by — drug vs procedure vs anatomy vs ... This is what the user actually cares about for deployment.
- KER by — the most informative single number in the leaderboard. The delta between and rows is the proof the SSML override pipeline is doing real work. Read this section aloud to the user.
- KER by — clinical environments are loud. rows are closer to reality than .
- Per-term KER (worst first) — these are your Stage 4 fine-tune targets.
A representative
split with the merriam-webster vs magpie_g2p delta interpretation:
references/scoring-recipes.md
§Representative ipa_source split. The delta tells the deployment story — if the user sees a wide gap and asks "should we fine-tune?", the answer is
not yet; route them back to
/digital-health-clinical-asr-build
's IPA QA pipeline (Stage 2d). See the decision tree below.
Decision tree (after eval)
Read the priority-category KER (drug KER for most clinical workflows, procedure KER for surgical workflows) and route:
| KER on priority category | Recommend |
|---|
| > 0.3 | /digital-health-clinical-asr-finetune
. Manifest is already NeMo-format-ready. Note: rows ≥ 100 is the minimum for a believable fine-tune signal; if the manifest is smaller, grow it first via /digital-health-clinical-asr-build
. |
| 0.1 – 0.3 | Either expand the term list (back to /digital-health-clinical-asr-build
with new domain terms — usually surfaces more failures cheaper than tuning) or fine-tune. On a first eval, expand. On a later eval where you've already grown the manifest, tune. |
| < 0.1 | Strong baseline. Don't tune yet — you'd be optimizing against a saturated metric. Push the eval harder: add voices, noise levels, contexts, adversarial terms. Loop back to /digital-health-clinical-asr-build
. |
Special case — rows score well but rows are bad. That's a pronunciation-hint coverage gap,
not a model gap. Route back to
/digital-health-clinical-asr-build
Step 2d (IPA QA review), not to
/digital-health-clinical-asr-finetune
. Fine-tuning over a TTS-pronunciation gap teaches the model to mis-recognize the model's own mistakes — the wrong fix.
Examples
Scenario A — first eval on a fresh cycle-1 manifest. User:
"I have with 200 clinical audio rows already, with and fields. How do I score it?" → Skip Stage 2 entirely. Run the audio-existence pre-flight. Pick
(default) and echo the choice + resolved function-id. Run the inlined Step 3b recipe (
). Score the four metrics. Produce the five-section leaderboard. Read the by-
split to the user. Apply the decision tree against drug KER.
Scenario B — interpreting a mixed result. User:
"Eval shows KER 0.05 on rows tagged but 0.40 on rows tagged . Should I fine-tune?" → No — this is the special case. The model is fine; the pronunciation hints aren't covering the long-tail terms. Route the user back to
/digital-health-clinical-asr-build
Step 2d to audition the
rows and append verified IPA to
pronunciation_overrides.csv
. Re-run Stage 3 after the rebuild before reconsidering Stage 4.
Artifacts produced
- — per-row transcription results with all clinical-extension fields preserved (the ASR joined to the manifest's and metadata)
- — per-row WER/CER/KER/SER scores
- — five-section markdown report
(File names are user-chosen; the names above are conventions the rest of this skill assumes.)
Troubleshooting
- "No manifest found" → user skipped Stage 2. Route to
/digital-health-clinical-asr-build
or confirm .
- All rows KER=1 → normalization mismatch between and . Apply the four normalization steps to both sides.
- All rows KER=0 but WER high → likely misaligned manifest (audio row mismatch). Spot-check a few pairs by hand.
- low, high → pronunciation-coverage gap. Route to
/digital-health-clinical-asr-build
Step 2d. Don't fine-tune — model isn't the problem.
- Both and high → real model gap. Stage 4 is the right route (manifest ≥ 100 rows).
- rows fine, balloons → robustness gap; expand noise diversity via
/digital-health-clinical-asr-build
.
- Riva-NIM and offline NeMo results diverge → Riva preprocessing / flags. Route to .
- on large manifests → retry after 30 s; slice + re-run dropped rows. Built-in backoff: .
Auth.__init__() got 'ssl_cert'
/ CUDA illegal-memory-access on Parakeet function ID: see references/offline-asr-recipe.md
(ssl_root_cert rename + §Whisper fallback).
Anything else: identify the upstream owner. ASR protocol / NIM deploy →
. Scoring → here.
Limitations
- English-only by default. Tokenization + normalization assume Latin script and en-US lexicon.
- Strict-contiguous KER is conservative. A near-miss like counts as a miss. That's intentional — pharmacy lookups fail on near-misses. Users wanting "soft" matching can switch to phoneme-level edit distance, which is a methodology extension, not a config tweak.
- One model per eval run. Comparing two models means running the eval twice and diffing the two files (or extending the recipe to write multi-model rows yourself).
- Hosted-only paths assumed. Self-hosted NIMs work but require first.
Next steps
- Forward (KER > 0.3, manifest ≥ 100 rows):
/digital-health-clinical-asr-finetune
.
- Back to build (KER 0.1–0.3 on first eval, or gap):
/digital-health-clinical-asr-build
.
- Stop (KER < 0.1): the eval is saturated. Harden it before declaring victory.
- Lateral for ASR protocol / auth / streaming / self-hosted NIM details: .
References
references/offline-asr-recipe.md
— full Step 3b Python recipe (, , ), function-ID catalog with call-shape notes, Whisper fallback, self-hosted Riva NIM setup
references/scoring-recipes.md
— pure-Python WER/CER/KER/SER scoring functions with the canonical 4-step normalization