diff options
author | Dylan William Hardison <dylan@hardison.net> | 2016-09-11 16:10:35 +0200 |
---|---|---|
committer | Dylan William Hardison <dylan@hardison.net> | 2017-02-26 02:46:14 +0100 |
commit | d3a58e043a804feacc1d401dd4e1be02c56fc93d (patch) | |
tree | 147a0fd604c7d0b76e0af06a735998f80f52c2ef | |
parent | 84ebf678974927ed94561b34c2f8a89f217d08a4 (diff) | |
download | bugzilla-d3a58e043a804feacc1d401dd4e1be02c56fc93d.tar.gz bugzilla-d3a58e043a804feacc1d401dd4e1be02c56fc93d.tar.xz |
Bug 1301951 - Fix Bugzilla::Bug memory leaks
-rw-r--r-- | Bugzilla.pm | 1 | ||||
-rw-r--r-- | Bugzilla/Attachment.pm | 9 | ||||
-rw-r--r-- | Bugzilla/Bug.pm | 21 | ||||
-rw-r--r-- | Bugzilla/Comment.pm | 9 |
4 files changed, 35 insertions, 5 deletions
diff --git a/Bugzilla.pm b/Bugzilla.pm index ada2cc326..6ad129412 100644 --- a/Bugzilla.pm +++ b/Bugzilla.pm @@ -686,6 +686,7 @@ sub _cleanup { my $smtp = $cache->{smtp}; $smtp->disconnect if $smtp; clear_request_cache(); + Bugzilla::Bug->CLEANUP(); # These are both set by CGI.pm but need to be undone so that # Apache can actually shut down its children if it needs to. diff --git a/Bugzilla/Attachment.pm b/Bugzilla/Attachment.pm index ec318b021..7d96beeda 100644 --- a/Bugzilla/Attachment.pm +++ b/Bugzilla/Attachment.pm @@ -46,6 +46,7 @@ use Bugzilla::Hook; use File::Copy; use List::Util qw(max); +use Scalar::Util qw(weaken isweak); use Storable qw(dclone); use parent qw(Bugzilla::Object); @@ -140,8 +141,14 @@ the bug object to which the attachment is attached =cut sub bug { + my ($self) = @_; require Bugzilla::Bug; - return $_[0]->{bug} //= Bugzilla::Bug->new({ id => $_[0]->bug_id, cache => 1 }); + return $self->{bug} if defined $self->{bug}; + + # note $bug exists as a strong reference to keep $self->{bug} defined until the end of this method + my $bug = $self->{bug} = Bugzilla::Bug->new({ id => $_[0]->bug_id, cache => 1 }); + weaken($self->{bug}) unless isweak($self->{bug}); + return $bug; } =over diff --git a/Bugzilla/Bug.pm b/Bugzilla/Bug.pm index f039ca759..38a27836a 100644 --- a/Bugzilla/Bug.pm +++ b/Bugzilla/Bug.pm @@ -34,7 +34,7 @@ use Bugzilla::BugUserLastVisit; use List::MoreUtils qw(firstidx uniq part); use List::Util qw(min max first); use Storable qw(dclone); -use Scalar::Util qw(blessed); +use Scalar::Util qw(blessed weaken); use parent qw(Bugzilla::Object Exporter); @Bugzilla::Bug::EXPORT = qw( @@ -43,6 +43,9 @@ use parent qw(Bugzilla::Object Exporter); editable_bug_fields ); +# This hash keeps a weak copy of every bug created. +my %CLEANUP; + ##################################################################### # Constants ##################################################################### @@ -361,6 +364,9 @@ sub new { return $error_self; } + $CLEANUP{$self->id} = $self; + weaken($CLEANUP{$self->id}); + return $self; } @@ -375,6 +381,18 @@ sub object_cache_key { return $key . ',' . Bugzilla->user->id; } +# This is called by Bugzilla::_cleanup() at the end of requests in a persistent environment +# (such as mod_perl) +sub CLEANUP { + foreach my $bug (values %CLEANUP) { + # $bug will be undef if there are no other references to it. + next unless $bug; + delete $bug->{depends_on_obj}; + delete $bug->{blocks_obj}; + } + %CLEANUP = (); +} + sub check { my $class = shift; my ($param, $field) = @_; @@ -3696,6 +3714,7 @@ sub comments { foreach my $comment (@{ $self->{'comments'} }) { $comment->{count} = $count++; $comment->{bug} = $self; + weaken($comment->{bug}); # XXX - hack for MySQL. Convert [U+....] back into its Unicode # equivalent for characters above U+FFFF as MySQL older than 5.5.3 # cannot store them, see Bugzilla::Comment::_check_thetext(). diff --git a/Bugzilla/Comment.pm b/Bugzilla/Comment.pm index d995d4509..2f063934a 100644 --- a/Bugzilla/Comment.pm +++ b/Bugzilla/Comment.pm @@ -21,7 +21,7 @@ use Bugzilla::User; use Bugzilla::Util; use List::Util qw(first); -use Scalar::Util qw(blessed); +use Scalar::Util qw(blessed weaken isweak); ############################### #### Initialization #### @@ -232,8 +232,11 @@ sub collapsed { sub bug { my $self = shift; require Bugzilla::Bug; - $self->{bug} ||= new Bugzilla::Bug($self->bug_id); - return $self->{bug}; + + # note $bug exists as a strong reference to keep $self->{bug} defined until the end of this method + my $bug = $self->{bug} ||= new Bugzilla::Bug($self->bug_id); + weaken($self->{bug}) unless isweak($self->{bug}); + return $bug; } sub is_about_attachment { |