Вход Регистрация
Файл: vendor/laravel/framework/src/Illuminate/Mail/Mailer.php
Строк: 784
<?php

namespace IlluminateMail;

use 
Closure;
use 
IlluminateContractsEventsDispatcher;
use 
IlluminateContractsMailMailable as MailableContract;
use 
IlluminateContractsMailMailer as MailerContract;
use 
IlluminateContractsMailMailQueue as MailQueueContract;
use 
IlluminateContractsQueueFactory as QueueContract;
use 
IlluminateContractsQueueShouldQueue;
use 
IlluminateContractsSupportHtmlable;
use 
IlluminateContractsViewFactory;
use 
IlluminateMailEventsMessageSending;
use 
IlluminateMailEventsMessageSent;
use 
IlluminateMailMailablesAddress;
use 
IlluminateSupportHtmlString;
use 
IlluminateSupportTraitsMacroable;
use 
InvalidArgumentException;
use 
SymfonyComponentMailerEnvelope;
use 
SymfonyComponentMailerTransportTransportInterface;
use 
SymfonyComponentMimeEmail;

class 
Mailer implements MailerContractMailQueueContract
{
    use 
Macroable;

    
/**
     * The name that is configured for the mailer.
     *
     * @var string
     */
    
protected $name;

    
/**
     * The view factory instance.
     *
     * @var IlluminateContractsViewFactory
     */
    
protected $views;

    
/**
     * The Symfony Transport instance.
     *
     * @var SymfonyComponentMailerTransportTransportInterface
     */
    
protected $transport;

    
/**
     * The event dispatcher instance.
     *
     * @var IlluminateContractsEventsDispatcher|null
     */
    
protected $events;

    
/**
     * The global from address and name.
     *
     * @var array
     */
    
protected $from;

    
/**
     * The global reply-to address and name.
     *
     * @var array
     */
    
protected $replyTo;

    
/**
     * The global return path address.
     *
     * @var array
     */
    
protected $returnPath;

    
/**
     * The global to address and name.
     *
     * @var array
     */
    
protected $to;

    
/**
     * The queue factory implementation.
     *
     * @var IlluminateContractsQueueFactory
     */
    
protected $queue;

    
/**
     * Create a new Mailer instance.
     *
     * @param  string  $name
     * @param  IlluminateContractsViewFactory  $views
     * @param  SymfonyComponentMailerTransportTransportInterface  $transport
     * @param  IlluminateContractsEventsDispatcher|null  $events
     * @return void
     */
    
public function __construct(string $nameFactory $viewsTransportInterface $transport, ?Dispatcher $events null)
    {
        
$this->name $name;
        
$this->views $views;
        
$this->events $events;
        
$this->transport $transport;
    }

    
/**
     * Set the global from address and name.
     *
     * @param  string  $address
     * @param  string|null  $name
     * @return void
     */
    
public function alwaysFrom($address$name null)
    {
        
$this->from compact('address''name');
    }

    
/**
     * Set the global reply-to address and name.
     *
     * @param  string  $address
     * @param  string|null  $name
     * @return void
     */
    
public function alwaysReplyTo($address$name null)
    {
        
$this->replyTo compact('address''name');
    }

    
/**
     * Set the global return path address.
     *
     * @param  string  $address
     * @return void
     */
    
public function alwaysReturnPath($address)
    {
        
$this->returnPath compact('address');
    }

    
/**
     * Set the global to address and name.
     *
     * @param  string  $address
     * @param  string|null  $name
     * @return void
     */
    
public function alwaysTo($address$name null)
    {
        
$this->to compact('address''name');
    }

    
/**
     * Begin the process of mailing a mailable class instance.
     *
     * @param  mixed  $users
     * @param  string|null  $name
     * @return IlluminateMailPendingMail
     */
    
public function to($users$name null)
    {
        if (! 
is_null($name) && is_string($users)) {
            
$users = new Address($users$name);
        }

        return (new 
PendingMail($this))->to($users);
    }

    
/**
     * Begin the process of mailing a mailable class instance.
     *
     * @param  mixed  $users
     * @param  string|null  $name
     * @return IlluminateMailPendingMail
     */
    
public function cc($users$name null)
    {
        if (! 
is_null($name) && is_string($users)) {
            
$users = new Address($users$name);
        }

        return (new 
PendingMail($this))->cc($users);
    }

    
/**
     * Begin the process of mailing a mailable class instance.
     *
     * @param  mixed  $users
     * @param  string|null  $name
     * @return IlluminateMailPendingMail
     */
    
public function bcc($users$name null)
    {
        if (! 
is_null($name) && is_string($users)) {
            
$users = new Address($users$name);
        }

        return (new 
PendingMail($this))->bcc($users);
    }

    
/**
     * Send a new message with only an HTML part.
     *
     * @param  string  $html
     * @param  mixed  $callback
     * @return IlluminateMailSentMessage|null
     */
    
public function html($html$callback)
    {
        return 
$this->send(['html' => new HtmlString($html)], [], $callback);
    }

    
/**
     * Send a new message with only a raw text part.
     *
     * @param  string  $text
     * @param  mixed  $callback
     * @return IlluminateMailSentMessage|null
     */
    
public function raw($text$callback)
    {
        return 
$this->send(['raw' => $text], [], $callback);
    }

    
/**
     * Send a new message with only a plain part.
     *
     * @param  string  $view
     * @param  array  $data
     * @param  mixed  $callback
     * @return IlluminateMailSentMessage|null
     */
    
public function plain($view, array $data$callback)
    {
        return 
$this->send(['text' => $view], $data$callback);
    }

    
/**
     * Render the given message as a view.
     *
     * @param  string|array  $view
     * @param  array  $data
     * @return string
     */
    
public function render($view, array $data = [])
    {
        
// First we need to parse the view, which could either be a string or an array
        // containing both an HTML and plain text versions of the view which should
        // be used when sending an e-mail. We will extract both of them out here.
        
[$view$plain$raw] = $this->parseView($view);

        
$data['message'] = $this->createMessage();

        return 
$this->replaceEmbeddedAttachments(
            
$this->renderView($view ?: $plain$data),
            
$data['message']->getSymfonyMessage()->getAttachments()
        );
    }

    
/**
     * Replace the embedded image attachments with raw, inline image data for browser rendering.
     *
     * @param  string  $renderedView
     * @param  array  $attachments
     * @return string
     */
    
protected function replaceEmbeddedAttachments(string $renderedView, array $attachments)
    {
        if (
preg_match_all('/<img.+?src=['"]cid:([^'"]+)['"].*?>/i'$renderedView$matches)) {
            foreach (
array_unique($matches[1]) as $image) {
                foreach (
$attachments as $attachment) {
                    if (
$attachment->getFilename() === $image) {
                        
$renderedView str_replace(
                            
'cid:'.$image,
                            
'data:'.$attachment->getContentType().';base64,'.$attachment->bodyToString(),
                            
$renderedView
                        
);

                        break;
                    }
                }
            }
        }

        return 
$renderedView;
    }

    
/**
     * Send a new message using a view.
     *
     * @param  IlluminateContractsMailMailable|string|array  $view
     * @param  array  $data
     * @param  Closure|string|null  $callback
     * @return IlluminateMailSentMessage|null
     */
    
public function send($view, array $data = [], $callback null)
    {
        if (
$view instanceof MailableContract) {
            return 
$this->sendMailable($view);
        }

        
$data['mailer'] = $this->name;

        
// First we need to parse the view, which could either be a string or an array
        // containing both an HTML and plain text versions of the view which should
        // be used when sending an e-mail. We will extract both of them out here.
        
[$view$plain$raw] = $this->parseView($view);

        
$data['message'] = $message $this->createMessage();

        
// Once we have retrieved the view content for the e-mail we will set the body
        // of this message using the HTML type, which will provide a simple wrapper
        // to creating view based emails that are able to receive arrays of data.
        
if (! is_null($callback)) {
            
$callback($message);
        }

        
$this->addContent($message$view$plain$raw$data);

        
// If a global "to" address has been set, we will set that address on the mail
        // message. This is primarily useful during local development in which each
        // message should be delivered into a single mail address for inspection.
        
if (isset($this->to['address'])) {
            
$this->setGlobalToAndRemoveCcAndBcc($message);
        }

        
// Next we will determine if the message should be sent. We give the developer
        // one final chance to stop this message and then we will send it to all of
        // its recipients. We will then fire the sent event for the sent message.
        
$symfonyMessage $message->getSymfonyMessage();

        if (
$this->shouldSendMessage($symfonyMessage$data)) {
            
$symfonySentMessage $this->sendSymfonyMessage($symfonyMessage);

            if (
$symfonySentMessage) {
                
$sentMessage = new SentMessage($symfonySentMessage);

                
$this->dispatchSentEvent($sentMessage$data);

                return 
$sentMessage;
            }
        }
    }

    
/**
     * Send the given mailable.
     *
     * @param  IlluminateContractsMailMailable  $mailable
     * @return IlluminateMailSentMessage|null
     */
    
protected function sendMailable(MailableContract $mailable)
    {
        return 
$mailable instanceof ShouldQueue
                        
$mailable->mailer($this->name)->queue($this->queue)
                        : 
$mailable->mailer($this->name)->send($this);
    }

    
/**
     * Parse the given view name or array.
     *
     * @param  Closure|array|string  $view
     * @return array
     *
     * @throws InvalidArgumentException
     */
    
protected function parseView($view)
    {
        if (
is_string($view) || $view instanceof Closure) {
            return [
$viewnullnull];
        }

        
// If the given view is an array with numeric keys, we will just assume that
        // both a "pretty" and "plain" view were provided, so we will return this
        // array as is, since it should contain both views with numerical keys.
        
if (is_array($view) && isset($view[0])) {
            return [
$view[0], $view[1], null];
        }

        
// If this view is an array but doesn't contain numeric keys, we will assume
        // the views are being explicitly specified and will extract them via the
        // named keys instead, allowing the developers to use one or the other.
        
if (is_array($view)) {
            return [
                
$view['html'] ?? null,
                
$view['text'] ?? null,
                
$view['raw'] ?? null,
            ];
        }

        throw new 
InvalidArgumentException('Invalid view.');
    }

    
/**
     * Add the content to a given message.
     *
     * @param  IlluminateMailMessage  $message
     * @param  string  $view
     * @param  string  $plain
     * @param  string  $raw
     * @param  array  $data
     * @return void
     */
    
protected function addContent($message$view$plain$raw$data)
    {
        if (isset(
$view)) {
            
$message->html($this->renderView($view$data) ?: ' ');
        }

        if (isset(
$plain)) {
            
$message->text($this->renderView($plain$data) ?: ' ');
        }

        if (isset(
$raw)) {
            
$message->text($raw);
        }
    }

    
/**
     * Render the given view.
     *
     * @param  Closure|string  $view
     * @param  array  $data
     * @return string
     */
    
protected function renderView($view$data)
    {
        
$view value($view$data);

        return 
$view instanceof Htmlable
                        
$view->toHtml()
                        : 
$this->views->make($view$data)->render();
    }

    
/**
     * Set the global "to" address on the given message.
     *
     * @param  IlluminateMailMessage  $message
     * @return void
     */
    
protected function setGlobalToAndRemoveCcAndBcc($message)
    {
        
$message->forgetTo();

        
$message->to($this->to['address'], $this->to['name'], true);

        
$message->forgetCc();
        
$message->forgetBcc();
    }

    
/**
     * Queue a new e-mail message for sending.
     *
     * @param  IlluminateContractsMailMailable|string|array  $view
     * @param  string|null  $queue
     * @return mixed
     *
     * @throws InvalidArgumentException
     */
    
public function queue($view$queue null)
    {
        if (! 
$view instanceof MailableContract) {
            throw new 
InvalidArgumentException('Only mailables may be queued.');
        }

        if (
is_string($queue)) {
            
$view->onQueue($queue);
        }

        return 
$view->mailer($this->name)->queue($this->queue);
    }

    
/**
     * Queue a new e-mail message for sending on the given queue.
     *
     * @param  string  $queue
     * @param  IlluminateContractsMailMailable  $view
     * @return mixed
     */
    
public function onQueue($queue$view)
    {
        return 
$this->queue($view$queue);
    }

    
/**
     * Queue a new e-mail message for sending on the given queue.
     *
     * This method didn't match rest of framework's "onQueue" phrasing. Added "onQueue".
     *
     * @param  string  $queue
     * @param  IlluminateContractsMailMailable  $view
     * @return mixed
     */
    
public function queueOn($queue$view)
    {
        return 
$this->onQueue($queue$view);
    }

    
/**
     * Queue a new e-mail message for sending after (n) seconds.
     *
     * @param  DateTimeInterface|DateInterval|int  $delay
     * @param  IlluminateContractsMailMailable  $view
     * @param  string|null  $queue
     * @return mixed
     *
     * @throws InvalidArgumentException
     */
    
public function later($delay$view$queue null)
    {
        if (! 
$view instanceof MailableContract) {
            throw new 
InvalidArgumentException('Only mailables may be queued.');
        }

        return 
$view->mailer($this->name)->later(
            
$delayis_null($queue) ? $this->queue $queue
        
);
    }

    
/**
     * Queue a new e-mail message for sending after (n) seconds on the given queue.
     *
     * @param  string  $queue
     * @param  DateTimeInterface|DateInterval|int  $delay
     * @param  IlluminateContractsMailMailable  $view
     * @return mixed
     */
    
public function laterOn($queue$delay$view)
    {
        return 
$this->later($delay$view$queue);
    }

    
/**
     * Create a new message instance.
     *
     * @return IlluminateMailMessage
     */
    
protected function createMessage()
    {
        
$message = new Message(new Email());

        
// If a global from address has been specified we will set it on every message
        // instance so the developer does not have to repeat themselves every time
        // they create a new message. We'll just go ahead and push this address.
        
if (! empty($this->from['address'])) {
            
$message->from($this->from['address'], $this->from['name']);
        }

        
// When a global reply address was specified we will set this on every message
        // instance so the developer does not have to repeat themselves every time
        // they create a new message. We will just go ahead and push this address.
        
if (! empty($this->replyTo['address'])) {
            
$message->replyTo($this->replyTo['address'], $this->replyTo['name']);
        }

        if (! empty(
$this->returnPath['address'])) {
            
$message->returnPath($this->returnPath['address']);
        }

        return 
$message;
    }

    
/**
     * Send a Symfony Email instance.
     *
     * @param  SymfonyComponentMimeEmail  $message
     * @return SymfonyComponentMailerSentMessage|null
     */
    
protected function sendSymfonyMessage(Email $message)
    {
        try {
            return 
$this->transport->send($messageEnvelope::create($message));
        } finally {
            
//
        
}
    }

    
/**
     * Determines if the email can be sent.
     *
     * @param  SymfonyComponentMimeEmail  $message
     * @param  array  $data
     * @return bool
     */
    
protected function shouldSendMessage($message$data = [])
    {
        if (! 
$this->events) {
            return 
true;
        }

        return 
$this->events->until(
            new 
MessageSending($message$data)
        ) !== 
false;
    }

    
/**
     * Dispatch the message sent event.
     *
     * @param  IlluminateMailSentMessage  $message
     * @param  array  $data
     * @return void
     */
    
protected function dispatchSentEvent($message$data = [])
    {
        if (
$this->events) {
            
$this->events->dispatch(
                new 
MessageSent($message$data)
            );
        }
    }

    
/**
     * Get the Symfony Transport instance.
     *
     * @return SymfonyComponentMailerTransportTransportInterface
     */
    
public function getSymfonyTransport()
    {
        return 
$this->transport;
    }

    
/**
     * Get the view factory instance.
     *
     * @return IlluminateContractsViewFactory
     */
    
public function getViewFactory()
    {
        return 
$this->views;
    }

    
/**
     * Set the Symfony Transport instance.
     *
     * @param  SymfonyComponentMailerTransportTransportInterface  $transport
     * @return void
     */
    
public function setSymfonyTransport(TransportInterface $transport)
    {
        
$this->transport $transport;
    }

    
/**
     * Set the queue manager instance.
     *
     * @param  IlluminateContractsQueueFactory  $queue
     * @return $this
     */
    
public function setQueue(QueueContract $queue)
    {
        
$this->queue $queue;

        return 
$this;
    }
}
Онлайн: 0
Реклама