Clinical Quality Language Specification
2.0.0-ballot - R2 STU 1 (Ballot)

This page is part of the Clinical Quality Language Specification (v2.0.0-ballot: R2 STU 1) based on FHIR (HL7® FHIR® Standard) R4. This version is a pre-release. The current official version is 1.5.3. For a full list of available versions, see the Directory of published versions

V2.0 Change Log

Page standards status: Informative Maturity Level: N/A

2.0 Change Log

Ballot Changes (v.2.0.0-ballot)

Change Summary

Clinical Quality Language has been published as an HL7 and ANSI Normative standard since December of 2020. ANSI Normative standards must be reaffirmed every 5 years, so the time has come to reaffirm the specification. In addition, we have been steadily gathering implementer feedback and author feature requests over the years. There have been 2 errata publications where we have found issues that needed correction or clarification, but feedback and requests that are substantive in nature have had to wait for the next release. We have discussed, resolved, and applied that feedback to the specification, and will be balloting the proposed update in the September cycle.

As a result, there are two ballots for CQL in the September cycle.

  1. Reaffirmation of Clinical Quality Language, Release 1
  2. STU Ballot of Clinical Quality Language, Release 2

Signup pools are still open through Thursday, August 7th.

This topic will provide an overview of the substantive changes to the specification proposed in the ballot.

Importantly, no breaking changes have been introduced in CQL R2.

Directives
  • FHIR-41326 - Added directive support and defined default comparison precision

Directives provide an ability for translator options to be provided as part of the language.

https://build.fhir.org/ig/HL7/cql/03-developersguide.html#directives

Default Comparison Precision

The first usage of directives is to provide a default comparison precision:

#DefaultComparisonPrecision: minutes

Means that all comparisons within the library that do not explicitly specify a precision will be performed to the minute.

#DefaultComparisonPrecision: minutes default included libraries

Means that this behavior will also be carried to included libraries that do not specify their own default comparison precision.

https://build.fhir.org/ig/HL7/cql/03-developersguide.html#defaultcomparisonprecision

Using Enhancements

CQL R2 introduces several enhancements to data model support.

Model Qualifiers

Using statements can now include namespace qualifiers, allowing models to be referenced globally in the same way that libraries are.

using hl7.fhir.us.core.USCore
Model Aliases

Using statements can now also include a called clause, allowing multiple versions of the same model to referenced:

using hl7.fhir.us.core.USCore version '7.0.0' called USCore7
using hl7.fhir.us.core.USCore version '8.0.0' called USCore8

https://build.fhir.org/ig/HL7/cql/03-developersguide.html#data-models-1

Model Definitions
  • FHIR-34014 - Added the ability to declare named tuple types
  • FHIR-48760 - Added the ability to define models including types, contexts, and conversions in CQL directly

In addition, the components of models can now be defined directly within CQL with the introduction of define type, define context, and define conversion statements:

https://build.fhir.org/ig/HL7/cql/03-developersguide.html#defining-models

Type Definitions

CQL R2 includes support for defining named types with the new define type statement:

define public type Quantity extends System.Any {
  value System.Decimal,
  unit System.String
}

Once defined in a library, these types function like any other type defined in a model brought in with a using statement, allowing authors to build data models directly in CQL.

https://build.fhir.org/ig/HL7/cql/03-developersguide.html#defining-class-types

Context Definitions

CQL R2 includes support for defining contexts with the new define context statement:

define context Patient of type FHIR.Patient with key { id }

Once a context is defined, it can be used in a context statement like any context brought in with a using statement.

In addition, types can declare their relationship to contexts with the related to clause of the define type statement:

define type Observation extends DomainResource {
  subject Patient,
  code CodeableConcept,
  value Choice<CodeableConcept, Quantity>
  ...
}
related to Patient by { subject }

https://build.fhir.org/ig/HL7/cql/03-developersguide.html#defining-contexts

Conversion Definitions

CQL R2 includes support for defining conversions with the new define conversion statement:

define implicit conversion 
  from FHIR.Period to System.Interval<System.DateTime> 
  using FHIRHelpers.ToInterval

Once an implicit conversion is defined, it can be used to support implicit conversions like any other conversion brought in with a using statement.

Once an explicit conversion is defined, it can be referenced with the convert function like any other conversion.

https://build.fhir.org/ig/HL7/cql/03-developersguide.html#defining-conversions

Parameter Enhancements
Parameter Constraints
  • FHIR-48682 - Added support for parameter constraints

CQL R2 includes support for parameter constraints:

/*
@constraint: error
@message: Measurement period must be a year
*/
define IsYearly: start of "Measurement Period" same year as end of "Measurement Period"

These constraints are expressions that must evaluate to true in order to make use of any expressions within the library.

https://build.fhir.org/ig/HL7/cql/03-developersguide.html#parameter-constraints

Parameter Binding

CQL R2 also introduces support for parameter binding:

include ColorectalCancerElements called CCE
  bind { AsOf: end of "Measurement Period" }

This example illustrates setting the AsOf parameter in the ColorectalCancerElements library to the end of "Measurement Period" in the current library.

https://build.fhir.org/ig/HL7/cql/03-developersguide.html#parameter-binding

Terminology Enhancements
Equivalent Contains
  • FHIR-31675 - Added equivalent contains (~contains)

CQL R2 introduces support for equivalent contains, a contains operator that uses equivalent semantics, rather than equality semantics.

define "EquivalentContainsIsTrue": { 'A', 'B', 'C' } ~contains 'a'
define "EquivalentContainsIsFalse": { 'B', 'C' } ~contains 'a'

This support enables not only equivalent contains use cases like the above examples, but terminological contains as well:

define CodeSystemContainsExample: SNOMED ~contains "Random SNOMED Code"
define ValueSetContainsExample: "Inpatient Encounter" ~contains "Random CPT Code"

In addition to providing a generally useful operation, this is used in the retrieve later to enable an important set of use cases involving negation and direct-reference codes.

https://build.fhir.org/ig/HL7/cql/09-b-cqlreference.html#equivalentcontains

https://build.fhir.org/ig/HL7/cql/09-b-cqlreference.html#contains-codesystem

https://build.fhir.org/ig/HL7/cql/09-b-cqlreference.html#in-codesystem

Equivalent In

CQL R2 introduced the equivalent in operator to clearly distinguish between equality- and equivalent-based membership:

define "EquivalentInIsTrue": 'a' ~in { 'A', 'B', 'C' }
define "EquivalentInIsFalse": 'a' ~in { 'B', 'C' }

And as with the ~contains operator, the terminological membership operators then make use of this new equivalent in:

define InCodeSystemExample: "Random SNOMED Code" ~in "SNOMED"
define InValueSetExample: "Random CPT Code" ~in "Inpatient Encounter"

For backwards compatibility, in still resolves to the terminological membership operators, but using the new ~in operator makes it explicit that equivalent in is being used.

https://build.fhir.org/ig/HL7/cql/09-b-cqlreference.html#equivalentin

https://build.fhir.org/ig/HL7/cql/09-b-cqlreference.html#in-codesystem

https://build.fhir.org/ig/HL7/cql/09-b-cqlreference.html#in-valueset

Contains In a Retrieve
  • FHIR-31392 - Added ~contains as a retrieve terminology comparator

A significant outstanding issue with supporting the use of direct-reference codes throughout CQL is the negation patterns. When the extent of an activity is represented with a value set, rather than a specific code, then the negation statement looking for negation of a particular code requires the use of the new ~contains operation. This can now be specified explicitly as in:

define "Reason for Macular Edema Absent Not Communicated (Explicit equivalent contains)":
  [CommunicationNotDone: reasonCode ~contains "Macular edema absent (situation)"]

And this now means that direct-reference codes can be the target of a negation retrieve, such as:

define "Reason for Macular Edema Absent Not Communicated (Implicit equivalent contains)":
  [CommunicationNotDone: "Macular edema absent (situation)"]

https://build.fhir.org/ig/HL7/cql/04-logicalspecification.html#codecomparator

Miscellaneous New Functions
Round(Quantity)

CQL R2 adds the ability to round Quantity values:

define "QuantityRound": Round(2.54 'cm') // 2 'cm'

https://build.fhir.org/ig/HL7/cql/09-b-cqlreference.html#round

MatchesFull

CQL R2 adds a MatchesFull function, which is a regex match that is required to match the entire string, as opposed to the Matches operation which by default uses partial matching.

define MatchesFullFalse: 'http://fhir.org/guides/cqf/common/Library/FHIR-ModelInfo|4.0.1'.matchesFull('Library') // returns false
define MatchesFullAlsoFalse: 'N8000123123'.matchesFull('N[0-9]{8}') // returns false as the string is not an 8 char number (it has 10)
define MatchesFullTrue: 'N8000123123'.matchesFull('N[0-9]{10}') // returns true as the string has an 10 number sequence in it starting with `N`

https://build.fhir.org/ig/HL7/cql/09-b-cqlreference.html#matchesfull

Slice

CQL R2 adds a Slice function that is a generalization of the Take, Skip, and Tail functions:

define SliceStart: Slice({ 1, 2, 3, 4, 5 }, 1) // { 2, 3, 4, 5 }
define SliceEnd: Slice({ 1, 2, 3, 4, 5 }, 1, 3) // { 2, 3 }

https://build.fhir.org/ig/HL7/cql/09-b-cqlreference.html#slice

FHIRPath Alignment
  • FHIR-48809 - Added FHIRPath mapping for additional string functions
  • FHIR-48829 - Added FHIRPath mapping for precision and boundary functions
  • FHIR-48828 - Added FHIRPath mapping for date/time extractors
  • FHIR-44827 - Added FHIRPath mapping for defineVariable

In addition, CQL R2 incorporates updates made to FHIRPath as part of a ballot earlier this year. In that update, FHIRPath added support for several capabilities and functions that were already defined in CQL. This support was added to FHIRPath with the same semantics as CQL, and now this update to CQL R2 adds FHIRPath mappings for that functionality to ensure that we maintain alignment between the two specifications.

https://build.fhir.org/ig/HL7/cql/16-i-fhirpathtranslation.html

Change Log

Compatible, Substantive
  • FHIR-31019: Support qualified identifiers for model specifiers
  • FHIR-31392: Support contains as a terminology comparator
  • FHIR-31397: Multi-filter support in Retrieve
  • FHIR-31675: Add terminological overloads for contains
  • FHIR-33126: Provide the ability to specify parameter binding in includes
  • FHIR-33247: Consider specific operators for equivalent membership
  • FHIR-34014: Ability to define named tuple types
  • FHIR-37827: Allow tags without a value
  • FHIR-40612: Would like Tail and Head functions with parameter
  • FHIR-41326: Provide a "default comparison precision"
  • FHIR-44827: Add FHIRPath mapping for defineVariable
  • FHIR-44828: Add FHIRPath mapping for date/time extractors
  • FHIR-44829: Add FHIRPath mapping for precision/boundary functions
  • FHIR-48682: Consider parameter constraints
  • FHIR-48760: Declare model artifacts in CQL itself instead of XML files
  • FHIR-48762: Add the ability to alias a model declaration
  • FHIR-48809: Add FHIRPath mapping for additional string functions
  • FHIR-50456: Support matchesFull
  • FHIR-51030: Round should be supported for Quantity types
Non-Substantive
  • FHIR-31398: Expand informative modelInfo description
  • FHIR-39612: Allow for concepts with no codes
  • FHIR-41169: Incorrect results from "CumulativeDuration" function in the documentation
  • FHIR-41731: Clarify compile-time vs run-time terms
  • FHIR-42731: Conditional branches should be of the same type
  • FHIR-44599: Enhance support for FHIR references
  • FHIR-44825: Clarify ToString behavior with quantity
  • FHIR-44910: Remove underscores in library name examples
  • FHIR-44922: Deprecate the NotEqual node
  • FHIR-46281: Clarify LowBoundary/HighBoundary semantics
  • FHIR-46817: Correct examples of lowBoundary and highBoundary functions
  • FHIR-48534: Change tests.zip to link to github tests
  • FHIR-48818: Document initial case
  • FHIR-48867: Clarify when seconds and milliseconds should be combined in comparisons and calculations
  • FHIR-49652: Reference to is QDM unhelpful
  • FHIR-49684: Date/time arithmetic - subtraction < 0
  • FHIR-51412: Divide by zero example is incorrect
  • FHIR-50198: Correction to disambiguate %context parameter from context definition
  • FHIR-50455: Error in matches example
  • FHIR-50938: Clarify successor/predecessor should be precision-aware for decimals