request.cgi 12.7 KB
Newer Older
1
#!/usr/bin/perl -wT
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
# -*- 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>
22
#                 Frédéric Buclin <LpSolit@gmail.com>
23 24 25 26 27 28 29 30 31

################################################################################
# Script Initialization
################################################################################

# Make it harder for us to do dangerous things in Perl.
use strict;

use lib qw(.);
32
require "globals.pl";
33
use Bugzilla;
34 35 36
use Bugzilla::Flag;
use Bugzilla::FlagType;
use Bugzilla::User;
37 38
use Bugzilla::Product;
use Bugzilla::Component;
39 40

# Make sure the user is logged in.
41 42 43 44
my $user = Bugzilla->login();
my $userid = $user->id;

my $cgi = Bugzilla->cgi;
45 46
my $template = Bugzilla->template;
my $vars = {};
47

48 49 50 51 52

################################################################################
# Main Body Execution
################################################################################

53 54 55 56 57 58 59 60 61 62 63 64
my $fields;
$fields->{'requester'}->{'type'} = 'single';
# If the user doesn't restrict his search to requests from the wind
# (requestee ne '-'), include the requestee for completion.
unless (defined $cgi->param('requestee')
        && $cgi->param('requestee') eq '-')
{
    $fields->{'requestee'}->{'type'} = 'single';
}

Bugzilla::User::match_field($cgi, $fields);

65 66 67 68 69 70 71 72
queue();
exit;

################################################################################
# Functions
################################################################################

sub queue {
73
    my $cgi = Bugzilla->cgi;
74
    my $dbh = Bugzilla->dbh;
75 76 77 78

    my $status = validateStatus($cgi->param('status'));
    my $form_group = validateGroup($cgi->param('group'));

79 80
    my $attach_join_clause = "flags.attach_id = attachments.attach_id";
    if (Param("insidergroup") && !UserInGroup(Param("insidergroup"))) {
81
        $attach_join_clause .= " AND attachments.isprivate < 1";
82 83 84 85 86 87 88 89 90 91 92 93
    }

    my $query = 
    # Select columns describing each flag, the bug/attachment on which
    # it has been set, who set it, and of whom they are requesting it.
    " SELECT    flags.id, flagtypes.name,
                flags.status,
                flags.bug_id, bugs.short_desc,
                products.name, components.name,
                flags.attach_id, attachments.description,
                requesters.realname, requesters.login_name,
                requestees.realname, requestees.login_name,
94
    " . $dbh->sql_date_format('flags.creation_date', '%Y.%m.%d %H:%i') .
95 96 97 98
    # Use the flags and flagtypes tables for information about the flags,
    # the bugs and attachments tables for target info, the profiles tables
    # for setter and requestee info, the products/components tables
    # so we can display product and component names, and the bug_group_map
99 100 101 102 103
    # table to help us weed out secure bugs to which the user should not have
    # access.
    "
      FROM           flags 
           LEFT JOIN attachments
104
                  ON ($attach_join_clause)
105
          INNER JOIN flagtypes
106
                  ON flags.type_id = flagtypes.id
107
          INNER JOIN profiles AS requesters
108
                  ON flags.setter_id = requesters.userid
109
           LEFT JOIN profiles AS requestees
110
                  ON flags.requestee_id  = requestees.userid
111
          INNER JOIN bugs
112
                  ON flags.bug_id = bugs.bug_id
113
          INNER JOIN products
114
                  ON bugs.product_id = products.id
115
          INNER JOIN components
116
                  ON bugs.component_id = components.id
117
           LEFT JOIN bug_group_map AS bgmap
118
                  ON bgmap.bug_id = bugs.bug_id
119
                 AND bgmap.group_id NOT IN (" .
120
                     join(', ', (-1, values(%{$user->groups}))) . ")
121
           LEFT JOIN cc AS ccmap
122 123
                  ON ccmap.who = $userid
                 AND ccmap.bug_id = bugs.bug_id
124 125 126 127 128
    " .

    # Weed out bug the user does not have access to
    " WHERE     ((bgmap.group_id IS NULL) OR
                 (ccmap.who IS NOT NULL AND cclist_accessible = 1) OR
129
                 (bugs.reporter = $userid AND bugs.reporter_accessible = 1) OR
130 131 132
                 (bugs.assigned_to = $userid) " .
                 (Param('useqacontact') ? "OR
                 (bugs.qa_contact = $userid))" : ")");
133
    
134
    # Limit query to pending requests.
135
    $query .= " AND flags.status = '?' " unless $status;
136 137 138 139

    # The set of criteria by which we filter records to display in the queue.
    my @criteria = ();
    
140 141 142 143 144 145 146 147 148
    # A list of columns to exclude from the report because the report conditions
    # limit the data being displayed to exact matches for those columns.
    # In other words, if we are only displaying "pending" , we don't
    # need to display a "status" column in the report because the value for that
    # column will always be the same.
    my @excluded_columns = ();
    
    # Filter requests by status: "pending", "granted", "denied", "all" 
    # (which means any), or "fulfilled" (which means "granted" or "denied").
149 150
    if ($status) {
        if ($status eq "+-") {
151
            push(@criteria, "flags.status IN ('+', '-')");
152
            push(@excluded_columns, 'status') unless $cgi->param('do_union');
153
        }
154 155
        elsif ($status ne "all") {
            push(@criteria, "flags.status = '$status'");
156
            push(@excluded_columns, 'status') unless $cgi->param('do_union');
157
        }
158 159 160
    }
    
    # Filter results by exact email address of requester or requestee.
161
    if (defined $cgi->param('requester') && $cgi->param('requester') ne "") {
162 163 164
        my $requester = $dbh->quote($cgi->param('requester'));
        trick_taint($requester); # Quoted above
        push(@criteria, $dbh->sql_istrcmp('requesters.login_name', $requester));
165
        push(@excluded_columns, 'requester') unless $cgi->param('do_union');
166
    }
167
    if (defined $cgi->param('requestee') && $cgi->param('requestee') ne "") {
168
        if ($cgi->param('requestee') ne "-") {
169 170
            my $requestee = $dbh->quote($cgi->param('requestee'));
            trick_taint($requestee); # Quoted above
171
            push(@criteria, $dbh->sql_istrcmp('requestees.login_name',
172
                            $requestee));
173 174
        }
        else { push(@criteria, "flags.requestee_id IS NULL") }
175
        push(@excluded_columns, 'requestee') unless $cgi->param('do_union');
176 177 178
    }
    
    # Filter results by exact product or component.
179
    if (defined $cgi->param('product') && $cgi->param('product') ne "") {
180 181 182 183 184 185 186 187
        my $product = Bugzilla::Product::check_product(scalar $cgi->param('product'));
        push(@criteria, "bugs.product_id = " . $product->id);
        push(@excluded_columns, 'product') unless $cgi->param('do_union');
        if (defined $cgi->param('component') && $cgi->param('component') ne "") {
            my $component =
                Bugzilla::Component::check_component($product, scalar $cgi->param('component'));
            push(@criteria, "bugs.component_id = " . $component->id);
            push(@excluded_columns, 'component') unless $cgi->param('do_union');
188 189
        }
    }
190

191
    # Filter results by flag types.
192 193
    my $form_type = $cgi->param('type');
    if (defined $form_type && !grep($form_type eq $_, ("", "all"))) {
194 195
        # Check if any matching types are for attachments.  If not, don't show
        # the attachment column in the report.
196
        my $types = Bugzilla::FlagType::match({ 'name' => $form_type });
197 198 199 200 201 202 203 204
        my $has_attachment_type = 0;
        foreach my $type (@$types) {
            if ($type->{'target_type'} eq "attachment") {
                $has_attachment_type = 1;
                last;
            }
        }
        if (!$has_attachment_type) { push(@excluded_columns, 'attachment') }
205 206 207 208

        my $quoted_form_type = $dbh->quote($form_type);
        trick_taint($quoted_form_type); # Already SQL quoted
        push(@criteria, "flagtypes.name = " . $quoted_form_type);
209
        push(@excluded_columns, 'type') unless $cgi->param('do_union');
210 211
    }
    
212 213 214
    # Add the criteria to the query.  We do an intersection by default 
    # but do a union if the "do_union" URL parameter (for which there is no UI 
    # because it's an advanced feature that people won't usually want) is true.
215
    my $and_or = $cgi->param('do_union') ? " OR " : " AND ";
216 217
    $query .= " AND (" . join($and_or, @criteria) . ") " if scalar(@criteria);
    
218 219 220
    # Group the records by flag ID so we don't get multiple rows of data
    # for each flag.  This is only necessary because of the code that
    # removes flags on bugs the user is unauthorized to access.
221 222 223 224 225 226 227 228
    $query .= ' ' . $dbh->sql_group_by('flags.id',
               'flagtypes.name, flags.status, flags.bug_id, bugs.short_desc,
                products.name, components.name, flags.attach_id,
                attachments.description, requesters.realname,
                requesters.login_name, requestees.realname,
                requestees.login_name, flags.creation_date,
                cclist_accessible, bugs.reporter, bugs.reporter_accessible,
                bugs.assigned_to');
229 230 231 232

    # Group the records, in other words order them by the group column
    # so the loop in the display template can break them up into separate
    # tables every time the value in the group column changes.
233 234 235

    $form_group ||= "requestee";
    if ($form_group eq "requester") {
236 237
        $query .= " ORDER BY requesters.realname, requesters.login_name";
    }
238
    elsif ($form_group eq "requestee") {
239 240
        $query .= " ORDER BY requestees.realname, requestees.login_name";
    }
241
    elsif ($form_group eq "category") {
242 243
        $query .= " ORDER BY products.name, components.name";
    }
244
    elsif ($form_group eq "type") {
245 246 247 248 249 250 251 252
        $query .= " ORDER BY flagtypes.name";
    }

    # Order the records (within each group).
    $query .= " , flags.creation_date";
    
    # Pass the query to the template for use when debugging this script.
    $vars->{'query'} = $query;
253
    $vars->{'debug'} = $cgi->param('debug') ? 1 : 0;
254
    
255
    my $results = $dbh->selectall_arrayref($query);
256
    my @requests = ();
257 258
    foreach my $result (@$results) {
        my @data = @$result;
259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276
        my $request = {
          'id'              => $data[0] , 
          'type'            => $data[1] , 
          'status'          => $data[2] , 
          'bug_id'          => $data[3] , 
          'bug_summary'     => $data[4] , 
          'category'        => "$data[5]: $data[6]" , 
          'attach_id'       => $data[7] , 
          'attach_summary'  => $data[8] ,
          'requester'       => ($data[9] ? "$data[9] <$data[10]>" : $data[10]) , 
          'requestee'       => ($data[11] ? "$data[11] <$data[12]>" : $data[12]) , 
          'created'         => $data[13]
        };
        push(@requests, $request);
    }

    # Get a list of request type names to use in the filter form.
    my @types = ("all");
277 278 279
    my $flagtypes = $dbh->selectcol_arrayref(
                         "SELECT DISTINCT(name) FROM flagtypes ORDER BY name");
    push(@types, @$flagtypes);
280
    
281
    $vars->{'products'} = $user->get_selectable_products;
282
    $vars->{'excluded_columns'} = \@excluded_columns;
283
    $vars->{'group_field'} = $form_group;
284 285 286 287
    $vars->{'requests'} = \@requests;
    $vars->{'types'} = \@types;

    # Return the appropriate HTTP response headers.
288
    print Bugzilla->cgi->header();
289 290 291 292 293 294 295 296 297 298 299

    # Generate and return the UI (HTML page) from the appropriate template.
    $template->process("request/queue.html.tmpl", $vars)
      || ThrowTemplateError($template->error());
}

################################################################################
# Data Validation / Security Authorization
################################################################################

sub validateStatus {
300
    my $status = shift;
301
    return if !defined $status;
302

303
    grep($status eq $_, qw(? +- + - all))
304
      || ThrowCodeError("flag_status_invalid",
305
                        { status => $status });
306 307
    trick_taint($status);
    return $status;
308 309 310
}

sub validateGroup {
311
    my $group = shift;
312
    return if !defined $group;
313

314
    grep($group eq $_, qw(requester requestee category type))
315
      || ThrowCodeError("request_queue_group_invalid", 
316
                        { group => $group });
317 318
    trick_taint($group);
    return $group;
319 320
}