WHO Immunization Implementation Guide
1.0.0 - release

WHO Immunization Implementation Guide, published by WHO. This guide is not an authorized publication; it is the continuous build for version 1.0.0 built by the FHIR (HL7® FHIR® Standard) CI Build. This version is based on the current content of https://github.com/WorldHealthOrganization/smart-immunizations/tree/132/merge and changes regularly. See the Directory of published versions

: IMMZCommon - JSON Representation

Draft as of 2026-03-31

Raw json | Download

{
  "resourceType" : "Library",
  "id" : "IMMZCommon",
  "meta" : {
    "profile" : [
      🔗 "http://hl7.org/fhir/uv/crmi/StructureDefinition/crmi-shareablelibrary"🔗 ,
      "http://hl7.org/fhir/uv/crmi/StructureDefinition/crmi-publishablelibrary"🔗 ,
      "http://hl7.org/fhir/uv/cql/StructureDefinition/cql-library"🔗 ,
      "http://hl7.org/fhir/uv/cql/StructureDefinition/cql-module"
    ]
  },
  "text" : {
    "status" : "extensions",
    "div" : "<div xmlns=\"http://www.w3.org/1999/xhtml\">\n<div>\n    <table class=\"grid dict\">\n        \n        \n        <tr>\n            <th scope=\"row\"><b>Title: </b></th>\n            <td style=\"padding-left: 4px;\">IMMZCommon</td>\n        </tr>\n        \n\n        \n        \n        <tr>\n            <th scope=\"row\"><b>Id: </b></th>\n            <td style=\"padding-left: 4px;\">IMMZCommon</td>\n        </tr>\n        \n\n        \n        \n        <tr>\n            <th scope=\"row\"><b>Version: </b></th>\n            <td style=\"padding-left: 4px;\">1.0.0</td>\n        </tr>\n        \n\n        \n        <tr>\n            <th scope=\"row\"><b>Url: </b></th>\n            <td style=\"padding-left: 4px;\"><a href=\"Library-IMMZCommon.html\">IMMZCommon</a></td>\n        </tr>\n        \n\n        \n\n        \n\n        \n        <tr>\n            <th scope=\"row\"><b>Status: </b></th>\n            <td style=\"padding-left: 4px;\">draft</td>\n        </tr>\n        \n\n        \n\n        \n        <tr>\n            <th scope=\"row\"><b>Type: </b></th>\n            <td style=\"padding-left: 4px;\">\n                \n                    \n                        \n                        <p style=\"margin-bottom: 5px;\">\n                            <b>system: </b> <span><a href=\"http://terminology.hl7.org/7.1.0/CodeSystem-library-type.html\">http://terminology.hl7.org/CodeSystem/library-type</a></span>\n                        </p>\n                        \n                        \n                        <p style=\"margin-bottom: 5px;\">\n                            <b>code: </b> <span>logic-library</span>\n                        </p>\n                        \n                        \n                    \n                \n                \n            </td>\n        </tr>\n        \n\n        \n\n        \n        <tr>\n            <th scope=\"row\"><b>Date: </b></th>\n            <td style=\"padding-left: 4px;\">2026-03-31 13:54:17+0000</td>\n        </tr>\n        \n\n        \n        <tr>\n            <th scope=\"row\"><b>Publisher: </b></th>\n            <td style=\"padding-left: 4px;\">WHO</td>\n        </tr>\n        \n\n        \n        <tr>\n            <th scope=\"row\"><b>Description: </b></th>\n            <td style=\"padding-left: 4px;\"><div><p>This library defines common terminologies and functions used throughout the Immunization CPG</p>\n</div></td>\n        </tr>\n        \n\n        \n\n        \n\n        \n\n        \n\n        \n\n        \n\n        \n\n        \n\n        \n\n        \n\n        \n        <tr>\n            <th scope=\"row\"><b>Related Artifacts: </b></th>\n            <td style=\"padding-left: 4px;\">\n                \n                \n                \n                <p><b>Dependencies</b></p>\n                <ul>\n                  \n                    <li><a href=\"Library-WHOCommon.html\">WHOCommon</a></li>\n                  \n                </ul>\n                \n                \n                \n                \n                \n            </td>\n        </tr>\n        \n\n        \n\n        \n        <tr>\n          <th scope=\"row\"><b>Data Requirements:</b></th>\n          <td style=\"padding-left: 4px;\">\n            <table class=\"grid-dict\">\n              <tr><th><b>Type</b></th><th><b>Profile</b></th><th><b>MS</b></th><th><b>Code Filter</b></th></tr>\n              \n                <tr>\n                  <th>Medication</th>\n                  <th>http://hl7.org/fhir/StructureDefinition/Medication</th>\n                  <th/>\n                  <th>\n                    \n                  </th>\n                </tr>\n              \n            </table>\n          </td>\n        </tr>\n        \n\n        \n        \n        <tr>\n          <td colspan=\"2\">\n            <table>\n              <tr><th><a id=\"cql-content\"><b>Content: </b></a> text/cql</th></tr>\n              <tr><td><pre><code class=\"language-cql\">library IMMZCommon\r\n\r\nusing FHIR version '4.0.1'\r\n\r\ninclude FHIRHelpers version '4.0.1'\r\ninclude WHOCommon called WC\r\n\r\n/**\r\n * @description Fetches a singleton protocol applied from an immunization\r\n * @comment The protocol list from the immunization\r\n */\r\ndefine function Only(protocols List&lt;FHIR.Immunization.ProtocolApplied&gt;):\r\n  singleton from protocols\r\n\r\n/**\r\n * @description Fetches a singleton protocol applied from an immunization\r\n * @comment The protocol list from the immunization\r\n */\r\ndefine fluent function only(protocols List&lt;FHIR.Immunization.ProtocolApplied&gt;):\r\n  singleton from protocols\r\n\r\n/**\r\n * @description Takes the date choice of a date/string choice (for Immunization date)\r\n */\r\ndefine function ToDate(choice Choice&lt;FHIR.date, FHIR.string&gt;):\r\n  case\r\n\t  when choice is FHIR.date then\r\n    \tchoice as FHIR.date\r\n\t\telse\r\n      Message(null as FHIR.date, true, '1', 'Error', 'Cannot compute a date from a String value')\r\n\tend\r\n\r\n/**\r\n * @description Takes the date choice of a date/string choice (for Immunization date)\r\n */\r\ndefine function ToDateTime(choice Choice&lt;FHIR.dateTime, FHIR.string&gt;):\r\n  case\r\n\t  when choice is FHIR.dateTime then\r\n    \tchoice as FHIR.dateTime\r\n\t\telse\r\n      Message(null as FHIR.dateTime, true, '1', 'Error', 'Cannot compute a date from a String value')\r\n\tend\r\n\r\n\r\n/**\r\n * @description Takes a choice of FHIR.string and FHIR.positiveInt and ensures the result is a FHIR.positiveInt\r\n */\r\ndefine function ToPositiveInt(choice Choice&lt;FHIR.positiveInt, FHIR.string&gt;):\r\n  case\r\n\t  when choice is FHIR.positiveInt then\r\n    \tchoice as FHIR.positiveInt\r\n\t\telse\r\n      Message(null as FHIR.positiveInt, true, '1', 'Error', 'Cannot compute a positive from a String value') // TODO: I'm sure that this is supported somehow?\r\n\tend\r\n\r\n\r\n/**\r\n * @description Takes a choice between a Medication and a CodeableConcept and returns just the code of the medication\r\n */\r\ndefine function ExtractMedicationCode(choice Choice&lt;FHIR.CodeableConcept, FHIR.Reference&gt;):\r\n  case\r\n\t  when choice is FHIR.CodeableConcept then\r\n    \tchoice as FHIR.CodeableConcept\r\n    when choice is FHIR.Reference then\r\n      First([Medication] M \r\n        where M.id = Last(Split(choice.reference, '/'))\r\n        return M.code as FHIR.CodeableConcept)\r\n\t\telse\r\n      Message(null as FHIR.CodeableConcept, true, '1', 'Error', 'Cannot compute a medication code') // TODO: I'm sure that this is supported somehow?\r\n\tend\r\n\r\n\r\n/**\r\n * @description Takes a choice between a Medication and a CodeableConcept and returns just the code of the medication\r\n */\r\ndefine function ExtractMedicationInitiationDate(choice Choice&lt;FHIR.dateTime, FHIR.Period&gt;):\r\n  case\r\n\t  when choice is FHIR.Period then\r\n    \tstart of (choice as FHIR.Period)\r\n    when choice is FHIR.dateTime then\r\n      choice as FHIR.dateTime\r\n\t\telse\r\n      Message(null as FHIR.dateTime, true, '1', 'Error', 'Cannot compute medication treatment initiation date') // TODO: I'm sure that this is supported somehow?\r\n\tend\r\n\r\n/**\r\n * @description: Gets the type of antigen dose extension value from an Immunization\r\n */\r\ndefine fluent function typeOfDose(immz Immunization):\r\n  (First(\r\n    immz.extension E where E.url = 'http://smart.who.int/immunizations/StructureDefinition/IMMZTypeOfDose'\r\n  )).value as FHIR.CodeableConcept\r\n\r\n/**\r\n * @description: Gets the type of antigen dose extension value from an Immunization\r\n */\r\ndefine fluent function brand(immz Immunization):\r\n  (First(\r\n    immz.extension E where E.url = 'http://smart.who.int/immunizations/StructureDefinition/IMMZVaccineBrand'\r\n  )).value as FHIR.CodeableConcept\r\n\r\n/**\r\n * @description: Gets a given immunization from a list that matches the dose number\r\n */\r\ndefine fluent function getDose(immunizations List&lt;Immunization&gt;, doseNumber String):\r\n  immunizations I where\r\n    exists( I.protocolApplied pa where pa.doseNumber = doseNumber )\r\n\r\n/**\r\n * @description: Gets a given immunization from a list that matches the dose number\r\n */\r\ndefine fluent function getDose(immunization Immunization, doseNumber String):\r\n  immunization I where\r\n    exists( I.protocolApplied pa where pa.doseNumber = doseNumber )\r\n\r\n/**\r\n * @description: Gets immunizations on or before a date\r\n */\r\ndefine fluent function onOrBefore(immunizations List&lt;Immunization&gt;, beforeDate Date):\r\n  immunizations I where\r\n    I.occurrence.toInterval() same day or before beforeDate\r\n\r\n/**\r\n * @description: Gets the series doses from an immunization\r\n */\r\ndefine fluent function seriesDoses(immunization Immunization):\r\n  First(immunization.protocolApplied pa where pa.seriesDoses is not null).seriesDoses\r\n\r\n/**\r\n * @description: Gets observation from an encounter or on or before a date\r\n */\r\ndefine fluent function encounterOrOnBefore(observations List&lt;Observation&gt;, EncounterId String, beforeDate Date):\r\n  observations O where\r\n    (O.encounter.references(EncounterId)\r\n      or O.effective.toInterval() starts same day or before beforeDate)\r\n\r\n/**\r\n * @description: Gets the doses from the primary series\r\n */\r\ndefine fluent function seriesPrimary(immunizations List&lt;Immunization&gt;):\r\n  immunizations I where\r\n    exists( I.protocolApplied pa where pa.series = 'Primary series' )\r\n\r\n/**\r\n * @description: Gets the doses from the dose 0 series\r\n */\r\ndefine fluent function seriesDose0(immunizations List&lt;Immunization&gt;):\r\n  immunizations I where\r\n    exists( I.protocolApplied pa where pa.series = 'Dose 0' )\r\n\r\n/**\r\n * @description: Gets the doses from the Booster series\r\n */\r\ndefine fluent function seriesBooster(immunizations List&lt;Immunization&gt;):\r\n  immunizations I where\r\n    exists( I.protocolApplied pa where pa.series = 'Booster dose' )\r\n\r\n/**\r\n * @description: Gets the doses from the Supplementary series\r\n */\r\ndefine fluent function seriesSupplementary(immunizations List&lt;Immunization&gt;):\r\n  immunizations I where\r\n    exists( I.protocolApplied pa where pa.series = 'Supplementary dose' )\r\n\r\n/**\r\n * @description: Sorts a list and returns the requested index\r\n */\r\ndefine fluent function sortedIndex(immunizations List&lt;Immunization&gt;, idx Integer):\r\n  if exists( immunizations ) then\r\n    (immunizations I sort by start of occurrence.toInterval())[idx]\r\n  else null\r\n</code></pre></td></tr>\n            </table>\n          </td>\n        </tr>\n        \n        \n        \n    </table>\n</div>\n</div>"
  },
  "extension" : [
    {
      "url" : "http://hl7.org/fhir/StructureDefinition/cqf-knowledgeCapability",
      "valueCode" : "computable"
    }
  ],
  "url" : "http://smart.who.int/immunizations/Library/IMMZCommon",
  "version" : "1.0.0",
  "name" : "IMMZCommon",
  "title" : "IMMZCommon",
  "status" : "draft",
  "experimental" : false,
  "type" : {
    "coding" : [
      {
        "system" : "http://terminology.hl7.org/CodeSystem/library-type",
        "code" : "logic-library"
      }
    ]
  },
  "date" : "2026-03-31T13:54:17+00:00",
  "publisher" : "WHO",
  "contact" : [
    {
      "name" : "WHO",
      "telecom" : [
        {
          "system" : "url",
          "value" : "http://who.int"
        }
      ]
    }
  ],
  "description" : "This library defines common terminologies and functions used throughout the Immunization CPG",
  "relatedArtifact" : [
    {
      "type" : "depends-on",
      "display" : "Library WC",
      "resource" : "http://smart.who.int/immunizations/Library/WHOCommon"
    }
  ],
  "dataRequirement" : [
    {
      "type" : "Medication",
      "profile" : [
        🔗 "http://hl7.org/fhir/StructureDefinition/Medication"
      ]
    }
  ],
  "content" : [
    {
      "contentType" : "text/cql",
      "data" : "bGlicmFyeSBJTU1aQ29tbW9uDQoNCnVzaW5nIEZISVIgdmVyc2lvbiAnNC4wLjEnDQoNCmluY2x1ZGUgRkhJUkhlbHBlcnMgdmVyc2lvbiAnNC4wLjEnDQppbmNsdWRlIFdIT0NvbW1vbiBjYWxsZWQgV0MNCg0KLyoqDQogKiBAZGVzY3JpcHRpb24gRmV0Y2hlcyBhIHNpbmdsZXRvbiBwcm90b2NvbCBhcHBsaWVkIGZyb20gYW4gaW1tdW5pemF0aW9uDQogKiBAY29tbWVudCBUaGUgcHJvdG9jb2wgbGlzdCBmcm9tIHRoZSBpbW11bml6YXRpb24NCiAqLw0KZGVmaW5lIGZ1bmN0aW9uIE9ubHkocHJvdG9jb2xzIExpc3Q8RkhJUi5JbW11bml6YXRpb24uUHJvdG9jb2xBcHBsaWVkPik6DQogIHNpbmdsZXRvbiBmcm9tIHByb3RvY29scw0KDQovKioNCiAqIEBkZXNjcmlwdGlvbiBGZXRjaGVzIGEgc2luZ2xldG9uIHByb3RvY29sIGFwcGxpZWQgZnJvbSBhbiBpbW11bml6YXRpb24NCiAqIEBjb21tZW50IFRoZSBwcm90b2NvbCBsaXN0IGZyb20gdGhlIGltbXVuaXphdGlvbg0KICovDQpkZWZpbmUgZmx1ZW50IGZ1bmN0aW9uIG9ubHkocHJvdG9jb2xzIExpc3Q8RkhJUi5JbW11bml6YXRpb24uUHJvdG9jb2xBcHBsaWVkPik6DQogIHNpbmdsZXRvbiBmcm9tIHByb3RvY29scw0KDQovKioNCiAqIEBkZXNjcmlwdGlvbiBUYWtlcyB0aGUgZGF0ZSBjaG9pY2Ugb2YgYSBkYXRlL3N0cmluZyBjaG9pY2UgKGZvciBJbW11bml6YXRpb24gZGF0ZSkNCiAqLw0KZGVmaW5lIGZ1bmN0aW9uIFRvRGF0ZShjaG9pY2UgQ2hvaWNlPEZISVIuZGF0ZSwgRkhJUi5zdHJpbmc+KToNCiAgY2FzZQ0KCSAgd2hlbiBjaG9pY2UgaXMgRkhJUi5kYXRlIHRoZW4NCiAgICAJY2hvaWNlIGFzIEZISVIuZGF0ZQ0KCQllbHNlDQogICAgICBNZXNzYWdlKG51bGwgYXMgRkhJUi5kYXRlLCB0cnVlLCAnMScsICdFcnJvcicsICdDYW5ub3QgY29tcHV0ZSBhIGRhdGUgZnJvbSBhIFN0cmluZyB2YWx1ZScpDQoJZW5kDQoNCi8qKg0KICogQGRlc2NyaXB0aW9uIFRha2VzIHRoZSBkYXRlIGNob2ljZSBvZiBhIGRhdGUvc3RyaW5nIGNob2ljZSAoZm9yIEltbXVuaXphdGlvbiBkYXRlKQ0KICovDQpkZWZpbmUgZnVuY3Rpb24gVG9EYXRlVGltZShjaG9pY2UgQ2hvaWNlPEZISVIuZGF0ZVRpbWUsIEZISVIuc3RyaW5nPik6DQogIGNhc2UNCgkgIHdoZW4gY2hvaWNlIGlzIEZISVIuZGF0ZVRpbWUgdGhlbg0KICAgIAljaG9pY2UgYXMgRkhJUi5kYXRlVGltZQ0KCQllbHNlDQogICAgICBNZXNzYWdlKG51bGwgYXMgRkhJUi5kYXRlVGltZSwgdHJ1ZSwgJzEnLCAnRXJyb3InLCAnQ2Fubm90IGNvbXB1dGUgYSBkYXRlIGZyb20gYSBTdHJpbmcgdmFsdWUnKQ0KCWVuZA0KDQoNCi8qKg0KICogQGRlc2NyaXB0aW9uIFRha2VzIGEgY2hvaWNlIG9mIEZISVIuc3RyaW5nIGFuZCBGSElSLnBvc2l0aXZlSW50IGFuZCBlbnN1cmVzIHRoZSByZXN1bHQgaXMgYSBGSElSLnBvc2l0aXZlSW50DQogKi8NCmRlZmluZSBmdW5jdGlvbiBUb1Bvc2l0aXZlSW50KGNob2ljZSBDaG9pY2U8RkhJUi5wb3NpdGl2ZUludCwgRkhJUi5zdHJpbmc+KToNCiAgY2FzZQ0KCSAgd2hlbiBjaG9pY2UgaXMgRkhJUi5wb3NpdGl2ZUludCB0aGVuDQogICAgCWNob2ljZSBhcyBGSElSLnBvc2l0aXZlSW50DQoJCWVsc2UNCiAgICAgIE1lc3NhZ2UobnVsbCBhcyBGSElSLnBvc2l0aXZlSW50LCB0cnVlLCAnMScsICdFcnJvcicsICdDYW5ub3QgY29tcHV0ZSBhIHBvc2l0aXZlIGZyb20gYSBTdHJpbmcgdmFsdWUnKSAvLyBUT0RPOiBJJ20gc3VyZSB0aGF0IHRoaXMgaXMgc3VwcG9ydGVkIHNvbWVob3c/DQoJZW5kDQoNCg0KLyoqDQogKiBAZGVzY3JpcHRpb24gVGFrZXMgYSBjaG9pY2UgYmV0d2VlbiBhIE1lZGljYXRpb24gYW5kIGEgQ29kZWFibGVDb25jZXB0IGFuZCByZXR1cm5zIGp1c3QgdGhlIGNvZGUgb2YgdGhlIG1lZGljYXRpb24NCiAqLw0KZGVmaW5lIGZ1bmN0aW9uIEV4dHJhY3RNZWRpY2F0aW9uQ29kZShjaG9pY2UgQ2hvaWNlPEZISVIuQ29kZWFibGVDb25jZXB0LCBGSElSLlJlZmVyZW5jZT4pOg0KICBjYXNlDQoJICB3aGVuIGNob2ljZSBpcyBGSElSLkNvZGVhYmxlQ29uY2VwdCB0aGVuDQogICAgCWNob2ljZSBhcyBGSElSLkNvZGVhYmxlQ29uY2VwdA0KICAgIHdoZW4gY2hvaWNlIGlzIEZISVIuUmVmZXJlbmNlIHRoZW4NCiAgICAgIEZpcnN0KFtNZWRpY2F0aW9uXSBNIA0KICAgICAgICB3aGVyZSBNLmlkID0gTGFzdChTcGxpdChjaG9pY2UucmVmZXJlbmNlLCAnLycpKQ0KICAgICAgICByZXR1cm4gTS5jb2RlIGFzIEZISVIuQ29kZWFibGVDb25jZXB0KQ0KCQllbHNlDQogICAgICBNZXNzYWdlKG51bGwgYXMgRkhJUi5Db2RlYWJsZUNvbmNlcHQsIHRydWUsICcxJywgJ0Vycm9yJywgJ0Nhbm5vdCBjb21wdXRlIGEgbWVkaWNhdGlvbiBjb2RlJykgLy8gVE9ETzogSSdtIHN1cmUgdGhhdCB0aGlzIGlzIHN1cHBvcnRlZCBzb21laG93Pw0KCWVuZA0KDQoNCi8qKg0KICogQGRlc2NyaXB0aW9uIFRha2VzIGEgY2hvaWNlIGJldHdlZW4gYSBNZWRpY2F0aW9uIGFuZCBhIENvZGVhYmxlQ29uY2VwdCBhbmQgcmV0dXJucyBqdXN0IHRoZSBjb2RlIG9mIHRoZSBtZWRpY2F0aW9uDQogKi8NCmRlZmluZSBmdW5jdGlvbiBFeHRyYWN0TWVkaWNhdGlvbkluaXRpYXRpb25EYXRlKGNob2ljZSBDaG9pY2U8RkhJUi5kYXRlVGltZSwgRkhJUi5QZXJpb2Q+KToNCiAgY2FzZQ0KCSAgd2hlbiBjaG9pY2UgaXMgRkhJUi5QZXJpb2QgdGhlbg0KICAgIAlzdGFydCBvZiAoY2hvaWNlIGFzIEZISVIuUGVyaW9kKQ0KICAgIHdoZW4gY2hvaWNlIGlzIEZISVIuZGF0ZVRpbWUgdGhlbg0KICAgICAgY2hvaWNlIGFzIEZISVIuZGF0ZVRpbWUNCgkJZWxzZQ0KICAgICAgTWVzc2FnZShudWxsIGFzIEZISVIuZGF0ZVRpbWUsIHRydWUsICcxJywgJ0Vycm9yJywgJ0Nhbm5vdCBjb21wdXRlIG1lZGljYXRpb24gdHJlYXRtZW50IGluaXRpYXRpb24gZGF0ZScpIC8vIFRPRE86IEknbSBzdXJlIHRoYXQgdGhpcyBpcyBzdXBwb3J0ZWQgc29tZWhvdz8NCgllbmQNCg0KLyoqDQogKiBAZGVzY3JpcHRpb246IEdldHMgdGhlIHR5cGUgb2YgYW50aWdlbiBkb3NlIGV4dGVuc2lvbiB2YWx1ZSBmcm9tIGFuIEltbXVuaXphdGlvbg0KICovDQpkZWZpbmUgZmx1ZW50IGZ1bmN0aW9uIHR5cGVPZkRvc2UoaW1teiBJbW11bml6YXRpb24pOg0KICAoRmlyc3QoDQogICAgaW1tei5leHRlbnNpb24gRSB3aGVyZSBFLnVybCA9ICdodHRwOi8vc21hcnQud2hvLmludC9pbW11bml6YXRpb25zL1N0cnVjdHVyZURlZmluaXRpb24vSU1NWlR5cGVPZkRvc2UnDQogICkpLnZhbHVlIGFzIEZISVIuQ29kZWFibGVDb25jZXB0DQoNCi8qKg0KICogQGRlc2NyaXB0aW9uOiBHZXRzIHRoZSB0eXBlIG9mIGFudGlnZW4gZG9zZSBleHRlbnNpb24gdmFsdWUgZnJvbSBhbiBJbW11bml6YXRpb24NCiAqLw0KZGVmaW5lIGZsdWVudCBmdW5jdGlvbiBicmFuZChpbW16IEltbXVuaXphdGlvbik6DQogIChGaXJzdCgNCiAgICBpbW16LmV4dGVuc2lvbiBFIHdoZXJlIEUudXJsID0gJ2h0dHA6Ly9zbWFydC53aG8uaW50L2ltbXVuaXphdGlvbnMvU3RydWN0dXJlRGVmaW5pdGlvbi9JTU1aVmFjY2luZUJyYW5kJw0KICApKS52YWx1ZSBhcyBGSElSLkNvZGVhYmxlQ29uY2VwdA0KDQovKioNCiAqIEBkZXNjcmlwdGlvbjogR2V0cyBhIGdpdmVuIGltbXVuaXphdGlvbiBmcm9tIGEgbGlzdCB0aGF0IG1hdGNoZXMgdGhlIGRvc2UgbnVtYmVyDQogKi8NCmRlZmluZSBmbHVlbnQgZnVuY3Rpb24gZ2V0RG9zZShpbW11bml6YXRpb25zIExpc3Q8SW1tdW5pemF0aW9uPiwgZG9zZU51bWJlciBTdHJpbmcpOg0KICBpbW11bml6YXRpb25zIEkgd2hlcmUNCiAgICBleGlzdHMoIEkucHJvdG9jb2xBcHBsaWVkIHBhIHdoZXJlIHBhLmRvc2VOdW1iZXIgPSBkb3NlTnVtYmVyICkNCg0KLyoqDQogKiBAZGVzY3JpcHRpb246IEdldHMgYSBnaXZlbiBpbW11bml6YXRpb24gZnJvbSBhIGxpc3QgdGhhdCBtYXRjaGVzIHRoZSBkb3NlIG51bWJlcg0KICovDQpkZWZpbmUgZmx1ZW50IGZ1bmN0aW9uIGdldERvc2UoaW1tdW5pemF0aW9uIEltbXVuaXphdGlvbiwgZG9zZU51bWJlciBTdHJpbmcpOg0KICBpbW11bml6YXRpb24gSSB3aGVyZQ0KICAgIGV4aXN0cyggSS5wcm90b2NvbEFwcGxpZWQgcGEgd2hlcmUgcGEuZG9zZU51bWJlciA9IGRvc2VOdW1iZXIgKQ0KDQovKioNCiAqIEBkZXNjcmlwdGlvbjogR2V0cyBpbW11bml6YXRpb25zIG9uIG9yIGJlZm9yZSBhIGRhdGUNCiAqLw0KZGVmaW5lIGZsdWVudCBmdW5jdGlvbiBvbk9yQmVmb3JlKGltbXVuaXphdGlvbnMgTGlzdDxJbW11bml6YXRpb24+LCBiZWZvcmVEYXRlIERhdGUpOg0KICBpbW11bml6YXRpb25zIEkgd2hlcmUNCiAgICBJLm9jY3VycmVuY2UudG9JbnRlcnZhbCgpIHNhbWUgZGF5IG9yIGJlZm9yZSBiZWZvcmVEYXRlDQoNCi8qKg0KICogQGRlc2NyaXB0aW9uOiBHZXRzIHRoZSBzZXJpZXMgZG9zZXMgZnJvbSBhbiBpbW11bml6YXRpb24NCiAqLw0KZGVmaW5lIGZsdWVudCBmdW5jdGlvbiBzZXJpZXNEb3NlcyhpbW11bml6YXRpb24gSW1tdW5pemF0aW9uKToNCiAgRmlyc3QoaW1tdW5pemF0aW9uLnByb3RvY29sQXBwbGllZCBwYSB3aGVyZSBwYS5zZXJpZXNEb3NlcyBpcyBub3QgbnVsbCkuc2VyaWVzRG9zZXMNCg0KLyoqDQogKiBAZGVzY3JpcHRpb246IEdldHMgb2JzZXJ2YXRpb24gZnJvbSBhbiBlbmNvdW50ZXIgb3Igb24gb3IgYmVmb3JlIGEgZGF0ZQ0KICovDQpkZWZpbmUgZmx1ZW50IGZ1bmN0aW9uIGVuY291bnRlck9yT25CZWZvcmUob2JzZXJ2YXRpb25zIExpc3Q8T2JzZXJ2YXRpb24+LCBFbmNvdW50ZXJJZCBTdHJpbmcsIGJlZm9yZURhdGUgRGF0ZSk6DQogIG9ic2VydmF0aW9ucyBPIHdoZXJlDQogICAgKE8uZW5jb3VudGVyLnJlZmVyZW5jZXMoRW5jb3VudGVySWQpDQogICAgICBvciBPLmVmZmVjdGl2ZS50b0ludGVydmFsKCkgc3RhcnRzIHNhbWUgZGF5IG9yIGJlZm9yZSBiZWZvcmVEYXRlKQ0KDQovKioNCiAqIEBkZXNjcmlwdGlvbjogR2V0cyB0aGUgZG9zZXMgZnJvbSB0aGUgcHJpbWFyeSBzZXJpZXMNCiAqLw0KZGVmaW5lIGZsdWVudCBmdW5jdGlvbiBzZXJpZXNQcmltYXJ5KGltbXVuaXphdGlvbnMgTGlzdDxJbW11bml6YXRpb24+KToNCiAgaW1tdW5pemF0aW9ucyBJIHdoZXJlDQogICAgZXhpc3RzKCBJLnByb3RvY29sQXBwbGllZCBwYSB3aGVyZSBwYS5zZXJpZXMgPSAnUHJpbWFyeSBzZXJpZXMnICkNCg0KLyoqDQogKiBAZGVzY3JpcHRpb246IEdldHMgdGhlIGRvc2VzIGZyb20gdGhlIGRvc2UgMCBzZXJpZXMNCiAqLw0KZGVmaW5lIGZsdWVudCBmdW5jdGlvbiBzZXJpZXNEb3NlMChpbW11bml6YXRpb25zIExpc3Q8SW1tdW5pemF0aW9uPik6DQogIGltbXVuaXphdGlvbnMgSSB3aGVyZQ0KICAgIGV4aXN0cyggSS5wcm90b2NvbEFwcGxpZWQgcGEgd2hlcmUgcGEuc2VyaWVzID0gJ0Rvc2UgMCcgKQ0KDQovKioNCiAqIEBkZXNjcmlwdGlvbjogR2V0cyB0aGUgZG9zZXMgZnJvbSB0aGUgQm9vc3RlciBzZXJpZXMNCiAqLw0KZGVmaW5lIGZsdWVudCBmdW5jdGlvbiBzZXJpZXNCb29zdGVyKGltbXVuaXphdGlvbnMgTGlzdDxJbW11bml6YXRpb24+KToNCiAgaW1tdW5pemF0aW9ucyBJIHdoZXJlDQogICAgZXhpc3RzKCBJLnByb3RvY29sQXBwbGllZCBwYSB3aGVyZSBwYS5zZXJpZXMgPSAnQm9vc3RlciBkb3NlJyApDQoNCi8qKg0KICogQGRlc2NyaXB0aW9uOiBHZXRzIHRoZSBkb3NlcyBmcm9tIHRoZSBTdXBwbGVtZW50YXJ5IHNlcmllcw0KICovDQpkZWZpbmUgZmx1ZW50IGZ1bmN0aW9uIHNlcmllc1N1cHBsZW1lbnRhcnkoaW1tdW5pemF0aW9ucyBMaXN0PEltbXVuaXphdGlvbj4pOg0KICBpbW11bml6YXRpb25zIEkgd2hlcmUNCiAgICBleGlzdHMoIEkucHJvdG9jb2xBcHBsaWVkIHBhIHdoZXJlIHBhLnNlcmllcyA9ICdTdXBwbGVtZW50YXJ5IGRvc2UnICkNCg0KLyoqDQogKiBAZGVzY3JpcHRpb246IFNvcnRzIGEgbGlzdCBhbmQgcmV0dXJucyB0aGUgcmVxdWVzdGVkIGluZGV4DQogKi8NCmRlZmluZSBmbHVlbnQgZnVuY3Rpb24gc29ydGVkSW5kZXgoaW1tdW5pemF0aW9ucyBMaXN0PEltbXVuaXphdGlvbj4sIGlkeCBJbnRlZ2VyKToNCiAgaWYgZXhpc3RzKCBpbW11bml6YXRpb25zICkgdGhlbg0KICAgIChpbW11bml6YXRpb25zIEkgc29ydCBieSBzdGFydCBvZiBvY2N1cnJlbmNlLnRvSW50ZXJ2YWwoKSlbaWR4XQ0KICBlbHNlIG51bGwNCg=="
    }
  ]
}