<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Abdelhadi Sabani</title>
    <description>The latest articles on DEV Community by Abdelhadi Sabani (@awbx).</description>
    <link>https://dev.to/awbx</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F720579%2Ffc166c75-7ffc-4083-8aaa-5cc24e4b7446.jpg</url>
      <title>DEV Community: Abdelhadi Sabani</title>
      <link>https://dev.to/awbx</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/awbx"/>
    <language>en</language>
    <item>
      <title>fhir-dsl: a type-safe FHIR toolchain for TypeScript</title>
      <dc:creator>Abdelhadi Sabani</dc:creator>
      <pubDate>Sat, 02 May 2026 00:59:39 +0000</pubDate>
      <link>https://dev.to/awbx/fhir-dsl-a-type-safe-fhir-toolchain-for-typescript-1gjk</link>
      <guid>https://dev.to/awbx/fhir-dsl-a-type-safe-fhir-toolchain-for-typescript-1gjk</guid>
      <description>&lt;p&gt;If you've written TypeScript against a FHIR API, you know the script:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;The "official" client returns &lt;code&gt;Bundle | unknown&lt;/code&gt;. Every property access is a &lt;code&gt;?.&lt;/code&gt; chain through fog.&lt;/li&gt;
&lt;li&gt;Search parameters are strings. &lt;code&gt;name&lt;/code&gt; works on Patient, but does &lt;code&gt;family&lt;/code&gt; work on Practitioner? Read the spec, hope you remember.&lt;/li&gt;
&lt;li&gt;Profile narrowing (US Core, IPS) doesn't exist at the type level. You assume the resource conforms; runtime tells you when it doesn't.&lt;/li&gt;
&lt;li&gt;FHIRPath is a string. The compiler doesn't know &lt;code&gt;Patient.name.given&lt;/code&gt; is valid and &lt;code&gt;Patient.name.givven&lt;/code&gt; isn't.&lt;/li&gt;
&lt;li&gt;Validation is "import a JSON schema and pray it matches the version you're talking to".&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;The frustration isn't FHIR. FHIR is &lt;em&gt;huge&lt;/em&gt; — ~146 resources in R4, hundreds more in R5, tens of thousands of bindings — but it's a &lt;strong&gt;precise&lt;/strong&gt; spec. Every resource, every search parameter, every value set is published as machine-readable JSON. There's no fundamental reason a client has to lie to your type system.&lt;/p&gt;

&lt;p&gt;Four months ago I started &lt;a href="https://github.com/awbx/fhir-dsl" rel="noopener noreferrer"&gt;&lt;code&gt;@fhir-dsl/*&lt;/code&gt;&lt;/a&gt; on the bet that if you lean hard on code generation and copy &lt;a href="https://kysely.dev/" rel="noopener noreferrer"&gt;Kysely&lt;/a&gt;'s design playbook, you can have full compile-time safety on every part of FHIR — searches, profiles, extensions, FHIRPath, validators, the lot — without runtime overhead.&lt;/p&gt;

&lt;p&gt;This post is a tour of what shipped in v1.x and the design decisions worth stealing if you're building a typed-DSL library for any precise-spec domain (insurance, banking, GraphQL meta-clients, scientific data formats).&lt;/p&gt;

&lt;h2&gt;
  
  
  A 30-second taste
&lt;/h2&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;createClient&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;./fhir/r4&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt; &lt;span class="c1"&gt;// generated by fhir-gen&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;fhir&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;createClient&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt; &lt;span class="na"&gt;baseUrl&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;https://hapi.fhir.org/baseR4&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt; &lt;span class="p"&gt;});&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;result&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;fhir&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;search&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Patient&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;where&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;family&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;eq&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Smith&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;where&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;birthdate&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;ge&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;1990-01-01&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;include&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;general-practitioner&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;sort&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;birthdate&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;desc&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;count&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="mi"&gt;20&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;execute&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt;

&lt;span class="c1"&gt;// result.data: Patient[]&lt;/span&gt;
&lt;span class="c1"&gt;// result.included: Practitioner[]&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Every string in that chain is autocompleted from the FHIR R4 schema:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;code&gt;"Patient"&lt;/code&gt; — narrowed to a real resource type&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;"family"&lt;/code&gt; — narrowed to Patient's actual search parameters; if you typed it on Practitioner you'd get a different list&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;"eq"&lt;/code&gt; — narrowed to the operator allowed for that parameter type (string params get &lt;code&gt;eq&lt;/code&gt;/&lt;code&gt;contains&lt;/code&gt;/&lt;code&gt;exact&lt;/code&gt;; date params get &lt;code&gt;eq&lt;/code&gt;/&lt;code&gt;gt&lt;/code&gt;/&lt;code&gt;ge&lt;/code&gt;/&lt;code&gt;lt&lt;/code&gt;/&lt;code&gt;le&lt;/code&gt;/&lt;code&gt;sa&lt;/code&gt;/&lt;code&gt;eb&lt;/code&gt;/&lt;code&gt;ap&lt;/code&gt;; tokens get &lt;code&gt;eq&lt;/code&gt;/&lt;code&gt;not&lt;/code&gt;/&lt;code&gt;in&lt;/code&gt;/&lt;code&gt;above&lt;/code&gt;/&lt;code&gt;below&lt;/code&gt;/&lt;code&gt;of-type&lt;/code&gt;/&lt;code&gt;text&lt;/code&gt;)&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;"general-practitioner"&lt;/code&gt; — narrowed to Patient's actual &lt;code&gt;_include&lt;/code&gt; targets&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;result.included&lt;/code&gt; — typed as &lt;code&gt;Practitioner[]&lt;/code&gt; because that's what &lt;code&gt;general-practitioner&lt;/code&gt; resolves to per the schema&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Mistype any of those and the build breaks. No runtime "I wonder if this search param exists" surprises.&lt;/p&gt;

&lt;h2&gt;
  
  
  The bet: generate everything from the spec
&lt;/h2&gt;

&lt;p&gt;The single biggest design call was leaning fully into code generation. Hand-writing types against ~146 resources × profiles × bindings × extensions is a fool's errand. The spec evolves; hand-maintained types decay.&lt;/p&gt;

&lt;p&gt;So &lt;code&gt;@fhir-dsl/cli&lt;/code&gt;'s &lt;code&gt;fhir-gen&lt;/code&gt; reads the FHIR spec — any version (R4, R4B, R5, R6), any Implementation Guide — and emits a typed schema:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx @fhir-dsl/cli generate &lt;span class="se"&gt;\&lt;/span&gt;
  &lt;span class="nt"&gt;--version&lt;/span&gt; r4 &lt;span class="se"&gt;\&lt;/span&gt;
  &lt;span class="nt"&gt;--ig&lt;/span&gt; hl7.fhir.us.core@6.1.0 &lt;span class="se"&gt;\&lt;/span&gt;
  &lt;span class="nt"&gt;--validator&lt;/span&gt; native &lt;span class="se"&gt;\&lt;/span&gt;
  &lt;span class="nt"&gt;--expand-valuesets&lt;/span&gt; &lt;span class="se"&gt;\&lt;/span&gt;
  &lt;span class="nt"&gt;--resolve-codesystems&lt;/span&gt; &lt;span class="se"&gt;\&lt;/span&gt;
  &lt;span class="nt"&gt;--out&lt;/span&gt; ./src/fhir
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Output:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;code&gt;types/&lt;/code&gt; — TypeScript interfaces for every resource, with branded primitives, discriminated &lt;code&gt;ChoiceOf&amp;lt;T, "value"&amp;gt;&lt;/code&gt; for &lt;code&gt;value[x]&lt;/code&gt; fields, primitive &lt;code&gt;_field&lt;/code&gt; siblings (&lt;code&gt;_id&lt;/code&gt;, &lt;code&gt;_extension&lt;/code&gt;, …), and the full 49-variant &lt;code&gt;Extension.value[x]&lt;/code&gt; union&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;profiles/&lt;/code&gt; — narrowed types for US Core (or any IG), with profile-required fields non-optional and slice-named optional fields exposed (&lt;code&gt;extension_usCoreRace?&lt;/code&gt;, &lt;code&gt;component_systolic?&lt;/code&gt;)&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;search-params/&lt;/code&gt; — typed search-parameter map per resource (this is what &lt;code&gt;where()&lt;/code&gt; reads)&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;schemas/&lt;/code&gt; — Standard Schema v1 validators per resource, with FHIRPath invariants (&lt;code&gt;pat-1&lt;/code&gt;, &lt;code&gt;dom-3&lt;/code&gt;, etc.) compiled to refinement predicates&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;terminology/&lt;/code&gt; — resolved value sets as TS unions (with &lt;code&gt;(string &amp;amp; {})&lt;/code&gt; for extensible bindings to keep autocomplete without boxing legitimate codes out)&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;client.ts&lt;/code&gt; — &lt;code&gt;createClient&amp;lt;R4Schema&amp;gt;()&lt;/code&gt; factory&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;You commit this. It's just TypeScript — no runtime FHIR-spec dependency, no third-party deps, fully tree-shakeable.&lt;/p&gt;

&lt;h2&gt;
  
  
  Where it gets interesting: the type system depth
&lt;/h2&gt;

&lt;p&gt;Type-checking the obvious stuff (resource names, primitive search params) is the price of entry. The real differentiator is what happens once you go beyond that.&lt;/p&gt;

&lt;h3&gt;
  
  
  Profile narrowing
&lt;/h3&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;vitals&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;fhir&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;search&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;
    &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Observation&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;http://hl7.org/fhir/us/core/StructureDefinition/us-core-vital-signs&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;where&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;patient&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;eq&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Patient/123&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;where&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;status&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;eq&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;final&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;execute&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt;

&lt;span class="c1"&gt;// vitals.data: USCoreVitalSignsProfile[]&lt;/span&gt;
&lt;span class="c1"&gt;// — `code`, `subject`, `effectiveDateTime`, `value[x]` are all non-optional&lt;/span&gt;
&lt;span class="c1"&gt;//   because the profile requires them&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;When you pass a profile URL to &lt;code&gt;.search()&lt;/code&gt;, the result type narrows to the profile's interface. Profile-required fields move from optional to required. Slices get their own typed accessors.&lt;/p&gt;

&lt;h3&gt;
  
  
  Typed extensions per IG
&lt;/h3&gt;

&lt;p&gt;US Core defines extensions like &lt;code&gt;us-core-race&lt;/code&gt;, &lt;code&gt;us-core-ethnicity&lt;/code&gt;, &lt;code&gt;us-core-birthsex&lt;/code&gt;. Each gets a branded &lt;code&gt;Extension&amp;lt;URL&amp;gt;&lt;/code&gt; interface in the generated output:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="kd"&gt;type&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;USCoreRaceExtension&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;./fhir/profiles/us-core&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="c1"&gt;// extension_usCoreRace?: USCoreRaceExtension&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;race&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;patient&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;extension_usCoreRace&lt;/span&gt;&lt;span class="p"&gt;?.&lt;/span&gt;&lt;span class="nx"&gt;extension&lt;/span&gt;&lt;span class="p"&gt;?.&lt;/span&gt;&lt;span class="nf"&gt;find&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;
  &lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;e&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nx"&gt;e&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;url&lt;/span&gt; &lt;span class="o"&gt;===&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;ombCategory&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="c1"&gt;// race.valueCoding has the branded Coding shape per the SD&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;You don't have to hunt through &lt;code&gt;extension[]&lt;/code&gt; looking for a URL match — slicing is materialized in the type.&lt;/p&gt;

&lt;h3&gt;
  
  
  Slicing on backbone elements
&lt;/h3&gt;

&lt;p&gt;Blood pressure observations have two &lt;code&gt;component[]&lt;/code&gt; entries with specific LOINC codes. The US Core BP profile slices &lt;code&gt;component&lt;/code&gt; into &lt;code&gt;component_systolic&lt;/code&gt; and &lt;code&gt;component_diastolic&lt;/code&gt;. The generator emits both as optional accessors:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// bp: USCoreBloodPressureProfile&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;systolic&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;bp&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;component_systolic&lt;/span&gt;&lt;span class="p"&gt;?.&lt;/span&gt;&lt;span class="nx"&gt;valueQuantity&lt;/span&gt;&lt;span class="p"&gt;?.&lt;/span&gt;&lt;span class="nx"&gt;value&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;diastolic&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;bp&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;component_diastolic&lt;/span&gt;&lt;span class="p"&gt;?.&lt;/span&gt;&lt;span class="nx"&gt;valueQuantity&lt;/span&gt;&lt;span class="p"&gt;?.&lt;/span&gt;&lt;span class="nx"&gt;value&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Runtime helpers (&lt;code&gt;extensionByUrl&lt;/code&gt;, &lt;code&gt;findSliceByPath&lt;/code&gt;) are also emitted for cases where you'd rather walk the array.&lt;/p&gt;

&lt;h3&gt;
  
  
  Chained, reverse-chained, composite
&lt;/h3&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// Chained: search Observations through their patient reference&lt;/span&gt;
&lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;fhir&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;search&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Observation&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;whereChained&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;subject&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Patient&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;family&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;eq&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Smith&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;execute&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt;

&lt;span class="c1"&gt;// _has: patients who have a specific observation&lt;/span&gt;
&lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;fhir&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;search&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Patient&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;has&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Observation&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;subject&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;code&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;eq&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;http://loinc.org|85354-9&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;execute&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt;

&lt;span class="c1"&gt;// Composite: code AND value-quantity together&lt;/span&gt;
&lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;fhir&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;search&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Observation&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;whereComposite&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;code-value-quantity&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="na"&gt;code&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;http://loinc.org|8480-6&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;value-quantity&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;60&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="p"&gt;})&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;execute&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Every chain hop, every &lt;code&gt;_has&lt;/code&gt; link, every composite component is typed against the schema. &lt;code&gt;whereChained("subject", "Practitioner", "family", "eq", "Smith")&lt;/code&gt; would fail at compile time because &lt;code&gt;Observation.subject&lt;/code&gt; doesn't reference Practitioner.&lt;/p&gt;

&lt;h3&gt;
  
  
  Capability-driven client narrowing
&lt;/h3&gt;

&lt;p&gt;If your FHIR server's CapabilityStatement says it doesn't support &lt;code&gt;Observation.write&lt;/code&gt;, you can wrap the client to remove &lt;code&gt;.create("Observation", …)&lt;/code&gt; from the type:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;createCapabilityGuard&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;@fhir-dsl/core&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;guarded&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;createCapabilityGuard&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;fhir&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;capabilityStatement&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="nx"&gt;guarded&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;create&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Observation&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="err"&gt;…&lt;/span&gt; &lt;span class="p"&gt;});&lt;/span&gt; &lt;span class="c1"&gt;// ← TypeError, server doesn't advertise it&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  FHIRPath, but typed
&lt;/h2&gt;

&lt;p&gt;&lt;code&gt;@fhir-dsl/fhirpath&lt;/code&gt; is a typed FHIRPath expression builder. You compose paths against generated resource types and either compile them to spec strings or evaluate them in-process:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;fhirpath&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;@fhir-dsl/fhirpath&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="kd"&gt;type&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;Patient&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;./fhir/r4&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;officialFamily&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;fhirpath&lt;/span&gt;&lt;span class="o"&gt;&amp;lt;&lt;/span&gt;&lt;span class="nx"&gt;Patient&lt;/span&gt;&lt;span class="o"&gt;&amp;gt;&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Patient&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;name&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;where&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;$&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nx"&gt;$&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;use&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;eq&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;official&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;))&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;family&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="nx"&gt;officialFamily&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;compile&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt;
&lt;span class="c1"&gt;// "Patient.name.where(use = 'official').family"&lt;/span&gt;

&lt;span class="nx"&gt;officialFamily&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;evaluate&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;somePatient&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="c1"&gt;// ["Smith"]&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Highlights:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;UCUM-aware Quantity comparisons.&lt;/strong&gt; &lt;code&gt;5 'mg' = 0.005 'g'&lt;/code&gt; returns &lt;code&gt;true&lt;/code&gt;. Same-dimension equality and ordering use a native UCUM core (SI base + prefixes, common healthcare units, single-&lt;code&gt;/&lt;/code&gt; compounds, bracketed &lt;code&gt;mm[Hg]&lt;/code&gt;). Offset, log, and multi-&lt;code&gt;/&lt;/code&gt; units throw &lt;code&gt;UcumError&lt;/code&gt; instead of silently wrong answers.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;&lt;code&gt;resolve()&lt;/code&gt; walks Bundles.&lt;/strong&gt; Inside &lt;code&gt;Bundle.entry.resource.subject.resolve()&lt;/code&gt;, the evaluator walks the surrounding Bundle to find the referenced resource — and falls through to &lt;code&gt;EvalOptions.resolveReference&lt;/code&gt; when not in scope.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Terminology hooks.&lt;/strong&gt; &lt;code&gt;conformsTo&lt;/code&gt;, &lt;code&gt;memberOf&lt;/code&gt;, &lt;code&gt;subsumes&lt;/code&gt;, &lt;code&gt;subsumedBy&lt;/code&gt; compile to spec strings and dispatch to &lt;code&gt;EvalOptions.terminology&lt;/code&gt; at evaluate time.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Invariants → OperationOutcome.&lt;/strong&gt; Every ElementDefinition.constraint compiles to a predicate via &lt;code&gt;compileInvariant&lt;/code&gt;, and &lt;code&gt;validateInvariants(resource, [inv])&lt;/code&gt; returns a real &lt;code&gt;OperationOutcome&lt;/code&gt;. This is wired automatically into the generated validators.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Write-back via JSON Patch.&lt;/strong&gt; &lt;code&gt;setValue&lt;/code&gt; and &lt;code&gt;createPatch&lt;/code&gt; invert an &lt;code&gt;eq&lt;/code&gt;-shaped predicate path into a deep-cloned next resource or an RFC 6902 patch. Useful for form-driven updates that don't blow away other fields.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Not every corner of the FHIRPath N1 spec is covered — only the practical subset that real FHIR invariants and navigation actually use. The &lt;a href="https://awbx.github.io/fhir-dsl/docs/fhirpath/overview#fhirpath-spec-coverage" rel="noopener noreferrer"&gt;coverage table&lt;/a&gt; is honest about gaps.&lt;/p&gt;

&lt;h2&gt;
  
  
  Validators that actually validate
&lt;/h2&gt;

&lt;p&gt;Pass &lt;code&gt;--validator native&lt;/code&gt; (or &lt;code&gt;--validator zod&lt;/code&gt;) to the generator and you get &lt;a href="https://standardschema.dev/" rel="noopener noreferrer"&gt;Standard Schema v1&lt;/a&gt; validators for every resource, datatype, binding, and profile. Native is zero-dep; zod uses &lt;code&gt;zod&lt;/code&gt;. Either way, downstream consumers see the same Standard Schema interface.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;PatientSchema&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;./fhir/schemas&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;result&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;PatientSchema&lt;/span&gt;&lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;~standard&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;].&lt;/span&gt;&lt;span class="nf"&gt;validate&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;unknownJson&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;result&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;issues&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="c1"&gt;// OperationOutcome-shaped diagnostics, including FHIRPath invariant failures&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Or chain &lt;code&gt;.validate()&lt;/code&gt; directly on a query:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;patient&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;fhir&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;read&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;Patient&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;123&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;validate&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt; &lt;span class="c1"&gt;// client-side schema check before returning&lt;/span&gt;
  &lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;execute&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;(Server-side &lt;code&gt;$validate&lt;/code&gt; is a separate operation — &lt;code&gt;client.operation("$validate", { resource })&lt;/code&gt; — for cases where you want the server's opinion.)&lt;/p&gt;

&lt;h2&gt;
  
  
  A real terminology engine
&lt;/h2&gt;

&lt;p&gt;&lt;code&gt;@fhir-dsl/terminology&lt;/code&gt; isn't a stub. It implements:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;ValueSet &lt;code&gt;$expand&lt;/code&gt; with filter operators: &lt;code&gt;is-a&lt;/code&gt;, &lt;code&gt;is-not-a&lt;/code&gt;, &lt;code&gt;descendent-of&lt;/code&gt;, &lt;code&gt;regex&lt;/code&gt;
&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;$validate-code&lt;/code&gt; against expanded sets&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;$lookup&lt;/code&gt; and &lt;code&gt;$translate&lt;/code&gt;
&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;$subsumes&lt;/code&gt; with transitive concept-graph traversal&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;So when a FHIR resource binds to a SNOMED ValueSet of "all descendants of &lt;code&gt;444971000124105&lt;/code&gt;" (Body Mass Index), the engine actually walks the hierarchy and gives you the closed enum at generate time. No "we'll just leave it as a string" cop-out.&lt;/p&gt;

&lt;h2&gt;
  
  
  SMART on FHIR v2 + an LLM bridge
&lt;/h2&gt;

&lt;p&gt;Two more pieces, both opt-in:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;&lt;code&gt;@fhir-dsl/smart&lt;/code&gt;&lt;/strong&gt; — full SMART v2 client: PKCE-S256 patient launch, backend-services with signed JWT (RS384/ES384), refresh-token rotation, scope DSL. Implements core's &lt;code&gt;AuthProvider&lt;/code&gt; interface so wiring it into &lt;code&gt;createClient({ auth: smartClient })&lt;/code&gt; is one line.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;&lt;code&gt;@fhir-dsl/mcp&lt;/code&gt;&lt;/strong&gt; — Model Context Protocol server that exposes ~10 generic FHIR verbs (read, vread, search, history, create, update, patch, delete, operation, capabilities) as tool calls. Bearer / backend-services / patient-launch auth, pluggable audit sinks, write gating with allowlists and dry-run, response-byte cap to prevent token economy blowups. Stdio and HTTP transports. So you can hand Claude a real FHIR backend without giving it raw HTTP access.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  The CLI surface
&lt;/h2&gt;

&lt;p&gt;&lt;code&gt;fhir-gen&lt;/code&gt; ships five commands:&lt;/p&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Command&lt;/th&gt;
&lt;th&gt;What it does&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;generate&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Generate types (+ optional MCP server, validators, IG profiles)&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;capability&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Snapshot a server's CapabilityStatement as a typed report&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;validate&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Structural sanity-check on a FHIR JSON resource (CI-friendly)&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;scaffold-ig&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Bootstrap a project pre-wired to an Implementation Guide&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;diff&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;Compare two generated outputs and exit-code on breaking changes&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;

&lt;p&gt;&lt;code&gt;diff&lt;/code&gt; is the one I'd point at: wire it into CI, and FHIR version bumps stop being scary because you get a machine-readable list of every removed field, every optional→required change, every type narrowing.&lt;/p&gt;

&lt;h2&gt;
  
  
  Design decisions worth calling out
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;Compile/execute split.&lt;/strong&gt; Every terminal builder exposes &lt;code&gt;compile()&lt;/code&gt; (synchronous, returns the wire shape) and &lt;code&gt;execute()&lt;/code&gt; (the network call). This lets snapshot tests skip the network, lets the React Query bindings derive a stable &lt;code&gt;queryKey&lt;/code&gt; without invoking anything, and lets MCP report the structured query shape to the LLM before executing. Same trick Kysely, Drizzle, and tRPC use.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Discriminated &lt;code&gt;FhirDslError&lt;/code&gt;.&lt;/strong&gt; Every error in the monorepo extends &lt;code&gt;FhirDslError&lt;/code&gt; with a &lt;code&gt;kind&lt;/code&gt; discriminator (&lt;code&gt;core.request&lt;/code&gt;, &lt;code&gt;core.validation&lt;/code&gt;, &lt;code&gt;smart.auth&lt;/code&gt;, &lt;code&gt;runtime.fhir&lt;/code&gt;, …) and structured &lt;code&gt;context&lt;/code&gt;. Plus a &lt;code&gt;Result&amp;lt;T, E&amp;gt;&lt;/code&gt; + &lt;code&gt;tryAsync&lt;/code&gt; toolkit if you want Effect-style typed handling without &lt;code&gt;try&lt;/code&gt;/&lt;code&gt;catch&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Surface-frozen at v1.x.&lt;/strong&gt; Public API across all 11 packages is locked at the &lt;a href="https://github.com/awbx/fhir-dsl/releases/tag/surface-v1.0.0" rel="noopener noreferrer"&gt;&lt;code&gt;surface-v1.0.0&lt;/code&gt;&lt;/a&gt; tag. Minor releases add to it, patch releases fix bugs in it, breaking changes wait for v2. Drift is caught by &lt;code&gt;pnpm audit:export-surface&lt;/code&gt; on every PR.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Pay-as-you-go packages.&lt;/strong&gt; Eleven small packages (~5KB to ~30KB each), not one monolith. The MCP package's SMART import is lazy-loaded so bearer-only deployments never pay the &lt;code&gt;jose&lt;/code&gt; cost.&lt;/p&gt;

&lt;h2&gt;
  
  
  What it isn't
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Not a FHIR server.&lt;/strong&gt; Use HAPI, Aidbox, Medplum.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Not a CDR.&lt;/strong&gt; No durable storage, no business logic.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Not "the FHIR Way".&lt;/strong&gt; Patient context, multi-tenant routing, pagination strategy — all your app's call.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Try it
&lt;/h2&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npm &lt;span class="nb"&gt;install&lt;/span&gt; @fhir-dsl/core @fhir-dsl/runtime
npx @fhir-dsl/cli generate &lt;span class="nt"&gt;--version&lt;/span&gt; r4 &lt;span class="nt"&gt;--out&lt;/span&gt; ./src/fhir
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Docs:&lt;/strong&gt; &lt;a href="https://awbx.github.io/fhir-dsl/" rel="noopener noreferrer"&gt;https://awbx.github.io/fhir-dsl/&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;GitHub:&lt;/strong&gt; &lt;a href="https://github.com/awbx/fhir-dsl" rel="noopener noreferrer"&gt;https://github.com/awbx/fhir-dsl&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Quick start:&lt;/strong&gt; &lt;a href="https://awbx.github.io/fhir-dsl/docs/getting-started/quick-start" rel="noopener noreferrer"&gt;https://awbx.github.io/fhir-dsl/docs/getting-started/quick-start&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;CLI reference:&lt;/strong&gt; &lt;a href="https://awbx.github.io/fhir-dsl/docs/cli/usage" rel="noopener noreferrer"&gt;https://awbx.github.io/fhir-dsl/docs/cli/usage&lt;/a&gt;
&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;If you've shipped FHIR integrations and have battle-scar feedback, I'd genuinely love it — open an issue, or DM. The surface is frozen but the docs and DX are still being polished, and outside critique is the only way to find the gaps insiders miss.&lt;/p&gt;

&lt;p&gt;If you're building a typed-DSL for some other precise-spec domain, the design playbook is the same: &lt;strong&gt;lean on code generation, split compile/execute, never lie to the type system, and don't let the spec be a "rough guide" — let it be the source of truth.&lt;/strong&gt; The upfront work pays off forever.&lt;/p&gt;

</description>
      <category>typescript</category>
      <category>healthcare</category>
      <category>fhir</category>
      <category>opensource</category>
    </item>
  </channel>
</rss>
