| #!/usr/bin/env perl -wT |
| # -*- Mode: perl; indent-tabs-mode: nil; cperl-indent-level: 4 -*- |
| # |
| # 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. |
| # |
| # The Original Code is the Bugzilla Bug Tracking System. |
| # |
| # The Initial Developer of the Original Code is Albert Ting |
| # |
| # Contributor(s): Albert Ting <alt@sonic.net> |
| # Max Kanat-Alexander <mkanat@bugzilla.org> |
| # Frédéric Buclin <LpSolit@gmail.com> |
| |
| |
| use strict; |
| use lib qw(. lib); |
| |
| use Bugzilla; |
| use Bugzilla::Constants; |
| use Bugzilla::Util; |
| use Bugzilla::Error; |
| use Bugzilla::Classification; |
| use Bugzilla::Token; |
| |
| my $dbh = Bugzilla->dbh; |
| my $cgi = Bugzilla->cgi; |
| my $template = Bugzilla->template; |
| local our $vars = {}; |
| |
| sub LoadTemplate { |
| my $action = shift; |
| my $cgi = Bugzilla->cgi; |
| my $template = Bugzilla->template; |
| |
| $vars->{'classifications'} = [Bugzilla::Classification::get_all_classifications()] |
| if ($action eq 'select'); |
| # There is currently only one section about classifications, |
| # so all pages point to it. Let's define it here. |
| $vars->{'doc_section'} = 'classifications.html'; |
| |
| $action =~ /(\w+)/; |
| $action = $1; |
| print $cgi->header(); |
| $template->process("admin/classifications/$action.html.tmpl", $vars) |
| || ThrowTemplateError($template->error()); |
| exit; |
| } |
| |
| # |
| # Preliminary checks: |
| # |
| |
| Bugzilla->login(LOGIN_REQUIRED); |
| |
| print $cgi->header(); |
| |
| exists Bugzilla->user->groups->{'editclassifications'} |
| || ThrowUserError("auth_failure", {group => "editclassifications", |
| action => "edit", |
| object => "classifications"}); |
| |
| ThrowUserError("auth_classification_not_enabled") |
| unless Bugzilla->params->{"useclassification"}; |
| |
| # |
| # often used variables |
| # |
| my $action = trim($cgi->param('action') || ''); |
| my $class_name = trim($cgi->param('classification') || ''); |
| my $token = $cgi->param('token'); |
| |
| # |
| # action='' -> Show nice list of classifications |
| # |
| LoadTemplate('select') unless $action; |
| |
| # |
| # action='add' -> present form for parameters for new classification |
| # |
| # (next action will be 'new') |
| # |
| |
| if ($action eq 'add') { |
| $vars->{'token'} = issue_session_token('add_classification'); |
| LoadTemplate($action); |
| } |
| |
| # |
| # action='new' -> add classification entered in the 'action=add' screen |
| # |
| |
| if ($action eq 'new') { |
| check_token_data($token, 'add_classification'); |
| |
| $class_name || ThrowUserError("classification_not_specified"); |
| |
| my $classification = |
| new Bugzilla::Classification({name => $class_name}); |
| |
| if ($classification) { |
| ThrowUserError("classification_already_exists", |
| { name => $classification->name }); |
| } |
| |
| my $description = trim($cgi->param('description') || ''); |
| |
| my $sortkey = trim($cgi->param('sortkey') || 0); |
| my $stored_sortkey = $sortkey; |
| detaint_natural($sortkey) |
| || ThrowUserError('classification_invalid_sortkey', {'name' => $class_name, |
| 'sortkey' => $stored_sortkey}); |
| |
| trick_taint($description); |
| trick_taint($class_name); |
| |
| # Add the new classification. |
| $dbh->do("INSERT INTO classifications (name, description, sortkey) |
| VALUES (?, ?, ?)", undef, ($class_name, $description, $sortkey)); |
| |
| delete_token($token); |
| |
| $vars->{'message'} = 'classification_created'; |
| $vars->{'classification'} = new Bugzilla::Classification({name => $class_name}); |
| $vars->{'classifications'} = [Bugzilla::Classification::get_all_classifications]; |
| $vars->{'token'} = issue_session_token('reclassify_classifications'); |
| LoadTemplate('reclassify'); |
| } |
| |
| # |
| # action='del' -> ask if user really wants to delete |
| # |
| # (next action would be 'delete') |
| # |
| |
| if ($action eq 'del') { |
| |
| my $classification = |
| Bugzilla::Classification::check_classification($class_name); |
| |
| if ($classification->id == 1) { |
| ThrowUserError("classification_not_deletable"); |
| } |
| |
| if ($classification->product_count()) { |
| ThrowUserError("classification_has_products"); |
| } |
| |
| $vars->{'classification'} = $classification; |
| $vars->{'token'} = issue_session_token('delete_classification'); |
| |
| LoadTemplate($action); |
| } |
| |
| # |
| # action='delete' -> really delete the classification |
| # |
| |
| if ($action eq 'delete') { |
| check_token_data($token, 'delete_classification'); |
| |
| my $classification = |
| Bugzilla::Classification::check_classification($class_name); |
| |
| if ($classification->id == 1) { |
| ThrowUserError("classification_not_deletable"); |
| } |
| |
| # lock the tables before we start to change everything: |
| $dbh->bz_start_transaction(); |
| |
| # update products just in case |
| $dbh->do("UPDATE products SET classification_id = 1 |
| WHERE classification_id = ?", undef, $classification->id); |
| |
| # delete |
| $dbh->do("DELETE FROM classifications WHERE id = ?", undef, |
| $classification->id); |
| |
| $dbh->bz_commit_transaction(); |
| |
| $vars->{'message'} = 'classification_deleted'; |
| $vars->{'classification'} = $class_name; |
| delete_token($token); |
| LoadTemplate('select'); |
| } |
| |
| # |
| # action='edit' -> present the edit classifications from |
| # |
| # (next action would be 'update') |
| # |
| |
| if ($action eq 'edit') { |
| |
| my $classification = |
| Bugzilla::Classification::check_classification($class_name); |
| |
| $vars->{'classification'} = $classification; |
| $vars->{'token'} = issue_session_token('edit_classification'); |
| |
| LoadTemplate($action); |
| } |
| |
| # |
| # action='update' -> update the classification |
| # |
| |
| if ($action eq 'update') { |
| check_token_data($token, 'edit_classification'); |
| |
| $class_name || ThrowUserError("classification_not_specified"); |
| |
| my $class_old_name = trim($cgi->param('classificationold') || ''); |
| |
| my $class_old = |
| Bugzilla::Classification::check_classification($class_old_name); |
| |
| my $description = trim($cgi->param('description') || ''); |
| |
| my $sortkey = trim($cgi->param('sortkey') || 0); |
| my $stored_sortkey = $sortkey; |
| detaint_natural($sortkey) |
| || ThrowUserError('classification_invalid_sortkey', {'name' => $class_old->name, |
| 'sortkey' => $stored_sortkey}); |
| |
| $dbh->bz_start_transaction(); |
| |
| if ($class_name ne $class_old->name) { |
| |
| my $class = new Bugzilla::Classification({name => $class_name}); |
| if ($class) { |
| ThrowUserError("classification_already_exists", |
| { name => $class->name }); |
| } |
| trick_taint($class_name); |
| $dbh->do("UPDATE classifications SET name = ? WHERE id = ?", |
| undef, ($class_name, $class_old->id)); |
| |
| $vars->{'updated_classification'} = 1; |
| } |
| |
| if ($description ne $class_old->description) { |
| trick_taint($description); |
| $dbh->do("UPDATE classifications SET description = ? |
| WHERE id = ?", undef, |
| ($description, $class_old->id)); |
| |
| $vars->{'updated_description'} = 1; |
| } |
| |
| if ($sortkey ne $class_old->sortkey) { |
| $dbh->do("UPDATE classifications SET sortkey = ? |
| WHERE id = ?", undef, |
| ($sortkey, $class_old->id)); |
| |
| $vars->{'updated_sortkey'} = 1; |
| } |
| |
| $dbh->bz_commit_transaction(); |
| |
| $vars->{'message'} = 'classification_updated'; |
| $vars->{'classification'} = $class_name; |
| delete_token($token); |
| LoadTemplate('select'); |
| } |
| |
| # |
| # action='reclassify' -> reclassify products for the classification |
| # |
| |
| if ($action eq 'reclassify') { |
| |
| my $classification = |
| Bugzilla::Classification::check_classification($class_name); |
| |
| my $sth = $dbh->prepare("UPDATE products SET classification_id = ? |
| WHERE name = ?"); |
| |
| if (defined $cgi->param('add_products')) { |
| check_token_data($token, 'reclassify_classifications'); |
| if (defined $cgi->param('prodlist')) { |
| foreach my $prod ($cgi->param("prodlist")) { |
| trick_taint($prod); |
| $sth->execute($classification->id, $prod); |
| } |
| } |
| delete_token($token); |
| } elsif (defined $cgi->param('remove_products')) { |
| check_token_data($token, 'reclassify_classifications'); |
| if (defined $cgi->param('myprodlist')) { |
| foreach my $prod ($cgi->param("myprodlist")) { |
| trick_taint($prod); |
| $sth->execute(1,$prod); |
| } |
| } |
| delete_token($token); |
| } |
| |
| my @classifications = |
| Bugzilla::Classification::get_all_classifications; |
| $vars->{'classifications'} = \@classifications; |
| $vars->{'classification'} = $classification; |
| $vars->{'token'} = issue_session_token('reclassify_classifications'); |
| |
| LoadTemplate($action); |
| } |
| |
| # |
| # No valid action found |
| # |
| |
| ThrowCodeError("action_unrecognized", {action => $action}); |