health-data-standards 2.2.1 → 3.0.3

Sign up to get free protection for your applications and to get access to all the features.
Files changed (264) hide show
  1. data/Gemfile +5 -0
  2. data/lib/health-data-standards.rb +55 -12
  3. data/lib/health-data-standards/export/c32.rb +7 -6
  4. data/lib/health-data-standards/export/cat_1.rb +18 -0
  5. data/lib/health-data-standards/export/ccda.rb +8 -8
  6. data/lib/health-data-standards/export/green_c32/entry.rb +8 -5
  7. data/lib/health-data-standards/export/green_c32/export_generator.rb +1 -1
  8. data/lib/health-data-standards/export/green_c32/record.rb +10 -7
  9. data/lib/health-data-standards/export/hdata/metadata.rb +7 -6
  10. data/lib/health-data-standards/export/helper/cat1_view_helper.rb +133 -0
  11. data/lib/health-data-standards/export/helper/gc32_view_helper.rb +39 -0
  12. data/lib/health-data-standards/export/helper/html_view_helper.rb +23 -0
  13. data/lib/health-data-standards/export/html.rb +22 -10
  14. data/lib/health-data-standards/export/qrda/entry_template_resolver.rb +31 -0
  15. data/lib/health-data-standards/export/qrda/hqmf-qrda-oids.json +638 -0
  16. data/lib/health-data-standards/export/rendering_context.rb +37 -3
  17. data/lib/health-data-standards/export/template_helper.rb +20 -23
  18. data/lib/health-data-standards/export/view_helper.rb +8 -42
  19. data/lib/health-data-standards/import/bundle/importer.rb +148 -0
  20. data/lib/health-data-standards/import/c32/care_goal_importer.rb +14 -26
  21. data/lib/health-data-standards/import/c32/condition_importer.rb +12 -51
  22. data/lib/health-data-standards/import/c32/immunization_importer.rb +9 -27
  23. data/lib/health-data-standards/import/c32/insurance_provider_importer.rb +24 -21
  24. data/lib/health-data-standards/import/c32/patient_importer.rb +27 -34
  25. data/lib/health-data-standards/import/cat1/diagnosis_active_importer.rb +19 -0
  26. data/lib/health-data-standards/import/cat1/diagnosis_inactive_importer.rb +19 -0
  27. data/lib/health-data-standards/import/cat1/diagnostic_study_order_importer.rb +24 -0
  28. data/lib/health-data-standards/import/cat1/encounter_order_importer.rb +24 -0
  29. data/lib/health-data-standards/import/cat1/entry_package.rb +25 -0
  30. data/lib/health-data-standards/import/cat1/gestational_age_importer.rb +42 -0
  31. data/lib/health-data-standards/import/cat1/lab_order_importer.rb +24 -0
  32. data/lib/health-data-standards/import/cat1/medication_dispensed_importer.rb +13 -0
  33. data/lib/health-data-standards/import/cat1/patient_importer.rb +104 -0
  34. data/lib/health-data-standards/import/cat1/procedure_intolerance_importer.rb +13 -0
  35. data/lib/health-data-standards/import/cat1/procedure_order_importer.rb +38 -0
  36. data/lib/health-data-standards/import/cat1/tobacco_use_importer.rb +19 -0
  37. data/lib/health-data-standards/import/ccda/allergy_importer.rb +2 -8
  38. data/lib/health-data-standards/import/ccda/care_goal_importer.rb +1 -1
  39. data/lib/health-data-standards/import/ccda/condition_importer.rb +2 -3
  40. data/lib/health-data-standards/import/ccda/encounter_importer.rb +2 -5
  41. data/lib/health-data-standards/import/ccda/immunization_importer.rb +1 -3
  42. data/lib/health-data-standards/import/ccda/insurance_provider_importer.rb +1 -1
  43. data/lib/health-data-standards/import/ccda/medical_equipment_importer.rb +2 -4
  44. data/lib/health-data-standards/import/ccda/medication_importer.rb +2 -4
  45. data/lib/health-data-standards/import/ccda/patient_importer.rb +1 -1
  46. data/lib/health-data-standards/import/ccda/procedure_importer.rb +2 -6
  47. data/lib/health-data-standards/import/ccda/result_importer.rb +2 -5
  48. data/lib/health-data-standards/import/ccda/vital_sign_importer.rb +2 -3
  49. data/lib/health-data-standards/import/cda/allergy_importer.rb +32 -0
  50. data/lib/health-data-standards/import/cda/condition_importer.rb +51 -0
  51. data/lib/health-data-standards/import/{c32 → cda}/encounter_importer.rb +9 -35
  52. data/lib/health-data-standards/import/cda/entry_finder.rb +22 -0
  53. data/lib/health-data-standards/import/{c32 → cda}/locatable_import_utils.rb +2 -3
  54. data/lib/health-data-standards/import/cda/medical_equipment_importer.rb +24 -0
  55. data/lib/health-data-standards/import/{c32 → cda}/medication_importer.rb +13 -15
  56. data/lib/health-data-standards/import/cda/narrative_reference_handler.rb +35 -0
  57. data/lib/health-data-standards/import/{c32 → cda}/organization_importer.rb +1 -1
  58. data/lib/health-data-standards/import/cda/procedure_importer.rb +35 -0
  59. data/lib/health-data-standards/import/{c32 → cda}/provider_importer.rb +4 -8
  60. data/lib/health-data-standards/import/cda/result_importer.rb +31 -0
  61. data/lib/health-data-standards/import/{c32 → cda}/section_importer.rb +26 -48
  62. data/lib/health-data-standards/import/{c32 → cda}/vital_sign_importer.rb +2 -3
  63. data/lib/health-data-standards/import/green_c32/section_importer.rb +8 -6
  64. data/lib/health-data-standards/import/provider_import_utils.rb +2 -2
  65. data/lib/health-data-standards/models/address.rb +1 -1
  66. data/lib/health-data-standards/models/condition.rb +7 -6
  67. data/lib/health-data-standards/models/cqm/bundle.rb +45 -0
  68. data/lib/health-data-standards/models/cqm/measure.rb +36 -0
  69. data/lib/health-data-standards/models/guarantor.rb +1 -0
  70. data/lib/health-data-standards/models/insurance_provider.rb +2 -0
  71. data/lib/health-data-standards/models/order_information.rb +2 -0
  72. data/lib/health-data-standards/models/record.rb +6 -5
  73. data/lib/health-data-standards/models/svs/value_set.rb +1 -1
  74. data/lib/health-data-standards/railtie.rb +11 -0
  75. data/lib/health-data-standards/tasks/bundle.rake +107 -0
  76. data/lib/health-data-standards/util/code_system_helper.rb +9 -1
  77. data/lib/hqmf-generator/attribute.xml.erb +11 -0
  78. data/lib/hqmf-generator/characteristic_criteria.xml.erb +21 -0
  79. data/lib/hqmf-generator/code.xml.erb +13 -0
  80. data/lib/hqmf-generator/condition_criteria.xml.erb +22 -0
  81. data/lib/hqmf-generator/derivation.xml.erb +6 -0
  82. data/lib/hqmf-generator/description.xml.erb +1 -0
  83. data/lib/hqmf-generator/document.xml.erb +63 -0
  84. data/lib/hqmf-generator/effective_time.xml.erb +4 -0
  85. data/lib/hqmf-generator/encounter_criteria.xml.erb +21 -0
  86. data/lib/hqmf-generator/field.xml.erb +28 -0
  87. data/lib/hqmf-generator/hqmf-generator.rb +292 -0
  88. data/lib/hqmf-generator/observation_criteria.xml.erb +25 -0
  89. data/lib/hqmf-generator/population_criteria.xml.erb +23 -0
  90. data/lib/hqmf-generator/precondition.xml.erb +14 -0
  91. data/lib/hqmf-generator/procedure_criteria.xml.erb +22 -0
  92. data/lib/hqmf-generator/reason.xml.erb +3 -0
  93. data/lib/hqmf-generator/reference.xml.erb +3 -0
  94. data/lib/hqmf-generator/source.xml.erb +6 -0
  95. data/lib/hqmf-generator/specific_occurrence.xml.erb +7 -0
  96. data/lib/hqmf-generator/subset.xml.erb +8 -0
  97. data/lib/hqmf-generator/substance_criteria.xml.erb +26 -0
  98. data/lib/hqmf-generator/supply_criteria.xml.erb +26 -0
  99. data/lib/hqmf-generator/template_id.xml.erb +5 -0
  100. data/lib/hqmf-generator/temporal_relationship.xml.erb +6 -0
  101. data/lib/hqmf-generator/value.xml.erb +24 -0
  102. data/lib/hqmf-generator/variable_criteria.xml.erb +12 -0
  103. data/lib/hqmf-model/attribute.rb +35 -0
  104. data/lib/hqmf-model/data_criteria.json +1123 -0
  105. data/lib/hqmf-model/data_criteria.rb +344 -0
  106. data/lib/hqmf-model/document.rb +178 -0
  107. data/lib/hqmf-model/population_criteria.rb +96 -0
  108. data/lib/hqmf-model/precondition.rb +91 -0
  109. data/lib/hqmf-model/types.rb +319 -0
  110. data/lib/hqmf-model/utilities.rb +52 -0
  111. data/lib/hqmf-parser.rb +56 -0
  112. data/lib/hqmf-parser/1.0/attribute.rb +68 -0
  113. data/lib/hqmf-parser/1.0/comparison.rb +34 -0
  114. data/lib/hqmf-parser/1.0/data_criteria.rb +92 -0
  115. data/lib/hqmf-parser/1.0/data_criteria_oid_xpath.json +91 -0
  116. data/lib/hqmf-parser/1.0/document.rb +203 -0
  117. data/lib/hqmf-parser/1.0/expression.rb +58 -0
  118. data/lib/hqmf-parser/1.0/observation.rb +61 -0
  119. data/lib/hqmf-parser/1.0/population_criteria.rb +75 -0
  120. data/lib/hqmf-parser/1.0/precondition.rb +89 -0
  121. data/lib/hqmf-parser/1.0/range.rb +65 -0
  122. data/lib/hqmf-parser/1.0/restriction.rb +160 -0
  123. data/lib/hqmf-parser/1.0/utilities.rb +41 -0
  124. data/lib/hqmf-parser/2.0/data_criteria.rb +319 -0
  125. data/lib/hqmf-parser/2.0/document.rb +165 -0
  126. data/lib/hqmf-parser/2.0/population_criteria.rb +53 -0
  127. data/lib/hqmf-parser/2.0/precondition.rb +44 -0
  128. data/lib/hqmf-parser/2.0/types.rb +223 -0
  129. data/lib/hqmf-parser/2.0/utilities.rb +30 -0
  130. data/lib/hqmf-parser/converter/pass1/data_criteria_converter.rb +252 -0
  131. data/lib/hqmf-parser/converter/pass1/document_converter.rb +185 -0
  132. data/lib/hqmf-parser/converter/pass1/population_criteria_converter.rb +165 -0
  133. data/lib/hqmf-parser/converter/pass1/precondition_converter.rb +173 -0
  134. data/lib/hqmf-parser/converter/pass1/precondition_extractor.rb +188 -0
  135. data/lib/hqmf-parser/converter/pass1/simple_data_criteria.rb +26 -0
  136. data/lib/hqmf-parser/converter/pass1/simple_operator.rb +89 -0
  137. data/lib/hqmf-parser/converter/pass1/simple_population_criteria.rb +10 -0
  138. data/lib/hqmf-parser/converter/pass1/simple_precondition.rb +51 -0
  139. data/lib/hqmf-parser/converter/pass1/simple_restriction.rb +64 -0
  140. data/lib/hqmf-parser/converter/pass2/comparison_converter.rb +112 -0
  141. data/lib/hqmf-parser/converter/pass2/operator_converter.rb +102 -0
  142. data/lib/hqmf-parser/parser.rb +54 -0
  143. data/lib/hqmf-parser/value_sets/value_set_parser.rb +241 -0
  144. data/lib/util/counter.rb +20 -0
  145. data/templates/{_allergies.c32.erb → c32/_allergies.c32.erb} +0 -0
  146. data/templates/{_allergies_no_current.c32.erb → c32/_allergies_no_current.c32.erb} +0 -0
  147. data/templates/{_care_goals.c32.erb → c32/_care_goals.c32.erb} +0 -0
  148. data/templates/{_code_with_reference.c32.erb → c32/_code_with_reference.c32.erb} +0 -0
  149. data/templates/{_conditions.c32.erb → c32/_conditions.c32.erb} +0 -0
  150. data/templates/{_conditions_no_current.c32.erb → c32/_conditions_no_current.c32.erb} +0 -0
  151. data/templates/{_encounters.c32.erb → c32/_encounters.c32.erb} +0 -0
  152. data/templates/{_immunizations.c32.erb → c32/_immunizations.c32.erb} +0 -0
  153. data/templates/{_medical_equipment.c32.erb → c32/_medical_equipment.c32.erb} +0 -0
  154. data/templates/{_medications.c32.erb → c32/_medications.c32.erb} +0 -0
  155. data/templates/{_medications_no_current.c32.erb → c32/_medications_no_current.c32.erb} +0 -0
  156. data/templates/{_narrative_block.c32.erb → c32/_narrative_block.c32.erb} +0 -0
  157. data/templates/{_procedures.c32.erb → c32/_procedures.c32.erb} +0 -0
  158. data/templates/{_results.c32.erb → c32/_results.c32.erb} +0 -0
  159. data/templates/{_social_history.c32.erb → c32/_social_history.c32.erb} +0 -0
  160. data/templates/{_vital_signs.c32.erb → c32/_vital_signs.c32.erb} +0 -0
  161. data/templates/{show.c32.erb → c32/show.c32.erb} +0 -0
  162. data/templates/cat1/_2.16.840.1.113883.10.20.22.4.85.cat1.erb +18 -0
  163. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.1.cat1.erb +14 -0
  164. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.101.cat1.erb +25 -0
  165. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.103.cat1.erb +12 -0
  166. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.105.cat1.erb +60 -0
  167. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.11.cat1.erb +41 -0
  168. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.12.cat1.erb +50 -0
  169. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.13.cat1.erb +37 -0
  170. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.14.cat1.erb +35 -0
  171. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.17.cat1.erb +22 -0
  172. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.18.cat1.erb +21 -0
  173. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.2.cat1.erb +28 -0
  174. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.20.cat1.erb +20 -0
  175. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.22.cat1.erb +21 -0
  176. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.23.cat1.erb +71 -0
  177. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.28.cat1.erb +20 -0
  178. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.3.cat1.erb +24 -0
  179. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.31.cat1.erb +20 -0
  180. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.32.cat1.erb +15 -0
  181. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.34.cat1.erb +58 -0
  182. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.37.cat1.erb +20 -0
  183. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.38.cat1.erb +16 -0
  184. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.4.cat1.erb +27 -0
  185. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.40.cat1.erb +17 -0
  186. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.41.cat1.erb +38 -0
  187. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.42.cat1.erb +38 -0
  188. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.43.cat1.erb +24 -0
  189. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.44.cat1.erb +24 -0
  190. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.45.cat1.erb +26 -0
  191. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.46.cat1.erb +30 -0
  192. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.47.cat1.erb +26 -0
  193. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.51.cat1.erb +13 -0
  194. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.54.cat1.erb +16 -0
  195. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.55.cat1.erb +10 -0
  196. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.57.cat1.erb +19 -0
  197. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.59.cat1.erb +17 -0
  198. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.62.cat1.erb +36 -0
  199. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.63.cat1.erb +23 -0
  200. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.64.cat1.erb +29 -0
  201. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.66.cat1.erb +34 -0
  202. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.69.cat1.erb +23 -0
  203. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.7.cat1.erb +30 -0
  204. data/templates/cat1/_2.16.840.1.113883.10.20.24.3.76.cat1.erb +32 -0
  205. data/templates/cat1/_measures.cat1.erb +66 -0
  206. data/templates/cat1/_medication_details.cat1.erb +9 -0
  207. data/templates/cat1/_ordinality.cat1.erb +4 -0
  208. data/templates/cat1/_patient_data.cat1.erb +14 -0
  209. data/templates/cat1/_reason.cat1.erb +16 -0
  210. data/templates/cat1/_record_target.cat1.erb +39 -0
  211. data/templates/cat1/_reporting_parameters.cat1.erb +24 -0
  212. data/templates/cat1/_result_value.cat1.erb +16 -0
  213. data/templates/cat1/show.cat1.erb +125 -0
  214. data/templates/{_address.gc32.erb → gc32/_address.gc32.erb} +1 -1
  215. data/templates/gc32/_advance_directive.gc32.erb +5 -0
  216. data/templates/gc32/_allergy.gc32.erb +12 -0
  217. data/templates/{_care_goal.gc32.erb → gc32/_care_goal.gc32.erb} +1 -1
  218. data/templates/gc32/_condition.gc32.erb +10 -0
  219. data/templates/gc32/_encounter.gc32.erb +28 -0
  220. data/templates/gc32/_entry.gc32.erb +3 -0
  221. data/templates/gc32/_entry_attributes.gc32.erb +10 -0
  222. data/templates/gc32/_immunization.gc32.erb +9 -0
  223. data/templates/gc32/_insurance_provider.gc32.erb +28 -0
  224. data/templates/gc32/_medical_equipment.gc32.erb +6 -0
  225. data/templates/gc32/_medication.gc32.erb +91 -0
  226. data/templates/{_name.gc32.erb → gc32/_name.gc32.erb} +0 -0
  227. data/templates/gc32/_organization.gc32.erb +10 -0
  228. data/templates/gc32/_person_attributes.gc32.erb +7 -0
  229. data/templates/gc32/_procedure.gc32.erb +9 -0
  230. data/templates/gc32/_provider.gc32.erb +9 -0
  231. data/templates/gc32/_result.gc32.erb +12 -0
  232. data/templates/gc32/_social_history.gc32.erb +6 -0
  233. data/templates/{_support.gc32.erb → gc32/_support.gc32.erb} +4 -3
  234. data/templates/gc32/_telecom.gc32.erb +1 -0
  235. data/templates/gc32/_vital_sign.gc32.erb +4 -0
  236. data/templates/{record.gc32.erb → gc32/record.gc32.erb} +26 -10
  237. data/templates/html/_entries_by_encounter.html.erb +2 -2
  238. data/templates/html/_entries_by_section.html.erb +1 -1
  239. data/templates/html/_entry.html.erb +16 -21
  240. data/templates/html/_header.html.erb +1 -1
  241. data/templates/html/_section.html.erb +1 -1
  242. data/templates/html/show.html.erb +23 -2
  243. data/templates/metadata.hdata.erb +3 -3
  244. metadata +282 -54
  245. data/lib/health-data-standards/import/c32/allergy_importer.rb +0 -47
  246. data/lib/health-data-standards/import/c32/medical_equipment_importer.rb +0 -45
  247. data/lib/health-data-standards/import/c32/procedure_importer.rb +0 -62
  248. data/lib/health-data-standards/import/c32/result_importer.rb +0 -56
  249. data/templates/_advance_directive.gc32.erb +0 -8
  250. data/templates/_allergy.gc32.erb +0 -23
  251. data/templates/_condition.gc32.erb +0 -9
  252. data/templates/_encounter.gc32.erb +0 -26
  253. data/templates/_entry.gc32.erb +0 -14
  254. data/templates/_immunization.gc32.erb +0 -11
  255. data/templates/_insurance_provider.gc32.erb +0 -0
  256. data/templates/_medical_equipment.gc32.erb +0 -7
  257. data/templates/_medication.gc32.erb +0 -72
  258. data/templates/_organization.gc32.erb +0 -10
  259. data/templates/_procedure.gc32.erb +0 -10
  260. data/templates/_provider.gc32.erb +0 -19
  261. data/templates/_result.gc32.erb +0 -16
  262. data/templates/_social_history.gc32.erb +0 -8
  263. data/templates/_telecom.gc32.erb +0 -1
  264. data/templates/_vital_sign.gc32.erb +0 -7
@@ -0,0 +1,344 @@
1
+ module HQMF
2
+ # Represents a data criteria specification
3
+ class DataCriteria
4
+
5
+ include HQMF::Conversion::Utilities
6
+
7
+ SOURCE_DATA_CRITERIA_TEMPLATE_ID = '2.16.840.1.113883.3.100.1.1'
8
+ SOURCE_DATA_CRITERIA_TEMPLATE_TITLE = 'Source data criteria'
9
+
10
+ XPRODUCT = 'XPRODUCT'
11
+ UNION = 'UNION'
12
+
13
+ FIELDS = {'SEVERITY' => {title:'Severity', coded_entry_method: :severity, code: 'SEV', code_system:'2.16.840.1.113883.5.4', template_id: '2.16.840.1.113883.3.560.1.1021.2', field_type: :value},
14
+ 'ORDINAL' => {title:'Ordinal', coded_entry_method: :ordinality, code: '117363000', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1012.2', field_type: :value},
15
+ 'REASON' => {title:'Reason', coded_entry_method: :reason, code: '410666004', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1017.2', field_type: :value},
16
+ 'SOURCE' => {title:'Source', coded_entry_method: :source, code: '260753009', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.2001.2', field_type: :value},
17
+ 'CUMULATIVE_MEDICATION_DURATION' => {title:'Cumulative Medication Duration', coded_entry_method: :cumulative_medication_duration, code: '363819003', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1001.3', field_type: :value},
18
+ 'FACILITY_LOCATION' => {title:'Facility Location', coded_entry_method: :facility, code: 'SDLOC', field_type: :value},
19
+ 'FACILITY_LOCATION_ARRIVAL_DATETIME' => {title:'Facility Location Arrival Date/Time', coded_entry_method: :facility_arrival, code: 'SDLOC_ARRIVAL', field_type: :nested_timestamp},
20
+ 'FACILITY_LOCATION_DEPARTURE_DATETIME' => {title:'Facility Location Departure Date/Time', coded_entry_method: :facility_departure, code: 'SDLOC_DEPARTURE', field_type: :nested_timestamp},
21
+ 'DISCHARGE_DATETIME' => {title:'Discharge Date/Time', coded_entry_method: :discharge_time, code: '442864001', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1025.1', field_type: :timestamp},
22
+ 'DISCHARGE_STATUS' => {title:'Discharge Status', coded_entry_method: :discharge_disposition, code: '309039003', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1003.2', field_type: :value},
23
+ 'ADMISSION_DATETIME' => {title:'Admission Date/Time', coded_entry_method: :admit_time, code: '399423000', code_system:'2.16.840.1.113883.6.96', field_type: :timestamp},
24
+ 'LENGTH_OF_STAY' => {title:'Length of Stay', coded_entry_method: :length_of_stay, code: '183797002', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1029.3', field_type: :value},
25
+ 'DOSE' => {title:'Dose', coded_entry_method: :dose, code: '398232005', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1004.1', field_type: :value},
26
+ 'ROUTE' => {title:'Route', coded_entry_method: :route, code: '263513008', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1020.2', field_type: :value},
27
+ 'START_DATETIME' => {title:'Start Date/Time', coded_entry_method: :start_date, code: '398201009', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1027.1', field_type: :timestamp},
28
+ 'FREQUENCY' => {title:'Frequency', coded_entry_method: :frequency, code: '260864003', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1006.1', field_type: :value},
29
+ 'ANATOMICAL_STRUCTURE' => {title:'Anatomical Structure', coded_entry_method: :anatomical_structure, code: '91723000', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1000.2', field_type: :value},
30
+ 'STOP_DATETIME' => {title:'Stop Date/Time', coded_entry_method: :end_date, code: '397898000', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1026.1', field_type: :timestamp},
31
+ 'INCISION_DATETIME' => {title:'Incision Date/Time', coded_entry_method: :incision_time, code: '34896006', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1007.1', field_type: :timestamp},
32
+ 'REMOVAL_DATETIME' => {title:'Removal Date/Time', coded_entry_method: :removal_time, code: '118292001', code_system:'2.16.840.1.113883.6.96', template_id: '2.16.840.1.113883.3.560.1.1032.1', field_type: :timestamp},
33
+ 'TRANSFER_TO' => {title:'Transfer To', coded_entry_method: :transfer_to, code: 'TRANSFER_TO', template_id: '2.16.840.1.113883.3.560.1.72', field_type: :value},
34
+ 'TRANSFER_FROM' => {title:'Transfer From', coded_entry_method: :transfer_from, code: 'TRANSFER_FROM', template_id: '2.16.840.1.113883.3.560.1.71', field_type: :value}
35
+ }
36
+
37
+ VALUE_FIELDS = {'SEV' => 'SEVERITY',
38
+ '117363000' => 'ORDINAL',
39
+ '410666004' => 'REASON',
40
+ '260753009' => 'SOURCE',
41
+ '363819003' => 'CUMULATIVE_MEDICATION_DURATION',
42
+ 'SDLOC' => 'FACILITY_LOCATION',
43
+ '442864001' => 'DISCHARGE_DATETIME',
44
+ '309039003' => 'DISCHARGE_STATUS',
45
+ '399423000' => 'ADMISSION_DATETIME',
46
+ '183797002' => 'LENGTH_OF_STAY',
47
+ '398232005' => 'DOSE',
48
+ '263513008' => 'ROUTE',
49
+ '398201009' => 'START_DATETIME',
50
+ '260864003' =>'FREQUENCY',
51
+ '91723000' => 'ANATOMICAL_STRUCTURE',
52
+ '397898000' => 'STOP_DATETIME',
53
+ '34896006' => 'INCISION_DATETIME',
54
+ '118292001' =>'REMOVAL_DATETIME'
55
+ }
56
+
57
+
58
+ attr_reader :title, :description, :code_list_id, :children_criteria, :derivation_operator , :specific_occurrence, :specific_occurrence_const, :source_data_criteria
59
+ attr_accessor :id, :value, :field_values, :effective_time, :status, :temporal_references, :subset_operators, :definition, :inline_code_list, :negation_code_list_id, :negation, :display_name
60
+
61
+ # Create a new data criteria instance
62
+ # @param [String] id
63
+ # @param [String] title
64
+ # @param [String] display_name
65
+ # @param [String] description
66
+ # @param [String] code_list_id
67
+ # @param [String] negation_code_list_id
68
+ # @param [List<String>] children_criteria (ids of children data criteria)
69
+ # @param [String] derivation_operator
70
+ # @param [String] definition
71
+ # @param [String] status
72
+ # @param [Value|Range|Coded] value
73
+ # @param [Hash<String,Value|Range|Coded>] field_values
74
+ # @param [Range] effective_time
75
+ # @param [Hash<String,[String]>] inline_code_list
76
+ # @param [boolean] negation
77
+ # @param [String] negation_code_list_id
78
+ # @param [List<TemporalReference>] temporal_references
79
+ # @param [List<SubsetOperator>] subset_operators
80
+ # @param [String] specific_occurrence
81
+ # @param [String] specific_occurrence_const
82
+ # @param [String] source_data_criteria (id for the source data criteria, important for specific occurrences)
83
+ def initialize(id, title, display_name, description, code_list_id, children_criteria, derivation_operator, definition, status, value, field_values, effective_time, inline_code_list, negation, negation_code_list_id, temporal_references, subset_operators, specific_occurrence, specific_occurrence_const, source_data_criteria=nil)
84
+
85
+ status = normalize_status(definition, status)
86
+ @settings = HQMF::DataCriteria.get_settings_for_definition(definition, status)
87
+
88
+ @id = id
89
+ @title = title
90
+ @description = description
91
+ @code_list_id = code_list_id
92
+ @negation_code_list_id = negation_code_list_id
93
+ @children_criteria = children_criteria
94
+ @derivation_operator = derivation_operator
95
+ @definition = definition
96
+ @status = status
97
+ @value = value
98
+ @field_values = field_values
99
+ @effective_time = effective_time
100
+ @inline_code_list = inline_code_list
101
+ @negation = negation
102
+ @negation_code_list_id = negation_code_list_id
103
+ @temporal_references = temporal_references
104
+ @subset_operators = subset_operators
105
+ @specific_occurrence = specific_occurrence
106
+ @specific_occurrence_const = specific_occurrence_const
107
+ @source_data_criteria = source_data_criteria || id
108
+ end
109
+
110
+ # create a new data criteria given a category and sub_category. A sub category can either be a status or a sub category
111
+ def self.create_from_category(id, title, description, code_list_id, category, sub_category=nil, negation=false, negation_code_list_id=nil)
112
+ settings = HQMF::DataCriteria.get_settings_for_definition(category, sub_category)
113
+ HQMF::DataCriteria.new(id, title, nil, description, code_list_id, nil, nil, settings['definition'], settings['status'], nil, nil, nil, nil, negation, negation_code_list_id, nil, nil, nil,nil)
114
+ end
115
+
116
+ def standard_category
117
+ @settings['standard_category']
118
+ end
119
+ def qds_data_type
120
+ @settings['qds_data_type']
121
+ end
122
+ def type
123
+ @settings['category'].to_sym
124
+ end
125
+ def property
126
+ @settings['property'].to_sym unless @settings['property'].nil?
127
+ end
128
+ def patient_api_function
129
+ @settings['patient_api_function'].to_sym unless @settings['patient_api_function'].empty?
130
+ end
131
+ def hard_status
132
+ @settings['hard_status']
133
+ end
134
+
135
+ def definition=(definition)
136
+ @definition = definition
137
+ @settings = HQMF::DataCriteria.get_settings_for_definition(@definition, @status)
138
+ end
139
+ def status=(status)
140
+ @status = status
141
+ @settings = HQMF::DataCriteria.get_settings_for_definition(@definition, @status)
142
+ end
143
+
144
+ # Create a new data criteria instance from a JSON hash keyed with symbols
145
+ def self.from_json(id, json)
146
+
147
+ title = json["title"] if json["title"]
148
+ display_name = json["display_name"] if json["display_name"]
149
+ description = json["description"] if json["description"]
150
+ code_list_id = json["code_list_id"] if json["code_list_id"]
151
+ children_criteria = json["children_criteria"] if json["children_criteria"]
152
+ derivation_operator = json["derivation_operator"] if json["derivation_operator"]
153
+ definition = json["definition"] if json["definition"]
154
+ status = json["status"] if json["status"]
155
+ value = convert_value(json["value"]) if json["value"]
156
+ field_values = json["field_values"].inject({}){|memo,(k,v)| memo[k.to_s] = convert_value(v); memo} if json["field_values"]
157
+ effective_time = HQMF::Range.from_json(json["effective_time"]) if json["effective_time"]
158
+ inline_code_list = json["inline_code_list"].inject({}){|memo,(k,v)| memo[k.to_s] = v; memo} if json["inline_code_list"]
159
+ negation = json["negation"] || false
160
+ negation_code_list_id = json['negation_code_list_id'] if json['negation_code_list_id']
161
+ temporal_references = json["temporal_references"].map {|reference| HQMF::TemporalReference.from_json(reference)} if json["temporal_references"]
162
+ subset_operators = json["subset_operators"].map {|operator| HQMF::SubsetOperator.from_json(operator)} if json["subset_operators"]
163
+ specific_occurrence = json['specific_occurrence'] if json['specific_occurrence']
164
+ specific_occurrence_const = json['specific_occurrence_const'] if json['specific_occurrence_const']
165
+ source_data_criteria = json['source_data_criteria'] if json['source_data_criteria']
166
+
167
+ HQMF::DataCriteria.new(id, title, display_name, description, code_list_id, children_criteria, derivation_operator, definition, status, value, field_values,
168
+ effective_time, inline_code_list, negation, negation_code_list_id, temporal_references, subset_operators,specific_occurrence,specific_occurrence_const,source_data_criteria)
169
+ end
170
+
171
+ def to_json
172
+ json = base_json
173
+ {self.id.to_s.to_sym => json}
174
+ end
175
+
176
+ def base_json
177
+ x = nil
178
+ json = build_hash(self, [:title,:display_name,:description,:standard_category,:qds_data_type,:code_list_id,:children_criteria, :derivation_operator, :property, :type, :definition, :status, :hard_status, :negation, :negation_code_list_id,:specific_occurrence,:specific_occurrence_const,:source_data_criteria])
179
+ json[:children_criteria] = @children_criteria unless @children_criteria.nil? || @children_criteria.empty?
180
+ json[:value] = ((@value.is_a? String) ? @value : @value.to_json) if @value
181
+ json[:field_values] = @field_values.inject({}) {|memo,(k,v)| memo[k] = (!v.nil? ? v.to_json : nil); memo} if @field_values
182
+ json[:effective_time] = @effective_time.to_json if @effective_time
183
+ json[:inline_code_list] = @inline_code_list if @inline_code_list
184
+ json[:temporal_references] = x if x = json_array(@temporal_references)
185
+ json[:subset_operators] = x if x = json_array(@subset_operators)
186
+ json
187
+ end
188
+
189
+ def has_temporal(temporal_reference)
190
+ @temporal_references.reduce(false) {|found, item| found ||= item == temporal_reference }
191
+ end
192
+ def has_subset(subset_operator)
193
+ @subset_operators.reduce(false) {|found, item| found ||= item == subset_operator }
194
+ end
195
+
196
+ def self.statuses_by_definition
197
+ settings_file = File.expand_path('../data_criteria.json', __FILE__)
198
+ settings_map = JSON.parse(File.read(settings_file))
199
+ all_defs = (settings_map.map {|key, value| {category: value['category'],definition:value['definition'],status:(value['status'].empty? ? nil : value['status']), sub_category: value['sub_category'],title:value['title']} unless value['not_supported']}).compact
200
+ by_categories = {}
201
+ all_defs.each do |definition|
202
+ by_categories[definition[:category]]||={}
203
+ status = definition[:status]
204
+ def_key = definition[:definition]
205
+ if status.nil? and definition[:sub_category] and !definition[:sub_category].empty?
206
+ status = definition[:sub_category]
207
+ def_key = def_key.gsub("_#{status}",'')
208
+ end
209
+ by_categories[definition[:category]][def_key]||={category:def_key,statuses:[]}
210
+ by_categories[definition[:category]][def_key][:statuses] << status unless status.nil?
211
+ end
212
+ status_by_category = {}
213
+ by_categories.each {|key, value| status_by_category[key] = value.values}
214
+ status_by_category.delete('derived')
215
+ status_by_category.delete('variable')
216
+ status_by_category.delete('measurement_period')
217
+ status_by_category.values.flatten
218
+ end
219
+
220
+ def referenced_data_criteria(document)
221
+ referenced = []
222
+ if (@children_criteria)
223
+ @children_criteria.each do |id|
224
+ dc = document.data_criteria(id)
225
+ referenced << id
226
+ referenced.concat(dc.referenced_data_criteria(document))
227
+ end
228
+ end
229
+ if (@temporal_references)
230
+ @temporal_references.each do |tr|
231
+ id = tr.reference.id
232
+ if (id != HQMF::Document::MEASURE_PERIOD_ID)
233
+ dc = document.data_criteria(id)
234
+ referenced << id
235
+ referenced.concat(dc.referenced_data_criteria(document))
236
+ end
237
+ end
238
+ end
239
+ referenced
240
+ end
241
+
242
+ def all_code_set_oids
243
+
244
+ # root oid
245
+ referenced_oids = [code_list_id]
246
+
247
+ # value oid
248
+ referenced_oids << value.code_list_id if value != nil and value.type == 'CD'
249
+
250
+ # negation oid
251
+ referenced_oids << negation_code_list_id if negation_code_list_id != nil
252
+
253
+ # field oids
254
+ if field_values != nil
255
+ referenced_oids.concat (field_values.map {|key,field| field.code_list_id if field != nil and field.type == 'CD'})
256
+ end
257
+
258
+ referenced_oids
259
+
260
+ end
261
+
262
+ def self.get_settings_for_definition(definition, status)
263
+ settings_file = File.expand_path('../data_criteria.json', __FILE__)
264
+ settings_map = JSON.parse(File.read(settings_file))
265
+ key = definition + ((status.nil? || status.empty?) ? '' : "_#{status}")
266
+ settings = settings_map[key]
267
+
268
+ raise "data criteria is not supported #{key}" if settings.nil? || settings["not_supported"]
269
+
270
+ settings
271
+ end
272
+
273
+ def self.definition_for_template_id(template_id)
274
+ get_template_id_map()[template_id]
275
+ end
276
+
277
+ def self.template_id_for_definition(definition, status, negation)
278
+ get_template_id_map().key({'definition' => definition, 'status' => status || '', 'negation' => negation})
279
+ end
280
+
281
+ def self.title_for_template_id(template_id)
282
+ value = get_template_id_map()[template_id]
283
+ if value
284
+ settings = self.get_settings_for_definition(value['definition'], value['status'])
285
+ if settings
286
+ settings['title']
287
+ else
288
+ 'Unknown data criteria'
289
+ end
290
+ else
291
+ 'Unknown template id'
292
+ end
293
+ end
294
+
295
+ def self.get_template_id_map
296
+ @@template_id_map ||= read_template_id_map
297
+ @@template_id_map
298
+ end
299
+
300
+ private
301
+
302
+ def self.read_template_id_map
303
+ HealthDataStandards::Util::HQMFTemplateHelper.template_id_map
304
+ end
305
+
306
+ def normalize_status(definition, status)
307
+ return status if status.nil?
308
+ case status.downcase
309
+ when 'completed', 'complete'
310
+ case definition
311
+ when 'diagnosis'
312
+ 'active'
313
+ else
314
+ 'performed'
315
+ end
316
+ when 'order'
317
+ 'ordered'
318
+ else
319
+ status.downcase
320
+ end
321
+ end
322
+
323
+ def self.convert_value(json)
324
+ return nil unless json.present?
325
+ type = json["type"]
326
+ case type
327
+ when 'TS', 'PQ'
328
+ value = HQMF::Value.from_json(json)
329
+ when 'IVL_PQ'
330
+ value = HQMF::Range.from_json(json)
331
+ when 'CD'
332
+ value = HQMF::Coded.from_json(json)
333
+ when 'ANYNonNull'
334
+ value = HQMF::AnyValue.from_json(json)
335
+ else
336
+ raise "Unknown value type [#{type}]"
337
+ end
338
+ value
339
+ end
340
+
341
+
342
+ end
343
+
344
+ end
@@ -0,0 +1,178 @@
1
+ module HQMF
2
+ # Class representing an HQMF document
3
+ class Document
4
+
5
+ MEASURE_PERIOD_ID = "MeasurePeriod"
6
+
7
+ STRATIFIED_POPULATION_TEMPLATE_ID = '2.16.840.1.113883.3.100.1.2'
8
+ STRATIFIED_POPULATION_TEMPLATE_TITLE = 'Stratified'
9
+
10
+ include HQMF::Conversion::Utilities
11
+
12
+ attr_reader :id, :title, :description, :measure_period, :attributes, :populations, :source_data_criteria, :hqmf_id, :hqmf_set_id, :hqmf_version_number, :cms_id
13
+
14
+ # Create a new HQMF::Document which can be converted to JavaScript
15
+ # @param [String] id
16
+ # @param [String] hqmf_id
17
+ # @param [String] hqmf_set_id
18
+ # @param [String] hqmf_version_number
19
+ # @param [String] title
20
+ # @param [String] description
21
+ # @param [Array#PopulationCritera] population_criteria
22
+ # @param [Array#DataCriteria] data_criteria
23
+ # @param [Array#DataCriteria] source_data_criteria
24
+ # @param [Array#Attribute] attributes
25
+ # @param [Array#Hash] populations
26
+ # @param [Range] measure_period
27
+ def initialize(id, hqmf_id, hqmf_set_id, hqmf_version_number, cms_id, title, description, population_criteria, data_criteria, source_data_criteria, attributes, measure_period, populations=nil)
28
+ @id = id
29
+ @hqmf_id = hqmf_id
30
+ @hqmf_set_id = hqmf_set_id
31
+ @hqmf_version_number = hqmf_version_number
32
+ @cms_id = cms_id
33
+ @title = title
34
+ @description = description
35
+ @population_criteria = population_criteria
36
+ @data_criteria = data_criteria
37
+ @source_data_criteria = source_data_criteria
38
+ @attributes = attributes
39
+ @populations = populations || [
40
+ {
41
+ HQMF::PopulationCriteria::IPP => HQMF::PopulationCriteria::IPP,
42
+ HQMF::PopulationCriteria::DENOM => HQMF::PopulationCriteria::DENOM,
43
+ HQMF::PopulationCriteria::NUMER => HQMF::PopulationCriteria::NUMER,
44
+ HQMF::PopulationCriteria::DENEXCEP => HQMF::PopulationCriteria::DENEXCEP,
45
+ HQMF::PopulationCriteria::DENEX => HQMF::PopulationCriteria::DENEX
46
+ }
47
+ ]
48
+ @measure_period = measure_period
49
+ end
50
+
51
+ # Create a new HQMF::Document from a JSON hash keyed with symbols
52
+ def self.from_json(json)
53
+ id = json["id"]
54
+ hqmf_id = json["hqmf_id"]
55
+ hqmf_set_id = json["hqmf_set_id"]
56
+ hqmf_version_number = json["hqmf_version_number"]
57
+ title = json["title"]
58
+ description = json["description"]
59
+ cms_id = json["cms_id"]
60
+
61
+ population_criterias = []
62
+ json["population_criteria"].each do |key, population_criteria|
63
+ population_criterias << HQMF::PopulationCriteria.from_json(key.to_s, population_criteria)
64
+ end if json['population_criteria']
65
+
66
+ data_criterias = []
67
+ json["data_criteria"].each do |key, data_criteria|
68
+ data_criterias << HQMF::DataCriteria.from_json(key.to_s, data_criteria)
69
+ end
70
+
71
+ source_data_criterias = []
72
+ json["source_data_criteria"].each do |key, data_criteria|
73
+ source_data_criterias << HQMF::DataCriteria.from_json(key.to_s, data_criteria)
74
+ end
75
+
76
+ populations = json["populations"] if json["populations"]
77
+
78
+ attributes = json["attributes"].map {|attribute| HQMF::Attribute.from_json(attribute)} if json["attributes"]
79
+
80
+ measure_period = HQMF::Range.from_json(json["measure_period"]) if json["measure_period"]
81
+ HQMF::Document.new(id, hqmf_id, hqmf_set_id, hqmf_version_number, cms_id, title, description, population_criterias, data_criterias, source_data_criterias, attributes, measure_period,populations)
82
+ end
83
+
84
+ def to_json
85
+ json = build_hash(self, [:id, :hqmf_id, :hqmf_set_id, :hqmf_version_number, :title, :description, :cms_id])
86
+
87
+ json[:population_criteria] = {}
88
+ @population_criteria.each do |population|
89
+ json[:population_criteria].merge! population.to_json
90
+ end
91
+
92
+ json[:data_criteria] = {}
93
+ @data_criteria.each do |data|
94
+ json[:data_criteria].merge! data.to_json
95
+ end
96
+
97
+ json[:source_data_criteria] = {}
98
+ @source_data_criteria.each do |data|
99
+ json[:source_data_criteria].merge! data.to_json
100
+ end
101
+
102
+ x = nil
103
+ json[:attributes] = x if x = json_array(@attributes)
104
+
105
+ json[:populations] = @populations
106
+
107
+ json[:measure_period] = @measure_period.to_json
108
+
109
+ json
110
+ end
111
+
112
+
113
+ # Get all the population criteria defined by the measure
114
+ # @return [Array] an array of HQMF::PopulationCriteria
115
+ def all_population_criteria
116
+ @population_criteria
117
+ end
118
+
119
+ # Get a specific population criteria by id.
120
+ # @param [String] id the population identifier
121
+ # @return [HQMF::PopulationCriteria] the matching criteria, raises an Exception if not found
122
+ def population_criteria(id)
123
+ find(@population_criteria, :id, id)
124
+ end
125
+
126
+ # Get all the data criteria defined by the measure
127
+ # @return [Array] an array of HQMF::DataCriteria describing the data elements used by the measure
128
+ def all_data_criteria
129
+ @data_criteria
130
+ end
131
+
132
+ def all_code_set_oids
133
+ (@data_criteria.map {|d| d.all_code_set_oids }).flatten.compact.uniq
134
+ end
135
+
136
+ # Get the source data criteria that are specific occurrences
137
+ # @return [Array] an array of HQMF::DataCriteria describing the data elements used by the measure that are specific occurrences
138
+ def specific_occurrence_source_data_criteria
139
+ return [] if @source_data_criteria.nil?
140
+ @source_data_criteria.select {|dc| !dc.specific_occurrence.nil?}
141
+ end
142
+
143
+
144
+ # @return [Array] an array of HQMF::DataCriteria ids that are actually used in the measure
145
+ def referenced_data_criteria
146
+ data_criteria_ids = []
147
+ @population_criteria.each do |population|
148
+ data_criteria_ids.concat(population.referenced_data_criteria)
149
+ end
150
+ references = []
151
+ data_criteria_ids.each do |id|
152
+ dc = data_criteria(id)
153
+ references << id
154
+ from_dc = dc.referenced_data_criteria(self)
155
+ references.concat(from_dc)
156
+ end
157
+ used_dc = []
158
+ references.uniq.each do |id|
159
+ used_dc << data_criteria(id)
160
+ end
161
+ used_dc
162
+ end
163
+
164
+
165
+ # Get a specific data criteria by id.
166
+ # @param [String] id the data criteria identifier
167
+ # @return [HQMF::DataCriteria] the matching data criteria, raises an Exception if not found
168
+ def data_criteria(id)
169
+ find(@data_criteria, :id, id)
170
+ end
171
+
172
+ private
173
+
174
+ def find(collection, attribute, value)
175
+ collection.find {|e| e.send(attribute)==value}
176
+ end
177
+ end
178
+ end