2 # BEGIN BPS TAGGED BLOCK {{{
6 # This software is Copyright (c) 1996-2015 Best Practical Solutions, LLC
7 # <sales@bestpractical.com>
9 # (Except where explicitly superseded by other copyright notices)
14 # This work is made available to you under the terms of Version 2 of
15 # the GNU General Public License. A copy of that license should have
16 # been provided with this software, but in any event can be snarfed
19 # This work is distributed in the hope that it will be useful, but
20 # WITHOUT ANY WARRANTY; without even the implied warranty of
21 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
22 # General Public License for more details.
24 # You should have received a copy of the GNU General Public License
25 # along with this program; if not, write to the Free Software
26 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
27 # 02110-1301 or visit their web page on the internet at
28 # http://www.gnu.org/licenses/old-licenses/gpl-2.0.html.
31 # CONTRIBUTION SUBMISSION POLICY:
33 # (The following paragraph is not intended to limit the rights granted
34 # to you to modify and distribute this software under the terms of
35 # the GNU General Public License and is only of importance to you if
36 # you choose to contribute your changes and enhancements to the
37 # community by submitting them to Best Practical Solutions, LLC.)
39 # By intentionally submitting any modifications, corrections or
40 # derivatives to this work, or any other work intended for use with
41 # Request Tracker, to Best Practical Solutions, LLC, you confirm that
42 # you are the copyright holder for those contributions and you grant
43 # Best Practical Solutions, LLC a nonexclusive, worldwide, irrevocable,
44 # royalty-free, perpetual, license to use, copy, create derivative
45 # works based on those contributions, and sublicense and distribute
46 # those contributions and any derivatives thereof.
48 # END BPS TAGGED BLOCK }}}
52 BEGIN { # BEGIN RT CMD BOILERPLATE
55 my @libs = ("@RT_LIB_PATH@", "@LOCAL_LIB_PATH@");
59 unless ( File::Spec->file_name_is_absolute($lib) ) {
60 $bin_path ||= ( File::Spec->splitpath(Cwd::abs_path(__FILE__)) )[1];
61 $lib = File::Spec->catfile( $bin_path, File::Spec->updir, $lib );
68 use Date::Format qw( strftime );
71 use RT::Interface::CLI qw( loc );
72 use RT::Interface::Email;
79 print loc("Usage:") . " $0 -m (daily|weekly) [--print] [--help]\n";
81 "[_1] is a utility, meant to be run from cron, that dispatches all deferred RT notifications as a per-user digest.",
84 print "\n\t-m, --mode\t"
85 . loc("Specify whether this is a daily or weekly run.") . "\n";
86 print "\t-p, --print\t"
87 . loc("Print the resulting digest messages to STDOUT; don't mail them. Do not mark them as sent")
89 print "\t-v, --verbose\t" . loc("Give output even on messages successfully sent") . "\n";
90 print "\t-h, --help\t" . loc("Print this message") . "\n";
92 if ( $error eq 'help' ) {
95 print loc("Error") . ": " . loc($error) . "\n";
100 my ( $frequency, $print, $verbose, $help ) = ( '', '', '', '' );
102 'mode=s' => \$frequency,
104 'verbose' => \$verbose,
108 usage('help') if $help;
109 usage("Mode argument must be 'daily' or 'weekly'")
110 unless $frequency =~ /^(daily|weekly)$/;
112 run( $frequency, $print );
115 my $frequency = shift;
118 ## Find all the tickets that have been modified within the time frame
119 ## described by $frequency.
121 my ( $all_digest, $sent_transactions ) = find_transactions($frequency);
123 ## Iterate through our huge hash constructing the digest message
124 ## for each user and sending it.
126 foreach my $user ( keys %$all_digest ) {
127 my ( $contents_list, $contents_body ) = build_digest_for_user( $user, $all_digest->{$user} );
128 # Now we have a content head and a content body. We can send a message.
129 if ( send_digest( $user, $contents_list, $contents_body ) ) {
130 print "Sent message to $user\n" if $verbose;
131 mark_transactions_sent( $frequency, $user, values %{$sent_transactions->{$user}} ) unless ($print);
133 print "Failed to send message to $user\n";
142 my ( $to, $index, $messages ) = @_;
144 # Combine the index and the messages.
146 my $body = "============== Tickets with activity in the last "
147 . ( $frequency eq 'daily' ? "day" : "seven days" ) . "\n\n";
150 $body .= "\n\n============== Messages recorded in the last "
151 . ( $frequency eq 'daily' ? "day" : "seven days" ) . "\n\n";
154 # Load our template. If we cannot load the template, abort
155 # immediately rather than failing through many loops.
156 my $digest_template = RT::Template->new( RT->SystemUser );
157 my ( $ret, $msg ) = $digest_template->Load('Email Digest');
159 print loc("Failed to load template")
160 . " 'Email Digest': "
162 . ". Cannot continue.\n";
165 ( $ret, $msg ) = $digest_template->Parse( Argument => $body );
167 print loc("Failed to parse template")
168 . " 'Email Digest'. Cannot continue.\n";
172 # Set our sender and recipient.
173 $digest_template->MIMEObj->head->replace(
174 'From', Encode::encode( "UTF-8", RT::Config->Get('CorrespondAddress') ) );
175 $digest_template->MIMEObj->head->replace(
176 'To', Encode::encode( "UTF-8", $to ) );
179 $digest_template->MIMEObj->print;
182 return RT::Interface::Email::SendEmail( Entity => $digest_template->MIMEObj)
186 # =item mark_transactions_sent( $frequency, $user, @txn_list );
188 # Takes a frequency string (either 'daily' or 'weekly'), a user and one or more
189 # transaction objects as its arguments. Marks the given deferred
190 # notifications as sent.
194 sub mark_transactions_sent {
195 my ( $freq, $user, @txns ) = @_;
196 return unless $freq =~ /(daily|weekly)/;
198 foreach my $txn (@txns) {
200 # Grab the attribute, mark the "sent" as true, and store the new
202 if ( my $attr = $txn->FirstAttribute('DeferredRecipients') ) {
203 my $deferred = $attr->Content;
204 $deferred->{$freq}->{$user}->{'_sent'} = 1;
206 Name => 'DeferredRecipients',
207 Description => 'Deferred recipients for this message',
208 Content => $deferred,
215 my $frequency = shift;
217 # Specify a short time for digest overlap, in case we aren't starting
218 # this process exactly on time.
219 my $OVERLAP_HEDGE = -30;
221 my $since_date = RT::Date->new( RT->SystemUser );
222 $since_date->Set( Format => 'unix', Value => time() );
223 if ( $frequency eq 'daily' ) {
224 $since_date->AddDays(-1);
226 $since_date->AddDays(-7);
229 $since_date->AddSeconds($OVERLAP_HEDGE);
234 sub find_transactions {
235 my $frequency = shift;
236 my $since_date = since_date($frequency);
238 my $txns = RT::Transactions->new( RT->SystemUser );
240 # First limit to recent transactions.
244 VALUE => $since_date->ISO
247 # Next limit to ticket transactions.
249 FIELD => 'ObjectType',
251 VALUE => 'RT::Ticket',
252 ENTRYAGGREGATOR => 'AND'
255 my $sent_transactions = {};
257 while ( my $txn = $txns->Next ) {
258 my $ticket = $txn->Ticket;
259 my $queue = $txn->TicketObj->QueueObj->Name;
260 # Xxx todo - may clobber if two queues have the same name
261 foreach my $user ( $txn->DeferredRecipients($frequency) ) {
262 $all_digest->{$user}->{$queue}->{$ticket}->{ $txn->id } = $txn;
263 $sent_transactions->{$user}->{ $txn->id } = $txn;
267 return ( $all_digest, $sent_transactions );
270 sub build_digest_for_user {
272 my $user_digest = shift;
274 my $contents_list = ''; # Holds the digest index.
275 my $contents_body = ''; # Holds the digest body.
277 # Has the user been disabled since a message was deferred on his/her
279 my $user_obj = RT::User->new( RT->SystemUser );
280 $user_obj->LoadByEmail($user);
281 if ( $user_obj->PrincipalObj->Disabled ) {
282 print STDERR loc("Skipping disabled user") . " $user\n";
286 print loc("Message for user") . " $user:\n\n" if $print;
287 foreach my $queue ( keys %$user_digest ) {
288 $contents_list .= "Queue $queue:\n";
289 $contents_body .= "Queue $queue:\n";
290 foreach my $ticket ( sort keys %{ $user_digest->{$queue} } ) {
291 my $tkt_txns = $user_digest->{$queue}->{$ticket};
292 my $ticket_obj = RT::Ticket->new( RT->SystemUser );
293 $ticket_obj->Load($ticket);
295 # Spit out the index entry for this ticket.
296 my $ticket_title = sprintf(
298 $ticket, $ticket_obj->Status, $ticket_obj->OwnerObj->Name,
301 $contents_list .= $ticket_title;
303 # Spit out the messages for the transactions on this ticket.
304 $contents_body .= "\n== $ticket_title\n";
305 foreach my $txn ( sort keys %$tkt_txns ) {
306 my $top = $tkt_txns->{$txn}->Attachments->First;
308 # $top contains the top-most RT::Attachment with our
309 # outgoing message. It may not be the MIME part with
310 # the content. Print a few headers from it for
312 $contents_body .= "From: " . $top->GetHeader('From') . "\n";
313 my $date = $top->GetHeader('Date ');
315 my $txn_obj = RT::Transaction->new( RT->SystemUser );
316 $txn_obj->Load($txn);
317 my $date_obj = RT::Date->new( RT->SystemUser );
320 Value => $txn_obj->Created
322 $date = strftime( '%a, %d %b %Y %H:%M:%S %z',
323 @{ [ localtime( $date_obj->Unix ) ] } );
325 $contents_body .= "Date: $date\n\n";
326 $contents_body .= $tkt_txns->{$txn}->ContentObj->Content . "\n";
327 $contents_body .= "-------\n";
328 } # foreach transaction
332 return ( $contents_list, $contents_body );
340 rt-email-digest - dispatch deferred notifications as a per-user digest
344 rt-email-digest -m (daily|weekly) [--print] [--help]
348 This script is a tool to dispatch all deferred RT notifications as a per-user
357 Specify whether this is a daily or weekly run.
359 --mode is equal to -m
363 Print the resulting digest messages to STDOUT; don't mail them. Do not mark them as sent
365 --print is equal to -p
371 --help is equal to -h