%PDF- %PDF-
Direktori : /home/graphicd/www/vebto/vendor/sentry/sentry-laravel/src/Sentry/Laravel/ |
Current File : /home/graphicd/www/vebto/vendor/sentry/sentry-laravel/src/Sentry/Laravel/ServiceProvider.php |
<?php namespace Sentry\Laravel; use Illuminate\Contracts\Http\Kernel as HttpKernelInterface; use Illuminate\Foundation\Application as Laravel; use Illuminate\Foundation\Http\Kernel as HttpKernel; use Illuminate\Log\LogManager; use Laravel\Lumen\Application as Lumen; use RuntimeException; use Sentry\ClientBuilder; use Sentry\ClientBuilderInterface; use Sentry\Integration as SdkIntegration; use Sentry\Laravel\Console\PublishCommand; use Sentry\Laravel\Console\TestCommand; use Sentry\Laravel\Http\LaravelRequestFetcher; use Sentry\Laravel\Http\SetRequestIpMiddleware; use Sentry\Laravel\Http\SetRequestMiddleware; use Sentry\Laravel\Tracing\ServiceProvider as TracingServiceProvider; use Sentry\SentrySdk; use Sentry\State\Hub; use Sentry\State\HubInterface; class ServiceProvider extends BaseServiceProvider { /** * List of configuration options that are Laravel specific and should not be sent to the base PHP SDK. */ private const LARAVEL_SPECIFIC_OPTIONS = [ // We do not want these settings to hit the PHP SDK because they are Laravel specific and the PHP SDK will throw errors 'tracing', 'breadcrumbs', // We resolve the integrations through the container later, so we initially do not pass it to the SDK yet 'integrations', // This is kept for backwards compatibility and can be dropped in a future breaking release 'breadcrumbs.sql_bindings', // The base namespace for controllers to strip of the beginning of controller class names 'controllers_base_namespace', ]; /** * Boot the service provider. */ public function boot(): void { $this->app->make(HubInterface::class); if ($this->hasDsnSet()) { $this->bindEvents(); if ($this->app instanceof Lumen) { $this->app->middleware(SetRequestMiddleware::class); $this->app->middleware(SetRequestIpMiddleware::class); } elseif ($this->app->bound(HttpKernelInterface::class)) { /** @var \Illuminate\Foundation\Http\Kernel $httpKernel */ $httpKernel = $this->app->make(HttpKernelInterface::class); if ($httpKernel instanceof HttpKernel) { $httpKernel->pushMiddleware(SetRequestMiddleware::class); $httpKernel->pushMiddleware(SetRequestIpMiddleware::class); } } } if ($this->app->runningInConsole()) { if ($this->app instanceof Laravel) { $this->publishes([ __DIR__ . '/../../../config/sentry.php' => config_path(static::$abstract . '.php'), ], 'config'); } $this->registerArtisanCommands(); } } /** * Register the service provider. */ public function register(): void { if ($this->app instanceof Lumen) { $this->app->configure(static::$abstract); } $this->mergeConfigFrom(__DIR__ . '/../../../config/sentry.php', static::$abstract); $this->configureAndRegisterClient($this->getUserConfig()); if (($logManager = $this->app->make('log')) instanceof LogManager) { $logManager->extend('sentry', function ($app, array $config) { return (new LogChannel($app))($config); }); } } /** * Bind to the Laravel event dispatcher to log events. */ protected function bindEvents(): void { $userConfig = $this->getUserConfig(); $handler = new EventHandler($this->app, $userConfig); $handler->subscribe(); if ($this->app->bound('octane')) { $handler->subscribeOctaneEvents(); } if ($this->app->bound('queue')) { $handler->subscribeQueueEvents($this->app->queue); } if (isset($userConfig['send_default_pii']) && $userConfig['send_default_pii'] !== false) { $handler->subscribeAuthEvents(); } } /** * Register the artisan commands. */ protected function registerArtisanCommands(): void { $this->commands([ TestCommand::class, PublishCommand::class, ]); } /** * Configure and register the Sentry client with the container. */ protected function configureAndRegisterClient(): void { $userConfig = $this->getUserConfig(); if (isset($userConfig['controllers_base_namespace'])) { Integration::setControllersBaseNamespace($userConfig['controllers_base_namespace']); } $this->app->bind(ClientBuilderInterface::class, function () { $basePath = base_path(); $userConfig = $this->getUserConfig(); foreach (self::LARAVEL_SPECIFIC_OPTIONS as $laravelSpecificOptionName) { unset($userConfig[$laravelSpecificOptionName]); } $options = \array_merge( [ 'prefixes' => [$basePath], 'in_app_exclude' => ["{$basePath}/vendor"], ], $userConfig ); // When we get no environment from the (user) configuration we default to the Laravel environment if (empty($options['environment'])) { $options['environment'] = $this->app->environment(); } $clientBuilder = ClientBuilder::create($options); // Set the Laravel SDK identifier and version $clientBuilder->setSdkIdentifier(Version::SDK_IDENTIFIER); $clientBuilder->setSdkVersion(Version::SDK_VERSION); return $clientBuilder; }); $this->app->singleton(HubInterface::class, function ($app) { /** @var \Sentry\ClientBuilderInterface $clientBuilder */ $clientBuilder = $this->app->make(ClientBuilderInterface::class); $options = $clientBuilder->getOptions(); $userIntegrations = $this->resolveIntegrationsFromUserConfig(); $options->setIntegrations(function (array $integrations) use ($options, $userIntegrations, $app) { if ($options->hasDefaultIntegrations()) { // Remove the default error and fatal exception listeners to let Laravel handle those // itself. These event are still bubbling up through the documented changes in the users // `ExceptionHandler` of their application or through the log channel integration to Sentry $integrations = array_filter($integrations, static function (SdkIntegration\IntegrationInterface $integration): bool { if ($integration instanceof SdkIntegration\ErrorListenerIntegration) { return false; } if ($integration instanceof SdkIntegration\ExceptionListenerIntegration) { return false; } if ($integration instanceof SdkIntegration\FatalErrorListenerIntegration) { return false; } // We also remove the default request integration so it can be readded // after with a Laravel specific request fetcher. This way we can resolve // the request from Laravel instead of constructing it from the global state if ($integration instanceof SdkIntegration\RequestIntegration) { return false; } return true; }); $integrations[] = new SdkIntegration\RequestIntegration( new LaravelRequestFetcher($app) ); } return array_merge($integrations, $userIntegrations); }); $hub = new Hub($clientBuilder->getClient()); SentrySdk::setCurrentHub($hub); return $hub; }); $this->app->alias(HubInterface::class, static::$abstract); } /** * Resolve the integrations from the user configuration with the container. * * @return array */ private function resolveIntegrationsFromUserConfig(): array { // Default Sentry Laravel SDK integrations $integrations = [ new Integration, new Integration\ExceptionContextIntegration, ]; $userConfig = $this->getUserConfig(); $integrationsToResolve = $userConfig['integrations'] ?? []; $enableDefaultTracingIntegrations = $userConfig['tracing']['default_integrations'] ?? true; if ($enableDefaultTracingIntegrations && $this->couldHavePerformanceTracingEnabled()) { $integrationsToResolve = array_merge($integrationsToResolve, TracingServiceProvider::DEFAULT_INTEGRATIONS); } foreach ($integrationsToResolve as $userIntegration) { if ($userIntegration instanceof SdkIntegration\IntegrationInterface) { $integrations[] = $userIntegration; } elseif (\is_string($userIntegration)) { $resolvedIntegration = $this->app->make($userIntegration); if (!$resolvedIntegration instanceof SdkIntegration\IntegrationInterface) { throw new RuntimeException( sprintf( 'Sentry integrations must be an instance of `%s` got `%s`.', SdkIntegration\IntegrationInterface::class, get_class($resolvedIntegration) ) ); } $integrations[] = $resolvedIntegration; } else { throw new RuntimeException( sprintf( 'Sentry integrations must either be a valid container reference or an instance of `%s`.', SdkIntegration\IntegrationInterface::class ) ); } } return $integrations; } /** * Get the services provided by the provider. * * @return array */ public function provides() { return [static::$abstract]; } }