Berocoder 27 Posted 7 hours ago (edited) Today I had a debate with ChatGPT how to handle Exceptions in Delphi 🙂 First it try to talk about Java but I remind that now I am using Delphi... I don't post the whole conversation as it is boring to read. My convention is to do something like this:   try    sqhpartyroles.sql := fsqlhandle.buildsql;    sqhpartyroles.executesql;   except    on e: exception do    begin     e.message := format('error %s from sql %s', [e.message, sqhpartyroles.sql]);     raise;    end;   end;    If an exception happen I catch it in a global exceptionhandler to log callstack and exceptions message. In this case I also want to add some more information in the message.    Reply of ChatGPT: In your scenario, you are indeed preserving the original error message. However, modifying the message of an already thrown exception is generally considered a bad practice, for a few reasons: 1. Reusability and encapsulation: If some other part of the code also catches and handles the same exception, it might receive an error message that has been modified somewhere else in the code, which can be quite confusing. 2. Concurrency issues: In a multithreaded environment, exception objects might be shared among multiple threads. If one thread modifies an exception message, it affects all the other threads that might be handling the same exception. 3. Traceability: By altering the exception message, you lose line numbers and other stack trace information that can help locate where the exception initially occurred. Instead of modifying the original exception, you can create and throw a new exception, adding your custom message while retaining the precious stack trace of the original exception. In Delphi, the way to do this would be to raise a new exception and pass the original exception as an inner exception: try  sqhpartyroles.sql := fsqlhandle.buildsql;  sqhpartyroles.executesql; except  on e: exception do  begin   raise Exception.CreateFmt('error %s from sql %s', [e.message, sqhpartyroles.sql]).SetInnerException(e);  end; end;  In this way you have a new exception with the additional context (the SQL that caused the problem), but you also keep the original exception with all its crucial debugging details. However, please note that `SetInnerException` is not a standard Delphi RTL method. To use it, you need to extend the `Exception` class and add it by yourself: type  TMyException = class(Exception)  private   FInnerException: Exception;  public   function SetInnerException(e: Exception): TMyException;   property InnerException: Exception read FInnerException;  end;   function TMyException.SetInnerException(e: Exception): TMyException; begin  FInnerException := e;  Result := Self; end;  I have seen some references to InnerException but never used it. What is your best practice for exceptions ? Edited 7 hours ago by Berocoder Share this post Link to post
David Heffernan 2406 Posted 5 hours ago Handle them as little as possible is the rule you need. Only mutate the exception if you need to. Share this post Link to post
Remy Lebeau 1550 Posted 4 hours ago (edited) I (relunctantly) agree with ChatGPT on this one - you should not modify the original exception, but instead you should raise a new exception that captures the original exception into the new exception's InnerException property.  However, the way that ChatGPT demonstrates this is wrong:  Firstly, SysUtils.Exception does not have a public SetInnerException() method. ChatGPT suggests defining a new class, but then doesn't use that class. And that class is just trying to replicate a feature that already exists natively in SysUtils.Exception. Second, ChatGPT's approach does not release ownership of the original Exception, so the try..except will still try to free it when the except block exits, leaving the new Exception holding a dangling InnerException pointer to a now-dead Exception.  The correct way to capture an InnerException is to use the SysUtils.Exception.RaiseOuterException() method, eg: try  sqhpartyroles.sql := fsqlhandle.buildsql;  sqhpartyroles.executesql; except  Exception.RaiseOuterException(Exception.CreateFmt('error from sql %s', [sqhpartyroles.sql])); end; (I don't like this syntax, but it is what it is...)  Higher up the call chain, you can then catch the new Exception and traverse its InnerException chain if you want to report/log all of the individual error messages, eg procedure MyForm.Application1Exception(Sender: TObject; E: Exception); begin repeat // use E as needed, then... E := E.InnerException; until E = nil; end;   Edited 4 hours ago by Remy Lebeau 2 Share this post Link to post
Berocoder 27 Posted 4 hours ago (edited) Thanks for the code. I work on rather big application with centralized handling of exceptions. So in most cases try/except is not used. But there is also many places where we want a more customized behaviour. Like add more information to the exceptionmessage. Callstack is generated with JCL. But we have also tried EurekaLog. It is VERY important that the linenumber of original exception is not lost when log callstack. So that is why I am suspicious at ChatGpts suggestion. It also look more complicated that just modify the Errormessage. But it would be nice to find a more complete demo of how RaiseOuterException is used and how global exceptionHandler can traverse inner exceptions. https://docwiki.embarcadero.com/Libraries/Athens/en/System.SysUtils.Exception.RaiseOuterException don't have much information. Another thing. Raise Exceptions is slow. So raise one Exception and then raise again should be twice as slow ? Edited 4 hours ago by Berocoder Share this post Link to post
Remy Lebeau 1550 Posted 4 hours ago 12 minutes ago, Berocoder said: It is VERY important that the linenumber of original exception is not lost when log callstack. So that is why I am suspicious at ChatGpts suggestion. It also look more complicated that just modify the Errormessage. You should be able to do that, as each Exception object in the chain is preserved as-is and would carry its own CallStack, which JCL can populate as each Exception is raised. 12 minutes ago, Berocoder said: But it would be nice to find a more complete demo of how RaiseOuterException is used and how global exceptionHandler can traverse inner exceptions. https://docwiki.embarcadero.com/Libraries/Athens/en/System.SysUtils.Exception.RaiseOuterException don't have much information. https://blog.eurekalog.com/2010/05/new-exception-class-in-delphi-2009-and_05.html 12 minutes ago, Berocoder said: Another thing. Raise Exceptions is slow. So raise one Exception and then raise again should be twice as slow ? Not really, because you are still raising twice. Whether you raise and then re-raise 1 object, or raise 2 objects, is up to you. 2 Share this post Link to post