HEX
Server: LiteSpeed
System: Linux php-prod-1.spaceapp.ru 5.15.0-160-generic #170-Ubuntu SMP Wed Oct 1 10:06:56 UTC 2025 x86_64
User: xnsbb3110 (1041)
PHP: 8.1.33
Disabled: NONE
Upload Files
File: //usr/local/CyberCP/public/snappymail/snappymail/v/2.38.2/app/libraries/MailSo/Mime/Parser.php
<?php

/*
 * This file is part of MailSo.
 *
 * (c) 2014 Usenko Timur
 *
 * For the full copyright and license information, please view the LICENSE
 * file that was distributed with this source code.
 */

namespace MailSo\Mime;

/**
 * @category MailSo
 * @package Mime
 */
abstract class Parser
{
	const
		POS_HEADERS = 1,
		POS_BODY = 2,
		POS_SUBPARTS = 3,
		POS_CLOSE_BOUNDARY = 4;

	private static array $LineParts = [];

	protected static function writeBody(Part $oPart, string $sBuffer) : void
	{
		if (null === $oPart->Body) {
			$oPart->Body = \MailSo\Base\ResourceRegistry::CreateMemoryResource();
		}
		if (\is_resource($oPart->Body)) {
			\fwrite($oPart->Body, $sBuffer);
		}
	}

	/**
	 * @param resource $rStreamHandle
	 */
	public static function parseStream($rStreamHandle) : ?Part
	{
		if (!\is_resource($rStreamHandle)) {
			return null;
		}

		$oPart = new Part;

		$bIsOef = false;
		$iOffset = 0;
		$sBuffer = '';
		$sPrevBuffer = '';
		$aBoundaryStack = array();

		static::$LineParts = [$oPart];

		static::parseStreamRecursive($oPart, $rStreamHandle, $iOffset,
			$sPrevBuffer, $sBuffer, $aBoundaryStack, $bIsOef);

		$oMimePart = null;
		$sFirstNotNullCharset = null;
		foreach (static::$LineParts as /* @var $oMimePart Part */ $oMimePart) {
			$sCharset = $oMimePart->HeaderCharset();
			if (\strlen($sCharset)) {
				$sFirstNotNullCharset = $sCharset;
				break;
			}
		}

		$sFirstNotNullCharset = (null !== $sFirstNotNullCharset)
			? $sFirstNotNullCharset : \MailSo\Base\Enumerations\Charset::ISO_8859_1;

		foreach (static::$LineParts as /* @var $oMimePart Part */ $oMimePart) {
			$sHeaderCharset = $oMimePart->HeaderCharset();
			$oMimePart->Headers->SetParentCharset($sHeaderCharset);
		}

		static::$LineParts = [];
		return $oPart;
	}

	/**
	 * @param resource $rStreamHandle
	 */
	public static function parseStreamRecursive(Part $oPart, $rStreamHandle, int &$iOffset,
		string &$sPrevBuffer, string &$sBuffer, array &$aBoundaryStack, bool &$bIsOef, bool $bNotFirstRead = false) : void
	{
		$iPos = 0;
		$iParsePosition = self::POS_HEADERS;
		$sCurrentBoundary = '';
		$bIsBoundaryCheck = false;
		$aHeadersLines = array();
		while (true) {
			if (!$bNotFirstRead) {
				$sPrevBuffer = $sBuffer;
				$sBuffer = '';
			}

			if (!$bIsOef && !\feof($rStreamHandle)) {
				if (!$bNotFirstRead) {
					$sBuffer = \fread($rStreamHandle, 8192);
					if (false === $sBuffer) {
						break;
					}
				} else {
					$bNotFirstRead = false;
				}
			} else if ($bIsOef && !\strlen($sBuffer)) {
				break;
			} else {
				$bIsOef = true;
			}

			while (true) {
				$sCurrentLine = $sPrevBuffer.$sBuffer;
				if (self::POS_HEADERS === $iParsePosition) {
					$iEndLen = 4;
					$iPos = \strpos($sCurrentLine, "\r\n\r\n", $iOffset);
					if (false === $iPos) {
						$iEndLen = 2;
						$iPos = \strpos($sCurrentLine, "\n\n", $iOffset);
					}

					if (false !== $iPos) {
						$aHeadersLines[] = \substr($sCurrentLine, $iOffset, $iPos + $iEndLen - $iOffset);

						$oPart->Headers->Parse(\implode($aHeadersLines))->SetParentCharset($oPart->HeaderCharset());
						$aHeadersLines = array();

						$sBoundary = $oPart->HeaderBoundary();
						if (\strlen($sBoundary))
						{
							$sBoundary = '--'.$sBoundary;
							$sCurrentBoundary = $sBoundary;
							\array_unshift($aBoundaryStack, $sBoundary);
						}

						$iOffset = $iPos + $iEndLen;
						$iParsePosition = self::POS_BODY;
						continue;
					} else {
						$iBufferLen = \strlen($sPrevBuffer);
						if ($iBufferLen > $iOffset) {
							$aHeadersLines[] = \substr($sPrevBuffer, $iOffset);
							$iOffset = 0;
						} else {
							$iOffset -= $iBufferLen;
						}
						break;
					}
				} else if (self::POS_BODY === $iParsePosition) {
					$iPos = false;
					$sBoundaryLen = 0;
					$bIsBoundaryEnd = false;
					$bCurrentPartBody = false;
					$bIsBoundaryCheck = \count($aBoundaryStack);

					foreach ($aBoundaryStack as $sKey => $sBoundary) {
						if (false !== ($iPos = \strpos($sCurrentLine, $sBoundary, $iOffset))) {
							if ($sCurrentBoundary === $sBoundary) {
								$bCurrentPartBody = true;
							}

							$sBoundaryLen = \strlen($sBoundary);
							if ('--' === \substr($sCurrentLine, $iPos + $sBoundaryLen, 2)) {
								$sBoundaryLen += 2;
								$bIsBoundaryEnd = true;
								unset($aBoundaryStack[$sKey]);
								$sCurrentBoundary = (isset($aBoundaryStack[$sKey + 1]))
									? $aBoundaryStack[$sKey + 1] : '';
							}

							break;
						}
					}

					if (false !== $iPos) {
						static::writeBody($oPart, \substr($sCurrentLine, $iOffset, $iPos - $iOffset));
						$iOffset = $iPos;

						if ($bCurrentPartBody) {
							$iParsePosition = self::POS_SUBPARTS;
							continue;
						}

						return;
					} else {
						$iBufferLen = \strlen($sPrevBuffer);
						if ($iBufferLen > $iOffset) {
							static::writeBody($oPart, \substr($sPrevBuffer, $iOffset));
							$iOffset = 0;
						} else {
							$iOffset -= $iBufferLen;
						}
						break;
					}
				} else if (self::POS_SUBPARTS === $iParsePosition) {
					$iPos = false;
					$iBoundaryLen = 0;
					$bIsBoundaryEnd = false;
					$bCurrentPartBody = false;
					$bIsBoundaryCheck = \count($aBoundaryStack);

					foreach ($aBoundaryStack as $sKey => $sBoundary) {
						$iPos = \strpos($sCurrentLine, $sBoundary, $iOffset);
						if (false !== $iPos) {
							if ($sCurrentBoundary === $sBoundary) {
								$bCurrentPartBody = true;
							}

							$iBoundaryLen = \strlen($sBoundary);
							if ('--' === \substr($sCurrentLine, $iPos + $iBoundaryLen, 2)) {
								$iBoundaryLen += 2;
								$bIsBoundaryEnd = true;
								unset($aBoundaryStack[$sKey]);
								$sCurrentBoundary = (isset($aBoundaryStack[$sKey + 1]))
									? $aBoundaryStack[$sKey + 1] : '';
							}
							break;
						}
					}

					if (false !== $iPos && $bCurrentPartBody) {
						$iOffset = $iPos + $iBoundaryLen;

						$oSubPart = new Part;

						static::parseStreamRecursive($oSubPart, $rStreamHandle,
							$iOffset, $sPrevBuffer, $sBuffer, $aBoundaryStack, $bIsOef, true);

						$oPart->SubParts->append($oSubPart);
						static::$LineParts[] = $oSubPart;
						//$iParsePosition = self::POS_HEADERS;
						unset($oSubPart);
					} else {
						return;
					}
				}
			}
		}

		if (\strlen($sPrevBuffer)) {
			if (self::POS_HEADERS === $iParsePosition) {
				$aHeadersLines[] = ($iOffset < \strlen($sPrevBuffer))
					? \substr($sPrevBuffer, $iOffset)
					: $sPrevBuffer;

				$oPart->Headers->Parse(\implode($aHeadersLines))->SetParentCharset($oPart->HeaderCharset());
				$aHeadersLines = array();
			} else if (!$bIsBoundaryCheck && self::POS_BODY === $iParsePosition) {
				static::writeBody($oPart, ($iOffset < \strlen($sPrevBuffer))
					? \substr($sPrevBuffer, $iOffset) : $sPrevBuffer);
			}
		} else {
			if (self::POS_HEADERS === $iParsePosition && \count($aHeadersLines)) {
				$oPart->Headers->Parse(\implode($aHeadersLines))->SetParentCharset($oPart->HeaderCharset());
				$aHeadersLines = array();
			}
		}

		return;
	}

}