_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()); } }