mirror of
https://github.com/the-djmaze/snappymail.git
synced 2025-01-05 06:22:52 +08:00
444 lines
13 KiB
PHP
444 lines
13 KiB
PHP
<?php
|
|
|
|
use \RainLoop\Exceptions\ClientException;
|
|
|
|
class TwoFactorAuthPlugin extends \RainLoop\Plugins\AbstractPlugin
|
|
{
|
|
const
|
|
NAME = 'Two Factor Authentication',
|
|
VERSION = '2.0',
|
|
RELEASE = '2021-04-13',
|
|
REQUIRED = '2.5.0',
|
|
CATEGORY = 'Login',
|
|
DESCRIPTION = 'This plugin allows you to to have TOTP';
|
|
|
|
// \RainLoop\Notifications\
|
|
const
|
|
AccountTwoFactorAuthRequired = 120,
|
|
AccountTwoFactorAuthError = 121,
|
|
|
|
Capa_TWO_FACTOR = 'TWO_FACTOR',
|
|
Capa_TWO_FACTOR_FORCE = 'TWO_FACTOR_FORCE';
|
|
|
|
/**
|
|
* @var \RainLoop\Providers\TwoFactorAuth
|
|
*/
|
|
private $oTwoFactorAuthProvider;
|
|
|
|
public function Init() : void
|
|
{
|
|
$this->UseLangs(true);
|
|
|
|
// $this->addCss('style.less');
|
|
$this->addJs('js/TwoFactorAuthSettings.js');
|
|
/*
|
|
$this->addHook('api.bootstrap.plugins');
|
|
$this->addHook('event.imap-post-login');
|
|
$this->addHook('event.imap-pre-connect');
|
|
$this->addHook('event.imap-pre-login');
|
|
$this->addHook('event.login-post-login-provide');
|
|
$this->addHook('event.login-pre-login-provide');
|
|
$this->addHook('event.sieve-post-connect');
|
|
$this->addHook('event.sieve-post-login');
|
|
$this->addHook('event.sieve-pre-connect');
|
|
$this->addHook('event.sieve-pre-login');
|
|
$this->addHook('event.smtp-post-connect');
|
|
$this->addHook('event.smtp-post-login');
|
|
$this->addHook('event.smtp-pre-connect');
|
|
$this->addHook('event.smtp-pre-login');
|
|
$this->addHook('filter.acount');
|
|
$this->addHook('filter.action-params');
|
|
$this->addHook('filter.app-data');
|
|
$this->addHook('filter.application-config');
|
|
$this->addHook('filter.build-message');
|
|
$this->addHook('filter.build-read-receipt-message');
|
|
$this->addHook('filter.domain');
|
|
$this->addHook('filter.fabrica');
|
|
$this->addHook('filter.folders-before');
|
|
$this->addHook('filter.folders-complete');
|
|
$this->addHook('filter.folders-post');
|
|
$this->addHook('filter.folders-system-types');
|
|
$this->addHook('filter.http-paths');
|
|
$this->addHook('filter.http-query');
|
|
$this->addHook('filter.imap-credentials');
|
|
$this->addHook('filter.json-response');
|
|
$this->addHook('filter.login-credentials');
|
|
$this->addHook('filter.login-credentials.step-1');
|
|
$this->addHook('filter.login-credentials.step-2');
|
|
$this->addHook('filter.message-html'
|
|
$this->addHook('filter.message-plain');
|
|
$this->addHook('filter.message-rcpt');
|
|
$this->addHook('filter.read-receipt-message-plain');
|
|
$this->addHook('filter.result-message');
|
|
$this->addHook('filter.save-message');
|
|
$this->addHook('filter.send-message');
|
|
$this->addHook('filter.send-message-stream');
|
|
$this->addHook('filter.send-read-receipt-message');
|
|
$this->addHook('filter.sieve-credentials');
|
|
$this->addHook('filter.smtp-credentials');
|
|
$this->addHook('filter.smtp-from');
|
|
$this->addHook('filter.smtp-hidden-rcpt');
|
|
$this->addHook('filter.smtp-message-stream');
|
|
$this->addHook('filter.system-folders-names');
|
|
$this->addHook('filter.upload-response');
|
|
$this->addHook('json.action-post-call');
|
|
$this->addHook('json.action-pre-call');
|
|
$this->addHook('json.suggestions-input-parameters');
|
|
$this->addHook('json.suggestions-post');
|
|
$this->addHook('json.suggestions-pre');
|
|
$this->addHook('main.default-response');
|
|
$this->addHook('main.default-response-data');
|
|
$this->addHook('main.default-response-data');
|
|
$this->addHook('main.default-response-error-data');
|
|
$this->addHook('main.fabrica');
|
|
$this->addHook('service.app-delay-start-begin');
|
|
$this->addHook('service.app-delay-start-end');
|
|
*/
|
|
|
|
$this->addJsonHook('GetTwoFactorInfo', 'DoGetTwoFactorInfo');
|
|
$this->addJsonHook('CreateTwoFactorSecret', 'DoCreateTwoFactorSecret');
|
|
$this->addJsonHook('ShowTwoFactorSecret', 'DoShowTwoFactorSecret');
|
|
$this->addJsonHook('EnableTwoFactor', 'DoEnableTwoFactor');
|
|
$this->addJsonHook('VerifyTwoFactorCode', 'DoVerifyTwoFactorCode');
|
|
$this->addJsonHook('ClearTwoFactorInfo', 'DoClearTwoFactorInfo');
|
|
|
|
$this->addTemplate('templates/TwoFactorAuthSettings.html');
|
|
$this->addTemplate('templates/PopupsTwoFactorAuthTest.html');
|
|
}
|
|
|
|
public function configMapping() : array
|
|
{
|
|
return [
|
|
\RainLoop\Plugins\Property::NewInstance('allow_two_factor_auth')
|
|
->SetLabel('TAB_SECURITY/LABEL_ALLOW_TWO_STEP')
|
|
->SetType(\RainLoop\Enumerations\PluginPropertyType::BOOL),
|
|
\RainLoop\Plugins\Property::NewInstance('force_two_factor_auth')
|
|
->SetLabel('TAB_SECURITY/LABEL_FORCE_TWO_STEP')
|
|
->SetType(\RainLoop\Enumerations\PluginPropertyType::BOOL)
|
|
];
|
|
}
|
|
|
|
public function DoGetTwoFactorInfo() : array
|
|
{
|
|
$oAccount = $this->getAccountFromToken();
|
|
|
|
if (!$this->TwoFactorAuthProvider() ||
|
|
!$this->GetCapa(false, static::Capa_TWO_FACTOR, $oAccount))
|
|
{
|
|
return $this->FalseResponse(__FUNCTION__);
|
|
}
|
|
|
|
return $this->DefaultResponse(__FUNCTION__,
|
|
$this->getTwoFactorInfo($oAccount, true));
|
|
}
|
|
|
|
public function DoCreateTwoFactorSecret() : array
|
|
{
|
|
$oAccount = $this->getAccountFromToken();
|
|
|
|
if (!$this->TwoFactorAuthProvider() ||
|
|
!$this->GetCapa(false, static::Capa_TWO_FACTOR, $oAccount))
|
|
{
|
|
return $this->FalseResponse(__FUNCTION__);
|
|
}
|
|
|
|
$sEmail = $oAccount->ParentEmailHelper();
|
|
|
|
$sSecret = $this->TwoFactorAuthProvider()->CreateSecret();
|
|
|
|
$aCodes = array();
|
|
for ($iIndex = 9; $iIndex > 0; $iIndex--)
|
|
{
|
|
$aCodes[] = \rand(100000000, 900000000);
|
|
}
|
|
|
|
$this->StorageProvider()->Put($oAccount,
|
|
\RainLoop\Providers\Storage\Enumerations\StorageType::CONFIG,
|
|
'two_factor',
|
|
\RainLoop\Utils::EncodeKeyValues(array(
|
|
'User' => $sEmail,
|
|
'Enable' => false,
|
|
'Secret' => $sSecret,
|
|
'BackupCodes' => \implode(' ', $aCodes)
|
|
))
|
|
);
|
|
|
|
$this->requestSleep();
|
|
|
|
return $this->DefaultResponse(__FUNCTION__,
|
|
$this->getTwoFactorInfo($oAccount));
|
|
}
|
|
|
|
public function DoShowTwoFactorSecret() : array
|
|
{
|
|
$oAccount = $this->getAccountFromToken();
|
|
|
|
if (!$this->TwoFactorAuthProvider() ||
|
|
!$this->GetCapa(false, static::Capa_TWO_FACTOR, $oAccount))
|
|
{
|
|
return $this->FalseResponse(__FUNCTION__);
|
|
}
|
|
|
|
$aResult = $this->getTwoFactorInfo($oAccount);
|
|
unset($aResult['BackupCodes']);
|
|
|
|
return $this->DefaultResponse(__FUNCTION__, $aResult);
|
|
}
|
|
|
|
public function DoEnableTwoFactor() : array
|
|
{
|
|
$oAccount = $this->getAccountFromToken();
|
|
|
|
if (!$this->TwoFactorAuthProvider() ||
|
|
!$this->GetCapa(false, static::Capa_TWO_FACTOR, $oAccount))
|
|
{
|
|
return $this->FalseResponse(__FUNCTION__);
|
|
}
|
|
|
|
// $this->setSettingsFromParams($oSettings, 'EnableTwoFactor', 'bool');
|
|
|
|
$sEmail = $oAccount->ParentEmailHelper();
|
|
|
|
$bResult = false;
|
|
$mData = $this->getTwoFactorInfo($oAccount);
|
|
if (isset($mData['Secret'], $mData['BackupCodes']))
|
|
{
|
|
$bResult = $this->StorageProvider()->Put($oAccount,
|
|
\RainLoop\Providers\Storage\Enumerations\StorageType::CONFIG,
|
|
'two_factor',
|
|
\RainLoop\Utils::EncodeKeyValues(array(
|
|
'User' => $sEmail,
|
|
'Enable' => '1' === \trim($this->jsonParam('Enable', '0')),
|
|
'Secret' => $mData['Secret'],
|
|
'BackupCodes' => $mData['BackupCodes']
|
|
))
|
|
);
|
|
}
|
|
|
|
return $this->DefaultResponse(__FUNCTION__, $bResult);
|
|
}
|
|
|
|
public function DoVerifyTwoFactorCode() : array
|
|
{
|
|
$oAccount = $this->getAccountFromToken();
|
|
|
|
if (!$this->TwoFactorAuthProvider() ||
|
|
!$this->GetCapa(false, static::Capa_TWO_FACTOR, $oAccount))
|
|
{
|
|
return $this->FalseResponse(__FUNCTION__);
|
|
}
|
|
|
|
$sCode = \trim($this->jsonParam('Code', ''));
|
|
|
|
$aData = $this->getTwoFactorInfo($oAccount);
|
|
$sSecret = !empty($aData['Secret']) ? $aData['Secret'] : '';
|
|
|
|
// $this->Logger()->WriteDump(array(
|
|
// $sCode, $sSecret, $aData,
|
|
// $this->TwoFactorAuthProvider()->VerifyCode($sSecret, $sCode)
|
|
// ));
|
|
|
|
$this->requestSleep();
|
|
|
|
return $this->DefaultResponse(__FUNCTION__,
|
|
$this->TwoFactorAuthProvider()->VerifyCode($sSecret, $sCode));
|
|
}
|
|
|
|
public function DoClearTwoFactorInfo() : array
|
|
{
|
|
$oAccount = $this->getAccountFromToken();
|
|
|
|
if (!$this->TwoFactorAuthProvider() ||
|
|
!$this->GetCapa(false, static::Capa_TWO_FACTOR, $oAccount))
|
|
{
|
|
return $this->FalseResponse(__FUNCTION__);
|
|
}
|
|
|
|
$this->StorageProvider()->Clear($oAccount,
|
|
\RainLoop\Providers\Storage\Enumerations\StorageType::CONFIG,
|
|
'two_factor'
|
|
);
|
|
|
|
return $this->DefaultResponse(__FUNCTION__,
|
|
$this->getTwoFactorInfo($oAccount, true));
|
|
}
|
|
|
|
protected function TwoFactorAuthProvider() : \RainLoop\Providers\TwoFactorAuth
|
|
{
|
|
// if ($this->Config()->Get('plugin', 'allow_two_factor_auth', 0))
|
|
// if ($this->Config()->Get('plugin', 'force_two_factor_auth', 0))
|
|
|
|
if (!$this->oTwoFactorAuthProvider) {
|
|
require __DIR__ . '/providers/interface.php';
|
|
require __DIR__ . '/providers/totp.php';
|
|
$this->oTwoFactorAuthProvider = new TwoFactorAuthTotp();
|
|
}
|
|
return $this->oTwoFactorAuthProvider;
|
|
}
|
|
|
|
protected function getTwoFactorInfo(\RainLoop\Model\Account $oAccount, bool $bRemoveSecret = false) : array
|
|
{
|
|
$sEmail = $oAccount->ParentEmailHelper();
|
|
|
|
$mData = null;
|
|
|
|
$aResult = array(
|
|
'User' => '',
|
|
'IsSet' => false,
|
|
'Enable' => false,
|
|
'Secret' => '',
|
|
'UrlTitle' => '',
|
|
'BackupCodes' => ''
|
|
);
|
|
|
|
if (!empty($sEmail))
|
|
{
|
|
$aResult['User'] = $sEmail;
|
|
|
|
$sData = $this->StorageProvider()->Get($oAccount,
|
|
\RainLoop\Providers\Storage\Enumerations\StorageType::CONFIG,
|
|
'two_factor'
|
|
);
|
|
|
|
if ($sData)
|
|
{
|
|
$mData = \RainLoop\Utils::DecodeKeyValues($sData);
|
|
}
|
|
}
|
|
|
|
if (!empty($aResult['User']) &&
|
|
!empty($mData['User']) && !empty($mData['Secret']) &&
|
|
!empty($mData['BackupCodes']) && $sEmail === $mData['User'])
|
|
{
|
|
$aResult['IsSet'] = true;
|
|
$aResult['Enable'] = isset($mData['Enable']) ? !!$mData['Enable'] : false;
|
|
$aResult['Secret'] = $mData['Secret'];
|
|
$aResult['BackupCodes'] = $mData['BackupCodes'];
|
|
$aResult['UrlTitle'] = $this->Config()->Get('webmail', 'title', '');
|
|
}
|
|
|
|
if ($bRemoveSecret)
|
|
{
|
|
if (isset($aResult['Secret']))
|
|
{
|
|
unset($aResult['Secret']);
|
|
}
|
|
|
|
if (isset($aResult['UrlTitle']))
|
|
{
|
|
unset($aResult['UrlTitle']);
|
|
}
|
|
|
|
if (isset($aResult['BackupCodes']))
|
|
{
|
|
unset($aResult['BackupCodes']);
|
|
}
|
|
}
|
|
|
|
return $aResult;
|
|
}
|
|
|
|
protected function removeBackupCodeFromTwoFactorInfo(\RainLoop\Model\Account $oAccount, string $sCode) : bool
|
|
{
|
|
if (!$oAccount || empty($sCode))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
$sData = $this->StorageProvider()->Get($oAccount,
|
|
\RainLoop\Providers\Storage\Enumerations\StorageType::CONFIG,
|
|
'two_factor'
|
|
);
|
|
|
|
if ($sData)
|
|
{
|
|
$mData = \RainLoop\Utils::DecodeKeyValues($sData);
|
|
|
|
if (!empty($mData['BackupCodes']))
|
|
{
|
|
$sBackupCodes = \preg_replace('/[^\d]+/', ' ', ' '.$mData['BackupCodes'].' ');
|
|
$sBackupCodes = \str_replace(' '.$sCode.' ', '', $sBackupCodes);
|
|
|
|
$mData['BackupCodes'] = \trim(\preg_replace('/[^\d]+/', ' ', $sBackupCodes));
|
|
|
|
return $this->StorageProvider()->Put($oAccount,
|
|
\RainLoop\Providers\Storage\Enumerations\StorageType::CONFIG,
|
|
'two_factor',
|
|
\RainLoop\Utils::EncodeKeyValues($mData)
|
|
);
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
public function ChangePassword()
|
|
{
|
|
$oActions = $this->Manager()->Actions();
|
|
$oAccount = $oActions->GetAccount();
|
|
|
|
if (!$oAccount->Email()) {
|
|
\trigger_error('ChangePassword failed: empty email address');
|
|
throw new ClientException(static::CouldNotSaveNewPassword);
|
|
}
|
|
|
|
$sPrevPassword = $this->jsonParam('PrevPassword');
|
|
if ($sPrevPassword !== $oAccount->Password()) {
|
|
throw new ClientException(static::CurrentPasswordIncorrect, null, $oActions->StaticI18N('NOTIFICATIONS/CURRENT_PASSWORD_INCORRECT'));
|
|
}
|
|
|
|
$sNewPassword = $this->jsonParam('NewPassword');
|
|
if ($this->Config()->Get('plugin', 'pass_min_length', 10) > \strlen($sNewPassword)) {
|
|
throw new ClientException(static::NewPasswordShort, null, $oActions->StaticI18N('NOTIFICATIONS/NEW_PASSWORD_SHORT'));
|
|
}
|
|
|
|
if ($this->Config()->Get('plugin', 'pass_min_strength', 70) > static::PasswordStrength($sNewPassword)) {
|
|
throw new ClientException(static::NewPasswordWeak, null, $oActions->StaticI18N('NOTIFICATIONS/NEW_PASSWORD_WEAK'));
|
|
}
|
|
|
|
$bResult = false;
|
|
$oConfig = $this->Config();
|
|
foreach ($this->getSupportedDrivers() as $name => $class) {
|
|
$sFoundedValue = '';
|
|
if (\RainLoop\Plugins\Helper::ValidateWildcardValues($oAccount->Email(), $oConfig->Get('plugin', "driver_{$name}_allowed_emails"), $sFoundedValue)) {
|
|
$name = $class::NAME;
|
|
$oLogger = $oActions->Logger();
|
|
try
|
|
{
|
|
$oDriver = new $class(
|
|
$oConfig,
|
|
$oLogger
|
|
);
|
|
if (!$oDriver->ChangePassword($oAccount, $sPrevPassword, $sNewPassword)) {
|
|
throw new ClientException(static::CouldNotSaveNewPassword);
|
|
}
|
|
$bResult = true;
|
|
if ($oLogger) {
|
|
$oLogger->Write("{$name} password changed for {$oAccount->Email()}");
|
|
}
|
|
}
|
|
catch (\Throwable $oException)
|
|
{
|
|
\trigger_error("{$class} failed: {$oException->getMessage()}");
|
|
if ($oLogger) {
|
|
$oLogger->Write("ERROR: {$name} password change for {$oAccount->Email()} failed");
|
|
$oLogger->WriteException($oException);
|
|
// $oLogger->WriteException($oException, \MailSo\Log\Enumerations\Type::WARNING, $name);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!$bResult) {
|
|
\trigger_error("ChangePassword failed");
|
|
throw new ClientException(static::CouldNotSaveNewPassword);
|
|
}
|
|
|
|
$oAccount->SetPassword($sNewPassword);
|
|
$oActions->SetAuthToken($oAccount);
|
|
|
|
return $this->jsonResponse(__FUNCTION__, $oActions->GetSpecAuthToken());
|
|
}
|
|
*/
|
|
}
|