Classification.pm 7.17 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::Classification;

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

14
use Bugzilla::Constants;
15
use Bugzilla::Field;
16
use Bugzilla::Util;
17
use Bugzilla::Error;
18
use Bugzilla::Product;
19

20
use base qw(Bugzilla::Field::ChoiceInterface Bugzilla::Object Exporter);
21
@Bugzilla::Classification::EXPORT = qw(sort_products_by_classification);
22

23 24 25 26
###############################
####    Initialization     ####
###############################

27 28
use constant IS_CONFIG => 1;

29
use constant DB_TABLE   => 'classifications';
30
use constant LIST_ORDER => 'sortkey, name';
31

32
use constant DB_COLUMNS => qw(
33 34 35 36
  id
  name
  description
  sortkey
37 38 39
);

use constant UPDATE_COLUMNS => qw(
40 41 42
  name
  description
  sortkey
43 44
);

45
use constant VALIDATORS => {
46 47 48
  name        => \&_check_name,
  description => \&_check_description,
  sortkey     => \&_check_sortkey,
49
};
50

51 52 53
###############################
####     Constructors     #####
###############################
54

55
sub remove_from_db {
56 57
  my $self = shift;
  my $dbh  = Bugzilla->dbh;
58

59
  ThrowUserError("classification_not_deletable") if ($self->id == 1);
60

61
  $dbh->bz_start_transaction();
62

63 64 65 66 67 68 69 70 71 72 73
  # Reclassify products to the default classification, if needed.
  my $product_ids
    = $dbh->selectcol_arrayref(
    'SELECT id FROM products WHERE classification_id = ?',
    undef, $self->id);

  if (@$product_ids) {
    $dbh->do('UPDATE products SET classification_id = 1 WHERE '
        . $dbh->sql_in('id', $product_ids));
    foreach my $id (@$product_ids) {
      Bugzilla->memcached->clear({table => 'products', id => $id});
74
    }
75 76
    Bugzilla->memcached->clear_config();
  }
77

78
  $self->SUPER::remove_from_db();
79

80
  $dbh->bz_commit_transaction();
81 82 83

}

84
###############################
85
####      Validators       ####
86 87
###############################

88
sub _check_name {
89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104
  my ($invocant, $name) = @_;

  $name = trim($name);
  $name || ThrowUserError('classification_not_specified');

  if (length($name) > MAX_CLASSIFICATION_SIZE) {
    ThrowUserError('classification_name_too_long', {'name' => $name});
  }

  my $classification = new Bugzilla::Classification({name => $name});
  if ($classification && (!ref $invocant || $classification->id != $invocant->id))
  {
    ThrowUserError("classification_already_exists",
      {name => $classification->name});
  }
  return $name;
105
}
106

107
sub _check_description {
108
  my ($invocant, $description) = @_;
109

110 111
  $description = trim($description || '');
  return $description;
112
}
113

114
sub _check_sortkey {
115 116 117 118 119 120 121 122 123
  my ($invocant, $sortkey) = @_;

  $sortkey ||= 0;
  my $stored_sortkey = $sortkey;
  if (!detaint_natural($sortkey) || $sortkey > MAX_SMALLINT) {
    ThrowUserError('classification_invalid_sortkey',
      {'sortkey' => $stored_sortkey});
  }
  return $sortkey;
124 125
}

126 127 128 129 130 131
#####################################
# Implement Bugzilla::Field::Choice #
#####################################

use constant FIELD_NAME => 'classification';
use constant is_default => 0;
132
use constant is_active  => 1;
133

134 135 136 137
###############################
####       Methods         ####
###############################

138
sub set_name        { $_[0]->set('name',        $_[1]); }
139
sub set_description { $_[0]->set('description', $_[1]); }
140
sub set_sortkey     { $_[0]->set('sortkey',     $_[1]); }
141

142
sub product_count {
143 144
  my $self = shift;
  my $dbh  = Bugzilla->dbh;
145

146 147 148
  if (!defined $self->{'product_count'}) {
    $self->{'product_count'} = $dbh->selectrow_array(
      q{
149
            SELECT COUNT(*) FROM products
150 151 152 153
            WHERE classification_id = ?}, undef, $self->id
    ) || 0;
  }
  return $self->{'product_count'};
154 155
}

156
sub products {
157 158
  my $self = shift;
  my $dbh  = Bugzilla->dbh;
159

160 161 162
  if (!$self->{'products'}) {
    my $product_ids = $dbh->selectcol_arrayref(
      q{
163
            SELECT id FROM products
164
            WHERE classification_id = ?
165 166
            ORDER BY name}, undef, $self->id
    );
167

168 169 170
    $self->{'products'} = Bugzilla::Product->new_from_list($product_ids);
  }
  return $self->{'products'};
171 172
}

173 174 175 176 177
###############################
####      Accessors        ####
###############################

sub description { return $_[0]->{'description'}; }
178
sub sortkey     { return $_[0]->{'sortkey'}; }
179

180 181 182 183 184 185 186 187 188

###############################
####       Helpers         ####
###############################

# This function is a helper to sort products to be listed
# in global/choose-product.html.tmpl.

sub sort_products_by_classification {
189 190 191 192 193 194 195 196 197 198 199 200 201 202
  my $products = shift;
  my $list;

  if (Bugzilla->params->{'useclassification'}) {
    my $class = {};

    # Get all classifications with at least one product.
    foreach my $product (@$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);
203
    }
204 205 206 207 208 209 210 211 212 213 214
    $list = [
      sort {
        $a->{'object'}->sortkey <=> $b->{'object'}->sortkey
          || lc($a->{'object'}->name) cmp lc($b->{'object'}->name)
      } (values %$class)
    ];
  }
  else {
    $list = [{object => undef, products => $products}];
  }
  return $list;
215 216
}

217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234
1;

__END__

=head1 NAME

Bugzilla::Classification - Bugzilla classification class.

=head1 SYNOPSIS

    use Bugzilla::Classification;

    my $classification = new Bugzilla::Classification(1);
    my $classification = new Bugzilla::Classification({name => 'Acme'});

    my $id = $classification->id;
    my $name = $classification->name;
    my $description = $classification->description;
235
    my $sortkey = $classification->sortkey;
236
    my $product_count = $classification->product_count;
237
    my $products = $classification->products;
238 239 240

=head1 DESCRIPTION

241 242 243 244 245 246
Classification.pm represents a classification 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::Classification> are listed
below.
247

248
A Classification is a higher-level grouping of Products.
249 250 251 252 253 254 255 256 257 258 259 260 261 262

=head1 METHODS

=over

=item C<product_count()>

 Description: Returns the total number of products that belong to
              the classification.

 Params:      none.

 Returns:     Integer - The total of products inside the classification.

263 264 265 266 267 268 269 270
=item C<products>

 Description: Returns all products of the classification.

 Params:      none.

 Returns:     A reference to an array of Bugzilla::Product objects.

271 272
=back

273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289
=head1 SUBROUTINES

=over

=item C<sort_products_by_classification>

 Description: This is a helper which returns a list of products sorted
              by classification in a form suitable to be passed to the
              global/choose-product.html.tmpl template.

 Params:      An arrayref of product objects.

 Returns:     An arrayref of hashes suitable to be passed to
              global/choose-product.html.tmpl.

=back

290
=cut
291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308

=head1 B<Methods in need of POD>

=over

=item set_description

=item sortkey

=item set_name

=item description

=item remove_from_db

=item set_sortkey

=back