How to perform an HTTP request from a Module#onBootstrap(...) in ZF2?

554 Views Asked by At

In a Module class of a Zend Framework 2 application I'm sending an HTTP request to a special endpoint to get some information about the user. It works, when I do this within a factory (s. the 1st approach in the code below). It also works within a listener definition in the Module#onBootstrap(...) (s. the 2nd approach). But when I try to perform the call directly from the Module#onBootstrap(...) (s. the 3d approach), the call fails with the error:

PHP Fatal error: Uncaught exception 'Zend\\Http\\Client\\Adapter\\Exception\\TimeoutException' with message 'Read timed out after 30 seconds' in /var/www/path/to/project/vendor/zendframework/zend-http/src/Client/Adapter/Socket.php:600
Stack trace:
#0 /var/www/path/to/project/vendor/zendframework/zend-http/src/Client/Adapter/Socket.php(412): Zend\\Http\\Client\\Adapter\\Socket->_checkSocketReadTimeout()
#1 /var/www/path/to/project/vendor/zendframework/zend-http/src/Client.php(1389): Zend\\Http\\Client\\Adapter\\Socket->read()
#2 /var/www/path/to/project/vendor/zendframework/zend-http/src/Client.php(893): Zend\\Http\\Client->doRequest(Object(Zend\\Uri\\Http), 'POST', false, Array, '')
#3 /var/www/path/to/project/module/MyApi/src/MyApi/Module.php(158): Zend\\Http\\Client->send()
#4 /var/www/path/to/project/module/MyApi/src/MyApi/Module.php(33): MyApi\\Module->retrieveUserInfosEndpoint(Object(ZF\\ContentNegotiation\\Request), 'http://my-project...')
#5 [intern in /var/www/path/to/project/vendor/zendframework/zend-http/src/Client/Adapter/Socket.php on line 600

Why does the error occur? How to send an HTTP request from a Module#onBootstrap(...)?


namespace MyModule;
...
class Module
{
    protected $userInfo;
    public function onBootstrap(MvcEvent $mvcEvent)
    {
        // 3d approach -- it does NOT work
        $userInfoEndpointUrl = $serviceManager->get('Config')['user_info_endpoint_url'];
        $request = $serviceManager->get('Request');
        $this->userInfo = $this->retrieveUserInfosEndpoint($request, $userInfoEndpointUrl);
        ...
        $halPlugin->getEventManager()->attach('myeventname', function ($event) use (..., $serviceManager) {
            // 2nd approach -- it works
            $userInfoEndpointUrl = $serviceManager->get('Config')['user_info_endpoint_url'];
            $request = $serviceManager->get('Request');
            $this->userInfo = $this->retrieveUserInfosEndpoint($request, $userInfoEndpointUrl);
            /*
            PHP Fatal error:
            Uncaught exception 'Zend\\Http\\Client\\Adapter\\Exception\\TimeoutException'
            with message 'Read timed out after 30 seconds'
            in /var/www/path/to/project/vendor/zendframework/zend-http/src/Client/Adapter/Socket.php:600
            */
            ...

        });
        ...
    }
    ...
    public function getServiceConfig()
    {
        return array(
            'factories' => array(
                'MyModule\\V1\\Rest\\Foo\\FooService' => function(ServiceManager $serviceManager) {
                    // 1st approach -- it works
                    $userInfoEndpointUrl = $serviceManager->get('Config')['user_info_endpoint_url'];
                    $request = $serviceManager->get('Request');
                    $this->userInfo = $this->retrieveUserInfosEndpoint($request, $userInfoEndpointUrl);
                    ...
                    return $fooService;
                },
                ...
            ),
            ...
        );
    }

    private function retrieveUserInfosEndpoint($request, $userInfoEndpointUrl)
    {
        $authorizationHeaderValue = $request->getHeader('Authorization')->getFieldValue();
        $client = new Client();
        $client->setUri($userInfoEndpointUrl);
        $client->setMethod('POST');
        $client->setOptions(['sslverifypeer' => false]);
        $client->setHeaders(['Authorization' => $authorizationHeaderValue]);
        $client->setOptions([
            'maxredirects' => 10,
            'timeout'      => 30,
        ]);
        $client->setParameterPost([]);
        $response = $client->send();
        $userInfo = json_decode($response->getContent(), true);
        return $userInfo;
    }

}
1

There are 1 best solutions below

2
On

Well, I see now -- it actually can not work. Performing a new request starts the whole event chain recursively and causes an infinite loop.

In my case to avoid multiple requesting my userinfo endpoint I put the call into a factory and can use it in other factories as well as in event listeners:

namespace MyModule;
...
class Module
{
    public function onBootstrap(MvcEvent $mvcEvent)
    {
        ...
        $halPlugin->getEventManager()->attach('myeventname', function ($event) use (..., $serviceManager) {
            ...
            $userInfo = $serviceManager->get('MyModule\\Service\\UserInfo');
            ...

        });
        ...
    }
    ...
    public function getServiceConfig()
    {
        return array(
            'factories' => array(
                'MyModule\\V1\\Rest\\Foo\\FooService' => function(ServiceManager $serviceManager) {
                    ...
                    $userInfo = $serviceManager->get('MyModule\\Service\\UserInfo');
                    ...
                    return $fooService;
                },
                'MyModule\\Service\\UserInfo' => function(ServiceManager $serviceManager) {
                    $userInfoEndpointUrl = $serviceManager->get('Config')['user_info_endpoint_url'];
                    $request = $serviceManager->get('Request');
                    $userInfo = $this->retrieveUserInfosEndpoint($request, $userInfoEndpointUrl);
                    return $userInfo;
                },
                ...
            ),
            ...
        );
    }
}