ZF2,在创建一个监听器时引发了异常。会话验证失败

时间:2015-01-20 08:30:02

标签: php zend-framework2 listener

让我们从错误的开始:我有一个acl监听器耦合到" route"事件最初似乎工作正常,但一段时间后会抛出一些错误。这些是错误:

  

致命错误:未捕获的异常   ' Zend的\会话\异常\的RuntimeException'与消息'会话   验证失败' ...

     

Zend \ ServiceManager \ Exception \ ServiceNotCreatedException:异常   在创建" authService";没有实例返回...

     

Zend \ ServiceManager \ Exception \ ServiceNotCreatedException:异常   在创建" MyAclListener";没有实例返回...

监听器的设置如下:

class MyAclListener implements ListenerAggregateInterface
{
    /** @var  AuthenticationServiceInterface */
    protected $_authService;

    function __construct (
        AuthenticationServiceInterface $authService ,
    )
    {
        $this->_authService     = $authService;
    }

    public function attach ( EventManagerInterface $events )
    {

        $this->listeners[ ] = $events->attach( 'route' , array( $this , 'checkAcl' ) );
    }

    public function checkAcl( MvcEvent $e ) {
\\\Do Something for which the authService is needed
}

然后,在module.config.php我有

$config = array(
    'service_manager' => array(
        'factories' => array(
            'MyAclListener' => 'MyAcl\Factory\MyAclListenerFactory'
        ) ,
    ) ,
    'listeners'       => array(
        'MyAclListener'
    ) ,
);

听众的工厂:

<?php

namespace MyAcl\Factory;

use MyAcl\Listener\MyAclListener;
use Zend\ServiceManager\FactoryInterface;
use Zend\ServiceManager\ServiceLocatorInterface;

class AclListenerFactory implements FactoryInterface
{

    public function createService(ServiceLocatorInterface $serviceLocator)
    {

        // Finally create the service
        return new MyAclListener(
            $serviceLocator->get('authService') ,
        );
    }

}

会话在Module.php中设置

   public function onBootstrap ( MvcEvent $e )
    {

        $serviceManager         = $e->getApplication()->getServiceManager();

        $config = $serviceManager->get('Configuration');
        $this->initSession( $config['session'] , $serviceManager->get('sessionCache')  );
}

public function initSession ( $config , $cache )
{

    $sessionConfig = new SessionConfig();
    $sessionConfig->setOptions( $config );
    $sessionManager = new SessionManager( $sessionConfig );

    $saveHandler = new Cache($cache);
    $sessionManager->setSaveHandler( $saveHandler );

    $sessionManager->getValidatorChain()->attach('session.validate', array(new HttpUserAgent(), 'isValid'));
    $sessionManager->getValidatorChain()->attach('session.validate', array(new RemoteAddr(), 'isValid'));

    $sessionManager->start();
    $sessionManager->regenerateId(true);

}

这些是本次会议的相关配置:

$config = array(
    'session'         => array(
        'cookie_lifetime' => 28800 ,
        'remember_me_seconds' => 28800 ,
        'use_cookies'         => true ,
        'cookie_httponly'     => true ,
    ) ,
    'caches'          => array(
        'sessionCache' => array(
            'adapter' => array(
                'name'     => 'memcached' ,
                'lifetime' => 604800 ,
                'options'  => array(
                    'servers'    => array(
                        array(
                            '127.0.0.1' , 11211
                        )
                    ) ,
                    'namespace'  => 'MySessionCache' ,
                    'liboptions' => array(
                        'COMPRESSION'     => true ,
                        'binary_protocol' => true ,
                        'no_block'        => true ,
                        'connect_timeout' => 100
                    )
                ) ,
                'plugins'  => array( 'serializer' )
            ) ,
        ) ,
    ) ,
)

在错误之后知道我的身份&#34;也许有用#34;丢失,登录用户已注销。

那么,我做错了什么?我该如何摆脱这些错误?甚至可能:是否有更好的解决方案来设置ACL检查和我的会话?

更新
我更改了会话设置:

在Module.php中

    public function onBootstrap ( MvcEvent $e )
    {

        $serviceManager = $e->getApplication()->getServiceManager();

        /* session */
        $this->expandInitialSessionConfig( $serviceManager , $serviceManager->get( 'sessionCache' ) );
}

public function expandInitialSessionConfig ( $sm , $cache )
{

    /** @var \Zend\Session\SessionManager $sessionManager */
    $sessionManager = $sm->get( 'Zend\Session\SessionManager' );

    $saveHandler = new Cache( $cache );
    $sessionManager->setSaveHandler( $saveHandler );

    $sessionManager->getValidatorChain()->attach( 'session.validate' , array( new HttpUserAgent() , 'isValid' ) );
    $sessionManager->getValidatorChain()->attach( 'session.validate' , array( new RemoteAddr() , 'isValid' ) );

}

在module.config.php

return array(
    'session_config'  => array(
        'cookie_lifetime'     => 28800 ,
        'remember_me_seconds' => 28800 ,
        'use_cookies'         => true ,
        'cookie_httponly'     => true ,
        'gc_probability'      => 10 ,
        'gc_maxlifetime'      => 7200 ,
        'cookie_domain'       => 'mydomain.com' ,
    ) ,
    'service_manager' => array(
        'factories' => array(
            'Zend\Session\SessionManager'         => 'Zend\Session\Service\SessionManagerFactory' ,
            'Zend\Session\Config\ConfigInterface' => 'Zend\Session\Service\SessionConfigFactory' ,
        ) ,
);

$ sessionManager-&gt; regenerateId(true)已被移动,因此它现在只在登录时发生。

2 个答案:

答案 0 :(得分:1)

您问ServiceLocator AclListenerFactory中的authService authService,但找不到任何类似名称导致第二个异常的内容。 因为找不到$serviceLocator->get('authService') , ,所以工厂不会返回监听器。这是最后一个例外。

在你的代码中写道:

authService

第一个例外与某些会话验证失败有关。所有问题的原因可能都在那里。没有会话验证结果可能没有有效的authService因此链接其他两个例外。 这个{{1}}在哪里注册或创建?这在您当前共享的代码中不可见。这项服务如何依赖于&#34;验证&#34;那是失败的......

如果您解决了会话验证问题,那么一切都可能没问题。

答案 1 :(得分:0)

我严重怀疑这是&#34;最佳实践&#34;,但是从我的监听器的构造函数中移除了那些依赖关系(自原始问题以来添加了更多)依赖于会话解决了问题。我知道在我的代码中我需要它们的第一时间从ServiceLocator获取它们并将它们存储在我的监听器类的受保护的参数中以供进一步使用。然而,就像我说的那样,这不再是适当的依赖注入,但至少它是有效的。

function __construct (Logger $_myLog )
{
    $this->_myLog          = $_myLog;
    // removed here: $this->_authService     = $authService;

}

然后:

public function checkAcl ( MvcEvent $e )
{
   $sm = $e->getApplication()->getServiceManager();
   $this->_authService = $sm->get('authService');
   ....
}