ghendricks%novell.com 6f67119dc2 Synching with revision 2385
git-svn-id: svn://10.0.0.236/trunk@215321 18797224-902f-48f8-a5cc-f745e15eee43
2006-11-15 21:30:13 +00:00

779 lines
19 KiB
Perl

# -*- 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 Test Runner System.
#
# The Initial Developer of the Original Code is Maciej Maczynski.
# Portions created by Maciej Maczynski are Copyright (C) 2001
# Maciej Maczynski. All Rights Reserved.
#
# Contributor(s): Greg Hendricks <ghendricks@novell.com>
# Michael Hight <mjhight@gmail.com>
# Garrett Braden <gbraden@novell.com>
=head1 NAME
Bugzilla::Testopia::Environment - A test environment
=head1 DESCRIPTION
Environments are a set of parameters dictating what conditions
a test was conducted in. Each test run must have an environment.
Environments can be very simple or very complex.
Environments are comprised of Elemements, Properties, and Values.
Elements can be nested within other elements. Each element can have
zero or more properties. Each property can only have one value selected
of the possible values.
=head1 SYNOPSIS
$env = Bugzilla::Testopia::Environment->new($env_id);
$env = Bugzilla::Testopia::Environment->new(\%env_hash);
=cut
package Bugzilla::Testopia::Environment;
use strict;
use Bugzilla::Util;
use Bugzilla::Error;
use Bugzilla::User;
###############################
#### Initialization ####
###############################
=head1 FIELDS
environment_id
product_id
name
isactive
=cut
use constant DB_COLUMNS => qw(
environment_id
product_id
name
isactive
);
our $columns = join(", ", DB_COLUMNS);
our constant $max_depth = 7;
###############################
#### Methods ####
###############################
=head1 METHODS
=head2 new
Instantiates a new Environment object
=cut
sub new {
my $invocant = shift;
my $class = ref($invocant) || $invocant;
my $self = {};
bless($self, $class);
return $self->_init(@_);
}
=head2 _init
Private constructor for the environment class
=cut
sub _init {
my $self = shift;
my ($param) = (@_);
my $dbh = Bugzilla->dbh;
my $id = $param unless (ref $param eq 'HASH' || ref $param eq 'Bugzilla::Testopia::Environment::Xml');
my $obj;
if (defined $id && detaint_natural($id)) {
$obj = $dbh->selectrow_hashref(qq{
SELECT $columns
FROM test_environments
WHERE environment_id = ?}, undef, ($id));
} elsif (ref $param eq 'HASH' || ref $param eq 'Bugzilla::Testopia::Environment::Xml'){
$obj = $param;
}
return undef unless (defined $obj);
foreach my $field (keys %$obj) {
$self->{$field} = $obj->{$field};
}
if(! ref $obj eq 'Bugzilla::Testopia::Environment::Xml'){
$self->get_environment_elements();
my @elements = $self->{'elements'};
foreach my $elem (@elements)
{
foreach my $element (@$elem)
{
my $elem_id = $element->{'element_id'};
my @properties = $element->{'properties'};
foreach my $prop (@properties)
{
foreach my $property (@$prop)
{
my $prop_id = $property->{'property_id'};
$property->{'value_selected'} = $self->get_value_selected($self->{'environment_id'},$elem_id,$prop_id);
}
}
}
}
}
return $self;
}
=head2 get element list for environment
Returns an array of element objects for an environment
=cut
sub get_environment_elements{
my $dbh = Bugzilla->dbh;
my $self = shift;
return $self->{'elements'} if exists $self->{'elements'};
my $id = $self->{'environment_id'};
my $ref = $dbh->selectcol_arrayref(qq{
SELECT DISTINCT tee.element_id
FROM test_environment_map as tem
JOIN test_environment_element as tee
ON tem.element_id = tee.element_id
WHERE tem.environment_id = ?},undef,$id);
my @elements;
foreach my $val (@$ref){
push @elements, Bugzilla::Testopia::Environment::Element->new($val);
}
$self->{'elements'} = \@elements;
return \@elements;
}
sub element_count {
my $self = shift;
return scalar(@{$self->get_environment_elements});
}
sub elements_to_json {
my $self = shift;
my $elements = $self->get_environment_elements;
my $json = '[';
foreach my $element (@$elements)
{
$json .= '{title:"'. $element->{'name'} .'",';
$json .= 'objectId:"'. $element->{'element_id'}. '",';
$json .= 'widgetId:"element'. $element->{'element_id'} .'",';
$json .= 'actionsDisabled:["addCategory","addValue","addChild"],';
$json .= 'isFolder:true,' if($element->check_for_children || $element->check_for_properties);
$json .= 'childIconSrc:"testopia/img/circle.gif"},';
}
chop $json;
$json .= ']';
return $json;
}
=head2 get_value_selected
Returns a selected value for the specified environment element property instance.
=cut
sub get_value_selected{
my $dbh = Bugzilla->dbh;
my $self = shift;
my ($environment,$element,$property) = (@_);
my ($var) = $dbh->selectrow_array(
"SELECT value_selected
FROM test_environment_map
WHERE environment_id = ?
AND element_id = ?
AND property_id = ?",
undef,($environment,$element,$property));
return $var;
}
=head2 get environment names
Returns the list of environment names and ids
=cut
sub get_environments{
my $dbh = Bugzilla->dbh;
my $self = shift;
my $ref = $dbh->selectall_arrayref(
"SELECT environment_id, name
FROM test_environments");
return $ref;
}
sub get_all_env_categories {
my $self = shift;
my ($byid) = @_;
my $dbh = Bugzilla->dbh;
my $idstr = $byid ? 'env_category_id' : 'DISTINCT name';
my $ref = $dbh->selectall_arrayref(
"SELECT $idstr AS id, name
FROM test_environment_category",
{'Slice' => {}});
return $ref;
}
sub get_all_visible_elements {
my $self = shift;
my ($byid) = @_;
my $dbh = Bugzilla->dbh;
my $idstr = $byid ? 'element_id' : 'DISTINCT name';
my $ref = $dbh->selectall_arrayref(
"SELECT $idstr AS id, name
FROM test_environment_element",
{'Slice' => {}});
return $ref;
}
sub get_all_element_properties {
my $self = shift;
my ($byid) = @_;
my $dbh = Bugzilla->dbh;
my $idstr = $byid ? 'property_id' : 'DISTINCT name';
my $ref = $dbh->selectall_arrayref(
"SELECT $idstr AS id, name, validexp
FROM test_environment_property",
{'Slice' => {}});
return $ref;
}
sub get_distinct_property_values {
my $self = shift;
my @exps;
foreach my $prop (@{$self->get_all_element_properties}){
push @exps, split(/\|/, $prop->{'validexp'})
}
my %seen;
foreach my $v (@exps){
$seen{$v} = $v;
}
my @values;
foreach my $v (keys %seen){
my %p;
$p{'id'} = $v;
$p{'name'} = $v;
push @values, \%p;
}
return \@values;
}
=head2 get all elements
Returns the list of element names, ids and category names
=cut
sub get_all_elements{
my $dbh = Bugzilla->dbh;
my $self = shift;
my $ref = $dbh->selectcol_arrayref(
"SELECT tee.element_id
FROM test_environment_map as tem
JOIN test_environment_element as tee
ON tem.element_id = tee.element_id",
undef);
my @elements;
foreach my $val (@$ref){
push @elements, Bugzilla::Testopia::Environment::Element->new($val);
}
return \@elements;
}
=head2 check environment name
Returns environment id if environment exists
=cut
sub check_environment{
my $self = shift;
my ($name, $product_id) = (@_);
my $dbh = Bugzilla->dbh;
my ($used) = $dbh->selectrow_array(
"SELECT environment_id
FROM test_environments
WHERE name = ? AND product_id = ?",
undef, ($name, $product_id));
return $used;
}
=head2 Check Environment Element Property Value Selected
Returns environment id if Environment Element Property Value Selected exists
=cut
sub check_value_selected {
my $self = shift;
my ($prop_id, $elem_id) = @_;
my $dbh = Bugzilla->dbh;
my ($used) = $dbh->selectrow_array(
"SELECT environment_id
FROM test_environment_map
WHERE environment_id = ?
AND property_id = ?
AND element_id = ?",
undef, ($self->{'environment_id'}, $prop_id, $elem_id));
return $used;
}
=head2 store
Serializes this environment to the database
=cut
sub store {
my $self = shift;
my $timestamp = Bugzilla::Testopia::Util::get_time_stamp();
#Verify Environment isn't already in use.
return undef if $self->check_environment($self->{'name'}, $self->{'product_id'});
my $dbh = Bugzilla->dbh;
$dbh->do("INSERT INTO test_environments ($columns)
VALUES (?,?,?,?)",
undef, (undef, $self->{'product_id'}, $self->{'name'}, $self->{'isactive'}));
my $key = $dbh->bz_last_key( 'test_environments', 'environment_id' );
my $elements = $self->{'elements'};
foreach my $element (@$elements)
{
$self->persist_environment_element_and_children(1, $element, "store");
}
return $key;
}
=head2 store property values
Serializes the property values to the database
=cut
sub store_property_value {
my $self = shift;
my ($prop_id,$elem_id,$value_selected) = @_;
return 0 if ($self->check_value_selected($prop_id, $elem_id, $value_selected));
my $dbh = Bugzilla->dbh;
$dbh->do("INSERT INTO test_environment_map (environment_id,property_id,element_id,value_selected)
VALUES (?,?,?,?)",undef, ($self->{'environment_id'}, $prop_id, $elem_id,$value_selected));
return 1;
}
=head2 store environment name
Serializes the environment name to the database
=cut
sub store_environment_name {
my $self = shift;
my ($name, $product_id) = (@_);
my $timestamp = Bugzilla::Testopia::Util::get_time_stamp();
return undef if $self->check_environment($name, $product_id);
my $dbh = Bugzilla->dbh;
$dbh->do("INSERT INTO test_environments ($columns)
VALUES (?,?,?,?)", undef,
(undef,$product_id,$name,1));
return $dbh->bz_last_key( 'test_environments', 'environment_id' );
}
=head2 update
Updates this environment object in the database.
Takes a reference to a hash whose keys match the fields of
an environment.
=cut
sub update {
my $self = shift;
my ($newvalues) = @_;
my $dbh = Bugzilla->dbh;
my $product_id;
$dbh->bz_lock_tables('test_environments WRITE');
foreach my $field (keys %{$newvalues}){
if ($self->{$field} ne $newvalues->{$field}){
# If the new name is already in use, return.
$product_id = $newvalues->{'product_id'} || $self->{'product_id'};
if ($product_id eq undef) {
$dbh->bz_unlock_tables;
return 0;
}
if ($field eq 'name' && $self->check_environment($newvalues->{'name'}, $product_id)) {
$dbh->bz_unlock_tables;
return 0;
}
trick_taint($newvalues->{$field});
$dbh->do("UPDATE test_environments
SET $field = ? WHERE environment_id = ?",
undef, $newvalues->{$field}, $self->{'environment_id'});
$self->{$field} = $newvalues->{$field};
}
}
$dbh->bz_unlock_tables;
my $elements = $self->{'elements'};
foreach my $element (@$elements)
{
$self->persist_environment_element_and_children(1, $element, "update");
}
return 1;
}
=head2 persist_environment_element_and_children
Persists Environment Element and Children Recursively.
=cut
sub persist_environment_element_and_children {
my $self = shift;
my ($depth, $element, $method) = @_;
if ($depth > $max_depth) {
return;
}
$depth++;
my $elem_id = $element->{'element_id'};
my $properties = $element->{'properties'};
foreach my $property (@$properties)
{
my $prop_id = $property->{'property_id'};
my $value_selected = $property->{'value_selected'};
my ($value_stored) = $self->get_value_selected($self->{'environment_id'},$prop_id,$elem_id);
if ($method eq "store" || $value_stored eq undef) {
$self->store_property_value($prop_id,$elem_id,$value_selected);
}
else {
$self->update_property_value($prop_id,$elem_id,$value_selected);
}
}
my $children = $element->{'children'};
foreach my $child_element (@$children) {
$self->persist_environment_element_and_children($depth, $child_element, $method);
}
}
=head2 update property value
Updates the property of the element in the database
=cut
sub update_property_value {
my $timestamp = Bugzilla::Testopia::Util::get_time_stamp();
my $self = shift;
my ($propID, $elemID, $valueSelected) = (@_);
my $dbh = Bugzilla->dbh;
$dbh->do("UPDATE test_environment_map
SET value_selected = ? WHERE environment_id = ? AND property_id = ? AND element_id = ?"
,undef, ($valueSelected,$self->{'environment_id'},$propID,$elemID));
return 1;
}
sub delete_element {
my $self = shift;
my ($element_id) = @_;
my $dbh = Bugzilla->dbh;
$dbh->do("DELETE FROM test_environment_map
WHERE environment_id = ? AND element_id = ?",
undef,($self->id, $element_id));
}
=head2 obliterate
Completely removes this environment from the database.
=cut
sub obliterate {
my $self = shift;
return 0 unless $self->candelete;
my $dbh = Bugzilla->dbh;
foreach my $obj (@{$self->runs}){
$obj->obliterate;
}
foreach my $obj (@{$self->caseruns}){
$obj->obliterate;
}
$dbh->do("DELETE FROM test_environment_map WHERE environment_id = ?", undef, $self->id);
$dbh->do("DELETE FROM test_environments WHERE environment_id = ?", undef, $self->id);
return 1;
}
=head2 archive
Archives this environment.
=cut
sub archive {
my $self = shift;
my $dbh = Bugzilla->dbh;
$dbh->bz_lock_tables('test_runs READ', 'test_environments WRITE','test_environment_map WRITE');
my $used = $dbh->selectrow_array("SELECT 1 FROM test_runs
WHERE environment_id = ?",
undef, $self->{'environment_id'});
if ($used) {
$dbh->bz_unlock_tables;
ThrowUserError("testopia-non-zero-run-count", {'object' => 'Environment'});
}
$dbh->do("UPDATE test_environments SET isactive = 0
WHERE environment_id = ?", undef, $self->{'environment_id'});
$dbh->bz_unlock_tables;
return 1;
}
=head2 get_run_list
Returns a list of run ids associated with this environment.
=cut
sub get_run_list {
my $self = shift;
my $dbh = Bugzilla->dbh;
my $ref = $dbh->selectcol_arrayref("SELECT run_id FROM test_runs
WHERE environment_id = ?",
undef, $self->{'environment_id'});
return join(",", @{$ref});
}
=head2 get_run_count
Returns a count of the runs associated with this environment
=cut
sub get_run_count {
my $self = shift;
my $dbh = Bugzilla->dbh;
my ($count) = $dbh->selectrow_array("SELECT COUNT(run_id) FROM test_runs
WHERE environment_id = ?",
undef, $self->{'environment_id'});
return $count;
}
=head2 canedit
Returns true if the logged in user has rights to edit this environment.
=cut
sub canedit {
my $self = shift;
return UserInGroup('managetestplans')
|| UserInGroup('edittestcases')
|| UserInGroup('runtests');
}
=head2 canview
Returns true if the logged in user has rights to view this environment.
=cut
sub canview {
my $self = shift;
return UserInGroup('managetestplans')
|| UserInGroup('edittestcases')
|| UserInGroup('runtests');
}
=head2 candelete
Returns true if the logged in user has rights to delete this environment.
=cut
sub candelete {
my $self = shift;
return 0 unless $self->canedit && Param("allow-test-deletion");
return 1 if Bugzilla->user->in_group("admin");
return 0;
}
###############################
#### Accessors ####
###############################
=head2 id
Returns the ID of this object
=head2 name
Returns the name of this object
=cut
sub id { return $_[0]->{'environment_id'}; }
sub product_id { return $_[0]->{'product_id'}; }
sub isactive { return $_[0]->{'isactive'}; }
sub name { return $_[0]->{'name'}; }
=head2 product
Returns the bugzilla product
=cut
sub product {
my $self = shift;
my $dbh = Bugzilla->dbh;
return $self->{'product'} if exists $self->{'product'};
$self->{'product'} = Bugzilla::Product->new($self->{'product_id'});
return $self->{'product'};
}
=head2 runs
Returns a reference to a list of test runs useing this environment
=cut
sub runs {
my ($self) = @_;
my $dbh = Bugzilla->dbh;
return $self->{'runs'} if exists $self->{'runs'};
my $runids = $dbh->selectcol_arrayref("SELECT run_id FROM test_runs
WHERE environment_id = ?",
undef, $self->id);
my @runs;
foreach my $id (@{$runids}){
push @runs, Bugzilla::Testopia::TestRun->new($id);
}
$self->{'runs'} = \@runs;
return $self->{'runs'};
}
=head2 caseruns
Returns a reference to a list of test caseruns useing this environment
=cut
sub caseruns {
my ($self) = @_;
my $dbh = Bugzilla->dbh;
return $self->{'caseruns'} if exists $self->{'caseruns'};
my $ids = $dbh->selectcol_arrayref("SELECT case_run_id FROM test_case_runs
WHERE environment_id = ?",
undef, $self->id);
my @caseruns;
foreach my $id (@{$ids}){
push @caseruns, Bugzilla::Testopia::TestCaseRun->new($id);
}
$self->{'caseruns'} = \@caseruns;
return $self->{'caseruns'};
}
=head1 TODO
=head1 SEE ALSO
TestRun
=head1 AUTHOR
Greg Hendricks <ghendricks@novell.com>
=cut
1;