View file upload/library/XenForo/Model/MailQueue.php

File size: 2.42Kb
<?php

class XenForo_Model_MailQueue extends XenForo_Model
{
	protected static $_deferredQueued = false;

	public function insertMailQueue(Zend_Mail $mailObj)
	{
		XenForo_Application::getDb()->insert('xf_mail_queue', array(
			'mail_data' => serialize($mailObj),
			'queue_date' => XenForo_Application::$time
		));

		if (!self::$_deferredQueued)
		{
			try
			{
				XenForo_Application::defer('MailQueue', array(), 'MailQueue');
				self::$_deferredQueued = true;
			}
			catch (Exception $e)
			{
				// need to just ignore this and let it get picked up later;
				// not doing this could lose email on a deadlock
			}
		}

		return true;
	}

	public function hasMailQueue()
	{
		$res = $this->_getDb()->fetchOne('
			SELECT MIN(mail_queue_id)
			FROM xf_mail_queue
		');
		return (bool)$res;
	}

	public function getMailQueue($limit = 20)
	{
		return $this->fetchAllKeyed($this->limitQueryResults(
			'
				SELECT *
				FROM xf_mail_queue
				ORDER BY queue_date
			', $limit
		), 'mail_queue_id');
	}

	public function runMailQueue($targetRunTime)
	{
		$s = microtime(true);
		$transport = XenForo_Mail::getTransport();
		$db = $this->_getDb();

		do
		{
			$queue = $this->getMailQueue($targetRunTime ? 20 : 0);

			foreach ($queue AS $id => $record)
			{
				if (!$db->delete('xf_mail_queue', 'mail_queue_id = ' . $db->quote($id)))
				{
					// already been deleted - run elsewhere
					continue;
				}

				$mailObj = @unserialize($record['mail_data']);
				if (!($mailObj instanceof Zend_Mail))
				{
					continue;
				}

				$thisTransport = XenForo_Mail::getFinalTransportForMail($mailObj, $transport);

				try
				{
					$mailObj->send($thisTransport);
				}
				catch (Exception $e)
				{
					$toEmails = implode(', ', $mailObj->getRecipients());

					if (method_exists($thisTransport, 'resetConnection'))
					{
						$thisTransport->resetConnection();

						try
						{
							$mailObj->send($thisTransport);
						}
						catch (Exception $e)
						{
							XenForo_Error::logException($e, false, "Email to $toEmails failed (after retry): ");
						}
					}
					else
					{
						XenForo_Error::logException($e, false, "Email to $toEmails failed: ");
					}
				}

				if ($targetRunTime && microtime(true) - $s > $targetRunTime)
				{
					$queue = false;
					break;
				}
			}
		}
		while ($queue);

		return $this->hasMailQueue();
	}
}