sanitycheck.cgi 6.04 KB
Newer Older
1 2
#!/usr/bonsaitools/bin/perl -w
# -*- Mode: perl; indent-tabs-mode: nil -*-
3
#
4 5 6 7 8 9 10 11 12 13
# 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.
#
14
# The Original Code is the Bugzilla Bug Tracking System.
15
#
16
# The Initial Developer of the Original Code is Netscape Communications
17 18 19 20
# Corporation. Portions created by Netscape are
# Copyright (C) 1998 Netscape Communications Corporation. All
# Rights Reserved.
#
21 22
# Contributor(s): Terry Weissman <terry@mozilla.org>

23 24
use diagnostics;
use strict;
25

26
require "CGI.pl";
27

28 29
use vars %::FORM;

30 31
print "Content-type: text/html\n";
print "\n";
32

33 34
ConnectToDatabase();

35
my $offervotecacherebuild = 0;
36 37 38 39

sub Status {
    my ($str) = (@_);
    print "$str <P>\n";
40 41
}

42 43 44
sub Alert {
    my ($str) = (@_);
    Status("<font color=red>$str</font>");
45 46
}

47 48
sub BugLink {
    my ($id) = (@_);
49
    return "<a href='show_bug.cgi?id=$id'>$id</a>";
50 51
}

52 53 54 55 56
sub AlertBadVoteCache {
    my ($id) = (@_);
    Alert("Bad vote cache for bug " . BugLink($id));
    $offervotecacherebuild = 1;
}
57

58

59
my @row;
60 61
my @checklist;

62 63 64 65 66
PutHeader("Bugzilla Sanity Check");

if (exists $::FORM{'rebuildvotecache'}) {
    Status("OK, now rebuilding vote cache.");
    SendSQL("lock tables bugs write, votes read");
67
    SendSQL("update bugs set votes = 0, delta_ts=delta_ts");
68 69 70 71 72 73 74
    SendSQL("select bug_id, sum(count) from votes group by bug_id");
    my %votes;
    while (@row = FetchSQLData()) {
        my ($id, $v) = (@row);
        $votes{$id} = $v;
    }
    foreach my $id (keys %votes) {
75
        SendSQL("update bugs set votes = $votes{$id}, delta_ts=delta_ts where bug_id = $id");
76 77
    }
    SendSQL("unlock tables");
78
    Status("Vote cache has been rebuilt.");
79 80 81 82
}

print "OK, now running sanity checks.<P>\n";

83 84 85 86 87 88 89 90
Status("Checking groups");
SendSQL("select bit from groups where bit != pow(2, round(log(bit) / log(2)))");
while (my $bit = FetchOneColumn()) {
    Alert("Illegal bit number found in group table: $bit");
}
    
SendSQL("select sum(bit) from groups where isbuggroup != 0");
my $buggroupset = FetchOneColumn();
terry%mozilla.org's avatar
terry%mozilla.org committed
91
if (!defined $buggroupset || $buggroupset eq "") {
92 93
    $buggroupset = 0;
}
94 95 96 97 98 99 100 101
SendSQL("select bug_id, groupset from bugs where groupset & $buggroupset != groupset");
while (@row = FetchSQLData()) {
    Alert("Bad groupset $row[1] found in bug " . BugLink($row[0]));
}




102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117
Status("Checking version/products");

SendSQL("select distinct product, version from bugs");
while (@row = FetchSQLData()) {
    my @copy = @row;
    push(@checklist, \@copy);
}

foreach my $ref (@checklist) {
    my ($product, $version) = (@$ref);
    SendSQL("select count(*) from versions where program = '$product' and value = '$version'");
    if (FetchOneColumn() != 1) {
        Alert("Bug(s) found with invalid product/version: $product/$version");
    }
}

118 119 120

Status("Checking components/products");

121
@checklist = ();
122 123 124 125 126 127 128 129 130 131 132 133 134 135 136
SendSQL("select distinct product, component from bugs");
while (@row = FetchSQLData()) {
    my @copy = @row;
    push(@checklist, \@copy);
}

foreach my $ref (@checklist) {
    my ($product, $component) = (@$ref);
    SendSQL("select count(*) from components where program = '$product' and value = '$component'");
    if (FetchOneColumn() != 1) {
        Alert("Bug(s) found with invalid product/component: $product/$component");
    }
}


137
Status("Checking profile ids...");
138

139
SendSQL("select userid,login_name from profiles");
140

141
my %profid;
142

143 144 145 146
while (@row = FetchSQLData()) {
    my ($id, $email) = (@row);
    if ($email =~ /^[^@, ]*@[^@, ]*\.[^@, ]*$/) {
        $profid{$id} = 1;
147
    } else {
148
        Alert "Bad profile id $id &lt;$email&gt;."
149 150 151 152
    }
}


153
undef $profid{0};
154 155


156
Status("Checking reporter/assigned_to/qa_contact ids");
157
SendSQL("select bug_id,reporter,assigned_to,qa_contact,votes from bugs");
158

159
my %votes;
160 161 162
my %bugid;

while (@row = FetchSQLData()) {
163
    my($id, $reporter, $assigned_to, $qa_contact, $v) = (@row);
164 165 166
    $bugid{$id} = 1;
    if (!defined $profid{$reporter}) {
        Alert("Bad reporter $reporter in " . BugLink($id));
167
    }
168 169
    if (!defined $profid{$assigned_to}) {
        Alert("Bad assigned_to $assigned_to in" . BugLink($id));
170
    }
171 172 173
    if ($qa_contact != 0 && !defined $profid{$qa_contact}) {
        Alert("Bad qa_contact $qa_contact in" . BugLink($id));
    }
174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194
    if ($v != 0) {
        $votes{$id} = $v;
    }
}

Status("Checking cached vote counts");
SendSQL("select bug_id, sum(count) from votes group by bug_id");

while (@row = FetchSQLData()) {
    my ($id, $v) = (@row);
    if ($v <= 0) {
        Alert("Bad vote sum for bug $id");
    } else {
        if (!defined $votes{$id} || $votes{$id} != $v) {
            AlertBadVoteCache($id);
        }
        delete $votes{$id};
    }
}
foreach my $id (keys %votes) {
    AlertBadVoteCache($id);
195 196
}

197 198 199 200 201
if ($offervotecacherebuild) {
    print qq{<a href="sanitycheck.cgi?rebuildvotecache=1">Click here to rebuild the vote cache</a><p>\n};
}


202
Status("Checking CC table");
203

204 205 206 207 208
SendSQL("select bug_id,who from cc");
while (@row = FetchSQLData()) {
    my ($id, $cc) = (@row);
    if (!defined $profid{$cc}) {
        Alert("Bad cc $cc in " . BugLink($id));
209 210 211 212
    }
}


213
Status("Checking activity table");
214

215
SendSQL("select bug_id,who from bugs_activity");
216

217 218 219 220
while (@row = FetchSQLData()) {
    my ($id, $who) = (@row);
    if (!defined $bugid{$id}) {
        Alert("Bad bugid " . BugLink($id));
221
    }
222 223
    if (!defined $profid{$who}) {
        Alert("Bad who $who in " . BugLink($id));
224 225
    }
}
226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243


Status("Checking dependency table");

SendSQL("select blocked, dependson from dependencies");
while (@row = FetchSQLData()) {
    my ($blocked, $dependson) = (@row);
    if (!defined $bugid{$blocked}) {
        Alert("Bad blocked " . BugLink($blocked));
    }
    if (!defined $bugid{$dependson}) {
        Alert("Bad dependson " . BugLink($dependson));
    }
}



Status("Sanity check completed.");