2020-03-11 21:17:52 +08:00
|
|
|
<?php
|
|
|
|
|
|
|
|
class RecaptchaPlugin extends \RainLoop\Plugins\AbstractPlugin
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function Init()
|
|
|
|
{
|
|
|
|
$this->UseLangs(true);
|
|
|
|
|
|
|
|
$this->addJs('js/recaptcha.js');
|
|
|
|
|
|
|
|
$this->addHook('ajax.action-pre-call', 'AjaxActionPreCall');
|
|
|
|
$this->addHook('filter.ajax-response', 'FilterAjaxResponse');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function configMapping()
|
|
|
|
{
|
|
|
|
return array(
|
|
|
|
\RainLoop\Plugins\Property::NewInstance('public_key')->SetLabel('Site key')
|
|
|
|
->SetAllowedInJs(true)
|
|
|
|
->SetDefaultValue(''),
|
|
|
|
\RainLoop\Plugins\Property::NewInstance('private_key')->SetLabel('Secret key')
|
|
|
|
->SetDefaultValue(''),
|
|
|
|
\RainLoop\Plugins\Property::NewInstance('theme')->SetLabel('Theme')
|
|
|
|
->SetAllowedInJs(true)
|
|
|
|
->SetType(\RainLoop\Enumerations\PluginPropertyType::SELECTION)
|
|
|
|
->SetDefaultValue(array('light', 'dark')),
|
|
|
|
\RainLoop\Plugins\Property::NewInstance('error_limit')->SetLabel('Limit')
|
|
|
|
->SetType(\RainLoop\Enumerations\PluginPropertyType::SELECTION)
|
|
|
|
->SetDefaultValue(array(0, 1, 2, 3, 4, 5))
|
|
|
|
->SetDescription('')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
private function getCaptchaCacherKey()
|
|
|
|
{
|
|
|
|
return 'CaptchaNew/Login/'.\RainLoop\Utils::GetConnectionToken();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
private function getLimit()
|
|
|
|
{
|
|
|
|
$iConfigLimit = $this->Config()->Get('plugin', 'error_limit', 0);
|
|
|
|
if (0 < $iConfigLimit)
|
|
|
|
{
|
|
|
|
$oCacher = $this->Manager()->Actions()->Cacher();
|
|
|
|
$sLimit = $oCacher && $oCacher->IsInited() ? $oCacher->Get($this->getCaptchaCacherKey()) : '0';
|
|
|
|
|
|
|
|
if (0 < \strlen($sLimit) && \is_numeric($sLimit))
|
|
|
|
{
|
|
|
|
$iConfigLimit -= (int) $sLimit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $iConfigLimit;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function FilterAppDataPluginSection($bAdmin, $bAuth, &$aData)
|
|
|
|
{
|
|
|
|
if (!$bAdmin && !$bAuth && \is_array($aData))
|
|
|
|
{
|
|
|
|
$aData['show_captcha_on_login'] = 1 > $this->getLimit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $sAction
|
|
|
|
*/
|
|
|
|
public function AjaxActionPreCall($sAction)
|
|
|
|
{
|
|
|
|
if ('Login' === $sAction && 0 >= $this->getLimit())
|
|
|
|
{
|
|
|
|
$bResult = false;
|
|
|
|
|
|
|
|
$sResult = $this->Manager()->Actions()->Http()->SendPostRequest(
|
|
|
|
'https://www.google.com/recaptcha/api/siteverify',
|
|
|
|
array(
|
|
|
|
'secret' => $this->Config()->Get('plugin', 'private_key', ''),
|
|
|
|
'response' => $this->Manager()->Actions()->GetActionParam('RecaptchaResponse', '')
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($sResult)
|
|
|
|
{
|
|
|
|
$aResp = @\json_decode($sResult, true);
|
|
|
|
if (\is_array($aResp) && isset($aResp['success']) && $aResp['success'])
|
|
|
|
{
|
|
|
|
$bResult = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$bResult)
|
|
|
|
{
|
|
|
|
$this->Manager()->Actions()->Logger()->Write('RecaptchaResponse:'.$sResult);
|
|
|
|
throw new \RainLoop\Exceptions\ClientException(\RainLoop\Notifications::CaptchaError);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $sAction
|
|
|
|
* @param array $aResponseItem
|
|
|
|
*/
|
|
|
|
public function FilterAjaxResponse($sAction, &$aResponseItem)
|
|
|
|
{
|
|
|
|
if ('Login' === $sAction && $aResponseItem && isset($aResponseItem['Result']))
|
|
|
|
{
|
|
|
|
$oCacher = $this->Manager()->Actions()->Cacher();
|
|
|
|
$iConfigLimit = (int) $this->Config()->Get('plugin', 'error_limit', 0);
|
|
|
|
|
|
|
|
$sKey = $this->getCaptchaCacherKey();
|
|
|
|
|
|
|
|
if (0 < $iConfigLimit && $oCacher && $oCacher->IsInited())
|
|
|
|
{
|
|
|
|
if (false === $aResponseItem['Result'])
|
|
|
|
{
|
|
|
|
$iLimit = 0;
|
|
|
|
$sLimut = $oCacher->Get($sKey);
|
|
|
|
if (0 < \strlen($sLimut) && \is_numeric($sLimut))
|
|
|
|
{
|
|
|
|
$iLimit = (int) $sLimut;
|
|
|
|
}
|
|
|
|
|
|
|
|
$oCacher->Set($sKey, ++$iLimit);
|
|
|
|
|
|
|
|
if ($iConfigLimit <= $iLimit)
|
|
|
|
{
|
|
|
|
$aResponseItem['Captcha'] = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$oCacher->Delete($sKey);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|