eCQM QICore Content Implementation Guide
2025.0.0 - CI Build
eCQM QICore Content Implementation Guide, published by cqframework. This guide is not an authorized publication; it is the continuous build for version 2025.0.0 built by the FHIR (HL7® FHIR® Standard) CI Build. This version is based on the current content of https://github.com/cqframework/ecqm-content-qicore-2025/ and changes regularly. See the Directory of published versions
Active as of 2025-05-27 |
<Library xmlns="http://hl7.org/fhir">
<id value="CMS832HHAKIFHIR"/>
<meta>
<profile
value="http://hl7.org/fhir/uv/crmi/StructureDefinition/crmi-shareablelibrary"/>
<profile
value="http://hl7.org/fhir/uv/crmi/StructureDefinition/crmi-computablelibrary"/>
<profile
value="http://hl7.org/fhir/uv/crmi/StructureDefinition/crmi-publishablelibrary"/>
<profile
value="http://hl7.org/fhir/uv/crmi/StructureDefinition/crmi-executablelibrary"/>
<profile
value="http://hl7.org/fhir/uv/cql/StructureDefinition/cql-library"/>
<profile
value="http://hl7.org/fhir/uv/cql/StructureDefinition/elm-json-library"/>
<profile
value="http://hl7.org/fhir/uv/cql/StructureDefinition/elm-xml-library"/>
</meta>
<language value="en"/>
<text>
<status value="extensions"/>
<div xmlns="http://www.w3.org/1999/xhtml">
<table class="grid dict">
<tr>
<th scope="row"><b>Title: </b></th>
<td style="padding-left: 4px;">CMS832HHAKIFHIR</td>
</tr>
<tr>
<th scope="row"><b>Id: </b></th>
<td style="padding-left: 4px;">CMS832HHAKIFHIR</td>
</tr>
<tr>
<th scope="row"><b>Version: </b></th>
<td style="padding-left: 4px;">0.2.000</td>
</tr>
<tr>
<th scope="row"><b>Url: </b></th>
<td style="padding-left: 4px;">https://madie.cms.gov/Library/CMS832HHAKIFHIR</td>
</tr>
<tr>
<th scope="row">
<b>
official
</b>
</th>
<td style="padding-left: 4px;">
<p style="margin-bottom: 5px;">
<span>CMS832HHAKIFHIR</span>
</p>
</td>
</tr>
<tr>
<th scope="row"><b>Type: </b></th>
<td style="padding-left: 4px;">
<p style="margin-bottom: 5px;">
<b>system: </b> <span>http://terminology.hl7.org/CodeSystem/library-type</span>
</p>
<p style="margin-bottom: 5px;">
<b>code: </b> <span>logic-library</span>
</p>
</td>
</tr>
<tr>
<th scope="row"><b>Date: </b></th>
<td style="padding-left: 4px;">2025-05-27T15:45:19+00:00</td>
</tr>
<tr>
<th scope="row"><b>Publisher: </b></th>
<td style="padding-left: 4px;">Centers for Medicare & Medicaid Services (CMS)</td>
</tr>
<tr>
<th scope="row"><b>Description: </b></th>
<td style="padding-left: 4px;">CMS832HHAKIFHIR</td>
</tr>
<tr>
<th scope="row"><b>Related Artifacts: </b></th>
<td style="padding-left: 4px;">
<p><b>Dependencies</b></p>
<ul>
<li>https://madie.cms.gov/Library/FHIRHelpers|4.4.000</li>
<li>https://madie.cms.gov/Library/CQMCommon|4.1.000</li>
<li>https://madie.cms.gov/Library/QICoreCommon|4.0.000</li>
<li>https://madie.cms.gov/Library/SupplementalDataElements|5.1.000</li>
<li>http://loinc.org</li>
<li>http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113883.3.666.5.307</li>
<li>http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1248.21</li>
<li>http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1248.33</li>
<li>http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1147.197</li>
<li>http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1248.199</li>
<li>http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1045.152</li>
<li>http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1248.12</li>
<li>http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1248.19</li>
</ul>
</td>
</tr>
<tr>
<th scope="row"><b>Data Requirements:</b></th>
<td style="padding-left: 4px;">
<table class="grid-dict">
<tr><th><b>Type</b></th><th><b>Profile</b></th><th><b>MS</b></th><th><b>Code Filter</b></th></tr>
<tr>
<th>Patient</th>
<th>http://hl7.org/fhir/us/qicore/StructureDefinition/qicore-patient</th>
<th>;;</th>
<th>
</th>
</tr>
<tr>
<th>Claim</th>
<th>http://hl7.org/fhir/us/qicore/StructureDefinition/qicore-claim</th>
<th>;;;;;;</th>
<th>
</th>
</tr>
<tr>
<th>Coverage</th>
<th>http://hl7.org/fhir/us/qicore/StructureDefinition/qicore-coverage</th>
<th>;;</th>
<th>
<b>code filter: </b>
<br/>
<span style="padding-left: 4px;">
<b>path: </b><span>type</span>
</span>
<br/>
<span style="padding-left: 4px;">
<b>value set: </b><span>http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.114222.4.11.3591</span>
</span>
</th>
</tr>
</table>
</td>
</tr>
<tr>
<td colspan="2">
<table>
<tr><th><a id="cql-content"><b>Content: </b></a> text/cql</th></tr>
<tr><td><pre><code class="language-cql">library CMS832HHAKIFHIR version '0.2.000'
using QICore version '6.0.0'
include FHIRHelpers version '4.4.000' called FHIRHelpers
include SupplementalDataElements version '5.1.000' called SDE
include CQMCommon version '4.1.000' called CQMCommon
include QICoreCommon version '4.0.000' called QICoreCommon
include CumulativeMedicationDuration version '5.0.000' called CMD
include HospitalHarm version '2.5.000' called HospitalHarm
codesystem "SNOMEDCT": 'http://snomed.info/sct'
codesystem "LOINC": 'http://loinc.org'
valueset "Body temperature": 'http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1045.152'
valueset "Creatinine Mass Per Volume": 'http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1248.21'
valueset "Emergency Department Visit": 'http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113883.3.117.1.7.1.292'
valueset "Encounter Inpatient": 'http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113883.3.666.5.307'
valueset "Glomerular Filtration Rate": 'http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113883.17.4077.2.2038'
valueset "High Risk Diagnosis for AKI": 'http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1248.12'
valueset "High Risk Procedures for AKI": 'http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1248.19'
valueset "Hospital Based Dialysis Services": 'http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1248.199'
valueset "Observation Services": 'http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1111.143'
valueset "Obstetrics and VTE Obstetrics": 'http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1248.33'
valueset "Present on Admission or Clinically Undetermined": 'http://cts.nlm.nih.gov/fhir/ValueSet/2.16.840.1.113762.1.4.1147.197'
code "Female (finding)": '248152002' from "SNOMEDCT" display 'Female'
code "Male (finding)": '248153007' from "SNOMEDCT" display 'Male'
code "Heart rate": '8867-4' from "LOINC" display 'Heart rate'
code "Respiratory rate": '9279-1' from "LOINC" display 'Respiratory rate'
code "Systolic blood pressure": '8480-6' from "LOINC" display 'Systolic blood pressure'
parameter "Measurement Period" Interval<DateTime>
context Patient
define "SDE Ethnicity":
SDE."SDE Ethnicity"
define "SDE Payer":
SDE."SDE Payer"
define "SDE Race":
SDE."SDE Race"
define "SDE Sex":
SDE."SDE Sex"
// -------------------------------- Population Definitions -----------------------------
define "Initial Population":
"Encounter with Creatinine and without Obstetrical Conditions"
define "Denominator":
"Initial Population"
define "Denominator Exclusion":
"Encounter with Less Than 2 Creatinine Results within 48 Hours of Arrival" // clause 1
union "Encounter with Index eGFR Less Than 60 within First 48 Hours" // clause 2
union "Encounter with 0.3 mg dL or More Increase in Creatinine" // clause 3
union "Encounter with Kidney Dialysis Started 48 Hours or Less After Arrival without High Creatinine" // clause 4
union "Encounter with High Risk Diagnosis for AKI" // clause 5
union "Encounter with High Risk Procedures for AKI" // clause 6
define "Numerator":
"Encounter with 2 Times Serum Creatinine Increase" // clause 1
union "Encounter with Kidney Dialysis Started More Than 48 Hours After Arrival without High Creatinine" // clause 2
//-------------------------------- Supporting Definitions for Initial Population -----------------------------
// Supports Initial Population
// Used in other definitions throughout
define "Encounter with Creatinine and without Obstetrical Conditions":
"Inpatient Encounter with Creatinine" EncounterWithCreatinine
where not exists ( ( EncounterWithCreatinine.encounterDiagnosis ( ) ) EncounterDiagnosis
where EncounterDiagnosis.code in "Obstetrics and VTE Obstetrics"
)
// Supports Initial Population
define "Inpatient Encounter with Creatinine":
from
"Encounter with Age 18 and Length of Stay 48 Hours or More" Encounter48Hours,
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] CreatinineTest
let HospitalizationPeriod: Encounter48Hours.hospitalizationWithObservation ( ),
CreatinineTestStart: CreatinineTest.effective.earliest ( )
where CreatinineTest.value is not null
and CreatinineTestStart during Interval[start of HospitalizationPeriod + 48 hours, end of HospitalizationPeriod]
and CreatinineTest.status in { 'final', 'amended', 'corrected' }
return Encounter48Hours
// Supports Initial Population
/*
define "Encounter with Age 18 and Length of Stay 48 Hours or More":
["Encounter": "Encounter Inpatient"] InpatientEncounter
where InpatientEncounter.period ends during day of "Measurement Period"
and AgeInYearsAt(date from start of InpatientEncounter.period)>= 18
and ( duration in hours of InpatientEncounter.hospitalizationWithObservation() >= 48 )*/
define "Encounter with Age 18 and Length of Stay 48 Hours or More":
["Encounter": "Encounter Inpatient"] InpatientEncounter
where Patient.sex in { '248153007', '248152002' }
and InpatientEncounter.period ends during day of "Measurement Period"
and AgeInYearsAt(date from start of InpatientEncounter.period) >= 18
and ( duration in hours of InpatientEncounter.hospitalizationWithObservation ( ) >= 48 )
// -------------------------------- Supporting Definitions for Denominator Exclusion -----------------------------
// Denominator Exclusion clause 1
define "Encounter with Less Than 2 Creatinine Results within 48 Hours of Arrival":
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
where ( Count("CreatinineLabTestwithResultwithinFirst48Hours"(QualifyingEncounter)) < 2 )
// Denonminator Exclusion clause 2
define "Encounter with Index eGFR Less Than 60 within First 48 Hours":
"Male Encounter with eGFR Less Than 60"
union "Female Encounter with eGFR Less Than 60"
// Supporting Denominator Exclusion clause 2
define "Female Encounter with eGFR Less Than 60":
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
where "FemaleeGFR"(QualifyingEncounter) is not null
and "FemaleeGFR"(QualifyingEncounter) as Decimal < 60
// Supporting Denominator Exclusion clause 2
define "Male Encounter with eGFR Less Than 60":
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
where "MaleeGFR"(QualifyingEncounter) is not null
and "MaleeGFR"(QualifyingEncounter) as Decimal < 60
// Denominator Exclusion clause 3 (Coalesce doesn't seem to work as intended)
define "Encounter with 0.3 mg dL or More Increase in Creatinine":
"Increase of 0.3 or More Using Lowest Creatinine within 24 Hours"
union "Increase of 0.3 or More Using First Creatinine within First 48 Hours"
//define "Encounter with 0.3 mg dL or More Increase in Creatinine":
//Coalesce("Increase of 0.3 or More Using Lowest Creatinine within 24 Hours","Increase of 0.3 or More Using First Creatinine within First 48 Hours")
/*
define "Encounter with 0.3 mg dL or More Increase in Creatinine":
case
when "Increase of 0.3 or More Using Lowest Creatinine within 24 Hours" is not null
then "Increase of 0.3 or More Using Lowest Creatinine within 24 Hours"
when "Increase of 0.3 or More Using First Creatinine within First 48 Hours" is not null
then "Increase of 0.3 or More Using First Creatinine within First 48 Hours"
else null
end */
// Supporting Denominator Exclusion clause 3
define "Increase of 0.3 or More Using Lowest Creatinine within 24 Hours":
from
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter,
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] IndexCreatinineLabResult,
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] SubsequentCreatinineLabResult
let IndexCreatinineLabResultTime: IndexCreatinineLabResult.effective.earliest ( ),
SubsequentCreatinineLabResultTime: SubsequentCreatinineLabResult.effective.earliest ( ),
HospitalWithObservationPeriod: QualifyingEncounter.hospitalizationWithObservation ( )
where IndexCreatinineLabResult.status in { 'final', 'amended', 'corrected' }
and SubsequentCreatinineLabResult.status in { 'final', 'amended', 'corrected' }
and ( ( SubsequentCreatinineLabResult.value as Quantity ) - ( IndexCreatinineLabResult.value as Quantity ) ) > 0.299 'mg/dL'
and IndexCreatinineLabResult.value = "LowestSerumCreatinineResult"(QualifyingEncounter)
and IndexCreatinineLabResultTime during Interval[SubsequentCreatinineLabResultTime - 48 hours, SubsequentCreatinineLabResultTime]
and IndexCreatinineLabResultTime during HospitalWithObservationPeriod
and IndexCreatinineLabResultTime during Interval[start of HospitalWithObservationPeriod, start of HospitalWithObservationPeriod + 24 hours]
and SubsequentCreatinineLabResultTime during HospitalWithObservationPeriod
and SubsequentCreatinineLabResultTime during Interval[start of HospitalWithObservationPeriod, start of HospitalWithObservationPeriod + 48 hours]
and IndexCreatinineLabResult.id != SubsequentCreatinineLabResult.id
return QualifyingEncounter
// Supporting Denominator Exclusion clause 3
define "Increase of 0.3 or More Using First Creatinine within First 48 Hours":
from
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter,
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] IndexCreatinineLabResult,
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] SubsequentCreatinineLabResult
let IndexCreatinineLabResultTime: IndexCreatinineLabResult.effective.earliest ( ),
SubsequentCreatinineLabResultTime: SubsequentCreatinineLabResult.effective.earliest ( ),
HospitalWithObservationPeriod: QualifyingEncounter.hospitalizationWithObservation ( )
where IndexCreatinineLabResult.status in { 'final', 'amended', 'corrected' }
and SubsequentCreatinineLabResult.status in { 'final', 'amended', 'corrected' }
and ( ( SubsequentCreatinineLabResult.value as Quantity ) - ( IndexCreatinineLabResult.value as Quantity ) ) > 0.299 'mg/dL'
and IndexCreatinineLabResult.value as Quantity = singleton from "EarliestSerumCreatinineResult"(QualifyingEncounter)
and IndexCreatinineLabResultTime during Interval[SubsequentCreatinineLabResultTime - 48 hours, SubsequentCreatinineLabResultTime]
and IndexCreatinineLabResultTime during HospitalWithObservationPeriod
and SubsequentCreatinineLabResultTime during Interval[start of HospitalWithObservationPeriod, start of HospitalWithObservationPeriod + 48 hours]
and SubsequentCreatinineLabResultTime during HospitalWithObservationPeriod
and IndexCreatinineLabResultTime during Interval[start of HospitalWithObservationPeriod, start of HospitalWithObservationPeriod + 48 hours]
and SubsequentCreatinineLabResultTime during Interval[start of HospitalWithObservationPeriod, start of HospitalWithObservationPeriod + 48 hours]
and IndexCreatinineLabResult.id != SubsequentCreatinineLabResult.id
return QualifyingEncounter
// Denominator Exclusion clause 4
define "Encounter with Kidney Dialysis Started 48 Hours or Less After Arrival without High Creatinine":
"Encounter with Kidney Dialysis Started 48 Hours or Less After Arrival" EncounterWithKidneyDialysis48HoursOrAfter
where not ( exists ( "Encounter with 2 Times Serum Creatinine Increase" EncounterWithHighCreatinine
where ( EncounterWithHighCreatinine.period includes EncounterWithKidneyDialysis48HoursOrAfter.period )
)
)
// Supporting Denominator Exclusion clause 4
define "Encounter with Kidney Dialysis Started 48 Hours or Less After Arrival":
from
["Procedure": "Hospital Based Dialysis Services"] Dialysis,
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
let HospitalWithObservationPeriod: QualifyingEncounter.hospitalizationWithObservation ( )
where Dialysis.performed.toInterval ( ) starts during Interval[start of HospitalWithObservationPeriod, start of HospitalWithObservationPeriod + 48 hours]
and Dialysis.performed.toInterval ( ) starts during HospitalWithObservationPeriod
return QualifyingEncounter
// Supporting Denominator Exclusion clause 4
// Also Supporting Numerator clause 1
define "Encounter with 2 Times Serum Creatinine Increase":
from
"Encounter with 1.5 Times Serum Creatinine Increase" EncounterWithHighCreatinine,
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] HighCreatinineTest,
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] LowCreatinineTest
let LowCreatinineTestTime: LowCreatinineTest.effective.earliest ( ),
HighCreatinineTestTime: HighCreatinineTest.effective.earliest ( ),
HospitalWithObservationPeriod: EncounterWithHighCreatinine.hospitalizationWithObservation ( )
where ( HighCreatinineTest.value > "Serum Creatinine Normal" )
and LowCreatinineTest.status in { 'final', 'amended', 'corrected' }
and HighCreatinineTest.status in { 'final', 'amended', 'corrected' }
and HighCreatinineTest.value = "HighestSerumCreatinineResult"(EncounterWithHighCreatinine)
and LowCreatinineTest.value = "LowestSerumCreatinineResult"(EncounterWithHighCreatinine)
and ( HighCreatinineTest.value as Quantity ) >= ( LowCreatinineTest.value as Quantity )
and LowCreatinineTestTime 7 days or less before HighCreatinineTestTime
and LowCreatinineTestTime during HospitalWithObservationPeriod
and HighCreatinineTestTime during Interval[start of HospitalWithObservationPeriod + 48 hours, start of HospitalWithObservationPeriod + 30 days]
and HighCreatinineTestTime during HospitalWithObservationPeriod
return EncounterWithHighCreatinine
// Supports "Encounter with 2 Times Serum Creatinine Increase"
// Supports "Encounter with 1.5 Times Serum Creatinine Increase"
define "Serum Creatinine Normal":
if ( Patient.sex = '248152002' ) then 1.02 'mg/dL'
else 1.18 'mg/dL'
// Denominator Exclusion clause 5
define "Encounter with High Risk Diagnosis for AKI":
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
where exists ( ( QualifyingEncounter.encounterDiagnosis ( ) ) EncounterDiagnosis
where ( EncounterDiagnosis.code in "High Risk Diagnosis for AKI" )
)
// Denominator Exclusion clause 6
define "Encounter with High Risk Procedures for AKI":
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
with ["Procedure": "High Risk Procedures for AKI"] HighRiskProcedures
such that HighRiskProcedures.performed.toInterval ( ) starts during QualifyingEncounter.hospitalizationWithObservation ( )
// -------------------------------- Supporting Definitions for Numerator -----------------------------
// Numerator clause 1: See "Encounter with 2 Times Serum Creatinine Increase"
// Numerator union clause 2
define "Encounter with Kidney Dialysis Started More Than 48 Hours After Arrival without High Creatinine":
"Encounter with Kidney Dialysis Started More Than 48 Hours After Arrival" EncounterWithDialysisAfter48Hours
where not ( exists ( "Encounter with 2 Times Serum Creatinine Increase" EncounterWithHighCreatinine
where ( EncounterWithHighCreatinine.period includes EncounterWithDialysisAfter48Hours.period )
)
)
// Supports Numerator union clause 2
define "Encounter with Kidney Dialysis Started More Than 48 Hours After Arrival":
from
["Procedure": "Hospital Based Dialysis Services"] Dialysis,
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
let HospitalWithObservationPeriod: QualifyingEncounter.hospitalizationWithObservation ( )
where Dialysis.performed.toInterval ( ) starts during Interval[start of HospitalWithObservationPeriod + 48 hours, end of HospitalWithObservationPeriod]
and Dialysis.performed.toInterval ( ) starts during HospitalWithObservationPeriod
return QualifyingEncounter
// -------------------------------- Supporting Definitions for Creatinine calculations -----------------------------
// Used throughout creatinine functions and defines
define "Qualifying Creatinine Lab Result by Time":
from
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter,
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] CreatinineTestByTime
let CrEncId: QualifyingEncounter.id,
CrHospPeriod: QualifyingEncounter.hospitalizationWithObservation ( ),
CrLabId: CreatinineTestByTime.id,
CrTime: CreatinineTestByTime.effective.earliest ( ),
CrTimeIssued: CreatinineTestByTime.issued,
CrResult: CreatinineTestByTime.value as Quantity,
CrResultValue: CrResult.value,
CrResultUnit: CrResult.unit
where CrTime during CrHospPeriod
and CreatinineTestByTime.isLaboratory ( )
and CreatinineTestByTime.status in { 'final', 'amended', 'corrected' }
and CrResultUnit = 'mg/dL'
and CreatinineTestByTime.value is not null
and CreatinineTestByTime.value as Quantity > 0 'mg/dL'
return Tuple {
CrEncInPtId: CrEncId,
CrHospitalization: CrHospPeriod,
CrLabObsId: CrLabId,
// LabCategory: CreatinineTestByQuantity.isLaboratory(),
CrLabObsCategory: if CreatinineTestByTime.isLaboratory ( ) then 'laboratory'
else CreatinineTestByTime.isLaboratory ( ),
CrLabObsStatus: CreatinineTestByTime.status,
CrLabResult: CrResult,
CrLabResultUnit: CrResultUnit,
CrLabResultValue: CrResultValue,
CrLabTime: CrTime,
CrLabTimeIssued: CrTimeIssued
}
sort by CrLabTime
// Supporting Denominator Exclusion clause 4
// Supporting Numerator
define "Encounter with 1.5 Times Serum Creatinine Increase":
from
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter,
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] HighCreatinineTest,
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] LowCreatinineTest
let LowCreatinineTestTime: LowCreatinineTest.effective.earliest ( ),
HighCreatinineTestTime: HighCreatinineTest.effective.earliest ( ),
HospitalWithObservationPeriod: QualifyingEncounter.hospitalizationWithObservation ( )
where ( HighCreatinineTest.value > "Serum Creatinine Normal" )
and LowCreatinineTest.status in { 'final', 'amended', 'corrected' }
and HighCreatinineTest.status in { 'final', 'amended', 'corrected' }
and HighCreatinineTest.value = "HighestSerumCreatinineResult"(QualifyingEncounter)
and LowCreatinineTest.value = "LowestSerumCreatinineResult"(QualifyingEncounter)
and "1.5IncreaseInCreatinine"(QualifyingEncounter) >= LowCreatinineTest.value
and LowCreatinineTestTime 7 days or less before HighCreatinineTestTime
and LowCreatinineTestTime during HospitalWithObservationPeriod
and HighCreatinineTestTime during Interval[start of HospitalWithObservationPeriod + 48 hours, start of HospitalWithObservationPeriod + 30 days]
and HighCreatinineTestTime during HospitalWithObservationPeriod
return QualifyingEncounter
/*
define "Encounter with 1.5 Times Serum Creatinine Increase":
from
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter,
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] CreatinineTest
let
LowestCreatinineTestWithin7DaysPrior: "LowestSerumCreatinineWithin7DaysPrior"(QualifyingEncounter, CreatinineTest),
CreatinineTestTime: CreatinineTest.effective.earliest(),
HospitalWithObservationPeriod: QualifyingEncounter.hospitalizationWithObservation()
where
CreatinineTest.value >= LowestCreatinineTestWithin7DaysPrior * 1.5
and CreatinineTest.value > "Serum Creatinine Normal"
and CreatinineTestTime during Interval[start of HospitalWithObservationPeriod + 48 hours, start of HospitalWithObservationPeriod + 30 days]
return QualifyingEncounter*/
// -------------------------------- Supporting Functions for Creatinine calculations -----------------------------
// Supports Denominator Exclusion clause 1
define function "CreatinineLabTestwithResultwithinFirst48Hours"(QualifyingEncounter Encounter):
from
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] CreatinineTest
where CreatinineTest.value is not null
and CreatinineTest.effective.earliest ( ) during Interval[start of QualifyingEncounter.hospitalizationWithObservation ( ), start of QualifyingEncounter.hospitalizationWithObservation ( ) + 48 hours]
and CreatinineTest.effective.earliest ( ) during QualifyingEncounter.hospitalizationWithObservation ( )
and CreatinineTest.status in { 'final', 'amended', 'corrected' }
return CreatinineTest
// eGFR functions return as decimal value rather than Quantity
// Supports Denominator Exclusion clause 2
// Supports risk variable "Risk Variable Estimated Glomerular Filtration Rate for Females"
define function "FemaleeGFR"(QualifyingEncounter Encounter):
if Patient.sex = '248152002' then ( 142 * Min({("IndexCreatinine"(QualifyingEncounter).value / 0.7), 1 }) ^ ( - 0.241 ) * Max({("IndexCreatinine"(QualifyingEncounter).value / 0.7), 1 }) ^ ( - 1.200 ) * 0.9938 ^ ( AgeInYearsAt(start of QualifyingEncounter.hospitalizationWithObservation()) ) * 1.012 )
else null
// eGFR functions return as decimal value rather than Quantity
// Supports Denominator Exclusion clause 2
// Supports risk variable "Risk Variable Estimated Glomerular Filtration Rate for Females"
define function "MaleeGFR"(QualifyingEncounter Encounter):
if Patient.sex = '248153007' then ( 142 * Min({("IndexCreatinine"(QualifyingEncounter).value / 0.9), 1 }) ^ ( - 0.302 ) * Max({("IndexCreatinine"(QualifyingEncounter).value / 0.9), 1 }) ^ ( - 1.200 ) * 0.9938 ^ ( AgeInYearsAt(start of QualifyingEncounter.hospitalizationWithObservation()) ) )
else null
// Used in GFR calculations, above
define function "IndexCreatinine"(QualifyingEncounter Encounter):
Coalesce("LowestSerumCreatinineIn24Hours"(QualifyingEncounter), singleton from "FirstSerumCreatinineIn48Hours"(QualifyingEncounter))
// Supports IndexCreatinine function
define function "LowestSerumCreatinineIn24Hours"(QualifyingEncounter Encounter):
Min((from
"Qualifying Creatinine Lab Result by Time" LabTestsLow
let LabResult: LabTestsLow.CrLabResult
where LabTestsLow.CrEncInPtId = QualifyingEncounter.id
and LabTestsLow.CrLabTime during Interval[start of QualifyingEncounter.hospitalizationWithObservation(), start of QualifyingEncounter.hospitalizationWithObservation() + 24 hours]
).CrLabResult
)
// Find result from earliest time in 48 hrs
// Supports IndexCreatinine function
define function "FirstSerumCreatinineIn48Hours"(QualifyingEncounter Encounter):
from
"Qualifying Creatinine Lab Result by Time" LabTests
let LabResult: LabTests.CrLabResult
where ( "EarliestSerumCreatinineTimeIn48Hours"(QualifyingEncounter) = LabTests.CrLabTime )
return LabResult as Quantity
// Find earliest time in 48 hrs
// Supports FirstSerumCreatinineIn48Hours function
define function "EarliestSerumCreatinineTimeIn48Hours"(QualifyingEncounter Encounter):
( Min((from
"Qualifying Creatinine Lab Result by Time" LabTests48
let LabResult48: LabTests48.CrLabResult
where LabTests48.CrEncInPtId = QualifyingEncounter.id
and LabTests48.CrLabTime during Interval[start of QualifyingEncounter.hospitalizationWithObservation(), start of QualifyingEncounter.hospitalizationWithObservation() + 48 hours]
).CrLabTime
)
)
// Supports Denominator Exclusion clause 3
define function "EarliestSerumCreatinineResult"(QualifyingEncounter Encounter):
from
"Qualifying Creatinine Lab Result by Time" LabTests
let LabResult: LabTests.CrLabResult
where ( "EarliestSerumCreatinineTime"(QualifyingEncounter) = LabTests.CrLabTime )
return LabResult as Quantity
// Supports Denominator Exclusion clause 3 via "EarliestSerumCreatinineResult" function
define function "EarliestSerumCreatinineTime"(QualifyingEncounter Encounter):
( Min((from
"Qualifying Creatinine Lab Result by Time" LabTestsEarly
let LabResultEarly: LabTestsEarly.CrLabResult
where LabTestsEarly.CrEncInPtId = QualifyingEncounter.id
).CrLabTime
)
)
// Not used anywhere
define function "SerumCreatinineSequencebyTime"(QualifyingEncounter Encounter):
["LaboratoryResultObservation": "Creatinine Mass Per Volume"] CreatinineTestByTime
let HospitalWithObservation: QualifyingEncounter.hospitalization ( )
where CreatinineTestByTime.effective.earliest ( ) during HospitalWithObservation
and CreatinineTestByTime.value is not null
and CreatinineTestByTime.isLaboratory ( )
and CreatinineTestByTime.status in { 'final', 'amended', 'corrected' }
return CreatinineTestByTime
// The following functions work with Cr result as Quantity (not decimal, but Quantity value is decimal)
// Supporting Denominator Exclusion clause 4 via "Encounter with 2 Times Serum Creatinine Increase"
// Supporting Numerator via "Encounter with 2 Times Serum Creatinine Increase"
define function "HighestSerumCreatinineResult"(QualifyingEncounter Encounter):
( Max((from
"Qualifying Creatinine Lab Result by Time" LabTests
let LabResult: LabTests.CrLabResult
where LabTests.CrEncInPtId = QualifyingEncounter.id
).CrLabResult
)
)
// Supporting Denominator Exclusion clause 4 via "Encounter with 1.5 Times Serum Creatinine Increase"
// Supporting Numerator via "Encounter with 1.5 Times Serum Creatinine Increase"
define function "1.5IncreaseInCreatinine"(QualifyingEncounter Encounter):
"HighestSerumCreatinineResult"(QualifyingEncounter) / 1.5
// Supporting Denominator Exclusion clause 4 via "Encounter with 1.5 Times Serum Creatinine Increase"
// Supporting Numerator via "Encounter with 1.5 Times Serum Creatinine Increase"
define function "2.0IncreaseInCreatinine"(QualifyingEncounter Encounter):
"HighestSerumCreatinineResult"(QualifyingEncounter) / 2
// Supporting Denominator Exclusion clause 3 via "Increase of 0.3 or More Using Lowest Creatinine within 24 Hours"
// and clause 4 via "Encounter with 1.5 Times Serum Creatinine Increase"
// Supporting Numerator via "Encounter with 1.5 Times Serum Creatinine Increase"
define function "LowestSerumCreatinineResult"(QualifyingEncounter Encounter):
( Min((from
"Qualifying Creatinine Lab Result by Time" LabTests
let LabResult: LabTests.CrLabResult
where LabTests.CrEncInPtId = QualifyingEncounter.id
).CrLabResult
)
)
// -------------------------------- Risk Variable Definitions -----------------------------
define "Risk Variable Estimated Glomerular Filtration Rate for Females":
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
return Tuple {
encounterId: QualifyingEncounter.id,
eGFR: "FemaleeGFR"(QualifyingEncounter)
}
define "Risk Variable Estimated Glomerular Filtration Rate for Males":
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
return Tuple {
encounterId: QualifyingEncounter.id,
eGFR: "MaleeGFR"(QualifyingEncounter)
}
define "Risk Variable All Encounter Diagnoses with POA Indication":
from
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter,
[Claim] claim
where claim.status = 'active'
and claim.use = 'claim'
and exists ( claim.item I
where I.encounter.references ( QualifyingEncounter )
and exists ( claim.diagnosis D
where D.sequence in claim.item.diagnosisSequence
and D.onAdmission in "Present on Admission or Clinically Undetermined"
)
)
define "Risk Variable First Heart Rate in Encounter":
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
return Tuple {
encounterId: QualifyingEncounter.id,
firstHeartRate: "FirstHeartRate"(QualifyingEncounter)
}
define "Risk Variable First Respiratory Rate in Encounter":
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
return Tuple {
encounterId: QualifyingEncounter.id,
firstRespiratoryRate: "FirstRespiratoryRate"(QualifyingEncounter)
}
define "Risk Variable First Systolic Blood Pressure in Encounter":
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
return Tuple {
encounterId: QualifyingEncounter.id,
firstSystolicBP: "FirstSystolicBloodPressure"(QualifyingEncounter)
}
define "Risk Variable First Temperature in Encounter":
"Encounter with Creatinine and without Obstetrical Conditions" QualifyingEncounter
return Tuple {
encounterId: QualifyingEncounter.id,
firstTemperature: "FirstBodyTemperature"(QualifyingEncounter)
}
// -------------------------------- Risk Variable supportin functions -----------------------------
define function "FirstBodyTemperature"(QualifyingEncounter Encounter):
First(["USCoreBodyTemperatureProfile": "Body temperature"] FirstTemperature
where FirstTemperature.effective.earliest() during QualifyingEncounter.hospitalizationWithObservation()
and FirstTemperature.value is not null
sort by effective.earliest()
).value as Quantity
define function "FirstHeartRate"(QualifyingEncounter Encounter):
First(["USCoreHeartRateProfile": "Heart rate"] FirstHeartBeats
where FirstHeartBeats.effective.earliest() during QualifyingEncounter.hospitalizationWithObservation()
and FirstHeartBeats.value is not null
sort by effective.earliest()
).value as Quantity
define function "FirstRespiratoryRate"(QualifyingEncounter Encounter):
First(["USCoreRespiratoryRateProfile": "Respiratory rate"] FirstRespiration
where FirstRespiration.effective.earliest() during QualifyingEncounter.hospitalizationWithObservation()
and FirstRespiration.value is not null
sort by effective.earliest()
).value as Quantity
define "Qualifying Systolic Blood Pressure Reading":
["USCoreBloodPressureProfile"] BloodPressure
where BloodPressure.effective.earliest ( ) during day of "Measurement Period"
define function "FirstSystolicBloodPressure"(QualifyingEncounter Encounter):
First("Qualifying Systolic Blood Pressure Reading" SBPReading
where SBPReading.effective.earliest() during QualifyingEncounter.hospitalizationWithObservation()
return singleton from(SBPReading.component SBPComponent
where SBPComponent.code ~ "Systolic blood pressure"
return SBPComponent.value as Quantity
)
)
/*
define function "FirstSystolicBloodPressureTest"(QualifyingEncounter Encounter ):
First(["USCoreBloodPressureProfile": "Systolic blood pressure"] FirstSystolic
where FirstSystolic.effective.earliest() during QualifyingEncounter.hospitalizationWithObservation()
and FirstSystolic.value is not null
sort by effective.earliest()
).value as Quantity
*/
// --------------------------------- Temporary debugging defines should go here ---------------------------------------------------
</code></pre></td></tr>
</table>
</td>
</tr>
</table>
</div>
</text>
<contained>
<Parameters>
<id value="options"/>
<parameter>
<name value="translatorVersion"/>
<valueString value="3.25.0"/>
</parameter>
<parameter>
<name value="option"/>
<valueString value="EnableAnnotations"/>
</parameter>
<parameter>
<name value="option"/>
<valueString value="EnableLocators"/>
</parameter>
<parameter>
<name value="option"/>
<valueString value="DisableListDemotion"/>
</parameter>
<parameter>
<name value="option"/>
<valueString value="DisableListPromotion"/>
</parameter>
<parameter>
<name value="format"/>
<valueString value="JSON"/>
</parameter>
<parameter>
<name value="format"/>
<valueString value="XML"/>
</parameter>
<parameter>
<name value="analyzeDataRequirements"/>
<valueBoolean value="false"/>
</parameter>
<parameter>
<name value="collapseDataRequirements"/>
<valueBoolean value="false"/>
</parameter>
<parameter>
<name value="compatibilityLevel"/>
<valueString value="1.5"/>
</parameter>
<parameter>
<name value="enableCqlOnly"/>
<valueBoolean value="false"/>
</parameter>
<parameter>
<name value="errorLevel"/>
<valueString value="Info"/>
</parameter>
<parameter>
<name value="signatureLevel"/>
<valueString value="Overloads"/>
</parameter>
<parameter>
<name value="validateUnits"/>
<valueBoolean value="true"/>
</parameter>
<parameter>
<name value="verifyOnly"/>
<valueBoolean value="false"/>
</parameter>
</Parameters>
</contained>
<extension
url="http://hl7.org/fhir/StructureDefinition/cqf-directReferenceCode">
<valueCoding>
<system value="http://snomed.info/sct"/>
<code value="248152002"/>
<display value="Female (finding)"/>
</valueCoding>
</extension>
<extension
url="http://hl7.org/fhir/StructureDefinition/cqf-directReferenceCode">
<valueCoding>
<system value="http://snomed.info/sct"/>
<code value="248153007"/>
<display value="Male (finding)"/>
</valueCoding>
</extension>
<extension
url="http://hl7.org/fhir/StructureDefinition/cqf-directReferenceCode">
<valueCoding>
<system value="http://loinc.org"/>
<code value="8867-4"/>
<display value="Heart rate"/>
</valueCoding>
</extension>
<extension
url="http://hl7.org/fhir/StructureDefinition/cqf-directReferenceCode">
<valueCoding>
<system value="http://loinc.org"/>
<code value="9279-1"/>
<display value="Respiratory rate"/>
</valueCoding>
</extension>
<extension
url="http://hl7.org/fhir/StructureDefinition/cqf-directReferenceCode">
<valueCoding>
<system value="http://loinc.org"/>
<code value="8480-6"/>
<display value="Systolic blood pressure"/>
</valueCoding>
</extension>
<extension url="http://hl7.org/fhir/StructureDefinition/cqf-cqlOptions">
<valueReference>
<reference value="#options"/>
</valueReference>
</extension>
<url value="https://madie.cms.gov/Library/CMS832HHAKIFHIR"/>
<identifier>
<use value="official"/>
<system value="https://madie.cms.gov/login"/>
<value value="CMS832HHAKIFHIR"/>
</identifier>
<version value="0.2.000"/>
<name value="CMS832HHAKIFHIR"/>
<title value="CMS832HHAKIFHIR"/>
<status value="active"/>
<experimental value="false"/>
<type>
<coding>
<system value="http://terminology.hl7.org/CodeSystem/library-type"/>
<code value="logic-library"/>
</coding>
</type>
<date value="2025-05-27T15:45:19+00:00"/>
<publisher value="Centers for Medicare & Medicaid Services (CMS)"/>
<description value="CMS832HHAKIFHIR"/>
<content>
<contentType value="text/cql"/>
<data
value="bGlicmFyeSBDTVM4MzJISEFLSUZISVIgdmVyc2lvbiAnMC4yLjAwMCcNCg0KdXNpbmcgUUlDb3JlIHZlcnNpb24gJzYuMC4wJw0KDQppbmNsdWRlIEZISVJIZWxwZXJzIHZlcnNpb24gJzQuNC4wMDAnIGNhbGxlZCBGSElSSGVscGVycw0KaW5jbHVkZSBTdXBwbGVtZW50YWxEYXRhRWxlbWVudHMgdmVyc2lvbiAnNS4xLjAwMCcgY2FsbGVkIFNERQ0KaW5jbHVkZSBDUU1Db21tb24gdmVyc2lvbiAnNC4xLjAwMCcgY2FsbGVkIENRTUNvbW1vbg0KaW5jbHVkZSBRSUNvcmVDb21tb24gdmVyc2lvbiAnNC4wLjAwMCcgY2FsbGVkIFFJQ29yZUNvbW1vbg0KaW5jbHVkZSBDdW11bGF0aXZlTWVkaWNhdGlvbkR1cmF0aW9uIHZlcnNpb24gJzUuMC4wMDAnIGNhbGxlZCBDTUQNCmluY2x1ZGUgSG9zcGl0YWxIYXJtIHZlcnNpb24gJzIuNS4wMDAnIGNhbGxlZCBIb3NwaXRhbEhhcm0NCg0KY29kZXN5c3RlbSAiU05PTUVEQ1QiOiAnaHR0cDovL3Nub21lZC5pbmZvL3NjdCcNCmNvZGVzeXN0ZW0gIkxPSU5DIjogJ2h0dHA6Ly9sb2luYy5vcmcnDQoNCnZhbHVlc2V0ICJCb2R5IHRlbXBlcmF0dXJlIjogJ2h0dHA6Ly9jdHMubmxtLm5paC5nb3YvZmhpci9WYWx1ZVNldC8yLjE2Ljg0MC4xLjExMzc2Mi4xLjQuMTA0NS4xNTInDQp2YWx1ZXNldCAiQ3JlYXRpbmluZSBNYXNzIFBlciBWb2x1bWUiOiAnaHR0cDovL2N0cy5ubG0ubmloLmdvdi9maGlyL1ZhbHVlU2V0LzIuMTYuODQwLjEuMTEzNzYyLjEuNC4xMjQ4LjIxJw0KdmFsdWVzZXQgIkVtZXJnZW5jeSBEZXBhcnRtZW50IFZpc2l0IjogJ2h0dHA6Ly9jdHMubmxtLm5paC5nb3YvZmhpci9WYWx1ZVNldC8yLjE2Ljg0MC4xLjExMzg4My4zLjExNy4xLjcuMS4yOTInDQp2YWx1ZXNldCAiRW5jb3VudGVyIElucGF0aWVudCI6ICdodHRwOi8vY3RzLm5sbS5uaWguZ292L2ZoaXIvVmFsdWVTZXQvMi4xNi44NDAuMS4xMTM4ODMuMy42NjYuNS4zMDcnDQp2YWx1ZXNldCAiR2xvbWVydWxhciBGaWx0cmF0aW9uIFJhdGUiOiAnaHR0cDovL2N0cy5ubG0ubmloLmdvdi9maGlyL1ZhbHVlU2V0LzIuMTYuODQwLjEuMTEzODgzLjE3LjQwNzcuMi4yMDM4Jw0KdmFsdWVzZXQgIkhpZ2ggUmlzayBEaWFnbm9zaXMgZm9yIEFLSSI6ICdodHRwOi8vY3RzLm5sbS5uaWguZ292L2ZoaXIvVmFsdWVTZXQvMi4xNi44NDAuMS4xMTM3NjIuMS40LjEyNDguMTInDQp2YWx1ZXNldCAiSGlnaCBSaXNrIFByb2NlZHVyZXMgZm9yIEFLSSI6ICdodHRwOi8vY3RzLm5sbS5uaWguZ292L2ZoaXIvVmFsdWVTZXQvMi4xNi44NDAuMS4xMTM3NjIuMS40LjEyNDguMTknDQp2YWx1ZXNldCAiSG9zcGl0YWwgQmFzZWQgRGlhbHlzaXMgU2VydmljZXMiOiAnaHR0cDovL2N0cy5ubG0ubmloLmdvdi9maGlyL1ZhbHVlU2V0LzIuMTYuODQwLjEuMTEzNzYyLjEuNC4xMjQ4LjE5OScNCnZhbHVlc2V0ICJPYnNlcnZhdGlvbiBTZXJ2aWNlcyI6ICdodHRwOi8vY3RzLm5sbS5uaWguZ292L2ZoaXIvVmFsdWVTZXQvMi4xNi44NDAuMS4xMTM3NjIuMS40LjExMTEuMTQzJw0KdmFsdWVzZXQgIk9ic3RldHJpY3MgYW5kIFZURSBPYnN0ZXRyaWNzIjogJ2h0dHA6Ly9jdHMubmxtLm5paC5nb3YvZmhpci9WYWx1ZVNldC8yLjE2Ljg0MC4xLjExMzc2Mi4xLjQuMTI0OC4zMycNCnZhbHVlc2V0ICJQcmVzZW50IG9uIEFkbWlzc2lvbiBvciBDbGluaWNhbGx5IFVuZGV0ZXJtaW5lZCI6ICdodHRwOi8vY3RzLm5sbS5uaWguZ292L2ZoaXIvVmFsdWVTZXQvMi4xNi44NDAuMS4xMTM3NjIuMS40LjExNDcuMTk3Jw0KDQpjb2RlICJGZW1hbGUgKGZpbmRpbmcpIjogJzI0ODE1MjAwMicgZnJvbSAiU05PTUVEQ1QiIGRpc3BsYXkgJ0ZlbWFsZScNCmNvZGUgIk1hbGUgKGZpbmRpbmcpIjogJzI0ODE1MzAwNycgZnJvbSAiU05PTUVEQ1QiIGRpc3BsYXkgJ01hbGUnDQpjb2RlICJIZWFydCByYXRlIjogJzg4NjctNCcgZnJvbSAiTE9JTkMiIGRpc3BsYXkgJ0hlYXJ0IHJhdGUnDQpjb2RlICJSZXNwaXJhdG9yeSByYXRlIjogJzkyNzktMScgZnJvbSAiTE9JTkMiIGRpc3BsYXkgJ1Jlc3BpcmF0b3J5IHJhdGUnDQpjb2RlICJTeXN0b2xpYyBibG9vZCBwcmVzc3VyZSI6ICc4NDgwLTYnIGZyb20gIkxPSU5DIiBkaXNwbGF5ICdTeXN0b2xpYyBibG9vZCBwcmVzc3VyZScNCg0KcGFyYW1ldGVyICJNZWFzdXJlbWVudCBQZXJpb2QiIEludGVydmFsPERhdGVUaW1lPg0KDQpjb250ZXh0IFBhdGllbnQNCg0KZGVmaW5lICJTREUgRXRobmljaXR5IjoNCiAgU0RFLiJTREUgRXRobmljaXR5Ig0KDQpkZWZpbmUgIlNERSBQYXllciI6DQogIFNERS4iU0RFIFBheWVyIg0KDQpkZWZpbmUgIlNERSBSYWNlIjoNCiAgU0RFLiJTREUgUmFjZSINCg0KZGVmaW5lICJTREUgU2V4IjoNCiAgU0RFLiJTREUgU2V4Ig0KDQovLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSBQb3B1bGF0aW9uIERlZmluaXRpb25zIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tDQoNCg0KZGVmaW5lICJJbml0aWFsIFBvcHVsYXRpb24iOg0KICAiRW5jb3VudGVyIHdpdGggQ3JlYXRpbmluZSBhbmQgd2l0aG91dCBPYnN0ZXRyaWNhbCBDb25kaXRpb25zIg0KDQpkZWZpbmUgIkRlbm9taW5hdG9yIjoNCiAgIkluaXRpYWwgUG9wdWxhdGlvbiINCg0KZGVmaW5lICJEZW5vbWluYXRvciBFeGNsdXNpb24iOg0KICAiRW5jb3VudGVyIHdpdGggTGVzcyBUaGFuIDIgQ3JlYXRpbmluZSBSZXN1bHRzIHdpdGhpbiA0OCBIb3VycyBvZiBBcnJpdmFsIiAvLyBjbGF1c2UgMQ0KICAgIA0KICAgIHVuaW9uICJFbmNvdW50ZXIgd2l0aCBJbmRleCBlR0ZSIExlc3MgVGhhbiA2MCB3aXRoaW4gRmlyc3QgNDggSG91cnMiICAgICAvLyBjbGF1c2UgMg0KICAgIA0KICAgIHVuaW9uICJFbmNvdW50ZXIgd2l0aCAwLjMgbWcgZEwgb3IgTW9yZSBJbmNyZWFzZSBpbiBDcmVhdGluaW5lIiAgICAgICAgICAvLyBjbGF1c2UgMw0KICAgIA0KICAgIHVuaW9uICJFbmNvdW50ZXIgd2l0aCBLaWRuZXkgRGlhbHlzaXMgU3RhcnRlZCA0OCBIb3VycyBvciBMZXNzIEFmdGVyIEFycml2YWwgd2l0aG91dCBIaWdoIENyZWF0aW5pbmUiICAvLyBjbGF1c2UgNA0KICAgIA0KICAgIHVuaW9uICJFbmNvdW50ZXIgd2l0aCBIaWdoIFJpc2sgRGlhZ25vc2lzIGZvciBBS0kiICAgICAgICAgICAgICAgICAgICAgICAvLyBjbGF1c2UgNQ0KICAgIA0KICAgIHVuaW9uICJFbmNvdW50ZXIgd2l0aCBIaWdoIFJpc2sgUHJvY2VkdXJlcyBmb3IgQUtJIiAgICAgICAgICAgICAgICAgICAgICAvLyBjbGF1c2UgNiANCg0KDQpkZWZpbmUgIk51bWVyYXRvciI6DQogICJFbmNvdW50ZXIgd2l0aCAyIFRpbWVzIFNlcnVtIENyZWF0aW5pbmUgSW5jcmVhc2UiIC8vIGNsYXVzZSAxDQogICAgDQogICAgdW5pb24gIkVuY291bnRlciB3aXRoIEtpZG5leSBEaWFseXNpcyBTdGFydGVkIE1vcmUgVGhhbiA0OCBIb3VycyBBZnRlciBBcnJpdmFsIHdpdGhvdXQgSGlnaCBDcmVhdGluaW5lIiAgIC8vIGNsYXVzZSAyDQoNCg0KLy8tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSBTdXBwb3J0aW5nIERlZmluaXRpb25zIGZvciBJbml0aWFsIFBvcHVsYXRpb24gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0NCg0KLy8gU3VwcG9ydHMgSW5pdGlhbCBQb3B1bGF0aW9uDQovLyBVc2VkIGluIG90aGVyIGRlZmluaXRpb25zIHRocm91Z2hvdXQNCg0KDQoNCg0KDQpkZWZpbmUgIkVuY291bnRlciB3aXRoIENyZWF0aW5pbmUgYW5kIHdpdGhvdXQgT2JzdGV0cmljYWwgQ29uZGl0aW9ucyI6DQogICJJbnBhdGllbnQgRW5jb3VudGVyIHdpdGggQ3JlYXRpbmluZSIgRW5jb3VudGVyV2l0aENyZWF0aW5pbmUNCiAgICB3aGVyZSBub3QgZXhpc3RzICggKCBFbmNvdW50ZXJXaXRoQ3JlYXRpbmluZS5lbmNvdW50ZXJEaWFnbm9zaXMgKCApICkgRW5jb3VudGVyRGlhZ25vc2lzDQogICAgICAgIHdoZXJlIEVuY291bnRlckRpYWdub3Npcy5jb2RlIGluICJPYnN0ZXRyaWNzIGFuZCBWVEUgT2JzdGV0cmljcyINCiAgICApDQoNCg0KLy8gU3VwcG9ydHMgSW5pdGlhbCBQb3B1bGF0aW9uDQoNCg0KZGVmaW5lICJJbnBhdGllbnQgRW5jb3VudGVyIHdpdGggQ3JlYXRpbmluZSI6DQogIGZyb20NCiAgICAiRW5jb3VudGVyIHdpdGggQWdlIDE4IGFuZCBMZW5ndGggb2YgU3RheSA0OCBIb3VycyBvciBNb3JlIiBFbmNvdW50ZXI0OEhvdXJzLA0KICAgIFsiTGFib3JhdG9yeVJlc3VsdE9ic2VydmF0aW9uIjogIkNyZWF0aW5pbmUgTWFzcyBQZXIgVm9sdW1lIl0gQ3JlYXRpbmluZVRlc3QNCiAgICBsZXQgSG9zcGl0YWxpemF0aW9uUGVyaW9kOiBFbmNvdW50ZXI0OEhvdXJzLmhvc3BpdGFsaXphdGlvbldpdGhPYnNlcnZhdGlvbiAoICksDQogICAgQ3JlYXRpbmluZVRlc3RTdGFydDogQ3JlYXRpbmluZVRlc3QuZWZmZWN0aXZlLmVhcmxpZXN0ICggKQ0KICAgIHdoZXJlIENyZWF0aW5pbmVUZXN0LnZhbHVlIGlzIG5vdCBudWxsDQogICAgICBhbmQgQ3JlYXRpbmluZVRlc3RTdGFydCBkdXJpbmcgSW50ZXJ2YWxbc3RhcnQgb2YgSG9zcGl0YWxpemF0aW9uUGVyaW9kICsgNDggaG91cnMsIGVuZCBvZiBIb3NwaXRhbGl6YXRpb25QZXJpb2RdDQogICAgICBhbmQgQ3JlYXRpbmluZVRlc3Quc3RhdHVzIGluIHsgJ2ZpbmFsJywgJ2FtZW5kZWQnLCAnY29ycmVjdGVkJyB9DQogICAgcmV0dXJuIEVuY291bnRlcjQ4SG91cnMNCiAgICAgICAgICAgICAgICANCi8vIFN1cHBvcnRzIEluaXRpYWwgUG9wdWxhdGlvbg0KDQovKg0KZGVmaW5lICJFbmNvdW50ZXIgd2l0aCBBZ2UgMTggYW5kIExlbmd0aCBvZiBTdGF5IDQ4IEhvdXJzIG9yIE1vcmUiOg0KICBbIkVuY291bnRlciI6ICJFbmNvdW50ZXIgSW5wYXRpZW50Il0gSW5wYXRpZW50RW5jb3VudGVyDQogICAgICAgICAgICAgICAgICAgIHdoZXJlIElucGF0aWVudEVuY291bnRlci5wZXJpb2QgZW5kcyBkdXJpbmcgZGF5IG9mICJNZWFzdXJlbWVudCBQZXJpb2QiDQogICAgICAgICAgICAgICAgICAgICAgICBhbmQgQWdlSW5ZZWFyc0F0KGRhdGUgZnJvbSBzdGFydCBvZiBJbnBhdGllbnRFbmNvdW50ZXIucGVyaW9kKT49IDE4DQogICAgICAgICAgICAgICAgICAgICAgICBhbmQgKCBkdXJhdGlvbiBpbiBob3VycyBvZiBJbnBhdGllbnRFbmNvdW50ZXIuaG9zcGl0YWxpemF0aW9uV2l0aE9ic2VydmF0aW9uKCkgPj0gNDggKSovIA0KDQoNCg0KZGVmaW5lICJFbmNvdW50ZXIgd2l0aCBBZ2UgMTggYW5kIExlbmd0aCBvZiBTdGF5IDQ4IEhvdXJzIG9yIE1vcmUiOg0KICBbIkVuY291bnRlciI6ICJFbmNvdW50ZXIgSW5wYXRpZW50Il0gSW5wYXRpZW50RW5jb3VudGVyDQogICAgd2hlcmUgUGF0aWVudC5zZXggaW4geyAnMjQ4MTUzMDA3JywgJzI0ODE1MjAwMicgfQ0KICAgICAgYW5kIElucGF0aWVudEVuY291bnRlci5wZXJpb2QgZW5kcyBkdXJpbmcgZGF5IG9mICJNZWFzdXJlbWVudCBQZXJpb2QiDQogICAgICBhbmQgQWdlSW5ZZWFyc0F0KGRhdGUgZnJvbSBzdGFydCBvZiBJbnBhdGllbnRFbmNvdW50ZXIucGVyaW9kKSA+PSAxOA0KICAgICAgYW5kICggZHVyYXRpb24gaW4gaG91cnMgb2YgSW5wYXRpZW50RW5jb3VudGVyLmhvc3BpdGFsaXphdGlvbldpdGhPYnNlcnZhdGlvbiAoICkgPj0gNDggKQ0KDQovLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSBTdXBwb3J0aW5nIERlZmluaXRpb25zIGZvciBEZW5vbWluYXRvciBFeGNsdXNpb24gIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tDQoNCi8vIERlbm9taW5hdG9yIEV4Y2x1c2lvbiBjbGF1c2UgMQ0KDQoNCg0KZGVmaW5lICJFbmNvdW50ZXIgd2l0aCBMZXNzIFRoYW4gMiBDcmVhdGluaW5lIFJlc3VsdHMgd2l0aGluIDQ4IEhvdXJzIG9mIEFycml2YWwiOg0KICAiRW5jb3VudGVyIHdpdGggQ3JlYXRpbmluZSBhbmQgd2l0aG91dCBPYnN0ZXRyaWNhbCBDb25kaXRpb25zIiBRdWFsaWZ5aW5nRW5jb3VudGVyDQogICAgd2hlcmUgKCBDb3VudCgiQ3JlYXRpbmluZUxhYlRlc3R3aXRoUmVzdWx0d2l0aGluRmlyc3Q0OEhvdXJzIihRdWFsaWZ5aW5nRW5jb3VudGVyKSkgPCAyICkgDQoNCi8vIERlbm9ubWluYXRvciBFeGNsdXNpb24gY2xhdXNlIDINCg0KDQpkZWZpbmUgIkVuY291bnRlciB3aXRoIEluZGV4IGVHRlIgTGVzcyBUaGFuIDYwIHdpdGhpbiBGaXJzdCA0OCBIb3VycyI6DQogICJNYWxlIEVuY291bnRlciB3aXRoIGVHRlIgTGVzcyBUaGFuIDYwIg0KICAgIHVuaW9uICJGZW1hbGUgRW5jb3VudGVyIHdpdGggZUdGUiBMZXNzIFRoYW4gNjAiDQoNCi8vIFN1cHBvcnRpbmcgRGVub21pbmF0b3IgRXhjbHVzaW9uIGNsYXVzZSAyICAgIA0KDQoNCmRlZmluZSAiRmVtYWxlIEVuY291bnRlciB3aXRoIGVHRlIgTGVzcyBUaGFuIDYwIjoNCiAgIkVuY291bnRlciB3aXRoIENyZWF0aW5pbmUgYW5kIHdpdGhvdXQgT2JzdGV0cmljYWwgQ29uZGl0aW9ucyIgUXVhbGlmeWluZ0VuY291bnRlcg0KICAgIHdoZXJlICJGZW1hbGVlR0ZSIihRdWFsaWZ5aW5nRW5jb3VudGVyKSBpcyBub3QgbnVsbA0KICAgICAgYW5kICJGZW1hbGVlR0ZSIihRdWFsaWZ5aW5nRW5jb3VudGVyKSBhcyBEZWNpbWFsIDwgNjANCg0KLy8gU3VwcG9ydGluZyBEZW5vbWluYXRvciBFeGNsdXNpb24gY2xhdXNlIDIgIA0KDQoNCmRlZmluZSAiTWFsZSBFbmNvdW50ZXIgd2l0aCBlR0ZSIExlc3MgVGhhbiA2MCI6DQogICJFbmNvdW50ZXIgd2l0aCBDcmVhdGluaW5lIGFuZCB3aXRob3V0IE9ic3RldHJpY2FsIENvbmRpdGlvbnMiIFF1YWxpZnlpbmdFbmNvdW50ZXINCiAgICB3aGVyZSAiTWFsZWVHRlIiKFF1YWxpZnlpbmdFbmNvdW50ZXIpIGlzIG5vdCBudWxsDQogICAgICBhbmQgIk1hbGVlR0ZSIihRdWFsaWZ5aW5nRW5jb3VudGVyKSBhcyBEZWNpbWFsIDwgNjANCg0KLy8gRGVub21pbmF0b3IgRXhjbHVzaW9uIGNsYXVzZSAzICAoQ29hbGVzY2UgZG9lc24ndCBzZWVtIHRvIHdvcmsgYXMgaW50ZW5kZWQpDQoNCg0KZGVmaW5lICJFbmNvdW50ZXIgd2l0aCAwLjMgbWcgZEwgb3IgTW9yZSBJbmNyZWFzZSBpbiBDcmVhdGluaW5lIjoNCiAgIkluY3JlYXNlIG9mIDAuMyBvciBNb3JlIFVzaW5nIExvd2VzdCBDcmVhdGluaW5lIHdpdGhpbiAyNCBIb3VycyINCiAgICB1bmlvbiAiSW5jcmVhc2Ugb2YgMC4zIG9yIE1vcmUgVXNpbmcgRmlyc3QgQ3JlYXRpbmluZSB3aXRoaW4gRmlyc3QgNDggSG91cnMiDQoNCiAgIA0KLy9kZWZpbmUgIkVuY291bnRlciB3aXRoIDAuMyBtZyBkTCBvciBNb3JlIEluY3JlYXNlIGluIENyZWF0aW5pbmUiOg0KICAvL0NvYWxlc2NlKCJJbmNyZWFzZSBvZiAwLjMgb3IgTW9yZSBVc2luZyBMb3dlc3QgQ3JlYXRpbmluZSB3aXRoaW4gMjQgSG91cnMiLCJJbmNyZWFzZSBvZiAwLjMgb3IgTW9yZSBVc2luZyBGaXJzdCBDcmVhdGluaW5lIHdpdGhpbiBGaXJzdCA0OCBIb3VycyIpDQoNCi8qDQpkZWZpbmUgIkVuY291bnRlciB3aXRoIDAuMyBtZyBkTCBvciBNb3JlIEluY3JlYXNlIGluIENyZWF0aW5pbmUiOg0KICBjYXNlDQogICAgd2hlbiAiSW5jcmVhc2Ugb2YgMC4zIG9yIE1vcmUgVXNpbmcgTG93ZXN0IENyZWF0aW5pbmUgd2l0aGluIDI0IEhvdXJzIiBpcyBub3QgbnVsbA0KICAgICAgdGhlbiAiSW5jcmVhc2Ugb2YgMC4zIG9yIE1vcmUgVXNpbmcgTG93ZXN0IENyZWF0aW5pbmUgd2l0aGluIDI0IEhvdXJzIg0KICAgIHdoZW4gIkluY3JlYXNlIG9mIDAuMyBvciBNb3JlIFVzaW5nIEZpcnN0IENyZWF0aW5pbmUgd2l0aGluIEZpcnN0IDQ4IEhvdXJzIiBpcyBub3QgbnVsbA0KICAgICAgdGhlbiAiSW5jcmVhc2Ugb2YgMC4zIG9yIE1vcmUgVXNpbmcgRmlyc3QgQ3JlYXRpbmluZSB3aXRoaW4gRmlyc3QgNDggSG91cnMiDQogICAgZWxzZSBudWxsDQogIGVuZCAqLw0KDQogIA0KLy8gU3VwcG9ydGluZyBEZW5vbWluYXRvciBFeGNsdXNpb24gY2xhdXNlIDMNCg0KDQoNCg0KDQpkZWZpbmUgIkluY3JlYXNlIG9mIDAuMyBvciBNb3JlIFVzaW5nIExvd2VzdCBDcmVhdGluaW5lIHdpdGhpbiAyNCBIb3VycyI6DQogIGZyb20NCiAgICAiRW5jb3VudGVyIHdpdGggQ3JlYXRpbmluZSBhbmQgd2l0aG91dCBPYnN0ZXRyaWNhbCBDb25kaXRpb25zIiBRdWFsaWZ5aW5nRW5jb3VudGVyLA0KICAgIFsiTGFib3JhdG9yeVJlc3VsdE9ic2VydmF0aW9uIjogIkNyZWF0aW5pbmUgTWFzcyBQZXIgVm9sdW1lIl0gSW5kZXhDcmVhdGluaW5lTGFiUmVzdWx0LA0KICAgIFsiTGFib3JhdG9yeVJlc3VsdE9ic2VydmF0aW9uIjogIkNyZWF0aW5pbmUgTWFzcyBQZXIgVm9sdW1lIl0gU3Vic2VxdWVudENyZWF0aW5pbmVMYWJSZXN1bHQNCiAgICBsZXQgSW5kZXhDcmVhdGluaW5lTGFiUmVzdWx0VGltZTogSW5kZXhDcmVhdGluaW5lTGFiUmVzdWx0LmVmZmVjdGl2ZS5lYXJsaWVzdCAoICksDQogICAgU3Vic2VxdWVudENyZWF0aW5pbmVMYWJSZXN1bHRUaW1lOiBTdWJzZXF1ZW50Q3JlYXRpbmluZUxhYlJlc3VsdC5lZmZlY3RpdmUuZWFybGllc3QgKCApLA0KICAgIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kOiBRdWFsaWZ5aW5nRW5jb3VudGVyLmhvc3BpdGFsaXphdGlvbldpdGhPYnNlcnZhdGlvbiAoICkNCiAgICB3aGVyZSBJbmRleENyZWF0aW5pbmVMYWJSZXN1bHQuc3RhdHVzIGluIHsgJ2ZpbmFsJywgJ2FtZW5kZWQnLCAnY29ycmVjdGVkJyB9DQogICAgICBhbmQgU3Vic2VxdWVudENyZWF0aW5pbmVMYWJSZXN1bHQuc3RhdHVzIGluIHsgJ2ZpbmFsJywgJ2FtZW5kZWQnLCAnY29ycmVjdGVkJyB9DQogICAgICBhbmQgKCAoIFN1YnNlcXVlbnRDcmVhdGluaW5lTGFiUmVzdWx0LnZhbHVlIGFzIFF1YW50aXR5ICkgLSAoIEluZGV4Q3JlYXRpbmluZUxhYlJlc3VsdC52YWx1ZSBhcyBRdWFudGl0eSApICkgPiAwLjI5OSAnbWcvZEwnDQogICAgICBhbmQgSW5kZXhDcmVhdGluaW5lTGFiUmVzdWx0LnZhbHVlID0gIkxvd2VzdFNlcnVtQ3JlYXRpbmluZVJlc3VsdCIoUXVhbGlmeWluZ0VuY291bnRlcikNCiAgICAgIGFuZCBJbmRleENyZWF0aW5pbmVMYWJSZXN1bHRUaW1lIGR1cmluZyBJbnRlcnZhbFtTdWJzZXF1ZW50Q3JlYXRpbmluZUxhYlJlc3VsdFRpbWUgLSA0OCBob3VycywgU3Vic2VxdWVudENyZWF0aW5pbmVMYWJSZXN1bHRUaW1lXQ0KICAgICAgYW5kIEluZGV4Q3JlYXRpbmluZUxhYlJlc3VsdFRpbWUgZHVyaW5nIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kDQogICAgICBhbmQgSW5kZXhDcmVhdGluaW5lTGFiUmVzdWx0VGltZSBkdXJpbmcgSW50ZXJ2YWxbc3RhcnQgb2YgSG9zcGl0YWxXaXRoT2JzZXJ2YXRpb25QZXJpb2QsIHN0YXJ0IG9mIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kICsgMjQgaG91cnNdDQogICAgICBhbmQgU3Vic2VxdWVudENyZWF0aW5pbmVMYWJSZXN1bHRUaW1lIGR1cmluZyBIb3NwaXRhbFdpdGhPYnNlcnZhdGlvblBlcmlvZA0KICAgICAgYW5kIFN1YnNlcXVlbnRDcmVhdGluaW5lTGFiUmVzdWx0VGltZSBkdXJpbmcgSW50ZXJ2YWxbc3RhcnQgb2YgSG9zcGl0YWxXaXRoT2JzZXJ2YXRpb25QZXJpb2QsIHN0YXJ0IG9mIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kICsgNDggaG91cnNdDQogICAgICBhbmQgSW5kZXhDcmVhdGluaW5lTGFiUmVzdWx0LmlkICE9IFN1YnNlcXVlbnRDcmVhdGluaW5lTGFiUmVzdWx0LmlkDQogICAgcmV0dXJuIFF1YWxpZnlpbmdFbmNvdW50ZXINCg0KLy8gU3VwcG9ydGluZyBEZW5vbWluYXRvciBFeGNsdXNpb24gY2xhdXNlIDMgICAgICAgICAgICANCg0KDQpkZWZpbmUgIkluY3JlYXNlIG9mIDAuMyBvciBNb3JlIFVzaW5nIEZpcnN0IENyZWF0aW5pbmUgd2l0aGluIEZpcnN0IDQ4IEhvdXJzIjoNCiAgZnJvbQ0KICAgICJFbmNvdW50ZXIgd2l0aCBDcmVhdGluaW5lIGFuZCB3aXRob3V0IE9ic3RldHJpY2FsIENvbmRpdGlvbnMiIFF1YWxpZnlpbmdFbmNvdW50ZXIsDQogICAgWyJMYWJvcmF0b3J5UmVzdWx0T2JzZXJ2YXRpb24iOiAiQ3JlYXRpbmluZSBNYXNzIFBlciBWb2x1bWUiXSBJbmRleENyZWF0aW5pbmVMYWJSZXN1bHQsDQogICAgWyJMYWJvcmF0b3J5UmVzdWx0T2JzZXJ2YXRpb24iOiAiQ3JlYXRpbmluZSBNYXNzIFBlciBWb2x1bWUiXSBTdWJzZXF1ZW50Q3JlYXRpbmluZUxhYlJlc3VsdA0KICAgIGxldCBJbmRleENyZWF0aW5pbmVMYWJSZXN1bHRUaW1lOiBJbmRleENyZWF0aW5pbmVMYWJSZXN1bHQuZWZmZWN0aXZlLmVhcmxpZXN0ICggKSwNCiAgICBTdWJzZXF1ZW50Q3JlYXRpbmluZUxhYlJlc3VsdFRpbWU6IFN1YnNlcXVlbnRDcmVhdGluaW5lTGFiUmVzdWx0LmVmZmVjdGl2ZS5lYXJsaWVzdCAoICksDQogICAgSG9zcGl0YWxXaXRoT2JzZXJ2YXRpb25QZXJpb2Q6IFF1YWxpZnlpbmdFbmNvdW50ZXIuaG9zcGl0YWxpemF0aW9uV2l0aE9ic2VydmF0aW9uICggKQ0KICAgIHdoZXJlIEluZGV4Q3JlYXRpbmluZUxhYlJlc3VsdC5zdGF0dXMgaW4geyAnZmluYWwnLCAnYW1lbmRlZCcsICdjb3JyZWN0ZWQnIH0NCiAgICAgIGFuZCBTdWJzZXF1ZW50Q3JlYXRpbmluZUxhYlJlc3VsdC5zdGF0dXMgaW4geyAnZmluYWwnLCAnYW1lbmRlZCcsICdjb3JyZWN0ZWQnIH0NCiAgICAgIGFuZCAoICggU3Vic2VxdWVudENyZWF0aW5pbmVMYWJSZXN1bHQudmFsdWUgYXMgUXVhbnRpdHkgKSAtICggSW5kZXhDcmVhdGluaW5lTGFiUmVzdWx0LnZhbHVlIGFzIFF1YW50aXR5ICkgKSA+IDAuMjk5ICdtZy9kTCcNCiAgICAgIGFuZCBJbmRleENyZWF0aW5pbmVMYWJSZXN1bHQudmFsdWUgYXMgUXVhbnRpdHkgPSBzaW5nbGV0b24gZnJvbSAiRWFybGllc3RTZXJ1bUNyZWF0aW5pbmVSZXN1bHQiKFF1YWxpZnlpbmdFbmNvdW50ZXIpDQogICAgICBhbmQgSW5kZXhDcmVhdGluaW5lTGFiUmVzdWx0VGltZSBkdXJpbmcgSW50ZXJ2YWxbU3Vic2VxdWVudENyZWF0aW5pbmVMYWJSZXN1bHRUaW1lIC0gNDggaG91cnMsIFN1YnNlcXVlbnRDcmVhdGluaW5lTGFiUmVzdWx0VGltZV0NCiAgICAgIGFuZCBJbmRleENyZWF0aW5pbmVMYWJSZXN1bHRUaW1lIGR1cmluZyBIb3NwaXRhbFdpdGhPYnNlcnZhdGlvblBlcmlvZA0KICAgICAgYW5kIFN1YnNlcXVlbnRDcmVhdGluaW5lTGFiUmVzdWx0VGltZSBkdXJpbmcgSW50ZXJ2YWxbc3RhcnQgb2YgSG9zcGl0YWxXaXRoT2JzZXJ2YXRpb25QZXJpb2QsIHN0YXJ0IG9mIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kICsgNDggaG91cnNdDQogICAgICBhbmQgU3Vic2VxdWVudENyZWF0aW5pbmVMYWJSZXN1bHRUaW1lIGR1cmluZyBIb3NwaXRhbFdpdGhPYnNlcnZhdGlvblBlcmlvZA0KICAgICAgYW5kIEluZGV4Q3JlYXRpbmluZUxhYlJlc3VsdFRpbWUgZHVyaW5nIEludGVydmFsW3N0YXJ0IG9mIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kLCBzdGFydCBvZiBIb3NwaXRhbFdpdGhPYnNlcnZhdGlvblBlcmlvZCArIDQ4IGhvdXJzXQ0KICAgICAgYW5kIFN1YnNlcXVlbnRDcmVhdGluaW5lTGFiUmVzdWx0VGltZSBkdXJpbmcgSW50ZXJ2YWxbc3RhcnQgb2YgSG9zcGl0YWxXaXRoT2JzZXJ2YXRpb25QZXJpb2QsIHN0YXJ0IG9mIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kICsgNDggaG91cnNdDQogICAgICBhbmQgSW5kZXhDcmVhdGluaW5lTGFiUmVzdWx0LmlkICE9IFN1YnNlcXVlbnRDcmVhdGluaW5lTGFiUmVzdWx0LmlkDQogICAgcmV0dXJuIFF1YWxpZnlpbmdFbmNvdW50ZXINCiANCi8vIERlbm9taW5hdG9yIEV4Y2x1c2lvbiBjbGF1c2UgNA0KDQoNCmRlZmluZSAiRW5jb3VudGVyIHdpdGggS2lkbmV5IERpYWx5c2lzIFN0YXJ0ZWQgNDggSG91cnMgb3IgTGVzcyBBZnRlciBBcnJpdmFsIHdpdGhvdXQgSGlnaCBDcmVhdGluaW5lIjoNCiAgIkVuY291bnRlciB3aXRoIEtpZG5leSBEaWFseXNpcyBTdGFydGVkIDQ4IEhvdXJzIG9yIExlc3MgQWZ0ZXIgQXJyaXZhbCIgRW5jb3VudGVyV2l0aEtpZG5leURpYWx5c2lzNDhIb3Vyc09yQWZ0ZXINCiAgICB3aGVyZSBub3QgKCBleGlzdHMgKCAiRW5jb3VudGVyIHdpdGggMiBUaW1lcyBTZXJ1bSBDcmVhdGluaW5lIEluY3JlYXNlIiBFbmNvdW50ZXJXaXRoSGlnaENyZWF0aW5pbmUNCiAgICAgICAgICB3aGVyZSAoIEVuY291bnRlcldpdGhIaWdoQ3JlYXRpbmluZS5wZXJpb2QgaW5jbHVkZXMgRW5jb3VudGVyV2l0aEtpZG5leURpYWx5c2lzNDhIb3Vyc09yQWZ0ZXIucGVyaW9kICkNCiAgICAgICkNCiAgICApDQogDQovLyBTdXBwb3J0aW5nIERlbm9taW5hdG9yIEV4Y2x1c2lvbiBjbGF1c2UgNA0KDQoNCmRlZmluZSAiRW5jb3VudGVyIHdpdGggS2lkbmV5IERpYWx5c2lzIFN0YXJ0ZWQgNDggSG91cnMgb3IgTGVzcyBBZnRlciBBcnJpdmFsIjoNCiAgZnJvbQ0KICAgIFsiUHJvY2VkdXJlIjogIkhvc3BpdGFsIEJhc2VkIERpYWx5c2lzIFNlcnZpY2VzIl0gRGlhbHlzaXMsDQogICAgIkVuY291bnRlciB3aXRoIENyZWF0aW5pbmUgYW5kIHdpdGhvdXQgT2JzdGV0cmljYWwgQ29uZGl0aW9ucyIgUXVhbGlmeWluZ0VuY291bnRlcg0KICAgIGxldCBIb3NwaXRhbFdpdGhPYnNlcnZhdGlvblBlcmlvZDogUXVhbGlmeWluZ0VuY291bnRlci5ob3NwaXRhbGl6YXRpb25XaXRoT2JzZXJ2YXRpb24gKCApDQogICAgd2hlcmUgRGlhbHlzaXMucGVyZm9ybWVkLnRvSW50ZXJ2YWwgKCApIHN0YXJ0cyBkdXJpbmcgSW50ZXJ2YWxbc3RhcnQgb2YgSG9zcGl0YWxXaXRoT2JzZXJ2YXRpb25QZXJpb2QsIHN0YXJ0IG9mIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kICsgNDggaG91cnNdDQogICAgICBhbmQgRGlhbHlzaXMucGVyZm9ybWVkLnRvSW50ZXJ2YWwgKCApIHN0YXJ0cyBkdXJpbmcgSG9zcGl0YWxXaXRoT2JzZXJ2YXRpb25QZXJpb2QNCiAgICByZXR1cm4gUXVhbGlmeWluZ0VuY291bnRlcg0KDQoNCi8vIFN1cHBvcnRpbmcgRGVub21pbmF0b3IgRXhjbHVzaW9uIGNsYXVzZSA0DQovLyBBbHNvIFN1cHBvcnRpbmcgTnVtZXJhdG9yIGNsYXVzZSAxDQoNCg0KDQpkZWZpbmUgIkVuY291bnRlciB3aXRoIDIgVGltZXMgU2VydW0gQ3JlYXRpbmluZSBJbmNyZWFzZSI6DQogIGZyb20NCiAgICAiRW5jb3VudGVyIHdpdGggMS41IFRpbWVzIFNlcnVtIENyZWF0aW5pbmUgSW5jcmVhc2UiIEVuY291bnRlcldpdGhIaWdoQ3JlYXRpbmluZSwNCiAgICBbIkxhYm9yYXRvcnlSZXN1bHRPYnNlcnZhdGlvbiI6ICJDcmVhdGluaW5lIE1hc3MgUGVyIFZvbHVtZSJdIEhpZ2hDcmVhdGluaW5lVGVzdCwNCiAgICBbIkxhYm9yYXRvcnlSZXN1bHRPYnNlcnZhdGlvbiI6ICJDcmVhdGluaW5lIE1hc3MgUGVyIFZvbHVtZSJdIExvd0NyZWF0aW5pbmVUZXN0DQogICAgbGV0IExvd0NyZWF0aW5pbmVUZXN0VGltZTogTG93Q3JlYXRpbmluZVRlc3QuZWZmZWN0aXZlLmVhcmxpZXN0ICggKSwNCiAgICBIaWdoQ3JlYXRpbmluZVRlc3RUaW1lOiBIaWdoQ3JlYXRpbmluZVRlc3QuZWZmZWN0aXZlLmVhcmxpZXN0ICggKSwNCiAgICBIb3NwaXRhbFdpdGhPYnNlcnZhdGlvblBlcmlvZDogRW5jb3VudGVyV2l0aEhpZ2hDcmVhdGluaW5lLmhvc3BpdGFsaXphdGlvbldpdGhPYnNlcnZhdGlvbiAoICkNCiAgICB3aGVyZSAoIEhpZ2hDcmVhdGluaW5lVGVzdC52YWx1ZSA+ICJTZXJ1bSBDcmVhdGluaW5lIE5vcm1hbCIgKQ0KICAgICAgYW5kIExvd0NyZWF0aW5pbmVUZXN0LnN0YXR1cyBpbiB7ICdmaW5hbCcsICdhbWVuZGVkJywgJ2NvcnJlY3RlZCcgfQ0KICAgICAgYW5kIEhpZ2hDcmVhdGluaW5lVGVzdC5zdGF0dXMgaW4geyAnZmluYWwnLCAnYW1lbmRlZCcsICdjb3JyZWN0ZWQnIH0NCiAgICAgIGFuZCBIaWdoQ3JlYXRpbmluZVRlc3QudmFsdWUgPSAiSGlnaGVzdFNlcnVtQ3JlYXRpbmluZVJlc3VsdCIoRW5jb3VudGVyV2l0aEhpZ2hDcmVhdGluaW5lKQ0KICAgICAgYW5kIExvd0NyZWF0aW5pbmVUZXN0LnZhbHVlID0gIkxvd2VzdFNlcnVtQ3JlYXRpbmluZVJlc3VsdCIoRW5jb3VudGVyV2l0aEhpZ2hDcmVhdGluaW5lKQ0KICAgICAgYW5kICggSGlnaENyZWF0aW5pbmVUZXN0LnZhbHVlIGFzIFF1YW50aXR5ICkgPj0gKCBMb3dDcmVhdGluaW5lVGVzdC52YWx1ZSBhcyBRdWFudGl0eSApDQogICAgICBhbmQgTG93Q3JlYXRpbmluZVRlc3RUaW1lIDcgZGF5cyBvciBsZXNzIGJlZm9yZSBIaWdoQ3JlYXRpbmluZVRlc3RUaW1lDQogICAgICBhbmQgTG93Q3JlYXRpbmluZVRlc3RUaW1lIGR1cmluZyBIb3NwaXRhbFdpdGhPYnNlcnZhdGlvblBlcmlvZA0KICAgICAgYW5kIEhpZ2hDcmVhdGluaW5lVGVzdFRpbWUgZHVyaW5nIEludGVydmFsW3N0YXJ0IG9mIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kICsgNDggaG91cnMsIHN0YXJ0IG9mIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kICsgMzAgZGF5c10NCiAgICAgIGFuZCBIaWdoQ3JlYXRpbmluZVRlc3RUaW1lIGR1cmluZyBIb3NwaXRhbFdpdGhPYnNlcnZhdGlvblBlcmlvZA0KICAgIHJldHVybiBFbmNvdW50ZXJXaXRoSGlnaENyZWF0aW5pbmUNCg0KDQoNCi8vIFN1cHBvcnRzICJFbmNvdW50ZXIgd2l0aCAyIFRpbWVzIFNlcnVtIENyZWF0aW5pbmUgSW5jcmVhc2UiDQovLyBTdXBwb3J0cyAiRW5jb3VudGVyIHdpdGggMS41IFRpbWVzIFNlcnVtIENyZWF0aW5pbmUgSW5jcmVhc2UiDQoNCg0KDQpkZWZpbmUgIlNlcnVtIENyZWF0aW5pbmUgTm9ybWFsIjoNCiAgaWYgKCBQYXRpZW50LnNleCA9ICcyNDgxNTIwMDInICkgdGhlbiAxLjAyICdtZy9kTCcgDQogICAgZWxzZSAxLjE4ICdtZy9kTCcgDQoNCg0KLy8gRGVub21pbmF0b3IgRXhjbHVzaW9uIGNsYXVzZSA1DQoNCg0KZGVmaW5lICJFbmNvdW50ZXIgd2l0aCBIaWdoIFJpc2sgRGlhZ25vc2lzIGZvciBBS0kiOg0KICAiRW5jb3VudGVyIHdpdGggQ3JlYXRpbmluZSBhbmQgd2l0aG91dCBPYnN0ZXRyaWNhbCBDb25kaXRpb25zIiBRdWFsaWZ5aW5nRW5jb3VudGVyDQogICAgd2hlcmUgZXhpc3RzICggKCBRdWFsaWZ5aW5nRW5jb3VudGVyLmVuY291bnRlckRpYWdub3NpcyAoICkgKSBFbmNvdW50ZXJEaWFnbm9zaXMNCiAgICAgICAgd2hlcmUgKCBFbmNvdW50ZXJEaWFnbm9zaXMuY29kZSBpbiAiSGlnaCBSaXNrIERpYWdub3NpcyBmb3IgQUtJIiApDQogICAgKQ0KDQoNCi8vIERlbm9taW5hdG9yIEV4Y2x1c2lvbiBjbGF1c2UgNg0KDQoNCmRlZmluZSAiRW5jb3VudGVyIHdpdGggSGlnaCBSaXNrIFByb2NlZHVyZXMgZm9yIEFLSSI6DQogICJFbmNvdW50ZXIgd2l0aCBDcmVhdGluaW5lIGFuZCB3aXRob3V0IE9ic3RldHJpY2FsIENvbmRpdGlvbnMiIFF1YWxpZnlpbmdFbmNvdW50ZXINCiAgICB3aXRoIFsiUHJvY2VkdXJlIjogIkhpZ2ggUmlzayBQcm9jZWR1cmVzIGZvciBBS0kiXSBIaWdoUmlza1Byb2NlZHVyZXMNCiAgICAgIHN1Y2ggdGhhdCBIaWdoUmlza1Byb2NlZHVyZXMucGVyZm9ybWVkLnRvSW50ZXJ2YWwgKCApIHN0YXJ0cyBkdXJpbmcgUXVhbGlmeWluZ0VuY291bnRlci5ob3NwaXRhbGl6YXRpb25XaXRoT2JzZXJ2YXRpb24gKCApDQoNCg0KLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gU3VwcG9ydGluZyBEZWZpbml0aW9ucyBmb3IgTnVtZXJhdG9yICAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQ0KDQovLyBOdW1lcmF0b3IgY2xhdXNlIDE6ICBTZWUgIkVuY291bnRlciB3aXRoIDIgVGltZXMgU2VydW0gQ3JlYXRpbmluZSBJbmNyZWFzZSINCg0KLy8gTnVtZXJhdG9yIHVuaW9uIGNsYXVzZSAyDQoNCg0KDQoNCmRlZmluZSAiRW5jb3VudGVyIHdpdGggS2lkbmV5IERpYWx5c2lzIFN0YXJ0ZWQgTW9yZSBUaGFuIDQ4IEhvdXJzIEFmdGVyIEFycml2YWwgd2l0aG91dCBIaWdoIENyZWF0aW5pbmUiOg0KICAiRW5jb3VudGVyIHdpdGggS2lkbmV5IERpYWx5c2lzIFN0YXJ0ZWQgTW9yZSBUaGFuIDQ4IEhvdXJzIEFmdGVyIEFycml2YWwiIEVuY291bnRlcldpdGhEaWFseXNpc0FmdGVyNDhIb3Vycw0KICAgIHdoZXJlIG5vdCAoIGV4aXN0cyAoICJFbmNvdW50ZXIgd2l0aCAyIFRpbWVzIFNlcnVtIENyZWF0aW5pbmUgSW5jcmVhc2UiIEVuY291bnRlcldpdGhIaWdoQ3JlYXRpbmluZQ0KICAgICAgICAgIHdoZXJlICggRW5jb3VudGVyV2l0aEhpZ2hDcmVhdGluaW5lLnBlcmlvZCBpbmNsdWRlcyBFbmNvdW50ZXJXaXRoRGlhbHlzaXNBZnRlcjQ4SG91cnMucGVyaW9kICkNCiAgICAgICkNCiAgICApDQoNCi8vIFN1cHBvcnRzIE51bWVyYXRvciB1bmlvbiBjbGF1c2UgMg0KDQoNCmRlZmluZSAiRW5jb3VudGVyIHdpdGggS2lkbmV5IERpYWx5c2lzIFN0YXJ0ZWQgTW9yZSBUaGFuIDQ4IEhvdXJzIEFmdGVyIEFycml2YWwiOg0KICBmcm9tDQogICAgWyJQcm9jZWR1cmUiOiAiSG9zcGl0YWwgQmFzZWQgRGlhbHlzaXMgU2VydmljZXMiXSBEaWFseXNpcywNCiAgICAiRW5jb3VudGVyIHdpdGggQ3JlYXRpbmluZSBhbmQgd2l0aG91dCBPYnN0ZXRyaWNhbCBDb25kaXRpb25zIiBRdWFsaWZ5aW5nRW5jb3VudGVyDQogICAgbGV0IEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kOiBRdWFsaWZ5aW5nRW5jb3VudGVyLmhvc3BpdGFsaXphdGlvbldpdGhPYnNlcnZhdGlvbiAoICkNCiAgICB3aGVyZSBEaWFseXNpcy5wZXJmb3JtZWQudG9JbnRlcnZhbCAoICkgc3RhcnRzIGR1cmluZyBJbnRlcnZhbFtzdGFydCBvZiBIb3NwaXRhbFdpdGhPYnNlcnZhdGlvblBlcmlvZCArIDQ4IGhvdXJzLCBlbmQgb2YgSG9zcGl0YWxXaXRoT2JzZXJ2YXRpb25QZXJpb2RdDQogICAgICBhbmQgRGlhbHlzaXMucGVyZm9ybWVkLnRvSW50ZXJ2YWwgKCApIHN0YXJ0cyBkdXJpbmcgSG9zcGl0YWxXaXRoT2JzZXJ2YXRpb25QZXJpb2QNCiAgICByZXR1cm4gUXVhbGlmeWluZ0VuY291bnRlcg0KDQoNCi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tIFN1cHBvcnRpbmcgRGVmaW5pdGlvbnMgZm9yIENyZWF0aW5pbmUgY2FsY3VsYXRpb25zIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tDQoNCi8vIFVzZWQgdGhyb3VnaG91dCBjcmVhdGluaW5lIGZ1bmN0aW9ucyBhbmQgZGVmaW5lcyAgDQoNCg0KDQpkZWZpbmUgIlF1YWxpZnlpbmcgQ3JlYXRpbmluZSBMYWIgUmVzdWx0IGJ5IFRpbWUiOg0KICBmcm9tDQogICAgIkVuY291bnRlciB3aXRoIENyZWF0aW5pbmUgYW5kIHdpdGhvdXQgT2JzdGV0cmljYWwgQ29uZGl0aW9ucyIgUXVhbGlmeWluZ0VuY291bnRlciwNCiAgICBbIkxhYm9yYXRvcnlSZXN1bHRPYnNlcnZhdGlvbiI6ICJDcmVhdGluaW5lIE1hc3MgUGVyIFZvbHVtZSJdIENyZWF0aW5pbmVUZXN0QnlUaW1lDQogICAgbGV0IENyRW5jSWQ6IFF1YWxpZnlpbmdFbmNvdW50ZXIuaWQsDQogICAgQ3JIb3NwUGVyaW9kOiBRdWFsaWZ5aW5nRW5jb3VudGVyLmhvc3BpdGFsaXphdGlvbldpdGhPYnNlcnZhdGlvbiAoICksDQogICAgQ3JMYWJJZDogQ3JlYXRpbmluZVRlc3RCeVRpbWUuaWQsDQogICAgQ3JUaW1lOiBDcmVhdGluaW5lVGVzdEJ5VGltZS5lZmZlY3RpdmUuZWFybGllc3QgKCApLA0KICAgIENyVGltZUlzc3VlZDogQ3JlYXRpbmluZVRlc3RCeVRpbWUuaXNzdWVkLA0KICAgIENyUmVzdWx0OiBDcmVhdGluaW5lVGVzdEJ5VGltZS52YWx1ZSBhcyBRdWFudGl0eSwNCiAgICBDclJlc3VsdFZhbHVlOiBDclJlc3VsdC52YWx1ZSwNCiAgICBDclJlc3VsdFVuaXQ6IENyUmVzdWx0LnVuaXQNCiAgICB3aGVyZSBDclRpbWUgZHVyaW5nIENySG9zcFBlcmlvZA0KICAgICAgYW5kIENyZWF0aW5pbmVUZXN0QnlUaW1lLmlzTGFib3JhdG9yeSAoICkNCiAgICAgIGFuZCBDcmVhdGluaW5lVGVzdEJ5VGltZS5zdGF0dXMgaW4geyAnZmluYWwnLCAnYW1lbmRlZCcsICdjb3JyZWN0ZWQnIH0NCiAgICAgIGFuZCBDclJlc3VsdFVuaXQgPSAnbWcvZEwnDQogICAgICBhbmQgQ3JlYXRpbmluZVRlc3RCeVRpbWUudmFsdWUgaXMgbm90IG51bGwNCiAgICAgIGFuZCBDcmVhdGluaW5lVGVzdEJ5VGltZS52YWx1ZSBhcyBRdWFudGl0eSA+IDAgJ21nL2RMJw0KICAgIHJldHVybiBUdXBsZSB7DQogICAgICBDckVuY0luUHRJZDogQ3JFbmNJZCwNCiAgICAgIENySG9zcGl0YWxpemF0aW9uOiBDckhvc3BQZXJpb2QsDQogICAgICBDckxhYk9ic0lkOiBDckxhYklkLA0KLy8gICAgICAgICAgICAgICAgICAgIExhYkNhdGVnb3J5OiBDcmVhdGluaW5lVGVzdEJ5UXVhbnRpdHkuaXNMYWJvcmF0b3J5KCksDQogICAgICANCiAgICAgIENyTGFiT2JzQ2F0ZWdvcnk6IGlmIENyZWF0aW5pbmVUZXN0QnlUaW1lLmlzTGFib3JhdG9yeSAoICkgdGhlbiAnbGFib3JhdG9yeScgDQogICAgICAgIGVsc2UgQ3JlYXRpbmluZVRlc3RCeVRpbWUuaXNMYWJvcmF0b3J5ICggKSwNCiAgICAgIENyTGFiT2JzU3RhdHVzOiBDcmVhdGluaW5lVGVzdEJ5VGltZS5zdGF0dXMsDQogICAgICBDckxhYlJlc3VsdDogQ3JSZXN1bHQsDQogICAgICBDckxhYlJlc3VsdFVuaXQ6IENyUmVzdWx0VW5pdCwNCiAgICAgIENyTGFiUmVzdWx0VmFsdWU6IENyUmVzdWx0VmFsdWUsDQogICAgICBDckxhYlRpbWU6IENyVGltZSwNCiAgICAgIENyTGFiVGltZUlzc3VlZDogQ3JUaW1lSXNzdWVkDQogICAgfQ0KICAgIHNvcnQgYnkgQ3JMYWJUaW1lICANCg0KLy8gU3VwcG9ydGluZyBEZW5vbWluYXRvciBFeGNsdXNpb24gY2xhdXNlIDQNCi8vIFN1cHBvcnRpbmcgTnVtZXJhdG9yICAgICAgICAgICAgICAgDQoNCg0KDQpkZWZpbmUgIkVuY291bnRlciB3aXRoIDEuNSBUaW1lcyBTZXJ1bSBDcmVhdGluaW5lIEluY3JlYXNlIjoNCiAgZnJvbQ0KICAgICJFbmNvdW50ZXIgd2l0aCBDcmVhdGluaW5lIGFuZCB3aXRob3V0IE9ic3RldHJpY2FsIENvbmRpdGlvbnMiIFF1YWxpZnlpbmdFbmNvdW50ZXIsDQogICAgWyJMYWJvcmF0b3J5UmVzdWx0T2JzZXJ2YXRpb24iOiAiQ3JlYXRpbmluZSBNYXNzIFBlciBWb2x1bWUiXSBIaWdoQ3JlYXRpbmluZVRlc3QsDQogICAgWyJMYWJvcmF0b3J5UmVzdWx0T2JzZXJ2YXRpb24iOiAiQ3JlYXRpbmluZSBNYXNzIFBlciBWb2x1bWUiXSBMb3dDcmVhdGluaW5lVGVzdA0KICAgIGxldCBMb3dDcmVhdGluaW5lVGVzdFRpbWU6IExvd0NyZWF0aW5pbmVUZXN0LmVmZmVjdGl2ZS5lYXJsaWVzdCAoICksDQogICAgSGlnaENyZWF0aW5pbmVUZXN0VGltZTogSGlnaENyZWF0aW5pbmVUZXN0LmVmZmVjdGl2ZS5lYXJsaWVzdCAoICksDQogICAgSG9zcGl0YWxXaXRoT2JzZXJ2YXRpb25QZXJpb2Q6IFF1YWxpZnlpbmdFbmNvdW50ZXIuaG9zcGl0YWxpemF0aW9uV2l0aE9ic2VydmF0aW9uICggKQ0KICAgIHdoZXJlICggSGlnaENyZWF0aW5pbmVUZXN0LnZhbHVlID4gIlNlcnVtIENyZWF0aW5pbmUgTm9ybWFsIiApDQogICAgICBhbmQgTG93Q3JlYXRpbmluZVRlc3Quc3RhdHVzIGluIHsgJ2ZpbmFsJywgJ2FtZW5kZWQnLCAnY29ycmVjdGVkJyB9DQogICAgICBhbmQgSGlnaENyZWF0aW5pbmVUZXN0LnN0YXR1cyBpbiB7ICdmaW5hbCcsICdhbWVuZGVkJywgJ2NvcnJlY3RlZCcgfQ0KICAgICAgYW5kIEhpZ2hDcmVhdGluaW5lVGVzdC52YWx1ZSA9ICJIaWdoZXN0U2VydW1DcmVhdGluaW5lUmVzdWx0IihRdWFsaWZ5aW5nRW5jb3VudGVyKQ0KICAgICAgYW5kIExvd0NyZWF0aW5pbmVUZXN0LnZhbHVlID0gIkxvd2VzdFNlcnVtQ3JlYXRpbmluZVJlc3VsdCIoUXVhbGlmeWluZ0VuY291bnRlcikNCiAgICAgIGFuZCAiMS41SW5jcmVhc2VJbkNyZWF0aW5pbmUiKFF1YWxpZnlpbmdFbmNvdW50ZXIpID49IExvd0NyZWF0aW5pbmVUZXN0LnZhbHVlDQogICAgICBhbmQgTG93Q3JlYXRpbmluZVRlc3RUaW1lIDcgZGF5cyBvciBsZXNzIGJlZm9yZSBIaWdoQ3JlYXRpbmluZVRlc3RUaW1lDQogICAgICBhbmQgTG93Q3JlYXRpbmluZVRlc3RUaW1lIGR1cmluZyBIb3NwaXRhbFdpdGhPYnNlcnZhdGlvblBlcmlvZA0KICAgICAgYW5kIEhpZ2hDcmVhdGluaW5lVGVzdFRpbWUgZHVyaW5nIEludGVydmFsW3N0YXJ0IG9mIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kICsgNDggaG91cnMsIHN0YXJ0IG9mIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kICsgMzAgZGF5c10NCiAgICAgIGFuZCBIaWdoQ3JlYXRpbmluZVRlc3RUaW1lIGR1cmluZyBIb3NwaXRhbFdpdGhPYnNlcnZhdGlvblBlcmlvZA0KICAgIHJldHVybiBRdWFsaWZ5aW5nRW5jb3VudGVyDQoNCi8qDQpkZWZpbmUgIkVuY291bnRlciB3aXRoIDEuNSBUaW1lcyBTZXJ1bSBDcmVhdGluaW5lIEluY3JlYXNlIjoNCiAgZnJvbQ0KICAgICJFbmNvdW50ZXIgd2l0aCBDcmVhdGluaW5lIGFuZCB3aXRob3V0IE9ic3RldHJpY2FsIENvbmRpdGlvbnMiIFF1YWxpZnlpbmdFbmNvdW50ZXIsDQogICAgWyJMYWJvcmF0b3J5UmVzdWx0T2JzZXJ2YXRpb24iOiAiQ3JlYXRpbmluZSBNYXNzIFBlciBWb2x1bWUiXSBDcmVhdGluaW5lVGVzdA0KICBsZXQgDQogICAgTG93ZXN0Q3JlYXRpbmluZVRlc3RXaXRoaW43RGF5c1ByaW9yOiAiTG93ZXN0U2VydW1DcmVhdGluaW5lV2l0aGluN0RheXNQcmlvciIoUXVhbGlmeWluZ0VuY291bnRlciwgQ3JlYXRpbmluZVRlc3QpLA0KICAgIENyZWF0aW5pbmVUZXN0VGltZTogQ3JlYXRpbmluZVRlc3QuZWZmZWN0aXZlLmVhcmxpZXN0KCksDQogICAgSG9zcGl0YWxXaXRoT2JzZXJ2YXRpb25QZXJpb2Q6IFF1YWxpZnlpbmdFbmNvdW50ZXIuaG9zcGl0YWxpemF0aW9uV2l0aE9ic2VydmF0aW9uKCkNCiAgd2hlcmUgDQogICAgQ3JlYXRpbmluZVRlc3QudmFsdWUgPj0gTG93ZXN0Q3JlYXRpbmluZVRlc3RXaXRoaW43RGF5c1ByaW9yICogMS41DQogICAgYW5kIENyZWF0aW5pbmVUZXN0LnZhbHVlID4gIlNlcnVtIENyZWF0aW5pbmUgTm9ybWFsIg0KICAgIGFuZCBDcmVhdGluaW5lVGVzdFRpbWUgZHVyaW5nIEludGVydmFsW3N0YXJ0IG9mIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kICsgNDggaG91cnMsIHN0YXJ0IG9mIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uUGVyaW9kICsgMzAgZGF5c10NCiAgcmV0dXJuIFF1YWxpZnlpbmdFbmNvdW50ZXIqLw0KICAgICAgICAgICAgDQoNCi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tIFN1cHBvcnRpbmcgRnVuY3Rpb25zICBmb3IgQ3JlYXRpbmluZSBjYWxjdWxhdGlvbnMgIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tDQoNCi8vIFN1cHBvcnRzIERlbm9taW5hdG9yIEV4Y2x1c2lvbiBjbGF1c2UgMQ0KDQoNCg0KDQpkZWZpbmUgZnVuY3Rpb24gIkNyZWF0aW5pbmVMYWJUZXN0d2l0aFJlc3VsdHdpdGhpbkZpcnN0NDhIb3VycyIoUXVhbGlmeWluZ0VuY291bnRlciBFbmNvdW50ZXIpOg0KICBmcm9tDQogICAgWyJMYWJvcmF0b3J5UmVzdWx0T2JzZXJ2YXRpb24iOiAiQ3JlYXRpbmluZSBNYXNzIFBlciBWb2x1bWUiXSBDcmVhdGluaW5lVGVzdA0KICAgIHdoZXJlIENyZWF0aW5pbmVUZXN0LnZhbHVlIGlzIG5vdCBudWxsDQogICAgICBhbmQgQ3JlYXRpbmluZVRlc3QuZWZmZWN0aXZlLmVhcmxpZXN0ICggKSBkdXJpbmcgSW50ZXJ2YWxbc3RhcnQgb2YgUXVhbGlmeWluZ0VuY291bnRlci5ob3NwaXRhbGl6YXRpb25XaXRoT2JzZXJ2YXRpb24gKCApLCBzdGFydCBvZiBRdWFsaWZ5aW5nRW5jb3VudGVyLmhvc3BpdGFsaXphdGlvbldpdGhPYnNlcnZhdGlvbiAoICkgKyA0OCBob3Vyc10NCiAgICAgIGFuZCBDcmVhdGluaW5lVGVzdC5lZmZlY3RpdmUuZWFybGllc3QgKCApIGR1cmluZyBRdWFsaWZ5aW5nRW5jb3VudGVyLmhvc3BpdGFsaXphdGlvbldpdGhPYnNlcnZhdGlvbiAoICkNCiAgICAgIGFuZCBDcmVhdGluaW5lVGVzdC5zdGF0dXMgaW4geyAnZmluYWwnLCAnYW1lbmRlZCcsICdjb3JyZWN0ZWQnIH0NCiAgICByZXR1cm4gQ3JlYXRpbmluZVRlc3QNCg0KLy8gZUdGUiBmdW5jdGlvbnMgcmV0dXJuIGFzIGRlY2ltYWwgdmFsdWUgcmF0aGVyIHRoYW4gUXVhbnRpdHkNCi8vIFN1cHBvcnRzIERlbm9taW5hdG9yIEV4Y2x1c2lvbiBjbGF1c2UgMg0KLy8gU3VwcG9ydHMgcmlzayB2YXJpYWJsZSAiUmlzayBWYXJpYWJsZSBFc3RpbWF0ZWQgR2xvbWVydWxhciBGaWx0cmF0aW9uIFJhdGUgZm9yIEZlbWFsZXMiDQoNCg0KDQoNCmRlZmluZSBmdW5jdGlvbiAiRmVtYWxlZUdGUiIoUXVhbGlmeWluZ0VuY291bnRlciBFbmNvdW50ZXIpOg0KICBpZiBQYXRpZW50LnNleCA9ICcyNDgxNTIwMDInIHRoZW4gKCAxNDIgKiBNaW4oeygiSW5kZXhDcmVhdGluaW5lIihRdWFsaWZ5aW5nRW5jb3VudGVyKS52YWx1ZSAvIDAuNyksIDEgfSkgXiAoIC0gMC4yNDEgKSAqIE1heCh7KCJJbmRleENyZWF0aW5pbmUiKFF1YWxpZnlpbmdFbmNvdW50ZXIpLnZhbHVlIC8gMC43KSwgMSB9KSBeICggLSAxLjIwMCApICogMC45OTM4IF4gKCBBZ2VJblllYXJzQXQoc3RhcnQgb2YgUXVhbGlmeWluZ0VuY291bnRlci5ob3NwaXRhbGl6YXRpb25XaXRoT2JzZXJ2YXRpb24oKSkgKSAqIDEuMDEyICkgDQogICAgZWxzZSBudWxsDQoNCi8vIGVHRlIgZnVuY3Rpb25zIHJldHVybiBhcyBkZWNpbWFsIHZhbHVlIHJhdGhlciB0aGFuIFF1YW50aXR5DQovLyBTdXBwb3J0cyBEZW5vbWluYXRvciBFeGNsdXNpb24gY2xhdXNlIDINCi8vIFN1cHBvcnRzIHJpc2sgdmFyaWFibGUgIlJpc2sgVmFyaWFibGUgRXN0aW1hdGVkIEdsb21lcnVsYXIgRmlsdHJhdGlvbiBSYXRlIGZvciBGZW1hbGVzIg0KDQoNCg0KDQpkZWZpbmUgZnVuY3Rpb24gIk1hbGVlR0ZSIihRdWFsaWZ5aW5nRW5jb3VudGVyIEVuY291bnRlcik6DQogIGlmIFBhdGllbnQuc2V4ID0gJzI0ODE1MzAwNycgdGhlbiAoIDE0MiAqIE1pbih7KCJJbmRleENyZWF0aW5pbmUiKFF1YWxpZnlpbmdFbmNvdW50ZXIpLnZhbHVlIC8gMC45KSwgMSB9KSBeICggLSAwLjMwMiApICogTWF4KHsoIkluZGV4Q3JlYXRpbmluZSIoUXVhbGlmeWluZ0VuY291bnRlcikudmFsdWUgLyAwLjkpLCAxIH0pIF4gKCAtIDEuMjAwICkgKiAwLjk5MzggXiAoIEFnZUluWWVhcnNBdChzdGFydCBvZiBRdWFsaWZ5aW5nRW5jb3VudGVyLmhvc3BpdGFsaXphdGlvbldpdGhPYnNlcnZhdGlvbigpKSApICkgDQogICAgZWxzZSBudWxsDQoNCg0KLy8gVXNlZCBpbiBHRlIgY2FsY3VsYXRpb25zLCBhYm92ZQ0KDQoNCmRlZmluZSBmdW5jdGlvbiAiSW5kZXhDcmVhdGluaW5lIihRdWFsaWZ5aW5nRW5jb3VudGVyIEVuY291bnRlcik6DQogIENvYWxlc2NlKCJMb3dlc3RTZXJ1bUNyZWF0aW5pbmVJbjI0SG91cnMiKFF1YWxpZnlpbmdFbmNvdW50ZXIpLCBzaW5nbGV0b24gZnJvbSAiRmlyc3RTZXJ1bUNyZWF0aW5pbmVJbjQ4SG91cnMiKFF1YWxpZnlpbmdFbmNvdW50ZXIpKSAgDQoNCi8vIFN1cHBvcnRzIEluZGV4Q3JlYXRpbmluZSBmdW5jdGlvbiAgDQoNCg0KZGVmaW5lIGZ1bmN0aW9uICJMb3dlc3RTZXJ1bUNyZWF0aW5pbmVJbjI0SG91cnMiKFF1YWxpZnlpbmdFbmNvdW50ZXIgRW5jb3VudGVyKToNCiAgTWluKChmcm9tDQogICAgICAgICJRdWFsaWZ5aW5nIENyZWF0aW5pbmUgTGFiIFJlc3VsdCBieSBUaW1lIiBMYWJUZXN0c0xvdw0KICAgICAgICBsZXQgTGFiUmVzdWx0OiBMYWJUZXN0c0xvdy5DckxhYlJlc3VsdA0KICAgICAgICB3aGVyZSBMYWJUZXN0c0xvdy5DckVuY0luUHRJZCA9IFF1YWxpZnlpbmdFbmNvdW50ZXIuaWQNCiAgICAgICAgICBhbmQgTGFiVGVzdHNMb3cuQ3JMYWJUaW1lIGR1cmluZyBJbnRlcnZhbFtzdGFydCBvZiBRdWFsaWZ5aW5nRW5jb3VudGVyLmhvc3BpdGFsaXphdGlvbldpdGhPYnNlcnZhdGlvbigpLCBzdGFydCBvZiBRdWFsaWZ5aW5nRW5jb3VudGVyLmhvc3BpdGFsaXphdGlvbldpdGhPYnNlcnZhdGlvbigpICsgMjQgaG91cnNdDQogICAgKS5DckxhYlJlc3VsdA0KICApIA0KDQovLyBGaW5kIHJlc3VsdCBmcm9tIGVhcmxpZXN0IHRpbWUgaW4gNDggaHJzDQovLyBTdXBwb3J0cyBJbmRleENyZWF0aW5pbmUgZnVuY3Rpb24NCg0KDQoNCmRlZmluZSBmdW5jdGlvbiAiRmlyc3RTZXJ1bUNyZWF0aW5pbmVJbjQ4SG91cnMiKFF1YWxpZnlpbmdFbmNvdW50ZXIgRW5jb3VudGVyKToNCiAgZnJvbQ0KICAgICJRdWFsaWZ5aW5nIENyZWF0aW5pbmUgTGFiIFJlc3VsdCBieSBUaW1lIiBMYWJUZXN0cw0KICAgIGxldCBMYWJSZXN1bHQ6IExhYlRlc3RzLkNyTGFiUmVzdWx0DQogICAgd2hlcmUgKCAiRWFybGllc3RTZXJ1bUNyZWF0aW5pbmVUaW1lSW40OEhvdXJzIihRdWFsaWZ5aW5nRW5jb3VudGVyKSA9IExhYlRlc3RzLkNyTGFiVGltZSApDQogICAgcmV0dXJuIExhYlJlc3VsdCBhcyBRdWFudGl0eQ0KDQoNCi8vIEZpbmQgZWFybGllc3QgdGltZSBpbiA0OCBocnMNCi8vIFN1cHBvcnRzIEZpcnN0U2VydW1DcmVhdGluaW5lSW40OEhvdXJzIGZ1bmN0aW9uDQoNCg0KDQpkZWZpbmUgZnVuY3Rpb24gIkVhcmxpZXN0U2VydW1DcmVhdGluaW5lVGltZUluNDhIb3VycyIoUXVhbGlmeWluZ0VuY291bnRlciBFbmNvdW50ZXIpOg0KICAoIE1pbigoZnJvbQ0KICAgICAgICAgICJRdWFsaWZ5aW5nIENyZWF0aW5pbmUgTGFiIFJlc3VsdCBieSBUaW1lIiBMYWJUZXN0czQ4DQogICAgICAgICAgbGV0IExhYlJlc3VsdDQ4OiBMYWJUZXN0czQ4LkNyTGFiUmVzdWx0DQogICAgICAgICAgd2hlcmUgTGFiVGVzdHM0OC5DckVuY0luUHRJZCA9IFF1YWxpZnlpbmdFbmNvdW50ZXIuaWQNCiAgICAgICAgICAgIGFuZCBMYWJUZXN0czQ4LkNyTGFiVGltZSBkdXJpbmcgSW50ZXJ2YWxbc3RhcnQgb2YgUXVhbGlmeWluZ0VuY291bnRlci5ob3NwaXRhbGl6YXRpb25XaXRoT2JzZXJ2YXRpb24oKSwgc3RhcnQgb2YgUXVhbGlmeWluZ0VuY291bnRlci5ob3NwaXRhbGl6YXRpb25XaXRoT2JzZXJ2YXRpb24oKSArIDQ4IGhvdXJzXQ0KICAgICAgKS5DckxhYlRpbWUNCiAgICApDQogICkNCg0KLy8gU3VwcG9ydHMgRGVub21pbmF0b3IgRXhjbHVzaW9uIGNsYXVzZSAzDQoNCg0KZGVmaW5lIGZ1bmN0aW9uICJFYXJsaWVzdFNlcnVtQ3JlYXRpbmluZVJlc3VsdCIoUXVhbGlmeWluZ0VuY291bnRlciBFbmNvdW50ZXIpOg0KICBmcm9tDQogICAgIlF1YWxpZnlpbmcgQ3JlYXRpbmluZSBMYWIgUmVzdWx0IGJ5IFRpbWUiIExhYlRlc3RzDQogICAgbGV0IExhYlJlc3VsdDogTGFiVGVzdHMuQ3JMYWJSZXN1bHQNCiAgICB3aGVyZSAoICJFYXJsaWVzdFNlcnVtQ3JlYXRpbmluZVRpbWUiKFF1YWxpZnlpbmdFbmNvdW50ZXIpID0gTGFiVGVzdHMuQ3JMYWJUaW1lICkNCiAgICByZXR1cm4gTGFiUmVzdWx0IGFzIFF1YW50aXR5DQoNCi8vICBTdXBwb3J0cyBEZW5vbWluYXRvciBFeGNsdXNpb24gY2xhdXNlIDMgdmlhICJFYXJsaWVzdFNlcnVtQ3JlYXRpbmluZVJlc3VsdCIgZnVuY3Rpb24gIA0KDQoNCmRlZmluZSBmdW5jdGlvbiAiRWFybGllc3RTZXJ1bUNyZWF0aW5pbmVUaW1lIihRdWFsaWZ5aW5nRW5jb3VudGVyIEVuY291bnRlcik6DQogICggTWluKChmcm9tDQogICAgICAgICAgIlF1YWxpZnlpbmcgQ3JlYXRpbmluZSBMYWIgUmVzdWx0IGJ5IFRpbWUiIExhYlRlc3RzRWFybHkNCiAgICAgICAgICBsZXQgTGFiUmVzdWx0RWFybHk6IExhYlRlc3RzRWFybHkuQ3JMYWJSZXN1bHQNCiAgICAgICAgICB3aGVyZSBMYWJUZXN0c0Vhcmx5LkNyRW5jSW5QdElkID0gUXVhbGlmeWluZ0VuY291bnRlci5pZA0KICAgICAgKS5DckxhYlRpbWUNCiAgICApDQogICkNCg0KLy8gTm90IHVzZWQgYW55d2hlcmUNCg0KDQpkZWZpbmUgZnVuY3Rpb24gIlNlcnVtQ3JlYXRpbmluZVNlcXVlbmNlYnlUaW1lIihRdWFsaWZ5aW5nRW5jb3VudGVyIEVuY291bnRlcik6DQogIFsiTGFib3JhdG9yeVJlc3VsdE9ic2VydmF0aW9uIjogIkNyZWF0aW5pbmUgTWFzcyBQZXIgVm9sdW1lIl0gQ3JlYXRpbmluZVRlc3RCeVRpbWUNCiAgICBsZXQgSG9zcGl0YWxXaXRoT2JzZXJ2YXRpb246IFF1YWxpZnlpbmdFbmNvdW50ZXIuaG9zcGl0YWxpemF0aW9uICggKQ0KICAgIHdoZXJlIENyZWF0aW5pbmVUZXN0QnlUaW1lLmVmZmVjdGl2ZS5lYXJsaWVzdCAoICkgZHVyaW5nIEhvc3BpdGFsV2l0aE9ic2VydmF0aW9uDQogICAgICBhbmQgQ3JlYXRpbmluZVRlc3RCeVRpbWUudmFsdWUgaXMgbm90IG51bGwNCiAgICAgIGFuZCBDcmVhdGluaW5lVGVzdEJ5VGltZS5pc0xhYm9yYXRvcnkgKCApDQogICAgICBhbmQgQ3JlYXRpbmluZVRlc3RCeVRpbWUuc3RhdHVzIGluIHsgJ2ZpbmFsJywgJ2FtZW5kZWQnLCAnY29ycmVjdGVkJyB9DQogICAgcmV0dXJuIENyZWF0aW5pbmVUZXN0QnlUaW1lDQoNCi8vIFRoZSBmb2xsb3dpbmcgZnVuY3Rpb25zIHdvcmsgd2l0aCBDciByZXN1bHQgYXMgUXVhbnRpdHkgKG5vdCBkZWNpbWFsLCBidXQgUXVhbnRpdHkgdmFsdWUgaXMgZGVjaW1hbCkNCg0KLy8gU3VwcG9ydGluZyBEZW5vbWluYXRvciBFeGNsdXNpb24gY2xhdXNlIDQgdmlhICJFbmNvdW50ZXIgd2l0aCAyIFRpbWVzIFNlcnVtIENyZWF0aW5pbmUgSW5jcmVhc2UiDQovLyBTdXBwb3J0aW5nIE51bWVyYXRvciB2aWEgIkVuY291bnRlciB3aXRoIDIgVGltZXMgU2VydW0gQ3JlYXRpbmluZSBJbmNyZWFzZSINCg0KDQoNCg0KZGVmaW5lIGZ1bmN0aW9uICJIaWdoZXN0U2VydW1DcmVhdGluaW5lUmVzdWx0IihRdWFsaWZ5aW5nRW5jb3VudGVyIEVuY291bnRlcik6DQogICggTWF4KChmcm9tDQogICAgICAgICAgIlF1YWxpZnlpbmcgQ3JlYXRpbmluZSBMYWIgUmVzdWx0IGJ5IFRpbWUiIExhYlRlc3RzDQogICAgICAgICAgbGV0IExhYlJlc3VsdDogTGFiVGVzdHMuQ3JMYWJSZXN1bHQNCiAgICAgICAgICB3aGVyZSBMYWJUZXN0cy5DckVuY0luUHRJZCA9IFF1YWxpZnlpbmdFbmNvdW50ZXIuaWQNCiAgICAgICkuQ3JMYWJSZXN1bHQNCiAgICApDQogICkNCg0KLy8gU3VwcG9ydGluZyBEZW5vbWluYXRvciBFeGNsdXNpb24gY2xhdXNlIDQgdmlhICJFbmNvdW50ZXIgd2l0aCAxLjUgVGltZXMgU2VydW0gQ3JlYXRpbmluZSBJbmNyZWFzZSINCi8vIFN1cHBvcnRpbmcgTnVtZXJhdG9yICB2aWEgIkVuY291bnRlciB3aXRoIDEuNSBUaW1lcyBTZXJ1bSBDcmVhdGluaW5lIEluY3JlYXNlIiAgICAgICAgICAgICANCg0KDQoNCmRlZmluZSBmdW5jdGlvbiAiMS41SW5jcmVhc2VJbkNyZWF0aW5pbmUiKFF1YWxpZnlpbmdFbmNvdW50ZXIgRW5jb3VudGVyKToNCiAgIkhpZ2hlc3RTZXJ1bUNyZWF0aW5pbmVSZXN1bHQiKFF1YWxpZnlpbmdFbmNvdW50ZXIpIC8gMS41DQoNCi8vIFN1cHBvcnRpbmcgRGVub21pbmF0b3IgRXhjbHVzaW9uIGNsYXVzZSA0IHZpYSAiRW5jb3VudGVyIHdpdGggMS41IFRpbWVzIFNlcnVtIENyZWF0aW5pbmUgSW5jcmVhc2UiDQovLyBTdXBwb3J0aW5nIE51bWVyYXRvciAgdmlhICJFbmNvdW50ZXIgd2l0aCAxLjUgVGltZXMgU2VydW0gQ3JlYXRpbmluZSBJbmNyZWFzZSIgICAgICAgICAgICAgDQoNCg0KDQpkZWZpbmUgZnVuY3Rpb24gIjIuMEluY3JlYXNlSW5DcmVhdGluaW5lIihRdWFsaWZ5aW5nRW5jb3VudGVyIEVuY291bnRlcik6DQogICJIaWdoZXN0U2VydW1DcmVhdGluaW5lUmVzdWx0IihRdWFsaWZ5aW5nRW5jb3VudGVyKSAvIDINCg0KLy8gU3VwcG9ydGluZyBEZW5vbWluYXRvciBFeGNsdXNpb24gY2xhdXNlIDMgdmlhICJJbmNyZWFzZSBvZiAwLjMgb3IgTW9yZSBVc2luZyBMb3dlc3QgQ3JlYXRpbmluZSB3aXRoaW4gMjQgSG91cnMiDQovLyBhbmQgY2xhdXNlIDQgdmlhICJFbmNvdW50ZXIgd2l0aCAxLjUgVGltZXMgU2VydW0gQ3JlYXRpbmluZSBJbmNyZWFzZSINCi8vIFN1cHBvcnRpbmcgTnVtZXJhdG9yICB2aWEgIkVuY291bnRlciB3aXRoIDEuNSBUaW1lcyBTZXJ1bSBDcmVhdGluaW5lIEluY3JlYXNlIiAgICAgICAgICAgICANCg0KDQoNCg0KZGVmaW5lIGZ1bmN0aW9uICJMb3dlc3RTZXJ1bUNyZWF0aW5pbmVSZXN1bHQiKFF1YWxpZnlpbmdFbmNvdW50ZXIgRW5jb3VudGVyKToNCiAgKCBNaW4oKGZyb20NCiAgICAgICAgICAiUXVhbGlmeWluZyBDcmVhdGluaW5lIExhYiBSZXN1bHQgYnkgVGltZSIgTGFiVGVzdHMNCiAgICAgICAgICBsZXQgTGFiUmVzdWx0OiBMYWJUZXN0cy5DckxhYlJlc3VsdA0KICAgICAgICAgIHdoZXJlIExhYlRlc3RzLkNyRW5jSW5QdElkID0gUXVhbGlmeWluZ0VuY291bnRlci5pZA0KICAgICAgKS5DckxhYlJlc3VsdA0KICAgICkNCiAgKQ0KDQovLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSBSaXNrIFZhcmlhYmxlIERlZmluaXRpb25zICAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQ0KDQoNCmRlZmluZSAiUmlzayBWYXJpYWJsZSBFc3RpbWF0ZWQgR2xvbWVydWxhciBGaWx0cmF0aW9uIFJhdGUgZm9yIEZlbWFsZXMiOg0KICAiRW5jb3VudGVyIHdpdGggQ3JlYXRpbmluZSBhbmQgd2l0aG91dCBPYnN0ZXRyaWNhbCBDb25kaXRpb25zIiBRdWFsaWZ5aW5nRW5jb3VudGVyDQogICAgcmV0dXJuIFR1cGxlIHsNCiAgICAgIGVuY291bnRlcklkOiBRdWFsaWZ5aW5nRW5jb3VudGVyLmlkLA0KICAgICAgZUdGUjogIkZlbWFsZWVHRlIiKFF1YWxpZnlpbmdFbmNvdW50ZXIpDQogICAgfQ0KDQpkZWZpbmUgIlJpc2sgVmFyaWFibGUgRXN0aW1hdGVkIEdsb21lcnVsYXIgRmlsdHJhdGlvbiBSYXRlIGZvciBNYWxlcyI6DQogICJFbmNvdW50ZXIgd2l0aCBDcmVhdGluaW5lIGFuZCB3aXRob3V0IE9ic3RldHJpY2FsIENvbmRpdGlvbnMiIFF1YWxpZnlpbmdFbmNvdW50ZXINCiAgICByZXR1cm4gVHVwbGUgew0KICAgICAgZW5jb3VudGVySWQ6IFF1YWxpZnlpbmdFbmNvdW50ZXIuaWQsDQogICAgICBlR0ZSOiAiTWFsZWVHRlIiKFF1YWxpZnlpbmdFbmNvdW50ZXIpDQogICAgfQ0KDQpkZWZpbmUgIlJpc2sgVmFyaWFibGUgQWxsIEVuY291bnRlciBEaWFnbm9zZXMgd2l0aCBQT0EgSW5kaWNhdGlvbiI6DQogIGZyb20NCiAgICAiRW5jb3VudGVyIHdpdGggQ3JlYXRpbmluZSBhbmQgd2l0aG91dCBPYnN0ZXRyaWNhbCBDb25kaXRpb25zIiBRdWFsaWZ5aW5nRW5jb3VudGVyLA0KICAgIFtDbGFpbV0gY2xhaW0NCiAgICB3aGVyZSBjbGFpbS5zdGF0dXMgPSAnYWN0aXZlJw0KICAgICAgYW5kIGNsYWltLnVzZSA9ICdjbGFpbScNCiAgICAgIGFuZCBleGlzdHMgKCBjbGFpbS5pdGVtIEkNCiAgICAgICAgICB3aGVyZSBJLmVuY291bnRlci5yZWZlcmVuY2VzICggUXVhbGlmeWluZ0VuY291bnRlciApDQogICAgICAgICAgICBhbmQgZXhpc3RzICggY2xhaW0uZGlhZ25vc2lzIEQNCiAgICAgICAgICAgICAgICB3aGVyZSBELnNlcXVlbmNlIGluIGNsYWltLml0ZW0uZGlhZ25vc2lzU2VxdWVuY2UNCiAgICAgICAgICAgICAgICAgIGFuZCBELm9uQWRtaXNzaW9uIGluICJQcmVzZW50IG9uIEFkbWlzc2lvbiBvciBDbGluaWNhbGx5IFVuZGV0ZXJtaW5lZCINCiAgICAgICAgICAgICkNCiAgICAgICkNCg0KZGVmaW5lICJSaXNrIFZhcmlhYmxlIEZpcnN0IEhlYXJ0IFJhdGUgaW4gRW5jb3VudGVyIjoNCiAgIkVuY291bnRlciB3aXRoIENyZWF0aW5pbmUgYW5kIHdpdGhvdXQgT2JzdGV0cmljYWwgQ29uZGl0aW9ucyIgUXVhbGlmeWluZ0VuY291bnRlcg0KICAgIHJldHVybiBUdXBsZSB7DQogICAgICBlbmNvdW50ZXJJZDogUXVhbGlmeWluZ0VuY291bnRlci5pZCwNCiAgICAgIGZpcnN0SGVhcnRSYXRlOiAiRmlyc3RIZWFydFJhdGUiKFF1YWxpZnlpbmdFbmNvdW50ZXIpDQogICAgfQ0KDQpkZWZpbmUgIlJpc2sgVmFyaWFibGUgRmlyc3QgUmVzcGlyYXRvcnkgUmF0ZSBpbiBFbmNvdW50ZXIiOg0KICAiRW5jb3VudGVyIHdpdGggQ3JlYXRpbmluZSBhbmQgd2l0aG91dCBPYnN0ZXRyaWNhbCBDb25kaXRpb25zIiBRdWFsaWZ5aW5nRW5jb3VudGVyDQogICAgcmV0dXJuIFR1cGxlIHsNCiAgICAgIGVuY291bnRlcklkOiBRdWFsaWZ5aW5nRW5jb3VudGVyLmlkLA0KICAgICAgZmlyc3RSZXNwaXJhdG9yeVJhdGU6ICJGaXJzdFJlc3BpcmF0b3J5UmF0ZSIoUXVhbGlmeWluZ0VuY291bnRlcikNCiAgICB9DQoNCmRlZmluZSAiUmlzayBWYXJpYWJsZSBGaXJzdCBTeXN0b2xpYyBCbG9vZCBQcmVzc3VyZSBpbiBFbmNvdW50ZXIiOg0KICAiRW5jb3VudGVyIHdpdGggQ3JlYXRpbmluZSBhbmQgd2l0aG91dCBPYnN0ZXRyaWNhbCBDb25kaXRpb25zIiBRdWFsaWZ5aW5nRW5jb3VudGVyDQogICAgcmV0dXJuIFR1cGxlIHsNCiAgICAgIGVuY291bnRlcklkOiBRdWFsaWZ5aW5nRW5jb3VudGVyLmlkLA0KICAgICAgZmlyc3RTeXN0b2xpY0JQOiAiRmlyc3RTeXN0b2xpY0Jsb29kUHJlc3N1cmUiKFF1YWxpZnlpbmdFbmNvdW50ZXIpDQogICAgfQ0KDQpkZWZpbmUgIlJpc2sgVmFyaWFibGUgRmlyc3QgVGVtcGVyYXR1cmUgaW4gRW5jb3VudGVyIjoNCiAgIkVuY291bnRlciB3aXRoIENyZWF0aW5pbmUgYW5kIHdpdGhvdXQgT2JzdGV0cmljYWwgQ29uZGl0aW9ucyIgUXVhbGlmeWluZ0VuY291bnRlcg0KICAgIHJldHVybiBUdXBsZSB7DQogICAgICBlbmNvdW50ZXJJZDogUXVhbGlmeWluZ0VuY291bnRlci5pZCwNCiAgICAgIGZpcnN0VGVtcGVyYXR1cmU6ICJGaXJzdEJvZHlUZW1wZXJhdHVyZSIoUXVhbGlmeWluZ0VuY291bnRlcikNCiAgICB9DQoNCi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tIFJpc2sgVmFyaWFibGUgc3VwcG9ydGluIGZ1bmN0aW9ucyAgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0NCg0KDQpkZWZpbmUgZnVuY3Rpb24gIkZpcnN0Qm9keVRlbXBlcmF0dXJlIihRdWFsaWZ5aW5nRW5jb3VudGVyIEVuY291bnRlcik6DQogIEZpcnN0KFsiVVNDb3JlQm9keVRlbXBlcmF0dXJlUHJvZmlsZSI6ICJCb2R5IHRlbXBlcmF0dXJlIl0gRmlyc3RUZW1wZXJhdHVyZQ0KICAgICAgd2hlcmUgRmlyc3RUZW1wZXJhdHVyZS5lZmZlY3RpdmUuZWFybGllc3QoKSBkdXJpbmcgUXVhbGlmeWluZ0VuY291bnRlci5ob3NwaXRhbGl6YXRpb25XaXRoT2JzZXJ2YXRpb24oKQ0KICAgICAgICBhbmQgRmlyc3RUZW1wZXJhdHVyZS52YWx1ZSBpcyBub3QgbnVsbA0KICAgICAgc29ydCBieSBlZmZlY3RpdmUuZWFybGllc3QoKQ0KICApLnZhbHVlIGFzIFF1YW50aXR5DQoNCmRlZmluZSBmdW5jdGlvbiAiRmlyc3RIZWFydFJhdGUiKFF1YWxpZnlpbmdFbmNvdW50ZXIgRW5jb3VudGVyKToNCiAgRmlyc3QoWyJVU0NvcmVIZWFydFJhdGVQcm9maWxlIjogIkhlYXJ0IHJhdGUiXSBGaXJzdEhlYXJ0QmVhdHMNCiAgICAgIHdoZXJlIEZpcnN0SGVhcnRCZWF0cy5lZmZlY3RpdmUuZWFybGllc3QoKSBkdXJpbmcgUXVhbGlmeWluZ0VuY291bnRlci5ob3NwaXRhbGl6YXRpb25XaXRoT2JzZXJ2YXRpb24oKQ0KICAgICAgICBhbmQgRmlyc3RIZWFydEJlYXRzLnZhbHVlIGlzIG5vdCBudWxsDQogICAgICBzb3J0IGJ5IGVmZmVjdGl2ZS5lYXJsaWVzdCgpDQogICkudmFsdWUgYXMgUXVhbnRpdHkNCg0KZGVmaW5lIGZ1bmN0aW9uICJGaXJzdFJlc3BpcmF0b3J5UmF0ZSIoUXVhbGlmeWluZ0VuY291bnRlciBFbmNvdW50ZXIpOg0KICBGaXJzdChbIlVTQ29yZVJlc3BpcmF0b3J5UmF0ZVByb2ZpbGUiOiAiUmVzcGlyYXRvcnkgcmF0ZSJdIEZpcnN0UmVzcGlyYXRpb24NCiAgICAgIHdoZXJlIEZpcnN0UmVzcGlyYXRpb24uZWZmZWN0aXZlLmVhcmxpZXN0KCkgZHVyaW5nIFF1YWxpZnlpbmdFbmNvdW50ZXIuaG9zcGl0YWxpemF0aW9uV2l0aE9ic2VydmF0aW9uKCkNCiAgICAgICAgYW5kIEZpcnN0UmVzcGlyYXRpb24udmFsdWUgaXMgbm90IG51bGwNCiAgICAgIHNvcnQgYnkgZWZmZWN0aXZlLmVhcmxpZXN0KCkNCiAgKS52YWx1ZSBhcyBRdWFudGl0eQ0KDQpkZWZpbmUgIlF1YWxpZnlpbmcgU3lzdG9saWMgQmxvb2QgUHJlc3N1cmUgUmVhZGluZyI6DQogIFsiVVNDb3JlQmxvb2RQcmVzc3VyZVByb2ZpbGUiXSBCbG9vZFByZXNzdXJlDQogICAgd2hlcmUgQmxvb2RQcmVzc3VyZS5lZmZlY3RpdmUuZWFybGllc3QgKCApIGR1cmluZyBkYXkgb2YgIk1lYXN1cmVtZW50IFBlcmlvZCINCg0KZGVmaW5lIGZ1bmN0aW9uICJGaXJzdFN5c3RvbGljQmxvb2RQcmVzc3VyZSIoUXVhbGlmeWluZ0VuY291bnRlciBFbmNvdW50ZXIpOg0KICBGaXJzdCgiUXVhbGlmeWluZyBTeXN0b2xpYyBCbG9vZCBQcmVzc3VyZSBSZWFkaW5nIiBTQlBSZWFkaW5nDQogICAgICB3aGVyZSBTQlBSZWFkaW5nLmVmZmVjdGl2ZS5lYXJsaWVzdCgpIGR1cmluZyBRdWFsaWZ5aW5nRW5jb3VudGVyLmhvc3BpdGFsaXphdGlvbldpdGhPYnNlcnZhdGlvbigpDQogICAgICByZXR1cm4gc2luZ2xldG9uIGZyb20oU0JQUmVhZGluZy5jb21wb25lbnQgU0JQQ29tcG9uZW50DQogICAgICAgICAgd2hlcmUgU0JQQ29tcG9uZW50LmNvZGUgfiAiU3lzdG9saWMgYmxvb2QgcHJlc3N1cmUiDQogICAgICAgICAgcmV0dXJuIFNCUENvbXBvbmVudC52YWx1ZSBhcyBRdWFudGl0eQ0KICAgICAgKQ0KICApDQovKiANCmRlZmluZSBmdW5jdGlvbiAiRmlyc3RTeXN0b2xpY0Jsb29kUHJlc3N1cmVUZXN0IihRdWFsaWZ5aW5nRW5jb3VudGVyIEVuY291bnRlciApOg0KICBGaXJzdChbIlVTQ29yZUJsb29kUHJlc3N1cmVQcm9maWxlIjogIlN5c3RvbGljIGJsb29kIHByZXNzdXJlIl0gRmlyc3RTeXN0b2xpYw0KICAgICAgd2hlcmUgRmlyc3RTeXN0b2xpYy5lZmZlY3RpdmUuZWFybGllc3QoKSBkdXJpbmcgUXVhbGlmeWluZ0VuY291bnRlci5ob3NwaXRhbGl6YXRpb25XaXRoT2JzZXJ2YXRpb24oKQ0KICAgICAgICBhbmQgRmlyc3RTeXN0b2xpYy52YWx1ZSBpcyBub3QgbnVsbA0KICAgICAgc29ydCBieSBlZmZlY3RpdmUuZWFybGllc3QoKQ0KICApLnZhbHVlIGFzIFF1YW50aXR5DQoqLw0KDQovLyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gVGVtcG9yYXJ5IGRlYnVnZ2luZyBkZWZpbmVzIHNob3VsZCBnbyBoZXJlIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQ0KDQo="/>
</content>
</Library>