I’ve been running a number of experiments recently that require a lot of computing time. “A lot” in this case being on the order of days. It would therefore be nice to have a script that would automatically E-mail me when my experiments finish, so I know to check the results. I fully expected there to be some magic shell script out there somewhere dedicated to this very purpose: sending out an E-mail when a specified process dies. Something like this:
$ ./run_experiments& [1] 1337 $ emailwhendone 1337 Awaiting process 1337's death...
As far as I can tell, however, there is no such script/program. So, as usual, I took it upon myself to write my own. The E-mailing part turned out to be a bit trickier than I had expected.
I didn’t want my script to be dependent on the existence of a local
mail server; therefore, I first tried using sSMTP. It turns out that sSMTP requires one to
hard-code the remote SMTP server address in a .conf
file,
so that approach was out.
Next I tried Mail::Sendmail
, however, that
module’s support for authentication is poor at best. That module also
doesn’t support SSL, so emailing through servers like Google Mail is
out.
Therefore, I finally settled on using Net::SMTP::SSL
,
which unfortunately has four dependencies. Luckily for me, those
dependencies are all easily installable on Gentoo:
- dev-perl/Authen-SASL
- dev-perl/IO-Socket-SSL
- dev-perl/Net-SSLeay
- dev-perl/Net-SMTP-SSL
I call my script emailwhendone
because, well, that’s exactly what it does. The code follows at the end of this post.
Disclaimer: I blatantly cribbed some of my code from Robert Maldon (for the MTA stuff) and Bill Luebkert (for the password input).
The script can be given one of two parameters: either the PID of the process for which to wait or the unique name of the process (if there are multiple processes with the same name you will need to use the PID). Right now I have the recipient E-mail address hard-coded; it should be fairly self evident from the code how to customize this. Here’s an example:
$ ./run_experiments& [1] 1337 $ emailwhendone 1337 Password for youremail@domain.com: ******************* Waiting for process 1337 (run_experiments) to finish... The process finished! Sending an email to youremail@domain.com... $
Here’s the code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 | #!/usr/bin/perl -w use Net::SMTP::SSL; use Term::ReadKey; END { ReadMode ( 'restore' ); } # just in case my $destination = 'youremail@domain.com' ; my $server = 'smtp.domain.com' ; my $port = 465; ##################################### sub usage { print " Usage: emailwhendone [PID|PROCESS_NAME]\n" ; } my $pid = $ARGV [0] or die &usage(); my $hostname = `hostname`; my $pidmatch = -1; my $processmatch = "" ; my @pidmatches ; open PRO, "/bin/ps axo pid,comm |" or die 'Failed to open pipe to `ps`' ; while (<pro>) { if ( $_ =~ m/^\s*(\d+)\s+(.+)$/) { my $matchpid = $1 ; my $matchprocess = $2 ; if ( $matchpid eq $pid ) { $pidmatch = $matchpid ; $processmatch = $matchprocess ; @pidmatches = [ $matchpid ]; last ; } elsif ( $pid =~ m/^\s* $matchprocess \s*$/) { $pidmatch = $matchpid ; push ( @pidmatches , $matchpid ); $processmatch = $matchprocess ; } } } close PRO; if ( scalar ( @pidmatches ) <= 0) { if ( $pid =~ m/^\s*\d+\s*$/) { print "Error: no process with ID " . $pid . "!\n" ; } else { print "Error: no process named \"" . $pid . "\"!\n" ; } exit (1); } elsif ( scalar ( @pidmatches ) > 1) { print "There are multiple PIDs that match this process name!\n" ; for my $match ( @pidmatches ) { print $match . "\t" . $pid . "\n" ; } exit (2); } sub get_passwd { # legal clear passwd chrs (26+26+10+24=86): "a-zA-Z0-9!#$%&()*+,-./:;<=> ?@[\]^"; my @legal_clear = ( 'a' .. 'z' , 'A' .. 'Z' , '0' .. '9' , split //, '!#$%&()*+,-./:;<=> ?@[\]^' ); my %legal_clear ; foreach ( @legal_clear ) { $legal_clear { $_ } = 1; } $| = 1; # unbuffer stdout to force unterminated line out ReadMode ( 'cbreak' ); my $ch = '' ; while ( defined ( $ch = ReadKey ())) { last if $ch eq "\x0D" or $ch eq "\x0A" ; if ( $ch eq "\x08" ) { # backspace print "\b \b" if $passwd ; # back up 1 chop $passwd ; next ; } if ( $ch eq "\x15" ) { # ^U print "\b \b" x length $passwd ; # back 1 for each char $passwd = '' ; next ; } if (not exists $legal_clear { $ch }) { print "\n'$ch' not a legal password character\n" ; print 'Password: ' , "*" x length $passwd ; # retype *'s next ; } $passwd .= $ch ; print '*' ; } print "\n" ; ReadMode ( 'restore' ); return $passwd ; } print "Password for " . $destination . ": " ; my $password = get_passwd(); sub send_mail { my $subject = $_ [0]; my $body = $_ [1]; my $smtp ; if (not $smtp = Net::SMTP::SSL->new( $server , Port => $port , Debug => 0)) { die "Could not connect to server.\n" ; } $smtp ->auth( $destination , $password ) || die "Authentication failed!\n" ; $smtp ->mail( $destination . "\n" ); $smtp ->to( $destination . "\n" ); $smtp ->data(); $smtp ->datasend( "From: " . $destination . "\n" ); $smtp ->datasend( "To: " . $destination . "\n" ); $smtp ->datasend( "Subject: " . $subject . "\n" ); $smtp ->datasend( "\n" ); $smtp ->datasend( $body . "\n" ); $smtp ->dataend(); $smtp ->quit; } print "Waiting for process " . $pidmatch . " (" . $processmatch . ") to finish..." ; my $done = 0; do { $done = 1; open PRO, "/bin/ps axo pid |" or die 'Failed to open pipe to `ps`' ; while (<pro>) { if ( $_ =~ m/^\s* $pidmatch \s*$/) { $done = 0; last ; } } close PRO; sleep (1); } while (! $done ); print "The process finished!\nSending an email to " . $destination . "..." ; &send_mail( 'Process ' . $pidmatch . ' (' . $processmatch . ') on ' . $hostname . ' finished!' , 'It\'s done!' ); print "\n" ; </pro></pro> |