420 lines
14 KiB
Perl
Executable File
420 lines
14 KiB
Perl
Executable File
#!/usr/bin/perl -wT
|
|
# -*- 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): Maciej Maczynski <macmac@xdsnet.pl>
|
|
# Ed Fuentetaja <efuentetaja@acm.org>
|
|
# Greg Hendricks <ghendricks@novell.com>
|
|
|
|
use strict;
|
|
use lib qw(. lib);
|
|
use Bugzilla::Constants;
|
|
use lib (bz_locations()->{extensionsdir} . '/testopia/lib');
|
|
|
|
use Bugzilla;
|
|
use Bugzilla::Util;
|
|
use Bugzilla::Error;
|
|
use Bugzilla::Field;
|
|
use Testopia::Constants;
|
|
use Testopia::Util;
|
|
use Testopia::Classification;
|
|
use Testopia::TestPlan;
|
|
use Testopia::TestRun;
|
|
use Testopia::TestCase;
|
|
use Testopia::Environment::Category;
|
|
use Testopia::Environment::Element;
|
|
use Testopia::Environment::Property;
|
|
use JSON;
|
|
|
|
local our $vars = {};
|
|
|
|
my $dbh = Bugzilla->dbh;
|
|
local our $cgi = Bugzilla->cgi;
|
|
local our $template = Bugzilla->template;
|
|
|
|
Bugzilla->login(LOGIN_REQUIRED);
|
|
Bugzilla->error_mode(ERROR_MODE_AJAX);
|
|
|
|
sub convert_bugzilla_fields {
|
|
my $field = shift;
|
|
my $values = get_legal_field_values($field);
|
|
my @vals;
|
|
foreach my $v (@$values){
|
|
push @vals, {name => $v, id => $v};
|
|
}
|
|
return \@vals;
|
|
}
|
|
|
|
sub get_searchable_objects{
|
|
my $object = shift;
|
|
my $dbh = Bugzilla->dbh;
|
|
|
|
my $products = Bugzilla->user->get_selectable_products;
|
|
my @ids;
|
|
foreach my $p (@{$products}){
|
|
push @ids, $p->id;
|
|
}
|
|
my $ref;
|
|
SWITCH: for ($object) {
|
|
/^components/ && do {
|
|
$ref = $dbh->selectall_arrayref(
|
|
"SELECT DISTINCT name AS id, name
|
|
FROM components WHERE product_id IN (". join(",",@ids) .")
|
|
ORDER BY name", {'Slice'=>{}});
|
|
last SWITCH;
|
|
};
|
|
|
|
/^categories/ && do {
|
|
$ref = $dbh->selectall_arrayref(
|
|
"SELECT DISTINCT name AS id, name
|
|
FROM test_case_categories WHERE product_id IN (". join(",",@ids) .")
|
|
ORDER BY name", {'Slice'=>{}});
|
|
last SWITCH;
|
|
};
|
|
|
|
/^builds/ && do {
|
|
$ref = $dbh->selectall_arrayref(
|
|
"SELECT DISTINCT name AS id, name
|
|
FROM test_builds WHERE product_id IN (". join(",",@ids) .")
|
|
ORDER BY name", {'Slice'=>{}});
|
|
last SWITCH;
|
|
};
|
|
|
|
/^versions/ && do {
|
|
$ref = $dbh->selectall_arrayref(
|
|
"SELECT DISTINCT value AS id, value AS name
|
|
FROM versions WHERE product_id IN (". join(",",@ids) .")
|
|
ORDER BY name", {'Slice'=>{}});
|
|
last SWITCH;
|
|
};
|
|
|
|
/^milestones/ && do {
|
|
$ref = $dbh->selectall_arrayref(
|
|
"SELECT DISTINCT value AS id, value AS name
|
|
FROM milestones WHERE product_id IN (". join(",",@ids) .")
|
|
ORDER BY name", {'Slice'=>{}});
|
|
last SWITCH;
|
|
};
|
|
|
|
/^environments/ && do {
|
|
$ref = $dbh->selectall_arrayref(
|
|
"SELECT DISTINCT name AS id, name
|
|
FROM test_environments WHERE product_id IN (". join(",",@ids) .")
|
|
ORDER BY name", {'Slice'=>{}});
|
|
last SWITCH;
|
|
};
|
|
}
|
|
return $ref;
|
|
}
|
|
|
|
my $action = $cgi->param('action') || '';
|
|
|
|
if ($action eq 'getversions'){
|
|
$cgi->send_header;
|
|
my @prod_ids;
|
|
|
|
my $type = $cgi->param('type');
|
|
my @values = split(",",$cgi->param('value'));
|
|
my $obj;
|
|
my @products;
|
|
my $prodinfo;
|
|
my $selectTypes = [qw{version milestone component build category environment}];
|
|
|
|
foreach my $value (@values){
|
|
if ($type eq 'classification'){
|
|
$obj = Testopia::Classification->new({name => $value});
|
|
next unless ($obj && scalar(grep {$_->name eq $obj->name} @{Bugzilla->user->get_selectable_classifications}));
|
|
push @products, @{$obj->user_visible_products()};
|
|
my @prod_names;
|
|
push @prod_names, $_->name foreach @products;
|
|
$prodinfo->{'product'} = \@prod_names;
|
|
unshift @$selectTypes, "product";
|
|
}
|
|
else {
|
|
trim($value);
|
|
$obj = Bugzilla::Product->new({name => $value});
|
|
next unless ($obj && Bugzilla->user->can_see_product($obj->name));
|
|
push @products, $obj;
|
|
}
|
|
}
|
|
|
|
unless (scalar @products > 0){
|
|
push @products, @{Bugzilla->user->get_selectable_products};
|
|
unshift @$selectTypes, "product";
|
|
my @prod_names;
|
|
push @prod_names, $_->name foreach @products;
|
|
$prodinfo->{'product'} = \@prod_names;
|
|
}
|
|
push @prod_ids, $_->id foreach (@products);
|
|
my $prod_ids = join(",", @prod_ids);
|
|
|
|
my $dbh = Bugzilla->dbh;
|
|
$prodinfo->{'version'} = $dbh->selectcol_arrayref("SELECT DISTINCT value FROM versions WHERE product_id IN ($prod_ids)");
|
|
$prodinfo->{'milestone'} = $dbh->selectcol_arrayref("SELECT DISTINCT value FROM milestones WHERE product_id IN ($prod_ids)");
|
|
$prodinfo->{'component'} = $dbh->selectcol_arrayref("SELECT DISTINCT name FROM components WHERE product_id IN ($prod_ids)");
|
|
$prodinfo->{'build'} = $dbh->selectcol_arrayref("SELECT DISTINCT name FROM test_builds WHERE product_id IN ($prod_ids)");
|
|
$prodinfo->{'category'} = $dbh->selectcol_arrayref("SELECT DISTINCT name FROM test_case_categories WHERE product_id IN ($prod_ids)");
|
|
$prodinfo->{'environment'} = $dbh->selectcol_arrayref("SELECT DISTINCT name FROM test_environments WHERE product_id IN ($prod_ids)");
|
|
|
|
# This list must match the name of the select fields and the names above
|
|
$prodinfo->{'selectTypes'} = $selectTypes;
|
|
|
|
my $json = new JSON;
|
|
print "{'success': true, objects: " . $json->encode($prodinfo) . "}";
|
|
}
|
|
|
|
elsif ($action eq 'get_products'){
|
|
$cgi->send_header;
|
|
my @prod;
|
|
if (Bugzilla->params->{'useclassification'}){
|
|
my @classes = $cgi->param('class_ids');
|
|
foreach my $id (@classes){
|
|
my $class = Testopia::Classification->new($id);
|
|
push @prod, @{$class->user_visible_products};
|
|
}
|
|
}
|
|
else {
|
|
@prod = Bugzilla->user->get_selectable_products;
|
|
}
|
|
|
|
my $ret;
|
|
foreach my $e (@prod){
|
|
$ret .= $e->{'id'}.'||'.$e->{'name'}.'|||';
|
|
}
|
|
chop($ret);
|
|
print $ret;
|
|
}
|
|
|
|
elsif ($action eq 'get_categories'){
|
|
$cgi->send_header;
|
|
my @prod_ids = $cgi->param('prod_id');
|
|
my $ret;
|
|
|
|
foreach my $prod_id (@prod_ids){
|
|
detaint_natural($prod_id);
|
|
my $cat = Testopia::Environment::Category->new({});
|
|
my $cats_ref = $cat->get_element_categories_by_product($prod_id);
|
|
|
|
foreach my $e (@{$cats_ref}){
|
|
$ret .= $e->id.'||'.$e->name.'|||';
|
|
}
|
|
}
|
|
chop($ret);
|
|
print $ret;
|
|
}
|
|
elsif ($action eq 'get_elements'){
|
|
$cgi->send_header;
|
|
my @cat_ids = $cgi->param('cat_id');
|
|
my $ret;
|
|
my @elmnts;
|
|
|
|
foreach my $cat_id (@cat_ids){
|
|
detaint_natural($cat_id);
|
|
|
|
my $cat = Testopia::Environment::Category->new($cat_id);
|
|
my $elmnts_ref = $cat->get_elements_by_category($cat->{'name'});
|
|
|
|
foreach my $e (@{$elmnts_ref}){
|
|
push @elmnts, Testopia::Environment::Element->new($e->{'element_id'});
|
|
}
|
|
}
|
|
|
|
foreach my $e (@elmnts){
|
|
$ret .= $e->id.'||'.$e->name.'|||';
|
|
}
|
|
chop($ret);
|
|
print $ret;
|
|
}
|
|
|
|
elsif ($action eq 'get_properties'){
|
|
$cgi->send_header;
|
|
my @elmnt_ids = $cgi->param('elmnt_id');
|
|
my $ret;
|
|
|
|
foreach my $elmnt_id (@elmnt_ids){
|
|
detaint_natural($elmnt_id);
|
|
|
|
my $elmnt = Testopia::Environment::Element->new($elmnt_id);
|
|
my $props = $elmnt->get_properties();
|
|
|
|
foreach my $e (@{$props}){
|
|
$ret .= $e->id.'||'.$e->name.'|||';
|
|
}
|
|
}
|
|
chop($ret);
|
|
print $ret;
|
|
}
|
|
elsif ($action eq 'get_valid_exp'){
|
|
$cgi->send_header;
|
|
my @prop_ids = $cgi->param('prop_id');
|
|
my $ret;
|
|
|
|
foreach my $prop_id (@prop_ids){
|
|
detaint_natural($prop_id);
|
|
|
|
my $prop = Testopia::Environment::Property->new($prop_id);
|
|
my $exp = $prop->validexp;
|
|
|
|
my @exps = split /\|/, $exp;
|
|
foreach my $exp (@exps){
|
|
$ret .=$exp.'|||';
|
|
}
|
|
}
|
|
chop($ret);
|
|
print $ret;
|
|
}
|
|
elsif ($action eq 'save_query'){
|
|
$cgi->send_header;
|
|
my $query = $cgi->param('query_part');
|
|
my $qname = $cgi->param('query_name');
|
|
my $type = $cgi->param('type');
|
|
|
|
ThrowUserError('query_name_missing') unless $qname;
|
|
|
|
trick_taint($query);
|
|
trick_taint($qname);
|
|
detaint_natural($type);
|
|
$type ||= SAVED_SEARCH;
|
|
|
|
my ($name) = $dbh->selectrow_array(
|
|
"SELECT name
|
|
FROM test_named_queries
|
|
WHERE userid = ?
|
|
AND name = ?",
|
|
undef,(Bugzilla->user->id, $qname));
|
|
|
|
if ($name){
|
|
$dbh->do(
|
|
"UPDATE test_named_queries
|
|
SET query = ?
|
|
WHERE userid = ?
|
|
AND name = ?",
|
|
undef,($query, Bugzilla->user->id, $qname));
|
|
$vars->{'tr_message'} = "Updated saved search '$qname'";
|
|
}
|
|
else{
|
|
my $quoted_qname = url_quote($qname);
|
|
$query .= "&qname=$quoted_qname" unless $type == 3;
|
|
$dbh->do("INSERT INTO test_named_queries
|
|
VALUES(?,?,?,?,?)",
|
|
undef, (Bugzilla->user->id, $qname, 1, $query, $type));
|
|
|
|
$vars->{'tr_message'} = "Search saved as '$qname'";
|
|
}
|
|
|
|
print "{'success': true}";
|
|
}
|
|
elsif ($action eq 'delete_query'){
|
|
$cgi->send_header;
|
|
my $qname = $cgi->param('query_name');
|
|
|
|
trick_taint($qname);
|
|
|
|
$dbh->do("DELETE FROM test_named_queries WHERE userid = ? AND name = ?",
|
|
undef, (Bugzilla->user->id, $qname));
|
|
$vars->{'tr_message'} = "Testopia Saved Search '$qname' Deleted";
|
|
|
|
print "{'success': true}";
|
|
}
|
|
|
|
elsif ($action eq 'get_saved_searches'){
|
|
$cgi->send_header;
|
|
my $type = $cgi->param('type');
|
|
my $user = $cgi->param('userid') || Bugzilla->user->id;
|
|
detaint_natural($type);
|
|
detaint_natural($user);
|
|
$type ||= SAVED_SEARCH;
|
|
|
|
my $ref = $dbh->selectall_arrayref(
|
|
"SELECT tnq.name, query, profiles.realname AS author, type
|
|
FROM test_named_queries AS tnq
|
|
INNER JOIN profiles ON tnq.userid = profiles.userid
|
|
WHERE tnq.userid = ? AND isvisible = 1
|
|
AND type = ?",
|
|
{'Slice' =>{}} ,($user, $type));
|
|
|
|
print "{'searches':" . to_json($ref) . "}";
|
|
}
|
|
|
|
else{
|
|
|
|
#TODO: Support default query
|
|
my $tab = $cgi->param('current_tab') || '';
|
|
if ($tab eq 'plan'){
|
|
$vars->{'plan'} = Testopia::TestPlan->new({});
|
|
$vars->{'title'} = "Search For Test Plans";
|
|
$vars->{'versions'} = get_searchable_objects('versions');
|
|
}
|
|
elsif ($tab eq 'run'){
|
|
$vars->{'title'} = "Search For Test Runs";
|
|
$vars->{'run'} = Testopia::TestRun->new({});;
|
|
$vars->{'versions'} = get_searchable_objects('versions');
|
|
$vars->{'milestones'} = get_searchable_objects('milestones');
|
|
$vars->{'builds'} = get_searchable_objects('builds');
|
|
$vars->{'environments'} = get_searchable_objects('environments');
|
|
}
|
|
elsif ($tab eq 'environment'){
|
|
$vars->{'title'} = "Search For Test Run Environments";
|
|
$vars->{'classifications'} = Bugzilla->user->get_selectable_classifications;
|
|
$vars->{'products'} = Bugzilla->user->get_selectable_products;
|
|
$vars->{'env'} = Testopia::Environment->new({});
|
|
}
|
|
elsif ($tab eq 'case_run'){
|
|
$vars->{'title'} = "Search For Test Case-Runs";
|
|
$vars->{'case'} = Testopia::TestCase->new({});
|
|
$vars->{'run'} = Testopia::TestRun->new({});
|
|
$vars->{'caserun'} = Testopia::TestCaseRun->new({});
|
|
|
|
$vars->{'versions'} = get_searchable_objects('versions');
|
|
$vars->{'milestones'} = get_searchable_objects('milestones');
|
|
$vars->{'builds'} = get_searchable_objects('builds');
|
|
$vars->{'environments'} = get_searchable_objects('environments');
|
|
$vars->{'components'} = get_searchable_objects('components');
|
|
$vars->{'categories'} = get_searchable_objects('categories');
|
|
$vars->{'bug_status'} = convert_bugzilla_fields('bug_status');
|
|
$vars->{'rep_platform'} = convert_bugzilla_fields('rep_platform');
|
|
$vars->{'op_sys'} = convert_bugzilla_fields('op_sys');
|
|
$vars->{'bug_priority'} = convert_bugzilla_fields('priority');
|
|
$vars->{'bug_severity'} = convert_bugzilla_fields('bug_severity');
|
|
$vars->{'resolution'} = convert_bugzilla_fields('resolution');
|
|
}
|
|
elsif ($tab eq 'case') {
|
|
$tab = 'case';
|
|
my $case = Testopia::TestCase->new({ 'case_id' => 0 });
|
|
$vars->{'title'} = "Search For Test Cases";
|
|
$vars->{'case'} = $case;
|
|
$vars->{'components'} = get_searchable_objects('components');
|
|
$vars->{'categories'} = get_searchable_objects('categories');
|
|
$vars->{'bug_status'} = convert_bugzilla_fields('bug_status');
|
|
$vars->{'rep_platform'} = convert_bugzilla_fields('rep_platform');
|
|
$vars->{'op_sys'} = convert_bugzilla_fields('op_sys');
|
|
$vars->{'bug_priority'} = convert_bugzilla_fields('priority');
|
|
$vars->{'bug_severity'} = convert_bugzilla_fields('bug_severity');
|
|
$vars->{'resolution'} = convert_bugzilla_fields('resolution');
|
|
}
|
|
else{
|
|
print "Location: tr_show_product.cgi?search=1 \n\n";
|
|
exit;
|
|
}
|
|
$cgi->send_header;
|
|
$vars->{'report'} = $cgi->param('report');
|
|
$vars->{'current_tab'} = $tab;
|
|
$template->process("testopia/search/advanced.html.tmpl", $vars)
|
|
|| ThrowTemplateError($template->error());
|
|
}
|