| 
									
										
										
										
											2013-11-30 13:37:20 -05:00
										 |  |  | #!/usr/bin/perl -w | 
					
						
							|  |  |  | # calendar-import.plx                                         -*- Perl -*- | 
					
						
							|  |  |  | # ==================================================================== | 
					
						
							| 
									
										
										
										
											2013-11-30 14:07:55 -05:00
										 |  |  | # NOTE: Overall license of this file is GPLv3-only, due (in part) to Software | 
					
						
							|  |  |  | # Freedom Law Center copyrights (see below).  Kuhn's personal copyrights are | 
					
						
							|  |  |  | # licensed GPLv3-or-later. | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:20 -05:00
										 |  |  | # | 
					
						
							|  |  |  | # ==================================================================== | 
					
						
							| 
									
										
										
										
											2013-11-30 14:07:55 -05:00
										 |  |  | # The sub's "safe_read_from_pipe" and read_from_process are: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:20 -05:00
										 |  |  | # Copyright (c) 2000-2004 CollabNet.  All rights reserved. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # This software is licensed as described in the file COPYING, which | 
					
						
							|  |  |  | # you should have received as part of this distribution.  The terms | 
					
						
							|  |  |  | # are also available at http://subversion.tigris.org/license-1.html. | 
					
						
							|  |  |  | # If newer versions of this license are posted there, you may use a | 
					
						
							|  |  |  | # newer version instead, at your option. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # This software consists of voluntary contributions made by many | 
					
						
							|  |  |  | # individuals.  For exact contribution history, see the revision | 
					
						
							|  |  |  | # history and logs, available at http://subversion.tigris.org/. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #  Note: bkuhn downloaded the license from | 
					
						
							|  |  |  | #  http://subversion.tigris.org/license-1.html on 2013-12-29 which said: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # The license of Subversion 1.7 and later is at | 
					
						
							|  |  |  | # http://svn.apache.org/repos/asf/subversion/trunk/LICENSE. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # The license of Subversion 1.6 and earlier can be found at | 
					
						
							|  |  |  | # http://svn.apache.org/repos/asf/subversion/tags/1.6.0/www/license-1.html. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Both license texts are now included, in APACHE-LICENSE and OLD-SVN-LICENSE, | 
					
						
							|  |  |  | # respectively. | 
					
						
							|  |  |  | # ==================================================================== | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Copyright © 2013 Bradley M. Kuhn <bkuhn@ebb.org> | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # This software's license gives you freedom; you can copy, convey, | 
					
						
							|  |  |  | # propogate, redistribute and/or modify this program under the terms of | 
					
						
							|  |  |  | # the GNU  General Public License (GPL) as published by the Free | 
					
						
							|  |  |  | # Software Foundation (FSF), either version 3 of the License, or (at your | 
					
						
							|  |  |  | # option) any later version of the GPL published by the FSF. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # This program is distributed in the hope that it will be useful, but | 
					
						
							|  |  |  | # WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
					
						
							|  |  |  | # General Public License for more details. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  | # along with this program in a file in the toplevel directory called | 
					
						
							|  |  |  | # "GPLv3".  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 14:07:55 -05:00
										 |  |  | # ==================================================================== | 
					
						
							|  |  |  | # The functions DoLog, BinarySearchForTZEntry, PrivatizeMergeAndTZIcalFile, | 
					
						
							|  |  |  | # BuildTZList, MergeLists, PrivacyFilterICalFiles, and FilterEmacsToICal material | 
					
						
							|  |  |  | # copyrighted and licensed as below: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Copyright © 2006 Software Freedom Law Center, Inc. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # This software gives you freedom; it is licensed to you under version 3 | 
					
						
							|  |  |  | # of the GNU General Public License. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # This software is distributed WITHOUT ANY WARRANTY, without even the | 
					
						
							|  |  |  | # implied warranties of MERCHANTABILITY and FITNESS FOR A PARTICULAR | 
					
						
							|  |  |  | # PURPOSE.  See the GNU General Public License for further details. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # You should have received a copy of the GNU General Public License, | 
					
						
							|  |  |  | # version 3.  If not, see <http://www.gnu.org/licenses/> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:20 -05:00
										 |  |  | use strict; | 
					
						
							|  |  |  | use warnings; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 14:14:46 -05:00
										 |  |  | use Carp; | 
					
						
							|  |  |  | use Data::ICal; | 
					
						
							|  |  |  | use File::Temp qw/:POSIX tempfile/; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:20 -05:00
										 |  |  | my($CONFIG_FILE) = (@ARGV); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if (@ARGV != 1) { | 
					
						
							|  |  |  |   print STDERR "usage: $0 <CONFIG_FILE>\n"; | 
					
						
							|  |  |  |   exit 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							| 
									
										
										
										
											2013-11-30 14:07:55 -05:00
										 |  |  | my $CALENDAR_LOCK_FILE = "$ENV{HOME}/.emacs-calendar-to-ics-lock"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | my $LOCK_CLEANUP_CODE = sub { | 
					
						
							|  |  |  |   return (unlink($CALENDAR_LOCK_FILE) != 1) ? | 
					
						
							|  |  |  |     "Failed unlink of $CALENDAR_LOCK_FILE.  Could cause trouble." : | 
					
						
							|  |  |  |     ""; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   my %messageHistory; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sub DoLog ($$$;$) { | 
					
						
							|  |  |  |     my($type, $user, $message, $cleanupCode) = @_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     use Date::Manip; | 
					
						
							|  |  |  |     my $NOW = ParseDate("now"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     my $lastTime = $messageHistory{$message}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     my $sendIt = 0; | 
					
						
							|  |  |  |     if (not defined $lastTime) { | 
					
						
							|  |  |  |       $sendIt = 1; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       my $err; | 
					
						
							|  |  |  |       my $sinceLast = DateCalc($lastTime,"+ 10 minutes",\$err); | 
					
						
							|  |  |  |       $sendIt = 1 if ($NOW gt $sinceLast); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if ($sendIt) { | 
					
						
							|  |  |  |       my  $fh = File::Temp->new(); | 
					
						
							|  |  |  |       $fh->unlink_on_destroy( 1 ); | 
					
						
							|  |  |  |       my  $fname = $fh->filename; | 
					
						
							|  |  |  |       print $fh "Calendar Export Failure: $message\n"; | 
					
						
							|  |  |  |       $fh->close(); | 
					
						
							|  |  |  |       system('/home/bkuhn/bin/myosd', $fname); | 
					
						
							|  |  |  |       unless (-f "$ENV{HOME}/.silent-running") { | 
					
						
							|  |  |  |         open(ESPEAK, "-|", "/usr/bin/espeak",  '-p', '45', '-s', '130', '-f', $fname, "--stdout"); | 
					
						
							|  |  |  |         open(PAPLAY, "|-", "/usr/bin/paplay"); | 
					
						
							|  |  |  |         my $data; | 
					
						
							|  |  |  |         while (read(ESPEAK, $data, 8) == 8) { | 
					
						
							|  |  |  |           print PAPLAY  $data; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         close PAPLAY; close ESPEAK; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       system('/usr/bin/notify-send', '-u', 'critical', '-t', '300000', | 
					
						
							|  |  |  |              'Failure', "Calendar export failure: $message"); | 
					
						
							|  |  |  |       $messageHistory{$message} = $NOW; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     my $more; | 
					
						
							|  |  |  |     $more = &$cleanupCode if defined $cleanupCode and ref $cleanupCode; | 
					
						
							|  |  |  |     $message .= "  $more" if (defined $more and $more !~ /^\s*$/); | 
					
						
							|  |  |  |     croak $message if $type eq "die"; | 
					
						
							|  |  |  |     warn $message; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   sub DieLog ($;$) { | 
					
						
							|  |  |  |     DoLog("die", undef, $_[0], $_[1]); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-11-30 14:08:56 -05:00
										 |  |  |   sub WarnLog ($;$) { | 
					
						
							| 
									
										
										
										
											2013-11-30 14:07:55 -05:00
										 |  |  |     DoLog("warn", $_[0], $_[1]); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:20 -05:00
										 |  |  | # Start a child process safely without using /bin/sh. | 
					
						
							|  |  |  | sub safe_read_from_pipe | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   unless (@_) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       DieLog("$0: safe_read_from_pipe passed no arguments."); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   my $pid = open(SAFE_READ, '-|'); | 
					
						
							|  |  |  |   unless (defined $pid) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       DieLog("$0: cannot fork: $!"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   unless ($pid) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       open(STDERR, ">&STDOUT") | 
					
						
							|  |  |  |         or DieLog("$0: cannot dup STDOUT: $!"); | 
					
						
							|  |  |  |       exec(@_) | 
					
						
							|  |  |  |         or DieLog("$0: cannot exec `@_': $!\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   my @output; | 
					
						
							|  |  |  |   while (<SAFE_READ>) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       s/[\r\n]+$//; | 
					
						
							|  |  |  |       push(@output, $_); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   close(SAFE_READ); | 
					
						
							|  |  |  |   my $result = $?; | 
					
						
							|  |  |  |   my $exit   = $result >> 8; | 
					
						
							|  |  |  |   my $signal = $result & 127; | 
					
						
							|  |  |  |   my $cd     = $result & 128 ? "with core dump" : ""; | 
					
						
							|  |  |  |   if ($signal or $cd) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       DieLog("$0: pipe from `@_' failed $cd: exit=$exit signal=$signal\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (wantarray) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       return ($result, @output); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       return $result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							|  |  |  | # Use safe_read_from_pipe to start a child process safely and return | 
					
						
							|  |  |  | # the output if it succeeded or an error message followed by the output | 
					
						
							|  |  |  | # if it failed. | 
					
						
							|  |  |  | sub read_from_process | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   unless (@_) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       DieLog("$0: read_from_process passed no arguments."); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   my ($status, @output) = &safe_read_from_pipe(@_); | 
					
						
							|  |  |  |   if ($status) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       return ("$0: `@_' failed with this output:", @output); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       return @output; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-12-29 16:22:45 -05:00
										 |  |  | my $PRINTED_CONKY_HEADER; | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:20 -05:00
										 |  |  | ############################################################################### | 
					
						
							| 
									
										
										
										
											2013-11-30 14:05:00 -05:00
										 |  |  | sub  ParseEventAndAddProposed ($$$) { | 
					
						
							|  |  |  |   my($config, $veventFile, $modString) = @_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   my $icsImportFile = tmpnam(); | 
					
						
							| 
									
										
										
										
											2013-11-30 14:37:30 -05:00
										 |  |  |   my $newDiaryTempFile = tmpnam(); | 
					
						
							| 
									
										
										
										
											2013-11-30 14:05:00 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   my $newCalendar = Data::ICal->new(data => <<END_ICAL | 
					
						
							|  |  |  | BEGIN:VCALENDAR | 
					
						
							|  |  |  | VERSION:2.0 | 
					
						
							|  |  |  | PRODID:-//Emacs//NONSGML icalendar.el//EN | 
					
						
							|  |  |  | END:VCALENDAR | 
					
						
							|  |  |  | END_ICAL | 
					
						
							|  |  |  | ); | 
					
						
							|  |  |  |   my $oldCalendar = Data::ICal->new(filename => $veventFile); | 
					
						
							|  |  |  |   my $entries = (defined $oldCalendar) ? $oldCalendar->entries : []; | 
					
						
							|  |  |  |   foreach my $entry (@{$entries}) { | 
					
						
							|  |  |  |     my $summary = $entry->property("summary"); | 
					
						
							|  |  |  |     if (defined $summary) { | 
					
						
							|  |  |  |       die("Multiple summary found in $veventFile") unless @{$summary} == 1; | 
					
						
							| 
									
										
										
										
											2013-11-30 14:49:37 -05:00
										 |  |  |       $summary->[0]->value($modString . ": " .  $summary->[0]->value . " (from $veventFile)"); | 
					
						
							| 
									
										
										
										
											2013-11-30 14:05:00 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     $newCalendar->add_entry($entry); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   open(SINGLE_EVENT_ICAL, ">", $icsImportFile) or DieLog("Unable to overwrite $icsImportFile: $!"); | 
					
						
							|  |  |  |   print SINGLE_EVENT_ICAL $newCalendar->as_string; | 
					
						
							|  |  |  |   close SINGLE_EVENT_ICAL; | 
					
						
							|  |  |  |   DieLog("Error ($?) while writing $icsImportFile ($?): $!") unless $? == 0; | 
					
						
							|  |  |  |   undef $newCalendar; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   my($elispFH, $elispFile) = tempfile(); | 
					
						
							|  |  |  |   print $elispFH "(setq-default european-calendar-style t)\n" | 
					
						
							|  |  |  |     if $config->{calendarStyle} =~  /european/i; | 
					
						
							|  |  |  |   print $elispFH <<ELISP_END | 
					
						
							|  |  |  | (setq icalendar-uid-format "emacs-%u-%h-%s") | 
					
						
							| 
									
										
										
										
											2013-11-30 14:37:30 -05:00
										 |  |  | (icalendar-import-file "$icsImportFile" "$newDiaryTempFile") | 
					
						
							| 
									
										
										
										
											2013-11-30 14:05:00 -05:00
										 |  |  | ELISP_END | 
					
						
							|  |  |  | ; | 
					
						
							|  |  |  |   $elispFH->close(); | 
					
						
							| 
									
										
										
										
											2013-11-30 14:06:14 -05:00
										 |  |  |   my @emacsOutput = read_from_process($config->{emacsBinary}, '--no-windows', | 
					
						
							| 
									
										
										
										
											2013-11-30 14:05:00 -05:00
										 |  |  |                  '--batch', '--no-site-file', '-l', $elispFile); | 
					
						
							| 
									
										
										
										
											2013-11-30 14:06:14 -05:00
										 |  |  |   DieLog("Emacs process for importing $veventFile and " . | 
					
						
							|  |  |  |          "$config->{proposedDiary} exited with non-zero exit status of " . | 
					
						
							| 
									
										
										
										
											2013-11-30 14:05:00 -05:00
										 |  |  |          "$? ($!), and output of:\n    " . join("\n   ", @emacsOutput)) | 
					
						
							|  |  |  |     if ($? != 0); | 
					
						
							|  |  |  |   my $goodCount =0; | 
					
						
							| 
									
										
										
										
											2013-11-30 14:37:30 -05:00
										 |  |  |   foreach my $line (@emacsOutput) { | 
					
						
							|  |  |  |     $goodCount++ if $line =~ /^((Saving.*file|Wrote)\s*$newDiaryTempFile)/;  } | 
					
						
							| 
									
										
										
										
											2013-11-30 14:05:00 -05:00
										 |  |  |   DieLog("Unexpected Emacs output: " . join("\n   ", @emacsOutput)) | 
					
						
							| 
									
										
										
										
											2013-11-30 14:37:30 -05:00
										 |  |  |     if ($goodCount != 2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   open(NEW_DIARY, "<", $newDiaryTempFile) | 
					
						
							|  |  |  |     or DieLog("unable to read temp file: $newDiaryTempFile: $!"); | 
					
						
							|  |  |  |   open(REAL_DIARY, '>>', $config->{proposedDiary}) | 
					
						
							|  |  |  |     or DieLog("unable to append to config->{proposedDiary}: $!"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 14:58:51 -05:00
										 |  |  |   while (my $line = <NEW_DIARY>) { | 
					
						
							| 
									
										
										
										
											2013-12-29 16:22:45 -05:00
										 |  |  |     print "\$hr\n\${font :size=17}Calendar Proposals:\n" | 
					
						
							|  |  |  |       unless defined $PRINTED_CONKY_HEADER; | 
					
						
							|  |  |  |     $PRINTED_CONKY_HEADER = 1; | 
					
						
							|  |  |  |     print "\${font Inconsolata:size=13}$line"; | 
					
						
							| 
									
										
										
										
											2013-11-30 14:58:51 -05:00
										 |  |  |     chomp $line; | 
					
						
							|  |  |  |     print REAL_DIARY "$line\n" unless defined $config->{__knownLines}{$line}; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-11-30 14:37:30 -05:00
										 |  |  |   close NEW_DIARY;  DieLog("error reading $newDiaryTempFile: $!") unless ($? == 0); | 
					
						
							|  |  |  |   close REAL_DIARY;  DieLog("error appending to $config->{proposedDiary}: $!") unless ($? == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   WarnLog("unable to remove temporary files, $icsImportFile and $newDiaryTempFile: $!") | 
					
						
							|  |  |  |           unless unlink($icsImportFile, $newDiaryTempFile) == 2; | 
					
						
							| 
									
										
										
										
											2013-11-30 14:05:00 -05:00
										 |  |  | } | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							| 
									
										
										
										
											2013-11-30 14:20:49 -05:00
										 |  |  | sub HandleProposedEvent ($$$) { | 
					
						
							| 
									
										
										
										
											2013-11-30 13:41:49 -05:00
										 |  |  |   my($config, $operation, $file) = @_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if ($operation eq 'A') { | 
					
						
							| 
									
										
										
										
											2013-12-29 16:22:45 -05:00
										 |  |  |     ParseEventAndAddProposed($config, $file, "ADDITION"); | 
					
						
							| 
									
										
										
										
											2013-11-30 13:41:49 -05:00
										 |  |  |   } elsif ($operation eq 'M') { | 
					
						
							| 
									
										
										
										
											2013-12-29 16:22:45 -05:00
										 |  |  |     ParseEventAndAddProposed($config, $file, "CHANGE"); | 
					
						
							| 
									
										
										
										
											2013-11-30 13:41:49 -05:00
										 |  |  |   } elsif ($operation eq 'D') { | 
					
						
							|  |  |  |     chdir $config->{gitDir} or DieLog("Unable to change directory to $config->{gitDir}"); | 
					
						
							| 
									
										
										
										
											2013-11-30 14:37:56 -05:00
										 |  |  |     system($config->{gitBinary}, 'checkout', '-q', $config->{myBranch}); | 
					
						
							| 
									
										
										
										
											2013-11-30 13:41:49 -05:00
										 |  |  |     DieLog("Unable to checkout $config->{myBranch} branch in git") unless ($? == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-29 16:22:45 -05:00
										 |  |  |     ParseEventAndAddProposed($config, $file, "DELETE"); | 
					
						
							| 
									
										
										
										
											2013-11-30 13:41:49 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Now, reset back to incoming branch, as GenerateDiaryFromNewEvents assumes that. | 
					
						
							|  |  |  |     chdir $config->{gitDir} or DieLog("Unable to change directory to $config->{gitDir}"); | 
					
						
							| 
									
										
										
										
											2013-11-30 14:37:56 -05:00
										 |  |  |     system($config->{gitBinary}, 'checkout', '-q', $config->{incomingBranch}); | 
					
						
							| 
									
										
										
										
											2013-11-30 13:41:49 -05:00
										 |  |  |     DieLog("Unable to checkout $config->{incomingBranch} branch in git") unless ($? == 0); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     DieLog("Invalid operation of $operation for $file"); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							| 
									
										
										
										
											2013-11-30 14:58:51 -05:00
										 |  |  | sub KnownLines ($) { | 
					
						
							|  |  |  |   my($file) = @_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   open(DATA, "<", $file) or DieLog("Unable to open $file for reading: $!"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   my %lines; | 
					
						
							|  |  |  |   while (my $line = <DATA>) { chomp $line; $lines{$line} = $line; } | 
					
						
							|  |  |  |   return \%lines; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:44 -05:00
										 |  |  | sub GenerateDiaryFromNewEvents ($) { | 
					
						
							|  |  |  |   my($config)  = @_; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 14:58:51 -05:00
										 |  |  |   $config->{__knownLines} = KnownLines($config->{proposedDiary}); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:44 -05:00
										 |  |  |   chdir $config->{gitDir} or DieLog("Unable to change directory to $config->{gitDir}"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 14:37:56 -05:00
										 |  |  |   system($config->{gitBinary}, 'checkout', '-q', $config->{incomingBranch}); | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:44 -05:00
										 |  |  |   DieLog("Unable to checkout $config->{incomingBranch} branch in git") unless ($? == 0); | 
					
						
							|  |  |  |   my @gitDiffSummaryOutput = | 
					
						
							| 
									
										
										
										
											2013-11-30 14:06:14 -05:00
										 |  |  |     read_from_process($config->{gitBinary}, 'diff-index', $config->{myBranch}); | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:44 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   foreach my $line (@gitDiffSummaryOutput) { | 
					
						
							| 
									
										
										
										
											2013-11-30 14:37:56 -05:00
										 |  |  |     next unless $line =~ /$ENV{USER}/;   # Ignore lines that aren't for my calendar. | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:44 -05:00
										 |  |  |     DieLog("odd line in diff-index output: $line") unless | 
					
						
							|  |  |  |       $line =~ /(A|D|M)\s+(\S+)$/; | 
					
						
							|  |  |  |     my($operation, $file) = ($1, $2); | 
					
						
							| 
									
										
										
										
											2013-11-30 13:41:49 -05:00
										 |  |  |     HandleProposedEvent($config, $operation, $file); | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:44 -05:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-11-30 14:58:51 -05:00
										 |  |  |   delete $config->{__knownLines}; | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:44 -05:00
										 |  |  | } | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							| 
									
										
										
										
											2013-11-30 14:11:02 -05:00
										 |  |  | sub ReadConfig($) { | 
					
						
							|  |  |  |   my($configFile) = @_; | 
					
						
							|  |  |  |   open (CONFIG_FILE, "<", $configFile) or DieLog("unable to read $configFile ($?): $!"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   my %config; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (my $line = <CONFIG_FILE>) { | 
					
						
							|  |  |  |     chomp $line; | 
					
						
							|  |  |  |     DieLog("Unable to parse $line in config file, $configFile") | 
					
						
							|  |  |  |       unless $line =~ /^\s*([^:]+)\s*:\s*([^:]+)\s*$/; | 
					
						
							|  |  |  |     $config{$1} = $2; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   close CONFIG_FILE;  DieLog("Error reading $configFile ($?): $!") if $? != 0; | 
					
						
							|  |  |  |   return \%config; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | ############################################################################### | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:20 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | system("/usr/bin/lockfile -r 8 $CALENDAR_LOCK_FILE"); | 
					
						
							|  |  |  | DieLog("Failure to acquire calendar lock on $CALENDAR_LOCK_FILE") unless ($? == 0); | 
					
						
							|  |  |  | my $config = ReadConfig($CONFIG_FILE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | $config->{scrubPrivate} = 0 if not defined $config->{scrubPrivate}; | 
					
						
							|  |  |  | $config->{reportProblems} = $config->{user} if not defined $config->{reportProblems}; | 
					
						
							|  |  |  | $config->{emacsBinary} = "/usr/bin/emacs" if not defined $config->{emacsBinary}; | 
					
						
							|  |  |  | $config->{calendarStyle} = 'plain' if not defined $config->{calendarStyle}; | 
					
						
							| 
									
										
										
										
											2013-11-30 14:12:28 -05:00
										 |  |  | foreach my $bin (qw/emacsBinary gitBinary/) { | 
					
						
							| 
									
										
										
										
											2013-11-30 14:20:07 -05:00
										 |  |  |   DieLog("$config->{$bin} doesn't appear to be executable for $bin") | 
					
						
							|  |  |  |     unless defined $config->{$bin} and -x $config->{$bin}; | 
					
						
							| 
									
										
										
										
											2013-11-30 14:12:28 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:20 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | DieLog("$CONFIG_FILE doesn't specify a (readable) Git directory via gitDir setting: $!") | 
					
						
							|  |  |  |   unless defined $config->{gitDir} and -d $config->{gitDir}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-30 14:11:02 -05:00
										 |  |  | GenerateDiaryFromNewEvents($config); | 
					
						
							| 
									
										
										
										
											2013-12-29 16:22:45 -05:00
										 |  |  | print "\$hr\n" if ($PRINTED_CONKY_HEADER); | 
					
						
							| 
									
										
										
										
											2013-11-30 13:37:20 -05:00
										 |  |  | &$LOCK_CLEANUP_CODE(); | 
					
						
							|  |  |  | __END__ | 
					
						
							|  |  |  | # Local variables: | 
					
						
							|  |  |  | # compile-command: "perl -c calendar-import.plx" | 
					
						
							|  |  |  | # End: |