<?php
final class ITSEC_Debug {
public static function print_r( $data, $args = array(), $echo = true ) {
$html = "<style>.wp-admin .it-debug-print-r { margin-left: 170px; } .wp-admin #wpcontent .it-debug-print-r { margin-left: 0; }</style>\n";
$html .= "<pre style='color:black;background:white;padding:15px;font-family:\"Courier New\",Courier,monospace;font-size:12px;white-space:pre-wrap;text-align:left;max-width:100%;' class='it-debug-print-r'>";
$html .= self::get_print_r( $data, $args );
$html .= "</pre>\n";
if ( $echo ) {
echo $html;
}
return $html;
}
public static function get_print_r( $data, $args = array() ) {
if ( is_bool( $args ) ) {
$args = array( 'expand_objects' => $args );
} else if ( is_numeric( $args ) ) {
$args = array( 'max_depth' => $args );
} else if ( ! is_array( $args ) ) {
$args = array();
}
// Create a deep copy so that variables aren't needlessly manipulated.
$data = unserialize( serialize( $data ) );
$default_args = array(
'expand_objects' => true,
'max_depth' => 10,
);
$args = array_merge( $default_args, $args );
if ( $args['max_depth'] < 1 ) {
$args['max_depth'] = 100;
}
return self::inspect_dive( $data, $args['expand_objects'], $args['max_depth'] );
}
public static function backtrace( $args = array() ) {
if ( is_string( $args ) ) {
$args = array( 'description' => $args );
} else if ( is_bool( $args ) ) {
$args = array( 'expand_objects' => $args );
} else if ( is_numeric( $args ) ) {
$args = array( 'max_depth' => $args );
} else if ( ! is_array( $args ) ) {
$args = array();
}
$default_args = array(
'description' => '',
'expand_objects' => false,
'max_depth' => 3,
'type' => '',
);
$args = array_merge( $default_args, $args );
if ( isset( $args['offset'] ) ) {
$args['offset']++;
} else {
$args['offset'] = 1;
}
$backtrace = self::get_backtrace( $args );
if ( 'string' == $args['type'] ) {
echo $backtrace;
} else {
$args['max_depth']++;
self::print_r( $backtrace, $args );
}
}
public static function get_backtrace( $args = array() ) {
if ( is_bool( $args ) ) {
$args = array( 'expand_objects' => $args );
} else if ( ! is_array( $args ) ) {
$args = array();
}
$default_args = array(
'expand_objects' => false,
'limit' => 0,
'offset' => 0,
'type' => 'array', // 'array' or 'string'
);
$args = array_merge( $default_args, $args );
$backtrace = debug_backtrace();
unset( $backtrace[0] );
if ( $args['offset'] > 0 ) {
$backtrace = array_slice( $backtrace, $args['offset'] );
}
if ( $args['limit'] > 0 ) {
$backtrace = array_slice( $backtrace, 0, $args['limit'] );
}
$backtrace = array_values( $backtrace );
if ( 'string' == $args['type'] ) {
$string_backtrace = '';
foreach ( $backtrace as $trace ) {
$string_backtrace .= self::get_backtrace_description( $trace, $args ) . "\n";
}
$backtrace = $string_backtrace;
}
return $backtrace;
}
private static function get_backtrace_description( $backtrace, $backtrace_args = array() ) {
$default_backtrace_args = array(
'remove_abspath' => true,
);
$backtrace_args = array_merge( $default_backtrace_args, $backtrace_args );
extract( $backtrace );
$args = self::flatten_backtrace_description_args( $args );
if ( $backtrace_args['remove_abspath'] && isset( $file ) ) {
$file = preg_replace( '/^' . preg_quote( ABSPATH, '/' ) . '/', '', $file );
}
if ( isset( $class ) && isset( $type ) && isset( $function ) && isset( $args ) ) {
return "<strong>$class$type$function(</strong>$args<strong>)</strong>";
} else if ( isset( $function ) && isset( $args ) ) {
if ( isset( $file ) && isset( $line ) ) {
return "<strong>$function(</strong>$args<strong>)</strong> on line $line of $file";
}
return "<strong>$function(</strong>$args<strong>)</strong>";
}
return 'String!';
}
private static function flatten_backtrace_description_args( $args, $max_depth = 2, $depth = 0 ) {
if ( is_string( $args ) ) {
return "'$args'";
} else if ( is_int( $args ) ) {
return "(int) $args";
} else if ( is_float( $args ) ) {
return "(float) $args";
} else if ( is_bool( $args ) ) {
return '(bool) ' . ( $args ? 'true' : 'false' );
} else if ( is_object( $args ) ) {
return '(object) ' . get_class( $args );
} else if ( ! is_array( $args ) ) {
return '[unknown]';
}
if ( $depth === $max_depth ) {
if ( empty( $args ) ) {
return 'array()';
} else {
return 'array( ' . count( $args ) . ' )';
}
}
$flat_args = array();
foreach ( $args as $arg ) {
$flat_args[] = self::flatten_backtrace_description_args( $arg, $max_depth, $depth + 1 );
}
$args = implode( ', ', $flat_args );
if ( ! empty( $args ) ) {
$args = " $args ";
}
if ( 0 === $depth ) {
return $args;
}
return "array($args)";
}
private static function pad( $depth, $pad = ' ' ) {
$retval = '';
for ( $x = 0; $x <= $depth; $x++ ) {
$retval .= $pad;
}
return $retval;
}
private static function is_callable_function( $function ) {
if ( ! is_callable( $function ) ) {
return false;
}
if ( ! isset( $GLOBALS['itsec_debug_cached_values'] ) ) {
$GLOBALS['itsec_debug_cached_values'] = array();
}
if ( ! isset( $GLOBALS['itsec_debug_cached_values']['ini_get:disable_functions'] ) ) {
$GLOBALS['itsec_debug_cached_values']['var:disable_functions'] = preg_split( '/\s*,\s*/', (string) ini_get( 'disable_functions' ) );
}
if ( in_array( $function, $GLOBALS['itsec_debug_cached_values']['var:disable_functions'] ) ) {
return false;
}
if ( ! isset( $GLOBALS['itsec_debug_cached_values']['ini_get:suhosin.executor.func.blacklist'] ) ) {
$GLOBALS['itsec_debug_cached_values']['ini_get:suhosin.executor.func.blacklist'] = preg_split( '/\s*,\s*/', (string) ini_get( 'suhosin.executor.func.blacklist' ) );
}
if ( in_array( $function, $GLOBALS['itsec_debug_cached_values']['ini_get:suhosin.executor.func.blacklist'] ) ) {
return false;
}
return true;
}
private static function inspect_dive( $data, $expand_objects, $max_depth, $depth = 0, $show_array_header = true ) {
$pad = self::pad( $depth, ' ' );
if ( is_string( $data ) ) {
if ( empty( $data ) ) {
return "<strong>[empty string]</strong>";
} else {
if ( self::is_callable_function( 'mb_detect_encoding' ) && ( 'UTF-8' !== mb_detect_encoding( $data, 'UTF-8', true ) ) && self::is_callable_function( 'utf8_encode' ) ) {
$data = utf8_encode( $data );
}
$flags = ENT_COMPAT;
if ( defined( 'ENT_HTML401' ) ) {
// phpcs:ignore PHPCompatibility.PHP.NewConstants.ent_html401Found -- Ensure that Tide doesn't reduce compatibility because of this code which is meant to improve compatibility.
$flags |= ENT_HTML401;
}
return htmlspecialchars( $data, $flags, 'UTF-8', false );
}
}
if ( is_bool( $data ) ) {
return ( $data ) ? '<strong>[boolean] true</strong>' : '<strong>[boolean] false</strong>';
}
if ( is_null( $data ) ) {
return '<strong>null</strong>';
}
if ( is_object( $data ) || 'object' === gettype( $data ) ) {
if ( ! is_object( $data ) || '__PHP_Incomplete_Class' === get_class( $data ) ) {
// Special handling for objects for classes that are not loaded.
$vars = get_object_vars( $data );
$class_name = $vars['__PHP_Incomplete_Class_Name'];
} else {
$class_name = get_class( $data );
}
$retval = "<strong>Object</strong> $class_name";
if ( ! $expand_objects || ( $depth == $max_depth ) ) {
return $retval;
}
if ( isset( $vars ) ) {
// Special handling for objects for classes that are not loaded.
unset( $vars['__PHP_Incomplete_Class_Name'] );
$new_vars = array();
foreach ( $vars as $key => $val ) {
$key = substr( $key, strlen( $class_name ) + 2 );
$new_vars[$key] = $val;
}
$vars = $new_vars;
} else {
$vars = get_object_vars( $data );
}
if ( empty( $vars ) ) {
$vars = '';
} else {
$vars = self::inspect_dive( $vars, $expand_objects, $max_depth, $depth, false );
}
$retval .= "$vars";
return $retval;
}
if ( is_array( $data ) ) {
$retval = ( $show_array_header ) ? '<strong>Array</strong>' : '';
if ( empty( $data ) ) {
return "$retval()";
}
if ( $depth == $max_depth ) {
return "$retval( " . count( $data ) . " )";
}
$max = 0;
foreach ( array_keys( $data ) as $index ) {
if ( strlen( $index ) > $max ) {
$max = strlen( $index );
}
}
foreach ( $data as $index => $val ) {
$spaces = self::pad( $max - strlen( $index ), ' ' );
$retval .= "\n$pad" . htmlspecialchars( $index ) . "$spaces <strong>=></strong> " . self::inspect_dive( $val, $expand_objects, $max_depth, $depth + 1 );
}
return $retval;
}
return '<strong>[' . gettype( $data ) . ']</strong> ' . $data;
}
}