432 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			432 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| namespace Ratchet\RFC6455\Test\Unit\Messaging;
 | |
| 
 | |
| use Ratchet\RFC6455\Messaging\Frame;
 | |
| use PHPUnit\Framework\TestCase;
 | |
| 
 | |
| /**
 | |
|  * @covers Ratchet\RFC6455\Messaging\Frame
 | |
|  * @todo getMaskingKey, getPayloadStartingByte don't have tests yet
 | |
|  * @todo Could use some clean up in general, I had to rush to fix a bug for a deadline, sorry.
 | |
|  */
 | |
| class FrameTest extends TestCase {
 | |
|     protected $_firstByteFinText    = '10000001';
 | |
| 
 | |
|     protected $_secondByteMaskedSPL = '11111101';
 | |
| 
 | |
|     /** @var Frame */
 | |
|     protected $_frame;
 | |
| 
 | |
|     protected $_packer;
 | |
| 
 | |
|     protected function setUp(): void {
 | |
|         $this->_frame = new Frame;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Encode the fake binary string to send over the wire
 | |
|      * @param string of 1's and 0's
 | |
|      * @return string
 | |
|      */
 | |
|     public static function encode(string $in): string {
 | |
|         if (strlen($in) > 8) {
 | |
|             $out = '';
 | |
|             while (strlen($in) >= 8) {
 | |
|                 $out .= static::encode(substr($in, 0, 8));
 | |
|                 $in   = substr($in, 8);
 | |
|             }
 | |
|             return $out;
 | |
|         }
 | |
|         return chr(bindec($in));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * This is a data provider
 | |
|      * param string The UTF8 message
 | |
|      * param string The WebSocket framed message, then base64_encoded
 | |
|      */
 | |
|     public static function UnframeMessageProvider(): array {
 | |
|         return array(
 | |
|             array('Hello World!', 'gYydAIfa1WXrtvIg0LXvbOP7'),
 | |
|             array('!@#$%^&*()-=_+[]{}\|/.,<>`~', 'gZv+h96r38f9j9vZ+IHWrvOWoayF9oX6gtfRqfKXwOeg'),
 | |
|             array('ಠ_ಠ', 'gYfnSpu5B/g75gf4Ow=='),
 | |
|             array(
 | |
|                 "The quick brown fox jumps over the lazy dog.  All work and no play makes Chris a dull boy.  I'm trying to get past 128 characters for a unit test here...",
 | |
|                 'gf4Amahb14P8M7Kj2S6+4MN7tfHHLLmjzjSvo8IuuvPbe7j1zSn398A+9+/JIa6jzDSwrYh7lu/Ee6Ds2jD34sY/9+3He6fvySL37skwsvCIGL/xwSj34og/ou/Ee7Xs0XX3o+F8uqPcKa7qxjz398d7sObce6fi2y/3sppj9+DAOqXiyy+y8dt7sezae7aj3TW+94gvsvDce7/m2j75rYY='
 | |
|             )
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     public static function underflowProvider(): array {
 | |
|         return array(
 | |
|             array('isFinal', ''),
 | |
|             array('getRsv1', ''),
 | |
|             array('getRsv2', ''),
 | |
|             array('getRsv3', ''),
 | |
|             array('getOpcode', ''),
 | |
|             array('isMasked', '10000001'),
 | |
|             array('getPayloadLength', '10000001'),
 | |
|             array('getPayloadLength', '1000000111111110'),
 | |
|             array('getMaskingKey', '1000000110000111'),
 | |
|             array('getPayload', '100000011000000100011100101010101001100111110100')
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider underflowProvider
 | |
|      */
 | |
|     public function testUnderflowExceptionFromAllTheMethodsMimickingBuffering(string $method, string $bin): void {
 | |
|         $this->expectException(\UnderflowException::class);
 | |
|         if (!empty($bin)) {
 | |
|             $this->_frame->addBuffer(static::encode($bin));
 | |
|         }
 | |
|         call_user_func(array($this->_frame, $method));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * A data provider for testing the first byte of a WebSocket frame
 | |
|      * param bool Given, is the byte indicate this is the final frame
 | |
|      * param int Given, what is the expected opcode
 | |
|      * param string of 0|1 Each character represents a bit in the byte
 | |
|      */
 | |
|     public static function firstByteProvider(): array {
 | |
|         return array(
 | |
|             array(false, false, false, true,   8, '00011000'),
 | |
|             array(true,  false, true,  false, 10, '10101010'),
 | |
|             array(false, false, false, false, 15, '00001111'),
 | |
|             array(true,  false, false, false,  1, '10000001'),
 | |
|             array(true,  true,  true,  true,  15, '11111111'),
 | |
|             array(true,  true,  false, false,  7, '11000111')
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider firstByteProvider
 | |
|      */
 | |
|     public function testFinCodeFromBits(bool $fin, bool $rsv1, bool $rsv2, bool $rsv3, int $opcode, string $bin): void {
 | |
|         $this->_frame->addBuffer(static::encode($bin));
 | |
|         $this->assertEquals($fin, $this->_frame->isFinal());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider firstByteProvider
 | |
|      */
 | |
|     public function testGetRsvFromBits(bool $fin, bool $rsv1, bool $rsv2, bool $rsv3, int $opcode, string $bin): void {
 | |
|         $this->_frame->addBuffer(static::encode($bin));
 | |
|         $this->assertEquals($rsv1, $this->_frame->getRsv1());
 | |
|         $this->assertEquals($rsv2, $this->_frame->getRsv2());
 | |
|         $this->assertEquals($rsv3, $this->_frame->getRsv3());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider firstByteProvider
 | |
|      */
 | |
|     public function testOpcodeFromBits(bool $fin, bool $rsv1, bool $rsv2, bool $rsv3, int $opcode, string $bin): void {
 | |
|         $this->_frame->addBuffer(static::encode($bin));
 | |
|         $this->assertEquals($opcode, $this->_frame->getOpcode());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider UnframeMessageProvider
 | |
|      */
 | |
|     public function testFinCodeFromFullMessage(string $msg, string $encoded): void {
 | |
|         $this->_frame->addBuffer(base64_decode($encoded));
 | |
|         $this->assertTrue($this->_frame->isFinal());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider UnframeMessageProvider
 | |
|      */
 | |
|     public function testOpcodeFromFullMessage(string $msg, string $encoded): void {
 | |
|         $this->_frame->addBuffer(base64_decode($encoded));
 | |
|         $this->assertEquals(1, $this->_frame->getOpcode());
 | |
|     }
 | |
| 
 | |
|     public static function payloadLengthDescriptionProvider(): array {
 | |
|         return array(
 | |
|             array(7,  '01110101'),
 | |
|             array(7,  '01111101'),
 | |
|             array(23, '01111110'),
 | |
|             array(71, '01111111'),
 | |
|             array(7,  '00000000'), // Should this throw an exception?  Can a payload be empty?
 | |
|             array(7,  '00000001')
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider payloadLengthDescriptionProvider
 | |
|      */
 | |
|     public function testFirstPayloadDesignationValue(int $bits, string $bin): void {
 | |
|         $this->_frame->addBuffer(static::encode($this->_firstByteFinText));
 | |
|         $this->_frame->addBuffer(static::encode($bin));
 | |
|         $ref = new \ReflectionClass($this->_frame);
 | |
|         $cb  = $ref->getMethod('getFirstPayloadVal');
 | |
|         $cb->setAccessible(true);
 | |
|         $this->assertEquals(bindec($bin), $cb->invoke($this->_frame));
 | |
|     }
 | |
| 
 | |
|     public function testFirstPayloadValUnderflow(): void {
 | |
|         $ref = new \ReflectionClass($this->_frame);
 | |
|         $cb  = $ref->getMethod('getFirstPayloadVal');
 | |
|         $cb->setAccessible(true);
 | |
|         $this->expectException(\UnderflowException::class);
 | |
|         $cb->invoke($this->_frame);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider payloadLengthDescriptionProvider
 | |
|      */
 | |
|     public function testDetermineHowManyBitsAreUsedToDescribePayload(int $expected_bits, string $bin): void {
 | |
|         $this->_frame->addBuffer(static::encode($this->_firstByteFinText));
 | |
|         $this->_frame->addBuffer(static::encode($bin));
 | |
|         $ref = new \ReflectionClass($this->_frame);
 | |
|         $cb  = $ref->getMethod('getNumPayloadBits');
 | |
|         $cb->setAccessible(true);
 | |
|         $this->assertEquals($expected_bits, $cb->invoke($this->_frame));
 | |
|     }
 | |
| 
 | |
|     public function testgetNumPayloadBitsUnderflow(): void {
 | |
|         $ref = new \ReflectionClass($this->_frame);
 | |
|         $cb  = $ref->getMethod('getNumPayloadBits');
 | |
|         $cb->setAccessible(true);
 | |
|         $this->expectException(\UnderflowException::class);
 | |
|         $cb->invoke($this->_frame);
 | |
|     }
 | |
| 
 | |
|     public function secondByteProvider(): array {
 | |
|         return array(
 | |
|             array(true,   1, '10000001'),
 | |
|             array(false,  1, '00000001'),
 | |
|             array(true, 125, $this->_secondByteMaskedSPL)
 | |
|         );
 | |
|     }
 | |
|     /**
 | |
|      * @dataProvider secondByteProvider
 | |
|      */
 | |
|     public function testIsMaskedReturnsExpectedValue(bool $masked, int $payload_length, string $bin): void {
 | |
|         $this->_frame->addBuffer(static::encode($this->_firstByteFinText));
 | |
|         $this->_frame->addBuffer(static::encode($bin));
 | |
|         $this->assertEquals($masked, $this->_frame->isMasked());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider UnframeMessageProvider
 | |
|      */
 | |
|     public function testIsMaskedFromFullMessage(string $msg, string $encoded): void {
 | |
|         $this->_frame->addBuffer(base64_decode($encoded));
 | |
|         $this->assertTrue($this->_frame->isMasked());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider secondByteProvider
 | |
|      */
 | |
|     public function testGetPayloadLengthWhenOnlyFirstFrameIsUsed(bool $masked, int $payload_length, string $bin): void {
 | |
|         $this->_frame->addBuffer(static::encode($this->_firstByteFinText));
 | |
|         $this->_frame->addBuffer(static::encode($bin));
 | |
|         $this->assertEquals($payload_length, $this->_frame->getPayloadLength());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider UnframeMessageProvider
 | |
|      * @todo Not yet testing when second additional payload length descriptor
 | |
|      */
 | |
|     public function testGetPayloadLengthFromFullMessage(string $msg, string $encoded): void {
 | |
|         $this->_frame->addBuffer(base64_decode($encoded));
 | |
|         $this->assertEquals(strlen($msg), $this->_frame->getPayloadLength());
 | |
|     }
 | |
| 
 | |
|     public function maskingKeyProvider(): array {
 | |
|         $frame = new Frame;
 | |
|         return array(
 | |
|             array($frame->generateMaskingKey()),
 | |
|             array($frame->generateMaskingKey()),
 | |
|             array($frame->generateMaskingKey())
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider maskingKeyProvider
 | |
|      * @todo I I wrote the dataProvider incorrectly, skipping for now
 | |
|      */
 | |
|     public function testGetMaskingKey(string $mask): void {
 | |
|         $this->_frame->addBuffer(static::encode($this->_firstByteFinText));
 | |
|         $this->_frame->addBuffer(static::encode($this->_secondByteMaskedSPL));
 | |
|         $this->_frame->addBuffer($mask);
 | |
|         $this->assertEquals($mask, $this->_frame->getMaskingKey());
 | |
|     }
 | |
| 
 | |
|     public function testGetMaskingKeyOnUnmaskedPayload(): void {
 | |
|         $frame = new Frame('Hello World!');
 | |
|         $this->assertEquals('', $frame->getMaskingKey());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider UnframeMessageProvider
 | |
|      * @todo Move this test to bottom as it requires all methods of the class
 | |
|      */
 | |
|     public function testUnframeFullMessage(string $unframed, string $base_framed): void {
 | |
|         $this->_frame->addBuffer(base64_decode($base_framed));
 | |
|         $this->assertEquals($unframed, $this->_frame->getPayload());
 | |
|     }
 | |
| 
 | |
|     public static function messageFragmentProvider(): array {
 | |
|         return array(
 | |
|             array(false, '', '', '', '', '')
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider UnframeMessageProvider
 | |
|      */
 | |
|     public function testCheckPiecingTogetherMessage(string $msg, string $encoded): void {
 | |
|         $framed = base64_decode($encoded);
 | |
|         for ($i = 0, $len = strlen($framed);$i < $len; $i++) {
 | |
|             $this->_frame->addBuffer(substr($framed, $i, 1));
 | |
|         }
 | |
|         $this->assertEquals($msg, $this->_frame->getPayload());
 | |
|     }
 | |
| 
 | |
|     public function testLongCreate(): void {
 | |
|         $len = 65525;
 | |
|         $pl  = $this->generateRandomString($len);
 | |
|         $frame = new Frame($pl, true, Frame::OP_PING);
 | |
|         $this->assertTrue($frame->isFinal());
 | |
|         $this->assertEquals(Frame::OP_PING, $frame->getOpcode());
 | |
|         $this->assertFalse($frame->isMasked());
 | |
|         $this->assertEquals($len, $frame->getPayloadLength());
 | |
|         $this->assertEquals($pl, $frame->getPayload());
 | |
|     }
 | |
| 
 | |
|     public function testReallyLongCreate(): void {
 | |
|         $len = 65575;
 | |
|         $frame = new Frame($this->generateRandomString($len));
 | |
|         $this->assertEquals($len, $frame->getPayloadLength());
 | |
|     }
 | |
| 
 | |
|     public function testExtractOverflow(): void {
 | |
|         $string1 = $this->generateRandomString();
 | |
|         $frame1  = new Frame($string1);
 | |
|         $string2 = $this->generateRandomString();
 | |
|         $frame2  = new Frame($string2);
 | |
|         $cat = new Frame;
 | |
|         $cat->addBuffer($frame1->getContents() . $frame2->getContents());
 | |
|         $this->assertEquals($frame1->getContents(), $cat->getContents());
 | |
|         $this->assertEquals($string1, $cat->getPayload());
 | |
|         $uncat = new Frame;
 | |
|         $uncat->addBuffer($cat->extractOverflow());
 | |
|         $this->assertEquals($string1, $cat->getPayload());
 | |
|         $this->assertEquals($string2, $uncat->getPayload());
 | |
|     }
 | |
| 
 | |
|     public function testEmptyExtractOverflow(): void {
 | |
|         $string = $this->generateRandomString();
 | |
|         $frame  = new Frame($string);
 | |
|         $this->assertEquals($string, $frame->getPayload());
 | |
|         $this->assertEquals('', $frame->extractOverflow());
 | |
|         $this->assertEquals($string, $frame->getPayload());
 | |
|     }
 | |
| 
 | |
|     public function testGetContents(): void {
 | |
|         $msg = 'The quick brown fox jumps over the lazy dog.';
 | |
|         $frame1 = new Frame($msg);
 | |
|         $frame2 = new Frame($msg);
 | |
|         $frame2->maskPayload();
 | |
|         $this->assertNotEquals($frame1->getContents(), $frame2->getContents());
 | |
|         $this->assertEquals(strlen($frame1->getContents()) + 4, strlen($frame2->getContents()));
 | |
|     }
 | |
| 
 | |
|     public function testMasking(): void {
 | |
|         $msg   = 'The quick brown fox jumps over the lazy dog.';
 | |
|         $frame = new Frame($msg);
 | |
|         $frame->maskPayload();
 | |
|         $this->assertTrue($frame->isMasked());
 | |
|         $this->assertEquals($msg, $frame->getPayload());
 | |
|     }
 | |
| 
 | |
|     public function testUnMaskPayload(): void {
 | |
|         $string = $this->generateRandomString();
 | |
|         $frame  = new Frame($string);
 | |
|         $frame->maskPayload()->unMaskPayload();
 | |
|         $this->assertFalse($frame->isMasked());
 | |
|         $this->assertEquals($string, $frame->getPayload());
 | |
|     }
 | |
| 
 | |
|     public function testGenerateMaskingKey(): void {
 | |
|         $dupe = false;
 | |
|         $done = array();
 | |
|         for ($i = 0; $i < 10; $i++) {
 | |
|             $new = $this->_frame->generateMaskingKey();
 | |
|             if (in_array($new, $done)) {
 | |
|                 $dupe = true;
 | |
|             }
 | |
|             $done[] = $new;
 | |
|         }
 | |
|         $this->assertEquals(4, strlen($new));
 | |
|         $this->assertFalse($dupe);
 | |
|     }
 | |
| 
 | |
|     public function testGivenMaskIsValid(): void {
 | |
|         $this->expectException(\InvalidArgumentException::class);
 | |
|         $this->_frame->maskPayload('hello world');
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @requires extension mbstring
 | |
|      */
 | |
|     public function testGivenMaskIsValidAscii(): void {
 | |
|         $this->expectException(\OutOfBoundsException::class);
 | |
|         $this->_frame->maskPayload('x✖');
 | |
|     }
 | |
| 
 | |
|     protected function generateRandomString(int $length = 10, bool $addSpaces = true, bool $addNumbers = true): string {
 | |
|         $characters = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ!"$%&/()=[]{}'; // ยง
 | |
|         $useChars = array();
 | |
|         for($i = 0; $i < $length; $i++) {
 | |
|             $useChars[] = $characters[mt_rand(0, strlen($characters) - 1)];
 | |
|         }
 | |
|         if($addSpaces === true) {
 | |
|             array_push($useChars, ' ', ' ', ' ', ' ', ' ', ' ');
 | |
|         }
 | |
|         if($addNumbers === true) {
 | |
|             array_push($useChars, rand(0, 9), rand(0, 9), rand(0, 9));
 | |
|         }
 | |
|         shuffle($useChars);
 | |
|         $randomString = trim(implode('', $useChars));
 | |
|         $randomString = substr($randomString, 0, $length);
 | |
|         return $randomString;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * There was a frame boundary issue when the first 3 bytes of a frame with a payload greater than
 | |
|      * 126 was added to the frame buffer and then Frame::getPayloadLength was called. It would cause the frame
 | |
|      * to set the payload length to 126 and then not recalculate it once the full length information was available.
 | |
|      *
 | |
|      * This is fixed by setting the defPayLen back to -1 before the underflow exception is thrown.
 | |
|      */
 | |
|     public function testFrameDeliveredOneByteAtATime(): void {
 | |
|         $startHeader = "\x01\x7e\x01\x00"; // header for a text frame of 256 - non-final
 | |
|         $framePayload = str_repeat("*", 256);
 | |
|         $rawOverflow = "xyz";
 | |
|         $rawFrame = $startHeader . $framePayload . $rawOverflow;
 | |
|         $frame = new Frame();
 | |
|         $payloadLen = 256;
 | |
|         for ($i = 0; $i < strlen($rawFrame); $i++) {
 | |
|             $frame->addBuffer($rawFrame[$i]);
 | |
|             try {
 | |
|                 // payloadLen will
 | |
|                 $payloadLen = $frame->getPayloadLength();
 | |
|             } catch (\UnderflowException $e) {
 | |
|                 if ($i > 2) { // we should get an underflow on 0,1,2
 | |
|                     $this->fail("Underflow exception when the frame length should be available");
 | |
|                 }
 | |
|             }
 | |
|             if ($payloadLen !== 256) {
 | |
|                 $this->fail("Payload length of " . $payloadLen . " should have been 256.");
 | |
|             }
 | |
|         }
 | |
|         // make sure the overflow is good
 | |
|         $this->assertEquals($rawOverflow, $frame->extractOverflow());
 | |
|     }
 | |
| }
 | 
