Nav apraksta

CliDumperTest.php 10KB

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