499f1a3e7e42162f7b6570cddb42508cb6e58948
[Packages/TYPO3.CMS.git] / typo3 / sysext / extbase / Classes / Utility / DebuggerUtility.php
1 <?php
2 namespace TYPO3\CMS\Extbase\Utility;
3
4 /* *
5 * This script belongs to the Extbase framework *
6 * *
7 * It is free software; you can redistribute it and/or modify it under *
8 * the terms of the GNU Lesser General Public License as published by the *
9 * Free Software Foundation, either version 3 of the License, or (at your *
10 * option) any later version. *
11 * *
12 * This script is distributed in the hope that it will be useful, but *
13 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHAN- *
14 * TABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser *
15 * General Public License for more details. *
16 * *
17 * You should have received a copy of the GNU Lesser General Public *
18 * License along with the script. *
19 * If not, see http://www.gnu.org/licenses/lgpl.html *
20 * *
21 * The TYPO3 project - inspiring people to share! *
22 * */
23 /**
24 * This class is a backport of the corresponding class of TYPO3 Flow.
25 * All credits go to the TYPO3 Flow team.
26 */
27 /**
28 * A debugging utility class
29 *
30 * @license http://www.gnu.org/licenses/lgpl.html GNU Lesser General Public License, version 3 or later
31 * @api
32 */
33 class DebuggerUtility {
34
35 const PLAINTEXT_INDENT = ' ';
36 const HTML_INDENT = '&nbsp;&nbsp;&nbsp;';
37 /**
38 * @var \TYPO3\CMS\Extbase\Persistence\ObjectStorage
39 */
40 static protected $renderedObjects;
41
42 /**
43 * Hardcoded list of Extbase class names (regex) which should not be displayed during debugging
44 *
45 * @var array
46 */
47 static protected $blacklistedClassNames = array(
48 'PHPUnit_Framework_MockObject_InvocationMocker',
49 'TYPO3\\CMS\\Extbase\\Persistence\\Generic\\IdentityMap',
50 'TYPO3\\CMS\\Extbase\\Reflection\\ReflectionService',
51 'TYPO3\\CMS\\Extbase\\Object\\ObjectManager',
52 'TYPO3\\CMS\\Extbase\\Persistence\\Generic\\Mapper\\DataMapper',
53 'TYPO3\\CMS\\Extbase\\Persistence\\Generic\\PersistenceManager',
54 'TYPO3\\CMS\\Extbase\\Persistence\\Generic\\Qom\\QueryObjectModelFactory',
55 'TYPO3\\CMS\\Frontend\\ContentObject\\ContentObjectRenderer'
56 );
57
58 /**
59 * Hardcoded list of property names (regex) which should not be displayed during debugging
60 *
61 * @var array
62 */
63 static protected $blacklistedPropertyNames = array('warning');
64
65 /**
66 * Is set to TRUE once the CSS file is included in the current page to prevent double inclusions of the CSS file.
67 *
68 * @var boolean
69 */
70 static protected $stylesheetEchoed = FALSE;
71
72 /**
73 * Defines the max recursion depth of the dump, set to 8 due to common memory limits
74 *
75 * @var int
76 */
77 static protected $maxDepth = 8;
78
79 /**
80 * Clear the state of the debugger
81 *
82 * @return void
83 */
84 static protected function clearState() {
85 self::$renderedObjects = new \TYPO3\CMS\Extbase\Persistence\ObjectStorage();
86 }
87
88 /**
89 * Renders a dump of the given value
90 *
91 * @param mixed $value
92 * @param integer $level
93 * @param boolean $plainText
94 * @param boolean $ansiColors
95 * @return string
96 */
97 static protected function renderDump($value, $level, $plainText, $ansiColors) {
98 $dump = '';
99 if (is_string($value)) {
100 $croppedValue = strlen($value) > 2000 ? substr($value, 0, 2000) . '...' : $value;
101 if ($plainText) {
102 $dump = self::ansiEscapeWrap(('"' . implode((PHP_EOL . str_repeat(self::PLAINTEXT_INDENT, ($level + 1))), str_split($croppedValue, 76)) . '"'), '33', $ansiColors) . ' (' . strlen($value) . ' chars)';
103 } else {
104 $dump = sprintf('\'<span class="debug-string">%s</span>\' (%s chars)', implode('<br />' . str_repeat(self::HTML_INDENT, ($level + 1)), str_split(htmlspecialchars($croppedValue), 76)), strlen($value));
105 }
106 } elseif (is_numeric($value)) {
107 $dump = sprintf('%s (%s)', self::ansiEscapeWrap($value, '35', $ansiColors), gettype($value));
108 } elseif (is_bool($value)) {
109 $dump = $value ? self::ansiEscapeWrap('TRUE', '32', $ansiColors) : self::ansiEscapeWrap('FALSE', '32', $ansiColors);
110 } elseif (is_null($value) || is_resource($value)) {
111 $dump = gettype($value);
112 } elseif (is_array($value)) {
113 $dump = self::renderArray($value, $level + 1, $plainText, $ansiColors);
114 } elseif (is_object($value)) {
115 $dump = self::renderObject($value, $level + 1, $plainText, $ansiColors);
116 }
117 return $dump;
118 }
119
120 /**
121 * Renders a dump of the given array
122 *
123 * @param array|\Traversable $array
124 * @param integer $level
125 * @param boolean $plainText
126 * @param boolean $ansiColors
127 * @return string
128 */
129 static protected function renderArray($array, $level, $plainText = FALSE, $ansiColors = FALSE) {
130 $type = is_array($array) ? 'array' : get_class($array);
131 if ($plainText) {
132 $header = self::ansiEscapeWrap($type, '36', $ansiColors);
133 } else {
134 $header = '<span class="debug-type">' . $type . '</span>';
135 }
136 $header .= count($array) > 0 ? ' (' . count($array) . ' items)' : ' (empty)';
137 if ($level >= self::$maxDepth) {
138 if ($plainText) {
139 $content = ' ' . self::ansiEscapeWrap('max depth', '47;30', $ansiColors);
140 } else {
141 $content = '<span class="debug-filtered">max depth</span>';
142 }
143 } else {
144 $content = self::renderCollection($array, $level, $plainText, $ansiColors);
145 }
146 if ($level > 1 && count($array) > 0 && !$plainText) {
147 $dump = '<span class="debug-tree"><input type="checkbox" /><span class="debug-header">' . $header . '</span><span class="debug-content">' . $content . '</span></span>';
148 } else {
149 $dump = $header . $content;
150 }
151 return $dump;
152 }
153
154 /**
155 * Renders a dump of the given object
156 *
157 * @param object $object
158 * @param integer $level
159 * @param boolean $plainText
160 * @param boolean $ansiColors
161 * @return string
162 */
163 static protected function renderObject($object, $level, $plainText = FALSE, $ansiColors = FALSE) {
164 if ($object instanceof \TYPO3\CMS\Extbase\Persistence\Generic\LazyLoadingProxy) {
165 $object = $object->_loadRealInstance();
166 }
167 $header = self::renderHeader($object, $level, $plainText, $ansiColors);
168 if ($level < self::$maxDepth && !self::isBlacklisted($object) && !(self::isAlreadyRendered($object) && $plainText !== TRUE)) {
169 $content = self::renderContent($object, $level, $plainText, $ansiColors);
170 } else {
171 $content = '';
172 }
173 if ($plainText) {
174 return $header . $content;
175 } else {
176 return '<span class="debug-tree">' . $header . '<span class="debug-content">' . $content . '</span></span>';
177 }
178 }
179
180 /**
181 * Checks if a given object or property should be excluded/filtered
182 *
183 * @param object $value An ReflectionProperty or other Object
184 * @return bool TRUE if the given object should be filtered
185 */
186 static protected function isBlacklisted($value) {
187 $result = FALSE;
188 if ($value instanceof \ReflectionProperty) {
189 $result = (strpos(implode('|', self::$blacklistedPropertyNames), $value->getName()) > 0);
190 } elseif (is_object($value)) {
191 $result = (strpos(implode('|', self::$blacklistedClassNames), get_class($value)) > 0);
192 }
193 return $result;
194 }
195
196 /**
197 * Checks if a given object was already rendered.
198 *
199 * @param $object
200 * @return bool TRUE if the given object was already rendered
201 */
202 static protected function isAlreadyRendered($object) {
203 return self::$renderedObjects->contains($object);
204 }
205
206 /**
207 * Renders the header of a given object/collection. It is usually the class name along with some flags.
208 *
209 * @param $object
210 * @param $level
211 * @param $plainText
212 * @param $ansiColors
213 * @return string The rendered header with tags
214 */
215 static protected function renderHeader($object, $level, $plainText, $ansiColors) {
216 $dump = '';
217 $persistenceType = '';
218 $className = get_class($object);
219 if ($plainText) {
220 $dump .= self::ansiEscapeWrap($className, '36', $ansiColors);
221 } else {
222 $dump .= '<span class="debug-type">' . $className . '</span>';
223 }
224 if ($object instanceof \TYPO3\CMS\Core\SingletonInterface) {
225 $scope = 'singleton';
226 } else {
227 $scope = 'prototype';
228 }
229 if ($plainText) {
230 $dump .= ' ' . self::ansiEscapeWrap($scope, '44;37', $ansiColors);
231 } else {
232 $dump .= $scope ? '<span class="debug-scope">' . $scope . '</span>' : '';
233 }
234 if ($object instanceof \TYPO3\CMS\Extbase\DomainObject\AbstractDomainObject) {
235 if ($object->_isDirty()) {
236 $persistenceType = 'modified';
237 } elseif ($object->_isNew()) {
238 $persistenceType = 'transient';
239 } else {
240 $persistenceType = 'persistent';
241 }
242 }
243 if ($object instanceof \TYPO3\CMS\Extbase\Persistence\ObjectStorage && $object->_isDirty()) {
244 $persistenceType = 'modified';
245 }
246 if ($object instanceof \TYPO3\CMS\Extbase\DomainObject\AbstractEntity) {
247 $domainObjectType = 'entity';
248 } elseif ($object instanceof \TYPO3\CMS\Extbase\DomainObject\AbstractValueObject) {
249 $domainObjectType = 'valueobject';
250 } else {
251 $domainObjectType = 'object';
252 }
253 if ($plainText) {
254 $dump .= ' ' . self::ansiEscapeWrap(($persistenceType . ' ' . $domainObjectType), '42;30', $ansiColors);
255 } else {
256 $dump .= '<span class="debug-ptype">' . $persistenceType . ' ' . $domainObjectType . '</span>';
257 }
258 if (strpos(implode('|', self::$blacklistedClassNames), get_class($object)) > 0) {
259 if ($plainText) {
260 $dump .= ' ' . self::ansiEscapeWrap('filtered', '47;30', $ansiColors);
261 } else {
262 $dump .= '<span class="debug-filtered">filtered</span>';
263 }
264 } elseif (self::$renderedObjects->contains($object) && !$plainText) {
265 $dump = '<a href="javascript:;" onclick="document.location.hash=\'#' . spl_object_hash($object) . '\';" class="debug-seeabove">' . $dump . '<span class="debug-filtered">see above</span></a>';
266 } elseif ($level >= self::$maxDepth && !$object instanceof \DateTime) {
267 if ($plainText) {
268 $dump .= ' ' . self::ansiEscapeWrap('max depth', '47;30', $ansiColors);
269 } else {
270 $dump .= '<span class="debug-filtered">max depth</span>';
271 }
272 } elseif ($level > 1 && !$object instanceof \DateTime && !$plainText) {
273 $dump = '<input type="checkbox" id="' . spl_object_hash($object) . '" /><span class="debug-header">' . $dump . '</span>';
274 }
275 if ($object instanceof \Countable) {
276 $dump .= count($object) > 0 ? ' (' . count($object) . ' items)' : ' (empty)';
277 }
278 if ($object instanceof \DateTime) {
279 $dump .= ' (' . $object->format(\DateTime::RFC3339) . ', ' . $object->getTimestamp() . ')';
280 }
281 if ($object instanceof \TYPO3\CMS\Extbase\DomainObject\DomainObjectInterface && !$object->_isNew()) {
282 $dump .= ' (uid=' . $object->getUid() . ', pid=' . $object->getPid() . ')';
283 }
284 return $dump;
285 }
286
287 /**
288 * @param $object
289 * @param $level
290 * @param $plainText
291 * @param $ansiColors
292 * @return string The rendered body content of the Object(Storage)
293 */
294 static protected function renderContent($object, $level, $plainText, $ansiColors) {
295 $dump = '';
296 if ($object instanceof \TYPO3\CMS\Extbase\Persistence\ObjectStorage || $object instanceof \Iterator) {
297 $dump .= self::renderCollection($object, $level, $plainText, $ansiColors);
298 } else {
299 self::$renderedObjects->attach($object);
300 if (!$plainText) {
301 $dump .= '<a name="' . spl_object_hash($object) . '" id="' . spl_object_hash($object) . '"></a>';
302 }
303 $classReflection = new \ReflectionClass(get_class($object));
304 $properties = $classReflection->getProperties();
305 foreach ($properties as $property) {
306 if (self::isBlacklisted($property)) {
307 continue;
308 }
309 $dump .= PHP_EOL . str_repeat(self::PLAINTEXT_INDENT, $level) . ($plainText ? '' : '<span class="debug-property">') . self::ansiEscapeWrap($property->getName(), '37', $ansiColors) . ($plainText ? '' : '</span>') . ' => ';
310 $property->setAccessible(TRUE);
311 $dump .= self::renderDump($property->getValue($object), $level, $plainText, $ansiColors);
312 if ($object instanceof \TYPO3\CMS\Extbase\DomainObject\AbstractDomainObject && !$object->_isNew() && $object->_isDirty($property->getName())) {
313 if ($plainText) {
314 $dump .= ' ' . self::ansiEscapeWrap('modified', '43;30', $ansiColors);
315 } else {
316 $dump .= '<span class="debug-dirty">modified</span>';
317 }
318 }
319 }
320 }
321 return $dump;
322 }
323
324 /**
325 * @param mixed $collection
326 * @param integer $level
327 * @param boolean $plainText
328 * @param boolean $ansiColors
329 * @return string
330 */
331 static protected function renderCollection($collection, $level, $plainText, $ansiColors) {
332 $dump = '';
333 foreach ($collection as $key => $value) {
334 $dump .= PHP_EOL . str_repeat(self::PLAINTEXT_INDENT, $level) . ($plainText ? '' : '<span class="debug-property">') . self::ansiEscapeWrap($key, '37', $ansiColors) . ($plainText ? '' : '</span>') . ' => ';
335 $dump .= self::renderDump($value, $level + 1, $plainText, $ansiColors);
336 }
337 return $dump;
338 }
339
340 /**
341 * Wrap a string with the ANSI escape sequence for colorful output
342 *
343 * @param string $string The string to wrap
344 * @param string $ansiColors The ansi color sequence (e.g. "1;37")
345 * @param boolean $enable If FALSE, the raw string will be returned
346 * @return string The wrapped or raw string
347 */
348 static protected function ansiEscapeWrap($string, $ansiColors, $enable = TRUE) {
349 if ($enable) {
350 return '\e[' . $ansiColors . 'm' . $string . '\e[0m';
351 } else {
352 return $string;
353 }
354 }
355
356 /**
357 * A var_dump function optimized for Extbase's object structures
358 *
359 * @param mixed $variable The value to dump
360 * @param string $title optional custom title for the debug output
361 * @param integer $maxDepth Sets the max recursion depth of the dump. De- or increase the number according to your needs and memory limit.
362 * @param boolean $plainText If TRUE, the dump is in plain text, if FALSE the debug output is in HTML format.
363 * @param boolean $ansiColors If TRUE (default), ANSI color codes is added to the output, if FALSE the debug output not colored.
364 * @param boolean $return if TRUE, the dump is returned for custom post-processing (e.g. embed in custom HTML). If FALSE (default), the dump is directly displayed.
365 * @param array $blacklistedClassNames An array of class names (RegEx) to be filtered. Default is an array of some common class names.
366 * @param array $blacklistedPropertyNames An array of property names and/or array keys (RegEx) to be filtered. Default is an array of some common property names.
367 * @return string if $return is TRUE, the dump is returned. By default, the dump is directly displayed, and nothing is returned.
368 * @api
369 */
370 static public function var_dump($variable, $title = NULL, $maxDepth = 8, $plainText = FALSE, $ansiColors = TRUE, $return = FALSE, $blacklistedClassNames = NULL, $blacklistedPropertyNames = NULL) {
371 self::$maxDepth = $maxDepth;
372 if ($title === NULL) {
373 $title = 'Extbase Variable Dump';
374 }
375 $ansiColors = $plainText && $ansiColors;
376 if ($ansiColors === TRUE) {
377 $title = '\e[1m' . $title . '\e[0m';
378 }
379 if (is_array($blacklistedClassNames)) {
380 self::$blacklistedClassNames = $blacklistedClassNames;
381 }
382 if (is_array($blacklistedPropertyNames)) {
383 self::$blacklistedPropertyNames = $blacklistedPropertyNames;
384 }
385 self::clearState();
386 if (!$plainText && self::$stylesheetEchoed === FALSE) {
387 echo '
388 <style type=\'text/css\'>
389 .debug-tree{position:relative;}
390 .debug-tree input{position:absolute;top:0;left:0;cursor:pointer;opacity:0;z-index:2;}
391 .debug-tree input ~ .debug-content{display:none;}
392 .debug-tree .debug-header:before{content:"+";padding:0 2px 0 2px;margin:0 3px 0 3px;font-size:1em;font-weight:bold;color:#004fb0;border:1px #004fb0 solid;}
393 .debug-tree input:checked ~ .debug-content{display:inline;}
394 .debug-tree input:checked ~ .debug-header:before{content:"-";}
395 .Extbase-Utility-Debugger-VarDump{display:block;text-align:left;background:#b9b9b9;border:10px solid #b9b9b9;-moz-border-radius:10px;-webkit-border-radius:10px;border-radius:10px;-moz-box-shadow:0 0 20px #333;-webkit-box-shadow:0 0 20px #333;box-shadow:0 0 20px #333;z-index:999;color:#000;margin:20px 0 0;}
396 .Extbase-Utility-Debugger-VarDump-Floating{position:relative;width:96%;margin:40px auto;}
397 .Extbase-Utility-Debugger-VarDump-Top{background:#eee;font:normal bold 12px \'Lucida Grande\',sans-serif;padding:5px;}
398 .Extbase-Utility-Debugger-VarDump-Center{background:#b9b9b9 url() 0 18px repeat;font:normal normal 11px/18px Monospaced,\'Lucida Console\',monospace;padding:18px 10px;}
399 .Extbase-Utility-Debugger-VarDump-Center pre{background-color:transparent;margin:0;}
400 .Extbase-Utility-Debugger-VarDump-Center,.Extbase-Utility-Debugger-VarDump-Center pre,.Extbase-Utility-Debugger-VarDump-Center p,.Extbase-Utility-Debugger-VarDump-Center a,.Extbase-Utility-Debugger-VarDump-Center strong,.Extbase-Utility-Debugger-VarDump-Center .debug-string{font:normal normal 11px/18px Monospaced,\'Lucida Console\',monospace;}
401 .Extbase-Utility-Debugger-VarDump-Center .debug-string{color:#000;white-space:normal;}
402 .Extbase-Utility-Debugger-VarDump-Center .debug-type{color:#004fb0;padding-right:4px;}
403 .Extbase-Utility-Debugger-VarDump-Center .debug-unregistered{background-color:#dce1e8;}
404 .Extbase-Utility-Debugger-VarDump-Center .debug-scope,.Extbase-Utility-Debugger-VarDump-Center .debug-ptype,.Extbase-Utility-Debugger-VarDump-Center .debug-proxy,.Extbase-Utility-Debugger-VarDump-Center .debug-filtered{color:#FFF;font-size:10px;line-height:16px;padding:1px 4px;}
405 .Extbase-Utility-Debugger-VarDump-Center .debug-scope{background-color:#3e7fe1;}
406 .Extbase-Utility-Debugger-VarDump-Center .debug-ptype{background-color:#6FBC16;}
407 .Extbase-Utility-Debugger-VarDump-Center .debug-dirty{background-color:#FFFF00;}
408 .Extbase-Utility-Debugger-VarDump-Center .debug-filtered{background-color:#8c8c8c;}
409 .Extbase-Utility-Debugger-VarDump-Center .debug-seeabove{text-decoration:none;font-style:italic;font-weight:400;}
410 .Extbase-Utility-Debugger-VarDump-Center .debug-property{color:#555;line-height:16px;padding:1px 2px;}
411 </style>';
412 self::$stylesheetEchoed = TRUE;
413 }
414 if ($plainText) {
415 $output = $title . PHP_EOL . self::renderDump($variable, 0, TRUE, $ansiColors) . PHP_EOL . PHP_EOL;
416 } else {
417 $output = '
418 <div class="Extbase-Utility-Debugger-VarDump ' . ($return ? 'Extbase-Utility-Debugger-VarDump-Inline' : 'Extbase-Utility-Debugger-VarDump-Floating') . '">
419 <div class="Extbase-Utility-Debugger-VarDump-Top">' . htmlspecialchars($title) . '</div>
420 <div class="Extbase-Utility-Debugger-VarDump-Center">
421 <pre dir="ltr">' . self::renderDump($variable, 0, FALSE, FALSE) . '</pre>
422 </div>
423 </div>
424 ';
425 }
426 if ($return === TRUE) {
427 return $output;
428 } else {
429 echo $output;
430 }
431 return '';
432 }
433
434 }
435
436
437 ?>