mirror of
https://we.phorge.it/source/phorge.git
synced 2024-11-13 18:32:41 +01:00
61a0c6d6e3
Summary: Ref T1536. Facebook currently does a check which should be on-login in registration hooks, and this is generally a reasonable hook to provide. The "will login" event allows listeners to reject or modify a login, or just log it or whatever. NOTE: This doesn't cover non-web logins right now -- notably Conduit. That's presumably fine. (This can't land for a while, it depends on about 10 uncommitted revisions.) Test Plan: Logged out and in again. Reviewers: wez, btrahan Reviewed By: btrahan CC: aran Maniphest Tasks: T1536 Differential Revision: https://secure.phabricator.com/D6202
215 lines
6.5 KiB
PHP
215 lines
6.5 KiB
PHP
<?php
|
|
|
|
final class PhabricatorAuthLoginController
|
|
extends PhabricatorAuthController {
|
|
|
|
private $providerKey;
|
|
private $provider;
|
|
|
|
public function shouldRequireLogin() {
|
|
return false;
|
|
}
|
|
|
|
public function willProcessRequest(array $data) {
|
|
$this->providerKey = $data['pkey'];
|
|
}
|
|
|
|
public function processRequest() {
|
|
$request = $this->getRequest();
|
|
$viewer = $request->getUser();
|
|
|
|
$response = $this->loadProvider();
|
|
if ($response) {
|
|
return $response;
|
|
}
|
|
|
|
$provider = $this->provider;
|
|
|
|
list($account, $response) = $provider->processLoginRequest($this);
|
|
if ($response) {
|
|
return $response;
|
|
}
|
|
|
|
if (!$account) {
|
|
throw new Exception(
|
|
"Auth provider failed to load an account from processLoginRequest()!");
|
|
}
|
|
|
|
if ($account->getUserPHID()) {
|
|
// The account is already attached to a Phabricator user, so this is
|
|
// either a login or a bad account link request.
|
|
if (!$viewer->isLoggedIn()) {
|
|
if ($provider->shouldAllowLogin()) {
|
|
return $this->processLoginUser($account);
|
|
} else {
|
|
return $this->renderError(
|
|
pht(
|
|
'The external account ("%s") you just authenticated with is '.
|
|
'not configured to allow logins on this Phabricator install. '.
|
|
'An administrator may have recently disabled it.',
|
|
$provider->getProviderName()));
|
|
}
|
|
} else if ($viewer->getPHID() == $account->getUserPHID()) {
|
|
return $this->renderError(
|
|
pht(
|
|
'This external account ("%s") is already linked to your '.
|
|
'Phabricator account.'));
|
|
} else {
|
|
return $this->renderError(
|
|
pht(
|
|
'The external account ("%s") you just used to login is alerady '.
|
|
'associated with another Phabricator user account. Login to the '.
|
|
'other Phabricator account and unlink the external account before '.
|
|
'linking it to a new Phabricator account.',
|
|
$provider->getProviderName()));
|
|
}
|
|
} else {
|
|
// The account is not yet attached to a Phabricator user, so this is
|
|
// either a registration or an account link request.
|
|
if (!$viewer->isLoggedIn()) {
|
|
if ($provider->shouldAllowRegistration()) {
|
|
return $this->processRegisterUser($account);
|
|
} else {
|
|
return $this->renderError(
|
|
pht(
|
|
'The external account ("%s") you just authenticated with is '.
|
|
'not configured to allow registration on this Phabricator '.
|
|
'install. An administrator may have recently disabled it.',
|
|
$provider->getProviderName()));
|
|
}
|
|
} else {
|
|
if ($provider->shouldAllowAccountLink()) {
|
|
return $this->processLinkUser($account);
|
|
} else {
|
|
return $this->renderError(
|
|
pht(
|
|
'The external account ("%s") you just authenticated with is '.
|
|
'not configured to allow account linking on this Phabricator '.
|
|
'install. An administrator may have recently disabled it.',
|
|
$provider->getProviderName()));
|
|
}
|
|
}
|
|
}
|
|
|
|
// This should be unreachable, but fail explicitly if we get here somehow.
|
|
return new Aphront400Response();
|
|
}
|
|
|
|
private function processLoginUser(PhabricatorExternalAccount $account) {
|
|
$user = id(new PhabricatorUser())->loadOneWhere(
|
|
'phid = %s',
|
|
$account->getUserPHID());
|
|
|
|
if (!$user) {
|
|
return $this->renderError(
|
|
pht(
|
|
'The external account you just logged in with is not associated '.
|
|
'with a valid Phabricator user.'));
|
|
}
|
|
|
|
return $this->loginUser($user);
|
|
}
|
|
|
|
private function processRegisterUser(PhabricatorExternalAccount $account) {
|
|
if ($account->getUserPHID()) {
|
|
throw new Exception("Account is already registered.");
|
|
}
|
|
|
|
// Regenerate the registration secret key, set it on the external account,
|
|
// set a cookie on the user's machine, and redirect them to registration.
|
|
// See PhabricatorAuthRegisterController for discussion of the registration
|
|
// key.
|
|
|
|
$registration_key = Filesystem::readRandomCharacters(32);
|
|
$account->setProperty(
|
|
'registrationKey',
|
|
PhabricatorHash::digest($registration_key));
|
|
|
|
$unguarded = AphrontWriteGuard::beginScopedUnguardedWrites();
|
|
$account->save();
|
|
unset($unguarded);
|
|
|
|
$this->getRequest()->setCookie('phreg', $registration_key);
|
|
|
|
$account_secret = $account->getAccountSecret();
|
|
$register_uri = $this->getApplicationURI('register/'.$account_secret.'/');
|
|
return id(new AphrontRedirectResponse())->setURI($register_uri);
|
|
}
|
|
|
|
private function processLinkUser(PhabricatorExternalAccount $account) {
|
|
// TODO: Implement.
|
|
return new Aphront404Response();
|
|
}
|
|
|
|
private function loadProvider() {
|
|
$provider = PhabricatorAuthProvider::getEnabledProviderByKey(
|
|
$this->providerKey);
|
|
|
|
if (!$provider) {
|
|
return $this->renderError(
|
|
pht(
|
|
'The account you are attempting to login with uses a nonexistent '.
|
|
'or disabled authentication provider (with key "%s"). An '.
|
|
'administrator may have recently disabled this provider.',
|
|
$this->providerKey));
|
|
}
|
|
|
|
$this->provider = $provider;
|
|
|
|
return null;
|
|
}
|
|
|
|
protected function renderError($message) {
|
|
$title = pht('Login Failed');
|
|
|
|
$view = new AphrontErrorView();
|
|
$view->setTitle($title);
|
|
$view->appendChild($message);
|
|
|
|
return $this->buildApplicationPage(
|
|
$view,
|
|
array(
|
|
'title' => $title,
|
|
'device' => true,
|
|
'dust' => true,
|
|
));
|
|
}
|
|
|
|
public function buildProviderPageResponse(
|
|
PhabricatorAuthProvider $provider,
|
|
$content) {
|
|
|
|
$crumbs = $this->buildApplicationCrumbs();
|
|
$crumbs->addCrumb(
|
|
id(new PhabricatorCrumbView())
|
|
->setName(pht('Login'))
|
|
->setHref($this->getApplicationURI('start/')));
|
|
$crumbs->addCrumb(
|
|
id(new PhabricatorCrumbView())
|
|
->setName($provider->getProviderName()));
|
|
|
|
return $this->buildApplicationPage(
|
|
array(
|
|
$crumbs,
|
|
$content,
|
|
),
|
|
array(
|
|
'title' => pht('Login'),
|
|
'device' => true,
|
|
'dust' => true,
|
|
));
|
|
}
|
|
|
|
public function buildProviderErrorResponse(
|
|
PhabricatorAuthProvider $provider,
|
|
$message) {
|
|
|
|
$message = pht(
|
|
'Authentication provider ("%s") encountered an error during login. %s',
|
|
$provider->getProviderName(),
|
|
$message);
|
|
|
|
return $this->renderError($message);
|
|
}
|
|
|
|
}
|