gemvc/apm-contracts

APM contracts, interface and abstract base class for GEMVC framework

Installs: 3

Dependents: 0

Suggesters: 0

Security: 0

Stars: 0

Watchers: 0

Forks: 0

Open Issues: 0

pkg:composer/gemvc/apm-contracts

1.1.0 2025-12-31 00:06 UTC

This package is auto-updated.

Last update: 2025-12-31 00:14:06 UTC


README

gemvc_header_for_github

GEMVC APM Contracts

License: MIT PHP Version

APM contracts, interfaces, and abstract base class for GEMVC framework. This package provides the foundation for all Application Performance Monitoring (APM) providers, enabling developers to create pluggable APM solutions that work seamlessly with GEMVC.

Table of Contents

Overview

The gemvc/apm-contracts package provides:

  • ApmInterface - Contract that all APM providers must implement
  • AbstractApm - Base class with shared functionality (request handling, utilities, configuration)
  • ApmFactory - Factory for creating APM provider instances based on configuration

This package is always installed with GEMVC (required by gemvc/library), providing a standardized way to integrate any APM solution (TraceKit, Datadog, New Relic, Elastic APM, OpenTelemetry, etc.) without changing application code.

πŸ“¦ Installation

This package is automatically installed when you install GEMVC:

composer require gemvc/library

It's also required by APM provider packages:

composer require gemvc/apm-tracekit
# This automatically installs gemvc/apm-contracts

πŸ—οΈ Architecture

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚                    gemvc/library                        β”‚
β”‚  (Core Framework - uses ApmFactory::create() -         β”‚
β”‚   Universal abstraction, like UniversalQueryExecuter)   β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”¬β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
                     β”‚ requires
                     β–Ό
β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚              gemvc/apm-contracts                        β”‚
β”‚  β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”  β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”  β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”   β”‚
β”‚  β”‚ ApmInterface β”‚  β”‚ AbstractApm  β”‚  β”‚ ApmFactory   β”‚   β”‚
β”‚  β”‚  (Contract)  β”‚  β”‚  (Base)      β”‚  β”‚ (Universal) β”‚   β”‚
β”‚  β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜  β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜  β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜   β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”¬β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
                     β”‚ implements/extends
                     β”‚ (Auto-discovered via dynamic instantiation)
        β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”Όβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
        β–Ό            β–Ό            β–Ό          
β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β” β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β” β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”   
β”‚ apm-tracekitβ”‚ β”‚ Datadog  β”‚ β”‚ NewRelic β”‚   
β”‚  Provider   β”‚ β”‚ Provider β”‚ β”‚ Provider β”‚   
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜ β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜ β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜    
     (Any provider - no factory modification needed!)

Universal Pattern: Just like UniversalQueryExecuter abstracts database connections (PDO/OpenSwoole/MongoDB), ApmFactory abstracts APM providers (TraceKit/Datadog/NewRelic). The core framework uses ApmFactory::create() without knowing which provider is installed.

Creating an APM Provider Package

Follow these steps to create your own APM provider package for GEMVC:

Step 1: Create Package Structure

mkdir gemvc-apm-yourprovider
cd gemvc-apm-yourprovider
composer init

Step 2: Set Up composer.json

{
  "name": "gemvc/apm-yourprovider",
  "description": "YourProvider APM provider for GEMVC framework",
  "type": "library",
  "license": "MIT",
  "require": {
    "php": ">=8.2",
    "gemvc/apm-contracts": "^1.0",
    "gemvc/library": "^5.2"
  },
  "autoload": {
    "psr-4": {
      "Gemvc\\Core\\Apm\\Providers\\YourProvider\\": "src/Gemvc/Core/Apm/Providers/YourProvider/"
    }
  }
}

Step 3: Create Provider Class

Create src/Gemvc/Core/Apm/Providers/YourProvider/YourProvider.php:

<?php
namespace Gemvc\Core\Apm\Providers\YourProvider;

use Gemvc\Core\Apm\AbstractApm;
use Gemvc\Helper\ProjectHelper;

/**
 * YourProvider APM Implementation
 */
class YourProvider extends AbstractApm
{
    // YourProvider-specific configuration
    private string $apiKey;
    private string $endpoint;
    
    /**
     * Load YourProvider-specific configuration
     */
    protected function loadConfiguration(array $config = []): void
    {
        // Load API key (check both YOURPROVIDER_API_KEY and APM_API_KEY)
        $this->apiKey = $config['api_key'] 
            ?? $_ENV['YOURPROVIDER_API_KEY'] 
            ?? $_ENV['APM_API_KEY'] 
            ?? '';
        
        // Load endpoint
        $this->endpoint = $config['endpoint'] 
            ?? $_ENV['YOURPROVIDER_ENDPOINT'] 
            ?? 'https://api.yourprovider.com/v1/traces';
        
        // Note: Common properties (enabled, sampleRate, traceResponse, etc.) are already
        // set in the constructor from $config or $_ENV. You can override them here if needed:
        // $this->enabled = $this->parseBooleanFlag($config, 'enabled', 'APM_ENABLED', true);
        // $this->sampleRate = $this->parseSampleRate($config, 'YOURPROVIDER_SAMPLE_RATE', 1.0);
        // $this->traceResponse = $this->parseBooleanFlag($config, 'trace_response', 'YOURPROVIDER_TRACE_RESPONSE', false);
        // $this->traceDbQuery = $this->parseBooleanFlag($config, 'trace_db_query', 'YOURPROVIDER_TRACE_DB_QUERY', false);
        // $this->traceRequestBody = $this->parseBooleanFlag($config, 'trace_request_body', 'YOURPROVIDER_TRACE_REQUEST_BODY', false);
        
        // Disable if no API key
        if (empty($this->apiKey)) {
            $this->enabled = false;
        }
    }
    
    /**
     * Initialize root trace from Request object
     */
    protected function initializeRootTrace(): void
    {
        if ($this->request === null) {
            return;
        }
        
        try {
            // Build root span attributes from Request
            $rootAttributes = [
                'http.method' => $this->request->getMethod(),
                'http.url' => $this->request->getUri(),
                'http.user_agent' => $this->request->getHeader('User-Agent') ?? 'unknown',
                'http.route' => $this->request->getServiceName() . '/' . $this->request->getMethodName(),
            ];
            
            // Optionally include request body if enabled
            if ($this->shouldTraceRequestBody()) {
                $requestBody = $this->getRequestBodyForTracing();
                if ($requestBody !== null) {
                    $rootAttributes['http.request.body'] = self::limitStringForTracing($requestBody);
                }
            }
            
            // Start root trace (implement your provider's trace creation)
            $this->rootSpan = $this->startSpan('http-request', $rootAttributes);
            
            if (empty($this->rootSpan)) {
                return;
            }
            
            // Register shutdown function to flush traces
            register_shutdown_function(function() {
                $this->flush();
            });
        } catch (\Throwable $e) {
            // Silently fail - don't let APM break the application
            if (ProjectHelper::isDevEnvironment()) {
                error_log("YourProvider: Failed to initialize root trace: " . $e->getMessage());
            }
        }
    }
    
    /**
     * Start a child span (implement your provider's span creation)
     */
    public function startSpan(string $operationName, array $attributes = [], int $kind = self::SPAN_KIND_INTERNAL): array
    {
        if (!$this->isEnabled()) {
            return [];
        }
        
        // Implement your provider's span creation logic
        // Return: ['span_id' => string, 'trace_id' => string, 'start_time' => int]
    }
    
    /**
     * End a span (implement your provider's span ending logic)
     */
    public function endSpan(array $spanData, array $finalAttributes = [], ?string $status = self::STATUS_OK): void
    {
        // Implement your provider's span ending logic
    }
    
    /**
     * Record an exception on a span
     */
    public function recordException(array $spanData, \Throwable $exception): array
    {
        // Implement your provider's exception recording logic
    }
    
    /**
     * Flush traces (send to your provider's service)
     */
    public function flush(): void
    {
        // Implement your provider's trace sending logic
        // Use AsyncApiCall for non-blocking sending
    }
    
    // Implement other required methods from ApmInterface...
}

Step 4: No Registration Required!

The factory automatically discovers your provider! No need to modify ApmFactory.php.

The factory uses dynamic class instantiation based on the APM_NAME environment variable:

  • APM_NAME="YourProvider" β†’ Automatically looks for Gemvc\Core\Apm\Providers\YourProvider\YourProviderProvider
  • If the class exists, it's instantiated automatically
  • If the class doesn't exist, the factory gracefully returns null

This follows the Open/Closed Principle - you can add new providers without modifying the factory!

Step 5: Configuration

Users configure your provider in .env:

APM_NAME="YourProvider"
APM_ENABLED="true"
YOURPROVIDER_API_KEY="your-api-key"
# or use unified key:
APM_API_KEY="your-api-key"

πŸ”§ How It Works

1. Framework Initialization

When GEMVC starts processing a request:

// In ApiService constructor
$this->tracekit = ApmFactory::create($this->request);

2. Factory Pattern

ApmFactory reads APM_NAME from environment and creates the appropriate provider:

$apmName = $_ENV['APM_NAME'] ?? 'TraceKit';
return match($apmName) {
    'tracekit' => new TraceKitProvider($request),
    'datadog' => new DatadogProvider($request),
    // ...
};

3. Provider Initialization

Each provider extends AbstractApm, which:

  1. Stores Request object - For accessing HTTP metadata
  2. Sets common properties - From $config array (if provided) or $_ENV variables:
    • $apmName - From $config['apm_name'] or $_ENV['APM_NAME']
    • $enabled - From $config['enabled'] or $_ENV['APM_ENABLED'] (accepts 'true', '1', 'false', '0', or boolean)
    • $sampleRate - From $config['sample_rate'] or $_ENV['APM_SAMPLE_RATE'] (clamped to 0.0-1.0)
    • $traceResponse, $traceDbQuery, $traceRequestBody - From config or environment
  3. Loads provider-specific configuration - Via loadConfiguration() (implemented by provider, can override common properties)
  4. Initializes root trace - Via initializeRootTrace() (implemented by provider)
  5. Registers shutdown function - To flush traces after HTTP response

4. Span Management

Throughout the request lifecycle, the framework calls:

  • startSpan() - Create child spans (database queries, controller operations, etc.)
  • endSpan() - End spans with final attributes and status
  • recordException() - Record exceptions on spans
  • flush() - Send traces to APM service (non-blocking)

5. Request Object Integration

The APM instance is stored in Request object for sharing:

$request->apm = $apmInstance;  // Primary property
$request->tracekit = $apmInstance;  // Backward compatibility

This allows all layers (Controller, UniversalQueryExecuter, Response, etc.) to access the same APM instance.

πŸ“š API Reference

ApmInterface

All APM providers must implement:

interface ApmInterface
{
    public function init(array $config = []): bool;
    public function isEnabled(): bool;
    public function startSpan(string $operationName, array $attributes = [], int $kind = self::SPAN_KIND_INTERNAL): array;
    public function endSpan(array $spanData, array $finalAttributes = [], ?string $status = self::STATUS_OK): void;
    public function recordException(array $spanData, \Throwable $exception): array;
    public function shouldTraceResponse(): bool;
    public function shouldTraceDbQuery(): bool;
    public function shouldTraceRequestBody(): bool;
    public static function determineStatusFromHttpCode(int $statusCode): string;
    public static function limitStringForTracing(string $value): string;
    public function getTraceId(): ?string;
    public function flush(): void;
    
    // Constants - OpenTelemetry standard
    public const SPAN_KIND_UNSPECIFIED = 0;
    public const SPAN_KIND_INTERNAL = 1;
    public const SPAN_KIND_SERVER = 2;
    public const SPAN_KIND_CLIENT = 3;
    public const SPAN_KIND_PRODUCER = 4;
    public const SPAN_KIND_CONSUMER = 5;
    
    public const STATUS_OK = 'OK';
    public const STATUS_ERROR = 'ERROR';
}

AbstractApm

Provides shared functionality:

  • Initialization - init() method for setup/configuration via CLI/GUI tools
  • Request management - getRequest(), getRequestBodyForTracing()
  • Configuration helpers - parseBooleanFlag(), parseSampleRate()
  • Sampling - shouldSample()
  • Utilities - limitStringForTracing(), determineStatusFromHttpCode()

Note: The init() method is designed for setup/configuration processes (CLI commands or GUI tools), not for runtime object creation. The constructor automatically loads configuration from environment variables during normal operation.

ApmFactory

Universal factory methods (no provider-specific code):

  • create(?Request $request, array $config = []): ?ApmInterface - Dynamically create APM instance based on APM_NAME. The $config array can override environment variables for runtime configuration.
  • isEnabled(): ?string - Lightweight check if APM is enabled. Returns APM provider name if enabled, null otherwise. Accepts APM_ENABLED values: "true", "1", or boolean true. Performance optimized - avoids double-checking APM_NAME.

Provider Naming Convention:

  • Provider names are standardized through the init() process by senior developers
  • Provider name from APM_NAME must be in correct PascalCase format (e.g., "TraceKit", "Datadog")
  • Class name format: Gemvc\Core\Apm\Providers\{ProviderName}\{ProviderName}Provider
  • Example: APM_NAME="TraceKit" β†’ Gemvc\Core\Apm\Providers\TraceKit\TraceKitProvider

πŸ’‘ Examples

Example: Simple APM Provider

class SimpleProvider extends AbstractApm
{
    protected function loadConfiguration(array $config = []): void
    {
        $this->enabled = !empty($_ENV['SIMPLE_API_KEY']);
    }
    
    protected function initializeRootTrace(): void
    {
        // Simple implementation
        $this->rootSpan = ['trace_id' => uniqid(), 'span_id' => uniqid()];
    }
    
    public function startSpan(string $operationName, array $attributes = [], int $kind = self::SPAN_KIND_INTERNAL): array
    {
        return ['span_id' => uniqid(), 'trace_id' => $this->getTraceId(), 'start_time' => time()];
    }
    
    public function endSpan(array $spanData, array $finalAttributes = [], ?string $status = self::STATUS_OK): void
    {
        // Log span data
        error_log("Span ended: " . json_encode($spanData));
    }
    
    public function recordException(array $spanData, \Throwable $exception): array
    {
        error_log("Exception: " . $exception->getMessage());
        return $spanData;
    }
    
    public function flush(): void
    {
        // Send traces
    }
}

Example: Using APM in Custom Code

// Get APM instance from Request
$apm = $request->apm;

if ($apm !== null && $apm->isEnabled()) {
    // Start a custom span
    $span = $apm->startSpan('custom-operation', [
        'custom.attribute' => 'value'
    ]);
    
    try {
        // Your code here
        $result = doSomething();
        
        // End span with success
        $apm->endSpan($span, ['result' => 'success'], ApmInterface::STATUS_OK);
    } catch (\Throwable $e) {
        // Record exception
        $apm->recordException($span, $e);
        $apm->endSpan($span, ['result' => 'error'], ApmInterface::STATUS_ERROR);
        throw $e;
    }
}

πŸ”— Related Packages

πŸ“ Environment Variables

Core APM Variables

  • APM_NAME - APM provider name (e.g., "TraceKit", "Datadog")
  • APM_ENABLED - Enable/disable APM (accepts "true", "1", "false", "0", or boolean; defaults to "true" if not set)
  • APM_SAMPLE_RATE - Sample rate for traces (0.0 to 1.0, where 1.0 = 100%; defaults to 1.0)
  • APM_TRACE_RESPONSE - Enable/disable response tracing (accepts "true", "1", "false", "0", or boolean; defaults to false)
  • APM_TRACE_DB_QUERY - Enable/disable database query tracing (accepts "true", "1", "false", "0", or boolean; defaults to false)
  • APM_TRACE_REQUEST_BODY - Enable/disable request body tracing (accepts "true", "1", "false", "0", or boolean; defaults to false)
  • APM_API_KEY - Unified API key (works for all providers)
  • APM_MAX_STRING_LENGTH - Maximum string length for tracing (default: 2000). Used by limitStringForTracing() to truncate long strings

Provider-Specific Variables

Each provider may define additional variables (e.g., TRACEKIT_API_KEY, DATADOG_API_KEY).

🀝 Contributing

To add a new APM provider:

  1. Create a new package following the structure above
  2. Submit a PR to apm-contracts to register your provider in ApmFactory
  3. Update documentation with your provider's configuration

πŸ“„ License

MIT License - see LICENSE file for details.

Credits

Part of the GEMVC PHP Framework built for Microservices ecosystem.