2003-03-27 03:07:02 +03:00
|
|
|
#!/usr/bin/perl -wT
|
2002-09-28 22:42:54 +04:00
|
|
|
# -*- Mode: perl; indent-tabs-mode: nil -*-
|
|
|
|
#
|
|
|
|
# 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 Netscape Communications
|
|
|
|
# Corporation. Portions created by Netscape are
|
|
|
|
# Copyright (C) 1998 Netscape Communications Corporation. All
|
|
|
|
# Rights Reserved.
|
|
|
|
#
|
|
|
|
# Contributor(s): Myk Melez <myk@mozilla.org>
|
|
|
|
|
|
|
|
################################################################################
|
|
|
|
# Script Initialization
|
|
|
|
################################################################################
|
|
|
|
|
|
|
|
# Make it harder for us to do dangerous things in Perl.
|
|
|
|
use strict;
|
|
|
|
use lib ".";
|
|
|
|
|
|
|
|
# Include the Bugzilla CGI and general utility library.
|
2005-08-10 05:30:41 +04:00
|
|
|
require "globals.pl";
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Use Bugzilla's flag modules for handling flag types.
|
2003-05-05 05:15:38 +04:00
|
|
|
use Bugzilla;
|
2004-03-27 06:51:44 +03:00
|
|
|
use Bugzilla::Constants;
|
2002-09-28 22:42:54 +04:00
|
|
|
use Bugzilla::Flag;
|
|
|
|
use Bugzilla::FlagType;
|
2005-08-30 20:47:33 +04:00
|
|
|
use Bugzilla::Group;
|
2005-03-16 01:10:14 +03:00
|
|
|
use Bugzilla::User;
|
2005-07-21 01:24:19 +04:00
|
|
|
use Bugzilla::Util;
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
use vars qw( $template $vars );
|
|
|
|
|
|
|
|
# Make sure the user is logged in and is an administrator.
|
2004-03-27 06:51:44 +03:00
|
|
|
Bugzilla->login(LOGIN_REQUIRED);
|
2002-09-28 22:42:54 +04:00
|
|
|
UserInGroup("editcomponents")
|
2005-01-16 17:07:31 +03:00
|
|
|
|| ThrowUserError("auth_failure", {group => "editcomponents",
|
|
|
|
action => "edit",
|
|
|
|
object => "flagtypes"});
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Suppress "used only once" warnings.
|
|
|
|
use vars qw(@legal_product @legal_components %components);
|
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
my $cgi = Bugzilla->cgi;
|
2002-09-28 22:42:54 +04:00
|
|
|
my $product_id;
|
|
|
|
my $component_id;
|
|
|
|
|
|
|
|
################################################################################
|
|
|
|
# Main Body Execution
|
|
|
|
################################################################################
|
|
|
|
|
|
|
|
# All calls to this script should contain an "action" variable whose value
|
|
|
|
# determines what the user wants to do. The code below checks the value of
|
|
|
|
# that variable and runs the appropriate code.
|
|
|
|
|
|
|
|
# Determine whether to use the action specified by the user or the default.
|
2005-04-10 03:12:11 +04:00
|
|
|
my $action = $cgi->param('action') || 'list';
|
2004-09-24 05:01:30 +04:00
|
|
|
my @categoryActions;
|
2002-09-28 22:42:54 +04:00
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
if (@categoryActions = grep(/^categoryAction-.+/, $cgi->param())) {
|
2004-09-24 05:01:30 +04:00
|
|
|
$categoryActions[0] =~ s/^categoryAction-//;
|
|
|
|
processCategoryChange($categoryActions[0]);
|
2002-09-28 22:42:54 +04:00
|
|
|
exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($action eq 'list') { list(); }
|
|
|
|
elsif ($action eq 'enter') { edit(); }
|
|
|
|
elsif ($action eq 'copy') { edit(); }
|
|
|
|
elsif ($action eq 'edit') { edit(); }
|
|
|
|
elsif ($action eq 'insert') { insert(); }
|
|
|
|
elsif ($action eq 'update') { update(); }
|
|
|
|
elsif ($action eq 'confirmdelete') { confirmDelete(); }
|
2004-09-03 13:58:00 +04:00
|
|
|
elsif ($action eq 'delete') { deleteType(); }
|
2002-09-28 22:42:54 +04:00
|
|
|
elsif ($action eq 'deactivate') { deactivate(); }
|
|
|
|
else {
|
|
|
|
ThrowCodeError("action_unrecognized", { action => $action });
|
|
|
|
}
|
|
|
|
|
|
|
|
exit;
|
|
|
|
|
|
|
|
################################################################################
|
|
|
|
# Functions
|
|
|
|
################################################################################
|
|
|
|
|
|
|
|
sub list {
|
|
|
|
# Define the variables and functions that will be passed to the UI template.
|
2004-11-24 01:41:44 +03:00
|
|
|
$vars->{'bug_types'} =
|
|
|
|
Bugzilla::FlagType::match({ 'target_type' => 'bug',
|
2005-04-10 03:12:11 +04:00
|
|
|
'group' => scalar $cgi->param('group') }, 1);
|
2002-09-28 22:42:54 +04:00
|
|
|
$vars->{'attachment_types'} =
|
2004-11-24 01:41:44 +03:00
|
|
|
Bugzilla::FlagType::match({ 'target_type' => 'attachment',
|
2005-04-10 03:12:11 +04:00
|
|
|
'group' => scalar $cgi->param('group') }, 1);
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Return the appropriate HTTP response headers.
|
2005-04-10 03:12:11 +04:00
|
|
|
print $cgi->header();
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Generate and return the UI (HTML page) from the appropriate template.
|
|
|
|
$template->process("admin/flag-type/list.html.tmpl", $vars)
|
|
|
|
|| ThrowTemplateError($template->error());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub edit {
|
2005-04-10 03:12:11 +04:00
|
|
|
$action eq 'enter' ? validateTargetType() : (my $id = validateID());
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Get this installation's products and components.
|
|
|
|
GetVersionTable();
|
|
|
|
|
|
|
|
# products and components and the function used to modify the components
|
|
|
|
# menu when the products menu changes; used by the template to populate
|
|
|
|
# the menus and keep the components menu consistent with the products menu
|
|
|
|
$vars->{'products'} = \@::legal_product;
|
|
|
|
$vars->{'components'} = \@::legal_components;
|
|
|
|
$vars->{'components_by_product'} = \%::components;
|
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
$vars->{'last_action'} = $cgi->param('action');
|
|
|
|
if ($cgi->param('action') eq 'enter' || $cgi->param('action') eq 'copy') {
|
2002-09-28 22:42:54 +04:00
|
|
|
$vars->{'action'} = "insert";
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
$vars->{'action'} = "update";
|
|
|
|
}
|
|
|
|
|
|
|
|
# If copying or editing an existing flag type, retrieve it.
|
2005-04-10 03:12:11 +04:00
|
|
|
if ($cgi->param('action') eq 'copy' || $cgi->param('action') eq 'edit') {
|
|
|
|
$vars->{'type'} = Bugzilla::FlagType::get($id);
|
|
|
|
$vars->{'type'}->{'inclusions'} = Bugzilla::FlagType::get_inclusions($id);
|
|
|
|
$vars->{'type'}->{'exclusions'} = Bugzilla::FlagType::get_exclusions($id);
|
2004-11-24 01:41:44 +03:00
|
|
|
# Users want to see group names, not IDs
|
|
|
|
foreach my $group ("grant_gid", "request_gid") {
|
|
|
|
my $gid = $vars->{'type'}->{$group};
|
|
|
|
next if (!$gid);
|
|
|
|
SendSQL("SELECT name FROM groups WHERE id = $gid");
|
|
|
|
$vars->{'type'}->{$group} = FetchOneColumn();
|
|
|
|
}
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
# Otherwise set the target type (the minimal information about the type
|
2002-11-06 02:45:56 +03:00
|
|
|
# that the template needs to know) from the URL parameter and default
|
|
|
|
# the list of inclusions to all categories.
|
2005-05-05 23:20:45 +04:00
|
|
|
else {
|
|
|
|
my %inclusions;
|
|
|
|
$inclusions{"__Any__:__Any__"} = "0:0";
|
2005-04-10 03:12:11 +04:00
|
|
|
$vars->{'type'} = { 'target_type' => scalar $cgi->param('target_type'),
|
2005-05-05 23:20:45 +04:00
|
|
|
'inclusions' => \%inclusions };
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
2005-08-30 20:47:33 +04:00
|
|
|
# Get a list of groups available to restrict this flag type against.
|
|
|
|
my @groups = Bugzilla::Group::get_all_groups();
|
|
|
|
$vars->{'groups'} = \@groups;
|
2002-09-28 22:42:54 +04:00
|
|
|
# Return the appropriate HTTP response headers.
|
2005-04-10 03:12:11 +04:00
|
|
|
print $cgi->header();
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Generate and return the UI (HTML page) from the appropriate template.
|
|
|
|
$template->process("admin/flag-type/edit.html.tmpl", $vars)
|
|
|
|
|| ThrowTemplateError($template->error());
|
|
|
|
}
|
|
|
|
|
|
|
|
sub processCategoryChange {
|
2004-09-24 05:01:30 +04:00
|
|
|
my $categoryAction = shift;
|
2002-09-28 22:42:54 +04:00
|
|
|
validateIsActive();
|
|
|
|
validateIsRequestable();
|
|
|
|
validateIsRequesteeble();
|
|
|
|
validateAllowMultiple();
|
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
my @inclusions = $cgi->param('inclusions');
|
|
|
|
my @exclusions = $cgi->param('exclusions');
|
2004-09-24 05:01:30 +04:00
|
|
|
if ($categoryAction eq 'include') {
|
2002-09-28 22:42:54 +04:00
|
|
|
validateProduct();
|
|
|
|
validateComponent();
|
2005-05-05 23:20:45 +04:00
|
|
|
my $category = ($product_id || 0) . ":" . ($component_id || 0);
|
2002-09-28 22:42:54 +04:00
|
|
|
push(@inclusions, $category) unless grep($_ eq $category, @inclusions);
|
|
|
|
}
|
2004-09-24 05:01:30 +04:00
|
|
|
elsif ($categoryAction eq 'exclude') {
|
2002-09-28 22:42:54 +04:00
|
|
|
validateProduct();
|
|
|
|
validateComponent();
|
2005-05-05 23:20:45 +04:00
|
|
|
my $category = ($product_id || 0) . ":" . ($component_id || 0);
|
2002-09-28 22:42:54 +04:00
|
|
|
push(@exclusions, $category) unless grep($_ eq $category, @exclusions);
|
|
|
|
}
|
2004-09-24 05:01:30 +04:00
|
|
|
elsif ($categoryAction eq 'removeInclusion') {
|
2005-04-10 03:12:11 +04:00
|
|
|
@inclusions = map(($_ eq $cgi->param('inclusion_to_remove') ? () : $_), @inclusions);
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
2004-09-24 05:01:30 +04:00
|
|
|
elsif ($categoryAction eq 'removeExclusion') {
|
2005-04-10 03:12:11 +04:00
|
|
|
@exclusions = map(($_ eq $cgi->param('exclusion_to_remove') ? () : $_), @exclusions);
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
|
2005-05-05 23:20:45 +04:00
|
|
|
# Convert the array @clusions('prod_ID:comp_ID') back to a hash of
|
|
|
|
# the form %clusions{'prod_name:comp_name'} = 'prod_ID:comp_ID'
|
|
|
|
my %inclusions = clusion_array_to_hash(\@inclusions);
|
|
|
|
my %exclusions = clusion_array_to_hash(\@exclusions);
|
|
|
|
|
2002-09-28 22:42:54 +04:00
|
|
|
# Get this installation's products and components.
|
|
|
|
GetVersionTable();
|
|
|
|
|
|
|
|
# products and components; used by the template to populate the menus
|
|
|
|
# and keep the components menu consistent with the products menu
|
|
|
|
$vars->{'products'} = \@::legal_product;
|
|
|
|
$vars->{'components'} = \@::legal_components;
|
|
|
|
$vars->{'components_by_product'} = \%::components;
|
2005-08-30 20:47:33 +04:00
|
|
|
my @groups = Bugzilla::Group::get_all_groups();
|
|
|
|
$vars->{'groups'} = \@groups;
|
2005-04-10 03:12:11 +04:00
|
|
|
$vars->{'action'} = $cgi->param('action');
|
2002-09-28 22:42:54 +04:00
|
|
|
my $type = {};
|
2005-04-10 03:12:11 +04:00
|
|
|
foreach my $key ($cgi->param()) { $type->{$key} = $cgi->param($key) }
|
2005-05-05 23:20:45 +04:00
|
|
|
$type->{'inclusions'} = \%inclusions;
|
|
|
|
$type->{'exclusions'} = \%exclusions;
|
2002-09-28 22:42:54 +04:00
|
|
|
$vars->{'type'} = $type;
|
|
|
|
|
|
|
|
# Return the appropriate HTTP response headers.
|
2005-04-10 03:12:11 +04:00
|
|
|
print $cgi->header();
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Generate and return the UI (HTML page) from the appropriate template.
|
|
|
|
$template->process("admin/flag-type/edit.html.tmpl", $vars)
|
|
|
|
|| ThrowTemplateError($template->error());
|
|
|
|
}
|
|
|
|
|
2005-05-05 23:20:45 +04:00
|
|
|
# Convert the array @clusions('prod_ID:comp_ID') back to a hash of
|
|
|
|
# the form %clusions{'prod_name:comp_name'} = 'prod_ID:comp_ID'
|
|
|
|
sub clusion_array_to_hash {
|
|
|
|
my $array = shift;
|
|
|
|
my %hash;
|
|
|
|
foreach my $ids (@$array) {
|
|
|
|
trick_taint($ids);
|
|
|
|
my ($product_id, $component_id) = split(":", $ids);
|
|
|
|
my $product_name = get_product_name($product_id) || "__Any__";
|
|
|
|
my $component_name = get_component_name($component_id) || "__Any__";
|
|
|
|
$hash{"$product_name:$component_name"} = $ids;
|
|
|
|
}
|
|
|
|
return %hash;
|
|
|
|
}
|
|
|
|
|
2002-09-28 22:42:54 +04:00
|
|
|
sub insert {
|
|
|
|
validateName();
|
|
|
|
validateDescription();
|
|
|
|
validateCCList();
|
|
|
|
validateTargetType();
|
|
|
|
validateSortKey();
|
|
|
|
validateIsActive();
|
|
|
|
validateIsRequestable();
|
|
|
|
validateIsRequesteeble();
|
|
|
|
validateAllowMultiple();
|
2004-11-24 01:41:44 +03:00
|
|
|
validateGroups();
|
2005-02-18 00:57:27 +03:00
|
|
|
|
|
|
|
my $dbh = Bugzilla->dbh;
|
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
my $name = SqlQuote($cgi->param('name'));
|
|
|
|
my $description = SqlQuote($cgi->param('description'));
|
|
|
|
my $cc_list = SqlQuote($cgi->param('cc_list'));
|
|
|
|
my $target_type = $cgi->param('target_type') eq "bug" ? "b" : "a";
|
2005-02-18 00:57:27 +03:00
|
|
|
|
|
|
|
$dbh->bz_lock_tables('flagtypes WRITE', 'products READ',
|
|
|
|
'components READ', 'flaginclusions WRITE',
|
|
|
|
'flagexclusions WRITE');
|
|
|
|
|
2002-09-28 22:42:54 +04:00
|
|
|
# Determine the new flag type's unique identifier.
|
|
|
|
SendSQL("SELECT MAX(id) FROM flagtypes");
|
|
|
|
my $id = FetchSQLData() + 1;
|
|
|
|
|
|
|
|
# Insert a record for the new flag type into the database.
|
|
|
|
SendSQL("INSERT INTO flagtypes (id, name, description, cc_list,
|
|
|
|
target_type, sortkey, is_active, is_requestable,
|
2004-11-24 01:41:44 +03:00
|
|
|
is_requesteeble, is_multiplicable,
|
|
|
|
grant_group_id, request_group_id)
|
2005-04-10 03:12:11 +04:00
|
|
|
VALUES ($id, $name, $description, $cc_list, '$target_type', " .
|
|
|
|
$cgi->param('sortkey') . ", " .
|
|
|
|
$cgi->param('is_active') . ", " .
|
|
|
|
$cgi->param('is_requestable') . ", " .
|
|
|
|
$cgi->param('is_requesteeble') . ", " .
|
|
|
|
$cgi->param('is_multiplicable') . ", " .
|
|
|
|
$cgi->param('grant_gid') . ", " .
|
|
|
|
$cgi->param('request_gid') . ")");
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Populate the list of inclusions/exclusions for this flag type.
|
2005-05-05 23:20:45 +04:00
|
|
|
validateAndSubmit($id);
|
2005-02-18 00:57:27 +03:00
|
|
|
|
|
|
|
$dbh->bz_unlock_tables();
|
2002-09-28 22:42:54 +04:00
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
$vars->{'name'} = $cgi->param('name');
|
2002-09-28 22:42:54 +04:00
|
|
|
$vars->{'message'} = "flag_type_created";
|
|
|
|
|
|
|
|
# Return the appropriate HTTP response headers.
|
2005-04-10 03:12:11 +04:00
|
|
|
print $cgi->header();
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Generate and return the UI (HTML page) from the appropriate template.
|
|
|
|
$template->process("global/message.html.tmpl", $vars)
|
|
|
|
|| ThrowTemplateError($template->error());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub update {
|
2005-04-10 03:12:11 +04:00
|
|
|
my $id = validateID();
|
2002-09-28 22:42:54 +04:00
|
|
|
validateName();
|
|
|
|
validateDescription();
|
|
|
|
validateCCList();
|
|
|
|
validateTargetType();
|
|
|
|
validateSortKey();
|
|
|
|
validateIsActive();
|
|
|
|
validateIsRequestable();
|
|
|
|
validateIsRequesteeble();
|
|
|
|
validateAllowMultiple();
|
2004-11-24 01:41:44 +03:00
|
|
|
validateGroups();
|
2005-02-18 00:57:27 +03:00
|
|
|
|
|
|
|
my $dbh = Bugzilla->dbh;
|
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
my $name = SqlQuote($cgi->param('name'));
|
|
|
|
my $description = SqlQuote($cgi->param('description'));
|
|
|
|
my $cc_list = SqlQuote($cgi->param('cc_list'));
|
2005-02-18 00:57:27 +03:00
|
|
|
|
|
|
|
$dbh->bz_lock_tables('flagtypes WRITE', 'products READ',
|
|
|
|
'components READ', 'flaginclusions WRITE',
|
|
|
|
'flagexclusions WRITE');
|
2002-09-28 22:42:54 +04:00
|
|
|
SendSQL("UPDATE flagtypes
|
|
|
|
SET name = $name ,
|
|
|
|
description = $description ,
|
|
|
|
cc_list = $cc_list ,
|
2005-04-10 03:12:11 +04:00
|
|
|
sortkey = " . $cgi->param('sortkey') . ",
|
|
|
|
is_active = " . $cgi->param('is_active') . ",
|
|
|
|
is_requestable = " . $cgi->param('is_requestable') . ",
|
|
|
|
is_requesteeble = " . $cgi->param('is_requesteeble') . ",
|
|
|
|
is_multiplicable = " . $cgi->param('is_multiplicable') . ",
|
|
|
|
grant_group_id = " . $cgi->param('grant_gid') . ",
|
|
|
|
request_group_id = " . $cgi->param('request_gid') . "
|
|
|
|
WHERE id = $id");
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Update the list of inclusions/exclusions for this flag type.
|
2005-05-05 23:20:45 +04:00
|
|
|
validateAndSubmit($id);
|
2002-09-28 22:42:54 +04:00
|
|
|
|
2005-02-18 00:57:27 +03:00
|
|
|
$dbh->bz_unlock_tables();
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Clear existing flags for bugs/attachments in categories no longer on
|
|
|
|
# the list of inclusions or that have been added to the list of exclusions.
|
|
|
|
SendSQL("
|
|
|
|
SELECT flags.id
|
2005-04-05 01:52:06 +04:00
|
|
|
FROM flags
|
|
|
|
INNER JOIN bugs
|
|
|
|
ON flags.bug_id = bugs.bug_id
|
|
|
|
LEFT OUTER JOIN flaginclusions AS i
|
|
|
|
ON (flags.type_id = i.type_id
|
2002-09-28 22:42:54 +04:00
|
|
|
AND (bugs.product_id = i.product_id OR i.product_id IS NULL)
|
|
|
|
AND (bugs.component_id = i.component_id OR i.component_id IS NULL))
|
2005-04-10 03:12:11 +04:00
|
|
|
WHERE flags.type_id = $id
|
2004-07-06 11:08:02 +04:00
|
|
|
AND flags.is_active = 1
|
2002-09-28 22:42:54 +04:00
|
|
|
AND i.type_id IS NULL
|
|
|
|
");
|
|
|
|
Bugzilla::Flag::clear(FetchOneColumn()) while MoreSQLData();
|
|
|
|
|
|
|
|
SendSQL("
|
|
|
|
SELECT flags.id
|
2005-04-05 01:52:06 +04:00
|
|
|
FROM flags
|
|
|
|
INNER JOIN bugs
|
|
|
|
ON flags.bug_id = bugs.bug_id
|
|
|
|
INNER JOIN flagexclusions AS e
|
|
|
|
ON flags.type_id = e.type_id
|
2005-04-10 03:12:11 +04:00
|
|
|
WHERE flags.type_id = $id
|
2004-07-06 11:08:02 +04:00
|
|
|
AND flags.is_active = 1
|
2002-09-28 22:42:54 +04:00
|
|
|
AND (bugs.product_id = e.product_id OR e.product_id IS NULL)
|
|
|
|
AND (bugs.component_id = e.component_id OR e.component_id IS NULL)
|
|
|
|
");
|
|
|
|
Bugzilla::Flag::clear(FetchOneColumn()) while MoreSQLData();
|
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
$vars->{'name'} = $cgi->param('name');
|
2002-09-28 22:42:54 +04:00
|
|
|
$vars->{'message'} = "flag_type_changes_saved";
|
|
|
|
|
|
|
|
# Return the appropriate HTTP response headers.
|
2005-04-10 03:12:11 +04:00
|
|
|
print $cgi->header();
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Generate and return the UI (HTML page) from the appropriate template.
|
|
|
|
$template->process("global/message.html.tmpl", $vars)
|
|
|
|
|| ThrowTemplateError($template->error());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub confirmDelete
|
|
|
|
{
|
2005-04-10 03:12:11 +04:00
|
|
|
my $id = validateID();
|
|
|
|
|
2002-09-28 22:42:54 +04:00
|
|
|
# check if we need confirmation to delete:
|
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
my $count = Bugzilla::Flag::count({ 'type_id' => $id,
|
2004-07-06 11:08:02 +04:00
|
|
|
'is_active' => 1 });
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
if ($count > 0) {
|
2005-04-10 03:12:11 +04:00
|
|
|
$vars->{'flag_type'} = Bugzilla::FlagType::get($id);
|
2002-09-28 22:42:54 +04:00
|
|
|
$vars->{'flag_count'} = scalar($count);
|
|
|
|
|
|
|
|
# Return the appropriate HTTP response headers.
|
2005-04-10 03:12:11 +04:00
|
|
|
print $cgi->header();
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Generate and return the UI (HTML page) from the appropriate template.
|
|
|
|
$template->process("admin/flag-type/confirm-delete.html.tmpl", $vars)
|
|
|
|
|| ThrowTemplateError($template->error());
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
deleteType();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-09-03 13:58:00 +04:00
|
|
|
sub deleteType {
|
2005-04-10 03:12:11 +04:00
|
|
|
my $id = validateID();
|
2005-02-18 00:57:27 +03:00
|
|
|
my $dbh = Bugzilla->dbh;
|
|
|
|
|
|
|
|
$dbh->bz_lock_tables('flagtypes WRITE', 'flags WRITE',
|
|
|
|
'flaginclusions WRITE', 'flagexclusions WRITE');
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Get the name of the flag type so we can tell users
|
|
|
|
# what was deleted.
|
2005-04-10 03:12:11 +04:00
|
|
|
SendSQL("SELECT name FROM flagtypes WHERE id = $id");
|
2002-09-28 22:42:54 +04:00
|
|
|
$vars->{'name'} = FetchOneColumn();
|
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
SendSQL("DELETE FROM flags WHERE type_id = $id");
|
|
|
|
SendSQL("DELETE FROM flaginclusions WHERE type_id = $id");
|
|
|
|
SendSQL("DELETE FROM flagexclusions WHERE type_id = $id");
|
|
|
|
SendSQL("DELETE FROM flagtypes WHERE id = $id");
|
2005-02-18 00:57:27 +03:00
|
|
|
$dbh->bz_unlock_tables();
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
$vars->{'message'} = "flag_type_deleted";
|
|
|
|
|
|
|
|
# Return the appropriate HTTP response headers.
|
2005-04-10 03:12:11 +04:00
|
|
|
print $cgi->header();
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Generate and return the UI (HTML page) from the appropriate template.
|
|
|
|
$template->process("global/message.html.tmpl", $vars)
|
|
|
|
|| ThrowTemplateError($template->error());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub deactivate {
|
2005-04-10 03:12:11 +04:00
|
|
|
my $id = validateID();
|
2002-09-28 22:42:54 +04:00
|
|
|
validateIsActive();
|
2005-02-18 00:57:27 +03:00
|
|
|
|
|
|
|
my $dbh = Bugzilla->dbh;
|
|
|
|
|
|
|
|
$dbh->bz_lock_tables('flagtypes WRITE');
|
2005-04-10 03:12:11 +04:00
|
|
|
SendSQL("UPDATE flagtypes SET is_active = 0 WHERE id = $id");
|
2005-02-18 00:57:27 +03:00
|
|
|
$dbh->bz_unlock_tables();
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
$vars->{'message'} = "flag_type_deactivated";
|
2005-04-10 03:12:11 +04:00
|
|
|
$vars->{'flag_type'} = Bugzilla::FlagType::get($id);
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Return the appropriate HTTP response headers.
|
2005-04-10 03:12:11 +04:00
|
|
|
print $cgi->header();
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
# Generate and return the UI (HTML page) from the appropriate template.
|
|
|
|
$template->process("global/message.html.tmpl", $vars)
|
|
|
|
|| ThrowTemplateError($template->error());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
################################################################################
|
|
|
|
# Data Validation / Security Authorization
|
|
|
|
################################################################################
|
|
|
|
|
|
|
|
sub validateID {
|
2005-04-10 03:12:11 +04:00
|
|
|
# $flagtype_id is destroyed if detaint_natural fails.
|
|
|
|
my $flagtype_id = $cgi->param('id');
|
|
|
|
detaint_natural($flagtype_id)
|
|
|
|
|| ThrowCodeError("flag_type_id_invalid",
|
|
|
|
{ id => scalar $cgi->param('id') });
|
2002-09-28 22:42:54 +04:00
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
SendSQL("SELECT 1 FROM flagtypes WHERE id = $flagtype_id");
|
2002-09-28 22:42:54 +04:00
|
|
|
FetchOneColumn()
|
2005-04-10 03:12:11 +04:00
|
|
|
|| ThrowCodeError("flag_type_nonexistent", { id => $flagtype_id });
|
|
|
|
|
|
|
|
return $flagtype_id;
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
sub validateName {
|
2005-04-10 03:12:11 +04:00
|
|
|
$cgi->param('name')
|
|
|
|
&& $cgi->param('name') !~ /[ ,]/
|
|
|
|
&& length($cgi->param('name')) <= 50
|
|
|
|
|| ThrowUserError("flag_type_name_invalid",
|
|
|
|
{ name => scalar $cgi->param('name') });
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
sub validateDescription {
|
2005-04-10 03:12:11 +04:00
|
|
|
length($cgi->param('description')) < 2**16-1
|
2002-09-28 22:42:54 +04:00
|
|
|
|| ThrowUserError("flag_type_description_invalid");
|
|
|
|
}
|
|
|
|
|
|
|
|
sub validateCCList {
|
2005-04-10 03:12:11 +04:00
|
|
|
length($cgi->param('cc_list')) <= 200
|
2002-09-28 22:42:54 +04:00
|
|
|
|| ThrowUserError("flag_type_cc_list_invalid",
|
2005-04-10 03:12:11 +04:00
|
|
|
{ cc_list => $cgi->param('cc_list') });
|
2002-09-28 22:42:54 +04:00
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
my @addresses = split(/[, ]+/, $cgi->param('cc_list'));
|
2005-08-15 21:43:38 +04:00
|
|
|
foreach my $address (@addresses) {
|
|
|
|
validate_email_syntax($address)
|
|
|
|
|| ThrowUserError('illegal_email_address', {addr => $address});
|
|
|
|
}
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
sub validateProduct {
|
2005-04-10 03:12:11 +04:00
|
|
|
return if !$cgi->param('product');
|
2002-09-28 22:42:54 +04:00
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
$product_id = get_product_id($cgi->param('product'));
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
defined($product_id)
|
|
|
|
|| ThrowCodeError("flag_type_product_nonexistent",
|
2005-04-10 03:12:11 +04:00
|
|
|
{ product => $cgi->param('product') });
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
sub validateComponent {
|
2005-04-10 03:12:11 +04:00
|
|
|
return if !$cgi->param('component');
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
$product_id
|
|
|
|
|| ThrowCodeError("flag_type_component_without_product");
|
|
|
|
|
2005-04-10 03:12:11 +04:00
|
|
|
$component_id = get_component_id($product_id, $cgi->param('component'));
|
2002-09-28 22:42:54 +04:00
|
|
|
|
|
|
|
defined($component_id)
|
|
|
|
|| ThrowCodeError("flag_type_component_nonexistent",
|
2005-04-10 03:12:11 +04:00
|
|
|
{ product => $cgi->param('product'),
|
|
|
|
name => $cgi->param('component') });
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
sub validateSortKey {
|
2005-04-10 03:12:11 +04:00
|
|
|
# $sortkey is destroyed if detaint_natural fails.
|
|
|
|
my $sortkey = $cgi->param('sortkey');
|
|
|
|
detaint_natural($sortkey)
|
|
|
|
&& $sortkey < 32768
|
2002-09-28 22:42:54 +04:00
|
|
|
|| ThrowUserError("flag_type_sortkey_invalid",
|
2005-04-10 03:12:11 +04:00
|
|
|
{ sortkey => scalar $cgi->param('sortkey') });
|
|
|
|
$cgi->param('sortkey', $sortkey);
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
sub validateTargetType {
|
2005-04-10 03:12:11 +04:00
|
|
|
grep($cgi->param('target_type') eq $_, ("bug", "attachment"))
|
2002-09-28 22:42:54 +04:00
|
|
|
|| ThrowCodeError("flag_type_target_type_invalid",
|
2005-04-10 03:12:11 +04:00
|
|
|
{ target_type => scalar $cgi->param('target_type') });
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
sub validateIsActive {
|
2005-04-10 03:12:11 +04:00
|
|
|
$cgi->param('is_active', $cgi->param('is_active') ? 1 : 0);
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
sub validateIsRequestable {
|
2005-04-10 03:12:11 +04:00
|
|
|
$cgi->param('is_requestable', $cgi->param('is_requestable') ? 1 : 0);
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
sub validateIsRequesteeble {
|
2005-04-10 03:12:11 +04:00
|
|
|
$cgi->param('is_requesteeble', $cgi->param('is_requesteeble') ? 1 : 0);
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
sub validateAllowMultiple {
|
2005-04-10 03:12:11 +04:00
|
|
|
$cgi->param('is_multiplicable', $cgi->param('is_multiplicable') ? 1 : 0);
|
2002-09-28 22:42:54 +04:00
|
|
|
}
|
|
|
|
|
2004-11-24 01:41:44 +03:00
|
|
|
sub validateGroups {
|
|
|
|
# Convert group names to group IDs
|
|
|
|
foreach my $col ("grant_gid", "request_gid") {
|
2005-04-10 03:12:11 +04:00
|
|
|
my $name = $cgi->param($col);
|
|
|
|
$cgi->param($col, "NULL") unless $name;
|
2004-11-24 01:41:44 +03:00
|
|
|
next if (!$name);
|
|
|
|
SendSQL("SELECT id FROM groups WHERE name = " . SqlQuote($name));
|
2005-04-10 03:12:11 +04:00
|
|
|
my $gid = FetchOneColumn();
|
|
|
|
if (!$gid) {
|
2004-11-24 01:41:44 +03:00
|
|
|
ThrowUserError("group_unknown", { name => $name });
|
|
|
|
}
|
2005-04-10 03:12:11 +04:00
|
|
|
$cgi->param($col, $gid);
|
2004-11-24 01:41:44 +03:00
|
|
|
}
|
|
|
|
}
|
2005-05-05 23:20:45 +04:00
|
|
|
|
|
|
|
# At this point, values either come the DB itself or have been recently
|
|
|
|
# added by the user and have passed all validation tests.
|
|
|
|
# The only way to have invalid product/component combinations is to
|
|
|
|
# hack the URL. So we silently ignore them, if any.
|
2005-08-13 16:27:04 +04:00
|
|
|
sub validateAndSubmit {
|
2005-05-05 23:20:45 +04:00
|
|
|
my ($id) = @_;
|
|
|
|
my $dbh = Bugzilla->dbh;
|
|
|
|
|
|
|
|
foreach my $category_type ("inclusions", "exclusions") {
|
|
|
|
# Will be used several times below.
|
|
|
|
my $sth = $dbh->prepare("INSERT INTO flag$category_type " .
|
|
|
|
"(type_id, product_id, component_id) " .
|
|
|
|
"VALUES (?, ?, ?)");
|
|
|
|
|
|
|
|
$dbh->do("DELETE FROM flag$category_type WHERE type_id = ?", undef, $id);
|
|
|
|
foreach my $category ($cgi->param($category_type)) {
|
|
|
|
trick_taint($category);
|
|
|
|
my ($product_id, $component_id) = split(":", $category);
|
|
|
|
# The product does not exist.
|
|
|
|
next if ($product_id && !get_product_name($product_id));
|
|
|
|
# A component was selected without a product being selected.
|
|
|
|
next if (!$product_id && $component_id);
|
|
|
|
# The component does not belong to this product.
|
|
|
|
next if ($component_id
|
|
|
|
&& !$dbh->selectrow_array("SELECT id FROM components
|
|
|
|
WHERE id = ? AND product_id = ?",
|
|
|
|
undef, ($component_id, $product_id)));
|
|
|
|
$product_id ||= undef;
|
|
|
|
$component_id ||= undef;
|
|
|
|
$sth->execute($id, $product_id, $component_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|