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:
parent
0bcbd0f158
commit
a100d97ed5
7 changed files with 40 additions and 7 deletions
|
@ -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(
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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(
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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');
|
||||||
|
|
|
@ -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());
|
||||||
|
|
Loading…
Reference in a new issue