197 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			197 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| /*
 | |
|  * This file is part of SwiftMailer.
 | |
|  * (c) 2004-2009 Chris Corbyn
 | |
|  *
 | |
|  * For the full copyright and license information, please view the LICENSE
 | |
|  * file that was distributed with this source code.
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * Throttles the rate at which emails are sent.
 | |
|  *
 | |
|  * @author Chris Corbyn
 | |
|  */
 | |
| class Swift_Plugins_ThrottlerPlugin extends Swift_Plugins_BandwidthMonitorPlugin implements Swift_Plugins_Sleeper, Swift_Plugins_Timer
 | |
| {
 | |
|     /** Flag for throttling in bytes per minute */
 | |
|     const BYTES_PER_MINUTE = 0x01;
 | |
| 
 | |
|     /** Flag for throttling in emails per second (Amazon SES) */
 | |
|     const MESSAGES_PER_SECOND = 0x11;
 | |
| 
 | |
|     /** Flag for throttling in emails per minute */
 | |
|     const MESSAGES_PER_MINUTE = 0x10;
 | |
| 
 | |
|     /**
 | |
|      * The Sleeper instance for sleeping.
 | |
|      *
 | |
|      * @var Swift_Plugins_Sleeper
 | |
|      */
 | |
|     private $sleeper;
 | |
| 
 | |
|     /**
 | |
|      * The Timer instance which provides the timestamp.
 | |
|      *
 | |
|      * @var Swift_Plugins_Timer
 | |
|      */
 | |
|     private $timer;
 | |
| 
 | |
|     /**
 | |
|      * The time at which the first email was sent.
 | |
|      *
 | |
|      * @var int
 | |
|      */
 | |
|     private $start;
 | |
| 
 | |
|     /**
 | |
|      * The rate at which messages should be sent.
 | |
|      *
 | |
|      * @var int
 | |
|      */
 | |
|     private $rate;
 | |
| 
 | |
|     /**
 | |
|      * The mode for throttling.
 | |
|      *
 | |
|      * This is {@link BYTES_PER_MINUTE} or {@link MESSAGES_PER_MINUTE}
 | |
|      *
 | |
|      * @var int
 | |
|      */
 | |
|     private $mode;
 | |
| 
 | |
|     /**
 | |
|      * An internal counter of the number of messages sent.
 | |
|      *
 | |
|      * @var int
 | |
|      */
 | |
|     private $messages = 0;
 | |
| 
 | |
|     /**
 | |
|      * Create a new ThrottlerPlugin.
 | |
|      *
 | |
|      * @param int                   $rate
 | |
|      * @param int                   $mode    defaults to {@link BYTES_PER_MINUTE}
 | |
|      * @param Swift_Plugins_Sleeper $sleeper (only needed in testing)
 | |
|      * @param Swift_Plugins_Timer   $timer   (only needed in testing)
 | |
|      */
 | |
|     public function __construct($rate, $mode = self::BYTES_PER_MINUTE, Swift_Plugins_Sleeper $sleeper = null, Swift_Plugins_Timer $timer = null)
 | |
|     {
 | |
|         $this->rate = $rate;
 | |
|         $this->mode = $mode;
 | |
|         $this->sleeper = $sleeper;
 | |
|         $this->timer = $timer;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Invoked immediately before the Message is sent.
 | |
|      */
 | |
|     public function beforeSendPerformed(Swift_Events_SendEvent $evt)
 | |
|     {
 | |
|         $time = $this->getTimestamp();
 | |
|         if (!isset($this->start)) {
 | |
|             $this->start = $time;
 | |
|         }
 | |
|         $duration = $time - $this->start;
 | |
| 
 | |
|         switch ($this->mode) {
 | |
|             case self::BYTES_PER_MINUTE:
 | |
|                 $sleep = $this->throttleBytesPerMinute($duration);
 | |
|                 break;
 | |
|             case self::MESSAGES_PER_SECOND:
 | |
|                 $sleep = $this->throttleMessagesPerSecond($duration);
 | |
|                 break;
 | |
|             case self::MESSAGES_PER_MINUTE:
 | |
|                 $sleep = $this->throttleMessagesPerMinute($duration);
 | |
|                 break;
 | |
|             default:
 | |
|                 $sleep = 0;
 | |
|                 break;
 | |
|         }
 | |
| 
 | |
|         if ($sleep > 0) {
 | |
|             $this->sleep($sleep);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Invoked when a Message is sent.
 | |
|      */
 | |
|     public function sendPerformed(Swift_Events_SendEvent $evt)
 | |
|     {
 | |
|         parent::sendPerformed($evt);
 | |
|         ++$this->messages;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Sleep for $seconds.
 | |
|      *
 | |
|      * @param int $seconds
 | |
|      */
 | |
|     public function sleep($seconds)
 | |
|     {
 | |
|         if (isset($this->sleeper)) {
 | |
|             $this->sleeper->sleep($seconds);
 | |
|         } else {
 | |
|             sleep($seconds);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the current UNIX timestamp.
 | |
|      *
 | |
|      * @return int
 | |
|      */
 | |
|     public function getTimestamp()
 | |
|     {
 | |
|         if (isset($this->timer)) {
 | |
|             return $this->timer->getTimestamp();
 | |
|         }
 | |
| 
 | |
|         return time();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a number of seconds to sleep for.
 | |
|      *
 | |
|      * @param int $timePassed
 | |
|      *
 | |
|      * @return int
 | |
|      */
 | |
|     private function throttleBytesPerMinute($timePassed)
 | |
|     {
 | |
|         $expectedDuration = $this->getBytesOut() / ($this->rate / 60);
 | |
| 
 | |
|         return (int) ceil($expectedDuration - $timePassed);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a number of seconds to sleep for.
 | |
|      *
 | |
|      * @param int $timePassed
 | |
|      *
 | |
|      * @return int
 | |
|      */
 | |
|     private function throttleMessagesPerSecond($timePassed)
 | |
|     {
 | |
|         $expectedDuration = $this->messages / $this->rate;
 | |
| 
 | |
|         return (int) ceil($expectedDuration - $timePassed);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get a number of seconds to sleep for.
 | |
|      *
 | |
|      * @param int $timePassed
 | |
|      *
 | |
|      * @return int
 | |
|      */
 | |
|     private function throttleMessagesPerMinute($timePassed)
 | |
|     {
 | |
|         $expectedDuration = $this->messages / ($this->rate / 60);
 | |
| 
 | |
|         return (int) ceil($expectedDuration - $timePassed);
 | |
|     }
 | |
| }
 |