1
0
Fork 0
mirror of https://we.phorge.it/source/phorge.git synced 2025-01-05 12:21:02 +01:00

Preserve "next" URI by using OAuth 'state' parameter

Summary:
When a user clicks a link like /T32 and has to login, redirect them
to the resource once they've authenticated if possible. OAuth has a param
specifically for this, called 'state', so use it if possible. Facebook
supports it but Github does not.

Test Plan:
logged in with facebook after viewing /D20

Reviewed By: aran
Reviewers: aran
CC: aran, epriestley
Differential Revision: 61
This commit is contained in:
epriestley 2011-03-07 19:29:51 -08:00
parent 0bcbd0f158
commit a100d97ed5
7 changed files with 40 additions and 7 deletions

View file

@ -30,6 +30,11 @@ class PhabricatorLoginController extends PhabricatorAuthController {
return id(new AphrontRedirectResponse())->setURI('/'); return id(new AphrontRedirectResponse())->setURI('/');
} }
$next_uri = $this->getRequest()->getPath();
if ($next_uri == '/login/') {
$next_uri = null;
}
$password_auth = PhabricatorEnv::getEnvConfig('auth.password-auth-enabled'); $password_auth = PhabricatorEnv::getEnvConfig('auth.password-auth-enabled');
$forms = array(); $forms = array();
@ -76,6 +81,7 @@ class PhabricatorLoginController extends PhabricatorAuthController {
$form $form
->setUser($request->getUser()) ->setUser($request->getUser())
->setAction('/login/') ->setAction('/login/')
->addHiddenInput('next', $next_uri)
->appendChild( ->appendChild(
id(new AphrontFormTextControl()) id(new AphrontFormTextControl())
->setLabel('Username/Email') ->setLabel('Username/Email')
@ -97,6 +103,8 @@ class PhabricatorLoginController extends PhabricatorAuthController {
$forms['Phabricator Login'] = $form; $forms['Phabricator Login'] = $form;
} }
$oauth_state = $next_uri;
$providers = array( $providers = array(
PhabricatorOAuthProvider::PROVIDER_FACEBOOK, PhabricatorOAuthProvider::PROVIDER_FACEBOOK,
PhabricatorOAuthProvider::PROVIDER_GITHUB, PhabricatorOAuthProvider::PROVIDER_GITHUB,
@ -140,6 +148,7 @@ class PhabricatorLoginController extends PhabricatorAuthController {
->addHiddenInput('client_id', $client_id) ->addHiddenInput('client_id', $client_id)
->addHiddenInput('redirect_uri', $redirect_uri) ->addHiddenInput('redirect_uri', $redirect_uri)
->addHiddenInput('scope', $minimum_scope) ->addHiddenInput('scope', $minimum_scope)
->addHiddenInput('state', $oauth_state)
->setUser($request->getUser()) ->setUser($request->getUser())
->setMethod('GET') ->setMethod('GET')
->appendChild( ->appendChild(

View file

@ -23,6 +23,7 @@ class PhabricatorOAuthLoginController extends PhabricatorAuthController {
private $accessToken; private $accessToken;
private $tokenExpires; private $tokenExpires;
private $oauthState;
public function shouldRequireLogin() { public function shouldRequireLogin() {
return false; return false;
@ -120,6 +121,7 @@ class PhabricatorOAuthLoginController extends PhabricatorAuthController {
'account?</p>'); 'account?</p>');
$dialog->addHiddenInput('token', $provider->getAccessToken()); $dialog->addHiddenInput('token', $provider->getAccessToken());
$dialog->addHiddenInput('expires', $oauth_info->getTokenExpires()); $dialog->addHiddenInput('expires', $oauth_info->getTokenExpires());
$dialog->addHiddenInput('state', $this->oauthState);
$dialog->addSubmitButton('Link Accounts'); $dialog->addSubmitButton('Link Accounts');
$dialog->addCancelButton('/settings/page/'.$provider_key.'/'); $dialog->addCancelButton('/settings/page/'.$provider_key.'/');
@ -133,6 +135,12 @@ class PhabricatorOAuthLoginController extends PhabricatorAuthController {
->setURI('/settings/page/'.$provider_key.'/'); ->setURI('/settings/page/'.$provider_key.'/');
} }
$next_uri = '/';
if ($this->oauthState) {
// Make sure a blind redirect to evil.com is impossible.
$uri = new PhutilURI($this->oauthState);
$next_uri = $uri->getPath();
}
// Login with known auth. // Login with known auth.
@ -151,7 +159,7 @@ class PhabricatorOAuthLoginController extends PhabricatorAuthController {
$request->setCookie('phusr', $known_user->getUsername()); $request->setCookie('phusr', $known_user->getUsername());
$request->setCookie('phsid', $session_key); $request->setCookie('phsid', $session_key);
return id(new AphrontRedirectResponse()) return id(new AphrontRedirectResponse())
->setURI('/'); ->setURI($next_uri);
} }
$oauth_email = $provider->retrieveUserEmail(); $oauth_email = $provider->retrieveUserEmail();
@ -193,6 +201,7 @@ class PhabricatorOAuthLoginController extends PhabricatorAuthController {
$controller->setOAuthProvider($provider); $controller->setOAuthProvider($provider);
$controller->setOAuthInfo($oauth_info); $controller->setOAuthInfo($oauth_info);
$controller->setOAuthState($this->oauthState);
return $this->delegateToController($controller); return $this->delegateToController($controller);
} }
@ -217,6 +226,7 @@ class PhabricatorOAuthLoginController extends PhabricatorAuthController {
if ($token) { if ($token) {
$this->tokenExpires = $request->getInt('expires'); $this->tokenExpires = $request->getInt('expires');
$this->accessToken = $token; $this->accessToken = $token;
$this->oauthState = $request->getStr('state');
return null; return null;
} }
@ -274,6 +284,7 @@ class PhabricatorOAuthLoginController extends PhabricatorAuthController {
} }
$this->accessToken = $token; $this->accessToken = $token;
$this->oauthState = $request->getStr('state');
return null; return null;
} }

View file

@ -21,6 +21,7 @@ abstract class PhabricatorOAuthRegistrationController
private $oauthProvider; private $oauthProvider;
private $oauthInfo; private $oauthInfo;
private $oauthState;
final public function setOAuthInfo($info) { final public function setOAuthInfo($info) {
$this->oauthInfo = $info; $this->oauthInfo = $info;
@ -40,4 +41,13 @@ abstract class PhabricatorOAuthRegistrationController
return $this->oauthProvider; return $this->oauthProvider;
} }
final public function setOAuthState($state) {
$this->oauthState = $state;
return $this;
}
final public function getOAuthState() {
return $this->oauthState;
}
} }

View file

@ -125,6 +125,7 @@ class PhabricatorOAuthDefaultRegistrationController
$form $form
->addHiddenInput('token', $provider->getAccessToken()) ->addHiddenInput('token', $provider->getAccessToken())
->addHiddenInput('expires', $oauth_info->getTokenExpires()) ->addHiddenInput('expires', $oauth_info->getTokenExpires())
->addHiddenInput('state', $this->getOAuthState())
->setUser($request->getUser()) ->setUser($request->getUser())
->setAction($provider->getRedirectURI()) ->setAction($provider->getRedirectURI())
->appendChild( ->appendChild(

View file

@ -56,7 +56,8 @@ abstract class PhabricatorController extends AphrontController {
} }
if ($this->shouldRequireLogin() && !$user->getPHID()) { if ($this->shouldRequireLogin() && !$user->getPHID()) {
throw new AphrontRedirectException('/login/'); $login_controller = new PhabricatorLoginController($request);
return $this->delegateToController($login_controller);
} }
} }

View file

@ -8,8 +8,8 @@
phutil_require_module('phabricator', 'aphront/console/core'); phutil_require_module('phabricator', 'aphront/console/core');
phutil_require_module('phabricator', 'aphront/controller'); phutil_require_module('phabricator', 'aphront/controller');
phutil_require_module('phabricator', 'aphront/exception/redirect');
phutil_require_module('phabricator', 'aphront/response/webpage'); phutil_require_module('phabricator', 'aphront/response/webpage');
phutil_require_module('phabricator', 'applications/auth/controller/login');
phutil_require_module('phabricator', 'applications/people/storage/user'); phutil_require_module('phabricator', 'applications/people/storage/user');
phutil_require_module('phabricator', 'infrastructure/env'); phutil_require_module('phabricator', 'infrastructure/env');
phutil_require_module('phabricator', 'storage/queryfx'); phutil_require_module('phabricator', 'storage/queryfx');

View file

@ -83,10 +83,11 @@ $request = $application->buildRequest();
$application->setRequest($request); $application->setRequest($request);
list($controller, $uri_data) = $application->buildController(); list($controller, $uri_data) = $application->buildController();
try { try {
$controller->willBeginExecution(); $response = $controller->willBeginExecution();
if (!$response) {
$controller->willProcessRequest($uri_data); $controller->willProcessRequest($uri_data);
$response = $controller->processRequest(); $response = $controller->processRequest();
}
} catch (AphrontRedirectException $ex) { } catch (AphrontRedirectException $ex) {
$response = id(new AphrontRedirectResponse()) $response = id(new AphrontRedirectResponse())
->setURI($ex->getURI()); ->setURI($ex->getURI());