' string, * mostly for security reasons. Use string values for global functions, * and array notation for class methods. * PHP's debug_backtrace() doesn't allow to inspect the argument names, * so all arguments of the provided functions will be filtered out. * @var array */ private static $ignore_function_args = []; /** * Return debug_backtrace() results with functions filtered * specific to the debugging system, and not the trace. * * @param null|array $ignoredFunctions If an array, filter these functions out of the trace * @return array */ public static function filtered_backtrace($ignoredFunctions = null) { return self::filter_backtrace(debug_backtrace(), $ignoredFunctions); } /** * Filter a backtrace so that it doesn't show the calls to the * debugging system, which is useless information. * * @param array $bt Backtrace to filter * @param null|array $ignoredFunctions List of extra functions to filter out * @return array */ public static function filter_backtrace($bt, $ignoredFunctions = null) { $defaultIgnoredFunctions = [ 'SilverStripe\\Logging\\Log::log', 'SilverStripe\\Dev\\Backtrace::backtrace', 'SilverStripe\\Dev\\Backtrace::filtered_backtrace', 'Zend_Log_Writer_Abstract->write', 'Zend_Log->log', 'Zend_Log->__call', 'Zend_Log->err', 'SilverStripe\\Dev\\DebugView->writeTrace', 'SilverStripe\\Dev\\CliDebugView->writeTrace', 'SilverStripe\\Dev\\Debug::emailError', 'SilverStripe\\Dev\\Debug::warningHandler', 'SilverStripe\\Dev\\Debug::noticeHandler', 'SilverStripe\\Dev\\Debug::fatalHandler', 'errorHandler', 'SilverStripe\\Dev\\Debug::showError', 'SilverStripe\\Dev\\Debug::backtrace', 'exceptionHandler' ]; if ($ignoredFunctions) { foreach ($ignoredFunctions as $ignoredFunction) { $defaultIgnoredFunctions[] = $ignoredFunction; } } while ($bt && in_array(self::full_func_name($bt[0]), $defaultIgnoredFunctions ?? [])) { array_shift($bt); } $ignoredArgs = static::config()->get('ignore_function_args'); // Filter out arguments foreach ($bt as $i => $frame) { $match = false; if (!empty($frame['class'])) { foreach ($ignoredArgs as $fnSpec) { if (is_array($fnSpec) && self::matchesFilterableClass($frame['class'], $fnSpec[0]) && $frame['function'] == $fnSpec[1] ) { $match = true; break; } } } else { if (in_array($frame['function'], $ignoredArgs ?? [])) { $match = true; } } if ($match) { foreach ($frame['args'] as $j => $arg) { $bt[$i]['args'][$j] = ''; } } } return $bt; } /** * Render or return a backtrace from the given scope. * * @param mixed $returnVal * @param bool $ignoreAjax * @param array $ignoredFunctions * @return mixed */ public static function backtrace($returnVal = false, $ignoreAjax = false, $ignoredFunctions = null) { $plainText = Director::is_cli() || (Director::is_ajax() && !$ignoreAjax); $result = self::get_rendered_backtrace(debug_backtrace(), $plainText, $ignoredFunctions); if ($returnVal) { return $result; } else { echo $result; return null; } } /** * Return the full function name. If showArgs is set to true, a string representation of the arguments will be * shown * * @param Object $item * @param bool $showArgs * @param int $argCharLimit * @return string */ public static function full_func_name($item, $showArgs = false, $argCharLimit = 10000) { $funcName = ''; if (isset($item['class'])) { $funcName .= $item['class']; } if (isset($item['type'])) { $funcName .= $item['type']; } if (isset($item['function'])) { $funcName .= $item['function']; } if ($showArgs && isset($item['args'])) { $args = []; foreach ($item['args'] as $arg) { if (!is_object($arg) || method_exists($arg, '__toString')) { $sarg = is_array($arg) ? 'Array' : strval($arg); $args[] = (strlen($sarg ?? '') > $argCharLimit) ? substr($sarg, 0, $argCharLimit) . '...' : $sarg; } else { $args[] = get_class($arg); } } $funcName .= "(" . implode(", ", $args) . ")"; } return $funcName; } /** * Render a backtrace array into an appropriate plain-text or HTML string. * * @param array $bt The trace array, as returned by debug_backtrace() or Exception::getTrace() * @param boolean $plainText Set to false for HTML output, or true for plain-text output * @param array $ignoredFunctions List of functions that should be ignored. If not set, a default is provided * @return string The rendered backtrace */ public static function get_rendered_backtrace($bt, $plainText = false, $ignoredFunctions = null) { if (empty($bt)) { return ''; } $bt = self::filter_backtrace($bt, $ignoredFunctions); $result = ($plainText) ? '' : ''; } return $result; } /** * Checks if the filterable class is wildcard, of if the class name is the filterable class, or a subclass of it, * or implements it. */ private static function matchesFilterableClass(string $className, string $filterableClass): bool { return $filterableClass === '*' || $className === $filterableClass || is_subclass_of($className, $filterableClass); } }