Make OpenTelemetry work with FiberBoundContextStorage

Created on 18 November 2024, 5 months ago

Problem/Motivation

The "open-telemetry/context" library started to use the FiberBoundContextStorage by default, which relies on the function Fiber::getCurrent().

But Drupal started to use the function Fiber::getCurrent() too, in the Renderer and BigPipe.

That leads to conflicts because Drupal Kernel calls the Fiber::getCurrent()->suspend() which puts the OpenTelemetry FiberBoundContextStorage service on suspend.

The temporary workaround is to force set the simple OpenTelemetry\Context\ContextStorage() on the initialization stage:

function OpentelemetryTraceManager::__construct() {
...
    // Force disable the FiberBoundContextStorage because of the conflict
    // with Drupal Renderer service.
    // @todo Make a proper fix to work well with the FiberBoundContextStorage.
    $contextStorage = new ContextStorage();
    Context::setStorage($contextStorage);
...

We need to investigate this and find a fix on the Drupal or OpenTelemetry side.

Steps to reproduce

Proposed resolution

Remaining tasks

User interface changes

API changes

Data model changes

📌 Task
Status

Active

Version

1.0

Component

Code

Created by

🇦🇲Armenia murz Yerevan, Armenia

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Comments & Activities

  • Issue created by @murz
  • 🇫🇷France andypost

    It may work if core patched to pass context into fiber but ++ to find out general approach without hacking every place where Fiber is created

    AUto-instrumentation using FFI+ZTS also overkill

  • 🇫🇷France andypost

    Module could provide hybrid context and raise required OT version to 1.1

    Kinda

    use OpenTelemetry\Context\ContextStorage;
    use OpenTelemetry\Context\FiberBoundContextStorage;
    
    class HybridContextStorage extends ContextStorage {
        private $fiberStorage;
        private $defaultStorage;
    
        public function __construct() {
            $this->fiberStorage = new FiberBoundContextStorage();
            $this->defaultStorage = new ContextStorage();
        }
    
        public function current(): Context {
            if (\Fiber::getCurrent()) {
                return $this->fiberStorage->current();
            }
            return $this->defaultStorage->current();
        }
    
        public function attach(Context $context): Scope {
            if (\Fiber::getCurrent()) {
                return $this->fiberStorage->attach($context);
            }
            return $this->defaultStorage->attach($context);
        }
    }
    
    
  • 🇫🇷France andypost

    Measuring queries executed in big-pipe and while rendering also can give wrong results on timing for db-queries

    Probably makes sense to patch core so it will react on fiber suspend,
    also maybe to add attribute to spans to point out that span generated inside of fiber and time could be inaccurate

Production build 0.71.5 2024