- Repocopy from www/<lang> to head/<lang>/htdocs to eliminate duplicate information in the www and the doc directory. - Add various administration files to svnadmin. Approved by: doceng (implicit)
		
			
				
	
	
		
			189 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			189 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
| #!/usr/bin/perl -Tw
 | |
| #------------------------------------------------------------------------------
 | |
| # Copyright (C) 2011, Shaun Amott <shaun@FreeBSD.org>
 | |
| # All rights reserved.
 | |
| #
 | |
| # Redistribution and use in source and binary forms, with or without
 | |
| # modification, are permitted provided that the following conditions
 | |
| # are met:
 | |
| # 1. Redistributions of source code must retain the above copyright
 | |
| #    notice, this list of conditions and the following disclaimer.
 | |
| # 2. Redistributions in binary form must reproduce the above copyright
 | |
| #    notice, this list of conditions and the following disclaimer in the
 | |
| #    documentation and/or other materials provided with the distribution.
 | |
| #
 | |
| # THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS ``AS IS'' AND
 | |
| # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | |
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 | |
| # ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE
 | |
| # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 | |
| # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 | |
| # OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 | |
| # HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 | |
| # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 | |
| # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 | |
| # SUCH DAMAGE.
 | |
| #
 | |
| # $FreeBSD$
 | |
| #------------------------------------------------------------------------------
 | |
| 
 | |
| package GnatsPR::Section::Email;
 | |
| 
 | |
| use GnatsPR::Section::MIME;
 | |
| 
 | |
| use strict;
 | |
| 
 | |
| require 5.006;
 | |
| 
 | |
| 
 | |
| #------------------------------------------------------------------------------
 | |
| # Func: new()
 | |
| # Desc: Constructor.
 | |
| #
 | |
| # Args: $header - Raw e-mail header.
 | |
| #       $body   - Raw message body.
 | |
| #
 | |
| # Retn: $self
 | |
| #------------------------------------------------------------------------------
 | |
| 
 | |
| sub new
 | |
| {
 | |
| 	my $class = shift;
 | |
| 	my ($header, $body) = @_;
 | |
| 
 | |
| 	my $self = {
 | |
| 		headerblob => '',
 | |
| 		bodyblob   => '',
 | |
| 
 | |
| 		headers    => {},
 | |
| 
 | |
| 		mimeparts  => []
 | |
| 	};
 | |
| 
 | |
| 	bless $self, $class;
 | |
| 
 | |
| 	$self->{headerblob} = $header;
 | |
| 	$self->{bodyblob} = $body;
 | |
| 
 | |
| 	$self->ParseHeader() if ($header);
 | |
| 	$self->ParseBody() if ($body);
 | |
| 
 | |
| 	return $self;
 | |
| }
 | |
| 
 | |
| 
 | |
| #------------------------------------------------------------------------------
 | |
| # Func: ParseHeader()
 | |
| # Desc: Parse header blob into fields.
 | |
| #
 | |
| # Args: n/a
 | |
| #
 | |
| # Retn: n/a
 | |
| #------------------------------------------------------------------------------
 | |
| 
 | |
| sub ParseHeader
 | |
| {
 | |
| 	my $self = shift;
 | |
| 
 | |
| 	my $key;
 | |
| 
 | |
| 	foreach my $line (split /\n/, $self->{headerblob}) {
 | |
| 		if ($line =~ /^(\S+):\s*(.*)$/) {
 | |
| 			my $val = $2;
 | |
| 			$key = lc $1;
 | |
| 
 | |
| 			# Ignore multiple defs (e.g. Received: headers)
 | |
| 			exists $self->{headers}->{$key}
 | |
| 				and next;
 | |
| 
 | |
| 			$self->{headers}->{$key} = $val;
 | |
| 		} elsif ($line =~ /^\s*(.*)$/) {
 | |
| 			my $val = $1;
 | |
| 
 | |
| 			defined $key
 | |
| 				or next;
 | |
| 
 | |
| 			# No field to append to
 | |
| 			exists $self->{headers}->{$key}
 | |
| 				or next;
 | |
| 
 | |
| 			$self->{headers}->{$key} .= ' '.$val;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| #------------------------------------------------------------------------------
 | |
| # Func: ParseBody()
 | |
| # Desc: Parse body blob.
 | |
| #
 | |
| # Args: n/a
 | |
| #
 | |
| # Retn: n/a
 | |
| #------------------------------------------------------------------------------
 | |
| 
 | |
| sub ParseBody
 | |
| {
 | |
| 	# XXX: recurse to second-level parts
 | |
| 
 | |
| 	my $self = shift;
 | |
| 
 | |
| 	$self->{mimeparts} = [];
 | |
| 
 | |
| 	# First of all - attempt to split into MIME parts
 | |
| 	# Note that since GNATS nukes a bunch of the headers
 | |
| 	# that we need, this is purely of a heuristic nature.
 | |
| 
 | |
| 	# Technically less permissive than RFC1341
 | |
| 
 | |
| 	my $nextbound = qr/^--([A-Za-z0-9'()+_,-.\/:=?]{6,70})$/m;
 | |
| 	my $first = 1;
 | |
| 
 | |
| 	while ($self->{bodyblob} =~ s/$nextbound//m) {
 | |
| 		my $last;
 | |
| 
 | |
| 		if ($first) {
 | |
| 			my $boundary = $1;
 | |
| 			$nextbound = qr/^--\Q$boundary\E(--)?$/m;
 | |
| 			$last = 0;
 | |
| 			$first = 0;
 | |
| 		} else {
 | |
| 			$last = ($2 and $2 eq '--');
 | |
| 		}
 | |
| 
 | |
| 		# Promote to MIME part
 | |
| 
 | |
| 		push @{$self->{mimeparts}},
 | |
| 			new GnatsPR::Section::MIME(
 | |
| 				substr($self->{bodyblob}, 0, $-[0], '')
 | |
| 			)
 | |
| 			unless ($-[0] == 0);
 | |
| 	}
 | |
| 
 | |
| 	if (!@{$self->{mimeparts}}) {
 | |
| 		# No parts - just plain text
 | |
| 		push @{$self->{mimeparts}},
 | |
| 			new GnatsPR::Section::MIME($self->{bodyblob});
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| #------------------------------------------------------------------------------
 | |
| # Func: Header()
 | |
| # Desc: Return a header field.
 | |
| #
 | |
| # Args: $key - Header name, case insensitive.
 | |
| #
 | |
| # Retn: $val - Value.
 | |
| #------------------------------------------------------------------------------
 | |
| 
 | |
| sub Header
 | |
| {
 | |
| 	my $self = shift;
 | |
| 	my ($key) = @_;
 | |
| 
 | |
| 	return $self->{headers}->{lc $key};
 | |
| }
 | |
| 
 | |
| 
 | |
| 1;
 |