BackgroundJobTest.php 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375
  1. <?php
  2. namespace Jobby\Tests;
  3. use Jobby\BackgroundJob;
  4. use Jobby\Helper;
  5. use Jobby\SerializerTrait;
  6. use Symfony\Component\Filesystem\Filesystem;
  7. /**
  8. * @coversDefaultClass Jobby\BackgroundJob
  9. */
  10. class BackgroundJobTest extends \PHPUnit_Framework_TestCase
  11. {
  12. use SerializerTrait;
  13. const JOB_NAME = 'name';
  14. /**
  15. * @var string
  16. */
  17. private $logFile;
  18. /**
  19. * @var Helper
  20. */
  21. private $helper;
  22. /**
  23. * {@inheritdoc}
  24. */
  25. protected function setUp()
  26. {
  27. $this->logFile = __DIR__ . '/_files/BackgroundJobTest.log';
  28. if (file_exists($this->logFile)) {
  29. unlink($this->logFile);
  30. }
  31. $this->helper = new Helper();
  32. }
  33. /**
  34. * {@inheritdoc}
  35. */
  36. protected function tearDown()
  37. {
  38. if (file_exists($this->logFile)) {
  39. unlink($this->logFile);
  40. }
  41. }
  42. public function runProvider()
  43. {
  44. $echo = function () {
  45. echo 'test';
  46. return true;
  47. };
  48. $uid = function () {
  49. echo getmyuid();
  50. return true;
  51. };
  52. $job = ['closure' => $echo];
  53. return [
  54. 'diabled, not run' => [$job + ['enabled' => false], ''],
  55. 'normal job, run' => [$job, 'test'],
  56. 'wrong host, not run' => [$job + ['runOnHost' => 'something that does not match'], ''],
  57. 'current user, run,' => [['closure' => $uid], getmyuid()],
  58. ];
  59. }
  60. /**
  61. * @covers ::getConfig
  62. */
  63. public function testGetConfig()
  64. {
  65. $job = new BackgroundJob('test job',[]);
  66. $this->assertInternalType('array',$job->getConfig());
  67. }
  68. /**
  69. * @dataProvider runProvider
  70. *
  71. * @covers ::run
  72. *
  73. * @param array $config
  74. * @param string $expectedOutput
  75. */
  76. public function testRun($config, $expectedOutput)
  77. {
  78. $this->runJob($config);
  79. $this->assertEquals($expectedOutput, $this->getLogContent());
  80. }
  81. /**
  82. * @covers ::runFile
  83. */
  84. public function testInvalidCommand()
  85. {
  86. $this->runJob(['command' => 'invalid-command']);
  87. $this->assertContains('invalid-command', $this->getLogContent());
  88. if ($this->helper->getPlatform() === Helper::UNIX) {
  89. $this->assertContains('not found', $this->getLogContent());
  90. $this->assertContains(
  91. "ERROR: Job exited with status '127'",
  92. $this->getLogContent()
  93. );
  94. } else {
  95. $this->assertContains(
  96. 'not recognized as an internal or external command',
  97. $this->getLogContent()
  98. );
  99. }
  100. }
  101. /**
  102. * @covers ::runFunction
  103. */
  104. public function testClosureNotReturnTrue()
  105. {
  106. $this->runJob(
  107. [
  108. 'closure' => function () {
  109. return false;
  110. },
  111. ]
  112. );
  113. $this->assertContains(
  114. 'ERROR: Closure did not return true! Returned:',
  115. $this->getLogContent()
  116. );
  117. }
  118. /**
  119. * @covers ::getLogFile
  120. */
  121. public function testHideStdOutByDefault()
  122. {
  123. ob_start();
  124. $this->runJob(
  125. [
  126. 'closure' => function () {
  127. echo 'foo bar';
  128. },
  129. 'output' => null,
  130. ]
  131. );
  132. $content = ob_get_contents();
  133. ob_end_clean();
  134. $this->assertEmpty($content);
  135. }
  136. /**
  137. * @covers ::getLogFile
  138. */
  139. public function testShouldCreateLogFolder()
  140. {
  141. $logfile = dirname($this->logFile) . '/foo/bar.log';
  142. $this->runJob(
  143. [
  144. 'closure' => function () {
  145. echo 'foo bar';
  146. },
  147. 'output' => $logfile,
  148. ]
  149. );
  150. $dirExists = file_exists(dirname($logfile));
  151. $isDir = is_dir(dirname($logfile));
  152. unlink($logfile);
  153. rmdir(dirname($logfile));
  154. $this->assertTrue($dirExists);
  155. $this->assertTrue($isDir);
  156. }
  157. /**
  158. * @covers ::mail
  159. */
  160. public function testNotSendMailOnMissingRecipients()
  161. {
  162. $helper = $this->getMock('Jobby\Helper', ['sendMail']);
  163. $helper->expects($this->never())
  164. ->method('sendMail')
  165. ;
  166. $this->runJob(
  167. [
  168. 'closure' => function () {
  169. return false;
  170. },
  171. 'recipients' => '',
  172. ],
  173. $helper
  174. );
  175. }
  176. /**
  177. * @covers ::mail
  178. */
  179. public function testMailShouldTriggerHelper()
  180. {
  181. $helper = $this->getMock('Jobby\Helper', ['sendMail']);
  182. $helper->expects($this->once())
  183. ->method('sendMail')
  184. ;
  185. $this->runJob(
  186. [
  187. 'closure' => function () {
  188. return false;
  189. },
  190. 'recipients' => 'test@example.com',
  191. ],
  192. $helper
  193. );
  194. }
  195. /**
  196. * @covers ::checkMaxRuntime
  197. */
  198. public function testCheckMaxRuntime()
  199. {
  200. if ($this->helper->getPlatform() !== Helper::UNIX) {
  201. $this->markTestSkipped("'maxRuntime' is not supported on Windows");
  202. }
  203. $helper = $this->getMock('Jobby\Helper', ['getLockLifetime']);
  204. $helper->expects($this->once())
  205. ->method('getLockLifetime')
  206. ->will($this->returnValue(0))
  207. ;
  208. $this->runJob(
  209. [
  210. 'command' => 'true',
  211. 'maxRuntime' => 1,
  212. ],
  213. $helper
  214. );
  215. $this->assertEmpty($this->getLogContent());
  216. }
  217. /**
  218. * @covers ::checkMaxRuntime
  219. */
  220. public function testCheckMaxRuntimeShouldFailIsExceeded()
  221. {
  222. if ($this->helper->getPlatform() !== Helper::UNIX) {
  223. $this->markTestSkipped("'maxRuntime' is not supported on Windows");
  224. }
  225. $helper = $this->getMock('Jobby\Helper', ['getLockLifetime']);
  226. $helper->expects($this->once())
  227. ->method('getLockLifetime')
  228. ->will($this->returnValue(2))
  229. ;
  230. $this->runJob(
  231. [
  232. 'command' => 'true',
  233. 'maxRuntime' => 1,
  234. ],
  235. $helper
  236. );
  237. $this->assertContains(
  238. 'MaxRuntime of 1 secs exceeded! Current runtime: 2 secs',
  239. $this->getLogContent()
  240. );
  241. }
  242. /**
  243. * @dataProvider haltDirProvider
  244. * @covers ::shouldRun
  245. *
  246. * @param bool $createFile
  247. * @param bool $jobRuns
  248. */
  249. public function testHaltDir($createFile, $jobRuns)
  250. {
  251. $dir = __DIR__ . '/_files';
  252. $file = $dir . '/' . static::JOB_NAME;
  253. $fs = new Filesystem();
  254. if ($createFile) {
  255. $fs->touch($file);
  256. }
  257. $this->runJob(
  258. [
  259. 'haltDir' => $dir,
  260. 'closure' => function () {
  261. echo 'test';
  262. return true;
  263. },
  264. ]
  265. );
  266. if ($createFile) {
  267. $fs->remove($file);
  268. }
  269. $content = $this->getLogContent();
  270. $this->assertEquals($jobRuns, is_string($content) && !empty($content));
  271. }
  272. public function haltDirProvider()
  273. {
  274. return [
  275. [true, false],
  276. [false, true],
  277. ];
  278. }
  279. /**
  280. * @param array $config
  281. * @param Helper $helper
  282. */
  283. private function runJob(array $config, Helper $helper = null)
  284. {
  285. $config = $this->getJobConfig($config);
  286. $job = new BackgroundJob(self::JOB_NAME, $config, $helper);
  287. $job->run();
  288. }
  289. /**
  290. * @param array $config
  291. *
  292. * @return array
  293. */
  294. private function getJobConfig(array $config)
  295. {
  296. $helper = new Helper();
  297. if (isset($config['closure'])) {
  298. $config['closure'] = $this->getSerializer()->serialize($config['closure']);
  299. }
  300. return array_merge(
  301. [
  302. 'enabled' => 1,
  303. 'haltDir' => null,
  304. 'runOnHost' => $helper->getHost(),
  305. 'dateFormat' => 'Y-m-d H:i:s',
  306. 'schedule' => '* * * * *',
  307. 'output' => $this->logFile,
  308. 'maxRuntime' => null,
  309. 'runAs' => null,
  310. ],
  311. $config
  312. );
  313. }
  314. /**
  315. * @return string
  316. */
  317. private function getLogContent()
  318. {
  319. return @file_get_contents($this->logFile);
  320. }
  321. }