2021-05-21 08:49:41 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace Xentral\Components\MailClient\Data;
|
|
|
|
|
|
|
|
use Xentral\Components\MailClient\Exception\InvalidArgumentException;
|
|
|
|
|
|
|
|
class MailAttachmentData implements MailAttachmentInterface
|
|
|
|
{
|
|
|
|
/** @var string $filename */
|
|
|
|
private $filename;
|
|
|
|
|
|
|
|
/** @var string $content */
|
|
|
|
private $content;
|
|
|
|
|
|
|
|
/** @var string $contentType */
|
|
|
|
private $contentType;
|
|
|
|
|
|
|
|
/** @var string $encoding */
|
|
|
|
private $encoding;
|
|
|
|
|
|
|
|
/** @var bool $isInlineAttachment*/
|
|
|
|
private $isInlineAttachment;
|
|
|
|
|
|
|
|
/** @var string|null $cid */
|
|
|
|
private $cid;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $filename
|
|
|
|
* @param string $content
|
|
|
|
* @param string $contentType
|
|
|
|
* @param string $encoding
|
|
|
|
* @param bool $isInlineAttachment
|
|
|
|
* @param string|null $cid
|
|
|
|
*/
|
|
|
|
public function __construct(
|
|
|
|
string $filename,
|
|
|
|
string $content,
|
|
|
|
string $contentType,
|
|
|
|
string $encoding,
|
|
|
|
bool $isInlineAttachment = false,
|
|
|
|
string $cid = null
|
|
|
|
)
|
|
|
|
{
|
|
|
|
$this->filename = $filename;
|
|
|
|
$this->content = $content;
|
|
|
|
$this->contentType = $contentType;
|
|
|
|
$this->encoding = $encoding;
|
|
|
|
$this->isInlineAttachment = $isInlineAttachment;
|
|
|
|
$this->cid = $cid;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param MailMessagePartInterface $part
|
|
|
|
*
|
|
|
|
* @throws InvalidArgumentException
|
|
|
|
*
|
|
|
|
* @return MailAttachmentData
|
|
|
|
*/
|
|
|
|
public static function fromMailMessagePart(MailMessagePartInterface $part): MailAttachmentData
|
|
|
|
{
|
2023-02-01 11:24:20 +01:00
|
|
|
|
|
|
|
$isInline = false;
|
|
|
|
|
2021-05-21 08:49:41 +02:00
|
|
|
$encodingHeader = $part->getHeader('content-transfer-encoding');
|
|
|
|
if ($encodingHeader === null) {
|
2023-01-12 00:01:20 +01:00
|
|
|
// Assume this is no error (?) throw new InvalidArgumentException('missing header: "Content-Transfer-Encoding"');
|
|
|
|
$encoding = '';
|
|
|
|
} else {
|
|
|
|
$encoding = $encodingHeader->getValue();
|
2021-05-21 08:49:41 +02:00
|
|
|
}
|
|
|
|
$dispositionHeader = $part->getHeader('content-disposition');
|
|
|
|
if ($dispositionHeader === null) {
|
|
|
|
throw new InvalidArgumentException('missing header: "Content-Disposition"');
|
|
|
|
}
|
2023-01-04 12:27:34 +01:00
|
|
|
$disposition = $dispositionHeader->getValue();
|
2022-07-27 18:05:24 +02:00
|
|
|
|
2023-01-04 12:27:34 +01:00
|
|
|
/*
|
|
|
|
Content-Disposition: inline
|
|
|
|
Content-Disposition: attachment
|
|
|
|
Content-Disposition: attachment; filename="filename.jpg"
|
|
|
|
|
|
|
|
This is not correctly implemented -> only the first string is evaluated
|
|
|
|
Content-Disposition: attachment; filename*0="filename_that_is_"
|
|
|
|
Content-Disposition: attachment; filename*1="very_long.jpg"
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (preg_match('/(.+);\s*filename(?:\*[0-9]){0,1}="([^"]+)".*$/m', $disposition, $matches)) {
|
|
|
|
$isInline = strtolower($matches[1]) === 'inline';
|
|
|
|
$filename = $matches[2];
|
|
|
|
}
|
|
|
|
else if ($disposition == 'attachment') {
|
|
|
|
// Filename is given in Content-Type e.g.
|
|
|
|
/* Content-Type: application/pdf; name="Filename.pdf"
|
|
|
|
Content-Transfer-Encoding: base64
|
|
|
|
Content-Disposition: attachment
|
|
|
|
*/
|
|
|
|
|
|
|
|
$contenttypeHeader = $part->getHeader('content-type');
|
|
|
|
if ($contenttypeHeader === null) {
|
|
|
|
throw new InvalidArgumentException('missing header: "Content-Type"');
|
|
|
|
}
|
|
|
|
$contenttype = $contenttypeHeader->getValue();
|
|
|
|
|
|
|
|
if (preg_match('/(.+);\s*name(?:\*[0-9]){0,1}="([^"]+)".*$/m', $contenttype, $matches)) {
|
|
|
|
$isInline = strtolower($matches[1]) === 'inline';
|
|
|
|
$filename = $matches[2];
|
|
|
|
} else {
|
|
|
|
throw new InvalidArgumentException(
|
|
|
|
sprintf('missing filename in header value "Content-Type" = "%s"', $contenttype)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if ($disposition == 'inline') {
|
|
|
|
$isInline = true;
|
2023-02-01 11:24:20 +01:00
|
|
|
$filename = "OpenXE_file.inline";
|
2023-01-12 00:01:20 +01:00
|
|
|
}
|
|
|
|
else if (strpos($disposition,'attachment;\n') == 0) { // No filename, check for content type message/rfc822
|
|
|
|
|
|
|
|
$contenttypeHeader = $part->getHeader('content-type');
|
|
|
|
if ($contenttypeHeader === null) {
|
|
|
|
throw new InvalidArgumentException('missing header: "Content-Type"');
|
|
|
|
}
|
|
|
|
$contenttype = $contenttypeHeader->getValue();
|
|
|
|
|
|
|
|
if ($contenttype == 'message/rfc822') {
|
|
|
|
$filename = 'ForwardedMessage.eml';
|
|
|
|
} else {
|
2023-02-01 11:24:20 +01:00
|
|
|
/* throw new InvalidArgumentException(
|
2023-01-12 00:01:20 +01:00
|
|
|
sprintf('unexpected header value "Content-Disposition" = "%s"', $disposition)
|
2023-02-01 11:24:20 +01:00
|
|
|
);*/
|
|
|
|
$filename = "OpenXE_file.unknown";
|
2023-01-12 00:01:20 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2021-05-21 08:49:41 +02:00
|
|
|
throw new InvalidArgumentException(
|
2023-01-12 00:01:20 +01:00
|
|
|
sprintf('unexpected header value "Content-Disposition" = "%s", not message/rfc822', $disposition)
|
2021-05-21 08:49:41 +02:00
|
|
|
);
|
|
|
|
}
|
2022-07-27 18:05:24 +02:00
|
|
|
|
|
|
|
// Thunderbird UTF URL-Format
|
|
|
|
$UTF_pos = strpos($filename,'UTF-8\'\'');
|
2023-01-04 12:27:34 +01:00
|
|
|
if ($UTF_pos !== false) {
|
2022-07-27 18:05:24 +02:00
|
|
|
$wasUTF = "JA";
|
|
|
|
$filename = substr($filename,$UTF_pos);
|
|
|
|
$filename = rawurldecode($filename);
|
|
|
|
}
|
|
|
|
|
2023-01-04 12:27:34 +01:00
|
|
|
|
2021-05-21 08:49:41 +02:00
|
|
|
$cid = null;
|
|
|
|
$contentIdHeader = $part->getHeader('content-id');
|
|
|
|
if ($contentIdHeader !== null) {
|
|
|
|
$cid = $contentIdHeader->getValue();
|
|
|
|
if (preg_match('/[<]?([^<>]+)[>]?$/', $cid, $cidMatches)) {
|
|
|
|
$cid = $cidMatches[1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-01 11:24:20 +01:00
|
|
|
$content = $part->getContent();
|
|
|
|
if ($content === null) { // This should not be
|
|
|
|
throw new InvalidArgumentException(
|
2023-02-01 19:02:16 +01:00
|
|
|
sprintf('content is null "%s"', substr(print_r($part,true),0,1000))
|
2023-02-01 11:24:20 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-05-21 08:49:41 +02:00
|
|
|
return new self(
|
|
|
|
$filename,
|
2023-02-01 11:24:20 +01:00
|
|
|
$content,
|
2021-05-21 08:49:41 +02:00
|
|
|
$part->getContentType(),
|
|
|
|
$encoding,
|
|
|
|
$isInline,
|
|
|
|
$cid
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getFileName(): string
|
|
|
|
{
|
|
|
|
return $this->filename;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getContent(): string
|
|
|
|
{
|
|
|
|
switch ($this->encoding) {
|
|
|
|
case 'base64':
|
|
|
|
return base64_decode($this->content);
|
|
|
|
|
|
|
|
default:
|
|
|
|
return $this->content;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getContentType(): string
|
|
|
|
{
|
|
|
|
return $this->contentType;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getTransferEncoding(): string
|
|
|
|
{
|
|
|
|
return $this->encoding;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function isInlineAttachment(): bool
|
|
|
|
{
|
|
|
|
return $this->isInlineAttachment;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string|null
|
|
|
|
*/
|
|
|
|
public function getCid(): ?string
|
|
|
|
{
|
|
|
|
return $this->cid;
|
|
|
|
}
|
|
|
|
}
|