ZF2在控制器外部使用重定向

时间:2013-01-22 21:19:08

标签: redirect controller zend-framework2

我正在处理一个在Module.php中调用并附加到bootstrap的ACL。

显然,ACL会限制对网站某些区域的访问,从而需要重定向。但是,当尝试使用控制器插件进行重定向时,它不起作用,因为插件似乎需要一个控制器。

从控制器外部重定向外部的最佳方法是什么? vanilla header()函数不合适,因为我需要使用已定义的路由。

任何帮助都会很棒!

干杯 -

1 个答案:

答案 0 :(得分:9)

通常,您希望通过返回响应来短路调度过程。在routedispatch期间,您可以返回响应以停止常规代码流停止并直接完成结果。如果是ACL检查,您很可能需要提前返回该响应并重定向到用户的登录页面。

您可以在控制器中构建响应,也可以检查插件的返回值,并在响应时重定向。请注意,第二种方法类似于how the PRG plugin works

第一种方法的一个例子:

use Zend\Mvc\Controller\AbstractActionController;

class MyController extends AbstractActionController
{
    public function fooAction()
    {
        if (!$this->aclAllowsAccess()) {
            // Use redirect plugin to redirect
            return $this->redirect('user/login');
        }

        // normal code flow
    }
}

像PRG插件这样的例子有效:

use Zend\Mvc\Controller\AbstractActionController;
use Zend\Http\Response;

class MyController extends AbstractActionController
{
    public function fooAction()
    {
        $result = $this->aclCheck();

        if ($result instanceof Response) {
            // Use return value to short-circuit
            return $result
        }

        // normal code flow
    }
}

插件可能看起来像这样(在第二种情况下):

use Zend\Mvc\Controller\Plugin\AbstractPlugin;

class AclCheck extends AbstractPlugin
{
    public function __invoke()
    {
        // Check the ACL
        if (false === $result) {
            $controller = $this->getController();
            $redirector = $controller->getPluginManager()->get('Redirect');
            $response = $redirector->toRoute('user/login');

            return $response;
        }
    }
}

在你的问题中你说:

  

[...]它不起作用,因为插件似乎需要一个控制器。

当您想在插件中执行$this->getController()时,这可能是控制器插件中的问题。您必须扩展Zend\Mvc\Controller\Plugin\AbstractPlugin或实施Zend\Mvc\Controller\Plugin\PluginInterface以确保您的ACL插件注入控制器。

如果您不想这样做,可以直接返回您自己创建的回复。它有点不太灵活,你创建一个响应对象,而已经有一个响应对象(导致可能与两个响应冲突),但插件代码将改变如下:

use Zend\Mvc\Controller\Plugin\AbstractPlugin;
use Zend\Http\PhpEnvironment\Response;

class AclCheck extends AbstractPlugin
{
    public function __invoke()
    {
        // Check the ACL
        if (false === $result) {
            $response = new Response;
            $response->setStatusCode(302);
            $response->getHeaders()
                     ->addHeaderLine('Location', '/user/login');

            return $response;
        }
    }
}