discourse-legacysite-perl/site/slowtwitch.com/cgi-bin/articles/admin/GT/Socket/Client/SSLHandle.pm
2024-06-17 21:49:12 +10:00

125 lines
3.4 KiB
Perl

# ====================================================================
# Gossamer Threads Module Library - http://gossamer-threads.com/
#
# GT::Socket::Client::SSLHandle
# Author: Jason Rhinelander
# CVS Info : 087,071,086,086,085
# $Id: SSLHandle.pm,v 1.3 2004/01/13 01:35:20 jagerman Exp $
#
# Copyright (c) 2004 Gossamer Threads Inc. All Rights Reserved.
# ====================================================================
#
# Description:
# A tied filehandle for SSL connections with GT::Socket::Client (via
# Net::SSLeay::Handle).
#
package GT::Socket::Client::SSLHandle;
use strict;
use vars qw/$VERSION $ERROR/;
use GT::Socket::Client;
use Net::SSLeay 1.06 qw/print_errs/;
*ERROR = \$GT::Socket::Client::ERROR;
$VERSION = sprintf "%d.%03d", q$Revision: 1.3 $ =~ /(\d+)\.(\d+)/;
Net::SSLeay::load_error_strings();
Net::SSLeay::SSLeay_add_ssl_algorithms();
Net::SSLeay::randomize();
sub TIEHANDLE {
my ($class, $socket) = @_;
my $ctx = Net::SSLeay::CTX_new()
or return ssl_err("Failed to create new SSL CTX: $!", "SSL CTX_new");
my $ssl = Net::SSLeay::new($ctx)
or return ssl_err("Failed to create SSL: $!", "SSL new");
my $fileno = fileno($socket);
Net::SSLeay::set_fd($ssl, $fileno);
my $connect = Net::SSLeay::connect($ssl);
${*$socket}{SSLHandle_ssl} = $ssl;
${*$socket}{SSLHandle_ctx} = $ctx;
${*$socket}{SSLHandle_fileno} = $fileno;
return bless $socket, $class;
}
sub PRINT {
my $socket = shift;
my $ssl = ${*$socket}{SSLHandle_ssl};
my $ret = 0;
for (@_) {
defined or last;
$ret = Net::SSLeay::write($ssl, $_);
if (!$ret) {
ssl_err("Could not write to SSL socket: $!", "SSL write");
last;
}
}
return $ret;
}
sub READLINE {
my $socket = shift;
my $ssl = ${*$socket}{SSLHandle_ssl};
my $line = Net::SSLeay::ssl_read_until($ssl);
if (!$line) {
ssl_err("Could not readline from SSL socket: $!", "SSL ssl_read_until");
return undef;
}
return $line;
}
sub READ {
my ($socket, $buffer, $length, $offset) = \(@_);
my $ssl = ${*$$socket}{SSLHandle_ssl};
if (defined $$offset) {
my $read = Net::SSLeay::ssl_read_all($ssl, $$length)
or return ssl_err("Could not read_all from SSL socket: $!", "SSL ssl_read_all");
my $buf_length = length($$buffer);
$$offset > $buf_length and $$buffer .= chr(0) x ($$offset - $buf_length);
substr($$buffer, $$offset) = $read;
return length($read);
}
else {
return length(
$$buffer = Net::SSLeay::ssl_read_all($ssl, $$length)
or return ssl_err("Could not read_all from SSL socket: $!", "SSL ssl_read_all")
);
}
}
sub WRITE {
my $socket = shift;
my ($buffer, $length, $offset) = @_;
$offset = 0 unless defined $offset;
# Return number of characters written
my $ssl = ${*$socket}{SSLHandle_ssl};
Net::SSLeay::write($ssl, substr($buffer, $offset, $length))
or return ssl_err("Could not write to SSL socket: $!", "SSL write");
return $length;
}
sub CLOSE {
my $socket = shift;
my $fileno = fileno($socket);
Net::SSLeay::free(${*$socket}{SSLHandle_ssl});
Net::SSLeay::CTX_free(${*$socket}{SSLHandle_ctx});
close $socket;
}
sub FILENO { fileno($_[0]) }
sub ssl_err {
my ($msg, $key) = @_;
$ERROR = "$msg\n" . print_errs($key); # Also sets $GT::Socket::Client::ERROR
return undef;
}
1;