Summary: Depends on D20117. Fixes T10071. When you're sent an email invitation, it's intended to allow you to register an account even if you otherwise could not (see D11737). Some time between D11737 and today, this stopped working (or perhaps it never worked and I got things wrong in D11737). I think this actually ended up not mattering for us, given the way Phacility auth was ultimately built. This feature generally seems reasonable, though, and probably //should// work. Make it work in the "password" and "oauth" cases, at least. This may //still// not work for LDAP, but testing that is nontrivial. Test Plan: - Enabled only passwords, turned off registration, sent an invite, registered with a password. - Enabled only Google OAuth, turned off registration, sent an invite, registered with Google OAuth. Reviewers: amckinley Reviewed By: amckinley Maniphest Tasks: T10071 Differential Revision: https://secure.phabricator.com/D20118
753 lines
24 KiB
PHP
753 lines
24 KiB
PHP
<?php
|
|
|
|
final class PhabricatorAuthRegisterController
|
|
extends PhabricatorAuthController {
|
|
|
|
public function shouldRequireLogin() {
|
|
return false;
|
|
}
|
|
|
|
public function handleRequest(AphrontRequest $request) {
|
|
$viewer = $this->getViewer();
|
|
$account_key = $request->getURIData('akey');
|
|
|
|
if ($viewer->isLoggedIn()) {
|
|
return id(new AphrontRedirectResponse())->setURI('/');
|
|
}
|
|
|
|
$invite = $this->loadInvite();
|
|
|
|
$is_setup = false;
|
|
if (strlen($account_key)) {
|
|
$result = $this->loadAccountForRegistrationOrLinking($account_key);
|
|
list($account, $provider, $response) = $result;
|
|
$is_default = false;
|
|
} else if ($this->isFirstTimeSetup()) {
|
|
$account = null;
|
|
$provider = null;
|
|
$response = null;
|
|
$is_default = true;
|
|
$is_setup = true;
|
|
} else {
|
|
list($account, $provider, $response) = $this->loadDefaultAccount($invite);
|
|
$is_default = true;
|
|
}
|
|
|
|
if ($response) {
|
|
return $response;
|
|
}
|
|
|
|
if (!$is_setup) {
|
|
if (!$provider->shouldAllowRegistration()) {
|
|
if ($invite) {
|
|
// If the user has an invite, we allow them to register with any
|
|
// provider, even a login-only provider.
|
|
} else {
|
|
// TODO: This is a routine error if you click "Login" on an external
|
|
// auth source which doesn't allow registration. The error should be
|
|
// more tailored.
|
|
|
|
return $this->renderError(
|
|
pht(
|
|
'The account you are attempting to register with uses an '.
|
|
'authentication provider ("%s") which does not allow '.
|
|
'registration. An administrator may have recently disabled '.
|
|
'registration with this provider.',
|
|
$provider->getProviderName()));
|
|
}
|
|
}
|
|
}
|
|
|
|
$errors = array();
|
|
|
|
$user = new PhabricatorUser();
|
|
|
|
if ($is_setup) {
|
|
$default_username = null;
|
|
$default_realname = null;
|
|
$default_email = null;
|
|
} else {
|
|
$default_username = $account->getUsername();
|
|
$default_realname = $account->getRealName();
|
|
$default_email = $account->getEmail();
|
|
}
|
|
|
|
$account_type = PhabricatorAuthPassword::PASSWORD_TYPE_ACCOUNT;
|
|
$content_source = PhabricatorContentSource::newFromRequest($request);
|
|
|
|
if ($invite) {
|
|
$default_email = $invite->getEmailAddress();
|
|
}
|
|
|
|
if ($default_email !== null) {
|
|
if (!PhabricatorUserEmail::isValidAddress($default_email)) {
|
|
$errors[] = pht(
|
|
'The email address associated with this external account ("%s") is '.
|
|
'not a valid email address and can not be used to register a '.
|
|
'Phabricator account. Choose a different, valid address.',
|
|
phutil_tag('strong', array(), $default_email));
|
|
$default_email = null;
|
|
}
|
|
}
|
|
|
|
if ($default_email !== null) {
|
|
// We should bypass policy here because e.g. limiting an application use
|
|
// to a subset of users should not allow the others to overwrite
|
|
// configured application emails.
|
|
$application_email = id(new PhabricatorMetaMTAApplicationEmailQuery())
|
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
|
->withAddresses(array($default_email))
|
|
->executeOne();
|
|
if ($application_email) {
|
|
$errors[] = pht(
|
|
'The email address associated with this account ("%s") is '.
|
|
'already in use by an application and can not be used to '.
|
|
'register a new Phabricator account. Choose a different, valid '.
|
|
'address.',
|
|
phutil_tag('strong', array(), $default_email));
|
|
$default_email = null;
|
|
}
|
|
}
|
|
|
|
$show_existing = null;
|
|
if ($default_email !== null) {
|
|
// If the account source provided an email, but it's not allowed by
|
|
// the configuration, roadblock the user. Previously, we let the user
|
|
// pick a valid email address instead, but this does not align well with
|
|
// user expectation and it's not clear the cases it enables are valuable.
|
|
// See discussion in T3472.
|
|
if (!PhabricatorUserEmail::isAllowedAddress($default_email)) {
|
|
$debug_email = new PHUIInvisibleCharacterView($default_email);
|
|
return $this->renderError(
|
|
array(
|
|
pht(
|
|
'The account you are attempting to register with has an invalid '.
|
|
'email address (%s). This Phabricator install only allows '.
|
|
'registration with specific email addresses:',
|
|
$debug_email),
|
|
phutil_tag('br'),
|
|
phutil_tag('br'),
|
|
PhabricatorUserEmail::describeAllowedAddresses(),
|
|
));
|
|
}
|
|
|
|
// If the account source provided an email, but another account already
|
|
// has that email, just pretend we didn't get an email.
|
|
if ($default_email !== null) {
|
|
$same_email = id(new PhabricatorUserEmail())->loadOneWhere(
|
|
'address = %s',
|
|
$default_email);
|
|
if ($same_email) {
|
|
if ($invite) {
|
|
// We're allowing this to continue. The fact that we loaded the
|
|
// invite means that the address is nonprimary and unverified and
|
|
// we're OK to steal it.
|
|
} else {
|
|
$show_existing = $default_email;
|
|
$default_email = null;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($show_existing !== null) {
|
|
if (!$request->getInt('phase')) {
|
|
return $this->newDialog()
|
|
->setTitle(pht('Email Address Already in Use'))
|
|
->addHiddenInput('phase', 1)
|
|
->appendParagraph(
|
|
pht(
|
|
'You are creating a new Phabricator account linked to an '.
|
|
'existing external account from outside Phabricator.'))
|
|
->appendParagraph(
|
|
pht(
|
|
'The email address ("%s") associated with the external account '.
|
|
'is already in use by an existing Phabricator account. Multiple '.
|
|
'Phabricator accounts may not have the same email address, so '.
|
|
'you can not use this email address to register a new '.
|
|
'Phabricator account.',
|
|
phutil_tag('strong', array(), $show_existing)))
|
|
->appendParagraph(
|
|
pht(
|
|
'If you want to register a new account, continue with this '.
|
|
'registration workflow and choose a new, unique email address '.
|
|
'for the new account.'))
|
|
->appendParagraph(
|
|
pht(
|
|
'If you want to link an existing Phabricator account to this '.
|
|
'external account, do not continue. Instead: log in to your '.
|
|
'existing account, then go to "Settings" and link the account '.
|
|
'in the "External Accounts" panel.'))
|
|
->appendParagraph(
|
|
pht(
|
|
'If you continue, you will create a new account. You will not '.
|
|
'be able to link this external account to an existing account.'))
|
|
->addCancelButton('/auth/login/', pht('Cancel'))
|
|
->addSubmitButton(pht('Create New Account'));
|
|
} else {
|
|
$errors[] = pht(
|
|
'The external account you are registering with has an email address '.
|
|
'that is already in use ("%s") by an existing Phabricator account. '.
|
|
'Choose a new, valid email address to register a new Phabricator '.
|
|
'account.',
|
|
phutil_tag('strong', array(), $show_existing));
|
|
}
|
|
}
|
|
|
|
$profile = id(new PhabricatorRegistrationProfile())
|
|
->setDefaultUsername($default_username)
|
|
->setDefaultEmail($default_email)
|
|
->setDefaultRealName($default_realname)
|
|
->setCanEditUsername(true)
|
|
->setCanEditEmail(($default_email === null))
|
|
->setCanEditRealName(true)
|
|
->setShouldVerifyEmail(false);
|
|
|
|
$event_type = PhabricatorEventType::TYPE_AUTH_WILLREGISTERUSER;
|
|
$event_data = array(
|
|
'account' => $account,
|
|
'profile' => $profile,
|
|
);
|
|
|
|
$event = id(new PhabricatorEvent($event_type, $event_data))
|
|
->setUser($user);
|
|
PhutilEventEngine::dispatchEvent($event);
|
|
|
|
$default_username = $profile->getDefaultUsername();
|
|
$default_email = $profile->getDefaultEmail();
|
|
$default_realname = $profile->getDefaultRealName();
|
|
|
|
$can_edit_username = $profile->getCanEditUsername();
|
|
$can_edit_email = $profile->getCanEditEmail();
|
|
$can_edit_realname = $profile->getCanEditRealName();
|
|
|
|
if ($is_setup) {
|
|
$must_set_password = false;
|
|
} else {
|
|
$must_set_password = $provider->shouldRequireRegistrationPassword();
|
|
}
|
|
|
|
$can_edit_anything = $profile->getCanEditAnything() || $must_set_password;
|
|
$force_verify = $profile->getShouldVerifyEmail();
|
|
|
|
// Automatically verify the administrator's email address during first-time
|
|
// setup.
|
|
if ($is_setup) {
|
|
$force_verify = true;
|
|
}
|
|
|
|
$value_username = $default_username;
|
|
$value_realname = $default_realname;
|
|
$value_email = $default_email;
|
|
$value_password = null;
|
|
|
|
$require_real_name = PhabricatorEnv::getEnvConfig('user.require-real-name');
|
|
|
|
$e_username = strlen($value_username) ? null : true;
|
|
$e_realname = $require_real_name ? true : null;
|
|
$e_email = strlen($value_email) ? null : true;
|
|
$e_password = true;
|
|
$e_captcha = true;
|
|
|
|
$skip_captcha = false;
|
|
if ($invite) {
|
|
// If the user is accepting an invite, assume they're trustworthy enough
|
|
// that we don't need to CAPTCHA them.
|
|
$skip_captcha = true;
|
|
}
|
|
|
|
$min_len = PhabricatorEnv::getEnvConfig('account.minimum-password-length');
|
|
$min_len = (int)$min_len;
|
|
|
|
$from_invite = $request->getStr('invite');
|
|
if ($from_invite && $can_edit_username) {
|
|
$value_username = $request->getStr('username');
|
|
$e_username = null;
|
|
}
|
|
|
|
$try_register =
|
|
($request->isFormPost() || !$can_edit_anything) &&
|
|
!$from_invite &&
|
|
($request->getInt('phase') != 1);
|
|
|
|
if ($try_register) {
|
|
$errors = array();
|
|
|
|
$unguarded = AphrontWriteGuard::beginScopedUnguardedWrites();
|
|
|
|
if ($must_set_password && !$skip_captcha) {
|
|
$e_captcha = pht('Again');
|
|
|
|
$captcha_ok = AphrontFormRecaptchaControl::processCaptcha($request);
|
|
if (!$captcha_ok) {
|
|
$errors[] = pht('Captcha response is incorrect, try again.');
|
|
$e_captcha = pht('Invalid');
|
|
}
|
|
}
|
|
|
|
if ($can_edit_username) {
|
|
$value_username = $request->getStr('username');
|
|
if (!strlen($value_username)) {
|
|
$e_username = pht('Required');
|
|
$errors[] = pht('Username is required.');
|
|
} else if (!PhabricatorUser::validateUsername($value_username)) {
|
|
$e_username = pht('Invalid');
|
|
$errors[] = PhabricatorUser::describeValidUsername();
|
|
} else {
|
|
$e_username = null;
|
|
}
|
|
}
|
|
|
|
if ($must_set_password) {
|
|
$value_password = $request->getStr('password');
|
|
$value_confirm = $request->getStr('confirm');
|
|
|
|
$password_envelope = new PhutilOpaqueEnvelope($value_password);
|
|
$confirm_envelope = new PhutilOpaqueEnvelope($value_confirm);
|
|
|
|
$engine = id(new PhabricatorAuthPasswordEngine())
|
|
->setViewer($user)
|
|
->setContentSource($content_source)
|
|
->setPasswordType($account_type)
|
|
->setObject($user);
|
|
|
|
try {
|
|
$engine->checkNewPassword($password_envelope, $confirm_envelope);
|
|
$e_password = null;
|
|
} catch (PhabricatorAuthPasswordException $ex) {
|
|
$errors[] = $ex->getMessage();
|
|
$e_password = $ex->getPasswordError();
|
|
}
|
|
}
|
|
|
|
if ($can_edit_email) {
|
|
$value_email = $request->getStr('email');
|
|
if (!strlen($value_email)) {
|
|
$e_email = pht('Required');
|
|
$errors[] = pht('Email is required.');
|
|
} else if (!PhabricatorUserEmail::isValidAddress($value_email)) {
|
|
$e_email = pht('Invalid');
|
|
$errors[] = PhabricatorUserEmail::describeValidAddresses();
|
|
} else if (!PhabricatorUserEmail::isAllowedAddress($value_email)) {
|
|
$e_email = pht('Disallowed');
|
|
$errors[] = PhabricatorUserEmail::describeAllowedAddresses();
|
|
} else {
|
|
$e_email = null;
|
|
}
|
|
}
|
|
|
|
if ($can_edit_realname) {
|
|
$value_realname = $request->getStr('realName');
|
|
if (!strlen($value_realname) && $require_real_name) {
|
|
$e_realname = pht('Required');
|
|
$errors[] = pht('Real name is required.');
|
|
} else {
|
|
$e_realname = null;
|
|
}
|
|
}
|
|
|
|
if (!$errors) {
|
|
if (!$is_setup) {
|
|
$image = $this->loadProfilePicture($account);
|
|
if ($image) {
|
|
$user->setProfileImagePHID($image->getPHID());
|
|
}
|
|
}
|
|
|
|
try {
|
|
$verify_email = false;
|
|
|
|
if ($force_verify) {
|
|
$verify_email = true;
|
|
}
|
|
|
|
if (!$is_setup) {
|
|
if ($value_email === $default_email) {
|
|
if ($account->getEmailVerified()) {
|
|
$verify_email = true;
|
|
}
|
|
|
|
if ($provider->shouldTrustEmails()) {
|
|
$verify_email = true;
|
|
}
|
|
|
|
if ($invite) {
|
|
$verify_email = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
$email_obj = null;
|
|
if ($invite) {
|
|
// If we have a valid invite, this email may exist but be
|
|
// nonprimary and unverified, so we'll reassign it.
|
|
$email_obj = id(new PhabricatorUserEmail())->loadOneWhere(
|
|
'address = %s',
|
|
$value_email);
|
|
}
|
|
if (!$email_obj) {
|
|
$email_obj = id(new PhabricatorUserEmail())
|
|
->setAddress($value_email);
|
|
}
|
|
|
|
$email_obj->setIsVerified((int)$verify_email);
|
|
|
|
$user->setUsername($value_username);
|
|
$user->setRealname($value_realname);
|
|
|
|
if ($is_setup) {
|
|
$must_approve = false;
|
|
} else if ($invite) {
|
|
$must_approve = false;
|
|
} else {
|
|
$must_approve = PhabricatorEnv::getEnvConfig(
|
|
'auth.require-approval');
|
|
}
|
|
|
|
if ($must_approve) {
|
|
$user->setIsApproved(0);
|
|
} else {
|
|
$user->setIsApproved(1);
|
|
}
|
|
|
|
if ($invite) {
|
|
$allow_reassign_email = true;
|
|
} else {
|
|
$allow_reassign_email = false;
|
|
}
|
|
|
|
$user->openTransaction();
|
|
|
|
$editor = id(new PhabricatorUserEditor())
|
|
->setActor($user);
|
|
|
|
$editor->createNewUser($user, $email_obj, $allow_reassign_email);
|
|
if ($must_set_password) {
|
|
$password_object = PhabricatorAuthPassword::initializeNewPassword(
|
|
$user,
|
|
$account_type);
|
|
|
|
$password_object
|
|
->setPassword($password_envelope, $user)
|
|
->save();
|
|
}
|
|
|
|
if ($is_setup) {
|
|
$xactions = array();
|
|
$xactions[] = id(new PhabricatorUserTransaction())
|
|
->setTransactionType(
|
|
PhabricatorUserEmpowerTransaction::TRANSACTIONTYPE)
|
|
->setNewValue(true);
|
|
|
|
$actor = PhabricatorUser::getOmnipotentUser();
|
|
$content_source = PhabricatorContentSource::newFromRequest(
|
|
$request);
|
|
|
|
$people_application_phid = id(new PhabricatorPeopleApplication())
|
|
->getPHID();
|
|
|
|
$transaction_editor = id(new PhabricatorUserTransactionEditor())
|
|
->setActor($actor)
|
|
->setActingAsPHID($people_application_phid)
|
|
->setContentSource($content_source)
|
|
->setContinueOnMissingFields(true);
|
|
|
|
$transaction_editor->applyTransactions($user, $xactions);
|
|
}
|
|
|
|
if (!$is_setup) {
|
|
$account->setUserPHID($user->getPHID());
|
|
$provider->willRegisterAccount($account);
|
|
$account->save();
|
|
}
|
|
|
|
$user->saveTransaction();
|
|
|
|
if (!$email_obj->getIsVerified()) {
|
|
$email_obj->sendVerificationEmail($user);
|
|
}
|
|
|
|
if ($must_approve) {
|
|
$this->sendWaitingForApprovalEmail($user);
|
|
}
|
|
|
|
if ($invite) {
|
|
$invite->setAcceptedByPHID($user->getPHID())->save();
|
|
}
|
|
|
|
return $this->loginUser($user);
|
|
} catch (AphrontDuplicateKeyQueryException $exception) {
|
|
$same_username = id(new PhabricatorUser())->loadOneWhere(
|
|
'userName = %s',
|
|
$user->getUserName());
|
|
|
|
$same_email = id(new PhabricatorUserEmail())->loadOneWhere(
|
|
'address = %s',
|
|
$value_email);
|
|
|
|
if ($same_username) {
|
|
$e_username = pht('Duplicate');
|
|
$errors[] = pht('Another user already has that username.');
|
|
}
|
|
|
|
if ($same_email) {
|
|
// TODO: See T3340.
|
|
$e_email = pht('Duplicate');
|
|
$errors[] = pht('Another user already has that email.');
|
|
}
|
|
|
|
if (!$same_username && !$same_email) {
|
|
throw $exception;
|
|
}
|
|
}
|
|
}
|
|
|
|
unset($unguarded);
|
|
}
|
|
|
|
$form = id(new AphrontFormView())
|
|
->setUser($request->getUser())
|
|
->addHiddenInput('phase', 2);
|
|
|
|
if (!$is_default) {
|
|
$form->appendChild(
|
|
id(new AphrontFormMarkupControl())
|
|
->setLabel(pht('External Account'))
|
|
->setValue(
|
|
id(new PhabricatorAuthAccountView())
|
|
->setUser($request->getUser())
|
|
->setExternalAccount($account)
|
|
->setAuthProvider($provider)));
|
|
}
|
|
|
|
if ($can_edit_username) {
|
|
$form->appendChild(
|
|
id(new AphrontFormTextControl())
|
|
->setLabel(pht('Username'))
|
|
->setName('username')
|
|
->setValue($value_username)
|
|
->setError($e_username));
|
|
} else {
|
|
$form->appendChild(
|
|
id(new AphrontFormMarkupControl())
|
|
->setLabel(pht('Username'))
|
|
->setValue($value_username)
|
|
->setError($e_username));
|
|
}
|
|
|
|
if ($can_edit_realname) {
|
|
$form->appendChild(
|
|
id(new AphrontFormTextControl())
|
|
->setLabel(pht('Real Name'))
|
|
->setName('realName')
|
|
->setValue($value_realname)
|
|
->setError($e_realname));
|
|
}
|
|
|
|
if ($must_set_password) {
|
|
$form->appendChild(
|
|
id(new AphrontFormPasswordControl())
|
|
->setLabel(pht('Password'))
|
|
->setName('password')
|
|
->setError($e_password));
|
|
$form->appendChild(
|
|
id(new AphrontFormPasswordControl())
|
|
->setLabel(pht('Confirm Password'))
|
|
->setName('confirm')
|
|
->setError($e_password)
|
|
->setCaption(
|
|
$min_len
|
|
? pht('Minimum length of %d characters.', $min_len)
|
|
: null));
|
|
}
|
|
|
|
if ($can_edit_email) {
|
|
$form->appendChild(
|
|
id(new AphrontFormTextControl())
|
|
->setLabel(pht('Email'))
|
|
->setName('email')
|
|
->setValue($value_email)
|
|
->setCaption(PhabricatorUserEmail::describeAllowedAddresses())
|
|
->setError($e_email));
|
|
}
|
|
|
|
if ($must_set_password && !$skip_captcha) {
|
|
$form->appendChild(
|
|
id(new AphrontFormRecaptchaControl())
|
|
->setLabel(pht('Captcha'))
|
|
->setError($e_captcha));
|
|
}
|
|
|
|
$submit = id(new AphrontFormSubmitControl());
|
|
|
|
if ($is_setup) {
|
|
$submit
|
|
->setValue(pht('Create Admin Account'));
|
|
} else {
|
|
$submit
|
|
->addCancelButton($this->getApplicationURI('start/'))
|
|
->setValue(pht('Register Account'));
|
|
}
|
|
|
|
|
|
$form->appendChild($submit);
|
|
|
|
$crumbs = $this->buildApplicationCrumbs();
|
|
|
|
if ($is_setup) {
|
|
$crumbs->addTextCrumb(pht('Setup Admin Account'));
|
|
$title = pht('Welcome to Phabricator');
|
|
} else {
|
|
$crumbs->addTextCrumb(pht('Register'));
|
|
$crumbs->addTextCrumb($provider->getProviderName());
|
|
$title = pht('Create a New Account');
|
|
}
|
|
$crumbs->setBorder(true);
|
|
|
|
$welcome_view = null;
|
|
if ($is_setup) {
|
|
$welcome_view = id(new PHUIInfoView())
|
|
->setSeverity(PHUIInfoView::SEVERITY_NOTICE)
|
|
->setTitle(pht('Welcome to Phabricator'))
|
|
->appendChild(
|
|
pht(
|
|
'Installation is complete. Register your administrator account '.
|
|
'below to log in. You will be able to configure options and add '.
|
|
'authentication mechanisms later on.'));
|
|
}
|
|
|
|
$object_box = id(new PHUIObjectBoxView())
|
|
->setForm($form)
|
|
->setFormErrors($errors);
|
|
|
|
$invite_header = null;
|
|
if ($invite) {
|
|
$invite_header = $this->renderInviteHeader($invite);
|
|
}
|
|
|
|
$header = id(new PHUIHeaderView())
|
|
->setHeader($title);
|
|
|
|
$view = id(new PHUITwoColumnView())
|
|
->setHeader($header)
|
|
->setFooter(
|
|
array(
|
|
$welcome_view,
|
|
$invite_header,
|
|
$object_box,
|
|
));
|
|
|
|
return $this->newPage()
|
|
->setTitle($title)
|
|
->setCrumbs($crumbs)
|
|
->appendChild($view);
|
|
}
|
|
|
|
private function loadDefaultAccount($invite) {
|
|
$providers = PhabricatorAuthProvider::getAllEnabledProviders();
|
|
$account = null;
|
|
$provider = null;
|
|
$response = null;
|
|
|
|
foreach ($providers as $key => $candidate_provider) {
|
|
if (!$invite) {
|
|
if (!$candidate_provider->shouldAllowRegistration()) {
|
|
unset($providers[$key]);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (!$candidate_provider->isDefaultRegistrationProvider()) {
|
|
unset($providers[$key]);
|
|
}
|
|
}
|
|
|
|
if (!$providers) {
|
|
$response = $this->renderError(
|
|
pht(
|
|
'There are no configured default registration providers.'));
|
|
return array($account, $provider, $response);
|
|
} else if (count($providers) > 1) {
|
|
$response = $this->renderError(
|
|
pht('There are too many configured default registration providers.'));
|
|
return array($account, $provider, $response);
|
|
}
|
|
|
|
$provider = head($providers);
|
|
$account = $provider->newDefaultExternalAccount();
|
|
|
|
return array($account, $provider, $response);
|
|
}
|
|
|
|
private function loadProfilePicture(PhabricatorExternalAccount $account) {
|
|
$phid = $account->getProfileImagePHID();
|
|
if (!$phid) {
|
|
return null;
|
|
}
|
|
|
|
// NOTE: Use of omnipotent user is okay here because the registering user
|
|
// can not control the field value, and we can't use their user object to
|
|
// do meaningful policy checks anyway since they have not registered yet.
|
|
// Reaching this means the user holds the account secret key and the
|
|
// registration secret key, and thus has permission to view the image.
|
|
|
|
$file = id(new PhabricatorFileQuery())
|
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
|
->withPHIDs(array($phid))
|
|
->executeOne();
|
|
if (!$file) {
|
|
return null;
|
|
}
|
|
|
|
$xform = PhabricatorFileTransform::getTransformByKey(
|
|
PhabricatorFileThumbnailTransform::TRANSFORM_PROFILE);
|
|
return $xform->executeTransform($file);
|
|
}
|
|
|
|
protected function renderError($message) {
|
|
return $this->renderErrorPage(
|
|
pht('Registration Failed'),
|
|
array($message));
|
|
}
|
|
|
|
private function sendWaitingForApprovalEmail(PhabricatorUser $user) {
|
|
$title = '[Phabricator] '.pht(
|
|
'New User "%s" Awaiting Approval',
|
|
$user->getUsername());
|
|
|
|
$body = new PhabricatorMetaMTAMailBody();
|
|
|
|
$body->addRawSection(
|
|
pht(
|
|
'Newly registered user "%s" is awaiting account approval by an '.
|
|
'administrator.',
|
|
$user->getUsername()));
|
|
|
|
$body->addLinkSection(
|
|
pht('APPROVAL QUEUE'),
|
|
PhabricatorEnv::getProductionURI(
|
|
'/people/query/approval/'));
|
|
|
|
$body->addLinkSection(
|
|
pht('DISABLE APPROVAL QUEUE'),
|
|
PhabricatorEnv::getProductionURI(
|
|
'/config/edit/auth.require-approval/'));
|
|
|
|
$admins = id(new PhabricatorPeopleQuery())
|
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
|
->withIsAdmin(true)
|
|
->execute();
|
|
|
|
if (!$admins) {
|
|
return;
|
|
}
|
|
|
|
$mail = id(new PhabricatorMetaMTAMail())
|
|
->addTos(mpull($admins, 'getPHID'))
|
|
->setSubject($title)
|
|
->setBody($body->render())
|
|
->saveAndSend();
|
|
}
|
|
|
|
}
|