Clinical Practice Guidelines
1.0.0 - ci-build International flag

Clinical Practice Guidelines, published by HL7 International - Clinical Decision Support WG. This is not an authorized publication; it is the continuous build for version 1.0.0). This version is based on the current content of https://github.com/HL7/cqf-recommendations/ and changes regularly. See the Directory of published versions

: Opioid CDS R4 Common Logic - TTL Representation

Active as of 2018-03-25

Raw ttl | Download

@prefix fhir: <http://hl7.org/fhir/> .
@prefix owl: <http://www.w3.org/2002/07/owl#> .
@prefix rdfs: <http://www.w3.org/2000/01/rdf-schema#> .
@prefix sct: <http://snomed.info/id/> .
@prefix xsd: <http://www.w3.org/2001/XMLSchema#> .

# - resource -------------------------------------------------------------------

 a fhir:Library ;
  fhir:nodeRole fhir:treeRoot ;
  fhir:id [ fhir:v "OpioidCDSR4Common"] ; # 
  fhir:text [
fhir:status [ fhir:v "generated" ] ;
fhir:div "<div xmlns=\"http://www.w3.org/1999/xhtml\"><h2>Participants</h2><table class=\"grid\"><tr><td>Author</td><td>Kensaku Kawamoto, MD, PhD, MHS</td></tr><tr><td>Author</td><td>Bryn Rhodes</td></tr><tr><td>Author</td><td>Floyd Eisenberg, MD, MPH</td></tr><tr><td>Author</td><td>Robert McClure, MD, MPH</td></tr></table><h2>Related Artifacts</h2><table class=\"grid\"><tr><td>documentation</td><td>CDC guideline for prescribing opioids for chronic pain</td><td>https://www.cdc.gov/mmwr/volumes/65/rr/rr6501e1.htm?CDC_AA_refVal=https%3A%2F%2Fwww.cdc.gov%2Fmmwr%2Fvolumes%2F65%2Frr%2Frr6501e1er.htm</td></tr></table><h2>Contents</h2><p><code>text/cql</code></p><pre><code class=\"language-sql\">library OpioidCDSR4Common version '0.1.0'\n\nusing FHIR version '4.0.0'\n\ninclude FHIRHelpers version '4.0.0' called FHIRHelpers\ninclude OMTKLogic version '0.0.0' called OMTKLogic\n\ncodesystem &quot;SNOMED&quot;: 'http://snomed.info/sct'\ncodesystem &quot;Medication Request Category Codes&quot;: 'http://terminology.hl7.org/CodeSystem/medicationrequest-category'\n\nvalueset &quot;Active Condition&quot;: 'http://hl7.org/fhir/uv/cpg/ValueSet/cpg-active-condition'\nvalueset &quot;Benzodiazepines&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/benzodiazepines'\n// TODO: Fix this name\nvalueset &quot;End of Life Conditions&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/end-of-life-conditions'\n// Harvested from VSAC - OID: 2.16.840.1.113762.1.4.1108.15\n// NOTE: This harvest note is incorrect, none of the following 3 value sets contain any of the codes in the above referenced valueset\n// Rob will construct an appropriate hospice value set aligned with current eCQM program usage and we will use that when available\nvalueset &quot;Hospice Disposition&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/hospice-disposition'\nvalueset &quot;Hospice Finding Codes&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/hospice-finding'\nvalueset &quot;Hospice Procedure Codes&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/hospice-procedure'\nvalueset &quot;Illicit Drug Screening&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/illicit-drug-urine-screening'\n// Harvested from VSAC - OID: 2.16.840.1.113883.3.526.3.1259\nvalueset &quot;Limited Life Expectancy Conditions&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/limited-life-expectancy-conditions'\nvalueset &quot;Long Acting Opioids&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/long-acting-opioids'\nvalueset &quot;Naloxone&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/naloxone'\nvalueset &quot;Risk Assessment&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/opioid-abuse-assessment'\nvalueset &quot;Opioid Drug Screening&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/opioid-urine-screening'\nvalueset &quot;Ambulatory Abuse Potential Opioids&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/opioids-abused-in-ambulatory-care'\nvalueset &quot;Substance Abuse&quot;: 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/substance-abuse'\n\n// TODO: Turn this into a valueset\ncode &quot;Referral to Hospice&quot;: '306205009' from &quot;SNOMED&quot;\n// TODO: Turn this into a valueset\ncode &quot;Outpatient&quot;: 'outpatient' from &quot;Medication Request Category Codes&quot;\n\n// TODO: Capture process decisions for long-term opioid use\ndefine IsForChronicPain: true\n\ndefine &quot;Active Ambulatory Opioid Rx&quot;:\n  [MedicationRequest: &quot;Ambulatory Abuse Potential Opioids&quot;] Rx\n    where Rx.status = 'active'\n      and ToCodes(Rx.category.coding) contains &quot;Outpatient&quot;\n\ndefine &quot;Active Ambulatory Benzodiazepine Rx&quot;:\n  [MedicationRequest: &quot;Benzodiazepines&quot;] Rx\n    where Rx.status = 'active'\n      and ToCodes(Rx.category.coding) contains &quot;Outpatient&quot;\n\ndefine &quot;Active Ambulatory Naloxone Rx&quot;:\n  [MedicationRequest: &quot;Naloxone&quot;] Rx\n    where Rx.status = 'active'\n      and ToCodes(Rx.category.coding) contains &quot;Outpatient&quot;\n\ndefine &quot;Ambulatory Opioid Rx&quot;:\n  [MedicationRequest: &quot;Ambulatory Abuse Potential Opioids&quot;] Rx\n      where ToCodes(Rx.category.coding) contains &quot;Outpatient&quot;\n\ndefine &quot;End of Life Assessment&quot;:\n  // 1. Conditions indicating end of life or with limited life expectancy\n  exists (\n    (\n      [Condition: &quot;End of Life Conditions&quot;] C\n        where C.clinicalStatus in &quot;Active Condition&quot;\n    )\n    union\n    (\n      [Condition: code in &quot;Limited Life Expectancy Conditions&quot;] C\n        where C.clinicalStatus in &quot;Active Condition&quot;\n    )\n  )\n  // 2. Admitted/referred/discharged to hospice care\n  or exists (\n    (\n      [ServiceRequest: code in &quot;Referral to Hospice&quot;] RR\n        where RR.status in { 'active', 'completed' }\n    )\n    union\n    (\n      [ServiceRequest: code in &quot;Hospice Procedure Codes&quot;] P\n        where P.status in { 'in-progress', 'completed' }\n    )\n    union\n    (\n      [ServiceRequest: code in &quot;Hospice Procedure Codes&quot;] E\n        where E.status in { 'planned', 'arrived', 'in-progress', 'finished', 'onleave' }\n    )\n    union\n    (\n      [Observation: code in &quot;Hospice Finding Codes&quot;] O\n        where not (O.status in { 'unknown', 'entered-in-error', 'cancelled' })\n    )\n    union\n    (\n      [Encounter] E\n        where\n          (\n            if E.hospitalization.dischargeDisposition.coding is null\n                or not exists (E.hospitalization.dischargeDisposition.coding)\n              then false\n            else E.hospitalization.dischargeDisposition in &quot;Hospice Disposition&quot;\n          )\n          and E.status in { 'planned', 'arrived', 'in-progress', 'finished', 'onleave' }\n    )\n  )\n\ndefine function Prescriptions(Orders List&lt;MedicationRequest&gt;):\n  Orders O\n    let\n      // NOTE: Assuming medication is specified as a CodeableConcept with a single RxNorm code\n      rxNormCode: ToCode(O.medication.coding[0]),\n      medicationName: OMTKLogic.GetMedicationName(rxNormCode),\n      // NOTE: Assuming a single dosage instruction element\n      dosageInstruction: O.dosageInstruction[0],\n      repeat: dosageInstruction.timing.repeat,\n      frequency: Coalesce(repeat.frequencyMax.value, repeat.frequency.value),\n      period: System.Quantity { value: repeat.period.value, unit: repeat.periodUnit.value },\n      doseDescription:\n        Coalesce(\n          // There should be a conversion from FHIR.SimpleQuantity to System.Quantity\n          if dosageInstruction.doseAndRate[0].dose is FHIR.Range\n            then ToString(ToQuantity(dosageInstruction.doseAndRate[0].dose.low))\n                          + '-' + ToString(ToQuantity(dosageInstruction.doseAndRate[0].dose.high))\n                          + dosageInstruction.doseAndRate[0].dose.high.unit.value\n            else ToString(ToQuantity(dosageInstruction.doseAndRate[0].dose)),\n            ''\n        ),\n      frequencyDescription:\n        ToString(dosageInstruction.timing.repeat.frequency.value) +\n          Coalesce(\n            '-' + ToString(dosageInstruction.timing.repeat.frequencyMax.value),\n            ''\n          )\n    return {\n      rxNormCode: rxNormCode,\n      isDraft: O.status.value = 'draft',\n      // NOTE: Assuming asNeeded is expressed as a boolean\n      isPRN: dosageInstruction.asNeeded.value,\n      prescription:\n        if dosageInstruction.text is not null then\n          medicationName + ' ' + dosageInstruction.text.value\n        else\n          // TODO: Shouldn't need the .value here on asNeededBoolean\n          medicationName + ' ' + doseDescription + ' q' + frequencyDescription + (if dosageInstruction.asNeeded.value then ' PRN' else ''),\n      // TODO: Shouldn't need the ToQuantity here...\n      dose: if dosageInstruction.doseAndRate[0].dose is FHIR.Range\n            then ToQuantity(dosageInstruction.doseAndRate[0].dose.high)\n            else ToQuantity(dosageInstruction.doseAndRate[0].dose),\n      dosesPerDay: Coalesce(OMTKLogic.ToDaily(frequency, period), 1.0)\n    }\n\ndefine function MME(prescriptions List&lt;MedicationRequest&gt;):\n  (Prescriptions(prescriptions)) P\n    let mme: SingletonFrom(OMTKLogic.CalculateMMEs({ { rxNormCode: P.rxNormCode, doseQuantity: P.dose, dosesPerDay: P.dosesPerDay } }))\n    return {\n      rxNormCode: P.rxNormCode,\n      isDraft: P.isDraft,\n      isPRN: P.isPRN,\n      prescription: P.prescription,\n      dailyDose: mme.dailyDoseDescription,\n      conversionFactor: mme.conversionFactor,\n      mme: mme.mme\n    }\n\ndefine function TotalMME(prescriptions List&lt;MedicationRequest&gt;):\n  System.Quantity {\n    value: Sum((MME(prescriptions)) M return M.mme.value),\n    unit: 'mg/d'\n  }\n\ndefine function ProbableDaysInRange(Orders List&lt;MedicationRequest&gt;, daysPast Integer, numDaysInDaysPast Integer):\n  Orders orders\n    let\n      frequency: orders.dosageInstruction[0].timing.repeat.frequency.value,\n      period: orders.dosageInstruction[0].timing.repeat.period.value,\n      periodDays: GetPeriodDays(orders.dosageInstruction[0].timing.repeat.periodUnit.value),\n      dosesPerDay:\n        if (frequency / (period * periodDays)) &gt;= 1.0\n        then 1.0\n        else frequency / (period * periodDays),\n      repeat: orders.dispenseRequest.numberOfRepeatsAllowed.value,\n      supplyDuration: GetDurationInDays(orders.dispenseRequest.expectedSupplyDuration),\n      validityPeriod: days between orders.dispenseRequest.validityPeriod.&quot;start&quot;.value and Today(),\n      endDifference:\n        if orders.dispenseRequest.validityPeriod.&quot;end&quot;.value &lt; Today()\n        then days between orders.dispenseRequest.validityPeriod.&quot;end&quot;.value and Today()\n        else 0\n    return\n      if (repeat * supplyDuration) &lt; numDaysInDaysPast then false\n      else\n        (dosesPerDay * ((repeat * supplyDuration) / validityPeriod) * (daysPast - endDifference)) &gt;= numDaysInDaysPast\n\ndefine function GetPeriodDays(value System.String):\n  case\n    when value = 'a' then 365.0\n    when value = 'mo' then 30.0\n    when value = 'h' then 1.0/24.0\n    when value = 'min' then 1.0/24.0*60.0\n    when value = 's' then 1.0/24.0*60.0*60.0\n    when value = 'ms' then 1.0/24.0*60.0*60.0*1000.0\n    else 1.0\n  end\n\n  define function GetDurationInDays(value FHIR.Duration):\n    case\n      when StartsWith(value.unit.value, 'a') then value.value.value * 365.0\n      when StartsWith(value.unit.value, 'mo') then value.value.value * 30.0\n      when StartsWith(value.unit.value, 'wk') then value.value.value * 7.0\n      when StartsWith(value.unit.value, 'd') then value.value.value\n      when StartsWith(value.unit.value, 'h') then value.value.value / 24.0\n      when StartsWith(value.unit.value, 'min') then value.value.value / 60.0 / 24.0\n      when StartsWith(value.unit.value, 's') then value.value.value / 60.0 / 60.0 / 24.0\n      when StartsWith(value.unit.value, 'ms') then value.value.value / 60.0 / 60.0 / 24.0 / 1000.0\n      else Message(1000, true, 'Undefined', 'Error', 'Unsupported duration unit')\n    end\n\ndefine function GetIngredient(rxNormCode Code):\n  OMTKLogic.GetIngredients(rxNormCode).ingredientName\n\ndefine function GetIngredients(rxNormCodes List&lt;Code&gt;):\n  rxNormCodes rnc return GetIngredient(rnc)\n\ndefine function GetMedicationNames(medications List&lt;MedicationRequest&gt;):\n  medications M\n    return OMTKLogic.GetIngredients(ToRxNormCode(M.medication.coding)).rxNormCode.display\n\n/*\n*  Conversion Functions\n*/\n\ndefine function ToCode(coding FHIR.Coding):\n  System.Code {\n    code: coding.code.value,\n    system: coding.system.value,\n    version: coding.version.value,\n    display: coding.display.value\n  }\n\ndefine function ToCodes(coding List&lt;FHIR.Coding&gt;):\n  coding c return ToCode(c)\n\ndefine function ToRxNormCode(coding List&lt;FHIR.Coding&gt;):\n  singleton from (\n    coding C where C.system = 'http://www.nlm.nih.gov/research/umls/rxnorm'\n  )\n\ndefine function ToQuantity(quantity FHIR.Quantity):\n  System.Quantity { value: quantity.value.value, unit: quantity.unit.value }\n</code></pre><p><code>No Content</code> (<code>application/elm+xml</code>)</p></div>"
  ] ; # 
  fhir:url [ fhir:v "http://fhir.org/guides/cdc/opioid-cds/Library/OpioidCDSR4Common"^^xsd:anyURI] ; # 
  fhir:version [ fhir:v "1.0.0"] ; # 
  fhir:name [ fhir:v "OpioidCDSR4Common"] ; # 
  fhir:title [ fhir:v "Opioid CDS R4 Common Logic"] ; # 
  fhir:status [ fhir:v "active"] ; # 
  fhir:experimental [ fhir:v "true"^^xsd:boolean] ; # 
  fhir:type [
    ( fhir:coding [
fhir:system [ fhir:v "http://terminology.hl7.org/CodeSystem/library-type"^^xsd:anyURI ] ;
fhir:code [ fhir:v "logic-library" ] ;
fhir:display [ fhir:v "Logic Library" ]     ] )
  ] ; # 
  fhir:date [ fhir:v "2018-03-25T13:49:09-06:00"^^xsd:dateTime] ; # 
  fhir:publisher [ fhir:v "HL7 International - Clinical Decision Support WG"] ; # 
  fhir:contact ( [
fhir:name [ fhir:v "HL7 International - Clinical Decision Support WG" ] ;
    ( fhir:telecom [
fhir:system [ fhir:v "url" ] ;
fhir:value [ fhir:v "http://www.hl7.org/Special/committees/dss/index.cfm" ]     ] )
  ] ) ; # 
  fhir:description [ fhir:v "Common Opioid Decision Support Logic for use in implementing CDC Opioid Prescribing Guidelines."] ; # 
  fhir:useContext ( [
fhir:code [
fhir:system [ fhir:v "http://terminology.hl7.org/CodeSystem/usage-context-type"^^xsd:anyURI ] ;
fhir:code [ fhir:v "focus" ] ;
fhir:display [ fhir:v "Clinical Focus" ]     ] ;
fhir:value [
a fhir:CodeableConcept ;
      ( fhir:coding [
a sct:182888003 ;
fhir:system [ fhir:v "http://snomed.info/sct"^^xsd:anyURI ] ;
fhir:code [ fhir:v "182888003" ] ;
fhir:display [ fhir:v "Medication requested (situation)" ]       ] )     ]
  ] [
fhir:code [
fhir:system [ fhir:v "http://terminology.hl7.org/CodeSystem/usage-context-type"^^xsd:anyURI ] ;
fhir:code [ fhir:v "focus" ] ;
fhir:display [ fhir:v "Clinical Focus" ]     ] ;
fhir:value [
a fhir:CodeableConcept ;
      ( fhir:coding [
a sct:82423001 ;
fhir:system [ fhir:v "http://snomed.info/sct"^^xsd:anyURI ] ;
fhir:code [ fhir:v "82423001" ] ;
fhir:display [ fhir:v "Chronic pain (finding)" ]       ] )     ]
  ] ) ; # 
  fhir:jurisdiction ( [
    ( fhir:coding [
fhir:system [ fhir:v "http://unstats.un.org/unsd/methods/m49/m49.htm"^^xsd:anyURI ] ;
fhir:code [ fhir:v "001" ] ;
fhir:display [ fhir:v "World" ]     ] )
  ] ) ; # 
  fhir:purpose [ fhir:v "This library contains common logic across recommendations including MME calculations, conversions, and looking up codes in valuesets."] ; # 
  fhir:usage [ fhir:v "This library is used for decision support for opioid guideline recommendations when applying PlanDefinitions."] ; # 
  fhir:copyright [ fhir:v "© CDC 2016+."] ; # 
  fhir:topic ( [
fhir:text [ fhir:v "Opioid Prescribing" ]
  ] ) ; # 
  fhir:author ( [
fhir:name [ fhir:v "Kensaku Kawamoto, MD, PhD, MHS" ]
  ] [
fhir:name [ fhir:v "Bryn Rhodes" ]
  ] [
fhir:name [ fhir:v "Floyd Eisenberg, MD, MPH" ]
  ] [
fhir:name [ fhir:v "Robert McClure, MD, MPH" ]
  ] ) ; # 
  fhir:relatedArtifact ( [
fhir:type [ fhir:v "documentation" ] ;
fhir:display [ fhir:v "CDC guideline for prescribing opioids for chronic pain" ] ;
fhir:url [ fhir:v "https://www.cdc.gov/mmwr/volumes/65/rr/rr6501e1.htm?CDC_AA_refVal=https%3A%2F%2Fwww.cdc.gov%2Fmmwr%2Fvolumes%2F65%2Frr%2Frr6501e1er.htm"^^xsd:anyURI ] ;
fhir:document [
fhir:url [ fhir:v "https://www.cdc.gov/mmwr/volumes/65/rr/rr6501e1.htm?CDC_AA_refVal=https%3A%2F%2Fwww.cdc.gov%2Fmmwr%2Fvolumes%2F65%2Frr%2Frr6501e1er.htm"^^xsd:anyURI ]     ]
  ] ) ; # 
  fhir:content ( [
fhir:contentType [ fhir:v "text/cql" ] ;
fhir:data [ fhir:v "library OpioidCDSR4Common version '0.1.0'

using FHIR version '4.0.0'

include FHIRHelpers version '4.0.0' called FHIRHelpers
include OMTKLogic version '0.0.0' called OMTKLogic

codesystem "SNOMED": 'http://snomed.info/sct'
codesystem "Medication Request Category Codes": 'http://terminology.hl7.org/CodeSystem/medicationrequest-category'

valueset "Active Condition": 'http://hl7.org/fhir/uv/cpg/ValueSet/cpg-active-condition'
valueset "Benzodiazepines": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/benzodiazepines'
// TODO: Fix this name
valueset "End of Life Conditions": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/end-of-life-conditions'
// Harvested from VSAC - OID: 2.16.840.1.113762.1.4.1108.15
// NOTE: This harvest note is incorrect, none of the following 3 value sets contain any of the codes in the above referenced valueset
// Rob will construct an appropriate hospice value set aligned with current eCQM program usage and we will use that when available
valueset "Hospice Disposition": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/hospice-disposition'
valueset "Hospice Finding Codes": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/hospice-finding'
valueset "Hospice Procedure Codes": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/hospice-procedure'
valueset "Illicit Drug Screening": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/illicit-drug-urine-screening'
// Harvested from VSAC - OID: 2.16.840.1.113883.3.526.3.1259
valueset "Limited Life Expectancy Conditions": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/limited-life-expectancy-conditions'
valueset "Long Acting Opioids": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/long-acting-opioids'
valueset "Naloxone": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/naloxone'
valueset "Risk Assessment": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/opioid-abuse-assessment'
valueset "Opioid Drug Screening": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/opioid-urine-screening'
valueset "Ambulatory Abuse Potential Opioids": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/opioids-abused-in-ambulatory-care'
valueset "Substance Abuse": 'http://fhir.org/guides/cdc/opioid-cds/ValueSet/substance-abuse'

// TODO: Turn this into a valueset
code "Referral to Hospice": '306205009' from "SNOMED"
// TODO: Turn this into a valueset
code "Outpatient": 'outpatient' from "Medication Request Category Codes"

// TODO: Capture process decisions for long-term opioid use
define IsForChronicPain: true

define "Active Ambulatory Opioid Rx":
  [MedicationRequest: "Ambulatory Abuse Potential Opioids"] Rx
    where Rx.status = 'active'
      and ToCodes(Rx.category.coding) contains "Outpatient"

define "Active Ambulatory Benzodiazepine Rx":
  [MedicationRequest: "Benzodiazepines"] Rx
    where Rx.status = 'active'
      and ToCodes(Rx.category.coding) contains "Outpatient"

define "Active Ambulatory Naloxone Rx":
  [MedicationRequest: "Naloxone"] Rx
    where Rx.status = 'active'
      and ToCodes(Rx.category.coding) contains "Outpatient"

define "Ambulatory Opioid Rx":
  [MedicationRequest: "Ambulatory Abuse Potential Opioids"] Rx
      where ToCodes(Rx.category.coding) contains "Outpatient"

define "End of Life Assessment":
  // 1. Conditions indicating end of life or with limited life expectancy
  exists (
    (
      [Condition: "End of Life Conditions"] C
        where C.clinicalStatus in "Active Condition"
    )
    union
    (
      [Condition: code in "Limited Life Expectancy Conditions"] C
        where C.clinicalStatus in "Active Condition"
    )
  )
  // 2. Admitted/referred/discharged to hospice care
  or exists (
    (
      [ServiceRequest: code in "Referral to Hospice"] RR
        where RR.status in { 'active', 'completed' }
    )
    union
    (
      [ServiceRequest: code in "Hospice Procedure Codes"] P
        where P.status in { 'in-progress', 'completed' }
    )
    union
    (
      [ServiceRequest: code in "Hospice Procedure Codes"] E
        where E.status in { 'planned', 'arrived', 'in-progress', 'finished', 'onleave' }
    )
    union
    (
      [Observation: code in "Hospice Finding Codes"] O
        where not (O.status in { 'unknown', 'entered-in-error', 'cancelled' })
    )
    union
    (
      [Encounter] E
        where
          (
            if E.hospitalization.dischargeDisposition.coding is null
                or not exists (E.hospitalization.dischargeDisposition.coding)
              then false
            else E.hospitalization.dischargeDisposition in "Hospice Disposition"
          )
          and E.status in { 'planned', 'arrived', 'in-progress', 'finished', 'onleave' }
    )
  )

define function Prescriptions(Orders List<MedicationRequest>):
  Orders O
    let
      // NOTE: Assuming medication is specified as a CodeableConcept with a single RxNorm code
      rxNormCode: ToCode(O.medication.coding[0]),
      medicationName: OMTKLogic.GetMedicationName(rxNormCode),
      // NOTE: Assuming a single dosage instruction element
      dosageInstruction: O.dosageInstruction[0],
      repeat: dosageInstruction.timing.repeat,
      frequency: Coalesce(repeat.frequencyMax.value, repeat.frequency.value),
      period: System.Quantity { value: repeat.period.value, unit: repeat.periodUnit.value },
      doseDescription:
        Coalesce(
          // There should be a conversion from FHIR.SimpleQuantity to System.Quantity
          if dosageInstruction.doseAndRate[0].dose is FHIR.Range
            then ToString(ToQuantity(dosageInstruction.doseAndRate[0].dose.low))
                          + '-' + ToString(ToQuantity(dosageInstruction.doseAndRate[0].dose.high))
                          + dosageInstruction.doseAndRate[0].dose.high.unit.value
            else ToString(ToQuantity(dosageInstruction.doseAndRate[0].dose)),
            ''
        ),
      frequencyDescription:
        ToString(dosageInstruction.timing.repeat.frequency.value) +
          Coalesce(
            '-' + ToString(dosageInstruction.timing.repeat.frequencyMax.value),
            ''
          )
    return {
      rxNormCode: rxNormCode,
      isDraft: O.status.value = 'draft',
      // NOTE: Assuming asNeeded is expressed as a boolean
      isPRN: dosageInstruction.asNeeded.value,
      prescription:
        if dosageInstruction.text is not null then
          medicationName + ' ' + dosageInstruction.text.value
        else
          // TODO: Shouldn't need the .value here on asNeededBoolean
          medicationName + ' ' + doseDescription + ' q' + frequencyDescription + (if dosageInstruction.asNeeded.value then ' PRN' else ''),
      // TODO: Shouldn't need the ToQuantity here...
      dose: if dosageInstruction.doseAndRate[0].dose is FHIR.Range
            then ToQuantity(dosageInstruction.doseAndRate[0].dose.high)
            else ToQuantity(dosageInstruction.doseAndRate[0].dose),
      dosesPerDay: Coalesce(OMTKLogic.ToDaily(frequency, period), 1.0)
    }

define function MME(prescriptions List<MedicationRequest>):
  (Prescriptions(prescriptions)) P
    let mme: SingletonFrom(OMTKLogic.CalculateMMEs({ { rxNormCode: P.rxNormCode, doseQuantity: P.dose, dosesPerDay: P.dosesPerDay } }))
    return {
      rxNormCode: P.rxNormCode,
      isDraft: P.isDraft,
      isPRN: P.isPRN,
      prescription: P.prescription,
      dailyDose: mme.dailyDoseDescription,
      conversionFactor: mme.conversionFactor,
      mme: mme.mme
    }

define function TotalMME(prescriptions List<MedicationRequest>):
  System.Quantity {
    value: Sum((MME(prescriptions)) M return M.mme.value),
    unit: 'mg/d'
  }

define function ProbableDaysInRange(Orders List<MedicationRequest>, daysPast Integer, numDaysInDaysPast Integer):
  Orders orders
    let
      frequency: orders.dosageInstruction[0].timing.repeat.frequency.value,
      period: orders.dosageInstruction[0].timing.repeat.period.value,
      periodDays: GetPeriodDays(orders.dosageInstruction[0].timing.repeat.periodUnit.value),
      dosesPerDay:
        if (frequency / (period * periodDays)) >= 1.0
        then 1.0
        else frequency / (period * periodDays),
      repeat: orders.dispenseRequest.numberOfRepeatsAllowed.value,
      supplyDuration: GetDurationInDays(orders.dispenseRequest.expectedSupplyDuration),
      validityPeriod: days between orders.dispenseRequest.validityPeriod."start".value and Today(),
      endDifference:
        if orders.dispenseRequest.validityPeriod."end".value < Today()
        then days between orders.dispenseRequest.validityPeriod."end".value and Today()
        else 0
    return
      if (repeat * supplyDuration) < numDaysInDaysPast then false
      else
        (dosesPerDay * ((repeat * supplyDuration) / validityPeriod) * (daysPast - endDifference)) >= numDaysInDaysPast

define function GetPeriodDays(value System.String):
  case
    when value = 'a' then 365.0
    when value = 'mo' then 30.0
    when value = 'h' then 1.0/24.0
    when value = 'min' then 1.0/24.0*60.0
    when value = 's' then 1.0/24.0*60.0*60.0
    when value = 'ms' then 1.0/24.0*60.0*60.0*1000.0
    else 1.0
  end

  define function GetDurationInDays(value FHIR.Duration):
    case
      when StartsWith(value.unit.value, 'a') then value.value.value * 365.0
      when StartsWith(value.unit.value, 'mo') then value.value.value * 30.0
      when StartsWith(value.unit.value, 'wk') then value.value.value * 7.0
      when StartsWith(value.unit.value, 'd') then value.value.value
      when StartsWith(value.unit.value, 'h') then value.value.value / 24.0
      when StartsWith(value.unit.value, 'min') then value.value.value / 60.0 / 24.0
      when StartsWith(value.unit.value, 's') then value.value.value / 60.0 / 60.0 / 24.0
      when StartsWith(value.unit.value, 'ms') then value.value.value / 60.0 / 60.0 / 24.0 / 1000.0
      else Message(1000, true, 'Undefined', 'Error', 'Unsupported duration unit')
    end

define function GetIngredient(rxNormCode Code):
  OMTKLogic.GetIngredients(rxNormCode).ingredientName

define function GetIngredients(rxNormCodes List<Code>):
  rxNormCodes rnc return GetIngredient(rnc)

define function GetMedicationNames(medications List<MedicationRequest>):
  medications M
    return OMTKLogic.GetIngredients(ToRxNormCode(M.medication.coding)).rxNormCode.display

/*
*  Conversion Functions
*/

define function ToCode(coding FHIR.Coding):
  System.Code {
    code: coding.code.value,
    system: coding.system.value,
    version: coding.version.value,
    display: coding.display.value
  }

define function ToCodes(coding List<FHIR.Coding>):
  coding c return ToCode(c)

define function ToRxNormCode(coding List<FHIR.Coding>):
  singleton from (
    coding C where C.system = 'http://www.nlm.nih.gov/research/umls/rxnorm'
  )

define function ToQuantity(quantity FHIR.Quantity):
  System.Quantity { value: quantity.value.value, unit: quantity.unit.value }
"^^xsd:base64Binary ]
  ] [
fhir:contentType [ fhir:v "application/elm+xml" ]
  ] ) . #