ProcessHandlerTest.php 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. <?php declare(strict_types=1);
  2. /*
  3. * This file is part of the Monolog package.
  4. *
  5. * (c) Jordi Boggiano <j.boggiano@seld.be>
  6. *
  7. * For the full copyright and license information, please view the LICENSE
  8. * file that was distributed with this source code.
  9. */
  10. namespace Monolog\Handler;
  11. use Monolog\Test\TestCase;
  12. use Monolog\Logger;
  13. class ProcessHandlerTest extends TestCase
  14. {
  15. /**
  16. * Dummy command to be used by tests that should not fail due to the command.
  17. *
  18. * @var string
  19. */
  20. const DUMMY_COMMAND = 'echo';
  21. /**
  22. * @covers Monolog\Handler\ProcessHandler::__construct
  23. * @covers Monolog\Handler\ProcessHandler::guardAgainstInvalidCommand
  24. * @covers Monolog\Handler\ProcessHandler::guardAgainstInvalidCwd
  25. * @covers Monolog\Handler\ProcessHandler::write
  26. * @covers Monolog\Handler\ProcessHandler::ensureProcessIsStarted
  27. * @covers Monolog\Handler\ProcessHandler::startProcess
  28. * @covers Monolog\Handler\ProcessHandler::handleStartupErrors
  29. */
  30. public function testWriteOpensProcessAndWritesToStdInOfProcess()
  31. {
  32. $fixtures = [
  33. 'chuck norris',
  34. 'foobar1337',
  35. ];
  36. $mockBuilder = $this->getMockBuilder('Monolog\Handler\ProcessHandler');
  37. $mockBuilder->setMethods(['writeProcessInput']);
  38. // using echo as command, as it is most probably available
  39. $mockBuilder->setConstructorArgs([self::DUMMY_COMMAND]);
  40. $handler = $mockBuilder->getMock();
  41. $handler->expects($this->exactly(2))
  42. ->method('writeProcessInput')
  43. ->withConsecutive($this->stringContains($fixtures[0]), $this->stringContains($fixtures[1]));
  44. /** @var ProcessHandler $handler */
  45. $handler->handle($this->getRecord(Logger::WARNING, $fixtures[0]));
  46. $handler->handle($this->getRecord(Logger::ERROR, $fixtures[1]));
  47. }
  48. /**
  49. * Data provider for invalid commands.
  50. *
  51. * @return array
  52. */
  53. public function invalidCommandProvider()
  54. {
  55. return [
  56. [1337, 'TypeError'],
  57. ['', 'InvalidArgumentException'],
  58. [null, 'TypeError'],
  59. [fopen('php://input', 'r'), 'TypeError'],
  60. ];
  61. }
  62. /**
  63. * @dataProvider invalidCommandProvider
  64. * @param mixed $invalidCommand
  65. * @covers Monolog\Handler\ProcessHandler::guardAgainstInvalidCommand
  66. */
  67. public function testConstructWithInvalidCommandThrowsInvalidArgumentException($invalidCommand, $expectedExcep)
  68. {
  69. $this->expectException($expectedExcep);
  70. new ProcessHandler($invalidCommand, Logger::DEBUG);
  71. }
  72. /**
  73. * Data provider for invalid CWDs.
  74. *
  75. * @return array
  76. */
  77. public function invalidCwdProvider()
  78. {
  79. return [
  80. [1337, 'TypeError'],
  81. ['', 'InvalidArgumentException'],
  82. [fopen('php://input', 'r'), 'TypeError'],
  83. ];
  84. }
  85. /**
  86. * @dataProvider invalidCwdProvider
  87. * @param mixed $invalidCwd
  88. * @covers Monolog\Handler\ProcessHandler::guardAgainstInvalidCwd
  89. */
  90. public function testConstructWithInvalidCwdThrowsInvalidArgumentException($invalidCwd, $expectedExcep)
  91. {
  92. $this->expectException($expectedExcep);
  93. new ProcessHandler(self::DUMMY_COMMAND, Logger::DEBUG, true, $invalidCwd);
  94. }
  95. /**
  96. * @covers Monolog\Handler\ProcessHandler::__construct
  97. * @covers Monolog\Handler\ProcessHandler::guardAgainstInvalidCwd
  98. */
  99. public function testConstructWithValidCwdWorks()
  100. {
  101. $handler = new ProcessHandler(self::DUMMY_COMMAND, Logger::DEBUG, true, sys_get_temp_dir());
  102. $this->assertInstanceOf(
  103. 'Monolog\Handler\ProcessHandler',
  104. $handler,
  105. 'Constructed handler is not a ProcessHandler.'
  106. );
  107. }
  108. /**
  109. * @covers Monolog\Handler\ProcessHandler::handleStartupErrors
  110. */
  111. public function testStartupWithFailingToSelectErrorStreamThrowsUnexpectedValueException()
  112. {
  113. $mockBuilder = $this->getMockBuilder('Monolog\Handler\ProcessHandler');
  114. $mockBuilder->setMethods(['selectErrorStream']);
  115. $mockBuilder->setConstructorArgs([self::DUMMY_COMMAND]);
  116. $handler = $mockBuilder->getMock();
  117. $handler->expects($this->once())
  118. ->method('selectErrorStream')
  119. ->will($this->returnValue(false));
  120. $this->expectException('\UnexpectedValueException');
  121. /** @var ProcessHandler $handler */
  122. $handler->handle($this->getRecord(Logger::WARNING, 'stream failing, whoops'));
  123. }
  124. /**
  125. * @covers Monolog\Handler\ProcessHandler::handleStartupErrors
  126. * @covers Monolog\Handler\ProcessHandler::selectErrorStream
  127. */
  128. public function testStartupWithErrorsThrowsUnexpectedValueException()
  129. {
  130. $handler = new ProcessHandler('>&2 echo "some fake error message"');
  131. $this->expectException('\UnexpectedValueException');
  132. $handler->handle($this->getRecord(Logger::WARNING, 'some warning in the house'));
  133. }
  134. /**
  135. * @covers Monolog\Handler\ProcessHandler::write
  136. */
  137. public function testWritingWithErrorsOnStdOutOfProcessThrowsInvalidArgumentException()
  138. {
  139. $mockBuilder = $this->getMockBuilder('Monolog\Handler\ProcessHandler');
  140. $mockBuilder->setMethods(['readProcessErrors']);
  141. // using echo as command, as it is most probably available
  142. $mockBuilder->setConstructorArgs([self::DUMMY_COMMAND]);
  143. $handler = $mockBuilder->getMock();
  144. $handler->expects($this->exactly(2))
  145. ->method('readProcessErrors')
  146. ->willReturnOnConsecutiveCalls('', $this->returnValue('some fake error message here'));
  147. $this->expectException('\UnexpectedValueException');
  148. /** @var ProcessHandler $handler */
  149. $handler->handle($this->getRecord(Logger::WARNING, 'some test stuff'));
  150. }
  151. /**
  152. * @covers Monolog\Handler\ProcessHandler::close
  153. */
  154. public function testCloseClosesProcess()
  155. {
  156. $class = new \ReflectionClass('Monolog\Handler\ProcessHandler');
  157. $property = $class->getProperty('process');
  158. $property->setAccessible(true);
  159. $handler = new ProcessHandler(self::DUMMY_COMMAND);
  160. $handler->handle($this->getRecord(Logger::WARNING, '21 is only the half truth'));
  161. $process = $property->getValue($handler);
  162. $this->assertTrue(is_resource($process), 'Process is not running although it should.');
  163. $handler->close();
  164. $process = $property->getValue($handler);
  165. $this->assertFalse(is_resource($process), 'Process is still running although it should not.');
  166. }
  167. }