enter_bug.cgi 22.3 KB
Newer Older
1
#!/usr/bin/perl -wT
2
# -*- Mode: perl; indent-tabs-mode: nil -*-
terry%netscape.com's avatar
terry%netscape.com committed
3
#
4 5 6 7 8 9 10 11 12 13
# 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.
#
terry%netscape.com's avatar
terry%netscape.com committed
14
# The Original Code is the Bugzilla Bug Tracking System.
15
# 
terry%netscape.com's avatar
terry%netscape.com committed
16
# The Initial Developer of the Original Code is Netscape Communications
17 18 19
# Corporation. Portions created by Netscape are Copyright (C) 1998
# Netscape Communications Corporation. All Rights Reserved.
# 
terry%netscape.com's avatar
terry%netscape.com committed
20
# Contributor(s): Terry Weissman <terry@mozilla.org>
21
#                 Dave Miller <justdave@syndicomm.com>
22
#                 Joe Robins <jmrobins@tgix.com>
23
#                 Gervase Markham <gerv@gerv.net>
24
#                 Shane H. W. Travis <travis@sedsystems.ca>
terry%netscape.com's avatar
terry%netscape.com committed
25

26
##############################################################################
27 28 29 30
#
# enter_bug.cgi
# -------------
# Displays bug entry form. Bug fields are specified through popup menus, 
31 32
# drop-down lists, or text fields. Default for these values can be 
# passed in as parameters to the cgi.
33
#
34
##############################################################################
35

36
use strict;
terry%netscape.com's avatar
terry%netscape.com committed
37

38 39
use lib qw(.);

40
use Bugzilla;
41
use Bugzilla::Constants;
42 43
use Bugzilla::Util;
use Bugzilla::Error;
44
use Bugzilla::Bug;
45
use Bugzilla::User;
46
use Bugzilla::Hook;
47
use Bugzilla::Product;
48
use Bugzilla::Classification;
49
use Bugzilla::Keyword;
50
use Bugzilla::Token;
51
use Bugzilla::Field;
52

53
my $user = Bugzilla->login(LOGIN_REQUIRED);
54

55 56 57
my $cloned_bug;
my $cloned_bug_id;

58
my $cgi = Bugzilla->cgi;
59
my $dbh = Bugzilla->dbh;
60 61
my $template = Bugzilla->template;
my $vars = {};
62

63 64 65
my $product_name = trim($cgi->param('product') || '');
# Will contain the product object the bug is created in.
my $product;
66

67
if ($product_name eq '') {
68 69 70 71
    # If the user cannot enter bugs in any product, stop here.
    my @enterable_products = @{$user->get_enterable_products};
    ThrowUserError('no_products') unless scalar(@enterable_products);

72
    my $classification = Bugzilla->params->{'useclassification'} ?
73
        scalar($cgi->param('classification')) : '__all';
74

75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95
    # Unless a real classification name is given, we sort products
    # by classification.
    my @classifications;

    unless ($classification && $classification ne '__all') {
        if (Bugzilla->params->{'useclassification'}) {
            my $class;
            # Get all classifications with at least one enterable product.
            foreach my $product (@enterable_products) {
                $class->{$product->classification_id}->{'object'} ||=
                    new Bugzilla::Classification($product->classification_id);
                # Nice way to group products per classification, without querying
                # the DB again.
                push(@{$class->{$product->classification_id}->{'products'}}, $product);
            }
            @classifications = sort {$a->{'object'}->sortkey <=> $b->{'object'}->sortkey
                                     || lc($a->{'object'}->name) cmp lc($b->{'object'}->name)}
                                    (values %$class);
        }
        else {
            @classifications = ({object => undef, products => \@enterable_products});
96
        }
97
    }
98

99
    unless ($classification) {
100 101 102
        # We know there is at least one classification available,
        # else we would have stopped earlier.
        if (scalar(@classifications) > 1) {
103 104
            # We only need classification objects.
            $vars->{'classifications'} = [map {$_->{'object'}} @classifications];
105 106 107 108 109 110 111 112

            $vars->{'target'} = "enter_bug.cgi";
            $vars->{'format'} = $cgi->param('format');
            $vars->{'cloned_bug_id'} = $cgi->param('cloned_bug_id');

            print $cgi->header();
            $template->process("global/choose-classification.html.tmpl", $vars)
               || ThrowTemplateError($template->error());
113
            exit;
114
        }
115
        # If we come here, then there is only one classification available.
116
        $classification = $classifications[0]->{'object'}->name;
117
    }
118

119 120 121 122 123 124 125
    # Keep only enterable products which are in the specified classification.
    if ($classification ne "__all") {
        my $class = new Bugzilla::Classification({'name' => $classification});
        # If the classification doesn't exist, then there is no product in it.
        if ($class) {
            @enterable_products
              = grep {$_->classification_id == $class->id} @enterable_products;
126
            @classifications = ({object => $class, products => \@enterable_products});
127
        }
128 129
        else {
            @enterable_products = ();
130
        }
131
    }
132

133 134 135 136
    if (scalar(@enterable_products) == 0) {
        ThrowUserError('no_products');
    }
    elsif (scalar(@enterable_products) > 1) {
137
        $vars->{'classifications'} = \@classifications;
138
        $vars->{'target'} = "enter_bug.cgi";
139
        $vars->{'format'} = $cgi->param('format');
140
        $vars->{'cloned_bug_id'} = $cgi->param('cloned_bug_id');
141

142
        print $cgi->header();
143 144
        $template->process("global/choose-product.html.tmpl", $vars)
          || ThrowTemplateError($template->error());
145
        exit;
146
    } else {
147
        # Only one product exists.
148
        $product = $enterable_products[0];
149
    }
150
}
151 152 153 154 155 156 157 158 159
else {
    # Do not use Bugzilla::Product::check_product() here, else the user
    # could know whether the product doesn't exist or is not accessible.
    $product = new Bugzilla::Product({'name' => $product_name});
}

# We need to check and make sure that the user has permission
# to enter a bug against this product.
$user->can_enter_product($product ? $product->name : $product_name, THROW_ERROR);
terry%netscape.com's avatar
terry%netscape.com committed
160

161 162 163
##############################################################################
# Useful Subroutines
##############################################################################
164 165
sub formvalue {
    my ($name, $default) = (@_);
166
    return Bugzilla->cgi->param($name) || $default || "";
167
}
terry%netscape.com's avatar
terry%netscape.com committed
168

169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184
# Takes the name of a field and a list of possible values for that 
# field. Returns the first value in the list that is actually a 
# valid value for that field.
# The field should be named after its DB table.
# Returns undef if none of the platforms match.
sub pick_valid_field_value (@) {
    my ($field, @values) = @_;
    my $dbh = Bugzilla->dbh;

    foreach my $value (@values) {
        return $value if $dbh->selectrow_array(
            "SELECT 1 FROM $field WHERE value = ?", undef, $value); 
    }
    return undef;
}

185
sub pickplatform {
186 187
    return formvalue("rep_platform") if formvalue("rep_platform");

188 189
    my @platform;

190 191
    if (Bugzilla->params->{'defaultplatform'}) {
        @platform = Bugzilla->params->{'defaultplatform'};
192
    } else {
193 194 195
        # If @platform is a list, this function will return the first
        # item in the list that is a valid platform choice. If
        # no choice is valid, we return "Other".
196
        for ($ENV{'HTTP_USER_AGENT'}) {
197
        #PowerPC
198 199 200
            /\(.*PowerPC.*\)/i && do {@platform = "Macintosh"; last;};
            /\(.*PPC.*\)/ && do {@platform = "Macintosh"; last;};
            /\(.*AIX.*\)/ && do {@platform = "Macintosh"; last;};
201
        #Intel x86
202
            /\(.*Intel.*\)/ && do {@platform = "PC"; last;};
203
            /\(.*[ix0-9]86.*\)/ && do {@platform = "PC"; last;};
204
        #Versions of Windows that only run on Intel x86
205 206
            /\(.*Win(?:dows |)[39M].*\)/ && do {@platform = "PC"; last};
            /\(.*Win(?:dows |)16.*\)/ && do {@platform = "PC"; last;};
207
        #Sparc
208 209
            /\(.*sparc.*\)/ && do {@platform = "Sun"; last;};
            /\(.*sun4.*\)/ && do {@platform = "Sun"; last;};
210
        #Alpha
211 212 213
            /\(.*AXP.*\)/i && do {@platform = "DEC"; last;};
            /\(.*[ _]Alpha.\D/i && do {@platform = "DEC"; last;};
            /\(.*[ _]Alpha\)/i && do {@platform = "DEC"; last;};
214
        #MIPS
215 216
            /\(.*IRIX.*\)/i && do {@platform = "SGI"; last;};
            /\(.*MIPS.*\)/i && do {@platform = "SGI"; last;};
217
        #68k
218 219
            /\(.*68K.*\)/ && do {@platform = "Macintosh"; last;};
            /\(.*680[x0]0.*\)/ && do {@platform = "Macintosh"; last;};
220
        #HP
221
            /\(.*9000.*\)/ && do {@platform = "HP"; last;};
222
        #ARM
223
#            /\(.*ARM.*\) && do {$platform = "ARM";};
224
        #Stereotypical and broken
225 226 227 228 229 230 231 232
            /\(.*Macintosh.*\)/ && do {@platform = "Macintosh"; last;};
            /\(.*Mac OS [89].*\)/ && do {@platform = "Macintosh"; last;};
            /\(Win.*\)/ && do {@platform = "PC"; last;};
            /\(.*Win(?:dows[ -])NT.*\)/ && do {@platform = "PC"; last;};
            /\(.*OSF.*\)/ && do {@platform = "DEC"; last;};
            /\(.*HP-?UX.*\)/i && do {@platform = "HP"; last;};
            /\(.*IRIX.*\)/i && do {@platform = "SGI"; last;};
            /\(.*(SunOS|Solaris).*\)/ && do {@platform = "Sun"; last;};
233
        #Braindead old browsers who didn't follow convention:
234 235
            /Amiga/ && do {@platform = "Macintosh"; last;};
            /WinMosaic/ && do {@platform = "PC"; last;};
236
        }
terry%netscape.com's avatar
terry%netscape.com committed
237
    }
238 239

    return pick_valid_field_value('rep_platform', @platform) || "Other";
terry%netscape.com's avatar
terry%netscape.com committed
240 241
}

242 243 244
sub pickos {
    if (formvalue('op_sys') ne "") {
        return formvalue('op_sys');
terry%netscape.com's avatar
terry%netscape.com committed
245
    }
246 247 248

    my @os;

249 250
    if (Bugzilla->params->{'defaultopsys'}) {
        @os = Bugzilla->params->{'defaultopsys'};
251
    } else {
252 253 254
        # This function will return the first
        # item in @os that is a valid platform choice. If
        # no choice is valid, we return "Other".
255
        for ($ENV{'HTTP_USER_AGENT'}) {
256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278
            /\(.*IRIX.*\)/ && do {@os = "IRIX"; last;};
            /\(.*OSF.*\)/ && do {@os = "OSF/1"; last;};
            /\(.*Linux.*\)/ && do {@os = "Linux"; last;};
            /\(.*Solaris.*\)/ && do {@os = "Solaris"; last;};
            /\(.*SunOS 5.*\)/ && do {@os = "Solaris"; last;};
            /\(.*SunOS.*sun4u.*\)/ && do {@os = "Solaris"; last;};
            /\(.*SunOS.*\)/ && do {@os = "SunOS"; last;};
            /\(.*HP-?UX.*\)/ && do {@os = "HP-UX"; last;};
            /\(.*BSD\/(?:OS|386).*\)/ && do {@os = "BSDI"; last;};
            /\(.*FreeBSD.*\)/ && do {@os = "FreeBSD"; last;};
            /\(.*OpenBSD.*\)/ && do {@os = "OpenBSD"; last;};
            /\(.*NetBSD.*\)/ && do {@os = "NetBSD"; last;};
            /\(.*BeOS.*\)/ && do {@os = "BeOS"; last;};
            /\(.*AIX.*\)/ && do {@os = "AIX"; last;};
            /\(.*OS\/2.*\)/ && do {@os = "OS/2"; last;};
            /\(.*QNX.*\)/ && do {@os = "Neutrino"; last;};
            /\(.*VMS.*\)/ && do {@os = "OpenVMS"; last;};
            /\(.*Windows XP.*\)/ && do {@os = "Windows XP"; last;};
            /\(.*Windows NT 5\.2.*\)/ && do {@os = "Windows Server 2003"; last;};
            /\(.*Windows NT 5\.1.*\)/ && do {@os = "Windows XP"; last;};
            /\(.*Windows 2000.*\)/ && do {@os = "Windows 2000"; last;};
            /\(.*Windows NT 5.*\)/ && do {@os = "Windows 2000"; last;};
            /\(.*Win.*9[8x].*4\.9.*\)/ && do {@os = "Windows ME"; last;};
279 280 281 282 283
            /\(.*Win(?:dows |)M[Ee].*\)/ && do {@os = "Windows ME"; last;};
            /\(.*Win(?:dows |)98.*\)/ && do {@os = "Windows 98"; last;};
            /\(.*Win(?:dows |)95.*\)/ && do {@os = "Windows 95"; last;};
            /\(.*Win(?:dows |)16.*\)/ && do {@os = "Windows 3.1"; last;};
            /\(.*Win(?:dows[ -]|)NT.*\)/ && do {@os = "Windows NT"; last;};
284 285 286 287 288 289
            /\(.*Windows.*NT.*\)/ && do {@os = "Windows NT"; last;};
            /\(.*32bit.*\)/ && do {@os = "Windows 95"; last;};
            /\(.*16bit.*\)/ && do {@os = "Windows 3.1"; last;};
            /\(.*Mac OS 9.*\)/ && do {@os = "Mac System 9.x"; last;};
            /\(.*Mac OS 8\.6.*\)/ && do {@os = "Mac System 8.6"; last;};
            /\(.*Mac OS 8\.5.*\)/ && do {@os = "Mac System 8.5"; last;};
290
        # Bugzilla doesn't have an entry for 8.1
291 292 293 294 295 296
            /\(.*Mac OS 8\.1.*\)/ && do {@os = "Mac System 8.0"; last;};
            /\(.*Mac OS 8\.0.*\)/ && do {@os = "Mac System 8.0"; last;};
            /\(.*Mac OS 8[^.].*\)/ && do {@os = "Mac System 8.0"; last;};
            /\(.*Mac OS 8.*\)/ && do {@os = "Mac System 8.6"; last;};
            /\(.*Mac OS X.*\)/ && do {@os = "Mac OS X 10.0"; last;};
            /\(.*Darwin.*\)/ && do {@os = "Mac OS X 10.0"; last;};
297
        # Silly
298 299 300
            /\(.*Mac.*PowerPC.*\)/ && do {@os = "Mac System 9.x"; last;};
            /\(.*Mac.*PPC.*\)/ && do {@os = "Mac System 9.x"; last;};
            /\(.*Mac.*68k.*\)/ && do {@os = "Mac System 8.0"; last;};
301
        # Evil
302 303 304 305 306
            /Amiga/i && do {@os = "Other"; last;};
            /WinMosaic/ && do {@os = "Windows 95"; last;};
            /\(.*PowerPC.*\)/ && do {@os = "Mac System 9.x"; last;};
            /\(.*PPC.*\)/ && do {@os = "Mac System 9.x"; last;};
            /\(.*68K.*\)/ && do {@os = "Mac System 8.0"; last;};
307
        }
terry%netscape.com's avatar
terry%netscape.com committed
308
    }
309 310 311 312 313

    push(@os, "Windows") if grep(/^Windows /, @os);
    push(@os, "Mac OS") if grep(/^Mac /, @os);

    return pick_valid_field_value('op_sys', @os) || "Other";
terry%netscape.com's avatar
terry%netscape.com committed
314
}
315 316 317
##############################################################################
# End of subroutines
##############################################################################
terry%netscape.com's avatar
terry%netscape.com committed
318

319 320 321
my $has_editbugs = $user->in_group('editbugs', $product->id);
my $has_canconfirm = $user->in_group('canconfirm', $product->id);

322 323 324 325 326 327 328
# If a user is trying to clone a bug
#   Check that the user has authorization to view the parent bug
#   Create an instance of Bug that holds the info from the parent
$cloned_bug_id = $cgi->param('cloned_bug_id');

if ($cloned_bug_id) {
    ValidateBugID($cloned_bug_id);
329
    $cloned_bug = new Bugzilla::Bug($cloned_bug_id);
330 331
}

332
if (scalar(@{$product->components}) == 1) {
333
    # Only one component; just pick it.
334
    $cgi->param('component', $product->components->[0]->name);
335 336
}

337
my %default;
terry%netscape.com's avatar
terry%netscape.com committed
338

339 340
$vars->{'product'}               = $product;

341 342 343 344
$vars->{'priority'}              = get_legal_field_values('priority');
$vars->{'bug_severity'}          = get_legal_field_values('bug_severity');
$vars->{'rep_platform'}          = get_legal_field_values('rep_platform');
$vars->{'op_sys'}                = get_legal_field_values('op_sys');
345

346
$vars->{'use_keywords'}          = 1 if Bugzilla::Keyword::keyword_count();
terry%netscape.com's avatar
terry%netscape.com committed
347

348
$vars->{'assigned_to'}           = formvalue('assigned_to');
349
$vars->{'assigned_to_disabled'}  = !$has_editbugs;
350
$vars->{'cc_disabled'}           = 0;
351

352
$vars->{'qa_contact'}           = formvalue('qa_contact');
353
$vars->{'qa_contact_disabled'}  = !$has_editbugs;
354

355
$vars->{'cloned_bug_id'}         = $cloned_bug_id;
356

357
$vars->{'token'}             = issue_session_token('createbug:');
358

359 360 361 362 363 364 365

my @enter_bug_fields = Bugzilla->get_fields({ custom => 1, obsolete => 0, 
                                              enter_bug => 1 });
foreach my $field (@enter_bug_fields) {
    $vars->{$field->name} = formvalue($field->name);
}

366
if ($cloned_bug_id) {
367

368
    $default{'component_'}    = $cloned_bug->component;
369 370 371 372
    $default{'priority'}      = $cloned_bug->{'priority'};
    $default{'bug_severity'}  = $cloned_bug->{'bug_severity'};
    $default{'rep_platform'}  = $cloned_bug->{'rep_platform'};
    $default{'op_sys'}        = $cloned_bug->{'op_sys'};
373

374 375
    $vars->{'short_desc'}     = $cloned_bug->{'short_desc'};
    $vars->{'bug_file_loc'}   = $cloned_bug->{'bug_file_loc'};
376
    $vars->{'keywords'}       = $cloned_bug->keywords;
377 378
    $vars->{'dependson'}      = $cloned_bug_id;
    $vars->{'blocked'}        = "";
379
    $vars->{'deadline'}       = $cloned_bug->{'deadline'};
380

381 382
    if (defined $cloned_bug->cc) {
        $vars->{'cc'}         = join (" ", @{$cloned_bug->cc});
383 384 385 386
    } else {
        $vars->{'cc'}         = formvalue('cc');
    }

387 388 389 390
    foreach my $field (@enter_bug_fields) {
        $vars->{$field->name} = $cloned_bug->{$field->name};
    }

391 392 393 394 395 396 397 398 399 400 401
# We need to ensure that we respect the 'insider' status of
# the first comment, if it has one. Either way, make a note
# that this bug was cloned from another bug.

    $cloned_bug->longdescs();
    my $isprivate             = $cloned_bug->{'longdescs'}->[0]->{'isprivate'};

    $vars->{'comment'}        = "";
    $vars->{'commentprivacy'} = 0;

    if ( !($isprivate) ||
402
         ( ( Bugzilla->params->{"insidergroup"} ) && 
403
           ( Bugzilla->user->in_group(Bugzilla->params->{"insidergroup"}) ) ) 
404 405 406 407
       ) {
        $vars->{'comment'}        = $cloned_bug->{'longdescs'}->[0]->{'body'};
        $vars->{'commentprivacy'} = $isprivate;
    }
408

409 410
# Ensure that the groupset information is set up for later use.
    $cloned_bug->groups();
411

412
} # end of cloned bug entry form
413

414
else {
415

416
    $default{'component_'}    = formvalue('component');
417 418
    $default{'priority'}      = formvalue('priority', Bugzilla->params->{'defaultpriority'});
    $default{'bug_severity'}  = formvalue('bug_severity', Bugzilla->params->{'defaultseverity'});
419 420 421 422 423 424 425 426
    $default{'rep_platform'}  = pickplatform();
    $default{'op_sys'}        = pickos();

    $vars->{'short_desc'}     = formvalue('short_desc');
    $vars->{'bug_file_loc'}   = formvalue('bug_file_loc', "http://");
    $vars->{'keywords'}       = formvalue('keywords');
    $vars->{'dependson'}      = formvalue('dependson');
    $vars->{'blocked'}        = formvalue('blocked');
427
    $vars->{'deadline'}       = formvalue('deadline');
428

429
    $vars->{'cc'}             = join(', ', $cgi->param('cc'));
430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447

    $vars->{'comment'}        = formvalue('comment');
    $vars->{'commentprivacy'} = formvalue('commentprivacy');

} # end of normal/bookmarked entry form


# IF this is a cloned bug,
# AND the clone's product is the same as the parent's
#   THEN use the version from the parent bug
# ELSE IF a version is supplied in the URL
#   THEN use it
# ELSE IF there is a version in the cookie
#   THEN use it (Posting a bug sets a cookie for the current version.)
# ELSE
#   The default version is the last one in the list (which, it is
#   hoped, will be the most recent one).
#
448 449
# Eventually maybe each product should have a "current version"
# parameter.
450
$vars->{'version'} = [map($_->name, @{$product->versions})];
451 452

if ( ($cloned_bug_id) &&
453
     ($product->name eq $cloned_bug->product ) ) {
454 455
    $default{'version'} = $cloned_bug->{'version'};
} elsif (formvalue('version')) {
456
    $default{'version'} = formvalue('version');
457 458 459
} elsif (defined $cgi->cookie("VERSION-" . $product->name) &&
    lsearch($vars->{'version'}, $cgi->cookie("VERSION-" . $product->name)) != -1) {
    $default{'version'} = $cgi->cookie("VERSION-" . $product->name);
460 461 462
} else {
    $default{'version'} = $vars->{'version'}->[$#{$vars->{'version'}}];
}
463

464
# Get list of milestones.
465
if ( Bugzilla->params->{'usetargetmilestone'} ) {
466
    $vars->{'target_milestone'} = [map($_->name, @{$product->milestones})];
467 468 469
    if (formvalue('target_milestone')) {
       $default{'target_milestone'} = formvalue('target_milestone');
    } else {
470
       $default{'target_milestone'} = $product->default_milestone;
471 472 473
    }
}

474 475 476 477 478 479 480 481 482 483 484
# Construct the list of allowable statuses.
#
# * If the product requires votes to confirm:
#   users with privs   : NEW + ASSI + UNCO
#   users with no privs: UNCO
#
# * If the product doesn't require votes to confirm:
#   users with privs   : NEW + ASSI
#   users with no privs: NEW (as these users cannot reassign
#                             bugs to them, it doesn't make sense
#                             to let them mark bugs as ASSIGNED)
485

486
my @status;
487
if ($has_editbugs || $has_canconfirm) {
488 489 490 491 492
    @status = ('NEW', 'ASSIGNED');
}
elsif (!$product->votes_to_confirm) {
    @status = ('NEW');
}
493
if ($product->votes_to_confirm) {
494
    push(@status, 'UNCONFIRMED');
495 496
}

497
$vars->{'bug_status'} = \@status; 
498

499 500 501 502 503 504 505 506 507
# Get the default from a template value if it is legitimate.
# Otherwise, set the default to the first item on the list.

if (formvalue('bug_status') && (lsearch(\@status, formvalue('bug_status')) >= 0)) {
    $default{'bug_status'} = formvalue('bug_status');
} else {
    $default{'bug_status'} = $status[0];
}
 
508 509 510 511 512 513 514
my $grouplist = $dbh->selectall_arrayref(
                  q{SELECT DISTINCT groups.id, groups.name, groups.description,
                                    membercontrol, othercontrol
                      FROM groups
                 LEFT JOIN group_control_map
                        ON group_id = id AND product_id = ?
                     WHERE isbuggroup != 0 AND isactive != 0
515
                  ORDER BY description}, undef, $product->id);
516

517
my @groups;
518

519 520
foreach my $row (@$grouplist) {
    my ($id, $groupname, $description, $membercontrol, $othercontrol) = @$row;
521 522 523 524 525 526
    # Only include groups if the entering user will have an option.
    next if ((!$membercontrol) 
               || ($membercontrol == CONTROLMAPNA) 
               || ($membercontrol == CONTROLMAPMANDATORY)
               || (($othercontrol != CONTROLMAPSHOWN) 
                    && ($othercontrol != CONTROLMAPDEFAULT)
527
                    && (!Bugzilla->user->in_group($groupname)))
528
             );
529 530
    my $check;

531 532 533 534 535 536 537 538 539
    # If this is a cloned bug, 
    # AND the product for this bug is the same as for the original
    #   THEN set a group's checkbox if the original also had it on
    # ELSE IF this is a bookmarked template
    #   THEN set a group's checkbox if was set in the bookmark
    # ELSE
    #   set a groups's checkbox based on the group control map
    #
    if ( ($cloned_bug_id) &&
540
         ($product->name eq $cloned_bug->product ) ) {
541 542 543 544 545 546 547
        foreach my $i (0..(@{$cloned_bug->{'groups'}}-1) ) {
            if ($cloned_bug->{'groups'}->[$i]->{'bit'} == $id) {
                $check = $cloned_bug->{'groups'}->[$i]->{'ison'};
            }
        }
    }
    elsif(formvalue("maketemplate") ne "") {
548 549 550
        $check = formvalue("bit-$id", 0);
    }
    else {
551 552 553
        # Checkbox is checked by default if $control is a default state.
        $check = (($membercontrol == CONTROLMAPDEFAULT)
                 || (($othercontrol == CONTROLMAPDEFAULT)
554
                      && (!Bugzilla->user->in_group($groupname))));
555 556
    }

557 558 559 560 561 562 563 564
    my $group = 
    {
        'bit' => $id , 
        'checked' => $check , 
        'description' => $description 
    };

    push @groups, $group;        
565
}
terry%netscape.com's avatar
terry%netscape.com committed
566

567 568
$vars->{'group'} = \@groups;

569
Bugzilla::Hook::process("enter_bug-entrydefaultvars", { vars => $vars });
570

571
$vars->{'default'} = \%default;
terry%netscape.com's avatar
terry%netscape.com committed
572

573 574 575
my $format = $template->get_format("bug/create/create",
                                   scalar $cgi->param('format'), 
                                   scalar $cgi->param('ctype'));
576

577
print $cgi->header($format->{'ctype'});
578
$template->process($format->{'template'}, $vars)
579
  || ThrowTemplateError($template->error());          
580