[Rt-commit] [rtir] 04/12: Wrap ACLEquivalenceObjects for tickets

Kevin Falcone falcone at bestpractical.com
Mon Apr 14 13:16:44 EDT 2014


This is an automated email from the git hooks/post-receive script.

falcone pushed a commit to branch 3.2/unwrap-hook-lexwrap
in repository rtir.

commit 7a07d8635442757298fc7dbd0f70d3fd037f1e14
Author: Kevin Falcone <falcone at bestpractical.com>
Date:   Wed Mar 26 13:58:22 2014 -0400

    Wrap ACLEquivalenceObjects for tickets
    
    Primary purpose of this method is to return two queues in the case that
    an Incident has a Constituency CF of EDUNET and the queue Incides - EDUNET
    exists.
    
    It has a lot of complexity to avoid running on non-IR queues
    and to try and cache whenever possible (although we no longer have a
    negative cache because of 1200cb16 which doesn't note the bug it fixes).
    
    While removing Hook::LexWrap and changing to return more traditionally,
    I also added a number of comments to document what I understand it to be
    doing and to note a few things that concern me and that can possibly be
    removed/cleaned up after tests pass.
---
 lib/RT/IR.pm | 77 +++++++++++++++++++++++++++++++++++++-----------------------
 1 file changed, 47 insertions(+), 30 deletions(-)

diff --git a/lib/RT/IR.pm b/lib/RT/IR.pm
index 061a88c..d2b71d8 100644
--- a/lib/RT/IR.pm
+++ b/lib/RT/IR.pm
@@ -656,46 +656,63 @@ if ( RT::IR->HasConstituency ) {
         return $self->RT::Record::_AddCustomFieldValue(@_);
     };
 
-    require RT::Ticket;
-    wrap 'RT::Ticket::ACLEquivalenceObjects', pre => sub {
+    # Tickets with Constituencies CF values that point to a Constituency
+    # Queue such as Incidents - EDUNET should return multiple Queues as
+    # EquivalenceObjects for checking rights. Constituency DutyTeams
+    # have their rights granted on the relevant Constituency Queues, so
+    # RT needs to know to check in both places for the combination of
+    # their rights.
+    my $orig_ACLEquivaluenceObjects = RT::Ticket->can('ACLEquivalenceObjects');
+    *RT::Ticket::ACLEquivalenceObjects = sub {
         my $self = shift;
 
         my $queue = RT::Queue->new(RT->SystemUser);
         $queue->Load($self->__Value('Queue'));
 
-        # We do this, rather than fall through to the orignal sub, as that
-        # interacts poorly with our overloaded QueueObj below
-        # Don't try and load Constituencies outside of RTIR.  It results
-        # in a lot of useless checks.
+        # For historical reasons, the System just returned a System Queue on this Ticket
+        # rather than delve into the overridden QueueObj.  It's possible this is no longer needed.
+        # We punt early on non-IR queues because otherwise we try to look for constituency Queues
+        # and the negative cache (_none) is disabled, leading to perf problems.
+        # It's also somewhat concerning that we return a SystemUser queue outside IR queues.
         if ( ( $self->CurrentUser->id == RT->SystemUser->id ) ||
              ( $queue->Name !~ /^(Incidents|Incident Reports|Investigations|Blocks)$/i ) ) {
-            $_[-1] =  [$queue];
-            return;
+            return [$queue];
         }
-        if ( UNIVERSAL::isa( $self, 'RT::Ticket' ) ) {
-            my $const = $RT::IR::ConstituencyCache{ $self->id };
-            if (!$const || $const eq '_none' ) {
-                my $systicket = RT::Ticket->new(RT->SystemUser);
-                $systicket->Load( $self->id );
-                $const = $RT::IR::ConstituencyCache{ $self->id } =
-                    $systicket->FirstCustomFieldValue('Constituency')
-                    || '_none';
-            }
-            return if $const eq '_none';
-            return if $RT::IR::HasNoQueueCache{ $const };
-
-            my $new_queue = RT::Queue->new(RT->SystemUser);
-            $new_queue->LoadByCols(
-                Name => $queue->Name . " - " . $const
-            );
-            unless ( $new_queue->id ) {
-                $RT::IR::HasNoQueueCache{$const} = 1;
-                return;
-            }
-            $_[-1] =  [$queue, $new_queue];
-        } else {
+
+        # Old old bulletproofing, can probably delete.
+        unless ( UNIVERSAL::isa( $self, 'RT::Ticket' ) ) {
             RT->Logger->crit("$self is not a ticket object like I expected");
+            return;
+        }
+
+        # We check both uncached constituencies and those explicitly negatively cached
+        # The commit that introduced this doesn't say what bug it was fixing, but it's definitely
+        # a perf hit.
+        my $const = $RT::IR::ConstituencyCache{ $self->id };
+        if (!$const || $const eq '_none' ) {
+            my $systicket = RT::Ticket->new(RT->SystemUser);
+            $systicket->Load( $self->id );
+            $const = $RT::IR::ConstituencyCache{ $self->id } =
+                $systicket->FirstCustomFieldValue('Constituency') || '_none';
+        }
+
+        # If this ticket still has no constituency, or the constituency it is assigned
+        # doesn't have a corresponding Constituency queue, return the default Queue.
+        if ( $const eq '_none' || $RT::IR::HasNoQueueCache{ $const } ) {
+            return $orig_ACLEquivaluenceObjects->($self, at _);
+        }
+
+        # Track that there is no Constituency queue for the Constituency recorded in the CF on this ticket.
+        my $constituency_queue = RT::Queue->new(RT->SystemUser);
+        $constituency_queue->LoadByCols( Name => $queue->Name . " - " . $const );
+        unless ( $constituency_queue->id ) {
+            $RT::IR::HasNoQueueCache{$const} = 1;
+            return $orig_ACLEquivaluenceObjects->($self, at _);
         }
+
+        # If a constituency queue such as 'Incidents - EDUNET' exists, we want to check
+        # ACLs on both of them.
+        return [$queue, $constituency_queue];
     };
 }
 

-- 
To stop receiving notification emails like this one, please contact
the administrator of this repository.


More information about the rt-commit mailing list