diff --git a/lib/Doctrine/ORM/Tools/Console/Command/RunDqlCommand.php b/lib/Doctrine/ORM/Tools/Console/Command/RunDqlCommand.php
index f8611da0b11..61f1eef78f1 100644
--- a/lib/Doctrine/ORM/Tools/Console/Command/RunDqlCommand.php
+++ b/lib/Doctrine/ORM/Tools/Console/Command/RunDqlCommand.php
@@ -4,8 +4,8 @@
namespace Doctrine\ORM\Tools\Console\Command;
-use Doctrine\Common\Util\Debug;
use Doctrine\ORM\Tools\Console\CommandCompatibility;
+use Doctrine\ORM\Tools\Debug;
use LogicException;
use RuntimeException;
use Symfony\Component\Console\Input\InputArgument;
@@ -116,7 +116,7 @@ private function doExecute(InputInterface $input, OutputInterface $output): int
$resultSet = $query->execute([], constant($hydrationMode));
- $ui->text(Debug::dump($resultSet, (int) $input->getOption('depth'), true, false));
+ $ui->text(Debug::dump($resultSet, (int) $input->getOption('depth')));
return 0;
}
diff --git a/lib/Doctrine/ORM/Tools/Debug.php b/lib/Doctrine/ORM/Tools/Debug.php
new file mode 100644
index 00000000000..8aa232c3869
--- /dev/null
+++ b/lib/Doctrine/ORM/Tools/Debug.php
@@ -0,0 +1,164 @@
+toArray();
+ }
+
+ if (! $maxDepth) {
+ return is_object($var) ? get_class($var)
+ : (is_array($var) ? 'Array(' . count($var) . ')' : $var);
+ }
+
+ if (is_array($var)) {
+ $return = [];
+
+ foreach ($var as $k => $v) {
+ $return[$k] = self::export($v, $maxDepth - 1);
+ }
+
+ return $return;
+ }
+
+ if (! $isObj) {
+ return $var;
+ }
+
+ $return = new stdClass();
+ if ($var instanceof DateTimeInterface) {
+ $return->__CLASS__ = get_class($var);
+ $return->date = $var->format('c');
+ $return->timezone = $var->getTimezone()->getName();
+
+ return $return;
+ }
+
+ $return->__CLASS__ = ClassUtils::getClass($var);
+
+ if ($var instanceof Proxy) {
+ $return->__IS_PROXY__ = true;
+ $return->__PROXY_INITIALIZED__ = $var->__isInitialized();
+ }
+
+ if ($var instanceof ArrayObject || $var instanceof ArrayIterator) {
+ $return->__STORAGE__ = self::export($var->getArrayCopy(), $maxDepth - 1);
+ }
+
+ return self::fillReturnWithClassAttributes($var, $return, $maxDepth);
+ }
+
+ /**
+ * Fill the $return variable with class attributes
+ * Based on obj2array function from {@see https://secure.php.net/manual/en/function.get-object-vars.php#47075}
+ *
+ * @param object $var
+ *
+ * @return mixed
+ */
+ private static function fillReturnWithClassAttributes($var, stdClass $return, int $maxDepth)
+ {
+ $clone = (array) $var;
+
+ foreach (array_keys($clone) as $key) {
+ $aux = explode("\0", (string) $key);
+ $name = end($aux);
+ if ($aux[0] === '') {
+ $name .= ':' . ($aux[1] === '*' ? 'protected' : $aux[1] . ':private');
+ }
+
+ $return->$name = self::export($clone[$key], $maxDepth - 1);
+ }
+
+ return $return;
+ }
+}
diff --git a/phpcs.xml.dist b/phpcs.xml.dist
index 420e6f91c1a..147b39dfb85 100644
--- a/phpcs.xml.dist
+++ b/phpcs.xml.dist
@@ -59,6 +59,11 @@
tests/*
+
+ lib/Doctrine/ORM/Tools/Debug.php
+ tests/Doctrine/Tests/ORM/Tools/DebugTest.php
+
+
lib/Doctrine/ORM/Events.php
lib/Doctrine/ORM/Tools/ToolEvents.php
diff --git a/psalm-baseline.xml b/psalm-baseline.xml
index 38a29728619..9d649f6defd 100644
--- a/psalm-baseline.xml
+++ b/psalm-baseline.xml
@@ -536,8 +536,8 @@
$class
$class
-
$platformFamily
+
new UuidGenerator()
@@ -2406,11 +2406,6 @@
getAllClassNames
-
-
- getOption('depth'), true, false)]]>
-
-
int
diff --git a/psalm.xml b/psalm.xml
index 81d8fe8476b..8905a3c4086 100644
--- a/psalm.xml
+++ b/psalm.xml
@@ -125,6 +125,11 @@
+
+
+
+
+
diff --git a/tests/Doctrine/Tests/ORM/Tools/DebugTest.php b/tests/Doctrine/Tests/ORM/Tools/DebugTest.php
new file mode 100644
index 00000000000..281d9afc4d0
--- /dev/null
+++ b/tests/Doctrine/Tests/ORM/Tools/DebugTest.php
@@ -0,0 +1,163 @@
+foo = 'bar';
+ $obj->bar = 1234;
+
+ $var = Debug::export($obj, 2);
+ self::assertEquals('stdClass', $var->__CLASS__);
+ }
+
+ public function testExportObjectWithReference(): void
+ {
+ $foo = 'bar';
+ $bar = ['foo' => & $foo];
+ $baz = (object) $bar;
+
+ $var = Debug::export($baz, 2);
+ $baz->foo = 'tab';
+
+ self::assertEquals('bar', $var->foo);
+ self::assertEquals('tab', $bar['foo']);
+ }
+
+ public function testExportArray(): void
+ {
+ $array = ['a' => 'b', 'b' => ['c', 'd' => ['e', 'f']]];
+ $var = Debug::export($array, 2);
+ $expected = $array;
+ $expected['b']['d'] = 'Array(2)';
+ self::assertEquals($expected, $var);
+ }
+
+ public function testExportDateTime(): void
+ {
+ $obj = new DateTime('2010-10-10 10:10:10', new DateTimeZone('UTC'));
+
+ $var = Debug::export($obj, 2);
+ self::assertEquals('DateTime', $var->__CLASS__);
+ self::assertEquals('2010-10-10T10:10:10+00:00', $var->date);
+ }
+
+ public function testExportDateTimeImmutable(): void
+ {
+ $obj = new DateTimeImmutable('2010-10-10 10:10:10', new DateTimeZone('UTC'));
+
+ $var = Debug::export($obj, 2);
+ self::assertEquals('DateTimeImmutable', $var->__CLASS__);
+ self::assertEquals('2010-10-10T10:10:10+00:00', $var->date);
+ }
+
+ public function testExportDateTimeZone(): void
+ {
+ $obj = new DateTimeImmutable('2010-10-10 12:34:56', new DateTimeZone('Europe/Rome'));
+
+ $var = Debug::export($obj, 2);
+ self::assertEquals('DateTimeImmutable', $var->__CLASS__);
+ self::assertEquals('2010-10-10T12:34:56+02:00', $var->date);
+ }
+
+ public function testExportArrayTraversable(): void
+ {
+ $obj = new ArrayObject(['foobar']);
+
+ $var = Debug::export($obj, 2);
+ self::assertContains('foobar', $var->__STORAGE__);
+
+ $it = new ArrayIterator(['foobar']);
+
+ $var = Debug::export($it, 5);
+ self::assertContains('foobar', $var->__STORAGE__);
+ }
+
+ /**
+ * @param array $expected
+ *
+ * @dataProvider provideAttributesCases
+ */
+ public function testExportParentAttributes(TestAsset\ParentClass $class, array $expected): void
+ {
+ $actualRepresentation = print_r($class, true);
+ $expectedRepresentation = print_r($expected, true);
+
+ $actualRepresentation = substr($actualRepresentation, strpos($actualRepresentation, '('));
+ $expectedRepresentation = substr($expectedRepresentation, strpos($expectedRepresentation, '('));
+
+ self::assertSame($expectedRepresentation, $actualRepresentation);
+
+ $var = Debug::export($class, 3);
+ $var = (array) $var;
+ unset($var['__CLASS__']);
+
+ self::assertSame($expected, $var);
+ }
+
+ /**
+ * @psalm-return array
+ */
+ public function provideAttributesCases(): iterable
+ {
+ return [
+ 'different-attributes' => [
+ new TestAsset\ChildClass(),
+ version_compare(PHP_VERSION, '8.1', '<') ?
+ [
+ 'childPublicAttribute' => 4,
+ 'childProtectedAttribute:protected' => 5,
+ 'childPrivateAttribute:Doctrine\Tests\ORM\Tools\TestAsset\ChildClass:private' => 6,
+ 'parentPublicAttribute' => 1,
+ 'parentProtectedAttribute:protected' => 2,
+ 'parentPrivateAttribute:Doctrine\Tests\ORM\Tools\TestAsset\ParentClass:private' => 3,
+ ] :
+ [
+ 'parentPublicAttribute' => 1,
+ 'parentProtectedAttribute:protected' => 2,
+ 'parentPrivateAttribute:Doctrine\Tests\ORM\Tools\TestAsset\ParentClass:private' => 3,
+ 'childPublicAttribute' => 4,
+ 'childProtectedAttribute:protected' => 5,
+ 'childPrivateAttribute:Doctrine\Tests\ORM\Tools\TestAsset\ChildClass:private' => 6,
+ ],
+ ],
+ 'same-attributes' => [
+ new TestAsset\ChildWithSameAttributesClass(),
+ version_compare(PHP_VERSION, '8.1', '<') ?
+ [
+ 'parentPublicAttribute' => 4,
+ 'parentProtectedAttribute:protected' => 5,
+ 'parentPrivateAttribute:Doctrine\Tests\ORM\Tools\TestAsset\ChildWithSameAttributesClass:private' => 6,
+ 'parentPrivateAttribute:Doctrine\Tests\ORM\Tools\TestAsset\ParentClass:private' => 3,
+ ] :
+ [
+ 'parentPublicAttribute' => 4,
+ 'parentProtectedAttribute:protected' => 5,
+ 'parentPrivateAttribute:Doctrine\Tests\ORM\Tools\TestAsset\ParentClass:private' => 3,
+ 'parentPrivateAttribute:Doctrine\Tests\ORM\Tools\TestAsset\ChildWithSameAttributesClass:private' => 6,
+ ],
+ ],
+ ];
+ }
+}
diff --git a/tests/Doctrine/Tests/ORM/Tools/TestAsset/ChildClass.php b/tests/Doctrine/Tests/ORM/Tools/TestAsset/ChildClass.php
new file mode 100644
index 00000000000..35e11456da1
--- /dev/null
+++ b/tests/Doctrine/Tests/ORM/Tools/TestAsset/ChildClass.php
@@ -0,0 +1,15 @@
+