2003-06-25 23:23:13 +00: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): Gervase Markham <gerv@gerv.net>
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
use lib ".";
|
|
|
|
|
|
|
|
# This module implements a series - a set of data to be plotted on a chart.
|
2003-12-07 22:43:20 +00:00
|
|
|
#
|
|
|
|
# This Series is in the database if and only if self->{'series_id'} is defined.
|
|
|
|
# Note that the series being in the database does not mean that the fields of
|
|
|
|
# this object are the same as the DB entries, as the object may have been
|
|
|
|
# altered.
|
|
|
|
|
2003-06-25 23:23:13 +00:00
|
|
|
package Bugzilla::Series;
|
|
|
|
|
|
|
|
use Bugzilla;
|
|
|
|
use Bugzilla::Util;
|
|
|
|
use Bugzilla::User;
|
|
|
|
|
2003-12-07 22:43:20 +00:00
|
|
|
use constant PUBLIC_USER_ID => 0;
|
|
|
|
|
2003-06-25 23:23:13 +00:00
|
|
|
sub new {
|
|
|
|
my $invocant = shift;
|
|
|
|
my $class = ref($invocant) || $invocant;
|
|
|
|
|
|
|
|
# Create a ref to an empty hash and bless it
|
|
|
|
my $self = {};
|
|
|
|
bless($self, $class);
|
|
|
|
|
2003-12-07 22:43:20 +00:00
|
|
|
my $arg_count = scalar(@_);
|
|
|
|
|
2004-08-29 21:29:36 +00:00
|
|
|
# new() can return undef if you pass in a series_id and the user doesn't
|
|
|
|
# have sufficient permissions. If you create a new series in this way,
|
|
|
|
# you need to check for an undef return, and act appropriately.
|
|
|
|
my $retval = $self;
|
|
|
|
|
2004-02-12 22:33:07 +00:00
|
|
|
# There are three ways of creating Series objects. Two (CGI and Parameters)
|
|
|
|
# are for use when creating a new series. One (Database) is for retrieving
|
|
|
|
# information on existing series.
|
2003-12-07 22:43:20 +00:00
|
|
|
if ($arg_count == 1) {
|
2003-06-25 23:23:13 +00:00
|
|
|
if (ref($_[0])) {
|
2003-11-08 00:25:31 +00:00
|
|
|
# We've been given a CGI object to create a new Series from.
|
2004-02-12 22:33:07 +00:00
|
|
|
# This series may already exist - external code needs to check
|
|
|
|
# before it calls writeToDatabase().
|
2003-12-07 22:43:20 +00:00
|
|
|
$self->initFromCGI($_[0]);
|
2003-06-25 23:23:13 +00:00
|
|
|
}
|
|
|
|
else {
|
2003-11-08 00:25:31 +00:00
|
|
|
# We've been given a series_id, which should represent an existing
|
|
|
|
# Series.
|
2004-08-29 21:29:36 +00:00
|
|
|
$retval = $self->initFromDatabase($_[0]);
|
2003-06-25 23:23:13 +00:00
|
|
|
}
|
|
|
|
}
|
2003-12-07 22:43:20 +00:00
|
|
|
elsif ($arg_count >= 6 && $arg_count <= 8) {
|
2003-11-08 00:25:31 +00:00
|
|
|
# We've been given a load of parameters to create a new Series from.
|
2004-02-12 22:33:07 +00:00
|
|
|
# Currently, undef is always passed as the first parameter; this allows
|
|
|
|
# you to call writeToDatabase() unconditionally.
|
2003-12-07 22:43:20 +00:00
|
|
|
$self->initFromParameters(@_);
|
2003-06-25 23:23:13 +00:00
|
|
|
}
|
|
|
|
else {
|
2003-12-07 22:43:20 +00:00
|
|
|
die("Bad parameters passed in - invalid number of args: $arg_count");
|
2003-06-25 23:23:13 +00:00
|
|
|
}
|
|
|
|
|
2004-08-29 21:29:36 +00:00
|
|
|
return $retval;
|
2003-06-25 23:23:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
sub initFromDatabase {
|
|
|
|
my $self = shift;
|
|
|
|
my $series_id = shift;
|
|
|
|
|
|
|
|
&::detaint_natural($series_id)
|
|
|
|
|| &::ThrowCodeError("invalid_series_id", { 'series_id' => $series_id });
|
|
|
|
|
|
|
|
my $dbh = Bugzilla->dbh;
|
|
|
|
my @series = $dbh->selectrow_array("SELECT series.series_id, cc1.name, " .
|
|
|
|
"cc2.name, series.name, series.creator, series.frequency, " .
|
2004-08-29 21:29:36 +00:00
|
|
|
"series.query, series.public " .
|
2003-06-25 23:23:13 +00:00
|
|
|
"FROM series " .
|
|
|
|
"LEFT JOIN series_categories AS cc1 " .
|
2004-08-29 21:29:36 +00:00
|
|
|
" ON series.category = cc1.id " .
|
2003-06-25 23:23:13 +00:00
|
|
|
"LEFT JOIN series_categories AS cc2 " .
|
2004-08-29 21:29:36 +00:00
|
|
|
" ON series.subcategory = cc2.id " .
|
|
|
|
"LEFT JOIN category_group_map AS cgm " .
|
|
|
|
" ON series.category = cgm.category_id " .
|
|
|
|
"LEFT JOIN user_group_map AS ugm " .
|
|
|
|
" ON cgm.group_id = ugm.group_id " .
|
|
|
|
" AND ugm.user_id = " . Bugzilla->user->id .
|
|
|
|
" AND isbless = 0 " .
|
|
|
|
"WHERE series.series_id = $series_id AND " .
|
|
|
|
"(public = 1 OR creator = " . Bugzilla->user->id . " OR " .
|
|
|
|
"(ugm.group_id IS NOT NULL)) " .
|
|
|
|
"GROUP BY series_id");
|
2003-06-25 23:23:13 +00:00
|
|
|
|
|
|
|
if (@series) {
|
|
|
|
$self->initFromParameters(@series);
|
2004-08-29 21:29:36 +00:00
|
|
|
return $self;
|
2003-06-25 23:23:13 +00:00
|
|
|
}
|
|
|
|
else {
|
2004-08-29 21:29:36 +00:00
|
|
|
return undef;
|
2003-06-25 23:23:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub initFromParameters {
|
2003-12-07 22:43:20 +00:00
|
|
|
# Pass undef as the first parameter if you are creating a new series.
|
2003-06-25 23:23:13 +00:00
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
($self->{'series_id'}, $self->{'category'}, $self->{'subcategory'},
|
|
|
|
$self->{'name'}, $self->{'creator'}, $self->{'frequency'},
|
2003-11-08 00:25:31 +00:00
|
|
|
$self->{'query'}, $self->{'public'}) = @_;
|
2003-06-25 23:23:13 +00:00
|
|
|
}
|
|
|
|
|
2003-12-07 22:43:20 +00:00
|
|
|
sub initFromCGI {
|
|
|
|
my $self = shift;
|
|
|
|
my $cgi = shift;
|
|
|
|
|
|
|
|
$self->{'series_id'} = $cgi->param('series_id') || undef;
|
|
|
|
if (defined($self->{'series_id'})) {
|
|
|
|
detaint_natural($self->{'series_id'})
|
|
|
|
|| &::ThrowCodeError("invalid_series_id",
|
|
|
|
{ 'series_id' => $self->{'series_id'} });
|
|
|
|
}
|
|
|
|
|
|
|
|
$self->{'category'} = $cgi->param('category')
|
|
|
|
|| $cgi->param('newcategory')
|
|
|
|
|| &::ThrowUserError("missing_category");
|
|
|
|
|
|
|
|
$self->{'subcategory'} = $cgi->param('subcategory')
|
|
|
|
|| $cgi->param('newsubcategory')
|
|
|
|
|| &::ThrowUserError("missing_subcategory");
|
|
|
|
|
|
|
|
$self->{'name'} = $cgi->param('name')
|
|
|
|
|| &::ThrowUserError("missing_name");
|
|
|
|
|
|
|
|
$self->{'creator'} = Bugzilla->user->id;
|
|
|
|
|
|
|
|
$self->{'frequency'} = $cgi->param('frequency');
|
|
|
|
detaint_natural($self->{'frequency'})
|
|
|
|
|| &::ThrowUserError("missing_frequency");
|
|
|
|
|
|
|
|
$self->{'query'} = $cgi->canonicalise_query("format", "ctype", "action",
|
|
|
|
"category", "subcategory", "name",
|
|
|
|
"frequency", "public", "query_format");
|
2004-08-29 21:29:36 +00:00
|
|
|
trick_taint($self->{'query'});
|
2003-12-07 22:43:20 +00:00
|
|
|
|
2004-08-29 21:29:36 +00:00
|
|
|
$self->{'public'} = $cgi->param('public') ? 1 : 0;
|
|
|
|
|
|
|
|
# Change 'admin' here and in series.html.tmpl, or remove the check
|
|
|
|
# completely, if you want to change who can make series public.
|
|
|
|
$self->{'public'} = 0 unless &::UserInGroup('admin');
|
2003-12-07 22:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
sub writeToDatabase {
|
2003-06-25 23:23:13 +00:00
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
my $dbh = Bugzilla->dbh;
|
2005-02-17 21:57:27 +00:00
|
|
|
$dbh->bz_lock_tables('series_categories WRITE', 'series WRITE');
|
2003-06-25 23:23:13 +00:00
|
|
|
|
|
|
|
my $category_id = getCategoryID($self->{'category'});
|
|
|
|
my $subcategory_id = getCategoryID($self->{'subcategory'});
|
|
|
|
|
2004-02-12 22:33:07 +00:00
|
|
|
my $exists;
|
|
|
|
if ($self->{'series_id'}) {
|
|
|
|
$exists =
|
|
|
|
$dbh->selectrow_array("SELECT series_id FROM series
|
|
|
|
WHERE series_id = $self->{'series_id'}");
|
|
|
|
}
|
|
|
|
|
2003-12-07 22:43:20 +00:00
|
|
|
# Is this already in the database?
|
2004-02-12 22:33:07 +00:00
|
|
|
if ($exists) {
|
2003-12-07 22:43:20 +00:00
|
|
|
# Update existing series
|
|
|
|
my $dbh = Bugzilla->dbh;
|
|
|
|
$dbh->do("UPDATE series SET " .
|
|
|
|
"category = ?, subcategory = ?," .
|
2004-08-29 21:29:36 +00:00
|
|
|
"name = ?, frequency = ?, public = ? " .
|
2003-12-07 22:43:20 +00:00
|
|
|
"WHERE series_id = ?", undef,
|
|
|
|
$category_id, $subcategory_id, $self->{'name'},
|
2004-08-29 21:29:36 +00:00
|
|
|
$self->{'frequency'}, $self->{'public'},
|
|
|
|
$self->{'series_id'});
|
2003-06-25 23:23:13 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
# Insert the new series into the series table
|
|
|
|
$dbh->do("INSERT INTO series (creator, category, subcategory, " .
|
2004-08-29 21:29:36 +00:00
|
|
|
"name, frequency, query, public) VALUES " .
|
|
|
|
"($self->{'creator'}, " .
|
2003-06-25 23:23:13 +00:00
|
|
|
"$category_id, $subcategory_id, " .
|
|
|
|
$dbh->quote($self->{'name'}) . ", $self->{'frequency'}," .
|
2004-08-29 21:29:36 +00:00
|
|
|
$dbh->quote($self->{'query'}) . ", $self->{'public'})");
|
2003-06-25 23:23:13 +00:00
|
|
|
|
|
|
|
# Retrieve series_id
|
|
|
|
$self->{'series_id'} = $dbh->selectrow_array("SELECT MAX(series_id) " .
|
|
|
|
"FROM series");
|
|
|
|
$self->{'series_id'}
|
|
|
|
|| &::ThrowCodeError("missing_series_id", { 'series' => $self });
|
2003-12-07 22:43:20 +00:00
|
|
|
}
|
|
|
|
|
2005-02-17 21:57:27 +00:00
|
|
|
$dbh->bz_unlock_tables();
|
2003-06-25 23:23:13 +00:00
|
|
|
}
|
|
|
|
|
2003-12-07 22:43:20 +00:00
|
|
|
# Check whether a series with this name, category and subcategory exists in
|
2004-02-12 22:33:07 +00:00
|
|
|
# the DB and, if so, returns its series_id.
|
2003-12-07 22:43:20 +00:00
|
|
|
sub existsInDatabase {
|
|
|
|
my $self = shift;
|
|
|
|
my $dbh = Bugzilla->dbh;
|
|
|
|
|
|
|
|
my $category_id = getCategoryID($self->{'category'});
|
|
|
|
my $subcategory_id = getCategoryID($self->{'subcategory'});
|
|
|
|
|
|
|
|
trick_taint($self->{'name'});
|
2004-02-12 22:33:07 +00:00
|
|
|
my $series_id = $dbh->selectrow_array("SELECT series_id " .
|
2003-12-07 22:43:20 +00:00
|
|
|
"FROM series WHERE category = $category_id " .
|
|
|
|
"AND subcategory = $subcategory_id AND name = " .
|
|
|
|
$dbh->quote($self->{'name'}));
|
|
|
|
|
2004-02-12 22:33:07 +00:00
|
|
|
return($series_id);
|
2003-12-07 22:43:20 +00:00
|
|
|
}
|
|
|
|
|
2003-06-25 23:23:13 +00:00
|
|
|
# Get a category or subcategory IDs, creating the category if it doesn't exist.
|
|
|
|
sub getCategoryID {
|
|
|
|
my ($category) = @_;
|
|
|
|
my $category_id;
|
|
|
|
my $dbh = Bugzilla->dbh;
|
|
|
|
|
|
|
|
# This seems for the best idiom for "Do A. Then maybe do B and A again."
|
|
|
|
while (1) {
|
|
|
|
# We are quoting this to put it in the DB, so we can remove taint
|
|
|
|
trick_taint($category);
|
|
|
|
|
2004-08-29 21:29:36 +00:00
|
|
|
$category_id = $dbh->selectrow_array("SELECT id " .
|
2003-06-25 23:23:13 +00:00
|
|
|
"from series_categories " .
|
|
|
|
"WHERE name =" . $dbh->quote($category));
|
2004-08-29 21:29:36 +00:00
|
|
|
|
|
|
|
last if defined($category_id);
|
2003-06-25 23:23:13 +00:00
|
|
|
|
|
|
|
$dbh->do("INSERT INTO series_categories (name) " .
|
|
|
|
"VALUES (" . $dbh->quote($category) . ")");
|
|
|
|
}
|
|
|
|
|
|
|
|
return $category_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
1;
|