Skip to main content
Drupal API
User account menu
  • Log in

Breadcrumb

  1. Drupal Core 11.1.x

ProtobufSerializer.php

Namespace

OpenTelemetry\Contrib\Otlp

File

vendor/open-telemetry/exporter-otlp/ProtobufSerializer.php

View source
<?php

declare (strict_types=1);
namespace OpenTelemetry\Contrib\Otlp;

use function base64_decode;
use function bin2hex;
use Exception;
use Google\Protobuf\Descriptor;
use Google\Protobuf\DescriptorPool;
use Google\Protobuf\FieldDescriptor;
use Google\Protobuf\Internal\GPBLabel;
use Google\Protobuf\Internal\GPBType;
use Google\Protobuf\Internal\Message;
use InvalidArgumentException;
use function json_decode;
use function json_encode;
use const JSON_UNESCAPED_SLASHES;
use const JSON_UNESCAPED_UNICODE;
use function lcfirst;
use OpenTelemetry\SDK\Common\Export\TransportInterface;
use function property_exists;
use function sprintf;
use function ucwords;

/**
 * @internal
 * @psalm-type SUPPORTED_CONTENT_TYPES = ContentTypes::PROTOBUF|ContentTypes::JSON|ContentTypes::NDJSON
 */
final class ProtobufSerializer {
    private function __construct(string $contentType) {
    }
    public static function getDefault() : ProtobufSerializer {
        return new self(ContentTypes::PROTOBUF);
    }
    
    /**
     * @psalm-param TransportInterface<SUPPORTED_CONTENT_TYPES> $transport
     */
    public static function forTransport(TransportInterface $transport) : ProtobufSerializer {
        return match ($contentType = $transport->contentType()) {    ContentTypes::PROTOBUF, ContentTypes::JSON, ContentTypes::NDJSON => new self($contentType),
            default => throw new InvalidArgumentException(sprintf('Not supported content type "%s"', $contentType)),
        
        };
    }
    public function serializeTraceId(string $traceId) : string {
        // @phpstan-ignore-next-line
        return match ($this->contentType) {    ContentTypes::PROTOBUF => $traceId,
            ContentTypes::JSON, ContentTypes::NDJSON => base64_decode(bin2hex($traceId)),
        
        };
    }
    public function serializeSpanId(string $spanId) : string {
        // @phpstan-ignore-next-line
        return match ($this->contentType) {    ContentTypes::PROTOBUF => $spanId,
            ContentTypes::JSON, ContentTypes::NDJSON => base64_decode(bin2hex($spanId)),
        
        };
    }
    public function serialize(Message $message) : string {
        // @phpstan-ignore-next-line
        return match ($this->contentType) {    ContentTypes::PROTOBUF => $message->serializeToString(),
            ContentTypes::JSON => self::postProcessJsonEnumValues($message, $message->serializeToJsonString()),
            ContentTypes::NDJSON => self::postProcessJsonEnumValues($message, $message->serializeToJsonString()) . "\n",
        
        };
    }
    
    /**
     * @phan-suppress PhanParamTooManyInternal (@see https://github.com/phan/phan/pull/4840)
     * @throws Exception
     */
    public function hydrate(Message $message, string $payload) : void {
        // @phpstan-ignore-next-line
        match ($this->contentType) {    ContentTypes::PROTOBUF => $message->mergeFromString($payload),
            ContentTypes::JSON, ContentTypes::NDJSON => $message->mergeFromJsonString($payload, true),
        
        };
    }
    
    /**
     * Workaround until protobuf exposes `FormatEnumsAsIntegers` option.
     *
     * [JSON Protobuf Encoding](https://opentelemetry.io/docs/specs/otlp/#json-protobuf-encoding):
     * > Values of enum fields MUST be encoded as integer values.
     *
     * @see https://github.com/open-telemetry/opentelemetry-php/issues/978
     * @see https://github.com/protocolbuffers/protobuf/pull/12707
     */
    private static function postProcessJsonEnumValues(Message $message, string $payload) : string {
        $pool = DescriptorPool::getGeneratedPool();
        $desc = $pool->getDescriptorByClassName($message::class);
        if (!$desc instanceof Descriptor) {
            return $payload;
        }
        $data = json_decode($payload);
        unset($payload);
        self::traverseDescriptor($data, $desc);
        return json_encode($data, JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE);
    }
    private static function traverseDescriptor(object $data, Descriptor $desc) : void {
        for ($i = 0, $n = $desc->getFieldCount(); $i < $n; $i++) {
            // @phan-suppress-next-line PhanParamTooManyInternal
            $field = $desc->getField($i);
            $name = lcfirst(strtr(ucwords((string) $field->getName(), '_'), [
                '_' => '',
            ]));
            if (!property_exists($data, $name)) {
                continue;
            }
            if ($field->getLabel() === GPBLabel::REPEATED) {
                foreach ($data->{$name} as $key => $value) {
                    $data->{$name}[$key] = self::traverseFieldDescriptor($value, $field);
                }
            }
            else {
                $data->{$name} = self::traverseFieldDescriptor($data->{$name}, $field);
            }
        }
    }
    private static function traverseFieldDescriptor($data, FieldDescriptor $field) {
        switch ($field->getType()) {
            case GPBType::MESSAGE:
                self::traverseDescriptor($data, $field->getMessageType());
                break;
            case GPBType::ENUM:
                $enum = $field->getEnumType();
                for ($i = 0, $n = $enum->getValueCount(); $i < $n; $i++) {
                    if ($data === $enum->getValue($i)
                        ->getName()) {
                        return $enum->getValue($i)
                            ->getNumber();
                    }
                }
                break;
        }
        return $data;
    }

}

Classes

Title Deprecated Summary
ProtobufSerializer @internal @psalm-type SUPPORTED_CONTENT_TYPES = ContentTypes::PROTOBUF|ContentTypes::JSON|ContentTypes::NDJSON
RSS feed
Powered by Drupal