2020-01-24 22:37:48 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Classes;
|
|
|
|
|
|
|
|
use OTPHP\TOTP;
|
|
|
|
use OTPHP\Factory;
|
|
|
|
use Assert\AssertionFailedException;
|
|
|
|
|
|
|
|
class OTP
|
|
|
|
{
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate a TOTP
|
|
|
|
*
|
|
|
|
* @param \App\TwoFAccount $twofaccount
|
2020-02-06 12:24:18 +01:00
|
|
|
* @param Boolean $isPreview Prevent updating storage in case of HOTP preview
|
2020-01-24 22:37:48 +01:00
|
|
|
* @return an array that represent the totp code
|
|
|
|
*/
|
2020-02-06 12:24:18 +01:00
|
|
|
public static function generate($uri, $isPreview = false)
|
2020-01-24 22:37:48 +01:00
|
|
|
{
|
|
|
|
|
2020-01-26 21:02:36 +01:00
|
|
|
$otp = OTP::get($uri);
|
2020-01-24 22:37:48 +01:00
|
|
|
|
|
|
|
if( get_class($otp) === 'OTPHP\TOTP' ) {
|
|
|
|
|
|
|
|
$currentPosition = time();
|
|
|
|
$PeriodCount = floor($currentPosition / $otp->getPeriod()); //nombre de période de x s depuis T0 (x=30 par défaut)
|
|
|
|
$currentPeriodStartAt = $PeriodCount * $otp->getPeriod();
|
|
|
|
$positionInCurrentPeriod = $currentPosition - $currentPeriodStartAt;
|
|
|
|
|
|
|
|
// For memo :
|
|
|
|
// $nextOtpAt = ($PeriodCount+1)*$period
|
|
|
|
// $remainingTime = $nextOtpAt - time()
|
|
|
|
|
|
|
|
return $totp = [
|
|
|
|
'otp' => $otp->now(),
|
|
|
|
'position' => $positionInCurrentPeriod
|
|
|
|
];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// It's a HOTP
|
|
|
|
$hotp = [
|
|
|
|
'otp' => $otp->at($otp->getCounter()),
|
2020-02-06 12:24:18 +01:00
|
|
|
'counter' => $otp->getCounter()
|
2020-01-24 22:37:48 +01:00
|
|
|
];
|
|
|
|
|
2020-02-06 12:24:18 +01:00
|
|
|
// now we update the counter for the next OTP generation
|
2020-01-24 22:37:48 +01:00
|
|
|
$otp->setParameter( 'counter', $otp->getcounter() + 1 );
|
2020-02-06 12:24:18 +01:00
|
|
|
$hotp['nextUri'] = urldecode($otp->getProvisioningUri());
|
2020-01-24 22:37:48 +01:00
|
|
|
|
2020-02-06 12:24:18 +01:00
|
|
|
if( !$isPreview ) {
|
|
|
|
$twofaccount = \App\TwoFAccount::where('uri', $uri)->first();
|
|
|
|
$twofaccount->uri = $hotp['nextUri'];
|
|
|
|
$twofaccount->save();
|
|
|
|
}
|
2020-01-24 22:37:48 +01:00
|
|
|
|
|
|
|
return $hotp;
|
|
|
|
}
|
|
|
|
|
2020-01-25 18:22:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* check if the provided uri is a valid OTP uri
|
|
|
|
*
|
|
|
|
* @param \App\TwoFAccount $twofaccount
|
|
|
|
* @return \Illuminate\Http\Response
|
|
|
|
*/
|
|
|
|
public static function get(String $uri) {
|
|
|
|
|
|
|
|
try {
|
|
|
|
return Factory::loadFromProvisioningUri($uri);
|
|
|
|
}
|
|
|
|
catch (AssertionFailedException $exception) {
|
|
|
|
$error = \Illuminate\Validation\ValidationException::withMessages([
|
|
|
|
'qrcode' => __('errors.response.no_valid_totp')
|
|
|
|
]);
|
|
|
|
|
|
|
|
throw $error;
|
|
|
|
}
|
2020-01-24 22:37:48 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|