747 lines
27 KiB
Perl
747 lines
27 KiB
Perl
# -*- Mode: perl; indent-tabs-mode: nil -*-
|
||
#
|
||
# The contents of this file are subject to the Mozilla Public
|
||
# License Version 1.1 (the "License"); you may not use this file
|
||
# except in compliance with the License. You may obtain a copy of
|
||
# the License at http://www.mozilla.org/MPL/
|
||
#
|
||
# Software distributed under the License is distributed on an "AS
|
||
# IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
|
||
# implied. See the License for the specific language governing
|
||
# rights and limitations under the License.
|
||
#
|
||
# The Original Code is the Bugzilla Bug Tracking System.
|
||
#
|
||
# The Initial Developer of the Original Code is Netscape Communications
|
||
# Corporation. Portions created by Netscape are
|
||
# Copyright (C) 1998 Netscape Communications Corporation. All
|
||
# Rights Reserved.
|
||
#
|
||
# Contributor(s): Terry Weissman <terry@mozilla.org>,
|
||
# Bryce Nesbitt <bryce-mozilla@nextbus.com>
|
||
# Dan Mosedale <dmose@mozilla.org>
|
||
# Alan Raetz <al_raetz@yahoo.com>
|
||
# Jacob Steenhagen <jake@actex.net>
|
||
# Matthew Tuck <matty@chariot.net.au>
|
||
# Bradley Baetz <bbaetz@student.usyd.edu.au>
|
||
# J. Paul Reed <preed@sigkill.com>
|
||
# Gervase Markham <gerv@gerv.net>
|
||
# Byron Jones <bugzilla@glob.com.au>
|
||
|
||
use strict;
|
||
|
||
package Bugzilla::BugMail;
|
||
|
||
use Bugzilla::Error;
|
||
use Bugzilla::User;
|
||
use Bugzilla::Constants;
|
||
use Bugzilla::Util;
|
||
use Bugzilla::Bug;
|
||
use Bugzilla::Classification;
|
||
use Bugzilla::Product;
|
||
use Bugzilla::Component;
|
||
use Bugzilla::Status;
|
||
use Bugzilla::Mailer;
|
||
use Bugzilla::CustisLocalBugzillas;
|
||
|
||
use Date::Parse;
|
||
use Date::Format;
|
||
|
||
use constant FORMAT_TRIPLE => "%19s|%-28s|%-28s";
|
||
use constant FORMAT_3_SIZE => [19,28,28];
|
||
use constant FORMAT_DOUBLE => "%19s %-55s";
|
||
use constant FORMAT_2_SIZE => [19,55];
|
||
|
||
use constant BIT_DIRECT => 1;
|
||
use constant BIT_WATCHING => 2;
|
||
|
||
# We need these strings for the X-Bugzilla-Reasons header
|
||
# Note: this hash uses "," rather than "=>" to avoid auto-quoting of the LHS.
|
||
use constant REL_NAMES => {
|
||
REL_ASSIGNEE , "AssignedTo",
|
||
REL_REPORTER , "Reporter",
|
||
REL_QA , "QAcontact",
|
||
REL_CC , "CC",
|
||
REL_VOTER , "Voter",
|
||
REL_GLOBAL_WATCHER, "GlobalWatcher"
|
||
};
|
||
|
||
# We use this instead of format because format doesn't deal well with
|
||
# multi-byte languages.
|
||
sub multiline_sprintf {
|
||
my ($format, $args, $sizes) = @_;
|
||
my @parts;
|
||
my @my_sizes = @$sizes; # Copy this so we don't modify the input array.
|
||
foreach my $string (@$args) {
|
||
my $size = shift @my_sizes;
|
||
my @pieces = split("\n", wrap_hard($string, $size));
|
||
push(@parts, \@pieces);
|
||
}
|
||
|
||
my $formatted;
|
||
while (1) {
|
||
# Get the first item of each part.
|
||
my @line = map { shift @$_ } @parts;
|
||
# If they're all undef, we're done.
|
||
last if !grep { defined $_ } @line;
|
||
# Make any single undef item into ''
|
||
@line = map { defined $_ ? $_ : '' } @line;
|
||
# And append a formatted line
|
||
$formatted .= sprintf($format, @line);
|
||
# Remove trailing spaces, or they become lots of =20's in
|
||
# quoted-printable emails.
|
||
$formatted =~ s/\s+$//;
|
||
$formatted .= "\n";
|
||
}
|
||
return $formatted;
|
||
}
|
||
|
||
sub three_columns {
|
||
return multiline_sprintf(FORMAT_TRIPLE, \@_, FORMAT_3_SIZE);
|
||
}
|
||
|
||
# This is a bit of a hack, basically keeping the old system()
|
||
# cmd line interface. Should clean this up at some point.
|
||
#
|
||
# args: bug_id, and an optional hash ref which may have keys for:
|
||
# changer, owner, qa, reporter, cc
|
||
# Optional hash contains values of people which will be forced to those
|
||
# roles when the email is sent.
|
||
# All the names are email addresses, not userids
|
||
# values are scalars, except for cc, which is a list
|
||
# This hash usually comes from the "mailrecipients" var in a template call.
|
||
sub Send {
|
||
my ($id, $forced) = (@_);
|
||
|
||
my @headerlist;
|
||
my %defmailhead;
|
||
my %fielddescription;
|
||
|
||
my $msg = "";
|
||
|
||
my $dbh = Bugzilla->dbh;
|
||
|
||
# XXX - These variables below are useless. We could use field object
|
||
# methods directly. But we first have to implement a cache in
|
||
# Bugzilla->get_fields to avoid querying the DB all the time.
|
||
foreach my $field (Bugzilla->get_fields({obsolete => 0})) {
|
||
push(@headerlist, $field->name);
|
||
$defmailhead{$field->name} = $field->in_new_bugmail;
|
||
$fielddescription{$field->name} = $field->description;
|
||
}
|
||
|
||
my %values = %{$dbh->selectrow_hashref(
|
||
'SELECT ' . join(',', editable_bug_fields()) . ', reporter,
|
||
lastdiffed AS start_time, LOCALTIMESTAMP(0) AS end_time
|
||
FROM bugs WHERE bug_id = ?',
|
||
undef, $id)};
|
||
|
||
my $product = new Bugzilla::Product($values{product_id});
|
||
$values{product} = $product->name;
|
||
if (Bugzilla->params->{'useclassification'}) {
|
||
$values{classification} = Bugzilla::Classification->new($product->classification_id)->name;
|
||
}
|
||
my $component = new Bugzilla::Component($values{component_id});
|
||
$values{component} = $component->name;
|
||
|
||
my ($start, $end) = ($values{start_time}, $values{end_time});
|
||
|
||
# User IDs of people in various roles. More than one person can 'have' a
|
||
# role, if the person in that role has changed, or people are watching.
|
||
my $reporter = $values{'reporter'};
|
||
my @assignees = ($values{'assigned_to'});
|
||
my @qa_contacts = ($values{'qa_contact'});
|
||
|
||
my $cc_users = $dbh->selectall_arrayref(
|
||
"SELECT cc.who, profiles.login_name
|
||
FROM cc
|
||
INNER JOIN profiles
|
||
ON cc.who = profiles.userid
|
||
WHERE bug_id = ?",
|
||
undef, $id);
|
||
|
||
my (@ccs, @cc_login_names);
|
||
foreach my $cc_user (@$cc_users) {
|
||
my ($user_id, $user_login) = @$cc_user;
|
||
push (@ccs, $user_id);
|
||
push (@cc_login_names, $user_login);
|
||
}
|
||
|
||
# Include the people passed in as being in particular roles.
|
||
# This can include people who used to hold those roles.
|
||
# At this point, we don't care if there are duplicates in these arrays.
|
||
my $changer = $forced->{'changer'};
|
||
if ($forced->{'owner'}) {
|
||
push (@assignees, login_to_id($forced->{'owner'}, THROW_ERROR));
|
||
}
|
||
|
||
if ($forced->{'qacontact'}) {
|
||
push (@qa_contacts, login_to_id($forced->{'qacontact'}, THROW_ERROR));
|
||
}
|
||
|
||
if ($forced->{'cc'}) {
|
||
foreach my $cc (@{$forced->{'cc'}}) {
|
||
push(@ccs, login_to_id($cc, THROW_ERROR));
|
||
}
|
||
}
|
||
|
||
# Convert to names, for later display
|
||
$values{'changer'} = $changer;
|
||
# If no changer is specified, then it has no name.
|
||
if ($changer) {
|
||
$values{'changername'} = Bugzilla::User->new({name => $changer})->name;
|
||
}
|
||
$values{'assigned_to'} = user_id_to_login($values{'assigned_to'});
|
||
$values{'reporter'} = user_id_to_login($values{'reporter'});
|
||
if ($values{'qa_contact'}) {
|
||
$values{'qa_contact'} = user_id_to_login($values{'qa_contact'});
|
||
}
|
||
$values{'cc'} = join(', ', @cc_login_names);
|
||
$values{'estimated_time'} = format_time_decimal($values{'estimated_time'});
|
||
|
||
if ($values{'deadline'}) {
|
||
$values{'deadline'} = time2str("%Y-%m-%d", str2time($values{'deadline'}));
|
||
}
|
||
|
||
my $dependslist = $dbh->selectcol_arrayref(
|
||
'SELECT dependson FROM dependencies
|
||
WHERE blocked = ? ORDER BY dependson',
|
||
undef, ($id));
|
||
|
||
$values{'dependson'} = join(",", @$dependslist);
|
||
|
||
my $blockedlist = $dbh->selectcol_arrayref(
|
||
'SELECT blocked FROM dependencies
|
||
WHERE dependson = ? ORDER BY blocked',
|
||
undef, ($id));
|
||
|
||
$values{'blocked'} = join(",", @$blockedlist);
|
||
|
||
my $grouplist = $dbh->selectcol_arrayref(
|
||
' SELECT name FROM groups
|
||
INNER JOIN bug_group_map
|
||
ON groups.id = bug_group_map.group_id
|
||
AND bug_group_map.bug_id = ?',
|
||
undef, ($id));
|
||
|
||
$values{'bug_group'} = join(', ', @$grouplist);
|
||
|
||
my @args = ($id);
|
||
|
||
# If lastdiffed is NULL, then we don't limit the search on time.
|
||
my $when_restriction = '';
|
||
if ($start) {
|
||
$when_restriction = ' AND bug_when > ? AND bug_when <= ?';
|
||
push @args, ($start, $end);
|
||
}
|
||
|
||
my $diffs = $dbh->selectall_arrayref(
|
||
"SELECT profiles.login_name, profiles.realname, fielddefs.description,
|
||
bugs_activity.bug_when, bugs_activity.removed,
|
||
bugs_activity.added, bugs_activity.attach_id, fielddefs.name
|
||
FROM bugs_activity
|
||
INNER JOIN fielddefs
|
||
ON fielddefs.id = bugs_activity.fieldid
|
||
INNER JOIN profiles
|
||
ON profiles.userid = bugs_activity.who
|
||
WHERE bugs_activity.bug_id = ?
|
||
$when_restriction
|
||
ORDER BY bugs_activity.bug_when", undef, @args);
|
||
|
||
my @diff_array = [];
|
||
|
||
my @new_depbugs;
|
||
my $difftext = "";
|
||
my $diffheader = "";
|
||
my @diffparts;
|
||
my $lastwho = "";
|
||
my $fullwho;
|
||
my @changedfields;
|
||
foreach my $ref (@$diffs) {
|
||
my ($who, $whoname, $what, $when, $old, $new, $attachid, $fieldname) = (@$ref);
|
||
my $diffpart = {};
|
||
if ($who ne $lastwho) {
|
||
$lastwho = $who;
|
||
$fullwho = $whoname ? "$whoname <$who>" : $who;
|
||
$diffheader = "\n$fullwho changed:\n\n";
|
||
$diffheader .= three_columns("What ", "Removed", "Added");
|
||
$diffheader .= ('-' x 76) . "\n";
|
||
push @diff_array, { type => 'who', name => $whoname, who => $who };
|
||
}
|
||
$what =~ s/^(Attachment )?/Attachment #$attachid / if $attachid;
|
||
if( $fieldname eq 'estimated_time' ||
|
||
$fieldname eq 'remaining_time' ) {
|
||
$old = format_time_decimal($old);
|
||
$new = format_time_decimal($new);
|
||
}
|
||
if ($fieldname eq 'dependson') {
|
||
push(@new_depbugs, grep {$_ =~ /^\d+$/} split(/[\s,]+/, $new));
|
||
}
|
||
if ($attachid) {
|
||
($diffpart->{'isprivate'}) = $dbh->selectrow_array(
|
||
'SELECT isprivate FROM attachments WHERE attach_id = ?',
|
||
undef, ($attachid));
|
||
}
|
||
$difftext = three_columns($what, $old, $new);
|
||
$diffpart->{'header'} = $diffheader;
|
||
$diffpart->{'fieldname'} = $fieldname;
|
||
$diffpart->{'text'} = $difftext;
|
||
push @diffparts, $diffpart;
|
||
push @changedfields, $what;
|
||
push @diff_array, {
|
||
type => 'change',
|
||
what => $what,
|
||
old => $old,
|
||
new => $new,
|
||
};
|
||
}
|
||
$values{'changed_fields'} = join(' ', @changedfields);
|
||
|
||
my @depbugs;
|
||
my $deptext = "";
|
||
# Do not include data about dependent bugs when they have just been added.
|
||
# Completely skip checking for dependent bugs on bug creation as all
|
||
# dependencies bugs will just have been added.
|
||
if ($start) {
|
||
my $dep_restriction = "";
|
||
if (scalar @new_depbugs) {
|
||
$dep_restriction = "AND bugs_activity.bug_id NOT IN (" .
|
||
join(", ", @new_depbugs) . ")";
|
||
}
|
||
|
||
my $dependency_diffs = $dbh->selectall_arrayref(
|
||
"SELECT bugs_activity.bug_id, bugs.short_desc, fielddefs.name,
|
||
bugs_activity.removed, bugs_activity.added
|
||
FROM bugs_activity
|
||
INNER JOIN bugs
|
||
ON bugs.bug_id = bugs_activity.bug_id
|
||
INNER JOIN dependencies
|
||
ON bugs_activity.bug_id = dependencies.dependson
|
||
INNER JOIN fielddefs
|
||
ON fielddefs.id = bugs_activity.fieldid
|
||
WHERE dependencies.blocked = ?
|
||
AND (fielddefs.name = 'bug_status'
|
||
OR fielddefs.name = 'resolution')
|
||
$when_restriction
|
||
$dep_restriction
|
||
ORDER BY bugs_activity.bug_when, bugs.bug_id", undef, @args);
|
||
|
||
my $thisdiff = "";
|
||
my $lastbug = "";
|
||
my $interestingchange = 0;
|
||
my @diff_tmp = ();
|
||
# TODO <20> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><><EFBFBD>, <20><><EFBFBD><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20> <20>**<2A>
|
||
foreach my $dependency_diff (@$dependency_diffs) {
|
||
my ($depbug, $summary, $what, $old, $new) = @$dependency_diff;
|
||
|
||
if ($depbug ne $lastbug) {
|
||
if ($interestingchange) {
|
||
$deptext .= $thisdiff;
|
||
push @diff_array, @diff_tmp;
|
||
@diff_tmp = ();
|
||
}
|
||
$lastbug = $depbug;
|
||
$thisdiff =
|
||
"\nBug $id depends on bug $depbug, which changed state.\n\n" .
|
||
"Bug $depbug Summary: $summary\n" .
|
||
correct_urlbase() . "show_bug.cgi?id=$depbug\n\n";
|
||
$thisdiff .= three_columns("What ", "Old Value", "New Value");
|
||
$thisdiff .= ('-' x 76) . "\n";
|
||
$interestingchange = 0;
|
||
push @diff_tmp, {
|
||
type => 'dep',
|
||
id => $id,
|
||
dep => $depbug,
|
||
summary => $summary,
|
||
};
|
||
}
|
||
$thisdiff .= three_columns($fielddescription{$what}, $old, $new);
|
||
if ($what eq 'bug_status'
|
||
&& is_open_state($old) ne is_open_state($new))
|
||
{
|
||
$interestingchange = 1;
|
||
}
|
||
push @depbugs, $depbug;
|
||
push @diff_tmp, {
|
||
type => 'change',
|
||
what => $fielddescription{$what},
|
||
old => $old,
|
||
new => $new,
|
||
};
|
||
}
|
||
|
||
if ($interestingchange)
|
||
{
|
||
$deptext .= $thisdiff;
|
||
push @diff_array, @diff_tmp;
|
||
@diff_tmp = ();
|
||
}
|
||
$deptext = trim($deptext);
|
||
|
||
if ($deptext)
|
||
{
|
||
push @diffparts, { text => "\n" . trim($deptext) };
|
||
}
|
||
}
|
||
|
||
my ($comments, $anyprivate) = get_comments_by_bug($id, $start, $end);
|
||
|
||
###########################################################################
|
||
# Start of email filtering code
|
||
###########################################################################
|
||
|
||
# A user_id => roles hash to keep track of people.
|
||
my %recipients;
|
||
my %watching;
|
||
|
||
# Now we work out all the people involved with this bug, and note all of
|
||
# the relationships in a hash. The keys are userids, the values are an
|
||
# array of role constants.
|
||
|
||
# Voters
|
||
my $voters = $dbh->selectcol_arrayref(
|
||
"SELECT who FROM votes WHERE bug_id = ?", undef, ($id));
|
||
|
||
$recipients{$_}->{+REL_VOTER} = BIT_DIRECT foreach (@$voters);
|
||
|
||
# CCs
|
||
$recipients{$_}->{+REL_CC} = BIT_DIRECT foreach (@ccs);
|
||
|
||
# Reporter (there's only ever one)
|
||
$recipients{$reporter}->{+REL_REPORTER} = BIT_DIRECT;
|
||
|
||
# QA Contact
|
||
if (Bugzilla->params->{'useqacontact'}) {
|
||
foreach (@qa_contacts) {
|
||
# QA Contact can be blank; ignore it if so.
|
||
$recipients{$_}->{+REL_QA} = BIT_DIRECT if $_;
|
||
}
|
||
}
|
||
|
||
# Assignee
|
||
$recipients{$_}->{+REL_ASSIGNEE} = BIT_DIRECT foreach (@assignees);
|
||
|
||
# The last relevant set of people are those who are being removed from
|
||
# their roles in this change. We get their names out of the diffs.
|
||
foreach my $ref (@$diffs) {
|
||
my ($who, $whoname, $what, $when, $old, $new) = (@$ref);
|
||
if ($old) {
|
||
# You can't stop being the reporter, and mail isn't sent if you
|
||
# remove your vote.
|
||
# Ignore people whose user account has been deleted or renamed.
|
||
if ($what eq "CC") {
|
||
foreach my $cc_user (split(/[\s,]+/, $old)) {
|
||
my $uid = login_to_id($cc_user);
|
||
$recipients{$uid}->{+REL_CC} = BIT_DIRECT if $uid;
|
||
}
|
||
}
|
||
elsif ($what eq "QAContact") {
|
||
my $uid = login_to_id($old);
|
||
$recipients{$uid}->{+REL_QA} = BIT_DIRECT if $uid;
|
||
}
|
||
elsif ($what eq "AssignedTo") {
|
||
my $uid = login_to_id($old);
|
||
$recipients{$uid}->{+REL_ASSIGNEE} = BIT_DIRECT if $uid;
|
||
}
|
||
}
|
||
}
|
||
|
||
# Find all those user-watching anyone on the current list, who is not
|
||
# on it already themselves.
|
||
my $involved = join(",", keys %recipients);
|
||
|
||
my $userwatchers =
|
||
$dbh->selectall_arrayref("SELECT watcher, watched FROM watch
|
||
WHERE watched IN ($involved)");
|
||
|
||
# Mark these people as having the role of the person they are watching
|
||
foreach my $watch (@$userwatchers) {
|
||
while (my ($role, $bits) = each %{$recipients{$watch->[1]}}) {
|
||
$recipients{$watch->[0]}->{$role} |= BIT_WATCHING
|
||
if $bits & BIT_DIRECT;
|
||
}
|
||
push(@{$watching{$watch->[0]}}, $watch->[1]);
|
||
}
|
||
|
||
# Global watcher
|
||
my @watchers = split(/[,\s]+/, Bugzilla->params->{'globalwatchers'});
|
||
foreach (@watchers) {
|
||
my $watcher_id = login_to_id($_);
|
||
next unless $watcher_id;
|
||
$recipients{$watcher_id}->{+REL_GLOBAL_WATCHER} = BIT_DIRECT;
|
||
}
|
||
|
||
# We now have a complete set of all the users, and their relationships to
|
||
# the bug in question. However, we are not necessarily going to mail them
|
||
# all - there are preferences, permissions checks and all sorts to do yet.
|
||
my @sent;
|
||
my @excluded;
|
||
|
||
foreach my $user_id (keys %recipients) {
|
||
my %rels_which_want;
|
||
my $sent_mail = 0;
|
||
|
||
my $user = new Bugzilla::User($user_id);
|
||
# Deleted users must be excluded.
|
||
next unless $user;
|
||
|
||
if ($user->can_see_bug($id)) {
|
||
# Go through each role the user has and see if they want mail in
|
||
# that role.
|
||
foreach my $relationship (keys %{$recipients{$user_id}}) {
|
||
if ($user->wants_bug_mail($id,
|
||
$relationship,
|
||
$diffs,
|
||
$comments,
|
||
$deptext,
|
||
$changer,
|
||
!$start))
|
||
{
|
||
$rels_which_want{$relationship} =
|
||
$recipients{$user_id}->{$relationship};
|
||
}
|
||
}
|
||
}
|
||
|
||
if (scalar(%rels_which_want)) {
|
||
# So the user exists, can see the bug, and wants mail in at least
|
||
# one role. But do we want to send it to them?
|
||
|
||
# If we are using insiders, and the comment is private, only send
|
||
# to insiders
|
||
my $insider_ok = 1;
|
||
$insider_ok = 0 if $anyprivate && !$user->is_insider;
|
||
|
||
# We shouldn't send mail if this is a dependency mail (i.e. there
|
||
# is something in @depbugs), and any of the depending bugs are not
|
||
# visible to the user. This is to avoid leaking the summaries of
|
||
# confidential bugs.
|
||
my $dep_ok = 1;
|
||
foreach my $dep_id (@depbugs) {
|
||
if (!$user->can_see_bug($dep_id)) {
|
||
$dep_ok = 0;
|
||
last;
|
||
}
|
||
}
|
||
|
||
# Make sure the user isn't in the nomail list, and the insider and
|
||
# dep checks passed.
|
||
if ($user->email_enabled &&
|
||
$insider_ok &&
|
||
$dep_ok)
|
||
{
|
||
# OK, OK, if we must. Email the user.
|
||
$sent_mail = sendMail(
|
||
user => $user,
|
||
headers => \@headerlist,
|
||
rels => \%rels_which_want,
|
||
values => \%values,
|
||
defhead => \%defmailhead,
|
||
fields => \%fielddescription,
|
||
diffs => \@diffparts,
|
||
diffar => \@diff_array,
|
||
newcomm => $comments,
|
||
anypriv => $anyprivate,
|
||
isnew => !$start,
|
||
id => $id,
|
||
watch => exists $watching{$user_id} ? $watching{$user_id} : undef,
|
||
);
|
||
}
|
||
}
|
||
|
||
if ($sent_mail) {
|
||
push(@sent, $user->login);
|
||
}
|
||
else {
|
||
push(@excluded, $user->login);
|
||
}
|
||
}
|
||
|
||
$dbh->do('UPDATE bugs SET lastdiffed = ? WHERE bug_id = ?',
|
||
undef, ($end, $id));
|
||
|
||
return {'sent' => \@sent, 'excluded' => \@excluded};
|
||
}
|
||
|
||
sub sendMail
|
||
{
|
||
my %arguments = @_;
|
||
my ($user, $hlRef, $relRef, $valueRef, $dmhRef, $fdRef,
|
||
$diffRef, $diffArray, $newcomments, $anyprivate, $isnew,
|
||
$id, $watchingRef
|
||
) = @arguments{qw(
|
||
user headers rels values defhead fields
|
||
diffs diffar newcomm anypriv isnew
|
||
id watch
|
||
)};
|
||
|
||
my %values = %$valueRef;
|
||
my @headerlist = @$hlRef;
|
||
my %mailhead = %$dmhRef;
|
||
my %fielddescription = %$fdRef;
|
||
my @diffparts = @$diffRef;
|
||
|
||
# Build difftext (the actions) by verifying the user should see them
|
||
my $difftext = "";
|
||
my $diffheader = "";
|
||
my $add_diff;
|
||
|
||
foreach my $diff (@diffparts)
|
||
{
|
||
$add_diff = 0;
|
||
|
||
if (exists($diff->{'fieldname'}) &&
|
||
($diff->{'fieldname'} eq 'estimated_time' ||
|
||
$diff->{'fieldname'} eq 'remaining_time' ||
|
||
$diff->{'fieldname'} eq 'work_time' ||
|
||
$diff->{'fieldname'} eq 'deadline'))
|
||
{
|
||
$add_diff = 1 if $user->is_timetracker;
|
||
} elsif ($diff->{'isprivate'}
|
||
&& !$user->is_insider)
|
||
{
|
||
$add_diff = 0;
|
||
} else {
|
||
$add_diff = 1;
|
||
}
|
||
|
||
if ($add_diff) {
|
||
if (exists($diff->{'header'}) &&
|
||
($diffheader ne $diff->{'header'})) {
|
||
$diffheader = $diff->{'header'};
|
||
$difftext .= $diffheader;
|
||
}
|
||
$difftext .= $diff->{'text'};
|
||
}
|
||
}
|
||
|
||
if ($difftext eq "" && !scalar(@$newcomments) && !$isnew) {
|
||
# Whoops, no differences!
|
||
return 0;
|
||
}
|
||
|
||
my $diffs = $difftext;
|
||
# Remove extra newlines.
|
||
$diffs =~ s/^\n+//s; $diffs =~ s/\n+$//s;
|
||
my @showfieldvalues = (); # for HTML emails
|
||
if ($isnew) {
|
||
my $head = "";
|
||
foreach my $f (@headerlist) {
|
||
next unless $mailhead{$f};
|
||
my $value = $values{$f};
|
||
# If there isn't anything to show, don't include this header.
|
||
next unless $value;
|
||
# Only send estimated_time if it is enabled and the user is in the group.
|
||
if (($f ne 'work_time' && $f ne 'estimated_time' && $f ne 'deadline') || $user->is_timetracker) {
|
||
my $desc = $fielddescription{$f};
|
||
$head .= multiline_sprintf(FORMAT_DOUBLE, ["$desc:", $value],
|
||
FORMAT_2_SIZE);
|
||
push @showfieldvalues, { desc => $desc, value => $value };
|
||
}
|
||
}
|
||
$diffs = $head . ($difftext ? "\n\n" : "") . $diffs;
|
||
}
|
||
|
||
my (@reasons, @reasons_watch);
|
||
while (my ($relationship, $bits) = each %{$relRef}) {
|
||
push(@reasons, $relationship) if ($bits & BIT_DIRECT);
|
||
push(@reasons_watch, $relationship) if ($bits & BIT_WATCHING);
|
||
}
|
||
|
||
my @headerrel = map { REL_NAMES->{$_} } @reasons;
|
||
my @watchingrel = map { REL_NAMES->{$_} } @reasons_watch;
|
||
push @headerrel, 'None' unless @headerrel;
|
||
push @watchingrel, 'None' unless @watchingrel;
|
||
push @watchingrel, map { user_id_to_login($_) } @$watchingRef;
|
||
|
||
my $vars = {
|
||
isnew => $isnew,
|
||
showfieldvalues => \@showfieldvalues,
|
||
to => $user->email,
|
||
bugid => $id,
|
||
alias => Bugzilla->params->{'usebugaliases'} ? $values{'alias'} : "",
|
||
classification => $values{'classification'},
|
||
product => $values{'product'},
|
||
comp => $values{'component'},
|
||
keywords => $values{'keywords'},
|
||
severity => $values{'bug_severity'},
|
||
status => $values{'bug_status'},
|
||
priority => $values{'priority'},
|
||
assignedto => $values{'assigned_to'},
|
||
assignedtoname => Bugzilla::User->new({name => $values{'assigned_to'}})->name,
|
||
targetmilestone => $values{'target_milestone'},
|
||
changedfields => $values{'changed_fields'},
|
||
summary => $values{'short_desc'},
|
||
reasons => \@reasons,
|
||
reasons_watch => \@reasons_watch,
|
||
reasonsheader => join(" ", @headerrel),
|
||
reasonswatchheader => join(" ", @watchingrel),
|
||
changer => $values{'changer'},
|
||
changername => $values{'changername'},
|
||
reporter => $values{'reporter'},
|
||
reportername => Bugzilla::User->new({name => $values{'reporter'}})->name,
|
||
diffs => $diffs,
|
||
diffarray => $diffArray,
|
||
new_comments => $newcomments,
|
||
threadingmarker => build_thread_marker($id, $user->id, $isnew),
|
||
};
|
||
|
||
# Hack into urlbase and set it to be correct for current user
|
||
Bugzilla::CustisLocalBugzillas::HackIntoUrlbase($user->email);
|
||
|
||
my $msg;
|
||
my $template = Bugzilla->template_inner($user->settings->{lang}->{value});
|
||
my $tmpl = '-'.$values{product};
|
||
$tmpl = '' unless $template->template_exists("email/newchangedmail$tmpl.txt.tmpl");
|
||
$template->process("email/newchangedmail$tmpl.txt.tmpl", $vars, \$msg)
|
||
|| ThrowTemplateError($template->error());
|
||
Bugzilla->template_inner("");
|
||
|
||
MessageToMTA($msg);
|
||
|
||
# Unhack urlbase :-)
|
||
Bugzilla::CustisLocalBugzillas::HackIntoUrlbase(undef);
|
||
|
||
return 1;
|
||
}
|
||
|
||
# Get bug comments for the given period.
|
||
sub get_comments_by_bug {
|
||
my ($id, $start, $end) = @_;
|
||
my $dbh = Bugzilla->dbh;
|
||
|
||
my $result = "";
|
||
my $count = 0;
|
||
my $anyprivate = 0;
|
||
|
||
# $start will be undef for new bugs, and defined for pre-existing bugs.
|
||
if ($start) {
|
||
# If $start is not NULL, obtain the count-index
|
||
# of this comment for the leading "Comment #xxx" line.
|
||
$count = $dbh->selectrow_array('SELECT COUNT(*) FROM longdescs
|
||
WHERE bug_id = ? AND bug_when <= ?',
|
||
undef, ($id, $start));
|
||
}
|
||
|
||
my $raw = 0; # Do not format comments which are not of type CMT_NORMAL.
|
||
my $comments = Bugzilla::Bug::GetComments($id, "oldest_to_newest", $start, $end, $raw);
|
||
my $attach_base = correct_urlbase() . 'attachment.cgi?id=';
|
||
|
||
foreach my $comment (@$comments) {
|
||
$comment->{count} = $count++;
|
||
# If an attachment was created, then add an URL. (Note: the 'g'lobal
|
||
# replace should work with comments with multiple attachments.)
|
||
if ($comment->{body} =~ /Created an attachment \(/) {
|
||
$comment->{body} =~ s/(Created an attachment \(id=([0-9]+)\))/$1\n --> \($attach_base$2\)/g;
|
||
}
|
||
}
|
||
|
||
if (Bugzilla->params->{'insidergroup'}) {
|
||
$anyprivate = 1 if scalar(grep {$_->{'isprivate'} > 0} @$comments);
|
||
}
|
||
|
||
return ($comments, $anyprivate);
|
||
}
|
||
|
||
1;
|