Skip to content

Commit

Permalink
Prevent leaking TracerProvider references in `register_shutdown_fun…
Browse files Browse the repository at this point in the history
…ction` (#716)
  • Loading branch information
Nevay authored Jun 21, 2022
1 parent ac95256 commit 122aff0
Show file tree
Hide file tree
Showing 2 changed files with 53 additions and 1 deletion.
41 changes: 40 additions & 1 deletion src/SDK/Trace/TracerProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -14,11 +14,15 @@
use OpenTelemetry\SDK\Trace\Sampler\AlwaysOnSampler;
use OpenTelemetry\SDK\Trace\Sampler\ParentBased;
use function register_shutdown_function;
use function spl_object_id;
use WeakReference;

final class TracerProvider implements API\TracerProviderInterface
{
public const DEFAULT_TRACER_NAME = 'io.opentelemetry.contrib.php';

/** @var array<int, WeakReference<self>>|null */
private static ?array $tracerProviders = null;
private static ?API\TracerInterface $defaultTracer = null;

/** @var array<string, API\TracerInterface> */
Expand Down Expand Up @@ -53,7 +57,7 @@ public function __construct(
$spanProcessors
);

register_shutdown_function([$this, 'shutdown']);
self::registerShutdownFunction($this);
}

public function forceFlush(): ?bool
Expand Down Expand Up @@ -116,6 +120,41 @@ public function shutdown(): bool
return true;
}

self::unregisterShutdownFunction($this);

return $this->tracerSharedState->shutdown();
}

public function __destruct()
{
$this->shutdown();
}

private static function registerShutdownFunction(TracerProvider $tracerProvider): void
{
if (self::$tracerProviders === null) {
register_shutdown_function(static function (): void {
$tracerProviders = self::$tracerProviders;
self::$tracerProviders = null;

// Push tracer provider shutdown to end of queue
// @phan-suppress-next-line PhanTypeMismatchArgumentInternal
register_shutdown_function(static function (array $tracerProviders): void {
foreach ($tracerProviders as $reference) {
if ($tracerProvider = $reference->get()) {
$tracerProvider->shutdown();
}
}
}, $tracerProviders);
});
}

self::$tracerProviders[spl_object_id($tracerProvider)] = WeakReference::create($tracerProvider);
}

private static function unregisterShutdownFunction(TracerProvider $tracerProvider): void
{
/** @psalm-suppress PossiblyNullArrayAccess */
unset(self::$tracerProviders[spl_object_id($tracerProvider)]);
}
}
13 changes: 13 additions & 0 deletions tests/Unit/SDK/Trace/TracerProviderTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
use OpenTelemetry\SDK\Trace\Tracer;
use OpenTelemetry\SDK\Trace\TracerProvider;
use PHPUnit\Framework\TestCase;
use WeakReference;

/**
* @coversDefaultClass \OpenTelemetry\SDK\Trace\TracerProvider
Expand Down Expand Up @@ -174,4 +175,16 @@ public function test_get_tracer_returns_noop_tracer_after_shutdown(): void
$provider->getTracer('foo')
);
}

/**
* @coversNothing
*/
public function test_tracer_register_shutdown_function_does_not_leak_reference(): void
{
$provider = new TracerProvider();
$reference = WeakReference::create($provider);

$provider = null;
$this->assertTrue($reference->get() === null);
}
}

0 comments on commit 122aff0

Please sign in to comment.