home *** CD-ROM | disk | FTP | other *** search
/ PC World Komputer 1995 November / PCWK1195.iso / inne / win95 / sieciowe / hotja32.lzh / hotjava / classsrc / net / telnetinputstream.java < prev    next >
Text File  |  1995-08-11  |  5KB  |  148 lines

  1. /*
  2.  * @(#)TelnetInputStream.java    1.11 95/05/10 Jonathan Payne
  3.  *
  4.  * Copyright (c) 1994 Sun Microsystems, Inc. All Rights Reserved.
  5.  *
  6.  * Permission to use, copy, modify, and distribute this software
  7.  * and its documentation for NON-COMMERCIAL purposes and without
  8.  * fee is hereby granted provided that this copyright notice
  9.  * appears in all copies. Please refer to the file "copyright.html"
  10.  * for further important copyright and licensing information.
  11.  *
  12.  * SUN MAKES NO REPRESENTATIONS OR WARRANTIES ABOUT THE SUITABILITY OF
  13.  * THE SOFTWARE, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
  14.  * TO THE IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
  15.  * PARTICULAR PURPOSE, OR NON-INFRINGEMENT. SUN SHALL NOT BE LIABLE FOR
  16.  * ANY DAMAGES SUFFERED BY LICENSEE AS A RESULT OF USING, MODIFYING OR
  17.  * DISTRIBUTING THIS SOFTWARE OR ITS DERIVATIVES.
  18.  */
  19.  
  20. package net;
  21.  
  22. import java.io.*;
  23.  
  24. /**
  25.  * This class provides input and output streams for telnet clients.
  26.  * This class overrides read to do CRLF processing as specified in
  27.  * RFC 854. The class assumes it is running on a system where lines
  28.  * are terminated with a single newline <LF> character.
  29.  *
  30.  * This is the relevant section of RFC 824 regarding CRLF processing:
  31.  *
  32.  * <pre>
  33.  * The sequence "CR LF", as defined, will cause the NVT to be
  34.  * positioned at the left margin of the next print line (as would,
  35.  * for example, the sequence "LF CR").  However, many systems and
  36.  * terminals do not treat CR and LF independently, and will have to
  37.  * go to some effort to simulate their effect.  (For example, some
  38.  * terminals do not have a CR independent of the LF, but on such
  39.  * terminals it may be possible to simulate a CR by backspacing.)
  40.  * Therefore, the sequence "CR LF" must be treated as a single "new
  41.  * line" character and used whenever their combined action is
  42.  * intended; the sequence "CR NUL" must be used where a carriage
  43.  * return alone is actually desired; and the CR character must be
  44.  * avoided in other contexts.  This rule gives assurance to systems
  45.  * which must decide whether to perform a "new line" function or a
  46.  * multiple-backspace that the TELNET stream contains a character
  47.  * following a CR that will allow a rational decision.
  48.  *
  49.  *    Note that "CR LF" or "CR NUL" is required in both directions
  50.  *    (in the default ASCII mode), to preserve the symmetry of the
  51.  *    NVT model.  Even though it may be known in some situations
  52.  *    (e.g., with remote echo and suppress go ahead options in
  53.  *    effect) that characters are not being sent to an actual
  54.  *    printer, nonetheless, for the sake of consistency, the protocol
  55.  *    requires that a NUL be inserted following a CR not followed by
  56.  *    a LF in the data stream.  The converse of this is that a NUL
  57.  *    received in the data stream after a CR (in the absence of
  58.  *    options negotiations which explicitly specify otherwise) should
  59.  *    be stripped out prior to applying the NVT to local character
  60.  *    set mapping.
  61.  * </pre>
  62.  *
  63.  * @version 1.11, 10 May 1995
  64.  * @author    Jonathan Payne
  65.  */
  66.  
  67. public class TelnetInputStream extends FilterInputStream {
  68.     /** If stickyCRLF is true, then we're a machine, like an IBM PC,
  69.     where a Newline is a CR followed by LF.  On UNIX, this is false
  70.     because Newline is represented with just a LF character. */
  71.     boolean        stickyCRLF = false;
  72.     boolean        seenCR = false;
  73.  
  74.     public boolean  binaryMode = false;
  75.  
  76.     public TelnetInputStream(InputStream fd, boolean binary) {
  77.     super(fd);
  78.     binaryMode = binary;
  79.     }
  80.  
  81.     public void setStickyCRLF(boolean on) {
  82.     stickyCRLF = on;
  83.     }
  84.  
  85.     public int read() {
  86.     if (binaryMode)
  87.         return super.read();
  88.  
  89.     int c;
  90.  
  91.     /* If last time we determined we saw a CRLF pair, and we're
  92.        not turning that into just a Newline (that is, we're
  93.        stickyCRLF), then return the LF part of that sticky
  94.        pair now. */
  95.  
  96.     if (seenCR) {
  97.         seenCR = false;
  98.         return '\n';
  99.     }
  100.  
  101.     if ((c = super.read()) == '\r') {    /* CR */
  102.         switch (c = super.read()) {
  103.         default:
  104.         case -1:                /* this is an error */
  105.         throw new TelnetProtocolException("misplaced CR in input");
  106.  
  107.         case 0:                /* NUL - treat CR as CR */
  108.         return '\r';
  109.  
  110.         case '\n':                /* CRLF - treat as NL */
  111.         if (stickyCRLF) {
  112.             seenCR = true;
  113.             return '\r';
  114.         } else {
  115.             return '\n';
  116.         }
  117.         }
  118.     }
  119.     return c;
  120.     }
  121.  
  122.     /** read into a byte array */
  123.     public int read(byte bytes[]) {
  124.     return read(bytes, 0, bytes.length);
  125.     }
  126.  
  127.     /** 
  128.      * Read into a byte array at offset <i>off</i> for length <i>length</i>
  129.      * bytes.
  130.      */
  131.     public int read(byte bytes[], int off, int length) {
  132.     if (binaryMode)
  133.         return super.read(bytes, off, length);
  134.  
  135.     int c;
  136.     int offStart = off;
  137.  
  138.     while (--length >= 0) {
  139.         c = read();
  140.         if (c == -1)
  141.         break;
  142.         bytes[off++] = (byte)c;
  143.     }
  144.     return (off > offStart) ? off - offStart : -1;
  145.     }
  146. }
  147.  
  148.