Asynchronous Operations: Exceptions

In general, an async operation has the following pattern:

  • Call an async function
  • Get an awaitable back
  • await the awaitable to get a result

If an exception is thrown within an async function body, the function does not technically throw - it returns an Awaitable that throws when resolved. This means that if an Awaitable is resolved multiple times, the same exception object instance will be thrown every time; as it is the same object every time, its data will be unchanged, including backtraces.

async function exception_thrower(): Awaitable<void> {
  throw new \Exception("Return exception handle");
}

async function basic_exception(): Awaitable<void> {
  // the handle does not throw, but result will be an Exception objection.
  // Remember, this is the same as:
  //   $handle = exception_thrower();
  //   await $handle;
  await exception_thrower();
}

<<__EntryPoint>>
function main(): void {
  HH\Asio\join(basic_exception());
}

The use of from_async ignores any successful awaitable results and just throw an exception of one of the awaitable results, if one of the results was an exception.

async function exception_thrower(): Awaitable<void> {
  throw new \Exception("Return exception handle");
}

async function non_exception_thrower(): Awaitable<int> {
  return 2;
}

async function multiple_waithandle_exception(): Awaitable<void> {
  $handles = vec[exception_thrower(), non_exception_thrower()];
  // You will get a fatal error here with the exception thrown
  $results = await Vec\from_async($handles);
  // This won't happen
  var_dump($results);
}

<<__EntryPoint>>
function main(): void {
  HH\Asio\join(multiple_waithandle_exception());
}

To get around this, and get the successful results as well, we can use the utility function HH\Asio\wrap. It takes an awaitable and returns the expected result or the exception if one was thrown. The exception it gives back is of the type ResultOrExceptionWrapper.

namespace HH\Asio {
  interface ResultOrExceptionWrapper<T> {
    public function isSucceeded(): bool;
    public function isFailed(): bool;
    public function getResult(): T;
    public function getException(): Exception;
  }
}

Taking the example above and using the wrapping mechanism, this is what the code looks like:

async function exception_thrower(): Awaitable<void> {
  throw new Exception();
}

async function non_exception_thrower(): Awaitable<int> {
  return 2;
}

async function wrapping_exceptions(): Awaitable<void> {
  $handles = vec[
    HH\Asio\wrap(exception_thrower()),
    HH\Asio\wrap(non_exception_thrower()),
  ];
  // Since we wrapped, the results will contain both the exception and the
  // integer result
  $results = await Vec\from_async($handles);
  var_dump($results);
}

<<__EntryPoint>>
function main(): void {
  HH\Asio\join(wrapping_exceptions());
}

Memoized Async Exceptions

Because __Memoize caches the awaitable itself, if an async function is memoized and throws, you will get the same exception backtrace on every failed call.

For example, both times an exception is caught here, foo is in the backtrace, but bar is not, as the call to foo led to the creation of the memoized awaitable:

<<__Memoize>>
async function throw_something(): Awaitable<int> {
  throw new Exception();
}

async function foo(): Awaitable<void> {
  await throw_something();
}

async function bar(): Awaitable<void> {
  await throw_something();
}

<<__EntryPoint>>
async function main(): Awaitable<void> {
  try {
    await foo();
  } catch (Exception $e) {
    var_dump($e->getTrace()[2] as shape('function' => string, ...)['function']);
  }
  try {
    await bar();
  } catch (Exception $e) {
    var_dump($e->getTrace()[2] as shape('function' => string, ...)['function']);
  }
}
Was This Page Useful?
Thank You!
Thank You! If you'd like to share more feedback, please file an issue.