| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338 |
- <?php declare(strict_types=1);
- /*
- * This file is part of the Monolog package.
- *
- * (c) Jordi Boggiano <j.boggiano@seld.be>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace Monolog\Handler;
- use Monolog\Test\TestCase;
- use Monolog\Level;
- use PHPUnit\Framework\MockObject\MockObject;
- /**
- * @author Pablo de Leon Belloc <pablolb@gmail.com>
- */
- class SocketHandlerTest extends TestCase
- {
- private SocketHandler&MockObject $handler;
- /**
- * @var resource
- */
- private $res;
- public function tearDown(): void
- {
- parent::tearDown();
- unset($this->res);
- unset($this->handler);
- }
- public function testInvalidHostname()
- {
- $this->expectException(\UnexpectedValueException::class);
- $handler = $this->createHandler('garbage://here');
- $handler->handle($this->getRecord(Level::Warning, 'data'));
- }
- public function testBadConnectionTimeout()
- {
- $this->expectException(\InvalidArgumentException::class);
- $handler = $this->createHandler('localhost:1234');
- $handler->setConnectionTimeout(-1);
- }
- public function testSetConnectionTimeout()
- {
- $handler = $this->createHandler('localhost:1234');
- $handler->setConnectionTimeout(10.1);
- $this->assertEquals(10.1, $handler->getConnectionTimeout());
- }
- public function testBadTimeout()
- {
- $this->expectException(\InvalidArgumentException::class);
- $handler = $this->createHandler('localhost:1234');
- $handler->setTimeout(-1);
- }
- public function testSetTimeout()
- {
- $handler = $this->createHandler('localhost:1234');
- $handler->setTimeout(10.25);
- $this->assertEquals(10.25, $handler->getTimeout());
- }
- public function testSetWritingTimeout()
- {
- $handler = $this->createHandler('localhost:1234');
- $handler->setWritingTimeout(10.25);
- $this->assertEquals(10.25, $handler->getWritingTimeout());
- }
- public function testSetChunkSize()
- {
- $handler = $this->createHandler('localhost:1234');
- $handler->setChunkSize(1025);
- $this->assertEquals(1025, $handler->getChunkSize());
- }
- public function testSetConnectionString()
- {
- $handler = $this->createHandler('tcp://localhost:9090');
- $this->assertEquals('tcp://localhost:9090', $handler->getConnectionString());
- }
- public function testExceptionIsThrownOnFsockopenError()
- {
- $this->setMockHandler(['fsockopen']);
- $this->handler->expects($this->once())
- ->method('fsockopen')
- ->willReturn(false);
- $this->expectException(\UnexpectedValueException::class);
- $this->writeRecord('Hello world');
- }
- public function testExceptionIsThrownOnPfsockopenError()
- {
- $this->setMockHandler(['pfsockopen']);
- $this->handler->expects($this->once())
- ->method('pfsockopen')
- ->willReturn(false);
- $this->handler->setPersistent(true);
- $this->expectException(\UnexpectedValueException::class);
- $this->writeRecord('Hello world');
- }
- public function testExceptionIsThrownIfCannotSetTimeout()
- {
- $this->setMockHandler(['streamSetTimeout']);
- $this->handler->expects($this->once())
- ->method('streamSetTimeout')
- ->willReturn(false);
- $this->expectException(\UnexpectedValueException::class);
- $this->writeRecord('Hello world');
- }
- public function testExceptionIsThrownIfCannotSetChunkSize()
- {
- $this->setMockHandler(['streamSetChunkSize']);
- $this->handler->setChunkSize(8192);
- $this->handler->expects($this->once())
- ->method('streamSetChunkSize')
- ->willReturn(false);
- $this->expectException(\UnexpectedValueException::class);
- $this->writeRecord('Hello world');
- }
- public function testWriteFailsOnIfFwriteReturnsFalse()
- {
- $this->setMockHandler(['fwrite']);
- $callback = function ($arg) {
- $map = [
- 'Hello world' => 6,
- 'world' => false,
- ];
- return $map[$arg];
- };
- $this->handler->expects($this->exactly(2))
- ->method('fwrite')
- ->willReturnCallback($callback);
- $this->expectException(\RuntimeException::class);
- $this->writeRecord('Hello world');
- }
- public function testWriteFailsIfStreamTimesOut()
- {
- $this->setMockHandler(['fwrite', 'streamGetMetadata']);
- $callback = function ($arg) {
- $map = [
- 'Hello world' => 6,
- 'world' => 5,
- ];
- return $map[$arg];
- };
- $this->handler->expects($this->exactly(1))
- ->method('fwrite')
- ->willReturnCallback($callback);
- $this->handler->expects($this->exactly(1))
- ->method('streamGetMetadata')
- ->willReturn(['timed_out' => true]);
- $this->expectException(\RuntimeException::class);
- $this->writeRecord('Hello world');
- }
- public function testWriteFailsOnIncompleteWrite()
- {
- $this->setMockHandler(['fwrite', 'streamGetMetadata']);
- $res = $this->res;
- $callback = function ($string) use ($res) {
- fclose($res);
- return \strlen('Hello');
- };
- $this->handler->expects($this->exactly(1))
- ->method('fwrite')
- ->willReturnCallback($callback);
- $this->handler->expects($this->exactly(1))
- ->method('streamGetMetadata')
- ->willReturn(['timed_out' => false]);
- $this->expectException(\RuntimeException::class);
- $this->writeRecord('Hello world');
- }
- public function testWriteWithMemoryFile()
- {
- $this->setMockHandler();
- $this->writeRecord('test1');
- $this->writeRecord('test2');
- $this->writeRecord('test3');
- fseek($this->res, 0);
- $this->assertEquals('test1test2test3', fread($this->res, 1024));
- }
- public function testWriteWithMock()
- {
- $this->setMockHandler(['fwrite']);
- $callback = function ($arg) {
- $map = [
- 'Hello world' => 6,
- 'world' => 5,
- ];
- return $map[$arg];
- };
- $this->handler->expects($this->exactly(2))
- ->method('fwrite')
- ->willReturnCallback($callback);
- $this->writeRecord('Hello world');
- }
- public function testClose()
- {
- $this->setMockHandler();
- $this->writeRecord('Hello world');
- $this->assertIsResource($this->res);
- $this->handler->close();
- $this->assertFalse(\is_resource($this->res), "Expected resource to be closed after closing handler");
- }
- public function testCloseDoesNotClosePersistentSocket()
- {
- $this->setMockHandler();
- $this->handler->setPersistent(true);
- $this->writeRecord('Hello world');
- $this->assertTrue(\is_resource($this->res));
- $this->handler->close();
- $this->assertTrue(\is_resource($this->res));
- }
- public function testAvoidInfiniteLoopWhenNoDataIsWrittenForAWritingTimeoutSeconds()
- {
- $this->setMockHandler(['fwrite', 'streamGetMetadata']);
- $this->handler->expects($this->any())
- ->method('fwrite')
- ->willReturn(0);
- $this->handler->expects($this->any())
- ->method('streamGetMetadata')
- ->willReturn(['timed_out' => false]);
- $this->handler->setWritingTimeout(1);
- $this->expectException(\RuntimeException::class);
- $this->writeRecord('Hello world');
- }
- private function createHandler(string $connectionString): SocketHandler
- {
- $handler = new SocketHandler($connectionString);
- $handler->setFormatter($this->getIdentityFormatter());
- return $handler;
- }
- private function writeRecord($string)
- {
- $this->handler->handle($this->getRecord(Level::Warning, $string));
- }
- private function setMockHandler(array $methods = [])
- {
- $this->res = fopen('php://memory', 'a');
- $defaultMethods = ['fsockopen', 'pfsockopen', 'streamSetTimeout', 'streamSetChunkSize'];
- $newMethods = array_diff($methods, $defaultMethods);
- $finalMethods = array_merge($defaultMethods, $newMethods);
- $this->handler = $this->getMockBuilder('Monolog\Handler\SocketHandler')
- ->onlyMethods($finalMethods)
- ->setConstructorArgs(['localhost:1234'])
- ->getMock();
- if (!\in_array('fsockopen', $methods)) {
- $this->handler->expects($this->any())
- ->method('fsockopen')
- ->willReturn($this->res);
- }
- if (!\in_array('pfsockopen', $methods)) {
- $this->handler->expects($this->any())
- ->method('pfsockopen')
- ->willReturn($this->res);
- }
- if (!\in_array('streamSetTimeout', $methods)) {
- $this->handler->expects($this->any())
- ->method('streamSetTimeout')
- ->willReturn(true);
- }
- if (!\in_array('streamSetChunkSize', $methods)) {
- $this->handler->expects($this->any())
- ->method('streamSetChunkSize')
- ->willReturn(8192);
- }
- $this->handler->setFormatter($this->getIdentityFormatter());
- }
- }
|