#!/usr/bin/perl
$|=1; # Run this with Faucet.
#use strict;
use IO::Socket;
use Net::hostent;
use bytes;
my $remote=$ARGV[0];
my $buf, $byte, @head, $up, $user, $err, $kidpid, $sock, $op, $server;
my $destip, $destport, $i;
### Read in first bytes of the Socks header.
read(STDIN,$buf,4);
@head=unpack("CCn",$buf);
read(STDIN,$buf,4);
$ip=inet_ntoa($buf);
$user='';
while(read(STDIN,$buf,1)) {
last unless(ord $buf);
$user .= $buf;
}
### Do some insanity checking.
$err=91; $op=-1; # Assume Not OK...
$destip=pack("N",0); $destport=0;
if($head[0] == 4) { # SOCKS 4
if($head[1] == 1) { # CONNECT
$sock=IO::Socket::INET->new(Proto=>"tcp",
PeerAddr=>$ip,
PeerPort=>$head[2]);
if($sock) {
$sock->autoflush(1);
$err=90;
}
} elsif($head[1] == 2) { # BIND
$destport=$$; $destip=inet_aton($remote);
$server=IO::Socket::INET->new(Proto=>'tcp',
LocalPort=>$$,
LocalAddr=>$remote,
Listen=>1,
Reuse=>1);
print pack("CCn",0,90,$destport).$destip;
if($sock=$server->accept()) {
$sock->autoflush(1);
if($sock->peerhost eq $ip) {
$err=90;
} else {
close $sock;
}
}
close $server;
}
}
print pack("CCn",0,$err,$destport).$destip;
exit if($err>90);
die "can't fork: $!" unless defined($kidpid=fork());
if($kidpid) {
while(read($sock,$byte,1)){
print STDOUT $byte;
}
kill("TERM", $kidpid);
} else {
while (read(STDIN,$byte,1)) {
print $sock $byte;
}
}
|