On Sat, Nov 8, 2025, at 1:54 PM, Rowan Tommins [IMSoP] wrote:
> On 05/11/2025 22:38, Bob Weinand wrote:
>> The choice of adding the exception to the exitContext() is 
>> interesting, but also very opinionated:
>>
>> - It means, that the only way to abort, in non-exceptional cases, is 
>> to throw yourself an exception. And put a try/catch around the with() 
>> {} block. Or manually use enterContext() & exitContext() - with a fake 
>> "new Exception" essentially.
>> - Maybe you want to hold a transaction, but just ensure that 
>> everything gets executed together (i.e. atomicity), but not care about 
>> whether everything actually went through (i.e. not force a rollback on 
>> exception). You'll now have to catch the exception, store it to a 
>> variable, use break and check for the exception after the with block. 
>> Or, yes, manually using enterContext() and exitContext(). 
>
>
> The Context Manager is *given knowledge of* the exception, but it's not 
> obliged to change its behaviour based on that knowledge. I don't think 
> that makes the interface opinionated, it makes it extremely flexible.
>
> It means you *can* write this, which is impossible in a destructor:
>
> function exitContext(?Throwable $exception) {
>      if ( $exception === null ) {
>          $this->commit();
>      } else {
>          $this->rollback();
>      }
> }
>
>
> But you could also write any of these, which are exactly the same as 
> they would be in __destruct():
>
> // Rollback unless explicitly committed
> function exitContext(?Throwable $exception) {
>      if ( ! $this->isCommitted ) {
>          $this->rollback();
>      }
> }
>
> // Expect explicit commit or rollback, but roll back as a safety net
> function exitContext(?Throwable $exception) {
>      if ( ! $this->isCommitted && ! $this->isRolledBack ) {
>          $this->logger->warn('Transaction went out of scope without 
> explicit rollback, rolling back now.');
>          $this->rollback();
>      }
> }
>
> // User can choose at any time which action will be taken on destruct / exit
> function exitContext(?Throwable $exception) {
>      if ( $this->shouldCommitOnExit ) {
>          $this->commit();
>      } else {
>          $this->rollback();
>      }
> }
>
>
> You could also combine different approaches, using the exception as an 
> extra signal only if the user hasn't chosen explicitly:
>
> function exitContext(?Throwable $exception) {
>      if ($this->isCommitted || $this->isRolledBack) {
>          return;
>      }
>      if ( $exception === null ) {
>          $this->logger->debug('Implicit commit - consider calling 
> commit() for clearer code.');
>          $this->commit();
>      } else {
>          $this->logger->debug('Implicit rollback - consider calling 
> rollback() for clearer code.');
>          $this->rollback();
>      }
> }
>
>
> -- 
> Rowan Tommins
> [IMSoP]

Though one point to note here, $this->isCommitted on the context MANAGER is not 
the same as isCommitted on the context VARIABLE.  So in the above examples you 
would have to either return $this from enterContext() (which is fine), or save 
a reference to the context variable in the manager and then check 
$this->txn->isCommitted (which is also fine).

Which you choose is mostly an implementation detail, and it's fine either way; 
I just want to emphasize that a lot of the flexibility of context managers 
comes from the separation of the context manager from the variable.

A context manager is not just an auto-unsetter, though that is part of what it 
does.  It is more properly a way to abstract out and package up setup/teardown 
lifecycle management, which can differentiate between a success or failure 
case.  (At least as much as PHP itself is able to right now.)  That has a wide 
variety of use cases, only a few of which a simple destructor could handle.  

--Larry Garfield

Reply via email to