#!/usr/bin/perl -T # # Given a filename, start offset and end offset of a mail message, # read the message and format it nicely using HTML. # # by John Fieber # February 26, 1998 # # $FreeBSD: www/en/cgi/getmsg.cgi,v 1.43 2007/07/16 18:12:04 wosch Exp $ # require "./cgi-lib.pl"; require "./cgi-style.pl"; use POSIX qw(strftime); # # Site design includes setting a:visited to the same as a:link, # which isn't good in archived messages, e.g., you want to follow # links in commit messages and know which links you've visited. # Override it inside the
 that is the message.
$t_style = qq`
`;
#
# Files MUST be fully qualified and MUST start with this path.
#
$messagepath = "/usr/local/www/db/text/";
$messagepathcurrent = "/usr/local/www/mid/archive/";
$ftparchive = 'ftp://ftp.FreeBSD.org/pub/FreeBSD/doc/mailing-lists/archive';
&ReadParse(*formdata);
&Fetch($formdata{'fetch'});
exit 0;
sub Fetch
{
    my ($docid) = @_;
    my ($start, $end, $file, $type) = split(/ /, $docid);
    my ($message, @finfo);
    #
    # Check to ensure that (a) the specified file starts
    # with an approved pathname and (b) that it contains no
    # relative components (eg ..).  This is so that arbitrary
    # files cannot be accessed.
    #
    $file =~ s/\.\.//g;
    $file =~ s|/+|/|;
    $file =~ s|^archive/|$messagepath/|;
    # read the full archive 
    if ($type eq 'archive') {
	# from the FreeBSD ftp server
	if ($file =~ s%^$messagepath%%o) {
	    print "Location: $ftparchive/$file.gz\n";
	    print "Content-type: text/plain\n\n";     
	    exit(0);
	}
	
	# from the local mail archive for current mails
	elsif ($file =~ m%^current/(cvs|svn|freebsd|p4|trustedbsd)-[a-z0-9-]+$% &&
	       open(DATA, "$messagepathcurrent$file")) {
	    print "Content-type: text/plain\n\n"; 
	    while() {
		print;
	    }
	    close(DATA);
	    exit(0);
	}
    }
    if (($file =~ /^$messagepath/ && -f $file && open(DATA, $file)) ||
	($file =~ m%^current/(cvs|svn|freebsd|p4|trustedbsd)-[a-z0-9-]+$% &&
	 open(DATA, "$messagepathcurrent$file")))
    {
	@finfo = stat DATA;
    	seek DATA, $start, 0;
	if ($end > $start && $start >= 0) {
	    read DATA, $message, $end - $start;
	} else {
	    # Unknown length, guess the end of the E-Mail
	    my($newline) = 0;
	    while() {
		last if ($newline && /^From .* \d{4}/);
		if (/^$/) { $newline = 1 } else { $newline = 0; }
		$message .= $_;
	    }
	}
    	close(DATA);
	print "last-modified: " .
	    POSIX::strftime("%a, %d %b %Y %T GMT", gmtime($finfo[9])) . "\n";
	# print E-Mail as plain ascii text
	if ($type eq 'raw') {
            print "Content-type: text/plain\n\n";
            print $message;
	    return;
        }	
	$message = &MessageToHTML($message, $file);
    }
    else
    {
    	$message = "The specified message cannot be accessed.
\n";
    }
    print &short_html_header("FreeBSD Mail Archives");
    print $message;
    print &html_footer;
}
sub EscapeHTML
{
    my ($text) = @_;
    $text =~ s/&/&/g;
    $text =~ s/</g;
    $text =~ s/>/>/g;
    return $text;
}
sub MessageToHTML
{
    my ($doc, $file) = @_;
    my ($header, $body) = split(/\n\n/, $doc, 2);
    my ($i, %hdr, $field, $data, $message);
    my ($mid) = 'mid.cgi';
    my ($mid_full_url) = 'http://docs.FreeBSD.org/cgi/mid.cgi';
    my ($tmid,$tirt,$tref);
    
    $body = &AddAnchors(&EscapeHTML($body));
    $header = &EscapeHTML($header);
    $header =~ s/\n[ \t]+/ /g;
    foreach $i (split(/\n/, $header)) {
    	($field, $data) = split(/ /, $i, 2);
	$field =~ y/A-Z/a-z/;
    	$hdr{$field} = $data;
    }
    $message = "\n";
    if (length($hdr{'date:'}) > 0) {
    	$message .= "Date:      $hdr{'date:'}\n";
    }
    if (length($hdr{'from:'}) > 0) {
    	$message .= "From:      $hdr{'from:'}\n";
    }
    if (length($hdr{'to:'}) > 0) {
    	$message .= "To:        $hdr{'to:'}\n";
    }
    if (length($hdr{'cc:'}) > 0) {
    	$message .= "Cc:        $hdr{'cc:'}\n";
    }
#    if (length($hdr{'sender:'}) > 0) {
#    	$message .= "Sender:    $hdr{'sender:'}\n";
#    }
    if (length($hdr{'subject:'}) > 0) {
    	$message .= "Subject:   $hdr{'subject:'}\n";
    }
    if ($hdr{'message-id:'}) {
	$tmid = $hdr{'message-id:'}; 
	$hdr{'message-id:'} =~ 
	    s%;([^&]+)&%;$1&%oi;
	$message .= "Message-ID:  $hdr{'message-id:'}\n";
    }
    if ($hdr{'resent-message-id:'}) {
	$hdr{'resent-message-id:'} =~ 
	    s%;([^&]+)&%;$1&%oi;
	$message .= "Resent-Message-ID: $hdr{'resent-message-id:'}\n";
    }
    if ($hdr{'in-reply-to:'}) {
	$tirt = $hdr{'in-reply-to:'};
	$hdr{'in-reply-to:'} =~
	    s%;([^&]+)&%;$1&%oi;
	$message .= "In-Reply-To: $hdr{'in-reply-to:'}\n";
    }
    if ($hdr{'references:'}) {
	$tref = $hdr{'references:'};
	$hdr{'references:'} =~
	    s%;([^&\s]+)&%;$1&%goi;
	$message .= "References:  $hdr{'references:'}\n";
    }
    $message .= "\n";
    $message .= "
\n";
    if ($tmid =~ m%;([^&]+)&%) {
	$message .= qq{Next in thread\n};
    }
    if ($tirt  =~ m%;([^&]+)&% ||
	$tref  =~ m%;([^&]+)&%) {
	$message .= qq{| Previous in thread\n};
    }
    $message .= qq{| Raw E-Mail\n};
    my $file2 = $file;
    if ($file2 =~ s%^$messagepath%archive/%oi ||
	$file2 =~ /^current/) {
    	$message .= qq{| Index\n};
    }
    $message .= qq{| Archive\n};
    $message .= qq{| Help\n};
    my $tid = $tmid;
    $tid =~ s/^<//;
    $tid =~ s/\@.*//;
    $message .= "
\n";
    #$message .= qq{\n};
    $message .= "\n$body\n
\n";
    #$message .= qq{\n};
    $message .= qq{
\nWant to link to this message? Use this URL: <};
    $message .= qq{$mid_full_url} . '?' . $tid . qq{>
};
    
    return $message;
}
sub strip_url
{
    my $url = shift;
    # strip trailing characters
    $url =~ s/>?$//;
    $url =~ s/[.,;>\s\)]*$//;
    return $url;
}
sub AddAnchors
{
    my ($text) = @_;
    my $cvsweb = 'http://cvsweb.FreeBSD.org';
    $text =~ s/(http|https|ftp)(:[\S]*?\/?)(\W?\s)/sprintf("%s<\/a>$3", &strip_url("$1$2"), "$1$2", $3)/egoi;
    if ($text =~ /Revision\s+Changes\s+Path/) {
	
	# match revsion and file name
	#   1.10      +2 -2      ports/audio/xmradio/Makefile
	# ->
	#  cvsweb.cgi/ports/audio/xmradio/Makefile.diff?r1=1.9&r2=r.10
	#
	$text =~ s!([\d.]+\.)(\d+)              # revision
		   (\s+[+-]\d+\s+[+-]\d+\s+)    # +- stuff
	           ([a-zA-Z\d_:.+/-]+)          # filename
		  !"$1$2" eq "1.1" ?
		    sprintf("%s%s%s%s", $1, $2, $3, $4, $4) :
		    sprintf("%s%s%s%s",
			    $1, $2, $3, $4, $1, $2 - 1, $1, $2, $4)!gex;
    }
    return $text;
}