mirror of
https://github.com/the-djmaze/snappymail.git
synced 2025-01-30 10:37:54 +08:00
0735071f1c
Filters interface fixes DKIM status a new message
480 lines
10 KiB
PHP
480 lines
10 KiB
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
|
|
*/
|
|
class HeaderCollection extends \MailSo\Base\Collection
|
|
{
|
|
/**
|
|
* @return string
|
|
*/
|
|
protected $sRawHeaders;
|
|
|
|
/**
|
|
* @var strign
|
|
*/
|
|
protected $sParentCharset;
|
|
|
|
/**
|
|
* @access protected
|
|
*
|
|
* @param string $sRawHeaders = ''
|
|
* @param bool $bStoreRawHeaders = true
|
|
*/
|
|
protected function __construct($sRawHeaders = '', $bStoreRawHeaders = true)
|
|
{
|
|
parent::__construct();
|
|
|
|
$this->sRawHeaders = '';
|
|
$this->sParentCharset = '';
|
|
|
|
if (0 < \strlen($sRawHeaders))
|
|
{
|
|
$this->Parse($sRawHeaders, $bStoreRawHeaders);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param string $sRawHeaders = ''
|
|
* @param bool $bStoreRawHeaders = true
|
|
*
|
|
* @return \MailSo\Mime\HeaderCollection
|
|
*/
|
|
public static function NewInstance($sRawHeaders = '', $bStoreRawHeaders = true)
|
|
{
|
|
return new self($sRawHeaders, $bStoreRawHeaders);
|
|
}
|
|
|
|
/**
|
|
* @param string $sName
|
|
* @param string $sValue
|
|
* @param bool $bToTop = false
|
|
*
|
|
* @return \MailSo\Mime\HeaderCollection
|
|
*
|
|
* @throws \MailSo\Base\Exceptions\InvalidArgumentException
|
|
*/
|
|
public function AddByName($sName, $sValue, $bToTop = false)
|
|
{
|
|
return $this->Add(Header::NewInstance($sName, $sValue), $bToTop);
|
|
}
|
|
|
|
/**
|
|
* @param string $sName
|
|
* @param string $sValue
|
|
* @param bool $bToTop = false
|
|
*
|
|
* @return \MailSo\Mime\HeaderCollection
|
|
*
|
|
* @throws \MailSo\Base\Exceptions\InvalidArgumentException
|
|
*/
|
|
public function SetByName($sName, $sValue, $bToTop = false)
|
|
{
|
|
return $this->RemoveByName($sName)->Add(Header::NewInstance($sName, $sValue), $bToTop);
|
|
}
|
|
|
|
/**
|
|
* @return \MailSo\Mime\Header | null
|
|
*/
|
|
public function &GetByIndex($iIndex)
|
|
{
|
|
$mResult = null;
|
|
$mResult =& parent::GetByIndex($iIndex);
|
|
return $mResult;
|
|
}
|
|
|
|
/**
|
|
* @param string $sHeaderName
|
|
* @param bool $bCharsetAutoDetect = false
|
|
* @return string
|
|
*/
|
|
public function ValueByName($sHeaderName, $bCharsetAutoDetect = false)
|
|
{
|
|
$oHeader = null;
|
|
$oHeader =& $this->GetByName($sHeaderName);
|
|
return (null !== $oHeader) ? ($bCharsetAutoDetect ? $oHeader->ValueWithCharsetAutoDetect() : $oHeader->Value()) : '';
|
|
}
|
|
|
|
/**
|
|
* @param string $sHeaderName
|
|
* @param bool $bCharsetAutoDetect = false
|
|
* @return array
|
|
*/
|
|
public function ValuesByName($sHeaderName, $bCharsetAutoDetect = false)
|
|
{
|
|
$aResult = array();
|
|
$oHeader = null;
|
|
|
|
$sHeaderNameLower = \strtolower($sHeaderName);
|
|
$aHeaders =& $this->GetAsArray();
|
|
foreach ($aHeaders as /* @var $oHeader \MailSo\Mime\Header */ &$oHeader)
|
|
{
|
|
if ($sHeaderNameLower === \strtolower($oHeader->Name()))
|
|
{
|
|
$aResult[] = $bCharsetAutoDetect ? $oHeader->ValueWithCharsetAutoDetect() : $oHeader->Value();
|
|
}
|
|
}
|
|
|
|
return $aResult;
|
|
}
|
|
|
|
/**
|
|
* @param string $sHeaderName
|
|
*
|
|
* @return \MailSo\Mime\HeaderCollection
|
|
*/
|
|
public function RemoveByName($sHeaderName)
|
|
{
|
|
$aResult = $this->FilterList(function ($oHeader) use ($sHeaderName) {
|
|
return $oHeader && \strtolower($oHeader->Name()) !== \strtolower($sHeaderName);
|
|
});
|
|
|
|
return $this->SetAsArray($aResult);
|
|
}
|
|
|
|
/**
|
|
* @param string $sHeaderName
|
|
* @param bool $bCharsetAutoDetect = false
|
|
*
|
|
* @return \MailSo\Mime\EmailCollection|null
|
|
*/
|
|
public function GetAsEmailCollection($sHeaderName, $bCharsetAutoDetect = false)
|
|
{
|
|
$oResult = null;
|
|
$sValue = $this->ValueByName($sHeaderName, $bCharsetAutoDetect);
|
|
if (0 < \strlen($sValue))
|
|
{
|
|
$oResult = \MailSo\Mime\EmailCollection::NewInstance($sValue);
|
|
}
|
|
|
|
return $oResult && 0 < $oResult->Count() ? $oResult : null;
|
|
}
|
|
|
|
/**
|
|
* @param string $sHeaderName
|
|
* @return \MailSo\Mime\ParameterCollection|null
|
|
*/
|
|
public function ParametersByName($sHeaderName)
|
|
{
|
|
$oParameters = $oHeader = null;
|
|
$oHeader =& $this->GetByName($sHeaderName);
|
|
if ($oHeader)
|
|
{
|
|
$oParameters = $oHeader->Parameters();
|
|
}
|
|
|
|
return $oParameters;
|
|
}
|
|
|
|
/**
|
|
* @param string $sHeaderName
|
|
* @param string $sParamName
|
|
* @return string
|
|
*/
|
|
public function ParameterValue($sHeaderName, $sParamName)
|
|
{
|
|
$oParameters = $this->ParametersByName($sHeaderName);
|
|
return (null !== $oParameters) ? $oParameters->ParameterValueByName($sParamName) : '';
|
|
}
|
|
|
|
/**
|
|
* @param string $sHeaderName
|
|
* @return \MailSo\Mime\Header | false
|
|
*/
|
|
public function &GetByName($sHeaderName)
|
|
{
|
|
$oResult = $oHeader = null;
|
|
|
|
$sHeaderNameLower = \strtolower($sHeaderName);
|
|
$aHeaders =& $this->GetAsArray();
|
|
foreach ($aHeaders as /* @var $oHeader \MailSo\Mime\Header */ &$oHeader)
|
|
{
|
|
if ($sHeaderNameLower === \strtolower($oHeader->Name()))
|
|
{
|
|
$oResult =& $oHeader;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return $oResult;
|
|
}
|
|
|
|
/**
|
|
* @param array $aList
|
|
* @return \MailSo\Mime\HeaderCollection
|
|
*
|
|
* @throws \MailSo\Base\Exceptions\InvalidArgumentException
|
|
*/
|
|
public function SetAsArray($aList)
|
|
{
|
|
parent::SetAsArray($aList);
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* @param string $sParentCharset
|
|
* @return \MailSo\Mime\HeaderCollection
|
|
*/
|
|
public function SetParentCharset($sParentCharset)
|
|
{
|
|
if (0 < \strlen($sParentCharset))
|
|
{
|
|
if ($this->sParentCharset !== $sParentCharset)
|
|
{
|
|
$oHeader = null;
|
|
$aHeaders =& $this->GetAsArray();
|
|
|
|
foreach ($aHeaders as /* @var $oHeader \MailSo\Mime\Header */ &$oHeader)
|
|
{
|
|
$oHeader->SetParentCharset($sParentCharset);
|
|
}
|
|
|
|
$this->sParentCharset = $sParentCharset;
|
|
}
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* @return void
|
|
*/
|
|
public function Clear()
|
|
{
|
|
parent::Clear();
|
|
|
|
$this->sRawHeaders = '';
|
|
}
|
|
|
|
/**
|
|
* @param string $sRawHeaders
|
|
* @param bool $bStoreRawHeaders = false
|
|
* @param string $sParentCharset = ''
|
|
*
|
|
* @return \MailSo\Mime\HeaderCollection
|
|
*/
|
|
public function Parse($sRawHeaders, $bStoreRawHeaders = false, $sParentCharset = '')
|
|
{
|
|
$this->Clear();
|
|
|
|
if ($bStoreRawHeaders)
|
|
{
|
|
$this->sRawHeaders = $sRawHeaders;
|
|
}
|
|
|
|
if (0 === \strlen($this->sParentCharset))
|
|
{
|
|
$this->sParentCharset = $sParentCharset;
|
|
}
|
|
|
|
$aHeaders = \explode("\n", \str_replace("\r", '', $sRawHeaders));
|
|
|
|
$sName = null;
|
|
$sValue = null;
|
|
foreach ($aHeaders as $sHeadersValue)
|
|
{
|
|
if (0 === strlen($sHeadersValue))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
$sFirstChar = \substr($sHeadersValue, 0, 1);
|
|
if ($sFirstChar !== ' ' && $sFirstChar !== "\t" && false === \strpos($sHeadersValue, ':'))
|
|
{
|
|
continue;
|
|
}
|
|
else if (null !== $sName && ($sFirstChar === ' ' || $sFirstChar === "\t"))
|
|
{
|
|
$sValue = \is_null($sValue) ? '' : $sValue;
|
|
|
|
if ('?=' === \substr(\rtrim($sHeadersValue), -2))
|
|
{
|
|
$sHeadersValue = \rtrim($sHeadersValue);
|
|
}
|
|
|
|
if ('=?' === \substr(\ltrim($sHeadersValue), 0, 2))
|
|
{
|
|
$sHeadersValue = \ltrim($sHeadersValue);
|
|
}
|
|
|
|
if ('=?' === \substr($sHeadersValue, 0, 2))
|
|
{
|
|
$sValue .= $sHeadersValue;
|
|
}
|
|
else
|
|
{
|
|
$sValue .= "\n".$sHeadersValue;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (null !== $sName)
|
|
{
|
|
$oHeader = Header::NewInstanceFromEncodedString($sName.': '.$sValue, $this->sParentCharset);
|
|
if ($oHeader)
|
|
{
|
|
$this->Add($oHeader);
|
|
}
|
|
|
|
$sName = null;
|
|
$sValue = null;
|
|
}
|
|
|
|
$aHeaderParts = \explode(':', $sHeadersValue, 2);
|
|
$sName = $aHeaderParts[0];
|
|
$sValue = isset($aHeaderParts[1]) ? $aHeaderParts[1] : '';
|
|
|
|
if ('?=' === \substr(\rtrim($sValue), -2))
|
|
{
|
|
$sValue = \rtrim($sValue);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (null !== $sName)
|
|
{
|
|
$oHeader = Header::NewInstanceFromEncodedString($sName.': '.$sValue, $this->sParentCharset);
|
|
if ($oHeader)
|
|
{
|
|
$this->Add($oHeader);
|
|
}
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* @return int
|
|
*/
|
|
public function DkimStatuses()
|
|
{
|
|
$aResult = array();
|
|
|
|
$aHeaders = $this->ValuesByName(\MailSo\Mime\Enumerations\Header::AUTHENTICATION_RESULTS);
|
|
if (\is_array($aHeaders) && 0 < \count($aHeaders))
|
|
{
|
|
foreach ($aHeaders as $sHeaderValue)
|
|
{
|
|
$sStatus = '';
|
|
$sHeader = '';
|
|
$sDkimLine = '';
|
|
|
|
$aMatch = array();
|
|
|
|
$sHeaderValue = \preg_replace('/[\r\n\t\s]+/', ' ', $sHeaderValue);
|
|
|
|
if (\preg_match('/dkim=[^;]+/i', $sHeaderValue, $aMatch) && !empty($aMatch[0]))
|
|
{
|
|
$sDkimLine = $aMatch[0];
|
|
|
|
$aMatch = array();
|
|
if (\preg_match('/dkim=([a-zA-Z0-9]+)/i', $sDkimLine, $aMatch) && !empty($aMatch[1]))
|
|
{
|
|
$sStatus = $aMatch[1];
|
|
}
|
|
|
|
$aMatch = array();
|
|
if (\preg_match('/header\.(d|i|from)=([^\s;]+)/i', $sDkimLine, $aMatch) && !empty($aMatch[2]))
|
|
{
|
|
$sHeader = \trim($aMatch[2]);
|
|
}
|
|
|
|
if (!empty($sStatus) && !empty($sHeader))
|
|
{
|
|
$aResult[] = array($sStatus, $sHeader, $sDkimLine);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// X-DKIM-Authentication-Results: signer="hostinger.com" status="pass"
|
|
$aHeaders = $this->ValuesByName(\MailSo\Mime\Enumerations\Header::X_DKIM_AUTHENTICATION_RESULTS);
|
|
if (\is_array($aHeaders) && 0 < \count($aHeaders))
|
|
{
|
|
foreach ($aHeaders as $sHeaderValue)
|
|
{
|
|
$sStatus = '';
|
|
$sHeader = '';
|
|
|
|
$aMatch = array();
|
|
|
|
$sHeaderValue = \preg_replace('/[\r\n\t\s]+/', ' ', $sHeaderValue);
|
|
|
|
if (\preg_match('/status[\s]?=[\s]?"([a-zA-Z0-9]+)"/i', $sHeaderValue, $aMatch) && !empty($aMatch[1]))
|
|
{
|
|
$sStatus = $aMatch[1];
|
|
}
|
|
|
|
if (\preg_match('/signer[\s]?=[\s]?"([^";]+)"/i', $sHeaderValue, $aMatch) && !empty($aMatch[1]))
|
|
{
|
|
$sHeader = \trim($aMatch[1]);
|
|
}
|
|
|
|
if (!empty($sStatus) && !empty($sHeader))
|
|
{
|
|
$aResult[] = array($sStatus, $sHeader, $sHeaderValue);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return $aResult;
|
|
}
|
|
|
|
/**
|
|
* @return int
|
|
*/
|
|
public function PopulateEmailColectionByDkim($oEmails)
|
|
{
|
|
if ($oEmails && $oEmails instanceof \MailSo\Mime\EmailCollection)
|
|
{
|
|
$aDkimStatuses = $this->DkimStatuses();
|
|
if (\is_array($aDkimStatuses) && 0 < \count($aDkimStatuses))
|
|
{
|
|
$oEmails->ForeachList(function (/* @var $oItem \MailSo\Mime\Email */ $oItem) use ($aDkimStatuses) {
|
|
if ($oItem && $oItem instanceof \MailSo\Mime\Email)
|
|
{
|
|
$sEmail = $oItem->GetEmail();
|
|
foreach ($aDkimStatuses as $aDkimData)
|
|
{
|
|
if (isset($aDkimData[0], $aDkimData[1]) &&
|
|
$aDkimData[1] === \strstr($sEmail, $aDkimData[1]))
|
|
{
|
|
$oItem->SetDkimStatusAndValue($aDkimData[0], empty($aDkimData[2]) ? '' : $aDkimData[2]);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return string
|
|
*/
|
|
public function ToEncodedString()
|
|
{
|
|
$aResult = array();
|
|
$aHeaders =& $this->GetAsArray();
|
|
foreach ($aHeaders as /* @var $oHeader \MailSo\Mime\Header */ &$oHeader)
|
|
{
|
|
$aResult[] = $oHeader->EncodedValue();
|
|
}
|
|
|
|
return \implode(\MailSo\Mime\Enumerations\Constants::CRLF, $aResult);
|
|
}
|
|
}
|