CliDumperTest.php 9.1 KB

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