
001package org.hl7.fhir.dstu3.elementmodel; 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.BufferedInputStream; 035import java.io.IOException; 036import java.io.InputStream; 037import java.io.InputStreamReader; 038import java.io.OutputStream; 039 040import org.hl7.fhir.dstu3.context.IWorkerContext; 041import org.hl7.fhir.dstu3.formats.IParser.OutputStyle; 042import org.hl7.fhir.dstu3.model.StructureDefinition; 043import org.hl7.fhir.exceptions.DefinitionException; 044import org.hl7.fhir.exceptions.FHIRException; 045import org.hl7.fhir.exceptions.FHIRFormatError; 046 047/** 048 * This class provides special support for parsing v2 by the v2 logical model 049 * For the logical model, see the FHIRPath spec 050 * 051 * @author Grahame Grieve 052 * 053 */ 054@Deprecated 055public class VerticalBarParser extends ParserBase { 056 057 /** 058 * Delimiters for a message. Note that the application rarely needs to concern 059 * itself with this information; it mainly exists for internal use. However if 060 * a message is being written to a spec that calls for non-standard delimiters, 061 * the application can set them here. 062 * 063 * @author Grahame 064 * 065 */ 066 public class Delimiters { 067 068 /** 069 * Hl7 defined default delimiter for a field 070 */ 071 public final static char DEFAULT_DELIMITER_FIELD = '|'; 072 073 /** 074 * Hl7 defined default delimiter for a component 075 */ 076 public final static char DEFAULT_DELIMITER_COMPONENT = '^'; 077 078 /** 079 * Hl7 defined default delimiter for a subcomponent 080 */ 081 public final static char DEFAULT_DELIMITER_SUBCOMPONENT = '&'; 082 083 /** 084 * Hl7 defined default delimiter for a repeat 085 */ 086 public final static char DEFAULT_DELIMITER_REPETITION = '~'; 087 088 /** 089 * Hl7 defined default delimiter for an escape 090 */ 091 public final static char DEFAULT_CHARACTER_ESCAPE = '\\'; 092 093 094 /** 095 * defined escape character for this message 096 */ 097 private char escapeCharacter; 098 099 /** 100 * defined repetition character for this message 101 */ 102 private char repetitionDelimiter; 103 104 /** 105 * defined field character for this message 106 */ 107 private char fieldDelimiter; 108 109 /** 110 * defined subComponent character for this message 111 */ 112 private char subComponentDelimiter; 113 114 /** 115 * defined component character for this message 116 */ 117 private char componentDelimiter; 118 119 /** 120 * create 121 * 122 */ 123 public Delimiters() { 124 super(); 125 reset(); 126 } 127 128 public boolean matches(Delimiters other) { 129 return escapeCharacter == other.escapeCharacter && 130 repetitionDelimiter == other.repetitionDelimiter && 131 fieldDelimiter == other.fieldDelimiter && 132 subComponentDelimiter == other.subComponentDelimiter && 133 componentDelimiter == other.componentDelimiter; 134 } 135 136 /** 137 * get defined component character for this message 138 * @return 139 */ 140 public char getComponentDelimiter() { 141 return componentDelimiter; 142 } 143 144 /** 145 * set defined component character for this message 146 * @param componentDelimiter 147 */ 148 public void setComponentDelimiter(char componentDelimiter) { 149 this.componentDelimiter = componentDelimiter; 150 } 151 152 /** 153 * get defined escape character for this message 154 * @return 155 */ 156 public char getEscapeCharacter() { 157 return escapeCharacter; 158 } 159 160 /** 161 * set defined escape character for this message 162 * @param escapeCharacter 163 */ 164 public void setEscapeCharacter(char escapeCharacter) { 165 this.escapeCharacter = escapeCharacter; 166 } 167 168 /** 169 * get defined field character for this message 170 * @return 171 */ 172 public char getFieldDelimiter() { 173 return fieldDelimiter; 174 } 175 176 /** 177 * set defined field character for this message 178 * @param fieldDelimiter 179 */ 180 public void setFieldDelimiter(char fieldDelimiter) { 181 this.fieldDelimiter = fieldDelimiter; 182 } 183 184 /** 185 * get repeat field character for this message 186 * @return 187 */ 188 public char getRepetitionDelimiter() { 189 return repetitionDelimiter; 190 } 191 192 /** 193 * set repeat field character for this message 194 * @param repetitionDelimiter 195 */ 196 public void setRepetitionDelimiter(char repetitionDelimiter) { 197 this.repetitionDelimiter = repetitionDelimiter; 198 } 199 200 /** 201 * get sub-component field character for this message 202 * @return 203 */ 204 public char getSubComponentDelimiter() { 205 return subComponentDelimiter; 206 } 207 208 /** 209 * set sub-component field character for this message 210 * @param subComponentDelimiter 211 */ 212 public void setSubComponentDelimiter(char subComponentDelimiter) { 213 this.subComponentDelimiter = subComponentDelimiter; 214 } 215 216 /** 217 * reset to default HL7 values 218 * 219 */ 220 public void reset () { 221 fieldDelimiter = DEFAULT_DELIMITER_FIELD; 222 componentDelimiter = DEFAULT_DELIMITER_COMPONENT; 223 subComponentDelimiter = DEFAULT_DELIMITER_SUBCOMPONENT; 224 repetitionDelimiter = DEFAULT_DELIMITER_REPETITION; 225 escapeCharacter = DEFAULT_CHARACTER_ESCAPE; 226 } 227 228 /** 229 * check that the delimiters are valid 230 * 231 * @throws FHIRException 232 */ 233 public void check() throws FHIRException { 234 rule(componentDelimiter != fieldDelimiter, "Delimiter Error: \""+componentDelimiter+"\" is used for both CPComponent and CPField"); 235 rule(subComponentDelimiter != fieldDelimiter, "Delimiter Error: \""+subComponentDelimiter+"\" is used for both CPSubComponent and CPField"); 236 rule(subComponentDelimiter != componentDelimiter, "Delimiter Error: \""+subComponentDelimiter+"\" is used for both CPSubComponent and CPComponent"); 237 rule(repetitionDelimiter != fieldDelimiter, "Delimiter Error: \""+repetitionDelimiter+"\" is used for both Repetition and CPField"); 238 rule(repetitionDelimiter != componentDelimiter, "Delimiter Error: \""+repetitionDelimiter+"\" is used for both Repetition and CPComponent"); 239 rule(repetitionDelimiter != subComponentDelimiter, "Delimiter Error: \""+repetitionDelimiter+"\" is used for both Repetition and CPSubComponent"); 240 rule(escapeCharacter != fieldDelimiter, "Delimiter Error: \""+escapeCharacter+"\" is used for both Escape and CPField"); 241 rule(escapeCharacter != componentDelimiter, "Delimiter Error: \""+escapeCharacter+"\" is used for both Escape and CPComponent"); 242 rule(escapeCharacter != subComponentDelimiter, "Delimiter Error: \""+escapeCharacter+"\" is used for both Escape and CPSubComponent"); 243 rule(escapeCharacter != repetitionDelimiter, "Delimiter Error: \""+escapeCharacter+"\" is used for both Escape and Repetition"); 244 } 245 246 /** 247 * check to see whether ch is a delimiter character (vertical bar parser support) 248 * @param ch 249 * @return 250 */ 251 public boolean isDelimiter(char ch) { 252 return ch == escapeCharacter || ch == repetitionDelimiter || ch == fieldDelimiter || ch == subComponentDelimiter || ch == componentDelimiter; 253 } 254 255 /** 256 * check to see whether ch is a cell delimiter char (vertical bar parser support) 257 * @param ch 258 * @return 259 */ 260 public boolean isCellDelimiter(char ch) { 261 return ch == repetitionDelimiter || ch == fieldDelimiter || ch == subComponentDelimiter || ch == componentDelimiter; 262 } 263 264 /** 265 * get the escape for a character 266 * @param ch 267 * @return 268 */ 269 public String getEscape(char ch) { 270 if (ch == escapeCharacter) 271 return escapeCharacter + "E" + escapeCharacter; 272 else if (ch == fieldDelimiter) 273 return escapeCharacter + "F" + escapeCharacter; 274 else if (ch == componentDelimiter) 275 return escapeCharacter + "S" + escapeCharacter; 276 else if (ch == subComponentDelimiter) 277 return escapeCharacter + "T" + escapeCharacter; 278 else if (ch == repetitionDelimiter) 279 return escapeCharacter + "R" + escapeCharacter; 280 else 281 return null; 282 } 283 284 /** 285 * build the MSH-2 content 286 * @return 287 */ 288 public String forMSH2() { 289 return "" + componentDelimiter + repetitionDelimiter + escapeCharacter + subComponentDelimiter; 290 } 291 292 /** 293 * check to see whether ch represents a delimiter escape 294 * @param ch 295 * @return 296 */ 297 public boolean isDelimiterEscape(char ch) { 298 return ch == 'F' || ch == 'S' || ch == 'E' || ch == 'T' || ch == 'R'; 299 } 300 301 /** 302 * get escape for ch in an escape 303 * @param ch 304 * @return 305 * @throws DefinitionException 306 * @throws FHIRException 307 */ 308 public char getDelimiterEscapeChar(char ch) throws DefinitionException { 309 if (ch == 'E') 310 return escapeCharacter; 311 else if (ch == 'F') 312 return fieldDelimiter; 313 else if (ch == 'S') 314 return componentDelimiter; 315 else if (ch == 'T') 316 return subComponentDelimiter; 317 else if (ch == 'R') 318 return repetitionDelimiter; 319 else 320 throw new DefinitionException("internal error in getDelimiterEscapeChar"); 321 } 322 } 323 324 public class VerticalBarParserReader { 325 326 327 private BufferedInputStream stream; 328 private String charsetName; 329 private InputStreamReader reader = null; 330 private boolean finished; 331 private char peeked; 332 private char lastValue; 333 private int offset; 334 private int lineNumber; 335 336 public VerticalBarParserReader(BufferedInputStream stream, String charsetName) throws FHIRException { 337 super(); 338 setStream(stream); 339 setCharsetName(charsetName); 340 open(); 341 } 342 343 public String getCharsetName() { 344 return charsetName; 345 } 346 347 public void setCharsetName(String charsetName) { 348 this.charsetName = charsetName; 349 } 350 351 public BufferedInputStream getStream() { 352 return stream; 353 } 354 355 public void setStream(BufferedInputStream stream) { 356 this.stream = stream; 357 } 358 359 private void open() throws FHIRException { 360 try { 361 stream.mark(2048); 362 reader = new InputStreamReader(stream, charsetName); 363 offset = 0; 364 lineNumber = 0; 365 lastValue = ' '; 366 next(); 367 } catch (Exception e) { 368 throw new FHIRException(e); 369 } 370 } 371 372 private void next() throws IOException, FHIRException { 373 finished = !reader.ready(); 374 if (!finished) { 375 char[] temp = new char[1]; 376 rule(reader.read(temp, 0, 1) == 1, "unable to read 1 character from the stream"); 377 peeked = temp[0]; 378 } 379 } 380 381 public String read(int charCount) throws FHIRException { 382 String value = ""; 383 for (int i = 0; i < charCount; i++) 384 value = value + read(); 385 return value; 386 } 387 388 public void skipEOL () throws FHIRException { 389 while (!finished && (peek() == '\r' || peek() == '\n')) 390 read(); 391 } 392 393 public char read () throws FHIRException { 394 rule(!finished, "No more content to read"); 395 char value = peek(); 396 offset++; 397 if (value == '\r' || value == '\n') { 398 if (lastValue != '\r' || value != '\n') 399 lineNumber++; 400 } 401 lastValue = value; 402 try { 403 next(); 404 } catch (Exception e) { 405 throw new FHIRException(e); 406 } 407 return value; 408 } 409 410 public boolean isFinished () { 411 return finished; 412 } 413 414 public char peek() throws FHIRException { 415 rule(!finished, "Cannot peek"); 416 return peeked; 417 } 418 419 public void mark() { 420 stream.mark(2048); 421 } 422 423 public void reset() throws FHIRException { 424 try { 425 stream.reset(); 426 } catch (IOException e) { 427 throw new FHIRException(e); 428 } 429 open(); 430 } 431 432 public boolean IsEOL() throws FHIRException { 433 return peek() == '\r' || peek() == '\n'; 434 } 435 436 public int getLineNumber() { 437 return lineNumber; 438 } 439 440 public int getOffset() { 441 return offset; 442 } 443 444 } 445 446 public VerticalBarParser(IWorkerContext context) { 447 super(context); 448 } 449 450 private String charset = "ASCII"; 451 private Delimiters delimiters = new Delimiters(); 452 453 @Override 454 public Element parse(InputStream stream) throws IOException, FHIRFormatError, DefinitionException, FHIRException { 455 StructureDefinition sd = context.fetchResource(StructureDefinition.class, "http://hl7.org/fhir/v2/StructureDefinition/Message"); 456 Element message = new Element("Message", new Property(context, sd.getSnapshot().getElementFirstRep(), sd)); 457 VerticalBarParserReader reader = new VerticalBarParserReader(new BufferedInputStream(stream), charset); 458 459 preDecode(reader); 460 while (!reader.isFinished()) // && (getOptions().getSegmentLimit() == 0 || getOptions().getSegmentLimit() > message.getSegments().size())) 461 readSegment(message, reader); 462 463 return message; 464 } 465 466 private void preDecode(VerticalBarParserReader reader) throws FHIRException { 467 reader.skipEOL(); 468 String temp = reader.read(3); 469 rule(temp.equals("MSH") || temp.equals("FHS"), "Found '" + temp + "' looking for 'MSH' or 'FHS'"); 470 readDelimiters(reader); 471 // readVersion(message); - probably don't need to do that? 472 // readCharacterSet(); 473 reader.reset(); // ready to read message now 474 } 475 476 private void rule(boolean test, String msg) throws FHIRException { 477 if (!test) 478 throw new FHIRException(msg); 479 } 480 481 private void readDelimiters(VerticalBarParserReader reader) throws FHIRException { 482 delimiters.setFieldDelimiter(reader.read()); 483 if (!(reader.peek() == delimiters.getFieldDelimiter())) 484 delimiters.setComponentDelimiter(reader.read()); 485 if (!(reader.peek() == delimiters.getFieldDelimiter())) 486 delimiters.setRepetitionDelimiter(reader.read()); 487 if (!(reader.peek() == delimiters.getFieldDelimiter())) 488 delimiters.setEscapeCharacter(reader.read()); 489 if (!(reader.peek() == delimiters.getFieldDelimiter())) 490 delimiters.setSubComponentDelimiter(reader.read()); 491 delimiters.check(); 492 } 493 494 private void readSegment(Element message, VerticalBarParserReader reader) throws FHIRException { 495 Element segment = new Element("segment", message.getProperty().getChild("segment")); 496 message.getChildren().add(segment); 497 Element segmentCode = new Element("code", segment.getProperty().getChild("code")); 498 segment.getChildren().add(segmentCode); 499 segmentCode.setValue(reader.read(3)); 500 501 int index = 0; 502 while (!reader.isFinished() && !reader.IsEOL()) { 503 index++; 504 readField(reader, segment, index); 505 if (!reader.isFinished() && !reader.IsEOL()) 506 rule(reader.read() == delimiters.getFieldDelimiter(), "Expected to find field delimiter"); 507 } 508 if (!reader.isFinished()) 509 reader.skipEOL(); 510 } 511 512 513 514 private void readField(VerticalBarParserReader reader, Element segment, int index) { 515 // TODO Auto-generated method stub 516 517 } 518 519 @Override 520 public void compose(Element e, OutputStream destination, OutputStyle style, String base) { 521 // TODO Auto-generated method stub 522 523 } 524 525}