CliDumperTest.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488
  1. <?php
  2. /*
  3. * This file is part of the Symfony package.
  4. *
  5. * (c) Fabien Potencier <fabien@symfony.com>
  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 Symfony\Component\VarDumper\Tests;
  11. use PHPUnit\Framework\TestCase;
  12. use Symfony\Component\VarDumper\Cloner\VarCloner;
  13. use Symfony\Component\VarDumper\Dumper\CliDumper;
  14. use Symfony\Component\VarDumper\Test\VarDumperTestTrait;
  15. /**
  16. * @author Nicolas Grekas <p@tchwork.com>
  17. */
  18. class CliDumperTest extends TestCase
  19. {
  20. use VarDumperTestTrait;
  21. public function testGet()
  22. {
  23. require __DIR__.'/Fixtures/dumb-var.php';
  24. $dumper = new CliDumper('php://output');
  25. $dumper->setColors(false);
  26. $cloner = new VarCloner();
  27. $cloner->addCasters(array(
  28. ':stream' => function ($res, $a) {
  29. unset($a['uri'], $a['wrapper_data']);
  30. return $a;
  31. },
  32. ));
  33. $data = $cloner->cloneVar($var);
  34. ob_start();
  35. $dumper->dump($data);
  36. $out = ob_get_clean();
  37. $out = preg_replace('/[ \t]+$/m', '', $out);
  38. $intMax = PHP_INT_MAX;
  39. $res = (int) $var['res'];
  40. $r = defined('HHVM_VERSION') ? '' : '#%d';
  41. $this->assertStringMatchesFormat(
  42. <<<EOTXT
  43. array:24 [
  44. "number" => 1
  45. 0 => &1 null
  46. "const" => 1.1
  47. 1 => true
  48. 2 => false
  49. 3 => NAN
  50. 4 => INF
  51. 5 => -INF
  52. 6 => {$intMax}
  53. "str" => "déjà\\n"
  54. 7 => b"é\\x00"
  55. "[]" => []
  56. "res" => stream resource {@{$res}
  57. %A wrapper_type: "plainfile"
  58. stream_type: "STDIO"
  59. mode: "r"
  60. unread_bytes: 0
  61. seekable: true
  62. %A options: []
  63. }
  64. "obj" => Symfony\Component\VarDumper\Tests\Fixture\DumbFoo {#%d
  65. +foo: "foo"
  66. +"bar": "bar"
  67. }
  68. "closure" => Closure {{$r}
  69. class: "Symfony\Component\VarDumper\Tests\CliDumperTest"
  70. this: Symfony\Component\VarDumper\Tests\CliDumperTest {{$r} …}
  71. parameters: {
  72. \$a: {}
  73. &\$b: {
  74. typeHint: "PDO"
  75. default: null
  76. }
  77. }
  78. file: "{$var['file']}"
  79. line: "{$var['line']} to {$var['line']}"
  80. }
  81. "line" => {$var['line']}
  82. "nobj" => array:1 [
  83. 0 => &3 {#%d}
  84. ]
  85. "recurs" => &4 array:1 [
  86. 0 => &4 array:1 [&4]
  87. ]
  88. 8 => &1 null
  89. "sobj" => Symfony\Component\VarDumper\Tests\Fixture\DumbFoo {#%d}
  90. "snobj" => &3 {#%d}
  91. "snobj2" => {#%d}
  92. "file" => "{$var['file']}"
  93. b"bin-key-é" => ""
  94. ]
  95. EOTXT
  96. ,
  97. $out
  98. );
  99. }
  100. /**
  101. * @requires extension xml
  102. */
  103. public function testXmlResource()
  104. {
  105. $var = xml_parser_create();
  106. $this->assertDumpMatchesFormat(
  107. <<<'EOTXT'
  108. xml resource {
  109. current_byte_index: %i
  110. current_column_number: %i
  111. current_line_number: 1
  112. error_code: XML_ERROR_NONE
  113. }
  114. EOTXT
  115. ,
  116. $var
  117. );
  118. }
  119. public function testJsonCast()
  120. {
  121. $var = (array) json_decode('{"0":{},"1":null}');
  122. foreach ($var as &$v) {
  123. }
  124. $var[] = &$v;
  125. $var[''] = 2;
  126. $this->assertDumpMatchesFormat(
  127. <<<'EOTXT'
  128. array:4 [
  129. "0" => {}
  130. "1" => &1 null
  131. 0 => &1 null
  132. "" => 2
  133. ]
  134. EOTXT
  135. ,
  136. $var
  137. );
  138. }
  139. public function testObjectCast()
  140. {
  141. $var = (object) array(1 => 1);
  142. $var->{1} = 2;
  143. $this->assertDumpMatchesFormat(
  144. <<<'EOTXT'
  145. {
  146. +1: 1
  147. +"1": 2
  148. }
  149. EOTXT
  150. ,
  151. $var
  152. );
  153. }
  154. public function testClosedResource()
  155. {
  156. if (defined('HHVM_VERSION') && HHVM_VERSION_ID < 30600) {
  157. $this->markTestSkipped();
  158. }
  159. $var = fopen(__FILE__, 'r');
  160. fclose($var);
  161. $dumper = new CliDumper('php://output');
  162. $dumper->setColors(false);
  163. $cloner = new VarCloner();
  164. $data = $cloner->cloneVar($var);
  165. ob_start();
  166. $dumper->dump($data);
  167. $out = ob_get_clean();
  168. $res = (int) $var;
  169. $this->assertStringMatchesFormat(
  170. <<<EOTXT
  171. Closed resource @{$res}
  172. EOTXT
  173. ,
  174. $out
  175. );
  176. }
  177. public function testFlags()
  178. {
  179. putenv('DUMP_LIGHT_ARRAY=1');
  180. putenv('DUMP_STRING_LENGTH=1');
  181. $var = array(
  182. range(1, 3),
  183. array('foo', 2 => 'bar'),
  184. );
  185. $this->assertDumpEquals(
  186. <<<EOTXT
  187. [
  188. [
  189. 1
  190. 2
  191. 3
  192. ]
  193. [
  194. 0 => (3) "foo"
  195. 2 => (3) "bar"
  196. ]
  197. ]
  198. EOTXT
  199. ,
  200. $var
  201. );
  202. putenv('DUMP_LIGHT_ARRAY=');
  203. putenv('DUMP_STRING_LENGTH=');
  204. }
  205. /**
  206. * @requires function Twig_Template::getSourceContext
  207. */
  208. public function testThrowingCaster()
  209. {
  210. $out = fopen('php://memory', 'r+b');
  211. require_once __DIR__.'/Fixtures/Twig.php';
  212. $twig = new \__TwigTemplate_VarDumperFixture_u75a09(new \Twig_Environment(new \Twig_Loader_Filesystem()));
  213. $dumper = new CliDumper();
  214. $dumper->setColors(false);
  215. $cloner = new VarCloner();
  216. $cloner->addCasters(array(
  217. ':stream' => function ($res, $a) {
  218. unset($a['wrapper_data']);
  219. return $a;
  220. },
  221. ));
  222. $cloner->addCasters(array(
  223. ':stream' => eval('return function () use ($twig) {
  224. try {
  225. $twig->render(array());
  226. } catch (\Twig_Error_Runtime $e) {
  227. throw $e->getPrevious();
  228. }
  229. };'),
  230. ));
  231. $line = __LINE__ - 2;
  232. $ref = (int) $out;
  233. $data = $cloner->cloneVar($out);
  234. $dumper->dump($data, $out);
  235. $out = stream_get_contents($out, -1, 0);
  236. $r = defined('HHVM_VERSION') ? '' : '#%d';
  237. $this->assertStringMatchesFormat(
  238. <<<EOTXT
  239. stream resource {@{$ref}
  240. ⚠: Symfony\Component\VarDumper\Exception\ThrowingCasterException {{$r}
  241. #message: "Unexpected Exception thrown from a caster: Foobar"
  242. -trace: {
  243. %sTwig.php:2: {
  244. : foo bar
  245. : twig source
  246. :
  247. }
  248. %sTemplate.php:%d: {
  249. : try {
  250. : \$this->doDisplay(\$context, \$blocks);
  251. : } catch (Twig_Error \$e) {
  252. }
  253. %sTemplate.php:%d: {
  254. : {
  255. : \$this->displayWithErrorHandling(\$this->env->mergeGlobals(\$context), array_merge(\$this->blocks, \$blocks));
  256. : }
  257. }
  258. %sTemplate.php:%d: {
  259. : try {
  260. : \$this->display(\$context);
  261. : } catch (%s \$e) {
  262. }
  263. %sCliDumperTest.php:{$line}: {
  264. : }
  265. : };'),
  266. : ));
  267. }
  268. }
  269. }
  270. %Awrapper_type: "PHP"
  271. stream_type: "MEMORY"
  272. mode: "%s+b"
  273. unread_bytes: 0
  274. seekable: true
  275. uri: "php://memory"
  276. %Aoptions: []
  277. }
  278. EOTXT
  279. ,
  280. $out
  281. );
  282. }
  283. public function testRefsInProperties()
  284. {
  285. $var = (object) array('foo' => 'foo');
  286. $var->bar = &$var->foo;
  287. $dumper = new CliDumper();
  288. $dumper->setColors(false);
  289. $cloner = new VarCloner();
  290. $data = $cloner->cloneVar($var);
  291. $out = $dumper->dump($data, true);
  292. $r = defined('HHVM_VERSION') ? '' : '#%d';
  293. $this->assertStringMatchesFormat(
  294. <<<EOTXT
  295. {{$r}
  296. +"foo": &1 "foo"
  297. +"bar": &1 "foo"
  298. }
  299. EOTXT
  300. ,
  301. $out
  302. );
  303. }
  304. /**
  305. * @runInSeparateProcess
  306. * @preserveGlobalState disabled
  307. * @requires PHP 5.6
  308. */
  309. public function testSpecialVars56()
  310. {
  311. $var = $this->getSpecialVars();
  312. $this->assertDumpEquals(
  313. <<<'EOTXT'
  314. array:3 [
  315. 0 => array:1 [
  316. 0 => &1 array:1 [
  317. 0 => &1 array:1 [&1]
  318. ]
  319. ]
  320. 1 => array:1 [
  321. "GLOBALS" => &2 array:1 [
  322. "GLOBALS" => &2 array:1 [&2]
  323. ]
  324. ]
  325. 2 => &2 array:1 [&2]
  326. ]
  327. EOTXT
  328. ,
  329. $var
  330. );
  331. }
  332. /**
  333. * @runInSeparateProcess
  334. * @preserveGlobalState disabled
  335. */
  336. public function testGlobalsNoExt()
  337. {
  338. $var = $this->getSpecialVars();
  339. unset($var[0]);
  340. $out = '';
  341. $dumper = new CliDumper(function ($line, $depth) use (&$out) {
  342. if ($depth >= 0) {
  343. $out .= str_repeat(' ', $depth).$line."\n";
  344. }
  345. });
  346. $dumper->setColors(false);
  347. $cloner = new VarCloner();
  348. $refl = new \ReflectionProperty($cloner, 'useExt');
  349. $refl->setAccessible(true);
  350. $refl->setValue($cloner, false);
  351. $data = $cloner->cloneVar($var);
  352. $dumper->dump($data);
  353. $this->assertSame(
  354. <<<'EOTXT'
  355. array:2 [
  356. 1 => array:1 [
  357. "GLOBALS" => &1 array:1 [
  358. "GLOBALS" => &1 array:1 [&1]
  359. ]
  360. ]
  361. 2 => &1 array:1 [&1]
  362. ]
  363. EOTXT
  364. ,
  365. $out
  366. );
  367. }
  368. /**
  369. * @runInSeparateProcess
  370. * @preserveGlobalState disabled
  371. */
  372. public function testBuggyRefs()
  373. {
  374. if (PHP_VERSION_ID >= 50600) {
  375. $this->markTestSkipped('PHP 5.6 fixed refs counting');
  376. }
  377. $var = $this->getSpecialVars();
  378. $var = $var[0];
  379. $dumper = new CliDumper();
  380. $dumper->setColors(false);
  381. $cloner = new VarCloner();
  382. $data = $cloner->cloneVar($var)->withMaxDepth(3);
  383. $out = '';
  384. $dumper->dump($data, function ($line, $depth) use (&$out) {
  385. if ($depth >= 0) {
  386. $out .= str_repeat(' ', $depth).$line."\n";
  387. }
  388. });
  389. $this->assertSame(
  390. <<<'EOTXT'
  391. array:1 [
  392. 0 => array:1 [
  393. 0 => array:1 [
  394. 0 => array:1 [ …1]
  395. ]
  396. ]
  397. ]
  398. EOTXT
  399. ,
  400. $out
  401. );
  402. }
  403. public function testIncompleteClass()
  404. {
  405. $unserializeCallbackHandler = ini_set('unserialize_callback_func', null);
  406. $var = unserialize('O:8:"Foo\Buzz":0:{}');
  407. ini_set('unserialize_callback_func', $unserializeCallbackHandler);
  408. $this->assertDumpMatchesFormat(
  409. <<<EOTXT
  410. __PHP_Incomplete_Class(Foo\Buzz) {}
  411. EOTXT
  412. ,
  413. $var
  414. );
  415. }
  416. private function getSpecialVars()
  417. {
  418. foreach (array_keys($GLOBALS) as $var) {
  419. if ('GLOBALS' !== $var) {
  420. unset($GLOBALS[$var]);
  421. }
  422. }
  423. $var = function &() {
  424. $var = array();
  425. $var[] = &$var;
  426. return $var;
  427. };
  428. return array($var(), $GLOBALS, &$GLOBALS);
  429. }
  430. }