All Downloads are FREE. Search and download functionalities are using the official Maven repository.

progress.Consultingwerk.Util.ErrorHelper.cls Maven / Gradle / Ivy

There is a newer version: 229
Show newest version
/**********************************************************************
 * Copyright (C) 2006-2013 by Consultingwerk Ltd. ("CW") -            *
 * www.consultingwerk.de and other contributors as listed             *
 * below.  All Rights Reserved.                                       *
 *                                                                    *
 *  Software is distributed on an "AS IS", WITHOUT WARRANTY OF ANY    *
 *   KIND, either express or implied.                                 *
 *                                                                    *
 *  Contributors:                                                     *
 *                                                                    *
 **********************************************************************/  
/*------------------------------------------------------------------------
    File        : ErrorHelper
    Purpose     : Various support routines for structured error handling
    Syntax      : Static methods only, private constructor to disallow 
                  instance creation
    Description : 
    Author(s)   : Mike Fechner / Consultingwerk Ltd.
    Created     : Sat Jul 04 13:29:48 CEST 2009
    Notes       : 
  ----------------------------------------------------------------------*/

ROUTINE-LEVEL ON ERROR UNDO, THROW.

{ Consultingwerk/products.i }

USING Consultingwerk.Exceptions.*            FROM PROPATH . 
USING Consultingwerk.Framework.*             FROM PROPATH .
USING Consultingwerk.Framework.Collections.* FROM PROPATH .
USING Consultingwerk.Util.*                  FROM PROPATH .
USING Consultingwerk.Util.Forms.*            FROM PROPATH . 
USING Progress.Lang.*                        FROM PROPATH .

CLASS Consultingwerk.Util.ErrorHelper: 

    /*------------------------------------------------------------------------------
        Purpose: Private default constructor. 
        Notes:   There's no need to create instances of the helper classes                                                                        
    ------------------------------------------------------------------------------*/
    CONSTRUCTOR PRIVATE ErrorHelper ():
        SUPER ().
        
    END CONSTRUCTOR.

    /*------------------------------------------------------------------------------
        Purpose: Returns the current ABL stack trace
        Notes:   Throws an AppError to receive the stack trace, works also when 
                 SESSION:ERROR-STATUS-TRACE is generally set to FALSE
        @return The current ABL Stacktrace 
    ------------------------------------------------------------------------------*/
    DEFINE PUBLIC STATIC PROPERTY CurrentStacktrace AS CHARACTER NO-UNDO 
    GET():
       
        DEFINE VARIABLE cStacktrace        AS CHARACTER NO-UNDO.
        DEFINE VARIABLE lCurrentErrorStack AS LOGICAL NO-UNDO.
        DEFINE VARIABLE iIndex AS INTEGER NO-UNDO.
       
        ASSIGN lCurrentErrorStack         = SESSION:ERROR-STACK-TRACE 
               SESSION:ERROR-STACK-TRACE  = TRUE .
       
        DO ON ERROR UNDO, THROW:
            UNDO, THROW NEW AppError () .     
            
            CATCH err AS Progress.Lang.Error:
                ASSIGN cStacktrace = err:CallStack .    
            END CATCH.
        END.
       
        ASSIGN iIndex = R-INDEX (cStacktrace, CHR (10)) .
           
        IF iIndex > 1 THEN 
            ASSIGN cStacktrace = SUBSTRING (cStacktrace, 1, iIndex - 1) .
        
        RETURN cStacktrace .
       
        FINALLY:
            ASSIGN SESSION:ERROR-STACK-TRACE = lCurrentErrorStack .    
        END FINALLY.
        
    END GET.

    /*------------------------------------------------------------------------------
        Purpose: Allows to control if the ShowErrorMessage methods are allowed 
                 to show an error message using a GUI for .NET Form                                                                      
        Notes:   When set to FALSE the message is delegated to the ShowErrorMessageBox
                 methods
    ------------------------------------------------------------------------------*/        
    DEFINE PUBLIC STATIC PROPERTY ErrorMessageDialogAllowed AS LOGICAL INITIAL TRUE NO-UNDO 
    GET.
    SET.
    
    /*------------------------------------------------------------------------------
        Purpose: The type name of the Form to be used as the ErrorMessageForm                                                                       
        Notes:   The type needs to implement the 
                 Consultingwerk.Util.Forms.IErrorMessageForm interface
    ------------------------------------------------------------------------------*/        
    DEFINE PUBLIC STATIC PROPERTY ErrorMessageFormType AS CHARACTER NO-UNDO 
         INIT "Consultingwerk.Windows.Util.Forms.ErrorMessageForm":U 
    GET.
    SET. 

     /*------------------------------------------------------------------------------
        Purpose: Allows to control if Table And Field information should be suppressed
                 with messages generated by the backend.                                                                        
        Notes:                                                                        
    ------------------------------------------------------------------------------*/    
    DEFINE PUBLIC STATIC PROPERTY SuppressTableAndFieldInfo AS LOGICAL INITIAL FALSE NO-UNDO 
    GET.
    SET.    

     /*------------------------------------------------------------------------------
        Purpose: Adds an ADM2 style formatted error message to an existing error 
                 message string
        Notes:                                                                        
        @param pcMessage The existing message text
        @param pcText The text of the additional message
        @param pcField The field that the message belongs to
        @param pcTable The table that the message belongs to
        @return The resulting message text
    ------------------------------------------------------------------------------*/    
    METHOD PUBLIC STATIC CHARACTER AddErrorMessage (pcMessage AS CHARACTER, 
                                                    pcText AS CHARACTER,
                                                    pcField AS CHARACTER,
                                                    pcTable AS CHARACTER):
        
        DEFINE VARIABLE iMsg     AS INTEGER   NO-UNDO.
        DEFINE VARIABLE iMsgCnt  AS INTEGER   NO-UNDO.
        DEFINE VARIABLE cMessage AS CHARACTER NO-UNDO.
  
        /* If one or more "raw" messages were passed (i.e., they are already in
           the delimited formatted created below), then these were returned from
           an AppServer object or other remote object. Just append them to any local
           messages. */
 
        IF pcText NE ? AND INDEX(pcText, CHR(4)) NE 0 THEN
          ASSIGN pcMessage = pcMessage +
            (IF pcMessage NE "":U THEN CHR(3) ELSE "":U) + pcText.
      
        ELSE DO:
            /* If there's no message passed, this means that a Progress ERROR-STATUS
               was detected, so we add a row for each of those messages. */
       
            iMsgCnt = IF pcText = ? THEN ERROR-STATUS:NUM-MESSAGES ELSE 1.
                 
            DO iMsg = 1 TO iMsgCnt:
              IF pcText = ? THEN
              DO:
              /* When logging ERROR-STATUS messages, remove any which directly 
                 reference the BUFFER-FIELD attribute; these errors are side-effects
                 of other assignment errors which should be reported to the user
                 instead. */
                cMessage = ERROR-STATUS:GET-MESSAGE(iMsg).
                IF INDEX(cMessage, "BUFFER-FIELD":U) NE 0 THEN NEXT.
              END. /* END DO IF pcText = ? */
              
              ASSIGN pcMessage = pcMessage +
                     (IF pcMessage NE "":U THEN CHR(3) ELSE "":U) +
                     (IF pcText = ? THEN cMessage ELSE pcText)
                     + CHR(4) + (IF pcField = ? THEN "":U ELSE pcField)
                     + CHR(4) + (IF pcTable = ? THEN "":U ELSE pcTable).
            END.   /* END DO iMsg */
        END.     /* END ELSE DO */
  
          RETURN pcMessage .

    END METHOD.

     /*------------------------------------------------------------------------------
        Purpose: Returns a single CHARACTER string composed of the error-strings 
                 of all records (from all tables in the given Dataset instance)                                                              
        Notes:      
        @param phDataset The handle of the dataset to return the error strings from
        @return The string composed of the error-strings of all records                                                                   
    ------------------------------------------------------------------------------*/     
    METHOD PUBLIC STATIC CHARACTER DatasetErrorStrings (phDataset AS HANDLE):
        
        DEFINE VARIABLE cErrorStrings AS CHARACTER NO-UNDO.
        
        DEFINE VARIABLE iBuffer       AS INTEGER NO-UNDO.
        DEFINE VARIABLE hBeforeQuery  AS HANDLE  NO-UNDO.
        DEFINE VARIABLE hBeforeTable  AS HANDLE  NO-UNDO.
        DEFINE VARIABLE hBeforeBuffer AS HANDLE  NO-UNDO.
    
        CREATE QUERY hBeforeQuery.
    
        DO iBuffer = 1 TO phDataset:NUM-BUFFERS:
           hBeforeTable = phDataset:GET-BUFFER-HANDLE(iBuffer):TABLE-HANDLE:BEFORE-TABLE.
           IF NOT VALID-HANDLE(hBeforeTable) OR NOT hBeforeTable:HAS-RECORDS THEN
               NEXT.
               
           hBeforeBuffer = phDataset:GET-BUFFER-HANDLE(iBuffer):BEFORE-BUFFER.
           hBeforeQuery:SET-BUFFERS (hBeforeBuffer).
           hBeforeQuery:QUERY-PREPARE (SUBSTITUTE ("FOR EACH &1":U, hBeforeBuffer:NAME)).
           hBeforeQuery:QUERY-OPEN ().
           hBeforeQuery:GET-FIRST ().
         
           DO WHILE hBeforeQuery:QUERY-OFF-END = FALSE.
               IF hBeforeBuffer:ERROR OR hBeforeBuffer:REJECTED THEN
               DO:
                   IF hBeforeBuffer:ERROR-STRING > "":U THEN
                   DO:
                       cErrorStrings = AddErrorMessage (cErrorStrings,
                                                        hBeforeBuffer:ERROR-STRING,
                                                        "":U,
                                                        phDataset:GET-BUFFER-HANDLE(iBuffer):TABLE-HANDLE:NAME).
                   END.
               END. 
               hBeforeQuery:GET-NEXT() .
           END.
        END.
                
        RETURN cErrorStrings .
        
        FINALLY:
            DELETE OBJECT hBeforeQuery.
        END FINALLY.            
    END METHOD .

    /*------------------------------------------------------------------------------
        Purpose: Returns the Typename of the error object                                                                        
        Notes:   Required as .NET Exceptions don't have an ABL Class 
                 (GetClass() returns ?)         
        @param e The Error (Progress error of .NET Exception) to return the type name for
        @return The type name (class name) of the error object
    ------------------------------------------------------------------------------*/    
    METHOD PUBLIC STATIC CHARACTER ErrorTypeName (e AS Progress.Lang.Error):

&IF DEFINED (DotNetAccessible) NE 0 &THEN       
        IF TYPE-OF (e, System.Exception) THEN 
            RETURN CAST(e, System.Exception):GetType():ToString() .
        ELSE 
&ENDIF
            RETURN e:GetClass():TypeName .    
              
    END METHOD.

     /*------------------------------------------------------------------------------
        Purpose: Returns a single CHARACTER string composed of the messages 
                 Contained in the Progress.Lang.Error object                                                                        
        Notes:    
        @param poError The Progress.Lang.Error to format
        @return A string containing all formatted messages from the Error object                                                                  
    ------------------------------------------------------------------------------*/    
    METHOD PUBLIC STATIC CHARACTER FormattedErrorMessages (poError AS Progress.Lang.Error):
        
        DEFINE VARIABLE iMessage   AS INTEGER        NO-UNDO.
        DEFINE VARIABLE cReturn    AS CHARACTER      NO-UNDO.
        DEFINE VARIABLE cMessage   AS CHARACTER      NO-UNDO.
        DEFINE VARIABLE cEntry     AS CHARACTER      NO-UNDO.
        DEFINE VARIABLE cFormatted AS CHARACTER      NO-UNDO.
        DEFINE VARIABLE iError     AS INTEGER        NO-UNDO.
        DEFINE VARIABLE oSoapFault AS SoapFaultError NO-UNDO .  
        
        DO iError = 1 TO poError:NumMessages:

            ASSIGN cMessage = poError:GetMessage(iError).
            
            /* Mike Fechner, Consultingwerk Ltd. 04.07.2009
               Formatting of ADM2 Style error messages */
            IF INDEX(cMessage, CHR(4)) > 0 THEN DO: 
                DO iMessage = 1 TO NUM-ENTRIES(cMessage, CHR(3)):
                    ASSIGN cEntry     = ENTRY(iMessage, cMessage, CHR(3))
                           cFormatted = ENTRY(1, cEntry, CHR(4)).                     
                    
                    /* Mike Fechner, Consultingwerk Ltd. 06.11.2009
                       Ability to suppress Table and Field Info with Dataset Error Messages */                    
                    IF ErrorHelper:SuppressTableAndFieldInfo = FALSE THEN DO:
                        IF NUM-ENTRIES(cEntry, CHR(4)) > 1 AND ENTRY(2, cEntry, CHR(4)) > "":U THEN
                            ASSIGN cFormatted = cFormatted + "~n  ":U + "Field: "{&TRAN} + ENTRY(2, cEntry, CHR(4)) .
                        IF NUM-ENTRIES(cEntry, CHR(4)) > 2 AND ENTRY(3, cEntry, CHR(4)) > "":U THEN
                            ASSIGN cFormatted = cFormatted + "~n  ":U + "Table: "{&TRAN} + ENTRY(3, cEntry, CHR(4)) .
                    END.
                        
                    ASSIGN cFormatted = cFormatted + "~n":U .                        
                END.
            
                ASSIGN cMessage = cFormatted .                
            END.
            
            cReturn = cReturn + 
                       (IF iError > 1 THEN "~n":U ELSE "":U) + 
                       cMessage .  
        END.
        
        /* Mike Fechner, Consultingwerk Ltd. 21.10.2011
           Output SOAP-FAULT */
        IF TYPE-OF (poError, Progress.Lang.SoapFaultError) THEN DO:
            oSoapFault = CAST (poError, Progress.Lang.SoapFaultError) .
            
            IF VALID-HANDLE (oSoapFault:SoapFault) THEN 
                cReturn = SUBSTITUTE ("&1~n~nSOAP fault: &2~nFault detail: &3~nFault code: &4~nFault Actor: &5"{&TRAN},
                                      cReturn, 
                                      oSoapFault:SoapFault:SOAP-FAULT-STRING,
                                      oSoapFault:SoapFault:SOAP-FAULT-DETAIL,
                                      oSoapFault:SoapFault:SOAP-FAULT-CODE,
                                      oSoapFault:SoapFault:SOAP-FAULT-ACTOR) .  
        END.
        
        RETURN cReturn .

    END METHOD .
    
    /*------------------------------------------------------------------------------
        Purpose: Returns a single CHARACTER string composed of the messages 
                 Contained in the Progress.Lang.Error object, including the stack 
                 trace and the text of inner exceptions when present                                                                        
        Notes:   Useful when logging complex error objects to a file 
        @param poError The Progress.Lang.Error to format
        @return A string containing all formatted messages from the Error object                                                                  
    ------------------------------------------------------------------------------*/
	METHOD PUBLIC STATIC CHARACTER FormattedErrorMessagesExt (poError AS Progress.Lang.Error):
		
        DEFINE VARIABLE cReturn     AS CHARACTER           NO-UNDO .
        DEFINE VARIABLE iIndent     AS INTEGER             NO-UNDO .
        
        DEFINE VARIABLE cType       AS CHARACTER           NO-UNDO .
        DEFINE VARIABLE cMessage    AS CHARACTER           NO-UNDO .
        DEFINE VARIABLE cStacktrace AS CHARACTER           NO-UNDO .
        DEFINE VARIABLE cPattern    AS CHARACTER           NO-UNDO INIT "An &1 has occured:"{&TRAN} .
        
        DEFINE VARIABLE oError      AS Progress.Lang.Error NO-UNDO . 
        
        oError = poError .
        
        DO WHILE VALID-OBJECT (oError) ON ERROR UNDO, THROW:
        
            ASSIGN cType       = ErrorHelper:ErrorTypeName (oError)
                   cMessage    = ErrorHelper:FormattedErrorMessages (oError)
                   cStacktrace = ErrorHelper:StackTrace (oError) 
                .

            IF cType > "":U THEN 
                cReturn = cReturn + 
                          StringHelper:Indent (SUBSTITUTE (cPattern, cType), 
                                               iIndent * 5) +
                          Consultingwerk.Environment:NewLine . 
            
            IF cMessage > "":U THEN 
                cReturn = cReturn + 
                          StringHelper:Indent (cMessage, 
                                               iIndent * 5) +
                          Consultingwerk.Environment:NewLine . 

            IF cStacktrace > "":U THEN 
                cReturn = cReturn + 
                          StringHelper:Indent (cStacktrace, 
                                               iIndent * 5) +
                          Consultingwerk.Environment:NewLine . 
            

            IF TYPE-OF (oError, ISupportsInnerException) THEN DO:
                ASSIGN oError   = CAST (oError, ISupportsInnerException):InnerException 
                       iIndent  = iIndent + 1 
                       cPattern = "Caused by an &1:"{&TRAN} . 
            END.
            ELSE 
                oError = ? . 
        END.
        
        RETURN cReturn .

	END METHOD .

    /*------------------------------------------------------------------------------
        Purpose: Display Error Message                                                                        
        Notes:   Stacktrack will be displayed when FrameworkSettings:DebugMode = TRUE           
        @param e The Error object to visualize
        @param pcTitle The title for the message message (dialog)
    ------------------------------------------------------------------------------*/    
    METHOD PUBLIC STATIC VOID ShowErrorMessage (e AS Progress.Lang.Error, 
                                                pcTitle AS CHARACTER):
        

        ErrorHelper:ShowErrorMessageBox (e, pcTitle) .

    END METHOD.

    /*------------------------------------------------------------------------------
        Purpose: Display Error Message                                                                        
        Notes:   Error Type (Class) will be used a default title    
        @param e The Error object to visualize
    ------------------------------------------------------------------------------*/    
    METHOD PUBLIC STATIC VOID ShowErrorMessage (e AS Progress.Lang.Error):
    
        DEFINE VARIABLE cTitle AS CHARACTER NO-UNDO.
                                
        ErrorHelper:ShowErrorMessageBox (e) .
                
    END METHOD.

    /*------------------------------------------------------------------------------
        Purpose: Display Error Message as an alert-box (to be used in functions 
                 and non-void methods due to WAIT-FOR limitations)                                                                      
        Notes:   Stacktrack will be displayed when FrameworkSettings:DebugMode = TRUE    
        @param e The Error object to visualize
        @param pcTitle The title for the message message (dialog)
    ------------------------------------------------------------------------------*/    
    METHOD PUBLIC STATIC VOID ShowErrorMessageBox (e AS Progress.Lang.Error, 
                                                   pcTitle AS CHARACTER):
        
        DEFINE VARIABLE cErrorString AS CHARACTER NO-UNDO.
        
        ASSIGN cErrorString = ErrorHelper:FormattedErrorMessages (e) .
        
        IF TYPE-OF (e, Progress.Lang.AppError) AND CAST (e, Progress.Lang.AppError):ReturnValue > "":U THEN 
            ASSIGN cErrorString = cErrorString + (IF cErrorString > "":U THEN "~n~n":U ELSE "":U) +
                                  CAST (e, Progress.Lang.AppError):ReturnValue .

        /* Mike Fechner, Consultingwerk Ltd. 14.07.2009
           If no error string was given, use the error class name as fall back */                   
        IF cErrorString > "":U THEN .
        ELSE cErrorString = ErrorHelper:ErrorTypeName(e) .

        MESSAGE cErrorString SKIP(2) 
                ErrorHelper:StackTrace (e)
           VIEW-AS ALERT-BOX ERROR TITLE pcTitle . 

    END METHOD.

    /*------------------------------------------------------------------------------
        Purpose: Display Error Message as an alert-box (to be used in functions 
                 and non-void methods due to WAIT-FOR limitations)                                                                        
        Notes:   Error Type (Class) will be used a default title               
        @param e The Error object to visualize
    ------------------------------------------------------------------------------*/    
    METHOD PUBLIC STATIC VOID ShowErrorMessageBox (e AS Progress.Lang.Error):

        DEFINE VARIABLE cTitle AS CHARACTER NO-UNDO.
    
        ASSIGN cTitle = ErrorHelper:ErrorTypeName(e) . 
    
        ErrorHelper:ShowErrorMessageBox (e, cTitle) .
                
    END METHOD.
    
    /*------------------------------------------------------------------------------
        Purpose: Reformats the Error Stack Trace (most current code block first, 
                 not last as provided by the CallStack attribute                                                                        
        Notes:    
        @param e The error object to return the formatted stacktrace for
        @return The formatted stack trace                                                                    
    ------------------------------------------------------------------------------*/
    METHOD PUBLIC STATIC LONGCHAR StackTrace (e AS Progress.Lang.Error):

        DEFINE VARIABLE i       AS INTEGER   NO-UNDO.
        DEFINE VARIABLE cReturn AS CHARACTER NO-UNDO.

        IF e:CallStack > "":U THEN DO:
            DO i = NUM-ENTRIES(e:CallStack, CHR(10)) TO 1 BY -1.  
                ASSIGN cReturn = cReturn + ENTRY(i, e:CallStack, CHR(10)) + CHR(10). 
            END.
        END.    
        ELSE 
            cReturn =  "Stacktrace not provided (-errorstack missing)."{&TRAN} + CHR(10) .        

&IF DEFINED (DotNetAccessible) NE 0 &THEN       
        IF TYPE-OF(e, System.Exception) THEN 
            ASSIGN cReturn = cReturn + CHR(10) +
                             ".NET Stack Trace:"{&TRAN} + CHR(10) + 
                             CAST(e, System.Exception):StackTrace .             
&ENDIF
        RETURN TRIM(cReturn, CHR(10)) .             

    END METHOD.

 END CLASS.




© 2015 - 2025 Weber Informatics LLC | Privacy Policy