
001package org.hl7.fhir.dstu3.terminologies; 002 003/* 004 Copyright (c) 2011+, HL7, Inc. 005 All rights reserved. 006 007 Redistribution and use in source and binary forms, with or without modification, 008 are permitted provided that the following conditions are met: 009 010 * Redistributions of source code must retain the above copyright notice, this 011 list of conditions and the following disclaimer. 012 * Redistributions in binary form must reproduce the above copyright notice, 013 this list of conditions and the following disclaimer in the documentation 014 and/or other materials provided with the distribution. 015 * Neither the name of HL7 nor the names of its contributors may be used to 016 endorse or promote products derived from this software without specific 017 prior written permission. 018 019 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND 020 ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 021 WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 022 IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, 023 INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 024 NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 025 PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, 026 WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 027 ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 028 POSSIBILITY OF SUCH DAMAGE. 029 030 */ 031 032 033 034import java.io.FileInputStream; 035import java.io.FileNotFoundException; 036import java.io.FileOutputStream; 037import java.io.IOException; 038 039import javax.xml.parsers.DocumentBuilder; 040import javax.xml.parsers.DocumentBuilderFactory; 041import javax.xml.parsers.ParserConfigurationException; 042 043import org.hl7.fhir.dstu3.formats.XmlParser; 044import org.hl7.fhir.dstu3.model.Bundle; 045import org.hl7.fhir.dstu3.model.Bundle.BundleEntryComponent; 046import org.hl7.fhir.dstu3.model.Bundle.BundleType; 047import org.hl7.fhir.dstu3.model.CodeableConcept; 048import org.hl7.fhir.dstu3.model.Coding; 049import org.hl7.fhir.dstu3.model.DataElement; 050import org.hl7.fhir.dstu3.model.DateTimeType; 051import org.hl7.fhir.dstu3.model.ElementDefinition; 052import org.hl7.fhir.dstu3.model.Enumerations.PublicationStatus; 053import org.hl7.fhir.dstu3.model.Identifier; 054import org.hl7.fhir.dstu3.model.InstantType; 055import org.hl7.fhir.dstu3.model.Meta; 056import org.hl7.fhir.dstu3.utils.ToolingExtensions; 057import org.hl7.fhir.exceptions.FHIRFormatError; 058import org.hl7.fhir.utilities.Utilities; 059import org.hl7.fhir.utilities.filesystem.ManagedFileAccess; 060import org.hl7.fhir.utilities.xml.XMLUtil; 061import org.w3c.dom.Document; 062import org.w3c.dom.Element; 063import org.xml.sax.SAXException; 064import org.xmlpull.v1.XmlPullParserException; 065 066/** 067 * This class converts the LOINC XML representation that the FHIR build tool uses internally to a set of DataElements in an atom feed 068 * 069 * @author Grahame 070 * 071 */ 072@Deprecated 073public class LoincToDEConvertor { 074 075 public static void main(String[] args) throws FHIRFormatError, IOException, XmlPullParserException, SAXException, ParserConfigurationException { 076 if (args.length == 0) { 077 System.out.println("FHIR LOINC to CDE convertor. "); 078 System.out.println(""); 079 System.out.println("This tool converts from LOINC to A set of DataElement definitions."); 080 System.out.println(""); 081 System.out.println("Usage: [jar(path?)] [dest] (-defn [definitions]) where: "); 082 System.out.println("* [dest] is a file name of the bundle to produce"); 083 System.out.println("* [definitions] is the file name of a file produced by exporting the main LOINC table from the mdb to XML"); 084 System.out.println(""); 085 } else { 086 LoincToDEConvertor exe = new LoincToDEConvertor(); 087 exe.setDest(args[0]); 088 for (int i = 1; i < args.length; i++) { 089 if (args[i].equals("-defn")) 090 exe.setDefinitions(args[i+1]); 091 } 092 exe.process(); 093 } 094 095 } 096 097 private String dest; 098 private String definitions; 099 public String getDest() { 100 return dest; 101 } 102 public void setDest(String dest) { 103 this.dest = dest; 104 } 105 public String getDefinitions() { 106 return definitions; 107 } 108 public void setDefinitions(String definitions) { 109 this.definitions = definitions; 110 } 111 112 private Document xml; 113 private Bundle bundle; 114 private DateTimeType now; 115 116 public Bundle process(String sourceFile) throws FileNotFoundException, SAXException, IOException, ParserConfigurationException { 117 this.definitions = sourceFile; 118 log("Begin. Produce Loinc CDEs in "+dest+" from "+definitions); 119 loadLoinc(); 120 log("LOINC loaded"); 121 122 now = DateTimeType.now(); 123 124 bundle = new Bundle(); 125 bundle.setType(BundleType.COLLECTION); 126 bundle.setId("http://hl7.org/fhir/commondataelement/loinc"); 127 bundle.setMeta(new Meta().setLastUpdatedElement(InstantType.now())); 128 129 processLoincCodes(); 130 return bundle; 131 } 132 133 public void process() throws FHIRFormatError, IOException, XmlPullParserException, SAXException, ParserConfigurationException { 134 log("Begin. Produce Loinc CDEs in "+dest+" from "+definitions); 135 loadLoinc(); 136 log("LOINC loaded"); 137 138 now = DateTimeType.now(); 139 140 bundle = new Bundle(); 141 bundle.setId("http://hl7.org/fhir/commondataelement/loinc"); 142 bundle.setMeta(new Meta().setLastUpdatedElement(InstantType.now())); 143 144 processLoincCodes(); 145 if (dest != null) { 146 log("Saving..."); 147 saveBundle(); 148 } 149 log("Done"); 150 151 } 152 153 private void log(String string) { 154 System.out.println(string); 155 156 } 157 private void loadLoinc() throws FileNotFoundException, SAXException, IOException, ParserConfigurationException { 158 DocumentBuilderFactory factory = XMLUtil.newXXEProtectedDocumentBuilderFactory(); 159 factory.setNamespaceAware(true); 160 DocumentBuilder builder = factory.newDocumentBuilder(); 161 162 xml = builder.parse(ManagedFileAccess.inStream(definitions)); 163 } 164 165 private void saveBundle() throws FHIRFormatError, IOException, XmlPullParserException { 166 XmlParser xml = new XmlParser(); 167 FileOutputStream s = ManagedFileAccess.outStream(dest); 168 xml.compose(s, bundle, true); 169 s.close(); 170 } 171 172 private String col(Element row, String name) { 173 Element e = XMLUtil.getNamedChild(row, name); 174 if (e == null) 175 return null; 176 String text = e.getTextContent(); 177 return text; 178 } 179 180 private boolean hasCol(Element row, String name) { 181 return Utilities.noString(col(row, name)); 182 } 183 184 private void processLoincCodes() { 185 Element row = XMLUtil.getFirstChild(xml.getDocumentElement()); 186 int i = 0; 187 while (row != null) { 188 i++; 189 if (i % 1000 == 0) 190 System.out.print("."); 191 String code = col(row, "LOINC_NUM"); 192 String comp = col(row, "COMPONENT"); 193 DataElement de = new DataElement(); 194 de.setId("loinc-"+code); 195 de.setMeta(new Meta().setLastUpdatedElement(InstantType.now())); 196 bundle.getEntry().add(new BundleEntryComponent().setResource(de)); 197 Identifier id = new Identifier(); 198 id.setSystem("http://hl7.org/fhir/commondataelement/loinc"); 199 id.setValue(code); 200 de.addIdentifier(id); 201 de.setPublisher("Regenstrief + FHIR Project Team"); 202 if (!col(row, "STATUS").equals("ACTIVE")) 203 de.setStatus(PublicationStatus.DRAFT); // till we get good at this 204 else 205 de.setStatus(PublicationStatus.RETIRED); 206 de.setDateElement(DateTimeType.now()); 207 de.setName(comp); 208 ElementDefinition dee = de.addElement(); 209 210 // PROPERTY ignore 211 // TIME_ASPCT 212 // SYSTEM 213 // SCALE_TYP 214 // METHOD_TYP 215 // dee.getCategory().add(new CodeableConcept().setText(col(row, "CLASS"))); 216 // SOURCE 217 // DATE_LAST_CHANGED - should be in ? 218 // CHNG_TYPE 219 dee.setComment(col(row , "COMMENTS")); 220 if (hasCol(row, "CONSUMER_NAME")) 221 dee.addAlias(col(row, "CONSUMER_NAME")); 222 // MOLAR_MASS 223 // CLASSTYPE 224 // FORMULA 225 // SPECIES 226 // EXMPL_ANSWERS 227 // ACSSYM 228 // BASE_NAME - ? this is a relationship 229 // NAACCR_ID 230 // ---------- CODE_TABLE todo 231 // SURVEY_QUEST_TEXT 232 // SURVEY_QUEST_SRC 233 if (hasCol(row, "RELATEDNAMES2")) { 234 String n = col(row, "RELATEDNAMES2"); 235 for (String s : n.split("\\;")) { 236 if (!Utilities.noString(s)) 237 dee.addAlias(s); 238 } 239 } 240 dee.addAlias(col(row, "SHORTNAME")); 241 // ORDER_OBS 242 // CDISC Code 243 // HL7_FIELD_SUBFIELD_ID 244 // ------------------ EXTERNAL_COPYRIGHT_NOTICE todo 245 dee.setDefinition(col(row, "LONG_COMMON_NAME")); 246 // HL7_V2_DATATYPE 247 String cc = makeType(col(row, "HL7_V3_DATATYPE"), code); 248 if (cc != null) 249 dee.addType().setCode(cc); 250 // todo... CURATED_RANGE_AND_UNITS 251 // todo: DOCUMENT_SECTION 252 // STATUS_REASON 253 // STATUS_TEXT 254 // CHANGE_REASON_PUBLIC 255 // COMMON_TEST_RANK 256 // COMMON_ORDER_RANK 257 // COMMON_SI_TEST_RANK 258 // HL7_ATTACHMENT_STRUCTURE 259 260 // units: 261 // UNITSREQUIRED 262 // SUBMITTED_UNITS 263 ToolingExtensions.setAllowableUnits(dee, makeUnits(col(row, "EXAMPLE_UNITS"), col(row, "EXAMPLE_UCUM_UNITS"))); 264 // EXAMPLE_SI_UCUM_UNITS 265 266 row = XMLUtil.getNextSibling(row); 267 } 268 System.out.println("done"); 269 } 270 271 private String makeType(String type, String id) { 272 if (Utilities.noString(type)) 273 return null; 274 if (type.equals("PQ")) 275 return "Quantity"; 276 else if (type.equals("ED")) 277 return "Attachment"; 278 else if (type.equals("TS")) 279 return "dateTime"; 280 else if (type.equals("ST")) 281 return "string"; 282 else if (type.equals("II")) 283 return "Identifier"; 284 else if (type.equals("CWE")) 285 return "CodeableConcept"; 286 else if (type.equals("CD") || type.equals("CO")) 287 return "CodeableConcept"; 288 else if (type.equals("PN")) 289 return "HumanName"; 290 else if (type.equals("EN")) 291 return "HumanName"; 292 else if (type.equals("AD")) 293 return "Address"; 294 else if (type.equals("BL")) 295 return "boolean"; 296 else if (type.equals("GTS")) 297 return "Schedule"; 298 else if (type.equals("INT")) 299 return "integer"; 300 else if (type.equals("CS")) 301 return "code"; 302 else if (type.equals("IVL_TS")) 303 return "Period"; 304 else if (type.equals("MMAT") || type.equals("PRF") || type.equals("TX") || type.equals("DT") || type.equals("FT")) 305 return null; 306 else 307 throw new Error("unmapped type "+type+" for LOINC code "+id); 308 } // 18606-4: MMAT. 18665-0: PRF. 18671-8: TX. 55400-6: DT; 8251-1: FT 309 310 private CodeableConcept makeUnits(String text, String ucum) { 311 if (Utilities.noString(text) && Utilities.noString(ucum)) 312 return null; 313 CodeableConcept cc = new CodeableConcept(); 314 cc.setText(text); 315 cc.getCoding().add(new Coding().setCode(ucum).setSystem("http://unitsofmeasure.org")); 316 return cc; 317 } 318 public Bundle getBundle() { 319 return bundle; 320 } 321}