X-Git-Url: https://git.donarmstrong.com/?a=blobdiff_plain;f=Debbugs%2FBugs.pm;h=e8d162edf279bd0d8c381e3696c6e6d57fb6775a;hb=24a9fde3b533874d639fd79b2dd9b54f287e3ba0;hp=e80a2bcc5043a4295c179f75e1ac371f7fdb2888;hpb=e1282fbc363a4a713064123b5a8b3570b95b8816;p=debbugs.git diff --git a/Debbugs/Bugs.pm b/Debbugs/Bugs.pm index e80a2bc..e8d162e 100644 --- a/Debbugs/Bugs.pm +++ b/Debbugs/Bugs.pm @@ -1,3 +1,9 @@ +# This module is part of debbugs, and is released +# under the terms of the GPL version 2, or any later +# version at your option. +# See the file README and COPYING for more information. +# +# Copyright 2007 by Don Armstrong . package Debbugs::Bugs; @@ -40,18 +46,20 @@ BEGIN{ @EXPORT = (); %EXPORT_TAGS = (); - @EXPORT_OK = (qw(get_bugs)); + @EXPORT_OK = (qw(get_bugs count_bugs newest_bug bug_filter)); $EXPORT_TAGS{all} = [@EXPORT_OK]; } use Debbugs::Config qw(:config); use Params::Validate qw(validate_with :types); use IO::File; -use Debbugs::Status qw(splitpackages); +use Debbugs::Status qw(splitpackages get_bug_status); use Debbugs::Packages qw(getsrcpkgs); -use Debbugs::Common qw(getparsedaddrs getmaintainers getmaintainers_reverse); +use Debbugs::Common qw(getparsedaddrs getmaintainers getmaintainers_reverse make_list); use Fcntl qw(O_RDONLY); use MLDBM qw(DB_File Storable); +use List::Util qw(first); +use Carp; =head2 get_bugs @@ -98,7 +106,8 @@ searches are performed. =over =item archive -- whether to search archived bugs or normal bugs; -defaults to false. +defaults to false. As a special case, if archive is 'both', but +archived and unarchived bugs are returned. =item usertags -- set of usertags and the bugs they are applied to @@ -174,7 +183,7 @@ sub get_bugs{ bugs => {type => SCALAR|ARRAYREF, optional => 1, }, - archive => {type => BOOLEAN, + archive => {type => BOOLEAN|SCALAR, default => 0, }, usertags => {type => HASHREF, @@ -186,6 +195,13 @@ sub get_bugs{ # Normalize options my %options = %param; my @bugs; + if ($options{archive} eq 'both') { + push @bugs, get_bugs(%options,archive=>0); + push @bugs, get_bugs(%options,archive=>1); + my %bugs; + @bugs{@bugs} = @bugs; + return keys %bugs; + } # A configuration option will set an array that we'll use here instead. for my $routine (qw(Debbugs::Bugs::get_bugs_by_idx Debbugs::Bugs::get_bugs_flatfile)) { my ($package) = $routine =~ m/^(.+)\:\:/; @@ -214,6 +230,141 @@ sub get_bugs{ return @bugs; } +=head2 count_bugs + + count_bugs(function => sub {...}) + +Uses a subroutine to classify bugs into categories and return the +number of bugs which fall into those categories + +=cut + +sub count_bugs { + my %param = validate_with(params => \@_, + spec => {function => {type => CODEREF, + }, + archive => {type => BOOLEAN, + default => 0, + }, + }, + ); + my $flatfile; + if ($param{archive}) { + $flatfile = IO::File->new("$config{spool_dir}/index.archive", 'r') + or die "Unable to open $config{spool_dir}/index.archive for reading: $!"; + } + else { + $flatfile = IO::File->new("$config{spool_dir}/index.db", 'r') + or die "Unable to open $config{spool_dir}/index.db for reading: $!"; + } + my %count = (); + while(<$flatfile>) { + if (m/^(\S+)\s+(\d+)\s+(\d+)\s+(\S+)\s+\[\s*([^]]*)\s*\]\s+(\w+)\s+(.*)$/) { + my @x = $param{function}->(pkg => $1, + bug => $2, + status => $4, + submitter => $5, + severity => $6, + tags => $7, + ); + local $_; + $count{$_}++ foreach @x; + } + } + close $flatfile; + return %count; +} + +=head2 newest_bug + + my $bug = newest_bug(); + +Returns the bug number of the newest bug, which is nextnumber-1. + +=cut + +sub newest_bug { + my $nn_fh = IO::File->new("$config{spool_dir}/nextnumber",'r') + or die "Unable to open $config{spool_dir}nextnumber for reading: $!"; + local $/; + my $next_number = <$nn_fh>; + close $nn_fh; + chomp $next_number; + return $next_number-1; +} + +=head2 bug_filter + + bug_filter + +Allows filtering bugs on commonly used criteria + +=cut + +sub bug_filter { + my %param = validate_with(params => \@_, + spec => {bug => {type => SCALAR, + regex => qr/^\d+$/, + }, + status => {type => HASHREF, + optional => 1, + }, + seen_merged => {type => HASHREF, + optional => 1, + }, + repeat_merged => {type => BOOLEAN, + optional => 1, + }, + include => {type => HASHREF, + optional => 1, + }, + exclude => {type => HASHREF, + optional => 1, + }, + min_days => {type => SCALAR, + optional => 1, + }, + max_days => {type => SCALAR, + optional => 1, + }, + }, + ); + if (exists $param{repeat_merged} and + not $param{repeat_merged} and + not defined $param{seen_merged}) { + croak "repeat_merged false requires seen_merged to be passed"; + } + + if (not exists $param{status}) { + my $location = getbuglocation($param{bug}, 'summary'); + return 0 if not defined $location or not length $location; + $param{status} = readbug( $param{bug}, $location ); + return 0 if not defined $param{status}; + } + + if (exists $param{include}) { + return 1 if (!__bug_matches($param{include}, $param{status})); + } + if (exists $param{exclude}) { + return 1 if (__bug_matches($param{exclude}, $param{status})); + } + if (exists $param{repeat_merged} and not $param{repeat_merged}) { + my @merged = sort {$a<=>$b} $param{bug}, split(/ /, $param{status}{mergedwith}); + return 1 if first {defined $_} @{$param{seen_merged}}{@merged}; + @{$param{seen_merged}}{@merged} = (1) x @merged; + } + my $daysold = int((time - $param{status}{date}) / 86400); # seconds to days + if (exists $param{min_days}) { + return 1 unless $param{min_days} <= $daysold; + } + if (exists $param{max_days}) { + return 1 unless $param{max_days} == -1 or + $param{max_days} >= $daysold; + } + return 0; +} + + =head2 get_bugs_by_idx This routine uses the by-$index.idx indicies to try to speed up @@ -248,6 +399,12 @@ sub get_bugs_by_idx{ maint => {type => SCALAR|ARRAYREF, optional => 1, }, + bugs => {type => SCALAR|ARRAYREF, + optional => 1, + }, + usertags => {type => HASHREF, + optional => 1, + }, }, ); my %bugs = (); @@ -264,25 +421,41 @@ sub get_bugs_by_idx{ delete @param{qw(maint src)}; $param{package} = [@packages]; } - my $keys = keys(%param) - 1; + my $keys = grep {$_ !~ /^(archive|usertags|bugs)$/} keys(%param); die "Need at least 1 key to search by" unless $keys; my $arc = $param{archive} ? '-arc':''; my %idx; - for my $key (grep {$_ ne 'archive'} keys %param) { + for my $key (grep {$_ !~ /^(archive|usertags|bugs)$/} keys %param) { my $index = $key; $index = 'submitter-email' if $key eq 'submitter'; $index = "$config{spool_dir}/by-${index}${arc}.idx"; tie(%idx, MLDBM => $index, O_RDONLY) or die "Unable to open $index: $!"; - for my $search (__make_list($param{$key})) { + my %bug_matching = (); + for my $search (make_list($param{$key})) { next unless defined $idx{$search}; for my $bug (keys %{$idx{$search}}) { + next if $bug_matching{$bug}; # increment the number of searches that this bug matched $bugs{$bug}++; + $bug_matching{$bug}=1; + } + } + if ($key eq 'tag' and exists $param{usertags}) { + for my $bug (make_list(grep {defined $_ } @{$param{usertags}}{make_list($param{tag})})) { + next if $bug_matching{$bug}; + $bugs{$bug}++; + $bug_matching{$bug}=1; } } untie %idx or die 'Unable to untie %idx'; } + if ($param{bugs}) { + $keys++; + for my $bug (make_list($param{bugs})) { + $bugs{$bug}++; + } + } # Throw out results that do not match all of the search specifications return map {$keys <= $bugs{$_}?($_):()} keys %bugs; } @@ -338,21 +511,19 @@ sub get_bugs_flatfile{ ); my $flatfile; if ($param{archive}) { - $flatfile = new IO::File "$debbugs::gSpoolDir/index.archive", 'r' - or die "Unable to open $debbugs::gSpoolDir/index.archive for reading: $!"; + $flatfile = IO::File->new("$config{spool_dir}/index.archive", 'r') + or die "Unable to open $config{spool_dir}/index.archive for reading: $!"; } else { - $flatfile = new IO::File "$debbugs::gSpoolDir/index.db", 'r' - or die "Unable to open $debbugs::gSpoolDir/index.db for reading: $!"; + $flatfile = IO::File->new("$config{spool_dir}/index.db", 'r') + or die "Unable to open $config{spool_dir}/index.db for reading: $!"; } my %usertag_bugs; if (exists $param{tag} and exists $param{usertags}) { - # This complex slice makes a hash with the bugs which have the # usertags passed in $param{tag} set. - @usertag_bugs{map {@{$_}} - @{$param{usertags}}{__make_list($param{tag})} - } = (1) x @{$param{usertags}}{__make_list($param{tag})} + @usertag_bugs{make_list(@{$param{usertags}}{make_list($param{tag})}) + } = (1) x make_list(@{$param{usertags}}{make_list($param{tag})}); } # We handle src packages, maint and maintenc by mapping to the # appropriate binary packages, then removing all packages which @@ -370,15 +541,15 @@ sub get_bugs_flatfile{ while (<$flatfile>) { next unless m/^(\S+)\s+(\d+)\s+(\d+)\s+(\S+)\s+\[\s*([^]]*)\s*\]\s+(\w+)\s+(.*)$/; my ($pkg,$bug,$time,$status,$submitter,$severity,$tags) = ($1,$2,$3,$4,$5,$6,$7); - next if exists $param{bug} and not grep {$bug == $_} __make_list($param{bugs}); - if (exists $param{pkg}) { + next if exists $param{bugs} and not grep {$bug == $_} make_list($param{bugs}); + if (exists $param{package}) { my @packages = splitpackages($pkg); next unless grep { my $pkg_list = $_; - grep {$pkg_list eq $_} __make_list($param{pkg}) + grep {$pkg_list eq $_} make_list($param{package}) } @packages; } if (exists $param{src}) { - my @src_packages = map { getsrcpkgs($_)} __make_list($param{src}); + my @src_packages = map { getsrcpkgs($_)} make_list($param{src}); my @packages = splitpackages($pkg); next unless grep { my $pkg_list = $_; grep {$pkg_list eq $_} @packages @@ -387,22 +558,22 @@ sub get_bugs_flatfile{ if (exists $param{submitter}) { my @p_addrs = map {lc($_->address)} map {getparsedaddrs($_)} - __make_list($param{submitter}); + make_list($param{submitter}); my @f_addrs = map {$_->address} getparsedaddrs($submitter||''); next unless grep { my $f_addr = $_; grep {$f_addr eq $_} @p_addrs } @f_addrs; } - next if exists $param{severity} and not grep {$severity eq $_} __make_list($param{severity}); - next if exists $param{status} and not grep {$status eq $_} __make_list($param{status}); + next if exists $param{severity} and not grep {$severity eq $_} make_list($param{severity}); + next if exists $param{status} and not grep {$status eq $_} make_list($param{status}); if (exists $param{tag}) { my $bug_ok = 0; # either a normal tag, or a usertag must be set $bug_ok = 1 if exists $param{usertags} and $usertag_bugs{$bug}; my @bug_tags = split ' ', $tags; $bug_ok = 1 if grep {my $bug_tag = $_; - grep {$bug_tag eq $_} __make_list($param{tag}); + grep {$bug_tag eq $_} make_list($param{tag}); } @bug_tags; next unless $bug_ok; } @@ -425,6 +596,18 @@ sub get_bugs_flatfile{ return @bugs; } +=head1 PRIVATE FUNCTIONS + +=head2 __handle_pkg_src_and_maint + + my @packages = __handle_pkg_src_and_maint(map {exists $param{$_}?($_,$param{$_}):()} + qw(package src maint) + ); + +Turn package/src/maint into a list of packages + +=cut + sub __handle_pkg_src_and_maint{ my %param = validate_with(params => \@_, spec => {package => {type => SCALAR|ARRAYREF, @@ -440,7 +623,8 @@ sub __handle_pkg_src_and_maint{ allow_extra => 1, ); - my @packages = __make_list($param{package}); + my @packages; + @packages = make_list($param{package}) if exists $param{package}; my $package_keys = @packages?1:0; my %packages; @packages{@packages} = (1) x @packages; @@ -448,7 +632,7 @@ sub __handle_pkg_src_and_maint{ # We only want to increment the number of keys if there is # something to match my $key_inc = 0; - for my $package ((map { getsrcpkgs($_)} __make_list($param{src})),__make_list($param{src})) { + for my $package ((map { getsrcpkgs($_)} make_list($param{src})),make_list($param{src})) { $packages{$package}++; $key_inc=1; } @@ -458,7 +642,7 @@ sub __handle_pkg_src_and_maint{ my $key_inc = 0; my $maint_rev = getmaintainers_reverse(); for my $package (map { exists $maint_rev->{$_}?@{$maint_rev->{$_}}:()} - __make_list($param{maint})) { + make_list($param{maint})) { $packages{$package}++; $key_inc = 1; } @@ -467,14 +651,52 @@ sub __handle_pkg_src_and_maint{ return grep {$packages{$_} >= $package_keys} keys %packages; } +my %field_match = ( + 'subject' => \&__contains_field_match, + 'tags' => sub { + my ($field, $values, $status) = @_; + my %values = map {$_=>1} @$values; + foreach my $t (split /\s+/, $status->{$field}) { + return 1 if (defined $values{$t}); + } + return 0; + }, + 'severity' => \&__exact_field_match, + 'pending' => \&__exact_field_match, + 'originator' => \&__contains_field_match, + 'forwarded' => \&__contains_field_match, + 'owner' => \&__contains_field_match, +); + +sub __bug_matches { + my ($hash, $status) = @_; + foreach my $key( keys( %$hash ) ) { + my $value = $hash->{$key}; + my $sub = $field_match{$key}; + return 1 if ($sub->($key, $value, $status)); + } + return 0; +} + +sub __exact_field_match { + my ($field, $values, $status) = @_; + my @values = @$values; + my @ret = grep {$_ eq $status->{$field} } @values; + $#ret != -1; +} -# This private subroutine takes a scalar and turns it into a list; -# transforming arrayrefs into their contents along the way. It also -# turns undef into the empty list. -sub __make_list{ - return map {defined $_?(ref($_) eq 'ARRAY'?@{$_}:$_):()} @_; +sub __contains_field_match { + my ($field, $values, $status) = @_; + foreach my $data (@$values) { + return 1 if (index($status->{$field}, $data) > -1); + } + return 0; } + + + + 1; __END__