enter_bug.cgi 24.1 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>
25
#                 Nitish Bezzala <nbezzala@yahoo.com>
terry%netscape.com's avatar
terry%netscape.com committed
26

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

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

39
use lib qw(. lib);
40

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

55
my $user = Bugzilla->login(LOGIN_REQUIRED);
56

57 58 59
my $cloned_bug;
my $cloned_bug_id;

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

65 66 67
# All pages point to the same part of the documentation.
$vars->{'doc_section'} = 'bugreports.html';

68 69 70
my $product_name = trim($cgi->param('product') || '');
# Will contain the product object the bug is created in.
my $product;
71

72
if ($product_name eq '') {
73 74 75 76
    # 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);

77
    my $classification = Bugzilla->params->{'useclassification'} ?
78
        scalar($cgi->param('classification')) : '__all';
79

80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100
    # 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});
101
        }
102
    }
103

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

            $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());
118
            exit;
119
        }
120
        # If we come here, then there is only one classification available.
121
        $classification = $classifications[0]->{'object'}->name;
122
    }
123

124 125 126 127 128 129 130
    # 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;
131
            @classifications = ({object => $class, products => \@enterable_products});
132
        }
133 134
        else {
            @enterable_products = ();
135
        }
136
    }
137

138 139 140 141
    if (scalar(@enterable_products) == 0) {
        ThrowUserError('no_products');
    }
    elsif (scalar(@enterable_products) > 1) {
142
        $vars->{'classifications'} = \@classifications;
143
        $vars->{'target'} = "enter_bug.cgi";
144
        $vars->{'format'} = $cgi->param('format');
145
        $vars->{'cloned_bug_id'} = $cgi->param('cloned_bug_id');
146

147
        print $cgi->header();
148 149
        $template->process("global/choose-product.html.tmpl", $vars)
          || ThrowTemplateError($template->error());
150
        exit;
151
    } else {
152
        # Only one product exists.
153
        $product = $enterable_products[0];
154
    }
155
}
156 157 158

# We need to check and make sure that the user has permission
# to enter a bug against this product.
159
$product = $user->can_enter_product($product || $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 201 202 203
            /\(.*PowerPC.*\)/i && do {push @platform, ("PowerPC", "Macintosh");};
        #AMD64, Intel x86_64
            /\(.*amd64.*\)/ && do {push @platform, ("AMD64", "x86_64", "PC");};
            /\(.*x86_64.*\)/ && do {push @platform, ("AMD64", "x86_64", "PC");};
        #Intel Itanium
            /\(.*IA64.*\)/ && do {push @platform, "IA64";};
204
        #Intel x86
205 206
            /\(.*Intel.*\)/ && do {push @platform, ("IA32", "x86", "PC");};
            /\(.*[ix0-9]86.*\)/ && do {push @platform, ("IA32", "x86", "PC");};
207
        #Versions of Windows that only run on Intel x86
208 209
            /\(.*Win(?:dows |)[39M].*\)/ && do {push @platform, ("IA32", "x86", "PC");};
            /\(.*Win(?:dows |)16.*\)/ && do {push @platform, ("IA32", "x86", "PC");};
210
        #Sparc
211 212
            /\(.*sparc.*\)/ && do {push @platform, ("Sparc", "Sun");};
            /\(.*sun4.*\)/ && do {push @platform, ("Sparc", "Sun");};
213
        #Alpha
214 215 216
            /\(.*AXP.*\)/i && do {push @platform, ("Alpha", "DEC");};
            /\(.*[ _]Alpha.\D/i && do {push @platform, ("Alpha", "DEC");};
            /\(.*[ _]Alpha\)/i && do {push @platform, ("Alpha", "DEC");};
217
        #MIPS
218 219
            /\(.*IRIX.*\)/i && do {push @platform, ("MIPS", "SGI");};
            /\(.*MIPS.*\)/i && do {push @platform, ("MIPS", "SGI");};
220
        #68k
221 222
            /\(.*68K.*\)/ && do {push @platform, ("68k", "Macintosh");};
            /\(.*680[x0]0.*\)/ && do {push @platform, ("68k", "Macintosh");};
223
        #HP
224
            /\(.*9000.*\)/ && do {push @platform, ("PA-RISC", "HP");};
225
        #ARM
226 227 228 229 230 231
            /\(.*ARM.*\)/ && do {push @platform, ("ARM", "PocketPC");};
        #PocketPC intentionally before PowerPC
            /\(.*Windows CE.*PPC.*\)/ && do {push @platform, ("ARM", "PocketPC");};
        #PowerPC
            /\(.*PPC.*\)/ && do {push @platform, ("PowerPC", "Macintosh");};
            /\(.*AIX.*\)/ && do {push @platform, ("PowerPC", "Macintosh");};
232
        #Stereotypical and broken
233 234 235 236 237 238 239 240 241 242
            /\(.*Windows CE.*\)/ && do {push @platform, ("ARM", "PocketPC");};
            /\(.*Macintosh.*\)/ && do {push @platform, ("68k", "Macintosh");};
            /\(.*Mac OS [89].*\)/ && do {push @platform, ("68k", "Macintosh");};
            /\(.*Win64.*\)/ && do {push @platform, "IA64";};
            /\(Win.*\)/ && do {push @platform, ("IA32", "x86", "PC");};
            /\(.*Win(?:dows[ -])NT.*\)/ && do {push @platform, ("IA32", "x86", "PC");};
            /\(.*OSF.*\)/ && do {push @platform, ("Alpha", "DEC");};
            /\(.*HP-?UX.*\)/i && do {push @platform, ("PA-RISC", "HP");};
            /\(.*IRIX.*\)/i && do {push @platform, ("MIPS", "SGI");};
            /\(.*(SunOS|Solaris).*\)/ && do {push @platform, ("Sparc", "Sun");};
243
        #Braindead old browsers who didn't follow convention:
244 245
            /Amiga/ && do {push @platform, ("68k", "Macintosh");};
            /WinMosaic/ && do {push @platform, ("IA32", "x86", "PC");};
246
        }
terry%netscape.com's avatar
terry%netscape.com committed
247
    }
248 249

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

252 253 254
sub pickos {
    if (formvalue('op_sys') ne "") {
        return formvalue('op_sys');
terry%netscape.com's avatar
terry%netscape.com committed
255
    }
256

257
    my @os = ();
258

259 260
    if (Bugzilla->params->{'defaultopsys'}) {
        @os = Bugzilla->params->{'defaultopsys'};
261
    } else {
262 263 264
        # This function will return the first
        # item in @os that is a valid platform choice. If
        # no choice is valid, we return "Other".
265
        for ($ENV{'HTTP_USER_AGENT'}) {
266
            /\(.*IRIX.*\)/ && do {push @os, "IRIX";};
267 268 269
            /\(.*OSF.*\)/ && do {push @os, "OSF/1";};
            /\(.*Linux.*\)/ && do {push @os, "Linux";};
            /\(.*Solaris.*\)/ && do {push @os, "Solaris";};
270 271 272 273 274 275 276 277 278 279 280 281 282
            /\(.*SunOS.*\)/ && do {
              /\(.*SunOS 5.11.*\)/ && do {push @os, ("OpenSolaris", "Opensolaris", "Solaris 11");};
              /\(.*SunOS 5.10.*\)/ && do {push @os, "Solaris 10";};
              /\(.*SunOS 5.9.*\)/ && do {push @os, "Solaris 9";};
              /\(.*SunOS 5.8.*\)/ && do {push @os, "Solaris 8";};
              /\(.*SunOS 5.7.*\)/ && do {push @os, "Solaris 7";};
              /\(.*SunOS 5.6.*\)/ && do {push @os, "Solaris 6";};
              /\(.*SunOS 5.5.*\)/ && do {push @os, "Solaris 5";};
              /\(.*SunOS 5.*\)/ && do {push @os, "Solaris";};
              /\(.*SunOS.*sun4u.*\)/ && do {push @os, "Solaris";};
              /\(.*SunOS.*i86pc.*\)/ && do {push @os, "Solaris";};
              /\(.*SunOS.*\)/ && do {push @os, "SunOS";};
            };
283
            /\(.*HP-?UX.*\)/ && do {push @os, "HP-UX";};
284 285 286 287 288 289
            /\(.*BSD.*\)/ && do {
              /\(.*BSD\/(?:OS|386).*\)/ && do {push @os, "BSDI";};
              /\(.*FreeBSD.*\)/ && do {push @os, "FreeBSD";};
              /\(.*OpenBSD.*\)/ && do {push @os, "OpenBSD";};
              /\(.*NetBSD.*\)/ && do {push @os, "NetBSD";};
            };
290 291 292 293 294
            /\(.*BeOS.*\)/ && do {push @os, "BeOS";};
            /\(.*AIX.*\)/ && do {push @os, "AIX";};
            /\(.*OS\/2.*\)/ && do {push @os, "OS/2";};
            /\(.*QNX.*\)/ && do {push @os, "Neutrino";};
            /\(.*VMS.*\)/ && do {push @os, "OpenVMS";};
295 296
            /\(.*Win.*\)/ && do {
              /\(.*Windows XP.*\)/ && do {push @os, "Windows XP";};
297
              /\(.*Windows NT 6\.1.*\)/ && do {push @os, "Windows 7";};
298 299 300 301 302 303 304 305 306 307 308 309 310 311
              /\(.*Windows NT 6\.0.*\)/ && do {push @os, "Windows Vista";};
              /\(.*Windows NT 5\.2.*\)/ && do {push @os, "Windows Server 2003";};
              /\(.*Windows NT 5\.1.*\)/ && do {push @os, "Windows XP";};
              /\(.*Windows 2000.*\)/ && do {push @os, "Windows 2000";};
              /\(.*Windows NT 5.*\)/ && do {push @os, "Windows 2000";};
              /\(.*Win.*9[8x].*4\.9.*\)/ && do {push @os, "Windows ME";};
              /\(.*Win(?:dows |)M[Ee].*\)/ && do {push @os, "Windows ME";};
              /\(.*Win(?:dows |)98.*\)/ && do {push @os, "Windows 98";};
              /\(.*Win(?:dows |)95.*\)/ && do {push @os, "Windows 95";};
              /\(.*Win(?:dows |)16.*\)/ && do {push @os, "Windows 3.1";};
              /\(.*Win(?:dows[ -]|)NT.*\)/ && do {push @os, "Windows NT";};
              /\(.*Windows.*NT.*\)/ && do {push @os, "Windows NT";};
            };
            /\(.*Mac OS X.*\)/ && do {
312 313 314 315 316 317 318 319 320
              /\(.*Mac OS X (?:|Mach-O |\()10.6.*\)/ && do {push @os, "Mac OS X 10.6";};
              /\(.*Mac OS X (?:|Mach-O |\()10.5.*\)/ && do {push @os, "Mac OS X 10.5";};
              /\(.*Mac OS X (?:|Mach-O |\()10.4.*\)/ && do {push @os, "Mac OS X 10.4";};
              /\(.*Mac OS X (?:|Mach-O |\()10.3.*\)/ && do {push @os, "Mac OS X 10.3";};
              /\(.*Mac OS X (?:|Mach-O |\()10.2.*\)/ && do {push @os, "Mac OS X 10.2";};
              /\(.*Mac OS X (?:|Mach-O |\()10.1.*\)/ && do {push @os, "Mac OS X 10.1";};
        # Unfortunately, OS X 10.4 was the first to support Intel. This is
        # fallback support because some browsers refused to include the OS
        # Version.
321
              /\(.*Intel.*Mac OS X.*\)/ && do {push @os, "Mac OS X 10.4";};
322 323
        # OS X 10.3 is the most likely default version of PowerPC Macs
        # OS X 10.0 is more for configurations which didn't setup 10.x versions
324 325
              /\(.*Mac OS X.*\)/ && do {push @os, ("Mac OS X 10.3", "Mac OS X 10.0", "Mac OS X");};
            };
326 327
            /\(.*32bit.*\)/ && do {push @os, "Windows 95";};
            /\(.*16bit.*\)/ && do {push @os, "Windows 3.1";};
328 329 330 331 332 333 334 335 336 337
            /\(.*Mac OS \d.*\)/ && do {
              /\(.*Mac OS 9.*\)/ && do {push @os, ("Mac System 9.x", "Mac System 9.0");};
              /\(.*Mac OS 8\.6.*\)/ && do {push @os, ("Mac System 8.6", "Mac System 8.5");};
              /\(.*Mac OS 8\.5.*\)/ && do {push @os, "Mac System 8.5";};
              /\(.*Mac OS 8\.1.*\)/ && do {push @os, ("Mac System 8.1", "Mac System 8.0");};
              /\(.*Mac OS 8\.0.*\)/ && do {push @os, "Mac System 8.0";};
              /\(.*Mac OS 8[^.].*\)/ && do {push @os, "Mac System 8.0";};
              /\(.*Mac OS 8.*\)/ && do {push @os, "Mac System 8.6";};
            };
            /\(.*Darwin.*\)/ && do {push @os, ("Mac OS X 10.0", "Mac OS X");};
338
        # Silly
339 340 341 342 343
            /\(.*Mac.*\)/ && do {
              /\(.*Mac.*PowerPC.*\)/ && do {push @os, "Mac System 9.x";};
              /\(.*Mac.*PPC.*\)/ && do {push @os, "Mac System 9.x";};
              /\(.*Mac.*68k.*\)/ && do {push @os, "Mac System 8.0";};
            };
344
        # Evil
345 346 347 348 349
            /Amiga/i && do {push @os, "Other";};
            /WinMosaic/ && do {push @os, "Windows 95";};
            /\(.*PowerPC.*\)/ && do {push @os, "Mac System 9.x";};
            /\(.*PPC.*\)/ && do {push @os, "Mac System 9.x";};
            /\(.*68K.*\)/ && do {push @os, "Mac System 8.0";};
350
        }
terry%netscape.com's avatar
terry%netscape.com committed
351
    }
352 353 354 355 356

    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
357
}
358 359 360
##############################################################################
# End of subroutines
##############################################################################
terry%netscape.com's avatar
terry%netscape.com committed
361

362 363 364
my $has_editbugs = $user->in_group('editbugs', $product->id);
my $has_canconfirm = $user->in_group('canconfirm', $product->id);

365 366 367 368 369 370
# 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) {
371 372
    $cloned_bug = Bugzilla::Bug->check($cloned_bug_id);
    $cloned_bug_id = $cloned_bug->id;
373 374
}

375
if (scalar(@{$product->components}) == 1) {
376
    # Only one component; just pick it.
377
    $cgi->param('component', $product->components->[0]->name);
378 379
}

380
my %default;
terry%netscape.com's avatar
terry%netscape.com committed
381

382 383
$vars->{'product'}               = $product;

384 385 386 387
$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');
388 389

$vars->{'assigned_to'}           = formvalue('assigned_to');
390
$vars->{'assigned_to_disabled'}  = !$has_editbugs;
391
$vars->{'cc_disabled'}           = 0;
392

393
$vars->{'qa_contact'}           = formvalue('qa_contact');
394
$vars->{'qa_contact_disabled'}  = !$has_editbugs;
395

396
$vars->{'cloned_bug_id'}         = $cloned_bug_id;
397

398
$vars->{'token'}             = issue_session_token('createbug:');
399

400

401
my @enter_bug_fields = grep { $_->enter_bug } Bugzilla->active_custom_fields;
402
foreach my $field (@enter_bug_fields) {
403 404 405 406 407 408 409 410
    my $cf_name = $field->name;
    my $cf_value = $cgi->param($cf_name);
    if (defined $cf_value) {
        if ($field->type == FIELD_TYPE_MULTI_SELECT) {
            $cf_value = [$cgi->param($cf_name)];
        }
        $default{$cf_name} = $vars->{$cf_name} = $cf_value;
    }
411 412
}

413
# This allows the Field visibility and value controls to work with the
414 415
# Classification and Product fields as a parent.
$default{'classification'} = $product->classification->name;
416 417
$default{'product'} = $product->name;

418
if ($cloned_bug_id) {
419

420
    $default{'component_'}    = $cloned_bug->component;
421 422 423 424
    $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;
425

426 427
    $vars->{'short_desc'}     = $cloned_bug->short_desc;
    $vars->{'bug_file_loc'}   = $cloned_bug->bug_file_loc;
428
    $vars->{'keywords'}       = $cloned_bug->keywords;
429 430
    $vars->{'dependson'}      = join (", ", $cloned_bug_id, @{$cloned_bug->dependson});
    $vars->{'blocked'}        = join (", ", @{$cloned_bug->blocked});
431
    $vars->{'deadline'}       = $cloned_bug->deadline;
432
    $vars->{'estimated_time'} = $cloned_bug->estimated_time;
433

434
    if (defined $cloned_bug->cc) {
435
        $vars->{'cc'}         = join (", ", @{$cloned_bug->cc});
436 437 438
    } else {
        $vars->{'cc'}         = formvalue('cc');
    }
439 440 441 442
    
    if ($cloned_bug->reporter->id != $user->id) {
        $vars->{'cc'} = join (", ", $cloned_bug->reporter->login, $vars->{'cc'}); 
    }
443

444
    foreach my $field (@enter_bug_fields) {
445 446
        my $field_name = $field->name;
        $vars->{$field_name} = $cloned_bug->$field_name;
447 448
    }

449 450 451
    # 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.
452 453
    my $bug_desc = $cloned_bug->comments({ order => 'oldest_to_newest' })->[0];
    my $isprivate = $bug_desc->is_private;
454 455 456 457

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

458
    if (!$isprivate || Bugzilla->user->is_insider) {
459 460 461
        # We use "body" to avoid any format_comment text, which would be
        # pointless to clone.
        $vars->{'comment'}        = $bug_desc->body;
462 463
        $vars->{'commentprivacy'} = $isprivate;
    }
464

465
} # end of cloned bug entry form
466

467
else {
468

469
    $default{'component_'}    = formvalue('component');
470 471
    $default{'priority'}      = formvalue('priority', Bugzilla->params->{'defaultpriority'});
    $default{'bug_severity'}  = formvalue('bug_severity', Bugzilla->params->{'defaultseverity'});
472 473 474
    $default{'rep_platform'}  = pickplatform();
    $default{'op_sys'}        = pickos();

475
    $vars->{'alias'}          = formvalue('alias');
476 477 478 479 480
    $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');
481
    $vars->{'deadline'}       = formvalue('deadline');
482
    $vars->{'estimated_time'} = formvalue('estimated_time');
483

484
    $vars->{'cc'}             = join(', ', $cgi->param('cc'));
485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502

    $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).
#
503 504
# Eventually maybe each product should have a "current version"
# parameter.
505
$vars->{'version'} = [map($_->name, @{$product->versions})];
506

507 508
my $version_cookie = $cgi->cookie("VERSION-" . $product->name);

509
if ( ($cloned_bug_id) &&
510
     ($product->name eq $cloned_bug->product ) ) {
511
    $default{'version'} = $cloned_bug->version;
512
} elsif (formvalue('version')) {
513
    $default{'version'} = formvalue('version');
514 515 516 517
} elsif (defined $version_cookie
         and grep { $_ eq $version_cookie } @{ $vars->{'version'} })
{
    $default{'version'} = $version_cookie;
518 519 520
} else {
    $default{'version'} = $vars->{'version'}->[$#{$vars->{'version'}}];
}
521

522
# Get list of milestones.
523
if ( Bugzilla->params->{'usetargetmilestone'} ) {
524
    $vars->{'target_milestone'} = [map($_->name, @{$product->milestones})];
525 526 527
    if (formvalue('target_milestone')) {
       $default{'target_milestone'} = formvalue('target_milestone');
    } else {
528
       $default{'target_milestone'} = $product->default_milestone;
529 530 531
    }
}

532
# Construct the list of allowable statuses.
533
my @statuses = @{ Bugzilla::Status->can_change_to() };
534 535
# Exclude closed states from the UI, even if the workflow allows them.
# The back-end code will still accept them, though.
536
@statuses = grep { $_->is_open } @statuses;
537

538 539
# UNCONFIRMED is illegal if allows_unconfirmed is false.
if (!$product->allows_unconfirmed) {
540
    @statuses = grep { $_->name ne 'UNCONFIRMED' } @statuses;
541
}
542
scalar(@statuses) || ThrowUserError('no_initial_bug_status');
543 544 545 546

# If the user has no privs...
unless ($has_editbugs || $has_canconfirm) {
    # ... use UNCONFIRMED if available, else use the first status of the list.
547 548
    my ($unconfirmed) = grep { $_->name eq 'UNCONFIRMED' } @statuses;
    @statuses = ($unconfirmed || $statuses[0]);
549 550
}

551
$vars->{'bug_status'} = \@statuses;
552

553
# Get the default from a template value if it is legitimate.
554 555
# Otherwise, and only if the user has privs, set the default
# to the first confirmed bug status on the list, if available.
556

557 558
my $picked_status = formvalue('bug_status');
if ($picked_status and grep($_->name eq $picked_status, @statuses)) {
559
    $default{'bug_status'} = formvalue('bug_status');
560 561
} elsif (scalar @statuses == 1) {
    $default{'bug_status'} = $statuses[0]->name;
562
}
563
else {
564 565
    $default{'bug_status'} = ($statuses[0]->name ne 'UNCONFIRMED') 
                             ? $statuses[0]->name : $statuses[1]->name;
566 567
}

568 569 570 571 572 573
my @groups = $cgi->param('groups');
if ($cloned_bug) {
    my @clone_groups = map { $_->name } @{ $cloned_bug->groups_in };
    # It doesn't matter if there are duplicate names, since all we check
    # for in the template is whether or not the group is set.
    push(@groups, @clone_groups);
574
}
575
$default{'groups'} = \@groups;
576

577
Bugzilla::Hook::process('enter_bug_entrydefaultvars', { vars => $vars });
578

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

581 582 583
my $format = $template->get_format("bug/create/create",
                                   scalar $cgi->param('format'), 
                                   scalar $cgi->param('ctype'));
584

585
print $cgi->header($format->{'ctype'});
586
$template->process($format->{'template'}, $vars)
587
  || ThrowTemplateError($template->error());          
588