generator = $generator; /** * @param \Throwable|null $exception Exception to be thrown into the generator. * @param mixed $value Value to be sent into the generator. */ $this->when = function ($exception, $value) { if ($this->depth > self::MAX_CONTINUATION_DEPTH) { // Defer continuation to avoid blowing up call stack. Loop::defer(function () use ($exception, $value) { ($this->when)($exception, $value); }); return; } try { if ($exception) { // Throw exception at current execution point. $yielded = $this->generator->throw($exception); } else { // Send the new value and execute to next yield statement. $yielded = $this->generator->send($value); } if (!$yielded instanceof Promise) { if (!$this->generator->valid()) { $this->resolve($this->generator->getReturn()); return; } if ($yielded instanceof ReactPromise) { $yielded = adapt($yielded); } else { throw new InvalidYieldError( $this->generator, \sprintf( "Unexpected yield; Expected an instance of %s or %s", Promise::class, ReactPromise::class ) ); } } ++$this->depth; $yielded->when($this->when); --$this->depth; } catch (\Throwable $exception) { $this->dispose($exception); } }; try { $yielded = $this->generator->current(); if (!$yielded instanceof Promise) { if (!$this->generator->valid()) { $this->resolve($this->generator->getReturn()); return; } if ($yielded instanceof ReactPromise) { $yielded = adapt($yielded); } else { throw new InvalidYieldError( $this->generator, \sprintf( "Unexpected yield; Expected an instance of %s or %s", Promise::class, ReactPromise::class ) ); } } ++$this->depth; $yielded->when($this->when); --$this->depth; } catch (\Throwable $exception) { $this->dispose($exception); } } /** * Runs the generator to completion then fails the coroutine with the given exception. * * @param \Throwable $exception */ private function dispose(\Throwable $exception) { if ($this->generator->valid()) { try { try { // Ensure generator has run to completion to avoid throws from finally blocks on destruction. do { $this->generator->throw($exception); } while ($this->generator->valid()); } finally { // Throw from finally to attach any exception thrown from generator as previous exception. throw $exception; } } catch (\Throwable $exception) { // $exception will be used to fail the coroutine. } } $this->fail($exception); } }