enter_bug.cgi 12.8 KB
Newer Older
1
#!/usr/bin/perl -T
2 3 4
# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
terry%netscape.com's avatar
terry%netscape.com committed
5
#
6 7
# This Source Code Form is "Incompatible With Secondary Licenses", as
# defined by the Mozilla Public License, v. 2.0.
terry%netscape.com's avatar
terry%netscape.com committed
8

9
##############################################################################
10 11 12 13
#
# enter_bug.cgi
# -------------
# Displays bug entry form. Bug fields are specified through popup menus, 
14 15
# drop-down lists, or text fields. Default for these values can be 
# passed in as parameters to the cgi.
16
#
17
##############################################################################
18

19
use 5.10.1;
20
use strict;
21 22
use warnings;

23
use lib qw(. lib);
24

25
use Bugzilla;
26
use Bugzilla::Constants;
27 28
use Bugzilla::Util;
use Bugzilla::Error;
29
use Bugzilla::Bug;
30
use Bugzilla::Hook;
31
use Bugzilla::Classification;
32
use Bugzilla::Token;
33
use Bugzilla::Field;
34
use Bugzilla::Status;
35
use Bugzilla::UserAgent;
36

37 38
use List::MoreUtils qw(none);

39
my $user = Bugzilla->login(LOGIN_REQUIRED);
40

41 42 43
my $cloned_bug;
my $cloned_bug_id;

44
my $cgi = Bugzilla->cgi;
45
my $dbh = Bugzilla->dbh;
46 47
my $template = Bugzilla->template;
my $vars = {};
48

49
# All pages point to the same part of the documentation.
50
$vars->{'doc_section'} = 'using/filing.html';
51

52 53 54
my $product_name = trim($cgi->param('product') || '');
# Will contain the product object the bug is created in.
my $product;
55

56
if ($product_name eq '') {
57 58 59 60
    # 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);

61
    my $classification = Bugzilla->params->{'useclassification'} ?
62
        scalar($cgi->param('classification')) : '__all';
63

64 65 66 67 68
    # Unless a real classification name is given, we sort products
    # by classification.
    my @classifications;

    unless ($classification && $classification ne '__all') {
69
        @classifications = @{sort_products_by_classification(\@enterable_products)};
70
    }
71

72
    unless ($classification) {
73 74 75
        # We know there is at least one classification available,
        # else we would have stopped earlier.
        if (scalar(@classifications) > 1) {
76 77
            # We only need classification objects.
            $vars->{'classifications'} = [map {$_->{'object'}} @classifications];
78 79 80 81 82 83

            $vars->{'target'} = "enter_bug.cgi";

            print $cgi->header();
            $template->process("global/choose-classification.html.tmpl", $vars)
               || ThrowTemplateError($template->error());
84
            exit;
85
        }
86
        # If we come here, then there is only one classification available.
87
        $classification = $classifications[0]->{'object'}->name;
88
    }
89

90 91 92 93 94 95 96
    # 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;
97
            @classifications = ({object => $class, products => \@enterable_products});
98
        }
99 100
        else {
            @enterable_products = ();
101
        }
102
    }
103

104 105 106 107
    if (scalar(@enterable_products) == 0) {
        ThrowUserError('no_products');
    }
    elsif (scalar(@enterable_products) > 1) {
108
        $vars->{'classifications'} = \@classifications;
109
        $vars->{'target'} = "enter_bug.cgi";
110

111
        print $cgi->header();
112 113
        $template->process("global/choose-product.html.tmpl", $vars)
          || ThrowTemplateError($template->error());
114
        exit;
115
    } else {
116
        # Only one product exists.
117
        $product = $enterable_products[0];
118
    }
119
}
120 121 122

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

125 126 127
##############################################################################
# Useful Subroutines
##############################################################################
128 129
sub formvalue {
    my ($name, $default) = (@_);
130
    return Bugzilla->cgi->param($name) || $default || "";
131
}
terry%netscape.com's avatar
terry%netscape.com committed
132

133 134 135
##############################################################################
# End of subroutines
##############################################################################
terry%netscape.com's avatar
terry%netscape.com committed
136

137 138 139
my $has_editbugs = $user->in_group('editbugs', $product->id);
my $has_canconfirm = $user->in_group('canconfirm', $product->id);

140 141 142 143 144 145
# 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) {
146 147
    $cloned_bug = Bugzilla::Bug->check($cloned_bug_id);
    $cloned_bug_id = $cloned_bug->id;
148 149
}

150 151 152 153 154 155 156 157 158 159
# If there is only one active component, choose it
my @active = grep { $_->is_active } @{$product->components};
if (scalar(@active) == 1) {
    $cgi->param('component', $active[0]->name);
}

# If there is only one active version, choose it
@active = grep { $_->is_active } @{$product->versions};
if (scalar(@active) == 1) {
    $cgi->param('version', $active[0]->name);
160 161
}

162
my %default;
terry%netscape.com's avatar
terry%netscape.com committed
163

164 165
$vars->{'product'}               = $product;

166 167 168 169
$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');
170 171

$vars->{'assigned_to'}           = formvalue('assigned_to');
172
$vars->{'assigned_to_disabled'}  = !$has_editbugs;
173
$vars->{'cc_disabled'}           = 0;
174

175
$vars->{'qa_contact'}           = formvalue('qa_contact');
176
$vars->{'qa_contact_disabled'}  = !$has_editbugs;
177

178
$vars->{'cloned_bug_id'}         = $cloned_bug_id;
179

180
$vars->{'token'} = issue_session_token('create_bug');
181

182

183
my @enter_bug_fields = grep { $_->enter_bug } Bugzilla->active_custom_fields;
184
foreach my $field (@enter_bug_fields) {
185 186 187 188 189 190 191 192
    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;
    }
193 194
}

195
# This allows the Field visibility and value controls to work with the
196 197
# Classification and Product fields as a parent.
$default{'classification'} = $product->classification->name;
198 199
$default{'product'} = $product->name;

200
if ($cloned_bug_id) {
201

202
    $default{'component_'}    = $cloned_bug->component;
203 204 205 206
    $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;
207

208 209
    $vars->{'short_desc'}     = $cloned_bug->short_desc;
    $vars->{'bug_file_loc'}   = $cloned_bug->bug_file_loc;
210
    $vars->{'keywords'}       = $cloned_bug->keywords;
211 212
    $vars->{'dependson'}      = join (", ", $cloned_bug_id, @{$cloned_bug->dependson});
    $vars->{'blocked'}        = join (", ", @{$cloned_bug->blocked});
213
    $vars->{'deadline'}       = $cloned_bug->deadline;
214
    $vars->{'estimated_time'} = $cloned_bug->estimated_time;
215

216
    if (scalar @{$cloned_bug->cc}) {
217
        $vars->{'cc'}         = join (", ", @{$cloned_bug->cc});
218 219 220
    } else {
        $vars->{'cc'}         = formvalue('cc');
    }
221 222 223 224 225 226 227 228

    foreach my $role (qw(reporter assigned_to qa_contact)) {
        if (defined($cloned_bug->$role)
            && $cloned_bug->$role->id != $user->id
            && none { $_ eq $cloned_bug->$role->login } @{$cloned_bug->cc})
        {
            $vars->{'cc'} = join (", ", $cloned_bug->$role->login, $vars->{'cc'});
        }
229
    }
230

231
    foreach my $field (@enter_bug_fields) {
232 233
        my $field_name = $field->name;
        $vars->{$field_name} = $cloned_bug->$field_name;
234 235
    }

236 237 238
    # 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.
239 240
    my $bug_desc = $cloned_bug->comments({ order => 'oldest_to_newest' })->[0];
    my $isprivate = $bug_desc->is_private;
241

242 243
    $vars->{'comment'} = "";
    $vars->{'comment_is_private'} = 0;
244

245
    if (!$isprivate || $user->is_insider) {
246 247
        # We use "body" to avoid any format_comment text, which would be
        # pointless to clone.
248 249
        $vars->{'comment'} = $bug_desc->body;
        $vars->{'comment_is_private'} = $isprivate;
250
    }
251

252
} # end of cloned bug entry form
253

254 255
else {
    $default{'component_'}    = formvalue('component');
256 257
    $default{'priority'}      = formvalue('priority', Bugzilla->params->{'defaultpriority'});
    $default{'bug_severity'}  = formvalue('bug_severity', Bugzilla->params->{'defaultseverity'});
258 259 260 261
    $default{'rep_platform'}  = formvalue('rep_platform', 
                                          Bugzilla->params->{'defaultplatform'} || detect_platform());
    $default{'op_sys'}        = formvalue('op_sys', 
                                          Bugzilla->params->{'defaultopsys'} || detect_op_sys());
262

263
    $vars->{'alias'}          = formvalue('alias');
264 265 266 267 268
    $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');
269
    $vars->{'deadline'}       = formvalue('deadline');
270
    $vars->{'estimated_time'} = formvalue('estimated_time');
271
    $vars->{'see_also'}       = formvalue('see_also');
272

273
    $vars->{'cc'}             = join(', ', $cgi->param('cc'));
274 275

    $vars->{'comment'}        = formvalue('comment');
276
    $vars->{'comment_is_private'} = formvalue('comment_is_private');
277 278 279 280 281 282 283 284 285 286 287 288 289 290 291

} # 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).
#
292 293
# Eventually maybe each product should have a "current version"
# parameter.
294
$vars->{'version'} = $product->versions;
295

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

298
if ( ($cloned_bug_id) &&
299
     ($product->name eq $cloned_bug->product ) ) {
300
    $default{'version'} = $cloned_bug->version;
301
} elsif (formvalue('version')) {
302
    $default{'version'} = formvalue('version');
303
} elsif (defined $version_cookie
304
         and grep { $_->name eq $version_cookie } @{ $vars->{'version'} })
305 306
{
    $default{'version'} = $version_cookie;
307
} else {
308
    $default{'version'} = $vars->{'version'}->[$#{$vars->{'version'}}]->name;
309
}
310

311
# Get list of milestones.
312
if ( Bugzilla->params->{'usetargetmilestone'} ) {
313
    $vars->{'target_milestone'} = $product->milestones;
314 315 316
    if (formvalue('target_milestone')) {
       $default{'target_milestone'} = formvalue('target_milestone');
    } else {
317
       $default{'target_milestone'} = $product->default_milestone;
318 319 320
    }
}

321
# Construct the list of allowable statuses.
322
my @statuses = @{ Bugzilla::Bug->new_bug_statuses($product) };
323 324
# Exclude closed states from the UI, even if the workflow allows them.
# The back-end code will still accept them, though.
325 326
# XXX We should remove this when the UI accepts closed statuses and update
# Bugzilla::Bug->default_bug_status.
327
@statuses = grep { $_->is_open } @statuses;
328

329
scalar(@statuses) || ThrowUserError('no_initial_bug_status');
330

331
$vars->{'bug_status'} = \@statuses;
332

333
# Get the default from a template value if it is legitimate.
334 335
# Otherwise, and only if the user has privs, set the default
# to the first confirmed bug status on the list, if available.
336

337 338
my $picked_status = formvalue('bug_status');
if ($picked_status and grep($_->name eq $picked_status, @statuses)) {
339
    $default{'bug_status'} = formvalue('bug_status');
340 341
} else {
    $default{'bug_status'} = Bugzilla::Bug->default_bug_status(@statuses);
342 343
}

344 345 346 347 348 349
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);
350
}
351
$default{'groups'} = \@groups;
352

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

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

357 358 359
my $format = $template->get_format("bug/create/create",
                                   scalar $cgi->param('format'), 
                                   scalar $cgi->param('ctype'));
360

361
print $cgi->header($format->{'ctype'});
362
$template->process($format->{'template'}, $vars)
363
  || ThrowTemplateError($template->error());