Webster/
Webster/Net/
package Net::Telnet;
require 5.002;

## User documentation in POD format is at end of this file.  Search for =head

use strict;

## Module export.
use vars qw(@EXPORT_OK);
@EXPORT_OK = qw(TELNET_IAC TELNET_DONT TELNET_DO TELNET_WONT TELNET_WILL
		TELNET_SB TELNET_GA TELNET_EL TELNET_EC TELNET_AYT TELNET_AO
		TELNET_IP TELNET_BREAK TELNET_DM TELNET_NOP TELNET_SE
		TELNET_EOR TELNET_ABORT TELNET_SUSP TELNET_EOF TELNET_SYNCH
		TELOPT_BINARY TELOPT_ECHO TELOPT_RCP TELOPT_SGA TELOPT_NAMS
		TELOPT_STATUS TELOPT_TM TELOPT_RCTE TELOPT_NAOL TELOPT_NAOP
		TELOPT_NAOCRD TELOPT_NAOHTS TELOPT_NAOHTD TELOPT_NAOFFD
		TELOPT_NAOVTS TELOPT_NAOVTD TELOPT_NAOLFD TELOPT_XASCII
		TELOPT_LOGOUT TELOPT_BM TELOPT_DET TELOPT_SUPDUP
		TELOPT_SUPDUPOUTPUT TELOPT_SNDLOC TELOPT_TTYPE TELOPT_EOR
		TELOPT_TUID TELOPT_OUTMRK TELOPT_TTYLOC TELOPT_3270REGIME
		TELOPT_X3PAD TELOPT_NAWS TELOPT_TSPEED TELOPT_LFLOW
		TELOPT_LINEMODE TELOPT_XDISPLOC TELOPT_OLD_ENVIRON
		TELOPT_AUTHENTICATION TELOPT_ENCRYPT TELOPT_NEW_ENVIRON
		TELOPT_EXOPL);

## Module import.
use Exporter ();
use Socket qw(AF_INET SOCK_STREAM inet_aton sockaddr_in);
use Symbol qw(qualify);

## Base class.
use vars qw(@ISA);
@ISA = qw(Exporter);
if (eval 'require IO::Socket') {
    push @ISA, 'IO::Socket::INET';
}
else {
    require FileHandle;
    push @ISA, 'FileHandle';
}

## Global variables.
use vars qw($Default_blocksize $VERSION @Telopts);
$Default_blocksize = 8192;
$VERSION = "3.01";
@Telopts = ("BINARY", "ECHO", "RCP", "SUPPRESS GO AHEAD", "NAME", "STATUS",
	    "TIMING MARK", "RCTE", "NAOL", "NAOP", "NAOCRD", "NAOHTS",
	    "NAOHTD", "NAOFFD", "NAOVTS", "NAOVTD", "NAOLFD", "EXTEND ASCII",
	    "LOGOUT", "BYTE MACRO", "DATA ENTRY TERMINAL", "SUPDUP",
	    "SUPDUP OUTPUT", "SEND LOCATION", "TERMINAL TYPE", "END OF RECORD",
	    "TACACS UID", "OUTPUT MARKING", "TTYLOC", "3270 REGIME", "X.3 PAD",
	    "NAWS", "TSPEED", "LFLOW", "LINEMODE", "XDISPLOC", "OLD-ENVIRON",
	    "AUTHENTICATION", "ENCRYPT", "NEW-ENVIRON");


########################### Public Methods ###########################


sub new {
    my($class) = @_;
    my(
       $fh_open,
       $host,
       $self,
       %args,
       );
    local $_;

    ## Create a new object with defaults.
    $self = $class->SUPER::new;
    $ {*$self}{net_telnet} = {
	bin_mode     => 0,
	blksize      => $Default_blocksize,
	buf          => "",
	cmd_prompt   => '/[\$%#>] $/',
	cmd_rm_mode  => "auto",
	dumplog      => '',
	eofile       => 1,
	errormode    => 'die',
	errormsg     => "",
	fdmask       => '',
	host         => "localhost",
	inputlog     => '',
	last_line    => "",
	maxbufsize   => 1024 * 1024,
	num_wrote    => 0,
	ofs          => "",
	opened       => '',
	opt_cback    => '',
	opt_log      => '',
	opts         => {},
	ors          => "\n",
	outputlog    => '',
	port         => 23,
	pushback_buf => "",
	rs           => "\n",
	telnet_mode  => 1,
	time_out     => 10,
	timedout     => '',
	unsent_opts  => "",
    };

    ## Indicate that we'll accept an offer from remote side for it to echo
    ## and suppress go aheads.
    &_opt_accept($self,
		 { option    => &TELOPT_ECHO,
		   is_remote => 1,
		   is_enable => 1 },
		 { option    => &TELOPT_SGA,
		   is_remote => 1,
		   is_enable => 1 },
		 );

    ## Parse the args.
    if (@_ == 2) {  # one positional arg given
	$host = $_[1];
    }
    elsif (@_ > 2) {  # named args given
	## Get the named args.
	(undef, %args) = @_;

	## Parse the errmode named arg first.
	foreach (keys %args) {
	    $self->errmode($args{$_})
		if /^-?errmode$/i;
	}

	## Parse all other named args.
	foreach (keys %args) {
	    if (/^-?binmode$/i) {
		$self->binmode($args{$_});
	    }
	    elsif (/^-?cmd_remove_mode$/i) {
		$self->cmd_remove_mode($args{$_});
	    }
	    elsif (/^-?dump_log$/i) {
		$self->dump_log($args{$_});
	    }
	    elsif (/^-?errmode$/i) {
		next;
	    }
	    elsif (/^-?fhopen$/i) {
		$fh_open = $args{$_};
	    }
	    elsif (/^-?host$/i) {
		$host = $args{$_};
	    }
	    elsif (/^-?input_log$/i) {
		$self->input_log($args{$_});
	    }
	    elsif (/^-?input_record_separator$/i) {
		$self->input_record_separator($args{$_});
	    }
	    elsif (/^-?option_log$/i) {
		$self->option_log($args{$_});
	    }
	    elsif (/^-?output_log$/i) {
		$self->output_log($args{$_});
	    }
	    elsif (/^-?output_record_separator$/i) {
		$self->output_record_separator($args{$_});
	    }
	    elsif (/^-?port$/i) {
		$self->port($args{$_})
		    or return;
	    }
	    elsif (/^-?prompt$/i) {
		$self->prompt($args{$_})
		    or return;
	    }
	    elsif (/^-?telnetmode$/i) {
		$self->telnetmode($args{$_});
	    }
	    elsif (/^-?timeout$/i) {
		$self->timeout($args{$_});
	    }
	    else {
		$self->error('usage: Net::Telnet->new(' .
			     '[Binmode => $mode,] ' .
			     '[Cmd_remove_mode => $mode,] ' .
			     '[Dump_Log => $filename,] ' .
			     '[Errmode => $errmode,] ' .
			     '[Fhopen => $filehandle,] ' .
			     '[Host => $host,] ' .
			     '[Input_log => $file,] ' .
			     '[Input_record_separator => $char,] ' .
			     '[Option_log => $file,] ' .
			     '[Output_log => $file,] ' .
			     '[Output_record_separator => $char,] '.
			     '[Port => $port,] ' .
			     '[Prompt => $matchop,] ' .
			     '[Telnetmode => $mode,] ' .
			     '[Timeout => $secs,])');
	    }
	}
    }

    if (defined $fh_open) {  # user wants us to attach to existing filehandle
	$self->fhopen($fh_open)
	    or return;
    }
    elsif (defined $host) {  # user wants us to open a connection to host
	$self->host($host);
	$self->open
	    or return;
    }

    $self;
} # end sub new


sub DESTROY {
} # end sub DESTROY


sub binmode {
    my($self, $mode) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{bin_mode};

    if (@_ >= 2) {
	$mode = 0
	    unless defined $mode;

	$stream->{bin_mode} = $mode;
    }

    $prev;
} # end sub binmode


sub break {
    my($self) = @_;
    my $stream = $ {*$self}{net_telnet};
    $stream->{timedout} = '';
    return if $stream->{eofile};
    local $stream->{ors} = '';

    $self->print("\xff\xf3");
} # end sub break


sub buffer {
    my($self) = @_;
    my $stream = $ {*$self}{net_telnet};

    \$stream->{buf};
} # end sub buffer


sub buffer_empty {
    my($self) = @_;
    my(
       $buffer,
       );

    $buffer = $self->buffer;
    $$buffer = "";
} # end sub buffer_empty


sub close {
    my($self) = @_;
    my $stream = $ {*$self}{net_telnet};

    $stream->{eofile} = 1;
    $stream->{opened} = '';
    close $self
	if defined fileno($self);

    1;
} # end sub close


sub cmd {
    my($self, @args) = @_;
    my(
       $arg,
       $buf,
       $cmd_remove_mode,
       $firstpos,
       $lastpos,
       $lines,
       $orig_errmode,
       $orig_prompt,
       $orig_timeout,
       $output,
       $output_ref,
       $prompt,
       $remove_echo,
       $rs,
       $rs_len,
       $telopt_echo,
       $timeout,
       @cmd,
       );
    local $_;

    ## Init vars.
    $output = [];
    $cmd_remove_mode = $self->cmd_remove_mode;
    $timeout = $self->timeout;
    $self->timed_out('');
    return if $self->eof;

    ## Parse args.
    if (@_ == 2) {  # one positional arg given
	push @cmd, $_[1];
    }
    elsif (@_ > 2) {  # named args given
	## Parse the named args.
	while (($_, $arg) = splice @args, 0, 2) {
	    if (/^-?cmd_remove/i) {
		$cmd_remove_mode = $arg;
		$cmd_remove_mode = "auto"
		    if $cmd_remove_mode =~ /^auto/i;
	    }
	    elsif (/^-?output$/i) {
		$output_ref = $arg;
		if (defined($output_ref) and ref($output_ref) eq "ARRAY") {
		    $output = $output_ref;
		}
	    }
	    elsif (/^-?prompt$/i) {
		$prompt = $arg;
	    }
	    elsif (/^-?string$/i) {
		push @cmd, $arg;
	    }
	    elsif (/^-?timeout$/i) {
		$timeout = &_parse_timeout($arg);
	    }
	    else {
		return $self->error('usage: $obj->cmd(',
				    '[Cmd_remove => $boolean,] ',
				    '[Output => $ref,] ',
				    '[Prompt => $match,] ',
				    '[String => $string,] ',
				    '[Timeout => $secs,])');
	    }
	}
    }

    ## Override some user settings.
    $orig_errmode = $self->errmode('return');
    $orig_timeout = $self->timeout(&_endtime($timeout));
    $orig_prompt  = $self->prompt($prompt) if defined $prompt;
    $self->errmsg('');

    ## Send command and wait for the prompt.
    $self->print(@cmd)
	and ($lines) = $self->waitfor($self->prompt);

    ## Restore user settings.
    $self->errmode($orig_errmode);
    $self->timeout($orig_timeout);
    $self->prompt($orig_prompt) if defined $orig_prompt;

    ## Check for failure.
    return $self->error("command timed-out") if $self->timed_out;
    return $self->error($self->errmsg) if $self->errmsg ne '';
    return if $self->eof;

    ## Split lines into an array, keeping record separator at end of line.
    $firstpos = 0;
    $rs = $self->input_record_separator;
    $rs_len = length $rs;
    while (($lastpos = index($lines, $rs, $firstpos)) > -1) {
	push(@$output,
	     substr($lines, $firstpos, $lastpos - $firstpos + $rs_len));
	$firstpos = $lastpos + $rs_len;
    }

    if ($firstpos < length $lines) {
	push @$output, substr($lines, $firstpos);
    }

    ## Determine if we should remove the first line of output based
    ## on the assumption that it's an echoed back command.
    if ($cmd_remove_mode eq "auto") {
	## See if remote side told us they'd echo.
	$telopt_echo = $self->option_state(&TELOPT_ECHO);
	$remove_echo = $telopt_echo->{remote_enabled};
    }
    else {  # user explicitly told us how many lines to remove.
	$remove_echo = $cmd_remove_mode;
    }

    ## Get rid of possible echo back command.
    while ($remove_echo--) {
	shift @$output;
    }

    ## Ensure at least a null string when there's no command output - so
    ## "true" is returned in a list context.
    unless (@$output) {
	@$output = ('');
    }

    ## Return command output via named arg, if requested.
    if (defined $output_ref) {
	if (ref($output_ref) eq "SCALAR") {
	    $$output_ref = join '', @$output;
	}
	elsif (ref($output_ref) eq "HASH") {
	    %$output_ref = @$output;
	}
    }

    wantarray ? @$output : 1;
} # end sub cmd


sub cmd_remove_mode {
    my($self, $mode) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{cmd_rm_mode};

    if (@_ >= 2) {
	if (! defined $mode) {
	    $mode = 0;
	}
	elsif ($mode =~ /^auto/i) {
	    $mode = "auto";
	}
	else {
	    ## Ensure it's a non-negative integer.
	    eval {
		local $^W = 1;
		local $SIG{'__WARN__'} = sub { die "non-numeric\n" };
		local $SIG{'__DIE__'} = 'DEFAULT';
		$mode = abs(int $mode);
	    };
	    if ($@) {
		$mode = 0;
	    }
	}

	$stream->{cmd_rm_mode} = $mode;
    }

    $prev;
} # end sub cmd_remove_mode


sub dump_log {
    my($self, $name) = @_;
    my(
       $fh,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $fh = $stream->{dumplog};

    if (@_ >= 2) {
	$fh = &_fname_to_handle($self, $name);
	$stream->{dumplog} = $fh;
    }

    $fh;
} # end sub dump_log


sub eof {
    my($self) = @_;

    $ {*$self}{net_telnet}{eofile};
} # end sub eof


sub errmode {
    my($self, $mode) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{errormode};

    if (@_ >= 2) {
	## Set the error mode.
	defined $mode or $mode = '';
	if (ref($mode) eq 'CODE') {
	    $stream->{errormode} = $mode;
	}
	elsif (ref($mode) eq 'ARRAY') {
	    unless (ref($mode->[0]) eq 'CODE') {
		&_carp($self,
		       'bad errmode: first item in list must be a code ref');
		$mode = 'die';
	    }
	    $stream->{errormode} = $mode;
	}
	elsif ($mode =~ /^return$/i) {
	    $stream->{errormode} = 'return';
	}
	else {
	    $stream->{errormode} = 'die';
	}
    }

    $prev;
} # end sub errmode


sub errmsg {
    my($self, @errmsgs) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{errormsg};

    if (@_ >= 2) {
	$stream->{errormsg} = join '', @errmsgs;
    }

    $prev;
} # end sub errmsg


sub error {
    my($self, @errmsg) = @_;
    my(
       $errmsg,
       $func,
       $mode,
       $stream,
       @args,
       );

    $stream = $ {*$self}{net_telnet};

    if (@_ >= 2) {
	## Put error message in the object.
	$errmsg = join '', @errmsg;
	$stream->{errormsg} = $errmsg;

	## Do the error action as described by error mode.
	$mode = $stream->{errormode};
	if (ref($mode) eq 'CODE') {
	    &$mode($errmsg);
	    return;
	}
	elsif (ref($mode) eq 'ARRAY') {
	    ($func, @args) = @$mode;
	    &$func(@args);
	    return;
	}
	elsif ($mode =~ /^return$/i) {
	    return;
	}
	else {  # die
	    if ($errmsg =~ /\n$/) {
		die $errmsg;
	    }
	    else {
		## Die and append caller's line number to message.
		&_croak($self, $errmsg);
	    }
	}
    }
    else {
	return $stream->{errormsg} ne '';
    }
} # end sub error


sub fhopen {
    my($self, $fh) = @_;
    my(
       $blksize,
       $globref,
       $stream,
       );

    ## Convert given filehandle to a typeglob reference, if necessary.
    $globref = &_qualify_fh($self, $fh);

    ## Ensure filehandle is already open.
    return $self->error("fhopen filehandle isn't already open")
	unless defined($globref) and defined(fileno $globref);

    ## Ensure we're closed.
    $self->close;

    ## Save our private data.
    $stream = $ {*$self}{net_telnet};

    ## Switch ourself with the given filehandle.
    *$self = *$globref;

    ## Restore our private data.
    $ {*$self}{net_telnet} = $stream;

    ## Re-initialize ourself.
    $self->autoflush;
    $stream = $ {*$self}{net_telnet};
    $blksize = (stat $self)[11];
    $stream->{blksize} = $blksize || $Default_blocksize;
    $stream->{buf} = "";
    $stream->{cmd_rm_mode} = 0;
    $stream->{eofile} = '';
    $stream->{errormsg} = "";
    vec($stream->{fdmask}='', fileno($self), 1) = 1;
    $stream->{host} = "";
    $stream->{last_line} = "";
    $stream->{num_wrote} = 0;
    $stream->{opened} = 1;
    $stream->{port} = '';
    $stream->{pushback_buf} = "";
    $stream->{telnet_mode} = 0;
    $stream->{timedout} = '';
    $stream->{unsent_opts} = "";
    &_reset_options($stream->{opts});

    1;
} # end sub fhopen


sub get {
    my($self, %args) = @_;
    my(
       $endtime,
       $line,
       $stream,
       $timeout,
       );
    local $_;

    ## Init vars.
    $stream = $ {*$self}{net_telnet};
    $timeout = $stream->{time_out};
    $stream->{timedout} = '';
    return if $stream->{eofile};

    ## Parse the named args.
    foreach (keys %args) {
	if (/^-?timeout$/i) {
	    $timeout = &_parse_timeout($args{$_});
	}
	else {
	    return $self->error('usage: $obj->get([Timeout => $secs,])');
	}
    }

    ## Set wall time when we time out.
    $endtime = &_endtime($timeout);

    ## Try to send any waiting option negotiation.
    if (length $stream->{unsent_opts}) {
	&_flush_opts($self, $stream);
    }

    ## Try to read just the waiting data using return error mode.
    {
	local $stream->{errormode} = 'return';
	$stream->{errormsg} = '';
	&_fillbuf($self, $stream, 0);
    }

    ## We're done if we timed-out and timeout value is set to "poll".
    return if $stream->{timedout} and defined($timeout) and $timeout == 0;

    ## We're done if we hit an error other than timing out.
    return $self->error($stream->{errormsg})
	if $stream->{errormsg} and ! $stream->{timedout};


    ## If buffer is still empty, try to read according to user's timeout.
    if (! length $stream->{buf}) {
	&_fillbuf($self, $stream, $endtime)
	    or do {
		return if $stream->{timedout};

		## We've reached end-of-file.
		$self->close;
		return;
	    };
    }

    ## Extract chars from buffer.
    $line = $stream->{buf};
    $stream->{buf} = '';

    $line;
} # end sub get


sub getline {
    my($self, %args) = @_;
    my(
       $endtime,
       $len,
       $line,
       $offset,
       $pos,
       $stream,
       $timeout,
       );
    local $_;

    ## Init vars.
    $stream = $ {*$self}{net_telnet};
    $timeout = $stream->{time_out};
    $stream->{timedout} = '';
    return if $stream->{eofile};

    ## Parse the named args.
    foreach (keys %args) {
	if (/^-?timeout$/i) {
	    $timeout = &_parse_timeout($args{$_});
	}
	else {
	    return $self->error('usage: $obj->getline([Timeout => $secs,])');
	}
    }

    ## Set wall time when we time out.
    $endtime = &_endtime($timeout);

    ## Try to send any waiting option negotiation.
    if (length $stream->{unsent_opts}) {
	&_flush_opts($self, $stream);
    }

    ## Keep reading into buffer until end-of-line is read.
    $offset = 0;
    while (($pos = index($stream->{buf}, $stream->{rs}, $offset)) == -1) {
	$offset = length $stream->{buf};
	&_fillbuf($self, $stream, $endtime)
	    or do {
		return if $stream->{timedout};

		## We've reached end-of-file.
		$self->close;
		if (length $stream->{buf}) {
		    return $stream->{buf};
		}
		else {
		    return;
		}
	    };
    }

    ## Extract line from buffer.
    $len = $pos + length $stream->{rs};
    $line = substr($stream->{buf}, 0, $len);
    substr($stream->{buf}, 0, $len) = '';

    $line;
} # end sub getline


sub getlines {
    my($self) = @_;
    my(
       $len,
       $line,
       $pos,
       $stream,
       @lines,
       );

    $stream = $ {*$self}{net_telnet};

    ## Fill buffer and get first line.
    $line = getline(@_)
	or return;
    push @lines, $line;

    ## Extract subsequent lines from buffer.
    while (($pos = index($stream->{buf}, $stream->{rs})) != -1) {
	$len = $pos + length $stream->{rs};
	push @lines, substr($stream->{buf}, 0, $len);
	substr($stream->{buf}, 0, $len) = '';
    }

    @lines;
} # end sub getlines


sub host {
    my($self, $host) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{host};

    if (@_ >= 2) {
	$host = ''
	    unless defined $host and length $host;

	$stream->{host} = $host;
    }

    $prev;
} # end sub host


sub input_log {
    my($self, $name) = @_;
    my(
       $fh,
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{inputlog};

    if (@_ >= 2) {
	$fh = &_fname_to_handle($self, $name);
	$stream->{inputlog} = $fh;
    }

    $prev;
} # end sub input_log


sub input_record_separator {
    my($self, $rs) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{rs};

    if (@_ >= 2) {
	defined $rs or $rs = '';
	$stream->{rs} = $rs;
    }

    $prev;
} # end sub input_record_separator


sub lastline {
    my($self, $line) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{last_line};

    if (@_ >= 2) {
	defined $line or $line = '';
	$stream->{last_line} = $line;
    }

    $prev;
} # end sub lastline


sub login {
    my($self) = @_;
    my(
       $cmd_prompt,
       $endtime,
       $error,
       $lastline,
       $match,
       $orig_errmode,
       $orig_timeout,
       $passwd,
       $prematch,
       $reset,
       $timeout,
       $usage,
       $username,
       %args,
       );
    local $_;

    ## Init vars.
    $timeout = $self->timeout;
    $self->timed_out('');
    return if $self->eof;
    $cmd_prompt = $self->prompt;
    $usage = 'usage: $obj->login(Name => $name, Password => $password, ' .
	'[Prompt => $match,] [Timeout => $secs,])';

    if (@_ == 3) {  # just username and passwd given
	$username = $_[1];
	$passwd = $_[2];
    }
    else {  # named args given
	## Get the named args.
	(undef, %args) = @_;

	## Parse the named args.
	foreach (keys %args) {
	    if (/^-?name$/i) {
		$username = $args{$_};
		defined($username)
		    or $username = "";
	    }
	    elsif (/^-?pass/i) {
		$passwd = $args{$_};
		defined($passwd)
		    or $passwd = "";
	    }
	    elsif (/^-?prompt$/i) {
		$cmd_prompt = $args{$_};
		defined $cmd_prompt
		    or $cmd_prompt = '';
		return $self->error("bad match operator: ",
				    "opening delimiter missing: $cmd_prompt")
		    unless ($cmd_prompt =~ m(^\s*/)
			    or $cmd_prompt =~ m(^\s*m\s*\W));
	    }
	    elsif (/^-?timeout$/i) {
		$timeout = &_parse_timeout($args{$_});
	    }
	    else {
		return $self->error($usage);
	    }
	}
    }

    return $self->error($usage)
	unless defined($username) and defined($passwd);

    ## Override these user set-able values.
    $endtime = &_endtime($timeout);
    $orig_timeout = $self->timeout($endtime);
    $orig_errmode = $self->errmode('return');

    ## Create a subroutine to reset to original values.
    $reset
	= sub {
	    $self->errmode($orig_errmode);
	    $self->timeout($orig_timeout);
	    1;
	};

    ## Create a subroutine to generate an error for user.
    $error
	= sub {
	    my($errmsg) = @_;

	    &$reset;
	    if ($self->timed_out) {
		return $self->error($errmsg);
	    }
	    elsif ($self->eof) {
		($lastline = $self->lastline) =~ s/\n+//;
		return $self->error($errmsg, ": ", $lastline);
	    }
	    else {
		return $self->error($self->errmsg);
	    }
	};

    ## Wait for login prompt.
    $self->waitfor(-match => '/login[: ]*$/i',
		   -match => '/username[: ]*$/i')
	or do {
	    return &$error("read eof waiting for login prompt")
		if $self->eof;
	    return &$error("timed-out waiting for login prompt");
	};

    ## Send login name.
    $self->print($username)
	or return &$error("login disconnected");

    ## Wait for password prompt.
    $self->waitfor(-match => '/password[: ]*$/i')
	or do {
	    return &$error("read eof waiting for password prompt")
		if $self->eof;
	    return &$error("timed-out waiting for password prompt");
	};

    ## Send password.
    $self->print($passwd)
	or return &$error("login disconnected");

    ## Wait for command prompt or another login prompt.
    ($prematch, $match) = $self->waitfor(-match => '/login[: ]*$/i',
					 -match => '/username[: ]*$/i',
					 -match => $cmd_prompt)
	or do {
	    return &$error("read eof waiting for command prompt")
		if $self->eof;
	    return &$error("timed-out waiting for command prompt");
	};

    ## Reset object to orig values.
    &$reset;

    ## It's a bad login if we got another login prompt.
    return $self->error("login failed: bad name or password")
	if $match =~ /login[: ]*$/i or $match =~ /username[: ]*$/i;

    1;
} # end sub login


sub max_buffer_length {
    my($self, $maxbufsize) = @_;
    my(
       $minbufsize,
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{maxbufsize};
    $minbufsize = 512;

    if (@_ >= 2) {
	## Ensure a valid max length.
	unless (defined $maxbufsize) {
	    $maxbufsize = $minbufsize;
	}

	## Test for non-numeric or negative values.
	eval {
	    local $^W = 1;
	    local $SIG{'__WARN__'} = sub { die "non-numeric\n" };
	    local $SIG{'__DIE__'} = 'DEFAULT';
	    $maxbufsize *= 1;
	};
	if ($@ or $maxbufsize < $minbufsize) {
	    $maxbufsize = $minbufsize;
	}

	$stream->{maxbufsize} = $maxbufsize;
    }

    $prev;
} # end sub max_buffer_length


sub open {
    my($self) = @_;
    my(
       $blksize,
       $errno,
       $host,
       $ip_addr,
       $port,
       $stream,
       $timeout,
       %args,
       );
    local $_;

    ## Init vars.
    $stream = $ {*$self}{net_telnet};
    $timeout = $stream->{time_out};
    $stream->{timedout} = '';

    if (@_ == 2) {  # one positional arg given
	$self->host($_[1]);
    }
    elsif (@_ > 2) {  # named args given
	## Get the named args.
	(undef, %args) = @_;

	## Parse the named args.
	foreach (keys %args) {
	    if (/^-?host$/i) {
		$self->host($args{$_});
	    }
	    elsif (/^-?port$/i) {
		$self->port($args{$_})
		    or return;
	    }
	    elsif (/^-?timeout$/i) {
		$timeout = &_parse_timeout($args{$_});
	    }
	    else {
		return $self->error('usage: $obj->open([Host => $host,] ',
				    '[Port => $service,] ',
				    '[Timeout => secs,])');
	    }
	}
    }

    ## Get host and port.
    $host = $self->host;
    $port = $self->port;

    ## Ensure we're already closed.
    $self->close;

    ## Don't use a timeout if we can't use the alarm signal.
    unless (&_have_alarm) {
	$timeout = undef;
    }

    if (defined $timeout) {  # use a timeout
	## Ensure a valid timeout value for alarm.
	if ($timeout < 1) {
	    $timeout = 1;
	}
	$timeout = int($timeout + 1.5);

	## Connect to server, timing out if it takes too long.
	eval {
	    ## Turn on timer.
	    local $SIG{'__DIE__'} = 'DEFAULT';
	    local $SIG{ALRM} = sub { die "timed-out\n" };
	    alarm $timeout;

	    ## Lookup server's IP address.
	    $ip_addr = inet_aton $host
		or die "unknown remote host: $host\n";

	    ## Create a socket and attach the filehandle to it.
	    socket $self, AF_INET, SOCK_STREAM, 0
		or die "problem creating socket: $!\n";

	    ## Open connection to server.
	    connect $self, sockaddr_in($port, $ip_addr)
		or die "problem connecting to \"$host\", port $port: $!\n";
	};
	alarm 0;

	## Check for error.
	if ($@ =~ /^timed-out$/) {  # time out failure
	    $stream->{timedout} = 1;
	    $self->close;
	    if (! $ip_addr) {
		return $self->error("unknown remote host: $host: ",
				    "name lookup timed-out");
	    }
	    else {
		return $self->error("problem connecting to \"$host\", ",
				    "port $port: connection timed-out");
	    }
	}
	elsif ($@) {  # hostname lookup or connect failure
	    $self->close;
	    chomp $@;
	    return $self->error($@);
	}
    }
    else {  # don't use a timeout
	## Lookup server's IP address.
	$ip_addr = inet_aton $host
	    or return $self->error("unknown remote host: $host");

	## Create a socket and attach the filehandle to it.
	socket $self, AF_INET, SOCK_STREAM, 0
	    or return $self->error("problem creating socket: $!");

	## Open connection to server.
	connect $self, sockaddr_in($port, $ip_addr)
	    or do {
		$errno = "$!";
		$self->close;
		return $self->error("problem connecting to \"$host\", ",
				    "port $port: $errno");
	    };
    }

    $self->autoflush;
    $blksize = (stat $self)[11];
    $stream->{blksize} = $blksize || $Default_blocksize;
    $stream->{buf} = "";
    $stream->{eofile} = '';
    $stream->{errormsg} = "";
    vec($stream->{fdmask}='', fileno($self), 1) = 1;
    $stream->{last_line} = "";
    $stream->{num_wrote} = 0;
    $stream->{opened} = 1;
    $stream->{pushback_buf} = "";
    $stream->{timedout} = '';
    $stream->{unsent_opts} = "";
    &_reset_options($stream->{opts});

    1;
} # end sub open


sub option_accept {
    my($self, @args) = @_;
    my(
       $arg,
       $option,
       $stream,
       @opt_args,
       );
    local $_;

    ## Init vars.
    $stream = $ {*$self}{net_telnet};

    ## Parse the named args.
    while (($_, $arg) = splice @args, 0, 2) {
	## Verify and save arguments.
	if (/^-?do$/i) {
	    ## Make sure a callback is defined.
	    return $self->error("usage: an option callback must already ",
				"be defined when enabling with $_")
		unless $stream->{opt_cback};

	    $option = &_verify_telopt_arg($self, $arg, $_)
		or return;
	    push @opt_args, { option    => $option,
			      is_remote => '',
			      is_enable => 1,
			  };
	}
	elsif (/^-?dont$/i) {
	    $option = &_verify_telopt_arg($self, $arg, $_)
		or return;
	    push @opt_args, { option    => $option,
			      is_remote => '',
			      is_enable => '',
			  };
	}
	elsif (/^-?will$/i) {
	    ## Make sure a callback is defined.
	    return $self->error("usage: an option callback must already ",
				"be defined when enabling with $_")
		unless $stream->{opt_cback};

	    $option = &_verify_telopt_arg($self, $arg, $_)
		or return;
	    push @opt_args, { option    => $option,
			      is_remote => 1,
			      is_enable => 1,
			  };
	}
	elsif (/^-?wont$/i) {
	    $option = &_verify_telopt_arg($self, $arg, $_)
		or return;
	    push @opt_args, { option    => $option,
			      is_remote => 1,
			      is_enable => '',
			  };
	}
	else {
	    return $self->error('usage: $obj->option_accept(' .
				'[Do => $telopt,] ',
				'[Dont => $telopt,] ',
				'[Will => $telopt,] ',
				'[Wont => $telopt,]');
	}
    }

    ## Set "receive ok" for options specified.
    &_opt_accept($self, @opt_args);
} # end sub option_accept


sub option_callback {
    my($self, $callback) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{opt_cback};

    if (@_ >= 2) {
	return $self->error("actual argument is not a code ref")
	    unless defined $callback and ref($callback) eq "CODE";

	$stream->{opt_cback} = $callback;
    }

    $prev;
} # end sub option_callback


sub option_log {
    my($self, $name) = @_;
    my(
       $fh,
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{opt_log};

    if (@_ >= 2) {
	$fh = &_fname_to_handle($self, $name);
	$stream->{opt_log} = $fh;
    }

    $prev;
} # end sub option_log


sub option_state {
    my($self, $option) = @_;
    my(
       $opt_state,
       $stream,
       %opt_state,
       );

    ## Ensure telnet option is non-negative integer.
    $option = &_verify_telopt_arg($self, $option)
	or return;

    ## Init vars.
    $stream = $ {*$self}{net_telnet};
    unless (defined $stream->{opts}{$option}) {
	&_set_default_option($stream, $option);
    }

    ## Return hashref to a copy of the values.
    $opt_state = $stream->{opts}{$option};
    %opt_state = %$opt_state;
    \%opt_state;
} # end sub option_state


sub output_field_separator {
    my($self, $ofs) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{ofs};

    if (@_ >= 2) {
	defined $ofs or $ofs = '';
	$stream->{ofs} = $ofs;
    }

    $prev;
} # end sub output_field_separator


sub output_log {
    my($self, $name) = @_;
    my(
       $fh,
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{outputlog};

    if (@_ >= 2) {
	$fh = &_fname_to_handle($self, $name);
	$stream->{outputlog} = $fh;
    }

    $prev;
} # end sub output_log


sub output_record_separator {
    my($self, $ors) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{ors};

    if (@_ >= 2) {
	defined $ors or $ors = '';
	$stream->{ors} = $ors;
    }

    $prev;
} # end sub output_record_separator


sub port {
    my($self, $port) = @_;
    my(
       $prev,
       $service,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{port};

    if (@_ >= 2) {
	return $self->error("bad port number: 0")
	    unless defined $port and $port;

	## Convert service to a port number.
	if ($port !~ /^\d+$/) {  # port isn't all digits
	    $service = $port;
	    $port = getservbyname($service, 'tcp')
		or return $self->error("unknown TCP service: $service");
	}

	$stream->{port} = $port;
    }

    $prev;
} # end sub port


sub print {
    my($self) = shift;
    my(
       $data,
       $endtime,
       $fh,
       $len,
       $nfound,
       $nwrote,
       $offset,
       $ready,
       $stream,
       $timed_out,
       $timeout,
       );

    $stream = $ {*$self}{net_telnet};
    $stream->{timedout} = '';
    $stream->{num_wrote} = 0;
    return $self->error("print failed: handle is closed")
	unless $stream->{opened};

    ## Try to send any waiting option negotiation.
    if (length $stream->{unsent_opts}) {
	&_flush_opts($self, $stream);
    }

    ## Add field and record separators.
    $data = join($stream->{ofs}, @_) . $stream->{ors};

    ## If requested, log the output.
    if ($stream->{outputlog}) {
	local $\ = '';
	$fh = $stream->{outputlog};
	$fh->print($data);
    }

    ## Convert newlines to carriage-return and linefeed.
    $data =~ s(\n)(\015\012)g
	unless $stream->{bin_mode};

    $offset = 0;
    $len = length $data;
    $endtime = &_endtime($stream->{time_out});
    while ($len) {
	## Set how long to wait for output ready.
	($timed_out, $timeout) = &_timeout_interval($endtime);
	if ($timed_out) {
	    $stream->{timedout} = 1;
	    return $self->error("print timed-out");
	}

	## Wait for output ready.
	$nfound = select '', $ready=$stream->{fdmask}, '', $timeout;
	if ($nfound > 0) {  # data can be written
	    if ($nwrote = syswrite $self, $data, $len, $offset) {
		## If requested, display network traffic.
		($stream->{dumplog})
		    and &_log_dump('>', $stream->{dumplog},
				   \$data, $offset, $nwrote);

		$stream->{num_wrote} += $nwrote;
		$offset += $nwrote;
		$len -= $nwrote;
		next;
	    }
	    elsif (! defined $nwrote) {  # write failed
		next if $! =~ /^Interrupted/;

		$stream->{opened} = '';
		return $self->error("unexpected write error: $!");
	    }
	    else {  # zero chars written
		$stream->{opened} = '';
		return $self->error("unexpected zero length write error: $!");
	    }
	}
	elsif ($nfound < 0) {  # select failure
	    next if $! =~ /^Interrupted/;

	    ## Failure equates to eof.
	    $stream->{opened} = '';
	    return $self->error("unexpected write error: $!");
	}
	else {  # timed-out
	    $stream->{timedout} = 1;
	    return $self->error("print timed-out");
	}
    }

    1;
} # end sub print


sub print_length {
    my($self) = @_;

    $ {*$self}{net_telnet}{num_wrote};
} # end sub print_length


sub prompt {
    my($self, $prompt) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{cmd_prompt};

    ## Parse args.
    if (@_ == 2) {
	defined $prompt or $prompt = '';
	return $self->error("bad match operator: ",
			    "opening delimiter missing: $prompt")
	    unless $prompt =~ m(^\s*/) or $prompt =~ m(^\s*m\s*\W);

	$stream->{cmd_prompt} = $prompt;
    }
    elsif (@_ > 2) {
	return $self->error('usage: $obj->prompt($match_op)');
    }

    $prev;
} # end sub prompt


sub telnetmode {
    my($self, $mode) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{telnet_mode};

    if (@_ >= 2) {
	$mode = 0
	    unless defined $mode;

	$stream->{telnet_mode} = $mode;
    }

    $prev;
} # end sub telnetmode


sub timed_out {
    my($self, $value) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{timedout};

    if (@_ >= 2) {
	defined $value or $value = '';
	$stream->{timedout} = $value;
    }

    $prev;
} # end sub timed_out


sub timeout {
    my($self, $timeout) = @_;
    my(
       $prev,
       $stream,
       );

    $stream = $ {*$self}{net_telnet};
    $prev = $stream->{time_out};

    if (@_ >= 2) {
	$stream->{time_out} = &_parse_timeout($timeout);
    }

    $prev;
} # end sub timeout


sub waitfor {
    my($self, @args) = @_;
    my(
       $arg,
       $endtime,
       $len,
       $match,
       $match_op,
       $pos,
       $prematch,
       $search,
       $search_cond,
       $stream,
       $string,
       $timeout,
       @match_cond,
       @match_ops,
       @search_cond,
       @string_cond,
       @warns,
       );
    local $_;

    ## Init vars.
    $stream = $ {*$self}{net_telnet};
    $timeout = $stream->{time_out};
    $stream->{timedout} = '';
    return if $stream->{eofile};
    return unless @args;

    ## Code template used build conditional to match a string.
    ## Values between array elements must be supplied later.
    @string_cond =
	('if (($pos = index $stream->{buf}, ', ') > -1) {
	    $len = ', ';
	    $prematch = substr $stream->{buf}, 0, $pos;
	    $match = substr $stream->{buf}, $pos, $len;
	    substr($stream->{buf}, 0, $pos + $len) = "";
	    last;
	}');

    ## Code template used build conditional to match a pattern.
    ## Values between array elements must be supplied later.
    @match_cond =
	('if ($stream->{buf} =~ ', ') {
	    $prematch = $`;
	    $match = $&;
	    substr($stream->{buf}, 0, length($`) + length($&)) = "";
	    last;
	}');

    ## Parse args.
    if (@_ == 2) {  # one positional arg given
	$arg = $_[1];
	return $self->error("bad match operator: ",
			    "opening delimiter missing: $arg")
	    unless $arg =~ m(^\s*/) or $arg =~ m(^\s*m\s*\W);

	## Fill in the blanks in the code template.
	push @match_ops, $arg;
	push @search_cond, join('', $match_cond[0], $arg, $match_cond[1]);
    }
    elsif (@_ > 2) {  # named args given
	## Parse the named args.
	while (($_, $arg) = splice @args, 0, 2) {
	    if (/^-?timeout$/i) {
		$timeout = &_parse_timeout($arg);
	    }
	    elsif (/^-?match$/i) {
		return $self->error("bad match operator: ",
				    "opening delimiter missing: $arg")
		    unless $arg =~ m(^\s*/) or $arg =~ m(^\s*m\s*\W);

		## Fill in the blanks in the code template.
		push @match_ops, $arg;
		push @search_cond, join('',
					$match_cond[0], $arg, $match_cond[1]);
	    }
	    elsif (/^-?string$/i) {
		## Fill in the blanks in the code template.
		$arg =~ s/'/\\'/g;  # quote ticks
		push @search_cond, join('',
					$string_cond[0], "'$arg'",
					$string_cond[1], length($arg),
					$string_cond[2]);
	    }
	    else {
		return $self->error('usage: $obj->waitfor([Match => ',
				    '$match_op,] [String => $string,] ',
				    '[Timeout => $secs,])');
	    }
	}
    }

    ## Construct conditional to check for requested string and pattern matches.
    ## Turn subsequent "if"s into "elsif".
    $search_cond = join "\n\tels", @search_cond;

    ## Construct loop to fill buffer until string/pattern, timeout, or eof.
    $search = join '', "
    while (1) {\n\t",
	$search_cond, '
	&_fillbuf($self, $stream, $endtime)
	    or do {
		last if $stream->{timedout};
		$self->close;
		last;
	    };
    }';

    ## Set wall time when we timeout.
    $endtime = &_endtime($timeout);

    ## Run the loop.
    {
	local $^W = 1;
	local $SIG{'__WARN__'} = sub { push @warns, @_ };
	local $stream->{errormode} = 'return';
	$stream->{errormsg} = '';
	eval $search;
    }

    ## Check for failure.
    return $self->error("pattern match timed-out") if $stream->{timedout};
    return $self->error($stream->{errormsg}) if $stream->{errormsg} ne '';
    return $self->error("pattern match read eof") if $stream->{eofile};

    ## Check for Perl syntax errors or warnings.
    if ($@ or @warns) {
	foreach $match_op (@match_ops) {
	    &_match_check($self, $match_op)
		or return;
	}
	return $self->error($@) if $@;
	return $self->error(@warns) if @warns;
    }

    wantarray ? ($prematch, $match) : 1;
} # end sub waitfor


######################## Private Subroutines #########################


sub _append_lineno {
    my($obj, @msgs) = @_;
    my(
       $file,
       $line,
       $pkg,
       );

    ## Find the caller that's not in object's class or one of its base classes.
    ($pkg, $file , $line) = &_user_caller($obj);
    join('', @msgs, " at ", $file, " line ", $line, "\n");
} # end sub _append_lineno


sub _carp {
    warn &_append_lineno(@_);
} # end sub _carp


sub _croak {
    die &_append_lineno(@_);
} # end sub _croak


sub _endtime {
    my($interval) = @_;

    ## Compute wall time when timeout occurs.
    if (defined $interval) {
	if ($interval >= $^T) {  # it's already an absolute time
	    return $interval;
	}
	elsif ($interval > 0) {  # it's relative to the current time
	    return int(time + 1.5 + $interval);
	}
	else {  # it's a one time poll
	    return 0;
	}
    }
    else {  # there's no timeout
	return undef;
    }
} # end sub _endtime


sub _fillbuf {
    my($self, $s, $endtime) = @_;
    my(
       $fh,
       $firstpos,
       $lastpos,
       $len_w_sep,
       $len_wo_sep,
       $nextchar,
       $nfound,
       $nread,
       $offset,
       $pos,
       $pushback_len,
       $ready,
       $timed_out,
       $timeout,
       );

    return unless $s->{opened};

    while (1) {
	## Ensure we haven't exceeded maximum buffer size.
	return $self->error("maximum input buffer length exceeded: ",
			    $s->{maxbufsize}, " bytes")
	    unless length($s->{buf}) <= $s->{maxbufsize};

	## Set how long to wait for input ready.
	($timed_out, $timeout) = &_timeout_interval($endtime);
	if ($timed_out) {
	    $s->{timedout} = 1;
	    return $self->error("read timed-out");
	}

	## Wait for input ready.
	$nfound = select $ready=$s->{fdmask}, '', '', $timeout;
	if ($nfound > 0) {  # data can be read
	    ## Append any partially read telnet char sequence.
	    $pushback_len = length $s->{pushback_buf};
	    if ($pushback_len) {
		$s->{buf} .= $s->{pushback_buf};
		$s->{pushback_buf} = '';
	    }

	    ## Do the read.
	    $offset = length $s->{buf};
	    if ($nread = sysread $self, $s->{buf}, $s->{blksize}, $offset) {
		## If requested, display network traffic.
		($s->{dumplog})
		    and &_log_dump('<', $s->{dumplog}, \$s->{buf}, $offset);

		## Process any telnet commands in the data stream.
		if ($s->{telnet_mode}
		    and index($s->{buf}, "\377", $offset - $pushback_len) > -1)
		{
		    &_interpret_cmd($self, $s, $offset - $pushback_len);
		}

		## Process carriage-return sequences in the data stream.
		$pos = $offset - $pushback_len;
		while (($pos = index($s->{buf}, "\015", $pos)) > -1) {
		    $nextchar = substr($s->{buf}, $pos + 1, 1);
		    if ($nextchar eq "\0") {
			## Convert CR NULL to CR
			substr($s->{buf}, $pos + 1, 1) = ''
			    if $s->{telnet_mode};
		    }
		    elsif ($nextchar eq "\012") {
			## Convert CR LF to newline when not in binary mode.
			substr($s->{buf}, $pos, 2) = "\n"
			    if ! $s->{bin_mode};
		    }
		    elsif (! length($nextchar) and $s->{telnet_mode}) {
			## Save CR for possible CR NULL conversion.
			$s->{pushback_buf} .= "\015";
			chop $s->{buf};
		    }

		    $pos++;
		}

		next if length $s->{buf} <= $offset;

		## If requested, log the input.
		if ($s->{inputlog}) {
		    local $\ = '';
		    $fh = $s->{inputlog};
		    $fh->print(substr $s->{buf}, $offset);
		}

		## Save last line in the buffer.
		if (($lastpos = rindex $s->{buf}, $s->{rs}) > -1) {
		    while (1) {
			## Find beginning of line.
			$firstpos = rindex $s->{buf}, $s->{rs}, $lastpos - 1;
			if ($firstpos == -1) {
			    $offset = 0;
			}
			else {
			    $offset = $firstpos + length $s->{rs};
			}

			## Determine length of line with and without separator.
			$len_wo_sep = $lastpos - $offset;
			$len_w_sep = $len_wo_sep + length $s->{rs};

			## Save line if it's not blank.
			if (substr($s->{buf}, $offset, $len_wo_sep)
			    !~ /^\s*$/)
			{
			    $s->{last_line} = substr($s->{buf},
						     $offset,
						     $len_w_sep);
			    last;
			}

			last if $firstpos == -1;

			$lastpos = $firstpos;
		    }
		}

		return 1;
	    }
	    elsif (! defined $nread) {  # read failed
		next if $! =~ /^Interrupted/;

		$s->{opened} = '';
		return $self->error("unexpected read error: $!");
	    }
	    else {  # read end-of-file
		$s->{opened} = '';
		return;
	    }
	}
	elsif ($nfound < 0) {  # select failure
	    next if $! =~ /^Interrupted/;

	    ## Failure equates to eof.
	    $s->{opened} = '';
	    return $self->error("unexpected read error: $!");
	}
	else {  # timed-out
	    $s->{timedout} = 1;
	    return $self->error("read timed-out");
	}
    }
} # end sub _fillbuf


sub _flush_opts {
    my($self, $s) = @_;
    my(
       $option_chars,
       );

    ## Get option and clear the output buf.
    $option_chars = $s->{unsent_opts};
    $s->{unsent_opts} = '';

    ## Try to send options without waiting.
    {
	local $s->{errormode} = 'return';
	local $s->{time_out} = 0;
	local $s->{ors} = '';
	$self->print($option_chars)
	    or do {
		## Save chars not printed for later.
		substr($option_chars, 0, $self->print_length) = '';
		$s->{unsent_opts} .= $option_chars;
	    };
    }

    1;
} # end sub _flush_opts


sub _fname_to_handle {
    my($self, $fh) = @_;
    my(
       $filename,
       );

    ## Default is off.
    if (!defined $fh or !length $fh) {
	return '';
    }

    ## Assume arg is a filename if it's not an open filehandle.
    no strict 'refs';
    if (!defined fileno($fh)) {
	$filename = $fh;
	$fh = &_new_handle();
	open $fh, ">$filename"
	    or do {
		&_carp($self, "problem creating $filename: $!");
		return '';
	    };
    }

    $fh->autoflush;
    $fh;
} # end sub _fname_to_handle


sub _have_alarm {
    eval {
	local $SIG{'__DIE__'} = 'DEFAULT';
	local $SIG{ALRM} = sub { die };
	alarm 0;
    };

    ! $@;
} # end sub _have_alarm


sub _interpret_cmd {
    my($self, $s, $offset) = @_;
    my(
       $endpos,
       $nextchar,
       $option,
       $pos,
       );

    ## Parse telnet commands in the data stream.
    $pos = $offset;
    while (($pos = index $s->{buf}, "\377", $pos) > -1) {  # unprocessed IAC
	$nextchar = substr $s->{buf}, $pos + 1, 1;

	## Save command if it's only partially read.
	if (! length $nextchar) {
	    $s->{pushback_buf} .= "\377";
	    chop $s->{buf};
	    last;
	}

	if ($nextchar eq "\377") {  # IAC is escaping "\377" char
	    ## Remove escape char from data stream.
	    substr($s->{buf}, $pos, 1) = '';
	    $pos++;
	}
	elsif ($nextchar eq "\375" or $nextchar eq "\373" or
	       $nextchar eq "\374" or $nextchar eq "\376") {  # opt negotiation
	    $option = substr $s->{buf}, $pos + 2, 1;

	    ## Save command if it's only partially read.
	    if (! length $option) {
		$s->{pushback_buf} .= "\377" . $nextchar;
		chop $s->{buf};
		chop $s->{buf};
		last;
	    }

	    ## Remove command from data stream.
	    substr($s->{buf}, $pos, 3) = '';

	    ## Handle option negotiation.
	    &_negotiate_recv($self, $s, $nextchar, ord($option), $pos);
	}
	elsif ($nextchar eq "\372") {  # start of subnegotiation parameters
	    ## Save command if it's only partially read.
	    $endpos = index $s->{buf}, "\360", $pos;
	    if ($endpos == -1) {
		$s->{pushback_buf} .= substr $s->{buf}, $pos;
		substr($s->{buf}, $pos) = '';
		last;
	    }

	    ## Ignore subnegotiation cmd.
	    substr($s->{buf}, $pos, $endpos - $pos + 1) = '';
	}
	else {  # various two char telnet commands
	    ## Ignore and remove command from data stream.
	    substr($s->{buf}, $pos, 2) = '';
	}
    }

    ## Try to send any waiting option negotiation.
    if (length $s->{unsent_opts}) {
	&_flush_opts($self, $s);
    }

    1;
} # end sub _interpret_cmd


sub _log_dump {
    my($direction, $fh, $data, $offset, $len) = @_;
    my(
       $addr,
       $hexvals,
       $line,
       );

    $addr = 0;
    $len = length($$data) - $offset
	unless defined $len;

    ## Print data in dump format.
    while ($len > 0) {
	## Convert up to the next 16 chars to hex, padding w/ spaces.
	if ($len >= 16) {
	    $line = substr $$data, $offset, 16;
	}
	else {
	    $line = substr $$data, $offset, $len;
	}
	$hexvals = unpack('H*', $line);
	$hexvals .= ' ' x (32 - length $hexvals);

	## Place in 16 columns, each containing two hex digits.
	$hexvals = sprintf("%s %s %s %s  " x 4,
			   unpack('a2' x 16, $hexvals));

	## For the ASCII column, change unprintable chars to a period.
	$line =~ s/[\000-\037,\177-\237]/./g;

	## Print the line in dump format.
	printf $fh "%s 0x%5.5lx: %s%s\n", $direction, $addr, $hexvals, $line;

	$addr += 16;
	$offset += 16;
	$len -= 16;
    }

    print $fh "\n";

    1;
} # end sub _log_dump


sub _log_option {
    my($fh, $direction, $request, $option) = @_;
    my(
       $name,
       );

    if ($option >= 0 and $option <= $#Telopts) {
	$name = $Telopts[$option];
    }
    else {
	$name = $option;
    }

    print $fh "$direction $request $name\n";
} # end sub _log_option


sub _match_check {
    my($self, $code) = @_;
    my $error;
    my @warns = ();

    ## Use eval to check for syntax errors or warnings.
    {
	local $^W = 1;
	local $SIG{'__WARN__'} = sub { push @warns, @_ };
	local $SIG{'__DIE__'} = 'DEFAULT';
	local $_ = '';
	eval "\$_ =~ $code;";
    }
    if ($@) {
	## Remove useless lines numbers from message.
	($error = $@) =~ s/ at \(eval \d+\) line \d+.?//;
	chomp $error;
	return $self->error("bad match operator: $error");
    }
    elsif (@warns) {
	## Remove useless lines numbers from message.
	($error = shift @warns) =~ s/ at \(eval \d+\) line \d+.?//;
	$error =~ s/ while "strict subs" in use//;
	chomp $error;
	return $self->error("bad match operator: $error");
    }

    1;
} # end sub _match_check


sub _negotiate_callback {
    my($self, $opt, $is_remote, $is_enabled, $was_enabled, $opt_bufpos) = @_;
    my(
       $callback,
       $opt_state,
       $stream,
       );

    ## Keep track of remote echo.
    if ($is_remote and $opt == &TELOPT_ECHO) {  # received WILL or WONT ECHO
	$stream = $ {*$self}{net_telnet};

	if ($is_enabled and !$was_enabled) {  # received WILL ECHO
	    $stream->{remote_echo} = 1;
	}
	elsif (!$is_enabled and $was_enabled) {  # received WONT ECHO
	    $stream->{remote_echo} = '';
	}
    }

    ## Invoke callback, if there is one.
    $callback = $self->option_callback;
    if ($callback) {
	&$callback($self, $opt, $is_remote,
		   $is_enabled, $was_enabled, $opt_bufpos);
    }

    1;
} # end sub _negotiate_callback


sub _negotiate_recv {
    my($self, $s, $opt_request, $opt, $opt_bufpos) = @_;

    ## Ensure data structure exists for this option.
    unless (defined $s->{opts}{$opt}) {
	&_set_default_option($s, $opt);
    }

    ## Process the option.
    if ($opt_request eq "\376") {  # DONT
	&_negotiate_recv_disable($self, $s, $opt, "dont", $opt_bufpos,
				 $s->{opts}{$opt}{local_enable_ok},
				 \$s->{opts}{$opt}{local_enabled},
				 \$s->{opts}{$opt}{local_state});
    }
    elsif ($opt_request eq "\375") {  # DO
	&_negotiate_recv_enable($self, $s, $opt, "do", $opt_bufpos,
				$s->{opts}{$opt}{local_enable_ok},
				\$s->{opts}{$opt}{local_enabled},
				\$s->{opts}{$opt}{local_state});
    }
    elsif ($opt_request eq "\374") {  # WONT
	&_negotiate_recv_disable($self, $s, $opt, "wont", $opt_bufpos,
				 $s->{opts}{$opt}{remote_enable_ok},
				 \$s->{opts}{$opt}{remote_enabled},
				 \$s->{opts}{$opt}{remote_state});
    }
    elsif ($opt_request eq "\373") {  # WILL
	&_negotiate_recv_enable($self, $s, $opt, "will", $opt_bufpos,
				$s->{opts}{$opt}{remote_enable_ok},
				\$s->{opts}{$opt}{remote_enabled},
				\$s->{opts}{$opt}{remote_state});
    }
    else {  # internal error
	die;
    }

    1;
} # end sub _negotiate_recv


sub _negotiate_recv_disable {
    my($self, $s, $opt, $opt_request,
       $opt_bufpos, $enable_ok, $is_enabled, $state) = @_;
    my(
       $ack,
       $disable_cmd,
       $enable_cmd,
       $is_remote,
       $nak,
       $was_enabled,
       );

    ## What do we use to request enable/disable or respond with ack/nak.
    if ($opt_request eq "wont") {
	$enable_cmd  = "\377\375" . pack('C', $opt);  # do command
	$disable_cmd = "\377\376" . pack('C', $opt);  # dont command
	$is_remote = 1;
	$ack = "DO";
	$nak = "DONT";

	&_log_option($s->{opt_log}, "RCVD", "WONT", $opt)
	    if $s->{opt_log};
    }
    elsif ($opt_request eq "dont") {
	$enable_cmd  = "\377\373" . pack('C', $opt);  # will command
	$disable_cmd = "\377\374" . pack('C', $opt);  # wont command
	$is_remote = '';
	$ack = "WILL";
	$nak = "WONT";

	&_log_option($s->{opt_log}, "RCVD", "DONT", $opt)
	    if $s->{opt_log};
    }
    else {  # internal error
	die;
    }

    ## Respond to WONT or DONT based on the current negotiation state.
    if ($$state eq "no") {  # state is already disabled
    }
    elsif ($$state eq "yes") {  # they're initiating disable
	$$is_enabled = '';
	$$state = "no";

	## Send positive acknowledgment.
	$s->{unsent_opts} .= $disable_cmd;
	&_log_option($s->{opt_log}, "SENT", $nak, $opt)
	    if $s->{opt_log};

	## Invoke callbacks.
	&_negotiate_callback($self, $opt, $is_remote,
			     $$is_enabled, $was_enabled, $opt_bufpos);
    }
    elsif ($$state eq "wantno") {  # they sent positive ack
	$$is_enabled = '';
	$$state = "no";

	## Invoke callback.
	&_negotiate_callback($self, $opt, $is_remote,
			     $$is_enabled, $was_enabled, $opt_bufpos);
    }
    elsif ($$state eq "wantno opposite") {  # pos ack but we changed our mind
	## Indicate disabled but now we want to enable.
	$$is_enabled = '';
	$$state = "wantyes";

	## Send queued request.
	$s->{unsent_opts} .= $enable_cmd;
	&_log_option($s->{opt_log}, "SENT", $ack, $opt)
	    if $s->{opt_log};

	## Invoke callback.
	&_negotiate_callback($self, $opt, $is_remote,
			     $$is_enabled, $was_enabled, $opt_bufpos);
    }
    elsif ($$state eq "wantyes") {  # they sent negative ack
	$$is_enabled = '';
	$$state = "no";

	## Invoke callback.
	&_negotiate_callback($self, $opt, $is_remote,
			     $$is_enabled, $was_enabled, $opt_bufpos);
    }
    elsif ($$state eq "wantyes opposite") {  # nak but we changed our mind
	$$is_enabled = '';
	$$state = "no";

	## Invoke callback.
	&_negotiate_callback($self, $opt, $is_remote,
			     $$is_enabled, $was_enabled, $opt_bufpos);
    }
} # end sub _negotiate_recv_disable


sub _negotiate_recv_enable {
    my($self, $s, $opt, $opt_request,
       $opt_bufpos, $enable_ok, $is_enabled, $state) = @_;
    my(
       $ack,
       $disable_cmd,
       $enable_cmd,
       $is_remote,
       $nak,
       $was_enabled,
       );

    ## What we use to send enable/disable request or send ack/nak response.
    if ($opt_request eq "will") {
	$enable_cmd  = "\377\375" . pack('C', $opt);  # do command
	$disable_cmd = "\377\376" . pack('C', $opt);  # dont command
	$is_remote = 1;
	$ack = "DO";
	$nak = "DONT";

	&_log_option($s->{opt_log}, "RCVD", "WILL", $opt)
	    if $s->{opt_log};
    }
    elsif ($opt_request eq "do") {
	$enable_cmd  = "\377\373" . pack('C', $opt);  # will command
	$disable_cmd = "\377\374" . pack('C', $opt);  # wont command
	$is_remote = '';
	$ack = "WILL";
	$nak = "WONT";

	&_log_option($s->{opt_log}, "RCVD", "DO", $opt)
	    if $s->{opt_log};
    }
    else {  # internal error
	die;
    }

    ## Save current enabled state.
    $was_enabled = $$is_enabled;

    ## Respond to WILL or DO based on the current negotiation state.
    if ($$state eq "no") {  # they're initiating enable
	if ($enable_ok) {  # we agree they/us should enable
	    $$is_enabled = 1;
	    $$state = "yes";

	    ## Send positive acknowledgment.
	    $s->{unsent_opts} .= $enable_cmd;
	    &_log_option($s->{opt_log}, "SENT", $ack, $opt)
		if $s->{opt_log};

	    ## Invoke callbacks.
	    &_negotiate_callback($self, $opt, $is_remote,
				 $$is_enabled, $was_enabled, $opt_bufpos);
	}
	else {  # we disagree they/us should enable
	    ## Send negative acknowledgment.
	    $s->{unsent_opts} .= $disable_cmd;
	    &_log_option($s->{opt_log}, "SENT", $nak, $opt)
		if $s->{opt_log};
	}
    }
    elsif ($$state eq "yes") {  # state is already enabled
    }
    elsif ($$state eq "wantno") {  # error: our disable req answered by enable
	$$is_enabled = '';
	$$state = "no";

	## Invoke callbacks.
	&_negotiate_callback($self, $opt, $is_remote,
			     $$is_enabled, $was_enabled, $opt_bufpos);
    }
    elsif ($$state eq "wantno opposite") { # err: disable req answerd by enable
	$$is_enabled = 1;
	$$state = "yes";

	## Invoke callbacks.
	&_negotiate_callback($self, $opt, $is_remote,
			     $$is_enabled, $was_enabled, $opt_bufpos);
    }
    elsif ($$state eq "wantyes") {  # they sent pos ack
	$$is_enabled = 1;
	$$state = "yes";

	## Invoke callback.
	&_negotiate_callback($self, $opt, $is_remote,
			     $$is_enabled, $was_enabled, $opt_bufpos);
    }
    elsif ($$state eq "wantyes opposite") {  # pos ack but we changed our mind
	## Indicate enabled but now we want to disable.
	$$is_enabled = 1;
	$$state = "wantno";

	## Inform other side we changed our mind.
	$s->{unsent_opts} .= $disable_cmd;
	&_log_option($s->{opt_log}, "SENT", $nak, $opt)
	    if $s->{opt_log};

	## Invoke callback.
	&_negotiate_callback($self, $opt, $is_remote,
			     $$is_enabled, $was_enabled, $opt_bufpos);
    }

    1;
} # end sub _negotiate_recv_enable


sub _new_handle {
    if ($INC{'IO/Handle.pm'}) {
	return IO::Handle->new;
    }
    else {
	require FileHandle;
	return FileHandle->new;
    }
} # end sub _new_handle


sub _opt_accept {
    my($self, @args) = @_;
    my(
       $arg,
       $option,
       $stream,
       );

    ## Init vars.
    $stream = $ {*$self}{net_telnet};

    foreach $arg (@args) {
	## Ensure data structure defined for this option.
	$option = $arg->{option};
	if (!defined $stream->{opts}{$option}) {
	    &_set_default_option($stream, $option);
	}

	## Save whether we'll accept or reject this option.
	if ($arg->{is_remote}) {
	    $stream->{opts}{$option}{remote_enable_ok} = $arg->{is_enable};
	}
	else {
	    $stream->{opts}{$option}{local_enable_ok} = $arg->{is_enable};
	}
    }

    1;
} # end sub _opt_accept


sub _parse_timeout {
    my($timeout) = @_;

    ## Ensure valid timeout.
    if (defined $timeout) {
	## Test for non-numeric or negative values.
	eval {
	    local $^W = 1;
	    local $SIG{'__WARN__'} = sub { die "non-numeric\n" };
	    local $SIG{'__DIE__'} = 'DEFAULT';
	    $timeout *= 1;
	};
	if ($@) {  # timeout arg is non-numeric
	    $timeout = undef;
	}
	elsif ($timeout < 0) {
	    $timeout = undef;
	}
    }

    $timeout;
} # end sub _parse_timeout


sub _qualify_fh {
    my($obj, $name) = @_;
    my(
       $user_class,
       );
    local $_;

    ## Get user's package name.
    ($user_class) = &_user_caller($obj);

    ## Ensure name is qualified with a package name.
    $name = qualify($name, $user_class);

    ## If it's not already, make it a typeglob ref.
    if (! ref $name) {
	no strict;
	local $^W = 0;

	$name =~ s/^\*+//;
	$name = eval "\\*$name";
	return unless ref $name;
    }

    $name;
} # end sub _qualify_fh


sub _reset_options {
    my($opts) = @_;
    my(
       $opt,
       );

    foreach $opt (keys %$opts) {
	$opts->{$opt}{remote_enabled} = '';
	$opts->{$opt}{remote_state} = "no";
	$opts->{$opt}{local_enabled} = '';
	$opts->{$opt}{local_state} = "no";
    }

    1;
} # end sub _reset_options


sub _set_default_option {
    my($stream, $option) = @_;

    $stream->{opts}{$option} = {
	remote_enabled   => '',
	remote_state     => "no",
	remote_enable_ok => '',
	local_enabled    => '',
	local_state      => "no",
	local_enable_ok  => '',
    };
} # end sub _set_default_option


sub _timeout_interval {
    my($endtime) = @_;
    my(
       $timeout,
       );

    ## Return timed-out boolean and timeout interval.
    if (defined $endtime) {
	## Is it a one-time poll.
	return ('', 0) if $endtime == 0;

	## Calculate the timeout interval.
	$timeout = $endtime - time;

	## Did we already timeout.
	return (1, 0) unless $timeout > 0;

	return ('', $timeout);
    }
    else {  # there is no timeout
	return ('', undef);
    }
} # end sub _timeout_interval


sub _user_caller {
    my($obj) = @_;
    my(
       $class,
       $curr_pkg,
       $file,
       $i,
       $line,
       $pkg,
       $isa,
       %isa,
       @isa,
       );

    ## Create a boolean hash to test for isa.  Make sure current
    ## package and the object's class are members.
    $class = ref $obj;
    @isa = eval "\@${class}::ISA";
    push @isa, $class;
    ($curr_pkg) = caller 1;
    push @isa, $curr_pkg;
    %isa = map {$_ => 1} @isa;

    ## Search back in call frames for a package that's not in isa.
    $i = 1;
    while (($pkg, $file, $line) = caller ++$i) {
	next if $isa{$pkg};

	return($pkg, $file, $line);
    }

    ## If not found, choose outer most call frame.
    ($pkg, $file, $line) = caller --$i;
    return($pkg, $file, $line);
} # end sub _user_caller


sub _verify_telopt_arg {
    my($self, $option, $argname) = @_;

    ## If provided, use argument name in error message.
    if (defined $argname) {
	$argname = "for arg $argname";
    }
    else {
	$argname = "";
    }

    ## Ensure telnet option is a non-negative integer.
    eval {
	local $^W = 1;
	local $SIG{'__WARN__'} = sub { die "non-numeric\n" };
	local $SIG{'__DIE__'} = 'DEFAULT';
	$option = abs(int $option);
    };
    return $self->error("bad telnet option $argname: non-numeric")
	if $@;

    return $self->error("bad telnet option $argname: option > 255")
	unless $option <= 255;

    $option;
} # end sub _verify_telopt_arg


######################## Exported Constants ##########################


sub TELNET_IAC ()	    {255}; # interpret as command:
sub TELNET_DONT	()	    {254}; # you are not to use option
sub TELNET_DO ()	    {253}; # please, you use option
sub TELNET_WONT ()	    {252}; # I won't use option
sub TELNET_WILL ()	    {251}; # I will use option
sub TELNET_SB ()	    {250}; # interpret as subnegotiation
sub TELNET_GA ()	    {249}; # you may reverse the line
sub TELNET_EL ()	    {248}; # erase the current line
sub TELNET_EC ()	    {247}; # erase the current character
sub TELNET_AYT ()	    {246}; # are you there
sub TELNET_AO ()	    {245}; # abort output--but let prog finish
sub TELNET_IP ()	    {244}; # interrupt process--permanently
sub TELNET_BREAK ()	    {243}; # break
sub TELNET_DM ()	    {242}; # data mark--for connect. cleaning
sub TELNET_NOP ()	    {241}; # nop
sub TELNET_SE ()	    {240}; # end sub negotiation
sub TELNET_EOR ()	    {239}; # end of record (transparent mode)
sub TELNET_ABORT ()	    {238}; # Abort process
sub TELNET_SUSP ()	    {237}; # Suspend process
sub TELNET_EOF ()	    {236}; # End of file
sub TELNET_SYNCH ()	    {242}; # for telfunc calls

sub TELOPT_BINARY ()	      {0}; # Binary Transmission
sub TELOPT_ECHO ()	      {1}; # Echo
sub TELOPT_RCP ()	      {2}; # Reconnection
sub TELOPT_SGA ()	      {3}; # Suppress Go Ahead
sub TELOPT_NAMS ()	      {4}; # Approx Message Size Negotiation
sub TELOPT_STATUS ()	      {5}; # Status
sub TELOPT_TM ()	      {6}; # Timing Mark
sub TELOPT_RCTE ()	      {7}; # Remote Controlled Trans and Echo
sub TELOPT_NAOL ()	      {8}; # Output Line Width
sub TELOPT_NAOP ()	      {9}; # Output Page Size
sub TELOPT_NAOCRD ()	     {10}; # Output Carriage-Return Disposition
sub TELOPT_NAOHTS ()	     {11}; # Output Horizontal Tab Stops
sub TELOPT_NAOHTD ()	     {12}; # Output Horizontal Tab Disposition
sub TELOPT_NAOFFD ()	     {13}; # Output Formfeed Disposition
sub TELOPT_NAOVTS ()	     {14}; # Output Vertical Tabstops
sub TELOPT_NAOVTD ()	     {15}; # Output Vertical Tab Disposition
sub TELOPT_NAOLFD ()	     {16}; # Output Linefeed Disposition
sub TELOPT_XASCII ()	     {17}; # Extended ASCII
sub TELOPT_LOGOUT ()	     {18}; # Logout
sub TELOPT_BM ()	     {19}; # Byte Macro
sub TELOPT_DET ()	     {20}; # Data Entry Terminal
sub TELOPT_SUPDUP ()	     {21}; # SUPDUP
sub TELOPT_SUPDUPOUTPUT ()   {22}; # SUPDUP Output
sub TELOPT_SNDLOC ()	     {23}; # Send Location
sub TELOPT_TTYPE ()	     {24}; # Terminal Type
sub TELOPT_EOR ()	     {25}; # End of Record
sub TELOPT_TUID ()	     {26}; # TACACS User Identification
sub TELOPT_OUTMRK ()	     {27}; # Output Marking
sub TELOPT_TTYLOC ()	     {28}; # Terminal Location Number
sub TELOPT_3270REGIME ()     {29}; # Telnet 3270 Regime
sub TELOPT_X3PAD ()	     {30}; # X.3 PAD
sub TELOPT_NAWS ()	     {31}; # Negotiate About Window Size
sub TELOPT_TSPEED ()	     {32}; # Terminal Speed
sub TELOPT_LFLOW ()	     {33}; # Remote Flow Control
sub TELOPT_LINEMODE ()	     {34}; # Linemode
sub TELOPT_XDISPLOC ()	     {35}; # X Display Location
sub TELOPT_OLD_ENVIRON ()    {36}; # Environment Option
sub TELOPT_AUTHENTICATION () {37}; # Authentication Option
sub TELOPT_ENCRYPT ()	     {38}; # Encryption Option
sub TELOPT_NEW_ENVIRON ()    {39}; # New Environment Option
sub TELOPT_EXOPL ()	    {255}; # Extended-Options-List


1;
__END__;


######################## User Documentation ##########################


## To format the following user documentation into a more readable
## format, use one of these programs: pod2man; pod2html; pod2text.

=head1 NAME

Net::Telnet - interact with TELNET port or other TCP ports

=head1 SYNOPSIS

C<use Net::Telnet ();>

see METHODS section below

=head1 DESCRIPTION

Net::Telnet allows you to make client connections to a TCP port and do
network I/O, especially to a port using the TELNET protocol.  Simple
I/O methods such as print, get, and getline are provided.  More
sophisticated interactive features are provided because connecting to
a TELNET port ultimately means communicating with a program designed
for human interaction.  These interactive features include the ability
to specify a timeout and to wait for patterns to appear in the input
stream, such as the prompt from a shell.

Other reasons to use this module than strictly with a TELNET port are:

=over 2

=item *

You're not familiar with sockets and you want a simple way to make
client connections to TCP services.

=item *

You want to be able to specify your own time-out while connecting,
reading, or writing.

=item *

You're communicating with an interactive program at the other end of
some socket or pipe and you want to wait for certain patterns to
appear.

=back

Here's an example that prints who's logged-on to the remote host
sparky.  In addition to a username and password, you must also know
the user's shell prompt, which for this example is C<bash$>

    use Net::Telnet ();
    $t = new Net::Telnet (Timeout => 10,
                          Prompt => '/bash\$ $/');
    $t->open("sparky");
    $t->login($username, $passwd);
    @lines = $t->cmd("/usr/bin/who");
    print @lines;

More examples are in the B<EXAMPLES> section below.

=head2 What To Know Before Using

=over 2

=item *

All output is flushed while all input is buffered.  Each object
contains an input buffer.

=item *

The output record separator for C<print()> and C<cmd()> is set to
C<"\n"> by default, so that you don't have to append all your commands
with a newline.  See C<output_record_separator()> to change the
behavior.

=item *

The methods C<login()> and C<cmd()> use the I<prompt> setting in the
object to determine when a login or remote command is complete.  The
method will fail with a time-out if you don't set the prompt
correctly.

=item *

To avoid unexpected backslash interpretation, always use single quotes
instead of double quotes to construct a match operator argument for
C<prompt()> and C<waitfor()> (e.g. C<'/bash\$ $/'>).  If you're
constructing a DOS like file path, you'll need to use four backslashes
to represent one (e.g. C<'/c:\\\\users\\\\billE<gt>$/i'>).

Of course don't forget about regexp metacharacters like C<.>, C<[>, or
C<$>.  You'll only need a single backslash to quote them.  The anchor
metacharacters C<^> and C<$> refer to positions in the input buffer.

=item *

Errors such as timing-out are handled according to the I<errmode>
action.  The default action is to print an error message to standard
error and have the program die.  Use C<errmode()> to change the
action.

Using an I<errmode> of C<"return"> arranges for the error message to
be saved and the offending method to return with an undefined value.
You can obtain the error message using C<errmsg()>.

=item *

You'll need to be running at least Perl version 5.002 to use this
module.  This module does not require any libraries that don't already
come with a standard Perl distribution.

If you have the IO:: libraries installed then IO::Socket::INET is used
as a base class, otherwise FileHandle is used.  The IO:: libraries now
come standard starting with perl5.004.

=item *

In the input stream, each sequence of I<carriage return> and I<line
feed> (i.e. C<"\015\012"> or CR LF) is converted to C<"\n">.  In the
output stream, each occurrence of C<"\n"> is converted to a sequence
of CR LF.  See C<binmode()> to change the behavior.  TCP protocols
typically use the ASCII sequence, carriage return and line feed to
designate a newline.

=item *

Timing-out while making a connection is disabled for machines that
don't support the C<alarm()> function.  Most notably these include
Win32 machines.

More universal support for connection time-outs will be in a later
release.

=item *

This is an alpha version - meaning that the interface may change in
future versions.  Contact me, Jay Rogers <jay@rgrs.com>, if you find
any bugs or have suggestions for improvement.

=back

=head2 Debugging

The typical bug causes a timeout error because you've made incorrect
assumptions about what the remote side actually sends.  The easiest
way to reconcile what the remote side sends with your expectations is
to use C<input_log()> or C<dump_log()>.

C<dump_log()> allows you to see the data being sent from the remote
side before any translation is done, while C<input_log()> shows you
the results after translation.  The translation includes converting
end of line characters and stripping and responding to TELNET protocol
commands.

=head2 Style of Named Parameters

Two different styles of named parameters are supported.  This document
only shows the IO:: style:

    Net::Telnet->new(Timeout => 20);

however the dash-option style is also allowed:

    Net::Telnet->new(-timeout => 20);

=head2 Connecting to a Remote Win32 Machine

By default, Windows machines don't come with a TELNET service.
However, third party TELNET servers can be acquired to provide access
to a DOS shell.  As is all too familiar, many of these servers are of
inferior quality.

One particularly nasty problem is a server that sends ANSI terminal
escape sequences despite being told not to.  These escape sequences
make your job of parsing the data much more difficult.  You can
sometimes avoid this with a server that prompts you for I<console
mode>.  Choose I<no> to console mode.


=head1 METHODS


=over 4

=item B<new> - create a new Net::Telnet object

    $obj = Net::Telnet->new([Binmode    => $mode,]
                            [Cmd_remove_mode => $mode,]
                            [Dump_Log   => $filename,]
                            [Errmode    => $errmode,]
                            [Fhopen     => $filehandle,]
                            [Host       => $host,]
                            [Input_log  => $file,]
                            [Input_record_separator => $char,]
                            [Option_log => $file,]
                            [Output_log => $file,]
                            [Output_record_separator => $char,]
                            [Port       => $port,]
                            [Prompt     => $matchop,]
                            [Telnetmode => $mode,]
                            [Timeout    => $secs,]);

This is the constructor for Net::Telnet objects.  A new object is
returned on success, the I<$errmode> action is performed on failure -
see C<errmode()>.  The arguments are short-cuts to methods of the same
name.

If the I<$host> argument is given then the object is opened by
connecting to TCP I<$port> on I<$host>.  Also see C<open()>.  The new
object returned is given the following defaults in the absence of
corresponding named arguments:

=over 4

=item

The default I<Host> is C<"localhost">

=item

The default I<Port> is C<23>

=item

The default I<Prompt> is C<'/[\$%#E<gt>] $/'>

=item

The default I<Timeout> is C<10>

=item

The default I<Errmode> is C<"die">

=item

The default I<Output_record_separator> is C<"\n">

=item

The default I<Input_record_separator> is C<"\n">

=item

The default I<Binmode> is C<0>, which means do newline translation

=item

The default I<Cmd_remove_mode> is C<"auto">.

=back

=back


=over 4

=item B<binmode> - toggle newline translation

    $mode = $obj->binmode;

    $prev = $obj->binmode($mode);

This method controls whether or not sequences of carriage returns and
line feeds (CR LF or more specifically C<"\015\012">) are translated.
By default they are translated (i.e. binmode is C<0>).

If no argument is given, the current mode is returned.

If I<$mode> is C<1> then binmode is I<on> and newline translation is
not done.

If I<$mode> is C<0> then binmode is I<off> and newline translation is
done.  In the input stream, each sequence of CR LF is converted to
C<"\n"> and in the output stream, each occurrence of C<"\n"> is
converted to a sequence of CR LF.

Note that input is always buffered.  Changing binmode doesn't effect
what's already been read into the buffer.  Output is not buffered and
changing binmode will have an immediate effect.

=back


=over 4

=item B<break> - send TELNET break character

    $ok = $obj->break;

This method sends the TELNET break character.  This character is
provided because it's a signal outside the USASCII character set which
is currently given local meaning within many systems.  It's intended
to indicate that the Break Key or the Attention Key was hit.

=back


=over 4

=item B<buffer> - scalar reference to object's input buffer

    $ref = $obj->buffer;

This method returns a scalar reference to the input buffer for
I<$obj>.  Data in the input buffer is data that has been read from the
remote side but has yet to be read by the user.  Modifications to the
input buffer are returned by a subsequent read.

=back


=over 4

=item B<buffer_empty> - discard all data in object's input buffer

    $obj->buffer_empty;

This method removes all data in the input buffer for I<$obj>.

=back


=over 4

=item B<close> - close object

    $ok = $obj->close;

This method closes the socket, file, or pipe associated with the
object.

=back


=over 4

=item B<cmd> - issue command and retrieve output

    $ok = $obj->cmd($string);
    $ok = $obj->cmd(String   => $string,
                    [Output  => $ref,]
                    [Prompt  => $match,]
                    [Timeout => $secs,]
                    [Cmd_remove_mode => $mode,]);

    @output = $obj->cmd($string);
    @output = $obj->cmd(String   => $string,
                        [Output  => $ref,]
                        [Prompt  => $match,]
                        [Timeout => $secs,]
                        [Cmd_remove_mode => $mode,]);

This method sends the command I<$string>, and reads the characters
sent back by the command up until and including the matching prompt.
It's assumed that the program to which you're sending is some kind of
command prompting interpreter such as a shell.

In a scalar context the characters read are discarded and a boolean is
returned indicating the success or failure of sending the command
string and reading the prompt.  Note that in order to return on error,
C<errmode()> must not be set to C<"die">.

In an array context, just the output generated by the command is
returned, one line per element.  In other words, all the characters in
between the echoed back command string and the prompt are returned.
If the command happens to return no output, an array containing one
element, the null string is returned.  This is so the array will
indicate true in a boolean context.

Many command interpreters echo back the command sent.  In most
situations, this method removes the first line returned from the
remote side (i.e. the echoed back command).  See C<cmd_remove_mode()>
for more control over this feature.

Use C<dump_log()> to debug when this method keeps timing-out and you
don't think it should.

Optional named arguments are provided to override the current settings
of prompt, timeout, and cmd_remove_mode.

The I<Output> named argument provides an alternative method of
receiving command output.  If you pass a scalar reference, all the
output (even if it contains multiple lines) is returned in the
referenced scalar.  If you pass an array or hash reference, the lines
of output are returned in the referenced array or hash.

=back


=over 4

=item B<cmd_remove_mode> - toggle removal of echoed commands

    $mode = $obj->cmd_remove_mode;

    $prev = $obj->cmd_remove_mode($mode);

This method controls how to deal with echoed back commands in the
output returned by cmd().  Typically, when you send a command to the
remote side, the first line of output returned is the command echoed
back.  Use this mode to remove the first line of output normally
returned by cmd().

If no argument is given, the current mode is returned.

If I<$mode> is C<0> then the command output returned from cmd() has no
lines removed.  If I<$mode> is a positive integer, then the first
I<$mode> lines of command output are stripped.

By default, I<$mode> is set to C<"auto">.  Auto means that whether or
not the first line of command output is stripped, depends on whether
or not the remote side offered to echo.  By default, Net::Telnet
always accepts an offer to echo by the remote side.  You can change
the default to reject such an offer using C<option_accept()>.

=back


=over 4

=item B<dump_log> - log all I/O in dump format

    $fh = $obj->dump_log;

    $fh = $obj->dump_log($fh);

    $fh = $obj->dump_log($filename);

This method starts or stops dump format logging of all the object's
input and output.  The dump format shows the blocks read and written
in a hexadecimal and printable character format.  This method is
useful when debugging, however you might want to first try
C<input_log()> as it's more readable.

If no argument is given, the current log filehandle is returned.  A
null string indicates logging is off.

To stop logging, use a null string as an argument.

If an open filehandle is given, it is used for logging and returned.
Otherwise, the argument is assumed to be the name of a file, the file
is opened and a filehandle to it is returned.

=back


=over 4

=item B<eof> - end of file indicator

    $eof = $obj->eof;

This method indicates if end of file has been read.  Because the input
is buffered this isn't the same thing as I<$obj> has closed.  In other
words I<$obj> can be closed but there still can be stuff in the buffer
to be read.  Under this condition you can still read but you won't be
able to write.

=back


=over 4

=item B<errmode> -  define action to be performed on error

    $mode = $obj->errmode;

    $prev = $obj->errmode($mode);

This method gets or sets the action used when errors are encountered
using the object.  The first calling sequence returns the current
error mode.  The second calling sequence sets it to I<$mode> and
returns the previous mode.  Valid values for I<$mode> are C<"die">
(the default), C<"return">, a I<coderef>, or an I<arrayref>.

When mode is C<"die"> and an error is encountered using the object,
then an error message is printed to standard error and the program
dies.

When mode is C<"return"> then the method generating the error places
an error message in the object and returns the undefined value in a
scalar context and a null list in list context.  The error message may
be obtained using C<errmsg()>.

When mode is a I<coderef>, then when an error is encountered
I<coderef> is called with the error message as its first argument.
Using this mode you may have your own subroutine handle errors.  If
I<coderef> itself returns then the method generating the error returns
undefined or a null list depending on context.

When mode is an I<arrayref>, the first element of the array must be a
I<coderef>.  Any elements that follow are the arguments to I<coderef>.
When an error is encountered, the I<coderef> is called with its
arguments.  Using this mode you may have your own subroutine handle
errors.  If the I<coderef> itself returns then the method generating
the error returns undefined or a null list depending on context.

=back


=over 4

=item B<errmsg> - most recent error message

    $msg = $obj->errmsg;

    $prev = $obj->errmsg(@msgs);

The first calling sequence returns the error message associated with
the object.  The null string is returned if no error has been
encountered yet.  The second calling sequence sets the error message
for the object to the concatenation of I<@msgs> and returns the
previous error message.  Normally, error messages are set internally
by a method when an error is encountered.

=back


=over 4

=item B<error> - perform the error mode action

    $obj->error(@msgs);

This method concatenates I<@msgs> into a string and places it in the
object as the error message.  Also see C<errmsg()>.  It then performs
the error mode.  Also see C<errmode()>.

If the error mode doesn't cause the program to die then the undefined
value or a null list is returned depending on context.

This method is primarily used by this class or a sub-class to perform
the user requested action when an error is encountered.

=back


=over 4

=item B<fhopen> - use already open filehandle for I/O

    $ok = $obj->fhopen($fh);

This method associates the open filehandle I<$fh> with I<$obj> for
further I/O.  Filehandle I<$fh> must already be opened.

Suppose you want to use the features of this module to do I/O to
something other than a TCP port, for example STDIN or a filehandle
opened to read from a process.  Instead of opening the object for I/O
to a TCP port by using C<open()> or C<new()>, call this method
instead.

=back


=over 4

=item B<get> - read block of data

    $data = $obj->get([Timeout => $secs,]);

This method reads a block of data from the object and returns it along
with any buffered data.  If no buffered data is available to return,
it will wait for data to read using the timeout specified in the
object.  You can override that timeout using I<$secs>.  Also see
C<timeout()>.  If buffered data is available to return, it also checks
for a block of data that can be immediately read.

On eof an undefined value is returned.  On time-out or other errors
the error mode action is performed.

An undefined value is returned for both eof and time-out when
I<errmode> is not set to C<"die">.  Use C<eof()> and C<timed_out()> to
distinguish.

=back


=over 4

=item B<getline> - read next line

    $line = $obj->getline([Timeout => $secs,]);

This method reads and returns the next line of data from the object.
You can use C<input_record_separator()> to change the notion of what
separates a line.  The default is C<"\n">.

If a line isn't immediately available, this method blocks waiting for
a line or the timeout.  You can override the object's timeout for this
method using I<$secs>.  Also see C<timeout()>.

On eof an undefined value is returned.  On timeout or other errors the
error mode action is performed.

An undefined value is returned for both eof and time-out when
I<errmode> is not set to C<"die">.  Use C<eof()> and C<timed_out()> to
distinguish.

=back


=over 4

=item B<getlines> - read next lines

    @lines = $obj->getlines([Timeout => $secs,]);

This method reads and returns the next available lines of data from
the object.  You can use C<input_record_separator()> to change the
notion of what separates a line.  The default is C<"\n">.

If a line isn't immediately available, this method blocks waiting for
one or more lines, or the timeout.  You can override the object's
timeout for this method using I<$secs>.  Also see C<timeout()>.

On eof a null array is returned.  On timeout or other errors the error
mode action is performed.

A null array is returned for both eof and time-out when I<errmode> is
not set to C<"die">.  Use C<eof()> and C<timed_out()> to distinguish.

=back


=over 4

=item B<host> - name of remote host

    $host = $obj->host;

    $prev = $obj->host($host);

This method designates the remote host.  With no argument this method
returns the current host name set in the object.  With an argument it
sets the current host name to I<$host> and returns the previous host
name.  You may indicate the remote host using either a hostname or an
IP address.

=back


=over 4

=item B<input_log> - log all input

    $fh = $obj->input_log;

    $fh = $obj->input_log($fh);

    $fh = $obj->input_log($filename);

This method starts or stops logging of input.  This is useful when
debugging.  Also see C<dump_log()>.  Because most command interpreters
echo back commands received, its likely all your output will also be
in this log.  Note that input logging occurs after newline
translation.  See C<binmode()> for details on newline translation.

If no argument is given, the log filehandle is returned.  A null
string indicates logging is off.

To stop logging, use a null string as an argument.

If an open filehandle is given, it is used for logging and returned.
Otherwise, the argument is assumed to be the name of a file, the file
is opened for logging and a filehandle to it is returned.

=back


=over 4

=item B<input_record_separator> - input line delimiter

    $rs = $obj->input_record_separator;

    $prev = $obj->input_record_separator($rs);

This method designates the line delimiter for input.  It's used with
C<getline()>, C<getlines()>, and C<cmd()> to determine lines in the
input.

With no argument this method returns the current input record
separator set in the object.  With an argument it sets the input
record separator to I<$rs> and returns the previous value.

=back


=over 4

=item B<lastline> - last line read

    $line = $obj->lastline;

    $prev = $obj->lastline($line);

This method saves the last line read from the object.  This may be a
useful error message when the remote side abnormally closes the
connection.  Typically the remote side will print an error message
before closing.

With no argument this method returns the last line read from the
object.  With an argument it sets the last line read to I<$line> and
returns the previous value.  Normally, only internal methods set the
last line.

=back


=over 4

=item B<login> - perform standard login

    $ok = $obj->login($username, $password);

    $ok = $obj->login(Name     => $username,
                      Password => $password,
                      [Prompt  => $match,]
                      [Timeout => $secs,]);

This method performs a standard login by waiting for a login prompt
and responding with I<$username>, then waiting for the password prompt
and responding with I<$password>, and then waiting for the command
interpreter prompt.  If any of those prompts sent by the remote side
don't match what's expected, this method will timeout - unless timeout
is turned off.

Login prompts must match either of the patterns:

    /login[: ]*$/i
    /username[: ]*$/i

Password prompts must match the pattern:

    /password[: ]*$/i

The command interpreter prompt must match the current setting of
prompt.

Use C<dump_log()> to debug when this method keeps timing-out and you
don't think it should.

Optional named arguments are provided to override the current settings
of prompt and timeout.

=back


=over 4

=item B<max_buffer_length> - maximum size of input buffer

    $len = $obj->max_buffer_length;

    $prev = $obj->max_buffer_length($len);

This method designates the maximum size of the input buffer.  An error
is generated when a read causes the buffer to exceed this limit.  The
default value is 1,048,576 bytes (1MB).  The input buffer can grow
much larger than the block size when you continuously read using
C<getline()> or C<waitfor()> and the data stream contains no newlines
or matching waitfor patterns.

With no argument this method returns the current maximum buffer length
set in the object.  With an argument it sets the maximum buffer length
to I<$len> and returns the previous value.

=back


=over 4

=item B<open> - connect to port on remote host

    $ok = $obj->open($host);

    $ok = $obj->open([Host    => $host,]
                     [Port    => $port,]
                     [Timeout => $secs,]);

This method opens a TCP connection to I<$port> on I<$host>.  If either
argument is missing then the current value of C<host()> or C<port()>
is used.  An optional named argument is provided to override the
current setting of timeout.

On time-out or other connection errors, the error mode action is
performed.

Timeouts don't work for this method on machines that don't implement
SIGALRM - most notably Win32 machines.  For those machines, an error
is returned when the system reaches its own time-out while trying to
connect.

A side effect of this method is to reset the alarm interval associated
with SIGALRM.

=back


=over 4

=item B<option_accept> - indicate willingness to accept a TELNET option

    $fh = $obj->option_accept([Do   => $telopt,]
                              [Dont => $telopt,]
                              [Will => $telopt,]
                              [Wont => $telopt,]);

This method is used to indicate whether to accept or reject an offer
to enable a TELNET option made by the remote side.  If you're using
I<Do> or I<Will> to indicate a willingness to enable, then a
notification callback must have already been defined by a prior call
to C<option_callback()>.  See C<option_callback()> for details on
receiving enable/disable notification of a TELNET option.

You can give multiple I<Do>, I<Dont>, I<Will>, or I<Wont> arguments
for different TELNET options in the same call to this method.

The following example describes the meaning of the named arguments.  A
TELNET option, such as C<TELOPT_ECHO> used below, is an integer
constant that you can import from Net::Telnet.  See the source in file
Telnet.pm for the complete list.

=over 4

=item

I<Do> => C<TELOPT_ECHO>

=over 4

=item

we'll accept an offer to enable the echo option on the local side

=back

=item

I<Dont> => C<TELOPT_ECHO>

=over 4

=item

we'll reject an offer to enable the echo option on the local side

=back

=item

I<Will> => C<TELOPT_ECHO>

=over 4

=item

we'll accept an offer to enable the echo option on the remote side

=back

=item

I<Wont> => C<TELOPT_ECHO>

=over 4

=item

we'll reject an offer to enable the echo option on the remote side

=back

=back

=item

Use C<option_send()> to send a request to the remote side to enable or
disable a particular TELNET option.

=back


=over 4

=item B<option_callback> - define the option negotiation callback

    $coderef = $obj->option_callback;

    $prev = $obj->option_callback($coderef);

This method defines the callback subroutine that's called when a
TELNET option is enabled or disabled.  Once defined, the
I<option_callback> may not be undefined.  However, calling this method
with a different I<$coderef> changes it.

Here are the circumstances that invoke I<$coderef>:

=over 4

=item

An option becomes enabled because the remote side requested an enable
and C<option_accept()> had been used to arrange that it be accepted.

=item

The remote side arbitrarily decides to disable an option that is
currently enabled.  Note that Net::Telnet always accepts a request to
disable from the remote side.

=item

C<option_send()> was used to send a request to enable or disable an
option and the response from the remote side has just been received.
Note, that if a request to enable is rejected then I<$coderef> is
still invoked even though the option didn't change.

=back

=item

Here are the arguments passed to I<&$coderef>:

    &$coderef($obj, $option, $is_remote,
              $is_enabled, $was_enabled, $buf_position);

=over 4

=item

1.  I<$obj> is the Net::Telnet object

=item

2.  I<$option> is the TELNET option.  Net::Telnet exports constants
for the various TELNET options which just equate to an integer.

=item

3.  I<$is_remote> is a boolean indicating for which side the option
applies.

=item

4.  I<$is_enabled> is a boolean indicating the option is enabled or
disabled

=item

5.  I<$was_enabled> is a boolean indicating the option was previously
enabled or disabled

=item

6.  I<$buf_position> is an integer indicating the position in the
object's input buffer where the option takes effect.  See C<buffer()>
to access the object's input buffer.

=back

=back


=over 4

=item B<option_log> - log all TELNET options sent or received

    $fh = $obj->option_log;

    $fh = $obj->option_log($fh);

    $fh = $obj->option_log($filename);

This method starts or stops logging of all TELNET options being sent
or received.  This is useful for debugging when you send options via
C<option_send()> or you arrange to accept option requests from the
remote side via C<option_accept()>.  Also see C<dump_log()>.

If no argument is given, the log filehandle is returned.  A null
string indicates logging is off.

To stop logging, use a null string as an argument.

If an open filehandle is given, it is used for logging and returned.
Otherwise, the argument is assumed to be the name of a file, the file
is opened for logging and a filehandle to it is returned.

=back


=over 4

=item B<option_send> - send TELNET option negotiation request

    $ok = $obj->option_send([Do    => $telopt,]
                            [Dont  => $telopt,]
                            [Will  => $telopt,]
                            [Wont  => $telopt,]
                            [Async => $boolean,]);

This method is not yet implemented.  Look for it in a future version.

=back


=over 4

=item B<option_state> - get current state of a TELNET option

    $hashref = $obj->option_state($telopt);

This method returns a hashref containing a copy of the current state
of TELNET option I<$telopt>.

Here are the values returned in the hash:

=over 4

=item

I<$hashref>->{remote_enabled}

=over 4

=item

boolean that indicates if the option is enabled on the remote side.

=back

=item

I<$hashref>->{remote_enable_ok}

=over 4

=item

boolean that indicates if it's ok to accept an offer to enable this
option on the remote side.

=back

=item

I<$hashref>->{remote_state}

=over 4

=item

string used to hold the internal state of option negotiation for this
option on the remote side.

=back

=item

I<$hashref>->{local_enabled}

=over 4

=item

boolean that indicates if the option is enabled on the local side.

=back

=item

I<$hashref>->{local_enable_ok}

=over 4

=item

boolean that indicates if it's ok to accept an offer to enable this
option on the local side.

=back

=item

I<$hashref>->{local_state}

=over 4

=item

string used to hold the internal state of option negotiation for this
option on the local side.

=back

=back

=back


=over 4

=item B<output_field_separator> - field separator for print

    $ofs = $obj->output_field_separator;

    $prev = $obj->output_field_separator($ofs);

This method designates the output field separator for C<print()>.
Ordinarily the print method simply prints out the comma separated
fields you specify.  Set this to specify what's printed between
fields.

With no argument this method returns the current output field
separator set in the object.  With an argument it sets the output
field separator to I<$ofs> and returns the previous value.

=back


=over 4

=item B<output_log> - log all output

    $fh = $obj->output_log;

    $fh = $obj->output_log($fh);

    $fh = $obj->output_log($filename);

This method starts or stops logging of output.  This is useful when
debugging.  Also see C<dump_log()>.  Because most command interpreters
echo back commands received, its likely all your output would also be
in an input log.  See C<input_log()>.  Note that output logging occurs
before newline translation.  See C<binmode()> for details on newline
translation.

If no argument is given, the log filehandle is returned.  A null
string indicates logging is off.

To stop logging, use a null string as an argument.

If an open filehandle is given, it is used for logging and returned.
Otherwise, the argument is assumed to be the name of a file, the file
is opened for logging and a filehandle to it is returned.

=back


=over 4

=item B<output_record_separator> - output line delimiter

    $ors = $obj->output_record_separator;

    $prev = $obj->output_record_separator($ors);

This method designates the output record separator for C<print()>.
Ordinarily the print operator simply prints out the comma separated
fields you specify, with no trailing newline or record separator
assumed.  Set this variable to specify what's printed at the end of
the print.

Note: the output record separator is set to C<"\n"> by default, so
there's no need to append all your commands with a newline.

With no argument this method returns the current output record
separator set in the object.  With an argument it sets the output
record separator to I<$ors> and returns the previous value.

=back


=over 4

=item B<port> - remote port

    $port = $obj->port;

    $prev = $obj->port($port);

This method designates the remote TCP port.  With no argument this
method returns the current port number.  With an argument it sets the
current port number to I<$port> and returns the previous port.  If
I<$port> is a service name, then first it's converted to a port number
using the perl function C<getservbyname()>.

=back


=over 4

=item B<print> - write to object

    $ok = $obj->print(@list);

This method prints a string or a comma-separated list of strings to
the opened object and returns non-zero if all data was successfully
written.

By default, the C<output_record_separator()> is set to C<"\n"> in order
to have your commands automatically end with a newline.  In most cases
your output is being read by a command interpreter which won't accept
a command until newline is read.  This is similar to someone typing a
command and hitting the return key.

On failure, it's possible that some data was written.  If you choose
to try and recover from a print timing-out, use C<print_length()> to
determine how much was written before time-out occurred.

=back


=over 4

=item B<print_length> - number of bytes written by print

    $num = $obj->print_length;

This returns the number of bytes successfully written by the most
recent C<print()>.

=back


=over 4

=item B<prompt> - pattern to match a prompt

    $matchop = $obj->prompt;

    $prev = $obj->prompt($matchop);

This method sets the pattern used to find a prompt in the input
stream.  It must be a string representing a valid perl pattern match
operator.  The methods C<login()> and C<cmd()> try to read until
matching the prompt.  They will fail with a time-out error if the
pattern you've chosen doesn't match what the remote side sends.

With no argument this method returns the prompt set in the object.
With an argument it sets the prompt to I<$matchop> and returns the
previous value.

The default prompt is C<'/[\$%#E<gt>] $/'>

Always use single quotes, instead of double quotes, to construct
I<$matchop> (e.g. C<'/bash\$ $/'>).  If you're constructing a DOS like
file path, you'll need to use four backslashes to represent one
(e.g. C<'/c:\\\\users\\\\billE<gt>$/i'>).

Of course don't forget about regexp metacharacters like C<.>, C<[>, or
C<$>.  You'll only need a single backslash to quote them.  The anchor
metacharacters C<^> and C<$> refer to positions in the input buffer.

=back


=over 4

=item B<telnetmode> - turn off/on telnet command interpretation

    $mode = $obj->telnetmode;

    $prev = $obj->telnetmode($mode);

This method controls whether or not TELNET commands in the data stream
are recognized and handled.  The TELNET protocol uses certain
character sequences sent in the data stream to control the session.
If the port you're connecting to isn't using the TELNET protocol, then
you should turn this mode off.  The default is I<on>.

If no argument is given, the current mode is returned.

If I<$mode> is C<0> then telnet mode is off.  If I<$mode> is C<1> then
telnet mode is on.

=back


=over 4

=item B<timed_out> - time-out indicator

    $boolean = $obj->timed_out;

    $prev = $obj->timed_out($boolean);

This method indicates if a previous read or write method timed-out.

With no argument this method returns true if a previous method
timed-out.  With an argument it sets the indicator.  Normally, only
internal methods set this indicator.

=back


=over 4

=item B<timeout> - I/O time-out interval

    $secs = $obj->timeout;

    $prev = $obj->timeout($secs);

This method sets the timeout interval that's used when performing I/O
or connecting to a port.  When a method doesn't complete within the
timeout interval then it's an error and the error mode action is
performed.

The timeout may be expressed as a relative or absolute value.  If
I<$secs> is greater than or equal to the time the program was started,
as determined by $^T, then it's the absolute time when timeout occurs.
Also see the perl function C<time()>.  A relative timeout happens
I<$secs> from when the I/O method begins.

If I<$secs> is C<0> then timeout occurs if the data cannot be
immediately read or written.  Use the undefined value to turn off
timing-out.

With no argument this method returns the timeout set in the object.
With an argument it sets the timeout to I<$secs> and returns the
previous value.

=back


=over 4

=item B<waitfor> - wait for pattern in the input

    $ok = $obj->waitfor($matchop);
    $ok = $obj->waitfor([Match   => $matchop,]
                        [String  => $string,]
                        [Timeout => $secs,]);

    ($prematch, $match) = $obj->waitfor($matchop);
    ($prematch, $match) = $obj->waitfor([Match   => $matchop,]
                                        [String  => $string,]
                                        [Timeout => $secs,]);

This method reads until a pattern match or string is found in the
input stream.  All the characters before and including the match are
removed from the input stream.  On time-out, eof, or other errors the
error mode action is performed.

In an array context the characters before the match and the matched
characters are returned in I<$prematch> and I<$match>.

You can specify more than one pattern or string by simply providing
multiple I<Match> and/or I<String> named arguments.  A I<$matchop>
must be a string representing a valid Perl pattern match operator.
The I<$string> is just a substring to find in the input stream.

Use C<dump_log()> to debug when this method keeps timing-out and you
don't think it should.

An optional named argument is provided to override the current setting
of timeout.

To avoid unexpected backslash interpretation, always use single quotes
instead of double quotes to construct a match operator argument for
C<prompt()> and C<waitfor()> (e.g. C<'/bash\$ $/'>).  If you're
constructing a DOS like file path, you'll need to use four backslashes
to represent one (e.g. C<'/c:\\\\users\\\\billE<gt>$/i'>).

Of course don't forget about regexp metacharacters like C<.>, C<[>, or
C<$>.  You'll only need a single backslash to quote them.  The anchor
metacharacters C<^> and C<$> refer to positions in the input buffer.

=back


=head1 SEE ALSO

=over 2

=item RFC 854

S<TELNET Protocol Specification>

S<ftp://ftp.isi.edu/in-notes/rfc854.txt>

=item RFC 1143

S<Q Method of Implementing TELNET Option Negotiation>

S<ftp://ftp.isi.edu/in-notes/rfc1143.txt>

=item TELNET Option Assignments

S<ftp://ftp.isi.edu/in-notes/iana/assignments/telnet-options>

=back


=head1 EXAMPLES

This example gets the current weather forecast for Brainerd, Minnesota.

    my($forecast, $t);

    use Net::Telnet ();
    $t = new Net::Telnet;
    $t->open("rainmaker.wunderground.com");

    ## Wait for first prompt and "hit return".
    $t->waitfor('/continue:.*$/');
    $t->print("");

    ## Wait for second prompt and respond with city code.
    $t->waitfor('/city code.*$/');
    $t->print("BRD");

    ## Read and print the first page of forecast.
    ($forecast) = $t->waitfor('/[ \t]+press return to continue/i');
    print $forecast;

    exit;


This example checks a POP server to see if you have mail.

    my($hostname, $line, $passwd, $pop, $username);

    $hostname = "your_destination_host_here";
    $username = "your_username_here";
    $passwd = "your_password_here";

    use Net::Telnet ();
    $pop = new Net::Telnet (Telnetmode => 0);
    $pop->open(Host => $hostname,
               Port => 110);


    ## Read connection message.
    $line = $pop->getline;
    die $line unless $line =~ /^\+OK/;

    ## Send user name.
    $pop->print("user $username");
    $line = $pop->getline;
    die $line unless $line =~ /^\+OK/;

    ## Send password.
    $pop->print("pass $passwd");
    $line = $pop->getline;
    die $line unless $line =~ /^\+OK/;

    ## Request status of messages.
    $pop->print("list");
    $line = $pop->getline;
    print $line;

    exit;


Here's an example you can use to down load a file of any type.  The
file is read from the remote host's standard output using cat.  To
prevent any output processing, the remote host's standard output is
put in raw mode using the Bourne shell.  The Bourne shell is used
because some shells, notably tcsh, prevent changing tty modes.  Upon
completion, FTP style statistics are printed to stderr.

    my($block, $filename, $host, $hostname, $k_per_sec, $line,
       $num_read, $passwd, $prevblock, $prompt, $size, $size_bsd,
       $size_sysv, $start_time, $total_time, $username);

    $hostname = "your_destination_host_here";
    $username = "your_username_here";
    $passwd = "your_password_here";
    $filename = "your_download_file_here";

    ## Connect and login.
    use Net::Telnet ();
    $host = new Net::Telnet (Timeout => 30,
                             Prompt => '/[%#>] $/');
    $host->open($hostname);
    $host->login($username, $passwd);

    ## Make sure prompt won't match anything in send data.
    $prompt = '_funkyPrompt_';
    $host->prompt("/$prompt\$/");
    $host->cmd("set prompt = '$prompt'");

    ## Get size of file.
    ($line) = $host->cmd("/usr/bin/ls -l $filename");
    ($size_bsd, $size_sysv) = (split ' ', $line)[3,4];
    if ($size_sysv =~ /^\d+$/) {
        $size = $size_sysv;
    }
    elsif ($size_bsd =~ /^\d+$/) {
        $size = $size_bsd;
    }
    else {
        die "$filename: no such file on $hostname";
    }

    ## Start sending the file.
    binmode STDOUT;
    $host->binmode(1);
    $host->print("/usr/bin/sh -c 'stty raw; cat $filename'");
    $host->getline;    # discard echoed back line

    ## Read file a block at a time.
    $num_read = 0;
    $prevblock = '';
    $start_time = time;
    while (($block = $host->get) and ($block !~ /$prompt$/o)) {
        if (length $block >= length $prompt) {
            print $prevblock;
            $num_read += length $prevblock;
            $prevblock = $block;
        }
        else {
            $prevblock .= $block;
        }

    }
    $host->close;

    ## Print last block without trailing prompt.
    $prevblock .= $block;
    $prevblock =~ s/$prompt$//;
    print $prevblock;
    $num_read += length $prevblock;
    die "error: expected size $size, received size $num_read\n"
        unless $num_read == $size;

    ## Print totals.
    $total_time = (time - $start_time) || 1;
    $k_per_sec = ($size / 1024) / $total_time;
    $k_per_sec = sprintf "%3.1f", $k_per_sec;
    warn("$num_read bytes received in $total_time seconds ",
         "($k_per_sec Kbytes/s)\n");

    exit;


Here's an example that shows how to talk to a program that
must communicate via a terminal.  In this case we're talking
to the telnet program via a pseudo-terminal.  We use the
Comm package to start the telnet program and return a
filehandle to the pseudo-terminal.  This example sends some
initial commands and then allows the user to type commands
to the telnet session.

    my($comm_pty, $host, $hostname, $passwd, $pty,
       $username, @lines);

    $hostname = "your_host_here";
    $username = "your_name_here";
    $passwd = "your_passwd_here";

    ## Start the telnet program so we can talk to it via a
    ## pseudo-terminal.
    {
        local $^W = 0;  # Comm.pl isn't warning clean

        require "Comm.pl";
        &Comm::init("close_it", "interact",
                    "open_proc", "stty_raw", "stty_sane");
        $comm_pty = &open_proc("telnet $hostname")
            or die "open_proc failed";

        ## Unfortunately the Comm package doesn't
        ## return us a fully qualified filehandle.  We
        ## must keep the filehandle Comm returned for
        ## its use and we must build another filehandle
        ## qualified with the current package for our
        ## use.
        $pty = "main::" . $comm_pty;
    }

    ## Obtain a new Net::Telnet object that does I/O to the
    ## pseudo-terminal attached to the running telnet
    ## program.  The "Telnetmode" is "off" because we're
    ## not talking directly to a telnet port as we normally
    ## do, we're talking to a pseudo-terminal.  The
    ## "Output_record_separator" is now a carriage return
    ## because that's what you'd normally hit when you get
    ## done typing a line at a terminal.
    use Net::Telnet ();
    $host = new Net::Telnet (Fhopen => $pty,
                             Timeout => 10,
                             Prompt => '/[\$%#>] $/',
                             Output_record_separator => "\r",
                             Telnetmode => 0,
                             Cmd_remove_mode => 1);

    ## Issue some commands.
    $host->login($username, $passwd);
    $host->cmd("setenv DISPLAY $ENV{DISPLAY}");
    print $host->cmd("who");

    ## Allow the user to interact with telnet program until
    ## they exit.
    {
        no strict 'subs';  # so we can refer to STDIN
        local $^W = 0;     # Comm.pl isn't warning clean

        &stty_raw(STDIN);
        &interact($comm_pty);
        &stty_sane(STDIN);
        &close_it($comm_pty);
    }

    print "Exited telnet\n";
    exit;


=head1 AUTHOR

Jay Rogers E<lt>jay@rgrs.comE<gt>


=head1 COPYRIGHT

Copyright (c) 1997 Jay Rogers. All rights reserved.  This program is
free software; you can redistribute it and/or modify it under the same
terms as Perl itself.