You need to sign in or sign up before continuing.
enter_bug.cgi 12.2 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
#
# enter_bug.cgi
# -------------
13 14
# Displays bug entry form. Bug fields are specified through popup menus,
# drop-down lists, or text fields. Default for these values can be
15
# 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 45
my $cgi      = Bugzilla->cgi;
my $dbh      = Bugzilla->dbh;
46
my $template = Bugzilla->template;
47
my $vars     = {};
48

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

52
my $product_name = trim($cgi->param('product') || '');
53

54 55
# Will contain the product object the bug is created in.
my $product;
56

57
if ($product_name eq '') {
58

59 60 61
  # 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);
62

63 64 65 66
  my $classification
    = Bugzilla->params->{'useclassification'}
    ? scalar($cgi->param('classification'))
    : '__all';
67

68 69 70
  # Unless a real classification name is given, we sort products
  # by classification.
  my @classifications;
71

72 73 74 75 76 77 78 79 80 81 82 83 84 85
  unless ($classification && $classification ne '__all') {
    @classifications = @{sort_products_by_classification(\@enterable_products)};
  }

  unless ($classification) {

    # We know there is at least one classification available,
    # else we would have stopped earlier.
    if (scalar(@classifications) > 1) {

      # We only need classification objects.
      $vars->{'classifications'} = [map { $_->{'object'} } @classifications];

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

87 88 89 90
      print $cgi->header();
      $template->process("global/choose-classification.html.tmpl", $vars)
        || ThrowTemplateError($template->error());
      exit;
91
    }
92

93 94 95 96 97 98 99 100 101 102 103 104 105
    # If we come here, then there is only one classification available.
    $classification = $classifications[0]->{'object'}->name;
  }

  # 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;
      @classifications = ({object => $class, products => \@enterable_products});
106
    }
107 108
    else {
      @enterable_products = ();
109
    }
110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127
  }

  if (scalar(@enterable_products) == 0) {
    ThrowUserError('no_products');
  }
  elsif (scalar(@enterable_products) > 1) {
    $vars->{'classifications'} = \@classifications;
    $vars->{'target'}          = "enter_bug.cgi";

    print $cgi->header();
    $template->process("global/choose-product.html.tmpl", $vars)
      || ThrowTemplateError($template->error());
    exit;
  }
  else {
    # Only one product exists.
    $product = $enterable_products[0];
  }
128
}
129 130 131

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

134 135 136
##############################################################################
# Useful Subroutines
##############################################################################
137
sub formvalue {
138 139
  my ($name, $default) = (@_);
  return Bugzilla->cgi->param($name) || $default || "";
140
}
terry%netscape.com's avatar
terry%netscape.com committed
141

142 143 144
##############################################################################
# End of subroutines
##############################################################################
terry%netscape.com's avatar
terry%netscape.com committed
145

146
my $has_editbugs   = $user->in_group('editbugs',   $product->id);
147 148
my $has_canconfirm = $user->in_group('canconfirm', $product->id);

149 150 151 152 153 154
# 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) {
155 156
  $cloned_bug    = Bugzilla::Bug->check($cloned_bug_id);
  $cloned_bug_id = $cloned_bug->id;
157 158
}

159 160 161
# If there is only one active component, choose it
my @active = grep { $_->is_active } @{$product->components};
if (scalar(@active) == 1) {
162
  $cgi->param('component', $active[0]->name);
163 164 165 166 167
}

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

171
my %default;
terry%netscape.com's avatar
terry%netscape.com committed
172

173
$vars->{'product'} = $product;
174

175 176 177 178
$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');
179

180 181 182
$vars->{'assigned_to'}          = formvalue('assigned_to');
$vars->{'assigned_to_disabled'} = !$has_editbugs;
$vars->{'cc_disabled'}          = 0;
183

184 185
$vars->{'qa_contact'}          = formvalue('qa_contact');
$vars->{'qa_contact_disabled'} = !$has_editbugs;
186

187
$vars->{'cloned_bug_id'} = $cloned_bug_id;
188

189
$vars->{'token'} = issue_session_token('create_bug');
190

191

192
my @enter_bug_fields = grep { $_->enter_bug } Bugzilla->active_custom_fields;
193
foreach my $field (@enter_bug_fields) {
194 195 196 197 198
  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)];
199
    }
200 201
    $default{$cf_name} = $vars->{$cf_name} = $cf_value;
  }
202 203
}

204
# This allows the Field visibility and value controls to work with the
205 206
# Classification and Product fields as a parent.
$default{'classification'} = $product->classification->name;
207
$default{'product'}        = $product->name;
208

209
if ($cloned_bug_id) {
210

211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237
  $default{'component_'}   = $cloned_bug->component;
  $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;

  $vars->{'short_desc'}   = $cloned_bug->short_desc;
  $vars->{'bug_file_loc'} = $cloned_bug->bug_file_loc;
  $vars->{'keywords'}     = $cloned_bug->keywords;
  $vars->{'dependson'}    = join(", ", $cloned_bug_id, @{$cloned_bug->dependson});
  $vars->{'blocked'}      = join(", ", @{$cloned_bug->blocked});
  $vars->{'deadline'}     = $cloned_bug->deadline;
  $vars->{'estimated_time'} = $cloned_bug->estimated_time;

  if (scalar @{$cloned_bug->cc}) {
    $vars->{'cc'} = join(", ", @{$cloned_bug->cc});
  }
  else {
    $vars->{'cc'} = formvalue('cc');
  }

  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'});
238
    }
239
  }
240

241 242 243 244
  foreach my $field (@enter_bug_fields) {
    my $field_name = $field->name;
    $vars->{$field_name} = $cloned_bug->$field_name;
  }
245

246 247 248
  # 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.
249
  my $bug_desc  = $cloned_bug->comments({order => 'oldest_to_newest'})->[0];
250
  my $isprivate = $bug_desc->is_private;
251

252 253
  $vars->{'comment'}            = "";
  $vars->{'comment_is_private'} = 0;
254

255
  if (!$isprivate || $user->is_insider) {
256

257 258 259 260 261
    # We use "body" to avoid any format_comment text, which would be
    # pointless to clone.
    $vars->{'comment'}            = $bug_desc->body;
    $vars->{'comment_is_private'} = $isprivate;
  }
262

263
}    # end of cloned bug entry form
264

265
else {
266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291
  $default{'component_'} = formvalue('component');
  $default{'priority'}
    = formvalue('priority', Bugzilla->params->{'defaultpriority'});
  $default{'bug_severity'}
    = formvalue('bug_severity', Bugzilla->params->{'defaultseverity'});
  $default{'rep_platform'} = formvalue('rep_platform',
    Bugzilla->params->{'defaultplatform'} || detect_platform());
  $default{'op_sys'}
    = formvalue('op_sys', Bugzilla->params->{'defaultopsys'} || detect_op_sys());

  $vars->{'alias'}          = formvalue('alias');
  $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');
  $vars->{'deadline'}       = formvalue('deadline');
  $vars->{'estimated_time'} = formvalue('estimated_time');
  $vars->{'see_also'}       = formvalue('see_also');

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

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

}    # end of normal/bookmarked entry form
292 293 294 295 296 297 298 299 300 301 302 303 304


# 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).
#
305 306
# Eventually maybe each product should have a "current version"
# parameter.
307
$vars->{'version'} = $product->versions;
308

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

311 312 313 314 315 316 317 318
if (($cloned_bug_id) && ($product->name eq $cloned_bug->product)) {
  $default{'version'} = $cloned_bug->version;
}
elsif (formvalue('version')) {
  $default{'version'} = formvalue('version');
}
elsif (defined $version_cookie
  and grep { $_->name eq $version_cookie } @{$vars->{'version'}})
319
{
320 321 322 323
  $default{'version'} = $version_cookie;
}
else {
  $default{'version'} = $vars->{'version'}->[$#{$vars->{'version'}}]->name;
324
}
325

326
# Get list of milestones.
327 328 329 330 331 332 333 334
if (Bugzilla->params->{'usetargetmilestone'}) {
  $vars->{'target_milestone'} = $product->milestones;
  if (formvalue('target_milestone')) {
    $default{'target_milestone'} = formvalue('target_milestone');
  }
  else {
    $default{'target_milestone'} = $product->default_milestone;
  }
335 336
}

337
# Construct the list of allowable statuses.
338 339
my @statuses = @{Bugzilla::Bug->new_bug_statuses($product)};

340 341
# Exclude closed states from the UI, even if the workflow allows them.
# The back-end code will still accept them, though.
342 343
# XXX We should remove this when the UI accepts closed statuses and update
# Bugzilla::Bug->default_bug_status.
344
@statuses = grep { $_->is_open } @statuses;
345

346
scalar(@statuses) || ThrowUserError('no_initial_bug_status');
347

348
$vars->{'bug_status'} = \@statuses;
349

350
# Get the default from a template value if it is legitimate.
351 352
# Otherwise, and only if the user has privs, set the default
# to the first confirmed bug status on the list, if available.
353

354 355
my $picked_status = formvalue('bug_status');
if ($picked_status and grep($_->name eq $picked_status, @statuses)) {
356 357 358 359
  $default{'bug_status'} = formvalue('bug_status');
}
else {
  $default{'bug_status'} = Bugzilla::Bug->default_bug_status(@statuses);
360 361
}

362 363
my @groups = $cgi->param('groups');
if ($cloned_bug) {
364 365 366 367 368
  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);
369
}
370
$default{'groups'} = \@groups;
371

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

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

376 377 378 379 380
my $format = $template->get_format(
  "bug/create/create",
  scalar $cgi->param('format'),
  scalar $cgi->param('ctype')
);
381

382
print $cgi->header($format->{'ctype'});
383
$template->process($format->{'template'}, $vars)
384
  || ThrowTemplateError($template->error());