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}