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

New redirect strategy #480

Open
wants to merge 15 commits into
base: master
Choose a base branch
from
7 changes: 5 additions & 2 deletions config/module.config.php
Original file line number Diff line number Diff line change
Expand Up @@ -16,14 +16,17 @@
),
),
'controllers' => array(
'invokables' => array(
'zfcuser' => 'ZfcUser\Controller\UserController',
'factories' => array(
'zfcuser' => 'ZfcUser\Factory\Controller\UserControllerFactory',
),
),
'service_manager' => array(
'aliases' => array(
'zfcuser_zend_db_adapter' => 'Zend\Db\Adapter\Adapter',
),
'factories' => array(
'zfcuser_redirect_callback' => 'ZfcUser\Factory\Controller\RedirectCallbackFactory'
)
),
'router' => array(
'routes' => array(
Expand Down
94 changes: 94 additions & 0 deletions src/ZfcUser/Controller/RedirectCallback.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,94 @@
<?php

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Missing headers

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As discussed, this may be done later

namespace ZfcUser\Controller;

use Zend\Mvc\Controller\Plugin\Redirect;
use Zend\Mvc\Router\RouteInterface;
use Zend\Mvc\Router\Exception;
use Zend\Http\PhpEnvironment\Request;
use Zend\Http\PhpEnvironment\Response;
use ZfcUser\Options\ModuleOptions;

class RedirectCallback
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Missing docblock

{
/** @var RouteInterface */
protected $router;

/** @var Response */
protected $response;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Request and response should not be localized as properties.


/** @var Request */
protected $request;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same as above


/** @var ModuleOptions */
protected $options;

/**
* @param RouteInterface $router
* @param Response $response
* @param Request $request
* @param ModuleOptions $options
*/
public function __construct(RouteInterface $router, Response $response, Request $request, ModuleOptions $options)
{
$this->router = $router;
$this->request = $request;
$this->response = $response;
$this->options = $options;
}

/**
* @return Response
*/
public function __invoke()
{
$routeMatch = $this->router->match($this->request);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You can inject the MVC routematch instead of matching again, see below in the factory

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

$response = $this->response;
$response->getHeaders()->addHeaderLine('Location', $this->getRedirect($routeMatch->getMatchedRouteName(), $routeMatch->getParam('redirect', false)));
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would extract a variable here to make it more readble:

$url = $this->getRedirectUrl(
    $routeMatch->getMatchedRouteName(),
    $routeMatch->getParam('redirect', false)
);

Note I also changed getRedirect() to getRedirectUrl() to show it should return a url.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Another thing: where do you the the route parameter from? I don't see a route definition in this PR which adds a parameter to the route called redirect, so I assume now this is always false?

$response->setStatusCode(302);
return $response;
}

/**
* @param $route
* @return bool
*/
protected function routeExists($route)
{
try{
$this->router->assemble($route);
} catch (Exception\RuntimeException $e) {
return false;
}
return true;
}

/**
* Returns the url to redirect to based on current route.
* If $redirect is set and the option to use redirect is set to true, it will return the $redirect url.
*
* @param string $currentRoute
* @param bool $redirect
* @return mixed
*/
protected function getRedirect($currentRoute, $redirect = false)
{
if (!$this->options->getUseRedirectParameterIfPresent() || ($redirect && !$this->routeExists($redirect))) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would make this more readable:

$useRedirect  = $this->options->getUseRedirectParameterIfPresent();
$routeExists  = ($redirect && $this->routeExists($redirect));

if (!$useRedirect || !$routeExists) {
    $redirect = false;
}

$redirect = false;
}

switch ($currentRoute) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If you make this thing a controller plugin (which I would advice), you can also reuse the url and/or redirect controller plugins. This just duplicates logic of assembling and returning responses

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agree, this should be a controller plugin. I am refactoring as we speak.

case 'zfcuser/login':
$route = ($redirect) ?: $this->options->getLoginRedirectRoute();
return $this->router->assemble([], ['name' => $route]);
break;
case 'zfcuser/logout':
$route = ($redirect) ?: $this->options->getLogoutRedirectRoute();
return $this->router->assemble([], ['name' => $route]);
break;
default:
return $this->router->assemble([], ['name' => 'zfcuser']);
}
}

}
22 changes: 14 additions & 8 deletions src/ZfcUser/Controller/UserController.php
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,16 @@ class UserController extends AbstractActionController
*/
protected $options;

/**
* @var Callable $redirectCallback
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

callable

*/
protected $redirectCallback;

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Remove this empty newline

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Was 2 before, 1 now as it should be.

public function __construct($redirectCallback)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does ZfcUser require 5.3? Otherwise, callable can be given as a hint. Also, missing docblock

{
$this->redirectCallback = $redirectCallback;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Check against is_callable() here

}

/**
* User page
*/
Expand Down Expand Up @@ -116,13 +126,8 @@ public function logoutAction()
$this->zfcUserAuthentication()->getAuthAdapter()->logoutAdapters();
$this->zfcUserAuthentication()->getAuthService()->clearIdentity();

$redirect = $this->params()->fromPost('redirect', $this->params()->fromQuery('redirect', false));

if ($this->getOptions()->getUseRedirectParameterIfPresent() && $redirect) {
return $this->redirect()->toRoute($redirect);
}

return $this->redirect()->toRoute($this->getOptions()->getLogoutRedirectRoute());
$redirect = $this->redirectCallback;
return $redirect();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Newline before this line if possible

}

/**
Expand Down Expand Up @@ -165,7 +170,8 @@ public function authenticateAction()
$route = $route($this->zfcUserAuthentication()->getIdentity());
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why is a callback support needed here?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Now that we have this redirect callback, it's not needed, but it would be BC to remove.
Now that i think about it, this PR is against 2.x branch, so i could probably remove.

}

return $this->redirect()->toRoute($route);
$redirect = $this->redirectCallback;
return $redirect();
}

/**
Expand Down
21 changes: 21 additions & 0 deletions src/ZfcUser/Factory/Controller/RedirectCallbackFactory.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
<?php
namespace ZfcUser\Factory\Controller;

use Zend\ServiceManager\FactoryInterface;
use Zend\ServiceManager\ServiceLocatorInterface;
use ZfcUser\Controller\RedirectCallback;

class RedirectCallbackFactory implements FactoryInterface
{
/**
* {@inheritDoc}
*/
public function createService(ServiceLocatorInterface $serviceLocator)
{
$router = $serviceLocator->get('Router');
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you add some IDE hints here? They help with static analysis

$response = $serviceLocator->get('Response');
$request = $serviceLocator->get('Request');
$options = $serviceLocator->get('zfcuser_module_options');
return new RedirectCallback($router, $response, $request, $options);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd inject the route match, so you don't need to match twice. On top of my head:

$mvcEvent   = $serviceLocator->get('Application')->getMvcEvent();
$routeMatch = $mvcEvent->getRouteMatch();

}
}
25 changes: 25 additions & 0 deletions src/ZfcUser/Factory/Controller/UserControllerFactory.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
<?php
namespace ZfcUser\Factory\Controller;

use Zend\Mvc\Controller\ControllerManager;
use Zend\ServiceManager\FactoryInterface;
use Zend\ServiceManager\ServiceLocatorInterface;
use ZfcUser\Authentication\Adapter;
use ZfcUser\Controller\UserController;

class UserControllerFactory implements FactoryInterface
{
/**
* {@inheritDoc}
*/
public function createService(ServiceLocatorInterface $controllerManager)
{
/** @var ControllerManager $controllerManager*/
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

/*, not /**

$serviceManager = $controllerManager->getServiceLocator();

$redirectCallback = $serviceManager->get('zfcuser_redirect_callback');
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same as above - hints

$controller = new UserController($redirectCallback);

return $controller;
}
}