403Webshell
Server IP : 80.87.202.40  /  Your IP : 216.73.216.169
Web Server : Apache
System : Linux rospirotorg.ru 5.14.0-539.el9.x86_64 #1 SMP PREEMPT_DYNAMIC Thu Dec 5 22:26:13 UTC 2024 x86_64
User : bitrix ( 600)
PHP Version : 8.2.27
Disable Function : NONE
MySQL : OFF |  cURL : ON |  WGET : ON |  Perl : ON |  Python : OFF |  Sudo : ON |  Pkexec : ON
Directory :  /home/bitrix/ext_www/rospirotorg.ru/bitrix/modules/main/lib/web/

Upload File :
current_dir [ Writeable] document_root [ Writeable]

 

Command :


[ Back ]     

Current File : /home/bitrix/ext_www/rospirotorg.ru/bitrix/modules/main/lib/web/cookiescrypter.php
<?php

namespace Bitrix\Main\Web;

use Bitrix\Main\Config;
use Bitrix\Main\Security\Cipher;
use Bitrix\Main\Security\SecurityException;
use Bitrix\Main\Session\KernelSession;
use Bitrix\Main\SystemException;

/**
 * Class CookiesCrypter
 * Designed to encrypt and decrypt cookies.
 */
final class CookiesCrypter
{
	public const COOKIE_MAX_SIZE = 4096;
	public const COOKIE_RESERVED_SUFFIX_BYTES = 3;

	private const SIGN_PREFIX = '-crpt-';
	private const CIPHER_KEY_SUFFIX = 'cookiecrypter';

	protected ?string $cipherKey;
	protected Cipher $cipher;

	public function __construct()
	{
	}

	/**
	 * Builds cipher object.
	 * @return $this
	 * @throws SystemException
	 */
	protected function buildCipher(): self
	{
		if (isset($this->cipher))
		{
			return $this;
		}

		$configuration = Config\Configuration::getInstance();

		$this->cipher = new Cipher();
		$this->cipherKey = $configuration->get('crypto')['crypto_key'] ?? null;
		if (!$this->cipherKey)
		{
			throw new SystemException('There is no crypto[crypto_key] in .settings.php. Generate it.');
		}
		$this->cipherKey = $this->prependSuffixToKey($this->cipherKey);

		return $this;
	}

	/**
	 * Prepends suffix to key.
	 * @param string $key Key to prepend suffix to.
	 * @return string
	 */
	protected function prependSuffixToKey(string $key): string
	{
		return $key . self::CIPHER_KEY_SUFFIX;
	}

	/**
	 * Packs and encrypts cookie.
	 * @param CryptoCookie $cookie Cookie to encrypt.
	 * @return iterable|Cookie[]
	 * @throws SecurityException
	 * @throws SystemException
	 */
	public function encrypt(CryptoCookie $cookie): iterable
	{
		$result = [];
		$encryptedValue = $this->encryptValue($cookie->getValue());
		foreach ($this->packCookie($cookie, $encryptedValue) as $partCookie)
		{
			$result[] = $partCookie;
		}

		return $result;
	}

	/**
	 * Decrypts cookie if needed.
	 * @param string $name Cookie name.
	 * @param string $value Encrypted cookie value.
	 * @param iterable $cookies Cookies to decrypt.
	 * @return string
	 */
	public function decrypt(string $name, string $value, iterable $cookies): string
	{
		if (!$this->shouldDecrypt($name, $value))
		{
			return $value;
		}

		try
		{
			return $this->unpackCookie($value, $cookies);
		}
		catch (SecurityException)
		{
			//just skip cookies which we can't decrypt.
		}

		return '';
	}

	/**
	 * Packs cookie into several parts to fit into cookie size limit.
	 * @param CryptoCookie $cookie Cookie to pack.
	 * @param string       $encryptedValue Encrypted cookie value.
	 * @return iterable|Cookie[]
	 */
	protected function packCookie(CryptoCookie $cookie, string $encryptedValue): iterable
	{
		$length = \strlen($encryptedValue);
		$maxContentLength = self::COOKIE_MAX_SIZE - self::COOKIE_RESERVED_SUFFIX_BYTES - \strlen($cookie->getName());

		$i = 0;
		$parts = ($length / $maxContentLength);
		$pack = [];
		do
		{
			$startPosition = $i * $maxContentLength;
			$partCookie = new Cookie("{$cookie->getName()}_{$i}", substr($encryptedValue, $startPosition, $maxContentLength));
			$cookie->copyAttributesTo($partCookie);
			$pack["{$cookie->getOriginalName()}_{$i}"] = $partCookie;

			$i++;
		}
		while($parts > $i);

		$mainCookie = new Cookie($cookie->getName(), $this->prependSign(implode(',', array_keys($pack))));
		$cookie->copyAttributesTo($mainCookie);

		array_unshift($pack, $mainCookie);

		return $pack;
	}

	/**
	 * Unpacks cookie from several parts.
	 * @param string $mainCookie Main cookie value.
	 * @param iterable $cookies Cookies to decrypt.
	 * @return string
	 */
	protected function unpackCookie(string $mainCookie, iterable $cookies): string
	{
		$mainCookie = $this->removeSign($mainCookie);
		$packedNames = array_flip(array_filter(explode(',', $mainCookie)));
		$parts = [];

		foreach ($cookies as $name => $value)
		{
			if (!isset($packedNames[$name]))
			{
				continue;
			}

			$parts[$packedNames[$name]] = $value;
			if (\count($parts) === \count($packedNames))
			{
				break;
			}
		}
		ksort($parts);
		$encryptedValue = implode('', $parts);

		return $this->decryptValue($encryptedValue);
	}

	/**
	 * Encrypts value.
	 * Also compresses value if possible.
	 * @param string $value Value to encrypt.
	 * @return string
	 * @throws SecurityException
	 * @throws SystemException
	 */
	protected function encryptValue(string $value): string
	{
		$this->buildCipher();
		if (\function_exists('gzencode'))
		{
			$value = gzencode($value);
		}

		return $this->encodeUrlSafeB64($this->cipher->encrypt($value, $this->getCipherKey()));
	}

	/**
	 * Decrypts value.
	 * @param string $value Value to decrypt.
	 * @return string
	 * @throws SecurityException
	 * @throws SystemException
	 */
	protected function decryptValue(string $value): string
	{
		$this->buildCipher();

		$value = $this->cipher->decrypt($this->decodeUrlSafeB64($value), $this->getCipherKey());
		if (\function_exists('gzdecode'))
		{
			$value = gzdecode($value);
		}

		return $value;
	}

	/**
	 * Decodes url safe base64.
	 * @param string $input Input string.
	 * @return string
	 */
	private function decodeUrlSafeB64(string $input): string
	{
		$padLength = 4 - \strlen($input) % 4;
		$input .= str_repeat('=', $padLength);

		return base64_decode(strtr($input, '-_', '+/')) ?: '';
	}

	/**
	 * Encodes url safe base64 to avoid issues with cookie values.
	 * @param string $input Input string.
	 * @return string
	 */
	private function encodeUrlSafeB64(string $input): string
	{
		return str_replace('=', '', strtr(base64_encode($input), '+/', '-_'));
	}

	/**
	 * Checks if cookie should be encrypted.
	 * @param Cookie $cookie Cookie to check.
	 * @return bool
	 */
	public function shouldEncrypt(Cookie $cookie): bool
	{
		return $cookie instanceof CryptoCookie;
	}

	/**
	 * Checks if cookie should be decrypted.
	 * @param string $cookieName Cookie name.
	 * @param string $cookieValue Cookie value.
	 * @return bool
	 */
	public function shouldDecrypt(string $cookieName, string $cookieValue): bool
	{
		if ($cookieName === KernelSession::COOKIE_NAME)
		{
			return true;
		}

		return str_starts_with($cookieValue, self::SIGN_PREFIX);
	}

	/**
	 * Prepends sign to value.
	 * @param string $value Value to prepend sign to.
	 * @return string
	 */
	protected function prependSign(string $value): string
	{
		return self::SIGN_PREFIX . $value;
	}

	/**
	 * Removes sign prefix from value.
	 * @param string $value Value to remove sign from.
	 * @return string
	 */
	protected function removeSign(string $value): string
	{
		return substr($value, \strlen(self::SIGN_PREFIX));
	}

	/**
	 * Returns cipher key.
	 * @return string
	 */
	public function getCipherKey(): string
	{
		return $this->cipherKey;
	}
}

Youez - 2016 - github.com/yon3zu
LinuXploit