Status.pm 8.51 KB
Newer Older
1 2 3
# 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/.
4
#
5 6
# This Source Code Form is "Incompatible With Secondary Licenses", as
# defined by the Mozilla Public License, v. 2.0.
7 8 9

package Bugzilla::Status;

10 11
use 5.10.1;
use strict;
12
use warnings;
13

14
# This subclasses Bugzilla::Field::Choice instead of implementing
15 16 17
# ChoiceInterface, because a bug status literally is a special type
# of Field::Choice, not just an object that happens to have the same
# methods.
18
use base qw(Bugzilla::Field::Choice Exporter);
19
@Bugzilla::Status::EXPORT = qw(
20 21
  BUG_STATE_OPEN
  SPECIAL_STATUS_WORKFLOW_ACTIONS
22

23 24
  is_open_state
  closed_bug_statuses
25
);
26

27 28
use Bugzilla::Error;

29 30 31 32
################################
#####   Initialization     #####
################################

33
use constant SPECIAL_STATUS_WORKFLOW_ACTIONS => qw(
34 35 36 37
  none
  duplicate
  change_resolution
  clearresolution
38 39
);

40 41
use constant DB_TABLE => 'bug_status';

42 43
# This has all the standard Bugzilla::Field::Choice columns plus "is_open"
sub DB_COLUMNS {
44
  return ($_[0]->SUPER::DB_COLUMNS, 'is_open');
45
}
46

47
sub VALIDATORS {
48 49 50 51 52
  my $invocant   = shift;
  my $validators = $invocant->SUPER::VALIDATORS;
  $validators->{is_open} = \&Bugzilla::Object::check_boolean;
  $validators->{value}   = \&_check_value;
  return $validators;
53 54 55 56 57 58 59
}

#########################
# Database Manipulation #
#########################

sub create {
60 61 62 63 64
  my $class = shift;
  my $self  = $class->SUPER::create(@_);
  delete Bugzilla->request_cache->{status_bug_state_open};
  add_missing_bug_status_transitions();
  return $self;
65
}
66

67
sub remove_from_db {
68 69 70
  my $self = shift;
  $self->SUPER::remove_from_db();
  delete Bugzilla->request_cache->{status_bug_state_open};
71 72
}

73 74 75 76 77
###############################
#####     Accessors        ####
###############################

sub is_active { return $_[0]->{'isactive'}; }
78
sub is_open   { return $_[0]->{'is_open'}; }
79

80
sub is_static {
81 82 83 84 85 86 87
  my $self = shift;
  if ( $self->name eq 'UNCONFIRMED'
    || $self->name eq Bugzilla->params->{'duplicate_or_move_bug_status'})
  {
    return 1;
  }
  return 0;
88 89
}

90 91 92 93 94
##############
# Validators #
##############

sub _check_value {
95 96 97 98 99 100 101 102
  my $invocant = shift;
  my $value    = $invocant->SUPER::_check_value(@_);

  if (grep { lc($value) eq lc($_) } SPECIAL_STATUS_WORKFLOW_ACTIONS) {
    ThrowUserError('fieldvalue_reserved_word',
      {field => $invocant->field, value => $value});
  }
  return $value;
103 104 105
}


106 107 108 109
###############################
#####       Methods        ####
###############################

110
sub BUG_STATE_OPEN {
111 112 113 114 115 116 117 118 119 120 121 122 123 124
  my $dbh           = Bugzilla->dbh;
  my $request_cache = Bugzilla->request_cache;
  my $cache_key     = 'status_bug_state_open';
  return @{$request_cache->{$cache_key}} if exists $request_cache->{$cache_key};

  my $rows = Bugzilla->memcached->get_config({key => $cache_key});
  if (!$rows) {
    $rows
      = $dbh->selectcol_arrayref('SELECT value FROM bug_status WHERE is_open = 1');
    Bugzilla->memcached->set_config({key => $cache_key, data => $rows});
  }

  $request_cache->{$cache_key} = $rows;
  return @$rows;
125 126 127 128
}

# Tells you whether or not the argument is a valid "open" state.
sub is_open_state {
129 130
  my ($state) = @_;
  return (grep($_ eq $state, BUG_STATE_OPEN) ? 1 : 0);
131 132
}

133
sub closed_bug_statuses {
134 135 136
  my @bug_statuses = Bugzilla::Status->get_all;
  @bug_statuses = grep { !$_->is_open } @bug_statuses;
  return @bug_statuses;
137 138
}

139
sub can_change_to {
140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158
  my $self = shift;
  my $dbh  = Bugzilla->dbh;

  if (!ref($self) || !defined $self->{'can_change_to'}) {
    my ($cond, @args, $self_exists);
    if (ref($self)) {
      $cond = '= ?';
      push(@args, $self->id);
      $self_exists = 1;
    }
    else {
      $cond = 'IS NULL';

      # Let's do it so that the code below works in all cases.
      $self = {};
    }

    my $new_status_ids = $dbh->selectcol_arrayref(
      "SELECT new_status
159 160 161 162
                                                         FROM status_workflow
                                                   INNER JOIN bug_status
                                                           ON id = new_status
                                                        WHERE isactive = 1
163
                                                          AND old_status $cond
164 165
                                                     ORDER BY sortkey", undef, @args
    );
166

167 168 169 170
    # Allow the bug status to remain unchanged.
    push(@$new_status_ids, $self->id) if $self_exists;
    $self->{'can_change_to'} = Bugzilla::Status->new_from_list($new_status_ids);
  }
171

172
  return $self->{'can_change_to'};
173 174
}

175
sub comment_required_on_change_from {
176 177 178 179 180 181 182 183
  my ($self, $old_status) = @_;
  my ($cond, $values)     = $self->_status_condition($old_status);

  my ($require_comment) = Bugzilla->dbh->selectrow_array(
    "SELECT require_comment FROM status_workflow
          WHERE $cond", undef, @$values
  );
  return $require_comment;
184 185 186 187 188
}

# Used as a helper for various functions that have to deal with old_status
# sometimes being NULL and sometimes having a value.
sub _status_condition {
189 190 191 192 193 194 195 196 197 198 199 200
  my ($self, $old_status) = @_;
  my @values;
  my $cond = 'old_status IS NULL';

  # We may pass a fake status object to represent the initial unset state.
  if ($old_status && $old_status->id) {
    $cond = 'old_status = ?';
    push(@values, $old_status->id);
  }
  $cond .= " AND new_status = ?";
  push(@values, $self->id);
  return ($cond, \@values);
201 202
}

203
sub add_missing_bug_status_transitions {
204 205 206 207 208 209
  my $bug_status = shift || Bugzilla->params->{'duplicate_or_move_bug_status'};
  my $dbh        = Bugzilla->dbh;
  my $new_status = new Bugzilla::Status({name => $bug_status});

  # Silently discard invalid bug statuses.
  $new_status || return;
210

211 212
  my $missing_statuses = $dbh->selectcol_arrayref(
    'SELECT id
213 214 215 216 217
                                                       FROM bug_status
                                                  LEFT JOIN status_workflow
                                                         ON old_status = id
                                                        AND new_status = ?
                                                      WHERE old_status IS NULL',
218 219 220 221 222 223 224 225 226 227 228 229
    undef, $new_status->id
  );

  my $sth = $dbh->prepare(
    'INSERT INTO status_workflow
                             (old_status, new_status) VALUES (?, ?)'
  );

  foreach my $old_status_id (@$missing_statuses) {
    next if ($old_status_id == $new_status->id);
    $sth->execute($old_status_id, $new_status->id);
  }
230
}
231 232 233 234 235 236 237 238 239 240 241 242 243

1;

__END__

=head1 NAME

Bugzilla::Status - Bug status class.

=head1 SYNOPSIS

    use Bugzilla::Status;

244
    my $bug_status = new Bugzilla::Status({ name => 'IN_PROGRESS' });
245 246
    my $bug_status = new Bugzilla::Status(4);

247
    my @closed_bug_statuses = closed_bug_statuses();
248 249 250

    Bugzilla::Status::add_missing_bug_status_transitions($bug_status);

251 252 253 254 255 256 257 258 259 260 261 262 263
=head1 DESCRIPTION

Status.pm represents a bug status object. It is an implementation
of L<Bugzilla::Object>, and thus provides all methods that
L<Bugzilla::Object> provides.

The methods that are specific to C<Bugzilla::Status> are listed
below.

=head1 METHODS

=over

264 265 266 267 268 269 270 271 272
=item C<closed_bug_statuses>

 Description: Returns a list of C<Bugzilla::Status> objects which can have
              a resolution associated with them ("closed" bug statuses).

 Params:      none.

 Returns:     A list of Bugzilla::Status objects.

273 274 275
=item C<can_change_to>

 Description: Returns the list of active statuses a bug can be changed to
276 277 278 279 280 281 282 283
              given the current bug status. If this method is called as a
              class method, then it returns all bug statuses available on
              bug creation.

 Params:      none.

 Returns:     A list of Bugzilla::Status objects.

284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306
=item C<comment_required_on_change_from>

=over

=item B<Description>

Checks if a comment is required to change to this status from another
status, according to the current settings in the workflow.

Note that this doesn't implement the checks enforced by the various
C<commenton> parameters--those are checked by internal checks in
L<Bugzilla::Bug>.

=item B<Params>

C<$old_status> - The status you're changing from.

=item B<Returns>

C<1> if a comment is required on this change, C<0> if not.

=back

307 308 309 310 311 312 313 314
=item C<add_missing_bug_status_transitions>

 Description: Insert all missing transitions to a given bug status.

 Params:      $bug_status - The value (name) of a bug status.

 Returns:     nothing.

315 316 317
=back

=cut
318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341

=head1 B<Methods in need of POD>

=over

=item create

=item BUG_STATE_OPEN

=item is_static

=item is_open_state

=item is_active

=item remove_from_db

=item DB_COLUMNS

=item is_open

=item VALIDATORS

=back