Telnet.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.BufferedOutputStream;
  20. import java.io.IOException;
  21. import java.io.OutputStream;
  22. import java.time.Duration;
  23. import java.util.Arrays;

  24. import org.apache.commons.net.SocketClient;

  25. class Telnet extends SocketClient {
  26.     static final boolean debug = /* true; */ false;

  27.     static final boolean debugoptions = /* true; */ false;

  28.     static final byte[] COMMAND_DO = { (byte) TelnetCommand.IAC, (byte) TelnetCommand.DO };

  29.     static final byte[] COMMAND_DONT = { (byte) TelnetCommand.IAC, (byte) TelnetCommand.DONT };

  30.     static final byte[] COMMAND_WILL = { (byte) TelnetCommand.IAC, (byte) TelnetCommand.WILL };

  31.     static final byte[] COMMAND_WONT = { (byte) TelnetCommand.IAC, (byte) TelnetCommand.WONT };

  32.     static final byte[] COMMAND_SB = { (byte) TelnetCommand.IAC, (byte) TelnetCommand.SB };

  33.     static final byte[] COMMAND_SE = { (byte) TelnetCommand.IAC, (byte) TelnetCommand.SE };

  34.     static final int WILL_MASK = 0x01;
  35.     static final int DO_MASK = 0x02;
  36.     static final int REQUESTED_WILL_MASK = 0x04;
  37.     static final int REQUESTED_DO_MASK = 0x08;

  38.     /* public */
  39.     static final int DEFAULT_PORT = 23;

  40.     /* TERMINAL-TYPE option (start) */
  41.     /**
  42.      * Terminal type option
  43.      */
  44.     protected static final int TERMINAL_TYPE = 24;
  45.     /**
  46.      * Send (for subnegotiation)
  47.      */
  48.     protected static final int TERMINAL_TYPE_SEND = 1;
  49.     /**
  50.      * Is (for subnegotiation)
  51.      */
  52.     protected static final int TERMINAL_TYPE_IS = 0;

  53.     /**
  54.      * Is sequence (for subnegotiation)
  55.      */
  56.     static final byte[] COMMAND_IS = { (byte) TERMINAL_TYPE, (byte) TERMINAL_TYPE_IS };

  57.     /* Code Section added for supporting AYT (start) */
  58.     /**
  59.      * AYT sequence
  60.      */
  61.     static final byte[] COMMAND_AYT = { (byte) TelnetCommand.IAC, (byte) TelnetCommand.AYT };

  62.     private final int[] doResponse;

  63.     private final int[] willResponse;

  64.     private final int[] options;

  65.     /**
  66.      * Terminal type
  67.      */
  68.     private String terminalType;
  69.     /* TERMINAL-TYPE option (end) */

  70.     /* open TelnetOptionHandler functionality (end) */

  71.     /* open TelnetOptionHandler functionality (start) */
  72.     /**
  73.      * Array of option handlers
  74.      */
  75.     private final TelnetOptionHandler[] optionHandlers;

  76.     /**
  77.      * monitor to wait for AYT
  78.      */
  79.     private final Object aytMonitor = new Object();

  80.     /**
  81.      * flag for AYT
  82.      */
  83.     private volatile boolean aytFlag = true;
  84.     /* Code Section added for supporting AYT (end) */

  85.     /**
  86.      * The stream on which to spy
  87.      */
  88.     private volatile OutputStream spyStream;

  89.     /**
  90.      * The notification handler
  91.      */
  92.     private TelnetNotificationHandler notifhand;

  93.     /**
  94.      * Empty Constructor
  95.      */
  96.     Telnet() {
  97.         setDefaultPort(DEFAULT_PORT);
  98.         doResponse = new int[TelnetOption.MAX_OPTION_VALUE + 1];
  99.         willResponse = new int[TelnetOption.MAX_OPTION_VALUE + 1];
  100.         options = new int[TelnetOption.MAX_OPTION_VALUE + 1];
  101.         optionHandlers = new TelnetOptionHandler[TelnetOption.MAX_OPTION_VALUE + 1];
  102.     }

  103.     /* TERMINAL-TYPE option (start) */
  104.     /**
  105.      * This constructor lets you specify the terminal type.
  106.      *
  107.      * @param termtype - terminal type to be negotiated (ej. VT100)
  108.      */
  109.     Telnet(final String termtype) {
  110.         setDefaultPort(DEFAULT_PORT);
  111.         doResponse = new int[TelnetOption.MAX_OPTION_VALUE + 1];
  112.         willResponse = new int[TelnetOption.MAX_OPTION_VALUE + 1];
  113.         options = new int[TelnetOption.MAX_OPTION_VALUE + 1];
  114.         terminalType = termtype;
  115.         optionHandlers = new TelnetOptionHandler[TelnetOption.MAX_OPTION_VALUE + 1];
  116.     }
  117.     /* TERMINAL-TYPE option (end) */

  118.     /**
  119.      * Called upon connection.
  120.      *
  121.      * @throws IOException - Exception in I/O.
  122.      */
  123.     @Override
  124.     protected void _connectAction_() throws IOException {
  125.         /* (start). BUGFIX: clean the option info for each connection */
  126.         for (int ii = 0; ii < TelnetOption.MAX_OPTION_VALUE + 1; ii++) {
  127.             doResponse[ii] = 0;
  128.             willResponse[ii] = 0;
  129.             options[ii] = 0;
  130.             if (optionHandlers[ii] != null) {
  131.                 optionHandlers[ii].setDo(false);
  132.                 optionHandlers[ii].setWill(false);
  133.             }
  134.         }
  135.         /* (end). BUGFIX: clean the option info for each connection */

  136.         super._connectAction_();
  137.         _input_ = new BufferedInputStream(_input_);
  138.         _output_ = new BufferedOutputStream(_output_);

  139.         /* open TelnetOptionHandler functionality (start) */
  140.         for (int ii = 0; ii < TelnetOption.MAX_OPTION_VALUE + 1; ii++) {
  141.             if (optionHandlers[ii] != null) {
  142.                 if (optionHandlers[ii].getInitLocal()) {
  143.                     requestWill(optionHandlers[ii].getOptionCode());
  144.                 }

  145.                 if (optionHandlers[ii].getInitRemote()) {
  146.                     requestDo(optionHandlers[ii].getOptionCode());
  147.                 }
  148.             }
  149.         }
  150.         /* open TelnetOptionHandler functionality (end) */
  151.     }

  152.     /* Code Section added for supporting spystreams (start) */
  153.     /**
  154.      * Registers an OutputStream for spying what's going on in the Telnet session.
  155.      *
  156.      * @param spystream - OutputStream on which session activity will be echoed.
  157.      */
  158.     void _registerSpyStream(final OutputStream spystream) {
  159.         spyStream = spystream;
  160.     }

  161.     /* Code Section added for supporting AYT (start) */
  162.     /**
  163.      * Sends an {@code Are You There (AYT)} sequence and waits for the result.
  164.      *
  165.      * @param timeout - Time to wait for a response.
  166.      * @throws IOException              - Exception in I/O.
  167.      * @throws IllegalArgumentException - Illegal argument
  168.      * @throws InterruptedException     - Interrupted during wait.
  169.      * @return true if AYT received a response, false otherwise
  170.      **/
  171.     final boolean _sendAYT(final Duration timeout) throws IOException, IllegalArgumentException, InterruptedException {
  172.         boolean retValue = false;
  173.         synchronized (aytMonitor) {
  174.             synchronized (this) {
  175.                 aytFlag = false;
  176.                 _output_.write(COMMAND_AYT);
  177.                 _output_.flush();
  178.             }
  179.             aytMonitor.wait(timeout.toMillis());
  180.             if (!aytFlag) {
  181.                 aytFlag = true;
  182.             } else {
  183.                 retValue = true;
  184.             }
  185.         }

  186.         return retValue;
  187.     }
  188.     /* Code Section added for supporting AYT (end) */

  189.     /**
  190.      * Sends a command, automatically adds IAC prefix and flushes the output.
  191.      *
  192.      * @param cmd - command data to be sent
  193.      * @throws IOException - Exception in I/O.
  194.      * @since 3.0
  195.      */
  196.     final synchronized void _sendCommand(final byte cmd) throws IOException {
  197.         _output_.write(TelnetCommand.IAC);
  198.         _output_.write(cmd);
  199.         _output_.flush();
  200.     }

  201.     /* open TelnetOptionHandler functionality (start) */
  202.     /**
  203.      * Manages subnegotiation for Terminal Type.
  204.      *
  205.      * @param subn - subnegotiation data to be sent
  206.      * @throws IOException - Exception in I/O.
  207.      **/
  208.     final synchronized void _sendSubnegotiation(final int[] subn) throws IOException {
  209.         if (debug) {
  210.             System.err.println("SEND SUBNEGOTIATION: ");
  211.             if (subn != null) {
  212.                 System.err.println(Arrays.toString(subn));
  213.             }
  214.         }
  215.         if (subn != null) {
  216.             _output_.write(COMMAND_SB);
  217.             // Note _output_ is buffered, so might as well simplify by writing single bytes
  218.             for (final int element : subn) {
  219.                 final byte b = (byte) element;
  220.                 if (b == (byte) TelnetCommand.IAC) { // cast is necessary because IAC is outside the signed byte range
  221.                     _output_.write(b); // double any IAC bytes
  222.                 }
  223.                 _output_.write(b);
  224.             }
  225.             _output_.write(COMMAND_SE);

  226.             /* Code Section added for sending the negotiation ASAP (start) */
  227.             _output_.flush();
  228.             /* Code Section added for sending the negotiation ASAP (end) */
  229.         }
  230.     }
  231.     /* open TelnetOptionHandler functionality (end) */

  232.     /**
  233.      * Stops spying this Telnet.
  234.      */
  235.     void _stopSpyStream() {
  236.         spyStream = null;
  237.     }

  238.     /**
  239.      * Registers a new TelnetOptionHandler for this telnet to use.
  240.      *
  241.      * @param opthand - option handler to be registered.
  242.      * @throws InvalidTelnetOptionException - The option code is invalid.
  243.      * @throws IOException                  on error
  244.      **/
  245.     void addOptionHandler(final TelnetOptionHandler opthand) throws InvalidTelnetOptionException, IOException {
  246.         final int optcode = opthand.getOptionCode();
  247.         if (!TelnetOption.isValidOption(optcode)) {
  248.             throw new InvalidTelnetOptionException("Invalid Option Code", optcode);
  249.         }
  250.         if (optionHandlers[optcode] != null) {
  251.             throw new InvalidTelnetOptionException("Already registered option", optcode);
  252.         }
  253.         optionHandlers[optcode] = opthand;
  254.         if (isConnected()) {
  255.             if (opthand.getInitLocal()) {
  256.                 requestWill(optcode);
  257.             }

  258.             if (opthand.getInitRemote()) {
  259.                 requestDo(optcode);
  260.             }
  261.         }
  262.     }

  263.     /**
  264.      * Unregisters a TelnetOptionHandler.
  265.      *
  266.      * @param optcode - Code of the option to be unregistered.
  267.      * @throws InvalidTelnetOptionException - The option code is invalid.
  268.      * @throws IOException                  on error
  269.      **/
  270.     void deleteOptionHandler(final int optcode) throws InvalidTelnetOptionException, IOException {
  271.         if (!TelnetOption.isValidOption(optcode)) {
  272.             throw new InvalidTelnetOptionException("Invalid Option Code", optcode);
  273.         }
  274.         if (optionHandlers[optcode] == null) {
  275.             throw new InvalidTelnetOptionException("Unregistered option", optcode);
  276.         }
  277.         final TelnetOptionHandler opthand = optionHandlers[optcode];
  278.         optionHandlers[optcode] = null;

  279.         if (opthand.getWill()) {
  280.             requestWont(optcode);
  281.         }

  282.         if (opthand.getDo()) {
  283.             requestDont(optcode);
  284.         }
  285.     }
  286.     /* open TelnetOptionHandler functionality (end) */

  287.     /* Code Section added for supporting AYT (start) */
  288.     /**
  289.      * Processes the response of an AYT
  290.      */
  291.     final synchronized void processAYTResponse() {
  292.         if (!aytFlag) {
  293.             synchronized (aytMonitor) {
  294.                 aytFlag = true;
  295.                 aytMonitor.notifyAll();
  296.             }
  297.         }
  298.     }
  299.     /* Code Section added for supporting AYT (end) */

  300.     /**
  301.      * Processes a COMMAND.
  302.      *
  303.      * @param command - option code to be set.
  304.      **/
  305.     void processCommand(final int command) {
  306.         if (debugoptions) {
  307.             System.err.println("RECEIVED COMMAND: " + command);
  308.         }

  309.         if (notifhand != null) {
  310.             notifhand.receivedNegotiation(TelnetNotificationHandler.RECEIVED_COMMAND, command);
  311.         }
  312.     }

  313.     /**
  314.      * Processes a {@code DO} request.
  315.      *
  316.      * @param option - option code to be set.
  317.      * @throws IOException - Exception in I/O.
  318.      **/
  319.     void processDo(final int option) throws IOException {
  320.         if (debugoptions) {
  321.             System.err.println("RECEIVED DO: " + TelnetOption.getOption(option));
  322.         }

  323.         if (notifhand != null) {
  324.             notifhand.receivedNegotiation(TelnetNotificationHandler.RECEIVED_DO, option);
  325.         }

  326.         boolean acceptNewState = false;

  327.         /* open TelnetOptionHandler functionality (start) */
  328.         if (optionHandlers[option] != null) {
  329.             acceptNewState = optionHandlers[option].getAcceptLocal();
  330.         } else if (option == TERMINAL_TYPE && terminalType != null && !terminalType.isEmpty()) {
  331.             acceptNewState = true;
  332.         }
  333.         /* TERMINAL-TYPE option (end) */
  334.         /* open TelnetOptionHandler functionality (start) */

  335.         if (willResponse[option] > 0) {
  336.             --willResponse[option];
  337.             if (willResponse[option] > 0 && stateIsWill(option)) {
  338.                 --willResponse[option];
  339.             }
  340.         }

  341.         if (willResponse[option] == 0) {
  342.             if (requestedWont(option)) {

  343.                 switch (option) {

  344.                 default:
  345.                     break;

  346.                 }

  347.                 if (acceptNewState) {
  348.                     setWantWill(option);
  349.                     sendWill(option);
  350.                 } else {
  351.                     ++willResponse[option];
  352.                     sendWont(option);
  353.                 }
  354.             } else {
  355.                 // Other end has acknowledged option.

  356.                 switch (option) {

  357.                 default:
  358.                     break;

  359.                 }

  360.             }
  361.         }

  362.         setWill(option);
  363.     }

  364.     /**
  365.      * Processes a {@code DONT} request.
  366.      *
  367.      * @param option - option code to be set.
  368.      * @throws IOException - Exception in I/O.
  369.      **/
  370.     void processDont(final int option) throws IOException {
  371.         if (debugoptions) {
  372.             System.err.println("RECEIVED DONT: " + TelnetOption.getOption(option));
  373.         }
  374.         if (notifhand != null) {
  375.             notifhand.receivedNegotiation(TelnetNotificationHandler.RECEIVED_DONT, option);
  376.         }
  377.         if (willResponse[option] > 0) {
  378.             --willResponse[option];
  379.             if (willResponse[option] > 0 && stateIsWont(option)) {
  380.                 --willResponse[option];
  381.             }
  382.         }

  383.         if (willResponse[option] == 0 && requestedWill(option)) {

  384.             switch (option) {

  385.             default:
  386.                 break;

  387.             }

  388.             /* FIX for a BUG in the negotiation (start) */
  389.             if (stateIsWill(option) || requestedWill(option)) {
  390.                 sendWont(option);
  391.             }

  392.             setWantWont(option);
  393.             /* FIX for a BUG in the negotiation (end) */
  394.         }

  395.         setWont(option);
  396.     }

  397.     /* TERMINAL-TYPE option (start) */
  398.     /**
  399.      * Processes a suboption negotiation.
  400.      *
  401.      * @param suboption       - subnegotiation data received
  402.      * @param suboptionLength - length of data received
  403.      * @throws IOException - Exception in I/O.
  404.      **/
  405.     void processSuboption(final int[] suboption, final int suboptionLength) throws IOException {
  406.         if (debug) {
  407.             System.err.println("PROCESS SUBOPTION.");
  408.         }

  409.         /* open TelnetOptionHandler functionality (start) */
  410.         if (suboptionLength > 0) {
  411.             if (optionHandlers[suboption[0]] != null) {
  412.                 final int[] responseSuboption = optionHandlers[suboption[0]].answerSubnegotiation(suboption, suboptionLength);
  413.                 _sendSubnegotiation(responseSuboption);
  414.             } else if (suboptionLength > 1) {
  415.                 if (debug) {
  416.                     for (int ii = 0; ii < suboptionLength; ii++) {
  417.                         System.err.println("SUB[" + ii + "]: " + suboption[ii]);
  418.                     }
  419.                 }
  420.                 if (suboption[0] == TERMINAL_TYPE && suboption[1] == TERMINAL_TYPE_SEND) {
  421.                     sendTerminalType();
  422.                 }
  423.             }
  424.         }
  425.         /* open TelnetOptionHandler functionality (end) */
  426.     }

  427.     /**
  428.      * Processes a {@code WILL} request.
  429.      *
  430.      * @param option - option code to be set.
  431.      * @throws IOException - Exception in I/O.
  432.      **/
  433.     void processWill(final int option) throws IOException {
  434.         if (debugoptions) {
  435.             System.err.println("RECEIVED WILL: " + TelnetOption.getOption(option));
  436.         }

  437.         if (notifhand != null) {
  438.             notifhand.receivedNegotiation(TelnetNotificationHandler.RECEIVED_WILL, option);
  439.         }

  440.         boolean acceptNewState = false;

  441.         /* open TelnetOptionHandler functionality (start) */
  442.         if (optionHandlers[option] != null) {
  443.             acceptNewState = optionHandlers[option].getAcceptRemote();
  444.         }
  445.         /* open TelnetOptionHandler functionality (end) */

  446.         if (doResponse[option] > 0) {
  447.             --doResponse[option];
  448.             if (doResponse[option] > 0 && stateIsDo(option)) {
  449.                 --doResponse[option];
  450.             }
  451.         }

  452.         if (doResponse[option] == 0 && requestedDont(option)) {

  453.             switch (option) {

  454.             default:
  455.                 break;

  456.             }

  457.             if (acceptNewState) {
  458.                 setWantDo(option);
  459.                 sendDo(option);
  460.             } else {
  461.                 ++doResponse[option];
  462.                 sendDont(option);
  463.             }
  464.         }

  465.         setDo(option);
  466.     }

  467.     /**
  468.      * Processes a {@code WONT} request.
  469.      *
  470.      * @param option - option code to be set.
  471.      * @throws IOException - Exception in I/O.
  472.      **/
  473.     void processWont(final int option) throws IOException {
  474.         if (debugoptions) {
  475.             System.err.println("RECEIVED WONT: " + TelnetOption.getOption(option));
  476.         }

  477.         if (notifhand != null) {
  478.             notifhand.receivedNegotiation(TelnetNotificationHandler.RECEIVED_WONT, option);
  479.         }

  480.         if (doResponse[option] > 0) {
  481.             --doResponse[option];
  482.             if (doResponse[option] > 0 && stateIsDont(option)) {
  483.                 --doResponse[option];
  484.             }
  485.         }

  486.         if (doResponse[option] == 0 && requestedDo(option)) {

  487.             switch (option) {

  488.             default:
  489.                 break;

  490.             }

  491.             /* FIX for a BUG in the negotiation (start) */
  492.             if (stateIsDo(option) || requestedDo(option)) {
  493.                 sendDont(option);
  494.             }

  495.             setWantDont(option);
  496.             /* FIX for a BUG in the negotiation (end) */
  497.         }

  498.         setDont(option);
  499.     }

  500.     /**
  501.      * Registers a notification handler to which will be sent notifications of received telnet option negotiation commands.
  502.      *
  503.      * @param notifhand - TelnetNotificationHandler to be registered
  504.      */
  505.     public void registerNotifHandler(final TelnetNotificationHandler notifhand) {
  506.         this.notifhand = notifhand;
  507.     }

  508.     /**
  509.      * Requests a DO.
  510.      *
  511.      * @param option - Option code.
  512.      * @throws IOException - Exception in I/O.
  513.      **/
  514.     final synchronized void requestDo(final int option) throws IOException {
  515.         if (doResponse[option] == 0 && stateIsDo(option) || requestedDo(option)) {
  516.             return;
  517.         }
  518.         setWantDo(option);
  519.         ++doResponse[option];
  520.         sendDo(option);
  521.     }

  522.     /**
  523.      * Requests a {@code DONT}.
  524.      *
  525.      * @param option - Option code.
  526.      * @throws IOException - Exception in I/O.
  527.      **/
  528.     final synchronized void requestDont(final int option) throws IOException {
  529.         if (doResponse[option] == 0 && stateIsDont(option) || requestedDont(option)) {
  530.             return;
  531.         }
  532.         setWantDont(option);
  533.         ++doResponse[option];
  534.         sendDont(option);
  535.     }

  536.     /**
  537.      * Looks for the state of the option.
  538.      *
  539.      * @return returns true if a {@code DO} has been requested.
  540.      *
  541.      * @param option - option code to be looked up.
  542.      */
  543.     boolean requestedDo(final int option) {
  544.         return (options[option] & REQUESTED_DO_MASK) != 0;
  545.     }

  546.     /**
  547.      * Looks for the state of the option.
  548.      *
  549.      * @return returns true if a {@code DONT} has been requested
  550.      *
  551.      * @param option - option code to be looked up.
  552.      */
  553.     boolean requestedDont(final int option) {
  554.         return !requestedDo(option);
  555.     }

  556.     /**
  557.      * Looks for the state of the option.
  558.      *
  559.      * @return returns true if a {@code WILL} has been requested
  560.      *
  561.      * @param option - option code to be looked up.
  562.      */
  563.     boolean requestedWill(final int option) {
  564.         return (options[option] & REQUESTED_WILL_MASK) != 0;
  565.     }

  566.     /**
  567.      * Looks for the state of the option.
  568.      *
  569.      * @return returns true if a {@code WONT} has been requested
  570.      *
  571.      * @param option - option code to be looked up.
  572.      */
  573.     boolean requestedWont(final int option) {
  574.         return !requestedWill(option);
  575.     }

  576.     /**
  577.      * Requests a {@code WILL}.
  578.      *
  579.      * @param option - Option code.
  580.      * @throws IOException - Exception in I/O.
  581.      **/
  582.     final synchronized void requestWill(final int option) throws IOException {
  583.         if (willResponse[option] == 0 && stateIsWill(option) || requestedWill(option)) {
  584.             return;
  585.         }
  586.         setWantWill(option);
  587.         ++doResponse[option];
  588.         sendWill(option);
  589.     }

  590.     /* TERMINAL-TYPE option (end) */

  591.     /**
  592.      * Requests a {@code WONT}.
  593.      *
  594.      * @param option - Option code.
  595.      * @throws IOException - Exception in I/O.
  596.      **/
  597.     final synchronized void requestWont(final int option) throws IOException {
  598.         if (willResponse[option] == 0 && stateIsWont(option) || requestedWont(option)) {
  599.             return;
  600.         }
  601.         setWantWont(option);
  602.         ++doResponse[option];
  603.         sendWont(option);
  604.     }

  605.     /**
  606.      * Sends a byte.
  607.      *
  608.      * @param b - byte to send
  609.      * @throws IOException - Exception in I/O.
  610.      **/
  611.     final synchronized void sendByte(final int b) throws IOException {
  612.         _output_.write(b);

  613.         /* Code Section added for supporting spystreams (start) */
  614.         spyWrite(b);
  615.         /* Code Section added for supporting spystreams (end) */

  616.     }

  617.     /**
  618.      * Sends a {@code DO}.
  619.      *
  620.      * @param option - Option code.
  621.      * @throws IOException - Exception in I/O.
  622.      **/
  623.     final synchronized void sendDo(final int option) throws IOException {
  624.         if (debug || debugoptions) {
  625.             System.err.println("DO: " + TelnetOption.getOption(option));
  626.         }
  627.         _output_.write(COMMAND_DO);
  628.         _output_.write(option);

  629.         /* Code Section added for sending the negotiation ASAP (start) */
  630.         _output_.flush();
  631.         /* Code Section added for sending the negotiation ASAP (end) */
  632.     }

  633.     /**
  634.      * Sends a {@code DONT}.
  635.      *
  636.      * @param option - Option code.
  637.      * @throws IOException - Exception in I/O.
  638.      **/
  639.     final synchronized void sendDont(final int option) throws IOException {
  640.         if (debug || debugoptions) {
  641.             System.err.println("DONT: " + TelnetOption.getOption(option));
  642.         }
  643.         _output_.write(COMMAND_DONT);
  644.         _output_.write(option);

  645.         /* Code Section added for sending the negotiation ASAP (start) */
  646.         _output_.flush();
  647.         /* Code Section added for sending the negotiation ASAP (end) */
  648.     }

  649.     /**
  650.      * Sends terminal type information.
  651.      *
  652.      * @throws IOException - Exception in I/O.
  653.      */
  654.     final synchronized void sendTerminalType() throws IOException {
  655.         if (debug) {
  656.             System.err.println("SEND TERMINAL-TYPE: " + terminalType);
  657.         }
  658.         if (terminalType != null) {
  659.             _output_.write(COMMAND_SB);
  660.             _output_.write(COMMAND_IS);
  661.             _output_.write(terminalType.getBytes(getCharset()));
  662.             _output_.write(COMMAND_SE);
  663.             _output_.flush();
  664.         }
  665.     }

  666.     /**
  667.      * Sends a {@code WILL}.
  668.      *
  669.      * @param option - Option code.
  670.      * @throws IOException - Exception in I/O.
  671.      **/
  672.     final synchronized void sendWill(final int option) throws IOException {
  673.         if (debug || debugoptions) {
  674.             System.err.println("WILL: " + TelnetOption.getOption(option));
  675.         }
  676.         _output_.write(COMMAND_WILL);
  677.         _output_.write(option);

  678.         /* Code Section added for sending the negotiation ASAP (start) */
  679.         _output_.flush();
  680.         /* Code Section added for sending the negotiation ASAP (end) */
  681.     }

  682.     /**
  683.      * Sends a {@code WONT}.
  684.      *
  685.      * @param option - Option code.
  686.      * @throws IOException - Exception in I/O.
  687.      **/
  688.     final synchronized void sendWont(final int option) throws IOException {
  689.         if (debug || debugoptions) {
  690.             System.err.println("WONT: " + TelnetOption.getOption(option));
  691.         }
  692.         _output_.write(COMMAND_WONT);
  693.         _output_.write(option);

  694.         /* Code Section added for sending the negotiation ASAP (start) */
  695.         _output_.flush();
  696.         /* Code Section added for sending the negotiation ASAP (end) */
  697.     }

  698.     /**
  699.      * Sets the state of the option.
  700.      *
  701.      * @param option - option code to be set.
  702.      * @throws IOException
  703.      */
  704.     void setDo(final int option) throws IOException {
  705.         options[option] |= DO_MASK;

  706.         /* open TelnetOptionHandler functionality (start) */
  707.         if (requestedDo(option) && optionHandlers[option] != null) {
  708.             optionHandlers[option].setDo(true);

  709.             final int[] subneg = optionHandlers[option].startSubnegotiationRemote();

  710.             if (subneg != null) {
  711.                 _sendSubnegotiation(subneg);
  712.             }
  713.         }
  714.         /* open TelnetOptionHandler functionality (end) */
  715.     }

  716.     /**
  717.      * Sets the state of the option.
  718.      *
  719.      * @param option - option code to be set.
  720.      */
  721.     void setDont(final int option) {
  722.         options[option] &= ~DO_MASK;

  723.         /* open TelnetOptionHandler functionality (start) */
  724.         if (optionHandlers[option] != null) {
  725.             optionHandlers[option].setDo(false);
  726.         }
  727.         /* open TelnetOptionHandler functionality (end) */
  728.     }

  729.     /**
  730.      * Sets the state of the option.
  731.      *
  732.      * @param option - option code to be set.
  733.      */
  734.     void setWantDo(final int option) {
  735.         options[option] |= REQUESTED_DO_MASK;
  736.     }

  737.     /**
  738.      * Sets the state of the option.
  739.      *
  740.      * @param option - option code to be set.
  741.      */
  742.     void setWantDont(final int option) {
  743.         options[option] &= ~REQUESTED_DO_MASK;
  744.     }

  745.     /**
  746.      * Sets the state of the option.
  747.      *
  748.      * @param option - option code to be set.
  749.      */
  750.     void setWantWill(final int option) {
  751.         options[option] |= REQUESTED_WILL_MASK;
  752.     }

  753.     /**
  754.      * Sets the state of the option.
  755.      *
  756.      * @param option - option code to be set.
  757.      */
  758.     void setWantWont(final int option) {
  759.         options[option] &= ~REQUESTED_WILL_MASK;
  760.     }

  761.     /**
  762.      * Sets the state of the option.
  763.      *
  764.      * @param option - option code to be set.
  765.      * @throws IOException
  766.      */
  767.     void setWill(final int option) throws IOException {
  768.         options[option] |= WILL_MASK;

  769.         /* open TelnetOptionHandler functionality (start) */
  770.         if (requestedWill(option) && optionHandlers[option] != null) {
  771.             optionHandlers[option].setWill(true);

  772.             final int[] subneg = optionHandlers[option].startSubnegotiationLocal();

  773.             if (subneg != null) {
  774.                 _sendSubnegotiation(subneg);
  775.             }
  776.         }
  777.         /* open TelnetOptionHandler functionality (end) */
  778.     }

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

  780.     /**
  781.      * Sets the state of the option.
  782.      *
  783.      * @param option - option code to be set.
  784.      */
  785.     void setWont(final int option) {
  786.         options[option] &= ~WILL_MASK;

  787.         /* open TelnetOptionHandler functionality (start) */
  788.         if (optionHandlers[option] != null) {
  789.             optionHandlers[option].setWill(false);
  790.         }
  791.         /* open TelnetOptionHandler functionality (end) */
  792.     }

  793.     /**
  794.      * Sends a read char on the spy stream.
  795.      *
  796.      * @param ch - character read from the session
  797.      */
  798.     void spyRead(final int ch) {
  799.         final OutputStream spy = spyStream;
  800.         if (spy != null) {
  801.             try {
  802.                 if (ch != '\r') // never write '\r' on its own
  803.                 {
  804.                     if (ch == '\n') {
  805.                         spy.write('\r'); // add '\r' before '\n'
  806.                     }
  807.                     spy.write(ch); // write original character
  808.                     spy.flush();
  809.                 }
  810.             } catch (final IOException e) {
  811.                 spyStream = null;
  812.             }
  813.         }
  814.     }

  815.     /**
  816.      * Sends a written char on the spy stream.
  817.      *
  818.      * @param ch - character written to the session
  819.      */
  820.     void spyWrite(final int ch) {
  821.         if (!(stateIsDo(TelnetOption.ECHO) && requestedDo(TelnetOption.ECHO))) {
  822.             final OutputStream spy = spyStream;
  823.             if (spy != null) {
  824.                 try {
  825.                     spy.write(ch);
  826.                     spy.flush();
  827.                 } catch (final IOException e) {
  828.                     spyStream = null;
  829.                 }
  830.             }
  831.         }
  832.     }
  833.     /* Code Section added for supporting spystreams (end) */

  834.     /**
  835.      * Looks for the state of the option.
  836.      *
  837.      * @return returns true if a {@code DO} has been acknowledged.
  838.      *
  839.      * @param option - option code to be looked up.
  840.      */
  841.     boolean stateIsDo(final int option) {
  842.         return (options[option] & DO_MASK) != 0;
  843.     }

  844.     /**
  845.      * Looks for the state of the option.
  846.      *
  847.      * @return returns true if a {@code DONT} has been acknowledged
  848.      *
  849.      * @param option - option code to be looked up.
  850.      */
  851.     boolean stateIsDont(final int option) {
  852.         return !stateIsDo(option);
  853.     }

  854.     /**
  855.      * Looks for the state of the option.
  856.      *
  857.      * @return returns true if a {@code WILL} has been acknowledged
  858.      *
  859.      * @param option - option code to be looked up.
  860.      */
  861.     boolean stateIsWill(final int option) {
  862.         return (options[option] & WILL_MASK) != 0;
  863.     }

  864.     /**
  865.      * Looks for the state of the option.
  866.      *
  867.      * @return returns true if a {@code WONT} has been acknowledged
  868.      *
  869.      * @param option - option code to be looked up.
  870.      */
  871.     boolean stateIsWont(final int option) {
  872.         return !stateIsWill(option);
  873.     }

  874.     /**
  875.      * Unregisters the current notification handler.
  876.      */
  877.     public void unregisterNotifHandler() {
  878.         this.notifhand = null;
  879.     }
  880. }