From 02835c333db25f18a0609c212d8d68d814cdded6 Mon Sep 17 00:00:00 2001 From: "kilian (ksmachome)" Date: Tue, 31 Dec 2019 13:30:05 +0100 Subject: [PATCH] v20191231 blocked toemail domains barreau.lu, juridig.lu --- backoffice/api/lib/session.pm | 7 +- backoffice/api/process.cgi | 4 +- backoffice/index.cgi | 2 +- backoffice/tmpl/module/profile/index.tt | 8 +- dev/lib.old/createpdfA4invoice | 249 --- dev/lib.old/download.cgi | 114 -- dev/lib.old/image.cgi | 69 - dev/lib.old/pgsql.pm | 327 ---- dev/lib.old/prefs.cgi | 77 - dev/lib.old/sendEmail | 2235 ----------------------- dev/lib.old/session.pm | 89 - dev/lib.old/test.jpeg | Bin 75102 -> 0 bytes dev/lib.old/test.png | Bin 4092 -> 0 bytes dev/lib.old/testfiledata.pl | 9 - dev/lib.old/upload.cgi | 266 --- index.cgi | 2 +- 16 files changed, 10 insertions(+), 3448 deletions(-) delete mode 100644 dev/lib.old/createpdfA4invoice delete mode 100644 dev/lib.old/download.cgi delete mode 100644 dev/lib.old/image.cgi delete mode 100644 dev/lib.old/pgsql.pm delete mode 100644 dev/lib.old/prefs.cgi delete mode 100644 dev/lib.old/sendEmail delete mode 100644 dev/lib.old/session.pm delete mode 100644 dev/lib.old/test.jpeg delete mode 100644 dev/lib.old/test.png delete mode 100644 dev/lib.old/testfiledata.pl delete mode 100644 dev/lib.old/upload.cgi diff --git a/backoffice/api/lib/session.pm b/backoffice/api/lib/session.pm index bbef43c..9063c5e 100644 --- a/backoffice/api/lib/session.pm +++ b/backoffice/api/lib/session.pm @@ -74,10 +74,13 @@ sub registeruser(){ foreach my $d (%{$data}){ $data->{$d} = $self->{db}->securetext($data->{$d}); } + # my $usergroup = 'user'; - # # if ($data->{email} =~ /\@barreau.lu$/) { + if (($data->{email} !~ /\@barreau.lu$/) || ($data->{email} !~ /\@juridig.lu$/)) { + $ret->{message} = "addresse mail ...@barreau.lu requis!"; + return $ret; # $usergroup = 'avocat'; - # # } + } my $user = $self->{db}->dbquerysorted("select id from users where lower(username)=lower('".$data->{email}."');"); if (keys(%{$user}) > 0){ diff --git a/backoffice/api/process.cgi b/backoffice/api/process.cgi index 73089c4..2e84e10 100755 --- a/backoffice/api/process.cgi +++ b/backoffice/api/process.cgi @@ -1,4 +1,4 @@ -#!/usr/bin/perl +#!/usr/local/bin/perl use strict; @@ -174,7 +174,7 @@ if (($cgi->request_method() eq "GET") || ($cgi->request_method() eq "POST")){ my $tmp_link=$db->textunidecode(lc($p->{$px})); $tmp_link =~ s/[^a-zA-Z0-9]/_/g; $p->{lawyercatalog_location_link} = $tmp_link; - } + } #$p->{$px} = $db->securetext($p->{$px}); diff --git a/backoffice/index.cgi b/backoffice/index.cgi index 9e52d11..4530606 100755 --- a/backoffice/index.cgi +++ b/backoffice/index.cgi @@ -1,4 +1,4 @@ -#!/Users/kilian/perl5/perlbrew/perls/perl-5.24.1/bin/perl +#!/usr/local/bin/perl use strict; use lib ('api/lib/perl5'); diff --git a/backoffice/tmpl/module/profile/index.tt b/backoffice/tmpl/module/profile/index.tt index fac46b6..086e9b6 100644 --- a/backoffice/tmpl/module/profile/index.tt +++ b/backoffice/tmpl/module/profile/index.tt @@ -17,7 +17,7 @@
- +
@@ -38,12 +38,6 @@ - - - - - - \ No newline at end of file diff --git a/dev/lib.old/createpdfA4invoice b/dev/lib.old/createpdfA4invoice deleted file mode 100644 index 0c569ab..0000000 --- a/dev/lib.old/createpdfA4invoice +++ /dev/null @@ -1,249 +0,0 @@ -#!/usr/bin/env perl - -use strict; -use File::Basename; -use Getopt::Long; -use utf8; -use Data::Dumper; -use lib (dirname($0).'/lib'); -use PDF::API2; -use PDF::Table; -use Image::Size; -use Encode; -use JSON::PP qw/encode_json decode_json/; - -#use Data::Dumper; -my $pdf = PDF::API2->new(); -my $strpdfdata =""; -my $templatefile = dirname($0)."/invoicetemplate.json"; -my $xindata = ""; -my $pdfout = ""; -my $lang = "fr"; -my $endpoints = (); -$| = 1; -# my $datafile=dirname($0).'/pdftest.pdf'; -GetOptions ("data|d=s" => \$xindata, "output|o=s" => \$pdfout, "lang|l=s" =>\$lang); - -#FUNCTIONS BEGIN -sub mergedata($$){ - my $strdtmpl = shift; - my $dvalues = shift; - #my $dvalues = JSON::PP::decode_json($strdvalues); - - foreach my $k (keys(%{$dvalues})){ - - if (ref($dvalues->{$k}) eq "ARRAY"){ - my $search = uc($k); - my $tblsearch = '\["%%'.$search.':START.*'.$search.':END%%"\]'; - - my ($tblrow) = $strdtmpl =~ m/\["%%${search}:START%%"\]\s*,\s*(.*)\s*,\s*\["%%${search}:END%%"\]/; - my @strrows = (); - foreach my $rk (sort @{$dvalues->{$k}}){ - my $nrow = $tblrow; - foreach my $sk (keys(%{$rk})){ - my $rsearch = '%%'.uc($sk).'%%'; - my $rreplace = $rk->{$sk}; - $nrow =~ s/$rsearch/$rreplace/g; - } - push(@strrows,$nrow); - - } - - - my $tblreplace = join(',',@strrows); - - $strdtmpl =~ s/$tblsearch/$tblreplace/; - } - else { - my $search = '%%'.uc($k).'%%'; - my $replace = $dvalues->{$k}; - $strdtmpl =~ s/$search/$replace/g; - } - } - - my $dtmpl = decode_json(encode("utf-8",$strdtmpl)); - return $dtmpl; -} - -sub addimage($$$$){ - my $pdf = shift; - my $page = shift; - my $section = shift; - my $data = shift; - my $gfx = $page->gfx(); - - if (! -e dirname($0).'/'.$data->{src}){ - # print "Image:".dirname($0).'/'.$data->{src}." does not exist!
"; - return $section->{y}; - } - $data->{src} = dirname($0).'/'.$data->{src}; - # print "Image datasource:".$data->{src}.'
'; - my ($iw, $ih) = imgsize($data->{src}); - - my $nimgw = $section->{width}; - my $nimgh = ($nimgw/$iw) * $ih; - $gfx->translate($section->{x},$section->{y}); - my $img = $pdf->image_png($data->{src}); - $gfx->image($img,0,0,$nimgw,$nimgh); - $gfx->translate(0,0); - $gfx->save; - return $nimgh + $section->{y}; -} - -sub addline($$){ - my $page = shift; - my $section = shift; - my $line = $page->gfx(); - $line->translate($section->{x},$section->{y}); - $line->strokecolor($section->{color}); - $line->linewidth($section->{width}); - $line->move( $section->{start}->{x}, $section->{start}->{y} ); - $line->line( $section->{end}->{x}, $section->{end}->{y} ); - $line->stroke; - $line->save; - return $section->{end}->{y}; -} - -sub addtext($$$$){ - my $pdf = shift; - my $page = shift; - my $section = shift; - my $data = shift; - - my $textobj = $page->text(); - if (exists($section->{y})){ - if ($section->{y} =~ /^%%.*%%$/){ - my ($sec,$k) = $section->{y} =~ /^%%(.+):(.+)%%$/; - $section->{y} = $endpoints->{$sec}->{$k}; - } - } - $textobj->translate($section->{x}, $section->{y}); - $textobj->font($pdf->corefont($section->{font}->{name}),$section->{font}->{size}); - foreach my $d (@{$data->{text}}){ - if ($section->{align} eq "center"){ - $textobj->text_center($d); - } elsif ($section->{align} eq "right"){ - $textobj->text_right($d); - } else { - $textobj->text($d); - } - $section->{y} = $section->{y} - $section->{lineheight}; - $textobj->translate($section->{x}, $section->{y}); - } - $textobj->save; - return $section->{y}; -} - -sub addtable($$$$){ - my $pdf = shift; - my $page = shift; - my $section = shift; - my $data = shift; - my $pdftbl = new PDF::Table; - delete($section->{type}); - if (exists($section->{font})){ - $section->{font} = $pdf->corefont($section->{font}); - } - if (exists($section->{start_y})){ - if ($section->{start_y} =~ /^%%.*%%$/){ - my ($sec,$k) = $section->{start_y} =~ /^%%(.+):(.+)%%$/; - $section->{start_y} = $endpoints->{$sec}->{$k}; - } - } - if (exists($section->{header_props})){ - foreach (my $i=0;$i{header_props}});$i++){ - my $tmpdata = @{$section->{header_props}}[$i]; - if (exists($tmpdata->{font})){ - $tmpdata->{font} = $pdf->corefont($tmpdata->{font}); - } - @{$section->{header_props}}[$i] = $tmpdata; - } - } - - if (exists($section->{column_props})){ - foreach (my $i=0;$i{column_props}});$i++){ - my $tmpdata = @{$section->{column_props}}[$i]; - if (exists($tmpdata->{font})){ - $tmpdata->{font} = $pdf->corefont($tmpdata->{font}); - } - @{$section->{column_props}}[$i] = $tmpdata; - } - } - - my ($lastpage, $tblpages, $final_y) = $pdftbl->table($pdf,$page, $data->{text}, %{$section}); - return ($lastpage, $tblpages, $final_y); -} -#FUNCTIONS END - - -if (! -e $templatefile){ - # print "file $templatefile does not exist!\n"; - exit(1); -} -$xindata =~ s/\r//g; -if (($xindata !~ /^{/) && ($xindata !~ /}$/)){ - # print "no data input!\n"; - exit(2); -} -if (-e $pdfout){ - unlink($pdfout); -} - -open(DATA,$templatefile); -while (my $l = ){ - $strpdfdata .= $l; -} -close(DATA); - -# print "Test get Template!\n"; -my $templatedata = decode_json($strpdfdata); -my $pdfdata->{section} = $templatedata->{section}; -my $decoded = decode("utf-8",encode_json($templatedata->{data}->{$lang})); -my $mergedata = (); -# eval { - $mergedata = decode_json($xindata); -# 1; -# } or do { -# my $e = $@; -# print "$e\n"; -# }; -$pdfdata->{data} = mergedata($decoded,$mergedata); -$pdf->preferences({-fitwindow => 1}); -# print "Start PDF creation!
"; - - -my $page = $pdf->page(); -$page->mediabox('A4'); - - -foreach my $s (sort keys(%{$pdfdata->{section}})){ - $endpoints->{$s} = {final_y => '', lastpage => 1,pages => 1}; - if ($pdfdata->{section}->{$s}->{type} eq "image"){ - # print "Add Image $s
"; - - $endpoints->{$s}->{y} = addimage($pdf,$page,$pdfdata->{section}->{$s},$pdfdata->{data}->{$s}); - } elsif ($pdfdata->{section}->{$s}->{type} eq "line"){ - # print "Add Line $s
"; - $endpoints->{$s}->{y} = addline($page,$pdfdata->{section}->{$s}); - } elsif ($pdfdata->{section}->{$s}->{type} eq "text"){ - # print "Add Text $s
"; - $endpoints->{$s}->{y} = addtext($pdf,$page,$pdfdata->{section}->{$s},$pdfdata->{data}->{$s}); - } elsif ($pdfdata->{section}->{$s}->{type} eq "table"){ - # print "Add Table $s
"; - ($endpoints->{$s}->{lastpage}, $endpoints->{$s}->{pages}, $endpoints->{$s}->{y}) = addtable($pdf,$page,$pdfdata->{section}->{$s},$pdfdata->{data}->{$s}); - } -} -# print "save pdf to:".$pdfout."
"; -$pdf->saveas($pdfout); -$pdf->end; -if (! -e $pdfout){ - exit(3); -} - -#Functions - - - - - - diff --git a/dev/lib.old/download.cgi b/dev/lib.old/download.cgi deleted file mode 100644 index 1ce7b36..0000000 --- a/dev/lib.old/download.cgi +++ /dev/null @@ -1,114 +0,0 @@ -#!/usr/bin/perl -use strict; -use CGI; -#use CGI::Carp qw(fatalsToBrowser warningsToBrowser); -use lib ('./lib'); -use config; -use data; -use Cwd; -use template; - -my $cwd = getcwd; -my $cgi = CGI->new(); -my $url = $cgi->url(-path_info=>1); -my $baseurl = $cgi->url(-base=>1); -my %param = (); -$param{'lang'} = 'fr'; -$param{'id'} = 0; -$param{'rub'} = 0; -$param{'cat'} = 0; -my @tp = $cgi->param; - - -foreach my $p (@tp){ - $param{$p} = $cgi->param($p); - -} - -$param{'user'}{'sid'} = $cgi->cookie('myoffersid'); - -if (!defined $param{'user'}{'sid'}){$param{'user'}{'sid'} = "";} - -my $config = config->new(); - -if ((exists $param{'user'}{'sid'}) && ($param{'user'}{'sid'} ne "")){ - my %result = $config->checksession($param{'user'}{'sid'}); - $param{'user'} = {%result}; -} -my $bok = 0; -if ((exists($param{'user'}{'sid'})) && ($param{'user'}{'sid'} ne "0")){ - my $d = data->new(); - if (exists $param{'fileid'}){ - my %data = $d->getspecialfilebyfid($param{'fileid'}); - if (scalar(keys(%data)) > 0){ - if ($param{'user'}{'usertypename'} eq 'admin'){ - $bok = 1; - } - elsif ($param{'user'}{'usertypename'} eq 'bussiness'){ - $bok = $d->ispartneringroup($param{'user'}{'iduser'},$data{'0'}{'idgroup'}); - }else{ - $bok = $d->isofferfromuser($data{'0'}{'idoffer'},$param{'user'}{'username'}); - } - if ($bok == 1){ - my $file = $cwd."/files/".$param{'rub'}."/".$param{'cat'}."/".$data{0}{'file'}; - if (-e $file){ - my $mt = `file -bi $file`; - chomp $mt; - print $cgi->header(-type=>$mt,-attachment=>$data{0}{'file'}); - open(DLD,$file); - binmode(DLD); - my $buffer = ''; - while (read(DLD,$buffer,4096)){ - print STDOUT $buffer; - } - close(DLD); - exit(0); - } - } - } - } -} -if (exists $param{'staticfile'}){ - - my $tmpl = template->new(); - my $doc = ' - - -MyOffer - - - - -
-%%LANG:descagbprivate%%
-'; - my $d = data->new(); - $doc = $tmpl->replace_langplaceholders($doc,$param{'lang'}); - my $myfid = $d->createsessionid(); - my $file = '/tmp/'.$myfid.'.pdf'; - open (TMPF,">/tmp/".$myfid.".html"); - print TMPF $doc; - close(TMPF); - system('htmldoc --bodycolor white --bodyfont sherif --charset iso-8859-1 --continuous --embedfonts -t pdf14 --portrait --right 2.54cm --no-toc --numbered --no-overflow --quiet --webpage /tmp/'.$myfid.'.html > '.$file); - my $mt = `file -bi $file`; - chomp $mt; - print $cgi->header(-type=>$mt,-attachment=>$param{'staticfile'}.'_'.$param{'lang'}.'.pdf'); - open(DLD,$file); - binmode(DLD); - my $buffer = ''; - while (read(DLD,$buffer,4096)){ - print STDOUT $buffer; - } - close(DLD); - unlink("/tmp/".$myfid.".html"); - unlink($file); - exit(0); -} - print $cgi->header(); - print '
No data found!
'; - exit(0); - - - diff --git a/dev/lib.old/image.cgi b/dev/lib.old/image.cgi deleted file mode 100644 index 30069a2..0000000 --- a/dev/lib.old/image.cgi +++ /dev/null @@ -1,69 +0,0 @@ -#!/usr/bin/perl -use strict; -use CGI; -use lib ('./lib'); -use config; -use data; -use Cwd; -my $cwd = getcwd; -my $cgi = CGI->new(); -my $url = $cgi->url(-path_info=>1); -my $baseurl = $cgi->url(-base=>1); -my %param = (); -$param{'num'} = 'png'; -$param{'num'} = 0; -$param{'id'} = 0; -$param{'rub'} = 0; -$param{'cat'} = 0; -my @tp = $cgi->param; - - -foreach my $p (@tp){ - $param{$p} = $cgi->param($p); - -} -my $file = ""; -if (exists ($param{'captcha'})){ - $file = $cwd."/pic/captcha/".$param{'captcha'}.'.png'; -}elsif (exists($param{'logo'})){ - $file = $cwd."/pic/logo/".$param{'logo'}.'.png'; -} -else { - $file = $cwd."/pic/".$param{'rub'}."/".$param{'cat'}."/".$param{'id'}."_".$param{'num'}.".".$param{'type'}; -} -#my $file = $cwd."/pic/".$param{'rub'}."/".$param{'cat'}."/".$param{'id'}."_".$param{'num'}.".".$param{'type'}; -if (-e $file){ - my $mt = `file -bi $file`; - chomp $mt; - if (exists ($param{'captcha'})){ - print $cgi->header(-type=>$mt,-attachment=>$param{'captcha'}.".png"); - }elsif (exists($param{'logo'})){ - print $cgi->header(-type=>$mt,-attachment=>$param{'logo'}.".png"); - } -else { - print $cgi->header(-type=>$mt,-attachment=>$param{'id'}."_".$param{'num'}.".png"); - } -# print $cgi->header(-type=>$mt,-attachment=>$param{'id'}."_".$param{'num'}.".png"); - open(DLD,$file); - binmode(DLD); - my $buffer = ''; - while (read(DLD,$buffer,4096)){ - print STDOUT $buffer; - } - close(DLD); - exit(0); -} -else { - $file = $cwd."/pic/default/error.png"; - my $mt = `file -bi $file`; - chomp $mt; - print $cgi->header(-type=>$mt,-attachment=>"error.png"); - open(DLD,$file); - binmode(DLD); - my $buffer = ''; - while (read(DLD,$buffer,4096)){ - print STDOUT $buffer; - } - close(DLD); - exit(0); -} diff --git a/dev/lib.old/pgsql.pm b/dev/lib.old/pgsql.pm deleted file mode 100644 index 7942c89..0000000 --- a/dev/lib.old/pgsql.pm +++ /dev/null @@ -1,327 +0,0 @@ -package pgsql; -use strict; -use DBI; -use File::Basename; -#use DBD::mysql; -use Data::Dumper; -use Digest::SHA::PurePerl qw(sha256_hex); - -# use lib (dirname($0).'/../lib'); -use lib ('.'); -use lib ('./lib'); -use DBD::PgPP; -use URI::Encode qw(uri_encode uri_decode); -#use DBD::SQLite; -use Encode; - -sub new { - my $class = shift; - my $p = shift; - my $self = bless {}, $class; - #foreach my $x (keys(%{$p})){ - # $self->{$x} = $p->{$x}; - #} - $self->{dbtype} = 'PgPP'; #PgPP,SQLite - #$self->{dbhost} = 'sql629.your-server.de'; - $self->{dbhost} = 'localhost'; - $self->{dbname} = 'juridiglu_db'; - $self->{dbuser} = 'juridiglu_user'; - $self->{dbpassword} = 'hfdR2C9pK9rQV4cH'; - return $self; -} - -sub securetext(){ - my $self = shift; - my $text = shift; - $text =~ s/'/''/g; - return $text; -} - -sub dbquery(){ - my $self = shift; - my $key = shift; - my $stat = shift; - #my $retempty = shift; - my $retdata =(); - my $dbh = DBI->connect('DBI:'.$self->{dbtype}.':dbname='.$self->{dbname}.';host='.$self->{dbhost},$self->{dbuser},$self->{dbpassword},{PrintError=>0,RaiseError=>0,AutoCommit=>1}) or return $retdata->{error} = "dbquery Connection Error!".$!; - $stat = encode("utf8", $stat); - - open FILE,">>sql.log"; - print FILE "$stat\n"; - close FILE; - my $sth = $dbh->prepare($stat); - $sth->execute() or return $retdata->{error} = "dbquery: ".$stat; - while(my $data = $sth->fetchrow_hashref()) - { - if (exists $data->{$key}){ - foreach my $k (keys %{$data}){ - #if ($k ne $key){ - - $retdata->{$data->{$key}}{$k} =$data->{$k}; - #} - } - } - } - if (keys(%{$retdata}) == 0){ - $retdata =(); - } - $sth->finish(); - $dbh->disconnect(); - return $retdata; -} - -sub dbquerysorted(){ - my $self = shift; - my $stat = shift; - my $vw_info = shift; - my $retdata; - my $dbh = DBI->connect('DBI:'.$self->{dbtype}.':dbname='.$self->{dbname}.';host='.$self->{dbhost},$self->{dbuser},$self->{dbpassword},{PrintError=>0,RaiseError=>0,AutoCommit=>1}) or return $retdata->{error} = "dbquery Connection Error!".$!; - $stat = encode("utf8", $stat); - open FILE,">>sql.log"; - print FILE "$stat\n"; - close FILE; - my $sth = $dbh->prepare($stat); - - - $sth->execute() or return $retdata->{error} = "dbquery: ".$sth->errstr; - my $count = 0; - - while(my $data = $sth->fetchrow_hashref()) - { - #$retdata->{$count} = $data; - foreach my $k (keys %{$data}){ - $retdata->{$count}->{$k} = $data->{$k}; - } - $count++; - } - -# my $qstruct = (); -# my $num_fields = $sth->{NUM_OF_FIELDS}; - -# for ( my $i=0; $i< $num_fields; $i++ ) { -# $qstruct->{$i}->{name} = $sth->{NAME}->[$i]; -# #$qstruct->{$i}->{type} = $sth->{COMMENT}->[$i]; -# #$qstruct->{$i}->{precision} = $sth->{PRECISION}->[$i]; -# } - - $sth->finish(); - $dbh->disconnect(); - - return $retdata; -} - -sub dbexec(){ - my $self = shift; - my $stat = shift; - my $retdata; - my $dbh = DBI->connect('DBI:'.$self->{dbtype}.':dbname='.$self->{dbname}.';host='.$self->{dbhost},$self->{dbuser},$self->{dbpassword},{PrintError=>0,RaiseError=>0,AutoCommit=>1}) or return $retdata->{error} = "dbquery Connection Error!".$!; - $stat = decode("UTF-8", $stat); - open FILE,">>sql.log"; - print FILE "$stat\n"; - close FILE; - my $sth = $dbh->prepare($stat); - $retdata->{success} = $dbh->do($stat) or return $retdata->{error} = "dbexec ".$dbh->errstr. "- SQL: ".$stat; - $dbh->disconnect(); - return $retdata; -} - -#sub dbqueryarray(){ -# my $self = shift; -# my $stat = shift; -# my @retdata = (); -# my $dbh = DBI->connect('DBI:SQLite:dbname='.$self->{dbfile},"","",{PrintError=>1,RaiseError=>1,AutoCommit=>1,mysql_enable_utf8 => 1}) or return $retdata[0]->{error} = "dbquery Connection Error!".$!; -# #$stat = encode("utf8", $stat); -# #open FILE,">>/tmp/sql.log"; -# #print "$stat\n"; -# # close FILE; -# my $sth = $dbh->prepare($stat); -# -# $sth->execute() or print "dbquery: ".$sth->errstr; -# my $count = 0; -# -# while(my $valdata = $sth->fetchrow_arrayref()) -# { -# if ($valdata == undef){ last;} -# my @rdata = (); -# foreach my $k (@{$valdata}){ -# push @rdata,decode( "utf8", $k); -# } -# push @retdata,\@rdata; -# } -# -# $sth->finish(); -# $dbh->disconnect(); -# #%retdata = sort {$a <=> $b} keys %retdata; -# return @retdata; -#} - - -sub create_ddl_insert(){ - my $self = shift; - my $data = shift; - my $fields = (); - my @ddl = (); - foreach my $f (keys(%{$data})){ - if ($f =~ /\_/){ - my $t = substr($f,0,index($f,"_")); - my $c = substr($f,length($t)+1); - #my ($t,$c) = $f =~ m/(.+)\_(.+)/; - $fields->{$t}->{$c} = $data->{$f}; - } - } - - foreach my $tb (keys(%{$fields})){ - my @sqlcol = (); - my @sqlval = (); - foreach my $c (keys(%{$fields->{$tb}})){ - my $v = $fields->{$tb}->{$c}; - $v =~ s/'/''/g; - push (@sqlcol,$c); - if ($v eq ''){ - $v = 'null'; - } else { - $v = "'".$v."'"; - } - push (@sqlval,$v); - } - push(@ddl,"INSERT INTO ".$tb." (".join(",",@sqlcol).") VALUES (".join(",",@sqlval).");"); - } - return @ddl; -} - -sub create_ddl_insert_json(){ - my $self = shift; - my $schema = shift; - my $table = shift; - my $columns = shift; - my $data = shift; - my @ddl = (); - my @sqlcol = (); - my @sqlval = (); - foreach my $c (keys(%{$data})){ - #if (exists($columns->{$c})){ - push (@sqlcol,'"'.$c.'"'); - my $v = $data->{$c}; - - if ($v eq ''){ - $v = 'null'; - }elsif ($v =~ /^data:.+;base64,/){ - $v =~ s/'/''/g; - $v = "'".$v."'"; - } - else { - $v= uri_decode($v); - $v =~ s/'/''/g; - if ($columns->{$c}->{data_type} eq "ARRAY"){ - if (ref($data->{$c}) eq "ARRAY"){ - $v = "{\"".join("\",\"",@{$data->{$c}})."\"}"; - } - else { - $v = 'null'; - } - $v =~ s/""/null/g; - }elsif ($columns->{$c}->{data_type} =~ /^timestamp/ ){ - - }elsif($columns->{$c}->{data_type} eq "date"){ - - }elsif($columns->{$c}->{data_type} eq "time"){ - - } - $v = "'".$v."'"; - } - push (@sqlval,$v); - #} - } - return "INSERT INTO ".$schema.".\"".$table."\" (".join(",",@sqlcol).") VALUES (".join(",",@sqlval).");"; -} - -sub create_ddl_update(){ - my $self = shift; - my $data = shift; - my $fields = (); - my @ddl = (); - foreach my $f (keys(%{$data})){ - if ($f =~ /^ident_/){ - my $fx = substr($f,6); - my $t = substr($fx,0,index($fx,"_")); - my $c = substr($fx,length($t)+1); - #my ($t,$c) = $f =~ m/^ident_(.+)\_([a-z0-9|\_]+)/; - $fields->{$t}->{cond}->{$c} = $data->{$f}; - } elsif ( ($f !~ /^ident/) && ($f =~ /.+\_.+/) ){ - my $t = substr($f,0,index($f,"_")); - my $c = substr($f,length($t)+1); - #my ($t,$c) = $f =~ m/^(.+)\_([a-z0-9|\_]+)/; - $fields->{$t}->{fields}->{$c} = $data->{$f}; - } - } - foreach my $tb (keys(%{$fields})){ - my @sqlupd = (); - my @sqlcond = (); - foreach my $c (keys(%{$fields->{$tb}->{fields}})){ - - my $v = $fields->{$tb}->{fields}->{$c}; - $v =~ s/'/''/g; - if ($v eq ''){ - $v = 'null'; - } else { - $v = "'".$v."'"; - } - push (@sqlupd,$c."=".$v); - } - foreach my $c (keys(%{$fields->{$tb}->{cond}})){ - my $v = $fields->{$tb}->{cond}->{$c}; - $v =~ s/'/''/g; - if ($v eq ''){ - $v = 'null'; - } else { - $v = "'".$v."'"; - } - push (@sqlcond,$c."=".$v); - } - push(@ddl,"UPDATE ".$tb." SET ".join(",",@sqlupd)." WHERE ".join(" AND ",@sqlcond).";"); - } - - return @ddl; -} - -sub create_ddl_delete(){ - my $self = shift; - my $data = shift; - my $fields = (); - my @ddl = (); - my @refcols = (); - my $refdata = (); - foreach my $f (keys(%{$data})){ - if ($f =~ /^ident_/){ - my ($t,$c) = $f =~ m/ident_(.+)\_(.+)/; - - $fields->{$t}->{cond}->{$c} = $data->{$f}; - push(@refcols,"'".$c.'_'.$t."'"); - $refdata->{$c.'_'.$t} = $data->{$f}; - } - } - -# my $ref = $self->dbquerysorted("select TABLE_NAME,COLUMN_NAME from information_schema.KEY_COLUMN_USAGE where COLUMN_NAME in (".join(",",@refcols).") and CONSTRAINT_SCHEMA='".$self->{dbname}."';"); -# foreach my $r (keys(%{$ref})){ -# my $refv = $refdata->{$ref->{$r}->{COLUMN_NAME}}; -# if ($refv eq ''){ -# $refv = ' is null'; -# } else { -# $refv =~ s/'/''/g; -# $refv = "='".$refv."'"; -# } -# push(@ddl,"DELETE from ".$ref->{$r}->{TABLE_NAME}." where ".$ref->{$r}->{COLUMN_NAME}.$refv.";"); -# } - foreach my $tb (keys(%{$fields})){ - my @sqlcond = (); - foreach my $c (keys(%{$fields->{$tb}->{cond}})){ - my $v = $fields->{$tb}->{cond}->{$c}; - $v =~ s/'/''/g; - push (@sqlcond,$c."='".$v."'"); - } - push(@ddl,"DELETE FROM ".$tb." WHERE ".join(" AND ",@sqlcond).";"); - } - return @ddl; -} - -1; diff --git a/dev/lib.old/prefs.cgi b/dev/lib.old/prefs.cgi deleted file mode 100644 index cb314e3..0000000 --- a/dev/lib.old/prefs.cgi +++ /dev/null @@ -1,77 +0,0 @@ -#!/usr/bin/perl - -use strict; -use lib ('../lib'); -use CGI; -use CGI::Cookie; -use CGI::Carp qw/fatalsToBrowser/; -use File::Basename; -use JSON::PP; -use lib ('./lib'); -use session; -use dksdb; -my $cgi = new CGI(); -my $scriptpath = $cgi->url(-absolute => 1); -my $p = (); -my @params = $cgi->param(); -foreach my $pe (@params){ - $p->{$pe} = $cgi->param($pe); -} -my $html->{result} = (); -$p->{sid} = $cgi->cookie('dksapps'); -my $se = session->new(); -my $sess = $se->getsession($p->{sid}); -print $cgi->header(-type=>"application/json", -charset => "utf-8"); -if ($sess == undef){ - $html->{error} = "No Authorisation"; - print JSON::PP::encode_json($html); - exit(0); -} -# $html->{p} = $p; -# $html->{sess} =$sess; -#my $datapath = $ENV{"DOCUMENT_ROOT"}.dirname(dirname($scriptpath)).'/data/'; -if (exists($p->{get}) ){ - if ($p->{app} ne "" && exists($sess->{$p->{app}})){ - if ($p->{get} eq "auth"){ - $sess->{$p->{app}} =~ s/\r//g; - $sess->{$p->{app}} =~ s/\n//g; - $sess->{$p->{app}} =~ s/\s+/ /g; - $html->{result} = JSON::PP::decode_json($sess->{$p->{app}}); - } else { - my $db = dksdb->new(); - my $prop = $db->dbquerysorted("select preference from modulepreferences where id_user=".$sess->{id}." and appident='".$p->{app}."' and page='".$p->{get}."';"); - if (keys(%{$prop}) > 0){ - my $pref = JSON::PP::decode_json($prop->{0}->{preference}); - $html->{result} = $pref; - } - } - } -} -if (exists($p->{set}) && (exists($p->{page}))){ - if ($p->{app} ne "" && exists($sess->{$p->{app}})){ - if (($p->{set} ne "") && ($p->{set} =~ /^{/) && ($p->{set} =~ /}$/)){ - my $nd = JSON::PP::decode_json($p->{set}); - my $db = dksdb->new(); - my $prop = $db->dbquerysorted("select preference from modulepreferences where id_user=".$sess->{id}." and appident='".$p->{app}."' and page='".$p->{page}."'"); - my $oprop = (); - if (keys(%{$prop}) > 0){ - $oprop = JSON::PP::decode_json($prop->{0}->{preference}); - } - foreach my $k (keys(%{$nd})){ - $oprop->{$k} = $nd->{$k}; - } - if (keys(%{$prop}) > 0){ - #update - $db->dbexec("update modulepreferences set preference='".JSON::PP::encode_json($oprop)."' where id_user=".$sess->{id}." and appident='".$p->{app}."' and page='".$p->{page}."'"); - } else { - #insert - - $db->dbexec("INSERT INTO modulepreferences (id_user,appident,page,preference) VALUES (".$sess->{id}.",'".$p->{app}."','".$p->{page}."','".JSON::PP::encode_json($oprop)."');"); - } - } - } - $html->{result} = 1; -} -print JSON::PP::encode_json($html); - - diff --git a/dev/lib.old/sendEmail b/dev/lib.old/sendEmail deleted file mode 100644 index c639439..0000000 --- a/dev/lib.old/sendEmail +++ /dev/null @@ -1,2235 +0,0 @@ -#!/usr/bin/perl -w -############################################################################## -## sendEmail -## Written by: Brandon Zehm -## -## License: -## sendEmail (hereafter referred to as "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. -## When redistributing modified versions of this source code it is recommended -## that that this disclaimer and the above coder's names are included in the -## modified code. -## -## Disclaimer: -## This program is provided with no warranty of any kind, either expressed or -## implied. It is the responsibility of the user (you) to fully research and -## comprehend the usage of this program. As with any tool, it can be misused, -## either intentionally (you're a vandal) or unintentionally (you're a moron). -## THE AUTHOR(S) IS(ARE) NOT RESPONSIBLE FOR ANYTHING YOU DO WITH THIS PROGRAM -## or anything that happens because of your use (or misuse) of this program, -## including but not limited to anything you, your lawyers, or anyone else -## can dream up. And now, a relevant quote directly from the GPL: -## -## NO WARRANTY -## -## 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY -## FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN -## OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES -## PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED -## OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF -## MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS -## TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE -## PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, -## REPAIR OR CORRECTION. -## -############################################################################## -use strict; -use IO::Socket; - - -######################## -## Global Variables ## -######################## - -my %conf = ( - ## General - "programName" => $0, ## The name of this program - "version" => '1.56', ## The version of this program - "authorName" => 'Brandon Zehm', ## Author's Name - "authorEmail" => 'caspian@dotconf.net', ## Author's Email Address - "timezone" => '+0000', ## We always use +0000 for the time zone - "hostname" => 'changeme', ## Used in printmsg() for all output (is updated later in the script). - "debug" => 0, ## Default debug level - "error" => '', ## Error messages will often be stored here - - ## Logging - "stdout" => 1, - "logging" => 0, ## If this is true the printmsg function prints to the log file - "logFile" => '', ## If this is specified (form the command line via -l) this file will be used for logging. - - ## Network - "server" => 'localhost', ## Default SMTP server - "port" => 25, ## Default port - "bindaddr" => '', ## Default local bind address - "alarm" => '', ## Default timeout for connects and reads, this gets set from $opt{'timeout'} - "tls_client" => 0, ## If TLS is supported by the client (us) - "tls_server" => 0, ## If TLS is supported by the remote SMTP server - - ## Email - "delimiter" => "----MIME delimiter for sendEmail-" ## MIME Delimiter - . rand(1000000), ## Add some randomness to the delimiter - "Message-ID" => rand(1000000) . "-sendEmail", ## Message-ID for email header - -); - - -## This hash stores the options passed on the command line via the -o option. -my %opt = ( - ## Addressing - "reply-to" => '', ## Reply-To field - - ## Message - "message-file" => '', ## File to read message body from - "message-header" => '', ## Additional email header line(s) - "message-format" => 'normal', ## If "raw" is specified the message is sent unmodified - "message-charset" => 'iso-8859-1', ## Message character-set - "message-content-type" => 'auto', ## auto, text, html or an actual string to put into the content-type header. - - ## Network - "timeout" => 60, ## Default timeout for connects and reads, this is copied to $conf{'alarm'} later. - "fqdn" => 'changeme', ## FQDN of this machine, used during SMTP communication (is updated later in the script). - - ## eSMTP - "username" => '', ## Username used in SMTP Auth - "password" => '', ## Password used in SMTP Auth - "tls" => 'auto', ## Enable or disable TLS support. Options: auto, yes, no - -); - -## More variables used later in the program -my $SERVER; -my $CRLF = "\015\012"; -my $subject = ''; -my $header = ''; -my $message = ''; -my $from = ''; -my @to = (); -my @cc = (); -my @bcc = (); -my @attachments = (); -my @attachments_names = (); - -## For printing colors to the console -my ${colorRed} = "\033[31;1m"; -my ${colorGreen} = "\033[32;1m"; -my ${colorCyan} = "\033[36;1m"; -my ${colorWhite} = "\033[37;1m"; -my ${colorNormal} = "\033[m"; -my ${colorBold} = "\033[1m"; -my ${colorNoBold} = "\033[0m"; - -## Don't use shell escape codes on Windows systems -if ($^O =~ /win/i) { - ${colorRed} = ${colorGreen} = ${colorCyan} = ${colorWhite} = ${colorNormal} = ${colorBold} = ${colorNoBold} = ""; -} - -## Load IO::Socket::SSL if it's available -eval { require IO::Socket::SSL; }; -if ($@) { $conf{'tls_client'} = 0; } -else { $conf{'tls_client'} = 1; } - - - - - - -############################# -## ## -## FUNCTIONS ## -## ## -############################# - - - - - -############################################################################################### -## Function: initialize () -## -## Does all the script startup jibberish. -## -############################################################################################### -sub initialize { - - ## Set STDOUT to flush immediatly after each print - $| = 1; - - ## Intercept signals - $SIG{'QUIT'} = sub { quit("EXITING: Received SIG$_[0]", 1); }; - $SIG{'INT'} = sub { quit("EXITING: Received SIG$_[0]", 1); }; - $SIG{'KILL'} = sub { quit("EXITING: Received SIG$_[0]", 1); }; - $SIG{'TERM'} = sub { quit("EXITING: Received SIG$_[0]", 1); }; - - ## ALARM and HUP signals are not supported in Win32 - unless ($^O =~ /win/i) { - $SIG{'HUP'} = sub { quit("EXITING: Received SIG$_[0]", 1); }; - $SIG{'ALRM'} = sub { quit("EXITING: Received SIG$_[0]", 1); }; - } - - ## Fixup $conf{'programName'} - $conf{'programName'} =~ s/(.)*[\/,\\]//; - $0 = $conf{'programName'} . " " . join(" ", @ARGV); - - ## Fixup $conf{'hostname'} and $opt{'fqdn'} - if ($opt{'fqdn'} eq 'changeme') { $opt{'fqdn'} = get_hostname(1); } - if ($conf{'hostname'} eq 'changeme') { $conf{'hostname'} = $opt{'fqdn'}; $conf{'hostname'} =~ s/\..*//; } - - return(1); -} - - - - - - - - - - - - - - - -############################################################################################### -## Function: processCommandLine () -## -## Processes command line storing important data in global vars (usually %conf) -## -############################################################################################### -sub processCommandLine { - - - ############################ - ## Process command line ## - ############################ - - my @ARGS = @ARGV; ## This is so later we can re-parse the command line args later if we need to - my $numargv = @ARGS; - help() unless ($numargv); - my $counter = 0; - - for ($counter = 0; $counter < $numargv; $counter++) { - - if ($ARGS[$counter] =~ /^-h$/i) { ## Help ## - help(); - } - - elsif ($ARGS[$counter] eq "") { ## Ignore null arguments - ## Do nothing - } - - elsif ($ARGS[$counter] =~ /^--help/) { ## Topical Help ## - $counter++; - if ($ARGS[$counter] && $ARGS[$counter] !~ /^-/) { - helpTopic($ARGS[$counter]); - } - else { - help(); - } - } - - elsif ($ARGS[$counter] =~ /^-o$/i) { ## Options specified with -o ## - $counter++; - ## Loop through each option passed after the -o - while ($ARGS[$counter] && $ARGS[$counter] !~ /^-/) { - - if ($ARGS[$counter] !~ /(\S+)=(\S.*)/) { - printmsg("WARNING => Name/Value pair [$ARGS[$counter]] is not properly formatted", 0); - printmsg("WARNING => Arguments proceeding -o should be in the form of \"name=value\"", 0); - } - else { - if (exists($opt{$1})) { - if ($1 eq 'message-header') { - $opt{$1} .= $2 . $CRLF; - } - else { - $opt{$1} = $2; - } - printmsg("DEBUG => Assigned \$opt{} key/value: $1 => $2", 3); - } - else { - printmsg("WARNING => Name/Value pair [$ARGS[$counter]] will be ignored: unknown key [$1]", 0); - printmsg("HINT => Try the --help option to find valid command line arguments", 1); - } - } - $counter++; - } $counter--; - } - - elsif ($ARGS[$counter] =~ /^-f$/) { ## From ## - $counter++; - if ($ARGS[$counter] && $ARGS[$counter] !~ /^-/) { $from = $ARGS[$counter]; } - else { printmsg("WARNING => The argument after -f was not an email address!", 0); $counter--; } - } - - elsif ($ARGS[$counter] =~ /^-t$/) { ## To ## - $counter++; - while ($ARGS[$counter] && ($ARGS[$counter] !~ /^-/)) { - if ($ARGS[$counter] =~ /[;,]/) { - push (@to, split(/[;,]/, $ARGS[$counter])); - } - else { - push (@to,$ARGS[$counter]); - } - $counter++; - } $counter--; - } - - elsif ($ARGS[$counter] =~ /^-cc$/) { ## Cc ## - $counter++; - while ($ARGS[$counter] && ($ARGS[$counter] !~ /^-/)) { - if ($ARGS[$counter] =~ /[;,]/) { - push (@cc, split(/[;,]/, $ARGS[$counter])); - } - else { - push (@cc,$ARGS[$counter]); - } - $counter++; - } $counter--; - } - - elsif ($ARGS[$counter] =~ /^-bcc$/) { ## Bcc ## - $counter++; - while ($ARGS[$counter] && ($ARGS[$counter] !~ /^-/)) { - if ($ARGS[$counter] =~ /[;,]/) { - push (@bcc, split(/[;,]/, $ARGS[$counter])); - } - else { - push (@bcc,$ARGS[$counter]); - } - $counter++; - } $counter--; - } - - elsif ($ARGS[$counter] =~ /^-m$/) { ## Message ## - $counter++; - $message = ""; - while ($ARGS[$counter] && $ARGS[$counter] !~ /^-/) { - if ($message) { $message .= " "; } - $message .= $ARGS[$counter]; - $counter++; - } $counter--; - - ## Replace '\n' with $CRLF. - ## This allows newlines with messages sent on the command line - $message =~ s/\\n/$CRLF/g; - } - - elsif ($ARGS[$counter] =~ /^-u$/) { ## Subject ## - $counter++; - $subject = ""; - while ($ARGS[$counter] && $ARGS[$counter] !~ /^-/) { - if ($subject) { $subject .= " "; } - $subject .= $ARGS[$counter]; - $counter++; - } $counter--; - } - - elsif ($ARGS[$counter] =~ /^-s$/) { ## Server ## - $counter++; - if ($ARGS[$counter] && $ARGS[$counter] !~ /^-/) { - $conf{'server'} = $ARGS[$counter]; - if ($conf{'server'} =~ /:/) { ## Port ## - ($conf{'server'},$conf{'port'}) = split(":",$conf{'server'}); - } - } - else { printmsg("WARNING - The argument after -s was not the server!", 0); $counter--; } - } - - elsif ($ARGS[$counter] =~ /^-b$/) { ## Bind Address ## - $counter++; - if ($ARGS[$counter] && $ARGS[$counter] !~ /^-/) { - $conf{'bindaddr'} = $ARGS[$counter]; - } - else { printmsg("WARNING - The argument after -b was not the bindaddr!", 0); $counter--; } - } - - elsif ($ARGS[$counter] =~ /^-a$/) { ## Attachments ## - $counter++; - while ($ARGS[$counter] && ($ARGS[$counter] !~ /^-/)) { - push (@attachments,$ARGS[$counter]); - $counter++; - } $counter--; - } - - elsif ($ARGS[$counter] =~ /^-xu$/) { ## AuthSMTP Username ## - $counter++; - if ($ARGS[$counter] && $ARGS[$counter] !~ /^-/) { - $opt{'username'} = $ARGS[$counter]; - } - else { - printmsg("WARNING => The argument after -xu was not valid username!", 0); - $counter--; - } - } - - elsif ($ARGS[$counter] =~ /^-xp$/) { ## AuthSMTP Password ## - $counter++; - if ($ARGS[$counter] && $ARGS[$counter] !~ /^-/) { - $opt{'password'} = $ARGS[$counter]; - } - else { - printmsg("WARNING => The argument after -xp was not valid password!", 0); - $counter--; - } - } - - elsif ($ARGS[$counter] =~ /^-l$/) { ## Logging ## - $counter++; - $conf{'logging'} = 1; - if ($ARGS[$counter] && $ARGS[$counter] !~ /^-/) { $conf{'logFile'} = $ARGS[$counter]; } - else { printmsg("WARNING - The argument after -l was not the log file!", 0); $counter--; } - } - - elsif ($ARGS[$counter] =~ s/^-v+//i) { ## Verbosity ## - my $tmp = (length($&) - 1); - $conf{'debug'} += $tmp; - } - - elsif ($ARGS[$counter] =~ /^-q$/) { ## Quiet ## - $conf{'stdout'} = 0; - } - - else { - printmsg("Error: \"$ARGS[$counter]\" is not a recognized option!", 0); - help(); - } - - } - - - - - - - - - ################################################### - ## Verify required variables are set correctly ## - ################################################### - - ## Make sure we have something in $conf{hostname} and $opt{fqdn} - if ($opt{'fqdn'} =~ /\./) { - $conf{'hostname'} = $opt{'fqdn'}; - $conf{'hostname'} =~ s/\..*//; - } - - if (!$conf{'server'}) { $conf{'server'} = 'localhost'; } - if (!$conf{'port'}) { $conf{'port'} = 25; } - if (!$from) { - quit("ERROR => You must specify a 'from' field! Try --help.", 1); - } - if ( ((scalar(@to)) + (scalar(@cc)) + (scalar(@bcc))) <= 0) { - quit("ERROR => You must specify at least one recipient via -t, -cc, or -bcc", 1); - } - - ## Make sure email addresses look OK. - foreach my $addr (@to, @cc, @bcc, $from, $opt{'reply-to'}) { - if ($addr) { - if (!returnAddressParts($addr)) { - printmsg("ERROR => Can't use improperly formatted email address: $addr", 0); - printmsg("HINT => Try viewing the extended help on addressing with \"--help addressing\"", 1); - quit("", 1); - } - } - } - - ## Make sure all attachments exist. - foreach my $file (@attachments) { - if ( (! -f $file) or (! -r $file) ) { - printmsg("ERROR => The attachment [$file] doesn't exist!", 0); - printmsg("HINT => Try specifying the full path to the file or reading extended help with \"--help message\"", 1); - quit("", 1); - } - } - - if ($conf{'logging'} and (!$conf{'logFile'})) { - quit("ERROR => You used -l to enable logging but didn't specify a log file!", 1); - } - - if ( $opt{'username'} ) { - if (!$opt{'password'}) { - ## Prompt for a password since one wasn't specified with the -xp option. - $SIG{'ALRM'} = sub { quit("ERROR => Timeout waiting for password inpupt", 1); }; - alarm(60) if ($^O !~ /win/i); ## alarm() doesn't work in win32 - print "Password: "; - $opt{'password'} = ; chomp $opt{'password'}; - if (!$opt{'password'}) { - quit("ERROR => A username for SMTP authentication was specified, but no password!", 1); - } - } - } - - ## Validate the TLS setting - $opt{'tls'} = lc($opt{'tls'}); - if ($opt{'tls'} !~ /^(auto|yes|no)$/) { - quit("ERROR => Invalid TLS setting ($opt{'tls'}). Must be one of auto, yes, or no.", 1); - } - - ## If TLS is set to "yes", make sure sendEmail loaded the libraries needed. - if ($opt{'tls'} eq 'yes' and $conf{'tls_client'} == 0) { - quit("ERROR => No TLS support! SendEmail can't load required libraries. (try installing Net::SSLeay and IO::Socket::SSL)", 1); - } - - ## Return 0 errors - return(0); -} - - - - - - - - - - - - - - - - -## getline($socketRef) -sub getline { - my ($socketRef) = @_; - local ($/) = "\r\n"; - return $$socketRef->getline; -} - - - - -## Receive a (multiline?) SMTP response from ($socketRef) -sub getResponse { - my ($socketRef) = @_; - my ($tmp, $reply); - local ($/) = "\r\n"; - return undef unless defined($tmp = getline($socketRef)); - return("getResponse() socket is not open") unless ($$socketRef->opened); - ## Keep reading lines if it's a multi-line response - while ($tmp =~ /^\d{3}-/o) { - $reply .= $tmp; - return undef unless defined($tmp = getline($socketRef)); - } - $reply .= $tmp; - $reply =~ s/\r?\n$//o; - return $reply; -} - - - - -############################################################################################### -## Function: SMTPchat ( [string $command] ) -## -## Description: Sends $command to the SMTP server (on SERVER) and awaits a successful -## reply form the server. If the server returns an error, or does not reply -## within $conf{'alarm'} seconds an error is generated. -## NOTE: $command is optional, if no command is specified then nothing will -## be sent to the server, but a valid response is still required from the server. -## -## Input: [$command] A (optional) valid SMTP command (ex. "HELO") -## -## -## Output: Returns zero on success, or non-zero on error. -## Error messages will be stored in $conf{'error'} -## A copy of the last SMTP response is stored in the global variable -## $conf{'SMTPchat_response'} -## -## -## Example: SMTPchat ("HELO mail.isp.net"); -############################################################################################### -sub SMTPchat { - my ($command) = @_; - - printmsg("INFO => Sending: \t$command", 1) if ($command); - - ## Send our command - print $SERVER "$command$CRLF" if ($command); - - ## Read a response from the server - $SIG{'ALRM'} = sub { $conf{'error'} = "alarm"; $SERVER->close(); }; - alarm($conf{'alarm'}) if ($^O !~ /win/i); ## alarm() doesn't work in win32; - my $result = $conf{'SMTPchat_response'} = getResponse(\$SERVER); - alarm(0) if ($^O !~ /win/i); ## alarm() doesn't work in win32; - - ## Generate an alert if we timed out - if ($conf{'error'} eq "alarm") { - $conf{'error'} = "ERROR => Timeout while reading from $conf{'server'}:$conf{'port'} There was no response after $conf{'alarm'} seconds."; - return(1); - } - - ## Make sure the server actually responded - if (!$result) { - $conf{'error'} = "ERROR => $conf{'server'}:$conf{'port'} returned a zero byte response to our query."; - return(2); - } - - ## Validate the response - if (evalSMTPresponse($result)) { - ## conf{'error'} will already be set here - return(2); - } - - ## Print the success messsage - printmsg($conf{'error'}, 1); - - ## Return Success - return(0); -} - - - - - - - - - - - - -############################################################################################### -## Function: evalSMTPresponse (string $message ) -## -## Description: Searches $message for either an SMTP success or error code, and returns -## 0 on success, and the actual error code on error. -## -## -## Input: $message Data received from a SMTP server (ex. "220 -## -## -## Output: Returns zero on success, or non-zero on error. -## Error messages will be stored in $conf{'error'} -## -## -## Example: SMTPchat ("HELO mail.isp.net"); -############################################################################################### -sub evalSMTPresponse { - my ($message) = @_; - - ## Validate input - if (!$message) { - $conf{'error'} = "ERROR => No message was passed to evalSMTPresponse(). What happened?"; - return(1) - } - - printmsg("DEBUG => evalSMTPresponse() - Checking for SMTP success or error status in the message: $message ", 3); - - ## Look for a SMTP success code - if ($message =~ /^([23]\d\d)/) { - printmsg("DEBUG => evalSMTPresponse() - Found SMTP success code: $1", 2); - $conf{'error'} = "SUCCESS => Received: \t$message"; - return(0); - } - - ## Look for a SMTP error code - if ($message =~ /^([45]\d\d)/) { - printmsg("DEBUG => evalSMTPresponse() - Found SMTP error code: $1", 2); - $conf{'error'} = "ERROR => Received: \t$message"; - return($1); - } - - ## If no SMTP codes were found return an error of 1 - $conf{'error'} = "ERROR => Received a message with no success or error code. The message received was: $message"; - return(2); - -} - - - - - - - - - - -######################################################### -# SUB: &return_month(0,1,etc) -# returns the name of the month that corrosponds -# with the number. returns 0 on error. -######################################################### -sub return_month { - my $x = $_[0]; - if ($x == 0) { return 'Jan'; } - if ($x == 1) { return 'Feb'; } - if ($x == 2) { return 'Mar'; } - if ($x == 3) { return 'Apr'; } - if ($x == 4) { return 'May'; } - if ($x == 5) { return 'Jun'; } - if ($x == 6) { return 'Jul'; } - if ($x == 7) { return 'Aug'; } - if ($x == 8) { return 'Sep'; } - if ($x == 9) { return 'Oct'; } - if ($x == 10) { return 'Nov'; } - if ($x == 11) { return 'Dec'; } - return (0); -} - - - - - - - - - - - - - - - - -######################################################### -# SUB: &return_day(0,1,etc) -# returns the name of the day that corrosponds -# with the number. returns 0 on error. -######################################################### -sub return_day { - my $x = $_[0]; - if ($x == 0) { return 'Sun'; } - if ($x == 1) { return 'Mon'; } - if ($x == 2) { return 'Tue'; } - if ($x == 3) { return 'Wed'; } - if ($x == 4) { return 'Thu'; } - if ($x == 5) { return 'Fri'; } - if ($x == 6) { return 'Sat'; } - return (0); -} - - - - - - - - - - - - - - - - -############################################################################################### -## Function: returnAddressParts(string $address) -## -## Description: Returns a two element array containing the "Name" and "Address" parts of -## an email address. -## -## Example: "Brandon Zehm " -## would return: ("Brandon Zehm", "caspian@dotconf.net"); -## -## "caspian@dotconf.net" -## would return: ("caspian@dotconf.net", "caspian@dotconf.net") -############################################################################################### -sub returnAddressParts { - my $input = $_[0]; - my $name = ""; - my $address = ""; - - ## Make sure to fail if it looks totally invalid - if ($input !~ /(\S+\@\S+)/) { - $conf{'error'} = "ERROR => The address [$input] doesn't look like a valid email address, ignoring it"; - return(undef()); - } - - ## Check 1, should find addresses like: "Brandon Zehm " - elsif ($input =~ /^\s*(\S(.*\S)?)\s*<(\S+\@\S+)>/o) { - ($name, $address) = ($1, $3); - } - - ## Otherwise if that failed, just get the address: - elsif ($input =~ /<(\S+\@\S+)>/o) { - $name = $address = $1; - } - - ## Or maybe it was formatted this way: caspian@dotconf.net - elsif ($input =~ /(\S+\@\S+)/o) { - $name = $address = $1; - } - - ## Something stupid happened, just return an error. - unless ($name and $address) { - printmsg("ERROR => Couldn't parse the address: $input", 0); - printmsg("HINT => If you think this should work, consider reporting this as a bug to $conf{'authorEmail'}", 1); - return(undef()); - } - - ## Make sure there aren't invalid characters in the address, and return it. - my $ctrl = '\000-\037'; - my $nonASCII = '\x80-\xff'; - if ($address =~ /[<> ,;:"'\[\]\\$ctrl$nonASCII]/) { - printmsg("WARNING => The address [$address] seems to contain invalid characters: continuing anyway", 0); - } - return($name, $address); -} - - - - - - - - - - - - - - - - -############################################################################################### -## Function: base64_encode(string $data, bool $chunk) -## -## Description: Returns $data as a base64 encoded string. -## If $chunk is true, the encoded data is returned in 76 character long lines -## with the final \CR\LF removed. -## -## Note: This is only used from the smtp auth section of code. -## At some point it would be nice to merge the code that encodes attachments and this. -############################################################################################### -sub base64_encode { - my $data = $_[0]; - my $chunk = $_[1]; - my $tmp = ''; - my $base64 = ''; - my $CRLF = "\r\n"; - - ################################### - ## Convert binary data to base64 ## - ################################### - while ($data =~ s/(.{45})//s) { ## Get 45 bytes from the binary string - $tmp = substr(pack('u', $&), 1); ## Convert the binary to uuencoded text - chop($tmp); - $tmp =~ tr|` -_|AA-Za-z0-9+/|; ## Translate from uuencode to base64 - $base64 .= $tmp; - } - - ########################## - ## Encode the leftovers ## - ########################## - my $padding = ""; - if ( ($data) and (length($data) > 0) ) { - $padding = (3 - length($data) % 3) % 3; ## Set flag if binary data isn't divisible by 3 - $tmp = substr(pack('u', $data), 1); ## Convert the binary to uuencoded text - chop($tmp); - $tmp =~ tr|` -_|AA-Za-z0-9+/|; ## Translate from uuencode to base64 - $base64 .= $tmp; - } - - ############################ - ## Fix padding at the end ## - ############################ - $data = ''; - $base64 =~ s/.{$padding}$/'=' x $padding/e if $padding; ## Fix the end padding if flag (from above) is set - if ($chunk) { - while ($base64 =~ s/(.{1,76})//s) { ## Put $CRLF after each 76 characters - $data .= "$1$CRLF"; - } - } - else { - $data = $base64; - } - - ## Remove any trailing CRLF's - $data =~ s/(\r|\n)*$//s; - return($data); -} - - - - - - - - - -######################################################### -# SUB: send_attachment("/path/filename") -# Sends the mime headers and base64 encoded file -# to the email server. -######################################################### -sub send_attachment { - my ($filename) = @_; ## Get filename passed - my (@fields, $y, $filename_name, $encoding, ## Local variables - @attachlines, $content_type); - my $bin = 1; - - @fields = split(/\/|\\/, $filename); ## Get the actual filename without the path - $filename_name = pop(@fields); - push @attachments_names, $filename_name; ## FIXME: This is only used later for putting in the log file - - ########################## - ## Autodetect Mime Type ## - ########################## - - @fields = split(/\./, $filename_name); - $encoding = $fields[$#fields]; - - if ($encoding =~ /txt|text|log|conf|^c$|cpp|^h$|inc|m3u/i) { $content_type = 'text/plain'; } - elsif ($encoding =~ /html|htm|shtml|shtm|asp|php|cfm/i) { $content_type = 'text/html'; } - elsif ($encoding =~ /sh$/i) { $content_type = 'application/x-sh'; } - elsif ($encoding =~ /tcl/i) { $content_type = 'application/x-tcl'; } - elsif ($encoding =~ /pl$/i) { $content_type = 'application/x-perl'; } - elsif ($encoding =~ /js$/i) { $content_type = 'application/x-javascript'; } - elsif ($encoding =~ /man/i) { $content_type = 'application/x-troff-man'; } - elsif ($encoding =~ /gif/i) { $content_type = 'image/gif'; } - elsif ($encoding =~ /jpg|jpeg|jpe|jfif|pjpeg|pjp/i) { $content_type = 'image/jpeg'; } - elsif ($encoding =~ /tif|tiff/i) { $content_type = 'image/tiff'; } - elsif ($encoding =~ /xpm/i) { $content_type = 'image/x-xpixmap'; } - elsif ($encoding =~ /bmp/i) { $content_type = 'image/x-MS-bmp'; } - elsif ($encoding =~ /pcd/i) { $content_type = 'image/x-photo-cd'; } - elsif ($encoding =~ /png/i) { $content_type = 'image/png'; } - elsif ($encoding =~ /aif|aiff/i) { $content_type = 'audio/x-aiff'; } - elsif ($encoding =~ /wav/i) { $content_type = 'audio/x-wav'; } - elsif ($encoding =~ /mp2|mp3|mpa/i) { $content_type = 'audio/x-mpeg'; } - elsif ($encoding =~ /ra$|ram/i) { $content_type = 'audio/x-pn-realaudio'; } - elsif ($encoding =~ /mpeg|mpg/i) { $content_type = 'video/mpeg'; } - elsif ($encoding =~ /mov|qt$/i) { $content_type = 'video/quicktime'; } - elsif ($encoding =~ /avi/i) { $content_type = 'video/x-msvideo'; } - elsif ($encoding =~ /zip/i) { $content_type = 'application/x-zip-compressed'; } - elsif ($encoding =~ /tar/i) { $content_type = 'application/x-tar'; } - elsif ($encoding =~ /jar/i) { $content_type = 'application/java-archive'; } - elsif ($encoding =~ /exe|bin/i) { $content_type = 'application/octet-stream'; } - elsif ($encoding =~ /ppt|pot|ppa|pps|pwz/i) { $content_type = 'application/vnd.ms-powerpoint'; } - elsif ($encoding =~ /mdb|mda|mde/i) { $content_type = 'application/vnd.ms-access'; } - elsif ($encoding =~ /xls|xlt|xlm|xld|xla|xlc|xlw|xll/i) { $content_type = 'application/vnd.ms-excel'; } - elsif ($encoding =~ /doc|dot/i) { $content_type = 'application/msword'; } - elsif ($encoding =~ /rtf/i) { $content_type = 'application/rtf'; } - elsif ($encoding =~ /pdf/i) { $content_type = 'application/pdf'; } - elsif ($encoding =~ /tex/i) { $content_type = 'application/x-tex'; } - elsif ($encoding =~ /latex/i) { $content_type = 'application/x-latex'; } - elsif ($encoding =~ /vcf/i) { $content_type = 'application/x-vcard'; } - else { $content_type = 'application/octet-stream'; } - - - ############################ - ## Process the attachment ## - ############################ - - ##################################### - ## Generate and print MIME headers ## - ##################################### - - $y = "$CRLF--$conf{'delimiter'}$CRLF"; - $y .= "Content-Type: $content_type;$CRLF"; - $y .= " name=\"$filename_name\"$CRLF"; - $y .= "Content-Transfer-Encoding: base64$CRLF"; - $y .= "Content-Disposition: attachment; filename=\"$filename_name\"$CRLF"; - $y .= "$CRLF"; - print $SERVER $y; - - - ########################################################### - ## Convert the file to base64 and print it to the server ## - ########################################################### - - open (FILETOATTACH, $filename) || do { - printmsg("ERROR => Opening the file [$filename] for attachment failed with the error: $!", 0); - return(1); - }; - binmode(FILETOATTACH); ## Hack to make Win32 work - - my $res = ""; - my $tmp = ""; - my $base64 = ""; - while () { ## Read a line from the (binary) file - $res .= $_; - - ################################### - ## Convert binary data to base64 ## - ################################### - while ($res =~ s/(.{45})//s) { ## Get 45 bytes from the binary string - $tmp = substr(pack('u', $&), 1); ## Convert the binary to uuencoded text - chop($tmp); - $tmp =~ tr|` -_|AA-Za-z0-9+/|; ## Translate from uuencode to base64 - $base64 .= $tmp; - } - - ################################ - ## Print chunks to the server ## - ################################ - while ($base64 =~ s/(.{76})//s) { - print $SERVER "$1$CRLF"; - } - - } - - ################################### - ## Encode and send the leftovers ## - ################################### - my $padding = ""; - if ( ($res) and (length($res) >= 1) ) { - $padding = (3 - length($res) % 3) % 3; ## Set flag if binary data isn't divisible by 3 - $res = substr(pack('u', $res), 1); ## Convert the binary to uuencoded text - chop($res); - $res =~ tr|` -_|AA-Za-z0-9+/|; ## Translate from uuencode to base64 - } - - ############################ - ## Fix padding at the end ## - ############################ - $res = $base64 . $res; ## Get left overs from above - $res =~ s/.{$padding}$/'=' x $padding/e if $padding; ## Fix the end padding if flag (from above) is set - if ($res) { - while ($res =~ s/(.{1,76})//s) { ## Send it to the email server. - print $SERVER "$1$CRLF"; - } - } - - close (FILETOATTACH) || do { - printmsg("ERROR - Closing the filehandle for file [$filename] failed with the error: $!", 0); - return(2); - }; - - ## Return 0 errors - return(0); - -} - - - - - - - - - -############################################################################################### -## Function: $string = get_hostname (boot $fqdn) -## -## Description: Tries really hard to returns the short (or FQDN) hostname of the current -## system. Uses techniques and code from the Sys-Hostname module. -## -## Input: $fqdn A true value (1) will cause this function to return a FQDN hostname -## rather than a short hostname. -## -## Output: Returns a string -############################################################################################### -sub get_hostname { - ## Assign incoming parameters to variables - my ( $fqdn ) = @_; - my $hostname = ""; - - ## STEP 1: Get short hostname - - ## Load Sys::Hostname if it's available - eval { require Sys::Hostname; }; - unless ($@) { - $hostname = Sys::Hostname::hostname(); - } - - ## If that didn't get us a hostname, try a few other things - else { - ## Windows systems - if ($^O !~ /win/i) { - if ($ENV{'COMPUTERNAME'}) { $hostname = $ENV{'COMPUTERNAME'}; } - if (!$hostname) { $hostname = gethostbyname('localhost'); } - if (!$hostname) { chomp($hostname = `hostname 2> NUL`) }; - } - - ## Unix systems - else { - local $ENV{PATH} = '/usr/bin:/bin:/usr/sbin:/sbin'; ## Paranoia - - ## Try the environment first (Help! What other variables could/should I be checking here?) - if ($ENV{'HOSTNAME'}) { $hostname = $ENV{'HOSTNAME'}; } - - ## Try the hostname command - eval { local $SIG{__DIE__}; local $SIG{CHLD}; $hostname = `hostname 2>/dev/null`; chomp($hostname); } || - - ## Try POSIX::uname(), which strictly can't be expected to be correct - eval { local $SIG{__DIE__}; require POSIX; $hostname = (POSIX::uname())[1]; } || - - ## Try the uname command - eval { local $SIG{__DIE__}; $hostname = `uname -n 2>/dev/null`; chomp($hostname); }; - - } - - ## If we can't find anything else, return "" - if (!$hostname) { - print "WARNING => No hostname could be determined, please specify one with -o fqdn=FQDN option!\n"; - return("unknown"); - } - } - - ## Return the short hostname - unless ($fqdn) { - $hostname =~ s/\..*//; - return(lc($hostname)); - } - - ## STEP 2: Determine the FQDN - - ## First, if we already have one return it. - if ($hostname =~ /\w\.\w/) { return(lc($hostname)); } - - ## Next try using - eval { $fqdn = (gethostbyname($hostname))[0]; }; - if ($fqdn) { return(lc($fqdn)); } - return(lc($hostname)); -} - - - - - - - - -############################################################################################### -## Function: printmsg (string $message, int $level) -## -## Description: Handles all messages - printing them to the screen only if the messages -## $level is >= the global debug level. If $conf{'logFile'} is defined it -## will also log the message to that file. -## -## Input: $message A message to be printed, logged, etc. -## $level The debug level of the message. If -## not defined 0 will be assumed. 0 is -## considered a normal message, 1 and -## higher is considered a debug message. -## -## Output: Prints to STDOUT -## -## Assumptions: $conf{'hostname'} should be the name of the computer we're running on. -## $conf{'stdout'} should be set to 1 if you want to print to stdout -## $conf{'logFile'} should be a full path to a log file if you want that -## $conf{'debug'} should be an integer between 0 and 10. -## -## Example: printmsg("WARNING: We believe in generic error messages... NOT!", 0); -############################################################################################### -sub printmsg { - ## Assign incoming parameters to variables - my ( $message, $level ) = @_; - - ## Make sure input is sane - $level = 0 if (!defined($level)); - $message =~ s/\s+$//sgo; - $message =~ s/\r?\n/, /sgo; - - ## Continue only if the debug level of the program is >= message debug level. - if ($conf{'debug'} >= $level) { - - ## Get the date in the format: Dec 3 11:14:04 - my ($sec, $min, $hour, $mday, $mon) = localtime(); - $mon = ('Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec')[$mon]; - my $date = sprintf("%s %02d %02d:%02d:%02d", $mon, $mday, $hour, $min, $sec); - - ## Print to STDOUT always if debugging is enabled, or if conf{stdout} is true. - if ( ($conf{'debug'} >= 1) or ($conf{'stdout'} == 1) ) { - print "$date $conf{'hostname'} $conf{'programName'}\[$$\]: $message\n"; - } - - ## Print to the log file if $conf{'logging'} is true - if ($conf{'logFile'}) { - if (openLogFile($conf{'logFile'})) { $conf{'logFile'} = ""; printmsg("ERROR => Opening the file [$conf{'logFile'}] for appending returned the error: $!", 1); } - print LOGFILE "$date $conf{'hostname'} $conf{'programName'}\[$$\]: $message\n"; - } - - } - - ## Return 0 errors - return(0); -} - - - - - - - - - - - - -############################################################################################### -## FUNCTION: -## openLogFile ( $filename ) -## -## -## DESCRIPTION: -## Opens the file $filename and attaches it to the filehandle "LOGFILE". Returns 0 on success -## and non-zero on failure. Error codes are listed below, and the error message gets set in -## global variable $!. -## -## -## Example: -## openFile ("/var/log/sendEmail.log"); -## -############################################################################################### -sub openLogFile { - ## Get the incoming filename - my $filename = $_[0]; - - ## Make sure our file exists, and if the file doesn't exist then create it - if ( ! -f $filename ) { - print STDERR "NOTICE: The log file [$filename] does not exist. Creating it now with mode [0600].\n" if ($conf{'stdout'}); - open (LOGFILE, ">>$filename"); - close LOGFILE; - chmod (0600, $filename); - } - - ## Now open the file and attach it to a filehandle - open (LOGFILE,">>$filename") or return (1); - - ## Put the file into non-buffering mode - select LOGFILE; - $| = 1; - select STDOUT; - - ## Return success - return(0); -} - - - - - - - - -############################################################################################### -## Function: read_file (string $filename) -## -## Description: Reads the contents of a file and returns a two part array: -## ($status, $file-contents) -## $status is 0 on success, non-zero on error. -## -## Example: ($status, $file) = read_file("/etc/passwd"); -############################################################################################### -sub read_file { - my ( $filename ) = @_; - - ## If the value specified is a file, load the file's contents - if ( (-e $filename and -r $filename) ) { - my $FILE; - if(!open($FILE, ' ' . $filename)) { - return((1, "")); - } - my $file = ''; - while (<$FILE>) { - $file .= $_; - } - ## Strip an ending \r\n - $file =~ s/\r?\n$//os; - } - return((1, "")); -} - - - - - - - - - -############################################################################################### -## Function: quit (string $message, int $errorLevel) -## -## Description: Exits the program, optionally printing $message. It -## returns an exit error level of $errorLevel to the -## system (0 means no errors, and is assumed if empty.) -## -## Example: quit("Exiting program normally", 0); -############################################################################################### -sub quit { - my ( $message, $errorLevel ) = @_; - $errorLevel = 0 if (!defined($errorLevel)); - - ## Print exit message - if ($message) { - printmsg($message, 0); - } - - ## Exit - exit($errorLevel); -} - - - - - - - - - - - - -############################################################################################### -## Function: help () -## -## Description: For all those newbies ;) -## Prints a help message and exits the program. -## -############################################################################################### -sub help { -exit(1) if (!$conf{'stdout'}); -print <${colorNoBold} - -Synopsis: $conf{'programName'} -f ADDRESS [options] - - ${colorRed}Required:${colorNormal} - -f ADDRESS from (sender) email address - * At least one recipient required via -t, -cc, or -bcc - * Message body required via -m, STDIN, or -o message-file=FILE - - ${colorGreen}Common:${colorNormal} - -t ADDRESS [ADDR ...] to email address(es) - -u SUBJECT message subject - -m MESSAGE message body - -s SERVER[:PORT] smtp mail relay, default is $conf{'server'}:$conf{'port'} - - ${colorGreen}Optional:${colorNormal} - -a FILE [FILE ...] file attachment(s) - -cc ADDRESS [ADDR ...] cc email address(es) - -bcc ADDRESS [ADDR ...] bcc email address(es) - -xu USERNAME username for SMTP authentication - -xp PASSWORD password for SMTP authentication - - ${colorGreen}Paranormal:${colorNormal} - -b BINDADDR[:PORT] local host bind address - -l LOGFILE log to the specified file - -v verbosity, use multiple times for greater effect - -q be quiet (i.e. no STDOUT output) - -o NAME=VALUE advanced options, for details try: --help misc - -o message-content-type= - -o message-file=FILE -o message-format=raw - -o message-header=HEADER -o message-charset=CHARSET - -o reply-to=ADDRESS -o timeout=SECONDS - -o username=USERNAME -o password=PASSWORD - -o tls= -o fqdn=FQDN - - - ${colorGreen}Help:${colorNormal} - --help the helpful overview you're reading now - --help addressing explain addressing and related options - --help message explain message body input and related options - --help networking explain -s, -b, etc - --help output explain logging and other output options - --help misc explain -o options, TLS, SMTP auth, and more - -EOM -exit(1); -} - - - - - - - - - -############################################################################################### -## Function: helpTopic ($topic) -## -## Description: For all those newbies ;) -## Prints a help message and exits the program. -## -############################################################################################### -sub helpTopic { - exit(1) if (!$conf{'stdout'}); - my ($topic) = @_; - - CASE: { - - - - -## ADDRESSING - ($topic eq 'addressing') && do { - print <" - Just Address: "john.doe\@gmail.com" - -The "Full Name" method is useful if you want a name, rather than a plain -email address, to be displayed in the recipient's From, To, or Cc fields -when they view the message. - - -${colorGreen}Multiple Recipients${colorNormal} -The -t, -cc, and -bcc options each accept multiple addresses. They may be -specified by separating them by either a white space, comma, or semi-colon -separated list. You may also specify the -t, -cc, and -bcc options multiple -times, each occurance will append the new recipients to the respective list. - -Examples: -(I used "-t" in these examples, but it can be "-cc" or "-bcc" as well) - - * Space separated list: - -t jane.doe\@yahoo.com "John Doe " - - * Semi-colon separated list: - -t "jane.doe\@yahoo.com; John Doe " - - * Comma separated list: - -t "jane.doe\@yahoo.com, John Doe " - - * Multiple -t, -cc, or -bcc options: - -t "jane.doe\@yahoo.com" -t "John Doe " - - -EOM - last CASE; - }; - - - - - - -## MESSAGE - ($topic eq 'message') && do { - print < - -o message-header=EMAIL HEADER - -o message-charset=CHARSET - -o message-format=raw - --u SUBJECT - This option allows you to specify the subject for your email message. - It is not required (anymore) that the subject be quoted, although it - is recommended. The subject will be read until an argument starting - with a hyphen (-) is found. - Examples: - -u "Contact information while on vacation" - -u New Microsoft vulnerability discovered - --m MESSAGE - This option is one of three methods that allow you to specify the message - body for your email. The message may be specified on the command line - with this -m option, read from a file with the -o message-file=FILE - option, or read from STDIN if neither of these options are present. - - It is not required (anymore) that the message be quoted, although it is - recommended. The message will be read until an argument starting with a - hyphen (-) is found. - Examples: - -m "See you in South Beach, Hawaii. -Todd" - -m Please ensure that you upgrade your systems right away - - Multi-line message bodies may be specified with the -m option by putting - a "\\n" into the message. Example: - -m "This is line 1.\\nAnd this is line 2." - - HTML messages are supported, simply begin your message with "" and - sendEmail will properly label the mime header so MUAs properly render - the message. It is currently not possible without "-o message-format=raw" - to send a message with both text and html parts with sendEmail. - --o message-file=FILE - This option is one of three methods that allow you to specify the message - body for your email. To use this option simply specify a text file - containing the body of your email message. Examples: - -o message-file=/root/message.txt - -o message-file="C:\\Program Files\\output.txt" - --o message-content-type= - This option allows you to specify the content-type of the email. If your - email message is an html message but is being displayed as a text message - just add "-o message-content-type=html" to the command line to force it - to display as an html message. This actually just changes the Content-Type: - header. Advanced users will be happy to know that if you specify anything - other than the three options listed above it will use that as the vaule - for the Content-Type header. - --o message-header=EMAIL HEADER - This option allows you to specify additional email headers to be included. - To add more than one message header simply use this option on the command - line more than once. If you specify a message header that sendEmail would - normally generate the one you specified will be used in it's place. - Do not use this unless you know what you are doing! - Example: - To scare a Microsoft Outlook user you may want to try this: - -o message-header="X-Message-Flag: Message contains illegal content" - Example: - To request a read-receipt try this: - -o message-header="Disposition-Notification-To: " - Example: - To set the message priority try this: - -o message-header="X-Priority: 1" - Priority reference: 1=highest, 2=high, 3=normal, 4=low, 5=lowest - --o message-charset=CHARSET - This option allows you to specify the character-set for the message body. - The default is iso-8859-1. - --o message-format=raw - This option instructs sendEmail to assume the message (specified with -m, - read from STDIN, or read from the file specified in -o message-file=FILE) - is already a *complete* email message. SendEmail will not generate any - headers and will transmit the message as-is to the remote SMTP server. - Due to the nature of this option the following command line options will - be ignored when this one is used: - -u SUBJECT - -o message-header=EMAIL HEADER - -o message-charset=CHARSET - -a ATTACHMENT - - -${colorGreen}The Message Body${colorNormal} -The email message body may be specified in one of three ways: - 1) Via the -m MESSAGE command line option. - Example: - -m "This is the message body" - - 2) By putting the message body in a file and using the -o message-file=FILE - command line option. - Example: - -o message-file=/root/message.txt - - 3) By piping the message body to sendEmail when nither of the above command - line options were specified. - Example: - grep "ERROR" /var/log/messages | sendEmail -t you\@domain.com ... - -If the message body begins with "" then the message will be treated as -an HTML message and the MIME headers will be written so that a HTML capable -email client will display the message in it's HTML form. -Any of the above methods may be used with the -o message-format=raw option -to deliver an already complete email message. - - -EOM - last CASE; - }; - - - - - - -## MISC - ($topic eq 'misc') && do { - print < - -o timeout=SECONDS - -o fqdn=FQDN - --a ATTACHMENT [ATTACHMENT ...] - This option allows you to attach any number of files to your email message. - To specify more than one attachment, simply separate each filename with a - space. Example: -a file1.txt file2.txt file3.txt - --xu USERNAME - Alias for -o username=USERNAME - --xp PASSWORD - Alias for -o password=PASSWORD - --o username=USERNAME (synonym for -xu) - These options allow specification of a username to be used with SMTP - servers that require authentication. If a username is specified but a - password is not, you will be prompted to enter one at runtime. - --o password=PASSWORD (synonym for -xp) - These options allow specification of a password to be used with SMTP - servers that require authentication. If a username is specified but a - password is not, you will be prompted to enter one at runtime. - --o tls= - This option allows you to specify if TLS (SSL for SMTP) should be enabled - or disabled. The default, auto, will use TLS automatically if your perl - installation has the IO::Socket::SSL and Net::SSLeay modules available, - and if the remote SMTP server supports TLS. To require TLS for message - delivery set this to yes. To disable TLS support set this to no. A debug - level of one or higher will reveal details about the status of TLS. - --o timeout=SECONDS - This option sets the timeout value in seconds used for all network reads, - writes, and a few other things. - --o fqdn=FQDN - This option sets the Fully Qualified Domain Name used during the initial - SMTP greeting. Normally this is automatically detected, but in case you - need to manually set it for some reason or get a warning about detection - failing, you can use this to override the default. - - -EOM - last CASE; - }; - - - - - - -## NETWORKING - ($topic eq 'networking') && do { - print < - -o timeout=SECONDS - --s SERVER[:PORT] - This option allows you to specify the SMTP server sendEmail should - connect to to deliver your email message to. If this option is not - specified sendEmail will try to connect to localhost:25 to deliver - the message. THIS IS MOST LIKELY NOT WHAT YOU WANT, AND WILL LIKELY - FAIL unless you have a email server (commonly known as an MTA) running - on your computer! - Typically you will need to specify your company or ISP's email server. - For example, if you use CableOne you will need to specify: - -s mail.cableone.net - If you have your own email server running on port 300 you would - probably use an option like this: - -s myserver.mydomain.com:300 - If you're a GMail user try: - -s smtp.gmail.com:587 -xu me\@gmail.com -xp PASSWD - --b BINDADDR[:PORT] - This option allows you to specify the local IP address (and optional - tcp port number) for sendEmail to bind to when connecting to the remote - SMTP server. This useful for people who need to send an email from a - specific network interface or source address and are running sendEmail on - a firewall or other host with several network interfaces. - --o tls= - This option allows you to specify if TLS (SSL for SMTP) should be enabled - or disabled. The default, auto, will use TLS automatically if your perl - installation has the IO::Socket::SSL and Net::SSLeay modules available, - and if the remote SMTP server supports TLS. To require TLS for message - delivery set this to yes. To disable TLS support set this to no. A debug - level of one or higher will reveal details about the status of TLS. - --o timeout=SECONDS - This option sets the timeout value in seconds used for all network reads, - writes, and a few other things. - - -EOM - last CASE; - }; - - - - - - -## OUTPUT - ($topic eq 'output') && do { - print < The help topic specified is not valid!", 1); - }; - -exit(1); -} - - - - - - - - - - - - - - - - - - - - - - -############################# -## ## -## MAIN PROGRAM ## -## ## -############################# - - -## Initialize -initialize(); - -## Process Command Line -processCommandLine(); -$conf{'alarm'} = $opt{'timeout'}; - -## Abort program after $conf{'alarm'} seconds to avoid infinite hangs -alarm($conf{'alarm'}) if ($^O !~ /win/i); ## alarm() doesn't work in win32 - - - - -################################################### -## Read $message from STDIN if -m was not used ## -################################################### - -if (!($message)) { - ## Read message body from a file specified with -o message-file= - if ($opt{'message-file'}) { - if (! -e $opt{'message-file'}) { - printmsg("ERROR => Message body file specified [$opt{'message-file'}] does not exist!", 0); - printmsg("HINT => 1) check spelling of your file; 2) fully qualify the path; 3) doubble quote it", 1); - quit("", 1); - } - if (! -r $opt{'message-file'}) { - printmsg("ERROR => Message body file specified can not be read due to restricted permissions!", 0); - printmsg("HINT => Check permissions on file specified to ensure it can be read", 1); - quit("", 1); - } - if (!open(MFILE, "< " . $opt{'message-file'})) { - printmsg("ERROR => Error opening message body file [$opt{'message-file'}]: $!", 0); - quit("", 1); - } - while () { - $message .= $_; - } - close(MFILE); - } - - ## Read message body from STDIN - else { - alarm($conf{'alarm'}) if ($^O !~ /win/i); ## alarm() doesn't work in win32 - if ($conf{'stdout'}) { - print "Reading message body from STDIN because the '-m' option was not used.\n"; - print "If you are manually typing in a message:\n"; - print " - First line must be received within $conf{'alarm'} seconds.\n" if ($^O !~ /win/i); - print " - End manual input with a CTRL-D on its own line.\n\n" if ($^O !~ /win/i); - print " - End manual input with a CTRL-Z on its own line.\n\n" if ($^O =~ /win/i); - } - while () { ## Read STDIN into $message - $message .= $_; - alarm(0) if ($^O !~ /win/i); ## Disable the alarm since at least one line was received - } - printmsg("Message input complete.", 0); - } -} - -## Replace bare LF's with CRLF's (\012 should always have \015 with it) -$message =~ s/(\015)?(\012|$)/\015\012/g; - -## Replace bare CR's with CRLF's (\015 should always have \012 with it) -$message =~ s/(\015)(\012|$)?/\015\012/g; - -## Check message for bare periods and encode them -$message =~ s/(^|$CRLF)(\.{1})($CRLF|$)/$1.$2$3/g; - -## Get the current date for the email header -my ($sec,$min,$hour,$mday,$mon,$year,$day) = gmtime(); -$year += 1900; $mon = return_month($mon); $day = return_day($day); -my $date = sprintf("%s, %s %s %d %.2d:%.2d:%.2d %s",$day, $mday, $mon, $year, $hour, $min, $sec, $conf{'timezone'}); - - - - -################################## -## Connect to the SMTP server ## -################################## -printmsg("DEBUG => Connecting to $conf{'server'}:$conf{'port'}", 1); -$SIG{'ALRM'} = sub { - printmsg("ERROR => Timeout while connecting to $conf{'server'}:$conf{'port'} There was no response after $conf{'alarm'} seconds.", 0); - printmsg("HINT => Try specifying a different mail relay with the -s option.", 1); - quit("", 1); -}; -alarm($conf{'alarm'}) if ($^O !~ /win/i); ## alarm() doesn't work in win32; -$SERVER = IO::Socket::INET->new( PeerAddr => $conf{'server'}, - PeerPort => $conf{'port'}, - LocalAddr => $conf{'bindaddr'}, - Proto => 'tcp', - Autoflush => 1, - timeout => $conf{'alarm'}, -); -alarm(0) if ($^O !~ /win/i); ## alarm() doesn't work in win32; - -## Make sure we got connected -if ( (!$SERVER) or (!$SERVER->opened()) ) { - printmsg("ERROR => Connection attempt to $conf{'server'}:$conf{'port'} failed: $@", 0); - printmsg("HINT => Try specifying a different mail relay with the -s option.", 1); - quit("", 1); -} - -## Save our IP address for later -$conf{'ip'} = $SERVER->sockhost(); -printmsg("DEBUG => My IP address is: $conf{'ip'}", 1); - - - - - - - -######################### -## Do the SMTP Dance ## -######################### - -## Read initial greeting to make sure we're talking to a live SMTP server -if (SMTPchat()) { quit($conf{'error'}, 1); } - -## We're about to use $opt{'fqdn'}, make sure it isn't empty -if (!$opt{'fqdn'}) { - ## Ok, that means we couldn't get a hostname, how about using the IP address for the HELO instead - $opt{'fqdn'} = "[" . $conf{'ip'} . "]"; -} - -## EHLO -if (SMTPchat('EHLO ' . $opt{'fqdn'})) { - printmsg($conf{'error'}, 0); - printmsg("NOTICE => EHLO command failed, attempting HELO instead"); - if (SMTPchat('HELO ' . $opt{'fqdn'})) { quit($conf{'error'}, 1); } - if ( $opt{'username'} and $opt{'password'} ) { - printmsg("WARNING => The mail server does not support SMTP authentication!", 0); - } -} -else { - - ## Determin if the server supports TLS - if ($conf{'SMTPchat_response'} =~ /STARTTLS/) { - $conf{'tls_server'} = 1; - printmsg("DEBUG => The remote SMTP server supports TLS :)", 2); - } - else { - $conf{'tls_server'} = 0; - printmsg("DEBUG => The remote SMTP server does NOT support TLS :(", 2); - } - - ## Start TLS if possible - if ($conf{'tls_server'} == 1 and $conf{'tls_client'} == 1 and $opt{'tls'} =~ /^(yes|auto)$/) { - printmsg("DEBUG => Starting TLS", 2); - if (SMTPchat('STARTTLS')) { quit($conf{'error'}, 1); } - if (! IO::Socket::SSL->start_SSL($SERVER, SSL_version => 'SSLv23:!SSLv3:!SSLv2', , SSL_verify_mode => 0)) { - quit("ERROR => TLS setup failed: " . IO::Socket::SSL::errstr(), 1); - } - printmsg("DEBUG => TLS: Using cipher: ". $SERVER->get_cipher(), 3); - printmsg("DEBUG => TLS session initialized :)", 1); - - ## Restart our SMTP session - if (SMTPchat('EHLO ' . $opt{'fqdn'})) { quit($conf{'error'}, 1); } - } - elsif ($opt{'tls'} eq 'yes' and $conf{'tls_server'} == 0) { - quit("ERROR => TLS not possible! Remote SMTP server, $conf{'server'}, does not support it.", 1); - } - - - ## Do SMTP Auth if required - if ( $opt{'username'} and $opt{'password'} ) { - if ($conf{'SMTPchat_response'} !~ /AUTH\s/) { - printmsg("NOTICE => Authentication not supported by the remote SMTP server!", 0); - } - else { - my $auth_succeeded = 0; - my $mutual_method = 0; - - # ## SASL CRAM-MD5 authentication method - # if ($conf{'SMTPchat_response'} =~ /\bCRAM-MD5\b/i) { - # printmsg("DEBUG => SMTP-AUTH: Using CRAM-MD5 authentication method", 1); - # if (SMTPchat('AUTH CRAM-MD5')) { quit($conf{'error'}, 1); } - # - # ## FIXME!! - # - # printmsg("DEBUG => User authentication was successful", 1); - # } - - ## SASL LOGIN authentication method - if ($auth_succeeded == 0 and $conf{'SMTPchat_response'} =~ /\bLOGIN\b/i) { - $mutual_method = 1; - printmsg("DEBUG => SMTP-AUTH: Using LOGIN authentication method", 1); - if (!SMTPchat('AUTH LOGIN')) { - if (!SMTPchat(base64_encode($opt{'username'}))) { - if (!SMTPchat(base64_encode($opt{'password'}))) { - $auth_succeeded = 1; - printmsg("DEBUG => User authentication was successful (Method: LOGIN)", 1); - } - } - } - if ($auth_succeeded == 0) { - printmsg("DEBUG => SMTP-AUTH: LOGIN authenticaion failed.", 1); - } - } - - ## SASL PLAIN authentication method - if ($auth_succeeded == 0 and $conf{'SMTPchat_response'} =~ /\bPLAIN\b/i) { - $mutual_method = 1; - printmsg("DEBUG => SMTP-AUTH: Using PLAIN authentication method", 1); - if (SMTPchat('AUTH PLAIN ' . base64_encode("$opt{'username'}\0$opt{'username'}\0$opt{'password'}"))) { - printmsg("DEBUG => SMTP-AUTH: PLAIN authenticaion failed.", 1); - } - else { - $auth_succeeded = 1; - printmsg("DEBUG => User authentication was successful (Method: PLAIN)", 1); - } - } - - ## If none of the authentication methods supported by sendEmail were supported by the server, let the user know - if ($mutual_method == 0) { - printmsg("WARNING => SMTP-AUTH: No mutually supported authentication methods available", 0); - } - - ## If we didn't get authenticated, log an error message and exit - if ($auth_succeeded == 0) { - quit("ERROR => ERROR => SMTP-AUTH: Authentication to $conf{'server'}:$conf{'port'} failed.", 1); - } - } - } -} - -## MAIL FROM -if (SMTPchat('MAIL FROM:<' .(returnAddressParts($from))[1]. '>')) { quit($conf{'error'}, 1); } - -## RCPT TO -my $oneRcptAccepted = 0; -foreach my $rcpt (@to, @cc, @bcc) { - my ($name, $address) = returnAddressParts($rcpt); - if (SMTPchat('RCPT TO:<' . $address . '>')) { - printmsg("WARNING => The recipient <$address> was rejected by the mail server, error follows:", 0); - $conf{'error'} =~ s/^ERROR/WARNING/o; - printmsg($conf{'error'}, 0); - } - elsif ($oneRcptAccepted == 0) { - $oneRcptAccepted = 1; - } -} -## If no recipients were accepted we need to exit with an error. -if ($oneRcptAccepted == 0) { - quit("ERROR => Exiting. No recipients were accepted for delivery by the mail server.", 1); -} - -## DATA -if (SMTPchat('DATA')) { quit($conf{'error'}, 1); } - - -############################### -## Build and send the body ## -############################### -printmsg("INFO => Sending message body",1); - -## If the message-format is raw just send the message as-is. -if ($opt{'message-format'} =~ /^raw$/i) { - print $SERVER $message; -} - -## If the message-format isn't raw, then build and send the message, -else { - - ## Message-ID: - if ($opt{'message-header'} !~ /^Message-ID:/iom) { - $header .= 'Message-ID: <' . $conf{'Message-ID'} . '@' . $conf{'hostname'} . '>' . $CRLF; - } - - ## From: "Name" (the pointless test below is just to keep scoping correct) - if ($from and $opt{'message-header'} !~ /^From:/iom) { - my ($name, $address) = returnAddressParts($from); - $header .= 'From: "' . $name . '" <' . $address . '>' . $CRLF; - } - - ## Reply-To: - if ($opt{'reply-to'} and $opt{'message-header'} !~ /^Reply-To:/iom) { - my ($name, $address) = returnAddressParts($opt{'reply-to'}); - $header .= 'Reply-To: "' . $name . '" <' . $address . '>' . $CRLF; - } - - ## To: "Name" - if ($opt{'message-header'} =~ /^To:/iom) { - ## The user put the To: header in via -o message-header - dont do anything - } - elsif (scalar(@to) > 0) { - $header .= "To:"; - for (my $a = 0; $a < scalar(@to); $a++) { - my $msg = ""; - - my ($name, $address) = returnAddressParts($to[$a]); - $msg = " \"$name\" <$address>"; - - ## If we're not on the last address add a comma to the end of the line. - if (($a + 1) != scalar(@to)) { - $msg .= ","; - } - - $header .= $msg . $CRLF; - } - } - ## We always want a To: line so if the only recipients were bcc'd they don't see who it was sent to - else { - $header .= "To: \"Undisclosed Recipients\" <>$CRLF"; - } - - if (scalar(@cc) > 0 and $opt{'message-header'} !~ /^Cc:/iom) { - $header .= "Cc:"; - for (my $a = 0; $a < scalar(@cc); $a++) { - my $msg = ""; - - my ($name, $address) = returnAddressParts($cc[$a]); - $msg = " \"$name\" <$address>"; - - ## If we're not on the last address add a comma to the end of the line. - if (($a + 1) != scalar(@cc)) { - $msg .= ","; - } - - $header .= $msg . $CRLF; - } - } - - if ($opt{'message-header'} !~ /^Subject:/iom) { - $header .= 'Subject: ' . $subject . $CRLF; ## Subject - } - if ($opt{'message-header'} !~ /^Date:/iom) { - $header .= 'Date: ' . $date . $CRLF; ## Date - } - if ($opt{'message-header'} !~ /^X-Mailer:/iom) { - $header .= 'X-Mailer: sendEmail-'.$conf{'version'}.$CRLF; ## X-Mailer - } - ## I wonder if I should put this in by default? - # if ($opt{'message-header'} !~ /^X-Originating-IP:/iom) { - # $header .= 'X-Originating-IP: ['.$conf{'ip'}.']'.$CRLF; ## X-Originating-IP - # } - - ## Encode all messages with MIME. - if ($opt{'message-header'} !~ /^MIME-Version:/iom) { - $header .= "MIME-Version: 1.0$CRLF"; - } - if ($opt{'message-header'} !~ /^Content-Type:/iom) { - my $content_type = 'multipart/mixed'; - if (scalar(@attachments) == 0) { $content_type = 'multipart/related'; } - $header .= "Content-Type: $content_type; boundary=\"$conf{'delimiter'}\"$CRLF"; - } - - ## Send additional message header line(s) if specified - if ($opt{'message-header'}) { - $header .= $opt{'message-header'}; - } - - ## Send the message header to the server - print $SERVER $header . $CRLF; - - ## Start sending the message body to the server - print $SERVER "This is a multi-part message in MIME format. To properly display this message you need a MIME-Version 1.0 compliant Email program.$CRLF"; - print $SERVER "$CRLF"; - - - ## Send message body - print $SERVER "--$conf{'delimiter'}$CRLF"; - ## Send a message content-type header: - ## If the message contains HTML... - if ($opt{'message-content-type'} eq 'html' or ($opt{'message-content-type'} eq 'auto' and $message =~ /^\s*( 0) { - ## Disable the alarm so people on modems can send big attachments - alarm(0) if ($^O !~ /win/i); ## alarm() doesn't work in win32 - - ## Send the attachments - foreach my $filename (@attachments) { - ## This is check 2, we already checked this above, but just in case... - if ( ! -f $filename ) { - printmsg("ERROR => The file [$filename] doesn't exist! Email will be sent, but without that attachment.", 0); - } - elsif ( ! -r $filename ) { - printmsg("ERROR => Couldn't open the file [$filename] for reading: $! Email will be sent, but without that attachment.", 0); - } - else { - printmsg("DEBUG => Sending the attachment [$filename]", 1); - send_attachment($filename); - } - } - } - - - ## End the mime encoded message - print $SERVER "$CRLF--$conf{'delimiter'}--$CRLF"; -} - - -## Tell the server we are done sending the email -print $SERVER "$CRLF.$CRLF"; -if (SMTPchat()) { quit($conf{'error'}, 1); } - - - -#################### -# We are done!!! # -#################### - -## Disconnect from the server (don't SMTPchat(), it breaks when using TLS) -print $SERVER "QUIT$CRLF"; -close $SERVER; - - - - - - -####################################### -## Generate exit message/log entry ## -####################################### - -if ($conf{'debug'} or $conf{'logging'}) { - printmsg("Generating a detailed exit message", 3); - - ## Put the message together - my $output = "Email was sent successfully! From: <" . (returnAddressParts($from))[1] . "> "; - - if (scalar(@to) > 0) { - $output .= "To: "; - for ($a = 0; $a < scalar(@to); $a++) { - $output .= "<" . (returnAddressParts($to[$a]))[1] . "> "; - } - } - if (scalar(@cc) > 0) { - $output .= "Cc: "; - for ($a = 0; $a < scalar(@cc); $a++) { - $output .= "<" . (returnAddressParts($cc[$a]))[1] . "> "; - } - } - if (scalar(@bcc) > 0) { - $output .= "Bcc: "; - for ($a = 0; $a < scalar(@bcc); $a++) { - $output .= "<" . (returnAddressParts($bcc[$a]))[1] . "> "; - } - } - $output .= "Subject: [$subject] " if ($subject); - if (scalar(@attachments_names) > 0) { - $output .= "Attachment(s): "; - foreach(@attachments_names) { - $output .= "[$_] "; - } - } - $output .= "Server: [$conf{'server'}:$conf{'port'}]"; - - -###################### -# Exit the program # -###################### - - ## Print / Log the detailed message - quit($output, 0); -} -else { - ## Or the standard message - quit("Email was sent successfully!", 0); -} - diff --git a/dev/lib.old/session.pm b/dev/lib.old/session.pm deleted file mode 100644 index cdf775f..0000000 --- a/dev/lib.old/session.pm +++ /dev/null @@ -1,89 +0,0 @@ -package session; - -use strict; -use File::Basename; -use Digest::SHA qw(sha256_hex); -use lib ('.'); -use lib ('.\\lib'); -use dksdb; -use Data::Dumper; - -sub new { - my $class = shift; - my $self = bless {}, $class; - $self->{db} = dksdb->new(); - return $self; -} - -sub checklogin(){ - my $self = shift; - my $login = shift; - my $password = shift; - my $pwd = sha256_hex($password); - my $sid = ""; - - my $sql = "select id from users where username='".$login."' and userpassword='".$pwd."' and activationkey is null and userblocked is null;"; - my $res = $self->{db}->dbquerysorted($sql); - if(keys(%{$res}) == 1){ - $sid = $self->createsessionid(); - $self->{db}->dbexec("UPDATE users set sessionid='".$sid."',remoteaddr='".$ENV{REMOTE_ADDR}."',lastlogin=CURRENT_TIMESTAMP,useragent='".$ENV{HTTP_USER_AGENT}."' WHERE username='".$login."' and userpassword='".$pwd."'"); - } - return $sid; -} - -sub getsession($){ - my $self = shift; - my $sid = shift; - my $sql = "SELECT us.id,us.usersurname,us.userprename,us.username,us.usergroup, - string_agg(CASE WHEN srv.expiration >= CURRENT_DATE or (prc.minmonth=0 and srv.expiration is null) then srv.service else null end,',') as service_valid, - string_agg(CASE WHEN srv.expiration < CURRENT_DATE then srv.service else null end,',') as service_expired - from users us - left join userservices srv on (srv.id_user=us.id) - left join appprices prc on (prc.id_app=srv.id_app) - where us.sessionid='".$sid."' and us.userblocked is null and us.activationkey is null and useragent='".$ENV{'HTTP_USER_AGENT'}."' and remoteaddr='".$ENV{'REMOTE_ADDR'}."' - group by us.id;"; - my $res= $self->{db}->dbquerysorted($sql); - my $ret = undef; - if (keys(%{$res}) > 0){ - return $res->{0}; - } - return $ret; -} - -sub createsessionid(){ - my $self = shift; - my $randnum =""; - my $i =0; - while ($i<50) - { - my $r1 = int(rand(3)) + 1 ; - if ($r1 == 1){ $randnum .= chr(int(rand(9) +48));$i++;} - if ($r1 == 2){ $randnum .= chr(int(rand(24) +65));$i++;} - if ($r1 == 3){ $randnum .= chr(int(rand(24) +97));$i++;} - } - return $randnum; -} - -sub deletesession(){ - my $self = shift; - my $sid = shift; - if ($sid ne ''){ - $self->{db}->dbexec("UPDATE users set sessionid=null where sessionid='".$sid."'"); - } -} - -sub resetpasswd(){ - -} - -sub registeruser(){ - -} - -sub randompasword(){ - my $self = shift; - my @alphanumeric = ('a'..'z', 'A'..'Z', 0..9); - my $randpassword = join '', map $alphanumeric[rand @alphanumeric], 0..12; - return $randpassword; -} -1; \ No newline at end of file diff --git a/dev/lib.old/test.jpeg b/dev/lib.old/test.jpeg deleted file mode 100644 index 5694063a87457fcaf822f4a5ac0aac4990cdee96..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 75102 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

aFuI-R|cjlO=Id!ZchF3jMuBuov!y5M>BN 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 YZ7d+h!t{}Ap6-m>eaPk^?*6O)+1E1`djJ3c diff --git a/dev/lib.old/test.png b/dev/lib.old/test.png deleted file mode 100644 index 19ef9d1c851f311ca9e6b4f9f4bb972f4c4884c1..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4092 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 -$sf->saveb64toFile('test','pathtofile','data:image/png;base64,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'); \ No newline at end of file diff --git a/dev/lib.old/upload.cgi b/dev/lib.old/upload.cgi deleted file mode 100644 index a21c5c8..0000000 --- a/dev/lib.old/upload.cgi +++ /dev/null @@ -1,266 +0,0 @@ -#!/usr/bin/perl -use strict; - -use CGI; -use lib ('./lib'); -use config; -use template; -use elements; -use check; -use save; -use Cwd; -use forms; -use data; -my $cwd = getcwd; -my $cgi = CGI->new(); -my $url = $cgi->url(-path_info=>1); -my $baseurl = substr($url,0,rindex($url,'/')); -my %param = (); -$param{'lang'} = 'fr'; -$param{'action'} = ''; -$param{'id'} = 0; -$param{'type'} = ''; -$param{'rub'} = 0; -$param{'cat'} = 0; -my @tp = $cgi->param; - - -foreach my $p (@tp){ - $param{$p} = $cgi->param($p); -} - -$param{'user'}{'sid'} = $cgi->cookie('myoffersid'); - -if (!defined $param{'user'}{'sid'}){$param{'user'}{'sid'} = "0";} - -my $config = config->new(); - - my %result = $config->checksession($param{'user'}{'sid'}); - $param{'user'} = {%result}; - -print $cgi->header(); -my $html = ""; -$html .= $cgi->start_html(-title => ":: MyOFFER :: Luxembourg", - -author=>'info@creatx.com', - -meta=>{'keywords'=>'', - 'copyright'=>'(c) copyright 2008 by creatX idea factory', - 'cache-control' => 'no-cache', - 'pragma' => 'no-cache', - 'robots' => 'noindex,nofollow', - 'description' => 'myoffer.lu'}, - -style=>[ - {'src'=>'css/myoffer.css'}, - {'src'=>'css/editor.css'}, - ], - -script=>[ - {-language=>'JavaScript',-src=>'js/myoffer.js'}, - {-language=>'JavaScript',-src=>'editor/fckeditor.js'} - ] - ); -my $f = forms->new(); -$html .= '

'; -if (!exists($param{'newuser'})){$param{'newuser'} = 0;} -if (!exists($param{'nextstep'})){$param{'nextstep'} = "";} -if ($param{'page'} eq 'frmspecialfile'){ - $html .= $f->get_frmspecialfile($param{'id'},$param{'lang'},$param{'retlink'},$param{'newuser'},$param{'nextstep'}); -}elsif ($param{'page'} eq 'frmpictures'){ - - $html .= $f->get_frmpictures($param{'id'},$param{'lang'},$param{'retlink'},$param{'newuser'},$param{'nextstep'}); -} -elsif ($param{'page'} eq 'frmaddlogo'){ - $html .= $f->get_frmlogo($param{'id'},$param{'lang'},$param{'retlink'}); -} -elsif ($param{'action'} eq 'save'){ - my %ret = (); - #foreach my $k (keys %param){ - # $html .= 'Key: '.$k.' value: '.$param{$k}.'
'; - #} - if ($param{'action2'} eq 'addfile'){ - #$html.= "add file"; - my $data = data->new(); - my %fdata = $data->getspecialfilebyid($param{'id'}); - #my $num = scalar(keys(%fdata))+1; - opendir(SPF,$cwd.'/files/'.$param{'rub'}.'/'.$param{'cat'}); - my $num = 0; - while (my $l = readdir(SPF)){ - if (index($l,$param{'id'}.'_') != -1){ - my ($y) = $l =~ m/\d+\_(\d+)\..+$/; - if ($y > $num){$num = $y;} - $num++; - } - } - close(SPF); - my $fh = $cgi->upload('txtspecialfile'); - my $suffix = substr($param{'txtspecialfile'},rindex($param{'txtspecialfile'},'.')+1); - if (! -d $cwd.'/files/'.$param{'rub'}){ - mkdir($cwd.'/files/'.$param{'rub'},0774) || die $!; - } - if (! -d $cwd.'/files/'.$param{'rub'}.'/'.$param{'cat'}){ - mkdir($cwd.'/files/'.$param{'rub'}.'/'.$param{'cat'},0774) || die $!; - } - open(SPF,'>'.$cwd.'/files/'.$param{'rub'}.'/'.$param{'cat'}.'/'.$param{'id'}.'_'.$num.'.'.$suffix) || die $!; - while (my $l = <$fh>) { - print SPF $l; - } - close(SPF); - - $data->setofferspecialfile($param{'id'}.'_'.$num.'.'.$suffix,$param{'id'},$param{'txtcomment'},$suffix); - $html.= $f->get_frmspecialfile($param{'id'},$param{'lang'},$param{'retlink'},$param{'newuser'},$param{'nextstep'}); - #$ret{'title'} = "%%LANG:headdatasaved%%"; - #$ret{'msg'} = "%%LANG:msgdatasaved%%"; - } - elsif ($param{'action2'} eq 'deletefile'){ - #$html.= "delete file"; - my $data = data->new(); - my %data = $data->getspecialfilebyfid($param{'fileid'}); - if ($data{0}{'file'} != ''){ - unlink($cwd.'/files/'.$param{'rub'}.'/'.$param{'cat'}.'/'.$data{0}{'file'}) || die $!; - } - $data->deleteofferspecialfile($data{0}{'fileid'}); - $html.= $f->get_frmspecialfile($param{'id'},$param{'lang'},$param{'retlink'},$param{'newuser'},$param{'nextstep'}); - } - elsif ($param{'type'} eq 'addlogo'){ - my $type = $cgi->uploadInfo($param{'txtnewlogo'})->{'Content-Type'}; - if (index($type,'image' == -1)){ - $html.= $f->get_frmlogo($param{'id'},$param{'lang'},$param{'retlink'}); - - } - else { - my $fh = $cgi->upload('txtnewlogo'); - my $fname = $cwd.'/pic/logo/'.$param{'id'}.'.png'; - open(SPF,'>'.$fname) || die $!; - while (my $l = <$fh>) { - print SPF $l; - } - close(SPF); - my $tdim = `identify -verbose $fname | grep Geometry`; - my ($pwidth,$pheight) = $tdim =~ m/\s+Geometry:\s(\d+)x(\d+)\+\d+\+\d+$/; - my $width=0; - my $height =0; - my $density = 72; - if ($pwidth>200){ - $width = 200; - } elsif ($height > 75) { - $height = 75; - } - if ($height == 0){ - $height = ($pheight/$pwidth) * $width; - #calc height - } - elsif ($width == 0) { - #calc width - $width = ($pwidth/$pheight) * $height; - } - if ($height > 75){ - $height = 75; - $width = ($width/$height) * $height; - } - my $status = system("mogrify -resize ".$width."x".$height." -density ".$density." ".$fname); - $html.= $f->get_frmlogo($param{'id'},$param{'lang'},$param{'retlink'}); - } - } - elsif ($param{'type'} eq 'addpicture'){ - my $type = $cgi->uploadInfo($param{'txtnewpic'})->{'Content-Type'}; - if (index($type,'image' == -1)){ - $html.= $f->get_frmpictures($param{'id'},$param{'lang'}); - # $ret{'title'} = "%%LANG:headerrormsg%%"; - # $ret{'msg'} = "%%LANG:msgonlypictures%%"; - } - else { - my $fh = $cgi->upload('txtnewpic'); - my $suffix = substr($param{'txtnewpic'},rindex($param{'txtnewpic'},'.')+1); - if (! -d $cwd.'/pic/'.$param{'rub'}){ - mkdir($cwd.'/pic/'.$param{'rub'},0774) || die $!; - } - if (! -d $cwd.'/pic/'.$param{'rub'}.'/'.$param{'cat'}){ - mkdir($cwd.'/pic/'.$param{'rub'}.'/'.$param{'cat'},0774) || die $!; - } - opendir(PICS,$cwd.'/pic/'.$param{'rub'}.'/'.$param{'cat'}); - my $x = 0; - while (my $l = readdir(PICS)){ - if (index($l,$param{'id'}.'_') != -1){ - my ($y) = $l =~ m/\d+\_(\d+)\..+$/; - if ($y > $x){$x = $y;} - $x++; - } - } - close(PICS); - open(SPF,'>'.$cwd.'/pic/'.$param{'rub'}.'/'.$param{'cat'}.'/'.$param{'id'}.'_'.$x.'.'.$suffix) || die $!; - while (my $l = <$fh>) { - print SPF $l; - } - close(SPF); - my $fname = $cwd.'/pic/'.$param{'rub'}.'/'.$param{'cat'}.'/'.$param{'id'}.'_'.$x.'.'.$suffix; - my $tdim = `identify -verbose $fname | grep Geometry`; - my ($pwidth,$pheight) = $tdim =~ m/\s+Geometry:\s(\d+)x(\d+)\+\d+\+\d+$/; - my $width=0; - my $height =0; - my $density = 72; - if ($pwidth>=$pheight){ - $width = 180; - } else { - $height = 135; - } - if ($height == 0){ - $height = ($pheight/$pwidth) * $width; - #calc height - } - elsif ($width == 0) { - #calc width - $width = ($pwidth/$pheight) * $height; - } - my $status = system("mogrify -resize ".$width."x".$height." -density ".$density." ".$fname); - if ($status == 0) { - my $data = data->new(); - $data->setofferpicture($param{'id'}.'_'.$x.'.'.$suffix,$param{'id'}); - } else { - die $!; - } - $html.= $f->get_frmpictures($param{'id'},$param{'lang'},$param{'retlink'},$param{'newuser'},$param{'nextstep'}); - } - } - elsif ($param{'type'} eq 'deletepictures'){ - my %delpic = (); - foreach my $k (keys %param){ - if (index($k,'chkdelpic') != -1) { - $delpic{$param{$k}} = ''; - } - } - my $data = data->new(); - my %pic = $data->getofferpictures($param{'id'}); - foreach my $p (keys %pic){ - #$html .= 'Picture '.$pic{$p}{'picture'}.' - '.$pic{$p}{'id'}."
"; - if (exists($delpic{$pic{$p}{'id'}})){ - #$html .= 'deleting '.$cwd.'/pic/'.$param{'rub'}.'/'.$param{'cat'}.'/'.$pic{$p}{'picture'}.'
'; - unlink($cwd.'/pic/'.$param{'rub'}.'/'.$param{'cat'}.'/'.$pic{$p}{'picture'}) || die $!; - $data->deleteofferpicture($pic{$p}{'id'}); - } - } - $html.= $f->get_frmpictures($param{'id'},$param{'lang'},$param{'retlink'},$param{'newuser'},$param{'nextstep'}); - } - - else { - $html .= '
No data found!
'; - - } - #if (keys(%ret) > 0){ - # $html .= '
'.$ret{'title'}.'
'."\n"; - # $html .= '
 
'."\n"; - # $html .= '
'.$ret{'msg'}.'
'."\n"; - #} -} -else { - $html .= '
No data found!
'; -} -$html .= '
'; -my $tmpl = template->new(); -$html = $tmpl->replace_simpleplaceholder($html,'%%BASEURL%%',$baseurl); -$html = $tmpl->replace_simpleplaceholder($html,'%%URL%%',$url); -$html = $tmpl->replace_langplaceholders($html,$param{'lang'}); - - -#close(DBG); -print $html; - -print $cgi->end_html(); - diff --git a/index.cgi b/index.cgi index 0924d67..6ae5772 100755 --- a/index.cgi +++ b/index.cgi @@ -1,4 +1,4 @@ -#!/Users/kilian/perl5/perlbrew/perls/perl-5.24.1/bin/perl +#!/usr/local/bin/perl use strict; use lib ('backoffice/api/lib/perl5'); -- 2.39.5