123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483 |
- <?php
-
- require_once 'Swift/Tests/SwiftUnitTestCase.php';
- require_once 'Swift/Mime/ContentEncoder/QpContentEncoder.php';
- require_once 'Swift/InputByteStream.php';
- require_once 'Swift/OutputByteStream.php';
- require_once 'Swift/CharacterStream.php';
-
- class Swift_StreamCollector implements Yay_Action {
- public $content = '';
- public function &invoke(Yay_Invocation $inv) {
- $args = $inv->getArguments();
- $this->content .= current($args);
- }
- public function describeTo(Yay_Description $description) {
- $description->appendText(' gathers input;');
- }
- }
-
- class Swift_Mime_ContentEncoder_QpContentEncoderTest
- extends Swift_Tests_SwiftUnitTestCase
- {
-
- public function testNameIsQuotedPrintable()
- {
- $encoder = new Swift_Mime_ContentEncoder_QpContentEncoder(
- $this->_createCharacterStream(true)
- );
- $this->assertEqual('quoted-printable', $encoder->getName());
- }
-
- /* -- RFC 2045, 6.7 --
- (1) (General 8bit representation) Any octet, except a CR or
- LF that is part of a CRLF line break of the canonical
- (standard) form of the data being encoded, may be
- represented by an "=" followed by a two digit
- hexadecimal representation of the octet's value. The
- digits of the hexadecimal alphabet, for this purpose,
- are "0123456789ABCDEF". Uppercase letters must be
- used; lowercase letters are not allowed. Thus, for
- example, the decimal value 12 (US-ASCII form feed) can
- be represented by "=0C", and the decimal value 61 (US-
- ASCII EQUAL SIGN) can be represented by "=3D". This
- rule must be followed except when the following rules
- allow an alternative encoding.
- */
-
- public function testPermittedCharactersAreNotEncoded()
- {
- /* -- RFC 2045, 6.7 --
- (2) (Literal representation) Octets with decimal values of
- 33 through 60 inclusive, and 62 through 126, inclusive,
- MAY be represented as the US-ASCII characters which
- correspond to those octets (EXCLAMATION POINT through
- LESS THAN, and GREATER THAN through TILDE,
- respectively).
- */
-
- foreach (array_merge(range(33, 60), range(62, 126)) as $ordinal)
- {
- $char = chr($ordinal);
-
- $os = $this->_createOutputByteStream(true);
- $charStream = $this->_createCharacterStream();
- $is = $this->_createInputByteStream();
- $collection = new Swift_StreamCollector();
-
- $this->_checking(Expectations::create()
- -> one($charStream)->flushContents()
- -> one($charStream)->importByteStream($os)
- -> one($charStream)->readBytes(any()) -> returns(array($ordinal))
- -> allowing($charStream)->readBytes(any()) -> returns(false)
-
- -> allowing($is)->write(any(), optional()) -> will($collection)
- -> ignoring($is)
-
- -> ignoring($os)
- );
-
- $encoder = new Swift_Mime_ContentEncoder_QpContentEncoder($charStream);
- $encoder->encodeByteStream($os, $is);
- $this->assertIdenticalBinary($char, $collection->content);
- }
- }
-
- public function testLinearWhiteSpaceAtLineEndingIsEncoded()
- {
- /* -- RFC 2045, 6.7 --
- (3) (White Space) Octets with values of 9 and 32 MAY be
- represented as US-ASCII TAB (HT) and SPACE characters,
- respectively, but MUST NOT be so represented at the end
- of an encoded line. Any TAB (HT) or SPACE characters
- on an encoded line MUST thus be followed on that line
- by a printable character. In particular, an "=" at the
- end of an encoded line, indicating a soft line break
- (see rule #5) may follow one or more TAB (HT) or SPACE
- characters. It follows that an octet with decimal
- value 9 or 32 appearing at the end of an encoded line
- must be represented according to Rule #1. This rule is
- necessary because some MTAs (Message Transport Agents,
- programs which transport messages from one user to
- another, or perform a portion of such transfers) are
- known to pad lines of text with SPACEs, and others are
- known to remove "white space" characters from the end
- of a line. Therefore, when decoding a Quoted-Printable
- body, any trailing white space on a line must be
- deleted, as it will necessarily have been added by
- intermediate transport agents.
- */
-
- $HT = chr(0x09); //9
- $SPACE = chr(0x20); //32
-
- //HT
- $os = $this->_createOutputByteStream(true);
- $charStream = $this->_createCharacterStream();
- $is = $this->_createInputByteStream();
- $collection = new Swift_StreamCollector();
-
- $this->_checking(Expectations::create()
- -> one($charStream)->flushContents()
- -> one($charStream)->importByteStream($os)
- -> one($charStream)->readBytes(any()) -> returns(array(ord('a')))
- -> one($charStream)->readBytes(any()) -> returns(array(0x09))
- -> one($charStream)->readBytes(any()) -> returns(array(0x09))
- -> one($charStream)->readBytes(any()) -> returns(array(0x0D))
- -> one($charStream)->readBytes(any()) -> returns(array(0x0A))
- -> one($charStream)->readBytes(any()) -> returns(array(ord('b')))
- -> allowing($charStream)->readBytes(any()) -> returns(false)
-
- -> allowing($is)->write(any(), optional()) -> will($collection)
- -> ignoring($is)
-
- -> ignoring($os)
- );
-
- $encoder = new Swift_Mime_ContentEncoder_QpContentEncoder($charStream);
- $encoder->encodeByteStream($os, $is);
-
- $this->assertEqual("a\t=09\r\nb", $collection->content);
-
- //SPACE
- $os = $this->_createOutputByteStream(true);
- $charStream = $this->_createCharacterStream();
- $is = $this->_createInputByteStream();
- $collection = new Swift_StreamCollector();
-
- $this->_checking(Expectations::create()
- -> one($charStream)->flushContents()
- -> one($charStream)->importByteStream($os)
- -> one($charStream)->readBytes(any()) -> returns(array(ord('a')))
- -> one($charStream)->readBytes(any()) -> returns(array(0x20))
- -> one($charStream)->readBytes(any()) -> returns(array(0x20))
- -> one($charStream)->readBytes(any()) -> returns(array(0x0D))
- -> one($charStream)->readBytes(any()) -> returns(array(0x0A))
- -> one($charStream)->readBytes(any()) -> returns(array(ord('b')))
- -> allowing($charStream)->readBytes(any()) -> returns(false)
-
- -> allowing($is)->write(any(), optional()) -> will($collection)
- -> ignoring($is)
-
- -> ignoring($os)
- );
-
- $encoder = new Swift_Mime_ContentEncoder_QpContentEncoder($charStream);
- $encoder->encodeByteStream($os, $is);
-
- $this->assertEqual("a =20\r\nb", $collection->content);
- }
-
- public function testCRLFIsLeftAlone()
- {
- /*
- (4) (Line Breaks) A line break in a text body, represented
- as a CRLF sequence in the text canonical form, must be
- represented by a (RFC 822) line break, which is also a
- CRLF sequence, in the Quoted-Printable encoding. Since
- the canonical representation of media types other than
- text do not generally include the representation of
- line breaks as CRLF sequences, no hard line breaks
- (i.e. line breaks that are intended to be meaningful
- and to be displayed to the user) can occur in the
- quoted-printable encoding of such types. Sequences
- like "=0D", "=0A", "=0A=0D" and "=0D=0A" will routinely
- appear in non-text data represented in quoted-
- printable, of course.
-
- Note that many implementations may elect to encode the
- local representation of various content types directly
- rather than converting to canonical form first,
- encoding, and then converting back to local
- representation. In particular, this may apply to plain
- text material on systems that use newline conventions
- other than a CRLF terminator sequence. Such an
- implementation optimization is permissible, but only
- when the combined canonicalization-encoding step is
- equivalent to performing the three steps separately.
- */
-
- $os = $this->_createOutputByteStream(true);
- $charStream = $this->_createCharacterStream();
- $is = $this->_createInputByteStream();
- $collection = new Swift_StreamCollector();
-
- $this->_checking(Expectations::create()
- -> one($charStream)->flushContents()
- -> one($charStream)->importByteStream($os)
- -> one($charStream)->readBytes(any()) -> returns(array(ord('a')))
- -> one($charStream)->readBytes(any()) -> returns(array(0x0D))
- -> one($charStream)->readBytes(any()) -> returns(array(0x0A))
- -> one($charStream)->readBytes(any()) -> returns(array(ord('b')))
- -> one($charStream)->readBytes(any()) -> returns(array(0x0D))
- -> one($charStream)->readBytes(any()) -> returns(array(0x0A))
- -> one($charStream)->readBytes(any()) -> returns(array(ord('c')))
- -> one($charStream)->readBytes(any()) -> returns(array(0x0D))
- -> one($charStream)->readBytes(any()) -> returns(array(0x0A))
- -> allowing($charStream)->readBytes(any()) -> returns(false)
-
- -> allowing($is)->write(any(), optional()) -> will($collection)
- -> ignoring($is)
-
- -> ignoring($os)
- );
-
- $encoder = new Swift_Mime_ContentEncoder_QpContentEncoder($charStream);
- $encoder->encodeByteStream($os, $is);
- $this->assertEqual("a\r\nb\r\nc\r\n", $collection->content);
- }
-
- public function testLinesLongerThan76CharactersAreSoftBroken()
- {
- /*
- (5) (Soft Line Breaks) The Quoted-Printable encoding
- REQUIRES that encoded lines be no more than 76
- characters long. If longer lines are to be encoded
- with the Quoted-Printable encoding, "soft" line breaks
- must be used. An equal sign as the last character on a
- encoded line indicates such a non-significant ("soft")
- line break in the encoded text.
- */
-
- $os = $this->_createOutputByteStream(true);
- $charStream = $this->_createCharacterStream();
- $is = $this->_createInputByteStream();
- $collection = new Swift_StreamCollector();
-
- $this->_checking(Expectations::create()
- -> one($charStream)->flushContents()
- -> one($charStream)->importByteStream($os)
-
- -> allowing($is)->write(any(), optional()) -> will($collection)
- -> ignoring($is)
-
- -> ignoring($os)
- );
-
- for ($seq = 0; $seq <= 140; ++$seq)
- {
- $this->_checking(Expectations::create()
- -> one($charStream)->readBytes(any()) -> returns(array(ord('a')))
- );
- }
- $this->_checking(Expectations::create()
- -> allowing($charStream)->readBytes(any()) -> returns(false)
- );
-
- $encoder = new Swift_Mime_ContentEncoder_QpContentEncoder($charStream);
- $encoder->encodeByteStream($os, $is);
- $this->assertEqual(str_repeat('a', 75) . "=\r\n" . str_repeat('a', 66), $collection->content);
- }
-
- public function testMaxLineLengthCanBeSpecified()
- {
- $os = $this->_createOutputByteStream(true);
- $charStream = $this->_createCharacterStream();
- $is = $this->_createInputByteStream();
- $collection = new Swift_StreamCollector();
-
- $this->_checking(Expectations::create()
- -> one($charStream)->flushContents()
- -> one($charStream)->importByteStream($os)
-
- -> allowing($is)->write(any(), optional()) -> will($collection)
- -> ignoring($is)
-
- -> ignoring($os)
- );
-
-
- for ($seq = 0; $seq <= 100; ++$seq)
- {
- $this->_checking(Expectations::create()
- -> one($charStream)->readBytes(any()) -> returns(array(ord('a')))
- );
- }
-
- $this->_checking(Expectations::create()
- -> one($charStream)->readBytes(any()) -> returns(false)
- );
-
- $encoder = new Swift_Mime_ContentEncoder_QpContentEncoder($charStream);
- $encoder->encodeByteStream($os, $is, 0, 54);
- $this->assertEqual(str_repeat('a', 53) . "=\r\n" . str_repeat('a', 48), $collection->content);
- }
-
- public function testBytesBelowPermittedRangeAreEncoded()
- {
- /*
- According to Rule (1 & 2)
- */
-
- foreach (range(0, 32) as $ordinal)
- {
- $char = chr($ordinal);
-
- $os = $this->_createOutputByteStream(true);
- $charStream = $this->_createCharacterStream();
- $is = $this->_createInputByteStream();
- $collection = new Swift_StreamCollector();
-
- $this->_checking(Expectations::create()
- -> one($charStream)->flushContents()
- -> one($charStream)->importByteStream($os)
-
- -> one($charStream)->readBytes(any()) -> returns(array($ordinal))
- -> allowing($charStream)->readBytes(any()) -> returns(false)
-
- -> allowing($is)->write(any(), optional()) -> will($collection)
- -> ignoring($is)
-
- -> ignoring($os)
- );
-
- $encoder = new Swift_Mime_ContentEncoder_QpContentEncoder($charStream);
- $encoder->encodeByteStream($os, $is);
- $this->assertEqual(sprintf('=%02X', $ordinal), $collection->content);
- }
- }
-
- public function testDecimalByte61IsEncoded()
- {
- /*
- According to Rule (1 & 2)
- */
-
- $char = chr(61);
-
- $os = $this->_createOutputByteStream(true);
- $charStream = $this->_createCharacterStream();
- $is = $this->_createInputByteStream();
- $collection = new Swift_StreamCollector();
-
- $this->_checking(Expectations::create()
- -> one($charStream)->flushContents()
- -> one($charStream)->importByteStream($os)
-
- -> one($charStream)->readBytes(any()) -> returns(array(61))
- -> allowing($charStream)->readBytes(any()) -> returns(false)
-
- -> allowing($is)->write(any(), optional()) -> will($collection)
- -> ignoring($is)
-
- -> ignoring($os)
- );
-
- $encoder = new Swift_Mime_ContentEncoder_QpContentEncoder($charStream);
- $encoder->encodeByteStream($os, $is);
- $this->assertEqual(sprintf('=%02X', 61), $collection->content);
- }
-
- public function testBytesAbovePermittedRangeAreEncoded()
- {
- /*
- According to Rule (1 & 2)
- */
-
- foreach (range(127, 255) as $ordinal)
- {
- $char = chr($ordinal);
-
- $os = $this->_createOutputByteStream(true);
- $charStream = $this->_createCharacterStream();
- $is = $this->_createInputByteStream();
- $collection = new Swift_StreamCollector();
-
- $this->_checking(Expectations::create()
- -> one($charStream)->flushContents()
- -> one($charStream)->importByteStream($os)
-
- -> one($charStream)->readBytes(any()) -> returns(array($ordinal))
- -> allowing($charStream)->readBytes(any()) -> returns(false)
-
- -> allowing($is)->write(any(), optional()) -> will($collection)
- -> ignoring($is)
-
- -> ignoring($os)
- );
-
- $encoder = new Swift_Mime_ContentEncoder_QpContentEncoder($charStream);
- $encoder->encodeByteStream($os, $is);
- $this->assertEqual(sprintf('=%02X', $ordinal), $collection->content);
- }
- }
-
- public function testFirstLineLengthCanBeDifferent()
- {
- $os = $this->_createOutputByteStream(true);
- $charStream = $this->_createCharacterStream();
- $is = $this->_createInputByteStream();
- $collection = new Swift_StreamCollector();
-
- $this->_checking(Expectations::create()
- -> one($charStream)->flushContents()
- -> one($charStream)->importByteStream($os)
-
- -> allowing($is)->write(any(), optional()) -> will($collection)
- -> ignoring($is)
-
- -> ignoring($os)
- );
-
- for ($seq = 0; $seq <= 140; ++$seq)
- {
- $this->_checking(Expectations::create()
- -> one($charStream)->readBytes(any()) -> returns(array(ord('a')))
- );
- }
- $this->_checking(Expectations::create()
- -> one($charStream)->readBytes(any()) -> returns(false)
- );
-
- $encoder = new Swift_Mime_ContentEncoder_QpContentEncoder($charStream);
- $encoder->encodeByteStream($os, $is, 22);
- $this->assertEqual(
- str_repeat('a', 53) . "=\r\n" . str_repeat('a', 75) . "=\r\n" . str_repeat('a', 13),
- $collection->content
- );
- }
-
- public function testObserverInterfaceCanChangeCharset()
- {
- $stream = $this->_createCharacterStream();
- $this->_checking(Expectations::create()
- -> one($stream)->setCharacterSet('windows-1252')
- -> ignoring($stream)
- );
- $encoder = new Swift_Mime_ContentEncoder_QpContentEncoder($stream);
- $encoder->charsetChanged('windows-1252');
- }
-
- // -- Creation Methods
-
- private function _createCharacterStream($stub = false)
- {
- return $stub
- ? $this->_stub('Swift_CharacterStream')
- : $this->_mock('Swift_CharacterStream')
- ;
- }
-
- private function _createEncoder($charStream)
- {
- return new Swift_Mime_HeaderEncoder_QpHeaderEncoder($charStream);
- }
-
- private function _createOutputByteStream($stub = false)
- {
- return $stub
- ? $this->_stub('Swift_OutputByteStream')
- : $this->_mock('Swift_OutputByteStream')
- ;
- }
-
- private function _createInputByteStream($stub = false)
- {
- return $stub
- ? $this->_stub('Swift_InputByteStream')
- : $this->_mock('Swift_InputByteStream')
- ;
- }
-
- }
|