TelnetClient.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */

  17. package org.apache.commons.net.telnet;

  18. import java.io.BufferedInputStream;
  19. import java.io.IOException;
  20. import java.io.InputStream;
  21. import java.io.OutputStream;
  22. import java.time.Duration;

  23. /**
  24.  * The TelnetClient class implements the simple network virtual terminal (NVT) for the Telnet protocol according to RFC 854. It does not implement any of the
  25.  * extra Telnet options because it is meant to be used within a Java program providing automated access to Telnet accessible resources.
  26.  * <p>
  27.  * The class can be used by first connecting to a server using the SocketClient {@link org.apache.commons.net.SocketClient#connect connect} method. Then an
  28.  * InputStream and OutputStream for sending and receiving data over the Telnet connection can be obtained by using the {@link #getInputStream getInputStream() }
  29.  * and {@link #getOutputStream getOutputStream() } methods. When you finish using the streams, you must call {@link #disconnect disconnect } rather than simply
  30.  * closing the streams.
  31.  * </p>
  32.  */
  33. public class TelnetClient extends Telnet {
  34.     private static final int DEFAULT_MAX_SUBNEGOTIATION_LENGTH = 512;

  35.     final int maxSubnegotiationLength;
  36.     private InputStream input;
  37.     private OutputStream output;
  38.     protected boolean readerThread = true;
  39.     private TelnetInputListener inputListener;

  40.     /**
  41.      * Default TelnetClient constructor, sets terminal-type {@code VT100}.
  42.      */
  43.     public TelnetClient() {
  44.         this("VT100", DEFAULT_MAX_SUBNEGOTIATION_LENGTH);
  45.     }

  46.     /**
  47.      * Constructs an instance with the specified max subnegotiation length and the default terminal-type {@code VT100}
  48.      *
  49.      * @param maxSubnegotiationLength the size of the subnegotiation buffer
  50.      */
  51.     public TelnetClient(final int maxSubnegotiationLength) {
  52.         this("VT100", maxSubnegotiationLength);
  53.     }

  54.     /**
  55.      * Constructs an instance with the specified terminal type.
  56.      *
  57.      * @param termtype the terminal type to use, e.g. {@code VT100}
  58.      */
  59.     public TelnetClient(final String termtype) {
  60.         this(termtype, DEFAULT_MAX_SUBNEGOTIATION_LENGTH);
  61.     }

  62.     /**
  63.      * Constructs an instance with the specified terminal type and max subnegotiation length
  64.      *
  65.      * @param termtype                the terminal type to use, e.g. {@code VT100}
  66.      * @param maxSubnegotiationLength the size of the subnegotiation buffer
  67.      */
  68.     public TelnetClient(final String termtype, final int maxSubnegotiationLength) {
  69.         /* TERMINAL-TYPE option (start) */
  70.         super(termtype);
  71.         /* TERMINAL-TYPE option (end) */
  72.         this.input = null;
  73.         this.output = null;
  74.         this.maxSubnegotiationLength = maxSubnegotiationLength;
  75.     }

  76.     /**
  77.      * Handles special connection requirements.
  78.      *
  79.      * @throws IOException If an error occurs during connection setup.
  80.      */
  81.     @Override
  82.     protected void _connectAction_() throws IOException {
  83.         super._connectAction_();
  84.         final TelnetInputStream tmp = new TelnetInputStream(_input_, this, readerThread);
  85.         if (readerThread) {
  86.             tmp.start();
  87.         }
  88.         // __input CANNOT refer to the TelnetInputStream. We run into
  89.         // blocking problems when some classes use TelnetInputStream, so
  90.         // we wrap it with a BufferedInputStream which we know is safe.
  91.         // This blocking behavior requires further investigation, but right
  92.         // now it looks like classes like InputStreamReader are not implemented
  93.         // in a safe manner.
  94.         input = new BufferedInputStream(tmp);
  95.         output = new TelnetOutputStream(this);
  96.     }

  97.     /**
  98.      * Registers a new TelnetOptionHandler for this telnet client to use.
  99.      *
  100.      * @param opthand - option handler to be registered.
  101.      *
  102.      * @throws InvalidTelnetOptionException on error
  103.      * @throws IOException                  on error
  104.      */
  105.     @Override
  106.     public void addOptionHandler(final TelnetOptionHandler opthand) throws InvalidTelnetOptionException, IOException {
  107.         super.addOptionHandler(opthand);
  108.     }
  109.     /* open TelnetOptionHandler functionality (end) */

  110.     void closeOutputStream() throws IOException {
  111.         if (_output_ == null) {
  112.             return;
  113.         }
  114.         try {
  115.             _output_.close();
  116.         } finally {
  117.             _output_ = null;
  118.         }
  119.     }

  120.     /**
  121.      * Unregisters a TelnetOptionHandler.
  122.      *
  123.      * @param optcode - Code of the option to be unregistered.
  124.      *
  125.      * @throws InvalidTelnetOptionException on error
  126.      * @throws IOException                  on error
  127.      */
  128.     @Override
  129.     public void deleteOptionHandler(final int optcode) throws InvalidTelnetOptionException, IOException {
  130.         super.deleteOptionHandler(optcode);
  131.     }

  132.     /**
  133.      * Disconnects the telnet session, closing the input and output streams as well as the socket. If you have references to the input and output streams of the
  134.      * telnet connection, you should not close them yourself, but rather call disconnect to properly close the connection.
  135.      */
  136.     @Override
  137.     public void disconnect() throws IOException {
  138.         try {
  139.             if (input != null) {
  140.                 input.close();
  141.             }
  142.             if (output != null) {
  143.                 output.close();
  144.             }
  145.         } finally { // NET-594
  146.             output = null;
  147.             input = null;
  148.             super.disconnect();
  149.         }
  150.     }

  151.     void flushOutputStream() throws IOException {
  152.         if (_output_ == null) {
  153.             throw new IOException("Stream closed");
  154.         }
  155.         _output_.flush();
  156.     }

  157.     /**
  158.      * Returns the telnet connection input stream. You should not close the stream when you finish with it. Rather, you should call {@link #disconnect
  159.      * disconnect }.
  160.      *
  161.      * @return The telnet connection input stream.
  162.      */
  163.     public InputStream getInputStream() {
  164.         return input;
  165.     }

  166.     /**
  167.      * Returns the state of the option on the local side.
  168.      *
  169.      * @param option - Option to be checked.
  170.      *
  171.      * @return The state of the option on the local side.
  172.      */
  173.     public boolean getLocalOptionState(final int option) {
  174.         /* BUG (option active when not already acknowledged) (start) */
  175.         return stateIsWill(option) && requestedWill(option);
  176.         /* BUG (option active when not already acknowledged) (end) */
  177.     }

  178.     /* Code Section added for supporting AYT (start) */

  179.     /**
  180.      * Returns the telnet connection output stream. You should not close the stream when you finish with it. Rather, you should call {@link #disconnect
  181.      * disconnect }.
  182.      *
  183.      * @return The telnet connection output stream.
  184.      */
  185.     public OutputStream getOutputStream() {
  186.         return output;
  187.     }

  188.     /**
  189.      * Gets the status of the reader thread.
  190.      *
  191.      * @return true if the reader thread is enabled, false otherwise
  192.      */
  193.     public boolean getReaderThread() {
  194.         return readerThread;
  195.     }

  196.     /**
  197.      * Returns the state of the option on the remote side.
  198.      *
  199.      * @param option - Option to be checked.
  200.      *
  201.      * @return The state of the option on the remote side.
  202.      */
  203.     public boolean getRemoteOptionState(final int option) {
  204.         /* BUG (option active when not already acknowledged) (start) */
  205.         return stateIsDo(option) && requestedDo(option);
  206.         /* BUG (option active when not already acknowledged) (end) */
  207.     }
  208.     /* open TelnetOptionHandler functionality (end) */

  209.     /* open TelnetOptionHandler functionality (start) */

  210.     // Notify input listener
  211.     void notifyInputListener() {
  212.         final TelnetInputListener listener;
  213.         synchronized (this) {
  214.             listener = this.inputListener;
  215.         }
  216.         if (listener != null) {
  217.             listener.telnetInputAvailable();
  218.         }
  219.     }

  220.     /**
  221.      * Register a listener to be notified when new incoming data is available to be read on the {@link #getInputStream input stream}. Only one listener is
  222.      * supported at a time.
  223.      *
  224.      * <p>
  225.      * More precisely, notifications are issued whenever the number of bytes available for immediate reading (i.e., the value returned by
  226.      * {@link InputStream#available}) transitions from zero to non-zero. Note that (in general) multiple reads may be required to empty the buffer and reset
  227.      * this notification, because incoming bytes are being added to the internal buffer asynchronously.
  228.      * </p>
  229.      *
  230.      * <p>
  231.      * Notifications are only supported when a {@link #setReaderThread reader thread} is enabled for the connection.
  232.      * </p>
  233.      *
  234.      * @param listener listener to be registered; replaces any previous
  235.      * @since 3.0
  236.      */
  237.     public synchronized void registerInputListener(final TelnetInputListener listener) {
  238.         this.inputListener = listener;
  239.     }

  240.     /**
  241.      * Registers a notification handler to which will be sent notifications of received telnet option negotiation commands.
  242.      *
  243.      * @param notifhand - TelnetNotificationHandler to be registered
  244.      */
  245.     @Override
  246.     public void registerNotifHandler(final TelnetNotificationHandler notifhand) {
  247.         super.registerNotifHandler(notifhand);
  248.     }

  249.     /* Code Section added for supporting spystreams (start) */
  250.     /**
  251.      * Registers an OutputStream for spying what's going on in the TelnetClient session.
  252.      *
  253.      * @param spystream - OutputStream on which session activity will be echoed.
  254.      */
  255.     public void registerSpyStream(final OutputStream spystream) {
  256.         super._registerSpyStream(spystream);
  257.     }

  258.     /**
  259.      * Sends an {@code Are You There (AYT)} sequence and waits for the result.
  260.      *
  261.      * @param timeout - Time to wait for a response.
  262.      *
  263.      * @return true if AYT received a response, false otherwise.
  264.      *
  265.      * @throws InterruptedException     on error
  266.      * @throws IllegalArgumentException on error
  267.      * @throws IOException              on error
  268.      * @since 3.10.0
  269.      */
  270.     public boolean sendAYT(final Duration timeout) throws IOException, IllegalArgumentException, InterruptedException {
  271.         return _sendAYT(timeout);
  272.     }

  273.     /**
  274.      * Sends an {@code Are You There (AYT)} sequence and waits for the result.
  275.      *
  276.      * @param timeout - Time to wait for a response (millis.)
  277.      *
  278.      * @return true if AYT received a response, false otherwise
  279.      *
  280.      * @throws InterruptedException     on error
  281.      * @throws IllegalArgumentException on error
  282.      * @throws IOException              on error
  283.      * @deprecated Use {@link #sendAYT(Duration)}.
  284.      */
  285.     @Deprecated
  286.     public boolean sendAYT(final long timeout) throws IOException, IllegalArgumentException, InterruptedException {
  287.         return _sendAYT(Duration.ofMillis(timeout));
  288.     }

  289.     /* Code Section added for supporting AYT (start) */

  290.     /**
  291.      * Sends a command byte to the remote peer, adding the IAC prefix.
  292.      *
  293.      * <p>
  294.      * This method does not wait for any response. Messages sent by the remote end can be handled by registering an approrpriate {@link TelnetOptionHandler}.
  295.      * </p>
  296.      *
  297.      * @param command the code for the command
  298.      * @throws IOException              if an I/O error occurs while writing the message
  299.      * @throws IllegalArgumentException on error
  300.      * @since 3.0
  301.      */
  302.     public void sendCommand(final byte command) throws IOException, IllegalArgumentException {
  303.         _sendCommand(command);
  304.     }

  305.     /**
  306.      * Sends a protocol-specific subnegotiation message to the remote peer. {@link TelnetClient} will add the IAC SB &amp; IAC SE framing bytes; the first byte
  307.      * in {@code message} should be the appropriate telnet option code.
  308.      *
  309.      * <p>
  310.      * This method does not wait for any response. Subnegotiation messages sent by the remote end can be handled by registering an approrpriate
  311.      * {@link TelnetOptionHandler}.
  312.      * </p>
  313.      *
  314.      * @param message option code followed by subnegotiation payload
  315.      * @throws IllegalArgumentException if {@code message} has length zero
  316.      * @throws IOException              if an I/O error occurs while writing the message
  317.      * @since 3.0
  318.      */
  319.     public void sendSubnegotiation(final int[] message) throws IOException, IllegalArgumentException {
  320.         if (message.length < 1) {
  321.             throw new IllegalArgumentException("zero length message");
  322.         }
  323.         _sendSubnegotiation(message);
  324.     }

  325.     /**
  326.      * Sets the status of the reader thread.
  327.      *
  328.      * <p>
  329.      * When enabled, a seaparate internal reader thread is created for new connections to read incoming data as it arrives. This results in immediate handling
  330.      * of option negotiation, notifications, etc. (at least until the fixed-size internal buffer fills up). Otherwise, no thread is created an all negotiation
  331.      * and option handling is deferred until a read() is performed on the {@link #getInputStream input stream}.
  332.      * </p>
  333.      *
  334.      * <p>
  335.      * The reader thread must be enabled for {@link TelnetInputListener} support.
  336.      * </p>
  337.      *
  338.      * <p>
  339.      * When this method is invoked, the reader thread status will apply to all subsequent connections; the current connection (if any) is not affected.
  340.      * </p>
  341.      *
  342.      * @param flag true to enable the reader thread, false to disable
  343.      * @see #registerInputListener
  344.      */
  345.     public void setReaderThread(final boolean flag) {
  346.         readerThread = flag;
  347.     }

  348.     /**
  349.      * Stops spying this TelnetClient.
  350.      */
  351.     public void stopSpyStream() {
  352.         super._stopSpyStream();
  353.     }
  354.     /* Code Section added for supporting spystreams (end) */

  355.     /**
  356.      * Unregisters the current {@link TelnetInputListener}, if any.
  357.      *
  358.      * @since 3.0
  359.      */
  360.     public synchronized void unregisterInputListener() {
  361.         this.inputListener = null;
  362.     }

  363.     /**
  364.      * Unregisters the current notification handler.
  365.      */
  366.     @Override
  367.     public void unregisterNotifHandler() {
  368.         super.unregisterNotifHandler();
  369.     }
  370. }