HEX
Server: Apache/2.4.6 (CentOS) OpenSSL/1.0.2k-fips PHP/7.4.30
System: Linux iZj6c1151k3ad370bosnmsZ 3.10.0-1160.76.1.el7.x86_64 #1 SMP Wed Aug 10 16:21:17 UTC 2022 x86_64
User: root (0)
PHP: 7.4.30
Disabled: NONE
Upload Files
File: /var/www/html/amberconcept/wp-content/plugins/goodbye-captcha/engine/GdbcBruteGuardian.php
<?php
/*
 * Copyright (C) 2015 Mihai Chelaru
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 */

final class GdbcBruteGuardian
{

	private static $arrSecurityDirectoryFiles = array(
		'.htaccess'  => 'deny from all',
		'index.php'  => '<?php',
		'index.html' => '',
	);

	private static $SITE_UNDER_ATTACK_FLAG   = 'wpbr-uad'; //under-attack-detected
	private static $SITE_ATTACK_WARNING_FLAG = 'wpbr-sid'; //suspicious-ips-detected

	public static function startGuarding()
	{
		if( !GdbcModulesController::isModuleRegistered(GdbcModulesController::MODULE_SETTINGS) )
			return;

		if( empty($_POST) )
			return;

		self::$SITE_UNDER_ATTACK_FLAG   .= '-' . get_current_blog_id();
		self::$SITE_ATTACK_WARNING_FLAG .= '-' . get_current_blog_id();


		$loginAttemptsHits      = 0;
		$arrLatestLoginAttempts = GdbcDbAccessController::getLatestLoginAttempts(60, false);
		foreach($arrLatestLoginAttempts as $index => $loginAttempt){
			$loginAttemptsHits += $loginAttempt->Hits;
			$arrLatestLoginAttempts[$loginAttempt->ClientIp] = $loginAttempt->Hits;
			unset($arrLatestLoginAttempts[$index]);
		}

		switch(true)
		{
			case ($loginAttemptsHits > 35 && !self::isSiteUnderAttack()) :

				self::triggerSiteUnderAttack();

				GdbcNotificationsController::sendBruteForceAttackDetectedEmailNotification($arrLatestLoginAttempts);

				break;

			case ($loginAttemptsHits > 25 && !self::isSiteUnderAttack()) :

				break;


			case ($loginAttemptsHits < 15) :

				if(self::isSiteUnderAttack()){
					self::unTriggerSiteUnderAttack();
				}

				break;
		}


		if(self::isSiteUnderAttack() && GdbcBruteForcePublicModule::getInstance()->getOption(GdbcBruteForceAdminModule::OPTION_AUTO_BLOCK_IP)) {
			if( isset($arrLatestLoginAttempts[GdbcIPUtils::getClientIpAddress()])  && $arrLatestLoginAttempts[GdbcIPUtils::getClientIpAddress()] > 4 && (!GdbcIPUtils::isClientIpBlackListed()) ){
				GdbcBlackListedIpsAdminModule::getInstance()->registerBlackListedIp(GdbcIPUtils::getClientIpAddress());
			}
		}

	}

	public static function isSiteUnderAttack()
	{
		static $siteUnderAttack = null;
		if(null !== $siteUnderAttack)
			return $siteUnderAttack;

		return $siteUnderAttack = self::flagExists(self::$SITE_UNDER_ATTACK_FLAG);
	}

	private static function triggerSiteUnderAttack()
	{
		self::setSiteFlag(self::$SITE_UNDER_ATTACK_FLAG, true);

		GdbcSettingsAdminModule::getInstance()->saveSecuredOptions(true);

	}

	private static function unTriggerSiteUnderAttack()
	{
		self::setSiteFlag(self::$SITE_UNDER_ATTACK_FLAG, false);
	}

	private static function flagExists($flagName)
	{
		$cacheHolder = GoodByeCaptchaUtils::getAvailableCacheStorage(self::getBaseCacheDirectoryPath());
		if(null === $cacheHolder)
			return 0;

		return $cacheHolder->has($flagName);
	}

	private static function setSiteFlag($flagName, $flagBooleanValue)
	{
		$cacheHolder = GoodByeCaptchaUtils::getAvailableCacheStorage(self::getBaseCacheDirectoryPath());
		if(null === $cacheHolder)
			return;

		if(false === $flagBooleanValue){
			return $cacheHolder->delete($flagName);
		}

		if($cacheHolder->getCacheStorage() instanceof MchGdbcCacheFileStorage)
		{
			foreach(self::$arrSecurityDirectoryFiles as $fileName => $fileContent)
			{
				$filePath = self::getBaseCacheDirectoryPath() . DIRECTORY_SEPARATOR . $fileName;
				if(MchGdbcWpUtils::fileExists($filePath))
					break;

				MchGdbcWpUtils::writeContentToFile($fileContent, $filePath, false);
			}
		}

		$cacheHolder->write($flagName, MchGdbcHttpRequest::getServerRequestTime(false));
	}

	private static function getBaseCacheDirectoryPath()
	{
		static $cacheDirectoryPath = false;
		if(false !== $cacheDirectoryPath)
			return $cacheDirectoryPath;

//		$cacheDirectoryPath = GdbcSettingsPublicModule::getInstance()->getOption(GdbcSettingsAdminModule::OPTION_CACHE_DIR_PATH);
//		if(!empty($cacheDirectoryPath)){
//			$cacheDirectoryPath .= DIRECTORY_SEPARATOR . GoodByeCaptcha::PLUGIN_SLUG  . DIRECTORY_SEPARATOR . get_current_blog_id();
//			if(MchGdbcWpUtils::isDirectoryUsable($cacheDirectoryPath, true)){
//				return $cacheDirectoryPath;
//			}
//		}

		$cacheDirectoryPath = MchGdbcWpUtils::getDirectoryPathForCache();
		if(null === $cacheDirectoryPath)
			return null;

		$cacheDirectoryPath .= DIRECTORY_SEPARATOR . GoodByeCaptcha::PLUGIN_SLUG  . DIRECTORY_SEPARATOR . get_current_blog_id();
		if(! MchGdbcWpUtils::isDirectoryUsable($cacheDirectoryPath, true) ) {
			return $cacheDirectoryPath = null;
		}

//		GdbcSettingsAdminModule::getInstance()->saveOption(GdbcSettingsAdminModule::OPTION_CACHE_DIR_PATH, MchGdbcWpUtils::getDirectoryPathForCache());

		return $cacheDirectoryPath;
	}


	public static function isFormEntryLogEnabled()
	{
		return (bool)GdbcSettingsPublicModule::getInstance()->getOption(GdbcSettingsAdminModule::OPTION_BLOCKED_CONTENT_LOG_DAYS);
	}

	public static function logRejectedAttempt(GdbcAttemptEntity $attemptEntity)
	{
		if( ((int)GdbcSettingsPublicModule::getInstance()->getOption(GdbcSettingsAdminModule::OPTION_MAX_LOGS_DAYS)) < 1 ) //logs are NOT enabled
			return;

		$attemptEntity->SiteId      = get_current_blog_id();
		$attemptEntity->CreatedDate = MchGdbcHttpRequest::getServerRequestTime();
		$attemptEntity->ClientIp    = GdbcIPUtils::getClientIpAddress();

		if(!self::isFormEntryLogEnabled()) //blocked content logs are NOT enabled
			$attemptEntity->Notes = null;

		GdbcDbAccessController::registerAttempt( $attemptEntity );

//		if(self::isSiteUnderAttack() && GoodByeCaptchaUtils::isLoginAttemptEntity($attemptEntity) && ( !GdbcIPUtils::isClientIpWhiteListed() )){
//			self::registerClientIpBruteForceRequest();
//		}

	}

	private static function getIpAddressDirPath($flagName, $ipAddress)
	{
		if (null === self::getBaseCacheDirectoryPath())
			return null;

		$subDirectoryName = null;
		switch ($flagName)
		{
			case self::$SITE_UNDER_ATTACK_FLAG   : $subDirectoryName = 'brute-ips'; break;
			case self::$SITE_ATTACK_WARNING_FLAG : $subDirectoryName = 'suspicious-ips'; break;
		}

		return self::getBaseCacheDirectoryPath() . DIRECTORY_SEPARATOR . $subDirectoryName . DIRECTORY_SEPARATOR . $ipAddress;
	}

	private static function isClientIpBruteForcing()
	{
		return @is_dir(self::getIpAddressDirPath(self::$SITE_UNDER_ATTACK_FLAG, GdbcIPUtils::getClientIpAddress()));
	}

	private static function registerClientIpBruteForceRequest()
	{
		if(self::isClientIpBruteForcing())
		{
			MchGdbcWpUtils::writeContentToFile(null, self::getIpAddressDirPath(self::$SITE_UNDER_ATTACK_FLAG, GdbcIPUtils::getClientIpAddress()) . DIRECTORY_SEPARATOR . MchGdbcHttpRequest::getServerRequestTime(true) . '.gdbc', false);
			return;
		}

		if( ! MchGdbcWpUtils::isDirectoryUsable(self::getIpAddressDirPath(self::$SITE_UNDER_ATTACK_FLAG, GdbcIPUtils::getClientIpAddress()), true) )
			return;

		foreach(self::$arrSecurityDirectoryFiles as $fileName => $fileContent){
			$filePath = self::getIpAddressDirPath(self::$SITE_UNDER_ATTACK_FLAG, GdbcIPUtils::getClientIpAddress()) . DIRECTORY_SEPARATOR . $fileName;
			MchGdbcWpUtils::writeContentToFile($fileContent, $filePath, false);
		}

		MchGdbcWpUtils::writeContentToFile(null, self::getIpAddressDirPath(self::$SITE_UNDER_ATTACK_FLAG, GdbcIPUtils::getClientIpAddress()) . DIRECTORY_SEPARATOR . MchGdbcHttpRequest::getServerRequestTime(true) . '.gdbc', false);
	}

	private function __construct()
	{}

}