@records-fhir/validator
v0.1.7
Published
Pure-TypeScript FHIR R4/R5/R6 validation engine. Validates resources against StructureDefinitions, terminology, references, and custom rules with no database required.
Maintainers
Readme
@records-fhir/validator
Pure TypeScript FHIR validation engine for R4, R5, and R6 resources.
The package validates FHIR resources against StructureDefinitions, FHIRPath constraints, terminology bindings, references, and optional custom rules without requiring a database or JVM. Records can wire database-backed profile and rule sources through dependency injection, while standalone consumers can run from local FHIR packages or optional bundled profiles.
This package is the open-source validator surface for medvertical/records-fhir-validator. The Records product itself is commercial closed source; it is not part of this package.
Repository Boundary
This package is designed to be published from the separate public repository
medvertical/records-fhir-validator.
Included in the public boundary:
- Pure TypeScript validator runtime.
- Validation orchestration and executors.
- StructureDefinition loading from filesystem/FHIR package sources.
- FHIRPath invariant execution.
- Terminology, reference, slicing, extension, metadata, and Bundle validators.
- OperationOutcome conversion helpers.
- Optional dependency injection hooks for host applications.
Excluded from the public boundary:
- Records web application.
- Records Express API routes and controllers.
- Records database schema, migrations, repositories, and storage services.
- Governance evidence reports and commercial workflow code.
- Customer-specific integrations, authentication, and deployment configuration.
Install
npm install @records-fhir/validator @records-fhir/validation-typesFor offline profile validation, install the optional profile bundle:
npm install @records-fhir/bundled-profilesPinning
Three valid ways to pin the GitHub Action that wraps this validator, depending on your trade-off between freshness and stability:
| Goal | Pin in uses: | Notes |
|---|---|---|
| Always-latest within current major | medvertical/records-fhir-validator@v0 | Force-moved on every stable release; never advances onto a prerelease |
| Specific minor/patch (recommended for production CI) | medvertical/[email protected] | Immutable once published |
| Bit-exact reproducibility | medvertical/records-fhir-validator@<commit-sha> | For audit / forensic builds |
The validator-v<semver> tag you may see on the public repo's release
page is the npm tarball mirror identifier — it titles the GitHub
release and lets you cross-reference a public-repo commit against an
npm tarball. It is not intended as a consumer pin; use v<semver> for
Action references.
Examples
Three copy-pasteable starting points ship in
examples/:
standalone-validate.mjs— validate a single JSON file from a Node script.bulk-folder-validate.mjs— walk a folder, validate every*.json, exit non-zero on any error.github-workflow.yml— drop into.github/workflows/to gate PRs with the composite Action.
Usage
Quick start (singleton)
For most use cases, use the lazy singleton — no class instantiation, profile source pre-wired to the package's defaults:
import {
recordsValidator,
setProfileSource,
createFilesystemProfileSource,
} from '@records-fhir/validator';
// Optional: point at a local FHIR package directory.
// Skip this if `~/.fhir/packages` is populated or you installed
// `@records-fhir/bundled-profiles`.
setProfileSource(createFilesystemProfileSource({
packageDirs: ['./fhir-packages'],
}));
const issues = await recordsValidator.validate(
{ resourceType: 'Patient', id: 'example', name: [{ family: 'Doe' }] },
'http://hl7.org/fhir/StructureDefinition/Patient',
'R4', // 'R4' | 'R4B' | 'R5' | 'R6'
);Class form (full control)
import { getRecordsValidatorClass } from '@records-fhir/validator';
const RecordsValidator = await getRecordsValidatorClass();
const validator = new RecordsValidator({
enableCaching: true,
strictMode: false,
});
const issues = await validator.validate({ resourceType: 'Patient' });FHIR version routing (R4B)
PublicFhirVersion accepts 'R4' | 'R4B' | 'R5' | 'R6'. R4B is
routed through the R4 internal path because R4B is a maintenance
release of R4 with the same StructureDefinitions and FHIRPath
context. Use toInternalFhirVersion to apply the same mapping in
embedder code:
import { toInternalFhirVersion, type PublicFhirVersion } from '@records-fhir/validator';
const v: PublicFhirVersion = 'R4B';
toInternalFhirVersion(v); // → 'R4'Apply a fix-suggestion patch
The catalog ships ~290 structured patches; applyFixPatch is the
executor:
import { applyFixPatch, getFixSuggestion } from '@records-fhir/validator';
const suggestion = getFixSuggestion('terminology-binding-required');
const result = applyFixPatch(
{ resourceType: 'Patient', id: 'p1' },
{ action: 'add', path: 'Patient.gender', value: 'other' },
);
// result.applied → true
// result.resource → { resourceType: 'Patient', id: 'p1', gender: 'other' }Custom Rules with the FHIRPath sandbox
User-defined Custom Rules go through a static safety pre-flight
before fhirpath.js evaluates them. Pathological expressions
(unbounded repeat(), deep where() nesting, megabyte regexes) are
rejected with a clear reason; legitimate FHIR core constraints sit
well under the limits.
import { checkFhirpathSandbox } from '@records-fhir/validator';
checkFhirpathSandbox('Patient.name.exists()');
// → { ok: true, metrics: { expressionLength: 21, functionCallCount: 1, nestingDepth: 1 } }
checkFhirpathSandbox('a'.repeat(5000));
// → { ok: false, reason: 'Expression length 5000 exceeds limit 4096', metrics: { ... } }Routing engine logs
By default the engine logs to console.{debug,info,warn,error}. Wire
in your own logger (Winston, pino, …) via setEngineLogger:
import { setEngineLogger } from '@records-fhir/validator';
setEngineLogger({
debug: () => {}, // silence debug
info: (msg, meta) => myLogger.info(msg, meta),
warn: (msg, meta) => myLogger.warn(msg, meta),
error: (msg, meta) => myLogger.error(msg, meta),
});Public Imports
Most consumers should import from the package root:
import { getRecordsValidatorClass, ValueSetValidator } from '@records-fhir/validator';Advanced integrations can use the explicitly exported subpaths in package.json, for example:
import { toOperationOutcome } from '@records-fhir/validator/core/operation-outcome-converter';Deep imports that are not listed in exports are internal and can change without notice.
Standalone vs. Records Integration
Standalone use does not require a database, Records API server, or Java. Host applications can provide optional integration hooks for profile resolution, custom business rules, and logging.
Records uses the same validator as its default engine, but the Records application code is not required to embed this package.
Embedder Boundaries
The package does not import Records server modules, database code, Express handlers, or application settings. Embedders can provide optional integration points:
setProfileSource()for database-backed or remote profile resolution.setCustomRulesSource()for user-defined business rules.setEngineLogger()for host logging.
Without these integrations, the validator uses no-op defaults and runs as a standalone offline validator.
Conformance
Current HL7 FHIR/fhir-test-cases status: 100.0% of executable comparison
tests passing. The latest report was generated on 2026-05-09 from upstream
commit e543043a076c493656fc8008df250659b15d02cb and is stored in the source
repository as conformance-results/report-2026-05-09.json.
The upstream manifest contains more than 900 entries. Records does not claim
that all manifest entries are executable in the current TypeScript validator
mode. The reported 100.0% applies to the JSON resource validation subset that
can be compared against the Java validator's expected OperationOutcome.
| Stage | Count | Meaning |
|---|---:|---|
| Upstream manifest entries | 969 | All entries in FHIR/fhir-test-cases/validator/manifest.json at commit e543043a. |
| Pre-filtered out | 438 | Not executable by this harness: the current comparison runner measures JSON FHIR resource validation against Java OperationOutcome baselines, not XML, non-resource formats, disabled upstream cases, unsupported modules, logical models, or cases without a Java baseline. |
| Candidate comparison set | 531 | R4/R5 or unversioned JSON-oriented entries with a declared Java baseline. |
| Runtime skipped | 35 | Candidate entries kept outside the headline JSON score because their Java baseline output is not available locally. |
| Executed and compared | 496 | Records result was normalized to OperationOutcome and diffed against Java. |
| Passed | 496 | All executable comparisons passed. |
Pre-filter exclusions:
| Reason | Count |
|---|---:|
| XML resources (Records validator is JSON-only) | 296 |
| Non-R4/R5 FHIR versions (3.0, 3.0.1, 1.4) | 47 |
| Unsupported modules: SHC, CDA, CDS Hooks, JSON5, XVer, DSIG, HL7 v2 | 74 |
| Disabled by upstream manifest (use-test: false) | 17 |
| No Java baseline declared in the manifest | 3 |
| Logical model test | 1 |
Runtime skips inside the 531 candidate set for the headline lane:
| Reason | Count | |---|---:| | Java baseline/parity backlog | 35 |
The Java baseline/parity backlog is measured separately with the explicit
--include-baseline-backlog discovery flag. The 2026-05-03 discovery run
resolves known upstream Java baseline path drift, includes explicit FML/NDJSON
parser-baseline fixtures, synthesizes the missing empty Java outcome for
cw-slice-compatible, admits JSON5 and DSIG JSON harness cases, and includes
two hidden Java-outcome fixtures. It runs the launch-discovery set:
547/547 passing, 0 skips, 0 failures. The report's passRate is 100.0%;
similarityScore may read 99.7% because it averages semantic diff similarity
for six approximate-but-passing Java parity cases. The backward-compatible
overallScore field remains an alias for similarityScore.
Excluded tests are tracked separately so the headline score does not imply XML, HL7 v2, CDA, CDS Hooks, DSIG, JSON5, SHC, or logical-model support.
Why So Many Manifest Entries Are Not Counted
The HL7 manifest is broader than "validate one FHIR JSON resource and compare the validator output." It also contains test assets for other formats, protocols, historical versions, and adjacent standards. Counting those as validator failures would make the metric less honest: it would mix unsupported product scope with actual JSON resource validation correctness.
| Excluded class | Why it is not part of this score | What would be needed to include it |
|---|---|---|
| XML resources | The package currently validates parsed JSON resources. XML requires parsing, XML-specific diagnostics, and stable XML-to-resource location mapping. | Add an XML parser/normalizer and an XML-aware diagnostic mapper, then run XML fixtures as a separate conformance lane. |
| CDA, HL7 v2, CDS Hooks, SHC, DSIG, JSON5, XVer | These are adjacent standards or special harnesses, not plain FHIR JSON resource validation. Some are transformation/signature/protocol tests rather than resource validation tests. | Build dedicated modules and dedicated conformance harnesses for each format/protocol. |
| Older FHIR versions (3.0, 3.0.1, 1.4) | The validator package targets R4, R5, and R6. Legacy STU3/DSTU-era behavior differs enough that it should not be silently mixed into the R4 score. | Add explicit legacy-version support and report it as a separate compatibility score. |
| Upstream-disabled tests | The upstream manifest marks them with use-test: false, so the reference suite itself does not treat them as active comparison cases. | Re-enable only if upstream enables them or if this project defines its own expected baseline. |
| Logical model tests | Logical models are not ordinary FHIR resource-instance validation cases. | Add logical-model validation support and a separate result category. |
| Missing Java baselines | The comparison metric is Java parity. Without an expected Java OperationOutcome, there is no objective diff target. | Generate and commit Java baselines, or define a Records-owned expected baseline with a different metric name. |
| .fml / .ndjson payloads | They pass the manifest filter but are not single JSON resource documents. | Add FML/NDJSON-specific loaders and compare them in dedicated lanes. |
For that reason, the headline number should be read as:
Records matches the Java validator on 496/496 currently executable FHIR JSON resource validation comparisons.
It should not be read as:
Records implements every format, protocol, legacy version, and adjacent standard represented somewhere in the upstream manifest.
The spec dispatch coverage report is also at 100% for the R4 base package
constraints measured by quality:spec-coverage.
MII 2026 Reference Scope
MII conformance is measured in a separate lane from the HL7
FHIR/fhir-test-cases score. The current scoped MII-2026 reference run was
generated on 2026-05-09 against the official MII FHIR Validator container at
http://localhost:8080. It matches the reference validator on 241/241 measured
resources from the refreshed MII 2026 corpus under the mii-2026-reference
profile scope and mii-local-blaze terminology mode, with 12 classified
corpus/profile-drift skips. The source-repository report is
conformance-results/mii-triangulation-2026-05-09.json.
This is a scoped parity claim for the measured package-example corpus. It is not an MII certification claim and does not imply full site-level MII Must-Support readiness.
The full scope-expansion plan is tracked in
docs/product/conformance-scope-roadmap.md in the Records source repository.
The public medvertical/records-fhir-validator export includes the same
roadmap under docs/conformance-scope-roadmap.md.
License
Apache-2.0 for this package. The surrounding Records application remains proprietary and is licensed separately.
