| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963 | 
							- <?php
 - 
 - require_once 'Swift/Tests/SwiftUnitTestCase.php';
 - require_once 'Swift/Transport/EsmtpHandler.php';
 - require_once 'Swift/Mime/Message.php';
 - require_once 'Swift/Transport/IoBuffer.php';
 - 
 - abstract class Swift_Transport_AbstractSmtpTest
 -   extends Swift_Tests_SwiftUnitTestCase
 - {
 -   
 -   /** Abstract test method */
 -   abstract protected function _getTransport($buf);
 -   
 -   public function testStartAccepts220ServiceGreeting()
 -   {
 -     /* -- RFC 2821, 4.2.
 -      
 -      Greeting = "220 " Domain [ SP text ] CRLF
 -      
 -      -- RFC 2822, 4.3.2.
 -      
 -      CONNECTION ESTABLISHMENT
 -          S: 220
 -          E: 554
 -     */
 -     
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $s = $this->_sequence('SMTP-convo');
 -     $this->_checking(Expectations::create()
 -       -> one($buf)->initialize() -> inSequence($s)
 -       -> one($buf)->readLine(0) -> inSequence($s) -> returns("220 some.server.tld bleh\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     try
 -     {
 -       $this->assertFalse($smtp->isStarted(), '%s: SMTP should begin non-started');
 -       $smtp->start();
 -       $this->assertTrue($smtp->isStarted(), '%s: start() should have started connection');
 -     }
 -     catch (Exception $e)
 -     {
 -       $this->fail('220 is a valid SMTP greeting and should be accepted');
 -     }
 -   }
 -   
 -   public function testBadGreetingCausesException()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $s = $this->_sequence('SMTP-convo');
 -     $this->_checking(Expectations::create()
 -       -> one($buf)->initialize() -> inSequence($s)
 -       -> one($buf)->readLine(0) -> inSequence($s) -> returns("554 I'm busy\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     try
 -     {
 -       $this->assertFalse($smtp->isStarted(), '%s: SMTP should begin non-started');
 -       $smtp->start();
 -       $this->fail('554 greeting indicates an error and should cause an exception');
 -     }
 -     catch (Exception $e)
 -     {
 -       $this->assertFalse($smtp->isStarted(), '%s: start() should have failed');
 -     }
 -   }
 -   
 -   public function testStartSendsHeloToInitiate()
 -   {
 -     /* -- RFC 2821, 3.2.
 -     
 -       3.2 Client Initiation
 - 
 -          Once the server has sent the welcoming message and the client has
 -          received it, the client normally sends the EHLO command to the
 -          server, indicating the client's identity.  In addition to opening the
 -          session, use of EHLO indicates that the client is able to process
 -          service extensions and requests that the server provide a list of the
 -          extensions it supports.  Older SMTP systems which are unable to
 -          support service extensions and contemporary clients which do not
 -          require service extensions in the mail session being initiated, MAY
 -          use HELO instead of EHLO.  Servers MUST NOT return the extended
 -          EHLO-style response to a HELO command.  For a particular connection
 -          attempt, if the server returns a "command not recognized" response to
 -          EHLO, the client SHOULD be able to fall back and send HELO.
 - 
 -          In the EHLO command the host sending the command identifies itself;
 -          the command may be interpreted as saying "Hello, I am <domain>" (and,
 -          in the case of EHLO, "and I support service extension requests").
 -          
 -        -- RFC 2281, 4.1.1.1.
 -        
 -        ehlo            = "EHLO" SP Domain CRLF
 -        helo            = "HELO" SP Domain CRLF
 -        
 -        -- RFC 2821, 4.3.2.
 -        
 -        EHLO or HELO
 -            S: 250
 -            E: 504, 550
 -        
 -      */
 -     
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $s = $this->_sequence('SMTP-convo');
 -     $this->_checking(Expectations::create()
 -       -> one($buf)->initialize() -> inSequence($s)
 -       -> one($buf)->readLine(0) -> inSequence($s) -> returns("220 some.server.tld bleh\r\n")
 -       -> one($buf)->write(pattern('~^HELO .*?\r\n$~D')) -> inSequence($s) -> returns(1)
 -       -> one($buf)->readLine(1) -> inSequence($s) -> returns('250 ServerName' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     try
 -     {
 -       $smtp->start();
 -     }
 -     catch (Exception $e)
 -     {
 -       $this->fail('Starting SMTP should send HELO and accept 250 response');
 -     }
 -   }
 -   
 -   public function testInvalidHeloResponseCausesException()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $s = $this->_sequence('SMTP-convo');
 -     $this->_checking(Expectations::create()
 -       -> one($buf)->initialize() -> inSequence($s)
 -       -> one($buf)->readLine(0) -> inSequence($s) -> returns("220 some.server.tld bleh\r\n")
 -       -> one($buf)->write(pattern('~^HELO .*?\r\n$~D')) -> inSequence($s) -> returns(1)
 -       -> one($buf)->readLine(1) -> inSequence($s) -> returns('504 WTF' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     try
 -     {
 -       $this->assertFalse($smtp->isStarted(), '%s: SMTP should begin non-started');
 -       $smtp->start();
 -       $this->fail('Non 250 HELO response should raise Exception');
 -     }
 -     catch (Exception $e)
 -     {
 -       $this->assertFalse($smtp->isStarted(), '%s: SMTP start() should have failed');
 -     }
 -   }
 -   
 -   public function testDomainNameIsPlacedInHelo()
 -   {
 -     /* -- RFC 2821, 4.1.4.
 -     
 -        The SMTP client MUST, if possible, ensure that the domain parameter
 -        to the EHLO command is a valid principal host name (not a CNAME or MX
 -        name) for its host.  If this is not possible (e.g., when the client's
 -        address is dynamically assigned and the client does not have an
 -        obvious name), an address literal SHOULD be substituted for the
 -        domain name and supplemental information provided that will assist in
 -        identifying the client.
 -     */
 -     
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $s = $this->_sequence('SMTP-convo');
 -     $this->_checking(Expectations::create()
 -       -> one($buf)->initialize() -> inSequence($s)
 -       -> one($buf)->readLine(0) -> inSequence($s) -> returns("220 some.server.tld bleh\r\n")
 -       -> one($buf)->write("HELO mydomain.com\r\n") -> inSequence($s) -> returns(1)
 -       -> one($buf)->readLine(1) -> inSequence($s) -> returns('250 ServerName' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->setLocalDomain('mydomain.com');
 -     $smtp->start();
 -   }
 -   
 -   public function testSuccessfulMailCommand()
 -   {
 -     /* -- RFC 2821, 3.3.
 -     
 -     There are three steps to SMTP mail transactions.  The transaction
 -     starts with a MAIL command which gives the sender identification.
 -     
 -     .....
 -     
 -     The first step in the procedure is the MAIL command.
 - 
 -       MAIL FROM:<reverse-path> [SP <mail-parameters> ] <CRLF>
 -       
 -     -- RFC 2821, 4.1.1.2.
 -     
 -     Syntax:
 - 
 -       "MAIL FROM:" ("<>" / Reverse-Path)
 -                        [SP Mail-parameters] CRLF
 -     -- RFC 2821, 4.1.2.
 -     
 -     Reverse-path = Path
 -       Forward-path = Path
 -       Path = "<" [ A-d-l ":" ] Mailbox ">"
 -       A-d-l = At-domain *( "," A-d-l )
 -             ; Note that this form, the so-called "source route",
 -             ; MUST BE accepted, SHOULD NOT be generated, and SHOULD be
 -             ; ignored.
 -       At-domain = "@" domain
 -       
 -     -- RFC 2821, 4.3.2.
 -     
 -     MAIL
 -       S: 250
 -       E: 552, 451, 452, 550, 553, 503
 -     */
 -     
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar'=>null))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("MAIL FROM: <me@domain.com>\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('250 OK' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     try
 -     {
 -       $smtp->start();
 -       $smtp->send($message);
 -     }
 -     catch (Exception $e)
 -     {
 -      $this->fail('MAIL FROM should accept a 250 response');
 -     }
 -   }
 -   
 -   public function testInvalidResponseCodeFromMailCausesException()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar'=>null))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("MAIL FROM: <me@domain.com>\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('553 Bad' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     try
 -     {
 -       $smtp->start();
 -       $smtp->send($message);
 -       $this->fail('MAIL FROM should accept a 250 response');
 -     }
 -     catch (Exception $e)
 -     {
 -     }
 -   }
 -   
 -   public function testSenderIsPreferredOverFrom()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getSender() -> returns(array('another@domain.com'=>'Someone'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar'=>null))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("MAIL FROM: <another@domain.com>\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('250 OK' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->start();
 -     $smtp->send($message);
 -   }
 -   
 -   public function testReturnPathIsPreferredOverSender()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getSender() -> returns(array('another@domain.com'=>'Someone'))
 -       -> allowing($message)->getReturnPath() -> returns('more@domain.com')
 -       -> allowing($message)->getTo() -> returns(array('foo@bar'=>null))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("MAIL FROM: <more@domain.com>\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('250 OK' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->start();
 -     $smtp->send($message);
 -   }
 -   
 -   public function testSuccessfulRcptCommandWith250Response()
 -   {
 -     /* -- RFC 2821, 3.3.
 -     
 -      The second step in the procedure is the RCPT command.
 - 
 -       RCPT TO:<forward-path> [ SP <rcpt-parameters> ] <CRLF>
 - 
 -      The first or only argument to this command includes a forward-path
 -      (normally a mailbox and domain, always surrounded by "<" and ">"
 -      brackets) identifying one recipient.  If accepted, the SMTP server
 -      returns a 250 OK reply and stores the forward-path.  If the recipient
 -      is known not to be a deliverable address, the SMTP server returns a
 -      550 reply, typically with a string such as "no such user - " and the
 -      mailbox name (other circumstances and reply codes are possible).
 -      This step of the procedure can be repeated any number of times.
 -     
 -     -- RFC 2821, 4.1.1.3.
 -     
 -     This command is used to identify an individual recipient of the mail
 -     data; multiple recipients are specified by multiple use of this
 -     command.  The argument field contains a forward-path and may contain
 -     optional parameters.
 - 
 -     The forward-path normally consists of the required destination
 -     mailbox.  Sending systems SHOULD not generate the optional list of
 -     hosts known as a source route.
 -     
 -     .......
 -     
 -     "RCPT TO:" ("<Postmaster@" domain ">" / "<Postmaster>" / Forward-Path)
 -                     [SP Rcpt-parameters] CRLF
 -                     
 -     -- RFC 2821, 4.2.2.
 -     
 -       250 Requested mail action okay, completed
 -       251 User not local; will forward to <forward-path>
 -          (See section 3.4)
 -       252 Cannot VRFY user, but will accept message and attempt
 -           delivery
 -     
 -     -- RFC 2821, 4.3.2.
 -     
 -     RCPT
 -       S: 250, 251 (but see section 3.4 for discussion of 251 and 551)
 -       E: 550, 551, 552, 553, 450, 451, 452, 503, 550
 -     */
 -     
 -     //We'll treat 252 as accepted since it isn't really a failure
 -     
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $s = $this->_sequence('SMTP-envelope');
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar'=>null))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("MAIL FROM: <me@domain.com>\r\n") -> inSequence($s) -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('250 OK' . "\r\n")
 -       -> one($buf)->write("RCPT TO: <foo@bar>\r\n") -> inSequence($s) -> returns(2)
 -       -> one($buf)->readLine(2) -> returns('250 OK' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     try
 -     {
 -       $smtp->start();
 -       $smtp->send($message);
 -     }
 -     catch (Exception $e)
 -     {
 -       $this->fail('RCPT TO should accept a 250 response');
 -     }
 -   }
 -   
 -   public function testMailFromCommandIsOnlySentOncePerMessage()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $s = $this->_sequence('SMTP-envelope');
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar'=>null))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("MAIL FROM: <me@domain.com>\r\n") -> inSequence($s) -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('250 OK' . "\r\n")
 -       -> one($buf)->write("RCPT TO: <foo@bar>\r\n") -> inSequence($s) -> returns(2)
 -       -> one($buf)->readLine(2) -> returns('250 OK' . "\r\n")
 -       -> never($buf)->write("MAIL FROM: <me@domain.com>\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->start();
 -     $smtp->send($message);
 -   }
 -   
 -   public function testMultipleRecipientsSendsMultipleRcpt()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array(
 -         'foo@bar' => null,
 -         'zip@button' => 'Zip Button',
 -         'test@domain' => 'Test user'
 -         ))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("RCPT TO: <foo@bar>\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('250 OK' . "\r\n")
 -       -> one($buf)->write("RCPT TO: <zip@button>\r\n") -> returns(2)
 -       -> one($buf)->readLine(2) -> returns('250 OK' . "\r\n")
 -       -> one($buf)->write("RCPT TO: <test@domain>\r\n") -> returns(3)
 -       -> one($buf)->readLine(3) -> returns('250 OK' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->start();
 -     $smtp->send($message);
 -   }
 -   
 -   public function testCcRecipientsSendsMultipleRcpt()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar' => null))
 -       -> allowing($message)->getCc() -> returns(array(
 -         'zip@button' => 'Zip Button',
 -         'test@domain' => 'Test user'
 -         ))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("RCPT TO: <foo@bar>\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('250 OK' . "\r\n")
 -       -> one($buf)->write("RCPT TO: <zip@button>\r\n") -> returns(2)
 -       -> one($buf)->readLine(2) -> returns('250 OK' . "\r\n")
 -       -> one($buf)->write("RCPT TO: <test@domain>\r\n") -> returns(3)
 -       -> one($buf)->readLine(3) -> returns('250 OK' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->start();
 -     $smtp->send($message);
 -   }
 -   
 -   public function testSendReturnsNumberOfSuccessfulRecipients()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar' => null))
 -       -> allowing($message)->getCc() -> returns(array(
 -         'zip@button' => 'Zip Button',
 -         'test@domain' => 'Test user'
 -         ))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("RCPT TO: <foo@bar>\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('250 OK' . "\r\n")
 -       -> one($buf)->write("RCPT TO: <zip@button>\r\n") -> returns(2)
 -       -> one($buf)->readLine(2) -> returns('501 Nobody here' . "\r\n")
 -       -> one($buf)->write("RCPT TO: <test@domain>\r\n") -> returns(3)
 -       -> one($buf)->readLine(3) -> returns('250 OK' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->start();
 -     $this->assertEqual(2, $smtp->send($message),
 -       '%s: 1 of 3 recipients failed so 2 should be returned'
 -       );
 -   }
 -   
 -   public function testRsetIsSentIfNoSuccessfulRecipients()
 -   {
 -     /* --RFC 2821, 4.1.1.5.
 -     
 -     This command specifies that the current mail transaction will be
 -     aborted.  Any stored sender, recipients, and mail data MUST be
 -     discarded, and all buffers and state tables cleared.  The receiver
 -     MUST send a "250 OK" reply to a RSET command with no arguments.  A
 -     reset command may be issued by the client at any time.
 -    
 -     -- RFC 2821, 4.3.2.
 -     
 -     RSET
 -       S: 250
 -     */
 -     
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar' => null))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("RCPT TO: <foo@bar>\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('503 Bad' . "\r\n")
 -       -> one($buf)->write("RSET\r\n") -> returns(2)
 -       -> one($buf)->readLine(2) -> returns('250 OK' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->start();
 -     $this->assertEqual(0, $smtp->send($message),
 -       '%s: 1 of 1 recipients failed so 0 should be returned'
 -       );
 -   }
 -   
 -   public function testSuccessfulDataCommand()
 -   {
 -     /* -- RFC 2821, 3.3.
 -     
 -     The third step in the procedure is the DATA command (or some
 -     alternative specified in a service extension).
 -     
 -           DATA <CRLF>
 - 
 -     If accepted, the SMTP server returns a 354 Intermediate reply and
 -     considers all succeeding lines up to but not including the end of
 -     mail data indicator to be the message text.
 -     
 -     -- RFC 2821, 4.1.1.4.
 -     
 -     The receiver normally sends a 354 response to DATA, and then treats
 -     the lines (strings ending in <CRLF> sequences, as described in
 -     section 2.3.7) following the command as mail data from the sender.
 -     This command causes the mail data to be appended to the mail data
 -     buffer.  The mail data may contain any of the 128 ASCII character
 -     codes, although experience has indicated that use of control
 -     characters other than SP, HT, CR, and LF may cause problems and
 -     SHOULD be avoided when possible.
 -    
 -     -- RFC 2821, 4.3.2.
 -     
 -     DATA
 -       I: 354 -> data -> S: 250
 -                         E: 552, 554, 451, 452
 -       E: 451, 554, 503
 -     */
 -     
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar' => null))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("DATA\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('354 Go ahead' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     try
 -     {
 -       $smtp->start();
 -       $smtp->send($message);
 -     }
 -     catch (Exception $e)
 -     {
 -       $this->fail('354 is the expected response to DATA');
 -     }
 -   }
 -   
 -   public function testBadDataResponseCausesException()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar' => null))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("DATA\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('451 Bad' . "\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     try
 -     {
 -       $smtp->start();
 -       $smtp->send($message);
 -       $this->fail('354 is the expected response to DATA (not observed)');
 -     }
 -     catch (Exception $e)
 -     {
 -     }
 -   }
 -   
 -   public function testMessageIsStreamedToBufferForData()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $s = $this->_sequence('DATA Streaming');
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar' => null))
 -       
 -       -> one($buf)->write("DATA\r\n") -> inSequence($s) -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('354 OK' . "\r\n")
 -       -> one($message)->toByteStream($buf) -> inSequence($s)
 -       -> one($buf)->write("\r\n.\r\n") -> inSequence($s) -> returns(2)
 -       -> one($buf)->readLine(2) -> returns('250 OK' . "\r\n")
 -       
 -       -> allowing($message)
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->start();
 -     $smtp->send($message);
 -   }
 -   
 -   public function testBadResponseAfterDataTransmissionCausesException()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $s = $this->_sequence('DATA Streaming');
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar' => null))
 -       
 -       -> one($buf)->write("DATA\r\n") -> inSequence($s) -> returns(1)
 -       -> one($buf)->readLine(1) -> returns('354 OK' . "\r\n")
 -       -> one($message)->toByteStream($buf) -> inSequence($s)
 -       -> one($buf)->write("\r\n.\r\n") -> inSequence($s) -> returns(2)
 -       -> one($buf)->readLine(2) -> returns('554 Error' . "\r\n")
 -       
 -       -> allowing($message)
 -       );
 -     $this->_finishBuffer($buf);
 -     try
 -     {
 -       $smtp->start();
 -       $smtp->send($message);
 -       $this->fail('250 is the expected response after a DATA transmission (not observed)');
 -     }
 -     catch (Exception $e)
 -     {
 -     }
 -   }
 -   
 -   public function testBccRecipientsAreRemovedFromHeaders()
 -   {
 -     /* -- RFC 2821, 7.2.
 -     
 -      Addresses that do not appear in the message headers may appear in the
 -      RCPT commands to an SMTP server for a number of reasons.  The two
 -      most common involve the use of a mailing address as a "list exploder"
 -      (a single address that resolves into multiple addresses) and the
 -      appearance of "blind copies".  Especially when more than one RCPT
 -      command is present, and in order to avoid defeating some of the
 -      purpose of these mechanisms, SMTP clients and servers SHOULD NOT copy
 -      the full set of RCPT command arguments into the headers, either as
 -      part of trace headers or as informational or private-extension
 -      headers.  Since this rule is often violated in practice, and cannot
 -      be enforced, sending SMTP systems that are aware of "bcc" use MAY
 -      find it helpful to send each blind copy as a separate message
 -      transaction containing only a single RCPT command.
 -      */
 -     
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar' => null))
 -       -> allowing($message)->getBcc() -> returns(array(
 -         'zip@button' => 'Zip Button',
 -         'test@domain' => 'Test user'
 -         ))
 -       -> atLeast(1)->of($message)->setBcc(array())
 -       -> allowing($message)
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->start();
 -     $smtp->send($message);
 -   }
 -   
 -   public function testEachBccRecipientIsSentASeparateMessage()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar' => null))
 -       -> allowing($message)->getBcc() -> returns(array(
 -         'zip@button' => 'Zip Button',
 -         'test@domain' => 'Test user'
 -         ))
 -       -> atLeast(1)->of($message)->setBcc(array())
 -       -> one($message)->setBcc(array('zip@button' => 'Zip Button'))
 -       -> one($message)->setBcc(array('test@domain' => 'Test user'))
 -       -> atLeast(1)->of($message)->setBcc(array(
 -         'zip@button' => 'Zip Button',
 -         'test@domain' => 'Test user'
 -         ))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("MAIL FROM: <me@domain.com>\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns("250 OK\r\n")
 -       -> one($buf)->write("RCPT TO: <foo@bar>\r\n") -> returns(2)
 -       -> one($buf)->readLine(2) -> returns("250 OK\r\n")
 -       -> one($buf)->write("DATA\r\n") -> returns(3)
 -       -> one($buf)->readLine(3) -> returns("354 OK\r\n")
 -       -> one($buf)->write("\r\n.\r\n") -> returns(4)
 -       -> one($buf)->readLine(4) -> returns("250 OK\r\n")
 -       
 -       -> one($buf)->write("MAIL FROM: <me@domain.com>\r\n") -> returns(5)
 -       -> one($buf)->readLine(5) -> returns("250 OK\r\n")
 -       -> one($buf)->write("RCPT TO: <zip@button>\r\n") -> returns(6)
 -       -> one($buf)->readLine(6) -> returns("250 OK\r\n")
 -       -> one($buf)->write("DATA\r\n") -> returns(7)
 -       -> one($buf)->readLine(7) -> returns("354 OK\r\n")
 -       -> one($buf)->write("\r\n.\r\n") -> returns(8)
 -       -> one($buf)->readLine(8) -> returns("250 OK\r\n")
 -       
 -       -> one($buf)->write("MAIL FROM: <me@domain.com>\r\n") -> returns(9)
 -       -> one($buf)->readLine(9) -> returns("250 OK\r\n")
 -       -> one($buf)->write("RCPT TO: <test@domain>\r\n") -> returns(10)
 -       -> one($buf)->readLine(10) -> returns("250 OK\r\n")
 -       -> one($buf)->write("DATA\r\n") -> returns(11)
 -       -> one($buf)->readLine(11) -> returns("354 OK\r\n")
 -       -> one($buf)->write("\r\n.\r\n") -> returns(12)
 -       -> one($buf)->readLine(12) -> returns("250 OK\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->start();
 -     $this->assertEqual(3, $smtp->send($message));
 -   }
 -   
 -   public function testMessageStateIsRestoredOnFailure()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar' => null))
 -       -> allowing($message)->getBcc() -> returns(array(
 -         'zip@button' => 'Zip Button',
 -         'test@domain' => 'Test user'
 -         ))
 -       -> one($message)->setBcc(array())
 -       -> one($message)->setBcc(array(
 -         'zip@button' => 'Zip Button',
 -         'test@domain' => 'Test user'
 -         ))
 -       -> allowing($message)
 -       -> one($buf)->write("MAIL FROM: <me@domain.com>\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns("250 OK\r\n")
 -       -> one($buf)->write("RCPT TO: <foo@bar>\r\n") -> returns(2)
 -       -> one($buf)->readLine(2) -> returns("250 OK\r\n")
 -       -> one($buf)->write("DATA\r\n") -> returns(3)
 -       -> one($buf)->readLine(3) -> returns("451 No\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     
 -     $smtp->start();
 -     try
 -     {
 -       $smtp->send($message);
 -       $this->fail('A bad response was given so exception is expected');
 -     }
 -     catch (Exception $e)
 -     {
 -     }
 -   }
 -   
 -   public function testStopSendsQuitCommand()
 -   {
 -     /* -- RFC 2821, 4.1.1.10.
 -     
 -     This command specifies that the receiver MUST send an OK reply, and
 -     then close the transmission channel.
 - 
 -     The receiver MUST NOT intentionally close the transmission channel
 -     until it receives and replies to a QUIT command (even if there was an
 -     error).  The sender MUST NOT intentionally close the transmission
 -     channel until it sends a QUIT command and SHOULD wait until it
 -     receives the reply (even if there was an error response to a previous
 -     command).  If the connection is closed prematurely due to violations
 -     of the above or system or network failure, the server MUST cancel any
 -     pending transaction, but not undo any previously completed
 -     transaction, and generally MUST act as if the command or transaction
 -     in progress had received a temporary error (i.e., a 4yz response).
 - 
 -     The QUIT command may be issued at any time.
 - 
 -     Syntax:
 -       "QUIT" CRLF
 -     */
 -     
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> one($buf)->initialize()
 -       -> one($buf)->write("QUIT\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns("221 Bye\r\n")
 -       -> one($buf)->terminate()
 -       );
 -     $this->_finishBuffer($buf);
 -     
 -     $this->assertFalse($smtp->isStarted());
 -     $smtp->start();
 -     $this->assertTrue($smtp->isStarted());
 -     $smtp->stop();
 -     $this->assertFalse($smtp->isStarted());
 -   }
 -   
 -   public function testBufferCanBeFetched()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $ref = $smtp->getBuffer();
 -     $this->assertReference($buf, $ref);
 -   }
 -   
 -   public function testBufferCanBeWrittenToUsingExecuteCommand()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> one($buf)->write("FOO\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns("250 OK\r\n")
 -       -> ignoring($buf)
 -       );
 -     
 -     $res = $smtp->executeCommand("FOO\r\n");
 -     $this->assertEqual("250 OK\r\n", $res);
 -   }
 -   
 -   public function testResponseCodesAreValidated()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> one($buf)->write("FOO\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns("551 Not ok\r\n")
 -       -> ignoring($buf)
 -       );
 -     
 -     try
 -     {
 -       $smtp->executeCommand("FOO\r\n", array(250, 251));
 -       $this->fail('A 250 or 251 response was needed but 551 was returned.');
 -     }
 -     catch (Exception $e)
 -     {
 -     }
 -   }
 -   
 -   public function testFailedRecipientsCanBeCollectedByReference()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar' => null))
 -       -> allowing($message)->getBcc() -> returns(array(
 -         'zip@button' => 'Zip Button',
 -         'test@domain' => 'Test user'
 -         ))
 -       -> atLeast(1)->of($message)->setBcc(array())
 -       -> one($message)->setBcc(array('zip@button' => 'Zip Button'))
 -       -> one($message)->setBcc(array('test@domain' => 'Test user'))
 -       -> atLeast(1)->of($message)->setBcc(array(
 -         'zip@button' => 'Zip Button',
 -         'test@domain' => 'Test user'
 -         ))
 -       -> allowing($message)
 -       
 -       -> one($buf)->write("MAIL FROM: <me@domain.com>\r\n") -> returns(1)
 -       -> one($buf)->readLine(1) -> returns("250 OK\r\n")
 -       -> one($buf)->write("RCPT TO: <foo@bar>\r\n") -> returns(2)
 -       -> one($buf)->readLine(2) -> returns("250 OK\r\n")
 -       -> one($buf)->write("DATA\r\n") -> returns(3)
 -       -> one($buf)->readLine(3) -> returns("354 OK\r\n")
 -       -> one($buf)->write("\r\n.\r\n") -> returns(4)
 -       -> one($buf)->readLine(4) -> returns("250 OK\r\n")
 -       
 -       -> one($buf)->write("MAIL FROM: <me@domain.com>\r\n") -> returns(5)
 -       -> one($buf)->readLine(5) -> returns("250 OK\r\n")
 -       -> one($buf)->write("RCPT TO: <zip@button>\r\n") -> returns(6)
 -       -> one($buf)->readLine(6) -> returns("500 Bad\r\n")
 -       -> one($buf)->write("RSET\r\n") -> returns(7)
 -       -> one($buf)->readLine(7) -> returns("250 OK\r\n")
 -       
 -       -> one($buf)->write("MAIL FROM: <me@domain.com>\r\n") -> returns(8)
 -       -> one($buf)->readLine(8) -> returns("250 OK\r\n")
 -       -> one($buf)->write("RCPT TO: <test@domain>\r\n") -> returns(9)
 -       -> one($buf)->readLine(9) -> returns("500 Bad\r\n")
 -       -> one($buf)->write("RSET\r\n") -> returns(10)
 -       -> one($buf)->readLine(10) -> returns("250 OK\r\n")
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->start();
 -     $this->assertEqual(1, $smtp->send($message, $failures));
 -     $this->assertEqual(array('zip@button', 'test@domain'), $failures,
 -       '%s: Failures should be caught in an array'
 -       );
 -   }
 -   
 -   public function testSendingRegeneratesMessageId()
 -   {
 -     $buf = $this->_getBuffer();
 -     $smtp = $this->_getTransport($buf);
 -     $message = $this->_createMessage();
 -     $this->_checking(Expectations::create()
 -       -> allowing($message)->getFrom() -> returns(array('me@domain.com'=>'Me'))
 -       -> allowing($message)->getTo() -> returns(array('foo@bar'=>null))
 -       -> one($message)->generateId()
 -       -> allowing($message)
 -       );
 -     $this->_finishBuffer($buf);
 -     $smtp->start();
 -     $smtp->send($message);
 -   }
 -   
 -   // -- Protected methods
 -   
 -   protected function _getBuffer()
 -   {
 -     return $this->_mock('Swift_Transport_IoBuffer');
 -   }
 -   
 -   protected function _createMessage()
 -   {
 -     return $this->_mock('Swift_Mime_Message');
 -   }
 -   
 -   protected function _finishBuffer($buf)
 -   {
 -     $this->_checking(Expectations::create()
 -       -> ignoring($buf)->readLine(0) -> returns('220 server.com foo' . "\r\n")
 -       -> ignoring($buf)->write(pattern('~^(EH|HE)LO .*?\r\n$~D')) -> returns($x = uniqid())
 -       -> ignoring($buf)->readLine($x) -> returns('250 ServerName' . "\r\n")
 -       -> ignoring($buf)->write(pattern('~^MAIL FROM: <.*?>\r\n$~D')) -> returns($x = uniqid())
 -       -> ignoring($buf)->readLine($x) -> returns('250 OK' . "\r\n")
 -       -> ignoring($buf)->write(pattern('~^RCPT TO: <.*?>\r\n$~D')) -> returns($x = uniqid())
 -       -> ignoring($buf)->readLine($x) -> returns('250 OK' . "\r\n")
 -       -> ignoring($buf)->write("DATA\r\n") -> returns($x = uniqid())
 -       -> ignoring($buf)->readLine($x) -> returns('354 OK' . "\r\n")
 -       -> ignoring($buf)->write("\r\n.\r\n") -> returns($x = uniqid())
 -       -> ignoring($buf)->readLine($x) -> returns('250 OK' . "\r\n")
 -       -> ignoring($buf)->write("RSET\r\n") -> returns($x = uniqid())
 -       -> ignoring($buf)->readLine($x) -> returns("250 OK\r\n")
 -       -> ignoring($buf) -> returns(false)
 -       );
 -   }
 -   
 - }
 
 
  |