-
Notifications
You must be signed in to change notification settings - Fork 24
Authoring Measures in CQL
This topic will provide a brief overview of using Clinical Quality Language (CQL) to author electronic Clinical Quality Measures (eCQMs). The discussion assumes familiarity with clinical quality measurement in general, and the representation of eCQMs using Health Quality Measure Format (HQMF) and Quality Data Model (QDM) in particular.
Clinical Quality Language (CQL) is a high-level query language, meaning that it can be used to write expressions that determine what data is to be returned, not how it is to be returned. How the data is to be returned is part of the implementation of a quality measure, and may be accomplished in any number of ways, so CQL is intentionally silent on any of those details, allowing the logic expressed by CQL queries to be used in a broad variety of implementation environments to achieve the same result.
One of the central constructs available in CQL is the expression, which is any sequence of CQL that returns a value. Values are the data that CQL operates on, and can be simple values like the integer 5
and the string John Doe
, or they can be complex structures (or rows, also called tuples) like a QDM Encounter, Performed
, which contains attributes such as admissionSource
and relevantPeriod
that are themselves values of different types.
The type of value that an expression returns is based on the contents of the expression. For example, an expression that consists only of a value, such as 5
will simply return that value. Operators, such as +
and *
can be used to create more complex expressions such as 2 + 3
, which will return the result of evaluating the operation, the value 5
in this case. Expressions can also make use of functions, such as CalculateAge()
that allow more complex operations to be defined and reused.
One of the most important types of value available in CQL is the list, which is a sequence of values of any type. Lists can contain simple values, like a list of integers, { 1, 2, 3 }
, or they can contain tuples, like a list of encounters.
CQL includes a full suite of operators and functions to allow expressions to be written that can describe the types of logic and criteria that are found in clinical quality measures. The following sections will discuss how to use this language to author a measure from the ground up.
One of the first considerations when expressing a clinical quality measure in CQL is determining what the measure is counting. Is the measure patient-based, meaning that the measure is expression as some relationship between populations of patients, or is the measure counting some other item such as encounters or procedures?
If the measure is patient-based, then the criteria we define will all return true
or false
for each specific patient, indicating whether the patient is part of the population being defined by that criteria. A true
or false
value is type Boolean in CQL, and many of the built-in operations such as comparison (e.g. value > 5
) return values of this type.
For example, to express the demographic criteria for a measure, we can define an In Demographic
expression:
define "In Demographic":
AgeInYearsAt(start of "Measurement Period") >= 13
This example returns true
or false
for a given patient, depending on whether they satisfy the criteria (13 years of age or older at the start of the measurement period).
Often, we want to identify whether a patient has had a particular type of encounter as part of a patient-based measure. For example:
define "Inpatient Encounters":
["Encounter, Performed": "Inpatient Encounter Codes"] Encounter
where Encounter.relevantPeriod during "Measurement Period"
This expression returns a list of encounters for the patient that match one of the codes in the Inpatient Encounter Codes
value set, and that occurred during the measurement period. Now because the result of this expression is a list of encounters, we use an exists
to turn the value into a Boolean so we combine it with the previous expression:
define "Initial Population":
"In Demographic"
and exists ("Inpatient Encounters")
However, if we are counting encounters in an episode-of-care measure, the population criteria will all need to result in lists of encounters:
define "Measurement Period Encounters":
["Encounter, Performed": "Ambulatory/ED Visit"] Encounter
where Encounter.relevantPeriod during "Measurement Period"
and "In Demographic"
define "Pharyngitis Encounters With Antibiotics":
"Measurement Period Encounters" Encounter
with "Pharyngitis" Pharyngitis
such that Common."Includes Or Starts During"(Pharyngitis, Encounter)
with "Antibiotics" Antibiotics
such that Antibiotics.authorDatetime 3 days or less after start of Encounter.relevantPeriod
Since the measure is counting encounters, we can return the list in the initial population:
define "Initial Population":
"Pharyngitis Encounters With Antibiotics"
Authoring Patterns - QICore v4.1.1
Authoring Patterns - QICore v5.0.0
Authoring Patterns - QICore v6.0.0
Cooking with CQL Q&A All Categories
Additional Q&A Examples
Developers Introduction to CQL
Specifying Population Criteria