]> git.p6c8.net - selfforum.git/blobdiff - selfforum-cgi/shared/Mail.pm
Initial release
[selfforum.git] / selfforum-cgi / shared / Mail.pm
diff --git a/selfforum-cgi/shared/Mail.pm b/selfforum-cgi/shared/Mail.pm
new file mode 100644 (file)
index 0000000..e46a81c
--- /dev/null
@@ -0,0 +1,294 @@
+# Mail.pm
+
+##############################################
+#                                            #
+# Autor: n.d.p. nd@o3media.de                #
+#                                            #
+# Letze Aenderung: n.d.p. / 2001-01-03       #
+#                                            #
+# ========================================== #
+#                                            #
+# Funktion:                                  #
+#      ganz simples Formatieren und Senden   #
+#      einer Mail im text/plain, qp-Format   #
+#                                            #
+##############################################
+
+use strict;
+
+package Mail;
+
+use vars qw($mailbox $mailprog @ISA @EXPORT);
+
+# ===================
+# Funktionsexport
+# ===================
+
+require Exporter;
+@ISA = qw(Exporter);
+@EXPORT = qw(is_mail_address send_mail);
+
+########################################
+# EXPORT
+# sub is_mail_address
+#
+# Funktion:
+#      Ueberpruefen der Syntax einer
+#      Email-Adresse
+#
+# Rueckgabe
+#      true/false
+########################################
+
+sub is_mail_address ($) {
+  return ($_[0] =~ /$mailbox/);
+}
+
+########################################
+# EXPORT
+# sub send_mail
+#
+# Funktion:
+#      Senden der Nachricht
+#      ueber open-print-close
+#      $Mail::mailprog enthaelt
+#      den vollstaendigen string fuer
+#      open, dass heisst, es kann
+#      auch ein Dateiname sein.
+#
+# Rueckgabe:
+#      true/false
+########################################
+
+sub send_mail {
+  my $param=shift;
+  local *MAIL;
+
+  open MAIL,$mailprog or return 0;
+    print MAIL &as_string ($param);
+  close MAIL and return 1;
+
+  # Hier muss irgendwas schiefgelaufen sein
+  0;
+}
+
+##########################################
+# PRIVAT
+# sub as_string
+#
+# Funktion:
+#      Bereitstellung der gesamten Mail
+#      als String.
+#
+# Rueckgabe:
+#      String
+##########################################
+
+sub as_string {
+  my $param=shift;
+
+  my $header=&header_as_string ($param);
+  my $body=&body_as_string ($param);
+
+  # Rueckgabe
+  "$header\n$body\n";
+}
+
+##########################################
+# PRIVAT
+# sub body_as_string
+#
+# Funktion:
+#      Bereitstellung des Bodys
+#      als (qp-codierten) String.
+#
+# Rueckgabe:
+#      String
+##########################################
+
+sub body_as_string {
+  my $param=shift;
+
+  &encode_qp($param->{body});
+}
+
+##########################################
+# PRIVAT
+# sub header_as_string
+#
+# Funktion:
+#      Bereitstellung des Headers
+#      als String.
+#
+# Rueckgabe:
+#      String
+##########################################
+
+sub header_as_string {
+  my $param=shift;
+
+  my $string="Content-Disposition: inline\n";
+     $string.="MIME-Version: 1.0\n";
+     $string.="Content-Transfer-Encoding: quoted-printable\n";
+     $string.="Content-Type: text/plain\n";
+     $string.="Date: ".&rfc822_date(time)."\n";
+     $string.="From: ".$param->{'from'}."\n";
+     $string.=&get_list('To',$param->{'to'});
+     $string.=&get_list('Cc',$param->{'cc'});
+     $string.=&get_list('Bcc',$param->{'bcc'});
+     $string.="Subject: ".encode_qp($param->{'subject'})."\n";
+
+  # Rueckgabe
+  $string;
+}
+
+#######################################
+# PRIVAT
+# sub encode_qp
+#
+# C&P aus dem Modul MIME::QuotedPrint
+# Thanx for that
+#######################################
+
+sub encode_qp ($)
+{
+    my $res = shift;
+    $res =~ s/([^ \t\n!-<>-~])/sprintf("=%02X", ord($1))/eg;  # rule #2,#3
+    $res =~ s/([ \t]+)$/
+      join('', map { sprintf("=%02X", ord($_)) }
+                   split('', $1)
+      )/egm;                        # rule #3 (encode whitespace at eol)
+
+    # rule #5 (lines must be shorter than 76 chars, but we are not allowed
+    # to break =XX escapes.  This makes things complicated :-( )
+    my $brokenlines = "";
+    $brokenlines .= "$1=\n"
+        while $res =~ s/(.*?^[^\n]{73} (?:
+                 [^=\n]{2} (?! [^=\n]{0,1} $) # 75 not followed by .?\n
+                |[^=\n]    (?! [^=\n]{0,2} $) # 74 not followed by .?.?\n
+                |          (?! [^=\n]{0,3} $) # 73 not followed by .?.?.?\n
+            ))//xsm;
+
+    "$brokenlines$res";
+}
+
+##############################################
+# PRIVAT
+# sub get_list
+#
+# Funktion:
+#      Aufbereitung einer Liste oder eines
+#      Strings fuer den Header (To, Cc, Bcc)
+#
+# Rueckgabe:
+#      Ergebnis oder nichts
+##############################################
+
+sub get_list ($$) {
+  my ($start,$list)=splice @_;
+  my $string="";
+
+  return "" unless (length($list));
+  if (ref($list)) {
+    return "" unless (@$list);
+    foreach (@$list) {
+      $string.="$start: $_\n";}}
+  else {
+    $string="$start: $list\n";}
+
+  # Rueckgabe
+  $string;
+}
+
+##############################################
+# PRIVAT
+# sub rfc822_date
+#
+# Funktion:
+#      Bereitstellung eines RFC-konformen
+#      Datumstrings
+#
+# Rueckgabe:
+#      Datumstring
+##############################################
+
+sub rfc822_date ($) {
+  my ($sek, $min, $std, $mtag, $mon, $jahr, $wtag) = gmtime (+shift);
+
+  sprintf ('%s, %02d %s %04d %02d:%02d:%02d GMT',
+             (qw(Sun Mon Tue Wed Thu Fri Sat))[$wtag],
+             $mtag,
+             (qw(Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec))[$mon],
+             $jahr+1900, $std, $min, $sek);
+}
+
+##############################################
+# Modulinitialisierung
+# BEGIN
+#
+# Funktion:
+#      Bereitstellung des Regexps und des
+#      Mailprogs
+##############################################
+
+BEGIN {
+    # Standard-Mailprogramm
+
+    # Dieser String wird so, wie er ist, an die open-Anweisung geschickt,
+    # -t  = tainted(?),der Header (=alles bis zur ersten Leerzeile)
+    #       wird nach To:, Cc: und evtl. Bcc: abgesucht.
+    # -oi = damit wird verhindert, dass sendmail, ein Zeile, wo nur ein
+    #       Punkt drinsteht, als Mailende erkennt( waere Standard ).
+    # ===================================================================
+
+    $mailprog = '|/usr/lib/sendmail -t -oi';
+
+    # Thanx to J. Friedl for this regex:
+
+    my ($address,$route_addr,$phrase,$addr_spec,$X,$phrase_char,$quoted_str,$comment,$word,$phrase_ctrl,$NonASCII,
+        $CloseBR,$OpenBR,$esc,$route,$domain,$local_part,$Period,$sub_domain,$domain_lit,$domain_ref,$quoted_pair,
+        $dtext,$atom,$qtext,$atom_char,$ctrl,$space,$tab,$CloseParen,$ctext,$Cnested,$OpenParen,$CRlist);
+
+    $esc        = '\\\\';
+    $Period      = '\.';
+    $space      = '\040';
+    $tab         = '\t';
+    $OpenBR     = '\[';
+    $CloseBR     = '\]';
+    $OpenParen  = '\(';
+    $CloseParen  = '\)';
+    $NonASCII   = '\x80-\xff';
+    $ctrl        = '\000-\037';
+    $CRlist     = '\n\015';
+    $qtext = qq/[^$esc$NonASCII$CRlist\"]/;
+    $dtext = qq/[^$esc$NonASCII$CRlist$OpenBR$CloseBR]/;
+    $quoted_pair = qq< $esc [^$NonASCII] >;
+    $ctext   = qq< [^$esc$NonASCII$CRlist()] >;
+    $Cnested = qq< $OpenParen $ctext* (?: $quoted_pair $ctext* )* $CloseParen >;
+    $comment = qq< $OpenParen $ctext* (?: (?: $quoted_pair | $Cnested ) $ctext* )* $CloseParen >;
+    $X = qq< [$space$tab]* (?: $comment [$space$tab]* )* >;
+    $atom_char   = qq/[^($space)<>\@,;:\".$esc$OpenBR$CloseBR$ctrl$NonASCII]/;
+    $atom = qq< $atom_char+ (?!$atom_char) >;
+    $quoted_str = qq< \" $qtext * (?: $quoted_pair $qtext * )* \" >;
+    $word = qq< (?: $atom | $quoted_str ) >;
+    $domain_ref  = $atom;
+    $domain_lit  = qq< $OpenBR (?: $dtext | $quoted_pair )* $CloseBR >;
+    $sub_domain  = qq< (?: $domain_ref | $domain_lit ) $X >;
+    $domain = qq< $sub_domain (?: $Period $X $sub_domain )* >;
+    $route = qq< \@ $X $domain (?: , $X \@ $X $domain )* : $X >;
+    $local_part = qq< $word $X (?: $Period $X $word $X )* >;
+    $addr_spec  = qq< $local_part \@ $X $domain >;
+    $route_addr = qq[ < $X (?: $route )? $addr_spec > ];
+    $phrase_ctrl = '\000-\010\012-\037';
+    $phrase_char = qq/[^()<>\@,;:\".$esc$OpenBR$CloseBR$NonASCII$phrase_ctrl]/;
+    $phrase = qq< $word $phrase_char * (?: (?: $comment | $quoted_str ) $phrase_char * )* >;
+    $mailbox = qq< $X (?: $addr_spec | $phrase  $route_addr ) >;
+    $mailbox = qr ~^$mailbox$~x;
+}
+
+# making 'require' happy
+1;
+
+#####################
+# end of Mail
+#####################
\ No newline at end of file

patrick-canterino.de