RotherOSS-LongEscalationPerformanceBoost
6.0.1
Rother OSS GmbH
https://otrs.ch/
GNU GENERAL PUBLIC LICENSE Version 3, 29 June 2007
Initial.
Caches parts of the TicketEscalationDateCalculation to boost performance of TicketGet and other modules for long escalation times.
6.0.x
2019-11-08 17:22:56
opms
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjIuMCIgaW5pdD0iQXBwbGljYXRpb24iPgogICAgPFNldHRpbmcgTmFtZT0iVGlja2V0OjpDdXN0b21Nb2R1bGUjIyNSb3RoZXJPU1MtTG9uZ0VzY2FsYXRpb25QZXJmb3JtYW5jZUJvb3N0IiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBUcmFuc2xhdGFibGU9IjEiPkNhY2hlcyBwYXJ0cyBvZiB0aGUgVGlja2V0RXNjYWxhdGlvbkRhdGVDYWxjdWxhdGlvbiB0byBib29zdCBwZXJmb3JtYW5jZSBvZiBUaWNrZXRHZXQgYW5kIG90aGVyIG1vZHVsZXMgZm9yIGxvbmcgZXNjYWxhdGlvbiB0aW1lcy48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxOYXZpZ2F0aW9uPkNvcmU6OlRpY2tldDwvTmF2aWdhdGlvbj4KICAgICAgICA8VmFsdWU+CiAgICAgICAgICAgIDxJdGVtIFZhbHVlVHlwZT0iU3RyaW5nIiBWYWx1ZVJlZ2V4PSIiPktlcm5lbDo6U3lzdGVtOjpUaWNrZXQ6OlJvdGhlck9TU0VzY2FsYXRpb25EYXRlQ2FsYzwvSXRlbT4KICAgICAgICA8L1ZhbHVlPgogICAgPC9TZXR0aW5nPgo8L290cnNfY29uZmlnPgo=
# --
# Copyright (C) 2001-2019 OTRS AG, https://otrs.com/
# Copyright (C) 2019 Rother OSS GmbH, https://otrs.ch/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --
# ---
# RotherOSS LongEscalationPerformanceBoost
# ---
# nofilter(TidyAll::Plugin::OTRS::Perl::PerlCritic)
# ---
package Kernel::System::Ticket::RotherOSSEscalationDateCalc;

use strict;
use warnings;

our $ObjectManagerDisabled = 1;

# disable redefine warnings in this scope
{
    no warnings 'redefine';

    # redefine TicketEscalationIndexBuild() of Kernel::System::Ticket
    sub Kernel::System::Ticket::TicketEscalationDateCalculation {
        my ( $Self, %Param ) = @_;

        # check needed stuff
        for my $Needed (qw(Ticket UserID)) {
            if ( !defined $Param{$Needed} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "Need $Needed!"
                );
                return;
            }
        }

        # get ticket attributes
        my %Ticket = %{ $Param{Ticket} };

        # do no escalations on (merge|close|remove) tickets
        return if $Ticket{StateType} eq 'merged';
        return if $Ticket{StateType} eq 'closed';
        return if $Ticket{StateType} eq 'removed';

        # get escalation properties
        my %Escalation = $Self->TicketEscalationPreferences(
            Ticket => $Param{Ticket},
            UserID => $Param{UserID} || 1,
        );

        # return if we do not have any escalation attributes
        my %Map = (
            EscalationResponseTime => 'FirstResponse',
            EscalationUpdateTime   => 'Update',
            EscalationSolutionTime => 'Solution',
        );
        my $EscalationAttribute;
        KEY:
        for my $Key ( sort keys %Map ) {
            if ( $Escalation{ $Map{$Key} . 'Time' } ) {
                $EscalationAttribute = 1;
                last KEY;
            }
        }

        return if !$EscalationAttribute;

        # create datetime object
        my $DateTimeObject = $Kernel::OM->Create('Kernel::System::DateTime');

        # calculate escalation times based on escalation properties
        my %Data;

# Rother OSS / avoid long escalation calculations
        my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');
        my $MaxUncached = 60 * 60 * 24 * 3; # three days TODO: SysConfigOption and test for optimum
# EO Rother

        TIME:
        for my $Key ( sort keys %Map ) {

            next TIME if !$Ticket{$Key};

            # get time before or over escalation (escalation_destination_unixtime - now)
            my $TimeTillEscalation = $Ticket{$Key} - $DateTimeObject->ToEpoch();

            # ticket is not escalated till now ($TimeTillEscalation > 0)
            my $WorkingTime = 0;

# Rother OSS / avoid long escalation calculations
            my $StoredTime;
            my $EscalationTime = $Ticket{$Key};
            my $Escalated = ( $TimeTillEscalation > 0 ) ? 0 : 1;
            # if more than $MaxUncached time would have to be calculated, use the cache
            if ( abs( $TimeTillEscalation ) > $MaxUncached ) {
                $StoredTime = $CacheObject->Get(
                    Type => 'EscalationWorkingTime',
                    Key  => $Ticket{TicketID}.'::'.$Key.'::'.$Escalated,
                );
        
                # if the cache applies to the same escalation time...
                if ( $StoredTime && $StoredTime->{EscalationTime} == $EscalationTime ) {
                    if ( $Escalated ) {
                        $WorkingTime = $StoredTime->{WorkingTime};
                        $EscalationTime = $StoredTime->{LastTime};
                    }
                    # if it is not escalated only use the cache, if the last update was not too long ago
                    elsif( $TimeTillEscalation > $DateTimeObject->ToEpoch() - $StoredTime->{LastTime} ) {
                        $WorkingTime = $StoredTime->{WorkingTime};
                        $EscalationTime = 2*$DateTimeObject->ToEpoch() - $StoredTime->{LastTime};
                    }
                }
            }
# EO Rother

            if ( $TimeTillEscalation > 0 ) {

                my $StopTimeObj = $Kernel::OM->Create(
                    'Kernel::System::DateTime',
                    ObjectParams => {
# Rother OSS / avoid long escalation calculations
#                       Epoch => $Ticket{$Key}
                        Epoch => $EscalationTime,
# EO Rother
                    }
                );

                my $DeltaObj = $DateTimeObject->Delta(
                    DateTimeObject => $StopTimeObj,
                    ForWorkingTime => 1,
                    Calendar       => $Escalation{Calendar},
                );

# Rother OSS / avoid long escalation calculations
#               $WorkingTime = $DeltaObj ? $DeltaObj->{AbsoluteSeconds} : 0;
                # subtract time between points from last working time
                $WorkingTime -= $DeltaObj ? $DeltaObj->{AbsoluteSeconds} : 0;
                $WorkingTime = abs( $WorkingTime );
# EO Rother

                # extract needed data
                my $Notify = $Escalation{ $Map{$Key} . 'Notify' };
                my $Time   = $Escalation{ $Map{$Key} . 'Time' };

                # set notification if notify % is reached
                if ( $Notify && $Time ) {

                    my $Reached = 100 - ( $WorkingTime / ( $Time * 60 / 100 ) );

                    if ( $Reached >= $Notify ) {
                        $Data{ $Map{$Key} . 'TimeNotification' } = 1;
                    }
                }
            }

            # ticket is overtime ($TimeTillEscalation < 0)
            else {
                my $StartTimeObj = $Kernel::OM->Create(
                    'Kernel::System::DateTime',
                    ObjectParams => {
# Rother OSS / avoid long escalation calculations
#                       Epoch => $Ticket{$Key}
                        Epoch => $EscalationTime,
# EO Rother
                    }
                );

                my $DeltaObj = $StartTimeObj->Delta(
                    DateTimeObject => $DateTimeObject,
                    ForWorkingTime => 1,
                    Calendar       => $Escalation{Calendar},
                );

# Rother OSS / avoid long escalation calculations
#               $WorkingTime = 0;
#               if ( $DeltaObj && $DeltaObj->{AbsoluteSeconds} ) {
#                   $WorkingTime = '-' . $DeltaObj->{AbsoluteSeconds};
#               }
                $WorkingTime -= $DeltaObj ? $DeltaObj->{AbsoluteSeconds} : 0;
# EO Rother

                # set escalation
                $Data{ $Map{$Key} . 'TimeEscalation' } = 1;
            }

# Rother OSS / avoid long escalation calculations
            # cache the values for long timespans
            if ( abs( $TimeTillEscalation ) > $MaxUncached &&
                # and a successful calculation of $WorkingTime
                ( ( !$StoredTime && $WorkingTime ) || ( $WorkingTime != $StoredTime->{WorkingTime} ) ) ) {

                $CacheObject->Set(
                    Type  => 'EscalationWorkingTime',
                    Key   => $Ticket{TicketID}.'::'.$Key.'::'.$Escalated,
                    Value => {
                        EscalationTime => $Ticket{ $Key },
                        WorkingTime    => $WorkingTime,
                        LastTime       => $DateTimeObject->ToEpoch(),
                    },
                    TTL   => 60 * 60 * 24 * 5, # five days
                );
            }
# EO Rother

            my $DestinationDate = $Kernel::OM->Create(
                'Kernel::System::DateTime',
                ObjectParams => {
                    Epoch => $Ticket{$Key}
                }
            );

            $Data{ $Map{$Key} . 'TimeDestinationTime' } = $Ticket{$Key};
            $Data{ $Map{$Key} . 'TimeDestinationDate' } = $DestinationDate->ToString();
            $Data{ $Map{$Key} . 'TimeWorkingTime' }     = $WorkingTime;
            $Data{ $Map{$Key} . 'Time' }                = $TimeTillEscalation;

            # set global escalation attributes (set the escalation which is the first in time)
            if (
                !$Data{EscalationDestinationTime}
                || $Data{EscalationDestinationTime} > $Ticket{$Key}
                )
            {
                $Data{EscalationDestinationTime} = $Ticket{$Key};
                $Data{EscalationDestinationDate} = $DestinationDate->ToString();
                $Data{EscalationTimeWorkingTime} = $WorkingTime;
                $Data{EscalationTime}            = $TimeTillEscalation;

                # escalation time in readable way
                $Data{EscalationDestinationIn} = '';
                $WorkingTime = abs($WorkingTime);
                if ( $WorkingTime >= 3600 ) {
                    $Data{EscalationDestinationIn} .= int( $WorkingTime / 3600 ) . 'h ';
                    $WorkingTime = $WorkingTime
                        - ( int( $WorkingTime / 3600 ) * 3600 );    # remove already shown hours
                }
                if ( $WorkingTime <= 3600 || int( $WorkingTime / 60 ) ) {
                    $Data{EscalationDestinationIn} .= int( $WorkingTime / 60 ) . 'm';
                }
            }
        }

        return %Data;
    }

}

1;

