first commit
This commit is contained in:
145
vendor/symfony/mailer/Transport/Smtp/Stream/AbstractStream.php
vendored
Normal file
145
vendor/symfony/mailer/Transport/Smtp/Stream/AbstractStream.php
vendored
Normal file
@ -0,0 +1,145 @@
|
||||
<?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\Mailer\Transport\Smtp\Stream;
|
||||
|
||||
use Symfony\Component\Mailer\Exception\TransportException;
|
||||
|
||||
/**
|
||||
* A stream supporting remote sockets and local processes.
|
||||
*
|
||||
* @author Fabien Potencier <fabien@symfony.com>
|
||||
* @author Nicolas Grekas <p@tchwork.com>
|
||||
* @author Chris Corbyn
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
abstract class AbstractStream
|
||||
{
|
||||
/** @var resource|null */
|
||||
protected $stream;
|
||||
/** @var resource|null */
|
||||
protected $in;
|
||||
/** @var resource|null */
|
||||
protected $out;
|
||||
protected $err;
|
||||
|
||||
private string $debug = '';
|
||||
|
||||
public function write(string $bytes, bool $debug = true): void
|
||||
{
|
||||
if ($debug) {
|
||||
foreach (explode("\n", trim($bytes)) as $line) {
|
||||
$this->debug .= sprintf("> %s\n", $line);
|
||||
}
|
||||
}
|
||||
|
||||
$bytesToWrite = \strlen($bytes);
|
||||
$totalBytesWritten = 0;
|
||||
while ($totalBytesWritten < $bytesToWrite) {
|
||||
$bytesWritten = @fwrite($this->in, substr($bytes, $totalBytesWritten));
|
||||
if (false === $bytesWritten || 0 === $bytesWritten) {
|
||||
throw new TransportException('Unable to write bytes on the wire.');
|
||||
}
|
||||
|
||||
$totalBytesWritten += $bytesWritten;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Flushes the contents of the stream (empty it) and set the internal pointer to the beginning.
|
||||
*/
|
||||
public function flush(): void
|
||||
{
|
||||
fflush($this->in);
|
||||
}
|
||||
|
||||
/**
|
||||
* Performs any initialization needed.
|
||||
*/
|
||||
abstract public function initialize(): void;
|
||||
|
||||
public function terminate(): void
|
||||
{
|
||||
$this->stream = $this->err = $this->out = $this->in = null;
|
||||
}
|
||||
|
||||
public function readLine(): string
|
||||
{
|
||||
if (feof($this->out)) {
|
||||
return '';
|
||||
}
|
||||
|
||||
$line = @fgets($this->out);
|
||||
if ('' === $line || false === $line) {
|
||||
$metas = stream_get_meta_data($this->out);
|
||||
if ($metas['timed_out']) {
|
||||
throw new TransportException(sprintf('Connection to "%s" timed out.', $this->getReadConnectionDescription()));
|
||||
}
|
||||
if ($metas['eof']) {
|
||||
throw new TransportException(sprintf('Connection to "%s" has been closed unexpectedly.', $this->getReadConnectionDescription()));
|
||||
}
|
||||
if (false === $line) {
|
||||
throw new TransportException(sprintf('Unable to read from connection to "%s": ', $this->getReadConnectionDescription()).error_get_last()['message']);
|
||||
}
|
||||
}
|
||||
|
||||
$this->debug .= sprintf('< %s', $line);
|
||||
|
||||
return $line;
|
||||
}
|
||||
|
||||
public function getDebug(): string
|
||||
{
|
||||
$debug = $this->debug;
|
||||
$this->debug = '';
|
||||
|
||||
return $debug;
|
||||
}
|
||||
|
||||
public static function replace(string $from, string $to, iterable $chunks): \Generator
|
||||
{
|
||||
if ('' === $from) {
|
||||
yield from $chunks;
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
$carry = '';
|
||||
$fromLen = \strlen($from);
|
||||
|
||||
foreach ($chunks as $chunk) {
|
||||
if ('' === $chunk = $carry.$chunk) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if (str_contains($chunk, $from)) {
|
||||
$chunk = explode($from, $chunk);
|
||||
$carry = array_pop($chunk);
|
||||
|
||||
yield implode($to, $chunk).$to;
|
||||
} else {
|
||||
$carry = $chunk;
|
||||
}
|
||||
|
||||
if (\strlen($carry) > $fromLen) {
|
||||
yield substr($carry, 0, -$fromLen);
|
||||
$carry = substr($carry, -$fromLen);
|
||||
}
|
||||
}
|
||||
|
||||
if ('' !== $carry) {
|
||||
yield $carry;
|
||||
}
|
||||
}
|
||||
|
||||
abstract protected function getReadConnectionDescription(): string;
|
||||
}
|
71
vendor/symfony/mailer/Transport/Smtp/Stream/ProcessStream.php
vendored
Normal file
71
vendor/symfony/mailer/Transport/Smtp/Stream/ProcessStream.php
vendored
Normal file
@ -0,0 +1,71 @@
|
||||
<?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\Mailer\Transport\Smtp\Stream;
|
||||
|
||||
use Symfony\Component\Mailer\Exception\TransportException;
|
||||
|
||||
/**
|
||||
* A stream supporting local processes.
|
||||
*
|
||||
* @author Fabien Potencier <fabien@symfony.com>
|
||||
* @author Chris Corbyn
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
final class ProcessStream extends AbstractStream
|
||||
{
|
||||
private string $command;
|
||||
|
||||
public function setCommand(string $command): void
|
||||
{
|
||||
$this->command = $command;
|
||||
}
|
||||
|
||||
public function initialize(): void
|
||||
{
|
||||
$descriptorSpec = [
|
||||
0 => ['pipe', 'r'],
|
||||
1 => ['pipe', 'w'],
|
||||
2 => ['pipe', '\\' === \DIRECTORY_SEPARATOR ? 'a' : 'w'],
|
||||
];
|
||||
$pipes = [];
|
||||
$this->stream = proc_open($this->command, $descriptorSpec, $pipes);
|
||||
stream_set_blocking($pipes[2], false);
|
||||
if ($err = stream_get_contents($pipes[2])) {
|
||||
throw new TransportException('Process could not be started: '.$err);
|
||||
}
|
||||
$this->in = &$pipes[0];
|
||||
$this->out = &$pipes[1];
|
||||
$this->err = &$pipes[2];
|
||||
}
|
||||
|
||||
public function terminate(): void
|
||||
{
|
||||
if (null !== $this->stream) {
|
||||
fclose($this->in);
|
||||
$out = stream_get_contents($this->out);
|
||||
fclose($this->out);
|
||||
$err = stream_get_contents($this->err);
|
||||
fclose($this->err);
|
||||
if (0 !== $exitCode = proc_close($this->stream)) {
|
||||
throw new TransportException('Process failed with exit code '.$exitCode.': '.$out.$err);
|
||||
}
|
||||
}
|
||||
|
||||
parent::terminate();
|
||||
}
|
||||
|
||||
protected function getReadConnectionDescription(): string
|
||||
{
|
||||
return 'process '.$this->command;
|
||||
}
|
||||
}
|
193
vendor/symfony/mailer/Transport/Smtp/Stream/SocketStream.php
vendored
Normal file
193
vendor/symfony/mailer/Transport/Smtp/Stream/SocketStream.php
vendored
Normal file
@ -0,0 +1,193 @@
|
||||
<?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\Mailer\Transport\Smtp\Stream;
|
||||
|
||||
use Symfony\Component\Mailer\Exception\TransportException;
|
||||
|
||||
/**
|
||||
* A stream supporting remote sockets.
|
||||
*
|
||||
* @author Fabien Potencier <fabien@symfony.com>
|
||||
* @author Chris Corbyn
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
final class SocketStream extends AbstractStream
|
||||
{
|
||||
private string $url;
|
||||
private string $host = 'localhost';
|
||||
private int $port = 465;
|
||||
private float $timeout;
|
||||
private bool $tls = true;
|
||||
private ?string $sourceIp = null;
|
||||
private array $streamContextOptions = [];
|
||||
|
||||
/**
|
||||
* @return $this
|
||||
*/
|
||||
public function setTimeout(float $timeout): static
|
||||
{
|
||||
$this->timeout = $timeout;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getTimeout(): float
|
||||
{
|
||||
return $this->timeout ?? (float) \ini_get('default_socket_timeout');
|
||||
}
|
||||
|
||||
/**
|
||||
* Literal IPv6 addresses should be wrapped in square brackets.
|
||||
*
|
||||
* @return $this
|
||||
*/
|
||||
public function setHost(string $host): static
|
||||
{
|
||||
$this->host = $host;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getHost(): string
|
||||
{
|
||||
return $this->host;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return $this
|
||||
*/
|
||||
public function setPort(int $port): static
|
||||
{
|
||||
$this->port = $port;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getPort(): int
|
||||
{
|
||||
return $this->port;
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets the TLS/SSL on the socket (disables STARTTLS).
|
||||
*
|
||||
* @return $this
|
||||
*/
|
||||
public function disableTls(): static
|
||||
{
|
||||
$this->tls = false;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function isTLS(): bool
|
||||
{
|
||||
return $this->tls;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return $this
|
||||
*/
|
||||
public function setStreamOptions(array $options): static
|
||||
{
|
||||
$this->streamContextOptions = $options;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getStreamOptions(): array
|
||||
{
|
||||
return $this->streamContextOptions;
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets the source IP.
|
||||
*
|
||||
* IPv6 addresses should be wrapped in square brackets.
|
||||
*
|
||||
* @return $this
|
||||
*/
|
||||
public function setSourceIp(string $ip): static
|
||||
{
|
||||
$this->sourceIp = $ip;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the IP used to connect to the destination.
|
||||
*/
|
||||
public function getSourceIp(): ?string
|
||||
{
|
||||
return $this->sourceIp;
|
||||
}
|
||||
|
||||
public function initialize(): void
|
||||
{
|
||||
$this->url = $this->host.':'.$this->port;
|
||||
if ($this->tls) {
|
||||
$this->url = 'ssl://'.$this->url;
|
||||
}
|
||||
$options = [];
|
||||
if ($this->sourceIp) {
|
||||
$options['socket']['bindto'] = $this->sourceIp.':0';
|
||||
}
|
||||
if ($this->streamContextOptions) {
|
||||
$options = array_merge($options, $this->streamContextOptions);
|
||||
}
|
||||
// do it unconditionally as it will be used by STARTTLS as well if supported
|
||||
$options['ssl']['crypto_method'] ??= \STREAM_CRYPTO_METHOD_TLS_CLIENT | \STREAM_CRYPTO_METHOD_TLSv1_2_CLIENT | \STREAM_CRYPTO_METHOD_TLSv1_1_CLIENT;
|
||||
$streamContext = stream_context_create($options);
|
||||
|
||||
$timeout = $this->getTimeout();
|
||||
set_error_handler(function ($type, $msg) {
|
||||
throw new TransportException(sprintf('Connection could not be established with host "%s": ', $this->url).$msg);
|
||||
});
|
||||
try {
|
||||
$this->stream = stream_socket_client($this->url, $errno, $errstr, $timeout, \STREAM_CLIENT_CONNECT, $streamContext);
|
||||
} finally {
|
||||
restore_error_handler();
|
||||
}
|
||||
|
||||
stream_set_blocking($this->stream, true);
|
||||
stream_set_timeout($this->stream, (int) $timeout, (int) (($timeout - (int) $timeout) * 1000000));
|
||||
$this->in = &$this->stream;
|
||||
$this->out = &$this->stream;
|
||||
}
|
||||
|
||||
public function startTLS(): bool
|
||||
{
|
||||
set_error_handler(function ($type, $msg) {
|
||||
throw new TransportException('Unable to connect with STARTTLS: '.$msg);
|
||||
});
|
||||
try {
|
||||
return stream_socket_enable_crypto($this->stream, true);
|
||||
} finally {
|
||||
restore_error_handler();
|
||||
}
|
||||
}
|
||||
|
||||
public function terminate(): void
|
||||
{
|
||||
if (null !== $this->stream) {
|
||||
fclose($this->stream);
|
||||
}
|
||||
|
||||
parent::terminate();
|
||||
}
|
||||
|
||||
protected function getReadConnectionDescription(): string
|
||||
{
|
||||
return $this->url;
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user