8c5391ab6e
Michael
420 lines
9.5 KiB
Perl
Executable File
420 lines
9.5 KiB
Perl
Executable File
#!/usr/bin/perl -w
|
|
|
|
# Simple WEB server.
|
|
#
|
|
# Inspired by some examples from the perlipc and other perl manual pages.
|
|
#
|
|
# Copyright (C) 2009 Michael Adam <obnox@samba.org>
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify it
|
|
# under the terms of the GNU General Public License as published by the Free
|
|
# Software Foundation; either version 2 of the License, or (at your option)
|
|
# any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful, but WITHOUT
|
|
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
# more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License along with
|
|
# this program; if not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
use strict;
|
|
|
|
use IO::Socket;
|
|
use IO::Select;
|
|
use Carp;
|
|
use POSIX qw(setsid :sys_wait_h);
|
|
use Errno;
|
|
use Getopt::Long;
|
|
use Pod::Usage;
|
|
use Fcntl ':flock'; # import LOCK_* constants
|
|
|
|
my $VERSION = "0.1";
|
|
|
|
my $EOL = "\015\012";
|
|
|
|
my $port = 2345;
|
|
my $proto = getprotobyname('tcp');
|
|
my $pid_file = "/tmp/webserver.pid";
|
|
my $log_dir = "/tmp";
|
|
my $access_log_file;
|
|
my $error_log_file;
|
|
my $document_root = "/tmp";
|
|
my $help = 0;
|
|
|
|
sub create_child($$$);
|
|
|
|
sub logmsg {
|
|
print STDERR "[", scalar localtime, ", $$] $0: @_\n";
|
|
}
|
|
|
|
sub start_server($$) {
|
|
my $proto = shift;
|
|
my $port = shift;
|
|
my $server;
|
|
|
|
$server = IO::Socket::INET->new(Proto => $proto,
|
|
LocalPort => $port,
|
|
Listen => SOMAXCONN,
|
|
Reuse => 1);
|
|
|
|
logmsg "server started listening on port $port";
|
|
|
|
return $server;
|
|
}
|
|
|
|
sub REAPER {
|
|
local $!; # don't let waitpid() overwrite current error
|
|
|
|
while ((my $pid = waitpid(-1,WNOHANG)) > 0 && WIFEXITED($?)) {
|
|
logmsg "reaped $pid" . ($? ? " with exit code $?" : '');
|
|
}
|
|
|
|
$SIG{CHLD} = \&REAPER;
|
|
}
|
|
|
|
sub parse_request($) {
|
|
my $client = shift;
|
|
my $request = {};
|
|
|
|
|
|
# parse the request line
|
|
|
|
my $request_line = <$client>;
|
|
if (!$request_line) {
|
|
$request->{error} = "emtpy request";
|
|
return $request;
|
|
}
|
|
|
|
chomp ($request_line);
|
|
|
|
my ($method, $object, $version) = split(" ", $request_line);
|
|
unless (defined($version) and $version) {
|
|
$request->{version} = "0.9";
|
|
} else {
|
|
if ($version !~ /HTTP\/(\d\.\d)/gi) {
|
|
$request->{error} = "illegal version ($version)";
|
|
return $request;
|
|
}
|
|
$request->{version} = $1;
|
|
}
|
|
$request->{method} = uc($method);
|
|
$request->{object} = $object;
|
|
|
|
# parse the request headers
|
|
|
|
my $current_header_line;
|
|
$request->{headers} = [];
|
|
while ($request_line = <$client>) {
|
|
if ($request_line =~ /^[ \t]/) {
|
|
# continued header line
|
|
chomp $request_line;
|
|
$current_header_line .= $request_line;
|
|
next;
|
|
}
|
|
|
|
if ($current_header_line) {
|
|
# finish current header line
|
|
my ($name, $value) = split(": ", $current_header_line);
|
|
push(@{$request->{headers}},
|
|
{ name => lc($name), value => $value });
|
|
}
|
|
|
|
last if ($request_line eq $EOL);
|
|
|
|
chomp $request_line;
|
|
$current_header_line = $request_line;
|
|
}
|
|
|
|
|
|
# parse entity (body)
|
|
|
|
$request->{entity} = "";
|
|
|
|
# skip for now, don't block...
|
|
# if ($request_line) {
|
|
# while ($request_line = <$client>) {
|
|
# logmsg "got line '$request_line'";
|
|
# $request->{entity} .= $request_line;
|
|
# }
|
|
# }
|
|
|
|
my @print_headers = ();
|
|
foreach my $header (@{$request->{headers}}) {
|
|
push @print_headers, $header->{name} . ": " . $header->{value};
|
|
}
|
|
logmsg "request:\n" .
|
|
"------------------------------\n" .
|
|
"Method: " . $request->{method} . "\n" .
|
|
"Object: " . $request->{object} . "\n" .
|
|
"Version: " . $request->{version} . "\n" .
|
|
"\n" .
|
|
"Headers:\n" .
|
|
join("\n", @print_headers) . "\n" .
|
|
#"\n" .
|
|
#"Body:\n" .
|
|
#"'" . $request->{entity} . "'\n" .
|
|
"------------------------------";
|
|
|
|
return $request;
|
|
}
|
|
|
|
sub child_action($) {
|
|
my $client = shift;
|
|
my $client_ip = shift;
|
|
|
|
logmsg "client_action: client $client_ip";
|
|
|
|
$client->autoflush();
|
|
|
|
my $fortune_bin = "/usr/games/fortune";
|
|
my $fortune = "";
|
|
if ( -x $fortune_bin) {
|
|
$fortune = qx(/usr/games/fortune);
|
|
$fortune =~ s/\n/$EOL/g;
|
|
}
|
|
|
|
my $request = parse_request($client);
|
|
|
|
if ($request->{error}) {
|
|
print $client "HTTP/1.0 400 Bad Request$EOL";
|
|
print $client "Server: Tinyproxy-Test-Web-Server/$VERSION$EOL";
|
|
print $client "Content-Type: text/html$EOL";
|
|
print $client "$EOL";
|
|
print $client "<html>$EOL";
|
|
print $client "<h1>400 Bad Request</h1>$EOL";
|
|
print $client "<p>Error: " . $request->{error} . "</p>$EOL";
|
|
print $client "</html>$EOL";
|
|
close $client;
|
|
return;
|
|
}
|
|
|
|
print $client "HTTP/1.0 200 OK$EOL";
|
|
print $client "Server: Tinyproxy-Test-Web-Server/$VERSION$EOL";
|
|
print $client "Content-Type: text/html$EOL";
|
|
print $client "$EOL";
|
|
print $client "<html>$EOL";
|
|
print $client "<h1>Tinyproxy test WEB server</h1>$EOL";
|
|
print $client "<h2>Fortune</h2>$EOL";
|
|
if ($fortune) {
|
|
print $client "<pre>$fortune</pre>$EOL";
|
|
} else {
|
|
print $client "Sorry, no $fortune_bin not found.$EOL";
|
|
}
|
|
|
|
my @print_headers = ();
|
|
foreach my $header (@{$request->{headers}}) {
|
|
push @print_headers, $header->{name} . ": " . $header->{value};
|
|
}
|
|
print $client "<h2>Your request:</h2>$EOL";
|
|
print $client "<pre>$EOL";
|
|
print $client "Method: " . $request->{method} . "\n" .
|
|
"Object: " . $request->{object} . "\n" .
|
|
"Version: " . $request->{version} . "\n" .
|
|
"\n" .
|
|
join("\n", @print_headers) .
|
|
"\n" .
|
|
"entity (body):\n" .
|
|
$request->{entity} . "\n";
|
|
print $client "</pre>$EOL";
|
|
print $client "</html>$EOL";
|
|
|
|
close $client;
|
|
|
|
return 0;
|
|
}
|
|
|
|
sub create_child($$$) {
|
|
my $client = shift;
|
|
my $action = shift;
|
|
my $client_ip = shift;
|
|
|
|
unless (@_ == 0 && $action && ref($action) eq 'CODE') {
|
|
confess "internal error. create_child needs code reference as argument";
|
|
}
|
|
|
|
my $pid = fork();
|
|
if (not defined($pid)) {
|
|
# error
|
|
logmsg "cannot fork: $!";
|
|
return;
|
|
} elsif ($pid) {
|
|
# parent
|
|
logmsg "child process created with pid $pid";
|
|
return;
|
|
} else {
|
|
# child
|
|
exit &$action($client, $client_ip);
|
|
}
|
|
}
|
|
|
|
sub process_options() {
|
|
my $result = GetOptions("help|?" => \$help,
|
|
"port=s" => \$port,
|
|
"pid-file=s" => \$pid_file,
|
|
"log-dir=s" => \$log_dir,
|
|
"root|document-root=s" => \$document_root);
|
|
die "Error reading cmdline options! $!" unless $result;
|
|
|
|
pod2usage(1) if $help;
|
|
|
|
# some post-processing:
|
|
|
|
($port) = $port =~ /^(\d+)$/ or die "invalid port";
|
|
$access_log_file = "$log_dir/webserver.access_log";
|
|
$error_log_file = "$log_dir/webserver.error_log";
|
|
}
|
|
|
|
sub daemonize() {
|
|
umask 0;
|
|
chdir "/" or die "daemonize: can't chdir to /: $!";
|
|
open STDIN, "/dev/null" or
|
|
die "daemonize: Can't read from /dev/null: $!";
|
|
|
|
my $pid = fork();
|
|
die "daemonize: can't fork: $!" if not defined($pid);
|
|
exit(0) if $pid != 0; # parent
|
|
|
|
# child (daemon)
|
|
setsid or die "damonize: Can't create a new session: $!";
|
|
}
|
|
|
|
sub reopen_logs() {
|
|
open STDOUT, ">> $access_log_file" or
|
|
die "daemonize: Can't write to '$access_log_file': $!";
|
|
open STDERR, ">> $error_log_file" or
|
|
die "daemonize: Can't write to '$error_log_file': $!";
|
|
}
|
|
|
|
sub get_pid_lock() {
|
|
# first make sure the file exists
|
|
open(LOCKFILE_W, ">> $pid_file") or
|
|
die "Error opening pid file '$pid_file' for writing: $!";
|
|
|
|
# open for reading and try to lock:
|
|
open(LOCKFILE, "< $pid_file") or
|
|
die "Error opening pid file '$pid_file' for reading: $!";
|
|
unless (flock(LOCKFILE, LOCK_EX|LOCK_NB)) {
|
|
print "pid file '$pid_file' is already locked.\n";
|
|
my $other_pid = <LOCKFILE>;
|
|
if (!defined($other_pid)) {
|
|
print "Error reading from pid file.\n";
|
|
} else {
|
|
chomp($other_pid);
|
|
if (!$other_pid) {
|
|
print "pid file is empty.\n";
|
|
} else {
|
|
print "Webserver is already running with pid '$other_pid'.\n";
|
|
}
|
|
}
|
|
close LOCKFILE;
|
|
exit(0);
|
|
}
|
|
|
|
# now re-open for recreating the file and write our pid
|
|
close(LOCKFILE_W);
|
|
open(LOCKFILE_W, "> $pid_file") or
|
|
die "Error opening pid file '$pid_file' for writing: $!";
|
|
LOCKFILE_W->autoflush(1);
|
|
print LOCKFILE_W "$$";
|
|
close(LOCKFILE_W);
|
|
}
|
|
|
|
sub release_pid_lock() {
|
|
flock(LOCKFILE, LOCK_UN);
|
|
close LOCKFILE;
|
|
}
|
|
|
|
# "main" ...
|
|
|
|
$|=1; # autoflush
|
|
|
|
process_options();
|
|
|
|
daemonize();
|
|
get_pid_lock();
|
|
reopen_logs();
|
|
|
|
$SIG{CHLD} = \&REAPER;
|
|
|
|
my $server = start_server($proto, $port);
|
|
my $slct = IO::Select->new($server);
|
|
|
|
while (1) {
|
|
my @ready_for_reading = $slct->can_read();
|
|
foreach my $fh (@ready_for_reading) {
|
|
if ($fh != $server) {
|
|
logmsg "select: fh ready for reading but not server",
|
|
"don't know what to do...";
|
|
}
|
|
|
|
# new connection:
|
|
my $client = $server->accept() or do {
|
|
# try again if accept() returned because
|
|
# a signal was received
|
|
if ($!{EINTR}) {
|
|
logmsg "accept: got signal EINTR ...";
|
|
next;
|
|
}
|
|
die "accept: $!";
|
|
};
|
|
my $client_ip = inet_ntoa($client->peeraddr);
|
|
logmsg "connection from $client_ip at port " . $client->peerport;
|
|
create_child($client, \&child_action, $client_ip);
|
|
close $client;
|
|
}
|
|
}
|
|
|
|
# never reached...
|
|
logmsg "Server done - ooops!\n";
|
|
|
|
release_pid_lock();
|
|
|
|
exit(0);
|
|
|
|
__END__
|
|
|
|
=head1 webserver.pl
|
|
|
|
A simple WEB server written in perl.
|
|
|
|
=head1 SYNOPSIS
|
|
|
|
webserver.pl [options]
|
|
|
|
=head1 OPTIONS
|
|
|
|
=over 8
|
|
|
|
=item B<--help>
|
|
|
|
Print a brief help message and exit.
|
|
|
|
=item B<--port>
|
|
|
|
Specify the port number for the server to listen on.
|
|
|
|
=item B<--root|--document-root>
|
|
|
|
Specify the document root directory from which to serve web content.
|
|
|
|
=item B<--log-dir>
|
|
|
|
Specify the directory where the log files should be stored.
|
|
|
|
=item B<--pid-file>
|
|
|
|
Specify the location of the pid lock file.
|
|
|
|
=back
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
This is a very simple web server. It currently does not deliver the specific web
|
|
page requested, but constructs the same kind of answer for each request, citing
|
|
a fortune if fortune is available, and printing the originating request.
|
|
|
|
=cut
|