mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-11 15:21:03 +01:00
Simplify and correct some challenge TTL lockout code
Summary: Depends on D19889. Ref T13222. Some of this logic is either not-quite-right or a little more complicated than it needs to be. Currently, we TTL TOTP challenges after three timesteps -- once the current code could no longer be used. But we actually have to TTL it after five timesteps -- once the most-future acceptable code could no longer be used. Otherwise, you can enter the most-future code now (perhaps the attacker compromises NTP and skews the server clock back by 75 seconds) and then an attacker can re-use it in three timesteps. Generally, simplify things a bit and trust TTLs more. This also makes the "wait" dialog friendlier since we can give users an exact number of seconds. The overall behavior here is still a little odd because we don't actually require you to respond to the challenge you were issued (right now, we check that the response is valid whenever you submit it, not that it's a valid response to the challenge we issued), but that will change in a future diff. This is just moving us generally in the right direction, and doesn't yet lock everything down properly. Test Plan: - Added a little snippet to the control caption to list all the valid codes to make this easier: ``` $key = new PhutilOpaqueEnvelope($config->getFactorSecret()); $valid = array(); foreach ($this->getAllowedTimesteps() as $step) { $valid[] = self::getTOTPCode($key, $step); } $control->setCaption( pht( 'Valid Codes: '.implode(', ', $valid))); ``` - Used the most-future code to sign `L3`. - Verified that `L4` did not unlock until the code for `L3` left the activation window. Reviewers: amckinley Reviewed By: amckinley Maniphest Tasks: T13222 Differential Revision: https://secure.phabricator.com/D19890
This commit is contained in:
parent
2de632d4fe
commit
0673e79d6d
2 changed files with 37 additions and 30 deletions
|
@ -9,7 +9,7 @@ final class PhabricatorAuthTryFactorAction extends PhabricatorSystemAction {
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getScoreThreshold() {
|
public function getScoreThreshold() {
|
||||||
return 10 / phutil_units('1 hour in seconds');
|
return 100 / phutil_units('1 hour in seconds');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getLimitExplanation() {
|
public function getLimitExplanation() {
|
||||||
|
|
|
@ -155,25 +155,43 @@ final class PhabricatorTOTPAuthFactor extends PhabricatorAuthFactor {
|
||||||
PhabricatorUser $viewer,
|
PhabricatorUser $viewer,
|
||||||
array $challenges) {
|
array $challenges) {
|
||||||
|
|
||||||
$now = $this->getCurrentTimestep();
|
$current_step = $this->getCurrentTimestep();
|
||||||
|
|
||||||
// If we already issued a valid challenge, don't issue a new one.
|
// If we already issued a valid challenge, don't issue a new one.
|
||||||
if ($challenges) {
|
if ($challenges) {
|
||||||
return array();
|
return array();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise, generate a new challenge for the current timestep. It TTLs
|
// Otherwise, generate a new challenge for the current timestep and compute
|
||||||
// after it would fall off the bottom of the window.
|
// the TTL.
|
||||||
$timesteps = $this->getAllowedTimesteps();
|
|
||||||
$min_step = min($timesteps);
|
|
||||||
|
|
||||||
|
// When computing the TTL, note that we accept codes within a certain
|
||||||
|
// window of the challenge timestep to account for clock skew and users
|
||||||
|
// needing time to enter codes.
|
||||||
|
|
||||||
|
// We don't want this challenge to expire until after all valid responses
|
||||||
|
// to it are no longer valid responses to any other challenge we might
|
||||||
|
// issue in the future. If the challenge expires too quickly, we may issue
|
||||||
|
// a new challenge which can accept the same TOTP code response.
|
||||||
|
|
||||||
|
// This means that we need to keep this challenge alive for double the
|
||||||
|
// window size: if we're currently at timestep 3, the user might respond
|
||||||
|
// with the code for timestep 5. This is valid, since timestep 5 is within
|
||||||
|
// the window for timestep 3.
|
||||||
|
|
||||||
|
// But the code for timestep 5 can be used to respond at timesteps 3, 4, 5,
|
||||||
|
// 6, and 7. To prevent any valid response to this challenge from being
|
||||||
|
// used again, we need to keep this challenge active until timestep 8.
|
||||||
|
|
||||||
|
$window_size = $this->getTimestepWindowSize();
|
||||||
$step_duration = $this->getTimestepDuration();
|
$step_duration = $this->getTimestepDuration();
|
||||||
$ttl_steps = ($now - $min_step) + 1;
|
|
||||||
|
$ttl_steps = ($window_size * 2) + 1;
|
||||||
$ttl_seconds = ($ttl_steps * $step_duration);
|
$ttl_seconds = ($ttl_steps * $step_duration);
|
||||||
|
|
||||||
return array(
|
return array(
|
||||||
$this->newChallenge($config, $viewer)
|
$this->newChallenge($config, $viewer)
|
||||||
->setChallengeKey($now)
|
->setChallengeKey($current_step)
|
||||||
->setChallengeTTL(PhabricatorTime::getNow() + $ttl_seconds),
|
->setChallengeTTL(PhabricatorTime::getNow() + $ttl_seconds),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -216,31 +234,15 @@ final class PhabricatorTOTPAuthFactor extends PhabricatorAuthFactor {
|
||||||
// nearby timestep, require that it was issued to the current session.
|
// nearby timestep, require that it was issued to the current session.
|
||||||
// This is defusing attacks where you (broadly) look at someone's phone
|
// This is defusing attacks where you (broadly) look at someone's phone
|
||||||
// and type the code in more quickly than they do.
|
// and type the code in more quickly than they do.
|
||||||
|
|
||||||
$step_duration = $this->getTimestepDuration();
|
|
||||||
$now = $this->getCurrentTimestep();
|
|
||||||
$timesteps = $this->getAllowedTimesteps();
|
|
||||||
$timesteps = array_fuse($timesteps);
|
|
||||||
$min_step = min($timesteps);
|
|
||||||
|
|
||||||
$session_phid = $viewer->getSession()->getPHID();
|
$session_phid = $viewer->getSession()->getPHID();
|
||||||
|
$now = PhabricatorTime::getNow();
|
||||||
|
|
||||||
$engine = $config->getSessionEngine();
|
$engine = $config->getSessionEngine();
|
||||||
$workflow_key = $engine->getWorkflowKey();
|
$workflow_key = $engine->getWorkflowKey();
|
||||||
|
|
||||||
foreach ($challenges as $challenge) {
|
foreach ($challenges as $challenge) {
|
||||||
$challenge_timestep = (int)$challenge->getChallengeKey();
|
$challenge_timestep = (int)$challenge->getChallengeKey();
|
||||||
|
$wait_duration = ($challenge->getChallengeTTL() - $now) + 1;
|
||||||
// This challenge isn't for one of the timesteps you'd be able to respond
|
|
||||||
// to if you submitted the form right now, so we're good to keep going.
|
|
||||||
if (!isset($timesteps[$challenge_timestep])) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// This is the number of timesteps you need to wait for the problem
|
|
||||||
// timestep to leave the window, rounded up.
|
|
||||||
$wait_steps = ($challenge_timestep - $min_step) + 1;
|
|
||||||
$wait_duration = ($wait_steps * $step_duration);
|
|
||||||
|
|
||||||
if ($challenge->getSessionPHID() !== $session_phid) {
|
if ($challenge->getSessionPHID() !== $session_phid) {
|
||||||
return $this->newResult()
|
return $this->newResult()
|
||||||
|
@ -248,7 +250,7 @@ final class PhabricatorTOTPAuthFactor extends PhabricatorAuthFactor {
|
||||||
->setErrorMessage(
|
->setErrorMessage(
|
||||||
pht(
|
pht(
|
||||||
'This factor recently issued a challenge to a different login '.
|
'This factor recently issued a challenge to a different login '.
|
||||||
'session. Wait %s seconds for the code to cycle, then try '.
|
'session. Wait %s second(s) for the code to cycle, then try '.
|
||||||
'again.',
|
'again.',
|
||||||
new PhutilNumber($wait_duration)));
|
new PhutilNumber($wait_duration)));
|
||||||
}
|
}
|
||||||
|
@ -259,7 +261,7 @@ final class PhabricatorTOTPAuthFactor extends PhabricatorAuthFactor {
|
||||||
->setErrorMessage(
|
->setErrorMessage(
|
||||||
pht(
|
pht(
|
||||||
'This factor recently issued a challenge for a different '.
|
'This factor recently issued a challenge for a different '.
|
||||||
'workflow. Wait %s seconds for the code to cycle, then try '.
|
'workflow. Wait %s second(s) for the code to cycle, then try '.
|
||||||
'again.',
|
'again.',
|
||||||
new PhutilNumber($wait_duration)));
|
new PhutilNumber($wait_duration)));
|
||||||
}
|
}
|
||||||
|
@ -423,8 +425,13 @@ final class PhabricatorTOTPAuthFactor extends PhabricatorAuthFactor {
|
||||||
}
|
}
|
||||||
|
|
||||||
private function getAllowedTimesteps() {
|
private function getAllowedTimesteps() {
|
||||||
$now = $this->getCurrentTimestep();
|
$current_step = $this->getCurrentTimestep();
|
||||||
return range($now - 2, $now + 2);
|
$window = $this->getTimestepWindowSize();
|
||||||
|
return range($current_step - $window, $current_step + $window);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getTimestepWindowSize() {
|
||||||
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue