Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

MonitoredMiddleware crashes if probe is not present #71

Open
Pixelshaped opened this issue Apr 8, 2024 · 6 comments
Open

MonitoredMiddleware crashes if probe is not present #71

Pixelshaped opened this issue Apr 8, 2024 · 6 comments

Comments

@Pixelshaped
Copy link

Pixelshaped commented Apr 8, 2024

Following the Symfony Messenger configuration https://docs.blackfire.io/php/integrations/symfony/messenger

framework:
  messenger:
    buses:
      messenger.bus.default:
        middleware:
          - 'Blackfire\Bridge\Symfony\MonitoredMiddleware'

services:
  Blackfire\Bridge\Symfony\MonitoredMiddleware: ~

I've got a crash in MonitoredMiddleware at line 32: Class "BlackfireProbe" not found.

And it's true: on my local machine, I don't have the probe installed.

I fixed it by adding

when@prod:

So that MonitoredMiddleware only works on my prod env.

But I still think that if phpversion('blackfire') returns false, it should fail silently or log.

@thomasdiluccio
Copy link
Contributor

I believe we should replace

if (version_compare(phpversion('blackfire'), '1.78.0', '>=')) {
 ....
}

with:

if (phpversion('blackfire') && version_compare(phpversion('blackfire'), '1.78.0', '>=')) {
 ....
}

to ensure the conditions fails if phpversion('blackfire') returns false;

@Pixelshaped
Copy link
Author

Pixelshaped commented Apr 8, 2024

Wouldn't be sufficient as the BlackfireProbeclass is used in both branches of the condition.

Something along those lines

<?php

/*
 * This file is part of the Blackfire SDK package.
 *
 * (c) Blackfire <[email protected]>
 *
 * For the full copyright and license information, please view the LICENSE
 * file that was distributed with this source code.
 */

namespace Blackfire\Bridge\Symfony;

use Symfony\Component\Messenger\Envelope;
use Symfony\Component\Messenger\Middleware\MiddlewareInterface;
use Symfony\Component\Messenger\Middleware\StackInterface;
use Symfony\Component\Messenger\Stamp\ConsumedByWorkerStamp;

class MonitoredMiddleware implements MiddlewareInterface
{
    public function handle(Envelope $envelope, StackInterface $stack): Envelope
    {
        if (null === $envelope->last(ConsumedByWorkerStamp::class)) {
            return $stack->next()->handle($envelope, $stack);
        }

        $txName = \get_class($envelope->getMessage());

        $blackfireVersion = phpversion('blackfire');
        if ($blackfireVersion !== false){
            if (version_compare($blackfireVersion, '1.78.0', '>=')) {
                \BlackfireProbe::startTransaction($txName);
            } else {
                \BlackfireProbe::startTransaction();
                \BlackfireProbe::setTransactionName($txName);
            }
        }

        try {
            return $stack->next()->handle($envelope, $stack);
        } finally {
            if($blackfireVersion !== false) {
                \BlackfireProbe::stopTransaction();
            }
        }
    }
}

Would work.

@romainneutron
Copy link
Member

Hello,

Silently failing is not an option. I prefer misconfiguration to explicitely fail so you can fix the configuration.
Using the Blackfire Middleware without the extension is unfortunately not possible, the only patch I could propose is an explicit RuntimeException:

diff --git a/languages/php/sdk/src/Blackfire/Bridge/Symfony/MonitoredMiddleware.php b/languages/php/sdk/src/Blackfire/Bridge/Symfony/MonitoredMiddleware.php
index cd58853e73..8fd10a1622 100644
--- a/languages/php/sdk/src/Blackfire/Bridge/Symfony/MonitoredMiddleware.php
+++ b/languages/php/sdk/src/Blackfire/Bridge/Symfony/MonitoredMiddleware.php
@@ -18,6 +18,13 @@ use Symfony\Component\Messenger\Stamp\ConsumedByWorkerStamp;
 
 class MonitoredMiddleware implements MiddlewareInterface
 {
+    public function __construct()
+    {
+        if (!extension_loaded('blackfire')) {
+            throw new \RuntimeException(sprintf('Extension "blackfire" is required to use "%s".', __CLASS__));
+        }
+    }
+
     public function handle(Envelope $envelope, StackInterface $stack): Envelope
     {
         if (null === $envelope->last(ConsumedByWorkerStamp::class)) {

Is it good for you ?

@Pixelshaped
Copy link
Author

Pixelshaped commented Apr 8, 2024

Not sure why silently failing is not an option.

As an example, when you implement the MonitorableCommandInterface on a Command (as per documentation https://docs.blackfire.io/php/integrations/symfony/cli-commands-monitoring) and you don't have the probe installed... nothing happens. The command runs and is not monitored and that's it.

How would you then justify this difference of behavior?

I think failing silently is a perfectly sane option in some cases.

If you want to throw an error though, it won't change much for me:

  • Program will halt several lines earlier
  • Message will be a little less cryptic than Class "BlackfireProbe" not found. but the previous one wasn't undecipherable either.
  • Notification will not be handled properly

Not sure why the absence of the Blackfire extension should prevent this aspect of the program to run normally, while about any other aspect of the program runs fine in the absence of the probe.

One might even say that it's the default behavior of Blackfire to never report that it's correctly installed 😄

@romainneutron
Copy link
Member

Hello,

Failing silently leads to situation where a user expects something that is not happening.
If a user explicitely uses a feature and it can not be used, it is expected to fail rather than do nothing silently.

Let's take an example: Imagine you're using Symfony, and depending on some environment configuration, some messages would not be dispatched by the event dispatcher, silently. There's high possibility it drives the user mad.

Here, it's the same kind of situation: you activated a feature, on purpose, in your configuration. But given your environment is missing an extension, it would not work. This would look like problematic for newcomers, they would not understand why it's not working.
If there's an explicit exception message, it's obvious.

Not sure why the absence of the Blackfire extension should prevent this aspect of the program to run normally, while about any other aspect of the program runs fine in the absence of the probe.

Because you explicitely activated the feature in your configuration.

As an example, when you implement the MonitorableCommandInterface on a Command (as per documentation https://docs.blackfire.io/php/integrations/symfony/cli-commands-monitoring) and you don't have the probe installed... nothing happens. The command runs and is not monitored and that's it.

That's correct, you're right. However we can not change the behavior without possibly breaking BC.

@Pixelshaped
Copy link
Author

I get what you're saying, but this behavior is a bit of a discrepancy vs how Blackfire works usually which is failing silently on about every call when the extension is not installed (as illustrated by Commands or any http request).

Do what you must.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

3 participants