module QRDA module Cat1 class SectionImporter attr_accessor :check_for_usable, :status_xpath, :code_xpath def initialize(entry_finder) @entry_finder = entry_finder @code_xpath = "./cda:code" @entry_id_map = {} @check_for_usable = true @entry_class = QDM::DataElement end # Traverses an HL7 CDA document passed in and creates an Array of Entry # objects based on what it finds # @param [Nokogiri::XML::Document] doc It is expected that the root node of this document # will have the "cda" namespace registered to "urn:hl7-org:v3" # measure definition # @return [Array] will be a list of Entry objects def create_entries(doc, nrh = NarrativeReferenceHandler.new) entry_list = [] @entry_id_map = {} entry_elements = @entry_finder.entries(doc) entry_elements.each do |entry_element| entry = create_entry(entry_element, nrh) if @check_for_usable entry_list << entry if usable_entry?(entry) else entry_list << entry end end [entry_list, @entry_id_map] end def usable_entry?(entry) entry.dataElementCodes.present? end def create_entry(entry_element, _nrh = NarrativeReferenceHandler.new) entry = @entry_class.new entry_qrda_id = extract_id(entry_element, @id_xpath) # Create a hash to map all of entry.ids to the same QRDA ids. This will be used to merge QRDA entries # that represent the same event. @entry_id_map["#{entry_qrda_id.value}_#{entry_qrda_id.namingSystem}"] ||= [] @entry_id_map["#{entry_qrda_id.value}_#{entry_qrda_id.namingSystem}"] << entry.id entry.dataElementCodes = extract_codes(entry_element, @code_xpath) extract_dates(entry_element, entry) if @result_xpath entry.result = extract_result_values(entry_element) end extract_negation(entry_element, entry) entry end private def extract_id(parent_element, id_xpath) id_element = parent_element.at_xpath(id_xpath) return unless id_element # If an extension is not included, use the root as the value. Other wise use the extension value = id_element['extension'] || id_element['root'] identifier = QDM::Id.new(value: value, namingSystem: id_element['root']) identifier end def extract_codes(coded_element, code_xpath) code_list = [] code_elements = coded_element.xpath(code_xpath) code_elements.each do |code_element| code_list << code_if_present(code_element) translations = code_element.xpath('cda:translation') translations.each do |translation| code_list << code_if_present(translation) end end code_list.compact end def code_if_present(code_element) return unless code_element && code_element['codeSystem'] && code_element['code'] QDM::Code.new(code_element['code'], HQMF::Util::CodeSystemHelper.code_system_for(code_element['codeSystem'])) end def extract_dates(parent_element, entry) entry.authorDatetime = extract_time(parent_element, @author_datetime_xpath) if @author_datetime_xpath entry.relevantPeriod = extract_interval(parent_element, @relevant_period_xpath) if @relevant_period_xpath entry.prevalencePeriod = extract_interval(parent_element, @prevalence_period_xpath) if @prevalence_period_xpath end def extract_interval(parent_element, interval_xpath) if parent_element.at_xpath("#{interval_xpath}/@value") low_time = DateTime.parse(parent_element.at_xpath(interval_xpath)['value']) high_time = DateTime.parse(parent_element.at_xpath(interval_xpath)['value']) end if parent_element.at_xpath("#{interval_xpath}/cda:low") low_time = if parent_element.at_xpath("#{interval_xpath}/cda:low")['value'] DateTime.parse(parent_element.at_xpath("#{interval_xpath}/cda:low")['value']) end end if parent_element.at_xpath("#{interval_xpath}/cda:high") high_time = if parent_element.at_xpath("#{interval_xpath}/cda:high")['value'] DateTime.parse(parent_element.at_xpath("#{interval_xpath}/cda:high")['value']) end end if parent_element.at_xpath("#{interval_xpath}/cda:center") low_time = Time.parse(parent_element.at_xpath("#{interval_xpath}/cda:center")['value']) high_time = Time.parse(parent_element.at_xpath("#{interval_xpath}/cda:center")['value']) end QDM::Interval.new(low_time, high_time).shift_dates(0) end def extract_time(parent_element, datetime_xpath) DateTime.parse(parent_element.at_xpath(datetime_xpath)['value']) if parent_element.at_xpath("#{datetime_xpath}/@value") end def extract_result_values(parent_element) result = [] parent_element.xpath(@result_xpath).each do |elem| result << extract_result_value(elem) end result.size > 1 ? result : result.first end def extract_result_value(value_element) return unless value_element && !value_element['nullFlavor'] value = value_element['value'] if value.present? return value.strip.to_f if (value_element['unit'] == "1" || value_element['unit'].nil?) return QDM::Quantity.new(value.strip.to_f, value_element['unit']) elsif value_element['code'].present? return code_if_present(value_element) end end def extract_reason(parent_element) return unless @reason_xpath reason_element = parent_element.xpath(@reason_xpath) negation_indicator = parent_element['negationInd'] # Return and do not set reason attribute if the entry is negated return nil if negation_indicator.eql?('true') reason_element.blank? ? nil : code_if_present(reason_element.first) end def extract_negation(parent_element, entry) negation_element = parent_element.xpath("./cda:entryRelationship[@typeCode='RSON']/cda:observation[cda:templateId/@root='2.16.840.1.113883.10.20.24.3.88']/cda:value") negation_indicator = parent_element['negationInd'] # Return and do not set negationRationale attribute if the entry is not negated return unless negation_indicator.eql?('true') entry.negationRationale = code_if_present(negation_element.first) unless negation_element.blank? extract_negated_code(parent_element, entry) end def extract_negated_code(parent_element, entry) code_elements = parent_element.xpath(@code_xpath) code_elements.each do |code_element| if code_element['nullFlavor'] == 'NA' && code_element['sdtc:valueSet'] entry.dataElementCodes = [{ code: code_element['sdtc:valueSet'], codeSystem: 'NA_VALUESET' }] end end end def extract_scalar(parent_element, scalar_xpath) scalar_element = parent_element.at_xpath(scalar_xpath) return unless scalar_element QDM::Quantity.new(scalar_element['value'].to_f, scalar_element['unit']) end def extract_components(parent_element) component_elements = parent_element.xpath(@components_xpath) components = [] component_elements&.each do |component_element| component = QDM::Component.new component.code = code_if_present(component_element.at_xpath('./cda:code')) component.result = extract_result_value(component_element.at_xpath('./cda:value')) components << component end components end def extract_facility(parent_element, entry) facility_element = parent_element.at_xpath(@facility_xpath) return unless facility_element facility = QDM::FacilityLocation.new participant_element = facility_element.at_xpath("./cda:participantRole[@classCode='SDLOC']/cda:code") facility.code = code_if_present(participant_element) facility.locationPeriod = extract_interval(facility_element, './cda:time') entry.facilityLocations = [facility] end def extract_related_to(parent_element) related_to_elements = parent_element.xpath(@related_to_xpath) related_ids = [] related_to_elements.each do |related_to_element| related_ids << extract_id(related_to_element, './sdtc:id') end related_ids end end end end