mirror of
https://github.com/Seldaek/monolog.git
synced 2025-10-24 10:06:08 +02:00
194 lines
6.5 KiB
PHP
194 lines
6.5 KiB
PHP
<?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;
|
|
|
|
class ProcessHandlerTest extends TestCase
|
|
{
|
|
/**
|
|
* Dummy command to be used by tests that should not fail due to the command.
|
|
*
|
|
* @var string
|
|
*/
|
|
const DUMMY_COMMAND = 'echo';
|
|
|
|
/**
|
|
* @covers Monolog\Handler\ProcessHandler::__construct
|
|
* @covers Monolog\Handler\ProcessHandler::guardAgainstInvalidCommand
|
|
* @covers Monolog\Handler\ProcessHandler::guardAgainstInvalidCwd
|
|
* @covers Monolog\Handler\ProcessHandler::write
|
|
* @covers Monolog\Handler\ProcessHandler::ensureProcessIsStarted
|
|
* @covers Monolog\Handler\ProcessHandler::startProcess
|
|
* @covers Monolog\Handler\ProcessHandler::handleStartupErrors
|
|
*/
|
|
public function testWriteOpensProcessAndWritesToStdInOfProcess()
|
|
{
|
|
$fixtures = [
|
|
'chuck norris',
|
|
'foobar1337',
|
|
];
|
|
|
|
$mockBuilder = $this->getMockBuilder('Monolog\Handler\ProcessHandler');
|
|
$mockBuilder->onlyMethods(['writeProcessInput']);
|
|
// using echo as command, as it is most probably available
|
|
$mockBuilder->setConstructorArgs([self::DUMMY_COMMAND]);
|
|
|
|
$handler = $mockBuilder->getMock();
|
|
|
|
$handler->expects($this->exactly(2))
|
|
->method('writeProcessInput')
|
|
->withConsecutive([$this->stringContains($fixtures[0])], [$this->stringContains($fixtures[1])]);
|
|
|
|
/** @var ProcessHandler $handler */
|
|
$handler->handle($this->getRecord(Level::Warning, $fixtures[0]));
|
|
$handler->handle($this->getRecord(Level::Error, $fixtures[1]));
|
|
}
|
|
|
|
/**
|
|
* Data provider for invalid commands.
|
|
*/
|
|
public function invalidCommandProvider(): array
|
|
{
|
|
return [
|
|
[1337, 'TypeError'],
|
|
['', 'InvalidArgumentException'],
|
|
[null, 'TypeError'],
|
|
[fopen('php://input', 'r'), 'TypeError'],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* @dataProvider invalidCommandProvider
|
|
* @param mixed $invalidCommand
|
|
* @covers Monolog\Handler\ProcessHandler::guardAgainstInvalidCommand
|
|
*/
|
|
public function testConstructWithInvalidCommandThrowsInvalidArgumentException($invalidCommand, $expectedExcep)
|
|
{
|
|
$this->expectException($expectedExcep);
|
|
new ProcessHandler($invalidCommand, Level::Debug);
|
|
}
|
|
|
|
/**
|
|
* Data provider for invalid CWDs.
|
|
*/
|
|
public function invalidCwdProvider(): array
|
|
{
|
|
return [
|
|
[1337, 'TypeError'],
|
|
['', 'InvalidArgumentException'],
|
|
[fopen('php://input', 'r'), 'TypeError'],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* @dataProvider invalidCwdProvider
|
|
* @param mixed $invalidCwd
|
|
* @covers Monolog\Handler\ProcessHandler::guardAgainstInvalidCwd
|
|
*/
|
|
public function testConstructWithInvalidCwdThrowsInvalidArgumentException($invalidCwd, $expectedExcep)
|
|
{
|
|
$this->expectException($expectedExcep);
|
|
new ProcessHandler(self::DUMMY_COMMAND, Level::Debug, true, $invalidCwd);
|
|
}
|
|
|
|
/**
|
|
* @covers Monolog\Handler\ProcessHandler::__construct
|
|
* @covers Monolog\Handler\ProcessHandler::guardAgainstInvalidCwd
|
|
*/
|
|
public function testConstructWithValidCwdWorks()
|
|
{
|
|
$handler = new ProcessHandler(self::DUMMY_COMMAND, Level::Debug, true, sys_get_temp_dir());
|
|
$this->assertInstanceOf(
|
|
'Monolog\Handler\ProcessHandler',
|
|
$handler,
|
|
'Constructed handler is not a ProcessHandler.'
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @covers Monolog\Handler\ProcessHandler::handleStartupErrors
|
|
*/
|
|
public function testStartupWithFailingToSelectErrorStreamThrowsUnexpectedValueException()
|
|
{
|
|
$mockBuilder = $this->getMockBuilder('Monolog\Handler\ProcessHandler');
|
|
$mockBuilder->onlyMethods(['selectErrorStream']);
|
|
$mockBuilder->setConstructorArgs([self::DUMMY_COMMAND]);
|
|
|
|
$handler = $mockBuilder->getMock();
|
|
|
|
$handler->expects($this->once())
|
|
->method('selectErrorStream')
|
|
->will($this->returnValue(false));
|
|
|
|
$this->expectException(\UnexpectedValueException::class);
|
|
/** @var ProcessHandler $handler */
|
|
$handler->handle($this->getRecord(Level::Warning, 'stream failing, whoops'));
|
|
}
|
|
|
|
/**
|
|
* @covers Monolog\Handler\ProcessHandler::handleStartupErrors
|
|
* @covers Monolog\Handler\ProcessHandler::selectErrorStream
|
|
*/
|
|
public function testStartupWithErrorsThrowsUnexpectedValueException()
|
|
{
|
|
$handler = new ProcessHandler('>&2 echo "some fake error message"');
|
|
|
|
$this->expectException(\UnexpectedValueException::class);
|
|
|
|
$handler->handle($this->getRecord(Level::Warning, 'some warning in the house'));
|
|
}
|
|
|
|
/**
|
|
* @covers Monolog\Handler\ProcessHandler::write
|
|
*/
|
|
public function testWritingWithErrorsOnStdOutOfProcessThrowsInvalidArgumentException()
|
|
{
|
|
$mockBuilder = $this->getMockBuilder('Monolog\Handler\ProcessHandler');
|
|
$mockBuilder->onlyMethods(['readProcessErrors']);
|
|
// using echo as command, as it is most probably available
|
|
$mockBuilder->setConstructorArgs([self::DUMMY_COMMAND]);
|
|
|
|
$handler = $mockBuilder->getMock();
|
|
|
|
$handler->expects($this->exactly(2))
|
|
->method('readProcessErrors')
|
|
->willReturnOnConsecutiveCalls('', $this->returnValue('some fake error message here'));
|
|
|
|
$this->expectException(\UnexpectedValueException::class);
|
|
/** @var ProcessHandler $handler */
|
|
$handler->handle($this->getRecord(Level::Warning, 'some test stuff'));
|
|
}
|
|
|
|
/**
|
|
* @covers Monolog\Handler\ProcessHandler::close
|
|
*/
|
|
public function testCloseClosesProcess()
|
|
{
|
|
$class = new \ReflectionClass('Monolog\Handler\ProcessHandler');
|
|
$property = $class->getProperty('process');
|
|
$property->setAccessible(true);
|
|
|
|
$handler = new ProcessHandler(self::DUMMY_COMMAND);
|
|
$handler->handle($this->getRecord(Level::Warning, '21 is only the half truth'));
|
|
|
|
$process = $property->getValue($handler);
|
|
$this->assertTrue(is_resource($process), 'Process is not running although it should.');
|
|
|
|
$handler->close();
|
|
|
|
$process = $property->getValue($handler);
|
|
$this->assertFalse(is_resource($process), 'Process is still running although it should not.');
|
|
}
|
|
}
|