205 lines
5.8 KiB
PHP
205 lines
5.8 KiB
PHP
|
<?php
|
||
|
|
||
|
/*
|
||
|
* This file is part of the Symfony package.
|
||
|
*
|
||
|
* (c) Fabien Potencier <fabien@symfony.com>
|
||
|
*
|
||
|
* For the full copyright and license information, please view the LICENSE
|
||
|
* file that was distributed with this source code.
|
||
|
*/
|
||
|
|
||
|
namespace Symfony\Component\Mime\Part;
|
||
|
|
||
|
use Symfony\Component\Mime\Encoder\Base64ContentEncoder;
|
||
|
use Symfony\Component\Mime\Encoder\ContentEncoderInterface;
|
||
|
use Symfony\Component\Mime\Encoder\EightBitContentEncoder;
|
||
|
use Symfony\Component\Mime\Encoder\QpContentEncoder;
|
||
|
use Symfony\Component\Mime\Exception\InvalidArgumentException;
|
||
|
use Symfony\Component\Mime\Header\Headers;
|
||
|
|
||
|
/**
|
||
|
* @author Fabien Potencier <fabien@symfony.com>
|
||
|
*/
|
||
|
class TextPart extends AbstractPart
|
||
|
{
|
||
|
private static $encoders = [];
|
||
|
|
||
|
private $body;
|
||
|
private $charset;
|
||
|
private $subtype;
|
||
|
private $disposition;
|
||
|
private $name;
|
||
|
private $encoding;
|
||
|
|
||
|
/**
|
||
|
* @param resource|string $body
|
||
|
*/
|
||
|
public function __construct($body, ?string $charset = 'utf-8', string $subtype = 'plain', string $encoding = null)
|
||
|
{
|
||
|
parent::__construct();
|
||
|
|
||
|
if (!\is_string($body) && !\is_resource($body)) {
|
||
|
throw new \TypeError(sprintf('The body of "%s" must be a string or a resource (got "%s").', self::class, \is_object($body) ? \get_class($body) : \gettype($body)));
|
||
|
}
|
||
|
|
||
|
$this->body = $body;
|
||
|
$this->charset = $charset;
|
||
|
$this->subtype = $subtype;
|
||
|
|
||
|
if (null === $encoding) {
|
||
|
$this->encoding = $this->chooseEncoding();
|
||
|
} else {
|
||
|
if ('quoted-printable' !== $encoding && 'base64' !== $encoding && '8bit' !== $encoding) {
|
||
|
throw new InvalidArgumentException(sprintf('The encoding must be one of "quoted-printable", "base64", or "8bit" ("%s" given).', $encoding));
|
||
|
}
|
||
|
$this->encoding = $encoding;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public function getMediaType(): string
|
||
|
{
|
||
|
return 'text';
|
||
|
}
|
||
|
|
||
|
public function getMediaSubtype(): string
|
||
|
{
|
||
|
return $this->subtype;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param string $disposition one of attachment, inline, or form-data
|
||
|
*
|
||
|
* @return $this
|
||
|
*/
|
||
|
public function setDisposition(string $disposition)
|
||
|
{
|
||
|
$this->disposition = $disposition;
|
||
|
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Sets the name of the file (used by FormDataPart).
|
||
|
*
|
||
|
* @return $this
|
||
|
*/
|
||
|
public function setName($name)
|
||
|
{
|
||
|
$this->name = $name;
|
||
|
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
public function getBody(): string
|
||
|
{
|
||
|
if (!\is_resource($this->body)) {
|
||
|
return $this->body;
|
||
|
}
|
||
|
|
||
|
if (stream_get_meta_data($this->body)['seekable'] ?? false) {
|
||
|
rewind($this->body);
|
||
|
}
|
||
|
|
||
|
return stream_get_contents($this->body) ?: '';
|
||
|
}
|
||
|
|
||
|
public function bodyToString(): string
|
||
|
{
|
||
|
return $this->getEncoder()->encodeString($this->getBody(), $this->charset);
|
||
|
}
|
||
|
|
||
|
public function bodyToIterable(): iterable
|
||
|
{
|
||
|
if (\is_resource($this->body)) {
|
||
|
if (stream_get_meta_data($this->body)['seekable'] ?? false) {
|
||
|
rewind($this->body);
|
||
|
}
|
||
|
yield from $this->getEncoder()->encodeByteStream($this->body);
|
||
|
} else {
|
||
|
yield $this->getEncoder()->encodeString($this->body);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public function getPreparedHeaders(): Headers
|
||
|
{
|
||
|
$headers = parent::getPreparedHeaders();
|
||
|
|
||
|
$headers->setHeaderBody('Parameterized', 'Content-Type', $this->getMediaType().'/'.$this->getMediaSubtype());
|
||
|
if ($this->charset) {
|
||
|
$headers->setHeaderParameter('Content-Type', 'charset', $this->charset);
|
||
|
}
|
||
|
if ($this->name && 'form-data' !== $this->disposition) {
|
||
|
$headers->setHeaderParameter('Content-Type', 'name', $this->name);
|
||
|
}
|
||
|
$headers->setHeaderBody('Text', 'Content-Transfer-Encoding', $this->encoding);
|
||
|
|
||
|
if (!$headers->has('Content-Disposition') && null !== $this->disposition) {
|
||
|
$headers->setHeaderBody('Parameterized', 'Content-Disposition', $this->disposition);
|
||
|
if ($this->name) {
|
||
|
$headers->setHeaderParameter('Content-Disposition', 'name', $this->name);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $headers;
|
||
|
}
|
||
|
|
||
|
public function asDebugString(): string
|
||
|
{
|
||
|
$str = parent::asDebugString();
|
||
|
if (null !== $this->charset) {
|
||
|
$str .= ' charset: '.$this->charset;
|
||
|
}
|
||
|
if (null !== $this->disposition) {
|
||
|
$str .= ' disposition: '.$this->disposition;
|
||
|
}
|
||
|
|
||
|
return $str;
|
||
|
}
|
||
|
|
||
|
private function getEncoder(): ContentEncoderInterface
|
||
|
{
|
||
|
if ('8bit' === $this->encoding) {
|
||
|
return self::$encoders[$this->encoding] ?? (self::$encoders[$this->encoding] = new EightBitContentEncoder());
|
||
|
}
|
||
|
|
||
|
if ('quoted-printable' === $this->encoding) {
|
||
|
return self::$encoders[$this->encoding] ?? (self::$encoders[$this->encoding] = new QpContentEncoder());
|
||
|
}
|
||
|
|
||
|
return self::$encoders[$this->encoding] ?? (self::$encoders[$this->encoding] = new Base64ContentEncoder());
|
||
|
}
|
||
|
|
||
|
private function chooseEncoding(): string
|
||
|
{
|
||
|
if (null === $this->charset) {
|
||
|
return 'base64';
|
||
|
}
|
||
|
|
||
|
return 'quoted-printable';
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @return array
|
||
|
*/
|
||
|
public function __sleep()
|
||
|
{
|
||
|
// convert resources to strings for serialization
|
||
|
if (\is_resource($this->body)) {
|
||
|
$this->body = $this->getBody();
|
||
|
}
|
||
|
|
||
|
$this->_headers = $this->getHeaders();
|
||
|
|
||
|
return ['_headers', 'body', 'charset', 'subtype', 'disposition', 'name', 'encoding'];
|
||
|
}
|
||
|
|
||
|
public function __wakeup()
|
||
|
{
|
||
|
$r = new \ReflectionProperty(AbstractPart::class, 'headers');
|
||
|
$r->setAccessible(true);
|
||
|
$r->setValue($this, $this->_headers);
|
||
|
unset($this->_headers);
|
||
|
}
|
||
|
}
|