592 lines
12 KiB
PHP
592 lines
12 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\Dumper;
|
||
|
||
use PHPUnit\Framework\TestCase;
|
||
use Symfony\Component\VarDumper\Cloner\VarCloner;
|
||
use Symfony\Component\VarDumper\Dumper\CliDumper;
|
||
use Symfony\Component\VarDumper\Test\VarDumperTestTrait;
|
||
use Twig\Environment;
|
||
use Twig\Loader\FilesystemLoader;
|
||
|
||
/**
|
||
* @author Nicolas Grekas <p@tchwork.com>
|
||
*/
|
||
class CliDumperTest extends TestCase
|
||
{
|
||
use VarDumperTestTrait;
|
||
|
||
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'];
|
||
|
||
$r = \defined('HHVM_VERSION') ? '' : '#%d';
|
||
$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}
|
||
class: "Symfony\Component\VarDumper\Tests\Dumper\CliDumperTest"
|
||
this: Symfony\Component\VarDumper\Tests\Dumper\CliDumperTest {{$r} …}
|
||
parameters: {
|
||
\$a: {}
|
||
&\$b: {
|
||
typeHint: "PDO"
|
||
default: null
|
||
}
|
||
}
|
||
file: "%s%eTests%eFixtures%edumb-var.php"
|
||
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
|
||
);
|
||
}
|
||
|
||
/**
|
||
* @dataProvider provideDumpWithCommaFlagTests
|
||
*/
|
||
public function testDumpWithCommaFlag($expected, $flags)
|
||
{
|
||
$dumper = new CliDumper(null, null, $flags);
|
||
$dumper->setColors(false);
|
||
$cloner = new VarCloner();
|
||
|
||
$var = array(
|
||
'array' => array('a', 'b'),
|
||
'string' => 'hello',
|
||
'multiline string' => "this\nis\na\multiline\nstring",
|
||
);
|
||
|
||
$dump = $dumper->dump($cloner->cloneVar($var), true);
|
||
|
||
$this->assertSame($expected, $dump);
|
||
}
|
||
|
||
public function testDumpWithCommaFlagsAndExceptionCodeExcerpt()
|
||
{
|
||
$dumper = new CliDumper(null, null, CliDumper::DUMP_TRAILING_COMMA);
|
||
$dumper->setColors(false);
|
||
$cloner = new VarCloner();
|
||
|
||
$ex = new \RuntimeException('foo');
|
||
|
||
$dump = $dumper->dump($cloner->cloneVar($ex)->withRefHandles(false), true);
|
||
|
||
$this->assertStringMatchesFormat(<<<'EOTXT'
|
||
RuntimeException {
|
||
#message: "foo"
|
||
#code: 0
|
||
#file: "%ACliDumperTest.php"
|
||
#line: %d
|
||
trace: {
|
||
%ACliDumperTest.php:%d {
|
||
›
|
||
› $ex = new \RuntimeException('foo');
|
||
›
|
||
}
|
||
%A
|
||
}
|
||
}
|
||
|
||
EOTXT
|
||
, $dump);
|
||
}
|
||
|
||
public function provideDumpWithCommaFlagTests()
|
||
{
|
||
$expected = <<<'EOTXT'
|
||
array:3 [
|
||
"array" => array:2 [
|
||
0 => "a",
|
||
1 => "b"
|
||
],
|
||
"string" => "hello",
|
||
"multiline string" => """
|
||
this\n
|
||
is\n
|
||
a\multiline\n
|
||
string
|
||
"""
|
||
]
|
||
|
||
EOTXT;
|
||
|
||
yield array($expected, CliDumper::DUMP_COMMA_SEPARATOR);
|
||
|
||
$expected = <<<'EOTXT'
|
||
array:3 [
|
||
"array" => array:2 [
|
||
0 => "a",
|
||
1 => "b",
|
||
],
|
||
"string" => "hello",
|
||
"multiline string" => """
|
||
this\n
|
||
is\n
|
||
a\multiline\n
|
||
string
|
||
""",
|
||
]
|
||
|
||
EOTXT;
|
||
|
||
yield array($expected, CliDumper::DUMP_TRAILING_COMMA);
|
||
}
|
||
|
||
/**
|
||
* @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;
|
||
|
||
if (\PHP_VERSION_ID >= 70200) {
|
||
$this->assertDumpMatchesFormat(
|
||
<<<'EOTXT'
|
||
array:4 [
|
||
0 => {}
|
||
1 => &1 null
|
||
2 => &1 null
|
||
"" => 2
|
||
]
|
||
EOTXT
|
||
,
|
||
$var
|
||
);
|
||
} else {
|
||
$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;
|
||
|
||
if (\PHP_VERSION_ID >= 70200) {
|
||
$this->assertDumpMatchesFormat(
|
||
<<<'EOTXT'
|
||
{
|
||
+"1": 2
|
||
}
|
||
EOTXT
|
||
,
|
||
$var
|
||
);
|
||
} else {
|
||
$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
|
||
);
|
||
}
|
||
|
||
public function testFlags()
|
||
{
|
||
putenv('DUMP_LIGHT_ARRAY=1');
|
||
putenv('DUMP_STRING_LENGTH=1');
|
||
|
||
$var = array(
|
||
range(1, 3),
|
||
array('foo', 2 => 'bar'),
|
||
);
|
||
|
||
$this->assertDumpEquals(
|
||
<<<EOTXT
|
||
[
|
||
[
|
||
1
|
||
2
|
||
3
|
||
]
|
||
[
|
||
0 => (3) "foo"
|
||
2 => (3) "bar"
|
||
]
|
||
]
|
||
EOTXT
|
||
,
|
||
$var
|
||
);
|
||
|
||
putenv('DUMP_LIGHT_ARRAY=');
|
||
putenv('DUMP_STRING_LENGTH=');
|
||
}
|
||
|
||
/**
|
||
* @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 Environment(new FilesystemLoader()));
|
||
|
||
$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\RuntimeError $e) {
|
||
throw $e->getPrevious();
|
||
}
|
||
};'),
|
||
));
|
||
$ref = (int) $out;
|
||
|
||
$data = $cloner->cloneVar($out);
|
||
$dumper->dump($data, $out);
|
||
$out = stream_get_contents($out, -1, 0);
|
||
|
||
$r = \defined('HHVM_VERSION') ? '' : '#%d';
|
||
$this->assertStringMatchesFormat(
|
||
<<<EOTXT
|
||
stream resource {@{$ref}
|
||
⚠: Symfony\Component\VarDumper\Exception\ThrowingCasterException {{$r}
|
||
#message: "Unexpected Exception thrown from a caster: Foobar"
|
||
trace: {
|
||
%sTwig.php:2 {
|
||
› foo bar
|
||
› twig source
|
||
›
|
||
}
|
||
%s%eTemplate.php:%d { …}
|
||
%s%eTemplate.php:%d { …}
|
||
%s%eTemplate.php:%d { …}
|
||
%s%eTests%eDumper%eCliDumperTest.php:%d { …}
|
||
%A }
|
||
}
|
||
%Awrapper_type: "PHP"
|
||
stream_type: "MEMORY"
|
||
mode: "%s+b"
|
||
unread_bytes: 0
|
||
seekable: true
|
||
uri: "php://memory"
|
||
%Aoptions: []
|
||
}
|
||
|
||
EOTXT
|
||
,
|
||
$out
|
||
);
|
||
}
|
||
|
||
public function testRefsInProperties()
|
||
{
|
||
$var = (object) array('foo' => 'foo');
|
||
$var->bar = &$var->foo;
|
||
|
||
$dumper = new CliDumper();
|
||
$dumper->setColors(false);
|
||
$cloner = new VarCloner();
|
||
|
||
$data = $cloner->cloneVar($var);
|
||
$out = $dumper->dump($data, true);
|
||
|
||
$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
|
||
);
|
||
}
|
||
|
||
public function testIncompleteClass()
|
||
{
|
||
$unserializeCallbackHandler = ini_set('unserialize_callback_func', null);
|
||
$var = unserialize('O:8:"Foo\Buzz":0:{}');
|
||
ini_set('unserialize_callback_func', $unserializeCallbackHandler);
|
||
|
||
$this->assertDumpMatchesFormat(
|
||
<<<EOTXT
|
||
__PHP_Incomplete_Class(Foo\Buzz) {}
|
||
EOTXT
|
||
,
|
||
$var
|
||
);
|
||
}
|
||
|
||
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);
|
||
}
|
||
}
|