6ef78ec553
* 2.7: Fix errors not rethrown even if not handled by console.error listeners [VarDumper] Fix dumping of non-nested stubs [Security] Avoid unnecessary route lookup for empty logout path
472 lines
10 KiB
PHP
472 lines
10 KiB
PHP
<?php
|
|
|
|
/*
|
|
* This file is part of the Symfony package.
|
|
*
|
|
* (c) Fabien Potencier <fabien@symfony.com>
|
|
*
|
|
* For the full copyright and license information, please view the LICENSE
|
|
* file that was distributed with this source code.
|
|
*/
|
|
|
|
namespace Symfony\Component\VarDumper\Tests;
|
|
|
|
use Symfony\Component\VarDumper\Cloner\VarCloner;
|
|
use Symfony\Component\VarDumper\Dumper\CliDumper;
|
|
use Symfony\Component\VarDumper\Test\VarDumperTestCase;
|
|
|
|
/**
|
|
* @author Nicolas Grekas <p@tchwork.com>
|
|
*/
|
|
class CliDumperTest extends VarDumperTestCase
|
|
{
|
|
public function testGet()
|
|
{
|
|
require __DIR__.'/Fixtures/dumb-var.php';
|
|
|
|
$dumper = new CliDumper('php://output');
|
|
$dumper->setColors(false);
|
|
$cloner = new VarCloner();
|
|
$cloner->addCasters(array(
|
|
':stream' => function ($res, $a) {
|
|
unset($a['uri'], $a['wrapper_data']);
|
|
|
|
return $a;
|
|
},
|
|
));
|
|
$data = $cloner->cloneVar($var);
|
|
|
|
ob_start();
|
|
$dumper->dump($data);
|
|
$out = ob_get_clean();
|
|
$out = preg_replace('/[ \t]+$/m', '', $out);
|
|
$intMax = PHP_INT_MAX;
|
|
$res = (int) $var['res'];
|
|
$closure54 = '';
|
|
$r = defined('HHVM_VERSION') ? '' : '#%d';
|
|
|
|
if (PHP_VERSION_ID >= 50400) {
|
|
$closure54 = <<<EOTXT
|
|
|
|
class: "Symfony\Component\VarDumper\Tests\CliDumperTest"
|
|
this: Symfony\Component\VarDumper\Tests\CliDumperTest {{$r} …}
|
|
EOTXT;
|
|
}
|
|
|
|
$this->assertStringMatchesFormat(
|
|
<<<EOTXT
|
|
array:24 [
|
|
"number" => 1
|
|
0 => &1 null
|
|
"const" => 1.1
|
|
1 => true
|
|
2 => false
|
|
3 => NAN
|
|
4 => INF
|
|
5 => -INF
|
|
6 => {$intMax}
|
|
"str" => "déjà\\n"
|
|
7 => b"é\\x00"
|
|
"[]" => []
|
|
"res" => stream resource {@{$res}
|
|
%A wrapper_type: "plainfile"
|
|
stream_type: "STDIO"
|
|
mode: "r"
|
|
unread_bytes: 0
|
|
seekable: true
|
|
%A options: []
|
|
}
|
|
"obj" => Symfony\Component\VarDumper\Tests\Fixture\DumbFoo {#%d
|
|
+foo: "foo"
|
|
+"bar": "bar"
|
|
}
|
|
"closure" => Closure {{$r}{$closure54}
|
|
parameters: {
|
|
\$a: {}
|
|
&\$b: {
|
|
typeHint: "PDO"
|
|
default: null
|
|
}
|
|
}
|
|
file: "{$var['file']}"
|
|
line: "{$var['line']} to {$var['line']}"
|
|
}
|
|
"line" => {$var['line']}
|
|
"nobj" => array:1 [
|
|
0 => &3 {#%d}
|
|
]
|
|
"recurs" => &4 array:1 [
|
|
0 => &4 array:1 [&4]
|
|
]
|
|
8 => &1 null
|
|
"sobj" => Symfony\Component\VarDumper\Tests\Fixture\DumbFoo {#%d}
|
|
"snobj" => &3 {#%d}
|
|
"snobj2" => {#%d}
|
|
"file" => "{$var['file']}"
|
|
b"bin-key-é" => ""
|
|
]
|
|
|
|
EOTXT
|
|
,
|
|
$out
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @requires extension xml
|
|
*/
|
|
public function testXmlResource()
|
|
{
|
|
$var = xml_parser_create();
|
|
|
|
$this->assertDumpMatchesFormat(
|
|
<<<'EOTXT'
|
|
xml resource {
|
|
current_byte_index: %i
|
|
current_column_number: %i
|
|
current_line_number: 1
|
|
error_code: XML_ERROR_NONE
|
|
}
|
|
EOTXT
|
|
,
|
|
$var
|
|
);
|
|
}
|
|
|
|
public function testJsonCast()
|
|
{
|
|
$var = (array) json_decode('{"0":{},"1":null}');
|
|
foreach ($var as &$v) {
|
|
}
|
|
$var[] = &$v;
|
|
$var[''] = 2;
|
|
|
|
$this->assertDumpMatchesFormat(
|
|
<<<'EOTXT'
|
|
array:4 [
|
|
"0" => {}
|
|
"1" => &1 null
|
|
0 => &1 null
|
|
"" => 2
|
|
]
|
|
EOTXT
|
|
,
|
|
$var
|
|
);
|
|
}
|
|
|
|
public function testObjectCast()
|
|
{
|
|
$var = (object) array(1 => 1);
|
|
$var->{1} = 2;
|
|
|
|
$this->assertDumpMatchesFormat(
|
|
<<<'EOTXT'
|
|
{
|
|
+1: 1
|
|
+"1": 2
|
|
}
|
|
EOTXT
|
|
,
|
|
$var
|
|
);
|
|
}
|
|
|
|
public function testClosedResource()
|
|
{
|
|
if (defined('HHVM_VERSION') && HHVM_VERSION_ID < 30600) {
|
|
$this->markTestSkipped();
|
|
}
|
|
|
|
$var = fopen(__FILE__, 'r');
|
|
fclose($var);
|
|
|
|
$dumper = new CliDumper('php://output');
|
|
$dumper->setColors(false);
|
|
$cloner = new VarCloner();
|
|
$data = $cloner->cloneVar($var);
|
|
|
|
ob_start();
|
|
$dumper->dump($data);
|
|
$out = ob_get_clean();
|
|
$res = (int) $var;
|
|
|
|
$this->assertStringMatchesFormat(
|
|
<<<EOTXT
|
|
Closed resource @{$res}
|
|
|
|
EOTXT
|
|
,
|
|
$out
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @requires function Twig_Template::getSourceContext
|
|
*/
|
|
public function testThrowingCaster()
|
|
{
|
|
$out = fopen('php://memory', 'r+b');
|
|
|
|
require_once __DIR__.'/Fixtures/Twig.php';
|
|
$twig = new \__TwigTemplate_VarDumperFixture_u75a09(new \Twig_Environment(new \Twig_Loader_Filesystem()));
|
|
|
|
$dumper = new CliDumper();
|
|
$dumper->setColors(false);
|
|
$cloner = new VarCloner();
|
|
$cloner->addCasters(array(
|
|
':stream' => function ($res, $a) {
|
|
unset($a['wrapper_data']);
|
|
|
|
return $a;
|
|
},
|
|
));
|
|
$cloner->addCasters(array(
|
|
':stream' => eval('return function () use ($twig) {
|
|
try {
|
|
$twig->render(array());
|
|
} catch (\Twig_Error_Runtime $e) {
|
|
throw $e->getPrevious();
|
|
}
|
|
};'),
|
|
));
|
|
$ref = (int) $out;
|
|
|
|
$data = $cloner->cloneVar($out);
|
|
$dumper->dump($data, $out);
|
|
rewind($out);
|
|
$out = stream_get_contents($out);
|
|
|
|
$r = defined('HHVM_VERSION') ? '' : '#%d';
|
|
$this->assertStringMatchesFormat(
|
|
<<<EOTXT
|
|
stream resource {@{$ref}
|
|
%Awrapper_type: "PHP"
|
|
stream_type: "MEMORY"
|
|
mode: "%s+b"
|
|
unread_bytes: 0
|
|
seekable: true
|
|
uri: "php://memory"
|
|
%Aoptions: []
|
|
⚠: Symfony\Component\VarDumper\Exception\ThrowingCasterException {{$r}
|
|
#message: "Unexpected Exception thrown from a caster: Foobar"
|
|
-trace: {
|
|
%d. __TwigTemplate_VarDumperFixture_u75a09->doDisplay() ==> new Exception(): {
|
|
src: {
|
|
%sTwig.php:21: """
|
|
// line 2\\n
|
|
throw new \Exception('Foobar');\\n
|
|
}\\n
|
|
"""
|
|
bar.twig:2: """
|
|
foo bar\\n
|
|
twig source\\n
|
|
\\n
|
|
"""
|
|
}
|
|
}
|
|
%d. Twig_Template->displayWithErrorHandling() ==> __TwigTemplate_VarDumperFixture_u75a09->doDisplay(): {
|
|
src: {
|
|
%sTemplate.php:%d: """
|
|
try {\\n
|
|
\$this->doDisplay(\$context, \$blocks);\\n
|
|
} catch (Twig_Error \$e) {\\n
|
|
"""
|
|
}
|
|
}
|
|
%d. Twig_Template->display() ==> Twig_Template->displayWithErrorHandling(): {
|
|
src: {
|
|
%sTemplate.php:%d: """
|
|
{\\n
|
|
\$this->displayWithErrorHandling(\$this->env->mergeGlobals(\$context), array_merge(\$this->blocks, \$blocks));\\n
|
|
}\\n
|
|
"""
|
|
}
|
|
}
|
|
%d. Twig_Template->render() ==> Twig_Template->display(): {
|
|
src: {
|
|
%sTemplate.php:%d: """
|
|
try {\\n
|
|
\$this->display(\$context);\\n
|
|
} catch (%s \$e) {\\n
|
|
"""
|
|
}
|
|
}
|
|
%d. %slosure%s() ==> Twig_Template->render(): {
|
|
src: {
|
|
%sCliDumperTest.php:%d: """
|
|
%A
|
|
"""
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
EOTXT
|
|
,
|
|
$out
|
|
);
|
|
}
|
|
|
|
public function testRefsInProperties()
|
|
{
|
|
$var = (object) array('foo' => 'foo');
|
|
$var->bar = &$var->foo;
|
|
|
|
$dumper = new CliDumper();
|
|
$dumper->setColors(false);
|
|
$cloner = new VarCloner();
|
|
|
|
$out = fopen('php://memory', 'r+b');
|
|
$data = $cloner->cloneVar($var);
|
|
$dumper->dump($data, $out);
|
|
rewind($out);
|
|
$out = stream_get_contents($out);
|
|
|
|
$r = defined('HHVM_VERSION') ? '' : '#%d';
|
|
$this->assertStringMatchesFormat(
|
|
<<<EOTXT
|
|
{{$r}
|
|
+"foo": &1 "foo"
|
|
+"bar": &1 "foo"
|
|
}
|
|
|
|
EOTXT
|
|
,
|
|
$out
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @runInSeparateProcess
|
|
* @preserveGlobalState disabled
|
|
* @requires PHP 5.6
|
|
*/
|
|
public function testSpecialVars56()
|
|
{
|
|
$var = $this->getSpecialVars();
|
|
|
|
$this->assertDumpEquals(
|
|
<<<'EOTXT'
|
|
array:3 [
|
|
0 => array:1 [
|
|
0 => &1 array:1 [
|
|
0 => &1 array:1 [&1]
|
|
]
|
|
]
|
|
1 => array:1 [
|
|
"GLOBALS" => &2 array:1 [
|
|
"GLOBALS" => &2 array:1 [&2]
|
|
]
|
|
]
|
|
2 => &2 array:1 [&2]
|
|
]
|
|
EOTXT
|
|
,
|
|
$var
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @runInSeparateProcess
|
|
* @preserveGlobalState disabled
|
|
*/
|
|
public function testGlobalsNoExt()
|
|
{
|
|
$var = $this->getSpecialVars();
|
|
unset($var[0]);
|
|
$out = '';
|
|
|
|
$dumper = new CliDumper(function ($line, $depth) use (&$out) {
|
|
if ($depth >= 0) {
|
|
$out .= str_repeat(' ', $depth).$line."\n";
|
|
}
|
|
});
|
|
$dumper->setColors(false);
|
|
$cloner = new VarCloner();
|
|
|
|
$refl = new \ReflectionProperty($cloner, 'useExt');
|
|
$refl->setAccessible(true);
|
|
$refl->setValue($cloner, false);
|
|
|
|
$data = $cloner->cloneVar($var);
|
|
$dumper->dump($data);
|
|
|
|
$this->assertSame(
|
|
<<<'EOTXT'
|
|
array:2 [
|
|
1 => array:1 [
|
|
"GLOBALS" => &1 array:1 [
|
|
"GLOBALS" => &1 array:1 [&1]
|
|
]
|
|
]
|
|
2 => &1 array:1 [&1]
|
|
]
|
|
|
|
EOTXT
|
|
,
|
|
$out
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @runInSeparateProcess
|
|
* @preserveGlobalState disabled
|
|
*/
|
|
public function testBuggyRefs()
|
|
{
|
|
if (PHP_VERSION_ID >= 50600) {
|
|
$this->markTestSkipped('PHP 5.6 fixed refs counting');
|
|
}
|
|
|
|
$var = $this->getSpecialVars();
|
|
$var = $var[0];
|
|
|
|
$dumper = new CliDumper();
|
|
$dumper->setColors(false);
|
|
$cloner = new VarCloner();
|
|
|
|
$data = $cloner->cloneVar($var)->withMaxDepth(3);
|
|
$out = '';
|
|
$dumper->dump($data, function ($line, $depth) use (&$out) {
|
|
if ($depth >= 0) {
|
|
$out .= str_repeat(' ', $depth).$line."\n";
|
|
}
|
|
});
|
|
|
|
$this->assertSame(
|
|
<<<'EOTXT'
|
|
array:1 [
|
|
0 => array:1 [
|
|
0 => array:1 [
|
|
0 => array:1 [ …1]
|
|
]
|
|
]
|
|
]
|
|
|
|
EOTXT
|
|
,
|
|
$out
|
|
);
|
|
}
|
|
|
|
private function getSpecialVars()
|
|
{
|
|
foreach (array_keys($GLOBALS) as $var) {
|
|
if ('GLOBALS' !== $var) {
|
|
unset($GLOBALS[$var]);
|
|
}
|
|
}
|
|
|
|
$var = function &() {
|
|
$var = array();
|
|
$var[] = &$var;
|
|
|
|
return $var;
|
|
};
|
|
|
|
return array($var(), $GLOBALS, &$GLOBALS);
|
|
}
|
|
}
|